Package base.topology

Source Code of base.topology.PreviewState

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

/*
*  @author  Anthony Chan
*/

package base.topology;

import java.awt.Graphics2D;
import java.awt.Insets;
import java.awt.Color;
import java.awt.Point;
import java.util.Arrays;

import base.drawable.CoordPixelXform;
import base.drawable.CategoryWeight;
import base.drawable.DrawnBox;
import base.drawable.Category;
import base.drawable.Shadow;

public class PreviewState
{
    // private static StateBorder BorderStyle = StateBorder.WHITE_RAISED_BORDER;
    private static StateBorder BorderStyle = StateBorder.COLOR_XOR_BORDER;

    public static void setBorderStyle( final StateBorder state_border )
    {
        BorderStyle = state_border;
    }

    // The constant String's should be the same as those in SummaryState
    public  static final String FIT_MOST_LEGENDS
                                = "FitMostLegends";
    private static final int    FIT_MOST_LEGENDS_ID          = 0;
    public  static final String OVERLAP_INCLUSION
                                = "OverlapInclusionRatio";
    private static final int    OVERLAP_INCLUSION_ID         = 1;
    public  static final String CUMULATIVE_INCLUSION
                                = "CumulativeInclusionRatio";
    private static final int    CUMULATIVE_INCLUSION_ID      = 2;
    public  static final String OVERLAP_EXCLUSION
                                = "OverlapExclusionRatio";
    private static final int    OVERLAP_EXCLUSION_ID         = 3;
    public  static final String CUMULATIVE_EXCLUSION
                                = "CumulativeExclusionRatio";
    private static final int    CUMULATIVE_EXCLUSION_ID      = 4;
    public  static final String CUMULATIVE_EXCLUSION_BASE
                                = "BaseAlignedCumulativeExclusionRatio";
    private static final int    CUMULATIVE_EXCLUSION_BASE_ID = 5;

    private static       int    DisplayType             = OVERLAP_INCLUSION_ID;

    public static void setDisplayType( String new_display_type )
    {
        if ( new_display_type.equals( FIT_MOST_LEGENDS ) )
            DisplayType = FIT_MOST_LEGENDS_ID;
        else if ( new_display_type.equals( OVERLAP_INCLUSION ) )
            DisplayType = OVERLAP_INCLUSION_ID;
        else if ( new_display_type.equals( CUMULATIVE_INCLUSION ) )
            DisplayType = CUMULATIVE_INCLUSION_ID;
        else if ( new_display_type.equals( OVERLAP_EXCLUSION ) )
            DisplayType = OVERLAP_EXCLUSION_ID;
        else if ( new_display_type.equals( CUMULATIVE_EXCLUSION ) )
            DisplayType = CUMULATIVE_EXCLUSION_ID;
        else if ( new_display_type.equals( CUMULATIVE_EXCLUSION_BASE ) )
            DisplayType = CUMULATIVE_EXCLUSION_BASE_ID;
        else
            DisplayType = OVERLAP_INCLUSION_ID;
    }

    private static        int    MinCategoryHeight          = 2
    private static        int    MinCategorySeparation      = 4

    public static void setMinCategoryHeight( int new_min_category_height )
    {
        MinCategoryHeight  = new_min_category_height;
    }

    /*
        Draw a Rectangle between left-upper vertex (start_time, start_ypos)
        and right-lower vertex (final_time, final_ypos)
        Assume caller guarantees the order of timestamps and ypos, such that
        start_time <= final_time  and  start_ypos <= final_ypos.
    */
    private static int  drawForward( Graphics2D g, Color color,
                                     Shadow shade, Insets insets,
                                     CoordPixelXform    coord_xform,
                                     DrawnBox           last_drawn_pos,
                                     double start_time, float start_ypos,
                                     double final_time, float final_ypos )
    {
        int      iStart, jStart, iFinal, jFinal;
        iStart   = coord_xform.convertTimeToPixel( start_time );
        iFinal   = coord_xform.convertTimeToPixel( final_time );

        /* Determine if State should be drawn */
        if ( last_drawn_pos.coversState( iStart, iFinal ) )
            return 0; // too small to be drawn in previously drawn location
        last_drawn_pos.set( iStart, iFinal );

        jStart   = coord_xform.convertRowToPixel( start_ypos );
        jFinal   = coord_xform.convertRowToPixel( final_ypos );

        if ( insets != null ) {
            iStart += insets.left;
            iFinal -= insets.right;
            jStart += insets.top;
            jFinal -= insets.bottom;
        }

        boolean  isStartVtxInImg, isFinalVtxInImg;
        int      iImageWidth = coord_xform.getImageWidth();
        isStartVtxInImg = ( iStart >= 0 ) ;
        isFinalVtxInImg = ( iFinal <  iImageWidth );

        int iHead, iTail, jHead, jTail;
        // jHead = slope * ( iHead - iStart ) + jStart
        if ( isStartVtxInImg )
            iHead = iStart;
        else
            iHead = 0;
            // iHead = -1;
        jHead    = jStart;

        // jTail = slope * ( iTail - iFinal ) + jFinal
        if ( isFinalVtxInImg )
            iTail = iFinal;
        else
            iTail = iImageWidth - 1;
            // iTail = iImageWidth;
        jTail    = jFinal;
           
        int iRange  = iFinal-iStart+1// width uncut by image border
        int iWidth  = iTail-iHead+1;    // width possibly cut by the image
        int jHeight = jTail-jHead+1;

        CategoryWeight[]  twgts;
        CategoryWeight    twgt = null;
        int               idx, twgts_length;
        float             tot_wgt, height_per_wgt;
        int               iLevel, iDelta, iCenter;
        int               jLevel, jDelta, jCenter;
        int               jDeltaTotal;
        boolean           isInclusive;

        jDeltaTotal  = 0;
        twgts        = shade.arrayOfCategoryWeights();
        twgts_length = twgts.length;
        if (    DisplayType == CUMULATIVE_INCLUSION_ID
             || DisplayType == CUMULATIVE_EXCLUSION_ID
             || DisplayType == CUMULATIVE_EXCLUSION_BASE_ID ) {
            isInclusive = ( DisplayType == CUMULATIVE_INCLUSION_ID );
            if ( isInclusive ) {
                Arrays.sort( twgts, CategoryWeight.INCL_RATIO_ORDER );
                // Compute the pixel height per unit weight
                tot_wgt = 0.0f;
                for ( idx = 0; idx < twgts_length; idx++ ) {
                    twgt = twgts[ idx ];
                    if ( twgt.getCategory().isVisible() )
                        tot_wgt += twgt.getRatio( isInclusive );
                }
                height_per_wgt = (float) jHeight / tot_wgt;
            }
            else {
                Arrays.sort( twgts, CategoryWeight.EXCL_RATIO_ORDER );
                height_per_wgt = jHeight;
            }

            // set sub-rectangles' height from the bottom, ie. jHead+jTail
            jLevel = jHead + jHeight;  // jLevel = jTail + 1
            for ( idx = twgts_length-1; idx >= 0; idx-- ) {
                twgt = twgts[ idx ];
                if ( twgt.getCategory().isVisible() ) {
                    jDelta = (int) ( height_per_wgt
                                   * twgt.getRatio( isInclusive ) + 0.5f );
                    if ( jDelta > 0 ) {
                        if ( jLevel > jHead ) {
                            if ( jLevel-jDelta >= jHead ) {
                                jLevel  -= jDelta;
                                twgt.setPixelHeight( jDelta );
                            }
                            else {
                                twgt.setPixelHeight( jLevel - jHead );
                                jLevel = jHead;
                            }
                        }
                        else
                            twgt.setPixelHeight( 0 );
                    }
                    else
                        twgt.setPixelHeight( 0 );
                }
                else
                    twgt.setPixelHeight( 0 );
                jDeltaTotal += twgt.getPixelHeight();
            }
            shade.setTotalPixelHeight( jDeltaTotal )// for isPixelIn()
        }
        else if (    DisplayType == OVERLAP_INCLUSION_ID
                  || DisplayType == OVERLAP_EXCLUSION_ID ) {
            isInclusive = ( DisplayType == OVERLAP_INCLUSION_ID );
            if ( isInclusive )
                Arrays.sort( twgts, CategoryWeight.INCL_RATIO_ORDER );
            else
                Arrays.sort( twgts, CategoryWeight.EXCL_RATIO_ORDER );
            jLevel = Integer.MAX_VALUE; // JLevel should be named as JDelta_prev
            iDelta = iRange;
            for ( idx = twgts_length-1; idx >= 0; idx-- ) {
                twgt = twgts[ idx ];
                if ( twgt.getCategory().isVisible() ) {
                    jDelta = (int) ( twgt.getRatio( isInclusive ) * jHeight
                                   + 0.5f );
                    twgt.setPixelHeight( jDelta );
                    if ( jDelta >= jLevel )
                        iDelta -= MinCategorySeparation;
                    twgt.setPixelWidth( iDelta );
                    jLevel = jDelta;
                }
                else
                    twgt.setPixelHeight( 0 );
            }
        }
        else { // if ( DisplayType == FIT_MOST_LEGENDS_ID )
            Arrays.sort( twgts, CategoryWeight.INCL_RATIO_ORDER );
            int num_visible_twgts = 0;
            for ( idx = 0; idx < twgts_length; idx++ ) {
                if ( twgts[ idx ].getCategory().isVisible() )
                    num_visible_twgts++;
            }
            jDelta = (int) ( (float) jHeight / num_visible_twgts );
            if ( jDelta < MinCategoryHeight )
                jDelta = MinCategoryHeight;
            // set sub-rectangles' height from the bottom, ie. jHead+jTail
            jLevel = jHead + jHeight;  // jLevel = jTail + 1
            for ( idx = twgts_length-1; idx >= 0; idx-- ) {
                twgt = twgts[ idx ];
                if ( twgt.getCategory().isVisible() ) {
                    if ( jLevel > jHead ) {
                        if ( jLevel-jDelta >= jHead ) {
                            jLevel  -= jDelta;
                            twgt.setPixelHeight( jDelta );
                        }
                        else {
                            twgt.setPixelHeight( jLevel - jHead );
                            jLevel = jHead;
                        }
                    }
                    else
                        twgt.setPixelHeight( 0 );
                }
                else
                    twgt.setPixelHeight( 0 );
            }
        }

        // Fill the color of the sub-rectangles from the bottom, ie. jHead+jTail
        int num_sub_rects_drawn = 0;
        if (    DisplayType == OVERLAP_INCLUSION_ID
             || DisplayType == OVERLAP_EXCLUSION_ID ) {
            // iBoxXXXX : variables that twgt isn't cut by image border
            int iBoxHead, iBoxTail, iBoxWidth;
            jCenter = jHead  + jHeight / 2; // i.e. jCenter % jHead & jTail
            iCenter = iStart + iRange / 2// i.e. iCenter % iStart & iFinal
            for ( idx = twgts_length-1; idx >= 0; idx-- ) {
                twgt       = twgts[ idx ];
                jDelta     = twgt.getPixelHeight();
                iBoxWidth  = twgt.getPixelWidth();
                if ( jDelta > 0 && iBoxWidth > 0 ) {
                    iBoxHead = iCenter - iBoxWidth / 2;
                    iBoxTail = iBoxHead + iBoxWidth;
                    iLevel   = ( iBoxHead >= 0 ? iBoxHead : 0 );
                    iDelta   = ( iBoxTail < iImageWidth ?
                                 iBoxTail : iImageWidth ) - iLevel;
                    g.setColor( twgt.getCategory().getColor() );
                    g.fillRect( iLevel, jCenter-jDelta/2, iDelta, jDelta );
                    num_sub_rects_drawn++;
                }
            }
        }
        else {
            /*
            if (    DisplayType == FIT_MOST_LEGENDS_ID )
                 || DisplayType == CUMULATIVE_INCLUSION_ID
                 || DisplayType == CUMULATIVE_EXCLUSION_ID
                 || DisplayType == CUMULATIVE_EXCLUSION_BASE_ID )
            */
            jLevel = jHead + jHeight;  // jLevel = jTail + 1
            if ( DisplayType == CUMULATIVE_EXCLUSION_ID )
                jLevel -= ( jHeight - jDeltaTotal ) / 2;
            for ( idx = twgts_length-1; idx >= 0; idx-- ) {
                twgt     = twgts[ idx ];
                jDelta   = twgt.getPixelHeight();
                if ( jDelta > 0 ) {
                    jLevel  -= jDelta;
                    g.setColor( twgt.getCategory().getColor() );
                    g.fillRect( iHead, jLevel, iWidth, jDelta );
                    num_sub_rects_drawn++;
                }
            }
        }

        if ( num_sub_rects_drawn > 0 )
            BorderStyle.paintStateBorder( g, color,
                                          iHead, jHead, isStartVtxInImg,
                                          iTail, jTail, isFinalVtxInImg );
        return 1;
    }

    /*
        Check if a point in pixel coordinate is in a Rectangle
        specified between left-upper vertex (start_time, start_ypos)
        and right-lower vertex (final_time, final_ypos)
        Assume caller guarantees the order of timestamps and ypos, such that
        start_time <= final_time  and  start_ypos <= final_ypos
    */
    private static Category isPixelIn( Shadow shade, Insets insets,
                                       CoordPixelXform coord_xform, Point pt,
                                       double start_time, float start_ypos,
                                       double final_time, float final_ypos )
    {
        int      iStart, jStart, iFinal, jFinal;
        int      pt_x, pt_y;

        pt_y     = pt.y;

        jStart   = coord_xform.convertRowToPixel( start_ypos );
        if ( pt_y < jStart  )
            return null;

        jFinal   = coord_xform.convertRowToPixel( final_ypos );
        if ( pt_y > jFinal )
            return null;

        pt_x     = pt.x;

        iStart   = coord_xform.convertTimeToPixel( start_time );
        if ( pt_x < iStart )
            return null;

        iFinal   = coord_xform.convertTimeToPixel( final_time );
        if ( pt_x > iFinal )
            return null;

        //  If the code gets to here, it means the pixel is within the Shadow.
        if ( insets != null ) {
            iStart += insets.left;
            iFinal -= insets.right;
            jStart += insets.top;
            jFinal -= insets.bottom;
        }

        int jHead, jTail, jHeight;
        jHead    = jStart;
        jTail    = jFinal;
        jHeight  = jTail-jHead+1;

        CategoryWeight[]  twgts;
        CategoryWeight    twgt = null;
        int               idx, twgts_length;

        twgts        = shade.arrayOfCategoryWeights();
        twgts_length = twgts.length;

        // Locate the sub-rectangle from the bottom, ie. jHead+jTail
        int iLevel, iDelta, iCenter;
        int jLevel, jDelta, jCenter;
        if (    DisplayType == OVERLAP_INCLUSION_ID
             || DisplayType == OVERLAP_EXCLUSION_ID ) {
            int iImageWidth, iRange;
            // iBoxXXXX : variables that twgt isn't cut by image border
            int iBoxHead, iBoxTail, iBoxWidth;
            iImageWidth = coord_xform.getImageWidth();
            iRange      = iFinal - iStart + 1; // width uncut by image border
            jCenter     = jHead  + jHeight / 2;// i.e. jCenter % jHead & jTail
            iCenter     = iStart + iRange / 2; // i.e. iCenter % iStart & iFinal
            for ( idx = 0; idx < twgts_length; idx++ ) {
                twgt      = twgts[ idx ];
                jDelta    = twgt.getPixelHeight();
                iBoxWidth = twgt.getPixelWidth();
                if ( jDelta > 0 && iBoxWidth > 0 ) {
                    jLevel = jCenter - jDelta / 2;
                    if ( pt_y >= jLevel && pt_y < jLevel+jDelta ) {
                        iBoxHead = iCenter - iBoxWidth / 2;
                        iBoxTail = iBoxHead + iBoxWidth;
                        iLevel   = ( iBoxHead >= 0 ? iBoxHead : 0 );
                        iDelta   = ( iBoxTail < iImageWidth ?
                                     iBoxTail : iImageWidth ) - iLevel;
                        if ( pt_x >= iLevel && pt_x < iLevel+iDelta )
                            return twgt.getCategory();
                    }
                }
            }
        }
        else {
            /*
            if (    DisplayType == FIT_MOST_LEGENDS_ID
                 || DisplayType == CUMULATIVE_INCLUSION_ID
                 || DisplayType == CUMULATIVE_EXCLUSION_ID
                 || DisplayType == CUMULATIVE_EXCLUSION_BASE_ID )
            */
            jLevel = jHead + jHeight;  // jLevel = jTail + 1
            if ( DisplayType == CUMULATIVE_EXCLUSION_ID )
                jLevel -= ( jHeight - shade.getTotalPixelHeight() ) / 2;
            for ( idx = twgts_length-1; idx >= 0; idx-- ) {
                twgt     = twgts[ idx ];
                jDelta   = twgt.getPixelHeight();
                if ( jDelta > 0 ) {
                    jLevel  -= jDelta;
                    if ( pt_y >= jLevel && pt_y < jLevel+jDelta )
                        return twgt.getCategory();
                }
            }
        }

        return null; // mean failure, need something other than null
    }


    public static int  draw( Graphics2D g, Color color,
                             Shadow shade, Insets insets,
                             CoordPixelXform    coord_xform,
                             DrawnBox           last_drawn_pos,
                             double start_time, float start_ypos,
                             double final_time, float final_ypos )
    {
         if ( start_time < final_time ) {
             if ( start_ypos < final_ypos )
                 return drawForward( g, color, shade, insets,
                                     coord_xform, last_drawn_pos,
                                     start_time, start_ypos,
                                     final_time, final_ypos );
             else
                 return drawForward( g, color, shade, insets,
                                     coord_xform, last_drawn_pos,
                                     start_time, final_ypos,
                                     final_time, start_ypos );
         }
         else {
             if ( start_ypos < final_ypos )
                 return drawForward( g, color, shade, insets,
                                     coord_xform, last_drawn_pos,
                                     final_time, start_ypos,
                                     start_time, final_ypos );
             else
                 return drawForward( g, color, shade, insets,
                                     coord_xform, last_drawn_pos,
                                     final_time, final_ypos,
                                     start_time, start_ypos );
         }
    }

    public static Category containsPixel( Shadow shade, Insets insets,
                                          CoordPixelXform coord_xform, Point pt,
                                          double start_time, float start_ypos,
                                          double final_time, float final_ypos )
    {
         if ( start_time < final_time ) {
             if ( start_ypos < final_ypos )
                 return isPixelIn( shade, insets, coord_xform, pt,
                                   start_time, start_ypos,
                                   final_time, final_ypos );
             else
                 return isPixelIn( shade, insets, coord_xform, pt,
                                   start_time, final_ypos,
                                   final_time, start_ypos );
         }
         else {
             if ( start_ypos < final_ypos )
                 return isPixelIn( shade, insets, coord_xform, pt,
                                   final_time, start_ypos,
                                   start_time, final_ypos );
             else
                 return isPixelIn( shade, insets, coord_xform, pt,
                                   final_time, final_ypos,
                                   start_time, start_ypos );
         }
    }
}
TOP

Related Classes of base.topology.PreviewState

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.