Skip to content

Commit 4a5cc2d

Browse files
jlskuzbep
authored andcommitted
images.Text: Add "alignx" option for horizontal alignment
Add an "alignx" option to the images.Text to control whether the value of the "x" option is the left border of the text (current behaviour), the center of each line or the right border. Fixes #10849
1 parent 9cad8d3 commit 4a5cc2d

File tree

3 files changed

+57
-13
lines changed

3 files changed

+57
-13
lines changed

‎docs/content/en/functions/images/Text.md

+4
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,10 @@ x
3737
y
3838
: (`int`) The vertical offset, in pixels, relative to the top of the image. Default is `10`.
3939

40+
alignx
41+
{{< new-in 0.141.0 >}}
42+
: (`string`) The horizontal alignment of the text relative to the `x` position. One of `left`, `center`, or `right`. Default is `left`.
43+
4044
[global resource]: /getting-started/glossary/#global-resource
4145
[page resource]: /getting-started/glossary/#page-resource
4246
[remote resource]: /getting-started/glossary/#remote-resource

‎resources/images/filters.go

+7
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,7 @@ func (*Filters) Text(text string, options ...any) gift.Filter {
6969
size: 20,
7070
x: 10,
7171
y: 10,
72+
alignx: "left",
7273
linespacing: 2,
7374
}
7475

@@ -87,6 +88,12 @@ func (*Filters) Text(text string, options ...any) gift.Filter {
8788
tf.x = cast.ToInt(v)
8889
case "y":
8990
tf.y = cast.ToInt(v)
91+
case "alignx":
92+
tf.alignx = cast.ToString(v)
93+
if tf.alignx != "left" && tf.alignx != "center" && tf.alignx != "right" {
94+
panic("alignx must be one of left, center, right")
95+
}
96+
9097
case "linespacing":
9198
tf.linespacing = cast.ToInt(v)
9299
case "font":

‎resources/images/text.go

+46-13
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@ type textFilter struct {
3535
text string
3636
color color.Color
3737
x, y int
38+
alignx string
3839
size float64
3940
linespacing int
4041
fontSource hugio.ReadSeekCloserProvider
@@ -77,30 +78,62 @@ func (f textFilter) Draw(dst draw.Image, src image.Image, options *gift.Options)
7778

7879
gift.New().Draw(dst, src)
7980

80-
// Draw text, consider and include linebreaks
8181
maxWidth := dst.Bounds().Dx() - 20
82+
83+
var availableWidth int
84+
switch f.alignx {
85+
case "right":
86+
availableWidth = f.x
87+
case "center":
88+
availableWidth = min((maxWidth-f.x), f.x) * 2
89+
case "left":
90+
availableWidth = maxWidth - f.x
91+
}
92+
8293
fontHeight := face.Metrics().Ascent.Ceil()
8394

95+
// Calculate lines, consider and include linebreaks
96+
finalLines := []string{}
97+
f.text = strings.ReplaceAll(f.text, "\r", "")
98+
for _, line := range strings.Split(f.text, "\n") {
99+
currentLine := ""
100+
// Break each line at the maximum width.
101+
for _, str := range strings.Fields(line) {
102+
fieldStrWidth := font.MeasureString(face, str)
103+
currentLineStrWidth := font.MeasureString(face, currentLine)
104+
105+
if (currentLineStrWidth.Ceil() + fieldStrWidth.Ceil()) >= availableWidth {
106+
finalLines = append(finalLines, currentLine)
107+
currentLine = ""
108+
}
109+
currentLine += str + " "
110+
}
111+
finalLines = append(finalLines, currentLine)
112+
}
113+
84114
// Correct y position based on font and size
85115
f.y = f.y + fontHeight
86116

87117
// Start position
88118
y := f.y
89-
d.Dot = fixed.P(f.x, f.y)
90119

91-
// Draw text line by line, breaking each line at the maximum width.
92-
f.text = strings.ReplaceAll(f.text, "\r", "")
93-
for _, line := range strings.Split(f.text, "\n") {
94-
for _, str := range strings.Fields(line) {
95-
strWidth := font.MeasureString(face, str)
96-
if (d.Dot.X.Ceil() + strWidth.Ceil()) >= maxWidth {
97-
y = y + fontHeight + f.linespacing
98-
d.Dot = fixed.P(f.x, y)
99-
}
100-
d.DrawString(str + " ")
120+
// Draw text line by line
121+
for _, line := range finalLines {
122+
line = strings.TrimSpace(line)
123+
strWidth := font.MeasureString(face, line)
124+
var x int
125+
switch f.alignx {
126+
case "right":
127+
x = f.x - strWidth.Ceil()
128+
case "center":
129+
x = f.x - (strWidth.Ceil() / 2)
130+
131+
case "left":
132+
x = f.x
101133
}
134+
d.Dot = fixed.P(x, y)
135+
d.DrawString(line)
102136
y = y + fontHeight + f.linespacing
103-
d.Dot = fixed.P(f.x, y)
104137
}
105138
}
106139

0 commit comments

Comments
 (0)