yaoh.wu
8 years ago
21 changed files with 4736 additions and 4713 deletions
@ -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,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 {
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();
} |
||||
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; |
||||
} |
||||
} |
@ -1,94 +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,189 +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 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; |
||||
} |
||||
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,363 +1,349 @@
|
||||
package com.fr.design.designer.beans.adapters.layout; |
||||
|
||||
import java.awt.*; |
||||
|
||||
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.painters.FRAbsoluteLayoutPainter; |
||||
import com.fr.design.designer.creator.*; |
||||
import com.fr.design.designer.properties.BoundsGroupModel; |
||||
import com.fr.design.designer.properties.FRAbsoluteLayoutPropertiesGroupModel; |
||||
import com.fr.form.ui.container.WAbsoluteLayout; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.design.utils.gui.LayoutUtils; |
||||
import com.fr.form.ui.widget.BoundsWidget; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.general.FRLogger; |
||||
|
||||
public class FRAbsoluteLayoutAdapter extends FRBodyLayoutAdapter { |
||||
//是不是添加到父容器上
|
||||
private boolean isAdd2ParentLayout = false; |
||||
private HoverPainter painter; |
||||
|
||||
public FRAbsoluteLayoutAdapter(XLayoutContainer container) { |
||||
super(container); |
||||
painter = new FRAbsoluteLayoutPainter(container); |
||||
initMinSize(); |
||||
} |
||||
|
||||
private void initMinSize() { |
||||
XWAbsoluteLayout layout = (XWAbsoluteLayout) container; |
||||
minWidth = layout.getActualMinWidth(); |
||||
minHeight = layout.getActualMinHeight(); |
||||
actualVal = layout.getAcualInterval(); |
||||
margin = layout.toData().getMargin(); |
||||
} |
||||
|
||||
@Override |
||||
public HoverPainter getPainter() { |
||||
return painter; |
||||
} |
||||
|
||||
/** |
||||
* 是否能在指定位置添加组件 |
||||
* |
||||
* @param creator 组件 |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
* @return 能则返回true |
||||
*/ |
||||
//这个地方的逻辑非常复杂,
|
||||
// 1.当前绝对布局是不可编辑且是最外层,那么其他控件添加在它周围,
|
||||
// 2.当前绝对布局是不可编辑且不是最外层,那么控件不可添加,(嵌套)
|
||||
// 3.当前绝对布局可编辑,那么控件添加
|
||||
@Override |
||||
public boolean accept(XCreator creator, int x, int y) { |
||||
Component comp = container.getComponentAt(x, y); |
||||
//布局控件要先判断是不是可编辑
|
||||
//可以编辑,按原有逻辑判断
|
||||
//不可编辑,当成一整个控件处理
|
||||
if (comp == null) { |
||||
return false; |
||||
} |
||||
//参数面板内的组件不允许拖往绝对布局中
|
||||
if (creator.getParent() != null && ((XCreator) creator.getParent()).acceptType(XWParameterLayout.class)) { |
||||
Rectangle rec = creator.getBounds(); |
||||
rec.y = creator.getParent().getHeight() - rec.height; |
||||
creator.setBounds(rec); |
||||
return false; |
||||
} |
||||
//判断组件能不能拖入绝对布局
|
||||
if (!creator.canEnterIntoAbsolutePane()) { |
||||
return false; |
||||
} |
||||
XLayoutContainer topLayout = XCreatorUtils.getHotspotContainer((XCreator) comp).getTopLayout(); |
||||
if (topLayout != null) { |
||||
if (topLayout.isEditable()) { |
||||
return topLayoutAccept(creator, x, y, topLayout); |
||||
} |
||||
//绝对布局嵌套,处于内层,不可编辑,不添加,topLayout只能获取到最外层可编辑的布局
|
||||
else if (((XLayoutContainer) topLayout.getParent()).acceptType(XWAbsoluteLayout.class)) { |
||||
return false; |
||||
} else { |
||||
return acceptWidget(creator, x, y); |
||||
} |
||||
} else { |
||||
FRLogger.getLogger().error("top layout is null!"); |
||||
} |
||||
|
||||
return false; |
||||
} |
||||
|
||||
//topLayout假如可以编辑的话就往里面添加组件
|
||||
private boolean topLayoutAccept(XCreator creator, int x, int y, XLayoutContainer topLayout) { |
||||
//允许组件重叠,可以不判断有没有和当前控件重叠
|
||||
//先计算当前控件的位置
|
||||
int creatorX, creatorY; |
||||
if (XCreatorUtils.getParentXLayoutContainer(creator) != null) { |
||||
Rectangle creatorRectangle = ComponentUtils.getRelativeBounds(creator); |
||||
creatorX = creatorRectangle.x; |
||||
creatorY = creatorRectangle.y; |
||||
} else { |
||||
//这边计算得到的组件其实位置是正确的,
|
||||
//因为传入的x和y已经加上了宽度或者高度的一半,再减去相同的宽度和高度的一半是没区别的,
|
||||
// 例如高度为21,那么就是+10-10;
|
||||
// 高度为20,那么就是+10-10; 没区别
|
||||
int w = creator.getWidth() / 2; |
||||
int h = creator.getHeight() / 2; |
||||
creatorX = x - w; |
||||
creatorY = y - h; |
||||
} |
||||
//frm 组件复用允许组件重叠
|
||||
//无须再判断和布局中其他控件重叠
|
||||
//Rectangle curRec = new Rectangle(creatorX, creatorY, creator.getWidth(), creator.getHeight());
|
||||
//WAbsoluteLayout wAbsoluteLayout = (WAbsoluteLayout) topLayout.toData();
|
||||
//for (int i = 0, count = wAbsoluteLayout.getWidgetCount(); i < count; i++) {
|
||||
//BoundsWidget temp = (BoundsWidget) wAbsoluteLayout.getWidget(i);
|
||||
//Rectangle rectangle = temp.getBounds();
|
||||
//if (curRec.intersects(rectangle)) {
|
||||
//允许组件重叠
|
||||
//return false;
|
||||
//}
|
||||
//}
|
||||
if (creatorX < 0 || creatorX + creator.getWidth() > container.getWidth()) { |
||||
return false; |
||||
} |
||||
if (creatorY < 0 || creatorY + creator.getHeight() > container.getHeight()) { |
||||
return false; |
||||
} |
||||
return x >= 0 && y >= 0 && creator.getHeight() <= container.getHeight() |
||||
&& creator.getWidth() <= container.getWidth(); |
||||
} |
||||
|
||||
/** |
||||
* 判断是否鼠标在组件的三等分区域,如果组件在布局管理器中间,上下左右都可能会三等分 |
||||
* |
||||
* @param parentComp 鼠标所在区域的组件 |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean isTrisectionArea(Component parentComp, int x, int y) { |
||||
XCreator creator = (XCreator) parentComp; |
||||
trisectAreaDirect = 0; |
||||
if (container.getComponentCount() <= 1) { |
||||
return false; |
||||
} |
||||
int maxWidth = parentComp.getWidth(); |
||||
int maxHeight = parentComp.getHeight(); |
||||
int xL = parentComp.getX(); |
||||
int yL = parentComp.getY(); |
||||
// 组件宽高的十分之一和默认值取大
|
||||
int minRangeWidth = Math.max(maxWidth / BORDER_PROPORTION, DEFAULT_AREA_LENGTH); |
||||
int minRangeHeight = Math.max(maxHeight / BORDER_PROPORTION, DEFAULT_AREA_LENGTH); |
||||
if (y < yL + minRangeHeight) { |
||||
// 在组件上侧三等分
|
||||
trisectAreaDirect = COMP_TOP; |
||||
} else if (y > yL + maxHeight - minRangeHeight) { |
||||
// 在组件下侧三等分
|
||||
trisectAreaDirect = COMP_BOTTOM; |
||||
} else if (x < xL + minRangeWidth) { |
||||
// 在组件左侧三等分
|
||||
trisectAreaDirect = COMP_LEFT; |
||||
} else if (x > xL + maxWidth - minRangeWidth) { |
||||
// 在组件右侧三等分
|
||||
trisectAreaDirect = COMP_RIGHT; |
||||
} |
||||
// tab布局的边界特殊处理,不进行三等分
|
||||
if (!creator.getTargetChildrenList().isEmpty()) { |
||||
return false; |
||||
} |
||||
|
||||
return !ComparatorUtils.equals(trisectAreaDirect, 0); |
||||
} |
||||
|
||||
//当前绝对布局不可编辑,就当成一个控件,组件添加在周围
|
||||
private boolean acceptWidget(XCreator creator, int x, int y) { |
||||
isFindRelatedComps = false; |
||||
//拖入组件判断时,先判断是否为交叉点区域,其次三等分区域,再次平分区域
|
||||
Component comp = container.getComponentAt(x, y); |
||||
boolean isMatchEdge = false; |
||||
//如果当前处于边缘地带, 那么就把他贴到父容器上
|
||||
XLayoutContainer parent = container.findNearestFit(); |
||||
container = parent != null ? parent : container; |
||||
isAdd2ParentLayout = true; |
||||
|
||||
int componentHeight = comp.getHeight(); |
||||
int componentWidth = comp.getWidth(); |
||||
//上半部分高度
|
||||
int upHeight = (int) (componentHeight * TOP_HALF) + comp.getY(); |
||||
//下半部分高度
|
||||
int downHeight = (int) (componentHeight * BOTTOM_HALF) + comp.getY(); |
||||
|
||||
if (isCrossPointArea(comp, x, y)) { |
||||
return canAcceptWhileCrossPoint(comp, x, y); |
||||
} |
||||
|
||||
if (isTrisectionArea(comp, x, y)) { |
||||
return canAcceptWhileTrisection(comp, x, y); |
||||
} |
||||
|
||||
boolean horizonValid = componentWidth >= minWidth * 2 + actualVal; |
||||
boolean verticalValid = componentHeight >= minHeight * 2 + actualVal; |
||||
return y > upHeight && y < downHeight ? horizonValid : verticalValid; |
||||
} |
||||
|
||||
/** |
||||
* 组件的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) { |
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
|
||||
int posX = x + rect.x; |
||||
int posY = y + rect.y; |
||||
if (!accept(creator, x, y)) { |
||||
return false; |
||||
} |
||||
addComp(creator, posX, posY); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
|
||||
@Override |
||||
protected void addComp(XCreator creator, int x, int y) { |
||||
if (!isAdd2ParentLayout) { |
||||
Rectangle r = ComponentUtils.getRelativeBounds(container); |
||||
x = x - r.x; |
||||
y = y - r.y; |
||||
if (XCreatorUtils.getParentXLayoutContainer(creator) != null) { |
||||
|
||||
Rectangle creatorRectangle = ComponentUtils.getRelativeBounds(creator); |
||||
x = creatorRectangle.x - r.x; |
||||
y = creatorRectangle.y - r.y; |
||||
} else { |
||||
int w = creator.getWidth() / 2; |
||||
int h = creator.getHeight() / 2; |
||||
x = x - w; |
||||
y = y - h; |
||||
} |
||||
fix(creator, x, y); |
||||
|
||||
if (creator.hasTitleStyle()) { |
||||
addParentCreator(creator); |
||||
} else { |
||||
container.add(creator, creator.toData().getWidgetName()); |
||||
} |
||||
XWAbsoluteLayout layout = (XWAbsoluteLayout) container; |
||||
layout.updateBoundsWidget(creator); |
||||
updateCreatorBackBound(); |
||||
LayoutUtils.layoutRootContainer(container); |
||||
} else { |
||||
fixAbsolute(creator, x, y); |
||||
if (creator.shouldScaleCreator() || creator.hasTitleStyle()) { |
||||
addParentCreator(creator); |
||||
} else { |
||||
container.add(creator, creator.toData().getWidgetName()); |
||||
} |
||||
XWFitLayout layout = (XWFitLayout) container; |
||||
// 更新对应的BoundsWidget
|
||||
layout.updateBoundsWidget(); |
||||
updateCreatorBackBound(); |
||||
} |
||||
} |
||||
|
||||
private void updateCreatorBackBound() { |
||||
for (int i = 0, size = container.getComponentCount(); i < size; i++) { |
||||
XCreator creator = (XCreator) container.getComponent(i); |
||||
creator.updateChildBound(minHeight); |
||||
creator.setBackupBound(creator.getBounds()); |
||||
} |
||||
} |
||||
|
||||
private void addParentCreator(XCreator child) { |
||||
XLayoutContainer parentPanel = child.initCreatorWrapper(child.getHeight()); |
||||
container.add(parentPanel, child.toData().getWidgetName()); |
||||
} |
||||
|
||||
/** |
||||
* 新拖入组件时,计算调整其他关联组件位置大小 |
||||
* |
||||
* @param child 新拖入的组件 |
||||
* @param x 鼠标所在x坐标 |
||||
* @param y 鼠标所在y坐标 |
||||
*/ |
||||
private void fixAbsolute(XCreator child, int x, int y) { |
||||
Component parentComp = container.getComponentAt(x, y); |
||||
if (container.getComponentCount() == 0) { |
||||
child.setLocation(0, 0); |
||||
child.setSize(parentComp.getWidth(), parentComp.getHeight()); |
||||
} else if (isCrossPointArea(parentComp, x, y)) { |
||||
//交叉区域插入组件时,根据具体位置进行上下或者左右或者相邻三个组件的位置大小插入
|
||||
fixCrossPointArea(parentComp, child, x, y); |
||||
//TODO 尽量不要出现这种写法吧?if else条件要么互斥,要么多个if判断return,不要在一条if else语句里面return吧?
|
||||
return; |
||||
} else if (isTrisectionArea(parentComp, x, y)) { |
||||
// 在边界三等分区域,就不再和组件二等分了
|
||||
fixTrisect(parentComp, child, x, y); |
||||
return; |
||||
} else { |
||||
fixHalve(parentComp, child, x, y); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 组件拖拽后调整大小 |
||||
* |
||||
* @param creator 组件 |
||||
*/ |
||||
@Override |
||||
public void fix(XCreator creator) { |
||||
WAbsoluteLayout wabs = (WAbsoluteLayout) container.toData(); |
||||
fix(creator, creator.getX(), creator.getY()); |
||||
wabs.setBounds(creator.toData(), creator.getBounds()); |
||||
|
||||
XWAbsoluteLayout layout = (XWAbsoluteLayout) container; |
||||
layout.updateBoundsWidget(creator); |
||||
} |
||||
|
||||
/** |
||||
* 调整组件大小到合适尺寸位置 |
||||
* |
||||
* @param creator 组件 |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
*/ |
||||
public void fix(XCreator creator, int x, int y) { |
||||
int height = creator.getHeight(); |
||||
int width = creator.getWidth(); |
||||
if (x < 0) { |
||||
width += x; |
||||
x = 0; |
||||
} else if (x + creator.getWidth() > container.getWidth()) { |
||||
width = container.getWidth() - x; |
||||
} |
||||
|
||||
if (y < 0) { |
||||
height += y; |
||||
y = 0; |
||||
} else if (y + creator.getHeight() > container.getHeight()) { |
||||
height = container.getHeight() - y; |
||||
} |
||||
|
||||
creator.setBounds(x, y, width, height); |
||||
} |
||||
|
||||
@Override |
||||
public ConstraintsGroupModel getLayoutConstraints(XCreator creator) { |
||||
return new BoundsGroupModel((XWAbsoluteLayout) container, creator); |
||||
} |
||||
|
||||
@Override |
||||
public GroupModel getLayoutProperties() { |
||||
XWAbsoluteLayout xwAbsoluteLayout = (XWAbsoluteLayout) container; |
||||
return new FRAbsoluteLayoutPropertiesGroupModel(xwAbsoluteLayout); |
||||
} |
||||
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.painters.FRAbsoluteLayoutPainter; |
||||
import com.fr.design.designer.creator.*; |
||||
import com.fr.design.designer.properties.BoundsGroupModel; |
||||
import com.fr.design.designer.properties.FRAbsoluteLayoutPropertiesGroupModel; |
||||
import com.fr.design.utils.ComponentUtils; |
||||
import com.fr.design.utils.gui.LayoutUtils; |
||||
import com.fr.form.ui.container.WAbsoluteLayout; |
||||
import com.fr.general.ComparatorUtils; |
||||
import com.fr.general.FRLogger; |
||||
|
||||
import java.awt.*; |
||||
|
||||
public class FRAbsoluteLayoutAdapter extends FRBodyLayoutAdapter { |
||||
//是不是添加到父容器上
|
||||
private boolean isAdd2ParentLayout = false; |
||||
private HoverPainter painter; |
||||
|
||||
public FRAbsoluteLayoutAdapter(XLayoutContainer container) { |
||||
super(container); |
||||
painter = new FRAbsoluteLayoutPainter(container); |
||||
initMinSize(); |
||||
} |
||||
|
||||
private void initMinSize() { |
||||
XWAbsoluteLayout layout = (XWAbsoluteLayout) container; |
||||
minWidth = layout.getActualMinWidth(); |
||||
minHeight = layout.getActualMinHeight(); |
||||
actualVal = layout.getAcualInterval(); |
||||
margin = layout.toData().getMargin(); |
||||
} |
||||
|
||||
@Override |
||||
public HoverPainter getPainter() { |
||||
return painter; |
||||
} |
||||
|
||||
/** |
||||
* 是否能在指定位置添加组件 |
||||
* |
||||
* @param creator 组件 |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
* @return 能则返回true |
||||
*/ |
||||
//这个地方的逻辑非常复杂,
|
||||
// 1.当前绝对布局是不可编辑且是最外层,那么其他控件添加在它周围,
|
||||
// 2.当前绝对布局是不可编辑且不是最外层,那么控件不可添加,(嵌套)
|
||||
// 3.当前绝对布局可编辑,那么控件添加
|
||||
@Override |
||||
public boolean accept(XCreator creator, int x, int y) { |
||||
Component comp = container.getComponentAt(x, y); |
||||
//布局控件要先判断是不是可编辑
|
||||
//可以编辑,按原有逻辑判断
|
||||
//不可编辑,当成一整个控件处理
|
||||
if (comp == null) { |
||||
return false; |
||||
} |
||||
//参数面板内的组件不允许拖往绝对布局中
|
||||
if (creator.getParent() != null && ((XCreator) creator.getParent()).acceptType(XWParameterLayout.class)) { |
||||
Rectangle rec = creator.getBounds(); |
||||
rec.y = creator.getParent().getHeight() - rec.height; |
||||
creator.setBounds(rec); |
||||
return false; |
||||
} |
||||
//判断组件能不能拖入绝对布局
|
||||
if (!creator.canEnterIntoAbsolutePane()) { |
||||
return false; |
||||
} |
||||
XLayoutContainer topLayout = XCreatorUtils.getHotspotContainer((XCreator) comp).getTopLayout(); |
||||
if (topLayout != null) { |
||||
if (topLayout.isEditable()) { |
||||
return topLayoutAccept(creator, x, y); |
||||
} |
||||
//绝对布局嵌套,处于内层,不可编辑,不添加,topLayout只能获取到最外层可编辑的布局
|
||||
else if (((XLayoutContainer) topLayout.getParent()).acceptType(XWAbsoluteLayout.class)) { |
||||
return false; |
||||
} else { |
||||
return acceptWidget(x, y); |
||||
} |
||||
} else { |
||||
FRLogger.getLogger().error("top layout is null!"); |
||||
} |
||||
|
||||
return false; |
||||
} |
||||
|
||||
//topLayout假如可以编辑的话就往里面添加组件
|
||||
private boolean topLayoutAccept(XCreator creator, int x, int y) { |
||||
//允许组件重叠,可以不判断有没有和当前控件重叠
|
||||
//先计算当前控件的位置
|
||||
int creatorX, creatorY; |
||||
if (XCreatorUtils.getParentXLayoutContainer(creator) != null) { |
||||
Rectangle creatorRectangle = ComponentUtils.getRelativeBounds(creator); |
||||
creatorX = creatorRectangle.x; |
||||
creatorY = creatorRectangle.y; |
||||
} else { |
||||
//这边计算得到的组件其实位置是正确的,
|
||||
//因为传入的x和y已经加上了宽度或者高度的一半,再减去相同的宽度和高度的一半是没区别的,
|
||||
// 例如高度为21,那么就是+10-10;
|
||||
// 高度为20,那么就是+10-10; 没区别
|
||||
int w = creator.getWidth() / 2; |
||||
int h = creator.getHeight() / 2; |
||||
creatorX = x - w; |
||||
creatorY = y - h; |
||||
} |
||||
if (creatorX < 0 || creatorX + creator.getWidth() > container.getWidth()) { |
||||
return false; |
||||
} |
||||
if (creatorY < 0 || creatorY + creator.getHeight() > container.getHeight()) { |
||||
return false; |
||||
} |
||||
return x >= 0 && y >= 0 && creator.getHeight() <= container.getHeight() |
||||
&& creator.getWidth() <= container.getWidth(); |
||||
} |
||||
|
||||
/** |
||||
* 判断是否鼠标在组件的三等分区域,如果组件在布局管理器中间,上下左右都可能会三等分 |
||||
* |
||||
* @param parentComp 鼠标所在区域的组件 |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
* @return 是则返回true |
||||
*/ |
||||
public boolean isTrisectionArea(Component parentComp, int x, int y) { |
||||
XCreator creator = (XCreator) parentComp; |
||||
trisectAreaDirect = 0; |
||||
if (container.getComponentCount() <= 1) { |
||||
return false; |
||||
} |
||||
int maxWidth = parentComp.getWidth(); |
||||
int maxHeight = parentComp.getHeight(); |
||||
int xL = parentComp.getX(); |
||||
int yL = parentComp.getY(); |
||||
// 组件宽高的十分之一和默认值取大
|
||||
int minRangeWidth = Math.max(maxWidth / BORDER_PROPORTION, DEFAULT_AREA_LENGTH); |
||||
int minRangeHeight = Math.max(maxHeight / BORDER_PROPORTION, DEFAULT_AREA_LENGTH); |
||||
if (y < yL + minRangeHeight) { |
||||
// 在组件上侧三等分
|
||||
trisectAreaDirect = COMP_TOP; |
||||
} else if (y > yL + maxHeight - minRangeHeight) { |
||||
// 在组件下侧三等分
|
||||
trisectAreaDirect = COMP_BOTTOM; |
||||
} else if (x < xL + minRangeWidth) { |
||||
// 在组件左侧三等分
|
||||
trisectAreaDirect = COMP_LEFT; |
||||
} else if (x > xL + maxWidth - minRangeWidth) { |
||||
// 在组件右侧三等分
|
||||
trisectAreaDirect = COMP_RIGHT; |
||||
} |
||||
// tab布局的边界特殊处理,不进行三等分
|
||||
if (!creator.getTargetChildrenList().isEmpty()) { |
||||
return false; |
||||
} |
||||
|
||||
return !ComparatorUtils.equals(trisectAreaDirect, 0); |
||||
} |
||||
|
||||
//当前绝对布局不可编辑,就当成一个控件,组件添加在周围
|
||||
private boolean acceptWidget(int x, int y) { |
||||
isFindRelatedComps = false; |
||||
//拖入组件判断时,先判断是否为交叉点区域,其次三等分区域,再次平分区域
|
||||
Component comp = container.getComponentAt(x, y); |
||||
//如果当前处于边缘地带, 那么就把他贴到父容器上
|
||||
XLayoutContainer parent = container.findNearestFit(); |
||||
container = parent != null ? parent : container; |
||||
isAdd2ParentLayout = true; |
||||
|
||||
int componentHeight = comp.getHeight(); |
||||
int componentWidth = comp.getWidth(); |
||||
//上半部分高度
|
||||
int upHeight = (int) (componentHeight * TOP_HALF) + comp.getY(); |
||||
//下半部分高度
|
||||
int downHeight = (int) (componentHeight * BOTTOM_HALF) + comp.getY(); |
||||
|
||||
if (isCrossPointArea(comp, x, y)) { |
||||
return canAcceptWhileCrossPoint(comp, x, y); |
||||
} |
||||
|
||||
if (isTrisectionArea(comp, x, y)) { |
||||
return canAcceptWhileTrisection(comp, x, y); |
||||
} |
||||
|
||||
boolean horizonValid = componentWidth >= minWidth * 2 + actualVal; |
||||
boolean verticalValid = componentHeight >= minHeight * 2 + actualVal; |
||||
return y > upHeight && y < downHeight ? horizonValid : verticalValid; |
||||
} |
||||
|
||||
/** |
||||
* 组件的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) { |
||||
Rectangle rect = ComponentUtils.getRelativeBounds(container); |
||||
|
||||
int posX = x + rect.x; |
||||
int posY = y + rect.y; |
||||
if (!accept(creator, x, y)) { |
||||
return false; |
||||
} |
||||
addComp(creator, posX, posY); |
||||
((XWidgetCreator) creator).recalculateChildrenSize(); |
||||
return true; |
||||
} |
||||
|
||||
@Override |
||||
protected void addComp(XCreator creator, int x, int y) { |
||||
if (!isAdd2ParentLayout) { |
||||
Rectangle r = ComponentUtils.getRelativeBounds(container); |
||||
x = x - r.x; |
||||
y = y - r.y; |
||||
if (XCreatorUtils.getParentXLayoutContainer(creator) != null) { |
||||
|
||||
Rectangle creatorRectangle = ComponentUtils.getRelativeBounds(creator); |
||||
x = creatorRectangle.x - r.x; |
||||
y = creatorRectangle.y - r.y; |
||||
} else { |
||||
int w = creator.getWidth() / 2; |
||||
int h = creator.getHeight() / 2; |
||||
x = x - w; |
||||
y = y - h; |
||||
} |
||||
fix(creator, x, y); |
||||
|
||||
if (creator.hasTitleStyle()) { |
||||
addParentCreator(creator); |
||||
} else { |
||||
container.add(creator, creator.toData().getWidgetName()); |
||||
} |
||||
XWAbsoluteLayout layout = (XWAbsoluteLayout) container; |
||||
layout.updateBoundsWidget(creator); |
||||
updateCreatorBackBound(); |
||||
LayoutUtils.layoutRootContainer(container); |
||||
} else { |
||||
fixAbsolute(creator, x, y); |
||||
if (creator.shouldScaleCreator() || creator.hasTitleStyle()) { |
||||
addParentCreator(creator); |
||||
} else { |
||||
container.add(creator, creator.toData().getWidgetName()); |
||||
} |
||||
XWFitLayout layout = (XWFitLayout) container; |
||||
// 更新对应的BoundsWidget
|
||||
layout.updateBoundsWidget(); |
||||
updateCreatorBackBound(); |
||||
} |
||||
} |
||||
|
||||
private void updateCreatorBackBound() { |
||||
for (int i = 0, size = container.getComponentCount(); i < size; i++) { |
||||
XCreator creator = (XCreator) container.getComponent(i); |
||||
creator.updateChildBound(minHeight); |
||||
creator.setBackupBound(creator.getBounds()); |
||||
} |
||||
} |
||||
|
||||
private void addParentCreator(XCreator child) { |
||||
XLayoutContainer parentPanel = child.initCreatorWrapper(child.getHeight()); |
||||
container.add(parentPanel, child.toData().getWidgetName()); |
||||
} |
||||
|
||||
/** |
||||
* 新拖入组件时,计算调整其他关联组件位置大小 |
||||
* |
||||
* @param child 新拖入的组件 |
||||
* @param x 鼠标所在x坐标 |
||||
* @param y 鼠标所在y坐标 |
||||
*/ |
||||
private void fixAbsolute(XCreator child, int x, int y) { |
||||
Component parentComp = container.getComponentAt(x, y); |
||||
if (container.getComponentCount() == 0) { |
||||
child.setLocation(0, 0); |
||||
child.setSize(parentComp.getWidth(), parentComp.getHeight()); |
||||
} else if (isCrossPointArea(parentComp, x, y)) { |
||||
//交叉区域插入组件时,根据具体位置进行上下或者左右或者相邻三个组件的位置大小插入
|
||||
fixCrossPointArea(parentComp, child, x, y); |
||||
//TODO 尽量不要出现这种写法吧?if else条件要么互斥,要么多个if判断return,不要在一条if else语句里面return吧?
|
||||
return; |
||||
} else if (isTrisectionArea(parentComp, x, y)) { |
||||
// 在边界三等分区域,就不再和组件二等分了
|
||||
fixTrisect(parentComp, child, x, y); |
||||
return; |
||||
} else { |
||||
fixHalve(parentComp, child, x, y); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 组件拖拽后调整大小 |
||||
* |
||||
* @param creator 组件 |
||||
*/ |
||||
@Override |
||||
public void fix(XCreator creator) { |
||||
WAbsoluteLayout wabs = (WAbsoluteLayout) container.toData(); |
||||
fix(creator, creator.getX(), creator.getY()); |
||||
wabs.setBounds(creator.toData(), creator.getBounds()); |
||||
|
||||
XWAbsoluteLayout layout = (XWAbsoluteLayout) container; |
||||
layout.updateBoundsWidget(creator); |
||||
} |
||||
|
||||
/** |
||||
* 调整组件大小到合适尺寸位置 |
||||
* |
||||
* @param creator 组件 |
||||
* @param x 坐标x |
||||
* @param y 坐标y |
||||
*/ |
||||
public void fix(XCreator creator, int x, int y) { |
||||
int height = creator.getHeight(); |
||||
int width = creator.getWidth(); |
||||
if (x < 0) { |
||||
width += x; |
||||
x = 0; |
||||
} else if (x + creator.getWidth() > container.getWidth()) { |
||||
width = container.getWidth() - x; |
||||
} |
||||
|
||||
if (y < 0) { |
||||
height += y; |
||||
y = 0; |
||||
} else if (y + creator.getHeight() > container.getHeight()) { |
||||
height = container.getHeight() - y; |
||||
} |
||||
|
||||
creator.setBounds(x, y, width, height); |
||||
} |
||||
|
||||
@Override |
||||
public ConstraintsGroupModel getLayoutConstraints(XCreator creator) { |
||||
return new BoundsGroupModel((XWAbsoluteLayout) container, creator); |
||||
} |
||||
|
||||
@Override |
||||
public GroupModel getLayoutProperties() { |
||||
XWAbsoluteLayout xwAbsoluteLayout = (XWAbsoluteLayout) container; |
||||
return new FRAbsoluteLayoutPropertiesGroupModel(xwAbsoluteLayout); |
||||
} |
||||
} |
File diff suppressed because it is too large
Load Diff
@ -1,107 +1,104 @@
|
||||
/** |
||||
* |
||||
*/ |
||||
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布局的边缘,则需要把新组件添加到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(); |
||||
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.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 { |
||||
|
||||
/** |
||||
* 构造函数 |
||||
* |
||||
* @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(); |
||||
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); |
||||
} |
||||
} |
@ -1,96 +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); |
||||
//实现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; |
||||
} |
||||
|
||||
/** |
||||
* |
||||
*/ |
||||
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,347 +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]; |
||||
} |
||||
|
||||
//选择的控件
|
||||
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; |
||||
} |
||||
|
||||
} |
||||
|
||||
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,372 +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); |
||||
//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; |
||||
} |
||||
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