Skip to content

Commit

Permalink
fix(input-number): many fixes
Browse files Browse the repository at this point in the history
- Fix `n-input-number` lacks `on-update-value` prop.
- Fix `n-input-number`'s value can't be null.
- Fix `n-input-number`'s button doesn't work after value is cleared, closes [#251](#251).
- `n-input-number` will focus directly, closes [#244](#244).
  • Loading branch information
07akioni committed Jun 23, 2021
1 parent fd23a30 commit a91dd59
Show file tree
Hide file tree
Showing 11 changed files with 156 additions and 62 deletions.
9 changes: 8 additions & 1 deletion CHANGELOG.en-US.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,14 @@

### Fixes

- `n-select` can't input in filterable mode in single mode in iOS Safari, closes [#230](https://github.com/TuSimple/naive-ui/issues/230)
- Fix `n-select` can't input in filterable mode in single mode in iOS Safari, closes [#230](https://github.com/TuSimple/naive-ui/issues/230)
- Fix `n-input-number` lacks `on-update-value` prop.
- Fix `n-input-number`'s value can't be null.
- Fix `n-input-number`'s button doesn't work after value is cleared, closes [#251](https://github.com/TuSimple/naive-ui/issues/251).

## Refactors

- `n-input-number` will focus directly, closes [#244](https://github.com/TuSimple/naive-ui/issues/244).

## 2.13.0 (2021-06-21)

Expand Down
9 changes: 8 additions & 1 deletion CHANGELOG.zh-CN.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,14 @@

### Fixes

- `n-select` 在可过滤单选模式下在 iOS Safari 无法输入,关闭 [#230](https://github.com/TuSimple/naive-ui/issues/230)
- 修复 `n-select` 在可过滤单选模式下在 iOS Safari 无法输入,关闭 [#230](https://github.com/TuSimple/naive-ui/issues/230)
- 修复 `n-input-number` 缺少 `on-update-value` 属性
- 修复 `n-input-number` 值无法为 `null`
- 修复 `n-input-number` 的按钮在值清空后无法使用,关闭 [#251](https://github.com/TuSimple/naive-ui/issues/251)

## Refactors

- `n-input-number` 会直接聚焦,关闭 [#244](https://github.com/TuSimple/naive-ui/issues/244)

## 2.13.0 (2021-06-21)

Expand Down
4 changes: 2 additions & 2 deletions src/input-number/demos/enUS/index.demo-entry.md
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ show-button
## Props

| Name | Type | Default | Description |
| --- | --- | --- | --- |
| --- | --- | --- | --- | --- |
| bordered | `boolean` | `true` | Whether to show the border. |
| default-value | `number \| null` | `null` | Default value in uncontrolled mode. |
| disabled | `boolean` | `false` | Whether to disable the input. |
Expand All @@ -32,4 +32,4 @@ show-button
| value | `number` | `undefined` | Value in controlled mode. |
| on-blur | `(event: FocusEvent) => void` | `undefined` | Callback when blur. |
| on-focus | `(event: FocusEvent) => void` | `undefined` | Callback when focused. |
| on-update:value | `(value: number) => void` | `undefined` | Callback when the component's value changes. |
| on-update:value | `(value: number | null) => void` | `undefined` | Callback when the component's value changes. |
33 changes: 33 additions & 0 deletions src/input-number/demos/zhCN/debug.demo.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
# Debug

```html
<n-input-number
v-model:value="value1"
placeholder="最小值"
:min="-3"
:max="5"
@update:value="handleUpdateValue1"
/>
{{ JSON.stringify(value1) }}
<n-input-number v-model:value="value2" @update:value="handleUpdateValue2" />
{{ JSON.stringify(value2) }}
```

```js
import { defineComponent, ref } from 'vue'

export default defineComponent({
setup () {
return {
handleUpdateValue1 (v) {
console.log(v)
},
handleUpdateValue2 (v) {
console.log(v)
},
value1: ref(null),
value2: ref(null)
}
}
})
```
5 changes: 3 additions & 2 deletions src/input-number/demos/zhCN/index.demo-entry.md
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,13 @@ size
step
validator
show-button
debug
```

## Props

| 名称 | 类型 | 默认值 | 说明 |
| --- | --- | --- | --- |
| --- | --- | --- | --- | --- |
| bordered | `boolean` | `true` | 是否有边框 |
| default-value | `number \| null` | `null` | 非受控模式下的默认值 |
| disabled | `boolean` | `false` | 是否禁用 |
Expand All @@ -32,4 +33,4 @@ show-button
| value | `number \| null` | `undefined` | 受控模式下的值 |
| on-blur | `(event: FocusEvent) => void` | `undefined` | 移除焦点的回调 |
| on-focus | `(event: FocusEvent) => void` | `undefined` | 获取焦点的回调 |
| on-update:value | `(value: number) => void` | `undefined` | 组件值发生变化的回调 |
| on-update:value | `(value: number | null) => void` | `undefined` | 组件值发生变化的回调 |
55 changes: 24 additions & 31 deletions src/input-number/src/InputNumber.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import type { ThemeProps } from '../../_mixins'
import { warn, call, MaybeArray, ExtractPublicPropTypes } from '../../_utils'
import { inputNumberLight, InputNumberTheme } from '../styles'
import { parse, validator, format, parseNumber } from './utils'
import type { OnUpdateValue } from './interface'

const inputNumberProps = {
...(useTheme.props as ThemeProps<InputNumberTheme>),
Expand All @@ -19,27 +20,15 @@ const inputNumberProps = {
type: Number as PropType<number | null>,
default: null
},
value: {
type: Number,
default: undefined
},
value: Number,
step: {
type: [Number, String],
default: 1
},
min: {
type: [Number, String],
default: undefined
},
max: {
type: [Number, String],
default: undefined
},
min: [Number, String],
max: [Number, String],
size: String as PropType<'small' | 'medium' | 'large'>,
disabled: {
type: Boolean,
default: false
},
disabled: Boolean,
validator: Function as PropType<(value: number) => boolean>,
bordered: {
type: Boolean as PropType<boolean | undefined>,
Expand All @@ -49,17 +38,13 @@ const inputNumberProps = {
type: Boolean,
default: true
},
// eslint-disable-next-line vue/prop-name-casing
'onUpdate:value': [Function, Array] as PropType<
MaybeArray<(value: number) => void>
>,
'onUpdate:value': [Function, Array] as PropType<MaybeArray<OnUpdateValue>>,
onUpdateValue: [Function, Array] as PropType<MaybeArray<OnUpdateValue>>,
onFocus: [Function, Array] as PropType<MaybeArray<(e: FocusEvent) => void>>,
onBlur: [Function, Array] as PropType<MaybeArray<(e: FocusEvent) => void>>,
// deprecated
onChange: {
type: [Function, Array] as PropType<
MaybeArray<(value: number) => void> | undefined
>,
type: [Function, Array] as PropType<MaybeArray<OnUpdateValue> | undefined>,
validator: () => {
if (__DEV__) {
warn(
Expand Down Expand Up @@ -124,39 +109,47 @@ export default defineComponent({
if (parsedNumber !== null) return parsedNumber
else return null
})
const doUpdateValue = (value: number): void => {
const doUpdateValue = (value: number | null): void => {
const { value: mergedValue } = mergedValueRef
if (value === mergedValue) return
const { 'onUpdate:value': onUpdateValue, onChange } = props
const {
'onUpdate:value': _onUpdateValue,
onUpdateValue,
onChange
} = props
const { nTriggerFormInput, nTriggerFormChange } = formItem
if (onChange) call(onChange, value)
if (onUpdateValue) call(onUpdateValue, value)
if (_onUpdateValue) call(_onUpdateValue, value)
uncontrolledValueRef.value = value
nTriggerFormInput()
nTriggerFormChange()
}
const deriveValueFromDisplayedValue = (
offset = 0,
postUpdateIfValid = true
doUpdateIfValid = true
): null | number | false => {
const { value: displayedValue } = displayedValueRef
const parsedValue = parse(displayedValue)
if (parsedValue === null) return null
if (parsedValue === null) {
if (doUpdateIfValid) doUpdateValue(null)
return null
}
if (validator(parsedValue)) {
let nextValue = parsedValue + offset
if (validator(nextValue)) {
const { value: mergedMax } = mergedMaxRef
const { value: mergedMin } = mergedMinRef
if (mergedMax !== null && nextValue > mergedMax) {
if (!postUpdateIfValid) return false
if (!doUpdateIfValid) return false
nextValue = mergedMax
}
if (mergedMin !== null && nextValue < mergedMin) {
if (!postUpdateIfValid) return false
if (!doUpdateIfValid) return false
nextValue = mergedMin
}
if (props.validator && !props.validator(nextValue)) return false
if (postUpdateIfValid) doUpdateValue(nextValue)
if (doUpdateIfValid) doUpdateValue(nextValue)
return nextValue
}
}
Expand Down Expand Up @@ -307,6 +300,7 @@ export default defineComponent({
}
function handleUpdateDisplayedValue (value: string): void {
displayedValueRef.value = value
deriveValueFromDisplayedValue()
}
watch(mergedValueRef, () => {
deriveDisplayedValueFromValue()
Expand Down Expand Up @@ -360,7 +354,6 @@ export default defineComponent({
bordered={this.mergedBordered}
value={this.displayedValue}
onUpdateValue={this.handleUpdateDisplayedValue}
passively-activated
theme={this.mergedTheme.peers.Input}
themeOverrides={this.mergedTheme.peerOverrides.Input}
builtinThemeOverrides={this.inputThemeOverrides}
Expand Down
1 change: 1 addition & 0 deletions src/input-number/src/interface.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export type OnUpdateValue = (value: number | null) => void
17 changes: 0 additions & 17 deletions src/input-number/tests/InputNumber.spec.ts

This file was deleted.

58 changes: 58 additions & 0 deletions src/input-number/tests/InputNumber.spec.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
import { mount } from '@vue/test-utils'
import { NInputNumber } from '../index'
import { NButton } from '../../button'

describe('n-input-number', () => {
it('should work with import on demand', () => {
mount(NInputNumber)
})

it('should work with `show-button` prop', async () => {
const wrapper = mount(NInputNumber)
expect(wrapper.findComponent(NButton).exists()).toBe(true)

await wrapper.setProps({ showButton: false })
expect(wrapper.findComponent(NButton).exists()).toBe(false)
})

it('should work with default value', async () => {
const wrapper = mount(NInputNumber, {
props: {
defaultValue: 1
}
})
expect(wrapper.find('input').element.value).toEqual('1')
})

it('should not trigger update if value is same', async () => {
const onUpdateValue = jest.fn()
const wrapper = mount(NInputNumber, {
attachTo: document.body,
props: {
defaultValue: 1,
onUpdateValue
}
})
wrapper.find('input').element.value = ''
await wrapper.find('input').trigger('input')
expect(onUpdateValue).toHaveBeenCalledWith(null)
wrapper.unmount()
})

it('trigger focus & blur event', () => {
const onFocus = jest.fn()
const onBlur = jest.fn()
const wrapper = mount(NInputNumber, {
attachTo: document.body,
props: {
onFocus,
onBlur
}
})
wrapper.find('input').element.focus()
expect(onFocus).toHaveBeenCalledTimes(1)
wrapper.find('input').element.blur()
expect(onBlur).toHaveBeenCalledTimes(1)
wrapper.unmount()
})
})
8 changes: 0 additions & 8 deletions src/input/tests/Input.spec.ts

This file was deleted.

19 changes: 19 additions & 0 deletions src/input/tests/Input.spec.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import { mount } from '@vue/test-utils'
import { NInput } from '../index'

describe('n-input', () => {
it('should work with import on demand', () => {
mount(NInput)
})
it('should call input callbacks', async () => {
const onUpdateValue = jest.fn()
const wrapper = mount(NInput, {
props: {
onUpdateValue
}
})
wrapper.find('input').element.value = 'cool'
await wrapper.find('input').trigger('input')
expect(onUpdateValue).toHaveBeenCalledWith('cool')
})
})

0 comments on commit a91dd59

Please sign in to comment.