Двухстороннее редактируемое связывание Vuex - PullRequest
0 голосов
/ 14 декабря 2018

У меня есть список позиций заказа.Каждая позиция имеет название продукта, количество, цену, сумму.Также есть поля ввода с суммой предоплаты в процентах и ​​в абсолютах (например, в долларах).Я использую геттер для полной предоплаты.Таким образом, чтобы отредактировать абсолютное значение из входных данных и чтобы процентное значение было актуальным, я должен обновить его в хранилище через процентное значение, то есть мне нужно рассчитать процентное значение, в котором абсолютное значение равно новому(просто набрал в поле ввода).Но вот проблема: когда я набираю число в поле ввода абсолютного значения, вычисление процента вызывает пересчет абсолютного получателя, который возвращает каретку в начало (ОК, здесь можно использовать функцию debounce), нотакже я чувствую, что это неправильно, и какое-то логическое нарушение имеет место.

Извините за беспорядок.Код:

store.js:

Vue.use(Vuex);

let product = {
    order: 1,
    name: '',
    size: '',
    content: '',
    price: 0,
    number: 1,
    discount: '',
    guarantee: 0,
    promotion: 0,
    location: '',
    sum: 0,
};
export default new Vuex.Store({
    strict: true,
    state: {
        products: [
            Object.assign({}, product),
        ],
        form: {
            prepaymentInPercent: 100,
        },

    },
    getters: {
        total(state) {
            return +(state.products.length > 0 && state.products.reduce(function (acc, cur) {
                return acc + cur.sum;
            }, 0));
        },
        rest(state, getters) {
            return Math.round(getters.total - getters.prepaymentInRub);
        },
        prepaymentInRub(state, getters) {
            return Math.round(getters.total * state.form.prepaymentInPercent / 100);
        },
    },
    mutations: {
        [ADD_PRODUCT](state, product) {
            state.products.push(product);
        },
        [UPDATE_PRODUCT](state, {updatedProduct, index}) {
            state.products.splice(index, 1, updatedProduct);
        },
        [DELETE_PRODUCT](state, index) {
            state.products.splice(index, 1);
        },
        [UPDATE_FORM](state, updatedFormFields) {
            state.form = Object.assign({}, state.form, updatedFormFields);
        },
    },
    actions: {
        addProduct({commit}) {
            let newProduct = Object.assign({}, product, {'order': product.order++});
            commit(ADD_PRODUCT, newProduct);
        },
        updateProduct: _.debounce(function ({commit}, {product, index}) {
            let updatedProduct = Object.assign({}, product);
            commit(UPDATE_PRODUCT, {updatedProduct, index});
        }, 200),
        deleteProduct({commit, state}, index) {
            state.products.length > 1 && commit(DELETE_PRODUCT, index);
        },
        updatePrepaymentInRub({commit, getters}, rubles) {
            let prepaymentInPercent = Math.round(rubles / getters.total * 100);
            commit(UPDATE_FORM, {prepaymentInPercent});
        },
    },
});

и

OrderForm.vue

<template>
    <form>
        <label>Prepayment, %
            <input id="prepaymentInPercent" type="number" min="0" max="100" v-model="prepaymentInPercent">
        </label>
        <label>Prepayment, rubles
            <input id="prepaymentInRub" min="0" type="number" :max="total" v-model="prepaymentInRubles">
        </label>
    </form>
</template>

<script>
    import {UPDATE_FORM} from "../mutation-types";
    import {mapGetters} from 'vuex';

    export default {
        name: "OrderForm",
        computed: {
            'prepaymentInRubles': {
                get() {
                    return this.$store.getters.prepaymentInRub;
                },
                set(value) {
                    this.$store.dispatch('updatePrepaymentInRub', value);
                }
            },
            ...mapGetters(['total']),
            'prepaymentInPercent': {
                get() {
                    return this.$store.state.form.prepaymentInPercent;
                },
                set(value) {
                    return this.$store.commit(UPDATE_FORM, {
                        'prepaymentInPercent': value
                    });
                }
            }
        },
    }
</script>

ProductRow.vue

<template>
    <tr>
        <td colspan="2" class="id">{{indexFrom1}}</td>
        <Editable v-model="product.name"/>
        <Editable v-model="product.size"/>
        <Editable v-model="product.content"/>
        <Editable v-model.number="product.price"/>
        <Editable v-model.number="product.number"/>
        <Editable v-model="product.discount"/>
        <td>
            <select v-model.number="product.promotion">
                <option selected value="0">No</option>
                <optgroup label="forNewsettlers">
                    <option value="5">-5%</option>
                    <option value="10">-10%</option>
                    <option value="15">-15%</option>
                </optgroup>
            </select>
        </td>
        <td>{{sum}}</td>
        <Editable v-model.number="product.guarantee"/>
        <td>
            <select v-model="product.location">
                <option selected value="SERVICE">Service</option>
                <option value="STOREHOUSE">Storehouse</option>
                <option value="SO">Special Order</option>
                <option value="from 1st shop">exhibition sample from 1st shop</option>
            </select>
        </td>
        <td>
            <span class="table-remove" @click="removeProduct(index)">Удалить</span>
        </td>
    </tr>
</template>

<script>
    import Editable from './EditableCell';

    export default {
        components: {
            Editable,
        },
        name: 'ProductRow',
        props: {
            initialProduct: Object,
            index: Number,
        },
        data() {
            return {
                product: Object.assign({}, this.initialProduct),
            };

        },
        computed: {
            sum() {
                let discountedPrice = this.isDiscountInPercent(this.product.discount) ?
                    this.product.price * this.getCoeffFromPercent(this.product.discount) :
                    this.product.price - this.product.discount;
                let priceWithPromotion = discountedPrice * this.getCoeffFromPercent(this.product.promotion);
                let sum = Math.round(priceWithPromotion * this.product.number);
                return sum > 0 ? sum : 0;
            },
            indexFrom1() {
                return this.index + 1;
            },
        },
        methods: {
            getCoeffFromPercent(percent) {
                return 1 - parseInt(percent) / 100;
            },
            isDiscountInPercent(discount) {
                return ~discount.indexOf('%') ? true : false;
            },
            removeProduct(index) {
                this.$store.dispatch('deleteProduct', index);
            }
        },
        watch: {
            sum() {
                this.product.sum = this.sum;
            },
            product: {
                handler(product) {
                    this.$store.dispatch('updateProduct', {
                        product: product,
                        index: this.index,
                    });
                },
                deep: true,
            },
        },
    };
</script>

1 Ответ

0 голосов
/ 14 декабря 2018

Я думаю, что ваша логика в порядке.В состоянии есть одно значение, которое имеет два разных представления и возможные мутации.

Чтобы избежать проблем с курсором, я бы изменил событие, которое вы слушаете, на входах.Помните, что v-model="prepaymentInPercent" эквивалентно @input="prepaymentInPercent = $event.target.value" :value="prepaymentInPercent".Событие input срабатывает при каждом изменении символа, но вы можете заменить его на событие change , которое будет срабатывать при размытии или вводе.например, @change="prepaymentInPercent = $event.target.value" :value="prepaymentInPercent"

Оказывается, есть модификатор lazy , который делает это автоматически, так что вы можете просто добавить .lazy к вашей v-модели.

...