diff --git a/packages/vant/src/config-provider/types.ts b/packages/vant/src/config-provider/types.ts index 639ac12a2..42656f04e 100644 --- a/packages/vant/src/config-provider/types.ts +++ b/packages/vant/src/config-provider/types.ts @@ -102,6 +102,7 @@ export type ConfigProviderThemeVars = BaseThemeVars & import('../dropdown-item').DropdownItemThemeVars & import('../dropdown-menu').DropdownMenuThemeVars & import('../empty').EmptyThemeVars & + import('../highlight').HighlightThemeVars & import('../field').FieldThemeVars & import('../floating-bubble').FloatingBubbleThemeVars & import('../floating-panel').FloatingPanelThemeVars & diff --git a/packages/vant/src/highlight/Highlight.tsx b/packages/vant/src/highlight/Highlight.tsx new file mode 100644 index 000000000..2e8542c5c --- /dev/null +++ b/packages/vant/src/highlight/Highlight.tsx @@ -0,0 +1,66 @@ +import { + ref, + watchEffect, + defineComponent, + type PropType, + type ExtractPropTypes, +} from 'vue'; + +import { truthProp, makeStringProp, createNamespace } from '../utils'; + +const [name, bem] = createNamespace('highlight'); + +export const highlightProps = { + keywords: [String, Array] as PropType, + autoEscape: truthProp, + sourceString: makeStringProp(''), + caseSensitive: Boolean, + highlightClassName: String, +}; + +export type HighlightProps = ExtractPropTypes; + +export default defineComponent({ + name, + + props: highlightProps, + + setup(props) { + const highlightedString = ref(''); + + const getHighlightClasses = () => + props.highlightClassName + ? props.highlightClassName + ` ${bem('tag')}` + : `${bem('tag')}`; + + const updateHighlight = () => { + const { keywords, sourceString, caseSensitive, autoEscape } = props; + const flags = caseSensitive ? 'g' : 'gi'; + + let _keywords = keywords; + + if (typeof keywords === 'string') { + _keywords = [keywords]; + } + + const regexPattern = (_keywords as string[]).map((keyword) => { + if (autoEscape) { + return keyword.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); + } + return keyword; + }); + + const regex = new RegExp(`(${regexPattern.join('|')})`, flags); + const highlighted = sourceString.replace( + regex, + `$1`, + ); + + highlightedString.value = highlighted; + }; + + watchEffect(() => updateHighlight()); + + return () =>
; + }, +}); diff --git a/packages/vant/src/highlight/README.md b/packages/vant/src/highlight/README.md new file mode 100644 index 000000000..6a5685998 --- /dev/null +++ b/packages/vant/src/highlight/README.md @@ -0,0 +1,128 @@ +# Highlight + +### Intro + +Highlight the specified text content. + +### Install + +Register component globally via `app.use`, refer to [Component Registration](#/en-US/advanced-usage#zu-jian-zhu-ce) for more registration ways. + +```js +import { createApp } from 'vue'; +import { Highlight } from 'vant'; + +const app = createApp(); +app.use(Highlight); +``` + +## Usage + +### Basic Usage + +You can specify keywords to be highlighted with `keywords` and source text with `source-string`. + +```html + +``` + +```ts +export default { + setup() { + const text = + 'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.'; + const keywords = 'questions'; + + return { + text, + keywords, + }; + }, +}; +``` + +### Multiple Keywords + +If you need to specific more than one keywords, you can pass in `keywords` as an array. + +```html + +``` + +```ts +export default { + setup() { + const text = + 'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.'; + const keywords = ['time', 'life', 'answer']; + + return { + text, + keywords, + }; + }, +}; +``` + +### Custom Class + +Set the `highlight-class-name` of the highlighted tag to customize the style. + +```html + +``` + +```ts +export default { + setup() { + const text = + 'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.'; + const keywords = 'life'; + + return { + text, + keywords, + }; + }, +}; +``` + +```css +.custom-class { + color: red; +} +``` + +## API + +### Props + +| Attribute | Description | Type | Default | +| --- | --- | --- | --- | +| keywords | Expected highlighted text | _string \| string[]_ | - | +| source-string | Source text | _string_ | - | +| auto-escape | Whether to automatically escape | _boolean_ | `true` | +| case-sensitive | Is case sensitive | _boolean_ | `false` | +| highlight-class-name | Class name of the highlight tag | _string_ | - | + +### Types + +The component exports the following type definitions: + +```ts +import type { HighlightProps, HighlightThemeVars } from 'vant'; +``` + +## Theming + +### CSS Variables + +The component provides the following CSS variables, which can be used to customize styles. Please refer to [ConfigProvider component](#/en-US/config-provider). + +| Name | Default Value | Description | +| --- | --- | --- | +| --van-highlight-tag-color | _var(--van-primary-color)_ | Color of highlighted text | diff --git a/packages/vant/src/highlight/README.zh-CN.md b/packages/vant/src/highlight/README.zh-CN.md new file mode 100644 index 000000000..7d43b86c1 --- /dev/null +++ b/packages/vant/src/highlight/README.zh-CN.md @@ -0,0 +1,125 @@ +# Highlight 高亮文本 + +### 介绍 + +高亮指定文本内容。 + +### 引入 + +通过以下方式来全局注册组件,更多注册方式请参考[组件注册](#/zh-CN/advanced-usage#zu-jian-zhu-ce)。 + +```js +import { createApp } from 'vue'; +import { Highlight } from 'vant'; + +const app = createApp(); +app.use(Highlight); +``` + +## 代码演示 + +### 基础用法 + +你可以通过 `keywords` 指定需要高亮的关键字,通过 `source-string` 指定源文本。 + +```html + +``` + +```ts +export default { + setup() { + const text = '慢慢来,不要急,生活给你出了难题,可也终有一天会给出答案。'; + const keywords = '难题'; + + return { + text, + keywords, + }; + }, +}; +``` + +### 多字符匹配 + +如果需要指定多个关键字,可以以数组的形式传入 `keywords`。 + +```html + +``` + +```ts +export default { + setup() { + const text = '慢慢来,不要急,生活给你出了难题,可也终有一天会给出答案。'; + const keywords = ['难题', '终有一天', '答案']; + + return { + text, + keywords, + }; + }, +}; +``` + +### 设置高亮标签类名 + +通过 `highlight-class-name` 可以设置高亮标签的类名,以便自定义样式。 + +```html + +``` + +```ts +export default { + setup() { + const text = '慢慢来,不要急,生活给你出了难题,可也终有一天会给出答案。'; + const keywords = '生活'; + + return { + text, + keywords, + }; + }, +}; +``` + +```css +.custom-class { + color: red; +} +``` + +## API + +### Props + +| 参数 | 说明 | 类型 | 默认值 | +| -------------------- | -------------- | -------------------- | ------- | +| keywords | 期望高亮的文本 | _string \| string[]_ | - | +| source-string | 源文本 | _string_ | - | +| auto-escape | 是否自动转义 | _boolean_ | `true` | +| case-sensitive | 是否区分大小写 | _boolean_ | `false` | +| highlight-class-name | 高亮标签的类名 | _string_ | - | + +### 类型定义 + +组件导出以下类型定义: + +```ts +import type { HighlightProps, HighlightThemeVars } from 'vant'; +``` + +## 主题定制 + +### 样式变量 + +组件提供了下列 CSS 变量,可用于自定义样式,使用方法请参考 [ConfigProvider 组件](#/zh-CN/config-provider)。 + +| 名称 | 默认值 | 描述 | +| ------------------------- | -------------------------- | ------------ | +| --van-highlight-tag-color | _var(--van-primary-color)_ | 高亮文本颜色 | diff --git a/packages/vant/src/highlight/demo/index.vue b/packages/vant/src/highlight/demo/index.vue new file mode 100644 index 000000000..b84f25a09 --- /dev/null +++ b/packages/vant/src/highlight/demo/index.vue @@ -0,0 +1,52 @@ + + + + + diff --git a/packages/vant/src/highlight/index.less b/packages/vant/src/highlight/index.less new file mode 100644 index 000000000..af348dfaa --- /dev/null +++ b/packages/vant/src/highlight/index.less @@ -0,0 +1,9 @@ +:root { + --van-highlight-tag-color: var(--van-primary-color); +} + +.van-highlight { + &__tag { + color: var(--van-highlight-tag-color); + } +} diff --git a/packages/vant/src/highlight/index.ts b/packages/vant/src/highlight/index.ts new file mode 100644 index 000000000..542def491 --- /dev/null +++ b/packages/vant/src/highlight/index.ts @@ -0,0 +1,16 @@ +import { withInstall } from '../utils'; +import _Highlight from './Highlight'; + +export const Highlight = withInstall(_Highlight); +export default Highlight; + +export { highlightProps } from './Highlight'; + +export type { HighlightProps } from './Highlight'; +export type { HighlightThemeVars } from './types'; + +declare module 'vue' { + export interface GlobalComponents { + vanHighlight: typeof Highlight; + } +} diff --git a/packages/vant/src/highlight/test/__snapshots__/demo-ssr.spec.ts.snap b/packages/vant/src/highlight/test/__snapshots__/demo-ssr.spec.ts.snap new file mode 100644 index 000000000..3349ad3f7 --- /dev/null +++ b/packages/vant/src/highlight/test/__snapshots__/demo-ssr.spec.ts.snap @@ -0,0 +1,43 @@ +// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html + +exports[`should render demo and match snapshot 1`] = ` + +
+ +
+ 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. +
+
+
+ +
+ Take your time and be patient. + + Life + + itself will eventually answer all those questions it once raised for you. +
+
+`; diff --git a/packages/vant/src/highlight/test/__snapshots__/demo.spec.ts.snap b/packages/vant/src/highlight/test/__snapshots__/demo.spec.ts.snap new file mode 100644 index 000000000..71f1b5125 --- /dev/null +++ b/packages/vant/src/highlight/test/__snapshots__/demo.spec.ts.snap @@ -0,0 +1,39 @@ +// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html + +exports[`should render demo and match snapshot 1`] = ` +
+
+ 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. +
+
+
+
+ Take your time and be patient. + + Life + + itself will eventually answer all those questions it once raised for you. +
+
+`; diff --git a/packages/vant/src/highlight/test/demo-ssr.spec.ts b/packages/vant/src/highlight/test/demo-ssr.spec.ts new file mode 100644 index 000000000..e00f7b093 --- /dev/null +++ b/packages/vant/src/highlight/test/demo-ssr.spec.ts @@ -0,0 +1,7 @@ +/** + * @vitest-environment node + */ +import Demo from '../demo/index.vue'; +import { snapshotDemo } from '../../../test/demo'; + +snapshotDemo(Demo, { ssr: true }); diff --git a/packages/vant/src/highlight/test/demo.spec.ts b/packages/vant/src/highlight/test/demo.spec.ts new file mode 100644 index 000000000..c0e0c95b9 --- /dev/null +++ b/packages/vant/src/highlight/test/demo.spec.ts @@ -0,0 +1,4 @@ +import Demo from '../demo/index.vue'; +import { snapshotDemo } from '../../../test/demo'; + +snapshotDemo(Demo); diff --git a/packages/vant/src/highlight/test/index.spec.ts b/packages/vant/src/highlight/test/index.spec.ts new file mode 100644 index 000000000..878e22a3b --- /dev/null +++ b/packages/vant/src/highlight/test/index.spec.ts @@ -0,0 +1,69 @@ +import { Highlight } from '..'; +import { mount } from '../../../test'; + +test('should render the specified text label highlighting correctly', () => { + const wrapper = mount(Highlight, { + props: { + keywords: 'questions', + sourceString: + 'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.', + }, + }); + + const highlight = wrapper.find('.van-highlight'); + const tagText = highlight.find('.van-highlight__tag').text(); + + expect(tagText).toEqual('questions'); +}); + +test('multiple keywords highlighting can be specified', () => { + const wrapper = mount(Highlight, { + props: { + keywords: ['time', 'life', 'questions'], + sourceString: + 'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.', + }, + }); + + const highlight = wrapper.find('.van-highlight'); + const tags = highlight.findAll('.van-highlight__tag'); + + expect(tags.length).toEqual(3); + expect(tags[0].text()).toEqual('time'); + expect(tags[1].text()).toEqual('Life'); + expect(tags[2].text()).toEqual('questions'); +}); + +test('should be correctly case sensitive', () => { + const wrapper = mount(Highlight, { + props: { + keywords: ['take', 'Life', 'questions'], + sourceString: + 'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.', + caseSensitive: true, + }, + }); + + const highlight = wrapper.find('.van-highlight'); + const tags = highlight.findAll('.van-highlight__tag'); + + expect(tags.length).toEqual(2); + expect(tags[0].text()).toEqual('Life'); + expect(tags[1].text()).toEqual('questions'); +}); + +test('should set custom class of the highlight tag', () => { + const wrapper = mount(Highlight, { + props: { + keywords: 'time', + sourceString: + 'Take your time and be patient. Life itself will eventually answer all those questions it once raised for you.', + highlightClassName: 'my-custom-class', + }, + }); + + const highlight = wrapper.find('.van-highlight'); + const tag = highlight.find('.van-highlight__tag'); + + expect(tag.classes()).toContain('my-custom-class'); +}); diff --git a/packages/vant/src/highlight/types.ts b/packages/vant/src/highlight/types.ts new file mode 100644 index 000000000..775d0ecdc --- /dev/null +++ b/packages/vant/src/highlight/types.ts @@ -0,0 +1,3 @@ +export type HighlightThemeVars = { + highlightTagColor?: string; +}; diff --git a/packages/vant/vant.config.mjs b/packages/vant/vant.config.mjs index 78640b77b..d65f0e3fd 100644 --- a/packages/vant/vant.config.mjs +++ b/packages/vant/vant.config.mjs @@ -312,6 +312,10 @@ location.href = location.href.replace('youzan.github.io', 'vant-ui.github.io'); path: 'empty', title: 'Empty 空状态', }, + { + path: 'highlight', + title: 'Highlight 高亮文本', + }, { path: 'image-preview', title: 'ImagePreview 图片预览', @@ -780,6 +784,10 @@ location.href = location.href.replace('youzan.github.io', 'vant-ui.github.io'); path: 'empty', title: 'Empty', }, + { + path: 'highlight', + title: 'Highlight', + }, { path: 'image-preview', title: 'ImagePreview',