blob: 4db8dc9682cebefd6224388322129afc6616cef9 [file] [log] [blame]
/*
* Copyright 2024 The Android Open Source Project
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/core/SkFontScanner.h"
#include "src/sfnt/SkOTUtils.h"
#include "src/ports/SkFontScanner_fontations.h"
#include "src/ports/SkTypeface_fontations_priv.h"
#include "src/ports/fontations/src/skpath_bridge.h"
using namespace skia_private;
namespace {
rust::Box<::fontations_ffi::BridgeFontRef> make_bridge_font_ref(sk_sp<SkData> fontData,
uint32_t index) {
rust::Slice<const uint8_t> slice{fontData->bytes(), fontData->size()};
return fontations_ffi::make_font_ref(slice, index);
}
}
SkFontScanner_Fontations::SkFontScanner_Fontations() {
}
SkFontScanner_Fontations::~SkFontScanner_Fontations() {
}
bool SkFontScanner_Fontations::scanFile(SkStreamAsset* stream, int* numFaces) const {
sk_sp<SkData> fontData = SkData::MakeFromStream(stream, stream->getLength());
stream->rewind();
rust::Slice<const uint8_t> slice{fontData->bytes(), fontData->size()};
::std::uint32_t num_fonts;
if (!fontations_ffi::font_or_collection(slice, num_fonts)) {
return false;
}
*numFaces = num_fonts == 0 ? 1 : num_fonts;
return true;
}
bool SkFontScanner_Fontations::scanFace(SkStreamAsset* stream,
int faceIndex,
int* numInstances) const {
rust::Box<fontations_ffi::BridgeFontRef> fontRef =
make_bridge_font_ref(SkData::MakeFromStream(stream, stream->getLength()), faceIndex);
stream->rewind();
if (!fontations_ffi::font_ref_is_valid(*fontRef)) {
return false;
}
// TODO: For now assume only the default instance (not variation)
*numInstances = 1;
return true;
}
// TODO: For now assume only the default instance (not variation)
bool SkFontScanner_Fontations::scanInstance(SkStreamAsset* stream,
int faceIndex,
int instanceIndex,
SkString* name,
SkFontStyle* style,
bool* isFixedPitch,
AxisDefinitions* axes) const {
SkASSERT(instanceIndex == 0);
rust::Box<fontations_ffi::BridgeFontRef> fontRef =
make_bridge_font_ref(SkData::MakeFromStream(stream, stream->getLength()), faceIndex);
if (!fontations_ffi::font_ref_is_valid(*fontRef)) {
return false;
}
if (name != nullptr) {
rust::String readFamilyName = fontations_ffi::family_name(*fontRef);
*name = SkString(readFamilyName.data(), readFamilyName.size());
}
if (style != nullptr) {
rust::Slice<const fontations_ffi::SkiaDesignCoordinate> coordinates;
rust::Box<fontations_ffi::BridgeNormalizedCoords> normalizedCoords =
resolve_into_normalized_coords(*fontRef, coordinates);
fontations_ffi::BridgeFontStyle fontStyle;
if (fontations_ffi::get_font_style(*fontRef, *normalizedCoords, fontStyle)) {
*style = SkFontStyle(fontStyle.weight, fontStyle.width, (SkFontStyle::Slant)fontStyle.slant);
} else {
*style = SkFontStyle::Normal();
}
}
return true;
}