Checkpoint
This commit is contained in:
parent
7885eaf974
commit
83dae46ec6
17 changed files with 2701 additions and 1193 deletions
44
Cargo.lock
generated
44
Cargo.lock
generated
|
@ -983,7 +983,7 @@ dependencies = [
|
||||||
"collections",
|
"collections",
|
||||||
"editor",
|
"editor",
|
||||||
"gpui",
|
"gpui",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"language",
|
"language",
|
||||||
"outline",
|
"outline",
|
||||||
"project",
|
"project",
|
||||||
|
@ -1960,7 +1960,7 @@ dependencies = [
|
||||||
"cranelift-codegen",
|
"cranelift-codegen",
|
||||||
"cranelift-entity",
|
"cranelift-entity",
|
||||||
"cranelift-frontend",
|
"cranelift-frontend",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"log",
|
"log",
|
||||||
"smallvec",
|
"smallvec",
|
||||||
"wasmparser",
|
"wasmparser",
|
||||||
|
@ -2392,7 +2392,7 @@ dependencies = [
|
||||||
"git",
|
"git",
|
||||||
"gpui",
|
"gpui",
|
||||||
"indoc",
|
"indoc",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"language",
|
"language",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"log",
|
"log",
|
||||||
|
@ -3273,7 +3273,7 @@ dependencies = [
|
||||||
"futures 0.3.28",
|
"futures 0.3.28",
|
||||||
"gpui_macros",
|
"gpui_macros",
|
||||||
"image",
|
"image",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"log",
|
"log",
|
||||||
"media",
|
"media",
|
||||||
|
@ -3365,7 +3365,7 @@ dependencies = [
|
||||||
"futures 0.3.28",
|
"futures 0.3.28",
|
||||||
"gpui_macros",
|
"gpui_macros",
|
||||||
"image",
|
"image",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"log",
|
"log",
|
||||||
"media",
|
"media",
|
||||||
|
@ -3374,6 +3374,7 @@ dependencies = [
|
||||||
"ordered-float",
|
"ordered-float",
|
||||||
"parking",
|
"parking",
|
||||||
"parking_lot 0.11.2",
|
"parking_lot 0.11.2",
|
||||||
|
"pathfinder_geometry",
|
||||||
"plane-split",
|
"plane-split",
|
||||||
"png",
|
"png",
|
||||||
"postage",
|
"postage",
|
||||||
|
@ -3971,15 +3972,6 @@ dependencies = [
|
||||||
"either",
|
"either",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "itertools"
|
|
||||||
version = "0.11.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "b1c173a5686ce8bfa551b3563d0c2170bf24ca44da99c7ca4bfdab5418c3fe57"
|
|
||||||
dependencies = [
|
|
||||||
"either",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "itoa"
|
name = "itoa"
|
||||||
version = "1.0.9"
|
version = "1.0.9"
|
||||||
|
@ -5772,7 +5764,7 @@ dependencies = [
|
||||||
"globset",
|
"globset",
|
||||||
"gpui",
|
"gpui",
|
||||||
"ignore",
|
"ignore",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"language",
|
"language",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"log",
|
"log",
|
||||||
|
@ -5896,7 +5888,7 @@ checksum = "62941722fb675d463659e49c4f3fe1fe792ff24fe5bbaa9c08cd3b98a1c354f5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bytes 1.4.0",
|
"bytes 1.4.0",
|
||||||
"heck 0.3.3",
|
"heck 0.3.3",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"log",
|
"log",
|
||||||
"multimap",
|
"multimap",
|
||||||
|
@ -5915,7 +5907,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "600d2f334aa05acb02a755e217ef1ab6dea4d51b58b7846588b747edec04efba"
|
checksum = "600d2f334aa05acb02a755e217ef1ab6dea4d51b58b7846588b747edec04efba"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 1.0.109",
|
"syn 1.0.109",
|
||||||
|
@ -5928,7 +5920,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f9cc1a3263e07e0bf68e96268f37665207b49560d98739662cdfaae215c720fe"
|
checksum = "f9cc1a3263e07e0bf68e96268f37665207b49560d98739662cdfaae215c720fe"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 1.0.109",
|
"syn 1.0.109",
|
||||||
|
@ -7568,7 +7560,7 @@ version = "0.2.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "0c12bc9199d1db8234678b7051747c07f517cdcf019262d1847b94ec8b1aee3e"
|
checksum = "0c12bc9199d1db8234678b7051747c07f517cdcf019262d1847b94ec8b1aee3e"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"nom",
|
"nom",
|
||||||
"unicode_categories",
|
"unicode_categories",
|
||||||
]
|
]
|
||||||
|
@ -7692,22 +7684,16 @@ name = "storybook"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"bytemuck",
|
|
||||||
"derive_more",
|
"derive_more",
|
||||||
"gpui2",
|
"gpui2",
|
||||||
"itertools 0.11.0",
|
|
||||||
"log",
|
"log",
|
||||||
"plane-split",
|
|
||||||
"raw-window-handle",
|
|
||||||
"refineable",
|
"refineable",
|
||||||
"rust-embed",
|
"rust-embed",
|
||||||
"serde",
|
"serde",
|
||||||
"settings",
|
"settings",
|
||||||
"simplelog",
|
"simplelog",
|
||||||
"slotmap",
|
|
||||||
"theme",
|
"theme",
|
||||||
"util",
|
"util",
|
||||||
"wgpu",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -7977,7 +7963,7 @@ dependencies = [
|
||||||
"dirs 4.0.0",
|
"dirs 4.0.0",
|
||||||
"futures 0.3.28",
|
"futures 0.3.28",
|
||||||
"gpui",
|
"gpui",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"libc",
|
"libc",
|
||||||
"mio-extras",
|
"mio-extras",
|
||||||
|
@ -8008,7 +7994,7 @@ dependencies = [
|
||||||
"editor",
|
"editor",
|
||||||
"futures 0.3.28",
|
"futures 0.3.28",
|
||||||
"gpui",
|
"gpui",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"language",
|
"language",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"libc",
|
"libc",
|
||||||
|
@ -9190,7 +9176,7 @@ dependencies = [
|
||||||
"futures 0.3.28",
|
"futures 0.3.28",
|
||||||
"gpui",
|
"gpui",
|
||||||
"indoc",
|
"indoc",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"language",
|
"language",
|
||||||
"language_selector",
|
"language_selector",
|
||||||
"log",
|
"log",
|
||||||
|
@ -10137,7 +10123,7 @@ dependencies = [
|
||||||
"gpui",
|
"gpui",
|
||||||
"indoc",
|
"indoc",
|
||||||
"install_cli",
|
"install_cli",
|
||||||
"itertools 0.10.5",
|
"itertools",
|
||||||
"language",
|
"language",
|
||||||
"lazy_static",
|
"lazy_static",
|
||||||
"log",
|
"log",
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use super::{
|
use super::{
|
||||||
event::key_to_native, screen::Screen, status_item::StatusItem, BoolExt as _, Dispatcher,
|
event::key_to_native, screen::Screen, status_item::StatusItem, BoolExt as _, Dispatcher,
|
||||||
FontSystem, MacWindow,
|
MacWindow, TextSystem,
|
||||||
};
|
};
|
||||||
use crate::{
|
use crate::{
|
||||||
executor,
|
executor,
|
||||||
|
@ -488,7 +488,7 @@ impl platform::ForegroundPlatform for MacForegroundPlatform {
|
||||||
|
|
||||||
pub struct MacPlatform {
|
pub struct MacPlatform {
|
||||||
dispatcher: Arc<Dispatcher>,
|
dispatcher: Arc<Dispatcher>,
|
||||||
fonts: Arc<FontSystem>,
|
fonts: Arc<TextSystem>,
|
||||||
pasteboard: id,
|
pasteboard: id,
|
||||||
text_hash_pasteboard_type: id,
|
text_hash_pasteboard_type: id,
|
||||||
metadata_pasteboard_type: id,
|
metadata_pasteboard_type: id,
|
||||||
|
@ -498,7 +498,7 @@ impl MacPlatform {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
Self {
|
Self {
|
||||||
dispatcher: Arc::new(Dispatcher),
|
dispatcher: Arc::new(Dispatcher),
|
||||||
fonts: Arc::new(FontSystem::new()),
|
fonts: Arc::new(TextSystem::new()),
|
||||||
pasteboard: unsafe { NSPasteboard::generalPasteboard(nil) },
|
pasteboard: unsafe { NSPasteboard::generalPasteboard(nil) },
|
||||||
text_hash_pasteboard_type: unsafe { ns_string("zed-text-hash") },
|
text_hash_pasteboard_type: unsafe { ns_string("zed-text-hash") },
|
||||||
metadata_pasteboard_type: unsafe { ns_string("zed-metadata") },
|
metadata_pasteboard_type: unsafe { ns_string("zed-metadata") },
|
||||||
|
|
|
@ -35,6 +35,7 @@ num_cpus = "1.13"
|
||||||
ordered-float.workspace = true
|
ordered-float.workspace = true
|
||||||
parking = "2.0.0"
|
parking = "2.0.0"
|
||||||
parking_lot.workspace = true
|
parking_lot.workspace = true
|
||||||
|
pathfinder_geometry = "0.5"
|
||||||
postage.workspace = true
|
postage.workspace = true
|
||||||
rand.workspace = true
|
rand.workspace = true
|
||||||
refineable.workspace = true
|
refineable.workspace = true
|
||||||
|
|
|
@ -1,14 +1,19 @@
|
||||||
|
use crate::{
|
||||||
|
Context, FontCache, LayoutId, Platform, Reference, View, Window, WindowContext, WindowHandle,
|
||||||
|
WindowId,
|
||||||
|
};
|
||||||
use anyhow::{anyhow, Result};
|
use anyhow::{anyhow, Result};
|
||||||
use slotmap::SlotMap;
|
use slotmap::SlotMap;
|
||||||
use std::{any::Any, marker::PhantomData, rc::Rc, sync::Arc};
|
use std::{any::Any, cell::RefCell, marker::PhantomData, rc::Rc, sync::Arc};
|
||||||
|
|
||||||
use crate::FontCache;
|
#[derive(Clone)]
|
||||||
|
pub struct App(Rc<RefCell<AppContext>>);
|
||||||
|
|
||||||
use super::{
|
impl App {
|
||||||
platform::Platform,
|
pub fn new(platform: Rc<dyn Platform>) -> Self {
|
||||||
window::{Window, WindowHandle, WindowId},
|
Self(Rc::new(RefCell::new(AppContext::new(platform))))
|
||||||
Context, LayoutId, Reference, View, WindowContext,
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
pub struct AppContext {
|
pub struct AppContext {
|
||||||
platform: Rc<dyn Platform>,
|
platform: Rc<dyn Platform>,
|
||||||
|
@ -21,7 +26,7 @@ pub struct AppContext {
|
||||||
|
|
||||||
impl AppContext {
|
impl AppContext {
|
||||||
pub fn new(platform: Rc<dyn Platform>) -> Self {
|
pub fn new(platform: Rc<dyn Platform>) -> Self {
|
||||||
let font_cache = Arc::new(FontCache::new(platform.font_system()));
|
let font_cache = Arc::new(FontCache::new(platform.text_system()));
|
||||||
AppContext {
|
AppContext {
|
||||||
platform,
|
platform,
|
||||||
font_cache,
|
font_cache,
|
||||||
|
|
|
@ -619,7 +619,7 @@ impl ExecutorEvent {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ForegroundExecutor {
|
impl ForegroundExecutor {
|
||||||
pub fn platform(dispatcher: Arc<dyn PlatformDispatcher>) -> Result<Self> {
|
pub fn new(dispatcher: Arc<dyn PlatformDispatcher>) -> Result<Self> {
|
||||||
if dispatcher.is_main_thread() {
|
if dispatcher.is_main_thread() {
|
||||||
Ok(Self::Platform {
|
Ok(Self::Platform {
|
||||||
dispatcher,
|
dispatcher,
|
||||||
|
|
|
@ -245,7 +245,7 @@ impl FontCache {
|
||||||
.typographic_bounds(font_id, glyph_id)
|
.typographic_bounds(font_id, glyph_id)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
bounds.size.width * self.em_size(font_id, font_size)
|
self.em_size(font_id, font_size) * bounds.size.width
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn em_advance(&self, font_id: FontId, font_size: Pixels) -> Pixels {
|
pub fn em_advance(&self, font_id: FontId, font_size: Pixels) -> Pixels {
|
||||||
|
@ -256,7 +256,7 @@ impl FontCache {
|
||||||
glyph_id = state.font_system.glyph_for_char(font_id, 'm').unwrap();
|
glyph_id = state.font_system.glyph_for_char(font_id, 'm').unwrap();
|
||||||
advance = state.font_system.advance(font_id, glyph_id).unwrap();
|
advance = state.font_system.advance(font_id, glyph_id).unwrap();
|
||||||
}
|
}
|
||||||
advance.x * self.em_size(font_id, font_size)
|
self.em_size(font_id, font_size) * advance.width
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn line_height(&self, font_size: Pixels) -> Pixels {
|
pub fn line_height(&self, font_size: Pixels) -> Pixels {
|
||||||
|
@ -356,7 +356,7 @@ mod tests {
|
||||||
#[test]
|
#[test]
|
||||||
fn test_select_font() {
|
fn test_select_font() {
|
||||||
let platform = TestPlatform::new();
|
let platform = TestPlatform::new();
|
||||||
let fonts = FontCache::new(platform.font_system());
|
let fonts = FontCache::new(platform.text_system());
|
||||||
let arial = fonts
|
let arial = fonts
|
||||||
.load_family(
|
.load_family(
|
||||||
&["Arial"],
|
&["Arial"],
|
||||||
|
|
|
@ -246,6 +246,12 @@ impl std::hash::Hash for Pixels {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<f64> for Pixels {
|
||||||
|
fn from(val: f64) -> Self {
|
||||||
|
Pixels(val as f32)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
unsafe impl bytemuck::Pod for Pixels {}
|
unsafe impl bytemuck::Pod for Pixels {}
|
||||||
unsafe impl bytemuck::Zeroable for Pixels {}
|
unsafe impl bytemuck::Zeroable for Pixels {}
|
||||||
|
|
||||||
|
|
|
@ -34,6 +34,7 @@ pub use styled::*;
|
||||||
pub use taffy::LayoutId;
|
pub use taffy::LayoutId;
|
||||||
use taffy::TaffyLayoutEngine;
|
use taffy::TaffyLayoutEngine;
|
||||||
use text::*;
|
use text::*;
|
||||||
|
pub use text::{Glyph, GlyphId};
|
||||||
pub use util::arc_cow::ArcCow;
|
pub use util::arc_cow::ArcCow;
|
||||||
pub use window::*;
|
pub use window::*;
|
||||||
|
|
||||||
|
|
|
@ -6,13 +6,26 @@ mod mac;
|
||||||
mod test;
|
mod test;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
AnyWindowHandle, Bounds, FontFeatures, FontId, FontMetrics, FontStyle, FontWeight, GlyphId,
|
AnyWindowHandle, Bounds, FontFeatures, FontId, FontMetrics, FontStyle, FontWeight,
|
||||||
LineLayout, Pixels, Point, RunStyle, SharedString, Size,
|
ForegroundExecutor, GlyphId, LineLayout, Pixels, Point, Result, RunStyle, SharedString, Size,
|
||||||
};
|
};
|
||||||
|
use anyhow::anyhow;
|
||||||
use async_task::Runnable;
|
use async_task::Runnable;
|
||||||
use futures::channel::oneshot;
|
use futures::channel::oneshot;
|
||||||
use raw_window_handle::{HasRawDisplayHandle, HasRawWindowHandle};
|
use raw_window_handle::{HasRawDisplayHandle, HasRawWindowHandle};
|
||||||
use std::{any::Any, fmt::Debug, ops::Range, rc::Rc, sync::Arc};
|
use seahash::SeaHasher;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
use std::hash::{Hash, Hasher};
|
||||||
|
use std::{
|
||||||
|
any::Any,
|
||||||
|
fmt::{self, Debug, Display},
|
||||||
|
ops::Range,
|
||||||
|
path::{Path, PathBuf},
|
||||||
|
rc::Rc,
|
||||||
|
str::FromStr,
|
||||||
|
sync::Arc,
|
||||||
|
};
|
||||||
|
pub use time::UtcOffset;
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
|
||||||
pub use events::*;
|
pub use events::*;
|
||||||
|
@ -22,15 +35,64 @@ pub use mac::*;
|
||||||
#[cfg(any(test, feature = "test"))]
|
#[cfg(any(test, feature = "test"))]
|
||||||
pub use test::*;
|
pub use test::*;
|
||||||
|
|
||||||
pub trait Platform {
|
// #[cfg(target_os = "macos")]
|
||||||
fn dispatcher(&self) -> Arc<dyn PlatformDispatcher>;
|
// pub fn current() -> Rc<dyn Platform> {
|
||||||
fn font_system(&self) -> Arc<dyn PlatformTextSystem>;
|
// MacPlatform
|
||||||
|
// }
|
||||||
|
|
||||||
|
pub trait Platform {
|
||||||
|
fn executor(&self) -> Rc<ForegroundExecutor>;
|
||||||
|
fn text_system(&self) -> Arc<dyn PlatformTextSystem>;
|
||||||
|
|
||||||
|
fn run(&self, on_finish_launching: Box<dyn FnOnce()>);
|
||||||
|
fn quit(&self);
|
||||||
|
fn restart(&self);
|
||||||
|
fn activate(&self, ignoring_other_apps: bool);
|
||||||
|
fn hide(&self);
|
||||||
|
fn hide_other_apps(&self);
|
||||||
|
fn unhide_other_apps(&self);
|
||||||
|
|
||||||
|
fn screens(&self) -> Vec<Rc<dyn PlatformScreen>>;
|
||||||
|
fn screen_by_id(&self, id: uuid::Uuid) -> Option<Rc<dyn PlatformScreen>>;
|
||||||
|
fn main_window(&self) -> Option<AnyWindowHandle>;
|
||||||
fn open_window(
|
fn open_window(
|
||||||
&self,
|
&self,
|
||||||
handle: AnyWindowHandle,
|
handle: AnyWindowHandle,
|
||||||
options: WindowOptions,
|
options: WindowOptions,
|
||||||
) -> Box<dyn PlatformWindow>;
|
) -> Box<dyn PlatformWindow>;
|
||||||
|
// fn add_status_item(&self, _handle: AnyWindowHandle) -> Box<dyn PlatformWindow>;
|
||||||
|
|
||||||
|
fn open_url(&self, url: &str);
|
||||||
|
fn on_open_urls(&self, callback: Box<dyn FnMut(Vec<String>)>);
|
||||||
|
fn prompt_for_paths(
|
||||||
|
&self,
|
||||||
|
options: PathPromptOptions,
|
||||||
|
) -> oneshot::Receiver<Option<Vec<PathBuf>>>;
|
||||||
|
fn prompt_for_new_path(&self, directory: &Path) -> oneshot::Receiver<Option<PathBuf>>;
|
||||||
|
fn reveal_path(&self, path: &Path);
|
||||||
|
|
||||||
|
fn on_become_active(&self, callback: Box<dyn FnMut()>);
|
||||||
|
fn on_resign_active(&self, callback: Box<dyn FnMut()>);
|
||||||
|
fn on_quit(&self, callback: Box<dyn FnMut()>);
|
||||||
|
fn on_reopen(&self, callback: Box<dyn FnMut()>);
|
||||||
|
fn on_event(&self, callback: Box<dyn FnMut(Event) -> bool>);
|
||||||
|
|
||||||
|
fn os_name(&self) -> &'static str;
|
||||||
|
fn os_version(&self) -> Result<SemanticVersion>;
|
||||||
|
fn app_version(&self) -> Result<SemanticVersion>;
|
||||||
|
fn app_path(&self) -> Result<PathBuf>;
|
||||||
|
fn local_timezone(&self) -> UtcOffset;
|
||||||
|
fn path_for_auxiliary_executable(&self, name: &str) -> Result<PathBuf>;
|
||||||
|
|
||||||
|
fn set_cursor_style(&self, style: CursorStyle);
|
||||||
|
fn should_auto_hide_scrollbars(&self) -> bool;
|
||||||
|
|
||||||
|
fn write_to_clipboard(&self, item: ClipboardItem);
|
||||||
|
fn read_from_clipboard(&self) -> Option<ClipboardItem>;
|
||||||
|
|
||||||
|
fn write_credentials(&self, url: &str, username: &str, password: &[u8]) -> Result<()>;
|
||||||
|
fn read_credentials(&self, url: &str) -> Result<Option<(String, Vec<u8>)>>;
|
||||||
|
fn delete_credentials(&self, url: &str) -> Result<()>;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait PlatformScreen: Debug {
|
pub trait PlatformScreen: Debug {
|
||||||
|
@ -90,12 +152,9 @@ pub trait PlatformTextSystem: Send + Sync {
|
||||||
style: FontStyle,
|
style: FontStyle,
|
||||||
) -> anyhow::Result<FontId>;
|
) -> anyhow::Result<FontId>;
|
||||||
fn font_metrics(&self, font_id: FontId) -> FontMetrics;
|
fn font_metrics(&self, font_id: FontId) -> FontMetrics;
|
||||||
fn typographic_bounds(
|
fn typographic_bounds(&self, font_id: FontId, glyph_id: GlyphId)
|
||||||
&self,
|
-> anyhow::Result<Bounds<f32>>;
|
||||||
font_id: FontId,
|
fn advance(&self, font_id: FontId, glyph_id: GlyphId) -> anyhow::Result<Size<f32>>;
|
||||||
glyph_id: GlyphId,
|
|
||||||
) -> anyhow::Result<Bounds<Pixels>>;
|
|
||||||
fn advance(&self, font_id: FontId, glyph_id: GlyphId) -> anyhow::Result<Point<Pixels>>;
|
|
||||||
fn glyph_for_char(&self, font_id: FontId, ch: char) -> Option<GlyphId>;
|
fn glyph_for_char(&self, font_id: FontId, ch: char) -> Option<GlyphId>;
|
||||||
fn rasterize_glyph(
|
fn rasterize_glyph(
|
||||||
&self,
|
&self,
|
||||||
|
@ -224,3 +283,109 @@ pub enum WindowPromptLevel {
|
||||||
Warning,
|
Warning,
|
||||||
Critical,
|
Critical,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug)]
|
||||||
|
pub struct PathPromptOptions {
|
||||||
|
pub files: bool,
|
||||||
|
pub directories: bool,
|
||||||
|
pub multiple: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug)]
|
||||||
|
pub enum PromptLevel {
|
||||||
|
Info,
|
||||||
|
Warning,
|
||||||
|
Critical,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug)]
|
||||||
|
pub enum CursorStyle {
|
||||||
|
Arrow,
|
||||||
|
ResizeLeftRight,
|
||||||
|
ResizeUpDown,
|
||||||
|
PointingHand,
|
||||||
|
IBeam,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for CursorStyle {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::Arrow
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
|
||||||
|
pub struct SemanticVersion {
|
||||||
|
major: usize,
|
||||||
|
minor: usize,
|
||||||
|
patch: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FromStr for SemanticVersion {
|
||||||
|
type Err = anyhow::Error;
|
||||||
|
|
||||||
|
fn from_str(s: &str) -> Result<Self> {
|
||||||
|
let mut components = s.trim().split('.');
|
||||||
|
let major = components
|
||||||
|
.next()
|
||||||
|
.ok_or_else(|| anyhow!("missing major version number"))?
|
||||||
|
.parse()?;
|
||||||
|
let minor = components
|
||||||
|
.next()
|
||||||
|
.ok_or_else(|| anyhow!("missing minor version number"))?
|
||||||
|
.parse()?;
|
||||||
|
let patch = components
|
||||||
|
.next()
|
||||||
|
.ok_or_else(|| anyhow!("missing patch version number"))?
|
||||||
|
.parse()?;
|
||||||
|
Ok(Self {
|
||||||
|
major,
|
||||||
|
minor,
|
||||||
|
patch,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Display for SemanticVersion {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
write!(f, "{}.{}.{}", self.major, self.minor, self.patch)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
||||||
|
pub struct ClipboardItem {
|
||||||
|
pub(crate) text: String,
|
||||||
|
pub(crate) metadata: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClipboardItem {
|
||||||
|
pub fn new(text: String) -> Self {
|
||||||
|
Self {
|
||||||
|
text,
|
||||||
|
metadata: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn with_metadata<T: Serialize>(mut self, metadata: T) -> Self {
|
||||||
|
self.metadata = Some(serde_json::to_string(&metadata).unwrap());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn text(&self) -> &String {
|
||||||
|
&self.text
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn metadata<T>(&self) -> Option<T>
|
||||||
|
where
|
||||||
|
T: for<'a> Deserialize<'a>,
|
||||||
|
{
|
||||||
|
self.metadata
|
||||||
|
.as_ref()
|
||||||
|
.and_then(|m| serde_json::from_str(m).ok())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn text_hash(text: &str) -> u64 {
|
||||||
|
let mut hasher = SeaHasher::new();
|
||||||
|
text.hash(&mut hasher);
|
||||||
|
hasher.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -2,18 +2,13 @@
|
||||||
///! an origin at the bottom left of the main display.
|
///! an origin at the bottom left of the main display.
|
||||||
mod dispatcher;
|
mod dispatcher;
|
||||||
mod events;
|
mod events;
|
||||||
|
mod open_type;
|
||||||
mod platform;
|
mod platform;
|
||||||
mod screen;
|
mod screen;
|
||||||
|
mod text_system;
|
||||||
mod window;
|
mod window;
|
||||||
mod window_appearence;
|
mod window_appearence;
|
||||||
|
|
||||||
use std::{
|
|
||||||
ffi::{c_char, CStr, OsStr},
|
|
||||||
ops::Range,
|
|
||||||
os::unix::prelude::OsStrExt,
|
|
||||||
path::PathBuf,
|
|
||||||
};
|
|
||||||
|
|
||||||
use crate::{px, size, Pixels, Size};
|
use crate::{px, size, Pixels, Size};
|
||||||
use anyhow::anyhow;
|
use anyhow::anyhow;
|
||||||
use cocoa::{
|
use cocoa::{
|
||||||
|
@ -25,8 +20,17 @@ use objc::{
|
||||||
runtime::{BOOL, NO, YES},
|
runtime::{BOOL, NO, YES},
|
||||||
sel, sel_impl,
|
sel, sel_impl,
|
||||||
};
|
};
|
||||||
|
use std::{
|
||||||
|
ffi::{c_char, CStr, OsStr},
|
||||||
|
ops::Range,
|
||||||
|
os::unix::prelude::OsStrExt,
|
||||||
|
path::PathBuf,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub use dispatcher::*;
|
||||||
pub use platform::*;
|
pub use platform::*;
|
||||||
pub use screen::*;
|
pub use screen::*;
|
||||||
|
pub use text_system::*;
|
||||||
pub use window::*;
|
pub use window::*;
|
||||||
|
|
||||||
trait BoolExt {
|
trait BoolExt {
|
||||||
|
|
395
crates/gpui3/src/platform/mac/open_type.rs
Normal file
395
crates/gpui3/src/platform/mac/open_type.rs
Normal file
|
@ -0,0 +1,395 @@
|
||||||
|
#![allow(unused, non_upper_case_globals)]
|
||||||
|
|
||||||
|
use std::ptr;
|
||||||
|
|
||||||
|
use crate::FontFeatures;
|
||||||
|
use cocoa::appkit::CGFloat;
|
||||||
|
use core_foundation::{base::TCFType, number::CFNumber};
|
||||||
|
use core_graphics::geometry::CGAffineTransform;
|
||||||
|
use core_text::{
|
||||||
|
font::{CTFont, CTFontRef},
|
||||||
|
font_descriptor::{
|
||||||
|
CTFontDescriptor, CTFontDescriptorCreateCopyWithFeature, CTFontDescriptorRef,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
use font_kit::font::Font;
|
||||||
|
|
||||||
|
const kCaseSensitiveLayoutOffSelector: i32 = 1;
|
||||||
|
const kCaseSensitiveLayoutOnSelector: i32 = 0;
|
||||||
|
const kCaseSensitiveLayoutType: i32 = 33;
|
||||||
|
const kCaseSensitiveSpacingOffSelector: i32 = 3;
|
||||||
|
const kCaseSensitiveSpacingOnSelector: i32 = 2;
|
||||||
|
const kCharacterAlternativesType: i32 = 17;
|
||||||
|
const kCommonLigaturesOffSelector: i32 = 3;
|
||||||
|
const kCommonLigaturesOnSelector: i32 = 2;
|
||||||
|
const kContextualAlternatesOffSelector: i32 = 1;
|
||||||
|
const kContextualAlternatesOnSelector: i32 = 0;
|
||||||
|
const kContextualAlternatesType: i32 = 36;
|
||||||
|
const kContextualLigaturesOffSelector: i32 = 19;
|
||||||
|
const kContextualLigaturesOnSelector: i32 = 18;
|
||||||
|
const kContextualSwashAlternatesOffSelector: i32 = 5;
|
||||||
|
const kContextualSwashAlternatesOnSelector: i32 = 4;
|
||||||
|
const kDefaultLowerCaseSelector: i32 = 0;
|
||||||
|
const kDefaultUpperCaseSelector: i32 = 0;
|
||||||
|
const kDiagonalFractionsSelector: i32 = 2;
|
||||||
|
const kFractionsType: i32 = 11;
|
||||||
|
const kHistoricalLigaturesOffSelector: i32 = 21;
|
||||||
|
const kHistoricalLigaturesOnSelector: i32 = 20;
|
||||||
|
const kHojoCharactersSelector: i32 = 12;
|
||||||
|
const kInferiorsSelector: i32 = 2;
|
||||||
|
const kJIS2004CharactersSelector: i32 = 11;
|
||||||
|
const kLigaturesType: i32 = 1;
|
||||||
|
const kLowerCasePetiteCapsSelector: i32 = 2;
|
||||||
|
const kLowerCaseSmallCapsSelector: i32 = 1;
|
||||||
|
const kLowerCaseType: i32 = 37;
|
||||||
|
const kLowerCaseNumbersSelector: i32 = 0;
|
||||||
|
const kMathematicalGreekOffSelector: i32 = 11;
|
||||||
|
const kMathematicalGreekOnSelector: i32 = 10;
|
||||||
|
const kMonospacedNumbersSelector: i32 = 0;
|
||||||
|
const kNLCCharactersSelector: i32 = 13;
|
||||||
|
const kNoFractionsSelector: i32 = 0;
|
||||||
|
const kNormalPositionSelector: i32 = 0;
|
||||||
|
const kNoStyleOptionsSelector: i32 = 0;
|
||||||
|
const kNumberCaseType: i32 = 21;
|
||||||
|
const kNumberSpacingType: i32 = 6;
|
||||||
|
const kOrdinalsSelector: i32 = 3;
|
||||||
|
const kProportionalNumbersSelector: i32 = 1;
|
||||||
|
const kQuarterWidthTextSelector: i32 = 4;
|
||||||
|
const kScientificInferiorsSelector: i32 = 4;
|
||||||
|
const kSlashedZeroOffSelector: i32 = 5;
|
||||||
|
const kSlashedZeroOnSelector: i32 = 4;
|
||||||
|
const kStyleOptionsType: i32 = 19;
|
||||||
|
const kStylisticAltEighteenOffSelector: i32 = 37;
|
||||||
|
const kStylisticAltEighteenOnSelector: i32 = 36;
|
||||||
|
const kStylisticAltEightOffSelector: i32 = 17;
|
||||||
|
const kStylisticAltEightOnSelector: i32 = 16;
|
||||||
|
const kStylisticAltElevenOffSelector: i32 = 23;
|
||||||
|
const kStylisticAltElevenOnSelector: i32 = 22;
|
||||||
|
const kStylisticAlternativesType: i32 = 35;
|
||||||
|
const kStylisticAltFifteenOffSelector: i32 = 31;
|
||||||
|
const kStylisticAltFifteenOnSelector: i32 = 30;
|
||||||
|
const kStylisticAltFiveOffSelector: i32 = 11;
|
||||||
|
const kStylisticAltFiveOnSelector: i32 = 10;
|
||||||
|
const kStylisticAltFourOffSelector: i32 = 9;
|
||||||
|
const kStylisticAltFourOnSelector: i32 = 8;
|
||||||
|
const kStylisticAltFourteenOffSelector: i32 = 29;
|
||||||
|
const kStylisticAltFourteenOnSelector: i32 = 28;
|
||||||
|
const kStylisticAltNineOffSelector: i32 = 19;
|
||||||
|
const kStylisticAltNineOnSelector: i32 = 18;
|
||||||
|
const kStylisticAltNineteenOffSelector: i32 = 39;
|
||||||
|
const kStylisticAltNineteenOnSelector: i32 = 38;
|
||||||
|
const kStylisticAltOneOffSelector: i32 = 3;
|
||||||
|
const kStylisticAltOneOnSelector: i32 = 2;
|
||||||
|
const kStylisticAltSevenOffSelector: i32 = 15;
|
||||||
|
const kStylisticAltSevenOnSelector: i32 = 14;
|
||||||
|
const kStylisticAltSeventeenOffSelector: i32 = 35;
|
||||||
|
const kStylisticAltSeventeenOnSelector: i32 = 34;
|
||||||
|
const kStylisticAltSixOffSelector: i32 = 13;
|
||||||
|
const kStylisticAltSixOnSelector: i32 = 12;
|
||||||
|
const kStylisticAltSixteenOffSelector: i32 = 33;
|
||||||
|
const kStylisticAltSixteenOnSelector: i32 = 32;
|
||||||
|
const kStylisticAltTenOffSelector: i32 = 21;
|
||||||
|
const kStylisticAltTenOnSelector: i32 = 20;
|
||||||
|
const kStylisticAltThirteenOffSelector: i32 = 27;
|
||||||
|
const kStylisticAltThirteenOnSelector: i32 = 26;
|
||||||
|
const kStylisticAltThreeOffSelector: i32 = 7;
|
||||||
|
const kStylisticAltThreeOnSelector: i32 = 6;
|
||||||
|
const kStylisticAltTwelveOffSelector: i32 = 25;
|
||||||
|
const kStylisticAltTwelveOnSelector: i32 = 24;
|
||||||
|
const kStylisticAltTwentyOffSelector: i32 = 41;
|
||||||
|
const kStylisticAltTwentyOnSelector: i32 = 40;
|
||||||
|
const kStylisticAltTwoOffSelector: i32 = 5;
|
||||||
|
const kStylisticAltTwoOnSelector: i32 = 4;
|
||||||
|
const kSuperiorsSelector: i32 = 1;
|
||||||
|
const kSwashAlternatesOffSelector: i32 = 3;
|
||||||
|
const kSwashAlternatesOnSelector: i32 = 2;
|
||||||
|
const kTitlingCapsSelector: i32 = 4;
|
||||||
|
const kTypographicExtrasType: i32 = 14;
|
||||||
|
const kVerticalFractionsSelector: i32 = 1;
|
||||||
|
const kVerticalPositionType: i32 = 10;
|
||||||
|
|
||||||
|
pub fn apply_features(font: &mut Font, features: &FontFeatures) {
|
||||||
|
// See https://chromium.googlesource.com/chromium/src/+/66.0.3359.158/third_party/harfbuzz-ng/src/hb-coretext.cc
|
||||||
|
// for a reference implementation.
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.calt,
|
||||||
|
kContextualAlternatesType,
|
||||||
|
kContextualAlternatesOnSelector,
|
||||||
|
kContextualAlternatesOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.case,
|
||||||
|
kCaseSensitiveLayoutType,
|
||||||
|
kCaseSensitiveLayoutOnSelector,
|
||||||
|
kCaseSensitiveLayoutOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.cpsp,
|
||||||
|
kCaseSensitiveLayoutType,
|
||||||
|
kCaseSensitiveSpacingOnSelector,
|
||||||
|
kCaseSensitiveSpacingOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.frac,
|
||||||
|
kFractionsType,
|
||||||
|
kDiagonalFractionsSelector,
|
||||||
|
kNoFractionsSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.liga,
|
||||||
|
kLigaturesType,
|
||||||
|
kCommonLigaturesOnSelector,
|
||||||
|
kCommonLigaturesOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.onum,
|
||||||
|
kNumberCaseType,
|
||||||
|
kLowerCaseNumbersSelector,
|
||||||
|
2,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ordn,
|
||||||
|
kVerticalPositionType,
|
||||||
|
kOrdinalsSelector,
|
||||||
|
kNormalPositionSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.pnum,
|
||||||
|
kNumberSpacingType,
|
||||||
|
kProportionalNumbersSelector,
|
||||||
|
4,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss01,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltOneOnSelector,
|
||||||
|
kStylisticAltOneOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss02,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltTwoOnSelector,
|
||||||
|
kStylisticAltTwoOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss03,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltThreeOnSelector,
|
||||||
|
kStylisticAltThreeOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss04,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltFourOnSelector,
|
||||||
|
kStylisticAltFourOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss05,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltFiveOnSelector,
|
||||||
|
kStylisticAltFiveOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss06,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltSixOnSelector,
|
||||||
|
kStylisticAltSixOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss07,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltSevenOnSelector,
|
||||||
|
kStylisticAltSevenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss08,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltEightOnSelector,
|
||||||
|
kStylisticAltEightOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss09,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltNineOnSelector,
|
||||||
|
kStylisticAltNineOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss10,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltTenOnSelector,
|
||||||
|
kStylisticAltTenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss11,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltElevenOnSelector,
|
||||||
|
kStylisticAltElevenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss12,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltTwelveOnSelector,
|
||||||
|
kStylisticAltTwelveOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss13,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltThirteenOnSelector,
|
||||||
|
kStylisticAltThirteenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss14,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltFourteenOnSelector,
|
||||||
|
kStylisticAltFourteenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss15,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltFifteenOnSelector,
|
||||||
|
kStylisticAltFifteenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss16,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltSixteenOnSelector,
|
||||||
|
kStylisticAltSixteenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss17,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltSeventeenOnSelector,
|
||||||
|
kStylisticAltSeventeenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss18,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltEighteenOnSelector,
|
||||||
|
kStylisticAltEighteenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss19,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltNineteenOnSelector,
|
||||||
|
kStylisticAltNineteenOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.ss20,
|
||||||
|
kStylisticAlternativesType,
|
||||||
|
kStylisticAltTwentyOnSelector,
|
||||||
|
kStylisticAltTwentyOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.subs,
|
||||||
|
kVerticalPositionType,
|
||||||
|
kInferiorsSelector,
|
||||||
|
kNormalPositionSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.sups,
|
||||||
|
kVerticalPositionType,
|
||||||
|
kSuperiorsSelector,
|
||||||
|
kNormalPositionSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.swsh,
|
||||||
|
kContextualAlternatesType,
|
||||||
|
kSwashAlternatesOnSelector,
|
||||||
|
kSwashAlternatesOffSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.titl,
|
||||||
|
kStyleOptionsType,
|
||||||
|
kTitlingCapsSelector,
|
||||||
|
kNoStyleOptionsSelector,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.tnum,
|
||||||
|
kNumberSpacingType,
|
||||||
|
kMonospacedNumbersSelector,
|
||||||
|
4,
|
||||||
|
);
|
||||||
|
toggle_open_type_feature(
|
||||||
|
font,
|
||||||
|
features.zero,
|
||||||
|
kTypographicExtrasType,
|
||||||
|
kSlashedZeroOnSelector,
|
||||||
|
kSlashedZeroOffSelector,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn toggle_open_type_feature(
|
||||||
|
font: &mut Font,
|
||||||
|
enabled: Option<bool>,
|
||||||
|
type_identifier: i32,
|
||||||
|
on_selector_identifier: i32,
|
||||||
|
off_selector_identifier: i32,
|
||||||
|
) {
|
||||||
|
if let Some(enabled) = enabled {
|
||||||
|
let native_font = font.native_font();
|
||||||
|
unsafe {
|
||||||
|
let selector_identifier = if enabled {
|
||||||
|
on_selector_identifier
|
||||||
|
} else {
|
||||||
|
off_selector_identifier
|
||||||
|
};
|
||||||
|
let new_descriptor = CTFontDescriptorCreateCopyWithFeature(
|
||||||
|
native_font.copy_descriptor().as_concrete_TypeRef(),
|
||||||
|
CFNumber::from(type_identifier).as_concrete_TypeRef(),
|
||||||
|
CFNumber::from(selector_identifier).as_concrete_TypeRef(),
|
||||||
|
);
|
||||||
|
let new_descriptor = CTFontDescriptor::wrap_under_create_rule(new_descriptor);
|
||||||
|
let new_font = CTFontCreateCopyWithAttributes(
|
||||||
|
font.native_font().as_concrete_TypeRef(),
|
||||||
|
0.0,
|
||||||
|
ptr::null(),
|
||||||
|
new_descriptor.as_concrete_TypeRef(),
|
||||||
|
);
|
||||||
|
let new_font = CTFont::wrap_under_create_rule(new_font);
|
||||||
|
*font = Font::from_native_font(new_font);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[link(name = "CoreText", kind = "framework")]
|
||||||
|
extern "C" {
|
||||||
|
fn CTFontCreateCopyWithAttributes(
|
||||||
|
font: CTFontRef,
|
||||||
|
size: CGFloat,
|
||||||
|
matrix: *const CGAffineTransform,
|
||||||
|
attributes: CTFontDescriptorRef,
|
||||||
|
) -> CTFontRef;
|
||||||
|
}
|
File diff suppressed because it is too large
Load diff
757
crates/gpui3/src/platform/mac/text_system.rs
Normal file
757
crates/gpui3/src/platform/mac/text_system.rs
Normal file
|
@ -0,0 +1,757 @@
|
||||||
|
use crate::{
|
||||||
|
point, px, size, Bounds, FontFeatures, FontId, FontMetrics, FontStyle, FontWeight, Glyph,
|
||||||
|
GlyphId, LineLayout, Pixels, PlatformTextSystem, Point, RasterizationOptions, Run, RunStyle,
|
||||||
|
Size,
|
||||||
|
};
|
||||||
|
use cocoa::appkit::{CGFloat, CGPoint};
|
||||||
|
use collections::HashMap;
|
||||||
|
use core_foundation::{
|
||||||
|
array::CFIndex,
|
||||||
|
attributed_string::{CFAttributedStringRef, CFMutableAttributedString},
|
||||||
|
base::{CFRange, TCFType},
|
||||||
|
string::CFString,
|
||||||
|
};
|
||||||
|
use core_graphics::{
|
||||||
|
base::{kCGImageAlphaPremultipliedLast, CGGlyph},
|
||||||
|
color_space::CGColorSpace,
|
||||||
|
context::CGContext,
|
||||||
|
};
|
||||||
|
use core_text::{font::CTFont, line::CTLine, string_attributes::kCTFontAttributeName};
|
||||||
|
use font_kit::{
|
||||||
|
handle::Handle, hinting::HintingOptions, metrics::Metrics, source::SystemSource,
|
||||||
|
sources::mem::MemSource,
|
||||||
|
};
|
||||||
|
use parking_lot::RwLock;
|
||||||
|
use pathfinder_geometry::{
|
||||||
|
rect::{RectF, RectI},
|
||||||
|
transform2d::Transform2F,
|
||||||
|
vector::{Vector2F, Vector2I},
|
||||||
|
};
|
||||||
|
use std::{cell::RefCell, char, cmp, convert::TryFrom, ffi::c_void, sync::Arc};
|
||||||
|
|
||||||
|
use super::open_type;
|
||||||
|
|
||||||
|
#[allow(non_upper_case_globals)]
|
||||||
|
const kCGImageAlphaOnly: u32 = 7;
|
||||||
|
|
||||||
|
pub struct MacTextSystem(RwLock<TextSystemState>);
|
||||||
|
|
||||||
|
struct TextSystemState {
|
||||||
|
memory_source: MemSource,
|
||||||
|
system_source: SystemSource,
|
||||||
|
fonts: Vec<font_kit::font::Font>,
|
||||||
|
font_ids_by_postscript_name: HashMap<String, FontId>,
|
||||||
|
postscript_names_by_font_id: HashMap<FontId, String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MacTextSystem {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self(RwLock::new(TextSystemState {
|
||||||
|
memory_source: MemSource::empty(),
|
||||||
|
system_source: SystemSource::new(),
|
||||||
|
fonts: Vec::new(),
|
||||||
|
font_ids_by_postscript_name: Default::default(),
|
||||||
|
postscript_names_by_font_id: Default::default(),
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for MacTextSystem {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PlatformTextSystem for MacTextSystem {
|
||||||
|
fn add_fonts(&self, fonts: &[Arc<Vec<u8>>]) -> anyhow::Result<()> {
|
||||||
|
self.0.write().add_fonts(fonts)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn all_families(&self) -> Vec<String> {
|
||||||
|
self.0
|
||||||
|
.read()
|
||||||
|
.system_source
|
||||||
|
.all_families()
|
||||||
|
.expect("core text should never return an error")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn load_family(&self, name: &str, features: &FontFeatures) -> anyhow::Result<Vec<FontId>> {
|
||||||
|
self.0.write().load_family(name, features)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn select_font(
|
||||||
|
&self,
|
||||||
|
font_ids: &[FontId],
|
||||||
|
weight: FontWeight,
|
||||||
|
style: FontStyle,
|
||||||
|
) -> anyhow::Result<FontId> {
|
||||||
|
self.0.read().select_font(font_ids, weight, style)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn font_metrics(&self, font_id: FontId) -> FontMetrics {
|
||||||
|
self.0.read().font_metrics(font_id)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn typographic_bounds(
|
||||||
|
&self,
|
||||||
|
font_id: FontId,
|
||||||
|
glyph_id: GlyphId,
|
||||||
|
) -> anyhow::Result<Bounds<f32>> {
|
||||||
|
self.0.read().typographic_bounds(font_id, glyph_id)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn advance(&self, font_id: FontId, glyph_id: GlyphId) -> anyhow::Result<Size<f32>> {
|
||||||
|
self.0.read().advance(font_id, glyph_id)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn glyph_for_char(&self, font_id: FontId, ch: char) -> Option<GlyphId> {
|
||||||
|
self.0.read().glyph_for_char(font_id, ch)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn rasterize_glyph(
|
||||||
|
&self,
|
||||||
|
font_id: FontId,
|
||||||
|
font_size: f32,
|
||||||
|
glyph_id: GlyphId,
|
||||||
|
subpixel_shift: Point<Pixels>,
|
||||||
|
scale_factor: f32,
|
||||||
|
options: RasterizationOptions,
|
||||||
|
) -> Option<(Bounds<u32>, Vec<u8>)> {
|
||||||
|
self.0.read().rasterize_glyph(
|
||||||
|
font_id,
|
||||||
|
font_size,
|
||||||
|
glyph_id,
|
||||||
|
subpixel_shift,
|
||||||
|
scale_factor,
|
||||||
|
options,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn layout_line(&self, text: &str, font_size: Pixels, runs: &[(usize, RunStyle)]) -> LineLayout {
|
||||||
|
self.0.write().layout_line(text, font_size, runs)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn wrap_line(
|
||||||
|
&self,
|
||||||
|
text: &str,
|
||||||
|
font_id: FontId,
|
||||||
|
font_size: Pixels,
|
||||||
|
width: Pixels,
|
||||||
|
) -> Vec<usize> {
|
||||||
|
self.0.read().wrap_line(text, font_id, font_size, width)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TextSystemState {
|
||||||
|
fn add_fonts(&mut self, fonts: &[Arc<Vec<u8>>]) -> anyhow::Result<()> {
|
||||||
|
self.memory_source.add_fonts(
|
||||||
|
fonts
|
||||||
|
.iter()
|
||||||
|
.map(|bytes| Handle::from_memory(bytes.clone(), 0)),
|
||||||
|
)?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn load_family(&mut self, name: &str, features: &FontFeatures) -> anyhow::Result<Vec<FontId>> {
|
||||||
|
let mut font_ids = Vec::new();
|
||||||
|
|
||||||
|
let family = self
|
||||||
|
.memory_source
|
||||||
|
.select_family_by_name(name)
|
||||||
|
.or_else(|_| self.system_source.select_family_by_name(name))?;
|
||||||
|
for font in family.fonts() {
|
||||||
|
let mut font = font.load()?;
|
||||||
|
open_type::apply_features(&mut font, features);
|
||||||
|
let font_id = FontId(self.fonts.len());
|
||||||
|
font_ids.push(font_id);
|
||||||
|
let postscript_name = font.postscript_name().unwrap();
|
||||||
|
self.font_ids_by_postscript_name
|
||||||
|
.insert(postscript_name.clone(), font_id);
|
||||||
|
self.postscript_names_by_font_id
|
||||||
|
.insert(font_id, postscript_name);
|
||||||
|
self.fonts.push(font);
|
||||||
|
}
|
||||||
|
Ok(font_ids)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn select_font(
|
||||||
|
&self,
|
||||||
|
font_ids: &[FontId],
|
||||||
|
weight: FontWeight,
|
||||||
|
style: FontStyle,
|
||||||
|
) -> anyhow::Result<FontId> {
|
||||||
|
let candidates = font_ids
|
||||||
|
.iter()
|
||||||
|
.map(|font_id| self.fonts[font_id.0].properties())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
let idx = font_kit::matching::find_best_match(
|
||||||
|
&candidates,
|
||||||
|
&font_kit::properties::Properties {
|
||||||
|
style,
|
||||||
|
weight,
|
||||||
|
stretch: Default::default(),
|
||||||
|
},
|
||||||
|
)?;
|
||||||
|
Ok(font_ids[idx])
|
||||||
|
}
|
||||||
|
|
||||||
|
fn font_metrics(&self, font_id: FontId) -> FontMetrics {
|
||||||
|
self.fonts[font_id.0].metrics().into()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn typographic_bounds(
|
||||||
|
&self,
|
||||||
|
font_id: FontId,
|
||||||
|
glyph_id: GlyphId,
|
||||||
|
) -> anyhow::Result<Bounds<f32>> {
|
||||||
|
Ok(self.fonts[font_id.0]
|
||||||
|
.typographic_bounds(glyph_id.into())?
|
||||||
|
.into())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn advance(&self, font_id: FontId, glyph_id: GlyphId) -> anyhow::Result<Size<f32>> {
|
||||||
|
Ok(self.fonts[font_id.0].advance(glyph_id.into())?.into())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn glyph_for_char(&self, font_id: FontId, ch: char) -> Option<GlyphId> {
|
||||||
|
self.fonts[font_id.0].glyph_for_char(ch).map(Into::into)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn id_for_native_font(&mut self, requested_font: CTFont) -> FontId {
|
||||||
|
let postscript_name = requested_font.postscript_name();
|
||||||
|
if let Some(font_id) = self.font_ids_by_postscript_name.get(&postscript_name) {
|
||||||
|
*font_id
|
||||||
|
} else {
|
||||||
|
let font_id = FontId(self.fonts.len());
|
||||||
|
self.font_ids_by_postscript_name
|
||||||
|
.insert(postscript_name.clone(), font_id);
|
||||||
|
self.postscript_names_by_font_id
|
||||||
|
.insert(font_id, postscript_name);
|
||||||
|
self.fonts
|
||||||
|
.push(font_kit::font::Font::from_core_graphics_font(
|
||||||
|
requested_font.copy_to_CGFont(),
|
||||||
|
));
|
||||||
|
font_id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_emoji(&self, font_id: FontId) -> bool {
|
||||||
|
self.postscript_names_by_font_id
|
||||||
|
.get(&font_id)
|
||||||
|
.map_or(false, |postscript_name| {
|
||||||
|
postscript_name == "AppleColorEmoji"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn rasterize_glyph(
|
||||||
|
&self,
|
||||||
|
font_id: FontId,
|
||||||
|
font_size: f32,
|
||||||
|
glyph_id: GlyphId,
|
||||||
|
subpixel_shift: Point<Pixels>,
|
||||||
|
scale_factor: f32,
|
||||||
|
options: RasterizationOptions,
|
||||||
|
) -> Option<(Bounds<u32>, Vec<u8>)> {
|
||||||
|
let font = &self.fonts[font_id.0];
|
||||||
|
let scale = Transform2F::from_scale(scale_factor);
|
||||||
|
let glyph_bounds = font
|
||||||
|
.raster_bounds(
|
||||||
|
glyph_id.into(),
|
||||||
|
font_size,
|
||||||
|
scale,
|
||||||
|
HintingOptions::None,
|
||||||
|
font_kit::canvas::RasterizationOptions::GrayscaleAa,
|
||||||
|
)
|
||||||
|
.ok()?;
|
||||||
|
|
||||||
|
if glyph_bounds.width() == 0 || glyph_bounds.height() == 0 {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
// Make room for subpixel variants.
|
||||||
|
let subpixel_padding = subpixel_shift.map(|v| f32::from(v).ceil() as u32);
|
||||||
|
let cx_bounds = RectI::new(
|
||||||
|
glyph_bounds.origin(),
|
||||||
|
glyph_bounds.size() + Vector2I::from(subpixel_padding),
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut bytes;
|
||||||
|
let cx;
|
||||||
|
match options {
|
||||||
|
RasterizationOptions::Alpha => {
|
||||||
|
bytes = vec![0; cx_bounds.width() as usize * cx_bounds.height() as usize];
|
||||||
|
cx = CGContext::create_bitmap_context(
|
||||||
|
Some(bytes.as_mut_ptr() as *mut _),
|
||||||
|
cx_bounds.width() as usize,
|
||||||
|
cx_bounds.height() as usize,
|
||||||
|
8,
|
||||||
|
cx_bounds.width() as usize,
|
||||||
|
&CGColorSpace::create_device_gray(),
|
||||||
|
kCGImageAlphaOnly,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
RasterizationOptions::Bgra => {
|
||||||
|
bytes = vec![0; cx_bounds.width() as usize * 4 * cx_bounds.height() as usize];
|
||||||
|
cx = CGContext::create_bitmap_context(
|
||||||
|
Some(bytes.as_mut_ptr() as *mut _),
|
||||||
|
cx_bounds.width() as usize,
|
||||||
|
cx_bounds.height() as usize,
|
||||||
|
8,
|
||||||
|
cx_bounds.width() as usize * 4,
|
||||||
|
&CGColorSpace::create_device_rgb(),
|
||||||
|
kCGImageAlphaPremultipliedLast,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Move the origin to bottom left and account for scaling, this
|
||||||
|
// makes drawing text consistent with the font-kit's raster_bounds.
|
||||||
|
cx.translate(
|
||||||
|
-glyph_bounds.origin_x() as CGFloat,
|
||||||
|
(glyph_bounds.origin_y() + glyph_bounds.height()) as CGFloat,
|
||||||
|
);
|
||||||
|
cx.scale(scale_factor as CGFloat, scale_factor as CGFloat);
|
||||||
|
|
||||||
|
cx.set_allows_font_subpixel_positioning(true);
|
||||||
|
cx.set_should_subpixel_position_fonts(true);
|
||||||
|
cx.set_allows_font_subpixel_quantization(false);
|
||||||
|
cx.set_should_subpixel_quantize_fonts(false);
|
||||||
|
font.native_font()
|
||||||
|
.clone_with_font_size(font_size as CGFloat)
|
||||||
|
.draw_glyphs(
|
||||||
|
&[u32::from(glyph_id) as CGGlyph],
|
||||||
|
&[CGPoint::new(
|
||||||
|
(f32::from(subpixel_shift.x) / scale_factor) as CGFloat,
|
||||||
|
(f32::from(subpixel_shift.y) / scale_factor) as CGFloat,
|
||||||
|
)],
|
||||||
|
cx,
|
||||||
|
);
|
||||||
|
|
||||||
|
if let RasterizationOptions::Bgra = options {
|
||||||
|
// Convert from RGBA with premultiplied alpha to BGRA with straight alpha.
|
||||||
|
for pixel in bytes.chunks_exact_mut(4) {
|
||||||
|
pixel.swap(0, 2);
|
||||||
|
let a = pixel[3] as f32 / 255.;
|
||||||
|
pixel[0] = (pixel[0] as f32 / a) as u8;
|
||||||
|
pixel[1] = (pixel[1] as f32 / a) as u8;
|
||||||
|
pixel[2] = (pixel[2] as f32 / a) as u8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Some((cx_bounds.into(), bytes))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn layout_line(
|
||||||
|
&mut self,
|
||||||
|
text: &str,
|
||||||
|
font_size: Pixels,
|
||||||
|
runs: &[(usize, RunStyle)],
|
||||||
|
) -> LineLayout {
|
||||||
|
// Construct the attributed string, converting UTF8 ranges to UTF16 ranges.
|
||||||
|
let mut string = CFMutableAttributedString::new();
|
||||||
|
{
|
||||||
|
string.replace_str(&CFString::new(text), CFRange::init(0, 0));
|
||||||
|
let utf16_line_len = string.char_len() as usize;
|
||||||
|
|
||||||
|
let last_run: RefCell<Option<(usize, FontId)>> = Default::default();
|
||||||
|
let font_runs = runs
|
||||||
|
.iter()
|
||||||
|
.filter_map(|(len, style)| {
|
||||||
|
let mut last_run = last_run.borrow_mut();
|
||||||
|
if let Some((last_len, last_font_id)) = last_run.as_mut() {
|
||||||
|
if style.font_id == *last_font_id {
|
||||||
|
*last_len += *len;
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
let result = (*last_len, *last_font_id);
|
||||||
|
*last_len = *len;
|
||||||
|
*last_font_id = style.font_id;
|
||||||
|
Some(result)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
*last_run = Some((*len, style.font_id));
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.chain(std::iter::from_fn(|| last_run.borrow_mut().take()));
|
||||||
|
|
||||||
|
let mut ix_converter = StringIndexConverter::new(text);
|
||||||
|
for (run_len, font_id) in font_runs {
|
||||||
|
let utf8_end = ix_converter.utf8_ix + run_len;
|
||||||
|
let utf16_start = ix_converter.utf16_ix;
|
||||||
|
|
||||||
|
if utf16_start >= utf16_line_len {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ix_converter.advance_to_utf8_ix(utf8_end);
|
||||||
|
let utf16_end = cmp::min(ix_converter.utf16_ix, utf16_line_len);
|
||||||
|
|
||||||
|
let cf_range =
|
||||||
|
CFRange::init(utf16_start as isize, (utf16_end - utf16_start) as isize);
|
||||||
|
let font = &self.fonts[font_id.0];
|
||||||
|
unsafe {
|
||||||
|
string.set_attribute(
|
||||||
|
cf_range,
|
||||||
|
kCTFontAttributeName,
|
||||||
|
&font.native_font().clone_with_font_size(font_size.into()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if utf16_end == utf16_line_len {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Retrieve the glyphs from the shaped line, converting UTF16 offsets to UTF8 offsets.
|
||||||
|
let line = CTLine::new_with_attributed_string(string.as_concrete_TypeRef());
|
||||||
|
|
||||||
|
let mut runs = Vec::new();
|
||||||
|
for run in line.glyph_runs().into_iter() {
|
||||||
|
let attributes = run.attributes().unwrap();
|
||||||
|
let font = unsafe {
|
||||||
|
attributes
|
||||||
|
.get(kCTFontAttributeName)
|
||||||
|
.downcast::<CTFont>()
|
||||||
|
.unwrap()
|
||||||
|
};
|
||||||
|
let font_id = self.id_for_native_font(font);
|
||||||
|
|
||||||
|
let mut ix_converter = StringIndexConverter::new(text);
|
||||||
|
let mut glyphs = Vec::new();
|
||||||
|
for ((glyph_id, position), glyph_utf16_ix) in run
|
||||||
|
.glyphs()
|
||||||
|
.iter()
|
||||||
|
.zip(run.positions().iter())
|
||||||
|
.zip(run.string_indices().iter())
|
||||||
|
{
|
||||||
|
let glyph_utf16_ix = usize::try_from(*glyph_utf16_ix).unwrap();
|
||||||
|
ix_converter.advance_to_utf16_ix(glyph_utf16_ix);
|
||||||
|
glyphs.push(Glyph {
|
||||||
|
id: (*glyph_id).into(),
|
||||||
|
position: point(position.x as f32, position.y as f32).map(px),
|
||||||
|
index: ix_converter.utf8_ix,
|
||||||
|
is_emoji: self.is_emoji(font_id),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
runs.push(Run { font_id, glyphs })
|
||||||
|
}
|
||||||
|
|
||||||
|
let typographic_bounds = line.get_typographic_bounds();
|
||||||
|
LineLayout {
|
||||||
|
width: typographic_bounds.width.into(),
|
||||||
|
ascent: typographic_bounds.ascent.into(),
|
||||||
|
descent: typographic_bounds.descent.into(),
|
||||||
|
runs,
|
||||||
|
font_size,
|
||||||
|
len: text.len(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn wrap_line(
|
||||||
|
&self,
|
||||||
|
text: &str,
|
||||||
|
font_id: FontId,
|
||||||
|
font_size: Pixels,
|
||||||
|
width: Pixels,
|
||||||
|
) -> Vec<usize> {
|
||||||
|
let mut string = CFMutableAttributedString::new();
|
||||||
|
string.replace_str(&CFString::new(text), CFRange::init(0, 0));
|
||||||
|
let cf_range = CFRange::init(0, text.encode_utf16().count() as isize);
|
||||||
|
let font = &self.fonts[font_id.0];
|
||||||
|
unsafe {
|
||||||
|
string.set_attribute(
|
||||||
|
cf_range,
|
||||||
|
kCTFontAttributeName,
|
||||||
|
&font.native_font().clone_with_font_size(font_size.into()),
|
||||||
|
);
|
||||||
|
|
||||||
|
let typesetter = CTTypesetterCreateWithAttributedString(string.as_concrete_TypeRef());
|
||||||
|
let mut ix_converter = StringIndexConverter::new(text);
|
||||||
|
let mut break_indices = Vec::new();
|
||||||
|
while ix_converter.utf8_ix < text.len() {
|
||||||
|
let utf16_len = CTTypesetterSuggestLineBreak(
|
||||||
|
typesetter,
|
||||||
|
ix_converter.utf16_ix as isize,
|
||||||
|
width.into(),
|
||||||
|
) as usize;
|
||||||
|
ix_converter.advance_to_utf16_ix(ix_converter.utf16_ix + utf16_len);
|
||||||
|
if ix_converter.utf8_ix >= text.len() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break_indices.push(ix_converter.utf8_ix as usize);
|
||||||
|
}
|
||||||
|
break_indices
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
struct StringIndexConverter<'a> {
|
||||||
|
text: &'a str,
|
||||||
|
utf8_ix: usize,
|
||||||
|
utf16_ix: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> StringIndexConverter<'a> {
|
||||||
|
fn new(text: &'a str) -> Self {
|
||||||
|
Self {
|
||||||
|
text,
|
||||||
|
utf8_ix: 0,
|
||||||
|
utf16_ix: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn advance_to_utf8_ix(&mut self, utf8_target: usize) {
|
||||||
|
for (ix, c) in self.text[self.utf8_ix..].char_indices() {
|
||||||
|
if self.utf8_ix + ix >= utf8_target {
|
||||||
|
self.utf8_ix += ix;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.utf16_ix += c.len_utf16();
|
||||||
|
}
|
||||||
|
self.utf8_ix = self.text.len();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn advance_to_utf16_ix(&mut self, utf16_target: usize) {
|
||||||
|
for (ix, c) in self.text[self.utf8_ix..].char_indices() {
|
||||||
|
if self.utf16_ix >= utf16_target {
|
||||||
|
self.utf8_ix += ix;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.utf16_ix += c.len_utf16();
|
||||||
|
}
|
||||||
|
self.utf8_ix = self.text.len();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct __CFTypesetter(c_void);
|
||||||
|
|
||||||
|
pub type CTTypesetterRef = *const __CFTypesetter;
|
||||||
|
|
||||||
|
#[link(name = "CoreText", kind = "framework")]
|
||||||
|
extern "C" {
|
||||||
|
fn CTTypesetterCreateWithAttributedString(string: CFAttributedStringRef) -> CTTypesetterRef;
|
||||||
|
|
||||||
|
fn CTTypesetterSuggestLineBreak(
|
||||||
|
typesetter: CTTypesetterRef,
|
||||||
|
start_index: CFIndex,
|
||||||
|
width: f64,
|
||||||
|
) -> CFIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Metrics> for FontMetrics {
|
||||||
|
fn from(metrics: Metrics) -> Self {
|
||||||
|
FontMetrics {
|
||||||
|
units_per_em: metrics.units_per_em,
|
||||||
|
ascent: metrics.ascent,
|
||||||
|
descent: metrics.descent,
|
||||||
|
line_gap: metrics.line_gap,
|
||||||
|
underline_position: metrics.underline_position,
|
||||||
|
underline_thickness: metrics.underline_thickness,
|
||||||
|
cap_height: metrics.cap_height,
|
||||||
|
x_height: metrics.x_height,
|
||||||
|
bounding_box: metrics.bounding_box.into(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<RectF> for Bounds<f32> {
|
||||||
|
fn from(rect: RectF) -> Self {
|
||||||
|
Bounds {
|
||||||
|
origin: point(rect.origin_x(), rect.origin_y()),
|
||||||
|
size: size(rect.width(), rect.height()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<RectI> for Bounds<u32> {
|
||||||
|
fn from(rect: RectI) -> Self {
|
||||||
|
Bounds {
|
||||||
|
origin: point(rect.origin_x() as u32, rect.origin_y() as u32),
|
||||||
|
size: size(rect.width() as u32, rect.height() as u32),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Point<u32>> for Vector2I {
|
||||||
|
fn from(size: Point<u32>) -> Self {
|
||||||
|
Vector2I::new(size.x as i32, size.y as i32)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Vector2F> for Size<f32> {
|
||||||
|
fn from(vec: Vector2F) -> Self {
|
||||||
|
size(vec.x(), vec.y())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// #[cfg(test)]
|
||||||
|
// mod tests {
|
||||||
|
// use super::*;
|
||||||
|
// use crate::AppContext;
|
||||||
|
// use font_kit::properties::{Style, Weight};
|
||||||
|
// use platform::FontSystem as _;
|
||||||
|
|
||||||
|
// #[crate::test(self, retries = 5)]
|
||||||
|
// fn test_layout_str(_: &mut AppContext) {
|
||||||
|
// // This is failing intermittently on CI and we don't have time to figure it out
|
||||||
|
// let fonts = FontSystem::new();
|
||||||
|
// let menlo = fonts.load_family("Menlo", &Default::default()).unwrap();
|
||||||
|
// let menlo_regular = RunStyle {
|
||||||
|
// font_id: fonts.select_font(&menlo, &Properties::new()).unwrap(),
|
||||||
|
// color: Default::default(),
|
||||||
|
// underline: Default::default(),
|
||||||
|
// };
|
||||||
|
// let menlo_italic = RunStyle {
|
||||||
|
// font_id: fonts
|
||||||
|
// .select_font(&menlo, Properties::new().style(Style::Italic))
|
||||||
|
// .unwrap(),
|
||||||
|
// color: Default::default(),
|
||||||
|
// underline: Default::default(),
|
||||||
|
// };
|
||||||
|
// let menlo_bold = RunStyle {
|
||||||
|
// font_id: fonts
|
||||||
|
// .select_font(&menlo, Properties::new().weight(Weight::BOLD))
|
||||||
|
// .unwrap(),
|
||||||
|
// color: Default::default(),
|
||||||
|
// underline: Default::default(),
|
||||||
|
// };
|
||||||
|
// assert_ne!(menlo_regular, menlo_italic);
|
||||||
|
// assert_ne!(menlo_regular, menlo_bold);
|
||||||
|
// assert_ne!(menlo_italic, menlo_bold);
|
||||||
|
|
||||||
|
// let line = fonts.layout_line(
|
||||||
|
// "hello world",
|
||||||
|
// 16.0,
|
||||||
|
// &[(2, menlo_bold), (4, menlo_italic), (5, menlo_regular)],
|
||||||
|
// );
|
||||||
|
// assert_eq!(line.runs.len(), 3);
|
||||||
|
// assert_eq!(line.runs[0].font_id, menlo_bold.font_id);
|
||||||
|
// assert_eq!(line.runs[0].glyphs.len(), 2);
|
||||||
|
// assert_eq!(line.runs[1].font_id, menlo_italic.font_id);
|
||||||
|
// assert_eq!(line.runs[1].glyphs.len(), 4);
|
||||||
|
// assert_eq!(line.runs[2].font_id, menlo_regular.font_id);
|
||||||
|
// assert_eq!(line.runs[2].glyphs.len(), 5);
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_glyph_offsets() -> anyhow::Result<()> {
|
||||||
|
// let fonts = FontSystem::new();
|
||||||
|
// let zapfino = fonts.load_family("Zapfino", &Default::default())?;
|
||||||
|
// let zapfino_regular = RunStyle {
|
||||||
|
// font_id: fonts.select_font(&zapfino, &Properties::new())?,
|
||||||
|
// color: Default::default(),
|
||||||
|
// underline: Default::default(),
|
||||||
|
// };
|
||||||
|
// let menlo = fonts.load_family("Menlo", &Default::default())?;
|
||||||
|
// let menlo_regular = RunStyle {
|
||||||
|
// font_id: fonts.select_font(&menlo, &Properties::new())?,
|
||||||
|
// color: Default::default(),
|
||||||
|
// underline: Default::default(),
|
||||||
|
// };
|
||||||
|
|
||||||
|
// let text = "This is, m𐍈re 𐍈r less, Zapfino!𐍈";
|
||||||
|
// let line = fonts.layout_line(
|
||||||
|
// text,
|
||||||
|
// 16.0,
|
||||||
|
// &[
|
||||||
|
// (9, zapfino_regular),
|
||||||
|
// (13, menlo_regular),
|
||||||
|
// (text.len() - 22, zapfino_regular),
|
||||||
|
// ],
|
||||||
|
// );
|
||||||
|
// assert_eq!(
|
||||||
|
// line.runs
|
||||||
|
// .iter()
|
||||||
|
// .flat_map(|r| r.glyphs.iter())
|
||||||
|
// .map(|g| g.index)
|
||||||
|
// .collect::<Vec<_>>(),
|
||||||
|
// vec![0, 2, 4, 5, 7, 8, 9, 10, 14, 15, 16, 17, 21, 22, 23, 24, 26, 27, 28, 29, 36, 37],
|
||||||
|
// );
|
||||||
|
// Ok(())
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// #[ignore]
|
||||||
|
// fn test_rasterize_glyph() {
|
||||||
|
// use std::{fs::File, io::BufWriter, path::Path};
|
||||||
|
|
||||||
|
// let fonts = FontSystem::new();
|
||||||
|
// let font_ids = fonts.load_family("Fira Code", &Default::default()).unwrap();
|
||||||
|
// let font_id = fonts.select_font(&font_ids, &Default::default()).unwrap();
|
||||||
|
// let glyph_id = fonts.glyph_for_char(font_id, 'G').unwrap();
|
||||||
|
|
||||||
|
// const VARIANTS: usize = 1;
|
||||||
|
// for i in 0..VARIANTS {
|
||||||
|
// let variant = i as f32 / VARIANTS as f32;
|
||||||
|
// let (bounds, bytes) = fonts
|
||||||
|
// .rasterize_glyph(
|
||||||
|
// font_id,
|
||||||
|
// 16.0,
|
||||||
|
// glyph_id,
|
||||||
|
// vec2f(variant, variant),
|
||||||
|
// 2.,
|
||||||
|
// RasterizationOptions::Alpha,
|
||||||
|
// )
|
||||||
|
// .unwrap();
|
||||||
|
|
||||||
|
// let name = format!("/Users/as-cii/Desktop/twog-{}.png", i);
|
||||||
|
// let path = Path::new(&name);
|
||||||
|
// let file = File::create(path).unwrap();
|
||||||
|
// let w = &mut BufWriter::new(file);
|
||||||
|
|
||||||
|
// let mut encoder = png::Encoder::new(w, bounds.width() as u32, bounds.height() as u32);
|
||||||
|
// encoder.set_color(png::ColorType::Grayscale);
|
||||||
|
// encoder.set_depth(png::BitDepth::Eight);
|
||||||
|
// let mut writer = encoder.write_header().unwrap();
|
||||||
|
// writer.write_image_data(&bytes).unwrap();
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_wrap_line() {
|
||||||
|
// let fonts = FontSystem::new();
|
||||||
|
// let font_ids = fonts.load_family("Helvetica", &Default::default()).unwrap();
|
||||||
|
// let font_id = fonts.select_font(&font_ids, &Default::default()).unwrap();
|
||||||
|
|
||||||
|
// let line = "one two three four five\n";
|
||||||
|
// let wrap_boundaries = fonts.wrap_line(line, font_id, 16., 64.0);
|
||||||
|
// assert_eq!(wrap_boundaries, &["one two ".len(), "one two three ".len()]);
|
||||||
|
|
||||||
|
// let line = "aaa ααα ✋✋✋ 🎉🎉🎉\n";
|
||||||
|
// let wrap_boundaries = fonts.wrap_line(line, font_id, 16., 64.0);
|
||||||
|
// assert_eq!(
|
||||||
|
// wrap_boundaries,
|
||||||
|
// &["aaa ααα ".len(), "aaa ααα ✋✋✋ ".len(),]
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
|
||||||
|
// #[test]
|
||||||
|
// fn test_layout_line_bom_char() {
|
||||||
|
// let fonts = FontSystem::new();
|
||||||
|
// let font_ids = fonts.load_family("Helvetica", &Default::default()).unwrap();
|
||||||
|
// let style = RunStyle {
|
||||||
|
// font_id: fonts.select_font(&font_ids, &Default::default()).unwrap(),
|
||||||
|
// color: Default::default(),
|
||||||
|
// underline: Default::default(),
|
||||||
|
// };
|
||||||
|
|
||||||
|
// let line = "\u{feff}";
|
||||||
|
// let layout = fonts.layout_line(line, 16., &[(line.len(), style)]);
|
||||||
|
// assert_eq!(layout.len, line.len());
|
||||||
|
// assert!(layout.runs.is_empty());
|
||||||
|
|
||||||
|
// let line = "a\u{feff}b";
|
||||||
|
// let layout = fonts.layout_line(line, 16., &[(line.len(), style)]);
|
||||||
|
// assert_eq!(layout.len, line.len());
|
||||||
|
// assert_eq!(layout.runs.len(), 1);
|
||||||
|
// assert_eq!(layout.runs[0].glyphs.len(), 2);
|
||||||
|
// assert_eq!(layout.runs[0].glyphs[0].id, 68); // a
|
||||||
|
// // There's no glyph for \u{feff}
|
||||||
|
// assert_eq!(layout.runs[0].glyphs[1].id, 69); // b
|
||||||
|
// }
|
||||||
|
// }
|
|
@ -9,7 +9,51 @@ impl TestPlatform {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Platform for TestPlatform {
|
impl Platform for TestPlatform {
|
||||||
fn font_system(&self) -> std::sync::Arc<dyn crate::PlatformTextSystem> {
|
fn executor(&self) -> std::rc::Rc<crate::ForegroundExecutor> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn text_system(&self) -> std::sync::Arc<dyn crate::PlatformTextSystem> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(&self, on_finish_launching: Box<dyn FnOnce()>) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn quit(&self) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn restart(&self) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn activate(&self, ignoring_other_apps: bool) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn hide(&self) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn hide_other_apps(&self) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn unhide_other_apps(&self) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn screens(&self) -> Vec<std::rc::Rc<dyn crate::PlatformScreen>> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn screen_by_id(&self, id: uuid::Uuid) -> Option<std::rc::Rc<dyn crate::PlatformScreen>> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main_window(&self) -> Option<crate::AnyWindowHandle> {
|
||||||
todo!()
|
todo!()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -21,7 +65,101 @@ impl Platform for TestPlatform {
|
||||||
todo!()
|
todo!()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn dispatcher(&self) -> std::sync::Arc<dyn crate::PlatformDispatcher> {
|
fn open_url(&self, url: &str) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_open_urls(&self, callback: Box<dyn FnMut(Vec<String>)>) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn prompt_for_paths(
|
||||||
|
&self,
|
||||||
|
options: crate::PathPromptOptions,
|
||||||
|
) -> futures::channel::oneshot::Receiver<Option<Vec<std::path::PathBuf>>> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn prompt_for_new_path(
|
||||||
|
&self,
|
||||||
|
directory: &std::path::Path,
|
||||||
|
) -> futures::channel::oneshot::Receiver<Option<std::path::PathBuf>> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn reveal_path(&self, path: &std::path::Path) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_become_active(&self, callback: Box<dyn FnMut()>) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_resign_active(&self, callback: Box<dyn FnMut()>) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_quit(&self, callback: Box<dyn FnMut()>) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_reopen(&self, callback: Box<dyn FnMut()>) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_event(&self, callback: Box<dyn FnMut(crate::Event) -> bool>) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn os_name(&self) -> &'static str {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn os_version(&self) -> anyhow::Result<crate::SemanticVersion> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn app_version(&self) -> anyhow::Result<crate::SemanticVersion> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn app_path(&self) -> anyhow::Result<std::path::PathBuf> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn local_timezone(&self) -> time::UtcOffset {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn path_for_auxiliary_executable(&self, name: &str) -> anyhow::Result<std::path::PathBuf> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn set_cursor_style(&self, style: crate::CursorStyle) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn should_auto_hide_scrollbars(&self) -> bool {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn write_to_clipboard(&self, item: crate::ClipboardItem) {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn read_from_clipboard(&self) -> Option<crate::ClipboardItem> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn write_credentials(&self, url: &str, username: &str, password: &[u8]) -> anyhow::Result<()> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn read_credentials(&self, url: &str) -> anyhow::Result<Option<(String, Vec<u8>)>> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn delete_credentials(&self, url: &str) -> anyhow::Result<()> {
|
||||||
todo!()
|
todo!()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
use crate::{text::GlyphId, FontId};
|
||||||
|
|
||||||
use super::{Bounds, Hsla, Pixels, Point};
|
use super::{Bounds, Hsla, Pixels, Point};
|
||||||
use bytemuck::{Pod, Zeroable};
|
use bytemuck::{Pod, Zeroable};
|
||||||
use plane_split::BspSplitter;
|
use plane_split::BspSplitter;
|
||||||
|
@ -39,7 +41,7 @@ impl Scene {
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub enum Primitive {
|
pub enum Primitive {
|
||||||
Quad(Quad),
|
Quad(Quad),
|
||||||
Glyph(Glyph),
|
Glyph(RenderedGlyph),
|
||||||
Underline(Underline),
|
Underline(Underline),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -58,7 +60,7 @@ impl Primitive {
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
pub struct PrimitiveBatch {
|
pub struct PrimitiveBatch {
|
||||||
pub quads: Vec<Quad>,
|
pub quads: Vec<Quad>,
|
||||||
pub glyphs: Vec<Glyph>,
|
pub glyphs: Vec<RenderedGlyph>,
|
||||||
pub underlines: Vec<Underline>,
|
pub underlines: Vec<Underline>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,26 +98,24 @@ unsafe impl Zeroable for Quad {}
|
||||||
|
|
||||||
unsafe impl Pod for Quad {}
|
unsafe impl Pod for Quad {}
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, Eq, PartialEq, Hash)]
|
|
||||||
pub struct GlyphId(u32);
|
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
|
||||||
pub struct Glyph {
|
|
||||||
pub id: GlyphId,
|
|
||||||
pub position: Point<Pixels>,
|
|
||||||
pub color: Hsla,
|
|
||||||
pub index: usize,
|
|
||||||
pub is_emoji: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<Quad> for Primitive {
|
impl From<Quad> for Primitive {
|
||||||
fn from(quad: Quad) -> Self {
|
fn from(quad: Quad) -> Self {
|
||||||
Primitive::Quad(quad)
|
Primitive::Quad(quad)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<Glyph> for Primitive {
|
#[derive(Debug, Clone, Copy)]
|
||||||
fn from(glyph: Glyph) -> Self {
|
#[repr(C)]
|
||||||
|
pub struct RenderedGlyph {
|
||||||
|
pub font_id: FontId,
|
||||||
|
pub font_size: f32,
|
||||||
|
pub id: GlyphId,
|
||||||
|
pub origin: Point<Pixels>,
|
||||||
|
pub color: Hsla,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<RenderedGlyph> for Primitive {
|
||||||
|
fn from(glyph: RenderedGlyph) -> Self {
|
||||||
Primitive::Glyph(glyph)
|
Primitive::Glyph(glyph)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
use crate::{black, px};
|
use crate::{black, px};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
point, Bounds, FontId, Glyph, Hsla, Pixels, PlatformTextSystem, Point, UnderlineStyle,
|
point, Bounds, FontId, Hsla, Pixels, PlatformTextSystem, Point, UnderlineStyle, WindowContext,
|
||||||
WindowContext,
|
|
||||||
};
|
};
|
||||||
use parking_lot::{Mutex, RwLock, RwLockUpgradableReadGuard};
|
use parking_lot::{Mutex, RwLock, RwLockUpgradableReadGuard};
|
||||||
use smallvec::SmallVec;
|
use smallvec::SmallVec;
|
||||||
|
@ -27,6 +26,35 @@ pub struct RunStyle {
|
||||||
pub underline: Option<UnderlineStyle>,
|
pub underline: Option<UnderlineStyle>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, Debug, Eq, PartialEq, Hash)]
|
||||||
|
pub struct GlyphId(u32);
|
||||||
|
|
||||||
|
impl From<GlyphId> for u32 {
|
||||||
|
fn from(value: GlyphId) -> Self {
|
||||||
|
value.0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<u16> for GlyphId {
|
||||||
|
fn from(num: u16) -> Self {
|
||||||
|
GlyphId(num as u32)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<u32> for GlyphId {
|
||||||
|
fn from(num: u32) -> Self {
|
||||||
|
GlyphId(num)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct Glyph {
|
||||||
|
pub id: GlyphId,
|
||||||
|
pub position: Point<Pixels>,
|
||||||
|
pub index: usize,
|
||||||
|
pub is_emoji: bool,
|
||||||
|
}
|
||||||
|
|
||||||
impl TextLayoutCache {
|
impl TextLayoutCache {
|
||||||
pub fn new(fonts: Arc<dyn PlatformTextSystem>) -> Self {
|
pub fn new(fonts: Arc<dyn PlatformTextSystem>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
|
@ -726,7 +754,7 @@ mod tests {
|
||||||
let cx = AppContext::test();
|
let cx = AppContext::test();
|
||||||
|
|
||||||
let font_cache = cx.font_cache().clone();
|
let font_cache = cx.font_cache().clone();
|
||||||
let font_system = cx.platform().font_system();
|
let font_system = cx.platform().text_system();
|
||||||
let family = font_cache
|
let family = font_cache
|
||||||
.load_family(&["Courier"], &Default::default())
|
.load_family(&["Courier"], &Default::default())
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
@ -793,7 +821,7 @@ mod tests {
|
||||||
fn test_wrap_shaped_line() {
|
fn test_wrap_shaped_line() {
|
||||||
let cx = AppContext::test();
|
let cx = AppContext::test();
|
||||||
let font_cache = cx.font_cache().clone();
|
let font_cache = cx.font_cache().clone();
|
||||||
let font_system = cx.platform().font_system();
|
let font_system = cx.platform().text_system();
|
||||||
let text_layout_cache = TextLayoutCache::new(font_system.clone());
|
let text_layout_cache = TextLayoutCache::new(font_system.clone());
|
||||||
|
|
||||||
let family = font_cache
|
let family = font_cache
|
||||||
|
|
|
@ -13,22 +13,16 @@ test-support = []
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
anyhow.workspace = true
|
anyhow.workspace = true
|
||||||
bytemuck = "1.14.0"
|
|
||||||
derive_more.workspace = true
|
derive_more.workspace = true
|
||||||
gpui2 = { path = "../gpui2" }
|
gpui2 = { path = "../gpui2" }
|
||||||
itertools = "0.11.0"
|
|
||||||
log.workspace = true
|
log.workspace = true
|
||||||
plane-split = "0.18.0"
|
|
||||||
raw-window-handle = "0.5.2"
|
|
||||||
refineable = { path = "../refineable" }
|
refineable = { path = "../refineable" }
|
||||||
rust-embed.workspace = true
|
rust-embed.workspace = true
|
||||||
serde.workspace = true
|
serde.workspace = true
|
||||||
settings = { path = "../settings" }
|
settings = { path = "../settings" }
|
||||||
simplelog = "0.9"
|
simplelog = "0.9"
|
||||||
slotmap = "1.0.6"
|
|
||||||
theme = { path = "../theme" }
|
theme = { path = "../theme" }
|
||||||
util = { path = "../util" }
|
util = { path = "../util" }
|
||||||
wgpu = "0.17.0"
|
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
gpui2 = { path = "../gpui2", features = ["test-support"] }
|
gpui2 = { path = "../gpui2", features = ["test-support"] }
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue