/*
* Freeplane - mind map editor
* Copyright (C) 2008 Joerg Mueller, Daniel Polansky, Christian Foltin, Dimitry Polivaev
*
* This file author is Christian Foltin
* It is modified by Dimitry Polivaev in 2008.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package org.freeplane.features.icon;
import java.util.Map;
import org.freeplane.core.extension.IExtension;
import org.freeplane.core.io.IReadCompletionListener;
import org.freeplane.features.map.IMapChangeListener;
import org.freeplane.features.map.INodeChangeListener;
import org.freeplane.features.map.MapChangeEvent;
import org.freeplane.features.map.MapModel;
import org.freeplane.features.map.NodeChangeEvent;
import org.freeplane.features.map.NodeModel;
import org.freeplane.features.mode.Controller;
import org.freeplane.features.mode.ModeController;
import org.freeplane.features.mode.NodeHookDescriptor;
import org.freeplane.features.mode.PersistentNodeHook;
import org.freeplane.features.styles.MapStyle;
import org.freeplane.features.styles.MapStyleModel;
import org.freeplane.n3.nanoxml.XMLElement;
/**
* @author Foltin
*/
@NodeHookDescriptor(hookName = "accessories/plugins/HierarchicalIcons.properties")
public class HierarchicalIcons extends PersistentNodeHook implements INodeChangeListener, IMapChangeListener,
IReadCompletionListener, IExtension {
public static final String ICONS = "hierarchical_icons";
public HierarchicalIcons() {
this(Mode.OR);
final ModeController modeController = Controller.getCurrentModeController();
IconController.getController(modeController).addStateIconProvider(new IStateIconProvider() {
public UIIcon getStateIcon(NodeModel node) {
AccumulatedIcons iconSet = node.getExtension(AccumulatedIcons.class);
if(iconSet != null)
return new UIIconSet(iconSet.getAccumulatedIcons(), 0.75f);
else
return null;
}
});
new HierarchicalIcons2();
}
protected HierarchicalIcons(Mode mode) {
super();
this.mode = mode;
final ModeController modeController = Controller.getCurrentModeController();
modeController.getMapController().getReadManager().addReadCompletionListener(this);
modeController.getMapController().addNodeChangeListener(this);
modeController.getMapController().addMapChangeListener(this);
}
@Override
protected void add(final NodeModel node, final IExtension extension) {
if(MapStyleModel.getExtension(node.getMap()) != null){
gatherLeavesAndSetStyle(node);
gatherLeavesAndSetParentsStyle(node);
}
super.add(node, extension);
}
@Override
protected IExtension createExtension(final NodeModel node, final XMLElement element) {
return this;
}
/**
*/
private void gatherLeavesAndSetParentsStyle(final NodeModel node) {
if (node.getChildCount() == 0) {
for (NodeModel parent = node.getParentNode(); parent != null; parent = parent.getParentNode()) {
AccumulatedIcons.setStyleCheckForChange(parent, mode);
}
return;
}
for (final NodeModel child : Controller.getCurrentModeController().getMapController().childrenUnfolded(node)) {
gatherLeavesAndSetParentsStyle(child);
}
}
/**
*/
private void gatherLeavesAndSetStyle(final NodeModel node) {
node.removeExtension(AccumulatedIcons.class);
if (node.getChildCount() == 0) {
AccumulatedIcons.setStyleCheckForChange(node, mode);
return;
}
for (final NodeModel child : Controller.getCurrentModeController().getMapController().childrenUnfolded(node)) {
gatherLeavesAndSetStyle(child);
}
}
public void mapChanged(final MapChangeEvent event) {
final MapModel map = event.getMap();
if(map == null){
return;
}
final NodeModel rootNode = map.getRootNode();
if (!isActive(rootNode)) {
return;
}
final Object property = event.getProperty();
if(! property.equals(MapStyle.MAP_STYLES)){
return;
}
gatherLeavesAndSetStyle(rootNode);
gatherLeavesAndSetParentsStyle(rootNode);
}
public void nodeChanged(final NodeChangeEvent event) {
final NodeModel node = event.getNode();
if (!isActive(node)) {
return;
}
setStyleRecursive(node);
}
public void onNodeDeleted(final NodeModel parent, final NodeModel child, final int index) {
if (!isActive(parent)) {
return;
}
setStyleRecursive(parent);
}
public void onNodeInserted(final NodeModel parent, final NodeModel child, final int newIndex) {
if (!isActive(parent)) {
return;
}
setStyleRecursive(child);
}
public void onNodeMoved(final NodeModel oldParent, final int oldIndex, final NodeModel newParent,
final NodeModel child, final int newIndex) {
if (!isActive(newParent)) {
return;
}
setStyleRecursive(oldParent);
setStyleRecursive(child);
}
public void onPreNodeDelete(final NodeModel parent, final NodeModel child, final int index) {
}
public void readingCompleted(final NodeModel topNode, final Map<String, String> newIds) {
if (!topNode.containsExtension(getClass()) && !topNode.getMap().getRootNode().containsExtension(getClass())) {
return;
}
gatherLeavesAndSetStyle(topNode);
gatherLeavesAndSetParentsStyle(topNode);
}
@Override
protected void remove(final NodeModel node, final IExtension extension) {
removeIcons(node);
super.remove(node, extension);
}
/**
*/
private void removeIcons(final NodeModel node) {
AccumulatedIcons icons = node.removeExtension(AccumulatedIcons.class);
if(icons != null){
Controller.getCurrentModeController().getMapController().delayedNodeRefresh(node, HierarchicalIcons.ICONS, null, null);
for (final NodeModel child : Controller.getCurrentModeController().getMapController().childrenUnfolded(node)) {
removeIcons(child);
}
}
}
public static enum Mode{AND, OR};
private Mode mode = Mode.OR;
/**
*/
private void setStyleRecursive(final NodeModel node) {
if (AccumulatedIcons.setStyleCheckForChange(node, mode) && node.getParentNode() != null) {
setStyleRecursive(node.getParentNode());
}
}
public void onPreNodeMoved(final NodeModel oldParent, final int oldIndex, final NodeModel newParent,
final NodeModel child, final int newIndex) {
}
}
@NodeHookDescriptor(hookName = "accessories/plugins/HierarchicalIcons2.properties")
class HierarchicalIcons2 extends HierarchicalIcons{
public HierarchicalIcons2() {
super(Mode.AND);
}
}