ttf_parser/tables/
name.rs#[cfg(feature = "std")]
use std::vec::Vec;
#[cfg(feature = "std")]
use std::string::String;
#[cfg(feature = "std")]
use crate::parser::LazyArray16;
use crate::parser::{Stream, FromData};
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;
}
#[derive(Clone, Copy, PartialEq, Debug)]
#[allow(missing_docs)]
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 => match encoding_id {
WINDOWS_SYMBOL_ENCODING_ID |
WINDOWS_UNICODE_BMP_ENCODING_ID => true,
_ => false,
}
_ => false,
}
}
#[derive(Clone, Copy)]
struct NameRecord {
platform_id: PlatformId,
encoding_id: u16,
language_id: u16,
name_id: u16,
length: u16,
offset: u16,
}
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::<u16>()?,
})
}
}
#[derive(Clone, Copy)]
pub struct Name<'a> {
data: NameRecord,
strings: &'a [u8],
}
impl<'a> Name<'a> {
pub fn platform_id(&self) -> PlatformId {
self.data.platform_id
}
pub fn encoding_id(&self) -> u16 {
self.data.encoding_id
}
pub fn language_id(&self) -> u16 {
self.data.language_id
}
pub fn name_id(&self) -> u16 {
self.data.name_id
}
pub fn name(&self) -> &'a [u8] {
let start = usize::from(self.data.offset);
let end = start + usize::from(self.data.length);
self.strings.get(start..end).unwrap_or(&[])
}
#[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()
}
}
#[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_ref().map(core::ops::Deref::deref)
.unwrap_or("unsupported encoding"))
.field("platform_id", &self.platform_id())
.field("encoding_id", &self.encoding_id())
.field("language_id", &self.language_id())
.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("name_id", &self.name_id())
.finish()
}
}
#[derive(Clone, Copy, Default)]
#[allow(missing_debug_implementations)]
pub struct Names<'a> {
names: &'a [u8],
storage: &'a [u8],
index: u16,
total: u16,
}
impl<'a> Names<'a> {
fn new(names: &'a [u8], storage: &'a [u8], total: u16) -> Self {
Names {
names,
storage,
index: 0,
total,
}
}
}
impl<'a> Iterator for Names<'a> {
type Item = Name<'a>;
fn next(&mut self) -> Option<Self::Item> {
if self.index < self.total {
let index = usize::from(self.index);
self.index += 1;
Some(Name {
data: Stream::read_at::<NameRecord>(self.names, NameRecord::SIZE * index)?,
strings: self.storage,
})
} else {
None
}
}
fn count(self) -> usize {
usize::from(self.total)
}
}
#[inline(never)]
pub(crate) fn parse(data: &[u8]) -> Option<Names> {
const LANG_TAG_RECORD_SIZE: u16 = 4;
let mut s = Stream::new(data);
let format: u16 = s.read()?;
let count: u16 = s.read()?;
s.skip::<u16>(); if format == 0 {
let names_data = s.read_bytes(NameRecord::SIZE * usize::from(count))?;
Some(Names::new(names_data, s.tail()?, count))
} else if format == 1 {
let lang_tag_count: u16 = s.read()?;
let lang_tag_len = lang_tag_count.checked_mul(LANG_TAG_RECORD_SIZE)?;
s.advance(usize::from(lang_tag_len)); let names_data = s.read_bytes(NameRecord::SIZE * usize::from(count))?;
Some(Names::new(names_data, s.tail()?, count))
} else {
None
}
}