布局和鼠标事件
Layouts and Mouse events
我正在尝试使用 SWT 创建一个自定义滑块,但我得到了一个奇怪的结果,我希望我在这里遗漏了一些东西,我想要一个具有内部 canvas 的 Composite顶部、左右和底部各有 15 个边距。
类似于此:
所以我写了这样一个主要的:
Main.java
import org.eclipse.swt.SWT;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.swt.widgets.Text;
public class Main {
public static void main (String [] args) {
Display display = new Display ();
Shell shell = new Shell(display);
shell.setLayout(new GridLayout(1,false));
shell.setSize(400,400);
CustomSlider slider= new CustomSlider(shell, SWT.NONE);
shell.pack();
shell.open ();
while (!shell.isDisposed ()) {
if (!display.readAndDispatch ()) display.sleep ();
}
display.dispose ();
}
}
以及扩展 Composite 的 customSlider class:
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.PaintEvent;
import org.eclipse.swt.events.PaintListener;
import org.eclipse.swt.graphics.GC;
import org.eclipse.swt.graphics.Point;
import org.eclipse.swt.graphics.Rectangle;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Canvas;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Listener;
import org.eclipse.swt.widgets.Widget;
public class CustomSlider extends Composite {
private Canvas progressBar;
private Rectangle progressBarRect;
private int knobSize=15;
private Point thisSize;
private boolean hover=false;
public CustomSlider(Composite arg0, int arg1) {
super(arg0, arg1);
thisSize= new Point(400, 50);
setSize(400, 50);
setBackground(arg0.getDisplay().getSystemColor(SWT.COLOR_MAGENTA));
GridLayout gl= new GridLayout(1,false);
setLayout(gl);
setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true));
progressBar = new Canvas(this, SWT.NONE);
GridLayout gl2= new GridLayout(1,false);
gl2.marginBottom=knobSize;
gl2.marginTop=knobSize;
gl2.marginLeft=knobSize;
gl2.marginRight=knobSize;
gl2.marginWidth=knobSize;
gl2.marginHeight=knobSize;
progressBar.setLayout(gl2);
GridData gd=new GridData(SWT.FILL, SWT.FILL, true, true);
progressBar.setLayoutData(gd);
progressBar.addPaintListener(new PaintListener() {
@Override
public void paintControl(PaintEvent arg0) {
Canvas canvasProgress=(Canvas)arg0.widget;
progressBarRect= canvasProgress.getBounds();
GC gc=arg0.gc;
gc.setForeground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_GREEN));
if (hover){
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_RED));
}else{
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_BLUE));
}
gc.fillRectangle(progressBarRect);
gc.drawRectangle(progressBarRect);
}
});
Listener listener = new Listener() {
public void handleEvent(Event event) {
System.out.println("---------------");
System.out.println("Canvas:"+((Canvas)event.widget).getBounds());
System.out.println("Composite:"+((Canvas)event.widget).getParent().getBounds());
System.out.println("Shell:"+((Canvas)event.widget).getParent().getParent().getBounds());
Display display=((Widget)event.widget).getDisplay();
switch (event.type) {
case SWT.MouseHover:
if (progressBarRect.contains(new Point(event.x, event.y))){
hover=true;
}else{
hover=false;
}
break;
case SWT.MouseDown:
break;
case SWT.MouseMove:
if (progressBarRect.contains(new Point(event.x, event.y))){
hover=true;
}else{
hover=false;
}
break;
case SWT.MouseEnter:
if (progressBarRect.contains(new Point(event.x, event.y))){
hover=true;
}else{
hover=false;
}
break;
case SWT.MouseExit:
if (progressBarRect.contains(new Point(event.x, event.y))){
System.out.println("progressBarRect:"+progressBarRect);
System.out.println("Point"+new Point(event.x, event.y));
System.out.println("HOVER!");
hover=true;
}else{
System.out.println("OUT!");
hover=false;
}
break;
case SWT.MouseUp:
break;
}
display.asyncExec(() -> progressBar.redraw());
}
};
progressBar.addListener(SWT.MouseDown, listener);
progressBar.addListener(SWT.MouseUp, listener);
progressBar.addListener(SWT.MouseMove, listener);
progressBar.addListener(SWT.MouseExit, listener);
progressBar.addListener(SWT.MouseEnter, listener);
progressBar.addListener(SWT.MouseHover, listener);
}
public Point getThisSize() {
return thisSize;
}
public void setThisSize(Point thisSize) {
this.thisSize = thisSize;
}
}
现在,首先是结果与我想要的不符,似乎没有考虑边距,或者最好只考虑两个(顶部和左侧):
其次,这些事件导致了意想不到的结果:MouseExit 事件表明我仍然悬停在 canvas 上,即使我不在合成中! :S
你能帮我理解这是怎么回事吗?
提前谢谢你...
编辑
按照 Greg 的建议,我按如下方式编辑代码:
主要:
import org.eclipse.swt.SWT;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
public class Main {
public static void main (String [] args) {
Display display = new Display ();
Shell shell = new Shell(display);
shell.setLayout(new GridLayout(1,false));
//shell.setSize(400,400);
CustomSlider2 slider= new CustomSlider2(shell, SWT.NONE);
shell.pack();
shell.open ();
while (!shell.isDisposed ()) {
if (!display.readAndDispatch ()) display.sleep ();
}
display.dispose ();
}
}
自定义滑块变为 CustomSlider2:
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.PaintEvent;
import org.eclipse.swt.events.PaintListener;
import org.eclipse.swt.graphics.GC;
import org.eclipse.swt.graphics.Rectangle;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Canvas;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Listener;
import org.eclipse.swt.widgets.Widget;
public class CustomSlider2 extends Composite {
private Canvas progressBar;
private Rectangle progressBarRect;
private int knobSize=15;
private boolean hover=false;
public CustomSlider2(Composite arg0, int arg1) {
super(arg0, arg1);
setBackground(arg0.getDisplay().getSystemColor(SWT.COLOR_MAGENTA));
setLayout(new GridLayout(1,false));
progressBar = new Canvas(this, SWT.CENTER);
final GridData gd=new GridData(SWT.FILL, SWT.FILL, true, true);
gd.widthHint = 100;
gd.heightHint = 50;
GridLayout gl= new GridLayout();
gl.numColumns=1;
gl.marginWidth=knobSize;
gl.marginHeight=knobSize;
progressBar.setLayoutData(gd);
progressBar.setLayout(gl);
progressBar.addPaintListener(new PaintListener() {
@Override
public void paintControl(PaintEvent arg0) {
Canvas canvasProgress=(Canvas)arg0.widget;
Rectangle clientArea= canvasProgress.getBounds();
progressBarRect=new Rectangle(clientArea.x, clientArea.y, clientArea.x + clientArea.width - 1, clientArea.y + clientArea.height - 1);
GC gc=arg0.gc;
gc.setForeground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_GREEN));
if (hover){
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_RED));
}else{
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_BLUE));
}
gc.fillRectangle(progressBarRect);
gc.drawRectangle(progressBarRect);
}
});
Listener listener = new Listener() {
public void handleEvent(Event event) {
System.out.println("---------------");
System.out.println("Canvas:"+((Canvas)event.widget).getBounds());
System.out.println("Composite:"+((Canvas)event.widget).getParent().getBounds());
System.out.println("Shell:"+((Canvas)event.widget).getParent().getParent().getBounds());
Display display=((Widget)event.widget).getDisplay();
switch (event.type) {
case SWT.MouseEnter:
hover=true;
break;
case SWT.MouseExit:
hover=false;
break;
}
display.asyncExec(() -> progressBar.redraw());
}
};
progressBar.addListener(SWT.MouseExit, listener);
progressBar.addListener(SWT.MouseEnter, listener);
}
}
因此事件问题似乎已解决,但布局问题并未解决...我仍然没有 canvas...
周围的余量
使用布局时,您可以使用布局宽度和高度提示来设置控件的建议大小。所以在这里你可以使用:
final GridData gd=new GridData(SWT.FILL, SWT.FILL, true, true);
gd.widthHint = 400;
gd.heightHint = 50;
progressBar.setLayoutData(gd);
在 paintControl
中,您使用的是 getBounds
,它为您提供控件的完整尺寸,包括布局添加的边框。但是诸如 fillRectangle
之类的操作总是在排除边框的 'client area' 中绘制。所以你的油漆应该是这样的:
@Override
public void paintControl(PaintEvent event) {
Canvas canvasProgress = (Canvas)event.widget;
Rectangle clientArea = canvasProgress.getClientArea(); // Client area
.. set colors
gc.fillRectangle(clientArea.x, clientArea.y, clientArea.x + clientArea.width - 1, clientArea.y + clientArea.height - 1);
gc.drawRectangle(clientArea.x, clientArea.y, clientArea.x + clientArea.width - 1, clientArea.y + clientArea.height - 1);
}
对于鼠标移动,无需检查事件中的位置。 MouseExit 仅在鼠标离开您的控件时调用。
我正在尝试使用 SWT 创建一个自定义滑块,但我得到了一个奇怪的结果,我希望我在这里遗漏了一些东西,我想要一个具有内部 canvas 的 Composite顶部、左右和底部各有 15 个边距。 类似于此:
所以我写了这样一个主要的:
Main.java
import org.eclipse.swt.SWT;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.swt.widgets.Text;
public class Main {
public static void main (String [] args) {
Display display = new Display ();
Shell shell = new Shell(display);
shell.setLayout(new GridLayout(1,false));
shell.setSize(400,400);
CustomSlider slider= new CustomSlider(shell, SWT.NONE);
shell.pack();
shell.open ();
while (!shell.isDisposed ()) {
if (!display.readAndDispatch ()) display.sleep ();
}
display.dispose ();
}
}
以及扩展 Composite 的 customSlider class:
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.PaintEvent;
import org.eclipse.swt.events.PaintListener;
import org.eclipse.swt.graphics.GC;
import org.eclipse.swt.graphics.Point;
import org.eclipse.swt.graphics.Rectangle;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Canvas;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Listener;
import org.eclipse.swt.widgets.Widget;
public class CustomSlider extends Composite {
private Canvas progressBar;
private Rectangle progressBarRect;
private int knobSize=15;
private Point thisSize;
private boolean hover=false;
public CustomSlider(Composite arg0, int arg1) {
super(arg0, arg1);
thisSize= new Point(400, 50);
setSize(400, 50);
setBackground(arg0.getDisplay().getSystemColor(SWT.COLOR_MAGENTA));
GridLayout gl= new GridLayout(1,false);
setLayout(gl);
setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true));
progressBar = new Canvas(this, SWT.NONE);
GridLayout gl2= new GridLayout(1,false);
gl2.marginBottom=knobSize;
gl2.marginTop=knobSize;
gl2.marginLeft=knobSize;
gl2.marginRight=knobSize;
gl2.marginWidth=knobSize;
gl2.marginHeight=knobSize;
progressBar.setLayout(gl2);
GridData gd=new GridData(SWT.FILL, SWT.FILL, true, true);
progressBar.setLayoutData(gd);
progressBar.addPaintListener(new PaintListener() {
@Override
public void paintControl(PaintEvent arg0) {
Canvas canvasProgress=(Canvas)arg0.widget;
progressBarRect= canvasProgress.getBounds();
GC gc=arg0.gc;
gc.setForeground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_GREEN));
if (hover){
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_RED));
}else{
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_BLUE));
}
gc.fillRectangle(progressBarRect);
gc.drawRectangle(progressBarRect);
}
});
Listener listener = new Listener() {
public void handleEvent(Event event) {
System.out.println("---------------");
System.out.println("Canvas:"+((Canvas)event.widget).getBounds());
System.out.println("Composite:"+((Canvas)event.widget).getParent().getBounds());
System.out.println("Shell:"+((Canvas)event.widget).getParent().getParent().getBounds());
Display display=((Widget)event.widget).getDisplay();
switch (event.type) {
case SWT.MouseHover:
if (progressBarRect.contains(new Point(event.x, event.y))){
hover=true;
}else{
hover=false;
}
break;
case SWT.MouseDown:
break;
case SWT.MouseMove:
if (progressBarRect.contains(new Point(event.x, event.y))){
hover=true;
}else{
hover=false;
}
break;
case SWT.MouseEnter:
if (progressBarRect.contains(new Point(event.x, event.y))){
hover=true;
}else{
hover=false;
}
break;
case SWT.MouseExit:
if (progressBarRect.contains(new Point(event.x, event.y))){
System.out.println("progressBarRect:"+progressBarRect);
System.out.println("Point"+new Point(event.x, event.y));
System.out.println("HOVER!");
hover=true;
}else{
System.out.println("OUT!");
hover=false;
}
break;
case SWT.MouseUp:
break;
}
display.asyncExec(() -> progressBar.redraw());
}
};
progressBar.addListener(SWT.MouseDown, listener);
progressBar.addListener(SWT.MouseUp, listener);
progressBar.addListener(SWT.MouseMove, listener);
progressBar.addListener(SWT.MouseExit, listener);
progressBar.addListener(SWT.MouseEnter, listener);
progressBar.addListener(SWT.MouseHover, listener);
}
public Point getThisSize() {
return thisSize;
}
public void setThisSize(Point thisSize) {
this.thisSize = thisSize;
}
}
现在,首先是结果与我想要的不符,似乎没有考虑边距,或者最好只考虑两个(顶部和左侧):
其次,这些事件导致了意想不到的结果:MouseExit 事件表明我仍然悬停在 canvas 上,即使我不在合成中! :S
你能帮我理解这是怎么回事吗?
提前谢谢你...
编辑 按照 Greg 的建议,我按如下方式编辑代码:
主要:
import org.eclipse.swt.SWT;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
public class Main {
public static void main (String [] args) {
Display display = new Display ();
Shell shell = new Shell(display);
shell.setLayout(new GridLayout(1,false));
//shell.setSize(400,400);
CustomSlider2 slider= new CustomSlider2(shell, SWT.NONE);
shell.pack();
shell.open ();
while (!shell.isDisposed ()) {
if (!display.readAndDispatch ()) display.sleep ();
}
display.dispose ();
}
}
自定义滑块变为 CustomSlider2:
import org.eclipse.swt.SWT;
import org.eclipse.swt.events.PaintEvent;
import org.eclipse.swt.events.PaintListener;
import org.eclipse.swt.graphics.GC;
import org.eclipse.swt.graphics.Rectangle;
import org.eclipse.swt.layout.GridData;
import org.eclipse.swt.layout.GridLayout;
import org.eclipse.swt.widgets.Canvas;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Listener;
import org.eclipse.swt.widgets.Widget;
public class CustomSlider2 extends Composite {
private Canvas progressBar;
private Rectangle progressBarRect;
private int knobSize=15;
private boolean hover=false;
public CustomSlider2(Composite arg0, int arg1) {
super(arg0, arg1);
setBackground(arg0.getDisplay().getSystemColor(SWT.COLOR_MAGENTA));
setLayout(new GridLayout(1,false));
progressBar = new Canvas(this, SWT.CENTER);
final GridData gd=new GridData(SWT.FILL, SWT.FILL, true, true);
gd.widthHint = 100;
gd.heightHint = 50;
GridLayout gl= new GridLayout();
gl.numColumns=1;
gl.marginWidth=knobSize;
gl.marginHeight=knobSize;
progressBar.setLayoutData(gd);
progressBar.setLayout(gl);
progressBar.addPaintListener(new PaintListener() {
@Override
public void paintControl(PaintEvent arg0) {
Canvas canvasProgress=(Canvas)arg0.widget;
Rectangle clientArea= canvasProgress.getBounds();
progressBarRect=new Rectangle(clientArea.x, clientArea.y, clientArea.x + clientArea.width - 1, clientArea.y + clientArea.height - 1);
GC gc=arg0.gc;
gc.setForeground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_GREEN));
if (hover){
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_RED));
}else{
gc.setBackground(canvasProgress.getDisplay().getSystemColor(SWT.COLOR_BLUE));
}
gc.fillRectangle(progressBarRect);
gc.drawRectangle(progressBarRect);
}
});
Listener listener = new Listener() {
public void handleEvent(Event event) {
System.out.println("---------------");
System.out.println("Canvas:"+((Canvas)event.widget).getBounds());
System.out.println("Composite:"+((Canvas)event.widget).getParent().getBounds());
System.out.println("Shell:"+((Canvas)event.widget).getParent().getParent().getBounds());
Display display=((Widget)event.widget).getDisplay();
switch (event.type) {
case SWT.MouseEnter:
hover=true;
break;
case SWT.MouseExit:
hover=false;
break;
}
display.asyncExec(() -> progressBar.redraw());
}
};
progressBar.addListener(SWT.MouseExit, listener);
progressBar.addListener(SWT.MouseEnter, listener);
}
}
因此事件问题似乎已解决,但布局问题并未解决...我仍然没有 canvas...
周围的余量使用布局时,您可以使用布局宽度和高度提示来设置控件的建议大小。所以在这里你可以使用:
final GridData gd=new GridData(SWT.FILL, SWT.FILL, true, true);
gd.widthHint = 400;
gd.heightHint = 50;
progressBar.setLayoutData(gd);
在 paintControl
中,您使用的是 getBounds
,它为您提供控件的完整尺寸,包括布局添加的边框。但是诸如 fillRectangle
之类的操作总是在排除边框的 'client area' 中绘制。所以你的油漆应该是这样的:
@Override
public void paintControl(PaintEvent event) {
Canvas canvasProgress = (Canvas)event.widget;
Rectangle clientArea = canvasProgress.getClientArea(); // Client area
.. set colors
gc.fillRectangle(clientArea.x, clientArea.y, clientArea.x + clientArea.width - 1, clientArea.y + clientArea.height - 1);
gc.drawRectangle(clientArea.x, clientArea.y, clientArea.x + clientArea.width - 1, clientArea.y + clientArea.height - 1);
}
对于鼠标移动,无需检查事件中的位置。 MouseExit 仅在鼠标离开您的控件时调用。