Skip to content
Snippets Groups Projects
Unverified Commit e8a853a6 authored by Caleb Yates's avatar Caleb Yates Committed by GitHub
Browse files

Code quality improvements (#164)

* fix: Implement reflect for FocusedWidget

* refactor: using `bevy_math::Rect.contains` implementation to clean up code

* fix: clippy is happy now

* fix: fmt is happy now

* add: Lots of Reflect impls and type registrations

* refactor: moved certain type registrations into their own modules
parent 397c9cde
No related branches found
No related tags found
No related merge requests found
...@@ -3,7 +3,7 @@ use bevy::prelude::*; ...@@ -3,7 +3,7 @@ use bevy::prelude::*;
use cosmic_text::{Attrs, AttrsOwned, Editor, FontSystem}; use cosmic_text::{Attrs, AttrsOwned, Editor, FontSystem};
/// Enum representing text wrapping in a cosmic [`Buffer`] /// Enum representing text wrapping in a cosmic [`Buffer`]
#[derive(Clone, Component, PartialEq, Default)] #[derive(Component, Reflect, Clone, PartialEq, Default)]
pub enum CosmicWrap { pub enum CosmicWrap {
InfiniteLine, InfiniteLine,
#[default] #[default]
...@@ -11,7 +11,7 @@ pub enum CosmicWrap { ...@@ -11,7 +11,7 @@ pub enum CosmicWrap {
} }
/// Enum representing the text alignment in a cosmic [`Buffer`] /// Enum representing the text alignment in a cosmic [`Buffer`]
#[derive(Clone, Component)] #[derive(Component, Reflect, Clone)]
pub enum CosmicTextAlign { pub enum CosmicTextAlign {
Center { padding: i32 }, Center { padding: i32 },
TopLeft { padding: i32 }, TopLeft { padding: i32 },
...@@ -30,7 +30,7 @@ impl Default for CosmicTextAlign { ...@@ -30,7 +30,7 @@ impl Default for CosmicTextAlign {
pub struct ReadOnly; // tag component pub struct ReadOnly; // tag component
/// Internal value used to decide what section of a [`Buffer`] to render /// Internal value used to decide what section of a [`Buffer`] to render
#[derive(Component, Debug, Default)] #[derive(Component, Reflect, Debug, Default)]
pub struct XOffset { pub struct XOffset {
pub left: f32, pub left: f32,
pub width: f32, pub width: f32,
...@@ -47,32 +47,32 @@ impl Default for DefaultAttrs { ...@@ -47,32 +47,32 @@ impl Default for DefaultAttrs {
} }
/// Image to be used as a buffer's background /// Image to be used as a buffer's background
#[derive(Component, Default)] #[derive(Component, Reflect, Default)]
pub struct CosmicBackgroundImage(pub Option<Handle<Image>>); pub struct CosmicBackgroundImage(pub Option<Handle<Image>>);
/// Color to be used as a buffer's background /// Color to be used as a buffer's background
#[derive(Component, Default, Deref)] #[derive(Component, Reflect, Default, Deref)]
pub struct CosmicBackgroundColor(pub Color); pub struct CosmicBackgroundColor(pub Color);
/// Color to be used for the text cursor /// Color to be used for the text cursor
#[derive(Component, Default, Deref)] #[derive(Component, Reflect, Default, Deref)]
pub struct CursorColor(pub Color); pub struct CursorColor(pub Color);
/// Color to be used as the selected text background /// Color to be used as the selected text background
#[derive(Component, Default, Deref)] #[derive(Component, Reflect, Default, Deref)]
pub struct SelectionColor(pub Color); pub struct SelectionColor(pub Color);
/// Color to be used for the selected text /// Color to be used for the selected text
#[derive(Component, Default, Deref)] #[derive(Component, Reflect, Default, Deref)]
pub struct SelectedTextColor(pub Color); pub struct SelectedTextColor(pub Color);
/// Maximum number of lines allowed in a buffer /// Maximum number of lines allowed in a buffer
#[derive(Component, Default)] #[derive(Component, Reflect, Default)]
pub struct MaxLines(pub usize); pub struct MaxLines(pub usize);
/// Maximum number of characters allowed in a buffer /// Maximum number of characters allowed in a buffer
// TODO: Check this functionality with widechars; Use graphemes to test? // TODO: Check this functionality with widechars; Use graphemes to test?
#[derive(Component, Default)] #[derive(Component, Reflect, Default)]
pub struct MaxChars(pub usize); pub struct MaxChars(pub usize);
/// Buffer does not respond to scroll events /// Buffer does not respond to scroll events
...@@ -111,7 +111,7 @@ pub struct ScrollDisabled; ...@@ -111,7 +111,7 @@ pub struct ScrollDisabled;
/// # .add_plugins(CosmicEditPlugin::default()) /// # .add_plugins(CosmicEditPlugin::default())
/// # .add_systems(Startup, setup); /// # .add_systems(Startup, setup);
/// # } /// # }
#[derive(Component)] #[derive(Component, Reflect)]
pub struct CosmicSource(pub Entity); pub struct CosmicSource(pub Entity);
/// A bundle containing all the required components for [`CosmicBuffer`] functionality. /// A bundle containing all the required components for [`CosmicBuffer`] functionality.
......
...@@ -23,11 +23,12 @@ impl Plugin for CursorPlugin { ...@@ -23,11 +23,12 @@ impl Plugin for CursorPlugin {
.run_if(not(resource_exists::<CursorPluginDisabled>)), .run_if(not(resource_exists::<CursorPluginDisabled>)),
) )
.add_event::<TextHoverIn>() .add_event::<TextHoverIn>()
.register_type::<TextHoverIn>()
.add_event::<TextHoverOut>(); .add_event::<TextHoverOut>();
} }
} }
#[derive(Component, Deref)] #[derive(Component, Reflect, Deref)]
pub struct HoverCursor(pub CursorIcon); pub struct HoverCursor(pub CursorIcon);
impl Default for HoverCursor { impl Default for HoverCursor {
...@@ -39,12 +40,12 @@ impl Default for HoverCursor { ...@@ -39,12 +40,12 @@ impl Default for HoverCursor {
/// For use with custom cursor control /// For use with custom cursor control
/// Event is emitted when cursor enters a text widget /// Event is emitted when cursor enters a text widget
/// Event contains the cursor from the buffer's [`HoverCursor`] /// Event contains the cursor from the buffer's [`HoverCursor`]
#[derive(Event, Deref)] #[derive(Event, Reflect, Deref, Debug)]
pub struct TextHoverIn(pub CursorIcon); pub struct TextHoverIn(pub CursorIcon);
/// For use with custom cursor control /// For use with custom cursor control
/// Event is emitted when cursor leaves a text widget /// Event is emitted when cursor leaves a text widget
#[derive(Event)] #[derive(Event, Debug)]
pub struct TextHoverOut; pub struct TextHoverOut;
pub(crate) fn change_cursor( pub(crate) fn change_cursor(
...@@ -109,17 +110,18 @@ pub(crate) fn hover_sprites( ...@@ -109,17 +110,18 @@ pub(crate) fn hover_sprites(
} }
let size = sprite.custom_size.unwrap_or(Vec2::ONE); let size = sprite.custom_size.unwrap_or(Vec2::ONE);
let x_min = node_transform.affine().translation.x - size.x / 2.; if get_node_cursor_pos(
let y_min = node_transform.affine().translation.y - size.y / 2.; window,
let x_max = node_transform.affine().translation.x + size.x / 2.; node_transform,
let y_max = node_transform.affine().translation.y + size.y / 2.; size,
if let Some(pos) = window.cursor_position() { false,
if let Some(pos) = camera.viewport_to_world_2d(camera_transform, pos) { camera,
if x_min < pos.x && pos.x < x_max && y_min < pos.y && pos.y < y_max { camera_transform,
*hovered = true; )
icon = hover.0; .is_some()
} {
} *hovered = true;
icon = hover.0;
} }
} }
......
...@@ -7,12 +7,13 @@ pub(crate) struct EventsPlugin; ...@@ -7,12 +7,13 @@ pub(crate) struct EventsPlugin;
impl Plugin for EventsPlugin { impl Plugin for EventsPlugin {
fn build(&self, app: &mut App) { fn build(&self, app: &mut App) {
app.add_event::<CosmicTextChanged>(); app.add_event::<CosmicTextChanged>()
.register_type::<CosmicTextChanged>();
} }
} }
/// Text change events /// Text change events
/// Sent when text is changed in a cosmic buffer /// Sent when text is changed in a cosmic buffer
/// Contains the entity on which the text was changed, and the new text as a [`String`] /// Contains the entity on which the text was changed, and the new text as a [`String`]
#[derive(Event, Debug)] #[derive(Event, Reflect, Debug)]
pub struct CosmicTextChanged(pub (Entity, String)); pub struct CosmicTextChanged(pub (Entity, String));
...@@ -17,12 +17,14 @@ impl Plugin for FocusPlugin { ...@@ -17,12 +17,14 @@ impl Plugin for FocusPlugin {
.in_set(FocusSet) .in_set(FocusSet)
.after(WidgetSet), .after(WidgetSet),
) )
.init_resource::<FocusedWidget>(); .init_resource::<FocusedWidget>()
.register_type::<FocusedWidget>();
} }
} }
/// Resource struct that keeps track of the currently active editor entity. /// Resource struct that keeps track of the currently active editor entity.
#[derive(Resource, Default, Deref, DerefMut)] #[derive(Resource, Reflect, Default, Deref, DerefMut)]
#[reflect(Resource)]
pub struct FocusedWidget(pub Option<Entity>); pub struct FocusedWidget(pub Option<Entity>);
pub(crate) fn add_editor_to_focused( pub(crate) fn add_editor_to_focused(
......
...@@ -125,8 +125,8 @@ pub(crate) fn input_mouse( ...@@ -125,8 +125,8 @@ pub(crate) fn input_mouse(
let mut is_ui_node = false; let mut is_ui_node = false;
let mut transform = sprite_transform; let mut transform = sprite_transform;
let (mut width, mut height) = let sprite_size = sprite.custom_size.expect("Must specify Sprite.custom_size");
(sprite.custom_size.unwrap().x, sprite.custom_size.unwrap().y); let (mut width, mut height) = (sprite_size.x, sprite_size.y);
// TODO: this is bad loop nesting, rethink system with relationships in mind // TODO: this is bad loop nesting, rethink system with relationships in mind
for (node, node_transform, source) in node_q.iter() { for (node, node_transform, source) in node_q.iter() {
...@@ -159,10 +159,10 @@ pub(crate) fn input_mouse( ...@@ -159,10 +159,10 @@ pub(crate) fn input_mouse(
get_y_offset_center(height * scale_factor, &buffer), get_y_offset_center(height * scale_factor, &buffer),
), ),
}; };
let point = |node_cursor_pos: (f32, f32)| { let point = |node_cursor_pos: Vec2| {
( (
(node_cursor_pos.0 * scale_factor) as i32 - padding_x, (node_cursor_pos.x * scale_factor) as i32 - padding_x,
(node_cursor_pos.1 * scale_factor) as i32 - padding_y, (node_cursor_pos.y * scale_factor) as i32 - padding_y,
) )
}; };
...@@ -173,7 +173,7 @@ pub(crate) fn input_mouse( ...@@ -173,7 +173,7 @@ pub(crate) fn input_mouse(
if let Some(node_cursor_pos) = get_node_cursor_pos( if let Some(node_cursor_pos) = get_node_cursor_pos(
primary_window, primary_window,
transform, transform,
(width, height), Vec2::new(width, height),
is_ui_node, is_ui_node,
camera, camera,
camera_transform, camera_transform,
...@@ -213,7 +213,7 @@ pub(crate) fn input_mouse( ...@@ -213,7 +213,7 @@ pub(crate) fn input_mouse(
if let Some(node_cursor_pos) = get_node_cursor_pos( if let Some(node_cursor_pos) = get_node_cursor_pos(
primary_window, primary_window,
transform, transform,
(width, height), Vec2::new(width, height),
is_ui_node, is_ui_node,
camera, camera,
camera_transform, camera_transform,
......
...@@ -138,6 +138,18 @@ impl Plugin for CosmicEditPlugin { ...@@ -138,6 +138,18 @@ impl Plugin for CosmicEditPlugin {
)) ))
.insert_resource(CosmicFontSystem(font_system)); .insert_resource(CosmicFontSystem(font_system));
app.register_type::<CosmicWrap>()
.register_type::<CosmicTextAlign>()
.register_type::<XOffset>()
.register_type::<CosmicBackgroundImage>()
.register_type::<CosmicBackgroundColor>()
.register_type::<CursorColor>()
.register_type::<SelectionColor>()
.register_type::<MaxLines>()
.register_type::<MaxChars>()
.register_type::<CosmicSource>()
.register_type::<HoverCursor>();
#[cfg(target_arch = "wasm32")] #[cfg(target_arch = "wasm32")]
{ {
let (tx, rx) = crossbeam_channel::bounded::<WasmPaste>(1); let (tx, rx) = crossbeam_channel::bounded::<WasmPaste>(1);
...@@ -148,7 +160,7 @@ impl Plugin for CosmicEditPlugin { ...@@ -148,7 +160,7 @@ impl Plugin for CosmicEditPlugin {
} }
/// Attach to primary camera, and enable the `multicam` feature to use multiple cameras. /// Attach to primary camera, and enable the `multicam` feature to use multiple cameras.
/// Will panic if no `Camera`s without this component exist and the `multicam` feature is enabled. /// Will panic if no Camera's without this component exist and the `multicam` feature is enabled.
/// ///
/// A very basic example which doesn't panic: /// A very basic example which doesn't panic:
/// ```rust /// ```rust
...@@ -176,7 +188,6 @@ impl Plugin for CosmicEditPlugin { ...@@ -176,7 +188,6 @@ impl Plugin for CosmicEditPlugin {
/// }); /// });
/// } /// }
/// ``` /// ```
#[cfg(feature = "multicam")]
#[derive(Component, Debug, Default)] #[derive(Component, Debug, Default)]
pub struct CosmicPrimaryCamera; pub struct CosmicPrimaryCamera;
......
...@@ -218,6 +218,7 @@ fn render_texture( ...@@ -218,6 +218,7 @@ fn render_texture(
if let Some(prev_image) = images.get_mut(canvas) { if let Some(prev_image) = images.get_mut(canvas) {
prev_image.data.clear(); prev_image.data.clear();
// Updates the stored asset image with the computed pixels
prev_image.data.extend_from_slice(pixels.as_slice()); prev_image.data.extend_from_slice(pixels.as_slice());
prev_image.resize(Extent3d { prev_image.resize(Extent3d {
width: size.0.x as u32, width: size.0.x as u32,
......
...@@ -36,22 +36,26 @@ pub fn deselect_editor_on_esc(i: Res<ButtonInput<KeyCode>>, mut focus: ResMut<Fo ...@@ -36,22 +36,26 @@ pub fn deselect_editor_on_esc(i: Res<ButtonInput<KeyCode>>, mut focus: ResMut<Fo
pub fn get_node_cursor_pos( pub fn get_node_cursor_pos(
window: &Window, window: &Window,
node_transform: &GlobalTransform, node_transform: &GlobalTransform,
size: (f32, f32), size: Vec2,
is_ui_node: bool, is_ui_node: bool,
camera: &Camera, camera: &Camera,
camera_transform: &GlobalTransform, camera_transform: &GlobalTransform,
) -> Option<(f32, f32)> { ) -> Option<Vec2> {
let (x_min, y_min, x_max, y_max) = ( let node_translation = node_transform.affine().translation;
node_transform.affine().translation.x - size.0 / 2., let node_bounds = Rect::new(
node_transform.affine().translation.y - size.1 / 2., node_translation.x - size.x / 2.,
node_transform.affine().translation.x + size.0 / 2., node_translation.y - size.y / 2.,
node_transform.affine().translation.y + size.1 / 2., node_translation.x + size.x / 2.,
node_translation.y + size.y / 2.,
); );
window.cursor_position().and_then(|pos| { window.cursor_position().and_then(|pos| {
if is_ui_node { if is_ui_node {
if x_min < pos.x && pos.x < x_max && y_min < pos.y && pos.y < y_max { if node_bounds.contains(pos) {
Some((pos.x - x_min, pos.y - y_min)) Some(Vec2::new(
pos.x - node_bounds.min.x,
pos.y - node_bounds.min.y,
))
} else { } else {
None None
} }
...@@ -59,8 +63,11 @@ pub fn get_node_cursor_pos( ...@@ -59,8 +63,11 @@ pub fn get_node_cursor_pos(
camera camera
.viewport_to_world_2d(camera_transform, pos) .viewport_to_world_2d(camera_transform, pos)
.and_then(|pos| { .and_then(|pos| {
if x_min < pos.x && pos.x < x_max && y_min < pos.y && pos.y < y_max { if node_bounds.contains(pos) {
Some((pos.x - x_min, y_max - pos.y)) Some(Vec2::new(
pos.x - node_bounds.min.x,
node_bounds.max.y - pos.y,
))
} else { } else {
None None
} }
......
...@@ -24,21 +24,23 @@ impl Plugin for WidgetPlugin { ...@@ -24,21 +24,23 @@ impl Plugin for WidgetPlugin {
.chain() .chain()
.in_set(WidgetSet) .in_set(WidgetSet)
.after(TransformSystem::TransformPropagate), .after(TransformSystem::TransformPropagate),
); )
.register_type::<CosmicPadding>()
.register_type::<CosmicWidgetSize>();
} }
} }
/// Wrapper for a [`Vec2`] describing the horizontal and vertical padding of a widget. /// Wrapper for a [`Vec2`] describing the horizontal and vertical padding of a widget.
/// This is set programatically, not for user modification. /// This is set programatically, not for user modification.
/// To set a widget's padding, use [`CosmicTextAlign`] /// To set a widget's padding, use [`CosmicTextAlign`]
#[derive(Component, Default, Deref, DerefMut, Debug)] #[derive(Component, Reflect, Default, Deref, DerefMut, Debug)]
pub struct CosmicPadding(pub Vec2); pub struct CosmicPadding(pub Vec2);
/// Wrapper for a [`Vec2`] describing the horizontal and vertical size of a widget. /// Wrapper for a [`Vec2`] describing the horizontal and vertical size of a widget.
/// This is set programatically, not for user modification. /// This is set programatically, not for user modification.
/// To set a widget's size, use either it's [`Sprite`] dimensions or modify the target UI element's /// To set a widget's size, use either it's [`Sprite`] dimensions or modify the target UI element's
/// size. /// size.
#[derive(Component, Default, Deref, DerefMut)] #[derive(Component, Reflect, Default, Deref, DerefMut)]
pub struct CosmicWidgetSize(pub Vec2); pub struct CosmicWidgetSize(pub Vec2);
/// Reshapes text in a [`CosmicEditor`] /// Reshapes text in a [`CosmicEditor`]
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment