Browse Source
# Conflicts: # designer_base/src/com/fr/design/locale/designer.properties # designer_base/src/com/fr/design/locale/designer_en_US.properties # designer_base/src/com/fr/design/locale/designer_ja_JP.properties # designer_base/src/com/fr/design/locale/designer_ko_KR.properties # designer_base/src/com/fr/design/locale/designer_zh_CN.properties # designer_chart/src/com/fr/design/chart/axis/MinMaxValuePane.java # designer_chart/src/com/fr/design/mainframe/chart/gui/style/axis/ChartValuePane.java # designer_form/src/com/fr/design/designer/beans/adapters/layout/FRAbsoluteLayoutAdapter.java # designer_form/src/com/fr/design/designer/beans/adapters/layout/FRFitLayoutAdapter.java # designer_form/src/com/fr/design/designer/beans/location/AccessDirection.java # designer_form/src/com/fr/design/designer/creator/XWAbsoluteLayout.javamaster
paul
8 years ago
80 changed files with 9722 additions and 6510 deletions
@ -0,0 +1,73 @@
|
||||
package com.fr.aspectj.designer; |
||||
|
||||
/** |
||||
* 记录模板过程 |
||||
* Created by plough on 2017/3/3. |
||||
*/ |
||||
|
||||
import com.fr.design.mainframe.templateinfo.TemplateInfoCollector; |
||||
import com.fr.grid.Grid; |
||||
import org.aspectj.lang.reflect.SourceLocation; |
||||
|
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.Date; |
||||
|
||||
public aspect TemplateProcessTracker { |
||||
//声明一个pointcut,匹配你需要的方法 |
||||
pointcut onMouseClicked(MouseEvent e) : |
||||
execution(* mouseClicked(MouseEvent)) && args(e); |
||||
pointcut onMousePressed(MouseEvent e) : |
||||
execution(* mousePressed(MouseEvent)) && args(e); |
||||
pointcut onMouseReleased(MouseEvent e) : |
||||
execution(* mouseReleased(MouseEvent)) && args(e); |
||||
pointcut onActionPerformed(ActionEvent e) : |
||||
execution(* actionPerformed(ActionEvent)) && args(e); |
||||
pointcut onSetValueAt(Object v, int r, int c) : |
||||
execution(* setValueAt(java.lang.Object, int, int)) && args(v, r, c); |
||||
pointcut onSetValue4EditingElement(Grid g, Object v) : |
||||
call(* setValue4EditingElement(java.lang.Object)) && target(g) && args(v); |
||||
|
||||
//before表示之前的意思 |
||||
//这整个表示在MouseAdapter的public void mouseXXX(MouseEvent)方法调用之前,你想要执行的代码 |
||||
before(MouseEvent e) : onMouseClicked(e) || onMousePressed(e) || onMouseReleased(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation();//切面对应的代码位置 |
||||
|
||||
// String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
} |
||||
//同上 |
||||
before(ActionEvent e) : onActionPerformed(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
// !within(LogHandlerBar) 没用, 手动过滤 |
||||
if (e.getSource().toString().contains("javax.swing.Timer")) { |
||||
return; |
||||
} |
||||
|
||||
//String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
|
||||
} |
||||
//同上 |
||||
before(Object v, int r, int c) : onSetValueAt(v, r, c) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
|
||||
//String log = String.format("%s:\n%s\nset value: %s at (%d, %d)\n\n", new Date(), sl, v, r, c); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
|
||||
} |
||||
//同上 |
||||
before(Grid g, Object v) : onSetValue4EditingElement(g, v) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
|
||||
// String v = "test"; |
||||
//String log = String.format("%s:\n%s\nset value: %s at %s\n\n", new Date(), sl, v, g.getEditingCellElement()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
} |
||||
|
||||
|
||||
} |
@ -0,0 +1,74 @@
|
||||
package com.fr.design.mainframe.templateinfo; |
||||
|
||||
import com.fr.base.parameter.ParameterUI; |
||||
import com.fr.main.impl.WorkBook; |
||||
import com.fr.report.cellcase.CellCase; |
||||
import com.fr.report.poly.PolyWorkSheet; |
||||
import com.fr.report.worksheet.WorkSheet; |
||||
|
||||
import java.util.Iterator; |
||||
|
||||
/** |
||||
* Created by plough on 2017/3/17. |
||||
*/ |
||||
public class JWorkBookProcessInfo extends TemplateProcessInfo<WorkBook> { |
||||
|
||||
public JWorkBookProcessInfo(WorkBook wb) { |
||||
super(wb); |
||||
} |
||||
|
||||
// 获取模板类型
|
||||
public int getReportType() { |
||||
return template.isElementCaseBook() ? 0 : 1; |
||||
} |
||||
|
||||
// 获取模板格子数
|
||||
public int getCellCount() { |
||||
int cellCount = 0; |
||||
if (template.isElementCaseBook()) { // 如果是普通报表
|
||||
for (int i = 0; i < template.getReportCount(); i++) { |
||||
WorkSheet r = (WorkSheet) template.getReport(i); |
||||
CellCase cc = r.getBlock().getCellCase(); |
||||
for (int j = 0; j < cc.getRowCount(); j++) { |
||||
Iterator iter = cc.getRow(j); |
||||
while (iter.hasNext()) { |
||||
cellCount ++; |
||||
iter.next(); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
return cellCount; |
||||
} |
||||
// 获取模板悬浮元素个数
|
||||
public int getFloatCount() { |
||||
int chartCount = 0; |
||||
if (template.isElementCaseBook()) { // 如果是普通报表
|
||||
for (int i = 0; i < template.getReportCount(); i++) { |
||||
WorkSheet r = (WorkSheet) template.getReport(i); |
||||
Iterator fiter = r.getBlock().floatIterator(); |
||||
while (fiter.hasNext()) { |
||||
chartCount ++; |
||||
fiter.next(); |
||||
} |
||||
} |
||||
} |
||||
return chartCount; |
||||
} |
||||
// 获取模板聚合块个数
|
||||
public int getBlockCount() { |
||||
int blockCount = 0; |
||||
if (!template.isElementCaseBook()) { // 如果是聚合报表
|
||||
for (int i = 0; i < template.getReportCount(); i++) { |
||||
PolyWorkSheet r = (PolyWorkSheet) template.getReport(i); |
||||
blockCount += r.getBlockCount(); |
||||
} |
||||
} |
||||
return blockCount; |
||||
} |
||||
// 获取模板控件数
|
||||
public int getWidgetCount() { |
||||
ParameterUI pui = template.getReportParameterAttr().getParameterUI(); |
||||
return pui == null ? 0 : (pui.getAllWidgets().length - 1); |
||||
} |
||||
} |
@ -0,0 +1,70 @@
|
||||
|
||||
apply plugin: 'java' |
||||
tasks.withType(JavaCompile){ |
||||
options.encoding = 'UTF-8' |
||||
} |
||||
//指定构建的jdk版本 |
||||
sourceCompatibility=1.7 |
||||
//指定生成jar包版本 |
||||
version='8.0' |
||||
//生成jar包重命名 |
||||
jar{ |
||||
baseName='fr-designer-core' |
||||
} |
||||
|
||||
|
||||
def srcDir="." |
||||
|
||||
//指定源码路径 |
||||
sourceSets{ |
||||
main{ |
||||
java{ |
||||
srcDirs=["${srcDir}/src"] |
||||
} |
||||
} |
||||
} |
||||
//获取什么分支名 |
||||
FileTree files =fileTree(dir:'./',include:'build.*.gradle') |
||||
def buildDir=files[0].path.substring(0,files[0].path.lastIndexOf ('\\')) |
||||
buildDir=buildDir.substring(0,buildDir.lastIndexOf ('\\')) |
||||
def branchName=buildDir.substring(buildDir.lastIndexOf ('\\')+1) |
||||
|
||||
//声明外部依赖 |
||||
dependencies{ |
||||
|
||||
compile fileTree(dir:"../../../finereport-lib-stable/${branchName}",include:'**/*.jar') |
||||
compile fileTree(dir:'../../../',include:"finereport-*-stable/${branchName}/**/build/libs/*.jar") |
||||
|
||||
testCompile 'junit:junit:4.12' |
||||
} |
||||
//复制非.java文件到classes文件夹下参与打包 |
||||
task copyFile(type:Copy,dependsOn:compileJava){ |
||||
copy{ |
||||
from ("${srcDir}/src"){ |
||||
exclude '**/.setting/**','.classpath','.project','**/*.java','**/*.db','**/*.g','**/package.html' |
||||
} |
||||
into 'build/classes/main' |
||||
} |
||||
|
||||
} |
||||
|
||||
|
||||
//压缩项目中的js文件 |
||||
task compressJS{ |
||||
ant.taskdef(name:'yuicompress',classname:'com.yahoo.platform.yui.compressor.YUICompressTask'){ |
||||
classpath { |
||||
fileset(dir:'../../../finereport-lib4build-stable',includes:'**/*.jar') |
||||
} |
||||
} |
||||
ant.yuicompress(linebreak:"500",warn:"false", munge:"yes",preserveallsemicolons:"false",charset:"utf-8",encoding:"utf-8",outputfolder:'build/classes/main'){ |
||||
fileset (dir:"${srcDir}/src"){ |
||||
include (name:'**/*.js') |
||||
include (name:'**/*.css') |
||||
} |
||||
|
||||
} |
||||
} |
||||
jar.dependsOn compressJS |
||||
|
||||
|
||||
|
@ -0,0 +1,60 @@
|
||||
package com.fr.aspectj.designerbase; |
||||
|
||||
/** |
||||
* 记录模板过程 |
||||
* Created by plough on 2017/3/3. |
||||
*/ |
||||
|
||||
import com.fr.design.mainframe.templateinfo.TemplateInfoCollector; |
||||
import org.aspectj.lang.reflect.SourceLocation; |
||||
|
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.Date; |
||||
|
||||
public aspect TemplateProcessTracker { |
||||
//声明一个pointcut,匹配你需要的方法 |
||||
pointcut onMouseClicked(MouseEvent e) : |
||||
execution(* mouseClicked(MouseEvent)) && args(e); |
||||
pointcut onMousePressed(MouseEvent e) : |
||||
execution(* mousePressed(MouseEvent)) && args(e); |
||||
pointcut onMouseReleased(MouseEvent e) : |
||||
execution(* mouseReleased(MouseEvent)) && args(e); |
||||
pointcut onActionPerformed(ActionEvent e) : |
||||
execution(* actionPerformed(ActionEvent)) && args(e); |
||||
pointcut onSetValueAt(Object v, int r, int c) : |
||||
execution(* setValueAt(java.lang.Object, int, int)) && args(v, r, c); |
||||
|
||||
//before表示之前的意思 |
||||
//这整个表示在MouseAdapter的public void mouseXXX(MouseEvent)方法调用之前,你想要执行的代码 |
||||
before(MouseEvent e) : onMouseClicked(e) || onMousePressed(e) || onMouseReleased(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation();//切面对应的代码位置 |
||||
|
||||
//String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
} |
||||
//同上 |
||||
before(ActionEvent e) : onActionPerformed(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
// !within(LogHandlerBar) 没用, 手动过滤 |
||||
if (e.getSource().toString().contains("javax.swing.Timer")) { |
||||
return; |
||||
} |
||||
|
||||
//String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
|
||||
} |
||||
//同上 |
||||
before(Object v, int r, int c) : onSetValueAt(v, r, c) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
|
||||
//String log = String.format("%s:\n%s\nset value: %s at (%d, %d)\n\n", new Date(), sl, v, r, c); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
} |
||||
|
||||
|
||||
} |
@ -1,37 +1,35 @@
|
||||
/* |
||||
* Copyright(c) 2001-2010, FineReport Inc, All Rights Reserved. |
||||
*/ |
||||
package com.fr.design.actions.edit; |
||||
|
||||
import java.awt.event.KeyEvent; |
||||
|
||||
import javax.swing.KeyStroke; |
||||
|
||||
import com.fr.base.BaseUtils; |
||||
import com.fr.design.actions.TemplateComponentAction; |
||||
import com.fr.design.designer.TargetComponent; |
||||
import com.fr.general.Inter; |
||||
|
||||
/** |
||||
* Copy. |
||||
*/ |
||||
public class CopyAction extends TemplateComponentAction { |
||||
public CopyAction(TargetComponent t) { |
||||
super(t); |
||||
|
||||
this.setName(Inter.getLocText("M_Edit-Copy")); |
||||
this.setMnemonic('C'); |
||||
this.setSmallIcon(BaseUtils.readIcon("/com/fr/design/images/m_edit/copy.png")); |
||||
this.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_C, KeyEvent.CTRL_MASK)); |
||||
} |
||||
|
||||
@Override |
||||
public boolean executeActionReturnUndoRecordNeeded() { |
||||
TargetComponent tc = getEditingComponent(); |
||||
if (tc != null) { |
||||
tc.copy(); |
||||
} |
||||
|
||||
return false; |
||||
} |
||||
/* |
||||
* Copyright(c) 2001-2010, FineReport Inc, All Rights Reserved. |
||||
*/ |
||||
package com.fr.design.actions.edit; |
||||
|
||||
import com.fr.base.BaseUtils; |
||||
import com.fr.design.actions.TemplateComponentAction; |
||||
import com.fr.design.designer.TargetComponent; |
||||
import com.fr.general.Inter; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.event.KeyEvent; |
||||
|
||||
/** |
||||
* Copy. |
||||
*/ |
||||
public class CopyAction extends TemplateComponentAction { |
||||
public CopyAction(TargetComponent t) { |
||||
super(t); |
||||
|
||||
this.setName(Inter.getLocText("M_Edit-Copy")); |
||||
this.setMnemonic('C'); |
||||
this.setSmallIcon(BaseUtils.readIcon("/com/fr/design/images/m_edit/copy.png")); |
||||
this.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_C, KeyEvent.CTRL_MASK)); |
||||
} |
||||
|
||||
@Override |
||||
public boolean executeActionReturnUndoRecordNeeded() { |
||||
TargetComponent tc = getEditingComponent(); |
||||
if (tc != null) { |
||||
tc.copy(); |
||||
} |
||||
return false; |
||||
} |
||||
} |
File diff suppressed because it is too large
Load Diff
@ -1 +1,66 @@
|
||||
package com.fr.design.mainframe;
import javax.swing.JComponent;
import com.fr.form.FormElementCaseContainerProvider;
/**
* Author : Shockway
* Date: 13-7-15
* Time: 上午10:28
*/
public interface BaseJForm extends JTemplateProvider{
public static final String XML_TAG = "JForm";
public static final int FORM_TAB = 0;
public static final int ELEMENTCASE_TAB = 1;
public static final int ELEMENTCASE_CHANGE_TAB = 2;
/**
* 刷新所有控件
*/
public void refreshAllNameWidgets();
/**
* 刷新参数
*/
public void populateParameter();
/**
* 刷新选中的控件
*/
public void refreshSelectedWidget();
/**
* 获取当前的Target
*/
public Object getTarget();
/**
* 执行撤销
*
* @param o 之前保存的状态
*/
public void applyUndoState4Form(BaseUndoState o);
/**
* 获取当前编辑的组件
*/
public JComponent getEditingPane();
/**
* 只在Form和ElementCase之间切换
* @param index 切换位置
*/
public void tabChanged(int index);
/**
* 在Form和ElementCase, 以及ElementCase和ElementCase之间切换
* @param index 切换位置
* @param ecContainer ElementCase所在container
*/
public void tabChanged(int index, FormElementCaseContainerProvider ecContainer);
} |
||||
package com.fr.design.mainframe; |
||||
|
||||
import javax.swing.JComponent; |
||||
|
||||
import com.fr.form.FormElementCaseContainerProvider; |
||||
|
||||
|
||||
/** |
||||
* Author : Shockway |
||||
* Date: 13-7-15 |
||||
* Time: 上午10:28 |
||||
*/ |
||||
public interface BaseJForm extends JTemplateProvider { |
||||
|
||||
String XML_TAG = "JForm"; |
||||
int FORM_TAB = 0; |
||||
int ELEMENTCASE_TAB = 1; |
||||
int ELEMENTCASE_CHANGE_TAB = 2; |
||||
|
||||
/** |
||||
* 刷新所有控件 |
||||
*/ |
||||
void refreshAllNameWidgets(); |
||||
|
||||
/** |
||||
* 刷新参数 |
||||
*/ |
||||
void populateParameter(); |
||||
|
||||
/** |
||||
* 刷新选中的控件 |
||||
*/ |
||||
void refreshSelectedWidget(); |
||||
|
||||
/** |
||||
* 获取当前的Target |
||||
*/ |
||||
Object getTarget(); |
||||
|
||||
/** |
||||
* 执行撤销 |
||||
* |
||||
* @param o 之前保存的状态 |
||||
*/ |
||||
void applyUndoState4Form(BaseUndoState o); |
||||
|
||||
/** |
||||
* 获取当前编辑的组件 |
||||
*/ |
||||
JComponent getEditingPane(); |
||||
|
||||
/** |
||||
* 只在Form和ElementCase之间切换 |
||||
* |
||||
* @param index 切换位置 |
||||
*/ |
||||
void tabChanged(int index); |
||||
|
||||
/** |
||||
* 在Form和ElementCase, 以及ElementCase和ElementCase之间切换 |
||||
* |
||||
* @param index 切换位置 |
||||
* @param ecContainer ElementCase所在container |
||||
*/ |
||||
void tabChanged(int index, FormElementCaseContainerProvider ecContainer); |
||||
} |
@ -0,0 +1,332 @@
|
||||
package com.fr.design.mainframe.templateinfo; |
||||
|
||||
import com.fr.base.FRContext; |
||||
import com.fr.base.io.IOFile; |
||||
import com.fr.design.DesignerEnvManager; |
||||
import com.fr.design.mainframe.DesignerContext; |
||||
import com.fr.design.mainframe.JTemplate; |
||||
import com.fr.env.RemoteEnv; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.general.FRLogger; |
||||
import com.fr.general.GeneralUtils; |
||||
import com.fr.general.SiteCenter; |
||||
import com.fr.general.http.HttpClient; |
||||
import com.fr.stable.*; |
||||
import org.json.JSONObject; |
||||
|
||||
import java.io.*; |
||||
import java.text.SimpleDateFormat; |
||||
import java.util.*; |
||||
|
||||
/** |
||||
* 做模板的过程和耗时收集,辅助类 |
||||
* Created by plough on 2017/2/21. |
||||
*/ |
||||
public class TemplateInfoCollector<T extends IOFile> implements Serializable { |
||||
private static final String FILE_NAME = "tplInfo.ser"; |
||||
private static TemplateInfoCollector instance; |
||||
private HashMap<String, HashMap<String, Object>> templateInfoList; |
||||
private String designerOpenDate; //设计器最近一次打开日期
|
||||
private static final int VALID_CELL_COUNT = 5; // 有效报表模板的格子数
|
||||
private static final int VALID_WIDGET_COUNT = 5; // 有效报表模板的控件数
|
||||
private static final int COMPLETE_DAY_COUNT = 15; // 判断模板是否完成的天数
|
||||
private static final int ONE_THOUSAND = 1000; |
||||
static final long serialVersionUID = 2007L; |
||||
|
||||
@SuppressWarnings("unchecked") |
||||
private TemplateInfoCollector() { |
||||
templateInfoList = new HashMap<>(); |
||||
setDesignerOpenDate(); |
||||
} |
||||
|
||||
/** |
||||
* 把设计器最近打开日期设定为当前日期 |
||||
*/ |
||||
private void setDesignerOpenDate() { |
||||
designerOpenDate = new SimpleDateFormat("yyyy-MM-dd").format(Calendar.getInstance().getTime()); |
||||
} |
||||
|
||||
/** |
||||
* 判断今天是否第一次打开设计器 |
||||
*/ |
||||
private boolean designerOpenFirstTime() { |
||||
String today = new SimpleDateFormat("yyyy-MM-dd").format(Calendar.getInstance().getTime()); |
||||
return !ComparatorUtils.equals(today, designerOpenDate); |
||||
} |
||||
|
||||
/** |
||||
* 获取缓存文件存放路径 |
||||
*/ |
||||
private static File getInfoFile() { |
||||
return new File(StableUtils.pathJoin(ProductConstants.getEnvHome(), FILE_NAME)); |
||||
} |
||||
|
||||
public static TemplateInfoCollector getInstance() { |
||||
if (instance == null) { |
||||
// 先尝试从文件读取
|
||||
try{ |
||||
ObjectInputStream is = new ObjectInputStream(new FileInputStream(getInfoFile())); |
||||
instance = (TemplateInfoCollector) is.readObject(); |
||||
} catch (FileNotFoundException ex) { |
||||
// 如果之前没有存储过,则创建新对象
|
||||
instance = new TemplateInfoCollector(); |
||||
} catch (InvalidClassException ex) { |
||||
// 如果 TemplateInfoCollecor 类结构有改动,则放弃之前收集的数据(下次保存时覆盖)
|
||||
// 这种情况主要在开发、测试过程中遇到,正式上线后不应该出现
|
||||
FRLogger.getLogger().info(ex.getMessage()); |
||||
FRLogger.getLogger().info("use a new instance"); |
||||
instance = new TemplateInfoCollector(); |
||||
} |
||||
catch (Exception ex) { |
||||
FRLogger.getLogger().error(ex.getMessage(), ex); |
||||
} |
||||
} |
||||
return instance; |
||||
} |
||||
|
||||
private boolean shouldCollectInfo() { |
||||
if (FRContext.getCurrentEnv() instanceof RemoteEnv) { // 远程设计不收集数据
|
||||
return false; |
||||
} |
||||
return DesignerEnvManager.getEnvManager().isJoinProductImprove() && FRContext.isChineseEnv(); |
||||
} |
||||
|
||||
public void appendProcess(String log) { |
||||
if (!shouldCollectInfo()) { |
||||
return; |
||||
} |
||||
// 获取当前编辑的模板
|
||||
JTemplate jt = DesignerContext.getDesignerFrame().getSelectedJTemplate(); |
||||
// 追加过程记录
|
||||
jt.appendProcess(log); |
||||
} |
||||
|
||||
/** |
||||
* 加载已经存储的模板过程 |
||||
*/ |
||||
@SuppressWarnings("unchecked") |
||||
public String loadProcess(T t) { |
||||
HashMap<String, Object> processMap = (HashMap<String, Object>) templateInfoList.get(t.getTemplateID()).get("processMap"); |
||||
return (String)processMap.get("process"); |
||||
} |
||||
|
||||
/** |
||||
* 根据模板ID是否在收集列表中,判断是否需要收集当前模板的信息 |
||||
*/ |
||||
public boolean inList(T t) { |
||||
return templateInfoList.containsKey(t.getTemplateID()); |
||||
} |
||||
|
||||
/** |
||||
* 将包含所有信息的对象保存到文件 |
||||
*/ |
||||
private void saveInfo() { |
||||
try { |
||||
ObjectOutputStream os = new ObjectOutputStream(new FileOutputStream(getInfoFile())); |
||||
String log = ""; |
||||
int count = 1; |
||||
for (String key : templateInfoList.keySet()) { |
||||
String createTime = ((HashMap)templateInfoList.get(key).get("consumingMap")).get("create_time").toString(); |
||||
log += (count + ". id: " + key + " " + createTime + "\n" + templateInfoList.get(key).toString() + "\n"); |
||||
count ++; |
||||
} |
||||
FRLogger.getLogger().info("writing tplInfo: \n" + log); |
||||
os.writeObject(instance); |
||||
os.close(); |
||||
} catch (Exception ex) { |
||||
FRLogger.getLogger().error(ex.getMessage()); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 更新 day_count:打开设计器却未编辑模板的连续日子 |
||||
*/ |
||||
private void addDayCount() { |
||||
if (designerOpenFirstTime()) { |
||||
for (String key : templateInfoList.keySet()) { |
||||
HashMap<String, Object> templateInfo = templateInfoList.get(key); |
||||
int dayCount = (int)templateInfo.get("day_count") + 1; |
||||
templateInfo.put("day_count", dayCount); |
||||
} |
||||
setDesignerOpenDate(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 收集模板信息。如果之前没有记录,则新增;如果已有记录,则更新。 |
||||
* 同时将最新数据保存到文件中。 |
||||
*/ |
||||
@SuppressWarnings("unchecked") |
||||
public void collectInfo(T t, JTemplate jt, long openTime, long saveTime) { |
||||
if (!shouldCollectInfo()) { |
||||
return; |
||||
} |
||||
|
||||
HashMap<String, Object> templateInfo; |
||||
|
||||
long timeConsume = ((saveTime - openTime) / ONE_THOUSAND); // 制作模板耗时(单位:s)
|
||||
String templateID = t.getTemplateID(); |
||||
|
||||
if (inList(t)) { // 已有记录
|
||||
templateInfo = templateInfoList.get(t.getTemplateID()); |
||||
// 更新 conusmingMap
|
||||
HashMap<String, Object> consumingMap = (HashMap<String, Object>) templateInfo.get("consumingMap"); |
||||
timeConsume += (long)consumingMap.get("time_consume"); // 加上之前的累计编辑时间
|
||||
consumingMap.put("time_consume", timeConsume); |
||||
} |
||||
else { // 新增
|
||||
templateInfo = new HashMap<>(); |
||||
templateInfo.put("consumingMap", getNewConsumingMap(templateID, openTime, timeConsume)); |
||||
} |
||||
|
||||
// 直接覆盖 processMap
|
||||
templateInfo.put("processMap", getProcessMap(templateID, jt)); |
||||
|
||||
// 保存模板时,让 day_count 归零
|
||||
templateInfo.put("day_count", 0); |
||||
|
||||
|
||||
templateInfoList.put(templateID, templateInfo); |
||||
|
||||
saveInfo(); // 每次更新之后,都同步到暂存文件中
|
||||
} |
||||
|
||||
private HashMap<String, Object> getNewConsumingMap(String templateID, long openTime, long timeConsume) { |
||||
HashMap<String, Object> consumingMap = new HashMap<>(); |
||||
|
||||
String username = DesignerEnvManager.getEnvManager().getBBSName(); |
||||
String uuid = DesignerEnvManager.getEnvManager().getUUID(); |
||||
String activitykey = DesignerEnvManager.getEnvManager().getActivationKey(); |
||||
String createTime = new SimpleDateFormat("yyyy-MM-dd HH:mm").format(Calendar.getInstance().getTime()); |
||||
String jarTime = GeneralUtils.readBuildNO(); |
||||
String version = ProductConstants.VERSION; |
||||
consumingMap.put("username", username); |
||||
consumingMap.put("uuid", uuid); |
||||
consumingMap.put("activitykey", activitykey); |
||||
consumingMap.put("templateID", templateID); |
||||
consumingMap.put("create_time", createTime); |
||||
consumingMap.put("time_consume", timeConsume); |
||||
consumingMap.put("jar_time", jarTime); |
||||
consumingMap.put("version", version); |
||||
|
||||
return consumingMap; |
||||
} |
||||
|
||||
private HashMap<String, Object> getProcessMap(String templateID, JTemplate jt) { |
||||
HashMap<String, Object> processMap = new HashMap<>(); |
||||
|
||||
processMap.put("templateID", templateID); |
||||
processMap.put("process", jt.getProcess()); |
||||
|
||||
TemplateProcessInfo info = jt.getProcessInfo(); |
||||
processMap.put("report_type", info.getReportType()); |
||||
processMap.put("cell_count", info.getCellCount()); |
||||
processMap.put("float_count", info.getFloatCount()); |
||||
processMap.put("block_count", info.getBlockCount()); |
||||
processMap.put("widget_count", info.getWidgetCount()); |
||||
|
||||
return processMap; |
||||
} |
||||
|
||||
/** |
||||
* 发送本地模板信息到服务器 |
||||
*/ |
||||
public void sendTemplateInfo() { |
||||
addDayCount(); |
||||
String consumingUrl = SiteCenter.getInstance().acquireUrlByKind("tempinfo.consuming") + "/single"; |
||||
String processUrl = SiteCenter.getInstance().acquireUrlByKind("tempinfo.process") + "/single"; |
||||
ArrayList<HashMap<String, String>> completeTemplatesInfo = getCompleteTemplatesInfo(); |
||||
for (HashMap<String, String> templateInfo : completeTemplatesInfo) { |
||||
String jsonConsumingMap = templateInfo.get("jsonConsumingMap"); |
||||
String jsonProcessMap = templateInfo.get("jsonProcessMap"); |
||||
if (sendSingleTemplateInfo(consumingUrl, jsonConsumingMap) && sendSingleTemplateInfo(processUrl, jsonProcessMap)) { |
||||
// 清空记录
|
||||
FRLogger.getLogger().info("successfully send " + templateInfo.get("templateID")); |
||||
removeFromTemplateInfoList(templateInfo.get("templateID")); |
||||
} else { |
||||
FRLogger.getLogger().info("send template info failed, will try next time, " + templateInfo.get("templateID")); |
||||
} |
||||
} |
||||
saveInfo(); |
||||
} |
||||
|
||||
private boolean sendSingleTemplateInfo(String url, String content) { |
||||
HashMap<String, String> para = new HashMap<>(); |
||||
String date = new SimpleDateFormat("yyyy-MM-dd").format(Calendar.getInstance().getTime()); |
||||
para.put("token", CodeUtils.md5Encode(date, "", "MD5")); |
||||
para.put("content", content); |
||||
HttpClient httpClient = new HttpClient(url, para, true); |
||||
httpClient.setTimeout(5000); |
||||
httpClient.asGet(); |
||||
|
||||
if (!httpClient.isServerAlive()) { |
||||
return false; |
||||
} |
||||
|
||||
String res = httpClient.getResponseText(); |
||||
boolean success = ComparatorUtils.equals(new JSONObject(res).get("status"), "success"); |
||||
return success; |
||||
} |
||||
|
||||
/** |
||||
* 返回已完成的模板信息 |
||||
*/ |
||||
@SuppressWarnings("unchecked") |
||||
private ArrayList<HashMap<String, String>> getCompleteTemplatesInfo() { |
||||
ArrayList<HashMap<String, String>> completeTemplatesInfo = new ArrayList<>(); |
||||
ArrayList<String> testTemplateKeys = new ArrayList<>(); // 保存测试模板的key
|
||||
for (String key : templateInfoList.keySet()) { |
||||
HashMap<String, Object> templateInfo = templateInfoList.get(key); |
||||
if ((int)templateInfo.get("day_count") <= COMPLETE_DAY_COUNT) { // 未完成模板
|
||||
continue; |
||||
} |
||||
if (isTestTemplate(templateInfo)) { |
||||
testTemplateKeys.add(key); |
||||
continue; |
||||
} |
||||
HashMap<String, Object> consumingMap = (HashMap<String, Object>) templateInfo.get("consumingMap"); |
||||
HashMap<String, Object> processMap = (HashMap<String, Object>) templateInfo.get("processMap"); |
||||
String jsonConsumingMap = new JSONObject(consumingMap).toString(); |
||||
String jsonProcessMap = new JSONObject(processMap).toString(); |
||||
HashMap<String, String> jsonTemplateInfo = new HashMap<>(); |
||||
jsonTemplateInfo.put("jsonConsumingMap", jsonConsumingMap); |
||||
jsonTemplateInfo.put("jsonProcessMap", jsonProcessMap); |
||||
jsonTemplateInfo.put("templateID", key); |
||||
completeTemplatesInfo.add(jsonTemplateInfo); |
||||
} |
||||
// 删除测试模板
|
||||
for (String key : testTemplateKeys) { |
||||
removeFromTemplateInfoList(key); |
||||
} |
||||
return completeTemplatesInfo; |
||||
} |
||||
|
||||
private void removeFromTemplateInfoList(String key) { |
||||
templateInfoList.remove(key); |
||||
FRLogger.getLogger().info(key + " is removed..."); |
||||
} |
||||
|
||||
@SuppressWarnings("unchecked") |
||||
private boolean isTestTemplate(HashMap<String, Object> templateInfo) { |
||||
HashMap<String, Object> processMap = (HashMap<String, Object>) templateInfo.get("processMap"); |
||||
int reportType = (int)processMap.get("report_type"); |
||||
int cellCount = (int)processMap.get("cell_count"); |
||||
int floatCount = (int)processMap.get("float_count"); |
||||
int blockCount = (int)processMap.get("block_count"); |
||||
int widgetCount = (int)processMap.get("widget_count"); |
||||
boolean isTestTemplate = false; |
||||
if (reportType == 0) { // 普通报表
|
||||
isTestTemplate = cellCount <= VALID_CELL_COUNT && floatCount <= 1 && widgetCount <= VALID_WIDGET_COUNT; |
||||
} else if (reportType == 1) { // 聚合报表
|
||||
isTestTemplate = blockCount <= 1 && widgetCount <= VALID_WIDGET_COUNT; |
||||
} else { // 表单(reportType == 2)
|
||||
isTestTemplate = widgetCount <= 1; |
||||
} |
||||
return isTestTemplate; |
||||
} |
||||
|
||||
public static void main(String[] args) { |
||||
TemplateInfoCollector tic = TemplateInfoCollector.getInstance(); |
||||
tic.sendTemplateInfo(); |
||||
} |
||||
} |
@ -0,0 +1,26 @@
|
||||
package com.fr.design.mainframe.templateinfo; |
||||
|
||||
import com.fr.base.io.IOFile; |
||||
|
||||
/** |
||||
* Created by plough on 2017/3/17. |
||||
*/ |
||||
public abstract class TemplateProcessInfo<T extends IOFile> { |
||||
|
||||
protected T template; |
||||
|
||||
public TemplateProcessInfo(T template) { |
||||
this.template = template; |
||||
} |
||||
|
||||
// 获取模板类型。0 代表普通报表,1 代表聚合报表,2 代表表单
|
||||
public abstract int getReportType(); |
||||
// 获取模板格子数
|
||||
public abstract int getCellCount(); |
||||
// 获取模板悬浮元素个数
|
||||
public abstract int getFloatCount(); |
||||
// 获取模板聚合块个数
|
||||
public abstract int getBlockCount(); |
||||
// 获取模板控件数
|
||||
public abstract int getWidgetCount(); |
||||
} |
@ -1 +1,72 @@
|
||||
package com.fr.design.parameter;
import com.fr.base.Parameter;
import com.fr.base.parameter.ParameterUI;
import com.fr.design.mainframe.AuthorityEditPane;
import javax.swing.*;
import java.awt.*;
/**
* 参数设计界面接口
*/
public interface ParameterDesignerProvider {
public void addListener(ParaDefinitePane paraDefinitePane);
public Component createWrapper();
public void setDesignHeight(int height);
public Dimension getDesignSize();
public Dimension getPreferredSize();
public void populate(ParameterUI p);
public void refreshAllNameWidgets();
public void refresh4TableData(String oldName, String newName);
public void refreshParameter(ParaDefinitePane paraDefinitePane);
public boolean isWithQueryButton();
public java.util.List<String> getAllXCreatorNameList();
public boolean isWithoutParaXCreator(Parameter[] ps);
public boolean isBlank();
public ParameterUI getParaTarget();
public boolean addingParameter2Editor(Parameter parameter, int index);
public boolean addingParameter2EditorWithQueryButton(Parameter parameter, int index);
public void addingAllParameter2Editor(Parameter[] parameterArray, int currentIndex);
public JPanel[] toolbarPanes4Form();
public JComponent[] toolBarButton4Form();
public void initBeforeUpEdit();
public void populateParameterPropertyPane(ParaDefinitePane p);
public void initWidgetToolbarPane();
public AuthorityEditPane getAuthorityEditPane();
public JPanel getEastUpPane();
public JPanel getEastDownPane();
public boolean isSupportAuthority();
public void removeSelection();
public ParameterBridge getParaComponent();
} |
||||
package com.fr.design.parameter; |
||||
|
||||
import com.fr.base.Parameter; |
||||
import com.fr.base.parameter.ParameterUI; |
||||
import com.fr.design.mainframe.AuthorityEditPane; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.*; |
||||
|
||||
|
||||
/** |
||||
* 参数设计界面接口 |
||||
*/ |
||||
public interface ParameterDesignerProvider { |
||||
|
||||
void addListener(ParaDefinitePane paraDefinitePane); |
||||
|
||||
Component createWrapper(); |
||||
|
||||
void setDesignHeight(int height); |
||||
|
||||
Dimension getDesignSize(); |
||||
|
||||
Dimension getPreferredSize(); |
||||
|
||||
void populate(ParameterUI p); |
||||
|
||||
void refreshAllNameWidgets(); |
||||
|
||||
void refresh4TableData(String oldName, String newName); |
||||
|
||||
void refreshParameter(ParaDefinitePane paraDefinitePane); |
||||
|
||||
boolean isWithQueryButton(); |
||||
|
||||
java.util.List<String> getAllXCreatorNameList(); |
||||
|
||||
boolean isWithoutParaXCreator(Parameter[] ps); |
||||
|
||||
boolean isBlank(); |
||||
|
||||
ParameterUI getParaTarget(); |
||||
|
||||
boolean addingParameter2Editor(Parameter parameter, int index); |
||||
|
||||
boolean addingParameter2EditorWithQueryButton(Parameter parameter, int index); |
||||
|
||||
void addingAllParameter2Editor(Parameter[] parameterArray, int currentIndex); |
||||
|
||||
JPanel[] toolbarPanes4Form(); |
||||
|
||||
JComponent[] toolBarButton4Form(); |
||||
|
||||
void initBeforeUpEdit(); |
||||
|
||||
void populateParameterPropertyPane(ParaDefinitePane p); |
||||
|
||||
void initWidgetToolbarPane(); |
||||
|
||||
AuthorityEditPane getAuthorityEditPane(); |
||||
|
||||
JPanel getEastUpPane(); |
||||
|
||||
JPanel getEastDownPane(); |
||||
|
||||
boolean isSupportAuthority(); |
||||
|
||||
void removeSelection(); |
||||
|
||||
ParameterBridge getParaComponent(); |
||||
|
||||
} |
@ -1,151 +1,154 @@
|
||||
package com.fr.design.utils; |
||||
|
||||
import com.fr.general.ComparatorUtils; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.*; |
||||
import java.awt.geom.Rectangle2D; |
||||
import java.util.ArrayList; |
||||
|
||||
/** |
||||
* 工具类,提供常用的工具方法 |
||||
*/ |
||||
public class ComponentUtils { |
||||
|
||||
public static boolean isComponentVisible(Component comp) { |
||||
if (!comp.isVisible() && !isRootComponent(comp)) { |
||||
return false; |
||||
} |
||||
Component parent = comp.getParent(); |
||||
|
||||
return parent == null || isComponentVisible(parent); |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 获取component所在的容器的绝对位置 |
||||
*/ |
||||
public static Rectangle getRelativeBounds(Component component) { |
||||
Rectangle bounds = new Rectangle(0, 0, component.getWidth(), component.getHeight()); |
||||
Container parent = component.getParent(); |
||||
|
||||
while (parent != null) { |
||||
bounds.x += component.getX(); |
||||
bounds.y += component.getY(); |
||||
component = parent; |
||||
parent = component.getParent(); |
||||
} |
||||
|
||||
return bounds; |
||||
} |
||||
|
||||
/** |
||||
* 恢复双缓冲状态,dbcomponents保存着初始状态为启动双缓冲的组件 |
||||
*/ |
||||
public static void resetBuffer(ArrayList<JComponent> dbcomponents) { |
||||
for (JComponent jcomponent : dbcomponents) { |
||||
jcomponent.setDoubleBuffered(true); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 禁止双缓冲状态,并将初始状态为启动双缓冲的组件保存到dbcomponents中 |
||||
*/ |
||||
public static void disableBuffer(Component comp, ArrayList<JComponent> dbcomponents) { |
||||
if ((comp instanceof JComponent) && comp.isDoubleBuffered()) { |
||||
JComponent jcomponent = (JComponent) comp; |
||||
|
||||
dbcomponents.add(jcomponent); |
||||
jcomponent.setDoubleBuffered(false); |
||||
} |
||||
|
||||
if (comp instanceof Container) { |
||||
Container container = (Container) comp; |
||||
int count = container.getComponentCount(); |
||||
|
||||
if (count > 0) { |
||||
for (int i = 0; i < count; i++) { |
||||
Component component = container.getComponent(i); |
||||
|
||||
disableBuffer(component, dbcomponents); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
public static int indexOfComponent(Container container, Component target) { |
||||
int count = container.getComponentCount(); |
||||
|
||||
for (int i = 0; i < count; i++) { |
||||
Component child = container.getComponent(i); |
||||
|
||||
if (child == target) { |
||||
return i; |
||||
} |
||||
} |
||||
|
||||
return -1; |
||||
} |
||||
|
||||
/** |
||||
* 计算组件root相对于其顶层容器的可见区域 |
||||
*/ |
||||
public static Rectangle computeVisibleRectRel2Root(Component root) { |
||||
Container container = findAncestorScrollPane(root); |
||||
|
||||
if (container == null) { |
||||
return getRelativeBounds(root); |
||||
} else { |
||||
// 如果是JScrollPane的子组件,需要计算其viewport与改组件的交叉的可见区域
|
||||
return getBoundsRel2Parent(root, container); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 计算组件root相对于其顶层容器的可见区域 |
||||
*/ |
||||
public static Rectangle computeVisibleRect(JComponent root) { |
||||
Rectangle root_bounds = ComponentUtils.getRelativeBounds(root); |
||||
Rectangle rect = computeVisibleRectRel2Root(root); |
||||
rect.x -= root_bounds.x; |
||||
rect.y -= root_bounds.y; |
||||
|
||||
return rect; |
||||
} |
||||
|
||||
private static Rectangle getBoundsRel2Parent(Component child, Container parent) { |
||||
Rectangle cRect = getRelativeBounds(child); |
||||
Rectangle pRect = getRelativeBounds(parent); |
||||
Rectangle bounds = new Rectangle(); |
||||
Rectangle2D.intersect(cRect, pRect, bounds); |
||||
|
||||
return bounds; |
||||
} |
||||
|
||||
public static Container findAncestorScrollPane(Component p) { |
||||
if ((p == null) || !(p instanceof Container)) { |
||||
return null; |
||||
} |
||||
|
||||
Container c = p.getParent(); |
||||
|
||||
return findAncestorScrollPane(c); |
||||
} |
||||
|
||||
public static boolean isRootComponent(Component root) { |
||||
Container parent = root.getParent(); |
||||
return parent == null; |
||||
} |
||||
|
||||
public static boolean isChildOf(Component component, Class parent) { |
||||
Container container = component.getParent(); |
||||
if (container != null) { |
||||
if (ComparatorUtils.equals(container.getClass(), parent)) { |
||||
return true; |
||||
} else { |
||||
return isChildOf(container, parent); |
||||
} |
||||
} |
||||
return false; |
||||
} |
||||
package com.fr.design.utils; |
||||
|
||||
import com.fr.general.ComparatorUtils; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.*; |
||||
import java.awt.geom.Rectangle2D; |
||||
import java.util.ArrayList; |
||||
|
||||
/** |
||||
* 工具类,提供常用的工具方法 |
||||
*/ |
||||
public class ComponentUtils { |
||||
|
||||
private ComponentUtils() { |
||||
} |
||||
|
||||
public static boolean isComponentVisible(Component comp) { |
||||
if (!comp.isVisible() && !isRootComponent(comp)) { |
||||
return false; |
||||
} |
||||
Component parent = comp.getParent(); |
||||
|
||||
return parent == null || isComponentVisible(parent); |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 获取component所在的容器的绝对位置 |
||||
*/ |
||||
public static Rectangle getRelativeBounds(Component component) { |
||||
Rectangle bounds = new Rectangle(0, 0, component.getWidth(), component.getHeight()); |
||||
Container parent = component.getParent(); |
||||
|
||||
while (parent != null) { |
||||
bounds.x += component.getX(); |
||||
bounds.y += component.getY(); |
||||
component = parent; |
||||
parent = component.getParent(); |
||||
} |
||||
|
||||
return bounds; |
||||
} |
||||
|
||||
/** |
||||
* 恢复双缓冲状态,dbcomponents保存着初始状态为启动双缓冲的组件 |
||||
*/ |
||||
public static void resetBuffer(ArrayList<JComponent> dbcomponents) { |
||||
for (JComponent jcomponent : dbcomponents) { |
||||
jcomponent.setDoubleBuffered(true); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 禁止双缓冲状态,并将初始状态为启动双缓冲的组件保存到dbcomponents中 |
||||
*/ |
||||
public static void disableBuffer(Component comp, ArrayList<JComponent> dbcomponents) { |
||||
if ((comp instanceof JComponent) && comp.isDoubleBuffered()) { |
||||
JComponent jcomponent = (JComponent) comp; |
||||
|
||||
dbcomponents.add(jcomponent); |
||||
jcomponent.setDoubleBuffered(false); |
||||
} |
||||
|
||||
if (comp instanceof Container) { |
||||
Container container = (Container) comp; |
||||
int count = container.getComponentCount(); |
||||
|
||||
if (count > 0) { |
||||
for (int i = 0; i < count; i++) { |
||||
Component component = container.getComponent(i); |
||||
|
||||
disableBuffer(component, dbcomponents); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
public static int indexOfComponent(Container container, Component target) { |
||||
int count = container.getComponentCount(); |
||||
|
||||
for (int i = 0; i < count; i++) { |
||||
Component child = container.getComponent(i); |
||||
|
||||
if (child.equals(target)) { |
||||
return i; |
||||
} |
||||
} |
||||
|
||||
return -1; |
||||
} |
||||
|
||||
/** |
||||
* 计算组件root相对于其顶层容器的可见区域 |
||||
*/ |
||||
public static Rectangle computeVisibleRectRel2Root(Component root) { |
||||
Container container = findAncestorScrollPane(root); |
||||
|
||||
if (container == null) { |
||||
return getRelativeBounds(root); |
||||
} else { |
||||
// 如果是JScrollPane的子组件,需要计算其viewport与改组件的交叉的可见区域
|
||||
return getBoundsRel2Parent(root, container); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 计算组件root相对于其顶层容器的可见区域 |
||||
*/ |
||||
public static Rectangle computeVisibleRect(JComponent root) { |
||||
Rectangle rootBounds = ComponentUtils.getRelativeBounds(root); |
||||
Rectangle rect = computeVisibleRectRel2Root(root); |
||||
rect.x -= rootBounds.x; |
||||
rect.y -= rootBounds.y; |
||||
|
||||
return rect; |
||||
} |
||||
|
||||
private static Rectangle getBoundsRel2Parent(Component child, Container parent) { |
||||
Rectangle cRect = getRelativeBounds(child); |
||||
Rectangle pRect = getRelativeBounds(parent); |
||||
Rectangle bounds = new Rectangle(); |
||||
Rectangle2D.intersect(cRect, pRect, bounds); |
||||
|
||||
return bounds; |
||||
} |
||||
|
||||
public static Container findAncestorScrollPane(Component p) { |
||||
if ((p == null) || !(p instanceof Container)) { |
||||
return null; |
||||
} |
||||
|
||||
Container c = p.getParent(); |
||||
|
||||
return findAncestorScrollPane(c); |
||||
} |
||||
|
||||
public static boolean isRootComponent(Component root) { |
||||
Container parent = root.getParent(); |
||||
return parent == null; |
||||
} |
||||
|
||||
public static boolean isChildOf(Component component, Class parent) { |
||||
Container container = component.getParent(); |
||||
if (container != null) { |
||||
if (ComparatorUtils.equals(container.getClass(), parent)) { |
||||
return true; |
||||
} else { |
||||
return isChildOf(container, parent); |
||||
} |
||||
} |
||||
return false; |
||||
} |
||||
} |
@ -0,0 +1,85 @@
|
||||
|
||||
apply plugin: 'java' |
||||
tasks.withType(JavaCompile){ |
||||
options.encoding = 'UTF-8' |
||||
} |
||||
|
||||
buildscript { |
||||
repositories { |
||||
maven { |
||||
url "http://www.eveoh.nl/files/maven2" |
||||
} |
||||
} |
||||
|
||||
dependencies { |
||||
classpath "nl.eveoh:gradle-aspectj:1.2" |
||||
} |
||||
} |
||||
|
||||
ext.aspectjVersion = '1.7.4' |
||||
apply plugin: 'aspectj' |
||||
|
||||
repositories { |
||||
mavenCentral() |
||||
} |
||||
//指定构建的jdk版本 |
||||
sourceCompatibility=1.7 |
||||
//指定生成jar包的版本 |
||||
version='8.0' |
||||
|
||||
def srcDir="." |
||||
|
||||
//对生成的jar包进行重命名 |
||||
|
||||
jar{ |
||||
baseName='fr-designer-chart' |
||||
} |
||||
|
||||
sourceSets{ |
||||
main{ |
||||
java{ |
||||
srcDirs=["${srcDir}/src"] |
||||
} |
||||
} |
||||
} |
||||
FileTree files =fileTree(dir:'./',include:'build.*.gradle') |
||||
def buildDir=files[0].path.substring(0,files[0].path.lastIndexOf ('\\')) |
||||
buildDir=buildDir.substring(0,buildDir.lastIndexOf ('\\')) |
||||
def branchName=buildDir.substring(buildDir.lastIndexOf ('\\')+1) |
||||
|
||||
//指定外部依赖 |
||||
dependencies{ |
||||
compile fileTree(dir:"../../../finereport-lib-stable/${branchName}",include:'**/*.jar') |
||||
compile fileTree(dir:'../../../',include:"finereport-*-stable/${branchName}/**/build/libs/*.jar") |
||||
|
||||
testCompile 'junit:junit:4.12' |
||||
} |
||||
//将非.java 文件复制到classes文件夹下参与打包 |
||||
task copyFile(type:Copy,dependsOn:compileJava){ |
||||
copy{ |
||||
from ("${srcDir}/src"){ |
||||
exclude '**/.setting/**','.classpath','.project','**/*.java','**/*.db','**/*.g','**/package.html' |
||||
|
||||
} |
||||
into 'build/classes/main' |
||||
} |
||||
|
||||
} |
||||
|
||||
//压缩项目中的js文件 |
||||
task compressJS{ |
||||
ant.taskdef(name:'yuicompress',classname:'com.yahoo.platform.yui.compressor.YUICompressTask'){ |
||||
classpath { |
||||
fileset(dir:'../../../finereport-lib4build-stable',includes:'**/*.jar') |
||||
} |
||||
} |
||||
ant.yuicompress(linebreak:"500",warn:"false", munge:"yes",preserveallsemicolons:"false",charset:"utf-8",encoding:"utf-8",outputfolder:'build/classes/main'){ |
||||
fileset (dir:"${srcDir}/src"){ |
||||
include (name:'**/*.js') |
||||
include (name:'**/*.css') |
||||
} |
||||
|
||||
} |
||||
} |
||||
jar.dependsOn compressJS |
||||
|
@ -0,0 +1,59 @@
|
||||
package com.fr.aspectj.designerchart; |
||||
|
||||
/** |
||||
* Created by plough on 2017/3/3. |
||||
*/ |
||||
import com.fr.chart.chartattr.Chart; |
||||
import com.fr.design.mainframe.templateinfo.TemplateInfoCollector; |
||||
import org.aspectj.lang.reflect.SourceLocation; |
||||
|
||||
import javax.swing.event.ListSelectionEvent; |
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.Date; |
||||
|
||||
public aspect TemplateProcessTracker { |
||||
//声明一个pointcut,匹配你需要的方法 |
||||
pointcut onMouseClicked(MouseEvent e) : |
||||
execution(* mouseClicked(MouseEvent)) && args(e); |
||||
pointcut onMousePressed(MouseEvent e) : |
||||
execution(* mousePressed(MouseEvent)) && args(e); |
||||
pointcut onMouseReleased(MouseEvent e) : |
||||
execution(* mouseReleased(MouseEvent)) && args(e); |
||||
pointcut onActionPerformed(ActionEvent e) : |
||||
execution(* actionPerformed(ActionEvent)) && args(e); |
||||
pointcut onChartUpdate(Chart c) : |
||||
execution(* update(Chart)) && args(c); |
||||
|
||||
//before表示之前的意思 |
||||
//这整个表示在MouseAdapter的public void mouseXXX(MouseEvent)方法调用之前,你想要执行的代码 |
||||
before(MouseEvent e) : onMouseClicked(e) || onMousePressed(e) || onMouseReleased(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation();//切面对应的代码位置 |
||||
|
||||
//String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
} |
||||
//同上 |
||||
before(ActionEvent e) : onActionPerformed(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
// !within(LogHandlerBar) 没用, 手动过滤 |
||||
if (e.getSource().toString().contains("javax.swing.Timer")) { |
||||
return; |
||||
} |
||||
|
||||
//String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
|
||||
} |
||||
//同上 |
||||
before(Chart c) : onChartUpdate(c) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
//String log = String.format("%s:\n%s\n插入新图表:%s\n\n", new Date(), sl, c.getChartName()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
|
||||
} |
||||
|
||||
} |
@ -0,0 +1,59 @@
|
||||
package com.fr.aspectj.designerform; |
||||
|
||||
/** |
||||
* Created by plough on 2017/3/3. |
||||
*/ |
||||
import com.fr.design.mainframe.templateinfo.TemplateInfoCollector; |
||||
import org.aspectj.lang.reflect.SourceLocation; |
||||
|
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.Date; |
||||
|
||||
public aspect TemplateProcessTracker { |
||||
//声明一个pointcut,匹配你需要的方法 |
||||
pointcut onMouseClicked(MouseEvent e) : |
||||
execution(* mouseClicked(MouseEvent)) && args(e); |
||||
pointcut onMousePressed(MouseEvent e) : |
||||
execution(* mousePressed(MouseEvent)) && args(e); |
||||
pointcut onMouseReleased(MouseEvent e) : |
||||
execution(* mouseReleased(MouseEvent)) && args(e); |
||||
pointcut onActionPerformed(ActionEvent e) : |
||||
execution(* actionPerformed(ActionEvent)) && args(e); |
||||
pointcut onSetValueAt(Object v, int r, int c) : |
||||
execution(* setValueAt(java.lang.Object, int, int)) && args(v, r, c); |
||||
|
||||
//before表示之前的意思 |
||||
//这整个表示在MouseAdapter的public void mouseXXX(MouseEvent)方法调用之前,你想要执行的代码 |
||||
before(MouseEvent e) : onMouseClicked(e) || onMousePressed(e) || onMouseReleased(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation();//切面对应的代码位置 |
||||
|
||||
//String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
} |
||||
//同上 |
||||
before(ActionEvent e) : onActionPerformed(e) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
// !within(LogHandlerBar) 没用, 手动过滤 |
||||
if (e.getSource().toString().contains("javax.swing.Timer")) { |
||||
return; |
||||
} |
||||
|
||||
//String log = String.format("%s:\n%s\n%s\n%s\n\n", new Date(), sl, e, e.getSource()); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
|
||||
} |
||||
//同上 |
||||
before(Object v, int r, int c) : onSetValueAt(v, r, c) { |
||||
SourceLocation sl = thisJoinPoint.getSourceLocation(); |
||||
|
||||
//String log = String.format("%s:\n%s\nset value: %s at (%d, %d)\n\n", new Date(), sl, v, r, c); |
||||
String log = ""; |
||||
TemplateInfoCollector.appendProcess(log); |
||||
|
||||
} |
||||
|
||||
|
||||
} |
@ -1,85 +1,94 @@
|
||||
package com.fr.design.designer.beans; |
||||
|
||||
import com.fr.design.beans.GroupModel; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
|
||||
/** |
||||
* 该接口是LayoutManager的BeanInfo类。标准Java平台没有提供布局管理器的BeanInfo类, |
||||
* 对于界面设计工具来说还需一些特殊的行为。 |
||||
* @since 6.5.3 |
||||
*/ |
||||
public interface LayoutAdapter { |
||||
|
||||
/** |
||||
* 在添加组件状态时,当鼠标移动到某个容器上方时,如果该容器有布局管理器,则会调用该布局 |
||||
* 管理适配器的accept来决定当前位置是否可以放置,并提供特殊的标识,比如红色区域标识。比 |
||||
* 如在BorderLayout中,如果某个方位已经放置了组件,则此时应该返回false标识该区域不可以 |
||||
* 放置。 |
||||
*@param creator 组件 |
||||
*@param x 添加的位置x,该位置是相对于container的 |
||||
*@param y 添加的位置y,该位置是相对于container的 |
||||
*@return 是否可以放置 |
||||
*/ |
||||
boolean accept(XCreator creator, int x, int y); |
||||
|
||||
/** |
||||
* 有的控件在拖拽调整大小后需要根据自身内容重新计算下当前的尺寸是否合适,如果不合适,就需要重新fix一下 |
||||
* @param creator 组件 |
||||
*/ |
||||
void fix(XCreator creator); |
||||
|
||||
/** |
||||
* 组件的ComponentAdapter在添加组件时,如果发现布局管理器不为空,会继而调用该布局管理器的 |
||||
* addComp方法来完成组件的具体添加。在该方法内,布局管理器可以提供额外的功能。 |
||||
* @param creator 被添加的新组件 |
||||
* @param x 添加的位置x,该位置是相对于container的 |
||||
* @param y 添加的位置y,该位置是相对于container的 |
||||
* @return 是否添加成功,成功返回true,否则false |
||||
*/ |
||||
boolean addBean(XCreator creator, int x, int y); |
||||
|
||||
/** |
||||
* 返回该布局管理适配器的Painter,为容器提供放置位置的标识。 |
||||
*/ |
||||
HoverPainter getPainter(); |
||||
|
||||
/** |
||||
* 显示parent的字组件child,解决CardLayout中显示某个非显示组件的特殊情况 |
||||
* @param child 组件 |
||||
*/ |
||||
void showComponent(XCreator child); |
||||
|
||||
void addNextComponent(XCreator dragged); |
||||
|
||||
/** |
||||
* 组件叠放顺序前插入 |
||||
* @param target 目标组件 |
||||
* @param added 插入组件 |
||||
*/ |
||||
void addBefore(XCreator target, XCreator added); |
||||
|
||||
/** |
||||
* 组件叠放顺序后插入 |
||||
* @param target 目标组件 |
||||
* @param added 放置组件 |
||||
*/ |
||||
void addAfter(XCreator target, XCreator added); |
||||
|
||||
/** |
||||
* 能否放置更多组件 |
||||
* @return 能则返回true |
||||
*/ |
||||
boolean canAcceptMoreComponent(); |
||||
|
||||
ConstraintsGroupModel getLayoutConstraints(XCreator creator); |
||||
|
||||
GroupModel getLayoutProperties(); |
||||
|
||||
/** |
||||
* 删除组件 |
||||
* @param creator 组件 |
||||
* @param initWidth 组件之前宽度 |
||||
* @param initHeight 组件之前高度 |
||||
*/ |
||||
void removeBean(XCreator creator, int initWidth, int initHeight); |
||||
package com.fr.design.designer.beans; |
||||
|
||||
import com.fr.design.beans.GroupModel; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
|
||||
/** |
||||
* 该接口是LayoutManager的BeanInfo类。标准Java平台没有提供布局管理器的BeanInfo类, |
||||
* 对于界面设计工具来说还需一些特殊的行为。 |
||||
* |
||||
* @since 6.5.3 |
||||
*/ |
||||
public interface LayoutAdapter { |
||||
|
||||
/** |
||||
* 在添加组件状态时,当鼠标移动到某个容器上方时,如果该容器有布局管理器,则会调用该布局 |
||||
* 管理适配器的accept来决定当前位置是否可以放置,并提供特殊的标识,比如红色区域标识。比 |
||||
* 如在BorderLayout中,如果某个方位已经放置了组件,则此时应该返回false标识该区域不可以 |
||||
* 放置。 |
||||
* |
||||
* @param creator 组件 |
||||
* @param x 添加的位置x,该位置是相对于container的 |
||||
* @param y 添加的位置y,该位置是相对于container的 |
||||
* @return 是否可以放置 |
||||
*/ |
||||
boolean accept(XCreator creator, int x, int y); |
||||
|
||||
/** |
||||
* 有的控件在拖拽调整大小后需要根据自身内容重新计算下当前的尺寸是否合适,如果不合适,就需要重新fix一下 |
||||
* |
||||
* @param creator 组件 |
||||
*/ |
||||
void fix(XCreator creator); |
||||
|
||||
/** |
||||
* 组件的ComponentAdapter在添加组件时,如果发现布局管理器不为空,会继而调用该布局管理器的 |
||||
* addComp方法来完成组件的具体添加。在该方法内,布局管理器可以提供额外的功能。 |
||||
* |
||||
* @param creator 被添加的新组件 |
||||
* @param x 添加的位置x,该位置是相对于container的 |
||||
* @param y 添加的位置y,该位置是相对于container的 |
||||
* @return 是否添加成功,成功返回true,否则false |
||||
*/ |
||||
boolean addBean(XCreator creator, int x, int y); |
||||
|
||||
/** |
||||
* 返回该布局管理适配器的Painter,为容器提供放置位置的标识。 |
||||
*/ |
||||
HoverPainter getPainter(); |
||||
|
||||
/** |
||||
* 显示parent的字组件child,解决CardLayout中显示某个非显示组件的特殊情况 |
||||
* |
||||
* @param child 组件 |
||||
*/ |
||||
void showComponent(XCreator child); |
||||
|
||||
void addNextComponent(XCreator dragged); |
||||
|
||||
/** |
||||
* 组件叠放顺序前插入 |
||||
* |
||||
* @param target 目标组件 |
||||
* @param added 插入组件 |
||||
*/ |
||||
void addBefore(XCreator target, XCreator added); |
||||
|
||||
/** |
||||
* 组件叠放顺序后插入 |
||||
* |
||||
* @param target 目标组件 |
||||
* @param added 放置组件 |
||||
*/ |
||||
void addAfter(XCreator target, XCreator added); |
||||
|
||||
/** |
||||
* 能否放置更多组件 |
||||
* |
||||
* @return 能则返回true |
||||
*/ |
||||
boolean canAcceptMoreComponent(); |
||||
|
||||
ConstraintsGroupModel getLayoutConstraints(XCreator creator); |
||||
|
||||
GroupModel getLayoutProperties(); |
||||
|
||||
/** |
||||
* 删除组件 |
||||
* |
||||
* @param creator 组件 |
||||
* @param initWidth 组件之前宽度 |
||||
* @param initHeight 组件之前高度 |
||||
*/ |
||||
void removeBean(XCreator creator, int initWidth, int initHeight); |
||||
} |
@ -1,31 +1,30 @@
|
||||
package com.fr.design.designer.beans.actions; |
||||
|
||||
import java.awt.event.InputEvent; |
||||
import java.awt.event.KeyEvent; |
||||
|
||||
import javax.swing.KeyStroke; |
||||
|
||||
import com.fr.base.BaseUtils; |
||||
import com.fr.general.Inter; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
|
||||
public class CopyAction extends FormEditAction { |
||||
|
||||
public CopyAction(FormDesigner t) { |
||||
super(t); |
||||
this.setName(Inter.getLocText("M_Edit-Copy")); |
||||
this.setMnemonic('C'); |
||||
this.setSmallIcon(BaseUtils.readIcon("/com/fr/design/images/m_edit/copy.png")); |
||||
this.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_C, InputEvent.CTRL_MASK)); |
||||
} |
||||
|
||||
@Override |
||||
public boolean executeActionReturnUndoRecordNeeded() { |
||||
FormDesigner tc = getEditingComponent(); |
||||
if (tc != null) { |
||||
tc.copy(); |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
package com.fr.design.designer.beans.actions; |
||||
|
||||
import com.fr.base.BaseUtils; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
import com.fr.general.Inter; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.event.InputEvent; |
||||
import java.awt.event.KeyEvent; |
||||
|
||||
public class CopyAction extends FormEditAction { |
||||
|
||||
public CopyAction(FormDesigner t) { |
||||
super(t); |
||||
this.setName(Inter.getLocText("M_Edit-Copy")); |
||||
this.setMnemonic('C'); |
||||
this.setSmallIcon(BaseUtils.readIcon("/com/fr/design/images/m_edit/copy.png")); |
||||
this.setAccelerator(KeyStroke.getKeyStroke(KeyEvent.VK_C, InputEvent.CTRL_MASK)); |
||||
} |
||||
|
||||
@Override |
||||
public boolean executeActionReturnUndoRecordNeeded() { |
||||
FormDesigner tc = getEditingComponent(); |
||||
if (tc != null) { |
||||
tc.copy(); |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
} |
@ -1,174 +1,189 @@
|
||||
package com.fr.design.designer.beans.adapters.layout; |
||||
|
||||
import java.awt.LayoutManager; |
||||
|
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.design.beans.GroupModel; |
||||
import com.fr.design.designer.beans.ConstraintsGroupModel; |
||||
import com.fr.design.designer.beans.HoverPainter; |
||||
import com.fr.design.designer.beans.LayoutAdapter; |
||||
import com.fr.design.designer.beans.painters.NullPainter; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWidgetCreator; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.design.utils.gui.LayoutUtils; |
||||
|
||||
public abstract class AbstractLayoutAdapter implements LayoutAdapter { |
||||
|
||||
protected XLayoutContainer container; |
||||
protected LayoutManager layout; |
||||
|
||||
public AbstractLayoutAdapter(XLayoutContainer container) { |
||||
this.container = container; |
||||
this.layout = container.getLayout(); |
||||
} |
||||
|
||||
/** |
||||
* 是否使用控件备份大小 |
||||
* @param xCreator 控件 |
||||
* @return 所在容器相同,且支持备份的话返回true |
||||
*/ |
||||
public boolean whetherUseBackupSize(XCreator xCreator) { |
||||
Class clazz = container.getClass(); |
||||
Class bkClazz = null; |
||||
if(xCreator.getBackupParent() != null) { |
||||
bkClazz = xCreator.getBackupParent().getClass(); |
||||
} |
||||
return ComparatorUtils.equals(bkClazz, clazz) |
||||
&& supportBackupSize(); |
||||
} |
||||
|
||||
/** |
||||
* 是否支持用备份大小 |
||||
* @return 否 |
||||
*/ |
||||
public boolean supportBackupSize() { |
||||
return false; |
||||
} |
||||
|
||||
/** |
||||
* 有的控件在拖拽调整大小后需要根据自身内容重新计算下当前的尺寸是否合适,如果不合适,就需要重新fix一下 |
||||
* @param creator 组件 |
||||
*/ |
||||
public void fix(XCreator creator) { |
||||
} |
||||
|
||||
/** |
||||
* 显示parent的字组件child,解决CardLayout中显示某个非显示组件的特殊情况 |
||||
* @param child 组件 |
||||
*/ |
||||
@Override |
||||
public void showComponent(XCreator child) { |
||||
child.setVisible(true); |
||||
} |
||||
|
||||
/** |
||||
* 组件的ComponentAdapter在添加组件时,如果发现布局管理器不为空,会继而调用该布局管理器的 |
||||
* addComp方法来完成组件的具体添加。在该方法内,布局管理器可以提供额外的功能。 |
||||
* @param creator 被添加的新组件 |
||||
* @param x 添加的位置x,该位置是相对于container的 |
||||
* @param y 添加的位置y,该位置是相对于container的 |
||||
* @return 是否添加成功,成功返回true,否则false |
||||
*/ |
||||
@Override |
||||
public boolean addBean(XCreator creator, int x, int y) { |
||||
if (!accept(creator, x, y)) { |
||||
return false; |
||||
} |
||||
addComp(creator, x, y); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
|
||||
/** |
||||
* 删除组件 |
||||
* @param creator 组件 |
||||
* @param initWidth 组件之前宽度 |
||||
* @param initHeight 组件之前高度 |
||||
*/ |
||||
public void removeBean(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
delete(creator, creatorWidth, creatorHeight); |
||||
} |
||||
|
||||
protected void delete(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
} |
||||
|
||||
protected abstract void addComp(XCreator creator, int x, int y); |
||||
|
||||
/** |
||||
* 增加下一个组件 |
||||
* @param dragged 组件 |
||||
*/ |
||||
@Override |
||||
public void addNextComponent(XCreator dragged) { |
||||
container.add(dragged); |
||||
LayoutUtils.layoutRootContainer(container); |
||||
} |
||||
|
||||
/** |
||||
* 目标控件位置插入组件 |
||||
* @param target 目标 |
||||
* @param added 增加组件 |
||||
*/ |
||||
@Override |
||||
public void addBefore(XCreator target, XCreator added) { |
||||
int index = ComponentUtils.indexOfComponent(container, target); |
||||
|
||||
if (index == -1) { |
||||
container.add(added, 0); |
||||
} else { |
||||
container.add(added, index); |
||||
} |
||||
|
||||
LayoutUtils.layoutRootContainer(container); |
||||
} |
||||
|
||||
/** |
||||
* 插在目标组件后面 |
||||
* @param target 目标 |
||||
* @param added 增加组件 |
||||
*/ |
||||
@Override |
||||
public void addAfter(XCreator target, XCreator added) { |
||||
int index = ComponentUtils.indexOfComponent(container, target); |
||||
|
||||
if (index == -1) { |
||||
container.add(added); |
||||
} else { |
||||
index++; |
||||
|
||||
if (index >= container.getComponentCount()) { |
||||
container.add(added); |
||||
} else { |
||||
container.add(added, index); |
||||
} |
||||
} |
||||
|
||||
LayoutUtils.layoutRootContainer(container); |
||||
} |
||||
|
||||
@Override |
||||
public HoverPainter getPainter() { |
||||
return new NullPainter(container); |
||||
} |
||||
|
||||
/** |
||||
* 是否能接收更多的组件 |
||||
* @return 能则返回true |
||||
*/ |
||||
@Override |
||||
public boolean canAcceptMoreComponent() { |
||||
return true; |
||||
} |
||||
|
||||
@Override |
||||
public ConstraintsGroupModel getLayoutConstraints(XCreator creator) { |
||||
return null; |
||||
} |
||||
|
||||
@Override |
||||
public GroupModel getLayoutProperties() { |
||||
return null; |
||||
} |
||||
package com.fr.design.designer.beans.adapters.layout; |
||||
|
||||
import com.fr.design.beans.GroupModel; |
||||
import com.fr.design.designer.beans.ConstraintsGroupModel; |
||||
import com.fr.design.designer.beans.HoverPainter; |
||||
import com.fr.design.designer.beans.LayoutAdapter; |
||||
import com.fr.design.designer.beans.painters.NullPainter; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWidgetCreator; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.design.utils.gui.LayoutUtils; |
||||
import com.fr.general.ComparatorUtils; |
||||
|
||||
import java.awt.*; |
||||
|
||||
public abstract class AbstractLayoutAdapter implements LayoutAdapter { |
||||
|
||||
protected XLayoutContainer container; |
||||
protected LayoutManager layout; |
||||
|
||||
public AbstractLayoutAdapter(XLayoutContainer container) { |
||||
this.container = container; |
||||
this.layout = container.getLayout(); |
||||
} |
||||
|
||||
/** |
||||
* 是否使用控件备份大小 |
||||
* |
||||
* @param xCreator 控件 |
||||
* @return 所在容器相同,且支持备份的话返回true |
||||
*/ |
||||
public boolean whetherUseBackupSize(XCreator xCreator) { |
||||
Class clazz = container.getClass(); |
||||
Class bkClazz = null; |
||||
if (xCreator.getBackupParent() != null) { |
||||
bkClazz = xCreator.getBackupParent().getClass(); |
||||
} |
||||
return ComparatorUtils.equals(bkClazz, clazz) |
||||
&& supportBackupSize(); |
||||
} |
||||
|
||||
/** |
||||
* 是否支持用备份大小 |
||||
* |
||||
* @return 否 |
||||
*/ |
||||
public boolean supportBackupSize() { |
||||
return false; |
||||
} |
||||
|
||||
/** |
||||
* 有的控件在拖拽调整大小后需要根据自身内容重新计算下当前的尺寸是否合适,如果不合适,就需要重新fix一下 |
||||
* |
||||
* @param creator 组件 |
||||
*/ |
||||
public void fix(XCreator creator) { |
||||
} |
||||
|
||||
/** |
||||
* 显示parent的字组件child,解决CardLayout中显示某个非显示组件的特殊情况 |
||||
* |
||||
* @param child 组件 |
||||
*/ |
||||
@Override |
||||
public void showComponent(XCreator child) { |
||||
child.setVisible(true); |
||||
} |
||||
|
||||
/** |
||||
* 组件的ComponentAdapter在添加组件时,如果发现布局管理器不为空,会继而调用该布局管理器的 |
||||
* addComp方法来完成组件的具体添加。在该方法内,布局管理器可以提供额外的功能。 |
||||
* |
||||
* @param creator 被添加的新组件 |
||||
* @param x 添加的位置x,该位置是相对于container的 |
||||
* @param y 添加的位置y,该位置是相对于container的 |
||||
* @return 是否添加成功,成功返回true,否则false |
||||
*/ |
||||
@Override |
||||
public boolean addBean(XCreator creator, int x, int y) { |
||||
if (!accept(creator, x, y)) { |
||||
return false; |
||||
} |
||||
addComp(creator, x, y); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
|
||||
/** |
||||
* 删除组件 |
||||
* |
||||
* @param creator 组件 |
||||
* @param creatorWidth |
||||
* @param creatorHeight |
||||
*/ |
||||
public void removeBean(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
delete(creator, creatorWidth, creatorHeight); |
||||
} |
||||
|
||||
protected void delete(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
} |
||||
|
||||
protected abstract void addComp(XCreator creator, int x, int y); |
||||
|
||||
/** |
||||
* 增加下一个组件 |
||||
* |
||||
* @param dragged 组件 |
||||
*/ |
||||
@Override |
||||
public void addNextComponent(XCreator dragged) { |
||||
container.add(dragged); |
||||
LayoutUtils.layoutRootContainer(container); |
||||
} |
||||
|
||||
/** |
||||
* 目标控件位置插入组件 |
||||
* |
||||
* @param target 目标 |
||||
* @param added 增加组件 |
||||
*/ |
||||
@Override |
||||
public void addBefore(XCreator target, XCreator added) { |
||||
int index = ComponentUtils.indexOfComponent(container, target); |
||||
|
||||
if (index == -1) { |
||||
container.add(added, 0); |
||||
} else { |
||||
container.add(added, index); |
||||
} |
||||
|
||||
LayoutUtils.layoutRootContainer(container); |
||||
} |
||||
|
||||
/** |
||||
* 插在目标组件后面 |
||||
* |
||||
* @param target 目标 |
||||
* @param added 增加组件 |
||||
*/ |
||||
@Override |
||||
public void addAfter(XCreator target, XCreator added) { |
||||
int index = ComponentUtils.indexOfComponent(container, target); |
||||
|
||||
if (index == -1) { |
||||
container.add(added); |
||||
} else { |
||||
index++; |
||||
|
||||
if (index >= container.getComponentCount()) { |
||||
container.add(added); |
||||
} else { |
||||
container.add(added, index); |
||||
} |
||||
} |
||||
|
||||
LayoutUtils.layoutRootContainer(container); |
||||
} |
||||
|
||||
@Override |
||||
public HoverPainter getPainter() { |
||||
return new NullPainter(container); |
||||
} |
||||
|
||||
/** |
||||
* 是否能接收更多的组件 |
||||
* |
||||
* @return 能则返回true |
||||
*/ |
||||
@Override |
||||
public boolean canAcceptMoreComponent() { |
||||
return true; |
||||
} |
||||
|
||||
@Override |
||||
public ConstraintsGroupModel getLayoutConstraints(XCreator creator) { |
||||
return null; |
||||
} |
||||
|
||||
@Override |
||||
public GroupModel getLayoutProperties() { |
||||
return null; |
||||
} |
||||
|
||||
|
||||
public XLayoutContainer getContainer() { |
||||
return this.container; |
||||
} |
||||
} |
@ -1,108 +1,105 @@
|
||||
/** |
||||
* |
||||
*/ |
||||
package com.fr.design.designer.beans.adapters.layout; |
||||
|
||||
|
||||
import com.fr.design.beans.GroupModel; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWParameterLayout; |
||||
import com.fr.design.designer.creator.XWidgetCreator; |
||||
import com.fr.design.designer.creator.cardlayout.XWCardLayout; |
||||
import com.fr.design.designer.creator.cardlayout.XWCardMainBorderLayout; |
||||
import com.fr.design.designer.creator.cardlayout.XWTabFitLayout; |
||||
import com.fr.design.designer.properties.FRTabFitLayoutPropertiesGroupModel; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.form.ui.LayoutBorderStyle; |
||||
import com.fr.form.ui.container.WBorderLayout; |
||||
import com.fr.form.ui.container.cardlayout.WCardMainBorderLayout; |
||||
import com.fr.general.ComparatorUtils; |
||||
|
||||
import java.awt.*; |
||||
|
||||
/** |
||||
* tab布局tabFit适配器 |
||||
* |
||||
* @author focus |
||||
* @date 2014-6-24 |
||||
*/ |
||||
public class FRTabFitLayoutAdapter extends FRFitLayoutAdapter { |
||||
//标题栏高度对tab布局内部组件的y坐标造成了偏移
|
||||
private static int TAB_HEIGHT = 40; |
||||
|
||||
/** |
||||
* 构造函数 |
||||
* |
||||
* @param container XWTabFitLayout容器 |
||||
*/ |
||||
public FRTabFitLayoutAdapter(XLayoutContainer container) { |
||||
super(container); |
||||
} |
||||
|
||||
/** |
||||
* 返回布局自身属性,方便一些特有设置在layout刷新时处理 |
||||
*/ |
||||
@Override |
||||
public GroupModel getLayoutProperties() { |
||||
XWTabFitLayout xfl = (XWTabFitLayout) container; |
||||
return new FRTabFitLayoutPropertiesGroupModel(xfl); |
||||
} |
||||
|
||||
/** |
||||
* 组件的ComponentAdapter在添加组件时,如果发现布局管理器不为空,会继而调用该布局管理器的 |
||||
* addComp方法来完成组件的具体添加。在该方法内,布局管理器可以提供额外的功能。 |
||||
* |
||||
* @param creator 被添加的新组件 |
||||
* @param x 添加的位置x,该位置是相对于container的 |
||||
* @param y 添加的位置y,该位置是相对于container的 |
||||
* @return 是否添加成功,成功返回true,否则false |
||||
*/ |
||||
@Override |
||||
public boolean addBean(XCreator creator, int x, int y) { |
||||
// 经过accept判断后,container会被改变,先备份
|
||||
XLayoutContainer backUpContainer = container; |
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
|
||||
int posX = x - rect.x; |
||||
int posY = y - rect.y; |
||||
if (!accept(creator, posX, posY)) { |
||||
return false; |
||||
} |
||||
// posX,posY是新拖入组件相对于容器的位置,若在tab布局的边缘,则需要把新组件添加到
|
||||
// 父层自适应布局中,这时候的添加位置就是tab布局所在的位置
|
||||
if (this.intersectsEdge(posX, posY, backUpContainer)) { |
||||
if (!ComparatorUtils.equals(backUpContainer.getOuterLayout(), backUpContainer.getBackupParent())) { |
||||
XWTabFitLayout tabLayout = (XWTabFitLayout) backUpContainer; |
||||
y = adjustY(y, tabLayout); |
||||
} |
||||
addComp(creator, x, y); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
// 如果不在边缘,容器为本自适应布局,增加组件的位置就是相对于容器的位置
|
||||
addComp(creator, posX, posY); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
|
||||
// tab布局的纵坐标受到tab高度的影响,判断的上边界取得是里面XWTabFitLayout的上边界,
|
||||
// 实际计算的时候的纵坐标用了外层的CardMainBorerLayout,需要将tab高度减掉
|
||||
//将y值变为相对坐标以实现获取到鼠标drop位置的控件
|
||||
//TODO 可以直接在这边将x,y都变成相对坐标,这样在后面判断拖进来的新控件放置方式的时候就不用再判断了
|
||||
private int adjustY(int y, XWTabFitLayout tabLayout) { |
||||
XWCardLayout cardLayout = (XWCardLayout) tabLayout.getBackupParent(); |
||||
LayoutBorderStyle style = cardLayout.toData().getBorderStyle(); |
||||
if (container.getLocation().y == WBorderLayout.DEFAULT_SIZE) { |
||||
y = y - WBorderLayout.DEFAULT_SIZE; |
||||
} |
||||
if (ComparatorUtils.equals(style.getType(), LayoutBorderStyle.TITLE)) { |
||||
y = y - WCardMainBorderLayout.TAB_HEIGHT; |
||||
} |
||||
return y; |
||||
} |
||||
|
||||
protected Rectangle getLayoutBound(XWCardMainBorderLayout mainLayout) { |
||||
return ComponentUtils.getRelativeBounds(mainLayout); |
||||
} |
||||
/** |
||||
* |
||||
*/ |
||||
package com.fr.design.designer.beans.adapters.layout; |
||||
|
||||
|
||||
import com.fr.design.beans.GroupModel; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWParameterLayout; |
||||
import com.fr.design.designer.creator.XWidgetCreator; |
||||
import com.fr.design.designer.creator.cardlayout.XWCardLayout; |
||||
import com.fr.design.designer.creator.cardlayout.XWCardMainBorderLayout; |
||||
import com.fr.design.designer.creator.cardlayout.XWTabFitLayout; |
||||
import com.fr.design.designer.properties.FRTabFitLayoutPropertiesGroupModel; |
||||
import com.fr.design.mainframe.widget.editors.ParameterEditor; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.form.ui.LayoutBorderStyle; |
||||
import com.fr.form.ui.container.WBorderLayout; |
||||
import com.fr.form.ui.container.cardlayout.WCardMainBorderLayout; |
||||
import com.fr.general.ComparatorUtils; |
||||
|
||||
import java.awt.*; |
||||
|
||||
/** |
||||
* tab布局tabFit适配器 |
||||
* |
||||
* @author focus |
||||
* @date 2014-6-24 |
||||
*/ |
||||
public class FRTabFitLayoutAdapter extends FRFitLayoutAdapter { |
||||
|
||||
/** |
||||
* 构造函数 |
||||
* |
||||
* @param container XWTabFitLayout容器 |
||||
*/ |
||||
public FRTabFitLayoutAdapter(XLayoutContainer container) { |
||||
super(container); |
||||
} |
||||
|
||||
/** |
||||
* 返回布局自身属性,方便一些特有设置在layout刷新时处理 |
||||
*/ |
||||
@Override |
||||
public GroupModel getLayoutProperties() { |
||||
XWTabFitLayout xfl = (XWTabFitLayout) container; |
||||
return new FRTabFitLayoutPropertiesGroupModel(xfl); |
||||
} |
||||
|
||||
/** |
||||
* 组件的ComponentAdapter在添加组件时,如果发现布局管理器不为空,会继而调用该布局管理器的 |
||||
* addComp方法来完成组件的具体添加。在该方法内,布局管理器可以提供额外的功能。 |
||||
* |
||||
* @param creator 被添加的新组件 |
||||
* @param x 添加的位置x,该位置是相对于container的 |
||||
* @param y 添加的位置y,该位置是相对于container的 |
||||
* @return 是否添加成功,成功返回true,否则false |
||||
*/ |
||||
@Override |
||||
public boolean addBean(XCreator creator, int x, int y) { |
||||
// 经过accept判断后,container会被改变,先备份
|
||||
XLayoutContainer backUpContainer = container; |
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
int posX = x - rect.x; |
||||
int posY = y - rect.y; |
||||
if (!accept(creator, posX, posY)) { |
||||
return false; |
||||
} |
||||
// posX,posY是新拖入组件相对于容器的位置,若在tab布局的边缘,则需要把新组件添加到l
|
||||
// 父层自适应布局中,这时候的添加位置就是tab布局所在的位置
|
||||
if (this.intersectsEdge(posX, posY, backUpContainer)) { |
||||
if (!ComparatorUtils.equals(backUpContainer.getOuterLayout(), backUpContainer.getBackupParent())) { |
||||
XWTabFitLayout tabLayout = (XWTabFitLayout) backUpContainer; |
||||
y = adjustY(y, tabLayout); |
||||
} |
||||
addComp(creator, x, y); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
// 如果不在边缘,容器为本自适应布局,增加组件的位置就是相对于容器的位置
|
||||
addComp(creator, posX, posY); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
|
||||
// tab布局的纵坐标受到tab高度以及参数面板高度的影响,判断的上边界取得是里面XWTabFitLayout的上边界,
|
||||
// 实际计算的时候的纵坐标用了外层的CardMainBorerLayout,需要将tab高度和参数面板高度减掉
|
||||
// 将y值变为相对坐标以实现获取到鼠标drop位置的控件
|
||||
// TODO 可以直接在这边将x,y都变成相对坐标,这样在后面判断拖进来的新控件放置方式的时候就不用再判断了
|
||||
private int adjustY(int y, XWTabFitLayout tabLayout) { |
||||
XWCardLayout cardLayout = (XWCardLayout) tabLayout.getBackupParent(); |
||||
LayoutBorderStyle style = cardLayout.toData().getBorderStyle(); |
||||
y = y - this.getParaEditorYOffset(); |
||||
if (ComparatorUtils.equals(style.getType(), LayoutBorderStyle.TITLE)) { |
||||
y = y - WCardMainBorderLayout.TAB_HEIGHT; |
||||
} |
||||
return y; |
||||
} |
||||
|
||||
protected Rectangle getLayoutBound(XWCardMainBorderLayout mainLayout) { |
||||
return ComponentUtils.getRelativeBounds(mainLayout); |
||||
} |
||||
|
||||
} |
@ -1,328 +1,461 @@
|
||||
package com.fr.design.designer.beans.models; |
||||
|
||||
import java.awt.LayoutManager; |
||||
import java.awt.Rectangle; |
||||
import java.awt.Toolkit; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.ArrayList; |
||||
|
||||
import com.fr.design.designer.beans.AdapterBus; |
||||
import com.fr.design.designer.beans.LayoutAdapter; |
||||
import com.fr.design.designer.beans.events.DesignerEvent; |
||||
import com.fr.design.designer.beans.location.Direction; |
||||
import com.fr.design.designer.beans.location.Location; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XCreatorUtils; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWFitLayout; |
||||
import com.fr.design.designer.creator.XWParameterLayout; |
||||
import com.fr.design.form.util.XCreatorConstants; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
import com.fr.design.mainframe.FormSelection; |
||||
import com.fr.design.mainframe.FormSelectionUtils; |
||||
import com.fr.design.utils.gui.LayoutUtils; |
||||
import com.fr.stable.ArrayUtils; |
||||
|
||||
/** |
||||
* 该model保存当前选择的组件和剪切版信息 |
||||
*/ |
||||
public class SelectionModel { |
||||
private static final int DELTA_X_Y = 20; //粘贴时候的偏移距离
|
||||
private static FormSelection CLIP_BOARD = new FormSelection(); |
||||
private FormDesigner designer; |
||||
private FormSelection selection; |
||||
private Rectangle hotspot_bounds; |
||||
|
||||
public SelectionModel(FormDesigner designer) { |
||||
this.designer = designer; |
||||
selection = new FormSelection(); |
||||
} |
||||
|
||||
/** |
||||
* 重置。清空formSelction以及选择区域 |
||||
*/ |
||||
public void reset() { |
||||
selection.reset(); |
||||
hotspot_bounds = null; |
||||
} |
||||
|
||||
/** |
||||
* formSelction是否为空 |
||||
* @return 是否为空 |
||||
*/ |
||||
public static boolean isEmpty(){ |
||||
return CLIP_BOARD.isEmpty(); |
||||
} |
||||
|
||||
/** |
||||
* 鼠标点击一下,所选中的单个组件。按下Ctrl或者shift键时鼠标可以进行多选 |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void selectACreatorAtMouseEvent(MouseEvent e) { |
||||
if (!e.isControlDown() && !e.isShiftDown()) { |
||||
// 如果Ctrl或者Shift键盘没有按下,则清除已经选择的组件
|
||||
selection.reset(); |
||||
} |
||||
|
||||
// 获取e所在的组件
|
||||
XCreator comp = designer.getComponentAt(e); |
||||
|
||||
//布局组件的顶层布局如不可编辑,要获取其顶层布局
|
||||
XLayoutContainer topLayout = XCreatorUtils.getHotspotContainer(comp).getTopLayout(); |
||||
if(topLayout != null && !topLayout.isEditable()){ |
||||
comp = topLayout; |
||||
} |
||||
|
||||
// 如果父层是scale和title两个专属容器,返回其父层,组件本身是不让被选中的
|
||||
if (comp != designer.getRootComponent() && comp != designer.getParaComponent()) { |
||||
XCreator parentContainer = (XCreator) comp.getParent(); |
||||
comp = parentContainer.isDedicateContainer() ? parentContainer : comp; |
||||
} |
||||
if (selection.removeSelectedCreator(comp) || selection.addSelectedCreator(comp)) { |
||||
designer.getEditListenerTable().fireCreatorModified(comp, DesignerEvent.CREATOR_SELECTED); |
||||
designer.repaint(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 将所选组件剪切到剪切板上 |
||||
*/ |
||||
public void cutSelectedCreator2ClipBoard() { |
||||
if (hasSelectionComponent()) { |
||||
selection.cut2ClipBoard(CLIP_BOARD); |
||||
designer.getEditListenerTable().fireCreatorModified(DesignerEvent.CREATOR_CUTED); |
||||
designer.repaint(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 复制当前选中的组件到剪切板 |
||||
*/ |
||||
public void copySelectedCreator2ClipBoard() { |
||||
if (!selection.isEmpty()) { |
||||
selection.copy2ClipBoard(CLIP_BOARD); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 从剪切板粘帖组件 |
||||
* @return 否 |
||||
*/ |
||||
public boolean pasteFromClipBoard() { |
||||
if (!CLIP_BOARD.isEmpty()) { |
||||
XLayoutContainer parent = null; |
||||
if (!hasSelectionComponent()) { |
||||
FormSelectionUtils.paste2Container(designer, designer.getRootComponent(),CLIP_BOARD, DELTA_X_Y, DELTA_X_Y); |
||||
} else { |
||||
parent = XCreatorUtils.getParentXLayoutContainer(selection.getSelectedCreator()); |
||||
if (parent != null) { |
||||
Rectangle rec = selection.getSelctionBounds(); |
||||
FormSelectionUtils.paste2Container(designer, parent,CLIP_BOARD, rec.x + DELTA_X_Y, rec.y + DELTA_X_Y); |
||||
} |
||||
} |
||||
} else { |
||||
Toolkit.getDefaultToolkit().beep(); |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
public FormSelection getSelection() { |
||||
return selection; |
||||
} |
||||
|
||||
/** |
||||
* 删除当前所有选择的组件 |
||||
*/ |
||||
public void deleteSelection() { |
||||
XCreator[] roots = selection.getSelectedCreators(); |
||||
|
||||
if (roots.length > 0) { |
||||
for (XCreator creator : roots) { |
||||
if(creator.acceptType(XWParameterLayout.class)){ |
||||
designer.removeParaComponent(); |
||||
} |
||||
|
||||
removeCreatorFromContainer(creator, creator.getWidth(), creator.getHeight()); |
||||
creator.removeAll(); |
||||
// 清除被选中的组件
|
||||
selection.reset(); |
||||
} |
||||
setSelectedCreator(designer.getRootComponent()); |
||||
// 触发事件
|
||||
designer.getEditListenerTable().fireCreatorModified(DesignerEvent.CREATOR_DELETED); |
||||
designer.repaint(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 从选择组件中删除某组件 |
||||
* |
||||
* @param creator 组件 |
||||
* @param creatorWidth 组件之前宽度 |
||||
* @param creatorHeight 组件之前高度 |
||||
*/ |
||||
public void removeCreator(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
selection.removeCreator(creator); |
||||
removeCreatorFromContainer(creator, creatorWidth, creatorHeight); |
||||
designer.repaint(); |
||||
} |
||||
|
||||
/** |
||||
* 设置选择区域 |
||||
*/ |
||||
public void setHotspotBounds(Rectangle rect) { |
||||
hotspot_bounds = rect; |
||||
} |
||||
|
||||
/** |
||||
* 获得当前选择区域 |
||||
*/ |
||||
public Rectangle getHotspotBounds() { |
||||
return hotspot_bounds; |
||||
} |
||||
|
||||
private void removeCreatorFromContainer(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
XLayoutContainer parent = XCreatorUtils.getParentXLayoutContainer(creator); |
||||
if (parent == null) { |
||||
return; |
||||
} |
||||
boolean changeCreator = creator.shouldScaleCreator() || creator.hasTitleStyle(); |
||||
if (parent.acceptType(XWFitLayout.class) && changeCreator) { |
||||
creator = (XCreator) creator.getParent(); |
||||
} |
||||
parent.getLayoutAdapter().removeBean(creator, creatorWidth, creatorHeight); |
||||
// 删除其根组件,同时就删除了同时被选择的叶子组件
|
||||
parent.remove(creator); |
||||
LayoutManager layout = parent.getLayout(); |
||||
|
||||
if (layout != null) { |
||||
// 刷新组件容器的布局
|
||||
LayoutUtils.layoutContainer(parent); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 是否有组件被选择。如果所选组件是最底层容器,也视为无选择 |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean hasSelectionComponent() { |
||||
return !selection.isEmpty() && selection.getSelectedCreator().getParent() != null; |
||||
} |
||||
|
||||
/** |
||||
* 移动组件至指定位置 |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
*/ |
||||
public void move(int x, int y) { |
||||
for (XCreator creator : selection.getSelectedCreators()) { |
||||
creator.setLocation(creator.getX() + x, creator.getY() + y); |
||||
LayoutAdapter layoutAdapter = AdapterBus.searchLayoutAdapter(designer, creator); |
||||
if (layoutAdapter != null) { |
||||
layoutAdapter.fix(creator); |
||||
} |
||||
} |
||||
designer.getEditListenerTable().fireCreatorModified(selection.getSelectedCreator(), |
||||
DesignerEvent.CREATOR_SELECTED); |
||||
} |
||||
|
||||
/** |
||||
* 释放捕获 |
||||
*/ |
||||
public void releaseDragging() { |
||||
designer.setPainter(null); |
||||
selection.fixCreator(designer); |
||||
designer.getEditListenerTable().fireCreatorModified(selection.getSelectedCreator(), |
||||
DesignerEvent.CREATOR_RESIZED); |
||||
} |
||||
|
||||
public Direction getDirectionAt(MouseEvent e) { |
||||
Direction dir; |
||||
if (e.isControlDown() || e.isShiftDown()) { |
||||
XCreator creator = designer.getComponentAt(e.getX(), e.getY(), selection.getSelectedCreators()); |
||||
if (creator != designer.getRootComponent() && selection.addedable(creator)) { |
||||
return Location.add; |
||||
} |
||||
} |
||||
if (hasSelectionComponent()) { |
||||
int x = e.getX() + designer.getArea().getHorizontalValue(); |
||||
int y = e.getY() + designer.getArea().getVerticalValue(); |
||||
dir = getDirection(selection.getRelativeBounds(), x, y); |
||||
if (selection.size() == 1) { |
||||
if (!ArrayUtils.contains(selection.getSelectedCreator().getDirections(), dir.getActual())) { |
||||
dir = Location.outer; |
||||
} |
||||
} |
||||
} else { |
||||
dir = Location.outer; |
||||
} |
||||
|
||||
if (designer.getDesignerMode().isFormParameterEditor() && dir == Location.outer) { |
||||
dir = designer.getLoc2Root(e); |
||||
} |
||||
return dir; |
||||
} |
||||
|
||||
private Direction getDirection(Rectangle bounds, int x, int y) { |
||||
if (x < (bounds.x - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((x >= (bounds.x - XCreatorConstants.RESIZE_BOX_SIZ)) && (x <= bounds.x)) { |
||||
if (y < (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((y >= (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) && (y <= bounds.y)) { |
||||
return Location.left_top; |
||||
} else if ((y > bounds.y) && (y < (bounds.y + bounds.height))) { |
||||
return Location.left; |
||||
} else if ((y >= (bounds.y + bounds.height)) |
||||
&& (y <= (bounds.y + bounds.height + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
return Location.left_bottom; |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} else if ((x > bounds.x) && (x < (bounds.x + bounds.width))) { |
||||
if (y < (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((y >= (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) && (y <= bounds.y)) { |
||||
return Location.top; |
||||
} else if ((y > bounds.y) && (y < (bounds.y + bounds.height))) { |
||||
return Location.inner; |
||||
} else if ((y >= (bounds.y + bounds.height)) |
||||
&& (y <= (bounds.y + bounds.height + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
return Location.bottom; |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} else if ((x >= (bounds.x + bounds.width)) |
||||
&& (x <= (bounds.x + bounds.width + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
if (y < (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((y >= (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) && (y <= bounds.y)) { |
||||
return Location.right_top; |
||||
} else if ((y > bounds.y) && (y < (bounds.y + bounds.height))) { |
||||
return Location.right; |
||||
} else if ((y >= (bounds.y + bounds.height)) |
||||
&& (y <= (bounds.y + bounds.height + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
return Location.right_bottom; |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} |
||||
|
||||
private void fireCreatorSelected() { |
||||
designer.getEditListenerTable().fireCreatorModified(selection.getSelectedCreator(), |
||||
DesignerEvent.CREATOR_SELECTED); |
||||
} |
||||
|
||||
public void setSelectedCreator(XCreator rootComponent) { |
||||
selection.setSelectedCreator(rootComponent); |
||||
fireCreatorSelected(); |
||||
} |
||||
|
||||
public void setSelectedCreators(ArrayList<XCreator> rebuildSelection) { |
||||
selection.setSelectedCreators(rebuildSelection); |
||||
fireCreatorSelected(); |
||||
} |
||||
package com.fr.design.designer.beans.models; |
||||
|
||||
import com.fr.design.designer.beans.AdapterBus; |
||||
import com.fr.design.designer.beans.LayoutAdapter; |
||||
import com.fr.design.designer.beans.events.DesignerEvent; |
||||
import com.fr.design.designer.beans.location.Direction; |
||||
import com.fr.design.designer.beans.location.Location; |
||||
import com.fr.design.designer.creator.*; |
||||
import com.fr.design.designer.creator.cardlayout.XWCardLayout; |
||||
import com.fr.design.designer.creator.cardlayout.XWCardMainBorderLayout; |
||||
import com.fr.design.designer.creator.cardlayout.XWTabFitLayout; |
||||
import com.fr.design.form.util.XCreatorConstants; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
import com.fr.design.mainframe.FormSelection; |
||||
import com.fr.design.mainframe.FormSelectionUtils; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.design.utils.gui.LayoutUtils; |
||||
import com.fr.stable.ArrayUtils; |
||||
|
||||
import java.awt.*; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.ArrayList; |
||||
|
||||
/** |
||||
* 该model保存当前选择的组件和剪切版信息 |
||||
*/ |
||||
public class SelectionModel { |
||||
//被粘贴组件在所选组件位置处往下、往右各错开20像素。执行多次粘贴时,在上一次粘贴的位置处错开20像素。
|
||||
private static final int DELTA_X_Y = 20; //粘贴时候的偏移距离
|
||||
private static final double OFFSET_RELATIVE = 0.80; |
||||
private static FormSelection clipboard = new FormSelection(); |
||||
private FormDesigner designer; |
||||
private FormSelection selection; |
||||
private Rectangle hotspotBounds; |
||||
|
||||
public SelectionModel(FormDesigner designer) { |
||||
this.designer = designer; |
||||
selection = new FormSelection(); |
||||
} |
||||
|
||||
/** |
||||
* 重置。清空formSelction以及选择区域 |
||||
*/ |
||||
public void reset() { |
||||
selection.reset(); |
||||
hotspotBounds = null; |
||||
} |
||||
|
||||
/** |
||||
* formSelction是否为空 |
||||
* |
||||
* @return 是否为空 |
||||
*/ |
||||
public static boolean isEmpty() { |
||||
return clipboard.isEmpty(); |
||||
} |
||||
|
||||
/** |
||||
* 鼠标点击一下,所选中的单个组件。按下Ctrl或者shift键时鼠标可以进行多选 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void selectACreatorAtMouseEvent(MouseEvent e) { |
||||
if (!e.isControlDown() && !e.isShiftDown()) { |
||||
// 如果Ctrl或者Shift键盘没有按下,则清除已经选择的组件
|
||||
selection.reset(); |
||||
} |
||||
// 获取e所在的组件
|
||||
XCreator comp = designer.getComponentAt(e); |
||||
|
||||
//布局组件的顶层布局如不可编辑,要获取其顶层布局
|
||||
XLayoutContainer topLayout = XCreatorUtils.getHotspotContainer(comp).getTopLayout(); |
||||
if (topLayout != null && !topLayout.isEditable()) { |
||||
comp = topLayout; |
||||
} |
||||
|
||||
// 如果父层是scale和title两个专属容器,返回其父层,组件本身是不让被选中的
|
||||
if (comp != designer.getRootComponent() && comp != designer.getParaComponent()) { |
||||
XCreator parentContainer = (XCreator) comp.getParent(); |
||||
comp = parentContainer.isDedicateContainer() ? parentContainer : comp; |
||||
} |
||||
if (selection.removeSelectedCreator(comp) || selection.addSelectedCreator(comp)) { |
||||
designer.getEditListenerTable().fireCreatorModified(comp, DesignerEvent.CREATOR_SELECTED); |
||||
designer.repaint(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 将所选组件剪切到剪切板上 |
||||
*/ |
||||
public void cutSelectedCreator2ClipBoard() { |
||||
if (hasSelectionComponent()) { |
||||
selection.cut2ClipBoard(clipboard); |
||||
designer.getEditListenerTable().fireCreatorModified(DesignerEvent.CREATOR_CUTED); |
||||
designer.repaint(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 复制当前选中的组件到剪切板 |
||||
*/ |
||||
public void copySelectedCreator2ClipBoard() { |
||||
if (!selection.isEmpty()) { |
||||
selection.copy2ClipBoard(clipboard); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 从剪切板粘帖组件 |
||||
* |
||||
* @return 否 |
||||
*/ |
||||
public boolean pasteFromClipBoard() { |
||||
if (!clipboard.isEmpty()) { |
||||
if (!hasSelectedPasteSource()) { |
||||
//未选
|
||||
unselectedPaste(); |
||||
} else { |
||||
//已选
|
||||
selectedPaste(); |
||||
} |
||||
} else { |
||||
Toolkit.getDefaultToolkit().beep(); |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
public FormSelection getSelection() { |
||||
return selection; |
||||
} |
||||
|
||||
/** |
||||
* 粘贴时未选择组件 |
||||
*/ |
||||
private void unselectedPaste() { |
||||
if (designer.getClass().equals(FormDesigner.class)) { |
||||
if (selection.getSelectedCreator() instanceof XWFitLayout) { |
||||
if (selection.getSelectedCreator().getClass().equals(XWTabFitLayout.class)) { |
||||
XLayoutContainer container = (XLayoutContainer) selection.getSelectedCreator(); |
||||
//tab布局编辑器内部左上角第一个坐标点
|
||||
int leftUpX = container.toData().getMargin().getLeft() + 1; |
||||
int leftUpY = container.toData().getMargin().getTop() + 1; |
||||
//选中第一个坐标点坐在的组件
|
||||
selection.setSelectedCreator((XCreator) container.getComponentAt(leftUpX, leftUpY)); |
||||
Rectangle rectangle = selection.getRelativeBounds(); |
||||
if (hasSelectedPasteSource()) { |
||||
selectedPaste(); |
||||
} else { |
||||
FormSelectionUtils.paste2Container(designer, container, clipboard, |
||||
rectangle.x + rectangle.width / 2, |
||||
rectangle.y + DELTA_X_Y); |
||||
} |
||||
} else { |
||||
//自适应布局编辑器内部左上角第一个坐标点
|
||||
int leftUpX = designer.getRootComponent().toData().getMargin().getLeft() + 1; |
||||
int leftUpY = designer.getRootComponent().toData().getMargin().getTop() + 1; |
||||
//选中第一个坐标点坐在的组件
|
||||
selection.setSelectedCreator((XCreator) designer.getRootComponent().getComponentAt(leftUpX, leftUpY)); |
||||
Rectangle rectangle = selection.getRelativeBounds(); |
||||
if (hasSelectedPasteSource()) { |
||||
selectedPaste(); |
||||
} else { |
||||
FormSelectionUtils.paste2Container(designer, designer.getRootComponent(), |
||||
clipboard, |
||||
rectangle.x + rectangle.width / 2, |
||||
rectangle.y + DELTA_X_Y); |
||||
} |
||||
} |
||||
} else { |
||||
//绝对布局
|
||||
//编辑器外面还有两层容器,使用designer.getRootComponent()获取到的是编辑器中层的容器,不是编辑器表层
|
||||
//当前选择的就是编辑器表层
|
||||
FormSelectionUtils.paste2Container(designer, (XLayoutContainer) selection.getSelectedCreator(), |
||||
clipboard, |
||||
DELTA_X_Y, |
||||
DELTA_X_Y); |
||||
} |
||||
} else { |
||||
//cpt本地组件复用,编辑器就一层,是最底层,使用designer.getRootComponent()就可以获取到
|
||||
//使用selection.getSelectedCreator()也应该是可以获取到的。
|
||||
FormSelectionUtils.paste2Container(designer, designer.getRootComponent(), |
||||
clipboard, |
||||
DELTA_X_Y, |
||||
DELTA_X_Y); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 粘贴时选择组件 |
||||
*/ |
||||
private void selectedPaste() { |
||||
XLayoutContainer container = null; |
||||
//获取到编辑器的表层容器(已选的组件的父容器就是表层容器)
|
||||
container = XCreatorUtils.getParentXLayoutContainer(selection.getSelectedCreator()); |
||||
if (container != null && selection.getSelectedCreator().getParent() instanceof XWFitLayout) { |
||||
//自适应布局
|
||||
|
||||
Rectangle selectionRec = selection.getRelativeBounds(); |
||||
//获取父容器位置,补充因参数面板高度导致的位置坐标计算偏移
|
||||
Rectangle containerRec = ComponentUtils.getRelativeBounds(container); |
||||
int positionX, positionY; |
||||
|
||||
if (container.getClass().equals(XWTabFitLayout.class)) { |
||||
//tab内部粘贴不补充高度偏移
|
||||
//且不计算参数面板造成的影响,因为在
|
||||
//@see com.fr.design.designer.beans.adapters.layout.FRTabFitLayoutAdapter#addBean中做了
|
||||
positionX = selectionRec.x + selectionRec.width / 2; |
||||
positionY = (int) (selectionRec.y + selectionRec.height * OFFSET_RELATIVE); |
||||
} else { |
||||
//计算自适应布局位置
|
||||
positionX = selectionRec.x - containerRec.x + selectionRec.width / 2; |
||||
positionY = (int) (selectionRec.y - containerRec.y + selectionRec.height * OFFSET_RELATIVE); |
||||
} |
||||
FormSelectionUtils.paste2Container(designer, container, clipboard, positionX, positionY); |
||||
} else if (container != null && selection.getSelectedCreator().getParent() instanceof XWAbsoluteLayout) { |
||||
//绝对布局
|
||||
Rectangle rec = selection.getSelctionBounds(); |
||||
FormSelectionUtils.paste2Container(designer, container, clipboard, rec.x + DELTA_X_Y, rec.y + DELTA_X_Y); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 删除当前所有选择的组件 |
||||
*/ |
||||
public void deleteSelection() { |
||||
XCreator[] roots = selection.getSelectedCreators(); |
||||
|
||||
if (roots.length > 0) { |
||||
for (XCreator creator : roots) { |
||||
if (creator.acceptType(XWParameterLayout.class)) { |
||||
designer.removeParaComponent(); |
||||
} |
||||
removeCreatorFromContainer(creator, creator.getWidth(), creator.getHeight()); |
||||
creator.removeAll(); |
||||
// 清除被选中的组件
|
||||
selection.reset(); |
||||
} |
||||
setSelectedCreator(designer.getRootComponent()); |
||||
FormSelectionUtils.rebuildSelection(designer); |
||||
// 触发事件
|
||||
designer.getEditListenerTable().fireCreatorModified(DesignerEvent.CREATOR_DELETED); |
||||
designer.repaint(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 从选择组件中删除某组件 |
||||
* |
||||
* @param creator 组件 |
||||
* @param creatorWidth 组件之前宽度 |
||||
* @param creatorHeight 组件之前高度 |
||||
*/ |
||||
public void removeCreator(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
selection.removeCreator(creator); |
||||
removeCreatorFromContainer(creator, creatorWidth, creatorHeight); |
||||
designer.repaint(); |
||||
} |
||||
|
||||
/** |
||||
* 设置选择区域 |
||||
*/ |
||||
public void setHotspotBounds(Rectangle rect) { |
||||
hotspotBounds = rect; |
||||
} |
||||
|
||||
/** |
||||
* 获得当前选择区域 |
||||
*/ |
||||
public Rectangle getHotspotBounds() { |
||||
return hotspotBounds; |
||||
} |
||||
|
||||
private void removeCreatorFromContainer(XCreator creator, int creatorWidth, int creatorHeight) { |
||||
XLayoutContainer parent = XCreatorUtils.getParentXLayoutContainer(creator); |
||||
if (parent == null) { |
||||
return; |
||||
} |
||||
boolean changeCreator = creator.shouldScaleCreator() || creator.hasTitleStyle(); |
||||
if (parent.acceptType(XWFitLayout.class) && changeCreator) { |
||||
creator = (XCreator) creator.getParent(); |
||||
} |
||||
parent.getLayoutAdapter().removeBean(creator, creatorWidth, creatorHeight); |
||||
// 删除其根组件,同时就删除了同时被选择的叶子组件
|
||||
parent.remove(creator); |
||||
LayoutManager layout = parent.getLayout(); |
||||
|
||||
if (layout != null) { |
||||
// 刷新组件容器的布局
|
||||
LayoutUtils.layoutContainer(parent); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 是否有组件被选择。如果所选组件是最底层容器,也视为无选择 |
||||
* |
||||
* @return 是则返回true |
||||
* yaoh.wu 不应该通过判断是否是最底层容器来判断是否选择了组件 |
||||
* 而是应该判断选择的容器是否是编辑器的最表层容器,也就是点击空白地方选择的容器 |
||||
* 但是直接判断选择的容器是否是编辑器最表层类型又会引发拖动时选不上的情况, |
||||
* 因此通过判断父容器来实现 |
||||
* <p> |
||||
* 举例:frm组件复用 绝对布局情况下,不选择时有三层容器: |
||||
* 底层@see {@link com.fr.design.designer.creator.XWBorderLayout} |
||||
* 中层@see {@link XWFitLayout} |
||||
* 表层@see {@link com.fr.design.designer.creator.XWAbsoluteBodyLayout} |
||||
* <p> |
||||
* 但是编辑窗口的最外层其实是表层@see {@link com.fr.design.designer.creator.XWAbsoluteBodyLayout}, |
||||
* 其他两层不是靠添加组件就可以编辑的。 |
||||
*/ |
||||
public boolean hasSelectedPasteSource() { |
||||
XCreator selectionXCreator = selection.getSelectedCreator(); |
||||
if (designer.getClass().equals(FormDesigner.class)) { |
||||
//frm本地组件复用
|
||||
if (selectionXCreator != null) { |
||||
//选中的是否是tab布局编辑器本身
|
||||
boolean tabEditor = selectionXCreator.getClass().equals(XWCardMainBorderLayout.class) |
||||
|| selectionXCreator.getClass().equals(XWCardLayout.class) |
||||
|| selectionXCreator.getClass().equals(XWTabFitLayout.class); |
||||
//选中的是否是frm绝对布局编辑器本身
|
||||
boolean absoluteEditor = selectionXCreator.getClass().equals(XWAbsoluteBodyLayout.class); |
||||
//选中是否是frm绝对画布块编辑器本身
|
||||
boolean absoluteCanvas = selectionXCreator.getClass().equals(XWAbsoluteLayout.class); |
||||
//选中的是否是相对布局编辑器本身
|
||||
boolean relativeEditor = selectionXCreator.getClass().equals(XWFitLayout.class); |
||||
|
||||
return !(tabEditor || absoluteEditor || absoluteCanvas || relativeEditor); |
||||
} else { |
||||
return false; |
||||
} |
||||
} else { |
||||
//cpt本地组件复用,selection.getSelectedCreator().getParent()=@XWParameterLayout instanceof @XWAbsoluteLayout
|
||||
return selectionXCreator != null && selectionXCreator.getParent() != null; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 是否有组件被选择。如果所选组件是最底层容器,也视为无选择 |
||||
* |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean hasSelectionComponent() { |
||||
return !selection.isEmpty() && selection.getSelectedCreator().getParent() != null; |
||||
} |
||||
|
||||
/** |
||||
* 移动组件至指定位置 |
||||
* |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
*/ |
||||
public void move(int x, int y) { |
||||
for (XCreator creator : selection.getSelectedCreators()) { |
||||
creator.setLocation(creator.getX() + x, creator.getY() + y); |
||||
LayoutAdapter layoutAdapter = AdapterBus.searchLayoutAdapter(designer, creator); |
||||
if (layoutAdapter != null) { |
||||
layoutAdapter.fix(creator); |
||||
} |
||||
} |
||||
designer.getEditListenerTable().fireCreatorModified(selection.getSelectedCreator(), |
||||
DesignerEvent.CREATOR_SELECTED); |
||||
} |
||||
|
||||
/** |
||||
* 释放捕获 |
||||
*/ |
||||
public void releaseDragging() { |
||||
designer.setPainter(null); |
||||
selection.fixCreator(designer); |
||||
designer.getEditListenerTable().fireCreatorModified(selection.getSelectedCreator(), |
||||
DesignerEvent.CREATOR_RESIZED); |
||||
} |
||||
|
||||
public Direction getDirectionAt(MouseEvent e) { |
||||
Direction dir; |
||||
if (e.isControlDown() || e.isShiftDown()) { |
||||
XCreator creator = designer.getComponentAt(e.getX(), e.getY(), selection.getSelectedCreators()); |
||||
if (creator != designer.getRootComponent() && selection.addedable(creator)) { |
||||
return Location.add; |
||||
} |
||||
} |
||||
if (hasSelectionComponent()) { |
||||
int x = e.getX() + designer.getArea().getHorizontalValue(); |
||||
int y = e.getY() + designer.getArea().getVerticalValue(); |
||||
dir = getDirection(selection.getRelativeBounds(), x, y); |
||||
if (selection.size() == 1 && !ArrayUtils.contains(selection.getSelectedCreator().getDirections(), dir |
||||
.getActual())) { |
||||
dir = Location.outer; |
||||
} |
||||
} else { |
||||
dir = Location.outer; |
||||
} |
||||
|
||||
if (designer.getDesignerMode().isFormParameterEditor() && dir == Location.outer) { |
||||
dir = designer.getLoc2Root(e); |
||||
} |
||||
return dir; |
||||
} |
||||
|
||||
private Direction getDirection(Rectangle bounds, int x, int y) { |
||||
if (x < (bounds.x - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((x >= (bounds.x - XCreatorConstants.RESIZE_BOX_SIZ)) && (x <= bounds.x)) { |
||||
if (y < (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((y >= (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) && (y <= bounds.y)) { |
||||
return Location.left_top; |
||||
} else if ((y > bounds.y) && (y < (bounds.y + bounds.height))) { |
||||
return Location.left; |
||||
} else if ((y >= (bounds.y + bounds.height)) |
||||
&& (y <= (bounds.y + bounds.height + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
return Location.left_bottom; |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} else if ((x > bounds.x) && (x < (bounds.x + bounds.width))) { |
||||
if (y < (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((y >= (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) && (y <= bounds.y)) { |
||||
return Location.top; |
||||
} else if ((y > bounds.y) && (y < (bounds.y + bounds.height))) { |
||||
return Location.inner; |
||||
} else if ((y >= (bounds.y + bounds.height)) |
||||
&& (y <= (bounds.y + bounds.height + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
return Location.bottom; |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} else if ((x >= (bounds.x + bounds.width)) |
||||
&& (x <= (bounds.x + bounds.width + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
if (y < (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) { |
||||
return Location.outer; |
||||
} else if ((y >= (bounds.y - XCreatorConstants.RESIZE_BOX_SIZ)) && (y <= bounds.y)) { |
||||
return Location.right_top; |
||||
} else if ((y > bounds.y) && (y < (bounds.y + bounds.height))) { |
||||
return Location.right; |
||||
} else if ((y >= (bounds.y + bounds.height)) |
||||
&& (y <= (bounds.y + bounds.height + XCreatorConstants.RESIZE_BOX_SIZ))) { |
||||
return Location.right_bottom; |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} else { |
||||
return Location.outer; |
||||
} |
||||
} |
||||
|
||||
private void fireCreatorSelected() { |
||||
designer.getEditListenerTable().fireCreatorModified(selection.getSelectedCreator(), |
||||
DesignerEvent.CREATOR_SELECTED); |
||||
} |
||||
|
||||
public void setSelectedCreator(XCreator rootComponent) { |
||||
selection.setSelectedCreator(rootComponent); |
||||
fireCreatorSelected(); |
||||
} |
||||
|
||||
public void setSelectedCreators(ArrayList<XCreator> rebuildSelection) { |
||||
selection.setSelectedCreators(rebuildSelection); |
||||
fireCreatorSelected(); |
||||
} |
||||
} |
@ -1,446 +1,476 @@
|
||||
package com.fr.design.designer.beans.models; |
||||
|
||||
import java.awt.Component; |
||||
import java.awt.Cursor; |
||||
import java.awt.Graphics; |
||||
import java.awt.Point; |
||||
import java.awt.Rectangle; |
||||
import java.awt.Toolkit; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.ArrayList; |
||||
|
||||
import com.fr.design.beans.location.Absorptionline; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
import com.fr.design.mainframe.FormSelectionUtils; |
||||
import com.fr.design.designer.beans.AdapterBus; |
||||
import com.fr.design.designer.beans.HoverPainter; |
||||
import com.fr.design.designer.beans.LayoutAdapter; |
||||
import com.fr.design.designer.beans.events.DesignerEvent; |
||||
import com.fr.design.designer.beans.location.Direction; |
||||
import com.fr.design.designer.beans.location.Location; |
||||
import com.fr.design.designer.creator.XConnector; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XCreatorUtils; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWAbsoluteLayout; |
||||
import com.fr.design.designer.creator.*; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
import com.fr.design.mainframe.FormSelectionUtils; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
|
||||
import java.awt.*; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.ArrayList; |
||||
|
||||
/** |
||||
* 普通模式下的状态model |
||||
*/ |
||||
public class StateModel { |
||||
// 对应的selection model
|
||||
|
||||
private SelectionModel selectionModel; |
||||
// 当前鼠标进入拖拽区域的位置类型
|
||||
private Direction driection; |
||||
|
||||
// 当前拖拽的起始位置
|
||||
private int current_x; |
||||
private int current_y; |
||||
|
||||
private Point startPoint = new Point(); |
||||
private Point currentPoint = new Point(); |
||||
|
||||
private Absorptionline lineInX; |
||||
private Absorptionline lineInY; |
||||
//等距线
|
||||
private Absorptionline lineEquidistant; |
||||
|
||||
// 当前是否处于拖拽选择状态
|
||||
private boolean selecting; |
||||
private boolean dragging; |
||||
|
||||
private boolean addable; |
||||
|
||||
private FormDesigner designer; |
||||
|
||||
public StateModel(FormDesigner designer) { |
||||
this.designer = designer; |
||||
selectionModel = designer.getSelectionModel(); |
||||
} |
||||
|
||||
/** |
||||
* 返回direction |
||||
* @return direction方向 |
||||
*/ |
||||
public Direction getDirection() { |
||||
return driection; |
||||
} |
||||
|
||||
/** |
||||
* 是否有组件正被选中 |
||||
* |
||||
* @return true 如果至少一个组件被选中 |
||||
*/ |
||||
public boolean isSelecting() { |
||||
return selecting; |
||||
} |
||||
// 对应的selection model
|
||||
|
||||
private SelectionModel selectionModel; |
||||
// 当前鼠标进入拖拽区域的位置类型
|
||||
private Direction driection; |
||||
|
||||
// 当前拖拽的起始位置
|
||||
private int currentX; |
||||
private int currentY; |
||||
|
||||
//拖拽组件原始位置大小备份
|
||||
private Rectangle selectedPositionBackup; |
||||
|
||||
private Point startPoint = new Point(); |
||||
private Point currentPoint = new Point(); |
||||
|
||||
private Absorptionline lineInX; |
||||
private Absorptionline lineInY; |
||||
//等距线
|
||||
private Absorptionline lineEquidistant; |
||||
|
||||
// 当前是否处于拖拽选择状态
|
||||
private boolean selecting; |
||||
private boolean dragging; |
||||
|
||||
private boolean addable; |
||||
|
||||
private FormDesigner designer; |
||||
|
||||
public StateModel(FormDesigner designer) { |
||||
this.designer = designer; |
||||
selectionModel = designer.getSelectionModel(); |
||||
} |
||||
|
||||
/** |
||||
*是否能拖拽 |
||||
* 返回direction |
||||
* |
||||
* @return direction方向 |
||||
*/ |
||||
public Direction getDirection() { |
||||
return driection; |
||||
} |
||||
|
||||
/** |
||||
* 是否有组件正被选中 |
||||
* |
||||
* @return true 如果至少一个组件被选中 |
||||
*/ |
||||
public boolean isSelecting() { |
||||
return selecting; |
||||
} |
||||
|
||||
/** |
||||
* 是否能拖拽 |
||||
* |
||||
* @return 非outer且选中为空 |
||||
*/ |
||||
public boolean dragable() { |
||||
return ((driection != Location.outer) && !selecting); |
||||
} |
||||
|
||||
/** |
||||
* 拖拽中是否可以转换为添加模式: |
||||
* 如果拖拽组件只有一个,鼠标当前所在位置的最底层表单容器与这个组件的容器不同; |
||||
* 如果拖拽组件为多个,鼠标当前所在位置的最底层表单容器除了要求要跟这些组件的容器不同外,还必须是绝对定位布局 |
||||
*/ |
||||
private void checkAddable(MouseEvent e) { |
||||
addable = false; |
||||
designer.setPainter(null); |
||||
|
||||
if (driection != Location.inner) { |
||||
return; |
||||
} |
||||
|
||||
XCreator comp = designer.getComponentAt(e.getX(), e.getY(), selectionModel.getSelection().getSelectedCreators()); |
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(comp); |
||||
XCreator creator = selectionModel.getSelection().getSelectedCreator(); |
||||
Component creatorContainer = XCreatorUtils.getParentXLayoutContainer(creator); |
||||
if (creatorContainer != null && creatorContainer != container |
||||
&& (selectionModel.getSelection().size() == 1 || container instanceof XWAbsoluteLayout)) { |
||||
HoverPainter painter = AdapterBus.getContainerPainter(designer, container); |
||||
designer.setPainter(painter); |
||||
if (painter != null) { |
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
rect.x -= designer.getArea().getHorizontalValue(); |
||||
rect.y -= designer.getArea().getVerticalValue(); |
||||
painter.setRenderingBounds(rect); |
||||
painter.setHotspot(new Point(e.getX(), e.getY())); |
||||
painter.setCreator(creator); |
||||
} |
||||
addable = true; |
||||
} |
||||
} |
||||
|
||||
private boolean addBean(XLayoutContainer container, int x, int y) { |
||||
LayoutAdapter adapter = container.getLayoutAdapter(); |
||||
Rectangle r = ComponentUtils.getRelativeBounds(container); |
||||
if (selectionModel.getSelection().size() == 1) { |
||||
return adapter.addBean(selectionModel.getSelection().getSelectedCreator(), x |
||||
+ designer.getArea().getHorizontalValue() - r.x, y + designer.getArea().getVerticalValue() - r.y); |
||||
} |
||||
for (XCreator creator : selectionModel.getSelection().getSelectedCreators()) { |
||||
adapter.addBean(creator, x + designer.getArea().getHorizontalValue() - r.x, y + designer.getArea().getVerticalValue()- r.y); |
||||
} |
||||
return true; |
||||
} |
||||
|
||||
private void adding(int x, int y) { |
||||
public boolean dragable() { |
||||
return ((driection != Location.outer) && !selecting); |
||||
} |
||||
|
||||
/** |
||||
* 拖拽中是否可以转换为添加模式: |
||||
* 如果拖拽组件只有一个,鼠标当前所在位置的最底层表单容器与这个组件的容器不同; |
||||
* 如果拖拽组件为多个,鼠标当前所在位置的最底层表单容器除了要求要跟这些组件的容器不同外,还必须是绝对定位布局 |
||||
*/ |
||||
private void checkAddable(MouseEvent e) { |
||||
addable = false; |
||||
designer.setPainter(null); |
||||
|
||||
if (driection != Location.inner) { |
||||
return; |
||||
} |
||||
|
||||
XCreator comp = designer.getComponentAt(e.getX(), e.getY(), selectionModel.getSelection().getSelectedCreators()); |
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(comp); |
||||
XCreator creator = selectionModel.getSelection().getSelectedCreator(); |
||||
Component creatorContainer = XCreatorUtils.getParentXLayoutContainer(creator); |
||||
if (creatorContainer != null && creatorContainer != container |
||||
&& (selectionModel.getSelection().size() == 1 || container instanceof XWAbsoluteLayout)) { |
||||
HoverPainter painter = AdapterBus.getContainerPainter(designer, container); |
||||
designer.setPainter(painter); |
||||
if (painter != null) { |
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
rect.x -= designer.getArea().getHorizontalValue(); |
||||
rect.y -= designer.getArea().getVerticalValue(); |
||||
painter.setRenderingBounds(rect); |
||||
painter.setHotspot(new Point(e.getX(), e.getY())); |
||||
painter.setCreator(creator); |
||||
} |
||||
addable = true; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* @param container 容器 |
||||
* @param mouseX 鼠标释放位置X |
||||
* @param mouseY 鼠标释放位置Y |
||||
* @return 是否成功 |
||||
*/ |
||||
private boolean addBean(XLayoutContainer container, int mouseX, int mouseY) { |
||||
LayoutAdapter adapter = container.getLayoutAdapter(); |
||||
Rectangle rectangleContainer = ComponentUtils.getRelativeBounds(container); |
||||
if (selectionModel.getSelection().size() == 1) { |
||||
return adapter.addBean(selectionModel.getSelection().getSelectedCreator(), |
||||
mouseX + designer.getArea().getHorizontalValue() - rectangleContainer.x, |
||||
mouseY + designer.getArea().getVerticalValue() - rectangleContainer.y); |
||||
} |
||||
for (XCreator creator : selectionModel.getSelection().getSelectedCreators()) { |
||||
adapter.addBean(creator, |
||||
mouseX + designer.getArea().getHorizontalValue() - rectangleContainer.x, |
||||
mouseY + designer.getArea().getVerticalValue() - rectangleContainer.y); |
||||
} |
||||
return true; |
||||
} |
||||
|
||||
/** |
||||
* @param mouseReleasedX 鼠标释放位置X |
||||
* @param mouseReleasedY 鼠标释放位置Y |
||||
*/ |
||||
private void adding(int mouseReleasedX, int mouseReleasedY) { |
||||
// 当前鼠标所在的组件
|
||||
XCreator hoveredComponent = designer.getComponentAt(x, y, selectionModel.getSelection().getSelectedCreators()); |
||||
XCreator hoveredComponent = designer.getComponentAt(mouseReleasedX, mouseReleasedY, selectionModel.getSelection().getSelectedCreators()); |
||||
|
||||
// 获取该组件所在的焦点容器
|
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(hoveredComponent); |
||||
|
||||
boolean success = false; |
||||
|
||||
if (container != null) { |
||||
// 如果是容器,则调用其acceptComponent接受组件
|
||||
success = addBean(container, x, y); |
||||
} |
||||
|
||||
if (success) { |
||||
FormSelectionUtils.rebuildSelection(designer); |
||||
designer.getEditListenerTable().fireCreatorModified( |
||||
selectionModel.getSelection().getSelectedCreator(), DesignerEvent.CREATOR_ADDED); |
||||
} else { |
||||
Toolkit.getDefaultToolkit().beep(); |
||||
} |
||||
|
||||
if (container != null) { |
||||
// 如果是容器,则调用其acceptComponent接受组件
|
||||
success = addBean(container, mouseReleasedX, mouseReleasedY); |
||||
} |
||||
|
||||
if (success) { |
||||
FormSelectionUtils.rebuildSelection(designer); |
||||
designer.getEditListenerTable().fireCreatorModified( |
||||
selectionModel.getSelection().getSelectedCreator(), DesignerEvent.CREATOR_ADDED); |
||||
} else { |
||||
selectionModel.getSelection().setSelectionBounds(selectedPositionBackup, designer); |
||||
Toolkit.getDefaultToolkit().beep(); |
||||
} |
||||
// 取消提示
|
||||
designer.setPainter(null); |
||||
} |
||||
|
||||
/** |
||||
*是否拖拽 |
||||
* 是否拖拽 |
||||
* |
||||
* @return dragging状态 |
||||
*/ |
||||
public boolean isDragging() { |
||||
return dragging; |
||||
} |
||||
public boolean isDragging() { |
||||
return dragging; |
||||
} |
||||
|
||||
/** |
||||
*是否可以开始画线 |
||||
* 是否可以开始画线 |
||||
* |
||||
* @return startPoint不为空返回true |
||||
*/ |
||||
public boolean prepareForDrawLining() { |
||||
return startPoint != null; |
||||
} |
||||
public boolean prepareForDrawLining() { |
||||
return startPoint != null; |
||||
} |
||||
|
||||
/** |
||||
*设置开始位置 |
||||
* 设置开始位置 |
||||
* |
||||
* @param p point位置 |
||||
*/ |
||||
public void setStartPoint(Point p) { |
||||
this.startPoint = p; |
||||
} |
||||
public void setStartPoint(Point p) { |
||||
this.startPoint = p; |
||||
} |
||||
|
||||
/** |
||||
*返回开始位置 |
||||
* 返回开始位置 |
||||
* |
||||
* @return 点位置 |
||||
*/ |
||||
public Point getStartPoint() { |
||||
return startPoint; |
||||
} |
||||
public Point getStartPoint() { |
||||
return startPoint; |
||||
} |
||||
|
||||
/** |
||||
*返回当前点位置 |
||||
* 返回当前点位置 |
||||
* |
||||
* @return 点位置 |
||||
*/ |
||||
public Point getEndPoint() { |
||||
return currentPoint; |
||||
} |
||||
return currentPoint; |
||||
} |
||||
|
||||
/** |
||||
*当前选中组件 |
||||
* 当前选中组件 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void startSelecting(MouseEvent e) { |
||||
selecting = true; |
||||
selectionModel.setHotspotBounds(new Rectangle()); |
||||
current_x = getMouseXY(e).x; |
||||
current_y = getMouseXY(e).y; |
||||
} |
||||
public void startSelecting(MouseEvent e) { |
||||
selecting = true; |
||||
selectionModel.setHotspotBounds(new Rectangle()); |
||||
currentX = getMouseXY(e).x; |
||||
currentY = getMouseXY(e).y; |
||||
} |
||||
|
||||
/** |
||||
*当前鼠标的xy |
||||
* 当前鼠标的xy |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void startResizing(MouseEvent e) { |
||||
if (!selectionModel.getSelection().isEmpty()) { |
||||
driection.backupBounds(designer); |
||||
} |
||||
current_x = getMouseXY(e).x; |
||||
current_y = getMouseXY(e).y; |
||||
} |
||||
public void startResizing(MouseEvent e) { |
||||
if (!selectionModel.getSelection().isEmpty()) { |
||||
driection.backupBounds(designer); |
||||
} |
||||
currentX = getMouseXY(e).x; |
||||
currentY = getMouseXY(e).y; |
||||
} |
||||
|
||||
/** |
||||
*起始点开始DrawLine |
||||
* 起始点开始DrawLine |
||||
* |
||||
* @param p 点位置 |
||||
*/ |
||||
public void startDrawLine(Point p) { |
||||
this.startPoint = p; |
||||
if(p != null) { |
||||
try { |
||||
designer.setCursor(XConnector.connectorCursor); |
||||
} catch (Exception e) { |
||||
} |
||||
} else { |
||||
designer.setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR)); |
||||
} |
||||
} |
||||
public void startDrawLine(Point p) { |
||||
this.startPoint = p; |
||||
if (p != null) { |
||||
try { |
||||
designer.setCursor(XConnector.connectorCursor); |
||||
} catch (Exception e) { |
||||
} |
||||
} else { |
||||
designer.setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR)); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*鼠标释放时所在的区域及圈中的组件 |
||||
* 鼠标释放时所在的区域及圈中的组件 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void selectCreators(MouseEvent e) { |
||||
int x = getMouseXY(e).x; |
||||
int y = getMouseXY(e).y; |
||||
public void selectCreators(MouseEvent e) { |
||||
int x = getMouseXY(e).x; |
||||
int y = getMouseXY(e).y; |
||||
|
||||
Rectangle bounds = createCurrentBounds(x, y); |
||||
Rectangle bounds = createCurrentBounds(x, y); |
||||
|
||||
if ((x != current_x) || (y != current_y)) { |
||||
selectionModel.setSelectedCreators(getHotspotCreators(bounds, designer.getRootComponent())); |
||||
} |
||||
selectionModel.setHotspotBounds(null); |
||||
} |
||||
if ((x != currentX) || (y != currentY)) { |
||||
selectionModel.setSelectedCreators(getHotspotCreators(bounds, designer.getRootComponent())); |
||||
} |
||||
selectionModel.setHotspotBounds(null); |
||||
} |
||||
|
||||
/** |
||||
*画所在区域线 |
||||
* 画所在区域线 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void drawLine(MouseEvent e) { |
||||
designer.getDrawLineHelper().setDrawLine(true); |
||||
Point p = designer.getDrawLineHelper().getNearWidgetPoint(e); |
||||
if (p != null) { |
||||
currentPoint = p; |
||||
} else { |
||||
currentPoint.x = e.getX() + designer.getArea().getHorizontalValue(); |
||||
currentPoint.y = e.getY() + designer.getArea().getVerticalValue(); |
||||
} |
||||
} |
||||
|
||||
private Rectangle createCurrentBounds(int x, int y) { |
||||
Rectangle bounds = new Rectangle(); |
||||
|
||||
bounds.x = Math.min(x, current_x); |
||||
bounds.y = Math.min(y, current_y); |
||||
bounds.width = Math.max(x, current_x) - bounds.x; |
||||
bounds.height = Math.max(y, current_y) - bounds.y; |
||||
|
||||
return bounds; |
||||
} |
||||
|
||||
private ArrayList<XCreator> getHotspotCreators(Rectangle selection, XCreator root) { |
||||
ArrayList<XCreator> creators = new ArrayList<XCreator>(); |
||||
|
||||
if (!root.isVisible() && !designer.isRoot(root)) { |
||||
return creators; |
||||
} |
||||
|
||||
if (root instanceof XLayoutContainer) { |
||||
XLayoutContainer container = (XLayoutContainer) root; |
||||
int count = container.getXCreatorCount(); |
||||
Rectangle clipped = new Rectangle(selection); |
||||
|
||||
for (int i = count - 1; i >= 0; i--) { |
||||
XCreator child = container.getXCreator(i); |
||||
|
||||
if (selection.contains(child.getBounds())) { |
||||
creators.add(child); |
||||
} else { |
||||
clipped.x = selection.x - child.getX(); |
||||
clipped.y = selection.y - child.getY(); |
||||
creators.addAll(getHotspotCreators(clipped, child)); |
||||
} |
||||
} |
||||
} |
||||
|
||||
return creators; |
||||
} |
||||
public void drawLine(MouseEvent e) { |
||||
designer.getDrawLineHelper().setDrawLine(true); |
||||
Point p = designer.getDrawLineHelper().getNearWidgetPoint(e); |
||||
if (p != null) { |
||||
currentPoint = p; |
||||
} else { |
||||
currentPoint.x = e.getX() + designer.getArea().getHorizontalValue(); |
||||
currentPoint.y = e.getY() + designer.getArea().getVerticalValue(); |
||||
} |
||||
} |
||||
|
||||
private Rectangle createCurrentBounds(int x, int y) { |
||||
Rectangle bounds = new Rectangle(); |
||||
|
||||
bounds.x = Math.min(x, currentX); |
||||
bounds.y = Math.min(y, currentY); |
||||
bounds.width = Math.max(x, currentX) - bounds.x; |
||||
bounds.height = Math.max(y, currentY) - bounds.y; |
||||
|
||||
return bounds; |
||||
} |
||||
|
||||
private ArrayList<XCreator> getHotspotCreators(Rectangle selection, XCreator root) { |
||||
ArrayList<XCreator> creators = new ArrayList<>(); |
||||
|
||||
if (!root.isVisible() && !designer.isRoot(root)) { |
||||
return creators; |
||||
} |
||||
|
||||
if (root instanceof XLayoutContainer) { |
||||
XLayoutContainer container = (XLayoutContainer) root; |
||||
int count = container.getXCreatorCount(); |
||||
Rectangle clipped = new Rectangle(selection); |
||||
|
||||
for (int i = count - 1; i >= 0; i--) { |
||||
XCreator child = container.getXCreator(i); |
||||
|
||||
if (selection.contains(child.getBounds())) { |
||||
creators.add(child); |
||||
} else { |
||||
clipped.x = selection.x - child.getX(); |
||||
clipped.y = selection.y - child.getY(); |
||||
creators.addAll(getHotspotCreators(clipped, child)); |
||||
} |
||||
} |
||||
} |
||||
|
||||
return creators; |
||||
} |
||||
|
||||
|
||||
/** |
||||
*重置model |
||||
* 重置model |
||||
*/ |
||||
public void resetModel() { |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
public void resetModel() { |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
|
||||
/** |
||||
*重置 |
||||
* 重置 |
||||
*/ |
||||
public void reset() { |
||||
driection = Location.outer; |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
public void reset() { |
||||
driection = Location.outer; |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
|
||||
/** |
||||
*取消拖拽 |
||||
* 取消拖拽 |
||||
*/ |
||||
public void draggingCancel() { |
||||
designer.repaint(); |
||||
reset(); |
||||
} |
||||
public void draggingCancel() { |
||||
designer.repaint(); |
||||
reset(); |
||||
} |
||||
|
||||
/** |
||||
*设置可拉伸方向 |
||||
* 设置可拉伸方向 |
||||
* |
||||
* @param dir 拉伸方向 |
||||
*/ |
||||
public void setDirection(Direction dir) { |
||||
if(driection != dir) { |
||||
this.driection = dir; |
||||
driection.updateCursor(designer); |
||||
} |
||||
} |
||||
public void setDirection(Direction dir) { |
||||
if (driection != dir) { |
||||
this.driection = dir; |
||||
driection.updateCursor(designer); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*x吸附线赋值 |
||||
* x吸附线赋值 |
||||
* |
||||
* @param line 线 |
||||
*/ |
||||
public void setXAbsorptionline(Absorptionline line) { |
||||
this.lineInX = line; |
||||
} |
||||
public void setXAbsorptionline(Absorptionline line) { |
||||
this.lineInX = line; |
||||
} |
||||
|
||||
/** |
||||
*y吸附线赋值 |
||||
* y吸附线赋值 |
||||
* |
||||
* @param line 线 |
||||
*/ |
||||
public void setYAbsorptionline(Absorptionline line) { |
||||
this.lineInY = line; |
||||
} |
||||
|
||||
/** |
||||
* 等距线赋值 |
||||
* |
||||
* @param line 线 |
||||
*/ |
||||
public void setYAbsorptionline(Absorptionline line) { |
||||
this.lineInY = line; |
||||
} |
||||
|
||||
/** |
||||
* 等距线赋值 |
||||
* @param line 线 |
||||
*/ |
||||
public void setEquidistantLine(Absorptionline line){ |
||||
this.lineEquidistant = line; |
||||
} |
||||
public void setEquidistantLine(Absorptionline line) { |
||||
this.lineEquidistant = line; |
||||
} |
||||
|
||||
/** |
||||
*画吸附线 |
||||
* 画吸附线 |
||||
* |
||||
* @param g Graphics类 |
||||
*/ |
||||
public void paintAbsorptionline(Graphics g) { |
||||
if(lineInX != null) { |
||||
lineInX.paint(g,designer.getArea()); |
||||
} |
||||
if(lineInY != null) { |
||||
lineInY.paint(g,designer.getArea()); |
||||
} |
||||
if(lineEquidistant != null){ |
||||
lineEquidistant.paint(g,designer.getArea()); |
||||
} |
||||
} |
||||
public void paintAbsorptionline(Graphics g) { |
||||
if (lineInX != null) { |
||||
lineInX.paint(g, designer.getArea()); |
||||
} |
||||
if (lineInY != null) { |
||||
lineInY.paint(g, designer.getArea()); |
||||
} |
||||
if (lineEquidistant != null) { |
||||
lineEquidistant.paint(g, designer.getArea()); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*拖拽 |
||||
* 拖拽 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void dragging(MouseEvent e) { |
||||
checkAddable(e); |
||||
setDependLinePainter(e); |
||||
driection.drag(getMouseXY(e).x-current_x, getMouseXY(e).y-current_y, designer); |
||||
this.dragging = true; |
||||
} |
||||
|
||||
// 拖拽时画依附线用到的painter
|
||||
private void setDependLinePainter(MouseEvent e){ |
||||
XCreator comp = designer.getComponentAt(e.getX(), e.getY(), selectionModel.getSelection().getSelectedCreators()); |
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(comp); |
||||
XCreator creator = selectionModel.getSelection().getSelectedCreator(); |
||||
HoverPainter painter = AdapterBus.getContainerPainter(designer, container); |
||||
designer.setPainter(painter); |
||||
if (painter != null) { |
||||
painter.setHotspot(new Point(e.getX(), e.getY())); |
||||
painter.setCreator(creator); |
||||
} |
||||
} |
||||
public void dragging(MouseEvent e) { |
||||
//进入dragging状态时备份组件大小和位置
|
||||
if (!dragging) { |
||||
selectedPositionBackup = selectionModel.getSelection().getRelativeBounds(); |
||||
} |
||||
checkAddable(e); |
||||
setDependLinePainter(e); |
||||
driection.drag(getMouseXY(e).x - currentX, getMouseXY(e).y - currentY, designer); |
||||
this.dragging = true; |
||||
} |
||||
|
||||
// 拖拽时画依附线用到的painter
|
||||
private void setDependLinePainter(MouseEvent e) { |
||||
XCreator comp = designer.getComponentAt(e.getX(), e.getY(), selectionModel.getSelection().getSelectedCreators()); |
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(comp); |
||||
XCreator creator = selectionModel.getSelection().getSelectedCreator(); |
||||
HoverPainter painter = AdapterBus.getContainerPainter(designer, container); |
||||
designer.setPainter(painter); |
||||
if (painter != null) { |
||||
painter.setHotspot(new Point(e.getX(), e.getY())); |
||||
painter.setCreator(creator); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*释放捕获 |
||||
* 释放捕获 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void releaseDragging(MouseEvent e) { |
||||
this.dragging = false; |
||||
if (addable) { |
||||
adding(e.getX(), e.getY()); |
||||
} else if (!selectionModel.getSelection().isEmpty()) { |
||||
selectionModel.releaseDragging(); |
||||
} |
||||
designer.repaint(); |
||||
} |
||||
public void releaseDragging(MouseEvent e) { |
||||
this.dragging = false; |
||||
if (addable) { |
||||
adding(e.getX(), e.getY()); |
||||
} else if (!selectionModel.getSelection().isEmpty()) { |
||||
selectionModel.releaseDragging(); |
||||
} |
||||
designer.repaint(); |
||||
} |
||||
|
||||
/** |
||||
*改变选择区域 |
||||
* 改变选择区域 |
||||
* |
||||
* @param e 鼠标事件 |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void changeSelection(MouseEvent e) { |
||||
Rectangle bounds = createCurrentBounds(getMouseXY(e).x, getMouseXY(e).y); |
||||
selectionModel.setHotspotBounds(bounds); |
||||
} |
||||
public void changeSelection(MouseEvent e) { |
||||
Rectangle bounds = createCurrentBounds(getMouseXY(e).x, getMouseXY(e).y); |
||||
selectionModel.setHotspotBounds(bounds); |
||||
} |
||||
|
||||
/** |
||||
*返回鼠标所在的x、y 考虑滚动条的值 |
||||
* 返回鼠标所在的x、y 考虑滚动条的值 |
||||
* |
||||
* @param e 鼠标事件 |
||||
* @return xy值 |
||||
*/ |
||||
public Point getMouseXY(MouseEvent e) { |
||||
Point p1 = new Point(e.getX() + designer.getArea().getHorizontalValue(), e.getY() |
||||
+ designer.getArea().getVerticalValue()); |
||||
return p1; |
||||
} |
||||
|
||||
public Point getMouseXY(MouseEvent e) { |
||||
Point p1 = new Point(e.getX() + designer.getArea().getHorizontalValue(), e.getY() |
||||
+ designer.getArea().getVerticalValue()); |
||||
return p1; |
||||
} |
||||
|
||||
} |
@ -1,88 +1,96 @@
|
||||
/** |
||||
* |
||||
*/ |
||||
package com.fr.design.designer.creator; |
||||
|
||||
import com.fr.form.ui.container.WLayout; |
||||
|
||||
import java.awt.*; |
||||
import java.beans.IntrospectionException; |
||||
import java.util.ArrayList; |
||||
|
||||
/** |
||||
* 一些控件专属的容器,如标题容器,sclae容器 |
||||
* @author jim |
||||
* @date 2014-11-7 |
||||
*/ |
||||
public abstract class DedicateLayoutContainer extends XLayoutContainer { |
||||
|
||||
public DedicateLayoutContainer(WLayout widget, Dimension initSize) { |
||||
super(widget, initSize); |
||||
} |
||||
|
||||
/** |
||||
* 得到属性名 |
||||
* @return 属性名 |
||||
* @throws IntrospectionException |
||||
*/ |
||||
public CRPropertyDescriptor[] supportedDescriptor() throws IntrospectionException { |
||||
return new CRPropertyDescriptor[0]; |
||||
} |
||||
|
||||
/** |
||||
* 返回容器图标 |
||||
* |
||||
* @return |
||||
*/ |
||||
@Override |
||||
public String getIconPath() { |
||||
if(this.getXCreator(XWScaleLayout.INDEX) != null){ |
||||
return this.getXCreator(XWScaleLayout.INDEX).getIconPath(); |
||||
} |
||||
return "/com/fr/web/images/form/resources/text_field_16.png"; |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 控件树不显示此组件 |
||||
* @param path 控件树list |
||||
*/ |
||||
public void notShowInComponentTree(ArrayList<Component> path) { |
||||
path.remove(path.size()-1); |
||||
} |
||||
|
||||
/** |
||||
* 重置组件的名称 |
||||
* @param name 名称 |
||||
*/ |
||||
public void resetCreatorName(String name) { |
||||
super.resetCreatorName(name); |
||||
XCreator child = getXCreator(XWScaleLayout.INDEX); |
||||
child.toData().setWidgetName(name); |
||||
} |
||||
|
||||
/** |
||||
* 返回对应属性表的组件,scale和title返回其子组件 |
||||
* @return 组件 |
||||
*/ |
||||
public XCreator getPropertyDescriptorCreator() { |
||||
return getXCreator(XWScaleLayout.INDEX); |
||||
} |
||||
|
||||
/** |
||||
* 是否作为控件树的叶子节点 |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean isComponentTreeLeaf() { |
||||
return true; |
||||
} |
||||
|
||||
/** |
||||
* 是否为sclae和title专属容器 |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean isDedicateContainer() { |
||||
return true; |
||||
} |
||||
|
||||
/** |
||||
* |
||||
*/ |
||||
package com.fr.design.designer.creator; |
||||
|
||||
import com.fr.form.ui.container.WLayout; |
||||
|
||||
import java.awt.*; |
||||
import java.beans.IntrospectionException; |
||||
import java.util.ArrayList; |
||||
|
||||
/** |
||||
* 一些控件专属的容器,如标题容器,sclae容器 |
||||
* |
||||
* @author jim |
||||
* @date 2014-11-7 |
||||
*/ |
||||
public abstract class DedicateLayoutContainer extends XLayoutContainer { |
||||
|
||||
public DedicateLayoutContainer(WLayout widget, Dimension initSize) { |
||||
super(widget, initSize); |
||||
} |
||||
|
||||
/** |
||||
* 得到属性名 |
||||
* |
||||
* @return 属性名 |
||||
* @throws IntrospectionException |
||||
*/ |
||||
public CRPropertyDescriptor[] supportedDescriptor() throws IntrospectionException { |
||||
return new CRPropertyDescriptor[0]; |
||||
} |
||||
|
||||
/** |
||||
* 返回容器图标 |
||||
* |
||||
* @return |
||||
*/ |
||||
@Override |
||||
public String getIconPath() { |
||||
if (this.getXCreator(XWScaleLayout.INDEX) != null) { |
||||
return this.getXCreator(XWScaleLayout.INDEX).getIconPath(); |
||||
} |
||||
return "/com/fr/web/images/form/resources/text_field_16.png"; |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 控件树不显示此组件 |
||||
* |
||||
* @param path 控件树list |
||||
*/ |
||||
public void notShowInComponentTree(ArrayList<Component> path) { |
||||
path.remove(path.size() - 1); |
||||
} |
||||
|
||||
/** |
||||
* 重置组件的名称 |
||||
* |
||||
* @param name 名称 |
||||
*/ |
||||
public void resetCreatorName(String name) { |
||||
super.resetCreatorName(name); |
||||
XCreator child = getXCreator(XWScaleLayout.INDEX); |
||||
//实现WTitleLayout的SetWidgetName
|
||||
child.toData().setWidgetName(name); |
||||
} |
||||
|
||||
/** |
||||
* 返回对应属性表的组件,scale和title返回其子组件 |
||||
* |
||||
* @return 组件 |
||||
*/ |
||||
public XCreator getPropertyDescriptorCreator() { |
||||
return getXCreator(XWScaleLayout.INDEX); |
||||
} |
||||
|
||||
/** |
||||
* 是否作为控件树的叶子节点 |
||||
* |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean isComponentTreeLeaf() { |
||||
return true; |
||||
} |
||||
|
||||
/** |
||||
* 是否为sclae和title专属容器 |
||||
* |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean isDedicateContainer() { |
||||
return true; |
||||
} |
||||
|
||||
} |
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,481 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.design.actions.UpdateAction; |
||||
import com.fr.design.dialog.BasicDialog; |
||||
import com.fr.design.dialog.BasicPane; |
||||
import com.fr.design.dialog.DialogActionAdapter; |
||||
import com.fr.design.gui.ibutton.UIButton; |
||||
import com.fr.design.gui.icheckbox.UICheckBox; |
||||
import com.fr.design.gui.ilable.UILabel; |
||||
import com.fr.design.gui.itextfield.UITextField; |
||||
import com.fr.design.gui.itoolbar.UIToolbar; |
||||
import com.fr.design.javascript.JavaScriptActionPane; |
||||
import com.fr.design.layout.FRGUIPaneFactory; |
||||
import com.fr.design.layout.TableLayout; |
||||
import com.fr.design.layout.TableLayoutHelper; |
||||
import com.fr.design.menu.ToolBarDef; |
||||
import com.fr.design.style.background.BackgroundPane; |
||||
import com.fr.design.utils.gui.GUICoreUtils; |
||||
import com.fr.design.widget.IconDefinePane; |
||||
import com.fr.form.ui.*; |
||||
import com.fr.form.ui.Button; |
||||
import com.fr.form.web.button.Export; |
||||
import com.fr.general.Background; |
||||
import com.fr.general.IOUtils; |
||||
import com.fr.general.Inter; |
||||
import com.fr.stable.StringUtils; |
||||
|
||||
import javax.swing.*; |
||||
import javax.swing.event.ListSelectionEvent; |
||||
import javax.swing.event.ListSelectionListener; |
||||
import java.awt.*; |
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.ActionListener; |
||||
import java.util.*; |
||||
|
||||
/** |
||||
* Created by harry on 2017-3-2. |
||||
*/ |
||||
public class FormEditToolBar extends BasicPane { |
||||
|
||||
private JList list; |
||||
private DefaultListModel listModel; |
||||
private JPanel right; |
||||
private CardLayout card; |
||||
private ButtonPane bp; |
||||
private FormToolBarButton lastButton; |
||||
private Background background = null; |
||||
private UICheckBox defaultCheckBox; |
||||
|
||||
private ListSelectionListener listSelectionListener = new ListSelectionListener() { |
||||
public void valueChanged(ListSelectionEvent evt) { |
||||
if (lastButton != null) { |
||||
lastButton.setWidget(bp.update()); |
||||
} |
||||
if (list.getSelectedValue() instanceof FormToolBarButton) { |
||||
lastButton = (FormToolBarButton) list.getSelectedValue(); |
||||
if (lastButton.getWidget() instanceof Button) { |
||||
card.show(right, "button"); |
||||
bp.populate(lastButton.getWidget()); |
||||
} else { |
||||
bp.populate(lastButton.getWidget()); |
||||
card.show(right, "none"); |
||||
} |
||||
} |
||||
} |
||||
}; |
||||
|
||||
|
||||
private ActionListener actioner = new ActionListener() { |
||||
/** |
||||
* |
||||
*/ |
||||
public void actionPerformed(ActionEvent arg0) { |
||||
final BackgroundPane backgroundPane = new BackgroundPane(); |
||||
BasicDialog dialog = backgroundPane.showWindow(DesignerContext.getDesignerFrame()); |
||||
backgroundPane.populate(FormEditToolBar.this.background); |
||||
dialog.addDialogActionListener(new DialogActionAdapter() { |
||||
public void doOk() { |
||||
FormEditToolBar.this.background = backgroundPane.update(); |
||||
if (FormEditToolBar.this.background != null) { |
||||
FormEditToolBar.this.defaultCheckBox.setSelected(false); |
||||
} |
||||
} |
||||
}); |
||||
dialog.setVisible(true); |
||||
} |
||||
}; |
||||
|
||||
public FormEditToolBar() { |
||||
initComponent(); |
||||
} |
||||
|
||||
/** |
||||
* 初始化 |
||||
*/ |
||||
public void initComponent() { |
||||
this.setLayout(FRGUIPaneFactory.createBorderLayout()); |
||||
JPanel left = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
listModel = new DefaultListModel(); |
||||
list = new JList(listModel); |
||||
list.setCellRenderer(render); |
||||
left.add(new JScrollPane(list), BorderLayout.CENTER); |
||||
if (listModel.getSize() > 0) { |
||||
list.setSelectedIndex(0); |
||||
} |
||||
|
||||
ToolBarDef toolbarDef = new ToolBarDef(); |
||||
toolbarDef.addShortCut(new MoveUpItemAction()); |
||||
toolbarDef.addShortCut(new MoveDownItemAction()); |
||||
toolbarDef.addShortCut(new RemoveAction()); |
||||
UIToolbar toolBar = ToolBarDef.createJToolBar(); |
||||
toolbarDef.updateToolBar(toolBar); |
||||
left.add(toolBar, BorderLayout.NORTH); |
||||
|
||||
right = FRGUIPaneFactory.createCardLayout_S_Pane(); |
||||
card = new CardLayout(); |
||||
right.setLayout(card); |
||||
bp = new ButtonPane(); |
||||
right.add("none", FRGUIPaneFactory.createNormalFlowInnerContainer_S_Pane()); |
||||
right.add("button", bp); |
||||
|
||||
JSplitPane splitPane = new JSplitPane(JSplitPane.HORIZONTAL_SPLIT, true, left, right); |
||||
// splitPane.setDividerLocation(left.getMinimumSize().width);
|
||||
splitPane.setDividerLocation(120); |
||||
this.add(splitPane); |
||||
list.addListSelectionListener(listSelectionListener); |
||||
JPanel backgroundPane = FRGUIPaneFactory.createNormalFlowInnerContainer_S_Pane(); |
||||
UIButton bgButton = new UIButton(Inter.getLocText("FR-Designer-Output_Background_Set")); |
||||
defaultCheckBox = new UICheckBox(Inter.getLocText("FR-Designer-Output_Default_Background")); |
||||
bgButton.addActionListener(actioner); |
||||
backgroundPane.add(defaultCheckBox); |
||||
backgroundPane.add(bgButton); |
||||
backgroundPane.setBorder(BorderFactory.createTitledBorder(Inter.getLocText("FR-Designer-Output_Background_Set"))); |
||||
this.add(backgroundPane, BorderLayout.SOUTH); |
||||
} |
||||
|
||||
ListCellRenderer render = new DefaultListCellRenderer() { |
||||
public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected, boolean cellHasFocus) { |
||||
super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus); |
||||
|
||||
if (value instanceof FormToolBarButton) { |
||||
FormToolBarButton button = (FormToolBarButton) value; |
||||
this.setText(button.getNameOption().optionName()); |
||||
this.setIcon(button.getNameOption().optionIcon()); |
||||
} |
||||
return this; |
||||
} |
||||
}; |
||||
|
||||
@Override |
||||
protected String title4PopupWindow() { |
||||
return Inter.getLocText("FR-Designer_Edit"); |
||||
} |
||||
|
||||
public void populate(FormToolBar ftoolbar) { |
||||
this.populate(ftoolbar, null); |
||||
} |
||||
|
||||
public void populate(FormToolBar ftoolbar, FormToolBarButton button) { |
||||
if (ftoolbar == null) { |
||||
return; |
||||
} |
||||
for (int i = 0; i < ftoolbar.getButtonlist().size(); i++) { |
||||
listModel.addElement(ftoolbar.getButtonlist().get(i)); |
||||
} |
||||
this.list.validate(); |
||||
this.list.repaint(); |
||||
if (ftoolbar.getButtonlist().size() > 0) { |
||||
this.list.setSelectedIndex(0); |
||||
} |
||||
if (button != null) { |
||||
this.list.setSelectedValue(button, true); |
||||
} |
||||
this.background = ftoolbar.getBackground(); |
||||
|
||||
this.defaultCheckBox.setSelected(ftoolbar.isDefault() ? true : false); |
||||
} |
||||
|
||||
public FormToolBar update() { |
||||
if (this.list.getSelectedIndex() > -1) { |
||||
for (int i = 0; i < listModel.getSize(); i++) { |
||||
this.list.setSelectedIndex(i); |
||||
FormToolBarButton toolBarButton = (FormToolBarButton) this.list.getSelectedValue(); |
||||
Widget widget = this.bp.update(); |
||||
toolBarButton.setWidget(widget); |
||||
if (widget instanceof Button) { |
||||
String iconname = ((Button) widget).getIconName(); |
||||
if (StringUtils.isNotBlank(iconname)) { |
||||
Image iimage = WidgetManager.getProviderInstance().getIconManager().getIconImage(iconname); |
||||
toolBarButton.setIcon(new ImageIcon(iimage)); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
java.util.List<FormToolBarButton> list = new ArrayList<FormToolBarButton>(); |
||||
for (int i = 0; i < listModel.size(); i++) { |
||||
list.add((FormToolBarButton) listModel.get(i)); |
||||
} |
||||
FormToolBar ftoolBar = new FormToolBar(); |
||||
ftoolBar.setButtonlist(list); |
||||
|
||||
ftoolBar.setDefault(this.defaultCheckBox.isSelected()); |
||||
if (!ftoolBar.isDefault()) { |
||||
ftoolBar.setBackground(this.background); |
||||
} |
||||
return ftoolBar; |
||||
} |
||||
|
||||
private class MoveUpItemAction extends UpdateAction { |
||||
public MoveUpItemAction() { |
||||
this.setName(Inter.getLocText("Utils-Move_Up")); |
||||
this.setMnemonic('U'); |
||||
this.setSmallIcon(IOUtils.readIcon("/com/fr/design/images/control/up.png")); |
||||
} |
||||
|
||||
/** |
||||
* |
||||
*/ |
||||
public void actionPerformed(ActionEvent evt) { |
||||
int selectedIndex = list.getSelectedIndex(); |
||||
if (selectedIndex == -1) { |
||||
return; |
||||
} |
||||
|
||||
// 上移
|
||||
if (selectedIndex > 0) { |
||||
DefaultListModel listModel = (DefaultListModel) list.getModel(); |
||||
|
||||
Object selecteObj1 = listModel.get(selectedIndex - 1); |
||||
listModel.set(selectedIndex - 1, listModel.get(selectedIndex)); |
||||
listModel.set(selectedIndex, selecteObj1); |
||||
|
||||
list.setSelectedIndex(selectedIndex - 1); |
||||
list.ensureIndexIsVisible(selectedIndex - 1); |
||||
list.validate(); |
||||
} |
||||
} |
||||
} |
||||
|
||||
private class MoveDownItemAction extends UpdateAction { |
||||
public MoveDownItemAction() { |
||||
this.setName(Inter.getLocText("Utils-Move_Down")); |
||||
this.setMnemonic('D'); |
||||
this.setSmallIcon(IOUtils.readIcon("/com/fr/design/images/control/down.png")); |
||||
} |
||||
|
||||
/** |
||||
* |
||||
*/ |
||||
public void actionPerformed(ActionEvent evt) { |
||||
int selectedIndex = list.getSelectedIndex(); |
||||
if (selectedIndex == -1) { |
||||
return; |
||||
} |
||||
|
||||
// 下移
|
||||
if (selectedIndex == -1) { |
||||
return; |
||||
} |
||||
|
||||
if (selectedIndex < list.getModel().getSize() - 1) { |
||||
DefaultListModel listModel = (DefaultListModel) list.getModel(); |
||||
|
||||
Object selecteObj1 = listModel.get(selectedIndex + 1); |
||||
listModel.set(selectedIndex + 1, listModel.get(selectedIndex)); |
||||
listModel.set(selectedIndex, selecteObj1); |
||||
|
||||
list.setSelectedIndex(selectedIndex + 1); |
||||
list.ensureIndexIsVisible(selectedIndex + 1); |
||||
list.validate(); |
||||
} |
||||
} |
||||
} |
||||
|
||||
public class RemoveAction extends UpdateAction { |
||||
public RemoveAction() { |
||||
this.setName(Inter.getLocText("FR-Designer_Delete")); |
||||
this.setSmallIcon(IOUtils.readIcon("/com/fr/base/images/cell/control/remove.png")); |
||||
} |
||||
|
||||
/** |
||||
* 动作 |
||||
* @param e 事件 |
||||
*/ |
||||
public void actionPerformed(ActionEvent e) { |
||||
int i = list.getSelectedIndex(); |
||||
if (i < 0 || !(listModel.getElementAt(i) instanceof FormToolBarButton)) { |
||||
return; |
||||
} |
||||
int val = JOptionPane.showConfirmDialog(FormEditToolBar.this, Inter.getLocText("FR-Designer_Are_You_Sure_To_Delete_The_Data") + "?", "Message", JOptionPane.YES_NO_OPTION); |
||||
if (val != JOptionPane.YES_OPTION) { |
||||
return; |
||||
} |
||||
listModel.removeElementAt(i); |
||||
list.validate(); |
||||
if (listModel.size() > 0) { |
||||
list.setSelectedIndex(0); |
||||
} else { |
||||
card.show(right, "none"); |
||||
} |
||||
} |
||||
} |
||||
|
||||
public class ButtonPane extends BasicPane { |
||||
private CardLayout card; |
||||
private JPanel centerPane; |
||||
private UICheckBox icon, text, excelP, excelO; |
||||
private Widget widget; |
||||
private UITextField nameField; |
||||
private IconDefinePane iconPane; |
||||
private UIButton button; |
||||
private JavaScriptActionPane javaScriptPane; |
||||
|
||||
public ButtonPane() { |
||||
this.initComponents(); |
||||
} |
||||
|
||||
/** |
||||
* 初始化元素 |
||||
*/ |
||||
public void initComponents() { |
||||
this.setLayout(FRGUIPaneFactory.createBorderLayout()); |
||||
JPanel north = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
icon = new UICheckBox(Inter.getLocText("FR-Designer_Show_Icon")); |
||||
text = new UICheckBox(Inter.getLocText("FR-Designer_Show_Text")); |
||||
|
||||
north.add(icon, BorderLayout.NORTH); |
||||
north.add(text, BorderLayout.CENTER); |
||||
|
||||
nameField = new UITextField(8); |
||||
iconPane = new IconDefinePane(); |
||||
javaScriptPane = JavaScriptActionPane.createDefault(); |
||||
|
||||
double p = TableLayout.PREFERRED; |
||||
double rowSize[] = {p, p}; |
||||
double columnSize[] = {p, p}; |
||||
|
||||
Component[][] coms = new Component[][]{{new UILabel(Inter.getLocText(new String[]{"Widget", "Printer-Alias"}) + ":"), nameField}, {new UILabel(Inter.getLocText(new String[]{"Widget", "Icon"}) + ":"), iconPane}}; |
||||
|
||||
JPanel nameIconPane = TableLayoutHelper.createTableLayoutPane(coms, rowSize, columnSize); |
||||
|
||||
north.add(nameIconPane, BorderLayout.SOUTH); |
||||
|
||||
north.setBorder(BorderFactory.createTitledBorder(Inter.getLocText(new String[]{"Form-Button", "Property", "Set"}))); |
||||
this.add(north, BorderLayout.NORTH); |
||||
JPanel none = FRGUIPaneFactory.createNormalFlowInnerContainer_S_Pane(); |
||||
centerPane = FRGUIPaneFactory.createCardLayout_S_Pane(); |
||||
card = new CardLayout(); |
||||
centerPane.setLayout(card); |
||||
centerPane.add("custom", getCustomPane()); |
||||
centerPane.add("export", getExport()); |
||||
centerPane.add("none", none); |
||||
|
||||
this.add(centerPane, BorderLayout.CENTER); |
||||
} |
||||
|
||||
|
||||
private JPanel getCustomPane() { |
||||
JPanel customPane = FRGUIPaneFactory.createCenterFlowInnerContainer_S_Pane(); |
||||
|
||||
button = new UIButton(Inter.getLocText("FR-Designer_User_Defined_Event")); |
||||
customPane.add(button); |
||||
customPane.setBorder(GUICoreUtils.createTitledBorder(Inter.getLocText("FR-Designer_Edit") + "JS", null)); |
||||
button.addActionListener(l); |
||||
return customPane; |
||||
} |
||||
|
||||
private JPanel getExport() { |
||||
JPanel export = FRGUIPaneFactory.createY_AXISBoxInnerContainer_L_Pane(); |
||||
// export.setLayout(new BoxLayout(export, BoxLayout.Y_AXIS));
|
||||
excelP = new UICheckBox(Inter.getLocText("FR-Designer-Output_Excel_Page")); |
||||
excelO = new UICheckBox(Inter.getLocText("FR-Designer-Output_Excel_Simple")); |
||||
export.add(excelP); |
||||
export.add(Box.createVerticalStrut(2)); |
||||
export.add(excelO); |
||||
export.add(Box.createVerticalStrut(2)); |
||||
|
||||
export.setBorder(GUICoreUtils.createTitledBorder(Inter.getLocText(new String[]{"Form-Button", "Property", "Set"}), null)); |
||||
return export; |
||||
} |
||||
|
||||
@Override |
||||
protected String title4PopupWindow() { |
||||
return "Button"; |
||||
} |
||||
|
||||
ActionListener l = new ActionListener() { |
||||
public void actionPerformed(ActionEvent e) { |
||||
if (!(widget instanceof CustomToolBarButton)) { |
||||
return; |
||||
} |
||||
if (javaScriptPane == null || ((CustomToolBarButton)widget).getJSImpl() == null) { |
||||
javaScriptPane = JavaScriptActionPane.createDefault(); |
||||
} |
||||
javaScriptPane.setPreferredSize(new Dimension(750, 500)); |
||||
BasicDialog dialog = javaScriptPane.showWindow(SwingUtilities.getWindowAncestor(ButtonPane.this)); |
||||
dialog.addDialogActionListener(new DialogActionAdapter() { |
||||
@Override |
||||
public void doOk() { |
||||
((CustomToolBarButton) widget).setJSImpl(javaScriptPane.updateBean()); |
||||
} |
||||
}); |
||||
dialog.setVisible(true); |
||||
} |
||||
}; |
||||
|
||||
/** |
||||
* 更新 |
||||
* @param widget 对应组件 |
||||
*/ |
||||
public void populate(Widget widget) { |
||||
this.widget = widget; |
||||
card.show(centerPane, "none"); |
||||
if (widget instanceof Button) { |
||||
populateDefault(); |
||||
} |
||||
if (widget instanceof Export) { |
||||
populateExport(); |
||||
} else if (widget instanceof CustomToolBarButton) { |
||||
populateCustomToolBarButton(); |
||||
} |
||||
} |
||||
|
||||
private void populateExport(){ |
||||
card.show(centerPane, "export"); |
||||
Export export = (Export) widget; |
||||
this.excelP.setSelected(export.isExcelPAvailable()); |
||||
this.excelO.setSelected(export.isExcelOAvailable()); |
||||
} |
||||
|
||||
private void populateCustomToolBarButton(){ |
||||
card.show(centerPane, "custom"); |
||||
CustomToolBarButton customToolBarButton = (CustomToolBarButton) widget; |
||||
if (customToolBarButton.getJSImpl() != null) { |
||||
this.javaScriptPane.populateBean(customToolBarButton.getJSImpl()); |
||||
} |
||||
} |
||||
|
||||
|
||||
|
||||
private void populateDefault(){ |
||||
Button button = (Button) widget; |
||||
this.icon.setSelected(button.isShowIcon()); |
||||
this.text.setSelected(button.isShowText()); |
||||
this.nameField.setText(button.getText()); |
||||
this.iconPane.populate(((Button) widget).getIconName()); |
||||
} |
||||
|
||||
/** |
||||
* 更新 |
||||
* |
||||
* @return 对应组件 |
||||
*/ |
||||
public Widget update() { |
||||
if (widget instanceof Export) { |
||||
updateExport(); |
||||
} else if (widget instanceof CustomToolBarButton) { |
||||
((CustomToolBarButton) widget).setJSImpl(this.javaScriptPane.updateBean()); |
||||
} |
||||
if (widget instanceof Button) { |
||||
updateDefault(); |
||||
} |
||||
|
||||
return widget; |
||||
} |
||||
|
||||
private void updateDefault(){ |
||||
((Button) widget).setShowIcon(this.icon.isSelected()); |
||||
((Button) widget).setShowText(this.text.isSelected()); |
||||
((Button) widget).setText(this.nameField.getText()); |
||||
((Button) widget).setIconName(this.iconPane.update()); |
||||
} |
||||
|
||||
|
||||
private void updateExport(){ |
||||
Export export = (Export) widget; |
||||
export.setExcelPAvailable(this.excelP.isSelected()); |
||||
export.setExcelOAvailable(this.excelO.isSelected()); |
||||
} |
||||
} |
||||
} |
@ -1,126 +1,253 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import java.awt.Component; |
||||
import java.awt.Rectangle; |
||||
import java.awt.Toolkit; |
||||
import java.util.ArrayList; |
||||
import java.util.Arrays; |
||||
import java.util.List; |
||||
|
||||
import com.fr.base.FRContext; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.design.designer.beans.LayoutAdapter; |
||||
import com.fr.design.designer.beans.events.DesignerEvent; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XCreatorUtils; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWAbsoluteLayout; |
||||
import com.fr.form.main.ClonedWidgetCreator; |
||||
import com.fr.form.ui.Widget; |
||||
|
||||
public class FormSelectionUtils { |
||||
|
||||
public static void paste2Container(FormDesigner designer, XLayoutContainer parent, FormSelection selection, int x, |
||||
int y) { |
||||
LayoutAdapter adapter = parent.getLayoutAdapter(); |
||||
if (selection.size() == 1) { |
||||
try { |
||||
XCreator creator = selection.getSelectedCreator(); |
||||
Widget cloned = new ClonedWidgetCreator(designer.getTarget()).clonedWidgetWithNoRepeatName(creator |
||||
.toData()); |
||||
XCreator clondCreator = XCreatorUtils.createXCreator(cloned, creator.getSize()); |
||||
if (adapter.addBean(clondCreator, x + clondCreator.getWidth() / 2, y + clondCreator.getHeight() / 2)) { |
||||
designer.getSelectionModel().getSelection().setSelectedCreator(clondCreator); |
||||
designer.getEditListenerTable().fireCreatorModified(clondCreator, DesignerEvent.CREATOR_PASTED); |
||||
return; |
||||
} |
||||
} catch (CloneNotSupportedException e) { |
||||
FRContext.getLogger().error(e.getMessage(), e); |
||||
} |
||||
} else if (selection.size() > 1) { |
||||
if (parent instanceof XWAbsoluteLayout) { |
||||
designer.getSelectionModel().getSelection().reset(); |
||||
Rectangle rec = selection.getSelctionBounds(); |
||||
for (XCreator creator : selection.getSelectedCreators()) { |
||||
try { |
||||
Widget cloned = new ClonedWidgetCreator(designer.getTarget()) |
||||
.clonedWidgetWithNoRepeatName(creator.toData()); |
||||
XCreator clondCreator = XCreatorUtils.createXCreator(cloned, creator.getSize()); |
||||
// 设置位置,移动20x20,防止被粘帖的组件重叠,照顾表单布局情况下
|
||||
adapter.addBean(clondCreator, x + creator.getX() - rec.x + clondCreator.getWidth() / 2, y |
||||
+ creator.getY() - rec.y + clondCreator.getHeight() / 2); |
||||
designer.getSelectionModel().getSelection().addSelectedCreator(clondCreator); |
||||
} catch (CloneNotSupportedException e) { |
||||
FRContext.getLogger().error(e.getMessage(), e); |
||||
} |
||||
} |
||||
designer.getEditListenerTable().fireCreatorModified( |
||||
designer.getSelectionModel().getSelection().getSelectedCreator(), DesignerEvent.CREATOR_PASTED); |
||||
return; |
||||
} |
||||
} |
||||
Toolkit.getDefaultToolkit().beep(); |
||||
} |
||||
|
||||
public static void rebuildSelection(FormDesigner designer) { |
||||
ArrayList<XCreator> newSelection = new ArrayList<XCreator>(); |
||||
List<Widget> widgetList = new ArrayList<Widget>(); |
||||
for (XCreator comp : designer.getSelectionModel().getSelection().getSelectedCreators()) { |
||||
widgetList.add(comp.toData()); |
||||
} |
||||
designer.getSelectionModel().setSelectedCreators( |
||||
rebuildSelection(designer.getRootComponent(), widgetList, newSelection)); |
||||
} |
||||
|
||||
public static ArrayList<XCreator> rebuildSelection(XCreator rootComponent, Widget[] selectWidgets) { |
||||
List<Widget> selectionWidget = new ArrayList<Widget>(); |
||||
if(selectWidgets != null){ |
||||
selectionWidget.addAll(Arrays.asList(selectWidgets)); |
||||
} |
||||
return FormSelectionUtils.rebuildSelection(rootComponent, selectionWidget, new ArrayList<XCreator>()); |
||||
} |
||||
|
||||
private static ArrayList<XCreator> rebuildSelection(XCreator rootComponent, List<Widget> selectionWidget, |
||||
ArrayList<XCreator> newSelection) { |
||||
FormSelectionUtils._rebuild(rootComponent, selectionWidget, newSelection); |
||||
if (newSelection.isEmpty()) { |
||||
newSelection.add(rootComponent); |
||||
} |
||||
return newSelection; |
||||
} |
||||
|
||||
private static void _rebuild(XCreator root, List<Widget> selectionWidget, List<XCreator> newSelection) { |
||||
if (selectionWidget.isEmpty()) { |
||||
return; |
||||
} |
||||
for (Widget x : selectionWidget) { |
||||
if (ComparatorUtils.equals(x, root.toData())) { |
||||
if (!newSelection.contains(root)) { |
||||
newSelection.add(root); |
||||
selectionWidget.remove(x); |
||||
} |
||||
break; |
||||
} |
||||
} |
||||
|
||||
int count = root.getComponentCount(); |
||||
for (int i = 0; i < count && !selectionWidget.isEmpty(); i++) { |
||||
Component c = root.getComponent(i); |
||||
if (c instanceof XCreator) { |
||||
XCreator creator = (XCreator) c; |
||||
for (Widget x : selectionWidget) { |
||||
if (ComparatorUtils.equals(x, creator.toData())) { |
||||
newSelection.add(creator); |
||||
selectionWidget.remove(x); |
||||
break; |
||||
} |
||||
} |
||||
if (c instanceof XLayoutContainer) { |
||||
_rebuild((XLayoutContainer) c, selectionWidget, newSelection); |
||||
} else { |
||||
continue; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.base.FRContext; |
||||
import com.fr.design.designer.beans.LayoutAdapter; |
||||
import com.fr.design.designer.beans.adapters.layout.AbstractLayoutAdapter; |
||||
import com.fr.design.designer.beans.adapters.layout.FRTabFitLayoutAdapter; |
||||
import com.fr.design.designer.beans.events.DesignerEvent; |
||||
import com.fr.design.designer.creator.*; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.form.ui.container.WTitleLayout; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.general.Inter; |
||||
|
||||
import java.awt.*; |
||||
import java.util.ArrayList; |
||||
import java.util.Arrays; |
||||
import java.util.List; |
||||
|
||||
public class FormSelectionUtils { |
||||
|
||||
//组件复制时坐标偏移
|
||||
private static final int DELAY_X_Y = 20; |
||||
|
||||
//组件重命名后缀
|
||||
private static final String POSTFIX = "_c"; |
||||
|
||||
private FormSelectionUtils() { |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 粘贴到容器 |
||||
*/ |
||||
public static void paste2Container(FormDesigner designer, XLayoutContainer parent, |
||||
FormSelection clipboard, int x, int y) { |
||||
LayoutAdapter adapter = parent.getLayoutAdapter(); |
||||
if (parent instanceof XWAbsoluteLayout) { |
||||
//绝对布局
|
||||
absolutePaste(designer, clipboard, adapter, x, y); |
||||
return; |
||||
} else if (parent instanceof XWFitLayout) { |
||||
//相对布局
|
||||
relativePaste(designer, clipboard, adapter, x, y); |
||||
return; |
||||
} |
||||
Toolkit.getDefaultToolkit().beep(); |
||||
} |
||||
|
||||
/** |
||||
* 绝对布局粘贴 |
||||
*/ |
||||
private static void absolutePaste(FormDesigner designer, FormSelection clipboard, LayoutAdapter adapter, int x, int y) { |
||||
|
||||
designer.getSelectionModel().getSelection().reset(); |
||||
Rectangle rec = clipboard.getSelctionBounds(); |
||||
for (XCreator creator : clipboard.getSelectedCreators()) { |
||||
try { |
||||
Widget copied = copyWidget(designer, creator); |
||||
XCreator copiedCreator = XCreatorUtils.createXCreator(copied, creator.getSize()); |
||||
// 获取位置
|
||||
Point point = getPasteLocation((AbstractLayoutAdapter) adapter, |
||||
copiedCreator, |
||||
x + creator.getX() - rec.x + copiedCreator.getWidth() / 2, |
||||
y + creator.getY() - rec.y + copiedCreator.getHeight() / 2); |
||||
if (!adapter.accept(copiedCreator, point.x, point.y)) { |
||||
designer.showMessageDialog(Inter.getLocText("FR-Designer_Too_Large_To_Paste")); |
||||
return; |
||||
} |
||||
boolean addSuccess = adapter.addBean(copiedCreator, point.x, point.y); |
||||
if (addSuccess) { |
||||
designer.getSelectionModel().getSelection().addSelectedCreator(copiedCreator); |
||||
} |
||||
} catch (CloneNotSupportedException e) { |
||||
FRContext.getLogger().error(e.getMessage(), e); |
||||
} |
||||
} |
||||
rebuildSelection(designer); |
||||
designer.getEditListenerTable().fireCreatorModified( |
||||
designer.getSelectionModel().getSelection().getSelectedCreator(), DesignerEvent.CREATOR_PASTED); |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 相对布局粘贴 |
||||
*/ |
||||
private static void relativePaste(FormDesigner designer, FormSelection clipboard, LayoutAdapter adapter, int x, int y) { |
||||
|
||||
//@see FRTabFitLayoutAdapter
|
||||
Rectangle tabContainerRect = ComponentUtils.getRelativeBounds(designer.getSelectionModel().getSelection() |
||||
.getSelectedCreator().getParent()); |
||||
|
||||
designer.getSelectionModel().getSelection().reset(); |
||||
for (XCreator creator : clipboard.getSelectedCreators()) { |
||||
try { |
||||
Widget copied = copyWidget(designer, creator); |
||||
XCreator copiedCreator = XCreatorUtils.createXCreator(copied, creator.getSize()); |
||||
if (adapter.getClass().equals(FRTabFitLayoutAdapter.class)) { |
||||
if (!adapter.accept(copiedCreator, x - tabContainerRect.x, y - tabContainerRect.y)) { |
||||
designer.showMessageDialog(Inter.getLocText("FR-Designer_Too_Small_To_Paste")); |
||||
return; |
||||
} |
||||
} else { |
||||
if (!adapter.accept(copiedCreator, x, y)) { |
||||
designer.showMessageDialog(Inter.getLocText("FR-Designer_Too_Small_To_Paste")); |
||||
return; |
||||
} |
||||
} |
||||
boolean addSuccess = adapter.addBean(copiedCreator, x, y); |
||||
if (addSuccess) { |
||||
designer.getSelectionModel().getSelection().addSelectedCreator(copiedCreator); |
||||
} |
||||
} catch (CloneNotSupportedException e) { |
||||
FRContext.getLogger().error(e.getMessage(), e); |
||||
} |
||||
} |
||||
rebuildSelection(designer); |
||||
designer.getEditListenerTable().fireCreatorModified( |
||||
designer.getSelectionModel().getSelection().getSelectedCreator(), DesignerEvent.CREATOR_PASTED); |
||||
} |
||||
|
||||
/** |
||||
* 组件复用绝对布局获取粘贴组件位置 |
||||
*/ |
||||
private static Point getPasteLocation(AbstractLayoutAdapter layoutAdapter, XCreator copiedCreator, int x, int y) { |
||||
//当宽度为奇数时 设置偏移
|
||||
int xoffset = (copiedCreator.getWidth() & 1) == 1 ? 1 : 0; |
||||
//当高度为奇数时 设置偏移
|
||||
int yoffset = (copiedCreator.getHeight() & 1) == 1 ? 1 : 0; |
||||
|
||||
if (!layoutAdapter.accept(copiedCreator, x, y)) { |
||||
XLayoutContainer container = layoutAdapter.getContainer(); |
||||
boolean xOut = x < 0 || x + copiedCreator.getWidth() / 2 + xoffset > container.getWidth(); |
||||
boolean yOut = y < 0 || y + copiedCreator.getHeight() / 2 + yoffset > container.getHeight(); |
||||
/* |
||||
* 组件原始位置位于布局的右下角, |
||||
* 和布局右下边界线紧挨, |
||||
* 粘贴时组件在原始位置向左错开20像素。 |
||||
* x,y同时越界 |
||||
*/ |
||||
if (xOut && yOut) { |
||||
//向左偏移
|
||||
x = container.getWidth() - copiedCreator.getWidth() / 2 - DELAY_X_Y - xoffset; |
||||
//紧贴下边界
|
||||
y = container.getHeight() - copiedCreator.getHeight() / 2 - yoffset; |
||||
return new Point(x, y); |
||||
} |
||||
/* |
||||
* 组件原始位置与布局边界距离小于20像素(下边界&右边界同时小于或者任意一个边界小于), |
||||
* 则粘贴时距离小于20像素一侧直接贴近布局边界, |
||||
* 距离大于20像素的一侧正常错开。 |
||||
* x,y中只有一个越界 |
||||
*/ |
||||
if ((xOut || yOut)) { |
||||
x = xOut ? container.getWidth() - copiedCreator.getWidth() / 2 - xoffset : x; |
||||
y = yOut ? container.getHeight() - copiedCreator.getHeight() / 2 - yoffset : y; |
||||
return new Point(x, y); |
||||
} |
||||
} |
||||
return new Point(x, y); |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 拷贝组件 |
||||
*/ |
||||
private static Widget copyWidget(FormDesigner formDesigner, XCreator xCreator) throws |
||||
CloneNotSupportedException { |
||||
ArrayList<String> nameSpace = new ArrayList<>(); |
||||
Widget copied = (Widget) xCreator.toData().clone(); |
||||
//重命名拷贝的组件
|
||||
String name = getCopiedName(formDesigner, copied, nameSpace); |
||||
if (copied instanceof WTitleLayout) { |
||||
XWTitleLayout xwTitleLayout = new XWTitleLayout((WTitleLayout) copied, xCreator.getSize()); |
||||
xwTitleLayout.resetCreatorName(name); |
||||
} else { |
||||
copied.setWidgetName(name); |
||||
} |
||||
return copied; |
||||
} |
||||
|
||||
/** |
||||
* 组件拷贝命名规则 |
||||
*/ |
||||
private static String getCopiedName(FormDesigner formDesigner, Widget copied, ArrayList<String> nameSpace) { |
||||
StringBuilder name = new StringBuilder(copied.getWidgetName()); |
||||
do { |
||||
name.append(POSTFIX); |
||||
} while (formDesigner.getTarget().isNameExist(name.toString()) || nameSpace.contains(name.toString())); |
||||
nameSpace.add(name.toString()); |
||||
return name.toString(); |
||||
} |
||||
|
||||
public static void rebuildSelection(FormDesigner designer) { |
||||
ArrayList<XCreator> newSelection = new ArrayList<>(); |
||||
List<Widget> widgetList = new ArrayList<>(); |
||||
for (XCreator comp : designer.getSelectionModel().getSelection().getSelectedCreators()) { |
||||
widgetList.add(comp.toData()); |
||||
} |
||||
designer.getSelectionModel().setSelectedCreators( |
||||
rebuildSelection(designer.getRootComponent(), widgetList, newSelection)); |
||||
} |
||||
|
||||
public static ArrayList<XCreator> rebuildSelection(XCreator rootComponent, Widget[] selectWidgets) { |
||||
List<Widget> selectionWidget = new ArrayList<>(); |
||||
if (selectWidgets != null) { |
||||
selectionWidget.addAll(Arrays.asList(selectWidgets)); |
||||
} |
||||
return FormSelectionUtils.rebuildSelection(rootComponent, selectionWidget, new ArrayList<XCreator>()); |
||||
} |
||||
|
||||
private static ArrayList<XCreator> rebuildSelection(XCreator rootComponent, List<Widget> selectionWidget, |
||||
ArrayList<XCreator> newSelection) { |
||||
FormSelectionUtils.rebuild(rootComponent, selectionWidget, newSelection); |
||||
if (newSelection.isEmpty()) { |
||||
newSelection.add(rootComponent); |
||||
} |
||||
return newSelection; |
||||
} |
||||
|
||||
private static void rebuild(XCreator root, List<Widget> selectionWidget, List<XCreator> newSelection) { |
||||
if (selectionWidget.isEmpty()) { |
||||
return; |
||||
} |
||||
for (Widget x : selectionWidget) { |
||||
if (ComparatorUtils.equals(x, root.toData())) { |
||||
if (!newSelection.contains(root)) { |
||||
newSelection.add(root); |
||||
selectionWidget.remove(x); |
||||
} |
||||
break; |
||||
} |
||||
} |
||||
|
||||
int count = root.getComponentCount(); |
||||
for (int i = 0; i < count && !selectionWidget.isEmpty(); i++) { |
||||
Component c = root.getComponent(i); |
||||
if (c instanceof XCreator) { |
||||
XCreator creator = (XCreator) c; |
||||
for (Widget x : selectionWidget) { |
||||
if (ComparatorUtils.equals(x, creator.toData())) { |
||||
newSelection.add(creator); |
||||
selectionWidget.remove(x); |
||||
break; |
||||
} |
||||
} |
||||
if (c instanceof XLayoutContainer) { |
||||
rebuild((XLayoutContainer) c, selectionWidget, newSelection); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,84 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.design.dialog.BasicDialog; |
||||
import com.fr.design.dialog.DialogActionAdapter; |
||||
import com.fr.design.gui.ibutton.UIButton; |
||||
import com.fr.design.gui.ilable.UILabel; |
||||
import com.fr.design.utils.gui.GUICoreUtils; |
||||
import com.fr.general.IOUtils; |
||||
import com.fr.general.Inter; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.*; |
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.ActionListener; |
||||
|
||||
/** |
||||
* Created by harry on 2017-3-2. |
||||
*/ |
||||
public class FormSettingToolBar extends JPanel { |
||||
private Icon setIcon = IOUtils.readIcon("com/fr/design/images/toolbarbtn/toolbarbtnsetting.png"); |
||||
private Icon delIcon = IOUtils.readIcon("com/fr/design/images/toolbarbtn/toolbarbtnclear.png"); |
||||
private UIButton setButton; |
||||
private UIButton delButton; |
||||
private FormToolBarPane toolBarPane; |
||||
|
||||
public FormSettingToolBar(String name, FormToolBarPane toolBarPane) { |
||||
super(); |
||||
this.setBackground(Color.lightGray); |
||||
this.add(new UILabel(name)); |
||||
this.toolBarPane = toolBarPane; |
||||
setButton = GUICoreUtils.createTransparentButton(setIcon, setIcon, setIcon); |
||||
setButton.setToolTipText(Inter.getLocText("FR-Designer_Edit_Button_ToolBar")); |
||||
setButton.setAction(new SetAction()); |
||||
delButton = GUICoreUtils.createTransparentButton(delIcon, delIcon, delIcon); |
||||
delButton.setToolTipText(Inter.getLocText("FR-Designer_Remove_Button_ToolBar")); |
||||
delButton.setAction(new DelAction()); |
||||
this.add(setButton); |
||||
this.add(delButton); |
||||
} |
||||
|
||||
public void setEnabled(boolean b) { |
||||
setButton.setEnabled(b); |
||||
delButton.setEnabled(b); |
||||
} |
||||
|
||||
public void addActionListener(ActionListener l){ |
||||
setButton.addActionListener(l); |
||||
delButton.addActionListener(l); |
||||
} |
||||
|
||||
private class SetAction extends AbstractAction { |
||||
|
||||
public SetAction() { |
||||
this.putValue(Action.SMALL_ICON, setIcon); |
||||
} |
||||
|
||||
@Override |
||||
public void actionPerformed(ActionEvent arg0) { |
||||
final FormEditToolBar tb = new FormEditToolBar(); |
||||
tb.populate(toolBarPane.getFToolBar()); |
||||
BasicDialog dialog = tb.showWindow(DesignerContext.getDesignerFrame()); |
||||
dialog.addDialogActionListener(new DialogActionAdapter() { |
||||
public void doOk() { |
||||
toolBarPane.setFToolBar(tb.update()); |
||||
} |
||||
}); |
||||
dialog.setVisible(true); |
||||
} |
||||
} |
||||
|
||||
private class DelAction extends AbstractAction { |
||||
|
||||
public DelAction() { |
||||
this.putValue(Action.SMALL_ICON, delIcon); |
||||
} |
||||
|
||||
@Override |
||||
public void actionPerformed(ActionEvent arg0) { |
||||
toolBarPane.removeAll(); |
||||
toolBarPane.removeButtonList(); |
||||
toolBarPane.repaint(); |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,68 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.form.ui.ToolBar; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.general.Background; |
||||
|
||||
import java.util.ArrayList; |
||||
import java.util.List; |
||||
|
||||
/** |
||||
* Created by harry on 2017-3-2. |
||||
*/ |
||||
public class FormToolBar { |
||||
private List<FormToolBarButton> buttonlist = new ArrayList<FormToolBarButton>(); |
||||
private Background background = null; |
||||
private boolean isDefault = true; |
||||
|
||||
public List<FormToolBarButton> getButtonlist() { |
||||
return buttonlist; |
||||
} |
||||
|
||||
public void setButtonlist(List<FormToolBarButton> buttonlist) { |
||||
if (buttonlist == null || buttonlist.size() < 0) { |
||||
this.buttonlist = new ArrayList<FormToolBarButton>(); |
||||
} else { |
||||
this.buttonlist = buttonlist; |
||||
} |
||||
} |
||||
|
||||
public void addButton(FormToolBarButton toolBarButton) { |
||||
this.buttonlist.add(toolBarButton); |
||||
} |
||||
|
||||
public void removeButton(FormToolBarButton toolBarButton) { |
||||
this.buttonlist.remove(toolBarButton); |
||||
} |
||||
|
||||
public void clearButton() { |
||||
this.buttonlist.clear(); |
||||
} |
||||
|
||||
public Background getBackground() { |
||||
return background; |
||||
} |
||||
|
||||
public void setBackground(Background background) { |
||||
this.background = background; |
||||
} |
||||
|
||||
public boolean isDefault() { |
||||
return isDefault; |
||||
} |
||||
|
||||
public void setDefault(boolean isDefault) { |
||||
this.isDefault = isDefault; |
||||
} |
||||
|
||||
public ToolBar getToolBar() { |
||||
Widget[] widgets = new Widget[this.getButtonlist().size()]; |
||||
for (int j = 0; j < this.getButtonlist().size(); j++) { |
||||
widgets[j] = this.getButtonlist().get(j).getWidget(); |
||||
} |
||||
ToolBar toolBar = new ToolBar(widgets); |
||||
toolBar.setBackground(this.background); |
||||
toolBar.setDefault(this.isDefault); |
||||
return toolBar; |
||||
} |
||||
} |
@ -0,0 +1,164 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.base.BaseUtils; |
||||
import com.fr.design.dialog.BasicDialog; |
||||
import com.fr.design.dialog.DialogActionAdapter; |
||||
import com.fr.design.gui.core.WidgetOption; |
||||
import com.fr.design.roleAuthority.ReportAndFSManagePane; |
||||
import com.fr.design.utils.gui.GUIPaintUtils; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.form.ui.WidgetManager; |
||||
import com.fr.stable.Constants; |
||||
import com.fr.stable.StringUtils; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.*; |
||||
import java.awt.event.MouseEvent; |
||||
import java.awt.event.MouseListener; |
||||
|
||||
/** |
||||
* Created by harry on 2017-3-2. |
||||
*/ |
||||
public class FormToolBarButton extends JButton implements MouseListener { |
||||
private Widget widget; |
||||
private WidgetOption widgetOption; |
||||
|
||||
public FormToolBarButton(Icon icon, Widget widget) { |
||||
this(null, icon, widget); |
||||
} |
||||
|
||||
public FormToolBarButton(String text, Icon icon, Widget widget) { |
||||
super(text, icon); |
||||
setBackground(null); |
||||
setRolloverEnabled(true); |
||||
this.widget = widget; |
||||
if (widget instanceof com.fr.form.ui.Button) { |
||||
com.fr.form.ui.Button button = (com.fr.form.ui.Button) widget; |
||||
String iconName = button.getIconName(); |
||||
if (StringUtils.isNotEmpty(iconName)) { |
||||
Image iimage = WidgetManager.getProviderInstance().getIconManager().getIconImage(iconName); |
||||
if (iimage != null) { |
||||
setIcon(new ImageIcon(iimage)); |
||||
} |
||||
} |
||||
} |
||||
this.addMouseListener(this); |
||||
setMargin(new Insets(0, 0, 0, 0)); |
||||
} |
||||
|
||||
/** |
||||
* 改变按钮的权限细粒度状态 |
||||
* |
||||
* @param selectedRole 选择的角色 |
||||
* @param isVisible 是否可见 |
||||
*/ |
||||
public void changeAuthorityState(String selectedRole, boolean isVisible) { |
||||
this.widget.changeOnlyVisibleAuthorityState(selectedRole, isVisible); |
||||
} |
||||
|
||||
/** |
||||
* 是都做过权限细粒度 |
||||
* |
||||
* @param role 选择的角色 |
||||
* @return 若是对应的该角色做过权限细粒度,则返回true |
||||
*/ |
||||
public boolean isDoneAuthorityEdited(String role) { |
||||
return this.widget.isDoneVisibleAuthority(role); |
||||
} |
||||
|
||||
public Widget getWidget() { |
||||
return this.widget; |
||||
} |
||||
|
||||
|
||||
public void setWidget(Widget widget) { |
||||
this.widget = widget; |
||||
} |
||||
|
||||
public WidgetOption getNameOption() { |
||||
return this.widgetOption; |
||||
} |
||||
|
||||
public void setNameOption(WidgetOption widgetOption) { |
||||
this.widgetOption = widgetOption; |
||||
} |
||||
|
||||
|
||||
protected void paintBorder(Graphics g) { |
||||
paintBorder(g, this); |
||||
} |
||||
|
||||
protected void paintBorder(Graphics g, FormToolBarButton b) { |
||||
String selectedRoles = ReportAndFSManagePane.getInstance().getRoleTree().getSelectedRoleName(); |
||||
GUIPaintUtils.drawBorder((Graphics2D) g, 0, 0, b.getWidth(), b.getHeight(), true, Constants.NULL, b.isDoneAuthorityEdited(selectedRoles)); |
||||
} |
||||
|
||||
/** |
||||
* 处理鼠标点击事件 |
||||
* |
||||
* @param e 鼠标点击事件 |
||||
*/ |
||||
public void mouseClicked(MouseEvent e) { |
||||
//该button只在报表块工具栏中使用,
|
||||
//parent只有FormToolBarPane一种,故可以直接强转
|
||||
final FormToolBarPane toolBarPane = (FormToolBarPane) this.getParent(); |
||||
if (BaseUtils.isAuthorityEditing()) { |
||||
auhtorityMouseAction(); |
||||
return; |
||||
} |
||||
if (e.getClickCount() >= 2) { |
||||
final FormEditToolBar editToolBar = new FormEditToolBar(); |
||||
editToolBar.populate(toolBarPane.getFToolBar(), this); |
||||
BasicDialog dialog = editToolBar.showWindow(DesignerContext.getDesignerFrame()); |
||||
dialog.addDialogActionListener(new DialogActionAdapter() { |
||||
public void doOk() { |
||||
toolBarPane.setFToolBar(editToolBar.update()); |
||||
} |
||||
}); |
||||
dialog.setVisible(true); |
||||
} |
||||
} |
||||
|
||||
|
||||
private void auhtorityMouseAction() { |
||||
if (this.isEnabled()) { |
||||
this.setSelected(!this.isSelected()); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标进入事件 |
||||
* |
||||
* @param e 鼠标进入事件 |
||||
*/ |
||||
public void mouseEntered(MouseEvent e) { |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 鼠标退出事件 |
||||
* |
||||
* @param e 鼠标退出事件 |
||||
*/ |
||||
public void mouseExited(MouseEvent e) { |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 鼠标按下事件 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void mousePressed(MouseEvent e) { |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 鼠标释放事件 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void mouseReleased(MouseEvent e) { |
||||
|
||||
} |
||||
} |
@ -0,0 +1,233 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.design.beans.BasicBeanPane; |
||||
import com.fr.design.dialog.BasicDialog; |
||||
import com.fr.design.dialog.DialogActionAdapter; |
||||
import com.fr.design.gui.core.WidgetOption; |
||||
import com.fr.design.layout.FRGUIPaneFactory; |
||||
import com.fr.form.ui.ToolBar; |
||||
import com.fr.form.ui.Widget; |
||||
|
||||
import javax.swing.*; |
||||
import java.awt.*; |
||||
import java.awt.datatransfer.DataFlavor; |
||||
import java.awt.datatransfer.UnsupportedFlavorException; |
||||
import java.awt.event.MouseAdapter; |
||||
import java.awt.event.MouseEvent; |
||||
import java.awt.event.MouseListener; |
||||
import java.util.List; |
||||
|
||||
/** |
||||
* Created by harry on 2017-3-2. |
||||
*/ |
||||
public class FormToolBarPane extends BasicBeanPane<ToolBar> { |
||||
|
||||
private FormToolBar ftoolbar = new FormToolBar(); |
||||
|
||||
public FormToolBarPane() { |
||||
super(); |
||||
this.initComponent(); |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 添加鼠标监听 |
||||
* |
||||
* @param mouselistener 鼠标监听 |
||||
*/ |
||||
public void addAuthorityListener(MouseListener mouselistener) { |
||||
List<FormToolBarButton> list = ftoolbar.getButtonlist(); |
||||
for (int i = 0; i < list.size(); i++) { |
||||
list.get(i).addMouseListener(mouselistener); |
||||
} |
||||
} |
||||
|
||||
public FormToolBarPane(FormToolBarButton button) { |
||||
super(); |
||||
this.initComponent(); |
||||
this.add(button); |
||||
} |
||||
|
||||
/** |
||||
* 初始化组件 |
||||
*/ |
||||
public void initComponent() { |
||||
this.addMouseListener(listener); |
||||
this.setLayout(FRGUIPaneFactory.createBoxFlowLayout()); |
||||
this.setTransferHandler(new ToolBarHandler(TransferHandler.COPY)); |
||||
this.setBorder(BorderFactory.createTitledBorder("")); |
||||
} |
||||
|
||||
/** |
||||
* 删除鼠标事件 |
||||
*/ |
||||
public void removeDefaultMouseListener() { |
||||
this.removeMouseListener(listener); |
||||
} |
||||
|
||||
@Override |
||||
protected String title4PopupWindow() { |
||||
return "Toolbar"; |
||||
} |
||||
|
||||
public void setSelectedButton(FormToolBarButton button) { |
||||
this.ftoolbar.addButton(button); |
||||
} |
||||
|
||||
/** |
||||
* 添加组件 |
||||
* |
||||
* @param comp 组件 |
||||
* |
||||
* @return 被添加的组件 |
||||
*/ |
||||
public Component add(Component comp) { |
||||
if (comp instanceof FormToolBarButton) { |
||||
this.ftoolbar.addButton((FormToolBarButton) comp); |
||||
} |
||||
return super.add(comp); |
||||
} |
||||
|
||||
private Component addComp(Component comp) { |
||||
return super.add(comp); |
||||
} |
||||
|
||||
|
||||
public void removeButtonList() { |
||||
this.ftoolbar.clearButton(); |
||||
} |
||||
|
||||
protected void setFToolBar(FormToolBar ftoolbar) { |
||||
if (ftoolbar == null) { |
||||
ftoolbar = new FormToolBar(); |
||||
} |
||||
this.ftoolbar = ftoolbar; |
||||
this.setToolBar(this.ftoolbar.getButtonlist()); |
||||
} |
||||
|
||||
public List<FormToolBarButton> getToolBarButtons() { |
||||
return ftoolbar.getButtonlist(); |
||||
} |
||||
|
||||
protected FormToolBar getFToolBar() { |
||||
return this.ftoolbar; |
||||
} |
||||
|
||||
public boolean isEmpty() { |
||||
return this.ftoolbar.getButtonlist().size() <= 0; |
||||
} |
||||
|
||||
private void setToolBar(List<FormToolBarButton> list) { |
||||
if (list == null || list.size() < 0) { |
||||
return; |
||||
} |
||||
this.removeAll(); |
||||
for (int i = 0; i < list.size(); i++) { |
||||
this.addComp(list.get(i)); |
||||
} |
||||
this.validate(); |
||||
this.repaint(); |
||||
|
||||
} |
||||
|
||||
@Override |
||||
public void populateBean(ToolBar toolbar) { |
||||
this.removeAll(); |
||||
this.getFToolBar().clearButton(); |
||||
for (int j = 0; j < toolbar.getWidgetSize(); j++) { |
||||
Widget widget = toolbar.getWidget(j); |
||||
WidgetOption no = WidgetOption.getToolBarButton(widget.getClass()); |
||||
if (no == null){ |
||||
//如果装了什么插件, 放到了工具栏上, 后来删除了插件, 模板里还存着之前的控件
|
||||
continue; |
||||
} |
||||
|
||||
FormToolBarButton button = new FormToolBarButton(no.optionIcon(), widget); |
||||
button.setNameOption(no); |
||||
this.add(button); |
||||
this.validate(); |
||||
this.repaint(); |
||||
} |
||||
this.getFToolBar().setBackground(toolbar.getBackground()); |
||||
this.getFToolBar().setDefault(toolbar.isDefault()); |
||||
} |
||||
|
||||
@Override |
||||
public ToolBar updateBean() { |
||||
return this.ftoolbar.getToolBar(); |
||||
} |
||||
|
||||
MouseListener listener = new MouseAdapter() { |
||||
@Override |
||||
public void mouseClicked(MouseEvent e) { |
||||
if (e.getClickCount() >= 2 && !SwingUtilities.isRightMouseButton(e)) { |
||||
final FormEditToolBar tb = new FormEditToolBar(); |
||||
tb.populate(getFToolBar()); |
||||
BasicDialog dialog = tb.showWindow(SwingUtilities.getWindowAncestor(FormToolBarPane.this)); |
||||
dialog.addDialogActionListener(new DialogActionAdapter() { |
||||
@Override |
||||
public void doOk() { |
||||
FormToolBarPane.this.setFToolBar(tb.update()); |
||||
} |
||||
}); |
||||
dialog.setVisible(true); |
||||
} |
||||
} |
||||
}; |
||||
|
||||
|
||||
/* |
||||
* 拖拽属性设置 |
||||
*/ |
||||
private class ToolBarHandler extends TransferHandler { |
||||
private int action; |
||||
|
||||
public ToolBarHandler(int action) { |
||||
this.action = action; |
||||
} |
||||
|
||||
|
||||
@Override |
||||
public boolean canImport(TransferHandler.TransferSupport support) { |
||||
if (!support.isDrop()) { |
||||
return false; |
||||
} |
||||
|
||||
if (!support.isDataFlavorSupported(DataFlavor.stringFlavor)) { |
||||
return false; |
||||
} |
||||
|
||||
boolean actionSupported = (action & support.getSourceDropActions()) == action; |
||||
if (actionSupported) { |
||||
support.setDropAction(action); |
||||
return true; |
||||
} |
||||
|
||||
return false; |
||||
} |
||||
|
||||
@Override |
||||
public boolean importData(TransferHandler.TransferSupport support) { |
||||
if (!canImport(support)) { |
||||
return false; |
||||
} |
||||
WidgetOption data; |
||||
try { |
||||
data = (WidgetOption) support.getTransferable().getTransferData(DataFlavor.stringFlavor); |
||||
} catch (UnsupportedFlavorException e) { |
||||
return false; |
||||
} catch (java.io.IOException e) { |
||||
return false; |
||||
} |
||||
|
||||
Widget widget = data.createWidget(); |
||||
FormToolBarButton btn = new FormToolBarButton(data.optionIcon(), widget); |
||||
btn.setNameOption(data); |
||||
FormToolBarPane.this.add(btn); |
||||
FormToolBarPane.this.validate(); |
||||
FormToolBarPane.this.repaint(); |
||||
return true; |
||||
} |
||||
|
||||
} |
||||
} |
@ -0,0 +1,54 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.design.gui.core.WidgetOption; |
||||
import com.fr.design.gui.core.WidgetOptionFactory; |
||||
import com.fr.form.ui.CustomToolBarButton; |
||||
import com.fr.form.web.button.ExcelO; |
||||
import com.fr.form.web.button.ExcelP; |
||||
import com.fr.form.web.button.Export; |
||||
import com.fr.form.web.button.page.*; |
||||
import com.fr.general.IOUtils; |
||||
import com.fr.general.Inter; |
||||
|
||||
/** |
||||
* Created by harry on 2017-3-3. |
||||
*/ |
||||
public class FormWebWidgetConstants { |
||||
|
||||
private FormWebWidgetConstants() { |
||||
} |
||||
|
||||
public static WidgetOption[] getFormElementCaseToolBarInstance() { |
||||
return new WidgetOption[]{FIRST, PREVIOUS, PAGENAVI, NEXT, LAST, EXPORT, EXCELP, EXCELO, CUSTOM_BUTTON}; |
||||
} |
||||
|
||||
// 当前页/总页数
|
||||
public static final WidgetOption PAGENAVI = WidgetOptionFactory.createByWidgetClass(Inter.getLocText(new String[]{"HJS-Current_Page", "HF-Number_of_Page"}, new String[]{"/", ""}), |
||||
IOUtils.readIcon("/com/fr/web/images/pageNumber.png"), PageNavi.class); |
||||
// 首页
|
||||
public static final WidgetOption FIRST = WidgetOptionFactory.createByWidgetClass(Inter.getLocText("FR-Engine_ReportServerP-First"), IOUtils.readIcon("/com/fr/web/images/first.png"), |
||||
First.class); |
||||
// 末页
|
||||
public static final WidgetOption LAST = WidgetOptionFactory.createByWidgetClass(Inter.getLocText("FR-Engine_ReportServerP-Last"), IOUtils.readIcon("/com/fr/web/images/last.png"), |
||||
Last.class); |
||||
// 前一页
|
||||
public static final WidgetOption PREVIOUS = WidgetOptionFactory.createByWidgetClass(Inter.getLocText("FR-Engine_ReportServerP-Previous"), |
||||
IOUtils.readIcon("/com/fr/web/images/previous.png"), Previous.class); |
||||
// 后一页
|
||||
public static final WidgetOption NEXT = WidgetOptionFactory.createByWidgetClass(Inter.getLocText("FR-Engine_ReportServerP-Next"), IOUtils.readIcon("/com/fr/web/images/next.png"), |
||||
Next.class); |
||||
|
||||
// 导出成Excel 分页导出
|
||||
public static final WidgetOption EXCELP = WidgetOptionFactory.createByWidgetClass(Inter.getLocText(new String[]{"Excel", "FR-Designer_Export_Excel_Page"}, new String[]{"(", ")"}), |
||||
IOUtils.readIcon("/com/fr/web/images/excel.png"), ExcelP.class); |
||||
// 导出成Excel 原样导出
|
||||
public static final WidgetOption EXCELO = WidgetOptionFactory.createByWidgetClass(Inter.getLocText(new String[]{"Excel", "FR-Designer_Export_Excel_Simple"}, new String[]{"(", ")"}), |
||||
IOUtils.readIcon("/com/fr/web/images/excel.png"), ExcelO.class); |
||||
// 导出
|
||||
public static final WidgetOption EXPORT = WidgetOptionFactory |
||||
.createByWidgetClass(Inter.getLocText("FR-Engine_Export"), IOUtils.readIcon("/com/fr/web/images/export.png"), Export.class); |
||||
|
||||
public static final WidgetOption CUSTOM_BUTTON = WidgetOptionFactory |
||||
.createByWidgetClass(Inter.getLocText(new String[]{"Custom", "Form-Button"}), CustomToolBarButton.class); |
||||
|
||||
} |
File diff suppressed because it is too large
Load Diff
@ -1,346 +1,348 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import java.awt.*; |
||||
import java.awt.event.MouseAdapter; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.List; |
||||
|
||||
import javax.swing.BorderFactory; |
||||
import javax.swing.JLabel; |
||||
import javax.swing.JTable; |
||||
import javax.swing.ListSelectionModel; |
||||
import javax.swing.table.*; |
||||
|
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.gui.ilable.UILabel; |
||||
import com.fr.design.gui.itable.GroupRenderer; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.form.ui.container.*; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.general.Inter; |
||||
import com.fr.stable.StringUtils; |
||||
|
||||
/** |
||||
* MobileWidgetTable类主要显示各种容器的控件列表(body,tab,绝对布局快,不包括参数面板) |
||||
* Created with IntelliJ IDEA. |
||||
* User: zx |
||||
* Date: 14-9-15 |
||||
* Time: 下午4:52 |
||||
* Modified by fanglei at 2017/01/23 |
||||
*/ |
||||
public class MobileWidgetTable extends JTable { |
||||
|
||||
private FormDesigner designer; |
||||
private String[][] cellData; |
||||
private String[] headers = {Inter.getLocText("Form-Widget_Name")}; |
||||
private static final int WIDGET_TABLE_ROW_HEIGHT = 22; |
||||
private UILabel moveComponent = new UILabel(); // 作为拖动时候随鼠标移动的那个半透明控件
|
||||
private int selectedRow = -1; |
||||
private int GAP = 11; |
||||
private boolean draging = false; |
||||
private boolean collapsed = false; // 控件列表是否折叠
|
||||
|
||||
@Override |
||||
public TableCellRenderer getCellRenderer(int row, int column) { |
||||
//第一行渲染成为标题的样子
|
||||
if (row == 0) { |
||||
return new GroupRenderer(); |
||||
} |
||||
return super.getCellRenderer(row, column); |
||||
} |
||||
|
||||
|
||||
public MobileWidgetTable(FormDesigner designer) { |
||||
this.designer = designer; |
||||
cellData = getData(); |
||||
this.setTableProperties(); |
||||
TableModel defaultModel = new BeanTableModel(); |
||||
this.setModel(defaultModel); |
||||
this.repaint(); |
||||
this.setDefaultRenderer(Object.class,new DefaultTableCellRenderer()); |
||||
refreshData(); |
||||
this.addMouseListener(mouseAdapter); |
||||
this.addMouseMotionListener(mouseAdapter); |
||||
add(moveComponent); |
||||
} |
||||
|
||||
private void setTableProperties() { |
||||
JTableHeader header = getTableHeader(); |
||||
header.setReorderingAllowed(false); |
||||
header.setPreferredSize(new Dimension(0, 0)); // 隐藏表头
|
||||
GroupRenderer headerRenderer = new GroupRenderer(); |
||||
headerRenderer.setPreferredSize(new Dimension(0, 0)); //这行代码隐藏表头。因为要实现折叠效果,表头不好监听事件
|
||||
headerRenderer.setHorizontalAlignment(JLabel.LEFT); |
||||
header.setDefaultRenderer(headerRenderer); |
||||
|
||||
this.setRowHeight(WIDGET_TABLE_ROW_HEIGHT); |
||||
this.setGridColor(new Color(212, 208, 200)); |
||||
this.setSelectionMode(ListSelectionModel.SINGLE_SELECTION); |
||||
this.setColumnSelectionAllowed(false); |
||||
this.setRowSelectionAllowed(false); |
||||
this.setFillsViewportHeight(false); |
||||
this.setAutoResizeMode(JTable.AUTO_RESIZE_ALL_COLUMNS); |
||||
} |
||||
|
||||
private MouseAdapter mouseAdapter = new MouseAdapter() { |
||||
/** |
||||
* 鼠标按下时处理的事件(设置当前选中的行列) |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mousePressed(MouseEvent e) { |
||||
getInstance().setCellSelected(); |
||||
if (selectedRow == 0 && !e.isPopupTrigger() && e.getClickCount() == 1 && e.getX() < WIDGET_TABLE_ROW_HEIGHT / 2){ // 如果是点击在第一行
|
||||
toggleCollapse(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标放开时处理的事件(如果是正在拖动则执行换位操作,重新绘制属性表,如果不是则什么也不做) |
||||
* 所谓的换行就是简单的重新拿到一次表格数据然后重新绘制表格 |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseReleased(MouseEvent e) { |
||||
if(!draging){ |
||||
return; |
||||
} |
||||
draging = false; |
||||
moveComponent.setVisible(false); |
||||
int toIndex = e.getY() < GAP ? 0 : (int)Math.rint((e.getY() - GAP)/WIDGET_TABLE_ROW_HEIGHT) + 1; |
||||
//当鼠标放开时,将选中的容器调整至新的顺序
|
||||
((WSortLayout)designer.getSelectionModel().getSelection().getSelectedCreator().toData()).adjustOrder(selectedRow - 1, toIndex - 1); |
||||
//拿取排序后表格数据,然后重绘表格
|
||||
getInstance().refreshData(); |
||||
getInstance().repaint(); |
||||
designer.fireTargetModified(); |
||||
getInstance().setCellSelected(); |
||||
} |
||||
|
||||
/** |
||||
* 设置鼠标在属性表区域移动时候的事件 |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseMoved(MouseEvent e) { |
||||
int overRow = 0; |
||||
for (int i = 0;i < getRowCount();i++) { |
||||
if (e.getY() > i * WIDGET_TABLE_ROW_HEIGHT && e.getY() <= (i + 1) * WIDGET_TABLE_ROW_HEIGHT){ |
||||
overRow = i; //判断鼠标在哪一行
|
||||
} |
||||
} |
||||
//如果鼠标移动到当前选中的行上面的时候
|
||||
if (overRow == selectedRow && selectedRow > 0) { |
||||
//把当前选中的那一行的光标改成(除了第一列)移动样式MOVE_CURSOR
|
||||
setCursor(Cursor.getPredefinedCursor(Cursor.MOVE_CURSOR)); |
||||
} else { |
||||
setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR)); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标拖动事件(如果鼠标当前是<code>MOVE_CURSOR</code>状态则执行开始拖动的代码, |
||||
* 绘制一个<code>moveComponent</code>来跟随鼠标移动) |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseDragged(MouseEvent e) { |
||||
int width = getColumnModel().getColumn(0).getWidth(); |
||||
//如果当前选中的行的范围是合理的话,就可以拖动
|
||||
if (selectedRow < getRowCount() && selectedRow > 0){ |
||||
setCursor(Cursor.getPredefinedCursor(Cursor.MOVE_CURSOR)); |
||||
draging = true; |
||||
moveComponent.setText(getValueAt(selectedRow,0).toString()); |
||||
moveComponent.setLocation(0, e.getY() - GAP); |
||||
moveComponent.setSize(new Dimension(width, WIDGET_TABLE_ROW_HEIGHT)); |
||||
moveComponent.setVisible(true); |
||||
moveComponent.setForeground(Color.lightGray); |
||||
moveComponent.setBorder(BorderFactory.createLineBorder(Color.lightGray)); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 设置鼠标单击时处理的事件(单击控件列表进入控件属性表) |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseClicked(MouseEvent e) { |
||||
if(selectedRow > 0){ |
||||
//当前点击的控件的名字
|
||||
String widgetName = cellData[selectedRow][0]; |
||||
if (StringUtils.isNotEmpty(widgetName)){ |
||||
//当前选择的容器
|
||||
XCreator selectedContainer = designer.getSelectionModel().getSelection().getSelectedCreator(); |
||||
WLayout selectedWidget = (WLayout)selectedContainer.toData(); |
||||
//当前选择的容器中的控件数量
|
||||
int count = selectedWidget.getWidgetCount(); |
||||
for (int i = 0;i < count ;i++){ |
||||
XCreator xCreator = (XCreator) selectedContainer.getComponent(i); |
||||
Widget widget = xCreator.toData(); |
||||
if (ComparatorUtils.equals(widgetName, widget.getWidgetName())) { |
||||
getEditingDesigner().getSelectionModel().setSelectedCreator(xCreator); |
||||
} |
||||
} |
||||
} |
||||
} else if (selectedRow == 0){ // 如果是点击在第一行
|
||||
if (!e.isPopupTrigger() && e.getClickCount() > 1) { |
||||
toggleCollapse(); |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标离开属性表区域事件 |
||||
* @param e |
||||
*/ |
||||
public void mouseExited(MouseEvent e) { |
||||
draging = false; |
||||
moveComponent.setVisible(false); |
||||
setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR)); |
||||
} |
||||
}; |
||||
|
||||
public MobileWidgetTable getInstance(){ |
||||
return this; |
||||
} |
||||
|
||||
public FormDesigner getEditingDesigner(){ |
||||
return designer; |
||||
} |
||||
|
||||
/** |
||||
* 设置当前get到的行列的单元格为选中状态 |
||||
*/ |
||||
private void setCellSelected() { |
||||
selectedRow = getSelectedRow(); |
||||
if (selectedRow != -1) { |
||||
this.setRowSelectionInterval(selectedRow, selectedRow); |
||||
this.setColumnSelectionInterval(0, 0); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 切换属性组折叠属性true/false |
||||
*/ |
||||
private void toggleCollapse() { |
||||
this.setCollapsed(!this.isCollapsed()); |
||||
//这里获取表格的父控件是为了当表格被折叠了后,装表格的父控件也要相应的重新布局一下
|
||||
//比如折叠之后表格行数应该比原来的少,占用父容器空间应该小点,不重新布局父容器,表格大小不会改变
|
||||
Container parent = MobileWidgetTable.this.getParent(); |
||||
if (parent != null) { |
||||
parent.revalidate(); |
||||
} |
||||
repaint(); |
||||
} |
||||
|
||||
/** |
||||
* 重新get排序后的数据 |
||||
*/ |
||||
public void refreshData(){ |
||||
cellData = getData(); |
||||
} |
||||
|
||||
/** |
||||
* 获取选中控件的控件列表 |
||||
* |
||||
* @return String[][] 二维数组,[0][0]widgetName |
||||
*/ |
||||
private String[][] getData(){ |
||||
if(designer.isFormParaDesigner()){ |
||||
return new String[0][0]; |
||||
} |
||||
|
||||
//选择的控件
|
||||
Widget selectedModel = designer.getSelectionModel().getSelection().getSelectedCreator().toData(); |
||||
|
||||
if(selectedModel == null){ |
||||
return new String[0][0]; |
||||
} |
||||
|
||||
// 选择的控件有两种类型,一种是WLayout,代表容器,一种是Widget,代表控件
|
||||
if (selectedModel.acceptType(WSortLayout.class)) { |
||||
List<String> mobileWidgetList = ((WSortLayout)selectedModel).getOrderedMobileWidgetList(); |
||||
String[][] widgetName = new String[mobileWidgetList.size() + 1][1]; |
||||
widgetName[0][0] = Inter.getLocText("FR-Designer_WidgetOrder"); |
||||
for (int i = 0; i < mobileWidgetList.size(); i++) { |
||||
widgetName[i + 1][0] = mobileWidgetList.get(i); |
||||
} |
||||
return widgetName; |
||||
} else { |
||||
return new String[0][0]; |
||||
} |
||||
} |
||||
|
||||
public boolean isCollapsed() { |
||||
return collapsed; |
||||
} |
||||
|
||||
public void setCollapsed(boolean collapsed) { |
||||
this.collapsed = collapsed; |
||||
} |
||||
|
||||
/** |
||||
* 自定义的tableEditor类 |
||||
*/ |
||||
public class BeanTableModel extends DefaultTableModel { |
||||
public BeanTableModel() { |
||||
super(cellData,headers); |
||||
} |
||||
|
||||
@Override |
||||
public int getRowCount() { |
||||
if (isCollapsed()) { |
||||
return 1; |
||||
} |
||||
return cellData.length; |
||||
} |
||||
|
||||
@Override |
||||
public int getColumnCount() { |
||||
return 1; |
||||
} |
||||
|
||||
|
||||
@Override |
||||
public Object getValueAt(int row, int column) { |
||||
if (row >= getRowCount() || column >= getColumnCount()) { |
||||
return null; |
||||
} |
||||
if (row == 0) { |
||||
return (isCollapsed()? "+" : "-") + cellData[row][0]; |
||||
} |
||||
|
||||
return cellData[row][0]; |
||||
} |
||||
|
||||
@Override |
||||
public String getColumnName(int column) { |
||||
return headers[0]; |
||||
} |
||||
|
||||
|
||||
@Override |
||||
public void setValueAt(Object aValue, int row, int column) { |
||||
if (row >= getRowCount() || column >= getColumnCount()) { |
||||
return; |
||||
} |
||||
if (aValue == null) { |
||||
cellData[row] = null; |
||||
return; |
||||
} |
||||
cellData[row][0] = aValue.toString(); |
||||
} |
||||
|
||||
/** |
||||
* 是否可编辑 |
||||
* @param row 行号 |
||||
* @param column 列号 |
||||
* @return 是否可编辑 |
||||
*/ |
||||
public boolean isCellEditable(int row, int column) { |
||||
return false; |
||||
} |
||||
|
||||
} |
||||
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import java.awt.*; |
||||
import java.awt.event.MouseAdapter; |
||||
import java.awt.event.MouseEvent; |
||||
import java.util.List; |
||||
|
||||
import javax.swing.BorderFactory; |
||||
import javax.swing.JLabel; |
||||
import javax.swing.JTable; |
||||
import javax.swing.ListSelectionModel; |
||||
import javax.swing.table.*; |
||||
|
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.gui.ilable.UILabel; |
||||
import com.fr.design.gui.itable.GroupRenderer; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.form.ui.container.*; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.general.Inter; |
||||
import com.fr.stable.StringUtils; |
||||
|
||||
/** |
||||
* MobileWidgetTable类主要显示各种容器的控件列表(body,tab,绝对布局快,不包括参数面板) |
||||
* Created with IntelliJ IDEA. |
||||
* User: zx |
||||
* Date: 14-9-15 |
||||
* Time: 下午4:52 |
||||
* Modified by fanglei at 2017/01/23 |
||||
*/ |
||||
public class MobileWidgetTable extends JTable { |
||||
|
||||
private FormDesigner designer; |
||||
private String[][] cellData; |
||||
private String[] headers = {Inter.getLocText("Form-Widget_Name")}; |
||||
private static final int WIDGET_TABLE_ROW_HEIGHT = 22; |
||||
private UILabel moveComponent = new UILabel(); // 作为拖动时候随鼠标移动的那个半透明控件
|
||||
private int selectedRow = -1; |
||||
private static final int GAP = 11; |
||||
private boolean draging = false; |
||||
private boolean collapsed = false; // 控件列表是否折叠
|
||||
|
||||
@Override |
||||
public TableCellRenderer getCellRenderer(int row, int column) { |
||||
//第一行渲染成为标题的样子
|
||||
if (row == 0) { |
||||
return new GroupRenderer(); |
||||
} |
||||
return super.getCellRenderer(row, column); |
||||
} |
||||
|
||||
|
||||
public MobileWidgetTable(FormDesigner designer) { |
||||
this.designer = designer; |
||||
cellData = getData(); |
||||
this.setTableProperties(); |
||||
TableModel defaultModel = new BeanTableModel(); |
||||
this.setModel(defaultModel); |
||||
this.repaint(); |
||||
this.setDefaultRenderer(Object.class, new DefaultTableCellRenderer()); |
||||
refreshData(); |
||||
this.addMouseListener(mouseAdapter); |
||||
this.addMouseMotionListener(mouseAdapter); |
||||
add(moveComponent); |
||||
} |
||||
|
||||
private void setTableProperties() { |
||||
JTableHeader header = getTableHeader(); |
||||
header.setReorderingAllowed(false); |
||||
header.setPreferredSize(new Dimension(0, 0)); // 隐藏表头
|
||||
GroupRenderer headerRenderer = new GroupRenderer(); |
||||
headerRenderer.setPreferredSize(new Dimension(0, 0)); //这行代码隐藏表头。因为要实现折叠效果,表头不好监听事件
|
||||
headerRenderer.setHorizontalAlignment(JLabel.LEFT); |
||||
header.setDefaultRenderer(headerRenderer); |
||||
|
||||
this.setRowHeight(WIDGET_TABLE_ROW_HEIGHT); |
||||
this.setGridColor(new Color(212, 208, 200)); |
||||
this.setSelectionMode(ListSelectionModel.SINGLE_SELECTION); |
||||
this.setColumnSelectionAllowed(false); |
||||
this.setRowSelectionAllowed(false); |
||||
this.setFillsViewportHeight(false); |
||||
this.setAutoResizeMode(JTable.AUTO_RESIZE_ALL_COLUMNS); |
||||
} |
||||
|
||||
private MouseAdapter mouseAdapter = new MouseAdapter() { |
||||
/** |
||||
* 鼠标按下时处理的事件(设置当前选中的行列) |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mousePressed(MouseEvent e) { |
||||
getInstance().setCellSelected(); |
||||
if (selectedRow == 0 && !e.isPopupTrigger() && e.getClickCount() == 1 && e.getX() < WIDGET_TABLE_ROW_HEIGHT / 2) { // 如果是点击在第一行
|
||||
toggleCollapse(); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标放开时处理的事件(如果是正在拖动则执行换位操作,重新绘制属性表,如果不是则什么也不做) |
||||
* 所谓的换行就是简单的重新拿到一次表格数据然后重新绘制表格 |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseReleased(MouseEvent e) { |
||||
if (!draging) { |
||||
return; |
||||
} |
||||
draging = false; |
||||
moveComponent.setVisible(false); |
||||
int toIndex = e.getY() < GAP ? 0 : (int) Math.rint((e.getY() - GAP) / WIDGET_TABLE_ROW_HEIGHT) + 1; |
||||
//当鼠标放开时,将选中的容器调整至新的顺序
|
||||
((WSortLayout) designer.getSelectionModel().getSelection().getSelectedCreator().toData()).adjustOrder(selectedRow - 1, toIndex - 1); |
||||
//拿取排序后表格数据,然后重绘表格
|
||||
getInstance().refreshData(); |
||||
getInstance().repaint(); |
||||
designer.fireTargetModified(); |
||||
getInstance().setCellSelected(); |
||||
} |
||||
|
||||
/** |
||||
* 设置鼠标在属性表区域移动时候的事件 |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseMoved(MouseEvent e) { |
||||
int overRow = 0; |
||||
for (int i = 0; i < getRowCount(); i++) { |
||||
if (e.getY() > i * WIDGET_TABLE_ROW_HEIGHT && e.getY() <= (i + 1) * WIDGET_TABLE_ROW_HEIGHT) { |
||||
overRow = i; //判断鼠标在哪一行
|
||||
} |
||||
} |
||||
//如果鼠标移动到当前选中的行上面的时候
|
||||
if (overRow == selectedRow && selectedRow > 0) { |
||||
//把当前选中的那一行的光标改成(除了第一列)移动样式MOVE_CURSOR
|
||||
setCursor(Cursor.getPredefinedCursor(Cursor.MOVE_CURSOR)); |
||||
} else { |
||||
setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR)); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标拖动事件(如果鼠标当前是<code>MOVE_CURSOR</code>状态则执行开始拖动的代码, |
||||
* 绘制一个<code>moveComponent</code>来跟随鼠标移动) |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseDragged(MouseEvent e) { |
||||
int width = getColumnModel().getColumn(0).getWidth(); |
||||
//如果当前选中的行的范围是合理的话,就可以拖动
|
||||
if (selectedRow < getRowCount() && selectedRow > 0) { |
||||
setCursor(Cursor.getPredefinedCursor(Cursor.MOVE_CURSOR)); |
||||
draging = true; |
||||
moveComponent.setText(getValueAt(selectedRow, 0).toString()); |
||||
moveComponent.setLocation(0, e.getY() - GAP); |
||||
moveComponent.setSize(new Dimension(width, WIDGET_TABLE_ROW_HEIGHT)); |
||||
moveComponent.setVisible(true); |
||||
moveComponent.setForeground(Color.lightGray); |
||||
moveComponent.setBorder(BorderFactory.createLineBorder(Color.lightGray)); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 设置鼠标单击时处理的事件(单击控件列表进入控件属性表) |
||||
* @param e |
||||
*/ |
||||
@Override |
||||
public void mouseClicked(MouseEvent e) { |
||||
if (selectedRow > 0) { |
||||
//当前点击的控件的名字
|
||||
String widgetName = cellData[selectedRow][0]; |
||||
if (StringUtils.isNotEmpty(widgetName)) { |
||||
//当前选择的容器
|
||||
XCreator selectedContainer = designer.getSelectionModel().getSelection().getSelectedCreator(); |
||||
WLayout selectedWidget = (WLayout) selectedContainer.toData(); |
||||
//当前选择的容器中的控件数量
|
||||
int count = selectedWidget.getWidgetCount(); |
||||
for (int i = 0; i < count; i++) { |
||||
XCreator xCreator = (XCreator) selectedContainer.getComponent(i); |
||||
Widget widget = xCreator.toData(); |
||||
if (ComparatorUtils.equals(widgetName, widget.getWidgetName())) { |
||||
getEditingDesigner().getSelectionModel().setSelectedCreator(xCreator); |
||||
} |
||||
} |
||||
} |
||||
} else if (selectedRow == 0) { // 如果是点击在第一行
|
||||
if (!e.isPopupTrigger() && e.getClickCount() > 1) { |
||||
toggleCollapse(); |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标离开属性表区域事件 |
||||
* @param e |
||||
*/ |
||||
public void mouseExited(MouseEvent e) { |
||||
draging = false; |
||||
moveComponent.setVisible(false); |
||||
setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR)); |
||||
} |
||||
}; |
||||
|
||||
public MobileWidgetTable getInstance() { |
||||
return this; |
||||
} |
||||
|
||||
public FormDesigner getEditingDesigner() { |
||||
return designer; |
||||
} |
||||
|
||||
/** |
||||
* 设置当前get到的行列的单元格为选中状态 |
||||
*/ |
||||
private void setCellSelected() { |
||||
selectedRow = getSelectedRow(); |
||||
if (selectedRow != -1) { |
||||
this.setRowSelectionInterval(selectedRow, selectedRow); |
||||
this.setColumnSelectionInterval(0, 0); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 切换属性组折叠属性true/false |
||||
*/ |
||||
private void toggleCollapse() { |
||||
this.setCollapsed(!this.isCollapsed()); |
||||
//这里获取表格的父控件是为了当表格被折叠了后,装表格的父控件也要相应的重新布局一下
|
||||
//比如折叠之后表格行数应该比原来的少,占用父容器空间应该小点,不重新布局父容器,表格大小不会改变
|
||||
Container parent = MobileWidgetTable.this.getParent(); |
||||
if (parent != null) { |
||||
parent.revalidate(); |
||||
} |
||||
repaint(); |
||||
} |
||||
|
||||
/** |
||||
* 重新get排序后的数据 |
||||
*/ |
||||
public void refreshData() { |
||||
cellData = getData(); |
||||
} |
||||
|
||||
/** |
||||
* 获取选中控件的控件列表 |
||||
* |
||||
* @return String[][] 二维数组,[0][0]widgetName |
||||
*/ |
||||
private String[][] getData() { |
||||
if (designer.isFormParaDesigner()) { |
||||
return new String[0][0]; |
||||
} |
||||
|
||||
//选择的控件
|
||||
XCreator selectedCreator = designer.getSelectionModel().getSelection().getSelectedCreator(); |
||||
Widget selectedModel = selectedCreator != null ? selectedCreator.toData() : null; |
||||
|
||||
if (selectedModel == null) { |
||||
return new String[0][0]; |
||||
} |
||||
|
||||
// 选择的控件有两种类型,一种是WLayout,代表容器,一种是Widget,代表控件
|
||||
if (selectedModel.acceptType(WSortLayout.class)) { |
||||
List<String> mobileWidgetList = ((WSortLayout) selectedModel).getOrderedMobileWidgetList(); |
||||
String[][] widgetName = new String[mobileWidgetList.size() + 1][1]; |
||||
widgetName[0][0] = Inter.getLocText("FR-Designer_WidgetOrder"); |
||||
for (int i = 0; i < mobileWidgetList.size(); i++) { |
||||
widgetName[i + 1][0] = mobileWidgetList.get(i); |
||||
} |
||||
return widgetName; |
||||
} else { |
||||
return new String[0][0]; |
||||
} |
||||
} |
||||
|
||||
public boolean isCollapsed() { |
||||
return collapsed; |
||||
} |
||||
|
||||
public void setCollapsed(boolean collapsed) { |
||||
this.collapsed = collapsed; |
||||
} |
||||
|
||||
/** |
||||
* 自定义的tableEditor类 |
||||
*/ |
||||
public class BeanTableModel extends DefaultTableModel { |
||||
public BeanTableModel() { |
||||
super(cellData, headers); |
||||
} |
||||
|
||||
@Override |
||||
public int getRowCount() { |
||||
if (isCollapsed()) { |
||||
return 1; |
||||
} |
||||
return cellData.length; |
||||
} |
||||
|
||||
@Override |
||||
public int getColumnCount() { |
||||
return 1; |
||||
} |
||||
|
||||
|
||||
@Override |
||||
public Object getValueAt(int row, int column) { |
||||
if (row >= getRowCount() || column >= getColumnCount()) { |
||||
return null; |
||||
} |
||||
if (row == 0) { |
||||
return (isCollapsed() ? "+" : "-") + cellData[row][0]; |
||||
} |
||||
|
||||
return cellData[row][0]; |
||||
} |
||||
|
||||
@Override |
||||
public String getColumnName(int column) { |
||||
return headers[0]; |
||||
} |
||||
|
||||
|
||||
@Override |
||||
public void setValueAt(Object aValue, int row, int column) { |
||||
if (row >= getRowCount() || column >= getColumnCount()) { |
||||
return; |
||||
} |
||||
if (aValue == null) { |
||||
cellData[row] = null; |
||||
return; |
||||
} |
||||
cellData[row][0] = aValue.toString(); |
||||
} |
||||
|
||||
/** |
||||
* 是否可编辑 |
||||
* |
||||
* @param row 行号 |
||||
* @param column 列号 |
||||
* @return 是否可编辑 |
||||
*/ |
||||
public boolean isCellEditable(int row, int column) { |
||||
return false; |
||||
} |
||||
|
||||
} |
||||
|
||||
} |
@ -1,366 +1,372 @@
|
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.base.BaseUtils; |
||||
import com.fr.design.ExtraDesignClassManager; |
||||
import com.fr.design.designer.beans.events.DesignerEditListener; |
||||
import com.fr.design.designer.beans.events.DesignerEvent; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XCreatorUtils; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWParameterLayout; |
||||
import com.fr.design.designer.properties.EventPropertyTable; |
||||
import com.fr.design.designer.properties.WidgetPropertyTable; |
||||
import com.fr.design.fun.WidgetPropertyUIProvider; |
||||
import com.fr.design.gui.frpane.UITabbedPane; |
||||
import com.fr.design.gui.icontainer.UIScrollPane; |
||||
import com.fr.design.gui.itable.AbstractPropertyTable; |
||||
import com.fr.design.layout.FRGUIPaneFactory; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.general.Inter; |
||||
import com.fr.stable.ArrayUtils; |
||||
|
||||
import javax.swing.*; |
||||
import javax.swing.border.LineBorder; |
||||
import javax.swing.table.JTableHeader; |
||||
import java.awt.*; |
||||
import java.util.ArrayList; |
||||
import java.util.List; |
||||
import java.util.Set; |
||||
|
||||
/** |
||||
* 控件属性表绘制 |
||||
* Modified by fanglei |
||||
*/ |
||||
public class WidgetPropertyPane extends FormDockView implements BaseWidgetPropertyPane { |
||||
|
||||
private static final String PARA = "para"; |
||||
private static final String BODY = "body"; |
||||
private WidgetPropertyTable propertyTable; // 控件的属性表
|
||||
private EventPropertyTable eventTable; // 控件的事件表
|
||||
private List<AbstractPropertyTable> widgetPropertyTables; // 这个变量应该是保存控件拓展的属性tab
|
||||
private FormDesigner designer; // 当前designer
|
||||
private UIScrollPane psp; // 用来装载属性表table的容器
|
||||
private UIScrollPane esp; //用来装载事件table的容器
|
||||
private JPanel wsp; // 装载移动端tab的容器,包括移动端属性表和控件拓展的移动端属性表
|
||||
private MobileParaWidgetTable mobileParaWidgetTable; // 参数面板的移动端属性tab(和body的移动端属性tab区别是没有标签名column)
|
||||
private MobileWidgetTable mobileWidgetTable; // body的移动端属性tab
|
||||
private UIScrollPane downPanel; // 这个滚动容器是用于装载centerPane的
|
||||
private JPanel centerPane; // 此centerPane采用的是cardLayout布局,装载着mobileWidgetTable和mobileBodyWidgetTable
|
||||
private CardLayout cardLayout; // 卡片布局,选中参数面板时显示mobileWidgetTable,选中body时显示mobileBodyWidgetTable
|
||||
private JTableHeader header;//把表头单独get出来作为一个组件
|
||||
|
||||
public static WidgetPropertyPane getInstance() { |
||||
if (HOLDER.singleton == null) { |
||||
HOLDER.singleton = new WidgetPropertyPane(); |
||||
} |
||||
return HOLDER.singleton; |
||||
} |
||||
|
||||
public static WidgetPropertyPane getInstance(FormDesigner formEditor) { |
||||
HOLDER.singleton.setEditingFormDesigner(formEditor); |
||||
HOLDER.singleton.refreshDockingView(); |
||||
return HOLDER.singleton; |
||||
} |
||||
|
||||
private static class HOLDER { |
||||
private static WidgetPropertyPane singleton = new WidgetPropertyPane(); |
||||
} |
||||
|
||||
private WidgetPropertyPane() { |
||||
setLayout(FRGUIPaneFactory.createBorderLayout()); |
||||
} |
||||
|
||||
@Override |
||||
public String getViewTitle() { |
||||
return Inter.getLocText("Form-Widget_Property_Table"); |
||||
} |
||||
|
||||
@Override |
||||
public Icon getViewIcon() { |
||||
return BaseUtils.readIcon("/com/fr/design/images/m_report/attributes.png"); |
||||
} |
||||
|
||||
@Override |
||||
/** |
||||
* 绘制属性表tab |
||||
*/ |
||||
public void refreshDockingView() { |
||||
designer = this.getEditingFormDesigner(); |
||||
removeAll(); |
||||
if (designer == null) { |
||||
clearDockingView(); |
||||
return; |
||||
} |
||||
widgetPropertyTables = new ArrayList<AbstractPropertyTable>(); |
||||
|
||||
//依次创建属性表、事件表、移动端表,再将它们整合到TabPane中去
|
||||
this.createPropertyTable(); |
||||
this.createEventTable(); |
||||
this.createMobileWidgetTable(); |
||||
this.createTabPane(); |
||||
|
||||
this.initTables(); |
||||
} |
||||
|
||||
/** |
||||
* 初始化属性表,事件表,移动端拓展的属性表 |
||||
*/ |
||||
private void initTables() { |
||||
propertyTable.initPropertyGroups(null); |
||||
eventTable.refresh(); |
||||
for (AbstractPropertyTable propertyTable : widgetPropertyTables) { |
||||
propertyTable.initPropertyGroups(designer); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 创建属性表table |
||||
*/ |
||||
private void createPropertyTable() { |
||||
propertyTable = new WidgetPropertyTable(designer); |
||||
designer.addDesignerEditListener(new WidgetPropertyDesignerAdapter(propertyTable)); |
||||
propertyTable.setBorder(null); |
||||
psp = new UIScrollPane(propertyTable); // 用来装载属性表table
|
||||
psp.setBorder(null); |
||||
} |
||||
|
||||
/** |
||||
* 创建事件表(事件选项卡不是JTable) |
||||
*/ |
||||
private void createEventTable() { |
||||
eventTable = new EventPropertyTable(designer); |
||||
designer.addDesignerEditListener(new EventPropertyDesignerAdapter(eventTable)); |
||||
eventTable.setBorder(null); |
||||
esp = new UIScrollPane(eventTable); //用来装载事件table
|
||||
esp.setBorder(null); |
||||
} |
||||
|
||||
/** |
||||
* 创建移动端控件列表 |
||||
*/ |
||||
private void createMobileWidgetTable() { |
||||
//加上表头后,这里不再使用borderLayout布局,而采用BoxLayout布局
|
||||
wsp = FRGUIPaneFactory.createY_AXISBoxInnerContainer_S_Pane(); |
||||
wsp.setBorder(null); |
||||
mobileParaWidgetTable = new MobileParaWidgetTable(designer); |
||||
mobileWidgetTable = new MobileWidgetTable(designer); |
||||
designer.addDesignerEditListener(new mobileWidgetDesignerAdapter()); |
||||
centerPane = FRGUIPaneFactory.createCardLayout_S_Pane(); |
||||
cardLayout = (CardLayout) centerPane.getLayout(); |
||||
centerPane.add(mobileParaWidgetTable, PARA);// 采用卡片布局的容器必须指定卡片名字,如果没有卡片名字
|
||||
// 就会出现:Exception in thread "main" java.lang.IllegalArgumentException:
|
||||
// cannot add to layout: constraint must be a string
|
||||
// 第二个参数代表卡片的名字。后来show方法调用时通过名字找到要显示的卡片
|
||||
centerPane.add(mobileWidgetTable, BODY); //这两句代码,是把JTable放到一个JPanel中去了,表头不会显示,
|
||||
//只有放到JScrollPanel中去表头才能正常显示,这就是MobileWidgetTable中定义了表头却没有显示的原因!
|
||||
//解决方案:MobileWidgetTable实在无法直接放到JScrollPanel中去的时候,应该把表头get出来单独作为一个组件显示
|
||||
|
||||
if(hasSelectParaPane(designer)){ |
||||
cardLayout.show(centerPane,PARA); |
||||
header = mobileParaWidgetTable.getTableHeader(); |
||||
} else { |
||||
cardLayout.show(centerPane,BODY); |
||||
header = mobileWidgetTable.getTableHeader(); |
||||
} |
||||
downPanel = new UIScrollPane(centerPane); |
||||
downPanel.setBorder(new LineBorder(Color.GRAY)); |
||||
|
||||
//获取拓展移动端属性tab
|
||||
WidgetPropertyUIProvider[] widgetAttrProviders = getExtraPropertyUIProviders(); |
||||
|
||||
addWidgetAttr(widgetAttrProviders); |
||||
} |
||||
|
||||
/** |
||||
* 将属性表,事件表,移动端控件列表整合到TabPane里面去 |
||||
*/ |
||||
private void createTabPane() { |
||||
UITabbedPane tabbedPane = new UITabbedPane(); // tab选项卡容器
|
||||
initTabPane(tabbedPane); |
||||
add(tabbedPane, BorderLayout.CENTER); |
||||
} |
||||
|
||||
/** |
||||
* 获取当前控件扩展的属性tab |
||||
* 来源有两个: |
||||
* 1, 各个控件从各自的Xcreator里扩展(例如手机重布局的tab就是从Xcreator中扩展的); |
||||
* 2, 所有的控件从插件里扩展. |
||||
* |
||||
* @return 扩展的tab |
||||
*/ |
||||
private WidgetPropertyUIProvider[] getExtraPropertyUIProviders() { |
||||
FormSelection selection = designer.getSelectionModel().getSelection(); |
||||
WidgetPropertyUIProvider[] embeddedPropertyUIProviders = null; |
||||
if (selection != null && selection.getSelectedCreator() != null) { |
||||
embeddedPropertyUIProviders = selection.getSelectedCreator().getWidgetPropertyUIProviders(); |
||||
} |
||||
Set<WidgetPropertyUIProvider> set = ExtraDesignClassManager.getInstance().getArray(WidgetPropertyUIProvider.XML_TAG); |
||||
return ArrayUtils.addAll(embeddedPropertyUIProviders, set.toArray(new WidgetPropertyUIProvider[set.size()])); |
||||
} |
||||
|
||||
/** |
||||
* 判断是将拓展的tab放入属性表还是将原来的tab放入属性表 |
||||
* @param widgetAttrProviders 拓展的tab |
||||
*/ |
||||
private void addWidgetAttr(WidgetPropertyUIProvider[] widgetAttrProviders) { |
||||
if (widgetAttrProviders.length == 0) { // 判断有没有拓展的tab,没有就使用原来的
|
||||
wsp.add(header); |
||||
wsp.add(downPanel); |
||||
} else { |
||||
for (WidgetPropertyUIProvider widgetAttrProvider : widgetAttrProviders) { |
||||
AbstractPropertyTable propertyTable = widgetAttrProvider.createWidgetAttrTable(); |
||||
widgetPropertyTables.add(propertyTable); |
||||
designer.addDesignerEditListener(new WidgetPropertyDesignerAdapter(propertyTable)); |
||||
UIScrollPane uiScrollPane = new UIScrollPane(getExtraBodyTable(propertyTable)); |
||||
wsp.add(uiScrollPane); |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 如果是body的拓展属性表,那么要额外加上一张控件顺序表 |
||||
* @return |
||||
*/ |
||||
private Component getExtraBodyTable(AbstractPropertyTable abstractPropertyTable) { |
||||
Widget selection = designer.getSelectionModel().getSelection().getSelectedCreator().toData(); |
||||
if (selection.getWidgetName().equals("body")) { |
||||
JPanel jPanel = FRGUIPaneFactory.createY_AXISBoxInnerContainer_S_Pane(); |
||||
jPanel.add(abstractPropertyTable); |
||||
MobileWidgetTable mobileWidgetTable = new MobileWidgetTable(designer); |
||||
jPanel.add(mobileWidgetTable.getTableHeader()); |
||||
jPanel.add(mobileWidgetTable); |
||||
return jPanel; |
||||
} |
||||
return abstractPropertyTable; |
||||
} |
||||
|
||||
private void initTabPane(UITabbedPane tabbedPane) { |
||||
tabbedPane.setOpaque(true); |
||||
tabbedPane.setBorder(null); |
||||
tabbedPane.setTabPlacement(SwingConstants.BOTTOM); |
||||
tabbedPane.addTab(Inter.getLocText("FR-Designer_Properties"), psp); |
||||
tabbedPane.addTab(Inter.getLocText("FR-Designer_Event"), esp); |
||||
tabbedPane.addTab(Inter.getLocText("FR-Widget_Mobile_Terminal"), wsp); |
||||
} |
||||
|
||||
//
|
||||
/** |
||||
* 选中的组件是否在参数面板里 |
||||
* @param designer 设计器 |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean hasSelectParaPane(FormDesigner designer){ |
||||
XCreator xCreator = designer.getSelectionModel().getSelection().getSelectedCreator(); |
||||
if(xCreator == null){ |
||||
xCreator = designer.getRootComponent(); |
||||
} |
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(xCreator); |
||||
return xCreator.acceptType(XWParameterLayout.class) || container.acceptType(XWParameterLayout.class); |
||||
} |
||||
|
||||
public void setEditingFormDesigner(BaseFormDesigner editor) { |
||||
FormDesigner fd = (FormDesigner) editor; |
||||
super.setEditingFormDesigner(fd); |
||||
} |
||||
|
||||
private void clearDockingView() { |
||||
propertyTable = null; |
||||
eventTable = null; |
||||
if (widgetPropertyTables != null) { |
||||
widgetPropertyTables.clear(); |
||||
} |
||||
JScrollPane psp = new JScrollPane(); |
||||
psp.setBorder(null); |
||||
this.add(psp, BorderLayout.CENTER); |
||||
} |
||||
|
||||
/** |
||||
* 属性表监听界面事件(编辑,删除,选中,改变大小) |
||||
*/ |
||||
private class WidgetPropertyDesignerAdapter implements DesignerEditListener { |
||||
AbstractPropertyTable propertyTable; |
||||
|
||||
WidgetPropertyDesignerAdapter(AbstractPropertyTable propertyTable) { |
||||
this.propertyTable = propertyTable; |
||||
} |
||||
|
||||
@Override |
||||
public void fireCreatorModified(DesignerEvent evt) { |
||||
if (evt.getCreatorEventID() == DesignerEvent.CREATOR_EDITED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_DELETED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_SELECTED) { |
||||
propertyTable.initPropertyGroups(designer); |
||||
} else if (evt.getCreatorEventID() == DesignerEvent.CREATOR_RESIZED) { |
||||
repaint(); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public boolean equals(Object o) { |
||||
return o instanceof WidgetPropertyDesignerAdapter && ((WidgetPropertyDesignerAdapter) o).propertyTable == this.propertyTable; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 事件表监听界面事件(编辑,选中) |
||||
*/ |
||||
private class EventPropertyDesignerAdapter implements DesignerEditListener { |
||||
EventPropertyTable propertyTable; |
||||
|
||||
EventPropertyDesignerAdapter(EventPropertyTable eventTable) { |
||||
this.propertyTable = eventTable; |
||||
} |
||||
|
||||
@Override |
||||
public void fireCreatorModified(DesignerEvent evt) { |
||||
if (evt.getCreatorEventID() == DesignerEvent.CREATOR_EDITED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_SELECTED) { |
||||
propertyTable.refresh(); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public boolean equals(Object o) { |
||||
return o instanceof EventPropertyDesignerAdapter; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 移动端属性表监听界面事件(改变大小,编辑,选中,增加控件) |
||||
*/ |
||||
private class mobileWidgetDesignerAdapter implements DesignerEditListener { |
||||
|
||||
mobileWidgetDesignerAdapter() { |
||||
} |
||||
|
||||
/** |
||||
* 响应界面改变事件 |
||||
* @param evt 事件 |
||||
*/ |
||||
public void fireCreatorModified(DesignerEvent evt) { |
||||
if (evt.getCreatorEventID() == DesignerEvent.CREATOR_RESIZED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_EDITED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_SELECTED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_ADDED) { |
||||
int value = downPanel.getVerticalScrollBar().getValue(); |
||||
if(hasSelectParaPane(getEditingFormDesigner())){ |
||||
cardLayout.show(centerPane,PARA); |
||||
mobileParaWidgetTable.refreshData(); |
||||
} else { |
||||
cardLayout.show(centerPane,BODY); |
||||
mobileWidgetTable.refreshData(); |
||||
} |
||||
//出现滚动条
|
||||
downPanel.doLayout(); |
||||
//控件列表选中某组件,触发表单中选中控件,选中事件又触发列表刷新,滚动条回到0
|
||||
//此处设置滚动条值为刷新前
|
||||
downPanel.getVerticalScrollBar().setValue(value); |
||||
} |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public Location preferredLocation() { |
||||
return Location.WEST_BELOW; |
||||
} |
||||
package com.fr.design.mainframe; |
||||
|
||||
import com.fr.base.BaseUtils; |
||||
import com.fr.design.ExtraDesignClassManager; |
||||
import com.fr.design.designer.beans.events.DesignerEditListener; |
||||
import com.fr.design.designer.beans.events.DesignerEvent; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XCreatorUtils; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.designer.creator.XWParameterLayout; |
||||
import com.fr.design.designer.properties.EventPropertyTable; |
||||
import com.fr.design.designer.properties.WidgetPropertyTable; |
||||
import com.fr.design.fun.WidgetPropertyUIProvider; |
||||
import com.fr.design.gui.frpane.UITabbedPane; |
||||
import com.fr.design.gui.icontainer.UIScrollPane; |
||||
import com.fr.design.gui.itable.AbstractPropertyTable; |
||||
import com.fr.design.layout.FRGUIPaneFactory; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.general.Inter; |
||||
import com.fr.stable.ArrayUtils; |
||||
|
||||
import javax.swing.*; |
||||
import javax.swing.border.LineBorder; |
||||
import javax.swing.table.JTableHeader; |
||||
import java.awt.*; |
||||
import java.util.ArrayList; |
||||
import java.util.List; |
||||
import java.util.Set; |
||||
|
||||
/** |
||||
* 控件属性表绘制 |
||||
* Modified by fanglei |
||||
*/ |
||||
public class WidgetPropertyPane extends FormDockView implements BaseWidgetPropertyPane { |
||||
|
||||
private static final String PARA = "para"; |
||||
private static final String BODY = "body"; |
||||
private WidgetPropertyTable propertyTable; // 控件的属性表
|
||||
private EventPropertyTable eventTable; // 控件的事件表
|
||||
private List<AbstractPropertyTable> widgetPropertyTables; // 这个变量应该是保存控件拓展的属性tab
|
||||
private FormDesigner designer; // 当前designer
|
||||
private UIScrollPane psp; // 用来装载属性表table的容器
|
||||
private UIScrollPane esp; //用来装载事件table的容器
|
||||
private JPanel wsp; // 装载移动端tab的容器,包括移动端属性表和控件拓展的移动端属性表
|
||||
private MobileParaWidgetTable mobileParaWidgetTable; // 参数面板的移动端属性tab(和body的移动端属性tab区别是没有标签名column)
|
||||
private MobileWidgetTable mobileWidgetTable; // body的移动端属性tab
|
||||
private UIScrollPane downPanel; // 这个滚动容器是用于装载centerPane的
|
||||
private JPanel centerPane; // 此centerPane采用的是cardLayout布局,装载着mobileWidgetTable和mobileBodyWidgetTable
|
||||
private CardLayout cardLayout; // 卡片布局,选中参数面板时显示mobileWidgetTable,选中body时显示mobileBodyWidgetTable
|
||||
private JTableHeader header;//把表头单独get出来作为一个组件
|
||||
|
||||
public static WidgetPropertyPane getInstance() { |
||||
if (HOLDER.singleton == null) { |
||||
HOLDER.singleton = new WidgetPropertyPane(); |
||||
} |
||||
return HOLDER.singleton; |
||||
} |
||||
|
||||
public static WidgetPropertyPane getInstance(FormDesigner formEditor) { |
||||
HOLDER.singleton.setEditingFormDesigner(formEditor); |
||||
HOLDER.singleton.refreshDockingView(); |
||||
return HOLDER.singleton; |
||||
} |
||||
|
||||
private static class HOLDER { |
||||
private static WidgetPropertyPane singleton = new WidgetPropertyPane(); |
||||
} |
||||
|
||||
private WidgetPropertyPane() { |
||||
setLayout(FRGUIPaneFactory.createBorderLayout()); |
||||
} |
||||
|
||||
@Override |
||||
public String getViewTitle() { |
||||
return Inter.getLocText("Form-Widget_Property_Table"); |
||||
} |
||||
|
||||
@Override |
||||
public Icon getViewIcon() { |
||||
return BaseUtils.readIcon("/com/fr/design/images/m_report/attributes.png"); |
||||
} |
||||
|
||||
@Override |
||||
/** |
||||
* 绘制属性表tab |
||||
*/ |
||||
public void refreshDockingView() { |
||||
designer = this.getEditingFormDesigner(); |
||||
removeAll(); |
||||
if (designer == null) { |
||||
clearDockingView(); |
||||
return; |
||||
} |
||||
widgetPropertyTables = new ArrayList<AbstractPropertyTable>(); |
||||
|
||||
//依次创建属性表、事件表、移动端表,再将它们整合到TabPane中去
|
||||
this.createPropertyTable(); |
||||
this.createEventTable(); |
||||
this.createMobileWidgetTable(); |
||||
this.createTabPane(); |
||||
|
||||
this.initTables(); |
||||
} |
||||
|
||||
/** |
||||
* 初始化属性表,事件表,移动端拓展的属性表 |
||||
*/ |
||||
private void initTables() { |
||||
propertyTable.initPropertyGroups(null); |
||||
eventTable.refresh(); |
||||
for (AbstractPropertyTable propertyTable : widgetPropertyTables) { |
||||
propertyTable.initPropertyGroups(designer); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 创建属性表table |
||||
*/ |
||||
private void createPropertyTable() { |
||||
propertyTable = new WidgetPropertyTable(designer); |
||||
designer.addDesignerEditListener(new WidgetPropertyDesignerAdapter(propertyTable)); |
||||
propertyTable.setBorder(null); |
||||
psp = new UIScrollPane(propertyTable); // 用来装载属性表table
|
||||
psp.setBorder(null); |
||||
} |
||||
|
||||
/** |
||||
* 创建事件表(事件选项卡不是JTable) |
||||
*/ |
||||
private void createEventTable() { |
||||
eventTable = new EventPropertyTable(designer); |
||||
designer.addDesignerEditListener(new EventPropertyDesignerAdapter(eventTable)); |
||||
eventTable.setBorder(null); |
||||
esp = new UIScrollPane(eventTable); //用来装载事件table
|
||||
esp.setBorder(null); |
||||
} |
||||
|
||||
/** |
||||
* 创建移动端控件列表 |
||||
*/ |
||||
private void createMobileWidgetTable() { |
||||
//加上表头后,这里不再使用borderLayout布局,而采用BoxLayout布局
|
||||
wsp = FRGUIPaneFactory.createY_AXISBoxInnerContainer_S_Pane(); |
||||
wsp.setBorder(null); |
||||
mobileParaWidgetTable = new MobileParaWidgetTable(designer); |
||||
mobileWidgetTable = new MobileWidgetTable(designer); |
||||
designer.addDesignerEditListener(new mobileWidgetDesignerAdapter()); |
||||
centerPane = FRGUIPaneFactory.createCardLayout_S_Pane(); |
||||
cardLayout = (CardLayout) centerPane.getLayout(); |
||||
centerPane.add(mobileParaWidgetTable, PARA); |
||||
// 采用卡片布局的容器必须指定卡片名字,如果没有卡片名字
|
||||
// 就会出现:Exception in thread "main" java.lang.IllegalArgumentException:
|
||||
// cannot add to layout: constraint must be a string
|
||||
// 第二个参数代表卡片的名字。后来show方法调用时通过名字找到要显示的卡片
|
||||
centerPane.add(mobileWidgetTable, BODY); //这两句代码,是把JTable放到一个JPanel中去了,表头不会显示,
|
||||
//只有放到JScrollPanel中去表头才能正常显示,这就是MobileWidgetTable中定义了表头却没有显示的原因!
|
||||
//解决方案:MobileWidgetTable实在无法直接放到JScrollPanel中去的时候,应该把表头get出来单独作为一个组件显示
|
||||
|
||||
if (hasSelectParaPane(designer)) { |
||||
cardLayout.show(centerPane, PARA); |
||||
header = mobileParaWidgetTable.getTableHeader(); |
||||
} else { |
||||
cardLayout.show(centerPane, BODY); |
||||
header = mobileWidgetTable.getTableHeader(); |
||||
} |
||||
downPanel = new UIScrollPane(centerPane); |
||||
downPanel.setBorder(new LineBorder(Color.GRAY)); |
||||
|
||||
//获取拓展移动端属性tab
|
||||
WidgetPropertyUIProvider[] widgetAttrProviders = getExtraPropertyUIProviders(); |
||||
|
||||
addWidgetAttr(widgetAttrProviders); |
||||
} |
||||
|
||||
/** |
||||
* 将属性表,事件表,移动端控件列表整合到TabPane里面去 |
||||
*/ |
||||
private void createTabPane() { |
||||
UITabbedPane tabbedPane = new UITabbedPane(); // tab选项卡容器
|
||||
initTabPane(tabbedPane); |
||||
add(tabbedPane, BorderLayout.CENTER); |
||||
} |
||||
|
||||
/** |
||||
* 获取当前控件扩展的属性tab |
||||
* 来源有两个: |
||||
* 1, 各个控件从各自的Xcreator里扩展(例如手机重布局的tab就是从Xcreator中扩展的); |
||||
* 2, 所有的控件从插件里扩展. |
||||
* |
||||
* @return 扩展的tab |
||||
*/ |
||||
private WidgetPropertyUIProvider[] getExtraPropertyUIProviders() { |
||||
FormSelection selection = designer.getSelectionModel().getSelection(); |
||||
WidgetPropertyUIProvider[] embeddedPropertyUIProviders = null; |
||||
if (selection != null && selection.getSelectedCreator() != null) { |
||||
embeddedPropertyUIProviders = selection.getSelectedCreator().getWidgetPropertyUIProviders(); |
||||
} |
||||
Set<WidgetPropertyUIProvider> set = ExtraDesignClassManager.getInstance().getArray(WidgetPropertyUIProvider.XML_TAG); |
||||
return ArrayUtils.addAll(embeddedPropertyUIProviders, set.toArray(new WidgetPropertyUIProvider[set.size()])); |
||||
} |
||||
|
||||
/** |
||||
* 判断是将拓展的tab放入属性表还是将原来的tab放入属性表 |
||||
* |
||||
* @param widgetAttrProviders 拓展的tab |
||||
*/ |
||||
private void addWidgetAttr(WidgetPropertyUIProvider[] widgetAttrProviders) { |
||||
if (widgetAttrProviders.length == 0) { // 判断有没有拓展的tab,没有就使用原来的
|
||||
wsp.add(header); |
||||
wsp.add(downPanel); |
||||
} else { |
||||
for (WidgetPropertyUIProvider widgetAttrProvider : widgetAttrProviders) { |
||||
AbstractPropertyTable propertyTable = widgetAttrProvider.createWidgetAttrTable(); |
||||
widgetPropertyTables.add(propertyTable); |
||||
designer.addDesignerEditListener(new WidgetPropertyDesignerAdapter(propertyTable)); |
||||
UIScrollPane uiScrollPane = new UIScrollPane(getExtraBodyTable(propertyTable)); |
||||
wsp.add(uiScrollPane); |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 如果是body的拓展属性表,那么要额外加上一张控件顺序表 |
||||
* |
||||
* @return |
||||
*/ |
||||
private Component getExtraBodyTable(AbstractPropertyTable abstractPropertyTable) { |
||||
Widget selection = designer.getSelectionModel().getSelection().getSelectedCreator().toData(); |
||||
if (selection.getWidgetName().equals("body")) { |
||||
JPanel jPanel = FRGUIPaneFactory.createY_AXISBoxInnerContainer_S_Pane(); |
||||
jPanel.add(abstractPropertyTable); |
||||
MobileWidgetTable mobileWidgetTable = new MobileWidgetTable(designer); |
||||
jPanel.add(mobileWidgetTable.getTableHeader()); |
||||
jPanel.add(mobileWidgetTable); |
||||
return jPanel; |
||||
} |
||||
return abstractPropertyTable; |
||||
} |
||||
|
||||
private void initTabPane(UITabbedPane tabbedPane) { |
||||
tabbedPane.setOpaque(true); |
||||
tabbedPane.setBorder(null); |
||||
tabbedPane.setTabPlacement(SwingConstants.BOTTOM); |
||||
tabbedPane.addTab(Inter.getLocText("FR-Designer_Properties"), psp); |
||||
tabbedPane.addTab(Inter.getLocText("FR-Designer_Event"), esp); |
||||
tabbedPane.addTab(Inter.getLocText("FR-Widget_Mobile_Terminal"), wsp); |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 选中的组件是否在参数面板里 |
||||
* |
||||
* @param designer 设计器 |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean hasSelectParaPane(FormDesigner designer) { |
||||
XCreator xCreator = designer.getSelectionModel().getSelection().getSelectedCreator(); |
||||
if (xCreator == null) { |
||||
xCreator = designer.getRootComponent(); |
||||
} |
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(xCreator); |
||||
//TODO container可能为空,引发空指针异常
|
||||
return xCreator.acceptType(XWParameterLayout.class) || container.acceptType(XWParameterLayout.class); |
||||
} |
||||
|
||||
public void setEditingFormDesigner(BaseFormDesigner editor) { |
||||
FormDesigner fd = (FormDesigner) editor; |
||||
super.setEditingFormDesigner(fd); |
||||
} |
||||
|
||||
private void clearDockingView() { |
||||
propertyTable = null; |
||||
eventTable = null; |
||||
if (widgetPropertyTables != null) { |
||||
widgetPropertyTables.clear(); |
||||
} |
||||
JScrollPane psp = new JScrollPane(); |
||||
psp.setBorder(null); |
||||
this.add(psp, BorderLayout.CENTER); |
||||
} |
||||
|
||||
/** |
||||
* 属性表监听界面事件(编辑,删除,选中,改变大小) |
||||
*/ |
||||
private class WidgetPropertyDesignerAdapter implements DesignerEditListener { |
||||
AbstractPropertyTable propertyTable; |
||||
|
||||
WidgetPropertyDesignerAdapter(AbstractPropertyTable propertyTable) { |
||||
this.propertyTable = propertyTable; |
||||
} |
||||
|
||||
@Override |
||||
public void fireCreatorModified(DesignerEvent evt) { |
||||
if (evt.getCreatorEventID() == DesignerEvent.CREATOR_EDITED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_DELETED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_SELECTED) { |
||||
propertyTable.initPropertyGroups(designer); |
||||
} else if (evt.getCreatorEventID() == DesignerEvent.CREATOR_RESIZED) { |
||||
repaint(); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public boolean equals(Object o) { |
||||
return o instanceof WidgetPropertyDesignerAdapter && ((WidgetPropertyDesignerAdapter) o).propertyTable == this.propertyTable; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 事件表监听界面事件(编辑,选中) |
||||
*/ |
||||
private class EventPropertyDesignerAdapter implements DesignerEditListener { |
||||
EventPropertyTable propertyTable; |
||||
|
||||
EventPropertyDesignerAdapter(EventPropertyTable eventTable) { |
||||
this.propertyTable = eventTable; |
||||
} |
||||
|
||||
@Override |
||||
public void fireCreatorModified(DesignerEvent evt) { |
||||
if (evt.getCreatorEventID() == DesignerEvent.CREATOR_EDITED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_SELECTED) { |
||||
propertyTable.refresh(); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public boolean equals(Object o) { |
||||
return o instanceof EventPropertyDesignerAdapter; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 移动端属性表监听界面事件(改变大小,编辑,选中,增加控件) |
||||
*/ |
||||
private class mobileWidgetDesignerAdapter implements DesignerEditListener { |
||||
|
||||
mobileWidgetDesignerAdapter() { |
||||
} |
||||
|
||||
/** |
||||
* 响应界面改变事件 |
||||
* |
||||
* @param evt 事件 |
||||
*/ |
||||
public void fireCreatorModified(DesignerEvent evt) { |
||||
if (evt.getCreatorEventID() == DesignerEvent.CREATOR_RESIZED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_EDITED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_SELECTED |
||||
|| evt.getCreatorEventID() == DesignerEvent.CREATOR_ADDED) { |
||||
int value = downPanel.getVerticalScrollBar().getValue(); |
||||
if (hasSelectParaPane(getEditingFormDesigner())) { |
||||
cardLayout.show(centerPane, PARA); |
||||
mobileParaWidgetTable.refreshData(); |
||||
} else { |
||||
cardLayout.show(centerPane, BODY); |
||||
mobileWidgetTable.refreshData(); |
||||
} |
||||
//出现滚动条
|
||||
downPanel.doLayout(); |
||||
//控件列表选中某组件,触发表单中选中控件,选中事件又触发列表刷新,滚动条回到0
|
||||
//此处设置滚动条值为刷新前
|
||||
downPanel.getVerticalScrollBar().setValue(value); |
||||
} |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public Location preferredLocation() { |
||||
return Location.WEST_BELOW; |
||||
} |
||||
} |
@ -0,0 +1,42 @@
|
||||
package com.fr.design.mainframe.templateinfo; |
||||
|
||||
import com.fr.form.main.Form; |
||||
import com.fr.form.ui.container.WLayout; |
||||
|
||||
/** |
||||
* Created by plough on 2017/3/17. |
||||
*/ |
||||
public class JFormProcessInfo extends TemplateProcessInfo<Form> { |
||||
public JFormProcessInfo(Form form) { |
||||
super(form); |
||||
} |
||||
|
||||
// 获取模板类型
|
||||
public int getReportType() { |
||||
return 2; |
||||
} |
||||
|
||||
// 获取模板格子数
|
||||
public int getCellCount() { |
||||
return 0; |
||||
} |
||||
// 获取模板悬浮元素个数
|
||||
public int getFloatCount() { |
||||
return 0; |
||||
} |
||||
// 获取模板聚合块个数
|
||||
public int getBlockCount() { |
||||
return 0; |
||||
} |
||||
// 获取模板控件数
|
||||
public int getWidgetCount() { |
||||
int widgetCount = 0; |
||||
for (int i = 0; i < template.getContainer().getWidgetCount(); i++) { |
||||
WLayout wl = (WLayout) template.getContainer().getWidget(i); |
||||
widgetCount += wl.getWidgetCount(); |
||||
} |
||||
int a = 1; |
||||
int b = 2; |
||||
return widgetCount; |
||||
} |
||||
} |
@ -0,0 +1,57 @@
|
||||
package com.fr.design.mainframe.widget.accessibles; |
||||
|
||||
import com.fr.design.dialog.BasicDialog; |
||||
import com.fr.design.dialog.DialogActionAdapter; |
||||
import com.fr.design.gui.core.WidgetOption; |
||||
import com.fr.design.mainframe.FormWebWidgetConstants; |
||||
import com.fr.design.mainframe.widget.editors.ElementCaseToolBarPane; |
||||
import com.fr.design.mainframe.widget.wrappers.ElementCaseToolBarWrapper; |
||||
import com.fr.form.web.FormToolBarManager; |
||||
import com.fr.stable.ArrayUtils; |
||||
|
||||
import javax.swing.*; |
||||
import java.util.ArrayList; |
||||
import java.util.Arrays; |
||||
import java.util.List; |
||||
|
||||
/** |
||||
* Created by harry on 2017-2-23. |
||||
*/ |
||||
public class AccessibleElementCaseToolBarEditor extends UneditableAccessibleEditor { |
||||
private ElementCaseToolBarPane pane; |
||||
|
||||
public AccessibleElementCaseToolBarEditor() { |
||||
super(new ElementCaseToolBarWrapper()); |
||||
} |
||||
|
||||
@Override |
||||
protected void showEditorPane() { |
||||
if (pane == null) { |
||||
pane = new ElementCaseToolBarPane(); |
||||
pane.setDefaultToolBar(getDefaultToolBarManager(), getToolBarInstance()); |
||||
} |
||||
BasicDialog dlg = pane.showToolBarWindow(SwingUtilities.getWindowAncestor(this), new DialogActionAdapter() { |
||||
|
||||
@Override |
||||
public void doOk() { |
||||
setValue(pane.updateBean()); |
||||
fireStateChanged(); |
||||
} |
||||
}); |
||||
FormToolBarManager[] managers = (FormToolBarManager[]) getValue(); |
||||
pane.setCheckBoxSelected(ArrayUtils.isNotEmpty(managers)); |
||||
pane.populateBean((FormToolBarManager[]) getValue()); |
||||
dlg.setVisible(true); |
||||
} |
||||
|
||||
private FormToolBarManager getDefaultToolBarManager() { |
||||
return FormToolBarManager.createDefaultToolBar(); |
||||
} |
||||
|
||||
private WidgetOption[] getToolBarInstance() { |
||||
List<WidgetOption> defaultOptions = Arrays.asList(FormWebWidgetConstants.getFormElementCaseToolBarInstance()); |
||||
List<WidgetOption> options = new ArrayList<WidgetOption>(); |
||||
options.addAll(defaultOptions); |
||||
return options.toArray(new WidgetOption[options.size()]); |
||||
} |
||||
} |
@ -0,0 +1,13 @@
|
||||
package com.fr.design.mainframe.widget.editors; |
||||
|
||||
import com.fr.design.mainframe.widget.accessibles.AccessibleElementCaseToolBarEditor; |
||||
import com.fr.design.mainframe.widget.accessibles.AccessiblePropertyEditor; |
||||
|
||||
/** |
||||
* Created by harry on 2017-2-23. |
||||
*/ |
||||
public class ElementCaseToolBarEditor extends AccessiblePropertyEditor { |
||||
public ElementCaseToolBarEditor() { |
||||
super(new AccessibleElementCaseToolBarEditor()); |
||||
} |
||||
} |
@ -0,0 +1,304 @@
|
||||
package com.fr.design.mainframe.widget.editors; |
||||
|
||||
import com.fr.design.beans.BasicBeanPane; |
||||
import com.fr.design.gui.core.WidgetOption; |
||||
import com.fr.design.gui.ibutton.UIButton; |
||||
import com.fr.design.gui.icheckbox.UICheckBox; |
||||
import com.fr.design.gui.ilable.UILabel; |
||||
import com.fr.design.layout.FRGUIPaneFactory; |
||||
import com.fr.design.mainframe.DesignerContext; |
||||
import com.fr.design.mainframe.FormSettingToolBar; |
||||
import com.fr.design.mainframe.FormToolBarButton; |
||||
import com.fr.design.mainframe.FormToolBarPane; |
||||
import com.fr.form.ui.Widget; |
||||
import com.fr.form.web.FormToolBarManager; |
||||
import com.fr.form.web.Location; |
||||
import com.fr.general.IOUtils; |
||||
import com.fr.general.Inter; |
||||
import com.fr.stable.ArrayUtils; |
||||
import com.fr.stable.Constants; |
||||
import com.fr.stable.StringUtils; |
||||
|
||||
import javax.swing.*; |
||||
import javax.swing.table.DefaultTableCellRenderer; |
||||
import javax.swing.table.DefaultTableModel; |
||||
import java.awt.*; |
||||
import java.awt.event.ActionEvent; |
||||
import java.awt.event.ActionListener; |
||||
import java.awt.event.MouseAdapter; |
||||
import java.awt.event.MouseEvent; |
||||
import java.awt.image.ImageObserver; |
||||
import java.util.*; |
||||
import java.util.List; |
||||
|
||||
/** |
||||
* Created by harry on 2017-2-23. |
||||
*/ |
||||
public class ElementCaseToolBarPane extends BasicBeanPane<FormToolBarManager[]> { |
||||
private static final int COLUMN = 2; |
||||
private int row = 6; |
||||
private DefaultTableModel toolbarButtonTableModel; |
||||
private JTable layoutTable; |
||||
private UICheckBox isUseToolBarCheckBox = new UICheckBox(Inter.getLocText("FR-Designer_Use_ToolBar") + ":"); // 是否使用工具栏
|
||||
|
||||
private FormToolBarPane northToolBar; |
||||
private FormToolBarPane southToolBar; |
||||
private FormToolBarManager defaultToolBar; |
||||
|
||||
|
||||
public ElementCaseToolBarPane() { |
||||
toolbarButtonTableModel = new TableModel(row ,COLUMN); |
||||
this.setLayout(FRGUIPaneFactory.createBorderLayout()); |
||||
JPanel north = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
UIButton defaultButton = new UIButton(Inter.getLocText("FR-Designer_Restore_Default")); |
||||
// 恢复默认按钮
|
||||
defaultButton.addActionListener(new ActionListener() { |
||||
public void actionPerformed(ActionEvent e) { |
||||
northToolBar.removeButtonList(); |
||||
northToolBar.removeAll(); |
||||
southToolBar.removeButtonList(); |
||||
southToolBar.removeAll(); |
||||
if (defaultToolBar == null) { |
||||
return; |
||||
} |
||||
FormToolBarManager toolBarManager = defaultToolBar; |
||||
toolBarManager.setToolBarLocation(Location.createTopEmbedLocation()); |
||||
FormToolBarManager[] tbm = new FormToolBarManager[] { toolBarManager }; |
||||
populateBean(tbm); |
||||
ElementCaseToolBarPane.this.repaint(); |
||||
} |
||||
}); |
||||
|
||||
north.add(isUseToolBarCheckBox, BorderLayout.WEST); |
||||
JPanel aa = FRGUIPaneFactory.createRightFlowInnerContainer_S_Pane(); |
||||
aa.add(defaultButton); |
||||
north.add(aa, BorderLayout.CENTER); |
||||
this.add(north, BorderLayout.NORTH); |
||||
|
||||
northToolBar = new FormToolBarPane(); |
||||
northToolBar.setPreferredSize(new Dimension(ImageObserver.WIDTH, 26)); |
||||
northToolBar.setBackground(Color.lightGray); |
||||
|
||||
UIButton topButton = new UIButton(IOUtils.readIcon("com/fr/design/images/arrow/arrow_up.png")); |
||||
topButton.setBorder(null); |
||||
topButton.setOpaque(false); |
||||
topButton.setContentAreaFilled(false); |
||||
topButton.setFocusPainted(false); |
||||
topButton.setRequestFocusEnabled(false); |
||||
topButton.addActionListener(new ActionListener() { |
||||
|
||||
@Override |
||||
public void actionPerformed(ActionEvent e) { |
||||
if (isSelectedtable()) { |
||||
WidgetOption no = (WidgetOption)layoutTable.getValueAt(layoutTable.getSelectedRow(), layoutTable.getSelectedColumn()); |
||||
Widget widget = no.createWidget(); |
||||
FormToolBarButton tb = new FormToolBarButton(no.optionIcon(), widget); |
||||
tb.setNameOption(no); |
||||
northToolBar.add(tb); |
||||
northToolBar.validate(); |
||||
northToolBar.repaint(); |
||||
} else { |
||||
JOptionPane.showMessageDialog(DesignerContext.getDesignerFrame(), Inter.getLocText("FR-Designer_ChooseOneButton")); |
||||
} |
||||
} |
||||
}); |
||||
|
||||
UIButton downButton = new UIButton(IOUtils.readIcon("com/fr/design/images/arrow/arrow_down.png")); |
||||
downButton.setBorder(null); |
||||
downButton.setMargin(null); |
||||
downButton.setOpaque(false); |
||||
downButton.setContentAreaFilled(false); |
||||
downButton.setFocusPainted(false); |
||||
downButton.setRequestFocusEnabled(false); |
||||
downButton.addActionListener(new ActionListener() { |
||||
|
||||
@Override |
||||
public void actionPerformed(ActionEvent e) { |
||||
if (isSelectedtable()) { |
||||
WidgetOption no = (WidgetOption)layoutTable.getValueAt(layoutTable.getSelectedRow(), layoutTable.getSelectedColumn()); |
||||
Widget widget = no.createWidget(); |
||||
FormToolBarButton tb = new FormToolBarButton(no.optionIcon(), widget); |
||||
tb.setNameOption(no); |
||||
southToolBar.add(tb); |
||||
southToolBar.validate(); |
||||
southToolBar.repaint(); |
||||
} else { |
||||
JOptionPane.showMessageDialog(DesignerContext.getDesignerFrame(), Inter.getLocText("FR-Designer_ChooseOneButton")); |
||||
} |
||||
} |
||||
}); |
||||
|
||||
initLayoutTable(); |
||||
|
||||
JPanel center = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
center.setBackground(Color.white); |
||||
center.add(topButton, BorderLayout.NORTH); |
||||
JPanel small = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
small.setBackground(Color.white); |
||||
small.add(new UILabel(StringUtils.BLANK), BorderLayout.NORTH); |
||||
small.add(layoutTable, BorderLayout.CENTER); |
||||
center.add(small, BorderLayout.CENTER); |
||||
center.add(downButton, BorderLayout.SOUTH); |
||||
southToolBar = new FormToolBarPane(); |
||||
southToolBar.setPreferredSize(new Dimension(ImageObserver.WIDTH, 26)); |
||||
southToolBar.setBackground(Color.lightGray); |
||||
JPanel movePane = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
JPanel northContentPane = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
FormSettingToolBar top = new FormSettingToolBar(Inter.getLocText("FR-Designer_ToolBar_Top"), northToolBar); |
||||
northContentPane.add(top, BorderLayout.EAST); |
||||
northContentPane.add(northToolBar, BorderLayout.CENTER); |
||||
northContentPane.setBackground(Color.lightGray); |
||||
|
||||
JPanel southContentPane = FRGUIPaneFactory.createBorderLayout_S_Pane(); |
||||
FormSettingToolBar bottom = new FormSettingToolBar(Inter.getLocText("FR-Designer_ToolBar_Bottom"), southToolBar); |
||||
southContentPane.add(bottom, BorderLayout.EAST); |
||||
southContentPane.add(southToolBar, BorderLayout.CENTER); |
||||
southContentPane.setBackground(Color.lightGray); |
||||
|
||||
movePane.add(northContentPane, BorderLayout.NORTH); |
||||
movePane.add(center, BorderLayout.CENTER); |
||||
movePane.add(southContentPane, BorderLayout.SOUTH); |
||||
|
||||
this.add(new JScrollPane(movePane), BorderLayout.CENTER); |
||||
|
||||
isUseToolBarCheckBox.setSelected(false); |
||||
} |
||||
|
||||
private void initLayoutTable() { |
||||
layoutTable = new JTable(toolbarButtonTableModel); |
||||
layoutTable.setDefaultRenderer(Object.class, tableRenderer); |
||||
layoutTable.setSelectionMode(ListSelectionModel.SINGLE_SELECTION); |
||||
layoutTable.setColumnSelectionAllowed(false); |
||||
layoutTable.setRowSelectionAllowed(false); |
||||
layoutTable.setBackground(Color.white); |
||||
int columnWidth = Integer.parseInt(Inter.getLocText("FR-Designer_LayoutTable_Column_Width")); |
||||
for (int i = 0; i < layoutTable.getColumnModel().getColumnCount(); i++) { |
||||
layoutTable.getColumnModel().getColumn(i).setPreferredWidth(columnWidth); |
||||
} |
||||
layoutTable.addMouseListener(new MouseAdapter() { |
||||
public void mouseClicked(MouseEvent e) { |
||||
if (e.getClickCount() > 1 && !SwingUtilities.isRightMouseButton(e)) { |
||||
WidgetOption no = (WidgetOption)layoutTable.getValueAt(layoutTable.getSelectedRow(), layoutTable.getSelectedColumn()); |
||||
Widget widget = no.createWidget(); |
||||
FormToolBarButton tb = new FormToolBarButton(no.optionIcon(), widget); |
||||
tb.setNameOption(no); |
||||
northToolBar.add(tb); |
||||
northToolBar.validate(); |
||||
northToolBar.repaint(); |
||||
} |
||||
} |
||||
}); |
||||
} |
||||
|
||||
private boolean isSelectedtable() { |
||||
for (int i = 0; i < layoutTable.getColumnCount(); i++) { |
||||
if (layoutTable.isColumnSelected(i)) { |
||||
return true; |
||||
} |
||||
} |
||||
return false; |
||||
} |
||||
|
||||
/** |
||||
* 是否被选中 |
||||
* @return 同上 |
||||
*/ |
||||
public boolean isUseToolbar() { |
||||
return this.isUseToolBarCheckBox.isSelected(); |
||||
} |
||||
|
||||
public void setDefaultToolBar(FormToolBarManager defaultToolBar, WidgetOption[] buttonArray) { |
||||
this.defaultToolBar = defaultToolBar; |
||||
if (buttonArray != null) { |
||||
for (int i = 0; i < buttonArray.length; i++) { |
||||
toolbarButtonTableModel.setValueAt(buttonArray[i], i % row, i / row); |
||||
} |
||||
} |
||||
|
||||
} |
||||
|
||||
DefaultTableCellRenderer tableRenderer = new DefaultTableCellRenderer() { |
||||
@Override |
||||
public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) { |
||||
super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column); |
||||
|
||||
if (value instanceof WidgetOption) { |
||||
WidgetOption nameOption = (WidgetOption)value; |
||||
this.setText(nameOption.optionName()); |
||||
|
||||
Icon icon = nameOption.optionIcon(); |
||||
if (icon != null) { |
||||
this.setIcon(icon); |
||||
} |
||||
} |
||||
if (value == null) { |
||||
this.setText(StringUtils.EMPTY); |
||||
this.setIcon(null); |
||||
} |
||||
return this; |
||||
} |
||||
}; |
||||
|
||||
@Override |
||||
protected String title4PopupWindow() { |
||||
return Inter.getLocText(new String[]{"Form-EC_toolbar", "Set"}); |
||||
} |
||||
|
||||
public void setCheckBoxSelected(boolean b) { |
||||
this.isUseToolBarCheckBox.setSelected(b); |
||||
} |
||||
|
||||
@Override |
||||
public void populateBean(FormToolBarManager[] toolBarManager) { |
||||
if (ArrayUtils.isEmpty(toolBarManager)) { |
||||
defaultToolBar.setToolBarLocation(Location.createTopEmbedLocation()); |
||||
toolBarManager = new FormToolBarManager[] { defaultToolBar }; |
||||
} |
||||
if (ArrayUtils.isEmpty(toolBarManager)) { |
||||
return; |
||||
} |
||||
for (int i = 0; i < toolBarManager.length; i++) { |
||||
Location location = toolBarManager[i].getToolBarLocation(); |
||||
if (location instanceof Location.Embed) { |
||||
if (((Location.Embed)location).getPosition() == Constants.TOP) { |
||||
northToolBar.populateBean(toolBarManager[i].getToolBar()); |
||||
} else if (((Location.Embed)location).getPosition() == Constants.BOTTOM) { |
||||
southToolBar.populateBean(toolBarManager[i].getToolBar()); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public FormToolBarManager[] updateBean() { |
||||
if(!isUseToolbar()){ |
||||
return new FormToolBarManager[0]; |
||||
} |
||||
List<FormToolBarManager> toolBarManagerList = new ArrayList<FormToolBarManager>(); |
||||
if (!northToolBar.isEmpty()) { |
||||
FormToolBarManager north = new FormToolBarManager(); |
||||
north.setToolBar(northToolBar.updateBean()); |
||||
north.setToolBarLocation(Location.createTopEmbedLocation()); |
||||
toolBarManagerList.add(north); |
||||
} |
||||
|
||||
if (!southToolBar.isEmpty()) { |
||||
FormToolBarManager south = new FormToolBarManager(); |
||||
south.setToolBar(southToolBar.updateBean()); |
||||
south.setToolBarLocation(Location.createBottomEmbedLocation()); |
||||
toolBarManagerList.add(south); |
||||
} |
||||
return toolBarManagerList.toArray(new FormToolBarManager[toolBarManagerList.size()]); |
||||
} |
||||
|
||||
private class TableModel extends DefaultTableModel { |
||||
public TableModel(int i, int j) { |
||||
super(i, j); |
||||
} |
||||
|
||||
// 禁止jtable的双击编辑功能
|
||||
public boolean isCellEditable(int row, int column) { |
||||
return false; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,34 @@
|
||||
package com.fr.design.mainframe.widget.wrappers; |
||||
|
||||
import com.fr.design.Exception.ValidationException; |
||||
import com.fr.design.designer.properties.Decoder; |
||||
import com.fr.design.designer.properties.Encoder; |
||||
import com.fr.form.web.FormToolBarManager; |
||||
import com.fr.general.Inter; |
||||
|
||||
/** |
||||
* Created by harry on 2017-3-1. |
||||
*/ |
||||
public class ElementCaseToolBarWrapper implements Encoder, Decoder { |
||||
@Override |
||||
public Object decode(String txt) { |
||||
return null; |
||||
} |
||||
|
||||
@Override |
||||
public void validate(String txt) throws ValidationException { |
||||
|
||||
} |
||||
|
||||
@Override |
||||
public String encode(Object v) { |
||||
if (v == null) { |
||||
return null; |
||||
} |
||||
FormToolBarManager[] toolBarManager = (FormToolBarManager[])v; |
||||
if (toolBarManager.length != 0) { |
||||
return Inter.getLocText("FR-Designer_Open"); |
||||
} |
||||
return Inter.getLocText("FR-Designer_Close"); |
||||
} |
||||
} |
Loading…
Reference in new issue