我实际上正在开发一个没有窗口装饰的 ScalaFX 应用程序。我想做的是让它可拖动,以便用户可以随意移动它。
我只需将拖动鼠标的坐标写入舞台 X/Y 坐标即可移动舞台。然而,这导致了一个滞后和闪烁的窗口。
ScalaFX中如何顺利实现拖拽舞台?
我实际上正在开发一个没有窗口装饰的 ScalaFX 应用程序。我想做的是让它可拖动,以便用户可以随意移动它。
我只需将拖动鼠标的坐标写入舞台 X/Y 坐标即可移动舞台。然而,这导致了一个滞后和闪烁的窗口。
ScalaFX中如何顺利实现拖拽舞台?
这是一个对我很有效的例子。它改编自“JavaFX 8 Introduction by Example”一书中的一个示例。它与您的尝试相比如何?
import scalafx.Includes._
import scalafx.application.JFXApp
import scalafx.application.JFXApp.PrimaryStage
import scalafx.geometry.Point2D
import scalafx.scene.Scene
import scalafx.scene.input.MouseEvent
import scalafx.scene.paint.Color
import scalafx.stage.{WindowEvent, StageStyle}
object DraggableApp extends JFXApp {
private var anchorPt: Point2D = null
private var previousLocation: Point2D = null
stage = new PrimaryStage {
initStyle(StageStyle.TRANSPARENT)
scene = new Scene {
fill = Color.rgb(0, 0, 0, 1.0)
}
}
// Initialize stage to be movable via mouse
initMovablePlayer()
/**
* Initialize the stage to allow the mouse cursor to move the application
* using dragging.
*/
private def initMovablePlayer(): Unit = {
val scene = stage.getScene
scene.onMousePressed = (event: MouseEvent) => anchorPt = new Point2D(event.screenX, event.screenY)
scene.onMouseDragged = (event: MouseEvent) =>
if (anchorPt != null && previousLocation != null) {
stage.x = previousLocation.x + event.screenX - anchorPt.x
stage.y = previousLocation.y + event.screenY - anchorPt.y
}
scene.onMouseReleased = (event: MouseEvent) => previousLocation = new Point2D(stage.getX, stage.getY)
stage.onShown = (event: WindowEvent) => previousLocation = new Point2D(stage.getX, stage.getY)
}
}
编辑:关于你关于调整舞台大小的问题,我尝试了以下变体,它在右键单击和拖动时调整舞台大小;我没有看到任何闪烁(在 OS X 上)。
import javafx.scene.{input => jfxsi}
// (...)
object DraggableApp extends JFXApp {
private var previousHeight = 0.0
private var previousWidth = 0.0
// (...)
private def initMovablePlayer(): Unit = {
val scene = stage.getScene
scene.onMousePressed = (event: MouseEvent) => anchorPt = new Point2D(event.screenX, event.screenY)
scene.onMouseDragged = (event: MouseEvent) =>
if (anchorPt != null && previousLocation != null) {
if (event.getButton == jfxsi.MouseButton.PRIMARY) {
stage.x = previousLocation.x + event.screenX - anchorPt.x
stage.y = previousLocation.y + event.screenY - anchorPt.y
} else if (event.getButton == jfxsi.MouseButton.SECONDARY) {
stage.width = previousWidth + event.screenX - anchorPt.x
stage.height = previousHeight + event.screenY - anchorPt.y
}
}
scene.onMouseReleased = (_: MouseEvent) => reset()
stage.onShown = (_: WindowEvent) => reset()
def reset (): Unit = {
previousLocation = new Point2D(stage.getX, stage.getY)
previousHeight = stage.getHeight
previousWidth = stage.getWidth
}
}
}
我知道这已经得到了回答,但为了完整起见,我想包括我过去这样做的方式。
致谢:可
拖动窗口:Pro JavaFX 8 示例
可调整大小的窗口:Little Child -允许用户调整未装饰舞台的大小
向上/向下还原:Undecorator - UndecoratorController.java
为(凌乱的)代码墙做好准备
class UndecoratedWindowHelper(val target: Stage) {
var anchorPt: Point2D = null
var prevLoc: Point2D = null
private var savedBounds: BoundingBox = _
private val _maximized: ReadOnlyBooleanWrapper = new ReadOnlyBooleanWrapper {value = false}
// begin init
val resizeListener = new ResizeListener(target, this)
target.scene.get.addEventHandler(jfxme.MOUSE_MOVED, resizeListener)
target.scene.get.addEventHandler(jfxme.MOUSE_PRESSED, resizeListener)
target.scene.get.addEventHandler(jfxme.MOUSE_DRAGGED, resizeListener)
// end init
def addWindowDragPoint(dragNode: Node): Unit = {
dragNode.onMousePressed = (event: MouseEvent) => {
anchorPt = new Point2D(event.screenX, event.screenY)
prevLoc = new Point2D(target.getX, target.getY)
}
dragNode.onMouseClicked = (event: MouseEvent) =>
if (event.clickCount == 2)
maximizeOrRestore()
dragNode.onMouseDragged = (event: MouseEvent) =>
if (resizeListener.cursorEvent == Cursor.DEFAULT && !maximized) {
target.x = prevLoc.x + event.screenX - anchorPt.x
target.y = prevLoc.y + event.screenY - anchorPt.y
}
target.onShown = (event: WindowEvent) => prevLoc = new Point2D(target.getX, target.getY)
}
def maximizeOrRestore() {
if (maximized) {
restoreSavedBounds()
savedBounds = null
_maximized set false
} else {
val screensForRectangle = Screen.screensForRectangle(target.getX, target.getY, target.getWidth, target.getHeight)
val screen = screensForRectangle.get(0)
val visualBounds = screen.visualBounds
savedBounds = new BoundingBox(target.getX, target.getY, target.getWidth, target.getHeight)
target.setX(visualBounds.getMinX)
target.setY(visualBounds.getMinY)
target.setWidth(visualBounds.getWidth)
target.setHeight(visualBounds.getHeight)
_maximized set true
}
}
def saveBounds() {
savedBounds = new BoundingBox(target.getX, target.getY, target.getWidth, target.getHeight)
}
def restoreSavedBounds() {
target.setX(savedBounds.getMinX)
target.setY(savedBounds.getMinY)
target.setWidth(savedBounds.getWidth)
target.setHeight(savedBounds.getHeight)
savedBounds = null
}
def maximized = _maximized.getValue
def maximizedProp = _maximized.getReadOnlyProperty
}
//formatter:off
protected class ResizeListener(
val stage: Stage, owner: UndecoratedWindowHelper) extends EventHandler[jfxme] {
var cursorEvent = Cursor.DEFAULT
val border = 4
var startX = 0d
var startY = 0d
//formatter:on
def handle(mouseEvent: jfxme) {
val mouseEventType = mouseEvent.getEventType
val scene = stage.getScene
val mouseEventX = mouseEvent.getSceneX
val mouseEventY = mouseEvent.getSceneY
val sceneWidth = scene.getWidth
val sceneHeight = scene.getHeight
if (jfxme.MOUSE_MOVED == mouseEventType) {
//@formatter:off
cursorEvent =
if (mouseEventX < border && mouseEventY < border) Cursor.NW_RESIZE
else if (mouseEventX < border && mouseEventY > sceneHeight - border) Cursor.SW_RESIZE
else if (mouseEventX > sceneWidth - border && mouseEventY < border) Cursor.NE_RESIZE
else if (mouseEventX > sceneWidth - border && mouseEventY > sceneHeight - border) Cursor.SE_RESIZE
else if (mouseEventX < border) Cursor.W_RESIZE
else if (mouseEventX > sceneWidth - border) Cursor.E_RESIZE
else if (mouseEventY < border) Cursor.N_RESIZE
else if (mouseEventY > sceneHeight - border) Cursor.S_RESIZE
else Cursor.DEFAULT
//@formatter:on
if (owner.maximized)
cursorEvent = Cursor.DEFAULT
scene.setCursor(cursorEvent)
} else if (mouseEventType == jfxme.MOUSE_PRESSED) {
startX = stage.getWidth - mouseEventX
startY = stage.getHeight - mouseEventY
} else if (mouseEventType == jfxme.MOUSE_DRAGGED) {
if (Cursor.DEFAULT != cursorEvent) {
if (Cursor.W_RESIZE != cursorEvent && Cursor.E_RESIZE != cursorEvent) {// not west or east
val minHeight = Math.max(stage.getMinHeight, border * 2)
if (Cursor.NW_RESIZE == cursorEvent || Cursor.N_RESIZE == cursorEvent || Cursor.NE_RESIZE == cursorEvent) {// NW or N or NE
val attemptedSize = stage.getY - mouseEvent.getScreenY + stage.getHeight
val actSize = Math.max(minHeight, attemptedSize)
val diff = actSize - attemptedSize
stage.setHeight(actSize)
stage.setY(mouseEvent.getScreenY - diff)
} else {// SW or S or SE
val attemptedSize = mouseEventY + startY
val actSize = Math.max(minHeight, attemptedSize)
stage.setHeight(actSize)
}
}
if (Cursor.N_RESIZE != cursorEvent && Cursor.S_RESIZE != cursorEvent) {
val minWidth = Math.max(stage.getMinWidth, border * 2)
if (Cursor.NW_RESIZE == cursorEvent || Cursor.W_RESIZE == cursorEvent || Cursor.SW_RESIZE == cursorEvent) {
val attemptedSize = stage.getX - mouseEvent.getScreenX + stage.getWidth
val actSize = Math.max(minWidth, attemptedSize)
val diff = actSize - attemptedSize
stage.setWidth(actSize)
stage.setX(mouseEvent.getScreenX - diff)
} else {
val attemptedSize = mouseEventX + startX
val actSize = Math.max(minWidth, attemptedSize)
stage.setWidth(actSize)
}
}
owner.saveBounds()
}
}
}
}