我有一个3D游戏,每次移动光标时,我都希望它重置到中间。问题是robot.mouseMove()调用MouseEvent(它确实有意义)并重置位置,因此我无法旋转。
谢谢!
答案 0 :(得分:3)
因为Robot
正在生成本机事件,所以事件将(最终)通过EDT处理事件队列。
这意味着如果你尝试做类似的事情......
removeMouseListener(...);
Robot.mouseMove(...);
addMouseListener(...);
它基本上没有任何效果,因为在事件处理的同一周期中发生了鼠标监听器的移除和附加,这意味着机器人已经引发的鼠标事件将不会被处理(或将出现后面的队列)...
相反,你需要提出一些你可以检测到的标志,然后忽略下一个传入的事件......
if (!ignoreMouseMove) {
ignoreMouseMove = true;
// Do your normal processing...
robot.mouseMove(...);
} else {
ignoreMouseMove = false;
}
下面的基本示例检测鼠标移动中心的距离,并更新一个简单的position
变量(基本上充当罗盘点)。这有助于说明运动,但更重要的是,我们正在打破事件周期......
import java.awt.AWTException;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.FontMetrics;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Robot;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
public class TestMouseMove {
public static void main(String[] args) {
new TestMouseMove();
}
public TestMouseMove() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
ex.printStackTrace();
}
JFrame frame = new JFrame("Testing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(new TestPane());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public class TestPane extends JPanel {
private Robot bot;
private int position = 0;
public TestPane() {
try {
bot = new Robot();
MouseAdapter ma = new MouseAdapter() {
boolean ignoreMouseMove = false;
@Override
public void mouseMoved(MouseEvent e) {
if (!ignoreMouseMove) {
ignoreMouseMove = true;
int x = getLocationOnScreen().x + (getWidth() / 2);
int y = getLocationOnScreen().y + (getHeight() / 2);
int distanceFromCenter = e.getPoint().x - (getWidth() / 2);
position += distanceFromCenter;
if (position < 0) {
position = 360 - position;
} else if (position > 360) {
position -= 360;
}
repaint();
bot.mouseMove(x, y);
} else {
ignoreMouseMove = false;
}
}
@Override
public void mouseClicked(MouseEvent e) {
System.exit(0);
}
};
addMouseListener(ma);
addMouseMotionListener(ma);
} catch (AWTException ex) {
ex.printStackTrace();;
}
}
@Override
public Dimension getPreferredSize() {
return new Dimension(200, 200);
}
@Override
protected void paintComponent(Graphics g) {
super.paintComponent(g);
Graphics2D g2d = (Graphics2D) g.create();
FontMetrics fm = g2d.getFontMetrics();
int x = getWidth() / 2;
int y = getHeight() / 2;
int amount = position;
while (x > 0) {
if (amount == position) {
g2d.drawLine(x, y, x, y - 40);
} else {
g2d.drawLine(x, y, x, y - 20);
}
String text = Integer.toString(amount);
g2d.drawString(text, x - (fm.stringWidth(text) / 2), y + fm.getHeight());
x -= 20;
amount--;
if (amount < 0) {
amount = 360 + amount;
}
}
amount = position + 1;
x = (getWidth() / 2) + 20;
while (x < getWidth()) {
g2d.drawLine(x, y, x, y - 20);
if (position > 360) {
position = 360 - position;
}
String text = Integer.toString(amount);
g2d.drawString(text, x - (fm.stringWidth(text) / 2), y + fm.getHeight());
x += 20;
amount++;
}
g2d.dispose();
}
}
}
答案 1 :(得分:2)
我更喜欢以下代码:
component.removeMouseListener(...);
Robot.doSomething();
component.addMouseListener(...);
而不是设置标志。使用此方法,管理侦听器的代码位于代码中的单个位置。
如果您使用标志,则需要
所以你最终在你班上的多个地方都有代码。
编辑:
关于机器人被添加到事件队列末尾的好处。那么,我将包装将MouseListener添加回SwingUtilities.invokeLater()
中的组件的代码答案 2 :(得分:0)
更一致的答案不是忽略鼠标事件,而是处理它,然后减去使用机器人移动的确切数量。这是jist:
public void mouseMoved(MouseEvent me) {
mouseDiffX += me.getX() - mouseX;
mouseDiffY += me.getY() - mouseY;
mouseX = me.getX();
mouseY = me.getY();
absoluteX = me.getLocationOnScreen().x;
absoluteY = me.getLocationOnScreen().y;
}
public void moveRobot() {
int newX = canvas.getWidth() / 2 + canvas.getLocationOnScreen().x;
int newY = canvas.getHeight() / 2 + canvas.getLocationOnScreen().y;
robotMoveX = newX - absoluteX;
robotMoveY = newY - absoluteY;
robotMoved = true;
}
public void update() {
if (robotMoved) {
mouseDiffX -= robotMoveX;
mouseDiffY -= robotMoveY;
robotMoved = false;
}
finalX += mouseDiffX;
finalY += mouseDiffY;
}