#[cfg(feature = "std")]
use std::string::String;
#[cfg(feature = "std")]
use std::vec::Vec;
use crate::parser::{FromData, LazyArray16, Offset, Offset16, Stream};
use crate::Language;
pub mod name_id {
#![allow(missing_docs)]
pub const COPYRIGHT_NOTICE: u16 = 0;
pub const FAMILY: u16 = 1;
pub const SUBFAMILY: u16 = 2;
pub const UNIQUE_ID: u16 = 3;
pub const FULL_NAME: u16 = 4;
pub const VERSION: u16 = 5;
pub const POST_SCRIPT_NAME: u16 = 6;
pub const TRADEMARK: u16 = 7;
pub const MANUFACTURER: u16 = 8;
pub const DESIGNER: u16 = 9;
pub const DESCRIPTION: u16 = 10;
pub const VENDOR_URL: u16 = 11;
pub const DESIGNER_URL: u16 = 12;
pub const LICENSE: u16 = 13;
pub const LICENSE_URL: u16 = 14;
pub const TYPOGRAPHIC_FAMILY: u16 = 16;
pub const TYPOGRAPHIC_SUBFAMILY: u16 = 17;
pub const COMPATIBLE_FULL: u16 = 18;
pub const SAMPLE_TEXT: u16 = 19;
pub const POST_SCRIPT_CID: u16 = 20;
pub const WWS_FAMILY: u16 = 21;
pub const WWS_SUBFAMILY: u16 = 22;
pub const LIGHT_BACKGROUND_PALETTE: u16 = 23;
pub const DARK_BACKGROUND_PALETTE: u16 = 24;
pub const VARIATIONS_POST_SCRIPT_NAME_PREFIX: u16 = 25;
}
#[allow(missing_docs)]
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
pub enum PlatformId {
Unicode,
Macintosh,
Iso,
Windows,
Custom,
}
impl FromData for PlatformId {
const SIZE: usize = 2;
#[inline]
fn parse(data: &[u8]) -> Option<Self> {
match u16::parse(data)? {
0 => Some(PlatformId::Unicode),
1 => Some(PlatformId::Macintosh),
2 => Some(PlatformId::Iso),
3 => Some(PlatformId::Windows),
4 => Some(PlatformId::Custom),
_ => None,
}
}
}
#[inline]
fn is_unicode_encoding(platform_id: PlatformId, encoding_id: u16) -> bool {
const WINDOWS_SYMBOL_ENCODING_ID: u16 = 0;
const WINDOWS_UNICODE_BMP_ENCODING_ID: u16 = 1;
match platform_id {
PlatformId::Unicode => true,
PlatformId::Windows => matches!(
encoding_id,
WINDOWS_SYMBOL_ENCODING_ID | WINDOWS_UNICODE_BMP_ENCODING_ID
),
_ => false,
}
}
#[derive(Clone, Copy)]
struct NameRecord {
platform_id: PlatformId,
encoding_id: u16,
language_id: u16,
name_id: u16,
length: u16,
offset: Offset16,
}
impl FromData for NameRecord {
const SIZE: usize = 12;
#[inline]
fn parse(data: &[u8]) -> Option<Self> {
let mut s = Stream::new(data);
Some(NameRecord {
platform_id: s.read::<PlatformId>()?,
encoding_id: s.read::<u16>()?,
language_id: s.read::<u16>()?,
name_id: s.read::<u16>()?,
length: s.read::<u16>()?,
offset: s.read::<Offset16>()?,
})
}
}
#[derive(Clone, Copy)]
pub struct Name<'a> {
pub platform_id: PlatformId,
pub encoding_id: u16,
pub language_id: u16,
pub name_id: u16,
pub name: &'a [u8],
}
impl<'a> Name<'a> {
#[cfg(feature = "std")]
#[inline(never)]
pub fn to_string(&self) -> Option<String> {
if self.is_unicode() {
self.name_from_utf16_be()
} else {
None
}
}
#[inline]
pub fn is_unicode(&self) -> bool {
is_unicode_encoding(self.platform_id, self.encoding_id)
}
#[cfg(feature = "std")]
#[inline(never)]
fn name_from_utf16_be(&self) -> Option<String> {
let mut name: Vec<u16> = Vec::new();
for c in LazyArray16::<u16>::new(self.name) {
name.push(c);
}
String::from_utf16(&name).ok()
}
pub fn language(&self) -> Language {
if self.platform_id == PlatformId::Windows {
Language::windows_language(self.language_id)
} else if self.platform_id == PlatformId::Macintosh
&& self.encoding_id == 0
&& self.language_id == 0
{
Language::English_UnitedStates
} else {
Language::Unknown
}
}
}
#[cfg(feature = "std")]
impl<'a> core::fmt::Debug for Name<'a> {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
let name = self.to_string();
f.debug_struct("Name")
.field("name", &name.as_deref().unwrap_or("unsupported encoding"))
.field("platform_id", &self.platform_id)
.field("encoding_id", &self.encoding_id)
.field("language_id", &self.language_id)
.field("language", &self.language())
.field("name_id", &self.name_id)
.finish()
}
}
#[cfg(not(feature = "std"))]
impl<'a> core::fmt::Debug for Name<'a> {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
f.debug_struct("Name")
.field("name", &self.name)
.field("platform_id", &self.platform_id)
.field("encoding_id", &self.encoding_id)
.field("language_id", &self.language_id)
.field("language", &self.language())
.field("name_id", &self.name_id)
.finish()
}
}
#[derive(Clone, Copy, Default)]
pub struct Names<'a> {
records: LazyArray16<'a, NameRecord>,
storage: &'a [u8],
}
impl<'a> Names<'a> {
pub fn get(&self, index: u16) -> Option<Name<'a>> {
let record = self.records.get(index)?;
let name_start = record.offset.to_usize();
let name_end = name_start + usize::from(record.length);
let name = self.storage.get(name_start..name_end)?;
Some(Name {
platform_id: record.platform_id,
encoding_id: record.encoding_id,
language_id: record.language_id,
name_id: record.name_id,
name,
})
}
pub fn len(&self) -> u16 {
self.records.len()
}
pub fn is_empty(&self) -> bool {
self.records.is_empty()
}
}
impl core::fmt::Debug for Names<'_> {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
write!(f, "Names {{ ... }}")
}
}
impl<'a> IntoIterator for Names<'a> {
type Item = Name<'a>;
type IntoIter = NamesIter<'a>;
#[inline]
fn into_iter(self) -> Self::IntoIter {
NamesIter {
names: self,
index: 0,
}
}
}
#[derive(Clone, Copy)]
#[allow(missing_debug_implementations)]
pub struct NamesIter<'a> {
names: Names<'a>,
index: u16,
}
impl<'a> Iterator for NamesIter<'a> {
type Item = Name<'a>;
fn next(&mut self) -> Option<Self::Item> {
if self.index < self.names.len() {
self.index += 1;
self.names.get(self.index - 1)
} else {
None
}
}
#[inline]
fn count(self) -> usize {
usize::from(self.names.len().saturating_sub(self.index))
}
}
#[derive(Clone, Copy, Default, Debug)]
pub struct Table<'a> {
pub names: Names<'a>,
}
impl<'a> Table<'a> {
pub fn parse(data: &'a [u8]) -> Option<Self> {
const LANG_TAG_RECORD_SIZE: u16 = 4;
let mut s = Stream::new(data);
let version = s.read::<u16>()?;
let count = s.read::<u16>()?;
let storage_offset = s.read::<Offset16>()?.to_usize();
if version == 0 {
} else if version == 1 {
let lang_tag_count = s.read::<u16>()?;
let lang_tag_len = lang_tag_count.checked_mul(LANG_TAG_RECORD_SIZE)?;
s.advance(usize::from(lang_tag_len)); } else {
return None;
}
let records = s.read_array16::<NameRecord>(count)?;
if s.offset() < storage_offset {
s.advance(storage_offset - s.offset());
}
let storage = s.tail()?;
Some(Table {
names: Names { records, storage },
})
}
}