forked from fanruan/design
daniel
8 years ago
72 changed files with 6619 additions and 5733 deletions
@ -1,6 +1,5 @@
|
||||
package com.fr.design.mainframe.templateinfo; |
||||
|
||||
import com.fr.base.io.IOFile; |
||||
import com.fr.base.parameter.ParameterUI; |
||||
import com.fr.main.impl.WorkBook; |
||||
import com.fr.report.cellcase.CellCase; |
@ -0,0 +1,89 @@
|
||||
|
||||
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' |
||||
//生成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 |
||||
|
||||
|
||||
|
@ -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; |
||||
} |
||||
} |
@ -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); |
||||
} |
@ -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 |
||||
|
@ -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,133 +1,141 @@
|
||||
package com.fr.design.designer.beans.models; |
||||
|
||||
import java.awt.Rectangle; |
||||
|
||||
import com.fr.design.designer.creator.XWAbsoluteLayout; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
import com.fr.design.designer.beans.AdapterBus; |
||||
import com.fr.design.designer.beans.ComponentAdapter; |
||||
import com.fr.design.designer.beans.adapters.component.CompositeComponentAdapter; |
||||
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.utils.ComponentUtils; |
||||
import com.fr.general.ComparatorUtils; |
||||
|
||||
/** |
||||
* 添加状态下的model |
||||
*/ |
||||
public class AddingModel { |
||||
|
||||
// 当前要添加的组件
|
||||
private XCreator creator; |
||||
// 记录当前鼠标的位置信息
|
||||
private int current_x; |
||||
private int current_y; |
||||
private boolean added; |
||||
|
||||
public AddingModel(FormDesigner designer, XCreator xCreator) { |
||||
String creatorName = getXCreatorName(designer, xCreator); |
||||
this.creator = xCreator; |
||||
instantiateCreator(designer, creatorName); |
||||
// 初始的时候隐藏该组件的图标
|
||||
current_x = -this.creator.getWidth(); |
||||
current_y = -this.creator.getHeight(); |
||||
} |
||||
|
||||
/** |
||||
* 待说明 |
||||
* @param designer 设计器 |
||||
* @param creatorName 组件名 |
||||
*/ |
||||
public void instantiateCreator(FormDesigner designer, String creatorName) { |
||||
creator.toData().setWidgetName(creatorName); |
||||
ComponentAdapter adapter = new CompositeComponentAdapter(designer, creator); |
||||
adapter.initialize(); |
||||
creator.addNotify(); |
||||
creator.putClientProperty(AdapterBus.CLIENT_PROPERTIES, adapter); |
||||
} |
||||
|
||||
public AddingModel(XCreator xCreator, int x, int y) { |
||||
this.creator = xCreator; |
||||
this.creator.backupCurrentSize(); |
||||
this.creator.backupParent(); |
||||
this.creator.setSize(xCreator.initEditorSize()); |
||||
current_x = x - (xCreator.getWidth() / 2); |
||||
current_y = y - (xCreator.getHeight() / 2); |
||||
} |
||||
|
||||
/** |
||||
* 隐藏当前组件的图标 |
||||
*/ |
||||
public void reset() { |
||||
current_x = -this.creator.getWidth(); |
||||
current_y = -this.creator.getHeight(); |
||||
} |
||||
|
||||
public String getXCreatorName(FormDesigner designer,XCreator x){ |
||||
String def= x.createDefaultName(); |
||||
if (x.acceptType(XWParameterLayout.class)) { |
||||
return def; |
||||
} |
||||
int i = 0; |
||||
while (designer.getTarget().isNameExist(def + i)) { |
||||
i++; |
||||
} |
||||
return def+i; |
||||
} |
||||
|
||||
public int getCurrentX() { |
||||
return current_x; |
||||
} |
||||
|
||||
public int getCurrentY() { |
||||
return current_y; |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 移动组件图标到鼠标事件发生的位置 |
||||
* @param x 坐标 |
||||
* @param y 坐标 |
||||
*/ |
||||
public void moveTo(int x, int y) { |
||||
current_x = x - (this.creator.getWidth() / 2); |
||||
current_y = y - (this.creator.getHeight() / 2); |
||||
} |
||||
|
||||
public XCreator getXCreator() { |
||||
return this.creator; |
||||
} |
||||
|
||||
/** |
||||
* 当前组件是否已经添加到某个容器中 |
||||
* @return 是返回true |
||||
*/ |
||||
public boolean isCreatorAdded() { |
||||
return added; |
||||
} |
||||
|
||||
/** |
||||
* 加入容器 |
||||
* @param designer 设计器 |
||||
* @param container 容器 |
||||
* @param x 坐标 |
||||
* @param y 坐标 |
||||
* @return 成功返回true |
||||
*/ |
||||
public boolean add2Container(FormDesigner designer, XLayoutContainer container, int x, int y) { |
||||
//考虑不同布局嵌套的情况,获取顶层容器
|
||||
XLayoutContainer xLayoutContainer = container.getTopLayout(); |
||||
if(xLayoutContainer != null && xLayoutContainer.acceptType(XWAbsoluteLayout.class)){ |
||||
container = xLayoutContainer; |
||||
} |
||||
|
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
if(!ComparatorUtils.equals(container.getOuterLayout(), container.getBackupParent())){ |
||||
return added = container.getLayoutAdapter().addBean(creator, x + designer.getArea().getHorizontalValue(), y + designer.getArea().getVerticalValue()); |
||||
} |
||||
return added = container.getLayoutAdapter().addBean(creator, |
||||
x + designer.getArea().getHorizontalValue() - rect.x, |
||||
y + designer.getArea().getVerticalValue() - rect.y); |
||||
} |
||||
package com.fr.design.designer.beans.models; |
||||
|
||||
import java.awt.Rectangle; |
||||
|
||||
import com.fr.design.designer.creator.XWAbsoluteLayout; |
||||
import com.fr.design.mainframe.FormDesigner; |
||||
import com.fr.design.designer.beans.AdapterBus; |
||||
import com.fr.design.designer.beans.ComponentAdapter; |
||||
import com.fr.design.designer.beans.adapters.component.CompositeComponentAdapter; |
||||
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.utils.ComponentUtils; |
||||
import com.fr.general.ComparatorUtils; |
||||
|
||||
/** |
||||
* 添加状态下的model |
||||
*/ |
||||
public class AddingModel { |
||||
|
||||
// 当前要添加的组件
|
||||
private XCreator creator; |
||||
// 记录当前鼠标的位置信息
|
||||
private int currentX; |
||||
private int currentY; |
||||
private boolean added; |
||||
|
||||
public AddingModel(FormDesigner designer, XCreator xCreator) { |
||||
String creatorName = getXCreatorName(designer, xCreator); |
||||
this.creator = xCreator; |
||||
instantiateCreator(designer, creatorName); |
||||
// 初始的时候隐藏该组件的图标
|
||||
currentY = -this.creator.getWidth(); |
||||
currentX = -this.creator.getHeight(); |
||||
} |
||||
|
||||
/** |
||||
* 待说明 |
||||
* |
||||
* @param designer 设计器 |
||||
* @param creatorName 组件名 |
||||
*/ |
||||
public void instantiateCreator(FormDesigner designer, String creatorName) { |
||||
creator.toData().setWidgetName(creatorName); |
||||
ComponentAdapter adapter = new CompositeComponentAdapter(designer, creator); |
||||
adapter.initialize(); |
||||
creator.addNotify(); |
||||
creator.putClientProperty(AdapterBus.CLIENT_PROPERTIES, adapter); |
||||
} |
||||
|
||||
public AddingModel(XCreator xCreator, int x, int y) { |
||||
this.creator = xCreator; |
||||
this.creator.backupCurrentSize(); |
||||
this.creator.backupParent(); |
||||
this.creator.setSize(xCreator.initEditorSize()); |
||||
currentX = x - (xCreator.getWidth() / 2); |
||||
currentY = y - (xCreator.getHeight() / 2); |
||||
} |
||||
|
||||
/** |
||||
* 隐藏当前组件的图标 |
||||
*/ |
||||
public void reset() { |
||||
currentX = -this.creator.getWidth(); |
||||
currentY = -this.creator.getHeight(); |
||||
} |
||||
|
||||
public String getXCreatorName(FormDesigner designer, XCreator x) { |
||||
String def = x.createDefaultName(); |
||||
if (x.acceptType(XWParameterLayout.class)) { |
||||
return def; |
||||
} |
||||
int i = 0; |
||||
while (designer.getTarget().isNameExist(def + i)) { |
||||
i++; |
||||
} |
||||
return def + i; |
||||
} |
||||
|
||||
public int getCurrentX() { |
||||
return currentX; |
||||
} |
||||
|
||||
public int getCurrentY() { |
||||
return currentY; |
||||
} |
||||
|
||||
|
||||
/** |
||||
* 移动组件图标到鼠标事件发生的位置 |
||||
* |
||||
* @param x 坐标 |
||||
* @param y 坐标 |
||||
*/ |
||||
public void moveTo(int x, int y) { |
||||
currentX = x - (this.creator.getWidth() / 2); |
||||
currentY = y - (this.creator.getHeight() / 2); |
||||
} |
||||
|
||||
public XCreator getXCreator() { |
||||
return this.creator; |
||||
} |
||||
|
||||
/** |
||||
* 当前组件是否已经添加到某个容器中 |
||||
* |
||||
* @return 是返回true |
||||
*/ |
||||
public boolean isCreatorAdded() { |
||||
return added; |
||||
} |
||||
|
||||
/** |
||||
* 加入容器 |
||||
* |
||||
* @param designer 设计器 |
||||
* @param container 容器 |
||||
* @param x 坐标 |
||||
* @param y 坐标 |
||||
* @return 成功返回true |
||||
*/ |
||||
public boolean add2Container(FormDesigner designer, XLayoutContainer container, int x, int y) { |
||||
//考虑不同布局嵌套的情况,获取顶层容器
|
||||
XLayoutContainer xLayoutContainer = container.getTopLayout(); |
||||
if (xLayoutContainer != null && xLayoutContainer.acceptType(XWAbsoluteLayout.class)) { |
||||
container = xLayoutContainer; |
||||
} |
||||
|
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
if (!ComparatorUtils.equals(container.getOuterLayout(), container.getBackupParent())) { |
||||
added = container.getLayoutAdapter().addBean(creator, |
||||
x + designer.getArea().getHorizontalValue(), |
||||
y + designer.getArea().getVerticalValue()); |
||||
return added; |
||||
} |
||||
added = container.getLayoutAdapter().addBean(creator, |
||||
x + designer.getArea().getHorizontalValue() - rect.x, |
||||
y + designer.getArea().getVerticalValue() - rect.y); |
||||
return added; |
||||
} |
||||
} |
@ -1,328 +1,434 @@
|
||||
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)) { |
||||
Rectangle rec = selection.getRelativeBounds(); |
||||
//Tab布局
|
||||
FormSelectionUtils.paste2Container(designer, (XLayoutContainer) selection.getSelectedCreator(), |
||||
clipboard, |
||||
rec.x + rec.width / 2, |
||||
rec.y + DELTA_X_Y); |
||||
} else { |
||||
Rectangle rec = selection.getRelativeBounds(); |
||||
//自适应布局
|
||||
FormSelectionUtils.paste2Container(designer, designer.getRootComponent(), |
||||
clipboard, |
||||
rec.x + rec.width / 2, |
||||
rec.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 = selectionRec.x - containerRec.x + selectionRec.width / 2; |
||||
int 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.utils.ComponentUtils; |
||||
|
||||
/** |
||||
* 普通模式下的状态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; |
||||
} |
||||
|
||||
/** |
||||
*是否能拖拽 |
||||
* @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) { |
||||
// 当前鼠标所在的组件
|
||||
XCreator hoveredComponent = designer.getComponentAt(x, y, 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(); |
||||
} |
||||
|
||||
// 取消提示
|
||||
designer.setPainter(null); |
||||
} |
||||
|
||||
/** |
||||
*是否拖拽 |
||||
* @return dragging状态 |
||||
*/ |
||||
public boolean isDragging() { |
||||
return dragging; |
||||
} |
||||
|
||||
/** |
||||
*是否可以开始画线 |
||||
* @return startPoint不为空返回true |
||||
*/ |
||||
public boolean prepareForDrawLining() { |
||||
return startPoint != null; |
||||
} |
||||
|
||||
/** |
||||
*设置开始位置 |
||||
* @param p point位置 |
||||
*/ |
||||
public void setStartPoint(Point p) { |
||||
this.startPoint = p; |
||||
} |
||||
|
||||
/** |
||||
*返回开始位置 |
||||
* @return 点位置 |
||||
*/ |
||||
public Point getStartPoint() { |
||||
return startPoint; |
||||
} |
||||
|
||||
/** |
||||
*返回当前点位置 |
||||
* @return 点位置 |
||||
*/ |
||||
public Point getEndPoint() { |
||||
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; |
||||
} |
||||
|
||||
/** |
||||
*当前鼠标的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; |
||||
} |
||||
|
||||
/** |
||||
*起始点开始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)); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*鼠标释放时所在的区域及圈中的组件 |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void selectCreators(MouseEvent e) { |
||||
int x = getMouseXY(e).x; |
||||
int y = getMouseXY(e).y; |
||||
|
||||
Rectangle bounds = createCurrentBounds(x, y); |
||||
|
||||
if ((x != current_x) || (y != current_y)) { |
||||
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; |
||||
} |
||||
|
||||
|
||||
/** |
||||
*重置model |
||||
*/ |
||||
public void resetModel() { |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
|
||||
/** |
||||
*重置 |
||||
*/ |
||||
public void reset() { |
||||
driection = Location.outer; |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
|
||||
/** |
||||
*取消拖拽 |
||||
*/ |
||||
public void draggingCancel() { |
||||
designer.repaint(); |
||||
reset(); |
||||
} |
||||
|
||||
/** |
||||
*设置可拉伸方向 |
||||
* @param dir 拉伸方向 |
||||
*/ |
||||
public void setDirection(Direction dir) { |
||||
if(driection != dir) { |
||||
this.driection = dir; |
||||
driection.updateCursor(designer); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*x吸附线赋值 |
||||
* @param line 线 |
||||
*/ |
||||
public void setXAbsorptionline(Absorptionline line) { |
||||
this.lineInX = line; |
||||
} |
||||
|
||||
/** |
||||
*y吸附线赋值 |
||||
* @param line 线 |
||||
*/ |
||||
public void setYAbsorptionline(Absorptionline line) { |
||||
this.lineInY = line; |
||||
} |
||||
|
||||
/** |
||||
* 等距线赋值 |
||||
* @param 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()); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*拖拽 |
||||
* @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); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
*释放捕获 |
||||
* @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(); |
||||
} |
||||
|
||||
/** |
||||
*改变选择区域 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void changeSelection(MouseEvent e) { |
||||
Rectangle bounds = createCurrentBounds(getMouseXY(e).x, getMouseXY(e).y); |
||||
selectionModel.setHotspotBounds(bounds); |
||||
} |
||||
|
||||
/** |
||||
*返回鼠标所在的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; |
||||
} |
||||
|
||||
package com.fr.design.designer.beans.models; |
||||
|
||||
import com.fr.design.beans.location.Absorptionline; |
||||
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.*; |
||||
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 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; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* @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(mouseReleasedX, mouseReleasedY, selectionModel.getSelection().getSelectedCreators()); |
||||
|
||||
// 获取该组件所在的焦点容器
|
||||
XLayoutContainer container = XCreatorUtils.getHotspotContainer(hoveredComponent); |
||||
|
||||
boolean success = false; |
||||
|
||||
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; |
||||
} |
||||
|
||||
/** |
||||
* 是否可以开始画线 |
||||
* |
||||
* @return startPoint不为空返回true |
||||
*/ |
||||
public boolean prepareForDrawLining() { |
||||
return startPoint != null; |
||||
} |
||||
|
||||
/** |
||||
* 设置开始位置 |
||||
* |
||||
* @param p point位置 |
||||
*/ |
||||
public void setStartPoint(Point p) { |
||||
this.startPoint = p; |
||||
} |
||||
|
||||
/** |
||||
* 返回开始位置 |
||||
* |
||||
* @return 点位置 |
||||
*/ |
||||
public Point getStartPoint() { |
||||
return startPoint; |
||||
} |
||||
|
||||
/** |
||||
* 返回当前点位置 |
||||
* |
||||
* @return 点位置 |
||||
*/ |
||||
public Point getEndPoint() { |
||||
return currentPoint; |
||||
} |
||||
|
||||
/** |
||||
* 当前选中组件 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void startSelecting(MouseEvent e) { |
||||
selecting = true; |
||||
selectionModel.setHotspotBounds(new Rectangle()); |
||||
currentX = getMouseXY(e).x; |
||||
currentY = getMouseXY(e).y; |
||||
} |
||||
|
||||
/** |
||||
* 当前鼠标的xy |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void startResizing(MouseEvent e) { |
||||
if (!selectionModel.getSelection().isEmpty()) { |
||||
driection.backupBounds(designer); |
||||
} |
||||
currentX = getMouseXY(e).x; |
||||
currentY = getMouseXY(e).y; |
||||
} |
||||
|
||||
/** |
||||
* 起始点开始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)); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 鼠标释放时所在的区域及圈中的组件 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void selectCreators(MouseEvent e) { |
||||
int x = getMouseXY(e).x; |
||||
int y = getMouseXY(e).y; |
||||
|
||||
Rectangle bounds = createCurrentBounds(x, y); |
||||
|
||||
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, 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 |
||||
*/ |
||||
public void resetModel() { |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
|
||||
/** |
||||
* 重置 |
||||
*/ |
||||
public void reset() { |
||||
driection = Location.outer; |
||||
dragging = false; |
||||
selecting = false; |
||||
} |
||||
|
||||
/** |
||||
* 取消拖拽 |
||||
*/ |
||||
public void draggingCancel() { |
||||
designer.repaint(); |
||||
reset(); |
||||
} |
||||
|
||||
/** |
||||
* 设置可拉伸方向 |
||||
* |
||||
* @param dir 拉伸方向 |
||||
*/ |
||||
public void setDirection(Direction dir) { |
||||
if (driection != dir) { |
||||
this.driection = dir; |
||||
driection.updateCursor(designer); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* x吸附线赋值 |
||||
* |
||||
* @param line 线 |
||||
*/ |
||||
public void setXAbsorptionline(Absorptionline line) { |
||||
this.lineInX = line; |
||||
} |
||||
|
||||
/** |
||||
* y吸附线赋值 |
||||
* |
||||
* @param line 线 |
||||
*/ |
||||
public void setYAbsorptionline(Absorptionline line) { |
||||
this.lineInY = line; |
||||
} |
||||
|
||||
/** |
||||
* 等距线赋值 |
||||
* |
||||
* @param 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()); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 拖拽 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
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(); |
||||
} |
||||
|
||||
/** |
||||
* 改变选择区域 |
||||
* |
||||
* @param e 鼠标事件 |
||||
*/ |
||||
public void changeSelection(MouseEvent e) { |
||||
Rectangle bounds = createCurrentBounds(getMouseXY(e).x, getMouseXY(e).y); |
||||
selectionModel.setHotspotBounds(bounds); |
||||
} |
||||
|
||||
/** |
||||
* 返回鼠标所在的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; |
||||
} |
||||
|
||||
} |
@ -1,93 +1,96 @@
|
||||
package com.fr.design.designer.beans.painters; |
||||
|
||||
import java.awt.Color; |
||||
import java.awt.Graphics; |
||||
import java.awt.Graphics2D; |
||||
import java.awt.Point; |
||||
import java.awt.Rectangle; |
||||
import java.awt.Stroke; |
||||
|
||||
import com.fr.design.designer.beans.HoverPainter; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.form.util.XCreatorConstants; |
||||
import com.fr.general.Inter; |
||||
|
||||
public abstract class AbstractPainter implements HoverPainter { |
||||
|
||||
protected Point hotspot; |
||||
protected Rectangle hotspot_bounds; |
||||
protected XLayoutContainer container; |
||||
protected XCreator creator; |
||||
|
||||
/** |
||||
* 构造函数 |
||||
* @param container 容器 |
||||
*/ |
||||
public AbstractPainter(XLayoutContainer container) { |
||||
this.container = container; |
||||
} |
||||
|
||||
@Override |
||||
public void setHotspot(Point p) { |
||||
hotspot = p; |
||||
} |
||||
|
||||
/** |
||||
* 画初始区域 |
||||
* @param g 画图类 |
||||
* @param startX 起始x位置 |
||||
* @param startY 起始y位置 |
||||
*/ |
||||
public void paint(Graphics g, int startX, int startY) { |
||||
if(hotspot_bounds != null){ |
||||
drawHotspot(g, hotspot_bounds.x, hotspot_bounds.y, hotspot_bounds.width, hotspot_bounds.height, Color.lightGray, true, false); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 设置边界 |
||||
* @param rect 位置 |
||||
*/ |
||||
@Override |
||||
public void setRenderingBounds(Rectangle rect) { |
||||
hotspot_bounds = rect; |
||||
} |
||||
|
||||
@Override |
||||
public void setCreator(XCreator component) { |
||||
this.creator = component; |
||||
} |
||||
|
||||
protected void drawHotspot(Graphics g, int x, int y, int width, int height, boolean accept) { |
||||
Color bColor = accept ? XCreatorConstants.LAYOUT_HOTSPOT_COLOR : XCreatorConstants.LAYOUT_FORBIDDEN_COLOR; |
||||
drawHotspot(g, x, y, width, height, bColor, accept, false); |
||||
} |
||||
|
||||
/** |
||||
* 自适应布局那边渲染提示,要画整个背景,不是画边框 |
||||
*/ |
||||
protected void drawRegionBackground(Graphics g, int x, int y, int width, int height, Color bColor, boolean accept) { |
||||
drawHotspot(g, x, y, width, height, bColor, accept, true); |
||||
} |
||||
|
||||
protected void drawHotspot(Graphics g, int x, int y, int width, int height, Color bColor, boolean accept, boolean drawBackground) { |
||||
Graphics2D g2d = (Graphics2D) g; |
||||
Color color = g2d.getColor(); |
||||
Stroke backup = g2d.getStroke(); |
||||
// 设置线条的样式
|
||||
g2d.setStroke(XCreatorConstants.STROKE); |
||||
g2d.setColor(bColor); |
||||
if (!accept) { |
||||
g2d.drawString(Inter.getLocText("Cannot-Add_To_This_Area") + "!", x + width / 3, y + height / 2); |
||||
} else if (drawBackground) { |
||||
g2d.fillRect(x, y, width, height); |
||||
} else { |
||||
g2d.drawRect(x, y, width, height); |
||||
} |
||||
g2d.setStroke(backup); |
||||
g2d.setColor(color); |
||||
} |
||||
|
||||
|
||||
package com.fr.design.designer.beans.painters; |
||||
|
||||
import java.awt.Color; |
||||
import java.awt.Graphics; |
||||
import java.awt.Graphics2D; |
||||
import java.awt.Point; |
||||
import java.awt.Rectangle; |
||||
import java.awt.Stroke; |
||||
|
||||
import com.fr.design.designer.beans.HoverPainter; |
||||
import com.fr.design.designer.creator.XCreator; |
||||
import com.fr.design.designer.creator.XLayoutContainer; |
||||
import com.fr.design.form.util.XCreatorConstants; |
||||
import com.fr.general.Inter; |
||||
|
||||
public abstract class AbstractPainter implements HoverPainter { |
||||
|
||||
protected Point hotspot; |
||||
protected Rectangle hotspot_bounds; |
||||
protected XLayoutContainer container; |
||||
protected XCreator creator; |
||||
|
||||
/** |
||||
* 构造函数 |
||||
* |
||||
* @param container 容器 |
||||
*/ |
||||
public AbstractPainter(XLayoutContainer container) { |
||||
this.container = container; |
||||
} |
||||
|
||||
@Override |
||||
public void setHotspot(Point p) { |
||||
hotspot = p; |
||||
} |
||||
|
||||
/** |
||||
* 画初始区域 |
||||
* |
||||
* @param g 画图类 |
||||
* @param startX 起始x位置 |
||||
* @param startY 起始y位置 |
||||
*/ |
||||
public void paint(Graphics g, int startX, int startY) { |
||||
if (hotspot_bounds != null) { |
||||
drawHotspot(g, hotspot_bounds.x, hotspot_bounds.y, hotspot_bounds.width, hotspot_bounds.height, Color.lightGray, true, false); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 设置边界 |
||||
* |
||||
* @param rect 位置 |
||||
*/ |
||||
@Override |
||||
public void setRenderingBounds(Rectangle rect) { |
||||
hotspot_bounds = rect; |
||||
} |
||||
|
||||
@Override |
||||
public void setCreator(XCreator component) { |
||||
this.creator = component; |
||||
} |
||||
|
||||
protected void drawHotspot(Graphics g, int x, int y, int width, int height, boolean accept) { |
||||
Color bColor = accept ? XCreatorConstants.LAYOUT_HOTSPOT_COLOR : XCreatorConstants.LAYOUT_FORBIDDEN_COLOR; |
||||
drawHotspot(g, x, y, width, height, bColor, accept, false); |
||||
} |
||||
|
||||
/** |
||||
* 自适应布局那边渲染提示,要画整个背景,不是画边框 |
||||
*/ |
||||
protected void drawRegionBackground(Graphics g, int x, int y, int width, int height, Color bColor, boolean accept) { |
||||
drawHotspot(g, x, y, width, height, bColor, accept, true); |
||||
} |
||||
|
||||
protected void drawHotspot(Graphics g, int x, int y, int width, int height, Color bColor, boolean accept, boolean drawBackground) { |
||||
Graphics2D g2d = (Graphics2D) g; |
||||
Color color = g2d.getColor(); |
||||
Stroke backup = g2d.getStroke(); |
||||
// 设置线条的样式
|
||||
g2d.setStroke(XCreatorConstants.STROKE); |
||||
g2d.setColor(bColor); |
||||
if (!accept) { |
||||
g2d.drawString(Inter.getLocText("Cannot-Add_To_This_Area") + "!", x + width / 3, y + height / 2); |
||||
} else if (drawBackground) { |
||||
g2d.fillRect(x, y, width, height); |
||||
} else { |
||||
g2d.drawRect(x, y, width, height); |
||||
} |
||||
g2d.setStroke(backup); |
||||
g2d.setColor(color); |
||||
} |
||||
|
||||
|
||||
} |
@ -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
@ -1,126 +1,239 @@
|
||||
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.events.DesignerEvent; |
||||
import com.fr.design.designer.creator.*; |
||||
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) { |
||||
designer.getSelectionModel().getSelection().reset(); |
||||
for (XCreator creator : clipboard.getSelectedCreators()) { |
||||
try { |
||||
Widget copied = copyWidget(designer, creator); |
||||
XCreator copiedCreator = XCreatorUtils.createXCreator(copied, creator.getSize()); |
||||
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); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
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; |
||||
} |
||||
} |
Loading…
Reference in new issue