zng/render.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
//! Frame builder and other types.
//!
//! Frame rendering means building a display list and updating all widget transforms, no actual pixel rendering happens
//! during the render pass, the built display list is send to the view-process where it is actually rendered.
//!
//! Widgets render is centered around [`UiNode::render`] and [`UiNode::render_update`] using the [`FrameBuilder`]
//! and [`FrameUpdate`] types. Render builds a display list, updates widget transforms and hit-test shapes, during
//! render some values in the display list can be bound to a [`FrameValueKey`], this key can be used during `render_update`
//! to replace the value in the last display list instead of rebuilding it.
//!
//! Note that even without render-updates all widgets that do not request render and are not ancestor to one are reused.
//! Reused widgets only include a range of display items to copy from the previous display list. A normal release built window
//! can easily achieve 60FPS rendering even without render-updates, but reusable components should try to achieve best performance.
//!
//! ```
//! use zng::prelude_wgt::*;
//!
//! /// Fills the available space with a centered circle of the color.
//! ///
//! /// This node disables inline layout for the widget.
//! pub fn color_circle(color: impl IntoVar<Rgba>) -> impl UiNode {
//! let color = color.into_var();
//! let mut area = PxRect::zero();
//!
//! // key to the color in a rendered frame,
//! // can be used to update the frame without rebuilding the display list
//! let color_key = FrameValueKey::new_unique();
//!
//! match_node_leaf(move |op| match op {
//! UiNodeOp::Init => {
//! // request a frame update when the color changes
//! WIDGET.sub_var_render_update(&color);
//! }
//! UiNodeOp::Measure { wm, desired_size } => {
//! wm.disable_inline(); // is inline-block
//! *desired_size = LAYOUT.constraints().fill_size();
//! }
//! UiNodeOp::Layout { final_size, .. } => {
//! *final_size = LAYOUT.constraints().fill_size();
//!
//! // centered square
//! let mut a = PxRect::from_size(*final_size);
//! if a.size.width < a.size.height {
//! a.origin.y = (a.size.height - a.size.width) / Px(2);
//! a.size.height = a.size.width;
//! } else {
//! a.origin.x = (a.size.width - a.size.height) / Px(2);
//! a.size.width = a.size.height;
//! }
//!
//! if a != area {
//! area = a;
//! // request a full render because are is not keyed for updates
//! WIDGET.render();
//! }
//! }
//! UiNodeOp::Render { frame } => {
//! // clip a circle at the area
//! frame.push_clip_rounded_rect(area, PxCornerRadius::new_all(area.size), false, false, |frame| {
//! // fill the are with color, bind the color_key to the color
//! frame.push_color(area, color_key.bind_var(&color, |&c| c.into()));
//! });
//! }
//! UiNodeOp::RenderUpdate { update } => {
//! // update the color in the existing frame, this is an optimization
//! update.update_color_opt(color_key.update_var(&color, |&c| c.into()));
//! }
//! _ => {}
//! })
//! }
//! ```
//!
//! The example above declares a simple node that draws a colored circle, the circle color is keyed for render updates.
//!
//! ```
//! # use zng::prelude::*;
//! # let _scope = APP.defaults();
//! # fn color_circle(_color: impl IntoVar<zng::color::Rgba>) -> impl UiNode { widget::node::FillUiNode }
//! let color = var(colors::RED);
//! let mut i = 0u8;
//! # let _ =
//! Container! {
//! child = color_circle(color.easing_with(1.secs(), easing::linear, color::rgba_sampler));
//! gesture::on_click = hn!(|_| {
//! color.set(match i {
//! 0 => colors::YELLOW,
//! 1 => colors::GREEN,
//! 2 => colors::RED,
//! _ => unreachable!(),
//! });
//! i += 1;
//! if i == 3 {
//! i = 0;
//! }
//! });
//! }
//! # ;
//! ```
//!
//! [`UiNode::render`]: crate::widget::node::UiNode::render
//! [`UiNode::render_update`]: crate::widget::node::UiNode::render_update
//!
//! # Full API
//!
//! See [`zng_app::render`] for the full API.
pub use zng_app::render::{
ClipBuilder, FontSynthesis, FrameBuilder, FrameUpdate, FrameValue, FrameValueKey, FrameValueUpdate, HitTestBuilder, HitTestClipBuilder,
ImageRendering, ReferenceFrameId, ReuseRange, SpatialFrameId,
};
pub use zng_view_api::window::FrameId;