From f7cf14f31ea2c35e84e3b2fa8d01503ec1156e27 Mon Sep 17 00:00:00 2001 From: 100pah Date: Sat, 16 Nov 2024 19:12:56 +0800 Subject: [PATCH] fix: `boundingRect.x` of text is incorrect when overflow: 'truncate'. Close apache/echarts#18306 . And ecomfe/zrender#1091 located the problematic code successfully but the fix was not entirely correct. --- src/graphic/Text.ts | 2 +- test/text-overflow.html | 128 ++++++++++++++++++++-------------------- 2 files changed, 66 insertions(+), 64 deletions(-) diff --git a/src/graphic/Text.ts b/src/graphic/Text.ts index d7fd9f27e..9dfe1fbf1 100644 --- a/src/graphic/Text.ts +++ b/src/graphic/Text.ts @@ -602,7 +602,7 @@ class ZRText extends Displayable implements GroupLike { if (fixedBoundingRect) { el.setBoundingRect(new BoundingRect( - adjustTextX(subElStyle.x, style.width, subElStyle.textAlign as TextAlign), + adjustTextX(subElStyle.x, contentWidth, subElStyle.textAlign as TextAlign), adjustTextY(subElStyle.y, calculatedLineHeight, subElStyle.textBaseline as TextVerticalAlign), /** * Text boundary should be the real text width. diff --git a/test/text-overflow.html b/test/text-overflow.html index ac9191f52..898b9b151 100644 --- a/test/text-overflow.html +++ b/test/text-overflow.html @@ -24,79 +24,90 @@ var zr = zrender.init(document.getElementById('main'), { renderer: window.__ZRENDER__DEFAULT__RENDERER__ }); + const config = { search: '', - width: 400, + width: 200, height: 400, fontSize: 13, fixedLineHeight: false, lineHeight: 14, rich: true, - breakAll: false + overflow: 'break', + lineOverflow: 'truncate', + align: null } - - var enText = new zrender.Text({ - style: { - text: LARGE_TEXT_EN, - fontSize: config.fontSize, - width: config.width, - padding: 10, - borderColor: '#000', - borderWidth: 1, - - overflow: 'break', - lineOverflow: 'truncate', - ellipsis: '…' - } - }); - zr.add(enText); - - var cnText = new zrender.Text({ - style: { - text: LARGE_TEXT_ZH, - fontSize: config.fontSize, - width: config.width, - padding: 10, - borderColor: '#000', - borderWidth: 1, - - overflow: 'break', - lineOverflow: 'truncate', - ellipsis: '…', - - x: config.width + 100 - } + const gui = new dat.GUI(); + gui.add(config, 'search').onChange(update); + gui.add(config, 'width', 10, 300).onChange(update); + gui.add(config, 'height', 10, 1600).onChange(update); + gui.add(config, 'fontSize', 1, 30).onChange(update); + gui.add(config, 'fixedLineHeight').onChange(update); + gui.add(config, 'lineHeight', 12, 50).onChange(update); + gui.add(config, 'rich').onChange(update); + gui.add(config, 'overflow', ['break', 'breakAll', 'truncate', 'none']).onChange(update); + gui.add(config, 'lineOverflow', ['truncate', null]).onChange(update); + gui.add(config, 'align', [null, 'left', 'center', 'right']).onChange(update); + + const texts = [ + LARGE_TEXT_EN, + LARGE_TEXT_ZH, + 'abcde', + '红黄蓝' + ]; + const textElementList = []; + texts.forEach(text => { + var el = new zrender.Text({ + style: { + text: text, + } + }); + zr.add(el); + textElementList.push(el); }); - zr.add(cnText); - const TEXTS = [LARGE_TEXT_EN, LARGE_TEXT_ZH]; function update() { - enText.style.width = cnText.style.width = config.width; - enText.style.height = cnText.style.height = config.height; - - enText.style.overflow = config.breakAll ? 'breakAll' : 'break'; - - cnText.style.x = config.width + 100; - - enText.style.rich = cnText.style.rich = config.rich ? { - highlight: { - // padding: 4, - backgroundColor: 'yellow', - fontSize: 20 + let lastTextElement = null; + + textElementList.forEach((text, idx) => { + + text.style.padding = 10; + text.style.borderColor = '#000'; + text.style.borderWidth = 1; + text.style.ellipsis = '…'; + + text.style.fontSize = config.fontSize; + text.style.width = config.width; + text.style.height = config.height; + text.style.overflow = config.overflow; + text.style.lineOverflow = config.lineOverflow; + text.style.align = config.align; + + text.style.rich = config.rich ? { + highlight: { + // padding: 4, + backgroundColor: 'yellow', + fontSize: 20 + } + } : null; + + text.style.x = lastTextElement + ? lastTextElement.style.x + lastTextElement.style.width + 50 + : 0; + lastTextElement = text; + + if (text.style.__originalText == null) { + text.style.__originalText = text.style.text; } - } : null; - - [enText, cnText].forEach((text, idx) => { - text.style.fontSize = cnText.style.fontSize = config.fontSize; if (config.search) { - text.style.text = TEXTS[idx].replace( + text.style.text = text.style.__originalText.replace( new RegExp(config.search, 'g'), `{highlight|${config.search}}` ); } else { - text.style.text = TEXTS[idx]; + text.style.text = text.style.__originalText; } if (!config.fixedLineHeight) { @@ -113,15 +124,6 @@ zr.refreshImmediately(); console.timeEnd('render'); } - const gui = new dat.GUI(); - gui.add(config, 'search').onChange(update); - gui.add(config, 'width', 100, 700).onChange(update); - gui.add(config, 'height', 10, 1600).onChange(update); - gui.add(config, 'fontSize', 1, 30).onChange(update); - gui.add(config, 'fixedLineHeight').onChange(update); - gui.add(config, 'lineHeight', 12, 50).onChange(update); - gui.add(config, 'rich').onChange(update); - gui.add(config, 'breakAll').onChange(update); update();