Desktop: Fix missing resize events causing all-gray window on Mac after launch (#3445)
* okayish solution should be improved at some point but for now it works well enough. * do leftover renames * better solution * less weird resize frames * move surface reconfiguration * fix recent desktop mac breakages * better looking resize on mac * fix background color * Fix blank screen on window initialization * cleanup --------- Co-authored-by: Keavon Chambers <keavon@keavon.com>
This commit is contained in:
parent
5b472a64b2
commit
2e4481880e
|
|
@ -5,7 +5,7 @@ use std::path::{Path, PathBuf};
|
||||||
use crate::common::*;
|
use crate::common::*;
|
||||||
|
|
||||||
const PACKAGE: &str = "graphite-desktop-platform-win";
|
const PACKAGE: &str = "graphite-desktop-platform-win";
|
||||||
const EXECUTABLE: &str = "graphite-editor.exe";
|
const EXECUTABLE: &str = "graphite.exe";
|
||||||
|
|
||||||
pub fn main() -> Result<(), Box<dyn Error>> {
|
pub fn main() -> Result<(), Box<dyn Error>> {
|
||||||
let app_bin = build_bin(PACKAGE, None)?;
|
let app_bin = build_bin(PACKAGE, None)?;
|
||||||
|
|
|
||||||
|
|
@ -18,22 +18,22 @@ use crate::cef;
|
||||||
use crate::consts::CEF_MESSAGE_LOOP_MAX_ITERATIONS;
|
use crate::consts::CEF_MESSAGE_LOOP_MAX_ITERATIONS;
|
||||||
use crate::event::{AppEvent, AppEventScheduler};
|
use crate::event::{AppEvent, AppEventScheduler};
|
||||||
use crate::persist::PersistentData;
|
use crate::persist::PersistentData;
|
||||||
use crate::render::GraphicsState;
|
use crate::render::{RenderError, RenderState};
|
||||||
use crate::window::Window;
|
use crate::window::Window;
|
||||||
use crate::wrapper::messages::{DesktopFrontendMessage, DesktopWrapperMessage, Platform};
|
use crate::wrapper::messages::{DesktopFrontendMessage, DesktopWrapperMessage, Platform};
|
||||||
use crate::wrapper::{DesktopWrapper, NodeGraphExecutionResult, WgpuContext, serialize_frontend_messages};
|
use crate::wrapper::{DesktopWrapper, NodeGraphExecutionResult, WgpuContext, serialize_frontend_messages};
|
||||||
|
|
||||||
pub(crate) struct App {
|
pub(crate) struct App {
|
||||||
cef_context: Box<dyn cef::CefContext>,
|
render_state: Option<RenderState>,
|
||||||
|
wgpu_context: WgpuContext,
|
||||||
window: Option<Window>,
|
window: Option<Window>,
|
||||||
window_scale: f64,
|
window_scale: f64,
|
||||||
cef_schedule: Option<Instant>,
|
|
||||||
cef_view_info_sender: Sender<cef::ViewInfoUpdate>,
|
|
||||||
graphics_state: Option<GraphicsState>,
|
|
||||||
wgpu_context: WgpuContext,
|
|
||||||
app_event_receiver: Receiver<AppEvent>,
|
app_event_receiver: Receiver<AppEvent>,
|
||||||
app_event_scheduler: AppEventScheduler,
|
app_event_scheduler: AppEventScheduler,
|
||||||
desktop_wrapper: DesktopWrapper,
|
desktop_wrapper: DesktopWrapper,
|
||||||
|
cef_context: Box<dyn cef::CefContext>,
|
||||||
|
cef_schedule: Option<Instant>,
|
||||||
|
cef_view_info_sender: Sender<cef::ViewInfoUpdate>,
|
||||||
last_ui_update: Instant,
|
last_ui_update: Instant,
|
||||||
avg_frame_time: f32,
|
avg_frame_time: f32,
|
||||||
start_render_sender: SyncSender<()>,
|
start_render_sender: SyncSender<()>,
|
||||||
|
|
@ -77,17 +77,17 @@ impl App {
|
||||||
persistent_data.load_from_disk();
|
persistent_data.load_from_disk();
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
cef_context,
|
render_state: None,
|
||||||
window: None,
|
|
||||||
window_scale: 1.0,
|
|
||||||
cef_schedule: Some(Instant::now()),
|
|
||||||
graphics_state: None,
|
|
||||||
cef_view_info_sender,
|
|
||||||
wgpu_context,
|
wgpu_context,
|
||||||
|
window: None,
|
||||||
|
window_scale: 1.,
|
||||||
app_event_receiver,
|
app_event_receiver,
|
||||||
app_event_scheduler,
|
app_event_scheduler,
|
||||||
desktop_wrapper: DesktopWrapper::new(),
|
desktop_wrapper: DesktopWrapper::new(),
|
||||||
last_ui_update: Instant::now(),
|
last_ui_update: Instant::now(),
|
||||||
|
cef_context,
|
||||||
|
cef_schedule: Some(Instant::now()),
|
||||||
|
cef_view_info_sender,
|
||||||
avg_frame_time: 0.,
|
avg_frame_time: 0.,
|
||||||
start_render_sender,
|
start_render_sender,
|
||||||
web_communication_initialized: false,
|
web_communication_initialized: false,
|
||||||
|
|
@ -162,23 +162,23 @@ impl App {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
DesktopFrontendMessage::UpdateViewportPhysicalBounds { x, y, width, height } => {
|
DesktopFrontendMessage::UpdateViewportPhysicalBounds { x, y, width, height } => {
|
||||||
if let Some(graphics_state) = &mut self.graphics_state
|
if let Some(render_state) = &mut self.render_state
|
||||||
&& let Some(window) = &self.window
|
&& let Some(window) = &self.window
|
||||||
{
|
{
|
||||||
let window_size = window.surface_size();
|
let window_size = window.surface_size();
|
||||||
|
|
||||||
let viewport_offset_x = x / window_size.width as f64;
|
let viewport_offset_x = x / window_size.width as f64;
|
||||||
let viewport_offset_y = y / window_size.height as f64;
|
let viewport_offset_y = y / window_size.height as f64;
|
||||||
graphics_state.set_viewport_offset([viewport_offset_x as f32, viewport_offset_y as f32]);
|
render_state.set_viewport_offset([viewport_offset_x as f32, viewport_offset_y as f32]);
|
||||||
|
|
||||||
let viewport_scale_x = if width != 0.0 { window_size.width as f64 / width } else { 1.0 };
|
let viewport_scale_x = if width != 0.0 { window_size.width as f64 / width } else { 1.0 };
|
||||||
let viewport_scale_y = if height != 0.0 { window_size.height as f64 / height } else { 1.0 };
|
let viewport_scale_y = if height != 0.0 { window_size.height as f64 / height } else { 1.0 };
|
||||||
graphics_state.set_viewport_scale([viewport_scale_x as f32, viewport_scale_y as f32]);
|
render_state.set_viewport_scale([viewport_scale_x as f32, viewport_scale_y as f32]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DesktopFrontendMessage::UpdateOverlays(scene) => {
|
DesktopFrontendMessage::UpdateOverlays(scene) => {
|
||||||
if let Some(graphics_state) = &mut self.graphics_state {
|
if let Some(render_state) = &mut self.render_state {
|
||||||
graphics_state.set_overlays_scene(scene);
|
render_state.set_overlays_scene(scene);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DesktopFrontendMessage::PersistenceWriteDocument { id, document } => {
|
DesktopFrontendMessage::PersistenceWriteDocument { id, document } => {
|
||||||
|
|
@ -331,19 +331,18 @@ impl App {
|
||||||
NodeGraphExecutionResult::HasRun(texture) => {
|
NodeGraphExecutionResult::HasRun(texture) => {
|
||||||
self.dispatch_desktop_wrapper_message(DesktopWrapperMessage::PollNodeGraphEvaluation);
|
self.dispatch_desktop_wrapper_message(DesktopWrapperMessage::PollNodeGraphEvaluation);
|
||||||
if let Some(texture) = texture
|
if let Some(texture) = texture
|
||||||
&& let Some(graphics_state) = self.graphics_state.as_mut()
|
&& let Some(render_state) = self.render_state.as_mut()
|
||||||
&& let Some(window) = self.window.as_ref()
|
&& let Some(window) = self.window.as_ref()
|
||||||
{
|
{
|
||||||
graphics_state.bind_viewport_texture(texture);
|
render_state.bind_viewport_texture(texture);
|
||||||
window.request_redraw();
|
window.request_redraw();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
NodeGraphExecutionResult::NotRun => {}
|
NodeGraphExecutionResult::NotRun => {}
|
||||||
},
|
},
|
||||||
AppEvent::UiUpdate(texture) => {
|
AppEvent::UiUpdate(texture) => {
|
||||||
if let Some(graphics_state) = self.graphics_state.as_mut() {
|
if let Some(render_state) = self.render_state.as_mut() {
|
||||||
graphics_state.resize(texture.width(), texture.height());
|
render_state.bind_ui_texture(texture);
|
||||||
graphics_state.bind_ui_texture(texture);
|
|
||||||
let elapsed = self.last_ui_update.elapsed().as_secs_f32();
|
let elapsed = self.last_ui_update.elapsed().as_secs_f32();
|
||||||
self.last_ui_update = Instant::now();
|
self.last_ui_update = Instant::now();
|
||||||
if elapsed < 0.5 {
|
if elapsed < 0.5 {
|
||||||
|
|
@ -385,13 +384,18 @@ impl ApplicationHandler for App {
|
||||||
|
|
||||||
self.window_scale = window.scale_factor();
|
self.window_scale = window.scale_factor();
|
||||||
let _ = self.cef_view_info_sender.send(cef::ViewInfoUpdate::Scale(self.window_scale));
|
let _ = self.cef_view_info_sender.send(cef::ViewInfoUpdate::Scale(self.window_scale));
|
||||||
|
|
||||||
|
// Ensures the CEF texture does not remain at 1x1 pixels until the window is resized by the user
|
||||||
|
// Affects only some Mac devices (issue found on 2023 M2 Mac Mini).
|
||||||
|
let PhysicalSize { width, height } = window.surface_size();
|
||||||
|
let _ = self.cef_view_info_sender.send(cef::ViewInfoUpdate::Size { width, height });
|
||||||
|
|
||||||
self.cef_context.notify_view_info_changed();
|
self.cef_context.notify_view_info_changed();
|
||||||
|
|
||||||
self.window = Some(window);
|
self.window = Some(window);
|
||||||
|
|
||||||
let graphics_state = GraphicsState::new(self.window.as_ref().unwrap(), self.wgpu_context.clone());
|
let render_state = RenderState::new(self.window.as_ref().unwrap(), self.wgpu_context.clone());
|
||||||
|
self.render_state = Some(render_state);
|
||||||
self.graphics_state = Some(graphics_state);
|
|
||||||
|
|
||||||
self.desktop_wrapper.init(self.wgpu_context.clone());
|
self.desktop_wrapper.init(self.wgpu_context.clone());
|
||||||
|
|
||||||
|
|
@ -418,14 +422,18 @@ impl ApplicationHandler for App {
|
||||||
self.app_event_scheduler.schedule(AppEvent::CloseWindow);
|
self.app_event_scheduler.schedule(AppEvent::CloseWindow);
|
||||||
}
|
}
|
||||||
WindowEvent::SurfaceResized(PhysicalSize { width, height }) => {
|
WindowEvent::SurfaceResized(PhysicalSize { width, height }) => {
|
||||||
let _ = self.cef_view_info_sender.send(cef::ViewInfoUpdate::Size {
|
let _ = self.cef_view_info_sender.send(cef::ViewInfoUpdate::Size { width, height });
|
||||||
width: width as usize,
|
|
||||||
height: height as usize,
|
|
||||||
});
|
|
||||||
self.cef_context.notify_view_info_changed();
|
self.cef_context.notify_view_info_changed();
|
||||||
|
|
||||||
|
if let Some(render_state) = &mut self.render_state {
|
||||||
|
render_state.resize(width, height);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(window) = &self.window {
|
if let Some(window) = &self.window {
|
||||||
let maximized = window.is_maximized();
|
let maximized = window.is_maximized();
|
||||||
self.app_event_scheduler.schedule(AppEvent::DesktopWrapperMessage(DesktopWrapperMessage::UpdateMaximized { maximized }));
|
self.app_event_scheduler.schedule(AppEvent::DesktopWrapperMessage(DesktopWrapperMessage::UpdateMaximized { maximized }));
|
||||||
|
|
||||||
|
window.request_redraw();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
WindowEvent::ScaleFactorChanged { scale_factor, .. } => {
|
WindowEvent::ScaleFactorChanged { scale_factor, .. } => {
|
||||||
|
|
@ -434,18 +442,24 @@ impl ApplicationHandler for App {
|
||||||
self.cef_context.notify_view_info_changed();
|
self.cef_context.notify_view_info_changed();
|
||||||
}
|
}
|
||||||
WindowEvent::RedrawRequested => {
|
WindowEvent::RedrawRequested => {
|
||||||
let Some(ref mut graphics_state) = self.graphics_state else { return };
|
let Some(render_state) = &mut self.render_state else { return };
|
||||||
// Only rerender once we have a new UI texture to display
|
|
||||||
if let Some(window) = &self.window {
|
if let Some(window) = &self.window {
|
||||||
match graphics_state.render(window) {
|
let size = window.surface_size();
|
||||||
|
render_state.resize(size.width, size.height);
|
||||||
|
|
||||||
|
match render_state.render(window) {
|
||||||
Ok(_) => {}
|
Ok(_) => {}
|
||||||
Err(wgpu::SurfaceError::Lost) => {
|
Err(RenderError::OutdatedUITextureError) => {
|
||||||
|
self.cef_context.notify_view_info_changed();
|
||||||
|
}
|
||||||
|
Err(RenderError::SurfaceError(wgpu::SurfaceError::Lost)) => {
|
||||||
tracing::warn!("lost surface");
|
tracing::warn!("lost surface");
|
||||||
}
|
}
|
||||||
Err(wgpu::SurfaceError::OutOfMemory) => {
|
Err(RenderError::SurfaceError(wgpu::SurfaceError::OutOfMemory)) => {
|
||||||
|
tracing::error!("GPU out of memory");
|
||||||
event_loop.exit();
|
event_loop.exit();
|
||||||
}
|
}
|
||||||
Err(e) => tracing::error!("{:?}", e),
|
Err(RenderError::SurfaceError(e)) => tracing::error!("Render error: {:?}", e),
|
||||||
}
|
}
|
||||||
let _ = self.start_render_sender.try_send(());
|
let _ = self.start_render_sender.try_send(());
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -55,8 +55,8 @@ pub(crate) trait CefEventHandler: Send + Sync + 'static {
|
||||||
|
|
||||||
#[derive(Clone, Copy)]
|
#[derive(Clone, Copy)]
|
||||||
pub(crate) struct ViewInfo {
|
pub(crate) struct ViewInfo {
|
||||||
width: usize,
|
width: u32,
|
||||||
height: usize,
|
height: u32,
|
||||||
scale: f64,
|
scale: f64,
|
||||||
}
|
}
|
||||||
impl ViewInfo {
|
impl ViewInfo {
|
||||||
|
|
@ -78,10 +78,10 @@ impl ViewInfo {
|
||||||
pub(crate) fn zoom(&self) -> f64 {
|
pub(crate) fn zoom(&self) -> f64 {
|
||||||
self.scale.ln() / 1.2_f64.ln()
|
self.scale.ln() / 1.2_f64.ln()
|
||||||
}
|
}
|
||||||
pub(crate) fn width(&self) -> usize {
|
pub(crate) fn width(&self) -> u32 {
|
||||||
self.width
|
self.width
|
||||||
}
|
}
|
||||||
pub(crate) fn height(&self) -> usize {
|
pub(crate) fn height(&self) -> u32 {
|
||||||
self.height
|
self.height
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -92,7 +92,7 @@ impl Default for ViewInfo {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) enum ViewInfoUpdate {
|
pub(crate) enum ViewInfoUpdate {
|
||||||
Size { width: usize, height: usize },
|
Size { width: u32, height: u32 },
|
||||||
Scale(f64),
|
Scale(f64),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
#[derive(clap::Parser)]
|
#[derive(clap::Parser)]
|
||||||
#[clap(name = "graphite-editor", version)]
|
#[clap(name = "graphite", version)]
|
||||||
pub struct Cli {
|
pub struct Cli {
|
||||||
#[arg(help = "Files to open on startup")]
|
#[arg(help = "Files to open on startup")]
|
||||||
pub files: Vec<std::path::PathBuf>,
|
pub files: Vec<std::path::PathBuf>,
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
pub(crate) const APP_NAME: &str = "Graphite";
|
pub(crate) const APP_NAME: &str = "Graphite";
|
||||||
|
#[cfg(target_os = "linux")]
|
||||||
pub(crate) const APP_ID: &str = "rs.graphite.Graphite";
|
pub(crate) const APP_ID: &str = "rs.graphite.Graphite";
|
||||||
|
|
||||||
pub(crate) const APP_DIRECTORY_NAME: &str = "graphite";
|
pub(crate) const APP_DIRECTORY_NAME: &str = "graphite";
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,5 @@
|
||||||
mod frame_buffer_ref;
|
mod frame_buffer_ref;
|
||||||
pub(crate) use frame_buffer_ref::FrameBufferRef;
|
pub(crate) use frame_buffer_ref::FrameBufferRef;
|
||||||
|
|
||||||
mod graphics_state;
|
mod state;
|
||||||
pub(crate) use graphics_state::GraphicsState;
|
pub(crate) use state::{RenderError, RenderState};
|
||||||
|
|
|
||||||
|
|
@ -23,6 +23,8 @@ fn vs_main(@builtin(vertex_index) vertex_index: u32) -> VertexOutput {
|
||||||
struct Constants {
|
struct Constants {
|
||||||
viewport_scale: vec2<f32>,
|
viewport_scale: vec2<f32>,
|
||||||
viewport_offset: vec2<f32>,
|
viewport_offset: vec2<f32>,
|
||||||
|
ui_scale: vec2<f32>,
|
||||||
|
background_color: vec4<f32>,
|
||||||
};
|
};
|
||||||
|
|
||||||
var<push_constant> constants: Constants;
|
var<push_constant> constants: Constants;
|
||||||
|
|
@ -38,19 +40,29 @@ var s_diffuse: sampler;
|
||||||
|
|
||||||
@fragment
|
@fragment
|
||||||
fn fs_main(in: VertexOutput) -> @location(0) vec4<f32> {
|
fn fs_main(in: VertexOutput) -> @location(0) vec4<f32> {
|
||||||
let ui_linear = srgb_to_linear(textureSample(t_ui, s_diffuse, in.tex_coords));
|
let ui_coordinate = in.tex_coords * constants.ui_scale;
|
||||||
|
if (ui_coordinate.x < 0.0 || ui_coordinate.x > 1.0 ||
|
||||||
|
ui_coordinate.y < 0.0 || ui_coordinate.y > 1.0) {
|
||||||
|
return srgb_to_linear(constants.background_color);
|
||||||
|
}
|
||||||
|
|
||||||
|
let ui_linear = srgb_to_linear(textureSample(t_ui, s_diffuse, ui_coordinate));
|
||||||
if (ui_linear.a >= 0.999) {
|
if (ui_linear.a >= 0.999) {
|
||||||
return ui_linear;
|
return ui_linear;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UI texture is premultiplied, we need to unpremultiply before blending
|
||||||
|
let ui_srgb = linear_to_srgb(unpremultiply(ui_linear));
|
||||||
|
|
||||||
let viewport_coordinate = (in.tex_coords - constants.viewport_offset) * constants.viewport_scale;
|
let viewport_coordinate = (in.tex_coords - constants.viewport_offset) * constants.viewport_scale;
|
||||||
|
if (viewport_coordinate.x < 0.0 || viewport_coordinate.x > 1.0 ||
|
||||||
|
viewport_coordinate.y < 0.0 || viewport_coordinate.y > 1.0) {
|
||||||
|
return srgb_to_linear(constants.background_color);
|
||||||
|
}
|
||||||
|
|
||||||
let overlay_srgb = textureSample(t_overlays, s_diffuse, viewport_coordinate);
|
let overlay_srgb = textureSample(t_overlays, s_diffuse, viewport_coordinate);
|
||||||
let viewport_srgb = textureSample(t_viewport, s_diffuse, viewport_coordinate);
|
let viewport_srgb = textureSample(t_viewport, s_diffuse, viewport_coordinate);
|
||||||
|
|
||||||
// UI texture is premultiplied, we need to unpremultiply before blending
|
|
||||||
let ui_srgb = linear_to_srgb(unpremultiply(ui_linear));
|
|
||||||
|
|
||||||
if (overlay_srgb.a < 0.001) {
|
if (overlay_srgb.a < 0.001) {
|
||||||
if (ui_srgb.a < 0.001) {
|
if (ui_srgb.a < 0.001) {
|
||||||
return srgb_to_linear(viewport_srgb);
|
return srgb_to_linear(viewport_srgb);
|
||||||
|
|
|
||||||
|
|
@ -4,7 +4,7 @@ use crate::wrapper::{Color, WgpuContext, WgpuExecutor};
|
||||||
|
|
||||||
#[derive(derivative::Derivative)]
|
#[derive(derivative::Derivative)]
|
||||||
#[derivative(Debug)]
|
#[derivative(Debug)]
|
||||||
pub(crate) struct GraphicsState {
|
pub(crate) struct RenderState {
|
||||||
surface: wgpu::Surface<'static>,
|
surface: wgpu::Surface<'static>,
|
||||||
context: WgpuContext,
|
context: WgpuContext,
|
||||||
executor: WgpuExecutor,
|
executor: WgpuExecutor,
|
||||||
|
|
@ -12,6 +12,8 @@ pub(crate) struct GraphicsState {
|
||||||
render_pipeline: wgpu::RenderPipeline,
|
render_pipeline: wgpu::RenderPipeline,
|
||||||
transparent_texture: wgpu::Texture,
|
transparent_texture: wgpu::Texture,
|
||||||
sampler: wgpu::Sampler,
|
sampler: wgpu::Sampler,
|
||||||
|
desired_width: u32,
|
||||||
|
desired_height: u32,
|
||||||
viewport_scale: [f32; 2],
|
viewport_scale: [f32; 2],
|
||||||
viewport_offset: [f32; 2],
|
viewport_offset: [f32; 2],
|
||||||
viewport_texture: Option<wgpu::Texture>,
|
viewport_texture: Option<wgpu::Texture>,
|
||||||
|
|
@ -22,7 +24,7 @@ pub(crate) struct GraphicsState {
|
||||||
overlays_scene: Option<vello::Scene>,
|
overlays_scene: Option<vello::Scene>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GraphicsState {
|
impl RenderState {
|
||||||
pub(crate) fn new(window: &Window, context: WgpuContext) -> Self {
|
pub(crate) fn new(window: &Window, context: WgpuContext) -> Self {
|
||||||
let size = window.surface_size();
|
let size = window.surface_size();
|
||||||
let surface = window.create_surface(context.instance.clone());
|
let surface = window.create_surface(context.instance.clone());
|
||||||
|
|
@ -171,6 +173,8 @@ impl GraphicsState {
|
||||||
render_pipeline,
|
render_pipeline,
|
||||||
transparent_texture,
|
transparent_texture,
|
||||||
sampler,
|
sampler,
|
||||||
|
desired_width: size.width,
|
||||||
|
desired_height: size.height,
|
||||||
viewport_scale: [1.0, 1.0],
|
viewport_scale: [1.0, 1.0],
|
||||||
viewport_offset: [0.0, 0.0],
|
viewport_offset: [0.0, 0.0],
|
||||||
viewport_texture: None,
|
viewport_texture: None,
|
||||||
|
|
@ -182,6 +186,13 @@ impl GraphicsState {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn resize(&mut self, width: u32, height: u32) {
|
pub(crate) fn resize(&mut self, width: u32, height: u32) {
|
||||||
|
if width == self.desired_width && height == self.desired_height {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.desired_width = width;
|
||||||
|
self.desired_height = height;
|
||||||
|
|
||||||
if width > 0 && height > 0 && (self.config.width != width || self.config.height != height) {
|
if width > 0 && height > 0 && (self.config.width != width || self.config.height != height) {
|
||||||
self.config.width = width;
|
self.config.width = width;
|
||||||
self.config.height = height;
|
self.config.height = height;
|
||||||
|
|
@ -230,24 +241,33 @@ impl GraphicsState {
|
||||||
self.bind_overlays_texture(texture);
|
self.bind_overlays_texture(texture);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn render(&mut self, window: &Window) -> Result<(), wgpu::SurfaceError> {
|
pub(crate) fn render(&mut self, window: &Window) -> Result<(), RenderError> {
|
||||||
|
let ui_scale = if let Some(ui_texture) = &self.ui_texture
|
||||||
|
&& (self.desired_width != ui_texture.width() || self.desired_height != ui_texture.height())
|
||||||
|
{
|
||||||
|
Some([self.desired_width as f32 / ui_texture.width() as f32, self.desired_height as f32 / ui_texture.height() as f32])
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
if let Some(scene) = self.overlays_scene.take() {
|
if let Some(scene) = self.overlays_scene.take() {
|
||||||
self.render_overlays(scene);
|
self.render_overlays(scene);
|
||||||
}
|
}
|
||||||
|
|
||||||
let output = self.surface.get_current_texture()?;
|
let output = self.surface.get_current_texture().map_err(RenderError::SurfaceError)?;
|
||||||
|
|
||||||
let view = output.texture.create_view(&wgpu::TextureViewDescriptor::default());
|
let view = output.texture.create_view(&wgpu::TextureViewDescriptor::default());
|
||||||
|
|
||||||
let mut encoder = self.context.device.create_command_encoder(&wgpu::CommandEncoderDescriptor { label: Some("Render Encoder") });
|
let mut encoder = self.context.device.create_command_encoder(&wgpu::CommandEncoderDescriptor { label: Some("Render Encoder") });
|
||||||
|
|
||||||
{
|
{
|
||||||
let mut render_pass = encoder.begin_render_pass(&wgpu::RenderPassDescriptor {
|
let mut render_pass = encoder.begin_render_pass(&wgpu::RenderPassDescriptor {
|
||||||
label: Some("Render Pass"),
|
label: Some("Graphite Composition Render Pass"),
|
||||||
color_attachments: &[Some(wgpu::RenderPassColorAttachment {
|
color_attachments: &[Some(wgpu::RenderPassColorAttachment {
|
||||||
view: &view,
|
view: &view,
|
||||||
resolve_target: None,
|
resolve_target: None,
|
||||||
ops: wgpu::Operations {
|
ops: wgpu::Operations {
|
||||||
load: wgpu::LoadOp::Clear(wgpu::Color { r: 0.01, g: 0.01, b: 0.01, a: 1.0 }),
|
load: wgpu::LoadOp::Clear(wgpu::Color { r: 0.01, g: 0.01, b: 0.01, a: 1. }),
|
||||||
store: wgpu::StoreOp::Store,
|
store: wgpu::StoreOp::Store,
|
||||||
},
|
},
|
||||||
depth_slice: None,
|
depth_slice: None,
|
||||||
|
|
@ -264,11 +284,14 @@ impl GraphicsState {
|
||||||
bytemuck::bytes_of(&Constants {
|
bytemuck::bytes_of(&Constants {
|
||||||
viewport_scale: self.viewport_scale,
|
viewport_scale: self.viewport_scale,
|
||||||
viewport_offset: self.viewport_offset,
|
viewport_offset: self.viewport_offset,
|
||||||
|
ui_scale: ui_scale.unwrap_or([1., 1.]),
|
||||||
|
_pad: [0., 0.],
|
||||||
|
background_color: [0x22 as f32 / 0xff as f32, 0x22 as f32 / 0xff as f32, 0x22 as f32 / 0xff as f32, 1.], // #222222
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
if let Some(bind_group) = &self.bind_group {
|
if let Some(bind_group) = &self.bind_group {
|
||||||
render_pass.set_bind_group(0, bind_group, &[]);
|
render_pass.set_bind_group(0, bind_group, &[]);
|
||||||
render_pass.draw(0..6, 0..1); // Draw 3 vertices for fullscreen triangle
|
render_pass.draw(0..3, 0..1); // Draw 3 vertices for fullscreen triangle
|
||||||
} else {
|
} else {
|
||||||
tracing::warn!("No bind group available - showing clear color only");
|
tracing::warn!("No bind group available - showing clear color only");
|
||||||
}
|
}
|
||||||
|
|
@ -277,6 +300,10 @@ impl GraphicsState {
|
||||||
window.pre_present_notify();
|
window.pre_present_notify();
|
||||||
output.present();
|
output.present();
|
||||||
|
|
||||||
|
if ui_scale.is_some() {
|
||||||
|
return Err(RenderError::OutdatedUITextureError);
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -312,9 +339,17 @@ impl GraphicsState {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) enum RenderError {
|
||||||
|
OutdatedUITextureError,
|
||||||
|
SurfaceError(wgpu::SurfaceError),
|
||||||
|
}
|
||||||
|
|
||||||
#[repr(C)]
|
#[repr(C)]
|
||||||
#[derive(Copy, Clone, bytemuck::Pod, bytemuck::Zeroable)]
|
#[derive(Copy, Clone, bytemuck::Pod, bytemuck::Zeroable)]
|
||||||
struct Constants {
|
struct Constants {
|
||||||
viewport_scale: [f32; 2],
|
viewport_scale: [f32; 2],
|
||||||
viewport_offset: [f32; 2],
|
viewport_offset: [f32; 2],
|
||||||
|
ui_scale: [f32; 2],
|
||||||
|
_pad: [f32; 2],
|
||||||
|
background_color: [f32; 4],
|
||||||
}
|
}
|
||||||
|
|
@ -111,10 +111,7 @@ pub(super) fn intercept_frontend_message(dispatcher: &mut DesktopWrapperMessageD
|
||||||
dispatcher.respond(DesktopFrontendMessage::PersistenceLoadPreferences);
|
dispatcher.respond(DesktopFrontendMessage::PersistenceLoadPreferences);
|
||||||
}
|
}
|
||||||
#[cfg(target_os = "macos")]
|
#[cfg(target_os = "macos")]
|
||||||
FrontendMessage::UpdateMenuBarLayout {
|
FrontendMessage::UpdateMenuBarLayout { diff } => {
|
||||||
layout_target: graphite_editor::messages::tool::tool_messages::tool_prelude::LayoutTarget::MenuBar,
|
|
||||||
diff,
|
|
||||||
} => {
|
|
||||||
use graphite_editor::messages::tool::tool_messages::tool_prelude::{DiffUpdate, WidgetDiff};
|
use graphite_editor::messages::tool::tool_messages::tool_prelude::{DiffUpdate, WidgetDiff};
|
||||||
match diff.as_slice() {
|
match diff.as_slice() {
|
||||||
[
|
[
|
||||||
|
|
|
||||||
|
|
@ -3,14 +3,14 @@ pub(crate) mod menu {
|
||||||
use base64::engine::Engine;
|
use base64::engine::Engine;
|
||||||
use base64::engine::general_purpose::STANDARD as BASE64;
|
use base64::engine::general_purpose::STANDARD as BASE64;
|
||||||
|
|
||||||
use graphite_editor::messages::input_mapper::utility_types::input_keyboard::{Key, LabeledKey, LabeledShortcut};
|
use graphite_editor::messages::input_mapper::utility_types::input_keyboard::{Key, LabeledKeyOrMouseMotion, LabeledShortcut};
|
||||||
use graphite_editor::messages::input_mapper::utility_types::misc::ActionShortcut;
|
use graphite_editor::messages::input_mapper::utility_types::misc::ActionShortcut;
|
||||||
use graphite_editor::messages::layout::LayoutMessage;
|
use graphite_editor::messages::layout::LayoutMessage;
|
||||||
use graphite_editor::messages::tool::tool_messages::tool_prelude::{Layout, LayoutGroup, LayoutTarget, MenuListEntry, Widget, WidgetId};
|
use graphite_editor::messages::tool::tool_messages::tool_prelude::{Layout, LayoutGroup, LayoutTarget, MenuListEntry, Widget, WidgetId};
|
||||||
|
|
||||||
use crate::messages::{EditorMessage, KeyCode, MenuItem, Modifiers, Shortcut};
|
use crate::messages::{EditorMessage, KeyCode, MenuItem, Modifiers, Shortcut};
|
||||||
|
|
||||||
pub(crate) fn convert_menu_bar_layout_to_menu_items(layout: &Layout) -> Vec<MenuItem> {
|
pub(crate) fn convert_menu_bar_layout_to_menu_items(Layout(layout): &Layout) -> Vec<MenuItem> {
|
||||||
let layout_group = match layout.as_slice() {
|
let layout_group = match layout.as_slice() {
|
||||||
[layout_group] => layout_group,
|
[layout_group] => layout_group,
|
||||||
_ => panic!("Menu bar layout is supposed to have exactly one layout group"),
|
_ => panic!("Menu bar layout is supposed to have exactly one layout group"),
|
||||||
|
|
@ -68,9 +68,9 @@ pub(crate) mod menu {
|
||||||
value,
|
value,
|
||||||
label,
|
label,
|
||||||
icon,
|
icon,
|
||||||
shortcut_keys,
|
|
||||||
children,
|
|
||||||
disabled,
|
disabled,
|
||||||
|
tooltip_shortcut,
|
||||||
|
children,
|
||||||
..
|
..
|
||||||
}: &MenuListEntry = entry;
|
}: &MenuListEntry = entry;
|
||||||
path.push(value.clone());
|
path.push(value.clone());
|
||||||
|
|
@ -83,7 +83,7 @@ pub(crate) mod menu {
|
||||||
return MenuItem::SubMenu { id, text, enabled, items };
|
return MenuItem::SubMenu { id, text, enabled, items };
|
||||||
}
|
}
|
||||||
|
|
||||||
let shortcut = match shortcut_keys {
|
let shortcut = match tooltip_shortcut {
|
||||||
Some(ActionShortcut::Shortcut(LabeledShortcut(shortcut))) => convert_labeled_keys_to_shortcut(shortcut),
|
Some(ActionShortcut::Shortcut(LabeledShortcut(shortcut))) => convert_labeled_keys_to_shortcut(shortcut),
|
||||||
_ => None,
|
_ => None,
|
||||||
};
|
};
|
||||||
|
|
@ -126,10 +126,14 @@ pub(crate) mod menu {
|
||||||
items
|
items
|
||||||
}
|
}
|
||||||
|
|
||||||
fn convert_labeled_keys_to_shortcut(labeled_keys: &Vec<LabeledKey>) -> Option<Shortcut> {
|
fn convert_labeled_keys_to_shortcut(labeled_keys: &Vec<LabeledKeyOrMouseMotion>) -> Option<Shortcut> {
|
||||||
let mut key: Option<KeyCode> = None;
|
let mut key: Option<KeyCode> = None;
|
||||||
let mut modifiers = Modifiers::default();
|
let mut modifiers = Modifiers::default();
|
||||||
for labeled_key in labeled_keys {
|
for labeled_key in labeled_keys {
|
||||||
|
let LabeledKeyOrMouseMotion::Key(labeled_key) = labeled_key else {
|
||||||
|
// Return None for shortcuts that include mouse motion because we can't show them in native menu
|
||||||
|
return None;
|
||||||
|
};
|
||||||
match labeled_key.key() {
|
match labeled_key.key() {
|
||||||
Key::Shift => modifiers |= Modifiers::SHIFT,
|
Key::Shift => modifiers |= Modifiers::SHIFT,
|
||||||
Key::Control => modifiers |= Modifiers::CONTROL,
|
Key::Control => modifiers |= Modifiers::CONTROL,
|
||||||
|
|
|
||||||
|
|
@ -488,7 +488,7 @@ impl LayoutMessageHandler {
|
||||||
if layout_target == LayoutTarget::MenuBar {
|
if layout_target == LayoutTarget::MenuBar {
|
||||||
widget_diffs = vec![WidgetDiff {
|
widget_diffs = vec![WidgetDiff {
|
||||||
widget_path: Vec::new(),
|
widget_path: Vec::new(),
|
||||||
new_value: DiffUpdate::Layout(current.layout.clone()),
|
new_value: DiffUpdate::Layout(self.layouts[LayoutTarget::MenuBar as usize].clone()),
|
||||||
}];
|
}];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue