0

我正在实现 .NET 的DragMove方法的 Rust 替代方法,但是结果导致应用程序在两个相对位置之间闪烁。

请参阅截屏视频示例项目

我用来执行拖动移动的代码:

let mut mouse_down = false;
let mut last_pos: Option<PhysicalPosition<f64>> = None;
event_loop.run(move |event, _, control_flow| match event {
Event::WindowEvent {
        event: WindowEvent::CursorMoved {
            position,
            ..
        },
        ..
    } => {
        let gl_window = display.gl_window();
        let window = gl_window.window();
        if mouse_down {
            if last_pos.is_some() {
                let previous_pos = last_pos.unwrap();
                let delta_x = previous_pos.x - position.x;
                let delta_y = previous_pos.y - position.y;
                window.set_outer_position(PhysicalPosition::new(position.x + delta_x, position.y + delta_y));
            }
            last_pos = Some(position);
        }
    }
    Event::WindowEvent {
        event: WindowEvent::MouseInput{
            state,
            button,
            ..
        },
        ..
    } => {
        mouse_down = button == MouseButton::Left && state == ElementState::Pressed;
        if !mouse_down {
            last_pos = None;
        }
    }
    _ => {}
});
4

1 回答 1

0

CursorMoved报告

(x,y)相对于窗口左上角的像素坐标。

当您稍后使用该位置时set_outer_position,您实际上是将窗口相对坐标重新解释为屏幕相对坐标。

您应该将偏移量应用于从external_position返回的位置。

虽然这解决了眼前的问题,但我不确定它是否足以解释窗口的移动。当您处理下一个 CursorMoved事件时,坐标仍然是相对于窗口的,但窗口已经移动了。这可能会产生伪影。

更强大的解决方案是在拖动移动操作开始时存储窗口的位置,并通过累积的增量偏移该位置。

于 2020-07-26T20:27:43.490 回答