mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
refactor(Highlight): refactor implementation & increase functionality (#12442)
This commit is contained in:
parent
f72903b576
commit
a75a458062
@ -1,21 +1,29 @@
|
|||||||
import {
|
import {
|
||||||
ref,
|
|
||||||
watchEffect,
|
|
||||||
defineComponent,
|
defineComponent,
|
||||||
type PropType,
|
computed,
|
||||||
type ExtractPropTypes,
|
type ExtractPropTypes,
|
||||||
|
type PropType,
|
||||||
} from 'vue';
|
} from 'vue';
|
||||||
|
|
||||||
import { truthProp, makeStringProp, createNamespace } from '../utils';
|
import {
|
||||||
|
createNamespace,
|
||||||
|
makeRequiredProp,
|
||||||
|
makeStringProp,
|
||||||
|
truthProp,
|
||||||
|
} from '../utils';
|
||||||
|
|
||||||
const [name, bem] = createNamespace('highlight');
|
const [name, bem] = createNamespace('highlight');
|
||||||
|
|
||||||
export const highlightProps = {
|
export const highlightProps = {
|
||||||
keywords: [String, Array] as PropType<string | string[]>,
|
|
||||||
autoEscape: truthProp,
|
autoEscape: truthProp,
|
||||||
sourceString: makeStringProp(''),
|
|
||||||
caseSensitive: Boolean,
|
caseSensitive: Boolean,
|
||||||
highlightClassName: String,
|
highlightClass: String,
|
||||||
|
highlightTag: makeStringProp<keyof HTMLElementTagNameMap>('span'),
|
||||||
|
keywords: makeRequiredProp<PropType<string | string[]>>([String, Array]),
|
||||||
|
sourceString: makeStringProp(''),
|
||||||
|
tag: makeStringProp<keyof HTMLElementTagNameMap>('div'),
|
||||||
|
unhighlightClass: String,
|
||||||
|
unhighlightTag: makeStringProp<keyof HTMLElementTagNameMap>('span'),
|
||||||
};
|
};
|
||||||
|
|
||||||
export type HighlightProps = ExtractPropTypes<typeof highlightProps>;
|
export type HighlightProps = ExtractPropTypes<typeof highlightProps>;
|
||||||
@ -26,41 +34,112 @@ export default defineComponent({
|
|||||||
props: highlightProps,
|
props: highlightProps,
|
||||||
|
|
||||||
setup(props) {
|
setup(props) {
|
||||||
const highlightedString = ref('');
|
const highlightChunks = computed(() => {
|
||||||
|
const { autoEscape, caseSensitive, keywords, sourceString } = props;
|
||||||
const getHighlightClasses = () =>
|
|
||||||
props.highlightClassName
|
|
||||||
? props.highlightClassName + ` ${bem('tag')}`
|
|
||||||
: `${bem('tag')}`;
|
|
||||||
|
|
||||||
const updateHighlight = () => {
|
|
||||||
const { keywords, sourceString, caseSensitive, autoEscape } = props;
|
|
||||||
const flags = caseSensitive ? 'g' : 'gi';
|
const flags = caseSensitive ? 'g' : 'gi';
|
||||||
|
const _keywords = Array.isArray(keywords) ? keywords : [keywords];
|
||||||
|
|
||||||
let _keywords = keywords;
|
// generate chunks
|
||||||
|
let chunks = _keywords
|
||||||
|
.filter((keyword) => keyword)
|
||||||
|
.reduce<Array<{ start: number; end: number; highlight: boolean }>>(
|
||||||
|
(chunks, keyword) => {
|
||||||
|
if (autoEscape) {
|
||||||
|
keyword = keyword.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
||||||
|
}
|
||||||
|
|
||||||
if (typeof keywords === 'string') {
|
const regex = new RegExp(keyword, flags);
|
||||||
_keywords = [keywords];
|
|
||||||
|
let match;
|
||||||
|
while ((match = regex.exec(sourceString))) {
|
||||||
|
const start = match.index;
|
||||||
|
const end = regex.lastIndex;
|
||||||
|
|
||||||
|
if (start >= end) {
|
||||||
|
regex.lastIndex++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
chunks.push({
|
||||||
|
start,
|
||||||
|
end,
|
||||||
|
highlight: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return chunks;
|
||||||
|
},
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
|
||||||
|
// merge chunks
|
||||||
|
chunks = chunks
|
||||||
|
.sort((a, b) => a.start - b.start)
|
||||||
|
.reduce<typeof chunks>((chunks, currentChunk) => {
|
||||||
|
const prevChunk = chunks[chunks.length - 1];
|
||||||
|
|
||||||
|
if (!prevChunk || currentChunk.start > prevChunk.end) {
|
||||||
|
const unhighlightStart = prevChunk ? prevChunk.end : 0;
|
||||||
|
const unhighlightEnd = currentChunk.start;
|
||||||
|
|
||||||
|
if (unhighlightStart !== unhighlightEnd) {
|
||||||
|
chunks.push({
|
||||||
|
start: unhighlightStart,
|
||||||
|
end: unhighlightEnd,
|
||||||
|
highlight: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
chunks.push(currentChunk);
|
||||||
|
} else {
|
||||||
|
prevChunk.end = Math.max(prevChunk.end, currentChunk.end);
|
||||||
|
}
|
||||||
|
|
||||||
|
return chunks;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const lastChunk = chunks[chunks.length - 1];
|
||||||
|
|
||||||
|
if (lastChunk && lastChunk.end < sourceString.length) {
|
||||||
|
chunks.push({
|
||||||
|
start: lastChunk.end,
|
||||||
|
end: sourceString.length,
|
||||||
|
highlight: false,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const regexPattern = (_keywords as string[]).map((keyword) => {
|
return chunks;
|
||||||
if (autoEscape) {
|
});
|
||||||
return keyword.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
|
||||||
|
const renderContent = () => {
|
||||||
|
const {
|
||||||
|
sourceString,
|
||||||
|
highlightClass,
|
||||||
|
unhighlightClass,
|
||||||
|
highlightTag,
|
||||||
|
unhighlightTag,
|
||||||
|
} = props;
|
||||||
|
|
||||||
|
return highlightChunks.value.map((chunk) => {
|
||||||
|
const { start, end, highlight } = chunk;
|
||||||
|
const text = sourceString.slice(start, end);
|
||||||
|
|
||||||
|
if (highlight) {
|
||||||
|
return (
|
||||||
|
<highlightTag class={[bem('tag'), highlightClass]}>
|
||||||
|
{text}
|
||||||
|
</highlightTag>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
return keyword;
|
|
||||||
|
return <unhighlightTag class={unhighlightClass}>{text}</unhighlightTag>;
|
||||||
});
|
});
|
||||||
|
|
||||||
const regex = new RegExp(`(${regexPattern.join('|')})`, flags);
|
|
||||||
const highlighted = sourceString.replace(
|
|
||||||
regex,
|
|
||||||
`<span class="${getHighlightClasses()}">$1</span>`,
|
|
||||||
);
|
|
||||||
|
|
||||||
highlightedString.value = highlighted;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
watchEffect(() => updateHighlight());
|
return () => {
|
||||||
|
const { tag } = props;
|
||||||
|
|
||||||
return () => <div class={bem()} innerHTML={highlightedString.value}></div>;
|
return <tag class={bem()}>{renderContent()}</tag>;
|
||||||
|
};
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -66,13 +66,13 @@ export default {
|
|||||||
|
|
||||||
### Custom Class
|
### Custom Class
|
||||||
|
|
||||||
Set the `highlight-class-name` of the highlighted tag to customize the style.
|
Set the `highlight-class` of the highlighted tag to customize the style.
|
||||||
|
|
||||||
```html
|
```html
|
||||||
<van-highlight
|
<van-highlight
|
||||||
:keywords="keywords"
|
:keywords="keywords"
|
||||||
:source-string="text"
|
:source-string="text"
|
||||||
highlight-class-name="custom-class"
|
highlight-class="custom-class"
|
||||||
/>
|
/>
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -103,11 +103,15 @@ export default {
|
|||||||
|
|
||||||
| Attribute | Description | Type | Default |
|
| Attribute | Description | Type | Default |
|
||||||
| --- | --- | --- | --- |
|
| --- | --- | --- | --- |
|
||||||
| keywords | Expected highlighted text | _string \| string[]_ | - |
|
|
||||||
| source-string | Source text | _string_ | - |
|
|
||||||
| auto-escape | Whether to automatically escape | _boolean_ | `true` |
|
| auto-escape | Whether to automatically escape | _boolean_ | `true` |
|
||||||
| case-sensitive | Is case sensitive | _boolean_ | `false` |
|
| case-sensitive | Is case sensitive | _boolean_ | `false` |
|
||||||
| highlight-class-name | Class name of the highlight tag | _string_ | - |
|
| highlight-class | Class name of the highlight element | _string_ | - |
|
||||||
|
| highlight-tag | HTML Tag of highlighted element | _string_ | `span` |
|
||||||
|
| keywords | Expected highlighted text | _string \| string[]_ | - |
|
||||||
|
| source-string | Source text | _string_ | - |
|
||||||
|
| tag | HTML Tag of root element | _string_ | `div` |
|
||||||
|
| unhighlight-class | Class name of the unhighlight element | _string_ | - |
|
||||||
|
| unhighlight-tag | HTML Tag of unhighlighted element | _string_ | `span` |
|
||||||
|
|
||||||
### Types
|
### Types
|
||||||
|
|
||||||
|
@ -64,13 +64,13 @@ export default {
|
|||||||
|
|
||||||
### 设置高亮标签类名
|
### 设置高亮标签类名
|
||||||
|
|
||||||
通过 `highlight-class-name` 可以设置高亮标签的类名,以便自定义样式。
|
通过 `highlight-class` 可以设置高亮标签的类名,以便自定义样式。
|
||||||
|
|
||||||
```html
|
```html
|
||||||
<van-highlight
|
<van-highlight
|
||||||
:keywords="keywords"
|
:keywords="keywords"
|
||||||
:source-string="text"
|
:source-string="text"
|
||||||
highlight-class-name="custom-class"
|
highlight-class="custom-class"
|
||||||
/>
|
/>
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -98,13 +98,17 @@ export default {
|
|||||||
|
|
||||||
### Props
|
### Props
|
||||||
|
|
||||||
| 参数 | 说明 | 类型 | 默认值 |
|
| 参数 | 说明 | 类型 | 默认值 |
|
||||||
| -------------------- | -------------- | -------------------- | ------- |
|
| --- | --- | --- | --- |
|
||||||
| keywords | 期望高亮的文本 | _string \| string[]_ | - |
|
| auto-escape | 是否自动转义 | _boolean_ | `true` |
|
||||||
| source-string | 源文本 | _string_ | - |
|
| case-sensitive | 是否区分大小写 | _boolean_ | `false` |
|
||||||
| auto-escape | 是否自动转义 | _boolean_ | `true` |
|
| highlight-class | 高亮元素的类名 | _string_ | - |
|
||||||
| case-sensitive | 是否区分大小写 | _boolean_ | `false` |
|
| highlight-tag | 高亮元素对应的 HTML 标签名 | _string_ | `span` |
|
||||||
| highlight-class-name | 高亮标签的类名 | _string_ | - |
|
| keywords | 期望高亮的文本 | _string \| string[]_ | - |
|
||||||
|
| source-string | 源文本 | _string_ | - |
|
||||||
|
| tag | 根节点对应的 HTML 标签名 | _string_ | `div` |
|
||||||
|
| unhighlight-class | 非高亮元素的类名 | _string_ | - |
|
||||||
|
| unhighlight-tag | 非高亮元素对应的 HTML 标签名 | _string_ | `span` |
|
||||||
|
|
||||||
### 类型定义
|
### 类型定义
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ const t = useTranslate({
|
|||||||
keywords2: ['难题', '终有一天', '答案'],
|
keywords2: ['难题', '终有一天', '答案'],
|
||||||
keywords3: '生活',
|
keywords3: '生活',
|
||||||
multipleKeywords: '多字符匹配',
|
multipleKeywords: '多字符匹配',
|
||||||
highlightClassName: '设置高亮标签类名',
|
highlightClass: '设置高亮标签类名',
|
||||||
},
|
},
|
||||||
'en-US': {
|
'en-US': {
|
||||||
text1:
|
text1:
|
||||||
@ -18,7 +18,7 @@ const t = useTranslate({
|
|||||||
keywords2: ['time', 'life', 'answer'],
|
keywords2: ['time', 'life', 'answer'],
|
||||||
keywords3: 'life',
|
keywords3: 'life',
|
||||||
multipleKeywords: 'Multiple Keywords',
|
multipleKeywords: 'Multiple Keywords',
|
||||||
highlightClassName: 'Highlight Class Name',
|
highlightClass: 'Highlight Class Name',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
@ -32,11 +32,11 @@ const t = useTranslate({
|
|||||||
<van-highlight :keywords="t('keywords2')" :source-string="t('text1')" />
|
<van-highlight :keywords="t('keywords2')" :source-string="t('text1')" />
|
||||||
</demo-block>
|
</demo-block>
|
||||||
|
|
||||||
<demo-block :title="t('highlightClassName')">
|
<demo-block :title="t('highlightClass')">
|
||||||
<van-highlight
|
<van-highlight
|
||||||
:keywords="t('keywords3')"
|
:keywords="t('keywords3')"
|
||||||
:source-string="t('text1')"
|
:source-string="t('text1')"
|
||||||
highlight-class-name="custom-class"
|
highlight-class="custom-class"
|
||||||
/>
|
/>
|
||||||
</demo-block>
|
</demo-block>
|
||||||
</template>
|
</template>
|
||||||
|
@ -5,39 +5,58 @@ exports[`should render demo and match snapshot 1`] = `
|
|||||||
<div>
|
<div>
|
||||||
<!--[-->
|
<!--[-->
|
||||||
<div class="van-highlight">
|
<div class="van-highlight">
|
||||||
Take your time and be patient. Life itself will eventually answer all those
|
<!--[-->
|
||||||
|
<span class>
|
||||||
|
Take your time and be patient. Life itself will eventually answer all those
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
questions
|
questions
|
||||||
</span>
|
</span>
|
||||||
it once raised for you.
|
<span class>
|
||||||
|
it once raised for you.
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<!--[-->
|
<!--[-->
|
||||||
<div class="van-highlight">
|
<div class="van-highlight">
|
||||||
Take your
|
<!--[-->
|
||||||
|
<span class>
|
||||||
|
Take your
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
time
|
time
|
||||||
</span>
|
</span>
|
||||||
and be patient.
|
<span class>
|
||||||
|
and be patient.
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
Life
|
Life
|
||||||
</span>
|
</span>
|
||||||
itself will eventually
|
<span class>
|
||||||
|
itself will eventually
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
answer
|
answer
|
||||||
</span>
|
</span>
|
||||||
all those questions it once raised for you.
|
<span class>
|
||||||
|
all those questions it once raised for you.
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<!--[-->
|
<!--[-->
|
||||||
<div class="van-highlight">
|
<div class="van-highlight">
|
||||||
Take your time and be patient.
|
<!--[-->
|
||||||
<span class="custom-class van-highlight__tag">
|
<span class>
|
||||||
|
Take your time and be patient.
|
||||||
|
</span>
|
||||||
|
<span class="van-highlight__tag custom-class">
|
||||||
Life
|
Life
|
||||||
</span>
|
</span>
|
||||||
itself will eventually answer all those questions it once raised for you.
|
<span class>
|
||||||
|
itself will eventually answer all those questions it once raised for you.
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
@ -3,37 +3,53 @@
|
|||||||
exports[`should render demo and match snapshot 1`] = `
|
exports[`should render demo and match snapshot 1`] = `
|
||||||
<div>
|
<div>
|
||||||
<div class="van-highlight">
|
<div class="van-highlight">
|
||||||
Take your time and be patient. Life itself will eventually answer all those
|
<span>
|
||||||
|
Take your time and be patient. Life itself will eventually answer all those
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
questions
|
questions
|
||||||
</span>
|
</span>
|
||||||
it once raised for you.
|
<span>
|
||||||
|
it once raised for you.
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div class="van-highlight">
|
<div class="van-highlight">
|
||||||
Take your
|
<span>
|
||||||
|
Take your
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
time
|
time
|
||||||
</span>
|
</span>
|
||||||
and be patient.
|
<span>
|
||||||
|
and be patient.
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
Life
|
Life
|
||||||
</span>
|
</span>
|
||||||
itself will eventually
|
<span>
|
||||||
|
itself will eventually
|
||||||
|
</span>
|
||||||
<span class="van-highlight__tag">
|
<span class="van-highlight__tag">
|
||||||
answer
|
answer
|
||||||
</span>
|
</span>
|
||||||
all those questions it once raised for you.
|
<span>
|
||||||
|
all those questions it once raised for you.
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<div class="van-highlight">
|
<div class="van-highlight">
|
||||||
Take your time and be patient.
|
<span>
|
||||||
<span class="custom-class van-highlight__tag">
|
Take your time and be patient.
|
||||||
|
</span>
|
||||||
|
<span class="van-highlight__tag custom-class">
|
||||||
Life
|
Life
|
||||||
</span>
|
</span>
|
||||||
itself will eventually answer all those questions it once raised for you.
|
<span>
|
||||||
|
itself will eventually answer all those questions it once raised for you.
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
@ -58,7 +58,7 @@ test('should set custom class of the highlight tag', () => {
|
|||||||
keywords: 'time',
|
keywords: 'time',
|
||||||
sourceString:
|
sourceString:
|
||||||
'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.',
|
'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.',
|
||||||
highlightClassName: 'my-custom-class',
|
highlightClass: 'my-custom-class',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -67,3 +67,31 @@ test('should set custom class of the highlight tag', () => {
|
|||||||
|
|
||||||
expect(tag.classes()).toContain('my-custom-class');
|
expect(tag.classes()).toContain('my-custom-class');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('should be merged when the highlighted content overlaps', () => {
|
||||||
|
const wrapper = mount(Highlight, {
|
||||||
|
props: {
|
||||||
|
keywords: ['ab', 'bc'],
|
||||||
|
sourceString: 'abcd',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const highlight = wrapper.find('.van-highlight');
|
||||||
|
const tags = highlight.findAll('.van-highlight__tag');
|
||||||
|
|
||||||
|
expect(tags[0].text()).toEqual('abc');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('empty text should not be matched', () => {
|
||||||
|
const wrapper = mount(Highlight, {
|
||||||
|
props: {
|
||||||
|
keywords: ['', 'bc'],
|
||||||
|
sourceString: 'abcd',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const highlight = wrapper.find('.van-highlight');
|
||||||
|
const tags = highlight.findAll('.van-highlight__tag');
|
||||||
|
|
||||||
|
expect(tags[0].text()).toEqual('bc');
|
||||||
|
});
|
||||||
|
Loading…
x
Reference in New Issue
Block a user