nplot-gtk/lib/LabelAxis.cs

357 строки
10 KiB
C#

/*
NPlot - A charting library for .NET
LabelAxis.cs
Copyright (C) 2003
Matt Howlett
Redistribution and use of NPlot or parts there-of in source and
binary forms, with or without modification, are permitted provided
that the following conditions are met:
1. Re-distributions in source form must retain at the head of each
source file the above copyright notice, this list of conditions
and the following disclaimer.
2. Any product ("the product") that makes use NPlot or parts
there-of must either:
(a) allow any user of the product to obtain a complete machine-
readable copy of the corresponding source code for the
product and the version of NPlot used for a charge no more
than your cost of physically performing source distribution,
on a medium customarily used for software interchange, or:
(b) reproduce the following text in the documentation, about
box or other materials intended to be read by human users
of the product that is provided to every human user of the
product:
"This product includes software developed as
part of the NPlot library project available
from: http://www.nplot.com/"
The words "This product" may optionally be replace with
the actual name of the product.
------------------------------------------------------------------------
THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
*/
using System;
using System.Collections;
using System.Drawing;
namespace NPlot
{
/// <summary>
/// Allows the creation of axes with any number of user defined labels at
/// user defined world values along the axis.
/// </summary>
public class LabelAxis : Axis
{
/// <summary>
/// Deep copy of LabelAxis.
/// </summary>
/// <returns>A copy of the LinearAxis Class.</returns>
public override object Clone()
{
LabelAxis a = new LabelAxis();
// ensure that this isn't being called on a derived type. If it is, then oh no!
if (this.GetType() != a.GetType())
{
throw new NPlotException( "Error. Clone method is not defined in derived type." );
}
DoClone( this, a );
return a;
}
/// <summary>
/// Helper method for Clone.
/// </summary>
/// <param name="a">The original object to clone.</param>
/// <param name="b">The cloned object.</param>
protected static void DoClone( LabelAxis b, LabelAxis a )
{
Axis.DoClone( b, a );
a.labels_ = (ArrayList)b.labels_.Clone();
a.numbers_ = (ArrayList)b.numbers_.Clone();
a.ticksBetweenText_ = b.ticksBetweenText_;
a.sortDataIfNecessary_ = b.sortDataIfNecessary_;
}
/// <summary>
/// Initialise LabelAxis to default state.
/// </summary>
private void Init()
{
labels_ = new ArrayList();
numbers_ = new ArrayList();
}
/// <summary>
/// Copy constructor
/// </summary>
/// <param name="a">The Axis to clone.</param>
/// <remarks>TODO: [review notes] I don't think this will work as desired.</remarks>
public LabelAxis( Axis a )
: base( a )
{
Init();
}
/// <summary>
/// Default constructor
/// </summary>
public LabelAxis()
: base()
{
Init();
}
/// <summary>
/// Constructor
/// </summary>
/// <param name="worldMin">Minimum world value</param>
/// <param name="worldMax">Maximum world value</param>
public LabelAxis( double worldMin, double worldMax )
: base( worldMin, worldMax )
{
Init();
}
/// <summary>
/// Adds a label to the axis
/// </summary>
/// <param name="name">The label</param>
/// <param name="val">The world value at which to place the label</param>
public void AddLabel( string name, double val )
{
labels_.Add( name );
numbers_.Add( val );
}
/// <summary>
/// Given Graphics surface, and physical extents of axis, draw ticks and
/// associated labels.
/// </summary>
/// <param name="g">The GDI+ Graphics surface on which to draw.</param>
/// <param name="physicalMin">The physical location of the world min point</param>
/// <param name="physicalMax">The physical location of the world max point</param>
/// <param name="boundingBox">out: smallest box that completely encompasses all of the ticks and tick labels.</param>
/// <param name="labelOffset">out: a suitable offset from the axis to draw the axis label.</param>
protected override void DrawTicks(
Graphics g,
Point physicalMin,
Point physicalMax,
out object labelOffset,
out object boundingBox )
{
Point tLabelOffset;
Rectangle tBoundingBox;
labelOffset = this.getDefaultLabelOffset( physicalMin, physicalMax );
boundingBox = null;
// draw the tick labels (but not the ticks).
PointF lastPos = WorldToPhysical( (double)numbers_[0], physicalMin, physicalMax, true );
for (int i=0; i<labels_.Count; ++i)
{
if ((double)numbers_[i] > WorldMin && (double)numbers_[i] < WorldMax)
{
// check to make sure labels are far enough appart.
PointF thisPos = WorldToPhysical( (double)numbers_[i], physicalMin, physicalMax, true );
float dist = Utils.Distance( thisPos, lastPos );
if ( i==0 || (dist > this.PhysicalSpacingMin) )
{
lastPos = thisPos;
this.DrawTick( g, (double)numbers_[i], 0,
(string)labels_[i],
new Point(0,0),
physicalMin, physicalMax,
out tLabelOffset, out tBoundingBox );
Axis.UpdateOffsetAndBounds(
ref labelOffset, ref boundingBox,
tLabelOffset, tBoundingBox );
}
}
}
// now draw the ticks (which might not be aligned with the tick text).
ArrayList largeTickPositions;
ArrayList smallTickPositions;
WorldTickPositions_FirstPass( physicalMin, physicalMax, out largeTickPositions, out smallTickPositions );
lastPos = WorldToPhysical( (double)largeTickPositions[0], physicalMin, physicalMax, true );
for (int i=0; i<largeTickPositions.Count; ++i)
{
double tickPos = (double)largeTickPositions[i];
// check to see that labels are far enough appart.
PointF thisPos = WorldToPhysical( tickPos, physicalMin, physicalMax, true );
float dist = Utils.Distance( thisPos, lastPos );
if ( (i==0) || (dist> this.PhysicalSpacingMin) )
{
lastPos = thisPos;
this.DrawTick( g, tickPos, LargeTickSize,
"",
new Point(0,0),
physicalMin, physicalMax,
out tLabelOffset, out tBoundingBox );
Axis.UpdateOffsetAndBounds(
ref labelOffset, ref boundingBox,
tLabelOffset, tBoundingBox );
}
}
}
/// <summary>
/// Determines the positions, in world coordinates, of the large ticks.
///
/// Label axes do not have small ticks.
/// </summary>
/// <param name="physicalMin">The physical position corresponding to the world minimum of the axis.</param>
/// <param name="physicalMax">The physical position corresponding to the world maximum of the axis.</param>
/// <param name="largeTickPositions">ArrayList containing the positions of the large ticks.</param>
/// <param name="smallTickPositions">null</param>
internal override void WorldTickPositions_FirstPass(
Point physicalMin,
Point physicalMax,
out ArrayList largeTickPositions,
out ArrayList smallTickPositions
)
{
smallTickPositions = null;
largeTickPositions = new ArrayList();
// if ticks correspond to position of text
if (!ticksBetweenText_)
{
for (int i=0; i<labels_.Count; ++i)
{
if ((double)numbers_[i] > WorldMin && (double)numbers_[i] < WorldMax)
{
largeTickPositions.Add( numbers_[i] );
}
}
}
// if ticks correspond to gaps between text
else
{
ArrayList numbers_copy;
if (sortDataIfNecessary_)
{
numbers_copy = (ArrayList)numbers_.Clone(); // shallow copy.
numbers_copy.Sort();
}
else
{
numbers_copy = numbers_;
}
for (int i=1; i<labels_.Count; ++i)
{
double worldPosition = ((double)numbers_copy[i] + (double)numbers_copy[i-1])/2.0;
if (worldPosition > WorldMin && worldPosition < WorldMax)
{
largeTickPositions.Add( worldPosition );
}
}
}
}
/// <summary>
/// If true, large ticks are drawn between the labels, rather
/// than at the position of the labels.
/// </summary>
public bool TicksBetweenText
{
get
{
return ticksBetweenText_;
}
set
{
ticksBetweenText_ = value;
}
}
private bool ticksBetweenText_ = false;
/// <summary>
/// If your data may be be specified out of order (that is
/// abscissa values with a higher index may be less than
/// abscissa values of a lower index), then data sorting
/// may be necessary to implement some of the functionality
/// of this object. If you know your data is already
/// ordered with abscissa values lowest -> highest, then
/// you may set this to false. It's default is true.
/// </summary>
public bool SortDataIfNecessary
{
get
{
return sortDataIfNecessary_;
}
set
{
sortDataIfNecessary_ = value;
}
}
private bool sortDataIfNecessary_ = true;
/// <summary>
/// If consecutive labels are less than this number of pixels appart,
/// some of the labels will not be drawn.
/// </summary>
public int PhysicalSpacingMin
{
get
{
return physicalSpacingMin_;
}
set
{
physicalSpacingMin_ = value;
}
}
private int physicalSpacingMin_ = 0;
private ArrayList labels_;
private ArrayList numbers_;
}
}