mirror of
https://github.com/zed-industries/zed.git
synced 2025-01-23 18:32:17 +00:00
Add ability to debug element trees as JSON
This commit is contained in:
parent
57a3207897
commit
0664321125
25 changed files with 554 additions and 41 deletions
6
Cargo.lock
generated
6
Cargo.lock
generated
|
@ -930,6 +930,8 @@ dependencies = [
|
||||||
"rand 0.8.3",
|
"rand 0.8.3",
|
||||||
"replace_with",
|
"replace_with",
|
||||||
"resvg",
|
"resvg",
|
||||||
|
"serde",
|
||||||
|
"serde_json",
|
||||||
"simplelog",
|
"simplelog",
|
||||||
"smallvec",
|
"smallvec",
|
||||||
"smol",
|
"smol",
|
||||||
|
@ -1691,9 +1693,9 @@ checksum = "388a1df253eca08550bef6c72392cfe7c30914bf41df5269b68cbd6ff8f570a3"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "serde"
|
name = "serde"
|
||||||
version = "1.0.124"
|
version = "1.0.125"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "bd761ff957cb2a45fbb9ab3da6512de9de55872866160b23c25f1a841e99d29f"
|
checksum = "558dc50e1a5a5fa7112ca2ce4effcb321b0300c0d4ccf0776a9f60cd89031171"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "serde_json"
|
name = "serde_json"
|
||||||
|
|
|
@ -18,6 +18,8 @@ pathfinder_geometry = "0.5"
|
||||||
rand = "0.8.3"
|
rand = "0.8.3"
|
||||||
replace_with = "0.1.7"
|
replace_with = "0.1.7"
|
||||||
resvg = "0.14"
|
resvg = "0.14"
|
||||||
|
serde = "1.0.125"
|
||||||
|
serde_json = "1.0.64"
|
||||||
smallvec = "1.6.1"
|
smallvec = "1.6.1"
|
||||||
smol = "1.2"
|
smol = "1.2"
|
||||||
tiny-skia = "0.5"
|
tiny-skia = "0.5"
|
||||||
|
|
|
@ -2,9 +2,10 @@ use gpui::{
|
||||||
color::ColorU,
|
color::ColorU,
|
||||||
fonts::{Properties, Weight},
|
fonts::{Properties, Weight},
|
||||||
platform::{current as platform, Runner},
|
platform::{current as platform, Runner},
|
||||||
Element as _, Quad,
|
DebugContext, Element as _, Quad,
|
||||||
};
|
};
|
||||||
use log::LevelFilter;
|
use log::LevelFilter;
|
||||||
|
use pathfinder_geometry::rect::RectF;
|
||||||
use simplelog::SimpleLogger;
|
use simplelog::SimpleLogger;
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
|
@ -59,7 +60,7 @@ impl gpui::Element for TextElement {
|
||||||
|
|
||||||
fn paint(
|
fn paint(
|
||||||
&mut self,
|
&mut self,
|
||||||
bounds: pathfinder_geometry::rect::RectF,
|
bounds: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
ctx: &mut gpui::PaintContext,
|
ctx: &mut gpui::PaintContext,
|
||||||
) -> Self::PaintState {
|
) -> Self::PaintState {
|
||||||
|
@ -109,11 +110,21 @@ impl gpui::Element for TextElement {
|
||||||
fn dispatch_event(
|
fn dispatch_event(
|
||||||
&mut self,
|
&mut self,
|
||||||
_: &gpui::Event,
|
_: &gpui::Event,
|
||||||
_: pathfinder_geometry::rect::RectF,
|
_: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
_: &mut Self::PaintState,
|
_: &mut Self::PaintState,
|
||||||
_: &mut gpui::EventContext,
|
_: &mut gpui::EventContext,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
_: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
_: &DebugContext,
|
||||||
|
) -> gpui::json::Value {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
9
gpui/src/color.rs
Normal file
9
gpui/src/color.rs
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
use crate::json::ToJson;
|
||||||
|
pub use pathfinder_color::*;
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
|
impl ToJson for ColorU {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
json!(format!("0x{:x}{:x}{:x}", self.r, self.g, self.b))
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,8 +1,10 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
AfterLayoutContext, Element, ElementBox, Event, EventContext, LayoutContext, PaintContext,
|
json, AfterLayoutContext, DebugContext, Element, ElementBox, Event, EventContext,
|
||||||
SizeConstraint,
|
LayoutContext, PaintContext, SizeConstraint,
|
||||||
};
|
};
|
||||||
|
use json::ToJson;
|
||||||
use pathfinder_geometry::vector::{vec2f, Vector2F};
|
use pathfinder_geometry::vector::{vec2f, Vector2F};
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
pub struct Align {
|
pub struct Align {
|
||||||
child: ElementBox,
|
child: ElementBox,
|
||||||
|
@ -79,4 +81,19 @@ impl Element for Align {
|
||||||
) -> bool {
|
) -> bool {
|
||||||
self.child.dispatch_event(event, ctx)
|
self.child.dispatch_event(event, ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: pathfinder_geometry::rect::RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "Align",
|
||||||
|
"alignment": self.alignment.to_json(),
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"child": self.child.debug(ctx),
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
use super::Element;
|
use super::Element;
|
||||||
use crate::PaintContext;
|
use crate::{
|
||||||
|
json::{self, json},
|
||||||
|
DebugContext, PaintContext,
|
||||||
|
};
|
||||||
|
use json::ToJson;
|
||||||
use pathfinder_geometry::{
|
use pathfinder_geometry::{
|
||||||
rect::RectF,
|
rect::RectF,
|
||||||
vector::{vec2f, Vector2F},
|
vector::{vec2f, Vector2F},
|
||||||
|
@ -70,4 +74,14 @@ where
|
||||||
) -> bool {
|
) -> bool {
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
_: &DebugContext,
|
||||||
|
) -> json::Value {
|
||||||
|
json!({"type": "Canvas", "bounds": bounds.to_json()})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
|
use json::ToJson;
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
AfterLayoutContext, Element, ElementBox, Event, EventContext, LayoutContext, PaintContext,
|
geometry::{rect::RectF, vector::Vector2F},
|
||||||
SizeConstraint,
|
json, AfterLayoutContext, DebugContext, Element, ElementBox, Event, EventContext,
|
||||||
|
LayoutContext, PaintContext, SizeConstraint,
|
||||||
};
|
};
|
||||||
use pathfinder_geometry::vector::Vector2F;
|
|
||||||
|
|
||||||
pub struct ConstrainedBox {
|
pub struct ConstrainedBox {
|
||||||
child: ElementBox,
|
child: ElementBox,
|
||||||
|
@ -63,7 +66,7 @@ impl Element for ConstrainedBox {
|
||||||
|
|
||||||
fn paint(
|
fn paint(
|
||||||
&mut self,
|
&mut self,
|
||||||
bounds: pathfinder_geometry::rect::RectF,
|
bounds: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
ctx: &mut PaintContext,
|
ctx: &mut PaintContext,
|
||||||
) -> Self::PaintState {
|
) -> Self::PaintState {
|
||||||
|
@ -73,11 +76,21 @@ impl Element for ConstrainedBox {
|
||||||
fn dispatch_event(
|
fn dispatch_event(
|
||||||
&mut self,
|
&mut self,
|
||||||
event: &Event,
|
event: &Event,
|
||||||
_: pathfinder_geometry::rect::RectF,
|
_: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
_: &mut Self::PaintState,
|
_: &mut Self::PaintState,
|
||||||
ctx: &mut EventContext,
|
ctx: &mut EventContext,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
self.child.dispatch_event(event, ctx)
|
self.child.dispatch_event(event, ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
_: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> json::Value {
|
||||||
|
json!({"type": "ConstrainedBox", "constraint": self.constraint.to_json(), "child": self.child.debug(ctx)})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
use pathfinder_geometry::rect::RectF;
|
use pathfinder_geometry::rect::RectF;
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
color::ColorU,
|
color::ColorU,
|
||||||
geometry::vector::{vec2f, Vector2F},
|
geometry::vector::{vec2f, Vector2F},
|
||||||
|
json::ToJson,
|
||||||
scene::{self, Border, Quad},
|
scene::{self, Border, Quad},
|
||||||
AfterLayoutContext, Element, ElementBox, Event, EventContext, LayoutContext, PaintContext,
|
AfterLayoutContext, Element, ElementBox, Event, EventContext, LayoutContext, PaintContext,
|
||||||
SizeConstraint,
|
SizeConstraint,
|
||||||
|
@ -189,6 +191,28 @@ impl Element for Container {
|
||||||
) -> bool {
|
) -> bool {
|
||||||
self.child.dispatch_event(event, ctx)
|
self.child.dispatch_event(event, ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &crate::DebugContext,
|
||||||
|
) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "Container",
|
||||||
|
"details": {
|
||||||
|
"margin": self.margin.to_json(),
|
||||||
|
"padding": self.padding.to_json(),
|
||||||
|
"background_color": self.background_color.to_json(),
|
||||||
|
"border": self.border.to_json(),
|
||||||
|
"corner_radius": self.corner_radius,
|
||||||
|
"shadow": self.shadow.to_json(),
|
||||||
|
},
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"child": self.child.debug(ctx),
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
|
@ -199,6 +223,25 @@ pub struct Margin {
|
||||||
right: f32,
|
right: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToJson for Margin {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
let mut value = json!({});
|
||||||
|
if self.top > 0. {
|
||||||
|
value["top"] = json!(self.top);
|
||||||
|
}
|
||||||
|
if self.right > 0. {
|
||||||
|
value["right"] = json!(self.right);
|
||||||
|
}
|
||||||
|
if self.bottom > 0. {
|
||||||
|
value["bottom"] = json!(self.bottom);
|
||||||
|
}
|
||||||
|
if self.left > 0. {
|
||||||
|
value["left"] = json!(self.left);
|
||||||
|
}
|
||||||
|
value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
pub struct Padding {
|
pub struct Padding {
|
||||||
top: f32,
|
top: f32,
|
||||||
|
@ -207,9 +250,38 @@ pub struct Padding {
|
||||||
right: f32,
|
right: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToJson for Padding {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
let mut value = json!({});
|
||||||
|
if self.top > 0. {
|
||||||
|
value["top"] = json!(self.top);
|
||||||
|
}
|
||||||
|
if self.right > 0. {
|
||||||
|
value["right"] = json!(self.right);
|
||||||
|
}
|
||||||
|
if self.bottom > 0. {
|
||||||
|
value["bottom"] = json!(self.bottom);
|
||||||
|
}
|
||||||
|
if self.left > 0. {
|
||||||
|
value["left"] = json!(self.left);
|
||||||
|
}
|
||||||
|
value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
pub struct Shadow {
|
pub struct Shadow {
|
||||||
offset: Vector2F,
|
offset: Vector2F,
|
||||||
blur: f32,
|
blur: f32,
|
||||||
color: ColorU,
|
color: ColorU,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToJson for Shadow {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"offset": self.offset.to_json(),
|
||||||
|
"blur": self.blur,
|
||||||
|
"color": self.color.to_json()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -1,6 +1,10 @@
|
||||||
use crate::geometry::{
|
use crate::{
|
||||||
rect::RectF,
|
geometry::{
|
||||||
vector::{vec2f, Vector2F},
|
rect::RectF,
|
||||||
|
vector::{vec2f, Vector2F},
|
||||||
|
},
|
||||||
|
json::{json, ToJson},
|
||||||
|
DebugContext,
|
||||||
};
|
};
|
||||||
use crate::{
|
use crate::{
|
||||||
AfterLayoutContext, Element, Event, EventContext, LayoutContext, PaintContext, SizeConstraint,
|
AfterLayoutContext, Element, Event, EventContext, LayoutContext, PaintContext, SizeConstraint,
|
||||||
|
@ -58,4 +62,17 @@ impl Element for Empty {
|
||||||
) -> bool {
|
) -> bool {
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
_: &DebugContext,
|
||||||
|
) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "Empty",
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
|
use pathfinder_geometry::rect::RectF;
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
geometry::vector::Vector2F, AfterLayoutContext, Element, ElementBox, Event, EventContext,
|
geometry::vector::Vector2F, AfterLayoutContext, DebugContext, Element, ElementBox, Event,
|
||||||
LayoutContext, PaintContext, SizeConstraint,
|
EventContext, LayoutContext, PaintContext, SizeConstraint,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct EventHandler {
|
pub struct EventHandler {
|
||||||
|
@ -49,7 +52,7 @@ impl Element for EventHandler {
|
||||||
|
|
||||||
fn paint(
|
fn paint(
|
||||||
&mut self,
|
&mut self,
|
||||||
bounds: pathfinder_geometry::rect::RectF,
|
bounds: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
ctx: &mut PaintContext,
|
ctx: &mut PaintContext,
|
||||||
) -> Self::PaintState {
|
) -> Self::PaintState {
|
||||||
|
@ -59,7 +62,7 @@ impl Element for EventHandler {
|
||||||
fn dispatch_event(
|
fn dispatch_event(
|
||||||
&mut self,
|
&mut self,
|
||||||
event: &Event,
|
event: &Event,
|
||||||
bounds: pathfinder_geometry::rect::RectF,
|
bounds: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
_: &mut Self::PaintState,
|
_: &mut Self::PaintState,
|
||||||
ctx: &mut EventContext,
|
ctx: &mut EventContext,
|
||||||
|
@ -80,4 +83,17 @@ impl Element for EventHandler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
_: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "EventHandler",
|
||||||
|
"child": self.child.debug(ctx),
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,15 @@
|
||||||
use std::any::Any;
|
use std::any::Any;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
AfterLayoutContext, Axis, Element, ElementBox, Event, EventContext, LayoutContext,
|
json::{self, ToJson, Value},
|
||||||
PaintContext, SizeConstraint, Vector2FExt,
|
AfterLayoutContext, Axis, DebugContext, Element, ElementBox, Event, EventContext,
|
||||||
|
LayoutContext, PaintContext, SizeConstraint, Vector2FExt,
|
||||||
};
|
};
|
||||||
use pathfinder_geometry::vector::{vec2f, Vector2F};
|
use pathfinder_geometry::{
|
||||||
|
rect::RectF,
|
||||||
|
vector::{vec2f, Vector2F},
|
||||||
|
};
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
pub struct Flex {
|
pub struct Flex {
|
||||||
axis: Axis,
|
axis: Axis,
|
||||||
|
@ -130,7 +135,7 @@ impl Element for Flex {
|
||||||
|
|
||||||
fn paint(
|
fn paint(
|
||||||
&mut self,
|
&mut self,
|
||||||
bounds: pathfinder_geometry::rect::RectF,
|
bounds: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
ctx: &mut PaintContext,
|
ctx: &mut PaintContext,
|
||||||
) -> Self::PaintState {
|
) -> Self::PaintState {
|
||||||
|
@ -147,7 +152,7 @@ impl Element for Flex {
|
||||||
fn dispatch_event(
|
fn dispatch_event(
|
||||||
&mut self,
|
&mut self,
|
||||||
event: &Event,
|
event: &Event,
|
||||||
_: pathfinder_geometry::rect::RectF,
|
_: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
_: &mut Self::PaintState,
|
_: &mut Self::PaintState,
|
||||||
ctx: &mut EventContext,
|
ctx: &mut EventContext,
|
||||||
|
@ -158,6 +163,21 @@ impl Element for Flex {
|
||||||
}
|
}
|
||||||
handled
|
handled
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "Flex",
|
||||||
|
"axis": self.axis.to_json(),
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"children": self.children.iter().map(|child| child.debug(ctx)).collect::<Vec<json::Value>>()
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct FlexParentData {
|
struct FlexParentData {
|
||||||
|
@ -202,7 +222,7 @@ impl Element for Expanded {
|
||||||
|
|
||||||
fn paint(
|
fn paint(
|
||||||
&mut self,
|
&mut self,
|
||||||
bounds: pathfinder_geometry::rect::RectF,
|
bounds: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
ctx: &mut PaintContext,
|
ctx: &mut PaintContext,
|
||||||
) -> Self::PaintState {
|
) -> Self::PaintState {
|
||||||
|
@ -212,7 +232,7 @@ impl Element for Expanded {
|
||||||
fn dispatch_event(
|
fn dispatch_event(
|
||||||
&mut self,
|
&mut self,
|
||||||
event: &Event,
|
event: &Event,
|
||||||
_: pathfinder_geometry::rect::RectF,
|
_: RectF,
|
||||||
_: &mut Self::LayoutState,
|
_: &mut Self::LayoutState,
|
||||||
_: &mut Self::PaintState,
|
_: &mut Self::PaintState,
|
||||||
ctx: &mut EventContext,
|
ctx: &mut EventContext,
|
||||||
|
@ -223,4 +243,18 @@ impl Element for Expanded {
|
||||||
fn metadata(&self) -> Option<&dyn Any> {
|
fn metadata(&self) -> Option<&dyn Any> {
|
||||||
Some(&self.metadata)
|
Some(&self.metadata)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
_: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> Value {
|
||||||
|
json!({
|
||||||
|
"type": "Expanded",
|
||||||
|
"flex": self.metadata.flex,
|
||||||
|
"child": self.child.debug(ctx)
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
color::ColorU,
|
color::ColorU,
|
||||||
font_cache::FamilyId,
|
font_cache::FamilyId,
|
||||||
|
@ -6,8 +8,10 @@ use crate::{
|
||||||
rect::RectF,
|
rect::RectF,
|
||||||
vector::{vec2f, Vector2F},
|
vector::{vec2f, Vector2F},
|
||||||
},
|
},
|
||||||
|
json::{ToJson, Value},
|
||||||
text_layout::Line,
|
text_layout::Line,
|
||||||
AfterLayoutContext, Element, Event, EventContext, LayoutContext, PaintContext, SizeConstraint,
|
AfterLayoutContext, DebugContext, Element, Event, EventContext, LayoutContext, PaintContext,
|
||||||
|
SizeConstraint,
|
||||||
};
|
};
|
||||||
use std::{ops::Range, sync::Arc};
|
use std::{ops::Range, sync::Arc};
|
||||||
|
|
||||||
|
@ -152,4 +156,32 @@ impl Element for Label {
|
||||||
) -> bool {
|
) -> bool {
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> Value {
|
||||||
|
json!({
|
||||||
|
"type": "Label",
|
||||||
|
"font_size": self.font_size,
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"text": &self.text,
|
||||||
|
"family_id": ctx.font_cache.family_name(self.family_id).unwrap(),
|
||||||
|
"font_properties": self.font_properties.to_json(),
|
||||||
|
"highlights": self.highlights.to_json(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToJson for Highlights {
|
||||||
|
fn to_json(&self) -> Value {
|
||||||
|
json!({
|
||||||
|
"color": self.color.to_json(),
|
||||||
|
"indices": self.indices,
|
||||||
|
"font_properties": self.font_properties.to_json(),
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,13 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
font_cache::FamilyId,
|
font_cache::FamilyId,
|
||||||
fonts::Properties,
|
fonts::Properties,
|
||||||
geometry::vector::{vec2f, Vector2F},
|
geometry::{
|
||||||
AfterLayoutContext, Element, ElementBox, Event, EventContext, LayoutContext, PaintContext,
|
rect::RectF,
|
||||||
SizeConstraint,
|
vector::{vec2f, Vector2F},
|
||||||
|
},
|
||||||
|
json::{json, ToJson},
|
||||||
|
AfterLayoutContext, DebugContext, Element, ElementBox, Event, EventContext, LayoutContext,
|
||||||
|
PaintContext, SizeConstraint,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct LineBox {
|
pub struct LineBox {
|
||||||
|
@ -85,4 +89,20 @@ impl Element for LineBox {
|
||||||
) -> bool {
|
) -> bool {
|
||||||
self.child.dispatch_event(event, ctx)
|
self.child.dispatch_event(event, ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"family_id": ctx.font_cache.family_name(self.family_id).unwrap(),
|
||||||
|
"font_size": self.font_size,
|
||||||
|
"font_properties": self.font_properties.to_json(),
|
||||||
|
"child": self.child.debug(ctx),
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
geometry::{rect::RectF, vector::Vector2F},
|
geometry::{rect::RectF, vector::Vector2F},
|
||||||
AfterLayoutContext, Event, EventContext, LayoutContext, PaintContext, SizeConstraint,
|
json, AfterLayoutContext, DebugContext, Event, EventContext, LayoutContext, PaintContext,
|
||||||
|
SizeConstraint,
|
||||||
};
|
};
|
||||||
use core::panic;
|
use core::panic;
|
||||||
use replace_with::replace_with_or_abort;
|
use replace_with::replace_with_or_abort;
|
||||||
|
@ -11,6 +12,7 @@ trait AnyElement {
|
||||||
fn after_layout(&mut self, _: &mut AfterLayoutContext) {}
|
fn after_layout(&mut self, _: &mut AfterLayoutContext) {}
|
||||||
fn paint(&mut self, origin: Vector2F, ctx: &mut PaintContext);
|
fn paint(&mut self, origin: Vector2F, ctx: &mut PaintContext);
|
||||||
fn dispatch_event(&mut self, event: &Event, ctx: &mut EventContext) -> bool;
|
fn dispatch_event(&mut self, event: &Event, ctx: &mut EventContext) -> bool;
|
||||||
|
fn debug(&self, ctx: &DebugContext) -> serde_json::Value;
|
||||||
|
|
||||||
fn size(&self) -> Vector2F;
|
fn size(&self) -> Vector2F;
|
||||||
fn metadata(&self) -> Option<&dyn Any>;
|
fn metadata(&self) -> Option<&dyn Any>;
|
||||||
|
@ -53,6 +55,14 @@ pub trait Element {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
layout: &Self::LayoutState,
|
||||||
|
paint: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> serde_json::Value;
|
||||||
|
|
||||||
fn boxed(self) -> ElementBox
|
fn boxed(self) -> ElementBox
|
||||||
where
|
where
|
||||||
Self: 'static + Sized,
|
Self: 'static + Sized,
|
||||||
|
@ -165,6 +175,18 @@ impl<T: Element> AnyElement for Lifecycle<T> {
|
||||||
| Lifecycle::PostPaint { element, .. } => element.metadata(),
|
| Lifecycle::PostPaint { element, .. } => element.metadata(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(&self, ctx: &DebugContext) -> serde_json::Value {
|
||||||
|
match self {
|
||||||
|
Lifecycle::PostPaint {
|
||||||
|
element,
|
||||||
|
bounds,
|
||||||
|
layout,
|
||||||
|
paint,
|
||||||
|
} => element.debug(*bounds, layout, paint, ctx),
|
||||||
|
_ => panic!("invalid element lifecycle state"),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ElementBox {
|
impl ElementBox {
|
||||||
|
@ -191,4 +213,8 @@ impl ElementBox {
|
||||||
pub fn metadata(&self) -> Option<&dyn Any> {
|
pub fn metadata(&self) -> Option<&dyn Any> {
|
||||||
self.0.metadata()
|
self.0.metadata()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn debug(&self, ctx: &DebugContext) -> json::Value {
|
||||||
|
self.0.debug(ctx)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
geometry::{rect::RectF, vector::Vector2F},
|
geometry::{rect::RectF, vector::Vector2F},
|
||||||
AfterLayoutContext, Element, ElementBox, Event, EventContext, LayoutContext, PaintContext,
|
json::{self, json, ToJson},
|
||||||
SizeConstraint,
|
AfterLayoutContext, DebugContext, Element, ElementBox, Event, EventContext, LayoutContext,
|
||||||
|
PaintContext, SizeConstraint,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct Stack {
|
pub struct Stack {
|
||||||
|
@ -71,6 +72,20 @@ impl Element for Stack {
|
||||||
}
|
}
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "Stack",
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"children": self.children.iter().map(|child| child.debug(ctx)).collect::<Vec<json::Value>>()
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Extend<ElementBox> for Stack {
|
impl Extend<ElementBox> for Stack {
|
||||||
|
|
|
@ -1,11 +1,13 @@
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
color::ColorU,
|
color::ColorU,
|
||||||
geometry::{
|
geometry::{
|
||||||
rect::RectF,
|
rect::RectF,
|
||||||
vector::{vec2f, Vector2F},
|
vector::{vec2f, Vector2F},
|
||||||
},
|
},
|
||||||
scene, AfterLayoutContext, Element, Event, EventContext, LayoutContext, PaintContext,
|
scene, AfterLayoutContext, DebugContext, Element, Event, EventContext, LayoutContext,
|
||||||
SizeConstraint,
|
PaintContext, SizeConstraint,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct Svg {
|
pub struct Svg {
|
||||||
|
@ -86,8 +88,25 @@ impl Element for Svg {
|
||||||
) -> bool {
|
) -> bool {
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
_: &DebugContext,
|
||||||
|
) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "Svg",
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"path": self.path,
|
||||||
|
"color": self.color.to_json(),
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
use crate::json::ToJson;
|
||||||
|
|
||||||
fn from_usvg_rect(rect: usvg::Rect) -> RectF {
|
fn from_usvg_rect(rect: usvg::Rect) -> RectF {
|
||||||
RectF::new(
|
RectF::new(
|
||||||
vec2f(rect.x() as f32, rect.y() as f32),
|
vec2f(rect.x() as f32, rect.y() as f32),
|
||||||
|
|
|
@ -7,8 +7,10 @@ use crate::{
|
||||||
rect::RectF,
|
rect::RectF,
|
||||||
vector::{vec2f, Vector2F},
|
vector::{vec2f, Vector2F},
|
||||||
},
|
},
|
||||||
|
json::{self, json},
|
||||||
ElementBox,
|
ElementBox,
|
||||||
};
|
};
|
||||||
|
use json::ToJson;
|
||||||
use parking_lot::Mutex;
|
use parking_lot::Mutex;
|
||||||
use std::{cmp, ops::Range, sync::Arc};
|
use std::{cmp, ops::Range, sync::Arc};
|
||||||
|
|
||||||
|
@ -236,4 +238,21 @@ where
|
||||||
|
|
||||||
handled
|
handled
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
layout: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &crate::DebugContext,
|
||||||
|
) -> json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "UniformList",
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"scroll_max": layout.scroll_max,
|
||||||
|
"item_height": layout.item_height,
|
||||||
|
"items": layout.items.iter().map(|item| item.debug(ctx)).collect::<Vec<json::Value>>()
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -36,6 +36,15 @@ impl FontCache {
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn family_name(&self, family_id: FamilyId) -> Result<String> {
|
||||||
|
self.0
|
||||||
|
.read()
|
||||||
|
.families
|
||||||
|
.get(family_id.0)
|
||||||
|
.ok_or_else(|| anyhow!("invalid family id"))
|
||||||
|
.map(|family| family.name.clone())
|
||||||
|
}
|
||||||
|
|
||||||
pub fn load_family(&self, names: &[&str]) -> Result<FamilyId> {
|
pub fn load_family(&self, names: &[&str]) -> Result<FamilyId> {
|
||||||
for name in names {
|
for name in names {
|
||||||
let state = self.0.upgradable_read();
|
let state = self.0.upgradable_read();
|
||||||
|
|
|
@ -1,7 +1,62 @@
|
||||||
|
use crate::json::json;
|
||||||
pub use font_kit::metrics::Metrics;
|
pub use font_kit::metrics::Metrics;
|
||||||
pub use font_kit::properties::{Properties, Weight};
|
pub use font_kit::properties::{Properties, Stretch, Style, Weight};
|
||||||
|
|
||||||
|
use crate::json::ToJson;
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug, Eq, PartialEq, Hash)]
|
#[derive(Copy, Clone, Debug, Eq, PartialEq, Hash)]
|
||||||
pub struct FontId(pub usize);
|
pub struct FontId(pub usize);
|
||||||
|
|
||||||
pub type GlyphId = u32;
|
pub type GlyphId = u32;
|
||||||
|
|
||||||
|
impl ToJson for Properties {
|
||||||
|
fn to_json(&self) -> crate::json::Value {
|
||||||
|
json!({
|
||||||
|
"style": self.style.to_json(),
|
||||||
|
"weight": self.weight.to_json(),
|
||||||
|
"stretch": self.stretch.to_json(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToJson for Style {
|
||||||
|
fn to_json(&self) -> crate::json::Value {
|
||||||
|
match self {
|
||||||
|
Style::Normal => json!("normal"),
|
||||||
|
Style::Italic => json!("italic"),
|
||||||
|
Style::Oblique => json!("oblique"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToJson for Weight {
|
||||||
|
fn to_json(&self) -> crate::json::Value {
|
||||||
|
if self.0 == Weight::THIN.0 {
|
||||||
|
json!("thin")
|
||||||
|
} else if self.0 == Weight::EXTRA_LIGHT.0 {
|
||||||
|
json!("extra light")
|
||||||
|
} else if self.0 == Weight::LIGHT.0 {
|
||||||
|
json!("light")
|
||||||
|
} else if self.0 == Weight::NORMAL.0 {
|
||||||
|
json!("normal")
|
||||||
|
} else if self.0 == Weight::MEDIUM.0 {
|
||||||
|
json!("medium")
|
||||||
|
} else if self.0 == Weight::SEMIBOLD.0 {
|
||||||
|
json!("semibold")
|
||||||
|
} else if self.0 == Weight::BOLD.0 {
|
||||||
|
json!("bold")
|
||||||
|
} else if self.0 == Weight::EXTRA_BOLD.0 {
|
||||||
|
json!("extra bold")
|
||||||
|
} else if self.0 == Weight::BLACK.0 {
|
||||||
|
json!("black")
|
||||||
|
} else {
|
||||||
|
json!(self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToJson for Stretch {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
json!(self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
use super::scene::{Path, PathVertex};
|
use super::scene::{Path, PathVertex};
|
||||||
use crate::color::ColorU;
|
use crate::{color::ColorU, json::ToJson};
|
||||||
pub use pathfinder_geometry::*;
|
pub use pathfinder_geometry::*;
|
||||||
use rect::RectF;
|
use rect::RectF;
|
||||||
|
use serde_json::json;
|
||||||
use vector::{vec2f, Vector2F};
|
use vector::{vec2f, Vector2F};
|
||||||
|
|
||||||
pub struct PathBuilder {
|
pub struct PathBuilder {
|
||||||
|
@ -106,3 +107,15 @@ impl PathBuilder {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToJson for Vector2F {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
json!([self.x(), self.y()])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ToJson for RectF {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
json!({"origin": self.origin().to_json(), "size": self.size().to_json()})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
15
gpui/src/json.rs
Normal file
15
gpui/src/json.rs
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
pub use serde_json::*;
|
||||||
|
|
||||||
|
pub trait ToJson {
|
||||||
|
fn to_json(&self) -> Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: ToJson> ToJson for Option<T> {
|
||||||
|
fn to_json(&self) -> Value {
|
||||||
|
if let Some(value) = self.as_ref() {
|
||||||
|
value.to_json()
|
||||||
|
} else {
|
||||||
|
json!(null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -18,11 +18,12 @@ mod util;
|
||||||
pub use elements::{Element, ElementBox};
|
pub use elements::{Element, ElementBox};
|
||||||
pub mod executor;
|
pub mod executor;
|
||||||
pub use executor::Task;
|
pub use executor::Task;
|
||||||
|
pub mod color;
|
||||||
|
pub mod json;
|
||||||
pub mod keymap;
|
pub mod keymap;
|
||||||
pub mod platform;
|
pub mod platform;
|
||||||
pub use pathfinder_color as color;
|
|
||||||
pub use platform::Event;
|
pub use platform::Event;
|
||||||
pub use presenter::{
|
pub use presenter::{
|
||||||
AfterLayoutContext, Axis, EventContext, LayoutContext, PaintContext, SizeConstraint,
|
AfterLayoutContext, Axis, DebugContext, EventContext, LayoutContext, PaintContext,
|
||||||
Vector2FExt,
|
SizeConstraint, Vector2FExt,
|
||||||
};
|
};
|
||||||
|
|
|
@ -2,11 +2,13 @@ use crate::{
|
||||||
app::{AppContext, MutableAppContext, WindowInvalidation},
|
app::{AppContext, MutableAppContext, WindowInvalidation},
|
||||||
elements::Element,
|
elements::Element,
|
||||||
font_cache::FontCache,
|
font_cache::FontCache,
|
||||||
|
json::ToJson,
|
||||||
platform::Event,
|
platform::Event,
|
||||||
text_layout::TextLayoutCache,
|
text_layout::TextLayoutCache,
|
||||||
AssetCache, ElementBox, Scene,
|
AssetCache, ElementBox, Scene,
|
||||||
};
|
};
|
||||||
use pathfinder_geometry::vector::{vec2f, Vector2F};
|
use pathfinder_geometry::vector::{vec2f, Vector2F};
|
||||||
|
use serde_json::json;
|
||||||
use std::{any::Any, collections::HashMap, sync::Arc};
|
use std::{any::Any, collections::HashMap, sync::Arc};
|
||||||
|
|
||||||
pub struct Presenter {
|
pub struct Presenter {
|
||||||
|
@ -224,6 +226,12 @@ impl<'a> EventContext<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct DebugContext<'a> {
|
||||||
|
rendered_views: &'a mut HashMap<usize, ElementBox>,
|
||||||
|
pub font_cache: &'a FontCache,
|
||||||
|
pub app: &'a AppContext,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
|
||||||
pub enum Axis {
|
pub enum Axis {
|
||||||
Horizontal,
|
Horizontal,
|
||||||
|
@ -239,6 +247,15 @@ impl Axis {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToJson for Axis {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
match self {
|
||||||
|
Axis::Horizontal => json!("horizontal"),
|
||||||
|
Axis::Vertical => json!("vertical"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub trait Vector2FExt {
|
pub trait Vector2FExt {
|
||||||
fn along(self, axis: Axis) -> f32;
|
fn along(self, axis: Axis) -> f32;
|
||||||
}
|
}
|
||||||
|
@ -291,6 +308,15 @@ impl SizeConstraint {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToJson for SizeConstraint {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"min": self.min.to_json(),
|
||||||
|
"max": self.max.to_json(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub struct ChildView {
|
pub struct ChildView {
|
||||||
view_id: usize,
|
view_id: usize,
|
||||||
}
|
}
|
||||||
|
@ -342,6 +368,25 @@ impl Element for ChildView {
|
||||||
) -> bool {
|
) -> bool {
|
||||||
ctx.dispatch_event(self.view_id, event)
|
ctx.dispatch_event(self.view_id, event)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: pathfinder_geometry::rect::RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
ctx: &DebugContext,
|
||||||
|
) -> serde_json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "ChildView",
|
||||||
|
"view_id": self.view_id,
|
||||||
|
"bounds": bounds.to_json(),
|
||||||
|
"child": if let Some(view) = ctx.rendered_views.get(&self.view_id) {
|
||||||
|
view.debug(ctx)
|
||||||
|
} else {
|
||||||
|
json!(null)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|
|
@ -1,7 +1,10 @@
|
||||||
|
use serde_json::json;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
color::ColorU,
|
color::ColorU,
|
||||||
fonts::{FontId, GlyphId},
|
fonts::{FontId, GlyphId},
|
||||||
geometry::{rect::RectF, vector::Vector2F},
|
geometry::{rect::RectF, vector::Vector2F},
|
||||||
|
json::ToJson,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct Scene {
|
pub struct Scene {
|
||||||
|
@ -258,3 +261,22 @@ impl Border {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToJson for Border {
|
||||||
|
fn to_json(&self) -> serde_json::Value {
|
||||||
|
let mut value = json!({});
|
||||||
|
if self.top {
|
||||||
|
value["top"] = json!(self.width);
|
||||||
|
}
|
||||||
|
if self.right {
|
||||||
|
value["right"] = json!(self.width);
|
||||||
|
}
|
||||||
|
if self.bottom {
|
||||||
|
value["bottom"] = json!(self.width);
|
||||||
|
}
|
||||||
|
if self.left {
|
||||||
|
value["left"] = json!(self.width);
|
||||||
|
}
|
||||||
|
value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -6,10 +6,12 @@ use gpui::{
|
||||||
vector::{vec2f, Vector2F},
|
vector::{vec2f, Vector2F},
|
||||||
PathBuilder,
|
PathBuilder,
|
||||||
},
|
},
|
||||||
|
json::{self, ToJson},
|
||||||
text_layout::{self, TextLayoutCache},
|
text_layout::{self, TextLayoutCache},
|
||||||
AfterLayoutContext, AppContext, Border, Element, Event, EventContext, FontCache, LayoutContext,
|
AfterLayoutContext, AppContext, Border, Element, Event, EventContext, FontCache, LayoutContext,
|
||||||
PaintContext, Quad, Scene, SizeConstraint, ViewHandle,
|
PaintContext, Quad, Scene, SizeConstraint, ViewHandle,
|
||||||
};
|
};
|
||||||
|
use json::json;
|
||||||
use smallvec::SmallVec;
|
use smallvec::SmallVec;
|
||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
use std::{
|
use std::{
|
||||||
|
@ -477,6 +479,19 @@ impl Element for BufferElement {
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn debug(
|
||||||
|
&self,
|
||||||
|
bounds: RectF,
|
||||||
|
_: &Self::LayoutState,
|
||||||
|
_: &Self::PaintState,
|
||||||
|
_: &gpui::DebugContext,
|
||||||
|
) -> json::Value {
|
||||||
|
json!({
|
||||||
|
"type": "BufferElement",
|
||||||
|
"bounds": bounds.to_json()
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct LayoutState {
|
pub struct LayoutState {
|
||||||
|
|
Loading…
Reference in a new issue