使用ImmerJS和Svelte不可变商店创建镜头,我缺少什么

如何解决使用ImmerJS和Svelte不可变商店创建镜头,我缺少什么

我正在研究svelte和ImmerJS。

Immerjs和svelte商店应该能够以一种非常优雅的方式构成。

我试图仅使用选择器表达式从父存储创建派生可写存储,其中选择器表达式(从.NET借用的术语)是一个函数lambda,它描述了如何访问对象树的某些子部分。

root => root.a.b[10].foo

通常人们必须对此进行解析,但是我认为immerjs已经使用代理和Draft<T>类完成了艰苦的工作。

因此,目的是能够执行以下操作。

import {writable} from "svelte/store"
import {lens} from "my_magic_lens_library_not_yet_written"

interface Foo {
  a: number
  b: string
}

interface Bar {
  foo1: Foo
  foo2: Foo
}

let bar:Bar = {
  foo1: {a:10,b:"monkey"},foo2: {a:20,b:"cat"}
}

let barStore:Writable<Bar> = writable(bar)

let foo1_a_Store:Writable<number> = lens(barStore,(b:Draft<Bar>) => b.foo1.a)

barStore.subscribe(v=>console.log(v))

foo1_a_Store.set(77)

我希望输出

{ foo1: { a: 10,b: 'monkey' },foo2: { a: 20,b: 'cat' } }
{ foo1: { a: 77,b: 'cat' } }

但是是

{ foo1: { a: 10,b: 'cat' } }
{ foo1: { a: 10,b: 'cat' } }

镜头的实现方式是

import {writable,Writable} from "svelte/store"
import {produce,Draft} from "immer"


type Updater<T> = (arg0:T)=>T

type Selector<T,U> = ((ar:T)=>U) & ((ar:Draft<T>)=>Draft<U>);

function lens<T,U>(store:Writable<T>,selector:Selector<T,U>):Writable<U>  
{

  let {subscribe,set,update} = store

  function subSet(v:U):void
  {
    let rootUpdater =  (oldValue:T) => {
      return produce(
        oldValue,(ds:Draft<T>) => {  
          let subDraft:Draft<U> = selector(ds)
          Object.assign(subDraft,v)
        }
      )
    }
    update(rootUpdater)
  }

  function subUpdate(updater:Updater<U>):void
  {
    let rootUpdater =  (oldValue:T) => {
      return produce(
          oldValue,(ds:Draft<T>) => {  
            let subDraft:Draft<U> = selector(ds)
            Object.assign(subDraft,updater(selector(oldValue)))  
          }
      )
    }
    update(rootUpdater) 
  }

  return {
      subscribe: subscriber => subscribe(v=>subscriber(selector(v))),set: subSet,update: subUpdate
  }
}

我很确定失败的行是

Object.assign(subDraft,updater(selector(oldValue))) 

在这里,我尝试将更新的值传播到子草稿中。谁知道这是否有可能?但这应该是。有人可以找出神奇的调味料来使它起作用吗?

repl.it上有一个实时版本

https://repl.it/@BradPhelan/Substore

解决方法

使用我写的名为immer-loves-svelte的新软件包来实现这一目标。导出了一个名为subStore的函数,使您可以从父级创建子级存储

import {writable,Writable} from "svelte/store"
import {produce,Draft,isDraft} from "immer"
import {subStore} from "immer-loves-svelte"

interface Foo {
  a: number
  b: string
}

interface Bar {
  foo1: Foo
  foo2: Foo
}

let bar:Bar = {
  foo1: {a:10,b:"monkey"},foo2: {a:20,b:"cat"}
}

let barStore:Writable<Bar> = writable(bar)

// magic happens at this line with subStore call
let foo1_a_Store:Writable<number> = 
   subStore(barStore,b => b.foo1.a)

barStore.subscribe(v=>console.log(v))

foo1_a_Store.set(77

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?