#[allow(unused_imports)]
use crate::codegen_prelude::*;
impl<'a> MinByteRange<'a> for Morx<'a> {
fn min_byte_range(&self) -> Range<usize> {
0..self.chains_byte_range().end
}
fn min_table_bytes(&self) -> &'a [u8] {
let range = self.min_byte_range();
self.data.as_bytes().get(range).unwrap_or_default()
}
}
impl TopLevelTable for Morx<'_> {
const TAG: Tag = Tag::new(b"morx");
}
impl<'a> FontRead<'a> for Morx<'a> {
fn read(data: FontData<'a>) -> Result<Self, ReadError> {
#[allow(clippy::absurd_extreme_comparisons)]
if data.len() < Self::MIN_SIZE {
return Err(ReadError::OutOfBounds);
}
Ok(Self { data })
}
}
#[derive(Clone)]
pub struct Morx<'a> {
data: FontData<'a>,
}
#[allow(clippy::needless_lifetimes)]
impl<'a> Morx<'a> {
pub const MIN_SIZE: usize = (u16::RAW_BYTE_LEN + u16::RAW_BYTE_LEN + u32::RAW_BYTE_LEN);
basic_table_impls!(impl_the_methods);
pub fn version(&self) -> u16 {
let range = self.version_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn n_chains(&self) -> u32 {
let range = self.n_chains_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn chains(&self) -> VarLenArray<'a, Chain<'a>> {
let range = self.chains_byte_range();
self.data
.split_off(range.start)
.and_then(|d| VarLenArray::read(d).ok())
.unwrap_or_default()
}
pub fn version_byte_range(&self) -> Range<usize> {
let start = 0;
start..start + u16::RAW_BYTE_LEN
}
pub fn unused_byte_range(&self) -> Range<usize> {
let start = self.version_byte_range().end;
start..start + u16::RAW_BYTE_LEN
}
pub fn n_chains_byte_range(&self) -> Range<usize> {
let start = self.unused_byte_range().end;
start..start + u32::RAW_BYTE_LEN
}
pub fn chains_byte_range(&self) -> Range<usize> {
let n_chains = self.n_chains();
let start = self.n_chains_byte_range().end;
start..start + {
let data = self.data.split_off(start).unwrap_or_default();
<Chain as VarSize>::total_len_for_count(data, n_chains as usize).unwrap_or(0)
}
}
}
#[cfg(feature = "experimental_traverse")]
impl<'a> SomeTable<'a> for Morx<'a> {
fn type_name(&self) -> &str {
"Morx"
}
fn get_field(&self, idx: usize) -> Option<Field<'a>> {
match idx {
0usize => Some(Field::new("version", self.version())),
1usize => Some(Field::new("n_chains", self.n_chains())),
2usize => Some(Field::new(
"chains",
traversal::FieldType::var_array("Chain", self.chains(), self.offset_data()),
)),
_ => None,
}
}
}
#[cfg(feature = "experimental_traverse")]
#[allow(clippy::needless_lifetimes)]
impl<'a> std::fmt::Debug for Morx<'a> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
(self as &dyn SomeTable<'a>).fmt(f)
}
}
impl<'a> MinByteRange<'a> for Chain<'a> {
fn min_byte_range(&self) -> Range<usize> {
0..self.subtables_byte_range().end
}
fn min_table_bytes(&self) -> &'a [u8] {
let range = self.min_byte_range();
self.data.as_bytes().get(range).unwrap_or_default()
}
}
impl<'a> FontRead<'a> for Chain<'a> {
fn read(data: FontData<'a>) -> Result<Self, ReadError> {
#[allow(clippy::absurd_extreme_comparisons)]
if data.len() < Self::MIN_SIZE {
return Err(ReadError::OutOfBounds);
}
Ok(Self { data })
}
}
#[derive(Clone)]
pub struct Chain<'a> {
data: FontData<'a>,
}
#[allow(clippy::needless_lifetimes)]
impl<'a> Chain<'a> {
pub const MIN_SIZE: usize =
(u32::RAW_BYTE_LEN + u32::RAW_BYTE_LEN + u32::RAW_BYTE_LEN + u32::RAW_BYTE_LEN);
basic_table_impls!(impl_the_methods);
pub fn default_flags(&self) -> u32 {
let range = self.default_flags_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn chain_length(&self) -> u32 {
let range = self.chain_length_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn n_feature_entries(&self) -> u32 {
let range = self.n_feature_entries_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn n_subtables(&self) -> u32 {
let range = self.n_subtables_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn features(&self) -> &'a [Feature] {
let range = self.features_byte_range();
self.data.read_array(range).ok().unwrap_or_default()
}
pub fn subtables(&self) -> VarLenArray<'a, Subtable<'a>> {
let range = self.subtables_byte_range();
self.data
.split_off(range.start)
.and_then(|d| VarLenArray::read(d).ok())
.unwrap_or_default()
}
pub fn default_flags_byte_range(&self) -> Range<usize> {
let start = 0;
start..start + u32::RAW_BYTE_LEN
}
pub fn chain_length_byte_range(&self) -> Range<usize> {
let start = self.default_flags_byte_range().end;
start..start + u32::RAW_BYTE_LEN
}
pub fn n_feature_entries_byte_range(&self) -> Range<usize> {
let start = self.chain_length_byte_range().end;
start..start + u32::RAW_BYTE_LEN
}
pub fn n_subtables_byte_range(&self) -> Range<usize> {
let start = self.n_feature_entries_byte_range().end;
start..start + u32::RAW_BYTE_LEN
}
pub fn features_byte_range(&self) -> Range<usize> {
let n_feature_entries = self.n_feature_entries();
let start = self.n_subtables_byte_range().end;
start..start + (n_feature_entries as usize).saturating_mul(Feature::RAW_BYTE_LEN)
}
pub fn subtables_byte_range(&self) -> Range<usize> {
let n_subtables = self.n_subtables();
let start = self.features_byte_range().end;
start..start + {
let data = self.data.split_off(start).unwrap_or_default();
<Subtable as VarSize>::total_len_for_count(data, n_subtables as usize).unwrap_or(0)
}
}
}
#[cfg(feature = "experimental_traverse")]
impl<'a> SomeTable<'a> for Chain<'a> {
fn type_name(&self) -> &str {
"Chain"
}
fn get_field(&self, idx: usize) -> Option<Field<'a>> {
match idx {
0usize => Some(Field::new("default_flags", self.default_flags())),
1usize => Some(Field::new("chain_length", self.chain_length())),
2usize => Some(Field::new("n_feature_entries", self.n_feature_entries())),
3usize => Some(Field::new("n_subtables", self.n_subtables())),
4usize => Some(Field::new(
"features",
traversal::FieldType::array_of_records(
stringify!(Feature),
self.features(),
self.offset_data(),
),
)),
5usize => Some(Field::new(
"subtables",
traversal::FieldType::var_array("Subtable", self.subtables(), self.offset_data()),
)),
_ => None,
}
}
}
#[cfg(feature = "experimental_traverse")]
#[allow(clippy::needless_lifetimes)]
impl<'a> std::fmt::Debug for Chain<'a> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
(self as &dyn SomeTable<'a>).fmt(f)
}
}
#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Copy, bytemuck :: AnyBitPattern)]
#[repr(C)]
#[repr(packed)]
pub struct Feature {
pub feature_type: BigEndian<u16>,
pub feature_settings: BigEndian<u16>,
pub enable_flags: BigEndian<u32>,
pub disable_flags: BigEndian<u32>,
}
impl Feature {
pub fn feature_type(&self) -> u16 {
self.feature_type.get()
}
pub fn feature_settings(&self) -> u16 {
self.feature_settings.get()
}
pub fn enable_flags(&self) -> u32 {
self.enable_flags.get()
}
pub fn disable_flags(&self) -> u32 {
self.disable_flags.get()
}
}
impl FixedSize for Feature {
const RAW_BYTE_LEN: usize =
u16::RAW_BYTE_LEN + u16::RAW_BYTE_LEN + u32::RAW_BYTE_LEN + u32::RAW_BYTE_LEN;
}
#[cfg(feature = "experimental_traverse")]
impl<'a> SomeRecord<'a> for Feature {
fn traverse(self, data: FontData<'a>) -> RecordResolver<'a> {
RecordResolver {
name: "Feature",
get_field: Box::new(move |idx, _data| match idx {
0usize => Some(Field::new("feature_type", self.feature_type())),
1usize => Some(Field::new("feature_settings", self.feature_settings())),
2usize => Some(Field::new("enable_flags", self.enable_flags())),
3usize => Some(Field::new("disable_flags", self.disable_flags())),
_ => None,
}),
data,
}
}
}
impl<'a> MinByteRange<'a> for Subtable<'a> {
fn min_byte_range(&self) -> Range<usize> {
0..self.data_byte_range().end
}
fn min_table_bytes(&self) -> &'a [u8] {
let range = self.min_byte_range();
self.data.as_bytes().get(range).unwrap_or_default()
}
}
impl<'a> FontRead<'a> for Subtable<'a> {
fn read(data: FontData<'a>) -> Result<Self, ReadError> {
#[allow(clippy::absurd_extreme_comparisons)]
if data.len() < Self::MIN_SIZE {
return Err(ReadError::OutOfBounds);
}
Ok(Self { data })
}
}
#[derive(Clone)]
pub struct Subtable<'a> {
data: FontData<'a>,
}
#[allow(clippy::needless_lifetimes)]
impl<'a> Subtable<'a> {
pub const MIN_SIZE: usize = (u32::RAW_BYTE_LEN + u32::RAW_BYTE_LEN + u32::RAW_BYTE_LEN);
basic_table_impls!(impl_the_methods);
pub fn length(&self) -> u32 {
let range = self.length_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn coverage(&self) -> u32 {
let range = self.coverage_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn sub_feature_flags(&self) -> u32 {
let range = self.sub_feature_flags_byte_range();
self.data.read_at(range.start).ok().unwrap()
}
pub fn data(&self) -> &'a [u8] {
let range = self.data_byte_range();
self.data.read_array(range).ok().unwrap_or_default()
}
pub fn length_byte_range(&self) -> Range<usize> {
let start = 0;
start..start + u32::RAW_BYTE_LEN
}
pub fn coverage_byte_range(&self) -> Range<usize> {
let start = self.length_byte_range().end;
start..start + u32::RAW_BYTE_LEN
}
pub fn sub_feature_flags_byte_range(&self) -> Range<usize> {
let start = self.coverage_byte_range().end;
start..start + u32::RAW_BYTE_LEN
}
pub fn data_byte_range(&self) -> Range<usize> {
let start = self.sub_feature_flags_byte_range().end;
start..start + self.data.len().saturating_sub(start) / u8::RAW_BYTE_LEN * u8::RAW_BYTE_LEN
}
}
#[cfg(feature = "experimental_traverse")]
impl<'a> SomeTable<'a> for Subtable<'a> {
fn type_name(&self) -> &str {
"Subtable"
}
fn get_field(&self, idx: usize) -> Option<Field<'a>> {
match idx {
0usize => Some(Field::new("length", self.length())),
1usize => Some(Field::new("coverage", self.coverage())),
2usize => Some(Field::new("sub_feature_flags", self.sub_feature_flags())),
3usize => Some(Field::new("data", self.data())),
_ => None,
}
}
}
#[cfg(feature = "experimental_traverse")]
#[allow(clippy::needless_lifetimes)]
impl<'a> std::fmt::Debug for Subtable<'a> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
(self as &dyn SomeTable<'a>).fmt(f)
}
}
#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Copy, bytemuck :: AnyBitPattern)]
#[repr(C)]
#[repr(packed)]
pub struct ContextualEntryData {
pub mark_index: BigEndian<u16>,
pub current_index: BigEndian<u16>,
}
impl ContextualEntryData {
pub fn mark_index(&self) -> u16 {
self.mark_index.get()
}
pub fn current_index(&self) -> u16 {
self.current_index.get()
}
}
impl FixedSize for ContextualEntryData {
const RAW_BYTE_LEN: usize = u16::RAW_BYTE_LEN + u16::RAW_BYTE_LEN;
}
#[cfg(feature = "experimental_traverse")]
impl<'a> SomeRecord<'a> for ContextualEntryData {
fn traverse(self, data: FontData<'a>) -> RecordResolver<'a> {
RecordResolver {
name: "ContextualEntryData",
get_field: Box::new(move |idx, _data| match idx {
0usize => Some(Field::new("mark_index", self.mark_index())),
1usize => Some(Field::new("current_index", self.current_index())),
_ => None,
}),
data,
}
}
}
#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash, Copy, bytemuck :: AnyBitPattern)]
#[repr(C)]
#[repr(packed)]
pub struct InsertionEntryData {
pub current_insert_index: BigEndian<u16>,
pub marked_insert_index: BigEndian<u16>,
}
impl InsertionEntryData {
pub fn current_insert_index(&self) -> u16 {
self.current_insert_index.get()
}
pub fn marked_insert_index(&self) -> u16 {
self.marked_insert_index.get()
}
}
impl FixedSize for InsertionEntryData {
const RAW_BYTE_LEN: usize = u16::RAW_BYTE_LEN + u16::RAW_BYTE_LEN;
}
#[cfg(feature = "experimental_traverse")]
impl<'a> SomeRecord<'a> for InsertionEntryData {
fn traverse(self, data: FontData<'a>) -> RecordResolver<'a> {
RecordResolver {
name: "InsertionEntryData",
get_field: Box::new(move |idx, _data| match idx {
0usize => Some(Field::new(
"current_insert_index",
self.current_insert_index(),
)),
1usize => Some(Field::new(
"marked_insert_index",
self.marked_insert_index(),
)),
_ => None,
}),
data,
}
}
}