[bugfix] Search: incomplete text display in iOS (#974)

This commit is contained in:
neverland 2018-05-02 20:22:12 +08:00 committed by GitHub
parent 8e088fd0f5
commit 593b09a026
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 31 additions and 123 deletions

View File

@ -1,20 +1,14 @@
<template> <template>
<div <div :class="b({ 'show-action': showAction })" :style="{ background }">
:class="b({ 'show-action': showAction })" <icon name="search" />
:style="{ 'background-color': background }" <field
> v-bind="$attrs"
<div :class="b('wrap')" v-clickoutside="onClickoutside"> v-on="listeners"
<icon name="search" /> :value="value"
<input type="search"
v-bind="$attrs" icon="clear"
v-on="listeners" @click-icon="$emit('input', '')"
v-refocus="focusStatus" />
type="search"
:class="b('input')"
:value="value"
>
<icon name="clear" v-show="isFocus && value" @click="onClean" />
</div>
<div v-if="showAction" :class="b('action')" > <div v-if="showAction" :class="b('action')" >
<slot name="action"> <slot name="action">
<div :class="b('cancel')" @click="onBack">{{ $t('cancel') }}</div> <div :class="b('cancel')" @click="onBack">{{ $t('cancel') }}</div>
@ -24,14 +18,18 @@
</template> </template>
<script> <script>
import Field from '../field';
import create from '../utils/create'; import create from '../utils/create';
import Clickoutside from '../utils/clickoutside';
export default create({ export default create({
name: 'search', name: 'search',
inheritAttrs: false, inheritAttrs: false,
components: {
Field
},
props: { props: {
value: String, value: String,
showAction: Boolean, showAction: Boolean,
@ -41,29 +39,10 @@ export default create({
} }
}, },
data() {
return {
isFocus: false,
focusStatus: false
};
},
directives: {
Clickoutside,
refocus: {
update: function(el, state) {
if (state.value) {
el.focus();
}
}
}
},
computed: { computed: {
listeners() { listeners() {
return { return {
...this.$listeners, ...this.$listeners,
focus: this.onFocus,
input: this.onInput, input: this.onInput,
keypress: this.onKeypress keypress: this.onKeypress
}; };
@ -71,13 +50,8 @@ export default create({
}, },
methods: { methods: {
onFocus() { onInput(value) {
this.isFocus = true; this.$emit('input', value);
this.$emit('focus');
},
onInput(event) {
this.$emit('input', event.target.value);
}, },
onKeypress(event) { onKeypress(event) {
@ -89,25 +63,9 @@ export default create({
this.$emit('keypress', event); this.$emit('keypress', event);
}, },
// refocus after click close icon
onClean() {
this.$emit('input', '');
this.focusStatus = true;
// ensure refocus can work after click clean icon
this.$nextTick(() => {
this.focusStatus = false;
});
},
onBack() { onBack() {
this.$emit('input', ''); this.$emit('input', '');
this.$emit('cancel'); this.$emit('cancel');
},
onClickoutside() {
this.isFocus = false;
this.focusStatus = false;
} }
} }
}); });

View File

@ -5,36 +5,19 @@
align-items: center; align-items: center;
box-sizing: border-box; box-sizing: border-box;
padding: 4px 15px; padding: 4px 15px;
position: relative;
&--show-action { &--show-action {
padding-right: 0; padding-right: 0;
} }
&__wrap { .van-cell {
position: relative;
flex: 1; flex: 1;
height: 34px;
box-sizing: border-box;
padding: 8px 24px 8px 35px;
border: 1px solid $gray-light;
border-radius: 4px; border-radius: 4px;
background-color: $white; padding: 5px 10px 5px 35px;
} }
&__input { input {
display: block;
width: 100%;
height: 16px;
line-height: 16px;
padding: 0;
font-size: 14px;
color: $gray-darker;
border: none;
&::placeholder {
color: $gray-dark;
}
&::-webkit-search-decoration, &::-webkit-search-decoration,
&::-webkit-search-cancel-button, &::-webkit-search-cancel-button,
&::-webkit-search-results-button, &::-webkit-search-results-button,
@ -59,21 +42,16 @@
} }
.van-icon-search { .van-icon-search {
top: 50%;
left: 25px;
z-index: 1;
color: $gray-darker; color: $gray-darker;
position: absolute; position: absolute;
top: 50%;
transform: translateY(-50%);
left: 10px;
font-size: 16px; font-size: 16px;
transform: translateY(-50%);
} }
.van-icon-clear { .van-icon-clear {
font-size: 14px;
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
padding: 5px; /* increase click area */
color: $gray-dark; color: $gray-dark;
} }
} }

View File

@ -12,17 +12,6 @@ describe('Search', () => {
wrapper = mount(Search); wrapper = mount(Search);
expect(wrapper.hasClass('van-search')).to.be.true; expect(wrapper.hasClass('van-search')).to.be.true;
expect(wrapper.data().focusStatus).to.be.false;
expect(wrapper.data().isFocus).to.be.false;
});
it('focus on input', () => {
wrapper = mount(Search);
const input = wrapper.find('.van-search__input')[0];
input.trigger('focus');
expect(wrapper.data().isFocus).to.be.true;
}); });
it('create a search with searchText', (done) => { it('create a search with searchText', (done) => {
@ -33,7 +22,7 @@ describe('Search', () => {
}); });
wrapper.vm.$nextTick(() => { wrapper.vm.$nextTick(() => {
const input = wrapper.find('.van-search__input')[0]; const input = wrapper.find('input')[0];
expect(input.element.value === 'search text').to.be.true; expect(input.element.value === 'search text').to.be.true;
done(); done();
}); });
@ -42,7 +31,7 @@ describe('Search', () => {
it('emit input event', () => { it('emit input event', () => {
wrapper = mount(Search); wrapper = mount(Search);
const input = wrapper.find('.van-search__input')[0]; const input = wrapper.find('input')[0];
const eventStub = sinon.stub(wrapper.vm, '$emit'); const eventStub = sinon.stub(wrapper.vm, '$emit');
input.trigger('input', { target: { value: 'search' }}); input.trigger('input', { target: { value: 'search' }});
@ -65,14 +54,13 @@ describe('Search', () => {
inputSpy(); inputSpy();
}); });
const input = wrapper.find('.van-search__input')[0]; const input = wrapper.find('input')[0];
input.trigger('focus'); input.trigger('focus');
const cleanBtn = wrapper.find('.van-icon-clear')[0]; const cleanBtn = wrapper.find('.van-field__icon')[0];
cleanBtn.trigger('click'); cleanBtn.trigger('touchstart');
wrapper.vm.$nextTick(() => { wrapper.vm.$nextTick(() => {
expect(focusSpy.calledOnce).to.be.true;
expect(inputSpy.calledOnce).to.be.true; expect(inputSpy.calledOnce).to.be.true;
expect(value).to.equal(''); expect(value).to.equal('');
done(); done();
@ -91,8 +79,6 @@ describe('Search', () => {
cancelBtn.trigger('click'); cancelBtn.trigger('click');
wrapper.vm.$nextTick(() => { wrapper.vm.$nextTick(() => {
expect(wrapper.vm.focusStatus).to.be.false;
expect(wrapper.vm.isFocus).to.be.false;
expect(eventStub.calledTwice).to.be.true; expect(eventStub.calledTwice).to.be.true;
expect(eventStub.calledWith('input')); expect(eventStub.calledWith('input'));
expect(eventStub.calledWith('change')); expect(eventStub.calledWith('change'));
@ -106,7 +92,7 @@ describe('Search', () => {
const searchSpy = sinon.spy(); const searchSpy = sinon.spy();
wrapper.vm.$on('search', searchSpy); wrapper.vm.$on('search', searchSpy);
const input = wrapper.find('.van-search__input')[0]; const input = wrapper.find('input')[0];
input.trigger('keypress.enter'); input.trigger('keypress.enter');
expect(searchSpy.calledOnce).to.be.true; expect(searchSpy.calledOnce).to.be.true;
@ -115,18 +101,4 @@ describe('Search', () => {
input.trigger('keypress.a'); input.trigger('keypress.a');
expect(keypressSpy.calledOnce).to.be.true; expect(keypressSpy.calledOnce).to.be.true;
}); });
it('blur after click outside', () => {
wrapper = mount(Search);
const input = wrapper.find('.van-search__input')[0];
input.trigger('focus');
expect(wrapper.vm.isFocus).to.be.true;
const body = document.body;
body.click();
expect(wrapper.vm.isFocus).to.be.false;
expect(wrapper.vm.focusStatus).to.be.false;
});
}); });