Серия исходных кодов Vue: простая реализация исходного кода Vuex

Vuex

Всем привет, я Lin Sanxin, новичок во фронтенде, сегодня я поделюсь с вами простой реализацией исходного кода Vuex.

На этот раз реализованы только состояние, геттеры, мутации, действия. На самом деле составом vuex является метод установки и класс хранилища, а в центре внимания находится миксин в установке, который является ядром vuex.

Это каталог файлов

vuex.js

// vuex.js
let Vue;

const install = (v) => {
    console.log(v)
    Vue = v;
    Vue.mixin({
        beforeCreate() {
            if (this.$options && this.$options.store) {
                // root
                this.$store = this.$options.store;
            } else {
                this.$store = this.$parent && this.$parent.$store;
            }
            console.log(this.$store);
        },
    })
}

class Store {
    constructor(options) {
        this.vm = new Vue({
            data: {
                state: options.state
            }
        });

        let getters = options.getters || {};
        this.getters = {};
        console.log(Object.keys(this.getters))
        Object.keys(getters).forEach(getterName => {
            Object.defineProperty(this.getters, getterName, {
                get: () => {
                    return getters[getterName](this.state);
                }
            })
        })

        let mutations = options.mutations || {};
        this.mutations = {};
        Object.keys(mutations).forEach(mutationName => {
            this.mutations[mutationName] = payload => {
                mutations[mutationName](this.state, payload);
            }
        })

        let actions = options.actions || {};
        this.actions = {};
        Object.keys(actions).forEach(actionName => {
            this.actions[actionName] = payload => {
                actions[actionName](this.state, payload);
            }
        })
    }
    get state() {
        return this.vm.state;
    }
    commit(name, payload) {
        this.mutations[name](payload);
    }
    dispatch(name, payload) {
        this.actions[name](payload);
    }
}

export default {
    install,
    Store
}

index.js

// index.js
import Vue from 'vue';
import vuex from './vuex';
Vue.use(vuex);


export default new vuex.Store({
    state: {
        num: 1
    },
    getters: {
        getNum(state) {
            return state.num * 2;
        }
    },
    mutations: { in (state, payload) {
            state.num += payload;
        },
        de(state, payload) {
            state.num -= payload;
        }
    },
    actions: { in (state, payload) {
            setTimeout(() => {
                state.num += payload;
            }, 2000)
        }
    }
})

main.js

// main.js
import Vue from 'vue';
import App from './App.vue'

import store from './store/index';


new Vue({
    store,
    el: '#app',
    components: {
        App
    },
    template: '<App/>',
})
let linsanxin = 1;
console.log(linsanxin)

App.vue

// App.vue
<template>
  <div>
    {{this.$store.state.num}}
    {{this.$store.getters.getNum}}
    <button @click="addNum">呵呵</button>
    <button @click="addNumByAsync">呵呵异步</button>
    <div id="a">asadasd</div>
    <button @click='add'>添加</button>
    <ul v-for="(item,index) in list" :key="index">
      <li class="ll">{{item.name}}</li>
    </ul>
  </div>
</template>

<script>
import img from "./asset/images/haha.jpg";
import "./asset/styles/test.css";
import "./asset/styles/global.stylus";
export default {
  name: "App",
  created() {
    console.log(img);
  },
  data() {
    return {
      list: [
        { name: "lin" },
        { name: "lin" },
        { name: "lin" },
        { name: "lin" },
        { name: "lin" }
      ]
    };
  },
  methods: {
      add(){
          this.list.push({name:'k'})
      },
      addNum(){
        this.$store.commit('in',2);
      },
      addNumByAsync(){
        this.$store.dispatch('in',3);
      }
  },
};
</script>

<style scoped lang='stylus'>
#a {
  background: red;
}
</style>

Эффект

Вывод: я обязательно стану великим богом, давай.