Package viewer.zoomable

Source Code of viewer.zoomable.ScrollableObject

/*
*  (C) 2001 by Argonne National Laboratory
*      See COPYRIGHT in top-level directory.
*/

/*
*  @author  Anthony Chan
*/

package viewer.zoomable;

import java.awt.*;
// import java.awt.event.*;
import javax.swing.*;

import base.drawable.TimeBoundingBox;

public abstract class ScrollableObject extends JComponent
                                       implements ScrollableView
{
    //  The least number of images for this class to work is "3"
    private   static final int   NumImages = 3;
    protected static final int   NumViewsPerImage = 2;
    protected static final int   NumViewsTotal = NumImages * NumViewsPerImage;

    private   ModelTime          model = null;
    private   JViewport          src_vport;
    private   Image              offscreenImages[ /* NumImages */ ];

    // The start and end of the image(s) in the user time coordinates
    private   TimeBoundingBox    tImages[ /* NumImages  */ ];
    private   double             tImage_extent;
    private   TimeBoundingBox    tImages_all;  // extremes of tImages[]

    // shorthand for some convenient constant
    private   int                half_NumImages;

    // There are 2 kinds of indexes to label the image in the array buffer. 
    // Both of the indexes are in the range of {0 : NumImages-1}.
    // - The 1st kind is called image order.  The image order for the
    //   image where viewport is in is always half_NumImages
    // - The 2nd kind is called image index.  The image index for the
    //   image where viewport is in can be any integer in {0 : NumImages-1}
    //   The variable, cur_img_idx, serves to keep track where this image
    //   is in the image array buffer.
    private   int                cur_img_idx;

    // The size of the visible view window in the viewport in pixel coordinates
    private   Dimension          visible_size;
    // The size of an image in pixel coordinates
    private   Dimension          image_size;
    // The size of this JCompoent in pixel coordinates
    private   Dimension          component_size;

    //  The following constructor is NOT meant to be called.
    public ScrollableObject( ModelTime model )
    {
        // Check if the number of images is an ODD number
        if ( NumImages % 2 == 0 || NumImages < 3 ) {
            String err_msg = "ScrollableObject(): NumImages = "
                           + NumImages + " which is invalid, "
                           + "i.e. either an EVEN number or < 3.";
            throw new IllegalStateException( err_msg );
            // System.exit( 1 );
        }

        this.model = model;
        offscreenImages = new Image[ NumImages ];
        tImages         = new TimeBoundingBox[ NumImages ];
        for ( int idx = 0; idx < NumImages; idx++ )
            tImages[ idx ] = new TimeBoundingBox();
        tImages_all     = new TimeBoundingBox();
        super.setDoubleBuffered( false );

        // Initialize the current image index and each image's time bound
        half_NumImages = NumImages / 2;
        setImagesInitTimeBounds();

        image_size     = new Dimension( 0, 0 );
        component_size = super.getSize();

        // Enable debugging graphics option
        setDebugGraphicsOptions( DebugGraphics.LOG_OPTION
                               | DebugGraphics.BUFFERED_OPTION
                               | DebugGraphics.FLASH_OPTION );
    }

    // tImages_all needs to be synchronized with tImages[]
    private void setImagesInitTimeBounds()
    {
        double model_view_extent = model.getTimeViewExtent();
        tImage_extent            = NumViewsPerImage * model_view_extent;
        tImages_all.reinitialize();

        int img_idx = 0;
        tImages[ img_idx ].setEarliestTime( model.getTimeViewPosition()
                                          - 0.5 * model_view_extent
                                          * (NumViewsTotal - 1) );
        tImages[ img_idx ].setLatestFromEarliest( tImage_extent );
        tImages_all.affectTimeBounds( tImages[ img_idx ] );
        for ( img_idx = 1; img_idx < NumImages; img_idx++ ) {
            tImages[ img_idx ].setEarliestTime(
                               tImages[ img_idx - 1 ].getLatestTime() );
            tImages[ img_idx ].setLatestFromEarliest( tImage_extent );
            tImages_all.affectTimeBounds( tImages[ img_idx ] );
        }
        // initialize cur_img_idx in offscreenImages[]
        cur_img_idx = half_NumImages;
    }

    public String getStringforTimesOfImages()
    {
        StringBuffer rep = new StringBuffer();
        for ( int img_idx = 0; img_idx < NumImages; img_idx++ )
            rep.append( "tImages[ " + img_idx + " ] = "
                      + tImages[ img_idx ] + "\n" );
        return rep.toString();
    }

    /*
    // tImages[ img_idx ] is guaranteed to be continuous in this class
    public boolean areImagesContinuousInTime()
    {
        double time_min;
        int    curr_img_idx, prev_img_idx, init_img_idx;

        init_img_idx = 0;
        time_min = tImages[ init_img_idx ].getEarliestTime();
        for ( int img_idx = 1; img_idx < NumImages; img_idx++ )
            if ( tImages[ img_idx ].getEarliestTime() < time_min ) {
                init_img_idx = img_idx;
                time_min = tImages[ init_img_idx ].getEarliestTime();
            }

        boolean isContinuous = true;
        for ( int ii = 1; ii < NumImages; ii++ ) {
            curr_img_idx = getValidImageIndex( init_img_idx + ii );
            prev_img_idx = getValidImageIndex( curr_img_idx - 1 );
            isContinuous = isContinuous
                         && (  tImages[ curr_img_idx ].getEarliestTime()
                            == tImages[ prev_img_idx ].getLatestTime() );
        }

        return isContinuous;
    }
    */

    //  Using cur_img_idx to compute the extreme times of tImages[] is
    //  NOT reliable when cur_img_idx is updated in checkToScrollView(). 
    //  getEarliestTimeOfImages() and getLatestTimeOfImages()
    //  are some of the examples.
    /*
    public double getEarliestTimeOfImages()
    {
        int head_img_idx;
        // compute the beginning image index in the image buffer
        head_img_idx = getValidImageIndex( cur_img_idx - half_NumImages - 1 );
        return tImages[ head_img_idx ].getEarliestTime();
    }

    // getLatestTimeOfImages() is NOT reliable in part of checkToScrollView()
    public double getLatestTimeOfImages()
    {
        int tail_img_idx;
        // compute the last image index in the image buffer
        tail_img_idx = getValidImageIndex( cur_img_idx + half_NumImages + 1 );
        return tImages[ tail_img_idx ].getLatestTime();
    }
    */

    public TimeBoundingBox getTimeBoundsOfImages()
    {
        return new TimeBoundingBox( tImages_all );
    }


    // getValidImageIndex() convert an index to be
    // { 0 <= image index < NumImages }
    //  i.e. implements periodic boundary condition
    private int getValidImageIndex( int img_idx )
    {
        int adj_img_idx;
        adj_img_idx = img_idx % NumImages;
        if ( adj_img_idx < 0 )
            return adj_img_idx + NumImages;
        else
            return adj_img_idx;
    }

    // getPrevImageIndex() and getNextImageIndex() implement circular buffer[]
    private int getNearPastImageIndex( int img_idx )
    {
        if ( img_idx == 0 )
            return NumImages - 1;
        else
            return img_idx - 1;
    }

    private int getNearFutureImageIndex( int img_idx )
    {
        if ( img_idx == NumImages - 1 )
            return 0 ;
        else
            return img_idx + 1;
    }

    /*
    private int getNearImageIndex( int img_idx, int dir )
    {
        if ( dir < 0 )
            return getNearPastImageIndex( img_idx );
        else
            return getNearFutureImageIndex( img_idx );
    }
    */

    //  Given a graphic context, create an offscreen image of specified size.
    private Image createImage( Dimension image_sz )
    {
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: createImage()'s image_sz = "
                         + image_sz );

        if ( image_sz.width > 0 && image_sz.height > 0 )
            return super.createImage( image_sz.width, image_sz.height );
        else
            return null;
    }

    private int getNumImagesMoved()
    {
        double cur_tView_init   = model.getTimeViewPosition();
        double cur_tView_extent = model.getTimeViewExtent();
        double cur_tView_final  = cur_tView_init + cur_tView_extent;

        if ( Debug.isActive() ) {
            Debug.println( "ScrollableObject: getNumImagesMoved()'s START: " );
            Debug.println( "cur_tView_init  = " + cur_tView_init + ",  "
                         + "cur_tView_final = " + cur_tView_final );
            Debug.println( "tImages[ cur ] = " + tImages[ cur_img_idx ] );
        }

        double view_init_in_imgs, view_final_in_imgs;
        int Nimages_moved_fwd, Nimages_moved_back, Nimages_moved;
        double tImages_init;

        // compute the beginning image index in the image buffer
        tImages_init = tImages_all.getEarliestTime();
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: getNumImagesMoved() "
                         + "tImages_init = " + tImages_init );

        // the integer part of view_init_in_imgs is the image order of
        // the image where cur_tView_init is in.  Nimages_moved_fwd is
        // the relative image order w.r.t. center image in the buffer.
        view_init_in_imgs  = ( cur_tView_init  - tImages_init )
                           / tImage_extent;
        Nimages_moved_fwd  = (int) Math.floor( view_init_in_imgs )
                           - half_NumImages;
        // the integer part of view_final_in_imgs is the image order of
        // the image where cur_tView_final is in.  Nimages_moved_back is
        // the relative image order w.r.t. center image in the buffer.
        view_final_in_imgs = ( cur_tView_final - tImages_init )
                           / tImage_extent;
        Nimages_moved_back = (int) Math.floor( view_final_in_imgs )
                           - half_NumImages;

        Nimages_moved = 0;
        if ( Nimages_moved_fwd > 0 )
            Nimages_moved = Nimages_moved_fwd;
        if ( Nimages_moved_back < 0 )
            Nimages_moved = Nimages_moved_back;

        if ( Debug.isActive() ) {
            Debug.println( "ScrollableObject: getNumImagesMoved() "
                         + "Nmages_moved = " + Nimages_moved );
            Debug.println( "ScrollableObject: getNumImagesMoved()'s END: " );
        }

        return Nimages_moved;
    }

    // scrollable_image interface when the view is zoomed in or out.
    public void checkToZoomView()
    {
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: checkToZoomView()'s START: " );
        double cur_tView_extent = model.getTimeViewExtent();
        if ( cur_tView_extent * NumViewsPerImage != tImage_extent ) {
            setImagesInitTimeBounds();

            initializeAllOffImages( tImages_all );
            for ( int img_idx = 0; img_idx < NumImages; img_idx++ )
            // for ( int img_idx = NumImages-1 ; img_idx >= 0 ; img_idx-- )
                drawOneOffImage( offscreenImages[ img_idx ],
                                 tImages[ img_idx ] );
            finalizeAllOffImages( tImages_all );
        }
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: checkToZoomView()'s END: " );
    }

    // scrollable_image interface when the view is scrolled by the scrollbar.
    public void checkToScrollView()
    {
        int Nimages_moved;
        int img_mv_dir, img_idx;
        int past_img_idx, future_img_idx;
        int start_idx;
        int idx;

        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: checkToScrollView()'s START: " );
        //  Using the old cur_img_idx as the center of images to locate
        //  the images needed to be redrawn
        img_mv_dir = 0;
        Nimages_moved = getNumImagesMoved();
        if ( Nimages_moved != 0 ) {
            if ( Math.abs( Nimages_moved ) <= NumImages ) {
                img_mv_dir = Nimages_moved / Math.abs( Nimages_moved );

                // locate the end image index in same direction as img_mv_dir
                start_idx = getValidImageIndex( cur_img_idx
                                              + img_mv_dir * half_NumImages);

                // Determine tImages_all first before invoking
                // initializeAllOffImages() and finalizeAllOffImages()
                for ( idx = 1; idx <= Math.abs( Nimages_moved ); idx++ ) {
                    img_idx = getValidImageIndex( start_idx
                                                + img_mv_dir * idx );
                    if ( Debug.isActive() )
                        Debug.println( "ScrollableObject: checkToScrollView() "
                                     + "cur_img_idx = " + cur_img_idx + ", "
                                     + "start_idx = " + start_idx + ", "
                                     + "img_idx = " + img_idx );

                    // synchronize tImages_all with tImages[]
                    // remove unneeded tImage[ img_idx ] from tImages_all
                    if ( ! tImages_all.remove( tImages[ img_idx ] ) )
                        System.err.println( "ScrollableObject: "
                                          + "checkToScrollView() "
                                          + "tImages[" + img_idx + "] = "
                                          + tImages[ img_idx ] + " does NOT "
                                          + "cover the end of tImages_all = "
                                          + tImages_all );
                    if ( img_mv_dir > 0 ) {
                        past_img_idx = getNearPastImageIndex( img_idx );
                        tImages[ img_idx ].setEarliestTime(
                               tImages[ past_img_idx ].getLatestTime() );
                        tImages[ img_idx ].setLatestFromEarliest(
                                           tImage_extent );
                    }
                    else { // img_mv_dir < 0
                        future_img_idx = getNearFutureImageIndex( img_idx );
                        tImages[ img_idx ].setLatestTime(
                               tImages[ future_img_idx ].getEarliestTime() );
                        tImages[ img_idx ].setEarliestFromLatest(
                                           tImage_extent );
                    }
                    // update tImages_all to reflect changes in tImages[]
                    // so drawOneOffImage() can use tImages_all
                    tImages_all.affectTimeBounds( tImages[ img_idx ] );
                }

                // Update the offscreenImages[] of those scrolled
                initializeAllOffImages( tImages_all );
                if ( img_mv_dir > 0 )
                    //for ( idx = 1; idx <= Math.abs( Nimages_moved ); idx++ ) {
                    for ( idx = Math.abs( Nimages_moved ); idx >=1; idx-- ) {
                        img_idx = getValidImageIndex( start_idx
                                                    + img_mv_dir * idx );
                        if ( offscreenImages[ img_idx ] != null )
                            drawOneOffImage( offscreenImages[ img_idx ],
                                             tImages[ img_idx ] );
                    }
                else
                    for ( idx = 1; idx <= Math.abs( Nimages_moved ); idx++ ) {
                        img_idx = getValidImageIndex( start_idx
                                                    + img_mv_dir * idx );
                        if ( offscreenImages[ img_idx ] != null )
                            drawOneOffImage( offscreenImages[ img_idx ],
                                             tImages[ img_idx ] );
                    }
                finalizeAllOffImages( tImages_all );

                // Update cur_img_idx in the offscreenImages[]
                cur_img_idx = getValidImageIndex( cur_img_idx + Nimages_moved );
            }
            else // Math.abs( Nimages_moved ) > NumImages
                if ( Debug.isActive() ) {
                    Debug.println( "****************************************" );
                    Debug.println( "ScrollableObject: checkToScrollView() "
                                 + "| Nimages_moved( " + Nimages_moved
                                 + " ) | >= NumImages( " + NumImages + " )" );
                }
                setImagesInitTimeBounds();

                initializeAllOffImages( tImages_all );
                for ( img_idx = 0; img_idx < NumImages; img_idx++ )
                // for ( img_idx = NumImages-1; img_idx >=0; img_idx-- )
                    drawOneOffImage( offscreenImages[ img_idx ],
                                     tImages[ img_idx ] );
                finalizeAllOffImages( tImages_all );
            }
        }   // Endof if ( Nimages_moved != 0 )
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: checkToScrollView()'s END: " );
    }

    protected int time2pixel( double time_coord )
    {
        return (int) Math.round( ( time_coord - tImages_all.getEarliestTime() )
                                 * model.getViewPixelsPerUnitTime() );
    }

    protected double pixel2time( int pixel_coord )
    {
        return (double) pixel_coord / model.getViewPixelsPerUnitTime()
                      + tImages_all.getEarliestTime();
    }

    // scrollable_image interface. This returns pixel coordinate in the image
    // buffer measured from the far left of the buffer.
    public int getXaxisViewPosition()
    {
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: getViewPosition() : "
                         + "model.getTimeViewPosition()="
                         + model.getTimeViewPosition() );
        return time2pixel( model.getTimeViewPosition() );
    }

    /*
        images_endtimes:  endtimes of the collective OffSreenImageS, offImage[]
     */
    protected abstract void
    initializeAllOffImages( final TimeBoundingBox  images_endtimes );

    /*
        image_endtimes:  endtimes of the OffScreenImage, image
     */
    protected abstract void
    drawOneOffImage( Image image, final TimeBoundingBox  image_endtimes );

    /*
        images_endtimes:  endtimes of the collective OffSreenImageS, offImage[]
     */
    protected abstract void
    finalizeAllOffImages( final TimeBoundingBox  images_endtimes );

    public void paintComponent( Graphics g )
    {
        if ( Debug.isActive() ) {
            Debug.println( "ScrollableObject : paintComponent()'s START : " );
            Debug.println( "ScrollableObject : paintComponent() "
                         + "g.getClipBounds() = " + g.getClipBounds() );
            Debug.println( "ScrollableObject : paintComponent() "
                         + "this = " + this );
            if ( src_vport != null )
                Debug.println( "ScrollableObject: paintComponent() "
                             + "src_vport.getViewPosition() = "
                             + src_vport.getViewPosition() );
        }
        /*
        //  These statements are moved to componentResized()
        // if ( offscreenImage == null ) {
        //     offscreenImage = this.createImage( null );
        //     drawOneOffImage( offscreenImage );
        // }
        */
        int img_width, vport_mv_dir;
        int img_idx, screen_img_pos;
        int side_idx, side_bit, side_offset;

        // vport_mv_dir = viewport's moving direction
        vport_mv_dir = model.getViewportMovingDir();
        if ( vport_mv_dir < 0 )
            vport_mv_dir = -1;
        else
            vport_mv_dir = 1;

        img_width = image_size.width;

        // draw Image in the middle of offscreenImages[]
        img_idx = cur_img_idx;
        screen_img_pos = half_NumImages * img_width;
        if ( offscreenImages[ img_idx ] != null )
            g.drawImage( offscreenImages[ img_idx ], screen_img_pos, 0, this );

        // Images are drawn alternatively around the middle of
        // offscreenImages[].  The drawing starts with image in the
        // same direction as viewport moving direction(i.e. opposite
        // to image moving direction), then jumps to the image on the
        // other side of the middle image in the image buffer.
        //
        // The order of drawing aims to optimize the refresh rate of
        // image in the buffer, so that user does NOT notice the image
        // is being redrawn.  The code is written to anticipate what
        // user wants to see next in the image buffer.  The assumption
        // is that when viewport moves, it will be more likely to keep
        // moving in the same direction.
        for ( side_idx = 1; side_idx <= half_NumImages; side_idx++ ) {
            for ( side_bit = 1; side_bit >= -1; side_bit -= 2 ) {
                // viewport_move_direction = -1 * image_move_direction
                side_offset = side_bit * vport_mv_dir * side_idx;
                img_idx = getValidImageIndex( cur_img_idx + side_offset );
                if ( Debug.isActive() )
                    Debug.println( "ScrollableObject: paintComponent() "
                                 + "side_offset = " + side_offset + ",  "
                                 + "img_idx = " + img_idx );
                screen_img_pos = ( half_NumImages + side_offset ) * img_width;
                if ( offscreenImages[ img_idx ] != null )
                    g.drawImage( offscreenImages[ img_idx ],
                                 screen_img_pos, 0, this );
            }
        }
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject : paintComponent()'s END : " );
    }

    public abstract int getJComponentHeight();

    /*
        componentResized() seems to be always been called when
        (or right after) the object is created.  Therefore, creatImage(),
        hence setSize(), are called inside componentResized().
    */
    // scrollable_image interface when the viewport is resized.
    public void componentResized( JViewport viewport )
    {
        /*
           The offscreenImage cannot be created inside the constructor,
           because image size cannot be determined before the object
           is created
        */
        if ( Debug.isActive() ) {
            Debug.println( "ScrollableObject: componentResized()'s START: " );
            Debug.println( "incoming viewport = " + viewport );
        }

        src_vport = viewport;
        if ( src_vport != null ) {
            visible_size = src_vport.getExtentSize();
            if ( Debug.isActive() )
                Debug.println( "ScrollableObject: componentResized()'s "
                             + "src_vport.getViewPosition() = "
                             + src_vport.getViewPosition() );
        }
        else
            visible_size = getVisibleRect().getSize();

        image_size.setSize( visible_size.width * NumViewsPerImage,
                            getJComponentHeight() );

        int NumOKImages = 0;
        for ( int img_idx = 0; img_idx < NumImages; img_idx++ ) {
            offscreenImages[ img_idx ] = this.createImage( image_size );
            if ( offscreenImages[ img_idx ] != null )
                NumOKImages += 1;
        }
        /*
           Init. of Time Model: Update the model's pixel coordinates.
           of time model during startup requires existence of offscreen
           image for its to do fireTimeChanged() to ViewportTime.
           i.e. model.updatePixelCoords() does model.fireTimeChanged()
        */
        if ( NumOKImages == NumImages && visible_size.width > 0 ) {
            model.setViewPixelsPerUnitTime( visible_size.width );
            model.updatePixelCoords();
        }
        /*
            It is very IMPORTANT to setSize() to indicate the width
            of this JComponent is longer than the viewport size, so
            the JViewport.setViewPosition() will work when it scrolls
            to position wider than the viewport size, i.e. without
            cutoff.  Defining getSize() for this class does NOT seem
            to help during initialization.  setSize() is a must here
            in componentResize()
        */
        component_size.setSize( image_size.width * NumImages,
                                image_size.height );
        setSize( component_size );

        // compute the last image index in the image buffer
        int img_idx = getValidImageIndex( cur_img_idx + half_NumImages + 1 );
        initializeAllOffImages( tImages_all );
        for ( int idx = 0; idx < NumImages; idx++ ) {
            drawOneOffImage( offscreenImages[ img_idx ],
                             tImages[ img_idx ] );
            // img_idx = getNearPastImageIndex( img_idx );
            img_idx = getNearFutureImageIndex( img_idx );
        }
        finalizeAllOffImages( tImages_all );
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: componentResized()'s END: " );
    }

    /*
        This method is for HierarchyBoundsListener, i.e
        ancestorMoved() or ancestorResized() methods.
        setJComponentSize() becomes irrelvant after defining getPreferredSize()
    */
    /*
    public void setJComponentSize()
    {
        setSize( component_size );
    }
    */

    /*
        Defining getPreferredSize() seems to make HierarchyBoundsListener
        for ViewportTime unnecessary.
    */
    public Dimension getPreferredSize()
    {
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: pref_size = " + component_size );
        return component_size;
    }

    public Dimension getSize()
    {
        if ( Debug.isActive() )
            Debug.println( "ScrollableObject: size = " + component_size );
        return component_size;
    }

    // Override the Component.setCursor()
    public void setCursor( Cursor new_cursor )
    {
        /*
            Make sure that the Cursor of this Component is not set,
            so the Cursor of its parent, ViewportTime, determines
            the cursor of the component.
        */
        // super.setCursor( null );
    }

    public abstract InfoDialog
    getPropertyAt( final Point            view_click,
                   final TimeBoundingBox  vport_times );

    /*
    protected Point getGlobalClickPoint( final Point local_click )
    {
        Point origin       = this.getLocationOnScreen();
        return   new Point( origin.x + local_click.x,
                            origin.y + local_click.y );
    }
    */

    protected InfoDialog getTimePropertyAt( final Point  local_click )
    {
        /* System.out.println( "\nshowPropertyAt() " + local_click ); */
        CoordPixelImage coord_xform;
        coord_xform = new CoordPixelImage( this, 0,
                                           this.getTimeBoundsOfImages() );
        Window          window;
        window = SwingUtilities.windowForComponent( this );
        if ( window instanceof Frame )
            return new InfoDialogForTime( (Frame) window,
                           coord_xform.convertPixelToTime( local_click.x ) );
        else // if ( window instanceof Dialog )
            return new InfoDialogForTime( (Dialog) window,
                           coord_xform.convertPixelToTime( local_click.x ) );
    }
}
TOP

Related Classes of viewer.zoomable.ScrollableObject

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.