本文不是对SWT做深入的研究,只是一些小技巧的介绍,通过构建一个自定义的分隔窗体来熟悉一些SWT的Widget和Layout。

使用过Swing的朋友们应该知道,在Swing里有一种分隔窗体叫JSplitPane,可以提供左右或者上下两个分隔窗体,并且可以通过中间的拖动条的拖拽来实现两个窗体大小的调整,拖动条还提供直接最大化某个窗体或者直接平分两个窗体的按钮。而在SWT里对应的有SashForm这个组件,同样可以实现左右拖拽,但并没有提供那些快捷键按钮,想通过继承SashForm来实现这个功能并不好做,那么这里我使用的是SWT的Sash和FormLayout的配合来实现一个类似于Swing JSplitPane的分割窗体。

先来看看我们最终实现的效果图:

   

好,那么我们就一起开始来设计这个分割窗体。首先做一下简单的分析:

1. 我们需要两个Composite来存放我们不同窗体的组件;

2. 我们需要一个可以拖拽的拖杠,使用SWT提供的Sash组件来实现;

3. 一个存放按钮的的Composite,并且在拖拽过程中要保持与拖杠同步;

4. 提供左右分隔和上下分隔两种模式。

一切从拖拽开始,那么什么是Sash呢?SWT的文档上是这么说的:

Instances of the receiver represent a selectable user interface object that allows the user to drag a rubber banded outline of the sash within the parent control.

Styles:
HORIZONTAL, VERTICAL, SMOOTH
Events:
Selection

就是说Sash能够在其Parent容器里进行拖拽并产生条纹线的回馈,当然还包括Selection事件。那么我们就可以这么设计,我们接收Sash的Selection事件,并相应的调整左右或者上下窗体的大小即可实现我们的需求。那么如何计算两个窗体的大小呢?这里最好就是从布局组件入手,使用SWT强大的FormLayout可以灵活的实现丰富的布局效果。

为什么使用FormLayout呢?它又有哪些功能?这里我只做简单的介绍,详细的大家可以参阅SWT的文档或其它网络文档。使用FormLayout和对应的FormData可以实现其布局中组件间的相对定位以及相对的偏移量,相对距离及偏移量是通过定义FormData的top/bottom/left/right决定的,其定义使用的类是FormAttachment。有了FormLayout和对应的FormData这就为我们Sash拖动后窗体的布局计算提供了基础,那么如何设置我们SplitPane中各个组件的FormData呢?如下图分析:

① 左边窗体,与父容器的左边距离为0,右边参照SashBar
② 右边窗体,与父容器的右边距离为0,左边参照SashBar
③ Sash,即我们的拖杠,左边参照SashBar,并且向左偏移5像素(这样Sash才会叠在SashBar上面),右边无参照,宽度定死为3
④ SashBar,即存放按钮的Composite,左右均无参照(左右的参照距离应该是在接收Sash的Selection事件时动态决定),与父容器的顶部距离为总高度的50%,并且向上偏移半个SashBar的高度,这样就能保证SashBar在垂直上居中

定义好布局后接下来要做的就是对Sash的Selection事件反馈作用到SashBar的FormData上,因为SashBar是其它所有组件的参照中心,大家都是围绕它来布局的。

 1 sash.addSelectionListener(new SelectionListener() {
 2     public void widgetDefaultSelected(SelectionEvent e) {
 3         widgetSelected(e);
 4     }
 5 
 6     public void widgetSelected(SelectionEvent e) {
 7         FormData data = (FormData) sashBar.getLayoutData();
 8         Rectangle rect = sashBar.getParent().getBounds();
 9         
10         data.left = null;
11         if(rect.width - e.x <= 7) {
12             data.right = new FormAttachment(100, 0);
13             return;
14         }
15         data.right = new FormAttachment(e.x, rect.width, 7);
16         
17         sashBar.setLayoutData(data);
18         sashBar.getParent().layout();
19     }
20 });

每次触发Sash的Selection事件,将SashBar的FormData取出,并获得父窗体的矩形区域,然后重新设置SashBar的FormData的右部,FormAttachment的分子设为鼠标的x轴坐标,分母设置为父容器矩形区域的宽度,并且向右偏移7个像素,因为e.x得到的是Sash的x轴坐标,而SashBar的右边应该比Sash的右边更靠右一些(Sash在SashBar的中间)。还有我们应该保证SashBar不被拖出右边区域,所以当判断到rect.width - e.x小于7时,就应该将SashBar的右边设为顶住父容器右边的极限值,不能再继续移动了。

这样,我们的SplitPane就基本完成了,剩下的就是如何提供两个窗体内容的部分。我采用的是定义一个ContentProvider接口,有两个抽象方法分别由实现类去构建两个窗体内容,返回值为初始化时两个窗口的比率,例如左窗口返回4,有窗口返回5,那么做窗口占4/9,右窗口占5/9:

1 public static interface ISashPanelContentProvider {
2     
3     public int positiveContent(Composite parent);
4     
5     public int negativeContent(Composite parent);
6     
7 }

通过上面的方法相信你应该知道基本的设计概念了,是不是很简单呢?只要我们发挥想象力,还有很多有用的扩展组件是可以制作出来的。那么SplitPane剩下的就是完善诸如上下分层还是左右分层、初始化构造这些周边工作,在我的源码工程里已经完成了这些功能,这篇随笔我就写到这里,点击这里下载源码。

转载于:https://www.cnblogs.com/louischx/archive/2008/10/25/1319501.html

基于SWT的Sash和FormLayout的自定义分隔窗体相关推荐

  1. swt能单独在linux运行么,java – 在Mac上运行基于SWT的跨平台jar

    我一直在开发一个基于SWT的项目,该项目旨在部署为Java Web Start,从而在多个平台上使用. 到目前为止,我已经设法解决了SWT所依赖的特定于系统的图书馆出现的出口问题(见相关的thread ...

  2. R语言构建logistic回归模型并评估模型:构建基于混淆矩阵计算分类评估指标的自定义函数、阳性样本比例(垃圾邮件比例)变化对应的分类器性能的变化、基于数据阳性样本比例选择合适的分类评估指标

    R语言构建logistic回归模型并评估模型:构建基于混淆矩阵计算分类评估指标的自定义函数.阳性样本比例(垃圾邮件比例)变化对应的分类器性能的变化.基于数据阳性样本比例选择合适的分类评估指标 目录

  3. sklearn基于make_scorer函数为Logistic模型构建自定义损失函数并可视化误差图(lambda selection)和系数图(trace plot)+代码实战

    sklearn基于make_scorer函数为Logistic模型构建自定义损失函数并可视化误差图(lambda selection)和系数图(trace plot)+代码实战 # 自定义损失函数 i ...

  4. sklearn基于make_scorer函数为Logistic模型构建自定义损失函数+代码实战(二元交叉熵损失 binary cross-entropy loss)

    sklearn基于make_scorer函数为Logistic模型构建自定义损失函数+代码实战(二元交叉熵损失 binary cross-entropy loss) # 广义线性模型中的各种连接函数: ...

  5. 小程序richtext_用于基于SWT的应用程序的RichText编辑器组件

    小程序richtext 本文将完成使用SWT实现我们自己的RichText编辑器组件的任务. 在为我的一位客户开发基于桌面的应用程序时,我遇到了这样的可视化组件的需求,并希望添加一项功能,以允许用户使 ...

  6. 用于基于SWT的应用程序的RichText编辑器组件

    本文将完成使用SWT实现我们自己的RichText编辑器组件的任务. 在为我的一位客户开发基于桌面的应用程序时,我遇到了这样一个可视化组件的需求,并希望添加一项功能,以允许用户使用粗体,斜体,删除线等 ...

  7. 最简单的基于DirectShow的示例:视频播放器自定义版

    ===================================================== 最简单的基于DirectShow的示例文章列表: 最简单的基于DirectShow的示例:视 ...

  8. 一些基于SWT的项目

    在 clientjava.com的blog上,列举了几个 基于SWT的项目,大部分是开源的(或者免费的)的SWT构件.这些包括: 1. SWTForm.这是一个将JGoodies的Form移植到SWT ...

  9. 【物体检测快速入门系列 | 01 】基于Tensorflow2.x Object Detection API构建自定义物体检测器

    这是机器未来的第1篇文章 原文首发地址:https://blog.csdn.net/RobotFutures/article/details/124745966 CSDN话题挑战赛第1期 活动详情地址 ...

  10. 基于windows10下使用bat脚本设置自定义开机启动项

    基于windows10下使用bat脚本设置自定义开机启动项 一.新建一个txt文件,然后写入内容,再重新命名修改文件成.bat脚本.博主举例新建一个rothschildlhl.txt空文本,然后复制下 ...

最新文章

  1. ACM MM:一种基于情感脑电信号时-频-空特征的3D密集连接网络
  2. python中chr函数的用法_python中hex,oct,chr,ord函数讲解
  3. 你可能没看懂Supercell的新游戏
  4. 罚款200元的交通违法行为
  5. Python 爬虫框架 - PySpider
  6. IJKMediaFramework框架的集成和使用实例一枚
  7. Atitit 学习记忆理论 教育理论 教学培训 目录 1. 视觉优先理论 1 2. 心理学的“认知负荷理论 2 2.1. 双重编码理论 2 2.2. 艺术化 原理动画 3 2.3. 艾宾浩斯遗忘曲
  8. mysql的索引(一)
  9. iphone 控制 android手机,苹果手机如何远程控制安卓手机
  10. Ubuntu18中,使用Python的matplotlib库设置simhei.ttf中文字体并显示
  11. 4G标准LTE FDD与LTE TDD的不同
  12. [统计学理论基础] 置信区间
  13. 构建地形系统(翻译)1
  14. 猫推荐算法大赛Top 9团队
  15. ijkplayer播放器崩溃问题
  16. php 微信支付 ca证书,微信企业付款 CA证书出错,请登录微信支付商户平台下载证书...
  17. 云服务器修改dns服务器为阿里云公共dns服务器
  18. 梅科尔工作室-江凌宇-鸿蒙笔记1
  19. centOS6使用NAT方式联网
  20. 360安全备份 android,手机里的数据怎么备份 360安全卫士备份手机数据图解

热门文章

  1. SQL数据库“单个用户”不能访问,设置为多个用户的解决方法
  2. 「LibreOJ β Round #4」多项式 (广义欧拉数论定理)
  3. UITableView的cell重用优化
  4. 获取winform应用程序集信息
  5. iOS 提交app到iTunes Connect预览截图截取方法及尺寸大小
  6. HDU 6611 K Subsequence(Dijkstra优化费用流 模板)题解
  7. Google Chrome不支持ClickOnce部署
  8. 不要奢望.NET能够跨平台
  9. 毕业后,她用1年时间拿下了30W年薪的阿里数据分析岗
  10. reticulate: R interface to Python