[Improvement] Sku: optimize DOM (#704)

This commit is contained in:
neverland 2018-03-16 20:27:59 +08:00 committed by GitHub
parent ffd72e5442
commit 4c195fd664
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
26 changed files with 228 additions and 265 deletions

View File

@ -33,6 +33,7 @@
:quota="$t('sku').quota" :quota="$t('sku').quota"
:quota-used="$t('sku').quota_used" :quota-used="$t('sku').quota_used"
:custom-stepper-config="customStepperConfig" :custom-stepper-config="customStepperConfig"
:message-config="messageConfig"
@buy-clicked="onBuyClicked" @buy-clicked="onBuyClicked"
@add-cart="onAddCartClicked" @add-cart="onAddCartClicked"
/> />
@ -54,6 +55,7 @@
show-add-cart-btn show-add-cart-btn
reset-stepper-on-hide reset-stepper-on-hide
:initial-sku="initialSku" :initial-sku="initialSku"
:message-config="messageConfig"
@buy-clicked="onBuyClicked" @buy-clicked="onBuyClicked"
@add-cart="onAddCartClicked" @add-cart="onAddCartClicked"
> >
@ -119,9 +121,9 @@ export default {
} }
}, },
messageConfig: { messageConfig: {
uploadImg: () => { uploadImg: (file, img) => {
return new Promise((resolve) => { return new Promise(resolve => {
setTimeout(() => resolve('https://img.yzcdn.cn/upload_files/2017/02/21/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg!100x100.jpg'), 1000); setTimeout(() => resolve(img), 1000);
}); });
}, },
uploadMaxSize: 3 uploadMaxSize: 3

View File

@ -58,10 +58,10 @@
/> />
</cell-group> </cell-group>
<div v-show="!hideBottomFields" class="van-address-edit__buttons"> <div v-show="!hideBottomFields" class="van-address-edit__buttons">
<van-button block :loading="isSaving" @click="onSaveAddress" type="primary"> <van-button block :loading="isSaving" @click="onSave" type="primary">
{{ $t('save') }} {{ $t('save') }}
</van-button> </van-button>
<van-button block :loading="isDeleting" @click="onDeleteAddress" v-if="isEdit"> <van-button block :loading="isDeleting" @click="onDelete" v-if="isEdit">
{{ $t('deleteAddress', computedAddressText) }} {{ $t('deleteAddress', computedAddressText) }}
</van-button> </van-button>
</div> </div>
@ -108,10 +108,10 @@ export default create({
components: { components: {
Field, Field,
SwitchCell,
VanButton,
Popup, Popup,
VanArea, VanArea,
VanButton,
SwitchCell,
AddressEditDetail AddressEditDetail
}, },
@ -223,7 +223,7 @@ export default create({
}); });
}, },
onSaveAddress() { onSave() {
const items = [ const items = [
'name', 'name',
'tel', 'tel',
@ -267,7 +267,7 @@ export default create({
} }
}, },
onDeleteAddress() { onDelete() {
if (this.isDeleting) { if (this.isDeleting) {
return; return;
} }

View File

@ -13,10 +13,10 @@
</radio-group> </radio-group>
<cell <cell
icon="add" icon="add"
class="van-address-list__add van-hairline--top"
@click="$emit('add')"
:title="addButtonText || $t('add')"
is-link is-link
class="van-address-list__add van-hairline--top"
:title="addButtonText || $t('add')"
@click="$emit('add')"
/> />
</div> </div>
</template> </template>

View File

@ -5,11 +5,9 @@
</template> </template>
<script> <script>
import install from '../utils/install'; import create from '../utils/create-basic';
export default {
install,
export default create({
name: 'cell-group', name: 'cell-group',
props: { props: {
@ -18,5 +16,5 @@ export default {
default: true default: true
} }
} }
}; });
</script> </script>

View File

@ -38,12 +38,10 @@
<script> <script>
import Icon from '../icon'; import Icon from '../icon';
import install from '../utils/install';
import RouterLink from '../mixins/router-link'; import RouterLink from '../mixins/router-link';
import create from '../utils/create-basic';
export default { export default create({
install,
name: 'cell', name: 'cell',
components: { components: {
@ -72,5 +70,5 @@ export default {
this.routerLink(); this.routerLink();
} }
} }
}; });
</script> </script>

View File

@ -13,10 +13,10 @@
</radio-group> </radio-group>
<cell <cell
icon="add" icon="add"
class="van-contact-list__add van-hairline--top"
@click="$emit('add')"
:title="addText || $t('addText')"
is-link is-link
class="van-contact-list__add van-hairline--top"
:title="addText || $t('addText')"
@click="$emit('add')"
/> />
</div> </div>
</template> </template>

View File

@ -6,16 +6,14 @@
</template> </template>
<script> <script>
import install from '../utils/install'; import create from '../utils/create-basic';
export default {
install,
export default create({
name: 'icon', name: 'icon',
props: { props: {
name: String, name: String,
info: String info: String
} }
}; });
</script> </script>

View File

@ -10,11 +10,9 @@
</template> </template>
<script> <script>
import install from '../utils/install'; import create from '../utils/create-basic';
export default {
install,
export default create({
name: 'loading', name: 'loading',
props: { props: {
@ -37,5 +35,5 @@ export default {
} : {}; } : {};
} }
} }
}; });
</script> </script>

View File

@ -3,11 +3,10 @@
v-if="!isSkuEmpty" v-if="!isSkuEmpty"
v-model="show" v-model="show"
position="bottom" position="bottom"
class="van-sku-container"
:close-on-click-overlay="closeOnClickOverlay" :close-on-click-overlay="closeOnClickOverlay"
:get-container="getContainer" :get-container="getContainer"
> >
<div class="van-sku-container">
<div class="van-sku-layout">
<!-- sku-header --> <!-- sku-header -->
<slot <slot
name="sku-header" name="sku-header"
@ -29,12 +28,9 @@
<!-- sku-group --> <!-- sku-group -->
<slot name="sku-group" :selected-sku="selectedSku" :sku-event-bus="skuEventBus"> <slot name="sku-group" :selected-sku="selectedSku" :sku-event-bus="skuEventBus">
<div v-if="hasSku" class="van-sku-group-container van-hairline--bottom"> <div v-if="hasSku" class="van-sku-group-container van-hairline--bottom">
<div
v-for="(skuTreeItem, index) in skuTree"
class="van-sku-row-group"
:key="index">
<sku-row <sku-row
:sku-event-bus="skuEventBus" v-for="(skuTreeItem, index) in skuTree"
:key="index"
:sku-row="skuTreeItem" :sku-row="skuTreeItem"
> >
<sku-row-item <sku-row-item
@ -48,7 +44,6 @@
/> />
</sku-row> </sku-row>
</div> </div>
</div>
</slot> </slot>
<!-- extra-sku-group --> <!-- extra-sku-group -->
<slot name="extra-sku-group" :sku-event-bus="skuEventBus"/> <slot name="extra-sku-group" :sku-event-bus="skuEventBus"/>
@ -93,8 +88,6 @@
:show-add-cart-btn="showAddCartBtn" :show-add-cart-btn="showAddCartBtn"
/> />
</slot> </slot>
</div>
</div>
</popup> </popup>
</template> </template>
@ -225,9 +218,8 @@ export default create({
return; return;
} }
const windowHeight = window.innerHeight;
// header82px, sku actions50pxbodyOffsetTop // header82px, sku actions50pxbodyOffsetTop
const maxHeight = windowHeight - this.bodyOffsetTop; const maxHeight = window.innerHeight - this.bodyOffsetTop;
return { return {
maxHeight: maxHeight + 'px' maxHeight: maxHeight + 'px'
@ -268,12 +260,12 @@ export default create({
const skuEventBus = new Vue(); const skuEventBus = new Vue();
this.skuEventBus = skuEventBus; this.skuEventBus = skuEventBus;
skuEventBus.$on('sku:close', this.onCloseClicked); skuEventBus.$on('sku:close', this.onClose);
skuEventBus.$on('sku:select', this.onSkuSelected); skuEventBus.$on('sku:select', this.onSelect);
skuEventBus.$on('sku:numChange', this.onNumChange); skuEventBus.$on('sku:numChange', this.onNumChange);
skuEventBus.$on('sku:overLimit', this.onOverLimit); skuEventBus.$on('sku:overLimit', this.onOverLimit);
skuEventBus.$on('sku:addCart', this.onAddCartClicked); skuEventBus.$on('sku:addCart', this.onAddCart);
skuEventBus.$on('sku:buy', this.onBuyClicked); skuEventBus.$on('sku:buy', this.onBuy);
this.resetSelectedSku(this.skuTree); this.resetSelectedSku(this.skuTree);
// skuEventBus // skuEventBus
@ -315,19 +307,17 @@ export default create({
} }
if (this.isSkuCombSelected) { if (this.isSkuCombSelected) {
const error = this.validateSkuMessages(); return this.validateSkuMessages();
// sku
return error;
} else {
return this.$t('spec');
} }
return this.$t('spec');
}, },
onCloseClicked() { onClose() {
this.show = false; this.show = false;
}, },
onSkuSelected(skuValue) { onSelect(skuValue) {
// sku // sku
this.selectedSku = this.selectedSku =
this.selectedSku[skuValue.skuKeyStr] === skuValue.id this.selectedSku[skuValue.skuKeyStr] === skuValue.id
@ -367,11 +357,11 @@ export default create({
} }
}, },
onAddCartClicked() { onAddCart() {
this.onBuyOrAddCart('add-cart'); this.onBuyOrAddCart('add-cart');
}, },
onBuyClicked() { onBuy() {
this.onBuyOrAddCart('buy-clicked'); this.onBuyOrAddCart('buy-clicked');
}, },

View File

@ -1,12 +1,14 @@
<template> <template>
<div class="van-sku-header van-hairline--bottom"> <div class="van-sku-header van-hairline--bottom">
<div class="van-sku-header__img-wrap"> <div class="van-sku-header__img-wrap">
<img class="van-sku__goods-img" :src="goodsImg" > <img :src="goodsImg" >
</div> </div>
<div class="van-sku-header__goods-info"> <div class="van-sku-header__goods-info">
<div class="van-sku__goods-name van-ellipsis">{{ goods.title }}</div> <div class="van-sku__goods-name van-ellipsis">{{ goods.title }}</div>
<div class="van-sku__goods-price"><span class="van-sku__price-symbol"></span><span class="van-sku__price-num">{{ price }}</span></div> <div class="van-sku__goods-price">
<span class="van-sku__close-icon" @click="skuEventBus.$emit('sku:close')" /> <span class="van-sku__price-symbol"></span><span class="van-sku__price-num">{{ price }}</span>
</div>
<icon name="close" class="van-sku__close-icon" @click="skuEventBus.$emit('sku:close')" />
</div> </div>
</div> </div>
</template> </template>
@ -18,11 +20,11 @@ export default create({
name: 'sku-header', name: 'sku-header',
props: { props: {
skuEventBus: Object,
sku: Object, sku: Object,
goods: Object,
skuEventBus: Object,
selectedSku: Object, selectedSku: Object,
selectedSkuComb: Object, selectedSkuComb: Object
goods: Object
}, },
computed: { computed: {
@ -32,6 +34,7 @@ export default create({
// 使sku // 使sku
return skuImg || this.goods.picture; return skuImg || this.goods.picture;
}, },
price() { price() {
if (this.selectedSkuComb) { if (this.selectedSkuComb) {
return (this.selectedSkuComb.price / 100).toFixed(2); return (this.selectedSkuComb.price / 100).toFixed(2);

View File

@ -2,56 +2,51 @@
<div class="van-sku-img-uploader"> <div class="van-sku-img-uploader">
<!-- 头部 --> <!-- 头部 -->
<van-uploader <van-uploader
:disabled="!canUpload" :disabled="!!paddingImg"
:before-read="beforeReadFile"
:after-read="afterReadFile" :after-read="afterReadFile"
accept="image/*"> :max-size="maxSize * 1024 * 1024"
accept="image/*"
@oversize="$toast($t('maxSize', maxSize))"
>
<div class="van-sku-img-uploader__header"> <div class="van-sku-img-uploader__header">
<div v-if="paddingImg">{{ $t('uploading') }}</div> <div v-if="paddingImg">{{ $t('uploading') }}</div>
<template v-else> <template v-else>
<van-icon name="photograph" /> <icon name="photograph" />
<span class="label">{{ getPhotoText(value) }}</span> {{ $t('or') }} <span class="label">{{ $t(value ? 'rephoto' : 'photo') }}</span> {{ $t('or') }}
<van-icon name="photo" /> <icon name="photo" />
<span class="label">{{ getPicText(value) }}</span> <span class="label">{{ $t(value ? 'reselect' : 'select') }}</span>
</template> </template>
</div> </div>
</van-uploader> </van-uploader>
<!-- 图片列表区域 --> <!-- 图片列表区域 -->
<div class="van-sku-img-uploader__imglist" v-if="paddingImg || imgList.length > 0"> <div class="van-clearfix" v-if="paddingImg || imgList.length > 0">
<!-- 已有的图片,图片右上角显示删除按钮 --> <!-- 已有的图片,图片右上角显示删除按钮 -->
<div <div
v-for="(img, index) in imgList" v-for="(img, index) in imgList"
:key="index" :key="index"
class="van-sku-img-uploader__img-container"> class="van-sku-img-uploader__img"
<span class="van-sku-img-uploader__delete-picture" @click="deleteImg(index)"> >
<van-icon name="clear" />
</span>
<img :src="img"> <img :src="img">
<icon name="clear" class="van-sku-img-uploader__delete" @click="$emit('input', '')" />
</div> </div>
<!-- 正在上传的图片,有上传等待提示 --> <!-- 正在上传的图片,有上传等待提示 -->
<div <div v-if="paddingImg" class="van-sku-img-uploader__img">
v-if="paddingImg"
class="van-sku-img-uploader__img-container">
<img :src="paddingImg"> <img :src="paddingImg">
<van-loading class="van-sku-img-uploader__uploading" type="spinner" color="black" /> <loading class="van-sku-img-uploader__uploading" type="spinner" color="black" />
</div> </div>
</div> </div>
</div> </div>
</template> </template>
<script> <script>
import Icon from '../../icon'; import VanUploader from '../../uploader';
import Uploader from '../../uploader';
import Loading from '../../loading';
import { create } from '../../utils'; import { create } from '../../utils';
export default create({ export default create({
name: 'sku-img-uploader', name: 'sku-img-uploader',
components: { components: {
'van-uploader': Uploader, VanUploader
'van-icon': Icon,
'van-loading': Loading
}, },
props: { props: {
@ -72,50 +67,25 @@ export default create({
paddingImg: '' paddingImg: ''
}; };
}, },
computed: { computed: {
imgList() { imgList() {
return this.value && !this.paddingImg ? [this.value] : []; return this.value && !this.paddingImg ? [this.value] : [];
},
canUpload() {
//
if (this.paddingImg) return false;
return true;
} }
}, },
methods: { methods: {
getPhotoText(value) {
return value ? this.$t('rephoto') : this.$t('photo');
},
getPicText(value) {
return value ? this.$t('reselect') : this.$t('select');
},
beforeReadFile(file) {
//
if (file.size > this.maxSize * 1024 * 1024) {
Toast(this.$t('maxSize', this.maxSize));
return false;
}
return true;
},
afterReadFile(file) { afterReadFile(file) {
// //
this.paddingImg = file.content; this.paddingImg = file.content;
this.uploadImg(file.file).then(img => { this.uploadImg(file.file, file.content).then(img => {
this.updateImg(img); this.$emit('input', img);
this.$nextTick(() => { this.$nextTick(() => {
this.paddingImg = ''; this.paddingImg = '';
}); });
}).catch(() => { }).catch(() => {
this.paddingImg = ''; this.paddingImg = '';
}); });
},
deleteImg() {
this.$emit('input', '');
},
updateImg(img) {
this.$emit('input', img);
} }
} }
}); });

View File

@ -7,7 +7,8 @@
:label="$t('onePic')" :label="$t('onePic')"
:key="`${goodsId}-${index}`" :key="`${goodsId}-${index}`"
:required="message.required == '1'" :required="message.required == '1'"
:title="message.name"> :title="message.name"
>
<sku-img-uploader <sku-img-uploader
v-model="messageValues[index].value" v-model="messageValues[index].value"
:upload-img="messageConfig.uploadImg" :upload-img="messageConfig.uploadImg"
@ -70,6 +71,7 @@ export default create({
resetMessageValues(messages) { resetMessageValues(messages) {
return (messages || []).map(() => ({ value: '' })); return (messages || []).map(() => ({ value: '' }));
}, },
getType(message) { getType(message) {
if (+message.multiple === 1) { if (+message.multiple === 1) {
return 'textarea'; return 'textarea';

View File

@ -1,10 +1,8 @@
<template> <template>
<div class="van-sku-row"> <div class="van-sku-row">
<div class="van-sku-row__title">{{ skuRow.k }}</div> <div class="van-sku-row__title">{{ skuRow.k }}</div>
<div class="van-sku-row__items">
<slot /> <slot />
</div> </div>
</div>
</template> </template>
<script> <script>

View File

@ -5,7 +5,7 @@
'van-sku-row__item--active': isChoosed, 'van-sku-row__item--active': isChoosed,
'van-sku-row__item--disabled': !isChoosable 'van-sku-row__item--disabled': !isChoosable
}" }"
@click="onSkuSelected" @click="onSelect"
> >
{{ skuValue.name }} {{ skuValue.name }}
</span> </span>
@ -41,14 +41,14 @@ export default create({
return matchedSku[skuKey] == sku[skuKey]; // eslint-disable-line return matchedSku[skuKey] == sku[skuKey]; // eslint-disable-line
}); });
}); });
const stock = filteredSku.reduce((total, sku) => (total += sku.stock_num), 0);
const stock = filteredSku.reduce((total, sku) => (total += sku.stock_num), 0);
return stock > 0; return stock > 0;
} }
}, },
methods: { methods: {
onSkuSelected() { onSelect() {
if (this.isChoosable) { if (this.isChoosable) {
this.skuEventBus.$emit('sku:select', { this.skuEventBus.$emit('sku:select', {
...this.skuValue, ...this.skuValue,

View File

@ -57,6 +57,7 @@ export default create({
currentNum(num) { currentNum(num) {
this.skuEventBus.$emit('sku:numChange', num); this.skuEventBus.$emit('sku:numChange', num);
}, },
stepperLimit(limit) { stepperLimit(limit) {
if (limit < this.currentNum) { if (limit < this.currentNum) {
this.currentNum = limit; this.currentNum = limit;
@ -71,6 +72,7 @@ export default create({
} }
return this.skuStockNum; return this.skuStockNum;
}, },
quotaText() { quotaText() {
const { quotaText } = this.customStepperConfig; const { quotaText } = this.customStepperConfig;
let text = ''; let text = '';
@ -83,6 +85,7 @@ export default create({
return text; return text;
}, },
stepperLimit() { stepperLimit() {
const quotaLimit = this.quota - this.quotaUsed; const quotaLimit = this.quota - this.quotaUsed;
let limit; let limit;
@ -105,6 +108,7 @@ export default create({
setCurrentNum(num) { setCurrentNum(num) {
this.currentNum = num; this.currentNum = num;
}, },
onOverLimit(action) { onOverLimit(action) {
this.skuEventBus.$emit('sku:overLimit', { this.skuEventBus.$emit('sku:overLimit', {
action, action,
@ -113,6 +117,7 @@ export default create({
quotaUsed: this.quotaUsed quotaUsed: this.quotaUsed
}); });
}, },
onChange(currentValue) { onChange(currentValue) {
const { handleStepperChange } = this.customStepperConfig; const { handleStepperChange } = this.customStepperConfig;
handleStepperChange && handleStepperChange(currentValue); handleStepperChange && handleStepperChange(currentValue);

View File

@ -43,7 +43,7 @@ export const normalizeSkuTree = (skuTree) => {
// 判断是否所有的sku都已经选中 // 判断是否所有的sku都已经选中
export const isAllSelected = (skuTree, selectedSku) => { export const isAllSelected = (skuTree, selectedSku) => {
// 筛选selectedSku对象中key值不为空的值 // 筛选selectedSku对象中key值不为空的值
const selected = Object.keys(selectedSku).filter(skuKeyStr => selectedSku[skuKeyStr] !== ''); const selected = Object.keys(selectedSku).filter(skuKeyStr => selectedSku[skuKeyStr]);
return skuTree.length === selected.length; return skuTree.length === selected.length;
}; };
@ -53,8 +53,8 @@ export const getSkuComb = (skuList, selectedSku) => {
return Object.keys(selectedSku).every(skuKeyStr => { return Object.keys(selectedSku).every(skuKeyStr => {
return String(skuComb[skuKeyStr]) === String(selectedSku[skuKeyStr]); // eslint-disable-line return String(skuComb[skuKeyStr]) === String(selectedSku[skuKeyStr]); // eslint-disable-line
}); });
})[0]; });
return skuComb; return skuComb[0];
}; };
// 获取已选择的sku名称 // 获取已选择的sku名称
@ -72,10 +72,9 @@ export const getSelectedSkuValues = (skuTree, selectedSku) => {
}, []); }, []);
}; };
const SkuHelper = { export default {
normalizeSkuTree, normalizeSkuTree,
isAllSelected, isAllSelected,
getSkuComb, getSkuComb,
getSelectedSkuValues getSelectedSkuValues
}; };
export default SkuHelper;

View File

@ -0,0 +1,18 @@
/**
* Create a basic component with common options
*/
import '../locale';
import i18n from '../mixins/i18n';
const install = function(Vue) {
Vue.component(this.name, this);
};
export default function(sfc) {
sfc.name = 'van-' + sfc.name;
sfc.install = sfc.install || install;
sfc.mixins = sfc.mixins || [];
sfc.mixins.push(i18n);
return sfc;
};

View File

@ -1,25 +1,18 @@
/** /**
* Create a component with common options * Create a component with common options
*/ */
import '../locale'; import createBasic from './create-basic';
import i18n from '../mixins/i18n';
import install from './install';
import Icon from '../icon'; import Icon from '../icon';
import Loading from '../loading'; import Loading from '../loading';
import Cell from '../cell'; import Cell from '../cell';
import CellGroup from '../cell-group'; import CellGroup from '../cell-group';
export default function(sfc) { export default function(sfc) {
sfc.name = 'van-' + sfc.name;
sfc.install = sfc.install || install;
sfc.mixins = sfc.mixins || [];
sfc.mixins.push(i18n);
sfc.components = Object.assign(sfc.components || {}, { sfc.components = Object.assign(sfc.components || {}, {
Icon, Icon,
Loading, Loading,
Cell, Cell,
CellGroup CellGroup
}); });
return createBasic(sfc);
return sfc;
}; };

View File

@ -1,6 +0,0 @@
/**
* Install function to register a component
*/
export default function(Vue) {
Vue.component(this.name, this);
}

View File

@ -1,3 +1,6 @@
/**
* VNode helper
*/
export default { export default {
name: 'van-node', name: 'van-node',
functional: true, functional: true,

View File

@ -5,6 +5,7 @@
@import "./common/var.css"; @import "./common/var.css";
@import "./common/normalize.css"; @import "./common/normalize.css";
@import "./common/ellipsis.css"; @import "./common/ellipsis.css";
@import "./common/clearfix.css";
@import "./common/hairline.css"; @import "./common/hairline.css";
@import "./common/animation.css"; @import "./common/animation.css";
@import './icon.css'; @import './icon.css';

View File

@ -0,0 +1,5 @@
@import '../mixins/clearfix.css';
.van-clearfix {
@mixin ellipsis;
}

View File

@ -3,10 +3,6 @@
.van-sku { .van-sku {
&-container { &-container {
background: $background-color;
}
&-layout {
background: $white; background: $white;
} }
@ -25,14 +21,6 @@
padding: 12px 0 2px; padding: 12px 0 2px;
} }
&-row-group {
margin: 0 15px 10px 0;
&:last-child {
margin-bottom: 0;
}
}
/* sku header */ /* sku header */
&-header { &-header {
margin-left: 15px; margin-left: 15px;
@ -46,7 +34,7 @@
background: $background-color; background: $background-color;
border-radius: 2px; border-radius: 2px;
.van-sku__goods-img { img {
position: absolute; position: absolute;
margin: auto; margin: auto;
top: 0; top: 0;
@ -87,19 +75,22 @@
} }
&__close-icon { &__close-icon {
top: 10px;
right: 15px;
font-size: 20px;
color: $gray-dark;
position: absolute; position: absolute;
top: 0; text-align: center;
right: 0;
width: 44px;
height: 44px;
background: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACwAAAAsCAMAAAApWqozAAAAUVBMVEUAAACfn5+bm5uampqZmZmoqKiampqampqZmZmampqampqbm5ubm5uZmZmqqqqampqampqZmZmampqampqampqbm5ubm5uampqdnZ2ampqZmZkjZXmqAAAAGnRSTlMAFYBC5wm+rdPv1Ckl8wzZxKWicm9eUjoN8nSyS9UAAAFVSURBVDjLvZXbrsIgEEWhtAUU6M1a3f//oSeHWKfuVhv7IC+TkJVhz4UZ9bsThq5uYmzqbgifSV04LI4r9Fs0FRHw/WhCugUz9h6IRdpmryVgzfLGWKC8brEF0E58ObVAsUIrC1y2fFwAW9GdhTfb6oyHZQ3+/C7us39VcgXE79o3sIgylayXdZdpIaJVH08rQnTE9BmeEPXsWMK1rV6m01Xz/dO1g5H3nH6yNcoZNnCPPoOXTjplemYlnR4h2wG9YppZ1WPItsOomGZWjeiyrWEU08wqgzrbBkERfWJWBTTZRiTFNLMqIWZ7x43atQYa+lA33EmG5KyB0yyDApT8/uvWFCCnTvIrtKRuLgqzQktRpNzMCi3l5kaymZXqiGS3blGX2Zl2ftGi6+avqtcsUvN/8a2++LCHRsH+kDkwvg4Mxt2Re3yY85pIsiaOLKD91faz8wcoUxux/aS9awAAAABJRU5ErkJggg==);
background-size: 22px 22px;
background-repeat: no-repeat;
background-position: 7px 10px;
} }
/* sku row */ /* sku row */
&-row { &-row {
margin: 0 15px 10px 0;
&:last-child {
margin-bottom: 0;
}
&__title { &__title {
font-size: 14px; font-size: 14px;
padding-bottom: 10px; padding-bottom: 10px;
@ -119,13 +110,13 @@
border-radius: 3px; border-radius: 3px;
box-sizing: border-box; box-sizing: border-box;
&.van-sku-row__item--active { &--active {
color: $white; color: $white;
border-color: $red; border-color: $red;
background: $red; background: $red;
} }
&.van-sku-row__item--disabled { &--disabled {
background: $active-color; background: $active-color;
border-color: $gray-light; border-color: $gray-light;
color: $gray; color: $gray;
@ -147,8 +138,6 @@
} }
&__stepper { &__stepper {
top: 7px;
left: 4px;
float: right; float: right;
&-title { &-title {
@ -172,6 +161,9 @@
} }
&-messages { &-messages {
padding-bottom: 10px;
background: $background-color;
&__image-cell { &__image-cell {
.van-cell__title { .van-cell__title {
width: 90px; width: 90px;
@ -187,7 +179,7 @@
&__header { &__header {
padding: 0 10px; padding: 0 10px;
border: 1px solid #e5e5e5; border: 1px solid $gray-light;
line-height: 24px; line-height: 24px;
border-radius: 3px; border-radius: 3px;
font-size: 12px; font-size: 12px;
@ -199,18 +191,13 @@
} }
} }
&__imglist { &__img {
@mixin clearfix;
}
&__img-container {
height: 60px; height: 60px;
width: 60px; width: 60px;
margin-top: 10px;
margin-right: 10px;
float: left; float: left;
margin: 10px 10px 0 0;
position: relative; position: relative;
border: #e5e5e5 1px solid; border: $gray-light 1px solid;
img { img {
max-width: 100%; max-width: 100%;
@ -221,14 +208,19 @@
} }
} }
&__delete-picture { &__delete {
position: absolute; position: absolute;
color: $red; color: $red;
top: -10px; top: -12px;
right: -17px; right: -14px;
z-index: 1; z-index: 1;
width: 22px; padding: 6px;
height: 22px;
&::before {
font-size: 14px;
border-radius: 14px;
background-color: $white;
}
} }
&__uploading { &__uploading {
@ -246,6 +238,5 @@
/* sku actions */ /* sku actions */
&-actions { &-actions {
display: flex; display: flex;
margin-top: 10px;
} }
} }

View File

@ -1,11 +1,8 @@
@import "./mixins/ellipsis.css";
@import "./mixins/clearfix.css";
@import './common/var.css'; @import './common/var.css';
.van-tree-select { .van-tree-select {
user-select: none; user-select: none;
position: relative; position: relative;
@mixin clearfix;
&__nav { &__nav {
width: 143px; width: 143px;

View File

@ -262,7 +262,7 @@ describe('AddressEdit', () => {
const deleteButton = wrapper.find('.van-button')[1]; const deleteButton = wrapper.find('.van-button')[1];
deleteButton.trigger('click'); deleteButton.trigger('click');
wrapper.vm.onDeleteAddress(); wrapper.vm.onDelete();
setTimeout(() => { setTimeout(() => {
wrapper.vm.isDeleting = false; wrapper.vm.isDeleting = false;

View File

@ -55,7 +55,7 @@ describe('Sku', (done) => {
'.van-stepper__input': '1' '.van-stepper__input': '1'
}, },
src: { src: {
'.van-sku__goods-img': 'https://img.yzcdn.cn/upload_files/2017/02/21/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg!100x100.jpg' '.van-sku-header__img-wrap img': 'https://img.yzcdn.cn/upload_files/2017/02/21/FjKTOxjVgnUuPmHJRdunvYky9OHP.jpg!100x100.jpg'
} }
}); });
@ -69,7 +69,7 @@ describe('Sku', (done) => {
wrapper.vm.$nextTick(() => { wrapper.vm.$nextTick(() => {
DOMChecker(wrapper, { DOMChecker(wrapper, {
src: { src: {
'.van-sku__goods-img': 'https://img.yzcdn.cn/upload_files/2017/03/16/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg?imageView2/2/w/100/h/100/q/75/format/jpg' '.van-sku-header__img-wrap img': 'https://img.yzcdn.cn/upload_files/2017/03/16/Fs_OMbSFPa183sBwvG_94llUYiLa.jpeg?imageView2/2/w/100/h/100/q/75/format/jpg'
} }
}); });
@ -111,7 +111,7 @@ describe('Sku', (done) => {
expect(buyCallback.calledOnce).to.be.false; expect(buyCallback.calledOnce).to.be.false;
// 选择完整规格时未填留言时弹出toast提示。 // 选择完整规格时未填留言时弹出toast提示。
wrapper.find('.van-sku-row-group')[1].find('.van-sku-row__item')[0].trigger('click'); wrapper.find('.van-sku-row')[1].find('.van-sku-row__item')[0].trigger('click');
buyBtn.trigger('click'); buyBtn.trigger('click');
wrapper.vm.$nextTick(() => { wrapper.vm.$nextTick(() => {
expect(toastText.textContent).to.equal('请填写留言1'); expect(toastText.textContent).to.equal('请填写留言1');