Skip to content

Commit 4b45b71

Browse files
authored
hplot: add Label plotter
This CL exports a new Label type that allows adding text labels on a canvas, either in user data space coordinates or in "normalized" coordinates. Fixes #790.
1 parent 262cb50 commit 4b45b71

6 files changed

+517
-0
lines changed

hplot/README.md

+65
Original file line numberDiff line numberDiff line change
@@ -1110,3 +1110,68 @@ func ExampleHStack_withLogY() {
11101110
}
11111111
}
11121112
```
1113+
1114+
## Labels
1115+
1116+
![label-example](https://github.com/go-hep/hep/raw/master/hplot/testdata/label_plot_golden.png)
1117+
1118+
[embedmd]:# (label_example_test.go go /func ExampleLabel/ /\n}/)
1119+
```go
1120+
func ExampleLabel() {
1121+
1122+
// Creating a new plot
1123+
p := hplot.New()
1124+
p.Title.Text = "Plot labels"
1125+
p.X.Min = -10
1126+
p.X.Max = +10
1127+
p.Y.Min = -10
1128+
p.Y.Max = +10
1129+
1130+
// Default labels
1131+
l1 := hplot.NewLabel(-8, 5, "(-8,5)\nDefault label")
1132+
p.Add(l1)
1133+
1134+
// Label with normalized coordinates.
1135+
l3 := hplot.NewLabel(
1136+
0.5, 0.5,
1137+
"(0.5,0.5)\nLabel with relative coords",
1138+
hplot.WithLabelNormalized(true),
1139+
)
1140+
p.Add(l3)
1141+
1142+
// Label with normalized coordinates and auto-adjustement.
1143+
l4 := hplot.NewLabel(
1144+
0.95, 0.95,
1145+
"(0.95,0.95)\nLabel at the canvas edge, with AutoAdjust",
1146+
hplot.WithLabelNormalized(true),
1147+
hplot.WithLabelAutoAdjust(true),
1148+
)
1149+
p.Add(l4)
1150+
1151+
// Label with a customed TextStyle
1152+
usrFont, err := vg.MakeFont("Courier-Bold", 12)
1153+
if err != nil {
1154+
panic(fmt.Errorf("could not create font (Courier-Bold, 12): %w", err))
1155+
}
1156+
sty := draw.TextStyle{
1157+
Color: plotutil.Color(2),
1158+
Font: usrFont,
1159+
}
1160+
l5 := hplot.NewLabel(
1161+
0.0, 0.1,
1162+
"(0.0,0.1)\nLabel with a user-defined font",
1163+
hplot.WithLabelTextStyle(sty),
1164+
hplot.WithLabelNormalized(true),
1165+
)
1166+
p.Add(l5)
1167+
1168+
p.Add(plotter.NewGlyphBoxes())
1169+
p.Add(hplot.NewGrid())
1170+
1171+
// Save the plot to a PNG file.
1172+
err = p.Save(15*vg.Centimeter, -1, "testdata/label_plot.png")
1173+
if err != nil {
1174+
log.Fatalf("error saving plot: %v\n", err)
1175+
}
1176+
}
1177+
```

hplot/label.go

+249
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,249 @@
1+
// Copyright ©2020 The go-hep Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package hplot
6+
7+
import (
8+
"fmt"
9+
"image/color"
10+
"math"
11+
12+
"gonum.org/v1/plot"
13+
"gonum.org/v1/plot/plotter"
14+
"gonum.org/v1/plot/vg/draw"
15+
)
16+
17+
// Label displays a user-defined text string on a plot.
18+
//
19+
// Fields of Label should not be modified once the Label has been
20+
// added to an hplot.Plot.
21+
type Label struct {
22+
Text string // Text of the label
23+
X, Y float64 // Position of the label
24+
TextStyle draw.TextStyle // Text style of the label
25+
26+
// Normalized indicates whether the label position
27+
// is in data coordinates or normalized with regard
28+
// to the canvas space.
29+
// When normalized, the label position is assumed
30+
// to fall in the [0, 1] interval. If true, NewLabel
31+
// panics if x or y are outside [0, 1].
32+
//
33+
// Normalized is false by default.
34+
Normalized bool
35+
36+
// AutoAdjust enables auto adjustment of the label
37+
// position, when Normalized is true and when x
38+
// and/or y are close to 1 and the label is partly
39+
// outside the canvas. If false and the label doesn't
40+
// fit in the canvas, an error is returned.
41+
//
42+
// AutoAdjust is false by default.
43+
AutoAdjust bool
44+
45+
// cache of gonum/plot.Labels
46+
plt *plotter.Labels
47+
}
48+
49+
// NewLabel creates a new txt label at position (x, y).
50+
func NewLabel(x, y float64, txt string, opts ...LabelOption) *Label {
51+
52+
style := draw.TextStyle{
53+
Color: color.Black,
54+
Font: DefaultStyle.Fonts.Tick, // FIXME(sbinet): add a field in Style?
55+
}
56+
57+
cfg := &labelConfig{
58+
TextStyle: style,
59+
}
60+
61+
for _, opt := range opts {
62+
opt(cfg)
63+
}
64+
65+
if cfg.Normalized {
66+
if !(0 <= x && x <= 1) {
67+
panic(fmt.Errorf(
68+
"hplot: normalized label x-position is outside [0,1]: %g", x,
69+
))
70+
}
71+
if !(0 <= y && y <= 1) {
72+
panic(fmt.Errorf(
73+
"hplot: normalized label y-position is outside [0,1]: %g", y,
74+
))
75+
}
76+
}
77+
78+
return &Label{
79+
Text: txt,
80+
X: x,
81+
Y: y,
82+
TextStyle: cfg.TextStyle,
83+
Normalized: cfg.Normalized,
84+
AutoAdjust: cfg.AutoAdjust,
85+
}
86+
}
87+
88+
// Plot implements the Plotter interface,
89+
// drawing the label on the canvas.
90+
func (lbl *Label) Plot(c draw.Canvas, p *plot.Plot) {
91+
lbl.labels(c, p).Plot(c, p)
92+
}
93+
94+
// DataRange returns the minimum and maximum x and
95+
// y values, implementing the plot.DataRanger interface.
96+
func (lbl *Label) DataRange() (xmin, xmax, ymin, ymax float64) {
97+
if lbl.Normalized {
98+
return math.Inf(+1), math.Inf(-1), math.Inf(+1), math.Inf(-1)
99+
}
100+
101+
return lbl.labels(draw.Canvas{}, nil).DataRange()
102+
}
103+
104+
// GlyphBoxes returns a GlyphBox, corresponding to the label.
105+
// GlyphBoxes implements the plot.GlyphBoxer interface.
106+
func (lbl *Label) GlyphBoxes(p *plot.Plot) []plot.GlyphBox {
107+
if lbl.plt == nil {
108+
return nil
109+
}
110+
// we expect Label.Plot(c,p) has already been called.
111+
return lbl.labels(draw.Canvas{}, p).GlyphBoxes(p)
112+
}
113+
114+
// Internal helper function to get plotter.Labels type.
115+
func (lbl *Label) labels(c draw.Canvas, p *plot.Plot) *plotter.Labels {
116+
if lbl.plt != nil {
117+
return lbl.plt
118+
}
119+
120+
var (
121+
x = lbl.X
122+
y = lbl.Y
123+
124+
err error
125+
)
126+
127+
if lbl.Normalized {
128+
// Check whether the label fits in the canvas
129+
box := lbl.TextStyle.Rectangle(lbl.Text)
130+
rect := c.Rectangle.Size()
131+
xmax := lbl.X + box.Max.X.Points()/rect.X.Points()
132+
ymax := lbl.Y + box.Max.Y.Points()/rect.Y.Points()
133+
if xmax > 1 || ymax > 1 {
134+
switch {
135+
case lbl.AutoAdjust:
136+
x, y = lbl.adjust(1/rect.X.Points(), 1/rect.Y.Points())
137+
default:
138+
panic(fmt.Errorf(
139+
"hplot: label (%g, %g) falls outside data canvas",
140+
x, y,
141+
))
142+
}
143+
}
144+
145+
// Turn relative into absolute coordinates
146+
x = lbl.scale(x, p.X.Min, p.X.Max, p.X.Scale)
147+
y = lbl.scale(y, p.Y.Min, p.Y.Max, p.Y.Scale)
148+
}
149+
150+
lbl.plt, err = plotter.NewLabels(plotter.XYLabels{
151+
XYs: []plotter.XY{{X: x, Y: y}},
152+
Labels: []string{lbl.Text},
153+
})
154+
if err != nil {
155+
panic(fmt.Errorf("hplot: could not create labels: %w", err))
156+
}
157+
158+
lbl.plt.TextStyle = []draw.TextStyle{lbl.TextStyle}
159+
160+
return lbl.plt
161+
}
162+
163+
func (lbl *Label) adjust(xnorm, ynorm float64) (x, y float64) {
164+
x = lbl.adjustX(xnorm)
165+
y = lbl.adjustY(ynorm)
166+
return x, y
167+
}
168+
169+
func (lbl *Label) adjustX(xnorm float64) float64 {
170+
var (
171+
box = lbl.TextStyle.Rectangle(lbl.Text)
172+
size = box.Size().X.Points() * xnorm
173+
x = lbl.X
174+
dx = size - (1 - x)
175+
)
176+
if x+size > 1 {
177+
x -= dx
178+
}
179+
if x < 0 {
180+
x = 0
181+
}
182+
return x
183+
}
184+
185+
func (lbl *Label) adjustY(ynorm float64) float64 {
186+
var (
187+
box = lbl.TextStyle.Rectangle(lbl.Text)
188+
size = box.Size().Y.Points() * ynorm
189+
y = lbl.Y
190+
dy = size - (1 - y)
191+
)
192+
if y+size > 1 {
193+
y -= dy
194+
}
195+
if y < 0 {
196+
y = 0
197+
}
198+
return y
199+
}
200+
201+
func (Label) scale(v, min, max float64, scaler plot.Normalizer) float64 {
202+
mid := min + 0.5*(max-min)
203+
if math.Abs(scaler.Normalize(min, max, mid)-0.5) < 1e-12 {
204+
return min + v*(max-min)
205+
}
206+
207+
// log-scale
208+
min = math.Log(min)
209+
max = math.Log(max)
210+
return math.Exp(min + v*(max-min))
211+
}
212+
213+
type labelConfig struct {
214+
TextStyle draw.TextStyle
215+
Normalized bool
216+
AutoAdjust bool
217+
}
218+
219+
// LabelOption handles various options to configure a Label.
220+
type LabelOption func(cfg *labelConfig)
221+
222+
// WithLabelTextStyle specifies the text style of the label.
223+
func WithLabelTextStyle(style draw.TextStyle) LabelOption {
224+
return func(cfg *labelConfig) {
225+
cfg.TextStyle = style
226+
}
227+
}
228+
229+
// WithLabelNormalized specifies whether the coordinates are
230+
// normalized to the canvas size.
231+
func WithLabelNormalized(norm bool) LabelOption {
232+
return func(cfg *labelConfig) {
233+
cfg.Normalized = norm
234+
}
235+
}
236+
237+
// WithLabelAutoAdjust specifies whether the coordinates are
238+
// automatically adjusted to the canvas size.
239+
func WithLabelAutoAdjust(auto bool) LabelOption {
240+
return func(cfg *labelConfig) {
241+
cfg.AutoAdjust = auto
242+
}
243+
}
244+
245+
var (
246+
_ plot.Plotter = (*Label)(nil)
247+
_ plot.DataRanger = (*Label)(nil)
248+
_ plot.GlyphBoxer = (*Label)(nil)
249+
)

hplot/label_example_test.go

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
// Copyright ©2020 The go-hep Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package hplot_test
6+
7+
import (
8+
"fmt"
9+
"log"
10+
11+
"gonum.org/v1/plot/plotter"
12+
"gonum.org/v1/plot/plotutil"
13+
"gonum.org/v1/plot/vg"
14+
"gonum.org/v1/plot/vg/draw"
15+
16+
"go-hep.org/x/hep/hplot"
17+
)
18+
19+
func ExampleLabel() {
20+
21+
// Creating a new plot
22+
p := hplot.New()
23+
p.Title.Text = "Plot labels"
24+
p.X.Min = -10
25+
p.X.Max = +10
26+
p.Y.Min = -10
27+
p.Y.Max = +10
28+
29+
// Default labels
30+
l1 := hplot.NewLabel(-8, 5, "(-8,5)\nDefault label")
31+
p.Add(l1)
32+
33+
// Label with normalized coordinates.
34+
l3 := hplot.NewLabel(
35+
0.5, 0.5,
36+
"(0.5,0.5)\nLabel with relative coords",
37+
hplot.WithLabelNormalized(true),
38+
)
39+
p.Add(l3)
40+
41+
// Label with normalized coordinates and auto-adjustement.
42+
l4 := hplot.NewLabel(
43+
0.95, 0.95,
44+
"(0.95,0.95)\nLabel at the canvas edge, with AutoAdjust",
45+
hplot.WithLabelNormalized(true),
46+
hplot.WithLabelAutoAdjust(true),
47+
)
48+
p.Add(l4)
49+
50+
// Label with a customed TextStyle
51+
usrFont, err := vg.MakeFont("Courier-Bold", 12)
52+
if err != nil {
53+
panic(fmt.Errorf("could not create font (Courier-Bold, 12): %w", err))
54+
}
55+
sty := draw.TextStyle{
56+
Color: plotutil.Color(2),
57+
Font: usrFont,
58+
}
59+
l5 := hplot.NewLabel(
60+
0.0, 0.1,
61+
"(0.0,0.1)\nLabel with a user-defined font",
62+
hplot.WithLabelTextStyle(sty),
63+
hplot.WithLabelNormalized(true),
64+
)
65+
p.Add(l5)
66+
67+
p.Add(plotter.NewGlyphBoxes())
68+
p.Add(hplot.NewGrid())
69+
70+
// Save the plot to a PNG file.
71+
err = p.Save(15*vg.Centimeter, -1, "testdata/label_plot.png")
72+
if err != nil {
73+
log.Fatalf("error saving plot: %v\n", err)
74+
}
75+
}

0 commit comments

Comments
 (0)