vue-next

vue-next项目地址

git地址:链接

git clone git@github.com:vuejs/vue-next.git
npm install
npm run dev // 生成一个未压缩的状态vue文件

在这里插入图片描述

入门小案例
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <script src="./vue.global.js"></script>
</head>
<body>
    <div id="app">

    </div>
    <script>
        const { createApp,reactive,computed,effect } = Vue;
        
        const YpComponent = {
            template:`
            <button @click="increment">Count is {{state.count}} double:{{state.double}}</button>
            `,
            setup(){
                const state = reactive({
                    count:0,
                    double:computed(()=>state.count*2)
                })
                effect(()=>{
                    // 副作用 可以监听变化
                    console.log('数字变化了',state.count)
                })
                function increment(){
                    state.count+=1
                }
                return {
                    state,
                    increment
                }
            }
        }
        createApp().mount(YpComponent,'#app')
    </script>
</body>
</html>
自己写一个reactive,computed,effect
<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <script src="./my-vue.js"></script>
</head>

<body>
    <div id="app"></div>
    <button id="btn">点我</button>

    <script>
        const root = document.querySelector('#app')
        const btn = document.querySelector('#btn')

        const obj = reactive({
            name: "vue",
            age: 6
        })

        let double= computed(()=>obj.age*2)

        effect(() => {
            root.innerHTML = `<h1>${obj.name}版本${obj.age} double:${double.value}</h1>`
        })
        btn.addEventListener('click', () => {
            obj.age += 1
        }, false)
    </script>
</body>

</html>

my-vue.js

let toProxy = new WeakMap() //存储  原始--响应
let toRaw = new WeakMap() //存储  响应--原始

const baseHander = {
    get(target, key) {
        const res = Reflect.get(target, key)

        //收集依赖
        track(target, key)
        // 递归查找
        return typeof res === 'object' ? reactive(res) : res
    },
    set(target, key, val) {
        const info = {
            oldValue: target[key],
            newValue: val
        }

        // if(target.hasOwnProperty(key)){ // 即如果触发的是私有属性,可以直接触发视图更新,length会屏蔽掉
        //     trigger();
        // }

        const res = Reflect.set(target, key, val)

        // 触发更新
        trigger(target, key, info)
        return res
    }
}

function reactive(target) {
    // 查询缓存
    let observed = toProxy.get(target)
    if (observed) {
        return observed;
    }
    if (toRaw.get(target)) {
        return target
    }
    observed = new Proxy(target, baseHander)
    //设置缓存
    toProxy.set(target, observed)
    toRaw.set(observed, target)
    return observed
}

let effectStack = []
let tagetMap = new WeakMap() //存储effect

function trigger(target, key, info) {
    const depsMap = tagetMap.get(target)

    if (depsMap === undefined) {
        return
    }
    const effects = new Set()
    const computedRuners = new Set()
    if (key) {
        let deps = depsMap.get(key)
        deps.forEach((effect) => {
            if(effect.computed){
                computedRuners.add(effect)
            }else{
                effects.add(effect)
            }

        })
    }
    // const run = effect =>effect()
    effects.forEach(effect => effect())
    computedRuners.forEach(effect => effect())
}


function track(target,key) {
    let effect = effectStack[effectStack.length - 1]
    if (effect) {
        let depsMap = tagetMap.get(target)
        if (depsMap === undefined) {
            depsMap = new Map()
            tagetMap.set(target, depsMap)
        }
        let dep = depsMap.get(key)
        if (dep === undefined) {
            dep = new Set()
            depsMap.set(key, dep)
        }
        if (!dep.has(effect)) {
            dep.add(effect)
            effect.deps.push(dep)
        }
    }
}

function effect(fn, options={}) {
    let e = createReactiveEffect(fn, options)
    if(!options.lazy){
        e()
    }
    return e
}

function createReactiveEffect(fn, options) {
    const effect = function effect(...args) {
        return run(effect, fn, args)
    }
    effect.deps = []

    effect.computed = options.computed
    effect.lazy = options.lazy
    return effect
}

function run(effect, fn, args) {
    if (effectStack.indexOf(effect) === -1) {
        try {
            effectStack.push(effect)
            return fn(...args)
        } finally {
            effectStack.pop()
        }
    }
}

function computed(fn) {
    const runner = effect(fn,{computed:true,lazy:true})
    return {
         effect:runner,
         get value(){
             return runner()
         }
    }
}
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值