您好,欢迎来到测品娱乐。
搜索
您的当前位置:首页学习RxJS之JavaScript框架Cycle.js

学习RxJS之JavaScript框架Cycle.js

来源:测品娱乐

有了.startWith() 提供的这个初始值,整个流程得以启动,自此形成一个闭环,一个事件驱动的永动机 :)

Drivers

driver 是 Cycle.js 主函数 main()和外部世界打交道的接口,比如HTTP请求,比如DOM操作,这些是由具体的driver 负责的,它的存在确保了 main()的纯函数特性,所有副作用和繁琐的细节皆由 driver来实施——所以 @cycle/core 才125 行,而@cycle/dom 却有 4052 行之巨。

driver也是一个函数,从流程上来说,driver 监听sinks(main()的输出)做为输入,执行一些命令式的副作用,并产生出sources做为main()的输入。

DOM Driver

即 @cycle/dom,是使用最为频繁的driver。实际应用中,我们的main()会与DOM进行交互:

  • 需要传递内容给用户时,main()会返新的DOM sinks,以触发domDriver()生成virtual-dom,并渲染
  • main()订阅domDriver()的输出值(做为输入),并据此进行响应
  • 组件化
    每个Cycle.js应用程序不管多复杂,都遵循一套输入输出的基本法,因此,组件化是很容易实现,无非就是函数对函数的组合调用

    实战

    准备工作

    安装全局模块

    依赖模块一览

    "devDependencies": {
    "babel-plugin-transform-react-jsx": "^6.8.0",
    "babel-preset-es2015": "^6.9.0",
    "babelify": "^7.3.0",
    "browserify": "^13.0.1",
    "uglifyify": "^3.0.1",
    "watchify": "^3.7.0"
    },
    "dependencies": {
    "@cycle/core": "^6.0.3",
    "@cycle/dom": "^9.4.0",
    "@cycle/http": "^8.2.2"
    }

    .babelrc (插件支持JSX语法)

    {
    "plugins": [
    ["transform-react-jsx", { "pragma": "hJSX" }]
    ],
    "presets": ["es2015"]
    }

    Scripts(热生成和运行服务器)

    "scripts": {
    "start": "http-server",
    "build": "../node_modules/.bin/watchify index.js -v -g uglifyify -t babelify -o bundle.js"
    }

    以下实例需要运行时,可以开两个shell,一个跑热编译,一个起http-server(爱用currently亦可

    交互实例1

    功能:两个button,一加一减, 从0起步,回显计数
    demo地址: http://output.jsbin.com/lamexacaku

    HTML代码

    <!DOCTYPE html>
    <html>
    <head>
    <meta charset="utf-8">
    <title>components</title>
    </head>
    <body>
    <div id="container"></div>
    <script src="bundle.js"></script>
    </body>
    </html>

    index.js

    import Cycle from '@cycle/core'
    import { makeDOMDriver, hJSX } from '@cycle/dom'
    function main({ DOM }) {
    const decrement$ = DOM.select('.decrement').events('click').map(_ => -1)
    const increment$ = DOM.select('.increment').events('click').map(_ => +1)
    const count$ = increment$.merge(decrement$)
    .scan((x, y) => x + y)
    .startWith(0)
    return {
    DOM: count$.map(count =>
    <div>
    <input type="button" className="decrement" value=" - "/>
    <input type="button" className="increment" value=" + "/>
    <div>
    Clicked {count} times~
    </div>
    </div>
    )
    }
    }
    Cycle.run(main, {
    DOM: makeDOMDriver('#container'),
    })

    不难看出:

  • main()是个纯函数,从始至终不依赖外部状态,它的所有动力来自于DOM事件源click,这个状态机依靠Observable.prototype.scan()得以计算和传递,最后生成sinks传递给DOM driver以渲染;
  • 启动了这个循环是 .startWith();
  • Cycle.run是应用程序的入口,加载main()和DOM driver,后者对一个HTML容器进行渲染输出
  • 交互实例2

    功能: 一个button一个框,输入并点button后,通过Github api搜索相关的Repo,回显总数并展示第一页Repo列表

    index.js

    import Cycle from '@cycle/core'
    import { makeDOMDriver, hJSX } from '@cycle/dom'
    import { makeHTTPDriver } from '@cycle/http'
    const GITHUB_SEARCH_URL = 'https://api.github.com/search/repositories?q='
    function main(responses$) {
    const search$ = responses$.DOM.select('input[type="button"]')
    .events('click')
    .map(_ => { return { url: GITHUB_SEARCH_URL } })
    const text$ = responses$.DOM.select('input[type="text"]')
    .events('input')
    .map(e => { return { keyword: e.target.value } })
    const http$ = search$.withLatestFrom(text$, (search, text)=> search.url + text.keyword)
    .map(state => { return { url: state, method: 'GET' } })
    const dom$ = responses$.HTTP
    .filter(res$ => res$.request.url && res$.request.url.startsWith(GITHUB_SEARCH_URL))
    .mergeAll()
    .map(res => JSON.parse(res.text))
    .startWith({ loading: true })
    .map(JSON => {
    return <div>
    <input type="text"/>
    <input type="button" value="search"/>
    <br/>
    <span>
    {JSON.loading ? 'Loading...' : `total: ${JSON.total_count}`}
    </span>
    <ol>
    {
    JSON.items && JSON.items.map(repo =>
    <div>
    <span>repo.full_name</span>
    <a href={ repo.html_url }>{ repo.html_url }</a>
    </div>
    )
    }
    </ol>
    </div>
    }
    )
    return {
    DOM: dom$,
    HTTP: http$,
    }
    }
    const driver = {
    DOM: makeDOMDriver('#container'),
    HTTP: makeHTTPDriver(),
    }
    Cycle.run(main, driver)

    有了实例1做铺垫,这段代码也就通俗易懂了,需要提示的是:

  • Rx的Observable对象,命名上约定以$符为结束,以示区分
  • Observable.prototype.withLatestFrom()的作用是:在当前Observable对象的事件触发时(不同于 combineLatest),去合并参数的目标Observable对象的最新状态,并传递给下一级Observer
  • 以上项目完整实例,可在 /rockdragon/rx_practise/tree/master/src/web 找到
  • 小结

    寥寥数语,并不足以概括Cycle.js,比如 MVI设计模式,Driver的编写,awesome-cycle 这些进阶项,还是留给看官们自行探索吧。

    Copyright © 2019- cepb.cn 版权所有 湘ICP备2022005869号-7

    违法及侵权请联系:TEL:199 18 7713 E-MAIL:2724546146@qq.com

    本站由北京市万商天勤律师事务所王兴未律师提供法律服务