@@ -14,15 +14,17 @@ import {
14
14
isUndefined ,
15
15
isString ,
16
16
lengthToNumber ,
17
+ isNumber ,
17
18
} from './utils.js'
18
19
import buildText , { container } from './builder/text.js'
19
20
import { buildDropShadow } from './builder/shadow.js'
20
21
import buildDecoration from './builder/text-decoration.js'
21
22
import { Locale } from './language.js'
22
23
import { FontEngine } from './font.js'
23
- import { Space , Tab } from './characters.js'
24
+ import { HorizontalEllipsis , Space , Tab } from './characters.js'
24
25
25
26
const skippedWordWhenFindingMissingFont = new Set ( [ Tab ] )
27
+
26
28
function shouldSkipWhenFindingMissingFont ( word : string ) : boolean {
27
29
return skippedWordWhenFindingMissingFont . has ( word )
28
30
}
@@ -49,7 +51,6 @@ export default async function* buildTextNodes(
49
51
50
52
const {
51
53
textAlign,
52
- textOverflow,
53
54
whiteSpace,
54
55
wordBreak,
55
56
lineHeight,
@@ -74,6 +75,11 @@ export default async function* buildTextNodes(
74
75
wordBreak as string
75
76
)
76
77
78
+ const [ lineLimit , blockEllipsis ] = processTextOverflow (
79
+ parentStyle ,
80
+ allowSoftWrap
81
+ )
82
+
77
83
const textContainer = createTextContainerNode ( Yoga , textAlign as string )
78
84
parent . insertChild ( textContainer , parent . getChildCount ( ) )
79
85
@@ -222,7 +228,7 @@ export default async function* buildTextNodes(
222
228
// @TODO : Support different writing modes.
223
229
// @TODO : Support RTL languages.
224
230
let i = 0
225
- while ( i < words . length ) {
231
+ while ( i < words . length && lines < lineLimit ) {
226
232
let word = words [ i ]
227
233
const forceBreak = requiredBreaks [ i ]
228
234
@@ -373,10 +379,12 @@ export default async function* buildTextNodes(
373
379
}
374
380
375
381
if ( currentWidth ) {
382
+ if ( lines < lineLimit ) {
383
+ height += currentLineHeight
384
+ }
376
385
lines ++
377
386
lineWidths . push ( currentWidth )
378
387
baselines . push ( currentBaselineOffset )
379
- height += currentLineHeight
380
388
}
381
389
382
390
// @TODO : Support `line-height`.
@@ -482,15 +490,14 @@ export default async function* buildTextNodes(
482
490
let mergedPath = ''
483
491
let extra = ''
484
492
let skippedLine = - 1
485
- let ellipsisWidth = textOverflow === 'ellipsis' ? measureGrapheme ( '…' ) : 0
486
- let spaceWidth = textOverflow === 'ellipsis' ? measureGrapheme ( ' ' ) : 0
487
493
let decorationLines : Record < number , null | number [ ] > = { }
488
494
let wordBuffer : string | null = null
489
495
let bufferedOffset = 0
490
496
491
497
for ( let i = 0 ; i < texts . length ; i ++ ) {
492
498
// Skip whitespace and empty characters.
493
499
const layout = wordPositionInLayout [ i ]
500
+ const nextLayout = wordPositionInLayout [ i + 1 ]
494
501
495
502
if ( ! layout ) continue
496
503
@@ -538,34 +545,80 @@ export default async function* buildTextNodes(
538
545
]
539
546
}
540
547
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
+ ) {
543
591
if (
544
- layout . x + width + ellipsisWidth + spaceWidth >
592
+ leftOffset + width + ellipsisWidth + spaceWidth >
545
593
parentContainerInnerWidth
546
594
) {
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
566
598
skippedLine = line
567
599
decorationLines [ line ] [ 1 ] = resolvedWidth
568
600
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
+ }
569
622
}
570
623
}
571
624
}
@@ -585,9 +638,9 @@ export default async function* buildTextNodes(
585
638
! text . includes ( Tab ) &&
586
639
! wordSeparators . includes ( text ) &&
587
640
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 &&
591
644
! isLastDisplayedBeforeEllipsis
592
645
) {
593
646
if ( wordBuffer === null ) {
@@ -648,7 +701,7 @@ export default async function* buildTextNodes(
648
701
// Get the decoration shape.
649
702
if ( parentStyle . textDecorationLine ) {
650
703
// 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 ) {
652
705
const deco = decorationLines [ line ]
653
706
if ( deco && ! deco [ 2 ] ) {
654
707
decorationShape += buildDecoration (
@@ -692,6 +745,10 @@ export default async function* buildTextNodes(
692
745
backgroundClipDef += shape
693
746
decorationShape = ''
694
747
}
748
+
749
+ if ( isLastDisplayedBeforeEllipsis ) {
750
+ break
751
+ }
695
752
}
696
753
697
754
// Embed the font as path.
@@ -764,6 +821,44 @@ function processTextTransform(
764
821
return content
765
822
}
766
823
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
+
767
862
function processWordBreak ( content , wordBreak : string ) {
768
863
const allowBreakWord = [ 'break-all' , 'break-word' ] . includes ( wordBreak )
769
864
@@ -863,3 +958,26 @@ function detectTabs(text: string):
863
958
tabCount : 0 ,
864
959
}
865
960
}
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
+ }
1 commit comments
vercel[bot] commentedon May 24, 2023
Successfully deployed to the following URLs:
satori-playground – ./
og-playground.vercel.app
og-playground.vercel.sh
satori-playground.vercel.sh
satori-playground-git-main.vercel.sh
satori-playground.vercel.app