1 | // font-kit/src/sources/mem.rs |
2 | // |
3 | // Copyright © 2018 The Pathfinder Project Developers. |
4 | // |
5 | // Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or |
6 | // http://www.apache.org/licenses/LICENSE-2.0> or the MIT license |
7 | // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your |
8 | // option. This file may not be copied, modified, or distributed |
9 | // except according to those terms. |
10 | |
11 | //! A source that keeps fonts in memory. |
12 | |
13 | use crate::error::{FontLoadingError, SelectionError}; |
14 | use crate::family_handle::FamilyHandle; |
15 | use crate::family_name::FamilyName; |
16 | use crate::font::Font; |
17 | use crate::handle::Handle; |
18 | use crate::properties::Properties; |
19 | use crate::source::Source; |
20 | use std::any::Any; |
21 | |
22 | /// A source that keeps fonts in memory. |
23 | #[allow (missing_debug_implementations)] |
24 | pub struct MemSource { |
25 | families: Vec<FamilyEntry>, |
26 | } |
27 | |
28 | impl MemSource { |
29 | /// Creates a new empty memory source. |
30 | pub fn empty() -> MemSource { |
31 | MemSource { families: vec![] } |
32 | } |
33 | |
34 | /// Creates a new memory source that contains the given set of font handles. |
35 | /// |
36 | /// The fonts referenced by the handles are eagerly loaded into memory. |
37 | pub fn from_fonts<I>(fonts: I) -> Result<MemSource, FontLoadingError> |
38 | where |
39 | I: Iterator<Item = Handle>, |
40 | { |
41 | let mut families = vec![]; |
42 | for handle in fonts { |
43 | add_font(handle, &mut families)?; |
44 | } |
45 | families.sort_by(|a, b| a.family_name.cmp(&b.family_name)); |
46 | Ok(MemSource { families }) |
47 | } |
48 | |
49 | /// Add an existing font handle to a `MemSource`. |
50 | /// |
51 | /// Returns the font that was just added. |
52 | /// |
53 | /// Note that adding fonts to an existing `MemSource` is slower than creating a new one from a |
54 | /// `Handle` iterator, since this method sorts after every addition, rather than once at the |
55 | /// end. |
56 | pub fn add_font(&mut self, handle: Handle) -> Result<Font, FontLoadingError> { |
57 | let font = add_font(handle, &mut self.families)?; |
58 | self.families |
59 | .sort_by(|a, b| a.family_name.cmp(&b.family_name)); |
60 | Ok(font) |
61 | } |
62 | |
63 | /// Add a number of existing font handles to a `MemSource`. |
64 | /// |
65 | /// Note that adding fonts to an existing `MemSource` is slower than creating a new one from a |
66 | /// `Handle` iterator, because extra unnecessary sorting with occur with every call to this |
67 | /// method. |
68 | pub fn add_fonts( |
69 | &mut self, |
70 | handles: impl Iterator<Item = Handle>, |
71 | ) -> Result<(), FontLoadingError> { |
72 | for handle in handles { |
73 | add_font(handle, &mut self.families)?; |
74 | } |
75 | self.families |
76 | .sort_by(|a, b| a.family_name.cmp(&b.family_name)); |
77 | Ok(()) |
78 | } |
79 | |
80 | /// Returns paths of all fonts installed on the system. |
81 | pub fn all_fonts(&self) -> Result<Vec<Handle>, SelectionError> { |
82 | Ok(self |
83 | .families |
84 | .iter() |
85 | .map(|family| family.font.clone()) |
86 | .collect()) |
87 | } |
88 | |
89 | /// Returns the names of all families installed on the system. |
90 | pub fn all_families(&self) -> Result<Vec<String>, SelectionError> { |
91 | let mut families = vec![]; |
92 | for family in &self.families { |
93 | if families.last() == Some(&family.family_name) { |
94 | continue; |
95 | } |
96 | |
97 | families.push(family.family_name.clone()); |
98 | } |
99 | |
100 | Ok(families) |
101 | } |
102 | |
103 | /// Looks up a font family by name and returns the handles of all the fonts in that family. |
104 | /// |
105 | /// FIXME(pcwalton): Case-insensitive comparison. |
106 | pub fn select_family_by_name(&self, family_name: &str) -> Result<FamilyHandle, SelectionError> { |
107 | let mut first_family_index = self |
108 | .families |
109 | .binary_search_by(|family| (&*family.family_name).cmp(family_name)) |
110 | .map_err(|_| SelectionError::NotFound)?; |
111 | |
112 | while first_family_index > 0 |
113 | && self.families[first_family_index - 1].family_name == family_name |
114 | { |
115 | first_family_index -= 1 |
116 | } |
117 | let mut last_family_index = first_family_index; |
118 | while last_family_index + 1 < self.families.len() |
119 | && self.families[last_family_index + 1].family_name == family_name |
120 | { |
121 | last_family_index += 1 |
122 | } |
123 | |
124 | let families = &self.families[first_family_index..(last_family_index + 1)]; |
125 | Ok(FamilyHandle::from_font_handles( |
126 | families.iter().map(|family| family.font.clone()), |
127 | )) |
128 | } |
129 | |
130 | /// Selects a font by PostScript name, which should be a unique identifier. |
131 | /// |
132 | /// The default implementation, which is used by the DirectWrite and the filesystem backends, |
133 | /// does a brute-force search of installed fonts to find the one that matches. |
134 | pub fn select_by_postscript_name( |
135 | &self, |
136 | postscript_name: &str, |
137 | ) -> Result<Handle, SelectionError> { |
138 | self.families |
139 | .iter() |
140 | .filter(|family_entry| family_entry.postscript_name == postscript_name) |
141 | .map(|family_entry| family_entry.font.clone()) |
142 | .next() |
143 | .ok_or(SelectionError::NotFound) |
144 | } |
145 | |
146 | /// Performs font matching according to the CSS Fonts Level 3 specification and returns the |
147 | /// handle. |
148 | #[inline ] |
149 | pub fn select_best_match( |
150 | &self, |
151 | family_names: &[FamilyName], |
152 | properties: &Properties, |
153 | ) -> Result<Handle, SelectionError> { |
154 | <Self as Source>::select_best_match(self, family_names, properties) |
155 | } |
156 | } |
157 | |
158 | impl Source for MemSource { |
159 | #[inline ] |
160 | fn all_fonts(&self) -> Result<Vec<Handle>, SelectionError> { |
161 | self.all_fonts() |
162 | } |
163 | |
164 | #[inline ] |
165 | fn all_families(&self) -> Result<Vec<String>, SelectionError> { |
166 | self.all_families() |
167 | } |
168 | |
169 | fn select_family_by_name(&self, family_name: &str) -> Result<FamilyHandle, SelectionError> { |
170 | self.select_family_by_name(family_name) |
171 | } |
172 | |
173 | fn select_by_postscript_name(&self, postscript_name: &str) -> Result<Handle, SelectionError> { |
174 | self.select_by_postscript_name(postscript_name) |
175 | } |
176 | |
177 | #[inline ] |
178 | fn as_any(&self) -> &dyn Any { |
179 | self |
180 | } |
181 | |
182 | #[inline ] |
183 | fn as_mut_any(&mut self) -> &mut dyn Any { |
184 | self |
185 | } |
186 | } |
187 | |
188 | /// Adds a font, but doesn't sort. Returns the font that was created to check for validity. |
189 | fn add_font(handle: Handle, families: &mut Vec<FamilyEntry>) -> Result<Font, FontLoadingError> { |
190 | let font: Font = Font::from_handle(&handle)?; |
191 | if let Some(postscript_name: String) = font.postscript_name() { |
192 | families.push(FamilyEntry { |
193 | family_name: font.family_name(), |
194 | postscript_name: postscript_name, |
195 | font: handle, |
196 | }) |
197 | } |
198 | Ok(font) |
199 | } |
200 | |
201 | struct FamilyEntry { |
202 | family_name: String, |
203 | postscript_name: String, |
204 | font: Handle, |
205 | } |
206 | |