From 41df4b481915c29aa8fed5e6cc48062de20fc43b Mon Sep 17 00:00:00 2001 From: neverland Date: Mon, 5 Feb 2018 17:52:43 +0800 Subject: [PATCH] [new feature] add Circle component (#608) --- docs/demos/common.js | 4 + docs/demos/index.js | 1 + docs/demos/views/circle.vue | 78 +++++++++++++++++++ docs/markdown/en-US/circle.md | 69 +++++++++++++++++ docs/markdown/index.js | 2 + docs/markdown/zh-CN/circle.md | 70 +++++++++++++++++ docs/src/doc.config.js | 8 ++ packages/circle/index.vue | 126 +++++++++++++++++++++++++++++++ packages/index.js | 3 + packages/vant-css/src/circle.css | 32 ++++++++ packages/vant-css/src/index.css | 3 +- test/unit/specs/circle.spec.js | 65 ++++++++++++++++ 12 files changed, 460 insertions(+), 1 deletion(-) create mode 100644 docs/demos/views/circle.vue create mode 100644 docs/markdown/en-US/circle.md create mode 100644 docs/markdown/zh-CN/circle.md create mode 100644 packages/circle/index.vue create mode 100644 packages/vant-css/src/circle.css create mode 100644 test/unit/specs/circle.spec.js diff --git a/docs/demos/common.js b/docs/demos/common.js index 21757d521..b2f7f208b 100644 --- a/docs/demos/common.js +++ b/docs/demos/common.js @@ -31,6 +31,8 @@ Vue.component('demo-section', DemoSection); Locale.add({ 'zh-CN': { + add: '增加', + decrease: '减少', red: '红色', orange: '橙色', yellow: '黄色', @@ -56,6 +58,8 @@ Locale.add({ passwordPlaceholder: '请输入密码' }, 'en-US': { + add: 'Add', + decrease: 'Decrease', red: 'Red', orange: 'Orange', yellow: 'Yellow', diff --git a/docs/demos/index.js b/docs/demos/index.js index 7a6113cf3..a3c210b92 100644 --- a/docs/demos/index.js +++ b/docs/demos/index.js @@ -31,6 +31,7 @@ export default { 'cell-swipe': asyncWrapper(r => require.ensure([], () => r(componentWrapper(require('./views/cell-swipe'), 'cell-swipe')), 'cell-swipe')), 'cell': asyncWrapper(r => require.ensure([], () => r(componentWrapper(require('./views/cell'), 'cell')), 'cell')), 'checkbox': asyncWrapper(r => require.ensure([], () => r(componentWrapper(require('./views/checkbox'), 'checkbox')), 'checkbox')), + 'circle': asyncWrapper(r => require.ensure([], () => r(componentWrapper(require('./views/circle'), 'circle')), 'circle')), 'contact': asyncWrapper(r => require.ensure([], () => r(componentWrapper(require('./views/contact'), 'contact')), 'contact')), 'coupon': asyncWrapper(r => require.ensure([], () => r(componentWrapper(require('./views/coupon'), 'coupon')), 'coupon')), 'datetime-picker': asyncWrapper(r => require.ensure([], () => r(componentWrapper(require('./views/datetime-picker'), 'datetime-picker')), 'datetime-picker')), diff --git a/docs/demos/views/circle.vue b/docs/demos/views/circle.vue new file mode 100644 index 000000000..276724745 --- /dev/null +++ b/docs/demos/views/circle.vue @@ -0,0 +1,78 @@ + + + + + diff --git a/docs/markdown/en-US/circle.md b/docs/markdown/en-US/circle.md new file mode 100644 index 000000000..5e1bb8cda --- /dev/null +++ b/docs/markdown/en-US/circle.md @@ -0,0 +1,69 @@ +## Circle + +### Install +``` javascript +import { Circle } from 'vant'; + +Vue.use(Circle); +``` + +### Usage + +#### Basic Usage + +```html + +``` + +``` javascript +export default { + data() { + return { + currentRate: 0 + }; + }, + computed: { + text() { + return this.currentRate.toFixed(0) + '%' + } + } +}; +``` + +#### Custom style + +```html + +``` + + +### API + +| Attribute | Description | Type | Default | Accepted Values | +|-----------|-----------|-----------|-------------|-------------| +| v-model | Current rate | `Number` | - | - | +| rate | Target rate | `Number` | `100` | - | +| size | Circle size | `String` | `100px` | - | +| color | Progress bar color | `String` | `#38f` | - | +| layer-color | Layer color | `String` | `#fff` | - | +| fill | Fill color | `String` | `none` | - | +| speed | Animate speed(rate/s)| `Number` | - | - | +| text | Text | `String` | - | - | +| stroke-width | Stroke width | `Number` | `40` | - | +| clockwise | Is clockwise | `Boolean` | `true` | - | diff --git a/docs/markdown/index.js b/docs/markdown/index.js index 124f4cefe..e5108c2a8 100644 --- a/docs/markdown/index.js +++ b/docs/markdown/index.js @@ -25,6 +25,7 @@ export default { 'zh-CN/changelog-generated': wrapper(r => require.ensure([], () => r(require('./zh-CN/changelog-generated.md')), 'zh-CN/changelog-generated')), 'zh-CN/changelog': wrapper(r => require.ensure([], () => r(require('./zh-CN/changelog.md')), 'zh-CN/changelog')), 'zh-CN/checkbox': wrapper(r => require.ensure([], () => r(require('./zh-CN/checkbox.md')), 'zh-CN/checkbox')), + 'zh-CN/circle': wrapper(r => require.ensure([], () => r(require('./zh-CN/circle.md')), 'zh-CN/circle')), 'zh-CN/contact': wrapper(r => require.ensure([], () => r(require('./zh-CN/contact.md')), 'zh-CN/contact')), 'zh-CN/coupon': wrapper(r => require.ensure([], () => r(require('./zh-CN/coupon.md')), 'zh-CN/coupon')), 'zh-CN/datetime-picker': wrapper(r => require.ensure([], () => r(require('./zh-CN/datetime-picker.md')), 'zh-CN/datetime-picker')), @@ -77,6 +78,7 @@ export default { 'en-US/cell': wrapper(r => require.ensure([], () => r(require('./en-US/cell.md')), 'en-US/cell')), 'en-US/changelog': wrapper(r => require.ensure([], () => r(require('./en-US/changelog.md')), 'en-US/changelog')), 'en-US/checkbox': wrapper(r => require.ensure([], () => r(require('./en-US/checkbox.md')), 'en-US/checkbox')), + 'en-US/circle': wrapper(r => require.ensure([], () => r(require('./en-US/circle.md')), 'en-US/circle')), 'en-US/contact': wrapper(r => require.ensure([], () => r(require('./en-US/contact.md')), 'en-US/contact')), 'en-US/coupon': wrapper(r => require.ensure([], () => r(require('./en-US/coupon.md')), 'en-US/coupon')), 'en-US/datetime-picker': wrapper(r => require.ensure([], () => r(require('./en-US/datetime-picker.md')), 'en-US/datetime-picker')), diff --git a/docs/markdown/zh-CN/circle.md b/docs/markdown/zh-CN/circle.md new file mode 100644 index 000000000..e0c2fee3e --- /dev/null +++ b/docs/markdown/zh-CN/circle.md @@ -0,0 +1,70 @@ +## Circle 环形进度条 + +### 使用指南 +``` javascript +import { Circle } from 'vant'; + +Vue.use(Circle); +``` + +### 代码演示 + +#### 基础用法 +通过 `rate` 指定目标进度,`v-model` 代表当前进度,`speed` 控制动画速度 + +```html + +``` + +``` javascript +export default { + data() { + return { + currentRate: 0 + }; + }, + computed: { + text() { + return this.currentRate.toFixed(0) + '%' + } + } +}; +``` + +#### 样式定制 + +```html + +``` + + +### API + +| 参数 | 说明 | 类型 | 默认值 | 可选值 | +|-----------|-----------|-----------|-------------|-------------| +| v-model | 当前进度 | `Number` | - | - | +| rate | 目标进度 | `Number` | `100` | - | +| size | 圆环直径 | `String` | `100px` | - | +| color | 进度条颜色 | `String` | `#38f` | - | +| layer-color | 轨道颜色 | `String` | `#fff` | - | +| fill | 填充颜色 | `String` | `none` | - | +| speed | 动画速度(单位为 rate/s)| `Number` | - | - | +| text | 文字 | `String` | - | - | +| stroke-width | 进度条宽度 | `Number` | `40` | - | +| clockwise | 是否顺时针增加 | `Boolean` | `true` | - | diff --git a/docs/src/doc.config.js b/docs/src/doc.config.js index c14fd4b24..f439eb013 100644 --- a/docs/src/doc.config.js +++ b/docs/src/doc.config.js @@ -80,6 +80,10 @@ module.exports = { path: '/cell', title: 'Cell - 单元格' }, + { + path: '/circle', + title: 'Circle - 环形进度条' + }, { path: '/icon', title: 'Icon - 图标' @@ -358,6 +362,10 @@ module.exports = { path: '/cell', title: 'Cell' }, + { + path: '/circle', + title: 'Circle' + }, { path: '/icon', title: 'Icon' diff --git a/packages/circle/index.vue b/packages/circle/index.vue new file mode 100644 index 000000000..1040124da --- /dev/null +++ b/packages/circle/index.vue @@ -0,0 +1,126 @@ + + + diff --git a/packages/index.js b/packages/index.js index cbd7e654f..f0c6cd13c 100644 --- a/packages/index.js +++ b/packages/index.js @@ -12,6 +12,7 @@ import CellGroup from './cell-group'; import CellSwipe from './cell-swipe'; import Checkbox from './checkbox'; import CheckboxGroup from './checkbox-group'; +import Circle from './circle'; import Col from './col'; import ContactCard from './contact-card'; import ContactEdit from './contact-edit'; @@ -77,6 +78,7 @@ const components = [ CellSwipe, Checkbox, CheckboxGroup, + Circle, Col, ContactCard, ContactEdit, @@ -149,6 +151,7 @@ export { CellSwipe, Checkbox, CheckboxGroup, + Circle, Col, ContactCard, ContactEdit, diff --git a/packages/vant-css/src/circle.css b/packages/vant-css/src/circle.css new file mode 100644 index 000000000..49eb936db --- /dev/null +++ b/packages/vant-css/src/circle.css @@ -0,0 +1,32 @@ +@import './common/var.css'; + +.van-circle { + position: relative; + text-align: center; + display: inline-block; + + svg { + top: 0; + left: 0; + width: 100%; + height: 100%; + position: absolute; + } + + &__layer { + fill: none; + stroke-dasharray: 3140px; + stroke-dashoffset: 3140px; + transform: rotate(90deg); + transform-origin: 530px 530px; + } + + &__text { + top: 50%; + left: 0; + width: 100%; + color: $text-color; + position: absolute; + transform: translateY(-50%); + } +} diff --git a/packages/vant-css/src/index.css b/packages/vant-css/src/index.css index a0ec70bd4..e2f140575 100644 --- a/packages/vant-css/src/index.css +++ b/packages/vant-css/src/index.css @@ -12,7 +12,7 @@ @import './badge.css'; @import './button.css'; @import './cell.css'; -@import './card.css'; +@import './circle.css'; @import './loading.css'; @import './nav-bar.css'; @import './notice-bar.css'; @@ -53,6 +53,7 @@ /* business components */ @import './address-edit.css'; @import './address-list.css'; +@import './card.css'; @import './contact-card.css'; @import './contact-list.css'; @import './contact-edit.css'; diff --git a/test/unit/specs/circle.spec.js b/test/unit/specs/circle.spec.js new file mode 100644 index 000000000..7fff94e7c --- /dev/null +++ b/test/unit/specs/circle.spec.js @@ -0,0 +1,65 @@ +import { mount } from 'avoriaz'; +import Circle from 'packages/circle'; + +describe('Circle', () => { + let wrapper; + afterEach(() => { + wrapper && wrapper.destroy(); + }); + + it('create a circle', () => { + wrapper = mount(Circle, { + propsData: { + text: 'test' + } + }); + + expect(wrapper.hasClass('van-circle')).to.be.true; + expect(wrapper.find('.van-circle__text')[0].text()).to.equal('test'); + }); + + it('circle rate', done => { + let currentRate = 0; + wrapper = mount(Circle, { + propsData: { + rate: 0, + value: 0, + clockwise: false + } + }); + wrapper.vm.$on('input', rate => { + currentRate = rate; + }); + wrapper.vm.rate = 50; + + setTimeout(() => { + expect(currentRate).to.equal(50); + done(); + }, 100); + }); + + it('circle animation', done => { + let currentRate = 0; + wrapper = mount(Circle, { + propsData: { + rate: 0, + value: 0, + speed: 500, + clockwise: false + } + }); + wrapper.vm.$on('input', rate => { + currentRate = rate; + }); + wrapper.vm.rate = 50; + + setTimeout(() => { + expect(currentRate === 50).to.be.false; + setTimeout(() => { + expect(currentRate === 50).to.be.true; + done(); + }, 200); + }, 50); + }); +}); +