All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.piccolo2d.extras.swing.PViewport Maven / Gradle / Ivy

There is a newer version: 3.0.1
Show newest version
/*
 * Copyright (c) 2008-2011, Piccolo2D project, http://piccolo2d.org
 * Copyright (c) 1998-2008, University of Maryland
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided
 * that the following conditions are met:
 *
 * Redistributions of source code must retain the above copyright notice, this list of conditions
 * and the following disclaimer.
 *
 * Redistributions in binary form must reproduce the above copyright notice, this list of conditions
 * and the following disclaimer in the documentation and/or other materials provided with the
 * distribution.
 *
 * None of the name of the University of Maryland, the name of the Piccolo2D project, or the names of its
 * contributors may be used to endorse or promote products derived from this software without specific
 * prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
 * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */
package org.piccolo2d.extras.swing;

import java.awt.Component;
import java.awt.Container;
import java.awt.Dimension;
import java.awt.LayoutManager;
import java.awt.Point;
import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;

import javax.swing.JViewport;
import javax.swing.ViewportLayout;

import org.piccolo2d.PCanvas;
import org.piccolo2d.util.PBounds;


/**
 * A subclass of JViewport that talks to the scroll director to negotiate the
 * view positions and sizes.
 * 
 * @author Lance Good
 */
public class PViewport extends JViewport {
    private static final long serialVersionUID = 1L;
    /** Controls what happens when scrolling occurs. */
    PScrollDirector scrollDirector;

    /** Pass constructor info to super. */
    public PViewport() {
        super();

        setScrollDirector(createScrollDirector());
    }

    /**
     * Subclasses can override this to install a different layout manager (or
     * null) in the constructor. Returns a new
     * ViewportLayout object.
     * 
     * @return a LayoutManager
     */
    protected LayoutManager createLayoutManager() {
        return new PViewportLayout();
    }

    /**
     * Subclasses can override this to install a different scroll director in
     * the constructor. Returns a new PScrollDirector object.
     * 
     * @return a PScrollDirector
     */
    protected PScrollDirector createScrollDirector() {
        return new PDefaultScrollDirector();
    }

    /**
     * Set the scroll director on this viewport.
     * 
     * @param scrollDirector The new scroll director
     */
    public void setScrollDirector(final PScrollDirector scrollDirector) {
        if (this.scrollDirector != null) {
            this.scrollDirector.unInstall();
        }
        this.scrollDirector = scrollDirector;
        if (scrollDirector != null) {
            this.scrollDirector.install(this, (PCanvas) getView());
        }
    }

    /**
     * Returns the scroll director on this viewport.
     * 
     * @return The scroll director on this viewport
     */
    public PScrollDirector getScrollDirector() {
        return scrollDirector;
    }

    /**
     * Overridden to throw an exception if the view is not a PCanvas.
     * 
     * @param view The new view - it better be a ZCanvas!
     */
    public void setView(final Component view) {
        if (!(view instanceof PCanvas)) {
            throw new UnsupportedOperationException("PViewport only supports ZCanvas");
        }

        super.setView(view);

        if (scrollDirector != null) {
            scrollDirector.install(this, (PCanvas) view);
        }
    }

    /**
     * Notifies all ChangeListeners when the views size, position,
     * or the viewports extent size has changed.
     * 
     * PDefaultScrollDirector calls this so it needs to be public.
     */
    public void fireStateChanged() {
        super.fireStateChanged();
    }

    /**
     * Sets the view coordinates that appear in the upper left hand corner of
     * the viewport, does nothing if there's no view.
     * 
     * @param p a Point object giving the upper left coordinates
     */
    public void setViewPosition(final Point p) {
        if (getView() == null) {
            return;
        }

        double oldX = 0, oldY = 0;
        final double x = p.x, y = p.y;

        final Point2D vp = getViewPosition();
        if (vp != null) {
            oldX = vp.getX();
            oldY = vp.getY();
        }

        // Send the scroll director the exact view position and let it interpret
        // it as needed
        final double newX = x;
        final double newY = y;

        if (oldX != newX || oldY != newY) {
            scrollUnderway = true;

            scrollDirector.setViewPosition(newX, newY);

            fireStateChanged();
        }
    }

    /**
     * Gets the view position from the scroll director based on the current
     * extent size.
     * 
     * @return The new view's position
     */
    public Point getViewPosition() {
        if (scrollDirector == null) {
            return null;
        }

        final Dimension extent = getExtentSize();
        return scrollDirector.getViewPosition(new PBounds(0, 0, extent.getWidth(), extent.getHeight()));
    }

    /**
     * Gets the view size from the scroll director based on the current extent
     * size.
     * 
     * @return The new view size
     */
    public Dimension getViewSize() {
        final Dimension extent = getExtentSize();
        return scrollDirector.getViewSize(new PBounds(0, 0, extent.getWidth(), extent.getHeight()));
    }

    /**
     * Gets the view size from the scroll director based on the specified extent
     * size.
     * 
     * @param r The extent size from which the view is computed
     * @return The new view size
     */
    public Dimension getViewSize(final Rectangle2D r) {
        return scrollDirector.getViewSize(r);
    }

    /**
     * A simple layout manager to give the ZCanvas the same size as the Viewport.
     */
    public static class PViewportLayout extends ViewportLayout {
        private static final long serialVersionUID = 1L;

        /**
         * Called when the specified container needs to be laid out.
         * 
         * @param parent the container to lay out
         */
        public void layoutContainer(final Container parent) {
            if (!(parent instanceof JViewport)) {
                throw new IllegalArgumentException("PViewport.layoutContainer may only be applied to JViewports");
            }
            final JViewport vp = (JViewport) parent;
            final Component view = vp.getView();

            if (view == null) {
                return;
            }

            final Dimension extentSize = vp.getSize();

            vp.setViewSize(extentSize);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy