Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Rewrite xilem_web to support new xilem_core #403

Merged
merged 57 commits into from
Jun 28, 2024
Merged
Show file tree
Hide file tree
Changes from 44 commits
Commits
Show all changes
57 commits
Select commit Hold shift + click to select a range
42ece98
xilem_core: Add `ViewSequence::count`
Philipp-M Jun 9, 2024
ac6fe76
xilem_core: Temporarily don't require `Send` bound on `Message` trait
Philipp-M Jun 9, 2024
918083f
xilem_web: Initial commit of rewrite
Philipp-M Jun 9, 2024
605361e
Slight cleanup and rename of `Attrs` to `Props` for better distinctio…
Philipp-M Jun 10, 2024
fc614af
Add experimental `OneOf2` View in xilem_core and xilem_web
Philipp-M Jun 10, 2024
06232f2
Small fix in classes, and more renaming
Philipp-M Jun 10, 2024
304b78d
xilem_web: Port counter example to new xilem_web
Philipp-M Jun 10, 2024
c422e4f
Refine OneOf2 view (reduce boilerplate) add generational ids and clea…
Philipp-M Jun 11, 2024
56473df
Add a OneOf2 `ViewSequence` impl
Philipp-M Jun 11, 2024
80c6213
Cleanup `OneOf2` view and sequence and document everything
Philipp-M Jun 12, 2024
731094e
Add tests for `OneOf2`
Philipp-M Jun 12, 2024
dc1e4e3
Add `OneOf2-9` implementations via a macro
Philipp-M Jun 12, 2024
4a3fa4b
Add orphan views
Philipp-M Jun 14, 2024
40406bb
A lot of fixes, and add DOM interfaces
Philipp-M Jun 15, 2024
f525a53
Port style support from old xilem_web and port the mathml_svg example…
Philipp-M Jun 15, 2024
ed2cc6c
Add parent to `PodMut` and fix `OneOf` view impl with it
Philipp-M Jun 15, 2024
ef2ccfa
Lazy access to attributes, which increases performance, when they're …
Philipp-M Jun 15, 2024
8f5efbf
Cleanup with `AnyPod` alias, and reduce macro-expanded boilerplate by…
Philipp-M Jun 15, 2024
71f6c85
Ported `Pointer` view from xilem_web
Philipp-M Jun 15, 2024
8862d77
Port `Fill` and `Stroke` views and add them to the relevant interfaces
Philipp-M Jun 15, 2024
1e3175e
xilem_core: Add support for kurbo shapes
Philipp-M Jun 15, 2024
b8bd42d
Port kurbo shapes and svgtoy example
Philipp-M Jun 15, 2024
f91e4c8
Port todomvc example
Philipp-M Jun 17, 2024
2092b85
Port custom_element
Philipp-M Jun 17, 2024
7675762
Add missing Adapt view
Philipp-M Jun 17, 2024
50b921a
Delete old xilem_web and rename new xilem_web2 to xilem_web
Philipp-M Jun 17, 2024
c732da5
Add a DomViewSequence as a workaround for regressing compile times be…
Philipp-M Jun 20, 2024
1be6bfa
Cleaned up and documented a lot
Philipp-M Jun 20, 2024
c6e053e
Merge branch 'main' into xilem_web2
Philipp-M Jun 20, 2024
06e5306
Correct license headers
Philipp-M Jun 20, 2024
f4c9cf0
Provide a way to add `View` implementations in downstream implementat…
Philipp-M Jun 17, 2024
4f6bfc4
Remove unneeded std feature
Philipp-M Jun 21, 2024
4b8335b
Cleanup a little bit and remove `AsOrphanView` in favor of `OrphanView`
Philipp-M Jun 22, 2024
627c16d
Document and port generic `Message`
Philipp-M Jun 22, 2024
c37a926
xilem_core: Add a test for the `OrphanView`
Philipp-M Jun 22, 2024
39f7790
Merge branch 'orphan-views' into xilem_web2
Philipp-M Jun 22, 2024
75847e4
Adjust to generic message of xilem_core (i.e. unhack `Send` bound fro…
Philipp-M Jun 22, 2024
6307703
xilem_core: Add a `ViewSequence` impl for `[impl ViewSequence; N]`
Philipp-M Jun 23, 2024
b7a5d66
Fix `set_attribute` when element is not an `HtmlInputElement`
Philipp-M Jun 23, 2024
1d60987
Add a lot of documentation to the interfaces and fix `OneOf` `WithSty…
Philipp-M Jun 23, 2024
52c8eb7
Add new example "blog" to show some of the new properties of xilem_web
Philipp-M Jun 23, 2024
e5dac12
Add missing `message.rs`
Philipp-M Jun 23, 2024
5101fc1
Add `README`
Philipp-M Jun 23, 2024
e482dc8
Fix some doc comments
Philipp-M Jun 23, 2024
0ed9231
Merge branch 'main' into xilem_web2
Philipp-M Jun 25, 2024
51f01ca
Fix CI
Philipp-M Jun 25, 2024
d5e420d
Remove blog example
Philipp-M Jun 27, 2024
1b53a9b
Make the `DomViewSequence` a private (crate level) implementation det…
Philipp-M Jun 27, 2024
e538bd0
Readd missing LICENSE to xilem_web
Philipp-M Jun 27, 2024
6aa53ab
Move `ViewCtx` to a separate module `context`
Philipp-M Jun 27, 2024
ac8caa6
Add builder methods from old `ViewExt` trait to `DomView` trait and f…
Philipp-M Jun 27, 2024
40745c8
Add missing License header
Philipp-M Jun 27, 2024
4e29707
Readd meta in Cargo.toml
Philipp-M Jun 27, 2024
4eb1e1d
... and readd package.metadata.docs for scrape-examples in Cargo.toml
Philipp-M Jun 27, 2024
bdf4259
Improve readme and use same trick as in Xilem Core
Philipp-M Jun 27, 2024
05038d6
Use `Rc<[ViewId]>` instead of `Vec<ViewId>` where possible
Philipp-M Jun 27, 2024
ca212a0
Address Review Feedback
Philipp-M Jun 28, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
40 changes: 12 additions & 28 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,10 @@ members = [
"masonry",

"xilem_web",
"xilem_web/xilem_web_core",
"xilem_web/web_examples/counter",
"xilem_web/web_examples/counter_custom_element",
"xilem_web/web_examples/todomvc",
"xilem_web/web_examples/blog",
"xilem_web/web_examples/mathml_svg",
"xilem_web/web_examples/svgtoy",
]
Expand Down
4 changes: 4 additions & 0 deletions xilem_core/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ publish = false # We'll publish this alongside Xilem 0.2

[dependencies]
tracing.workspace = true
kurbo = { optional = true, workspace = true }

[lints]
workspace = true
Expand All @@ -20,3 +21,6 @@ workspace = true
default-target = "x86_64-unknown-linux-gnu"
# xilem_core is entirely platform-agnostic, so only display docs for one platform
targets = []

[features]
kurbo = ["dep:kurbo"]
1 change: 1 addition & 0 deletions xilem_core/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ pub use view::{View, ViewId, ViewPathTracker};
mod views;
pub use views::{
adapt, map_action, map_state, memoize, one_of, Adapt, AdaptThunk, MapAction, MapState, Memoize,
OrphanView,
};

mod message;
Expand Down
79 changes: 78 additions & 1 deletion xilem_core/src/sequence.rs
Original file line number Diff line number Diff line change
Expand Up @@ -511,7 +511,7 @@ where
) -> MessageResult<Action, Message> {
let (start, rest) = id_path
.split_first()
.expect("Id path has elements for Option<ViewSequence>");
.expect("Id path has elements for Vec<ViewSequence>");
let (index, generation) = view_id_to_index_generation(*start);
let stored_generation = &seq_state.generations[index];
if *stored_generation != generation {
Expand All @@ -524,6 +524,83 @@ where
}
}

impl<State, Action, Context, Element, Marker, Seq, Message, const N: usize>
ViewSequence<State, Action, Context, Element, Vec<Marker>, Message> for [Seq; N]
where
Seq: ViewSequence<State, Action, Context, Element, Marker, Message>,
Context: ViewPathTracker,
Element: ViewElement,
{
type SeqState = [Seq::SeqState; N];

#[doc(hidden)]
fn seq_build(&self, ctx: &mut Context, elements: &mut AppendVec<Element>) -> Self::SeqState {
// there's no enumerate directly on an array
let mut generation = 0;
self.each_ref().map(|vs| {
let state = ctx.with_id(ViewId::new(generation), |ctx| vs.seq_build(ctx, elements));
generation += 1;
state
})
}

#[doc(hidden)]
fn seq_rebuild(
&self,
prev: &Self,
seq_state: &mut Self::SeqState,
ctx: &mut Context,
elements: &mut impl ElementSplice<Element>,
) {
for (gen, ((seq, prev_seq), state)) in self.iter().zip(prev).zip(seq_state).enumerate() {
ctx.with_id(
ViewId::new(gen.try_into().expect("usize should fit in u64")),
|ctx| {
seq.seq_rebuild(prev_seq, state, ctx, elements);
},
);
}
}

#[doc(hidden)]
fn seq_message(
&self,
seq_state: &mut Self::SeqState,
id_path: &[ViewId],
message: Message,
app_state: &mut State,
) -> MessageResult<Action, Message> {
let (start, rest) = id_path
.split_first()
.expect("Id path has elements for [ViewSequence; N]");

let index: usize = start
.routing_id()
.try_into()
.expect("u64 should fit in usize");
// Panics if index is out of bounds, but we know it isn't because this is the same generation
let inner_state = &mut seq_state[index];
self[index].seq_message(inner_state, rest, message, app_state)
}

#[doc(hidden)]
fn seq_teardown(
&self,
seq_state: &mut Self::SeqState,
ctx: &mut Context,
elements: &mut impl ElementSplice<Element>,
) {
for (gen, (seq, state)) in self.iter().zip(seq_state).enumerate() {
ctx.with_id(
ViewId::new(gen.try_into().expect("usize should fit in u64")),
|ctx| {
seq.seq_teardown(state, ctx, elements);
},
);
}
}
}

impl<State, Action, Context, Element, Message>
ViewSequence<State, Action, Context, Element, (), Message> for ()
where
Expand Down
3 changes: 3 additions & 0 deletions xilem_core/src/views/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,3 +15,6 @@ pub use memoize::{memoize, Memoize};

/// Statically typed alternatives to the type-erased [`AnyView`](`crate::AnyView`).
pub mod one_of;

mod orphan;
pub use orphan::OrphanView;
125 changes: 125 additions & 0 deletions xilem_core/src/views/orphan.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,125 @@
// Copyright 2024 the Xilem Authors
// SPDX-License-Identifier: Apache-2.0

use crate::{DynMessage, MessageResult, Mut, View, ViewElement, ViewId, ViewPathTracker};

/// This trait provides a way to add [`View`] implementations for types that would be restricted otherwise by the orphan rules.
/// Every type that can be supported with this trait, needs a concrete `View` implementation in xilem_core, possibly feature-gated.
pub trait OrphanView<V, State, Action, Message = DynMessage>: ViewPathTracker + Sized {
/// See [`View::Element`]
type OrphanElement: ViewElement;
/// See [`View::ViewState`]
type OrphanViewState;

/// See [`View::build`]
fn orphan_build(view: &V, ctx: &mut Self) -> (Self::OrphanElement, Self::OrphanViewState);

/// See [`View::rebuild`]
fn orphan_rebuild<'el>(
new: &V,
prev: &V,
view_state: &mut Self::OrphanViewState,
ctx: &mut Self,
element: Mut<'el, Self::OrphanElement>,
) -> Mut<'el, Self::OrphanElement>;

/// See [`View::teardown`]
fn orphan_teardown(
view: &V,
view_state: &mut Self::OrphanViewState,
ctx: &mut Self,
element: Mut<'_, Self::OrphanElement>,
);

/// See [`View::message`]
fn orphan_message(
view: &V,
view_state: &mut Self::OrphanViewState,
id_path: &[ViewId],
message: Message,
app_state: &mut State,
) -> MessageResult<Action, Message>;
}

macro_rules! impl_orphan_view_for {
($ty: ty) => {
impl<State, Action, Context, Message> View<State, Action, Context, Message> for $ty
where
Context: OrphanView<$ty, State, Action, Message>,
{
type Element = Context::OrphanElement;

type ViewState = Context::OrphanViewState;

fn build(&self, ctx: &mut Context) -> (Self::Element, Self::ViewState) {
Context::orphan_build(self, ctx)
}

fn rebuild<'el>(
&self,
prev: &Self,
view_state: &mut Self::ViewState,
ctx: &mut Context,
element: Mut<'el, Self::Element>,
) -> Mut<'el, Self::Element> {
Context::orphan_rebuild(self, prev, view_state, ctx, element)
}

fn teardown(
&self,
view_state: &mut Self::ViewState,
ctx: &mut Context,
element: Mut<'_, Self::Element>,
) {
Context::orphan_teardown(self, view_state, ctx, element);
}

fn message(
&self,
view_state: &mut Self::ViewState,
id_path: &[ViewId],
message: Message,
app_state: &mut State,
) -> MessageResult<Action, Message> {
Context::orphan_message(self, view_state, id_path, message, app_state)
}
}
};
}

// string impls
impl_orphan_view_for!(&'static str);
impl_orphan_view_for!(alloc::string::String);
impl_orphan_view_for!(alloc::borrow::Cow<'static, str>);

// number impls
impl_orphan_view_for!(f32);
impl_orphan_view_for!(f64);
impl_orphan_view_for!(i8);
impl_orphan_view_for!(u8);
impl_orphan_view_for!(i16);
impl_orphan_view_for!(u16);
impl_orphan_view_for!(i32);
impl_orphan_view_for!(u32);
impl_orphan_view_for!(i64);
impl_orphan_view_for!(u64);
impl_orphan_view_for!(u128);
impl_orphan_view_for!(isize);
impl_orphan_view_for!(usize);

#[cfg(feature = "kurbo")]
mod kurbo {
use super::OrphanView;
use crate::{MessageResult, Mut, View, ViewId};
impl_orphan_view_for!(kurbo::PathSeg);
impl_orphan_view_for!(kurbo::Arc);
impl_orphan_view_for!(kurbo::BezPath);
impl_orphan_view_for!(kurbo::Circle);
impl_orphan_view_for!(kurbo::CircleSegment);
impl_orphan_view_for!(kurbo::CubicBez);
impl_orphan_view_for!(kurbo::Ellipse);
impl_orphan_view_for!(kurbo::Line);
impl_orphan_view_for!(kurbo::QuadBez);
impl_orphan_view_for!(kurbo::Rect);
impl_orphan_view_for!(kurbo::RoundedRect);
}
Loading
Loading