在侦听器方法中使用GC

时间:2015-10-05 13:41:10

标签: java events view swt mouselistener

我使用this教程创建了自定义视图。我能够在paintControl()方法中绘制我想要的所有内容。我想要做的是,除了我的MouseMoveListener之外还有一个PaintListener,它应该在鼠标悬停时绘制一些附加内容:

@Override
public void createPartControl(Composite parent) {

    canvas = new Canvas(parent, SWT.NONE);
    canvas.addPaintListener(new PaintListener() {
        public void paintControl(PaintEvent e) {
            // draw something
        }
    });

    canvas.addMouseMoveListener(new MouseMoveListener() {
        @Override
        public void mouseMove(MouseEvent e) {
            system.out.println(e.x + ", e.y");
            // draw something
        }
    });
}

是否可以在GC方法中使用mouseMove()来绘制除paintControl()中创建的现有形状之外的形状?我尝试将GC设置为实例变量,并将其设置为paintControl()以在mouseOver()中重复使用,但不幸的是,null就是ls -lrt

2 个答案:

答案 0 :(得分:1)

作为@Kuba sais,您无法在PaintEvent之外的GC上绘图。

要重绘画布,请从鼠标事件中调用其redraw方法,如下所示:

canvas.addMouseMoveListener( new MouseMoveListener() {
    @Override
    public void mouseMove( MouseEvent event ) {
      canvas.redraw();
    }
} );

答案 1 :(得分:1)

要绘制任何控件(可绘制,扩展Drawable),您需要创建GC的实例并使用它来绘制您需要的任何内容。不要忘记处置GC实例。

我修改了Snippet10以显示如何执行此操作:

private static boolean mouseDown = false;

public static void main(String[] args) {
    final Display display = new Display();
    final Shell shell = new Shell(display);
    shell.setText("Advanced Graphics");
    FontData fd = shell.getFont().getFontData()[0];
    final Font font = new Font(display, fd.getName(), 60, SWT.BOLD | SWT.ITALIC);
    final Image image = new Image(display, 640, 480);
    final Rectangle rect = image.getBounds();
    GC gc = new GC(image);
    gc.setBackground(display.getSystemColor(SWT.COLOR_RED));
    gc.fillOval(rect.x, rect.y, rect.width, rect.height);
    gc.dispose();
    shell.addListener(SWT.Paint, new Listener() {
        public void handleEvent(Event event) {
            GC gc = event.gc;               
            Transform tr = new Transform(display);
            tr.translate(50, 120);
            tr.rotate(-30);
            gc.drawImage(image, 0, 0, rect.width, rect.height, 0, 0, rect.width / 2, rect.height / 2);
            gc.setAlpha(100);
            gc.setTransform(tr);
            Path path = new Path(display);
            path.addString("SWT", 0, 0, font);
            gc.setBackground(display.getSystemColor(SWT.COLOR_GREEN));
            gc.setForeground(display.getSystemColor(SWT.COLOR_BLUE));
            gc.fillPath(path);
            gc.drawPath(path);
            tr.dispose();
            path.dispose();
        }           
    });
    shell.addMouseListener(new MouseListener() {
        @Override
        public void mouseUp(MouseEvent e) {
            mouseDown = false;
        }

        @Override
        public void mouseDown(MouseEvent e) {
            mouseDown = true;
        }

        @Override
        public void mouseDoubleClick(MouseEvent e) {}
    });
    shell.addMouseMoveListener(new MouseMoveListener() {
        @Override
        public void mouseMove(MouseEvent e) {
            if (!mouseDown)
                return;

            GC gc = new GC(shell);
            gc.drawPoint(e.x, e.y);
            gc.dispose();
        }
    });

    shell.setSize(shell.computeSize(rect.width / 2, rect.height / 2));
    shell.open();
    while (!shell.isDisposed()) {
        if (!display.readAndDispatch())
            display.sleep();
    }
    image.dispose();
    font.dispose();
    display.dispose();
}

注意:无论何时在控件上触发绘制事件,您的自定义绘图都将丢失。所以也许最好是在图像上画画。然后在控件上调用redraw,它始终假定它必须绘制背景,然后在背景上绘制该图像。这样,在每次重绘时,您都可以保持鼠标绘图。