summaryrefslogtreecommitdiff
path: root/src/style
diff options
context:
space:
mode:
authorLaurenz <laurmaedje@gmail.com>2021-12-07 16:36:39 +0100
committerLaurenz <laurmaedje@gmail.com>2021-12-07 16:36:39 +0100
commit40b87d4066fe85cb3fde6cf84cd60d748273ae25 (patch)
tree792b2e5edd8e72649d9fdcac24dc07620bf0f15c /src/style
parent26bdc1f0f6fe8113d7fcfb4d5aca46aa5238ccd8 (diff)
Set Rules Episode II: Attack of the properties
Diffstat (limited to 'src/style')
-rw-r--r--src/style/mod.rs419
-rw-r--r--src/style/paper.rs233
2 files changed, 0 insertions, 652 deletions
diff --git a/src/style/mod.rs b/src/style/mod.rs
deleted file mode 100644
index 45dbeb54..00000000
--- a/src/style/mod.rs
+++ /dev/null
@@ -1,419 +0,0 @@
-//! Style properties.
-
-mod paper;
-
-pub use paper::*;
-
-use std::fmt::{self, Debug, Formatter};
-use std::rc::Rc;
-
-use ttf_parser::Tag;
-
-use crate::eval::Smart;
-use crate::font::*;
-use crate::geom::*;
-use crate::util::EcoString;
-
-/// Defines a set of properties a template can be instantiated with.
-#[derive(Debug, Clone, Eq, PartialEq, Hash)]
-pub struct Style {
- /// The page settings.
- pub page: Rc<PageStyle>,
- /// The paragraph settings.
- pub par: Rc<ParStyle>,
- /// The current text settings.
- pub text: Rc<TextStyle>,
-}
-
-impl Style {
- /// Access the `page` style mutably.
- pub fn page_mut(&mut self) -> &mut PageStyle {
- Rc::make_mut(&mut self.page)
- }
-
- /// Access the `par` style mutably.
- pub fn par_mut(&mut self) -> &mut ParStyle {
- Rc::make_mut(&mut self.par)
- }
-
- /// Access the `text` style mutably.
- pub fn text_mut(&mut self) -> &mut TextStyle {
- Rc::make_mut(&mut self.text)
- }
-
- /// The resolved line spacing.
- pub fn leading(&self) -> Length {
- self.par.leading.resolve(self.text.size)
- }
-
- /// The resolved paragraph spacing.
- pub fn par_spacing(&self) -> Length {
- self.par.spacing.resolve(self.text.size)
- }
-}
-
-impl Default for Style {
- fn default() -> Self {
- Self {
- page: Rc::new(PageStyle::default()),
- par: Rc::new(ParStyle::default()),
- text: Rc::new(TextStyle::default()),
- }
- }
-}
-
-/// Defines style properties of pages.
-#[derive(Debug, Clone, Eq, PartialEq, Hash)]
-pub struct PageStyle {
- /// The class of this page.
- pub class: PaperClass,
- /// The width and height of the page.
- pub size: Size,
- /// The amount of white space on each side of the page. If a side is set to
- /// `None`, the default for the paper class is used.
- pub margins: Sides<Smart<Linear>>,
- /// The background fill of the page.
- pub fill: Option<Paint>,
-}
-
-impl PageStyle {
- /// The resolved margins.
- pub fn margins(&self) -> Sides<Linear> {
- let default = self.class.default_margins();
- Sides {
- left: self.margins.left.unwrap_or(default.left),
- top: self.margins.top.unwrap_or(default.top),
- right: self.margins.right.unwrap_or(default.right),
- bottom: self.margins.bottom.unwrap_or(default.bottom),
- }
- }
-}
-
-impl Default for PageStyle {
- fn default() -> Self {
- let paper = Paper::A4;
- Self {
- class: paper.class(),
- size: paper.size(),
- margins: Sides::splat(Smart::Auto),
- fill: None,
- }
- }
-}
-
-/// Defines style properties of paragraphs.
-#[derive(Debug, Clone, Eq, PartialEq, Hash)]
-pub struct ParStyle {
- /// The direction for text and inline objects.
- pub dir: Dir,
- /// How to align text and inline objects in their line.
- pub align: Align,
- /// The spacing between lines (dependent on scaled font size).
- pub leading: Linear,
- /// The spacing between paragraphs (dependent on scaled font size).
- pub spacing: Linear,
-}
-
-impl Default for ParStyle {
- fn default() -> Self {
- Self {
- dir: Dir::LTR,
- align: Align::Left,
- leading: Relative::new(0.65).into(),
- spacing: Relative::new(1.2).into(),
- }
- }
-}
-
-/// Defines style properties of text.
-#[derive(Debug, Clone, Eq, PartialEq, Hash)]
-pub struct TextStyle {
- /// The font size.
- pub size: Length,
- /// The selected font variant (the final variant also depends on `strong`
- /// and `emph`).
- pub variant: FontVariant,
- /// The top end of the text bounding box.
- pub top_edge: VerticalFontMetric,
- /// The bottom end of the text bounding box.
- pub bottom_edge: VerticalFontMetric,
- /// Glyph color.
- pub fill: Paint,
- /// A list of font families with generic class definitions (the final
- /// family list also depends on `monospace`).
- pub families: Rc<FamilyStyle>,
- /// OpenType features.
- pub features: Rc<FontFeatures>,
- /// The amount of space that should be added between character.
- pub tracking: Em,
- /// Whether 300 extra font weight should be added to what is defined by the
- /// `variant`.
- pub strong: bool,
- /// Whether the the font style defined by the `variant` should be inverted.
- pub emph: bool,
- /// Whether a monospace font should be preferred.
- pub monospace: bool,
- /// Whether font fallback to a base list should occur.
- pub fallback: bool,
-}
-
-impl TextStyle {
- /// The resolved variant with `strong` and `emph` factored in.
- pub fn variant(&self) -> FontVariant {
- let mut variant = self.variant;
-
- if self.strong {
- variant.weight = variant.weight.thicken(300);
- }
-
- if self.emph {
- variant.style = match variant.style {
- FontStyle::Normal => FontStyle::Italic,
- FontStyle::Italic => FontStyle::Normal,
- FontStyle::Oblique => FontStyle::Normal,
- }
- }
-
- variant
- }
-
- /// The resolved family iterator.
- pub fn families(&self) -> impl Iterator<Item = &str> + Clone {
- let head = if self.monospace {
- self.families.monospace.as_slice()
- } else {
- &[]
- };
-
- let core = self.families.list.iter().flat_map(move |family| {
- match family {
- FontFamily::Named(name) => std::slice::from_ref(name),
- FontFamily::Serif => &self.families.serif,
- FontFamily::SansSerif => &self.families.sans_serif,
- FontFamily::Monospace => &self.families.monospace,
- }
- });
-
- let tail = if self.fallback {
- self.families.base.as_slice()
- } else {
- &[]
- };
-
- head.iter().chain(core).chain(tail).map(EcoString::as_str)
- }
-
- /// Access the `families` style mutably.
- pub fn families_mut(&mut self) -> &mut FamilyStyle {
- Rc::make_mut(&mut self.families)
- }
-
- /// Access the font `features` mutably.
- pub fn features_mut(&mut self) -> &mut FontFeatures {
- Rc::make_mut(&mut self.features)
- }
-}
-
-impl Default for TextStyle {
- fn default() -> Self {
- Self {
- size: Length::pt(11.0),
- variant: FontVariant {
- style: FontStyle::Normal,
- weight: FontWeight::REGULAR,
- stretch: FontStretch::NORMAL,
- },
- top_edge: VerticalFontMetric::CapHeight,
- bottom_edge: VerticalFontMetric::Baseline,
- fill: RgbaColor::BLACK.into(),
- families: Rc::new(FamilyStyle::default()),
- features: Rc::new(FontFeatures::default()),
- tracking: Em::zero(),
- strong: false,
- emph: false,
- monospace: false,
- fallback: true,
- }
- }
-}
-
-/// Font list with family definitions.
-#[derive(Debug, Clone, Eq, PartialEq, Hash)]
-pub struct FamilyStyle {
- /// The user-defined list of font families.
- pub list: Vec<FontFamily>,
- /// Definition of serif font families.
- pub serif: Vec<EcoString>,
- /// Definition of sans-serif font families.
- pub sans_serif: Vec<EcoString>,
- /// Definition of monospace font families used for raw text.
- pub monospace: Vec<EcoString>,
- /// Base fonts that are tried as last resort.
- pub base: Vec<EcoString>,
-}
-
-impl Default for FamilyStyle {
- fn default() -> Self {
- Self {
- list: vec![FontFamily::SansSerif],
- serif: vec!["ibm plex serif".into()],
- sans_serif: vec!["ibm plex sans".into()],
- monospace: vec!["ibm plex mono".into()],
- base: vec![
- "ibm plex sans".into(),
- "latin modern math".into(),
- "twitter color emoji".into(),
- ],
- }
- }
-}
-
-/// A generic or named font family.
-#[derive(Clone, Eq, PartialEq, Hash)]
-pub enum FontFamily {
- /// A family that has "serifs", small strokes attached to letters.
- Serif,
- /// A family in which glyphs do not have "serifs", small attached strokes.
- SansSerif,
- /// A family in which (almost) all glyphs are of equal width.
- Monospace,
- /// A specific family with a name.
- Named(EcoString),
-}
-
-impl Debug for FontFamily {
- fn fmt(&self, f: &mut Formatter) -> fmt::Result {
- f.pad(match self {
- Self::Serif => "serif",
- Self::SansSerif => "sans-serif",
- Self::Monospace => "monospace",
- Self::Named(s) => s,
- })
- }
-}
-
-/// Whether various kinds of ligatures should appear.
-#[derive(Debug, Clone, Eq, PartialEq, Hash)]
-pub struct FontFeatures {
- /// Whether to apply kerning ("kern").
- pub kerning: bool,
- /// Whether the text should use small caps. ("smcp")
- pub smallcaps: bool,
- /// Whether to apply stylistic alternates. ("salt")
- pub alternates: bool,
- /// Which stylistic set to apply. ("ss01" - "ss20")
- pub stylistic_set: Option<StylisticSet>,
- /// Configuration of ligature features.
- pub ligatures: LigatureFeatures,
- /// Configuration of numbers features.
- pub numbers: NumberFeatures,
- /// Raw OpenType features to apply.
- pub raw: Vec<(Tag, u32)>,
-}
-
-impl Default for FontFeatures {
- fn default() -> Self {
- Self {
- kerning: true,
- smallcaps: false,
- alternates: false,
- stylistic_set: None,
- ligatures: LigatureFeatures::default(),
- numbers: NumberFeatures::default(),
- raw: vec![],
- }
- }
-}
-
-/// A stylistic set in a font face.
-#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
-pub struct StylisticSet(u8);
-
-impl StylisticSet {
- /// Creates a new set, clamping to 1-20.
- pub fn new(index: u8) -> Self {
- Self(index.clamp(1, 20))
- }
-
- /// Get the value, guaranteed to be 1-20.
- pub fn get(self) -> u8 {
- self.0
- }
-}
-
-/// Whether various kinds of ligatures should appear.
-#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
-pub struct LigatureFeatures {
- /// Standard ligatures. ("liga", "clig")
- pub standard: bool,
- /// Ligatures that should be used sparringly. ("dlig")
- pub discretionary: bool,
- /// Historical ligatures. ("hlig")
- pub historical: bool,
-}
-
-impl Default for LigatureFeatures {
- fn default() -> Self {
- Self {
- standard: true,
- discretionary: false,
- historical: false,
- }
- }
-}
-
-/// Defines the style of numbers.
-#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
-pub struct NumberFeatures {
- /// Whether to use lining or old-style numbers.
- pub type_: Smart<NumberType>,
- /// Whether to use proportional or tabular numbers.
- pub width: Smart<NumberWidth>,
- /// How to position numbers vertically.
- pub position: NumberPosition,
- /// Whether to have a slash through the zero glyph. ("zero")
- pub slashed_zero: bool,
- /// Whether to convert fractions. ("frac")
- pub fractions: bool,
-}
-
-impl Default for NumberFeatures {
- fn default() -> Self {
- Self {
- type_: Smart::Auto,
- width: Smart::Auto,
- position: NumberPosition::Normal,
- slashed_zero: false,
- fractions: false,
- }
- }
-}
-
-/// Which kind of numbers / figures to select.
-#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
-pub enum NumberType {
- /// Numbers that fit well with capital text. ("lnum")
- Lining,
- /// Numbers that fit well into flow of upper- and lowercase text. ("onum")
- OldStyle,
-}
-
-/// The width of numbers / figures.
-#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
-pub enum NumberWidth {
- /// Number widths are glyph specific. ("pnum")
- Proportional,
- /// All numbers are of equal width / monospaced. ("tnum")
- Tabular,
-}
-
-/// How to position numbers.
-#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
-pub enum NumberPosition {
- /// Numbers are positioned on the same baseline as text.
- Normal,
- /// Numbers are smaller and placed at the bottom. ("subs")
- Subscript,
- /// Numbers are smaller and placed at the top. ("sups")
- Superscript,
-}
diff --git a/src/style/paper.rs b/src/style/paper.rs
deleted file mode 100644
index 252d293d..00000000
--- a/src/style/paper.rs
+++ /dev/null
@@ -1,233 +0,0 @@
-use crate::geom::{Length, Linear, Relative, Sides, Size};
-
-/// Specification of a paper.
-#[derive(Debug, Copy, Clone)]
-pub struct Paper {
- /// The broad class this paper belongs to.
- class: PaperClass,
- /// The width of the paper in millimeters.
- width: f64,
- /// The height of the paper in millimeters.
- height: f64,
-}
-
-/// Defines default margins for a class of related papers.
-#[derive(Debug, Copy, Clone, Eq, PartialEq, Hash)]
-pub enum PaperClass {
- Custom,
- Base,
- US,
- Newspaper,
- Book,
-}
-
-impl PaperClass {
- /// The default margins for this page class.
- pub fn default_margins(self) -> Sides<Linear> {
- let f = |r| Relative::new(r).into();
- let s = |l, t, r, b| Sides::new(f(l), f(t), f(r), f(b));
- match self {
- Self::Custom => s(0.1190, 0.0842, 0.1190, 0.0842),
- Self::Base => s(0.1190, 0.0842, 0.1190, 0.0842),
- Self::US => s(0.1760, 0.1092, 0.1760, 0.0910),
- Self::Newspaper => s(0.0455, 0.0587, 0.0455, 0.0294),
- Self::Book => s(0.1200, 0.0852, 0.1500, 0.0965),
- }
- }
-}
-
-macro_rules! papers {
- ($(($var:ident: $class:ident, $width:expr, $height: expr, $($pats:tt)*))*) => {
- impl Paper {
- /// Parse a paper from its name.
- ///
- /// Both lower and upper case are fine.
- pub fn from_name(name: &str) -> Option<Self> {
- match name.to_lowercase().as_str() {
- $($($pats)* => Some(Self::$var),)*
- _ => None,
- }
- }
-
- /// The class of the paper.
- pub fn class(self) -> PaperClass {
- self.class
- }
-
- /// The size of the paper.
- pub fn size(self) -> Size {
- Size::new(Length::mm(self.width), Length::mm(self.height))
- }
- }
-
- /// Predefined papers.
- ///
- /// Each paper is parsable from its name in kebab-case.
- impl Paper {
- $(papers!(@$var, stringify!($($pats)*), $class, $width, $height);)*
- }
- };
-
- (@$var:ident, $names:expr, $class:ident, $width:expr, $height:expr) => {
- pub const $var: Self = Self {
- class: PaperClass::$class,
- width: $width,
- height: $height,
- };
- };
-}
-
-// All paper sizes in mm.
-//
-// Resources:
-// - https://papersizes.io/
-// - https://en.wikipedia.org/wiki/Paper_size
-// - https://www.theedkins.co.uk/jo/units/oldunits/print.htm
-// - https://vintagepaper.co/blogs/news/traditional-paper-sizes
-papers! {
- // ---------------------------------------------------------------------- //
- // ISO 216 A Series
- (A0: Base, 841.0, 1189.0, "a0")
- (A1: Base, 594.0, 841.0, "a1")
- (A2: Base, 420.0, 594.0, "a2")
- (A3: Base, 297.0, 420.0, "a3")
- (A4: Base, 210.0, 297.0, "a4")
- (A5: Base, 148.0, 210.0, "a5")
- (A6: Book, 105.0, 148.0, "a6")
- (A7: Base, 74.0, 105.0, "a7")
- (A8: Base, 52.0, 74.0, "a8")
- (A9: Base, 37.0, 52.0, "a9")
- (A10: Base, 26.0, 37.0, "a10")
- (A11: Base, 18.0, 26.0, "a11")
-
- // ISO 216 B Series
- (ISO_B1: Base, 707.0, 1000.0, "iso-b1")
- (ISO_B2: Base, 500.0, 707.0, "iso-b2")
- (ISO_B3: Base, 353.0, 500.0, "iso-b3")
- (ISO_B4: Base, 250.0, 353.0, "iso-b4")
- (ISO_B5: Book, 176.0, 250.0, "iso-b5")
- (ISO_B6: Book, 125.0, 176.0, "iso-b6")
- (ISO_B7: Base, 88.0, 125.0, "iso-b7")
- (ISO_B8: Base, 62.0, 88.0, "iso-b8")
-
- // ISO 216 C Series
- (ISO_C3: Base, 324.0, 458.0, "iso-c3")
- (ISO_C4: Base, 229.0, 324.0, "iso-c4")
- (ISO_C5: Base, 162.0, 229.0, "iso-c5")
- (ISO_C6: Base, 114.0, 162.0, "iso-c6")
- (ISO_C7: Base, 81.0, 114.0, "iso-c7")
- (ISO_C8: Base, 57.0, 81.0, "iso-c8")
-
- // DIN D Series (extension to ISO)
- (DIN_D3: Base, 272.0, 385.0, "din-d3")
- (DIN_D4: Base, 192.0, 272.0, "din-d4")
- (DIN_D5: Base, 136.0, 192.0, "din-d5")
- (DIN_D6: Base, 96.0, 136.0, "din-d6")
- (DIN_D7: Base, 68.0, 96.0, "din-d7")
- (DIN_D8: Base, 48.0, 68.0, "din-d8")
-
- // SIS (used in academia)
- (SIS_G5: Base, 169.0, 239.0, "sis-g5")
- (SIS_E5: Base, 115.0, 220.0, "sis-e5")
-
- // ANSI Extensions
- (ANSI_A: Base, 216.0, 279.0, "ansi-a")
- (ANSI_B: Base, 279.0, 432.0, "ansi-b")
- (ANSI_C: Base, 432.0, 559.0, "ansi-c")
- (ANSI_D: Base, 559.0, 864.0, "ansi-d")
- (ANSI_E: Base, 864.0, 1118.0, "ansi-e")
-
- // ANSI Architectural Paper
- (ARCH_A: Base, 229.0, 305.0, "arch-a")
- (ARCH_B: Base, 305.0, 457.0, "arch-b")
- (ARCH_C: Base, 457.0, 610.0, "arch-c")
- (ARCH_D: Base, 610.0, 914.0, "arch-d")
- (ARCH_E1: Base, 762.0, 1067.0, "arch-e1")
- (ARCH_E: Base, 914.0, 1219.0, "arch-e")
-
- // JIS B Series
- (JIS_B0: Base, 1030.0, 1456.0, "jis-b0")
- (JIS_B1: Base, 728.0, 1030.0, "jis-b1")
- (JIS_B2: Base, 515.0, 728.0, "jis-b2")
- (JIS_B3: Base, 364.0, 515.0, "jis-b3")
- (JIS_B4: Base, 257.0, 364.0, "jis-b4")
- (JIS_B5: Base, 182.0, 257.0, "jis-b5")
- (JIS_B6: Base, 128.0, 182.0, "jis-b6")
- (JIS_B7: Base, 91.0, 128.0, "jis-b7")
- (JIS_B8: Base, 64.0, 91.0, "jis-b8")
- (JIS_B9: Base, 45.0, 64.0, "jis-b9")
- (JIS_B10: Base, 32.0, 45.0, "jis-b10")
- (JIS_B11: Base, 22.0, 32.0, "jis-b11")
-
- // SAC D Series
- (SAC_D0: Base, 764.0, 1064.0, "sac-d0")
- (SAC_D1: Base, 532.0, 760.0, "sac-d1")
- (SAC_D2: Base, 380.0, 528.0, "sac-d2")
- (SAC_D3: Base, 264.0, 376.0, "sac-d3")
- (SAC_D4: Base, 188.0, 260.0, "sac-d4")
- (SAC_D5: Base, 130.0, 184.0, "sac-d5")
- (SAC_D6: Base, 92.0, 126.0, "sac-d6")
-
- // ISO 7810 ID
- (ISO_ID_1: Base, 85.6, 53.98, "iso-id-1")
- (ISO_ID_2: Base, 74.0, 105.0, "iso-id-2")
- (ISO_ID_3: Base, 88.0, 125.0, "iso-id-3")
-
- // ---------------------------------------------------------------------- //
- // Asia
- (ASIA_F4: Base, 210.0, 330.0, "asia-f4")
-
- // Japan
- (JP_SHIROKU_BAN_4: Base, 264.0, 379.0, "jp-shiroku-ban-4")
- (JP_SHIROKU_BAN_5: Base, 189.0, 262.0, "jp-shiroku-ban-5")
- (JP_SHIROKU_BAN_6: Base, 127.0, 188.0, "jp-shiroku-ban-6")
- (JP_KIKU_4: Base, 227.0, 306.0, "jp-kiku-4")
- (JP_KIKU_5: Base, 151.0, 227.0, "jp-kiku-5")
- (JP_BUSINESS_CARD: Base, 91.0, 55.0, "jp-business-card")
-
- // China
- (CN_BUSINESS_CARD: Base, 90.0, 54.0, "cn-business-card")
-
- // Europe
- (EU_BUSINESS_CARD: Base, 85.0, 55.0, "eu-business-card")
-
- // French Traditional (AFNOR)
- (FR_TELLIERE: Base, 340.0, 440.0, "fr-tellière")
- (FR_COURONNE_ECRITURE: Base, 360.0, 460.0, "fr-couronne-écriture")
- (FR_COURONNE_EDITION: Base, 370.0, 470.0, "fr-couronne-édition")
- (FR_RAISIN: Base, 500.0, 650.0, "fr-raisin")
- (FR_CARRE: Base, 450.0, 560.0, "fr-carré")
- (FR_JESUS: Base, 560.0, 760.0, "fr-jésus")
-
- // United Kingdom Imperial
- (UK_BRIEF: Base, 406.4, 342.9, "uk-brief")
- (UK_DRAFT: Base, 254.0, 406.4, "uk-draft")
- (UK_FOOLSCAP: Base, 203.2, 330.2, "uk-foolscap")
- (UK_QUARTO: Base, 203.2, 254.0, "uk-quarto")
- (UK_CROWN: Base, 508.0, 381.0, "uk-crown")
- (UK_BOOK_A: Book, 111.0, 178.0, "uk-book-a")
- (UK_BOOK_B: Book, 129.0, 198.0, "uk-book-b")
-
- // Unites States
- (US_LETTER: US, 215.9, 279.4, "us-letter")
- (US_LEGAL: US, 215.9, 355.6, "us-legal")
- (US_TABLOID: US, 279.4, 431.8, "us-tabloid")
- (US_EXECUTIVE: US, 184.15, 266.7, "us-executive")
- (US_FOOLSCAP_FOLIO: US, 215.9, 342.9, "us-foolscap-folio")
- (US_STATEMENT: US, 139.7, 215.9, "us-statement")
- (US_LEDGER: US, 431.8, 279.4, "us-ledger")
- (US_OFICIO: US, 215.9, 340.36, "us-oficio")
- (US_GOV_LETTER: US, 203.2, 266.7, "us-gov-letter")
- (US_GOV_LEGAL: US, 215.9, 330.2, "us-gov-legal")
- (US_BUSINESS_CARD: Base, 88.9, 50.8, "us-business-card")
- (US_DIGEST: Book, 139.7, 215.9, "us-digest")
- (US_TRADE: Book, 152.4, 228.6, "us-trade")
-
- // ---------------------------------------------------------------------- //
- // Other
- (NEWSPAPER_COMPACT: Newspaper, 280.0, 430.0, "newspaper-compact")
- (NEWSPAPER_BERLINER: Newspaper, 315.0, 470.0, "newspaper-berliner")
- (NEWSPAPER_BROADSHEET: Newspaper, 381.0, 578.0, "newspaper-broadsheet")
- (PRESENTATION_16_9: Base, 297.0, 167.0625, "presentation-16-9")
- (PRESENTATION_4_3: Base, 280.0, 210.0, "presentation-4-3")
-}