SWT Canvas位置

时间:2009-05-01 20:57:58

标签: resize canvas swt

我正在使用最新的稳定版SWT。

我有一个包含ToolBar,ScrolledComposite和Canvas的Shell。 Canvas被设置为ScrolledComposite的内容(例如scrolledComposite.setContent(canvas))。 Canvas的创建具有永不改变的特定大小(例如,400 x 400)。而ScrolledComposite不断增长或缩小以填充可用的父shell的客户区。

我有一个附加到父Shell的调整大小侦听器,它尝试执行以下操作:a)如上所述增长ScrolledComposite,b)在ScrolledComposite中水平和垂直居中画布(请参阅下面的示例代码)。 / p>

这在Mac OS X上完全正常,但是在Windows上,会调用resize事件并正确计算新位置,但最终Canvas会快速恢复到0,0。另外一小部分信息是,如果你不断调整窗口的大小,你可以看到画布闪烁,看起来好像是在正确的位置画出来的。

_shell.addListener (SWT.Resize, new Listener () {
    public void handleEvent (Event e)
    {
        int toolBarOffset = _toolBar.getSize().y;

        Rectangle clientArea = _shell.getClientArea();

        _scrollComposite.setBounds(0, toolBarOffset, clientArea.width, clientArea.height - toolBarOffset);

        Point canvasSize = _canvas.getSize();

        int canvasX = Math.max(0, (clientArea.width / 2) - (canvasSize.x / 2));
        int canvasY = Math.max(0, ((clientArea.height - toolBarOffset) / 2) - (canvasSize.y / 2));

        _canvas.setLocation(canvasX, canvasY);
    }
});

1 个答案:

答案 0 :(得分:1)

不确定您是否已经弄明白,但问题是ScrolledComposite在调整大小时始终将内容设置为0/0。我不确定为什么你的方法适用于OS X,我没有测试我的例子,因为我这里没有Mac。

解决方案是使用填充复合材料,该填充复合材料始终至少与ScrolledComposite中的客户区一样大。在该填充器中,您可以正确地居中Canvas。

我做了一个小例子,作为额外的奖励,如果SC的客户区小于画布,它也会使画布居中(因为我首先认为这是你的问题:))

您可能需要进行一些小的调整,我猜OS X上有一些关于该代码的故障......

package test;

import org.eclipse.swt.SWT;
import org.eclipse.swt.custom.ScrolledComposite;
import org.eclipse.swt.events.ControlAdapter;
import org.eclipse.swt.events.ControlEvent;
import org.eclipse.swt.events.PaintEvent;
import org.eclipse.swt.events.PaintListener;
import org.eclipse.swt.graphics.Point;
import org.eclipse.swt.graphics.Rectangle;
import org.eclipse.swt.layout.FillLayout;
import org.eclipse.swt.widgets.Canvas;
import org.eclipse.swt.widgets.Composite;

import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;

public class Test {

    public static void main(final String[] args) {
        final Display display = new Display();
        final Shell shell = new Shell(display);

        shell.setLayout( new FillLayout() );

        final ScrolledComposite sComp = new ScrolledComposite( shell, SWT.H_SCROLL | SWT.V_SCROLL );
        final Composite fillComp = new Composite( sComp, SWT.NONE );

        sComp.setLayout( new FillLayout() );

        final Canvas c = new Canvas( fillComp, SWT.DOUBLE_BUFFERED );
        c.addPaintListener( new PaintListener() {
            public void paintControl(PaintEvent e) {
                Point p = c.getSize();
                e.gc.setBackground( display.getSystemColor( SWT.COLOR_RED ));
                e.gc.fillRectangle( 0, 0, p.x, p.y );

                e.gc.setBackground( display.getSystemColor( SWT.COLOR_BLUE ));
                e.gc.fillRectangle( p.x / 2 - 10, p.y / 2 - 10, 20, 20 );
            }
        });

        c.setSize( 400, 400 );
        sComp.setContent( fillComp );

        sComp.addControlListener( new ControlAdapter() {
            @Override
            public void controlResized(ControlEvent e) {
                Rectangle clientArea = sComp.getClientArea();

                Point cSize = c.getSize();
                int fillX = Math.max( clientArea.width, cSize.x );
                int fillY = Math.max( clientArea.height, cSize.y );
                fillComp.setSize( fillX, fillY );

                int cX = ( clientArea.width - cSize.x ) / 2;
                int cY = ( clientArea.height - cSize.y ) / 2;

                sComp.setOrigin( -cX, -cY );

                int locX = Math.max( cX, 0 );
                int locY = Math.max( cY, 0 );
                c.setLocation( locX, locY );
            }
        });

        shell.open();
        shell.pack();
        while (!shell.isDisposed()) {
            if (!display.readAndDispatch()) {
                display.sleep();
            }
        }   
    }
}