1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268
|
// Copyright ©2015 The Gonum Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Package plotter defines a variety of standard Plotters for the
// plot package.
//
// Plotters use the primitives provided by the plot package to draw to
// the data area of a plot. This package provides some standard data
// styles such as lines, scatter plots, box plots, labels, and more.
//
// New* functions return an error if the data contains Inf, NaN, or is
// empty. Some of the New* functions return other plotter-specific errors
// too.
package plotter // import "gonum.org/v1/plot/plotter"
import (
"errors"
"image/color"
"math"
"gonum.org/v1/plot/vg"
"gonum.org/v1/plot/vg/draw"
)
var (
// DefaultLineStyle is the default style for drawing
// lines.
DefaultLineStyle = draw.LineStyle{
Color: color.Black,
Width: vg.Points(1),
Dashes: []vg.Length{},
DashOffs: 0,
}
// DefaultGlyphStyle is the default style used
// for gyph marks.
DefaultGlyphStyle = draw.GlyphStyle{
Color: color.Black,
Radius: vg.Points(2.5),
Shape: draw.RingGlyph{},
}
)
// Valuer wraps the Len and Value methods.
type Valuer interface {
// Len returns the number of values.
Len() int
// Value returns a value.
Value(int) float64
}
// Range returns the minimum and maximum values.
func Range(vs Valuer) (min, max float64) {
min = math.Inf(1)
max = math.Inf(-1)
for i := 0; i < vs.Len(); i++ {
v := vs.Value(i)
min = math.Min(min, v)
max = math.Max(max, v)
}
return
}
// Values implements the Valuer interface.
type Values []float64
var (
ErrInfinity = errors.New("Infinite data point")
ErrNaN = errors.New("NaN data point")
ErrNoData = errors.New("No data points")
)
// CheckFloats returns an error if any of the arguments are NaN or Infinity.
func CheckFloats(fs ...float64) error {
for _, f := range fs {
switch {
case math.IsNaN(f):
return ErrNaN
case math.IsInf(f, 0):
return ErrInfinity
}
}
return nil
}
// CopyValues returns a Values that is a copy of the values
// from a Valuer, or an error if there are no values, or if one of
// the copied values is a NaN or Infinity.
func CopyValues(vs Valuer) (Values, error) {
if vs.Len() == 0 {
return nil, ErrNoData
}
cpy := make(Values, vs.Len())
for i := 0; i < vs.Len(); i++ {
cpy[i] = vs.Value(i)
if err := CheckFloats(cpy[i]); err != nil {
return nil, err
}
}
return cpy, nil
}
func (vs Values) Len() int {
return len(vs)
}
func (vs Values) Value(i int) float64 {
return vs[i]
}
// XYer wraps the Len and XY methods.
type XYer interface {
// Len returns the number of x, y pairs.
Len() int
// XY returns an x, y pair.
XY(int) (x, y float64)
}
// XYRange returns the minimum and maximum
// x and y values.
func XYRange(xys XYer) (xmin, xmax, ymin, ymax float64) {
xmin, xmax = Range(XValues{xys})
ymin, ymax = Range(YValues{xys})
return
}
// XYs implements the XYer interface.
type XYs []XY
// XY is an x and y value.
type XY struct{ X, Y float64 }
// CopyXYs returns an XYs that is a copy of the x and y values from
// an XYer, or an error if one of the data points contains a NaN or
// Infinity.
func CopyXYs(data XYer) (XYs, error) {
cpy := make(XYs, data.Len())
for i := range cpy {
cpy[i].X, cpy[i].Y = data.XY(i)
if err := CheckFloats(cpy[i].X, cpy[i].Y); err != nil {
return nil, err
}
}
return cpy, nil
}
func (xys XYs) Len() int {
return len(xys)
}
func (xys XYs) XY(i int) (float64, float64) {
return xys[i].X, xys[i].Y
}
// XValues implements the Valuer interface,
// returning the x value from an XYer.
type XValues struct {
XYer
}
func (xs XValues) Value(i int) float64 {
x, _ := xs.XY(i)
return x
}
// YValues implements the Valuer interface,
// returning the y value from an XYer.
type YValues struct {
XYer
}
func (ys YValues) Value(i int) float64 {
_, y := ys.XY(i)
return y
}
// XYZer wraps the Len and XYZ methods.
type XYZer interface {
// Len returns the number of x, y, z triples.
Len() int
// XYZ returns an x, y, z triple.
XYZ(int) (float64, float64, float64)
// XY returns an x, y pair.
XY(int) (float64, float64)
}
// XYZs implements the XYZer interface using a slice.
type XYZs []XYZ
// XYZ is an x, y and z value.
type XYZ struct{ X, Y, Z float64 }
// Len implements the Len method of the XYZer interface.
func (xyz XYZs) Len() int {
return len(xyz)
}
// XYZ implements the XYZ method of the XYZer interface.
func (xyz XYZs) XYZ(i int) (float64, float64, float64) {
return xyz[i].X, xyz[i].Y, xyz[i].Z
}
// XY implements the XY method of the XYer interface.
func (xyz XYZs) XY(i int) (float64, float64) {
return xyz[i].X, xyz[i].Y
}
// CopyXYZs copies an XYZer.
func CopyXYZs(data XYZer) (XYZs, error) {
cpy := make(XYZs, data.Len())
for i := range cpy {
cpy[i].X, cpy[i].Y, cpy[i].Z = data.XYZ(i)
if err := CheckFloats(cpy[i].X, cpy[i].Y, cpy[i].Z); err != nil {
return nil, err
}
}
return cpy, nil
}
// XYValues implements the XYer interface, returning
// the x and y values from an XYZer.
type XYValues struct{ XYZer }
// XY implements the XY method of the XYer interface.
func (xy XYValues) XY(i int) (float64, float64) {
x, y, _ := xy.XYZ(i)
return x, y
}
// Labeller wraps the Label methods.
type Labeller interface {
// Label returns a label.
Label(int) string
}
// XErrorer wraps the XError method.
type XErrorer interface {
// XError returns two error values for X data.
XError(int) (float64, float64)
}
// Errors is a slice of low and high error values.
type Errors []struct{ Low, High float64 }
// XErrors implements the XErrorer interface.
type XErrors Errors
func (xe XErrors) XError(i int) (float64, float64) {
return xe[i].Low, xe[i].High
}
// YErrorer wraps the YError method.
type YErrorer interface {
// YError returns two error values for Y data.
YError(int) (float64, float64)
}
// YErrors implements the YErrorer interface.
type YErrors Errors
func (ye YErrors) YError(i int) (float64, float64) {
return ye[i].Low, ye[i].High
}
|