Package org.openscience.jchempaint.controller

Source Code of org.openscience.jchempaint.controller.AbstractSelectModule

/* $Revision$ $Author$ $Date$
*
*  Copyright (C) 2008  Gilleain Torrance <gilleain.torrance@gmail.com>
*
*  Contact: cdk-devel@list.sourceforge.net
*
*  This program is free software; you can redistribute it and/or
*  modify it under the terms of the GNU Lesser General Public License
*  as published by the Free Software Foundation; either version 2.1
*  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 Lesser General Public License for more details.
*
*  You should have received a copy of the GNU Lesser General Public License
*  along with this program; if not, write to the Free Software
*  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*/
package org.openscience.jchempaint.controller;

import java.awt.Cursor;
import java.awt.event.KeyEvent;
import java.awt.event.MouseEvent;
import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;

import javax.vecmath.Point2d;

import org.openscience.cdk.AtomContainer;
import org.openscience.cdk.interfaces.IAtom;
import org.openscience.cdk.interfaces.IAtomContainer;
import org.openscience.cdk.interfaces.IBond;
import org.openscience.cdk.interfaces.IChemObject;
import org.openscience.cdk.renderer.selection.AbstractSelection;
import org.openscience.jchempaint.renderer.BoundsCalculator;
import org.openscience.jchempaint.renderer.JChemPaintRendererModel;
import org.openscience.cdk.renderer.selection.IChemObjectSelection;
import org.openscience.jchempaint.renderer.selection.LogicalSelection;
import org.openscience.jchempaint.renderer.selection.ShapeSelection;
import org.openscience.jchempaint.renderer.selection.SingleSelection;

/**
* An abstract base class for select modules. Classes extending this must
* implement getDrawModeString and set selection to the appropriate type in
* their constructor.
*
* @cdk.module controlbasic
*/
public abstract class AbstractSelectModule extends ControllerModuleAdapter {
   
    protected ShapeSelection selection;
    private Point2d startPoint;
    private String ID;
   
    public AbstractSelectModule(IChemModelRelay chemModelRelay) {
        super(chemModelRelay);
    }

  private IAtom getClosestSelAtom(Point2d worldCoord) {
    IAtom closestAtom = null;
    IChemObjectSelection sel = chemModelRelay.getRenderer().getRenderer2DModel().getSelection();
    double closestDistanceSQ = Double.MAX_VALUE;

    for (IAtom atom : sel.elements(IAtom.class)) {
      if (atom.getPoint2d() != null) {
        double distanceSQ = atom.getPoint2d().distanceSquared(
            worldCoord);
        if (distanceSQ < closestDistanceSQ) {
          closestAtom = atom;
          closestDistanceSQ = distanceSQ;
        }
      }
    }
    return closestAtom;
  }
 
  private IBond getClosestSelBond(Point2d worldCoord) {
    IBond closestBond = null;
    IChemObjectSelection sel = chemModelRelay.getRenderer().getRenderer2DModel().getSelection();
    double closestDistanceSQ = Double.MAX_VALUE;

    for (IBond bond : sel.elements(IBond.class)) {
      if (bond.get2DCenter() != null) {
        double distanceSQ = bond.get2DCenter().distanceSquared(
            worldCoord);
        if (distanceSQ < closestDistanceSQ) {
          closestBond = bond;
          closestDistanceSQ = distanceSQ;
        }
      }
    }
    return closestBond;
  }

       
   
    public void mouseDrag(Point2d from, Point2d to) {
        Rectangle2D bounds=null;
        boolean inSelectionCircle = false;
        JChemPaintRendererModel model = chemModelRelay.getRenderer().getRenderer2DModel();
        IChemObjectSelection sel = model.getSelection();
        if (sel == null) return;
        double d = model.getSelectionRadius() / model.getScale();
        IAtom closestAtom = null;
        IBond closestBond = null;
        IAtom highlitAtom = model.getHighlightedAtom();
        IBond highlitBond = model.getHighlightedBond();
        if(from.equals(startPoint)) {
          LogicalSelection lsel = null;
          boolean isAllSelected = false;
            if (sel.getClass().isAssignableFrom(LogicalSelection.class)) {
              lsel = (LogicalSelection)sel;
                isAllSelected = lsel.getType() == LogicalSelection.Type.ALL
                && sel.isFilled();
          }
          if (!isAllSelected && sel != null && sel.isFilled()) {
              bounds = BoundsCalculator.calculateBounds(sel.getConnectedAtomContainer());
              closestAtom = getClosestSelAtom(startPoint);
              closestBond = getClosestSelBond(startPoint);
          }
            inSelectionCircle = (closestAtom != null && closestAtom.getPoint2d().distance(startPoint) < 4*d)
                  || (closestBond != null && closestBond.get2DCenter().distance(startPoint) < 4*d)
                || highlitAtom != null || highlitBond != null;
          //in case the user either starts dragging inside the current selection
          //rectangle or in a selection circle, we switch to move mode,
          //else we start a new selection
          if(inSelectionCircle || isAllSelected
              || (bounds !=null
              && bounds.contains(new Point2D.Double(startPoint.x, startPoint.y)))) {
                  IControllerModule newActiveModule = new MoveModule(this.chemModelRelay, this);
                  newActiveModule.setID("move");
                  chemModelRelay.setActiveDrawModule(newActiveModule);
                  Point2d sf = chemModelRelay.getRenderer().toScreenCoordinates(startPoint.x, startPoint.y);
                  ((IMouseEventRelay)this.chemModelRelay).mouseClickedDown((int)sf.x,(int)sf.y);
                  Point2d st = chemModelRelay.getRenderer().toScreenCoordinates(to.x, to.y);
                  ((IMouseEventRelay)this.chemModelRelay).mouseDrag((int)sf.x,(int)sf.y,(int)st.x,(int)st.y);
                  return;
          }
        }
        this.selection.addPoint(to);
        this.chemModelRelay.select(selection);
        this.chemModelRelay.getRenderer().getRenderer2DModel().setSelection(this.selection);
        this.chemModelRelay.updateView();
    }
   
    public void mouseMove(Point2d p){
        showMouseCursor(p, this.chemModelRelay);
    }

    public static void showMouseCursor(Point2d p, IChemModelRelay chemModelRelay){
       
        //We look if we the user would move when clicking and show arrows then
        Rectangle2D bounds=null;
        if(chemModelRelay.getRenderer().getRenderer2DModel().getSelection()!=null
                    && chemModelRelay.getRenderer().getRenderer2DModel()
                    .getSelection().isFilled())
            try {
                bounds = BoundsCalculator.calculateBounds(chemModelRelay
                        .getRenderer().getRenderer2DModel().getSelection()
                        .getConnectedAtomContainer());
            } catch (NullPointerException e) {
               
                bounds=null;
            }
        IChemObject highlightedAtom = chemModelRelay.getRenderer().getRenderer2DModel().getHighlightedAtom();
        IChemObject highlightedBond = chemModelRelay.getRenderer().getRenderer2DModel().getHighlightedBond();
        if((bounds !=null && bounds.contains(new Point2D.Double(p.x, p.y)))
                || highlightedAtom!=null || highlightedBond!=null){
            chemModelRelay.setCursor(Cursor.HAND_CURSOR);
        }else{
            chemModelRelay.setCursor(Cursor.DEFAULT_CURSOR);
        }
    }

    @Override
    public void mouseClickedDown(Point2d p) {
        mouseClickedDown(p, 0);
    }

    @Override
    public void mouseClickedUp(Point2d p) {
        mouseClickedUp(p, 0);
    }

    @Override
    public void mouseClickedDown(Point2d p, int modifiers) {
        startPoint=p;
    }

    @Override
    public void mouseClickedUp(Point2d p, int modifiers) {
        chemModelRelay.select(selection);
        selection.reset();

        boolean shiftPressed = (modifiers & MouseEvent.SHIFT_DOWN_MASK) != 0;

        if(p.equals(startPoint)){
           
            IAtom closestAtom = chemModelRelay.getClosestAtom(p);
            IBond closestBond = chemModelRelay.getClosestBond(p);
            IChemObject singleSelection = getHighlighted( p,
                    closestAtom,
                    closestBond );

            IChemObjectSelection curSel = chemModelRelay.getRenderer().getRenderer2DModel().getSelection();
            IChemObjectSelection sel = null;
            if (shiftPressed && curSel != null && curSel.isFilled()) {
                IAtomContainer container = new AtomContainer(curSel.getConnectedAtomContainer());               
                if (singleSelection instanceof IAtom) {
                    IAtom atom = (IAtom) singleSelection;
                    if (!container.contains(atom))  container.addAtom(atom);
                    else container.removeAtom(atom);
                } else if (singleSelection instanceof IBond) {
                    IBond bond = (IBond) singleSelection;
                    if (!container.contains(bond))  container.addBond(bond);
                    else container.addBond(bond);
                }
                LogicalSelection logSel = new LogicalSelection(LogicalSelection.Type.ALL);
                logSel.select(container);
                sel = logSel;
            } else {
                if (singleSelection == null) {
                    selection.clear();
                    sel = AbstractSelection.EMPTY_SELECTION;
                }
                else if (singleSelection instanceof IAtom) {
                    sel = new SingleSelection<IAtom>((IAtom) singleSelection);
                }
                else if (singleSelection instanceof IBond) {
                    sel = new SingleSelection<IBond>((IBond) singleSelection);
                }
            }      
            this.chemModelRelay.select(sel);
        }

        this.chemModelRelay.updateView();
    }
   
    public String getID() {
        return ID;
    }

    public void setID(String ID) {
        this.ID=ID;
    }

}
TOP

Related Classes of org.openscience.jchempaint.controller.AbstractSelectModule

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.