Skip to content

Commit e479559

Browse files
LuciNyanshuding
andauthoredMay 24, 2023
feat: Multiline text overflow ellipsis (#480)
### Description 1. Support multiline text overflow ellipsis(webkit-line-clamp + webkit-box-orient). 2. Fixed a bug where overflow: 'hidden' should not be inherited by child elements. 3. Support the 'line-clamp' property, including custom block ellipsis. I added display: block here because whether it's required by the CSS draft or needed to implement this feature (must be different from display: flex), it seems I have to do so. One thing I want to discuss here is whether we can set the default display to flex, because this is more in line with the original behavior of the browser and does not seem to cause any problems (can pass all unit tests). https://github.com/vercel/satori/assets/22126563/5ec006a7-9cfb-4fc4-9000-adc9fb042c17 https://github.com/vercel/satori/assets/22126563/d672942e-e2c9-423c-8bcf-23a932e7cfb3 Closes: #253 --------- Co-authored-by: Shu Ding <g@shud.in>
1 parent 341bfab commit e479559

12 files changed

+374
-33
lines changed
 

‎src/characters.ts

+1
Original file line numberDiff line numberDiff line change
@@ -6,3 +6,4 @@ export function stringFromCode(code: string): string {
66

77
export const Space = stringFromCode('U+0020')
88
export const Tab = stringFromCode('U+0009')
9+
export const HorizontalEllipsis = stringFromCode('U+2026')

‎src/handler/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -175,7 +175,9 @@ export default async function handler(
175175
style.display,
176176
{
177177
flex: Yoga.DISPLAY_FLEX,
178+
block: Yoga.DISPLAY_FLEX,
178179
none: Yoga.DISPLAY_NONE,
180+
'-webkit-box': Yoga.DISPLAY_FLEX,
179181
},
180182
Yoga.DISPLAY_FLEX,
181183
'display'

‎src/text.ts

+150-32
Original file line numberDiff line numberDiff line change
@@ -14,15 +14,17 @@ import {
1414
isUndefined,
1515
isString,
1616
lengthToNumber,
17+
isNumber,
1718
} from './utils.js'
1819
import buildText, { container } from './builder/text.js'
1920
import { buildDropShadow } from './builder/shadow.js'
2021
import buildDecoration from './builder/text-decoration.js'
2122
import { Locale } from './language.js'
2223
import { FontEngine } from './font.js'
23-
import { Space, Tab } from './characters.js'
24+
import { HorizontalEllipsis, Space, Tab } from './characters.js'
2425

2526
const skippedWordWhenFindingMissingFont = new Set([Tab])
27+
2628
function shouldSkipWhenFindingMissingFont(word: string): boolean {
2729
return skippedWordWhenFindingMissingFont.has(word)
2830
}
@@ -49,7 +51,6 @@ export default async function* buildTextNodes(
4951

5052
const {
5153
textAlign,
52-
textOverflow,
5354
whiteSpace,
5455
wordBreak,
5556
lineHeight,
@@ -74,6 +75,11 @@ export default async function* buildTextNodes(
7475
wordBreak as string
7576
)
7677

78+
const [lineLimit, blockEllipsis] = processTextOverflow(
79+
parentStyle,
80+
allowSoftWrap
81+
)
82+
7783
const textContainer = createTextContainerNode(Yoga, textAlign as string)
7884
parent.insertChild(textContainer, parent.getChildCount())
7985

@@ -222,7 +228,7 @@ export default async function* buildTextNodes(
222228
// @TODO: Support different writing modes.
223229
// @TODO: Support RTL languages.
224230
let i = 0
225-
while (i < words.length) {
231+
while (i < words.length && lines < lineLimit) {
226232
let word = words[i]
227233
const forceBreak = requiredBreaks[i]
228234

@@ -373,10 +379,12 @@ export default async function* buildTextNodes(
373379
}
374380

375381
if (currentWidth) {
382+
if (lines < lineLimit) {
383+
height += currentLineHeight
384+
}
376385
lines++
377386
lineWidths.push(currentWidth)
378387
baselines.push(currentBaselineOffset)
379-
height += currentLineHeight
380388
}
381389

382390
// @TODO: Support `line-height`.
@@ -482,15 +490,14 @@ export default async function* buildTextNodes(
482490
let mergedPath = ''
483491
let extra = ''
484492
let skippedLine = -1
485-
let ellipsisWidth = textOverflow === 'ellipsis' ? measureGrapheme('…') : 0
486-
let spaceWidth = textOverflow === 'ellipsis' ? measureGrapheme(' ') : 0
487493
let decorationLines: Record<number, null | number[]> = {}
488494
let wordBuffer: string | null = null
489495
let bufferedOffset = 0
490496

491497
for (let i = 0; i < texts.length; i++) {
492498
// Skip whitespace and empty characters.
493499
const layout = wordPositionInLayout[i]
500+
const nextLayout = wordPositionInLayout[i + 1]
494501

495502
if (!layout) continue
496503

@@ -538,34 +545,80 @@ export default async function* buildTextNodes(
538545
]
539546
}
540547

541-
if (textOverflow === 'ellipsis') {
542-
if (lineWidths[line] > parentContainerInnerWidth) {
548+
if (lineLimit !== Infinity) {
549+
let _blockEllipsis = blockEllipsis
550+
let ellipsisWidth = measureGrapheme(blockEllipsis)
551+
if (ellipsisWidth > parentContainerInnerWidth) {
552+
_blockEllipsis = HorizontalEllipsis
553+
ellipsisWidth = measureGrapheme(_blockEllipsis)
554+
}
555+
const spaceWidth = measureGrapheme(Space)
556+
const isNotLastLine = line < lineWidths.length - 1
557+
const isLastAllowedLine = line + 1 === lineLimit
558+
559+
function calcEllipsis(baseWidth: number, _text: string) {
560+
const chars = segment(_text, 'grapheme', locale)
561+
562+
let subset = ''
563+
let resolvedWidth = 0
564+
565+
for (const char of chars) {
566+
const w = baseWidth + measureGraphemeArray([subset + char])
567+
if (
568+
// Keep at least one character:
569+
// > The first character or atomic inline-level element on a line
570+
// must be clipped rather than ellipsed.
571+
// https://drafts.csswg.org/css-overflow/#text-overflow
572+
subset &&
573+
w + ellipsisWidth > parentContainerInnerWidth
574+
) {
575+
break
576+
}
577+
subset += char
578+
resolvedWidth = w
579+
}
580+
581+
return {
582+
subset,
583+
resolvedWidth,
584+
}
585+
}
586+
587+
if (
588+
isLastAllowedLine &&
589+
(isNotLastLine || lineWidths[line] > parentContainerInnerWidth)
590+
) {
543591
if (
544-
layout.x + width + ellipsisWidth + spaceWidth >
592+
leftOffset + width + ellipsisWidth + spaceWidth >
545593
parentContainerInnerWidth
546594
) {
547-
const chars = segment(text, 'grapheme', locale)
548-
let subset = ''
549-
let resolvedWidth = 0
550-
for (const char of chars) {
551-
const w = layout.x + measureGraphemeArray([subset + char])
552-
if (
553-
// Keep at least one character:
554-
// > The first character or atomic inline-level element on a line
555-
// must be clipped rather than ellipsed.
556-
// https://drafts.csswg.org/css-overflow/#text-overflow
557-
subset &&
558-
w + ellipsisWidth > parentContainerInnerWidth
559-
) {
560-
break
561-
}
562-
subset += char
563-
resolvedWidth = w
564-
}
565-
text = subset + '…'
595+
const { subset, resolvedWidth } = calcEllipsis(leftOffset, text)
596+
597+
text = subset + _blockEllipsis
566598
skippedLine = line
567599
decorationLines[line][1] = resolvedWidth
568600
isLastDisplayedBeforeEllipsis = true
601+
} else if (nextLayout && nextLayout.line !== line) {
602+
if (textAlign === 'center') {
603+
const { subset, resolvedWidth } = calcEllipsis(leftOffset, text)
604+
605+
text = subset + _blockEllipsis
606+
skippedLine = line
607+
decorationLines[line][1] = resolvedWidth
608+
isLastDisplayedBeforeEllipsis = true
609+
} else {
610+
const nextLineText = texts[i + 1]
611+
612+
const { subset, resolvedWidth } = calcEllipsis(
613+
width + leftOffset,
614+
nextLineText
615+
)
616+
617+
text = text + subset + _blockEllipsis
618+
skippedLine = line
619+
decorationLines[line][1] = resolvedWidth
620+
isLastDisplayedBeforeEllipsis = true
621+
}
569622
}
570623
}
571624
}
@@ -585,9 +638,9 @@ export default async function* buildTextNodes(
585638
!text.includes(Tab) &&
586639
!wordSeparators.includes(text) &&
587640
texts[i + 1] &&
588-
wordPositionInLayout[i + 1] &&
589-
!wordPositionInLayout[i + 1].isImage &&
590-
topOffset === wordPositionInLayout[i + 1].y &&
641+
nextLayout &&
642+
!nextLayout.isImage &&
643+
topOffset === nextLayout.y &&
591644
!isLastDisplayedBeforeEllipsis
592645
) {
593646
if (wordBuffer === null) {
@@ -648,7 +701,7 @@ export default async function* buildTextNodes(
648701
// Get the decoration shape.
649702
if (parentStyle.textDecorationLine) {
650703
// If it's the last word in the current line.
651-
if (line !== wordPositionInLayout[i + 1]?.line || skippedLine === line) {
704+
if (line !== nextLayout?.line || skippedLine === line) {
652705
const deco = decorationLines[line]
653706
if (deco && !deco[2]) {
654707
decorationShape += buildDecoration(
@@ -692,6 +745,10 @@ export default async function* buildTextNodes(
692745
backgroundClipDef += shape
693746
decorationShape = ''
694747
}
748+
749+
if (isLastDisplayedBeforeEllipsis) {
750+
break
751+
}
695752
}
696753

697754
// Embed the font as path.
@@ -764,6 +821,44 @@ function processTextTransform(
764821
return content
765822
}
766823

824+
function processTextOverflow(
825+
parentStyle: Record<string, string | number>,
826+
allowSoftWrap: boolean
827+
): [number, string?] {
828+
const {
829+
textOverflow,
830+
lineClamp,
831+
WebkitLineClamp,
832+
WebkitBoxOrient,
833+
overflow,
834+
display,
835+
} = parentStyle
836+
837+
if (display === 'block' && lineClamp) {
838+
const [lineLimit, blockEllipsis = HorizontalEllipsis] =
839+
parseLineClamp(lineClamp)
840+
if (lineLimit) {
841+
return [lineLimit, blockEllipsis]
842+
}
843+
}
844+
845+
if (
846+
textOverflow === 'ellipsis' &&
847+
display === '-webkit-box' &&
848+
WebkitBoxOrient === 'vertical' &&
849+
isNumber(WebkitLineClamp) &&
850+
WebkitLineClamp > 0
851+
) {
852+
return [WebkitLineClamp, HorizontalEllipsis]
853+
}
854+
855+
if (textOverflow === 'ellipsis' && overflow === 'hidden' && !allowSoftWrap) {
856+
return [1, HorizontalEllipsis]
857+
}
858+
859+
return [Infinity]
860+
}
861+
767862
function processWordBreak(content, wordBreak: string) {
768863
const allowBreakWord = ['break-all', 'break-word'].includes(wordBreak)
769864

@@ -863,3 +958,26 @@ function detectTabs(text: string):
863958
tabCount: 0,
864959
}
865960
}
961+
962+
function parseLineClamp(input: number | string): [number?, string?] {
963+
if (typeof input === 'number') return [input]
964+
965+
const regex1 = /^(\d+)\s*"(.*)"$/
966+
const regex2 = /^(\d+)\s*'(.*)'$/
967+
const match1 = regex1.exec(input)
968+
const match2 = regex2.exec(input)
969+
970+
if (match1) {
971+
const number = +match1[1]
972+
const text = match1[2]
973+
974+
return [number, text]
975+
} else if (match2) {
976+
const number = +match2[1]
977+
const text = match2[2]
978+
979+
return [number, text]
980+
}
981+
982+
return []
983+
}

‎src/utils.ts

+4
Original file line numberDiff line numberDiff line change
@@ -266,6 +266,10 @@ export function isString(x: unknown): x is string {
266266
return typeof x === 'string'
267267
}
268268

269+
export function isNumber(x: unknown): x is number {
270+
return typeof x === 'number'
271+
}
272+
269273
export function isUndefined(x: unknown): x is undefined {
270274
return toString(x) === '[object Undefined]'
271275
}

‎test/error.test.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@ describe('Error', () => {
4040
}
4141
)
4242
expect(result).rejects.toThrowError(
43-
`Invalid value for CSS property "display". Allowed values: "flex" | "none". Received: "inline-block".`
43+
`Invalid value for CSS property "display". Allowed values: "flex" | "block" | "none" | "-webkit-box". Received: "inline-block".`
4444
)
4545
})
4646

‎test/line-clamp.test.tsx

+167
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,167 @@
1+
import { it, describe, expect } from 'vitest'
2+
3+
import { initFonts, toImage } from './utils.js'
4+
import satori from '../src/index.js'
5+
6+
describe('Line Clamp', () => {
7+
let fonts
8+
initFonts((f) => (fonts = f))
9+
10+
it('Should work correctly', async () => {
11+
const svg = await satori(
12+
<div
13+
style={{
14+
height: '100%',
15+
width: '100%',
16+
display: 'flex',
17+
flexDirection: 'column',
18+
alignItems: 'center',
19+
justifyContent: 'center',
20+
backgroundColor: '#fff',
21+
}}
22+
>
23+
<div
24+
style={{
25+
width: '100%',
26+
display: 'block',
27+
lineClamp: 2,
28+
}}
29+
>
30+
lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do
31+
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
32+
minim veniam, quis nostrud exercitation ullamco laboris nisi ut
33+
aliquip ex ea commodo consequat.
34+
</div>
35+
<div
36+
style={{
37+
width: '100%',
38+
display: 'block',
39+
lineClamp: '2',
40+
}}
41+
>
42+
lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do
43+
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
44+
minim veniam, quis nostrud exercitation ullamco laboris nisi ut
45+
aliquip ex ea commodo consequat.
46+
</div>
47+
<div
48+
style={{
49+
width: '100%',
50+
display: 'block',
51+
lineClamp: '2 "… (continued)"',
52+
}}
53+
>
54+
lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do
55+
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
56+
minim veniam, quis nostrud exercitation ullamco laboris nisi ut
57+
aliquip ex ea commodo consequat.
58+
</div>
59+
<div
60+
style={{
61+
width: '100%',
62+
display: 'block',
63+
lineClamp: "2 '… (continued)'",
64+
}}
65+
>
66+
lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do
67+
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
68+
minim veniam, quis nostrud exercitation ullamco laboris nisi ut
69+
aliquip ex ea commodo consequat.
70+
</div>
71+
</div>,
72+
{ width: 200, height: 200, fonts, embedFont: true }
73+
)
74+
expect(toImage(svg, 200)).toMatchImageSnapshot()
75+
})
76+
77+
it('Should replace custom block ellipsis with default ellipsis when too long', async () => {
78+
const svg = await satori(
79+
<div
80+
style={{
81+
height: '100%',
82+
width: '100%',
83+
display: 'flex',
84+
flexDirection: 'column',
85+
alignItems: 'center',
86+
justifyContent: 'center',
87+
backgroundColor: '#fff',
88+
}}
89+
>
90+
<div
91+
style={{
92+
width: '100%',
93+
display: 'block',
94+
lineClamp: '2 "… (loooooooooooooooooooooooooog text)"',
95+
}}
96+
>
97+
lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do
98+
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
99+
minim veniam, quis nostrud exercitation ullamco laboris nisi ut
100+
aliquip ex ea commodo consequat.
101+
</div>
102+
</div>,
103+
{ width: 200, height: 200, fonts, embedFont: true }
104+
)
105+
expect(toImage(svg, 200)).toMatchImageSnapshot()
106+
})
107+
108+
it('Should not work when display is not set to block', async () => {
109+
const svg = await satori(
110+
<div
111+
style={{
112+
height: '100%',
113+
width: '100%',
114+
display: 'flex',
115+
flexDirection: 'column',
116+
alignItems: 'center',
117+
justifyContent: 'center',
118+
backgroundColor: '#fff',
119+
}}
120+
>
121+
<div
122+
style={{
123+
width: '100%',
124+
lineClamp: 2,
125+
}}
126+
>
127+
lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do
128+
eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad
129+
minim veniam, quis nostrud exercitation ullamco laboris nisi ut
130+
aliquip ex ea commodo consequat.
131+
</div>
132+
</div>,
133+
{ width: 200, height: 200, fonts, embedFont: true }
134+
)
135+
expect(toImage(svg, 200)).toMatchImageSnapshot()
136+
})
137+
138+
it('Should work correctly when `text-align: center`', async () => {
139+
const svg = await satori(
140+
<div
141+
style={{
142+
height: '100%',
143+
width: '100%',
144+
display: 'flex',
145+
flexDirection: 'column',
146+
alignItems: 'center',
147+
justifyContent: 'center',
148+
backgroundColor: '#fff',
149+
}}
150+
>
151+
<div
152+
style={{
153+
width: '100%',
154+
display: 'block',
155+
fontSize: 32,
156+
textAlign: 'center',
157+
lineClamp: 2,
158+
}}
159+
>
160+
Making the Web. Superfast
161+
</div>
162+
</div>,
163+
{ width: 200, height: 200, fonts, embedFont: true }
164+
)
165+
expect(toImage(svg, 200)).toMatchImageSnapshot()
166+
})
167+
})

‎test/overflow.test.tsx

+49
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,55 @@ describe('Overflow', () => {
6868
})
6969

7070
it('should work with ellipsis, nowrap', async () => {
71+
const svg = await satori(
72+
<div
73+
style={{
74+
display: 'flex',
75+
height: '100%',
76+
width: '100%',
77+
alignItems: 'center',
78+
justifyContent: 'center',
79+
flexDirection: 'column',
80+
backgroundColor: 'white',
81+
fontSize: 60,
82+
fontWeight: 400,
83+
}}
84+
>
85+
<div
86+
style={{
87+
display: 'flex',
88+
flexDirection: 'column',
89+
width: 450,
90+
overflow: 'hidden',
91+
whiteSpace: 'nowrap',
92+
}}
93+
>
94+
<div
95+
style={{
96+
width: 450,
97+
textOverflow: 'ellipsis',
98+
overflow: 'hidden',
99+
}}
100+
>
101+
{'LuciNyan 1 2 345'}
102+
</div>
103+
<div
104+
style={{
105+
width: 450,
106+
textOverflow: 'ellipsis',
107+
overflow: 'hidden',
108+
}}
109+
>
110+
{'LuciNyan 1 2 345 6'}
111+
</div>
112+
</div>
113+
</div>,
114+
{ width: 450, height: 450, fonts, embedFont: true }
115+
)
116+
expect(toImage(svg, 450)).toMatchImageSnapshot()
117+
})
118+
119+
it("should not work when overflow is not 'hidden' and overflow property should not be inherited", async () => {
71120
const svg = await satori(
72121
<div
73122
style={{

1 commit comments

Comments
 (1)

vercel[bot] commented on May 24, 2023

@vercel[bot]
Please sign in to comment.