From 2c8333ed50d3e4d1bc97ef631245346b3c622948 Mon Sep 17 00:00:00 2001 From: "zhanfeng.hzf" Date: Tue, 10 Sep 2019 15:25:44 +0800 Subject: [PATCH] =?UTF-8?q?=E5=A2=9E=E5=8A=A0=E7=A4=BA=E4=BE=8B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- docs/data.js | 2 +- docs/index.html | 16 ++++------------ docs/logo.jpg | Bin 25969 -> 0 bytes docs/zh/guide/autorun.html | 24 ------------------------ docs/zh/guide/binding.html | 24 ------------------------ docs/zh/guide/hook_model.html | 24 ------------------------ docs/zh/guide/mapping.html | 24 ------------------------ docs/zh/guide/model.html | 24 ------------------------ docs/zh/guide/quick.html | 24 ------------------------ docs/zh/guide/typescript.html | 24 ------------------------ docs/zh/guide/use_model.html | 24 ------------------------ docs/zh/guide/watch.html | 24 ------------------------ markdowns/quick.md | 2 ++ 13 files changed, 7 insertions(+), 229 deletions(-) delete mode 100644 docs/logo.jpg delete mode 100644 docs/zh/guide/autorun.html delete mode 100644 docs/zh/guide/binding.html delete mode 100644 docs/zh/guide/hook_model.html delete mode 100644 docs/zh/guide/mapping.html delete mode 100644 docs/zh/guide/model.html delete mode 100644 docs/zh/guide/quick.html delete mode 100644 docs/zh/guide/typescript.html delete mode 100644 docs/zh/guide/use_model.html delete mode 100644 docs/zh/guide/watch.html diff --git a/docs/data.js b/docs/data.js index bed0f3b..ca9adb9 100644 --- a/docs/data.js +++ b/docs/data.js @@ -1 +1 @@ -window.DOC_DATA={"locales":[{"name":"zh","text":"中文 (CN)","title":"Mota","groups":[{"name":"guide","text":"使用指南","docs":[{"group":"guide","name":"quick","title":"快速开始","index":1,"source":"# Mota\n\n## 简述 \n\nReact 是一个「视图层」的 UI 框架,以常见的 MVC 来讲 React 仅是 View,而我们在编写应用时,通常还需要关注更加重要的 model,对于 React 来讲,我们常常需要一个「状态管理」库。然而,目前大多数针对 React 的状态管理库都是「强依赖」过多的侵入本应该独立的业务模型中,导致「业务逻辑」对应的代码并不能轻易在其它地方重用,往往这些框架还具有「强排它性」,但是「业务模型」应该是没有过多依赖,应该是无关框架的,它应该随时可以被用在任何合适的 JavaScript 环境中,使用 mota 你可以用原生的普通的 JavaScript 代码编写你的「业务模型」,并让你的「业务模型」在不同框架、不同运行环境下重用更为容易。\n\nMota 是一个响应式的 React 应用状态管理库,基于 Mota 你可以用单纯无依赖的 JavaScript 为应用编写「业务模型」,并轻易的将「业务模型」关联到 React 应用中。\n\n## 示例\n\n[在线 TodoList 示例](http://houfeng.net/dn-template-mota/example/)\n([示例源码](https://github.com/Houfeng/dn-template-mota))\n\n## 安装\n\n通过 npm 安装,如下\n```sh\n$ npm i mota --save \n```\n\n或通过 `dawn` 脚手脚加创建工程,如下\n\n```sh\n$ mkdir your_path\n$ cd your_path\n$ dn init -t mota\n$ dn dev\n```\n\n需要先安装 dawn([Dawn 安装及使用文档](https://alibaba.github.io/dawn/docs/))\n\n## 结构\n\n一个 `mota` 工程的通常结构如下\n\n```sh\n.\n├── README.md\n├── package.json\n└── src\n ├── assets\n │   ├── common.less\n │   ├── favicon.ico\n │   └── index.html\n ├── components\n │   ├── todoApp.js\n │   └── todoItem.js\n ├── index.js\n └── models\n ├── TodoItem.js\n ├── TodoList.js\n └── index.js\n```\n\n## 文档\n- [快速开始](http://houfeng.net/mota/#!/zh/guide/quick)\n- [编写业务模型](http://houfeng.net/mota/#!/zh/guide/model)\n- [将组件属性映射到模型](http://houfeng.net/mota/#!/zh/guide/mapping)\n- [自执行函数](http://houfeng.net/mota/#!/zh/guide/autorun)\n- [监听模型变化](http://houfeng.net/mota/#!/zh/guide/watch)\n- [将模型数据与表单绑定](http://houfeng.net/mota/#!/zh/guide/binding)\n\n## 链接\n- [版本发布日志](https://github.com/Houfeng/mota/releases)\n- [MIT 开源协议](https://tldrlegal.com/license/mit-license)","filename":"markdowns/quick.md","root":"/Users/Houfeng/my/dev/mota","result":"

Mota

\n

简述

\n

React 是一个「视图层」的 UI 框架,以常见的 MVC 来讲 React 仅是 View,而我们在编写应用时,通常还需要关注更加重要的 model,对于 React 来讲,我们常常需要一个「状态管理」库。然而,目前大多数针对 React 的状态管理库都是「强依赖」过多的侵入本应该独立的业务模型中,导致「业务逻辑」对应的代码并不能轻易在其它地方重用,往往这些框架还具有「强排它性」,但是「业务模型」应该是没有过多依赖,应该是无关框架的,它应该随时可以被用在任何合适的 JavaScript 环境中,使用 mota 你可以用原生的普通的 JavaScript 代码编写你的「业务模型」,并让你的「业务模型」在不同框架、不同运行环境下重用更为容易。

\n

Mota 是一个响应式的 React 应用状态管理库,基于 Mota 你可以用单纯无依赖的 JavaScript 为应用编写「业务模型」,并轻易的将「业务模型」关联到 React 应用中。

\n

示例

\n

在线 TodoList 示例\n(示例源码)

\n

安装

\n

通过 npm 安装,如下

\n
$ npm i mota --save \n
\n

或通过 dawn 脚手脚加创建工程,如下

\n
$ mkdir your_path\n$ cd your_path\n$ dn init -t mota\n$ dn dev\n
\n

需要先安装 dawn(Dawn 安装及使用文档

\n

结构

\n

一个 mota 工程的通常结构如下

\n
.\n├── README.md\n├── package.json\n└── src\n    ├── assets\n    │   ├── common.less\n    │   ├── favicon.ico\n    │   └── index.html\n    ├── components\n    │   ├── todoApp.js\n    │   └── todoItem.js\n    ├── index.js\n    └── models\n        ├── TodoItem.js\n        ├── TodoList.js\n        └── index.js\n
\n

文档

\n\n

链接

\n\n"},{"group":"guide","name":"model","title":"编写业务模型","index":2,"source":"# 编写业务模型\n\n在你编写模型之前,先放下 React 也放下 Mota,就用单纯的 JavaScript 去编写你的业务模型,或有一个或多个类、或就是几个 Object,依它们应有的、自然的关系去抽像就行了,业务模型不依赖于 UI、也不依赖于某个框架,它易于测试,你可以针对它做单元测试。它易于重用,你可以将它用在合适的地方。最后, Mota 只是出场把它关联到 React。\n\n在 Mota 中「模型」可以是由一个 `class` 或普通的的 `Object`,整个「业务模型层」会由多个 `class` 和多个 `Object` 组成,而编写模型所需要的知识就是 JavaScript 固有的编程的知识。\n\n如下示例通过编写一个名为 `User` 的 `class` 创建了一个「用户模型」\n\n```js\nexport default class User {\n firstName = 'Jack';\n lastName = 'Hou';\n get fullName(){\n reutrn `${this.firstName} ${this.lastName}`;\n }\n}\n```\n\n也可以是一个 `Object`,通常这个模型需要是「单例」时,可采用这种方式,如下\n\n```js\nexport default {\n firstName: 'Jack',\n lastName: 'Hou',\n get fullName(){\n reutrn `${this.firstName} ${this.lastName}`;\n }\n};\n```\n\n在「业务模型」编写完成后,可以通过 `@model` 将某个「类」或「类的实例」关联到指定组件,关联后便可以在组件中使用 `this.model` 访问「模型的成员变量或方法」了,Mota 还会自动「收集组件依赖」,在组件「依赖的模型数据」发生变化时,自动响应变化并「驱动组件重新渲染」,如下\n\n```js\nimport { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\nclass App extends React.Component {\n\n onChange(field,event){\n this.model[field] = event.target.value;\n }\n\n render(){\n return
\n

{this.model.fullName}

\n

\n \n
\n \n

\n
;\n }\n}\n\nReactDOM.render(, mountNode);\n```\n\n值得注意的是,在使用 `@model` 时如果传入的是一个 `class` 最终每个组件实例都会自动创建一个 `独立的实例`,这样带来的好处是「当一个页面中有同一个组件的多个实例时,不会相互影响」。","filename":"markdowns/model.md","root":"/Users/Houfeng/my/dev/mota","result":"

编写业务模型

\n

在你编写模型之前,先放下 React 也放下 Mota,就用单纯的 JavaScript 去编写你的业务模型,或有一个或多个类、或就是几个 Object,依它们应有的、自然的关系去抽像就行了,业务模型不依赖于 UI、也不依赖于某个框架,它易于测试,你可以针对它做单元测试。它易于重用,你可以将它用在合适的地方。最后, Mota 只是出场把它关联到 React。

\n

在 Mota 中「模型」可以是由一个 class 或普通的的 Object,整个「业务模型层」会由多个 class 和多个 Object 组成,而编写模型所需要的知识就是 JavaScript 固有的编程的知识。

\n

如下示例通过编写一个名为 Userclass 创建了一个「用户模型」

\n
export default class User {\n  firstName = 'Jack';\n  lastName = 'Hou';\n  get fullName(){\n    reutrn `${this.firstName} ${this.lastName}`;\n  }\n}\n
\n

也可以是一个 Object,通常这个模型需要是「单例」时,可采用这种方式,如下

\n
export default {\n  firstName: 'Jack',\n  lastName: 'Hou',\n  get fullName(){\n    reutrn `${this.firstName} ${this.lastName}`;\n  }\n};\n
\n

在「业务模型」编写完成后,可以通过 @model 将某个「类」或「类的实例」关联到指定组件,关联后便可以在组件中使用 this.model 访问「模型的成员变量或方法」了,Mota 还会自动「收集组件依赖」,在组件「依赖的模型数据」发生变化时,自动响应变化并「驱动组件重新渲染」,如下

\n
import { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\nclass App extends React.Component {\n\n  onChange(field,event){\n    this.model[field] = event.target.value;\n  }\n\n  render(){\n    return <div>\n      <p>{this.model.fullName}</p>\n      <p>\n        <input onChange={this.onChange.bind(this,'firstName')}/>\n        <br/>\n        <input onChange={this.onChange.bind(this,'lastName')}/>\n      </p>\n    </div>;\n  }\n}\n\nReactDOM.render(<App/>, mountNode);\n
\n

值得注意的是,在使用 @model 时如果传入的是一个 class 最终每个组件实例都会自动创建一个 独立的实例,这样带来的好处是「当一个页面中有同一个组件的多个实例时,不会相互影响」。

\n"},{"group":"guide","name":"mapping","title":"属性映射","index":3,"source":"# 属性映射\n\n在 React 中通常会将应用折分为多个组件重用它们,并在用时传递给它「属性」,Mota 提供了将「组件属性」映射到「模型数据」的能力,基于 `model` 编程会让「视图层」的编写更为方例,专注于 UI 的呈现,如下\n\n```js\n@model({ value: 'demo' })\n@mapping(['value'])\nclass Demo extends React.Component {\n render () {\n return
{this.model.value}
;\n }\n}\n```\n\n上边的代码通过 `mapping` 将 `Demo` 这个组件的 `value` 属性映射到了 `model.value` 上,在组件的属性 `value` 发生变化时,会自动同步到 `model.value` 中。\n\n通过一个 `map` 进行映射,还可以让「组件属性」和「模型的成员」使用不同名称,如下:\n\n```js\n@model({ value: 'demo' })\n@mapping({ content: 'value' })\nclass Demo extends React.Component {\n render () {\n return
{this.model.value}
;\n }\n}\n```\n\n上边的代码,将组件 demo 的 `content` 属性映射到了 `model.value` 上,那么这个组件就可以这样使用了\n\n```js\nfunction App(){\n return ;\n}\n```\n\n`Demo` 组件的 `content` 属性,将自动被赋值给 `model.value`,如果没有 `mapping`,通常我们就需要在 `componentDidMount` 和 `componentWillReceiveProps` 之类的生命周函数去处理。其实,`mapping` 就像是一个语法糖,使用它将不再需要手动处理 prop->model 的更新了。","filename":"markdowns/mapping.md","root":"/Users/Houfeng/my/dev/mota","result":"

属性映射

\n

在 React 中通常会将应用折分为多个组件重用它们,并在用时传递给它「属性」,Mota 提供了将「组件属性」映射到「模型数据」的能力,基于 model 编程会让「视图层」的编写更为方例,专注于 UI 的呈现,如下

\n
@model({ value: 'demo' })\n@mapping(['value'])\nclass Demo extends React.Component {\n  render () {\n    return <div>{this.model.value}</div>;\n  }\n}\n
\n

上边的代码通过 mappingDemo 这个组件的 value 属性映射到了 model.value 上,在组件的属性 value 发生变化时,会自动同步到 model.value 中。

\n

通过一个 map 进行映射,还可以让「组件属性」和「模型的成员」使用不同名称,如下:

\n
@model({ value: 'demo' })\n@mapping({ content: 'value' })\nclass Demo extends React.Component {\n  render () {\n    return <div>{this.model.value}</div>;\n  }\n}\n
\n

上边的代码,将组件 demo 的 content 属性映射到了 model.value 上,那么这个组件就可以这样使用了

\n
function App(){\n  return <Demo content={'yyyy'} />;\n}\n
\n

Demo 组件的 content 属性,将自动被赋值给 model.value,如果没有 mapping,通常我们就需要在 componentDidMountcomponentWillReceiveProps 之类的生命周函数去处理。其实,mapping 就像是一个语法糖,使用它将不再需要手动处理 prop->model 的更新了。

\n"},{"group":"guide","name":"autorun","title":"自执行函数","index":4,"source":"# 自执行函数\n\nMota 中提供了一个 `autorun` 函数,可用于装饰 React 组件的成员方法,被装饰的「成员方法」将会在组件挂载后自动执行一次,Mota 将「收集方法中依赖的模型数据」,在依赖的模型数据发生变化时会「自动重新执行」对应的组件方法。\n\n示例\n\n```js\nimport { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n @autorun\n test() {\n console.log(this.model.name);\n }\n\n}\n```\n\n上边的示例代码中,组件在被挂载后将会自动执行 `test` 方法,同时 mota 会发现方法中依赖了 `model.name`,那么,在 `model.name` 发生变化时,就会重新执行 `test` 方法。","filename":"markdowns/autorun.md","root":"/Users/Houfeng/my/dev/mota","result":"

自执行函数

\n

Mota 中提供了一个 autorun 函数,可用于装饰 React 组件的成员方法,被装饰的「成员方法」将会在组件挂载后自动执行一次,Mota 将「收集方法中依赖的模型数据」,在依赖的模型数据发生变化时会「自动重新执行」对应的组件方法。

\n

示例

\n
import { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n  @autorun\n  test() {\n    console.log(this.model.name);\n  }\n\n}\n
\n

上边的示例代码中,组件在被挂载后将会自动执行 test 方法,同时 mota 会发现方法中依赖了 model.name,那么,在 model.name 发生变化时,就会重新执行 test 方法。

\n"},{"group":"guide","name":"watch","title":"监听模型变化","index":5,"source":"# 监听模型变化\n\n\nMota 中提供了一个 `watch` 函数,可用于装饰 React 组件的成员方法,`watch` 可以指定要观察的「模型数据」,在模型数据发变化时,就会自动执行「被装饰的组件方法」,`watch` 还可以像 `autorun` 一样自动执行一次,但它和 `autorun` 还是不尽相同,主要有如下区别\n\n- `autorun` 会自动收集依赖,而 `watch` 不会关心组件方法中有何依赖,需要手动指定依赖的模型数据\n- `watch` 默认不会「自动执行」,需显式的指定「立即执行参数为 true」,才会自动执行首次。\n- `autorun` 依赖的是「模型数据」本身,而 `watch` 依赖的是「计算函数」每次的「计算结果」\n \n示例\n\n```js\nimport { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n @watch(model=>model.name)\n test() {\n console.log('name 发生了变化');\n }\n\n}\n```\n\n上边的代码,通过 `watch` 装饰了 `test` 方法,并指定了观察的模型数据 `model.name`,那么每当 `model.name` 发生变\b化时,都会打印 `name 发生了变化`.\n\n`watch` 是否重新执行,取决于 `watch` 的作为第一个参数传给它的「计算函数」的计算结果,每当依赖的模型数据发生变化时 `watch` 都会重执行计算函数,当计算结果有变化时,才会执行被装饰的「组件方法」,示例\n\n```js\nexport default Demo extends Component {\n\n @watch(model=>model.name+model.age)\n test() {\n console.log('name 发生变化');\n }\n\n}\n```\n\n有时,我们希望 `watch` 能首先自动执行一次,那么可通过向第二个参数传一个 `true` 声明这个 `watch` 要自动执行一次。\n\n```js\nexport default Demo extends Component {\n\n @watch(model=>model.name,true)\n test() {\n console.log('name 发生变化');\n }\n\n}\n```\n\n上边的 `test` 方法,将会在「组件挂载之后自动执行」,之后在 `model.name` 发生变化时也将自动重新执行。","filename":"markdowns/watch.md","root":"/Users/Houfeng/my/dev/mota","result":"

监听模型变化

\n

Mota 中提供了一个 watch 函数,可用于装饰 React 组件的成员方法,watch 可以指定要观察的「模型数据」,在模型数据发变化时,就会自动执行「被装饰的组件方法」,watch 还可以像 autorun 一样自动执行一次,但它和 autorun 还是不尽相同,主要有如下区别

\n
    \n
  • autorun 会自动收集依赖,而 watch 不会关心组件方法中有何依赖,需要手动指定依赖的模型数据
  • \n
  • watch 默认不会「自动执行」,需显式的指定「立即执行参数为 true」,才会自动执行首次。
  • \n
  • autorun 依赖的是「模型数据」本身,而 watch 依赖的是「计算函数」每次的「计算结果」
  • \n
\n

示例

\n
import { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n  @watch(model=>model.name)\n  test() {\n    console.log('name 发生了变化');\n  }\n\n}\n
\n

上边的代码,通过 watch 装饰了 test 方法,并指定了观察的模型数据 model.name,那么每当 model.name 发生变\b化时,都会打印 name 发生了变化.

\n

watch 是否重新执行,取决于 watch 的作为第一个参数传给它的「计算函数」的计算结果,每当依赖的模型数据发生变化时 watch 都会重执行计算函数,当计算结果有变化时,才会执行被装饰的「组件方法」,示例

\n
export default Demo extends Component {\n\n  @watch(model=>model.name+model.age)\n  test() {\n    console.log('name 发生变化');\n  }\n\n}\n
\n

有时,我们希望 watch 能首先自动执行一次,那么可通过向第二个参数传一个 true 声明这个 watch 要自动执行一次。

\n
export default Demo extends Component {\n\n  @watch(model=>model.name,true)\n  test() {\n    console.log('name 发生变化');\n  }\n\n}\n
\n

上边的 test 方法,将会在「组件挂载之后自动执行」,之后在 model.name 发生变化时也将自动重新执行。

\n"},{"group":"guide","name":"binding","title":"数据绑定","index":6,"source":"# 数据绑定\n\n\n### 基本用法\n\n不要惊诧,就是「双向绑定」。Mota 不排斥「双向绑定」,使用 Mota 能够实现类似 `ng` 或 `vue` 的绑定效果。还是前边小节中的模型,我们来稍微改动一下组件的代码\n\n```js\nimport { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\n@binding\nclass App extends React.Component {\n render(){\n const { fullName, firstName, popup } = this.model;\n return
\n

{fullName}

\n

\n \n \n

\n
;\n }\n}\nReactDOM.render(, mountNode);\n```\n\n~~其中的「关键」就是 `@binding`,使用 `@binding` 后~~ (>=1.2.0 的版本将会自动处理,不必显示的启用),组件便具备了「双向绑定」的能力,在 `jsx` 中便可以通过名为 `data-bind` 的自定义 `attribute` 进行绑定了,`data-bind` 的值是一个「绑定表达式字符串」,绑定表达式执行的 `scope` 是 `model` 而不是 `this`,也就是只能与 `模型的成员` 进行绑定。\n\n会有一种情况是当要绑定的数据是一个循环变量时,「绑定表达式」写起会较麻烦也稍显长,比如\n\n```js\n@model(userModel)\n@binding\nclass App extends React.Component {\n render(){\n const { userList } = this.model;\n return
    \n {userList.map((user,index)=>(\n
  • \n \n {user.name}\n
  • \n ))}\n
;\n }\n}\n```\n\n因为「绑定表达式」的执行 `scope` 默认是 `this.model`,以及「表达式是个字符串」,看一下 `userList[${index}].selected` 这并不友好,为此 Mota 还提供了一个名为 `data-scope` 的 `attribute`,通过它能改变要绑定的 `scope`,参考如下示例\n\n```js\n@model(userModel)\n@binding\nclass App extends React.Component {\n render(){\n const { userList } = this.model;\n return
    \n {userList.map(user=>(\n
  • \n \n {user.name}\n
  • \n ))}\n
;\n }\n}\n```\n\n通过 `data-scope` 将 `input` 的绑定上下文对象声明为当前循环变量 `user`,这样就可以用 `data-bind` 直接绑定到对应 `user` 的属性上了。\n\n### 原生表单控件\n\n所有的原生表单控件,比如「普通 input、checkbox、radio、textarea、select」都可以直接进行绑定。其中,「普通 input 和 textrea」比较简单,将一个字符类型的模型数据与控件绑定就行了,而对于「checkbox 和 radio」 有多种不同的绑定形式。\n\n将「checkbox 或 radio」绑定到一个 `boolean` 值,此时会将 checkbox 或 radio 的 `checked` 属性和模型数据建立绑定,`checked` 反应了 `boolean` 变量的值,参考如下示例\n\n```js\n@model({ selected:false })\n@binding\nclass App extends React.Component {\n render(){\n return
\n \n \n
;\n }\n}\n```\n\n如上示例通过 `this.model.selected` 就能拿到当前 checkbox 或 radio 的选中状态。\n\n\n将 checkbox 绑定到一个「数组」,通常是多个 checkbox 绑定同一个数组变量上,此时和数据建立绑定的是 checkbox 的 value,数据中会包含当前选中的 checkbox 的 value,如下\n\n```js\n@model({ selected:[] })\n@binding\nclass App extends React.Component {\n render(){\n return
\n \n \n
;\n }\n}\n```\n\n如上示例,通过 `this.selected` 就能知道当前有哪些 checkbox 被选中了,并拿到所有选中的 value\n\n\n将多个 radio 绑定我到一个「字符类型的变量」,此时和数据建立绑定的是 raido 的 value,因为 radio 是单选的,所以对应的数据是当前选中的 radio 的 value,如下\n\n```js\n@model({ selected:'' })\n@binding\nclass App extends React.Component {\n render(){\n return
\n \n \n
;\n }\n}\n```\n通过 `this.model.selected` 就能拿到当前选中的 radio 的 `value`\n\n\n### 自定义组件\n\n但是对于一些「组件库」中的「部分表单组件」不能直接绑定,因为 Mota 并没有什么依据可以判断这是一个什么组件。所以 Mota 提供了一个名为 `bindable` 的函数,用将任意组件包装成「可绑定组件」。\n\nbindable 有两种个参数,用于分别指定「原始组件」和「包装选项」\n\n```js\n//可以这样\nconst MyComponent = bindable(opts, Component);\n//也可这样\nconst MyCompoent = bindable(Component, opts);\n```\n\n关建是 `bindable` 需要的 `opts`,通过 `opts` 我们可以造诉 Mota 如何绑定这个组件,`opts` 中有两个重要的成员,它的结构如下\n\n```js\n{\n value: ['value 对应的属性名'],\n event: ['value 改变的事件名']\n}\n```\n\n所以,我们可以这样包装一个自定义文本输入框\n\n```js\nconst MyInput = bindable(Input,{\n value: ['value'],\n event: ['onChange']\n});\n```\n\n对这种「value 不需要转换,`change` 能通过 `event` 或 `event.target.value` 拿到值」的组件,通过如上的代码就能完成包装了。\n\n对于有 `onChange` 和 `value` 的这类文本输入组件,因为 opts 的默认值就是\n\n```js\n{\n value: ['value'],\n event: ['onChange']\n}\n```\n\n所以,可以更简单,这样就行,\n```js\nconst MyInput = bindable(Input);\n```\n\n而对于 checkbox 和 radio 来讲,如上边讲到的它「根据不同的数据型有不同的绑定形式」,这就需要指定处理函数了,如下\n\n```js\nconst radioOpts = {\n prop: ['checked', (ctx, props) => {\n const mValue = ctx.getValue();\n if (typeof mValue == 'boolean') {\n return !!mValue;\n } else {\n return mValue == props.value;\n }\n }],\n event: ['onChange', (ctx, event) => {\n const { value, checked } = event.target;\n const mValue = ctx.getValue();\n if (typeof mValue == 'boolean') {\n ctx.setValue(checked);\n } else if (checked) ctx.setValue(value);\n }]\n};\n```\n\n通过 `prop` 的第二个值,能指定「属性处理函数」,`event` 的第二个值能指取「事件处理函数」,处理函数的 `ctx` 是个特殊的对象 \n\n- `ctx.getValue` 能获取「当前绑定的模型数据」\n- `ctx.setValue` 能设置「当前绑定的模型数据」\n\n上边是 `radio` 的配置,首先,在「属性处理函数」中通过绑定的「模型数据的类型」决定 `checked` 最终的状态是什么,并在函数中返回。再次,在「事件处理函数」中通过绑定的「模型数据的类型」决定将什么值回写到模型中。\n\n通过「属性处理函数」和「事件处理函数」几乎就能将任意的自定义组件转换为「可绑定组件」了。\n\n另外,对于常见的 `CheckBox` 和 `Radio` 类型的组件 Mota 也提供了内建的 `opts` 配置支持,如果一个自定义组件拥有和「原生 checkbox 一致的属性和事件模型」,那边可以直接用简单的方式去包装,如下\n\n```js\nconst MyCheckBox = bindable('checkbox',CheckBox);\nconst MyRadio = bindable('radio',Radio);\n```\n\n好了,关于绑定就这些了。","filename":"markdowns/binding.md","root":"/Users/Houfeng/my/dev/mota","result":"

数据绑定

\n

基本用法

\n

不要惊诧,就是「双向绑定」。Mota 不排斥「双向绑定」,使用 Mota 能够实现类似 ngvue 的绑定效果。还是前边小节中的模型,我们来稍微改动一下组件的代码

\n
import { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\n@binding\nclass App extends React.Component {\n  render(){\n    const { fullName, firstName, popup } = this.model;\n    return <div>\n      <p>{fullName}</p>\n      <p>\n        <input data-bind=\"firstName\"/>\n        <button onClick={popup}> click me </button>\n      </p>\n    </div>;\n  }\n}\nReactDOM.render(<App/>, mountNode);\n
\n

其中的「关键」就是 @binding,使用 @binding (>=1.2.0 的版本将会自动处理,不必显示的启用),组件便具备了「双向绑定」的能力,在 jsx 中便可以通过名为 data-bind 的自定义 attribute 进行绑定了,data-bind 的值是一个「绑定表达式字符串」,绑定表达式执行的 scopemodel 而不是 this,也就是只能与 模型的成员 进行绑定。

\n

会有一种情况是当要绑定的数据是一个循环变量时,「绑定表达式」写起会较麻烦也稍显长,比如

\n
@model(userModel)\n@binding\nclass App extends React.Component {\n  render(){\n    const { userList } = this.model;\n    return <ul>\n     {userList.map((user,index)=>(\n       <li key={user.id}>\n         <input type=\"checkobx\" data-bind={`userList[${index}].selected`}>\n         {user.name}\n       </li>\n     ))}\n    </ul>;\n  }\n}\n
\n

因为「绑定表达式」的执行 scope 默认是 this.model,以及「表达式是个字符串」,看一下 userList[${index}].selected 这并不友好,为此 Mota 还提供了一个名为 data-scopeattribute,通过它能改变要绑定的 scope,参考如下示例

\n
@model(userModel)\n@binding\nclass App extends React.Component {\n  render(){\n    const { userList } = this.model;\n    return <ul>\n     {userList.map(user=>(\n       <li key={user.id}>\n         <input type=\"checkobx\" data-scope={user} data-bind=\"selected\">\n         {user.name}\n       </li>\n     ))}\n    </ul>;\n  }\n}\n
\n

通过 data-scopeinput 的绑定上下文对象声明为当前循环变量 user,这样就可以用 data-bind 直接绑定到对应 user 的属性上了。

\n

原生表单控件

\n

所有的原生表单控件,比如「普通 input、checkbox、radio、textarea、select」都可以直接进行绑定。其中,「普通 input 和 textrea」比较简单,将一个字符类型的模型数据与控件绑定就行了,而对于「checkbox 和 radio」 有多种不同的绑定形式。

\n

将「checkbox 或 radio」绑定到一个 boolean 值,此时会将 checkbox 或 radio 的 checked 属性和模型数据建立绑定,checked 反应了 boolean 变量的值,参考如下示例

\n
@model({ selected:false })\n@binding\nclass App extends React.Component {\n  render(){\n    return <div>\n      <input type=\"checkbox\" data-bind=\"selected\"/>\n      <input type=\"radio\" data-bind=\"selected\"/>\n    </div>;\n  }\n}\n
\n

如上示例通过 this.model.selected 就能拿到当前 checkbox 或 radio 的选中状态。

\n

将 checkbox 绑定到一个「数组」,通常是多个 checkbox 绑定同一个数组变量上,此时和数据建立绑定的是 checkbox 的 value,数据中会包含当前选中的 checkbox 的 value,如下

\n
@model({ selected:[] })\n@binding\nclass App extends React.Component {\n  render(){\n    return <div>\n      <input type=\"checkbox\" data-bind=\"selected\" value=\"1\"/>\n      <input type=\"checkbox\" data-bind=\"selected\" value=\"2\"/>\n    </div>;\n  }\n}\n
\n

如上示例,通过 this.selected 就能知道当前有哪些 checkbox 被选中了,并拿到所有选中的 value

\n

将多个 radio 绑定我到一个「字符类型的变量」,此时和数据建立绑定的是 raido 的 value,因为 radio 是单选的,所以对应的数据是当前选中的 radio 的 value,如下

\n
@model({ selected:'' })\n@binding\nclass App extends React.Component {\n  render(){\n    return <div>\n      <input type=\"radio\" data-bind=\"selected\" value=\"1\"/>\n      <input type=\"radio\" data-bind=\"selected\" value=\"2\"/>\n    </div>;\n  }\n}\n
\n

通过 this.model.selected 就能拿到当前选中的 radio 的 value

\n

自定义组件

\n

但是对于一些「组件库」中的「部分表单组件」不能直接绑定,因为 Mota 并没有什么依据可以判断这是一个什么组件。所以 Mota 提供了一个名为 bindable 的函数,用将任意组件包装成「可绑定组件」。

\n

bindable 有两种个参数,用于分别指定「原始组件」和「包装选项」

\n
//可以这样\nconst MyComponent = bindable(opts, Component);\n//也可这样\nconst MyCompoent = bindable(Component, opts);\n
\n

关建是 bindable 需要的 opts,通过 opts 我们可以造诉 Mota 如何绑定这个组件,opts 中有两个重要的成员,它的结构如下

\n
{\n  value: ['value 对应的属性名'],\n  event: ['value 改变的事件名']\n}\n
\n

所以,我们可以这样包装一个自定义文本输入框

\n
const MyInput = bindable(Input,{\n  value: ['value'],\n  event: ['onChange']\n});\n
\n

对这种「value 不需要转换,change 能通过 eventevent.target.value 拿到值」的组件,通过如上的代码就能完成包装了。

\n

对于有 onChangevalue 的这类文本输入组件,因为 opts 的默认值就是

\n
{\n  value: ['value'],\n  event: ['onChange']\n}\n
\n

所以,可以更简单,这样就行,

\n
const MyInput = bindable(Input);\n
\n

而对于 checkbox 和 radio 来讲,如上边讲到的它「根据不同的数据型有不同的绑定形式」,这就需要指定处理函数了,如下

\n
const radioOpts = {\n  prop: ['checked', (ctx, props) => {\n    const mValue = ctx.getValue();\n    if (typeof mValue == 'boolean') {\n      return !!mValue;\n    } else {\n      return mValue == props.value;\n    }\n  }],\n  event: ['onChange', (ctx, event) => {\n    const { value, checked } = event.target;\n    const mValue = ctx.getValue();\n    if (typeof mValue == 'boolean') {\n      ctx.setValue(checked);\n    } else if (checked) ctx.setValue(value);\n  }]\n};\n
\n

通过 prop 的第二个值,能指定「属性处理函数」,event 的第二个值能指取「事件处理函数」,处理函数的 ctx 是个特殊的对象

\n
    \n
  • ctx.getValue 能获取「当前绑定的模型数据」
  • \n
  • ctx.setValue 能设置「当前绑定的模型数据」
  • \n
\n

上边是 radio 的配置,首先,在「属性处理函数」中通过绑定的「模型数据的类型」决定 checked 最终的状态是什么,并在函数中返回。再次,在「事件处理函数」中通过绑定的「模型数据的类型」决定将什么值回写到模型中。

\n

通过「属性处理函数」和「事件处理函数」几乎就能将任意的自定义组件转换为「可绑定组件」了。

\n

另外,对于常见的 CheckBoxRadio 类型的组件 Mota 也提供了内建的 opts 配置支持,如果一个自定义组件拥有和「原生 checkbox 一致的属性和事件模型」,那边可以直接用简单的方式去包装,如下

\n
const MyCheckBox = bindable('checkbox',CheckBox);\nconst MyRadio = bindable('radio',Radio);\n
\n

好了,关于绑定就这些了。

\n"},{"group":"guide","name":"use_model","title":"使用 Hook API","index":7,"source":"# 使用 Hook API\n\n### 简单介绍\n在 React 发布包含 `Hooks` 的 `alpah` 版后,Mota 也在 `next` 版本中新增支持了 Hook 风格的 API,随着 React `v16.8` 版本的发布带来了稳定版的 `Hooks` 支持。\n\n目前,Mota \b已经在稳定版中,提供了 `Hook API` 的支持,利用 React 的 `Hooks` 可以让你在不编写类的情况下使用 `state` 和 React 的其他功能。而使用 Mota 极少的 `Hook API` 将给应用带来 Hook 风格可响应的全局状态管理支持。\n\n### 基本用法\n\n```js\nimport React from 'react';\nimport { render } from 'react-dom';\n\nfunction App(){\n //通过 useModel 拿到一个可响应的 model\n const model = useModel({ count:0 });\n //定义累加按钮事件\n const onClick = useCallback(()=>model.count++);\n //--\n return
\n
{model.count}
\n \n
;\n}\n\nrender(, document.getElementId('root'));\n```\n\n仅有一个新增 API `useModel`,通过 `useModel` 可在一个 `Function Component` 中使用 `model`,如同在 `Class Component` 中的 `@model`,此时的 `model` 依然是可响应的,执行时会对组件进行「依赖收集」,当操作 `model` 的成员时(比如 `model.count=1` 的赋值操作),Mota 会自动发现组件依赖的数据发生了变化并通知组件进行更新。\n\n### 进阶说明\n\n在基本用法中提到了一个关键词「依赖收集」,通过 `useModel` 拿到了可响应的 `model`,默认情况下只有被组件依赖的模型数据发生了变化组件才会更新,比如下边的示例代码中,只有在 `model.a` 发生变化时,组件才会重新渲染。\n\n\n```js\nfunction Demo(){\n const model = useModel({ a:0, b:1 });\n ...\n return
{model.a}
\n}\n```\n\n实际开发过程中有时「组件依赖了模型上的某个对象,但希望这个对象的子成员发生变化时,组件也要重新渲染」\n\n```js\nfunction Demo(){\n const { info } = useModel({ info: { name: 'test'} });\n ...\n return \n}\n```\n\n因为对于 `Demo` 来说只依赖了 `info`,而 `info` 的引用是一直没有变化的,所以在 `info.name` 发生变化时 `Demo` 并不会重新渲染。那这样 `Info` 组件会一直显示旧的数据。\n\n如何处理这个问题?\n\n一个方法是让 `Info` 也通过 `useModel` 有自已的 `model`,那 `Info` 的依赖会被独立解析,比如\n\n```js\nfunction Info(props){\n const info = useModel(props.data);\n return
{info.name}
\n}\n```\n\n这个虽然 `Demo` 不会重新渲染,但 Mota 会发现 `Info` 依赖了 `info.name`,但发现数据变化时,`Info` 会自动更新。\n\n还有一个方法是,在更新 `info.name` 时换一个写法\n\n```js\n//通常的直接给 name 赋值\nmodel.info.name = 'test';\n//如下的给 info 赋值的写法,会让 Demo 发现 info 的变化\nmodel.info = {...model.info, name: 'test'};\n```\n\n除了上述的两个方法,还有一个方法就是通过 `useModel` 的第二个参数显示的声明额外的依赖,第二个参数可是一个数组,数组中是显式声明的依赖,格式为子成员的路径,如下\n\n```js\nfunction Demo(){\n const { info } = useModel({ \n info: { name: 'test'} \n }, ['info.name']);\n ...\n return \n}\n```\n\n但有时时模型数据是一个数组,我们无法直接指定每个子元素的路径,这时第二个参数还可以是一个函数,函数的参数是「变化的模型数据的路径」,可参函数中返回 `boolean` 值决定是否需要更新组件,如下\n\n```js\nfunction Demo(){\n const { info } = useModel({ \n info: [{name: 'test1'}] \n }, p=> p.endsWith('.name'));\n ...\n return \n}\n```\n\n示例中通过用 `endsWith` 路径是不是 `*.name` 结尾的决定要不要更新,当然也可以用更多的判断方法决定要不要更新。","filename":"markdowns/use_model.md","root":"/Users/Houfeng/my/dev/mota","result":"

使用 Hook API

\n

简单介绍

\n

在 React 发布包含 Hooksalpah 版后,Mota 也在 next 版本中新增支持了 Hook 风格的 API,随着 React v16.8 版本的发布带来了稳定版的 Hooks 支持。

\n

目前,Mota \b已经在稳定版中,提供了 Hook API 的支持,利用 React 的 Hooks 可以让你在不编写类的情况下使用 state 和 React 的其他功能。而使用 Mota 极少的 Hook API 将给应用带来 Hook 风格可响应的全局状态管理支持。

\n

基本用法

\n
import React from 'react';\nimport { render } from 'react-dom';\n\nfunction App(){\n  //通过 useModel 拿到一个可响应的 model\n  const model = useModel({ count:0 });\n  //定义累加按钮事件\n  const onClick = useCallback(()=>model.count++);\n  //--\n  return <div>\n    <div>{model.count}</div>\n    <button onClick={onClick}>click me</button>\n  </div>;\n}\n\nrender(<App/>, document.getElementId('root'));\n
\n

仅有一个新增 API useModel,通过 useModel 可在一个 Function Component 中使用 model,如同在 Class Component 中的 @model,此时的 model 依然是可响应的,执行时会对组件进行「依赖收集」,当操作 model 的成员时(比如 model.count=1 的赋值操作),Mota 会自动发现组件依赖的数据发生了变化并通知组件进行更新。

\n

进阶说明

\n

在基本用法中提到了一个关键词「依赖收集」,通过 useModel 拿到了可响应的 model,默认情况下只有被组件依赖的模型数据发生了变化组件才会更新,比如下边的示例代码中,只有在 model.a 发生变化时,组件才会重新渲染。

\n
function Demo(){\n  const model = useModel({ a:0, b:1 });\n  ...\n  return <div>{model.a}</div>\n}\n
\n

实际开发过程中有时「组件依赖了模型上的某个对象,但希望这个对象的子成员发生变化时,组件也要重新渲染」

\n
function Demo(){\n  const { info } = useModel({ info: { name: 'test'} });\n  ...\n  return <Info data={info}/>\n}\n
\n

因为对于 Demo 来说只依赖了 info,而 info 的引用是一直没有变化的,所以在 info.name 发生变化时 Demo 并不会重新渲染。那这样 Info 组件会一直显示旧的数据。

\n

如何处理这个问题?

\n

一个方法是让 Info 也通过 useModel 有自已的 model,那 Info 的依赖会被独立解析,比如

\n
function Info(props){\n  const info = useModel(props.data);\n  return <div>{info.name}</div>\n}\n
\n

这个虽然 Demo 不会重新渲染,但 Mota 会发现 Info 依赖了 info.name,但发现数据变化时,Info 会自动更新。

\n

还有一个方法是,在更新 info.name 时换一个写法

\n
//通常的直接给 name 赋值\nmodel.info.name = 'test';\n//如下的给 info 赋值的写法,会让 Demo 发现 info 的变化\nmodel.info = {...model.info, name: 'test'};\n
\n

除了上述的两个方法,还有一个方法就是通过 useModel 的第二个参数显示的声明额外的依赖,第二个参数可是一个数组,数组中是显式声明的依赖,格式为子成员的路径,如下

\n
function Demo(){\n  const { info } = useModel({ \n    info: { name: 'test'} \n  }, ['info.name']);\n  ...\n  return <Info data={info}/>\n}\n
\n

但有时时模型数据是一个数组,我们无法直接指定每个子元素的路径,这时第二个参数还可以是一个函数,函数的参数是「变化的模型数据的路径」,可参函数中返回 boolean 值决定是否需要更新组件,如下

\n
function Demo(){\n  const { info } = useModel({ \n    info: [{name: 'test1'}] \n  }, p=> p.endsWith('.name'));\n  ...\n  return <Info data={info}/>\n}\n
\n

示例中通过用 endsWith 路径是不是 *.name 结尾的决定要不要更新,当然也可以用更多的判断方法决定要不要更新。

\n"},{"group":"guide","name":"hook_model","title":"面向 Hook 的模型","index":8,"source":"# 面向 Hook 的模型\n\n### 一些说明\n\n针对 Mota 的 `useModel` 的模型本质上和针对 `@model` 的模型并无本质区别,用以往的风格编写的「模型类」或「单例的普通 Object」,除了能用于 `@model` 也是能用于 `useModel` 的。\n\n既然用了 `Hook API`,是不是可在编写模型也避免再写「类」或「单例的 Object」?为此 `useModel` 还提供了用 `ES Module` 作为模型的支持,如用其他风格的模型一样,`ES Module` 风格的模型也不需要引用额外的依赖,仅用 `ES` 原生语法即可。\n\n### 用 ES Module 编写模型\n\n通过 `ES Module` 直接作为 `model` 的优点时「简单、直接」,同时由于为了保证「可响应」不被破坏,需要一点点约束,就是「必须导出一个 state 对象」,如下\n\n```js\nexport const state = {\n name: 'test'\n}\n\nexport function setName(name){\n state.name = name\n}\n```\n\n上边的代码是一个最简单的可当作 `model` 的 `ES Module`,一个包含「state 和一组件函数」的 `ES Module` 就是一个可被 `useModel` 使用的 `model`,参考如下代码\n\n```js\nimport * as demo from './models/demo';\n\nfunction App(){\n const { name } = useModel(demo);\n return
{name}
;\n}\n```\n\n注意:必须 export 一个 state 的约束只针对于 `ES Module`,用 `class` 或 `object` 的风格编写的 `model` 无任何约束。","filename":"markdowns/hook_model.md","root":"/Users/Houfeng/my/dev/mota","result":"

面向 Hook 的模型

\n

一些说明

\n

针对 Mota 的 useModel 的模型本质上和针对 @model 的模型并无本质区别,用以往的风格编写的「模型类」或「单例的普通 Object」,除了能用于 @model 也是能用于 useModel 的。

\n

既然用了 Hook API,是不是可在编写模型也避免再写「类」或「单例的 Object」?为此 useModel 还提供了用 ES Module 作为模型的支持,如用其他风格的模型一样,ES Module 风格的模型也不需要引用额外的依赖,仅用 ES 原生语法即可。

\n

用 ES Module 编写模型

\n

通过 ES Module 直接作为 model 的优点时「简单、直接」,同时由于为了保证「可响应」不被破坏,需要一点点约束,就是「必须导出一个 state 对象」,如下

\n
export const state = {\n  name: 'test'\n}\n\nexport function setName(name){\n  state.name = name\n}\n
\n

上边的代码是一个最简单的可当作 modelES Module,一个包含「state 和一组件函数」的 ES Module 就是一个可被 useModel 使用的 model,参考如下代码

\n
import * as demo from './models/demo';\n\nfunction App(){\n  const { name } = useModel(demo);\n  return <div>{name}</div>;\n}\n
\n

注意:必须 export 一个 state 的约束只针对于 ES Module,用 classobject 的风格编写的 model 无任何约束。

\n"},{"group":"guide","name":"typescript","title":"在 TS 中使用","index":8,"source":"# 在 TS 中使用 Mota\n\nMota 的 `Package` 中自带了「类型定义文件」,无论使用 `Class + Decorator` 风格的 API 或使用 `Hooks` 风格的 API,都能愉快的使用 TypeScript,下边有两个小提示。\n\n### 提示一:使用 @model \n\n在通过 `@model` 为组件关联了一个 `model` 后,需要声明 `this.model` 的类型,参考如下代码\n\n```js\nimport * as React from \"react\";\nimport { model, watch, mapping } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\n@model(DemoModel)\nexport class Demo extends React.Component {\n\n //需要声明 model 的类型\n model: DemoModel;\n\n render() {\n //便能让 this.model 具备完整的类型提示了\n const { name } = this.model;\n return
\n {name} \n
;\n }\n}\n```\n\n### 提示一:使用 useModel\n\n完整的 `useModel` 的定义为 `useModel(model:T)=>T`,但使用 `useModel` 时一般不需做特别的声明,默认情况下 `TS` 就能完成类型推导\n\n```js\nimport * as React from \"react\";\nimport { useModel } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\nexport function Demo {\n const { name } = useModel(DemoModel);\n return
\n {name} \n
;\n}\n```","filename":"markdowns/typescript.md","root":"/Users/Houfeng/my/dev/mota","result":"

在 TS 中使用 Mota

\n

Mota 的 Package 中自带了「类型定义文件」,无论使用 Class + Decorator 风格的 API 或使用 Hooks 风格的 API,都能愉快的使用 TypeScript,下边有两个小提示。

\n

提示一:使用 @model

\n

在通过 @model 为组件关联了一个 model 后,需要声明 this.model 的类型,参考如下代码

\n
import * as React from \"react\";\nimport { model, watch, mapping } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\n@model(DemoModel)\nexport class Demo extends React.Component {\n\n  //需要声明 model 的类型\n  model: DemoModel;\n\n  render() {\n    //便能让 this.model 具备完整的类型提示了\n    const { name } = this.model;\n    return <div className=\"demo\">  \n      {name}  \n    </div>;\n  }\n}\n
\n

提示一:使用 useModel

\n

完整的 useModel 的定义为 useModel<T>(model:T)=>T,但使用 useModel 时一般不需做特别的声明,默认情况下 TS 就能完成类型推导

\n
import * as React from \"react\";\nimport { useModel } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\nexport function Demo {\n  const { name } = useModel(DemoModel);\n  return <div className=\"demo\">  \n    {name}  \n  </div>;\n}\n
\n"}]}],"links":[{"text":"状态管理 (Mota)","url":"//houfeng.net/mota/"},{"text":"数据验证 (Validation)","url":"//houfeng.net/mota-validation/"},{"text":"表单组件 (Form)","url":"//houfeng.net/mota-form/"},{"text":"源码 (GitHub)","url":"//github.com/Houfeng/mota"}]}],"plugins":[{"name":"doczilla-place","options":{}},{"name":"doczilla-include","options":{}},{"name":"doczilla-container","options":{}},{"name":"doczilla-highlight","options":{}},{"name":"doczilla-details","options":{}},{"name":"doczilla-card","options":{}},{"name":"doczilla-anchor","options":{}}],"mode":"static","baseUri":"","extname":".html"}; \ No newline at end of file +window.DOC_DATA={"locales":[{"name":"zh","text":"中文 (CN)","title":"Mota","groups":[{"name":"guide","text":"使用指南","docs":[{"group":"guide","name":"quick","title":"快速开始","index":1,"source":"# Mota\n\n## 简述 \n\nReact 是一个「视图层」的 UI 框架,以常见的 MVC 来讲 React 仅是 View,而我们在编写应用时,通常还需要关注更加重要的 model,对于 React 来讲,我们常常需要一个「状态管理」库。然而,目前大多数针对 React 的状态管理库都是「强依赖」过多的侵入本应该独立的业务模型中,导致「业务逻辑」对应的代码并不能轻易在其它地方重用,往往这些框架还具有「强排它性」,但是「业务模型」应该是没有过多依赖,应该是无关框架的,它应该随时可以被用在任何合适的 JavaScript 环境中,使用 mota 你可以用原生的普通的 JavaScript 代码编写你的「业务模型」,并让你的「业务模型」在不同框架、不同运行环境下重用更为容易。\n\nMota 是一个响应式的 React 应用状态管理库,基于 Mota 你可以用单纯无依赖的 JavaScript 为应用编写「业务模型」,并轻易的将「业务模型」关联到 React 应用中。\n\n## 示例\n\n[在线 TodoList 示例](http://houfeng.net/dn-template-mota/example/)\n([示例源码](https://github.com/Houfeng/dn-template-mota))\n\n\n\n## 安装\n\n通过 npm 安装,如下\n```sh\n$ npm i mota --save \n```\n\n或通过 `dawn` 脚手脚加创建工程,如下\n\n```sh\n$ mkdir your_path\n$ cd your_path\n$ dn init -t mota\n$ dn dev\n```\n\n需要先安装 dawn([Dawn 安装及使用文档](https://alibaba.github.io/dawn/docs/))\n\n## 结构\n\n一个 `mota` 工程的通常结构如下\n\n```sh\n.\n├── README.md\n├── package.json\n└── src\n ├── assets\n │   ├── common.less\n │   ├── favicon.ico\n │   └── index.html\n ├── components\n │   ├── todoApp.js\n │   └── todoItem.js\n ├── index.js\n └── models\n ├── TodoItem.js\n ├── TodoList.js\n └── index.js\n```\n\n## 文档\n- [快速开始](http://houfeng.net/mota/#!/zh/guide/quick)\n- [编写业务模型](http://houfeng.net/mota/#!/zh/guide/model)\n- [将组件属性映射到模型](http://houfeng.net/mota/#!/zh/guide/mapping)\n- [自执行函数](http://houfeng.net/mota/#!/zh/guide/autorun)\n- [监听模型变化](http://houfeng.net/mota/#!/zh/guide/watch)\n- [将模型数据与表单绑定](http://houfeng.net/mota/#!/zh/guide/binding)\n\n## 链接\n- [版本发布日志](https://github.com/Houfeng/mota/releases)\n- [MIT 开源协议](https://tldrlegal.com/license/mit-license)","filename":"markdowns/quick.md","root":"/Users/Houfeng/my/dev/mota","result":"

Mota

\n

简述

\n

React 是一个「视图层」的 UI 框架,以常见的 MVC 来讲 React 仅是 View,而我们在编写应用时,通常还需要关注更加重要的 model,对于 React 来讲,我们常常需要一个「状态管理」库。然而,目前大多数针对 React 的状态管理库都是「强依赖」过多的侵入本应该独立的业务模型中,导致「业务逻辑」对应的代码并不能轻易在其它地方重用,往往这些框架还具有「强排它性」,但是「业务模型」应该是没有过多依赖,应该是无关框架的,它应该随时可以被用在任何合适的 JavaScript 环境中,使用 mota 你可以用原生的普通的 JavaScript 代码编写你的「业务模型」,并让你的「业务模型」在不同框架、不同运行环境下重用更为容易。

\n

Mota 是一个响应式的 React 应用状态管理库,基于 Mota 你可以用单纯无依赖的 JavaScript 为应用编写「业务模型」,并轻易的将「业务模型」关联到 React 应用中。

\n

示例

\n

在线 TodoList 示例\n(示例源码)

\n\n

安装

\n

通过 npm 安装,如下

\n
$ npm i mota --save \n
\n

或通过 dawn 脚手脚加创建工程,如下

\n
$ mkdir your_path\n$ cd your_path\n$ dn init -t mota\n$ dn dev\n
\n

需要先安装 dawn(Dawn 安装及使用文档

\n

结构

\n

一个 mota 工程的通常结构如下

\n
.\n├── README.md\n├── package.json\n└── src\n    ├── assets\n    │   ├── common.less\n    │   ├── favicon.ico\n    │   └── index.html\n    ├── components\n    │   ├── todoApp.js\n    │   └── todoItem.js\n    ├── index.js\n    └── models\n        ├── TodoItem.js\n        ├── TodoList.js\n        └── index.js\n
\n

文档

\n\n

链接

\n\n"},{"group":"guide","name":"model","title":"编写业务模型","index":2,"source":"# 编写业务模型\n\n在你编写模型之前,先放下 React 也放下 Mota,就用单纯的 JavaScript 去编写你的业务模型,或有一个或多个类、或就是几个 Object,依它们应有的、自然的关系去抽像就行了,业务模型不依赖于 UI、也不依赖于某个框架,它易于测试,你可以针对它做单元测试。它易于重用,你可以将它用在合适的地方。最后, Mota 只是出场把它关联到 React。\n\n在 Mota 中「模型」可以是由一个 `class` 或普通的的 `Object`,整个「业务模型层」会由多个 `class` 和多个 `Object` 组成,而编写模型所需要的知识就是 JavaScript 固有的编程的知识。\n\n如下示例通过编写一个名为 `User` 的 `class` 创建了一个「用户模型」\n\n```js\nexport default class User {\n firstName = 'Jack';\n lastName = 'Hou';\n get fullName(){\n reutrn `${this.firstName} ${this.lastName}`;\n }\n}\n```\n\n也可以是一个 `Object`,通常这个模型需要是「单例」时,可采用这种方式,如下\n\n```js\nexport default {\n firstName: 'Jack',\n lastName: 'Hou',\n get fullName(){\n reutrn `${this.firstName} ${this.lastName}`;\n }\n};\n```\n\n在「业务模型」编写完成后,可以通过 `@model` 将某个「类」或「类的实例」关联到指定组件,关联后便可以在组件中使用 `this.model` 访问「模型的成员变量或方法」了,Mota 还会自动「收集组件依赖」,在组件「依赖的模型数据」发生变化时,自动响应变化并「驱动组件重新渲染」,如下\n\n```js\nimport { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\nclass App extends React.Component {\n\n onChange(field,event){\n this.model[field] = event.target.value;\n }\n\n render(){\n return
\n

{this.model.fullName}

\n

\n \n
\n \n

\n
;\n }\n}\n\nReactDOM.render(, mountNode);\n```\n\n值得注意的是,在使用 `@model` 时如果传入的是一个 `class` 最终每个组件实例都会自动创建一个 `独立的实例`,这样带来的好处是「当一个页面中有同一个组件的多个实例时,不会相互影响」。","filename":"markdowns/model.md","root":"/Users/Houfeng/my/dev/mota","result":"

编写业务模型

\n

在你编写模型之前,先放下 React 也放下 Mota,就用单纯的 JavaScript 去编写你的业务模型,或有一个或多个类、或就是几个 Object,依它们应有的、自然的关系去抽像就行了,业务模型不依赖于 UI、也不依赖于某个框架,它易于测试,你可以针对它做单元测试。它易于重用,你可以将它用在合适的地方。最后, Mota 只是出场把它关联到 React。

\n

在 Mota 中「模型」可以是由一个 class 或普通的的 Object,整个「业务模型层」会由多个 class 和多个 Object 组成,而编写模型所需要的知识就是 JavaScript 固有的编程的知识。

\n

如下示例通过编写一个名为 Userclass 创建了一个「用户模型」

\n
export default class User {\n  firstName = 'Jack';\n  lastName = 'Hou';\n  get fullName(){\n    reutrn `${this.firstName} ${this.lastName}`;\n  }\n}\n
\n

也可以是一个 Object,通常这个模型需要是「单例」时,可采用这种方式,如下

\n
export default {\n  firstName: 'Jack',\n  lastName: 'Hou',\n  get fullName(){\n    reutrn `${this.firstName} ${this.lastName}`;\n  }\n};\n
\n

在「业务模型」编写完成后,可以通过 @model 将某个「类」或「类的实例」关联到指定组件,关联后便可以在组件中使用 this.model 访问「模型的成员变量或方法」了,Mota 还会自动「收集组件依赖」,在组件「依赖的模型数据」发生变化时,自动响应变化并「驱动组件重新渲染」,如下

\n
import { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\nclass App extends React.Component {\n\n  onChange(field,event){\n    this.model[field] = event.target.value;\n  }\n\n  render(){\n    return <div>\n      <p>{this.model.fullName}</p>\n      <p>\n        <input onChange={this.onChange.bind(this,'firstName')}/>\n        <br/>\n        <input onChange={this.onChange.bind(this,'lastName')}/>\n      </p>\n    </div>;\n  }\n}\n\nReactDOM.render(<App/>, mountNode);\n
\n

值得注意的是,在使用 @model 时如果传入的是一个 class 最终每个组件实例都会自动创建一个 独立的实例,这样带来的好处是「当一个页面中有同一个组件的多个实例时,不会相互影响」。

\n"},{"group":"guide","name":"mapping","title":"属性映射","index":3,"source":"# 属性映射\n\n在 React 中通常会将应用折分为多个组件重用它们,并在用时传递给它「属性」,Mota 提供了将「组件属性」映射到「模型数据」的能力,基于 `model` 编程会让「视图层」的编写更为方例,专注于 UI 的呈现,如下\n\n```js\n@model({ value: 'demo' })\n@mapping(['value'])\nclass Demo extends React.Component {\n render () {\n return
{this.model.value}
;\n }\n}\n```\n\n上边的代码通过 `mapping` 将 `Demo` 这个组件的 `value` 属性映射到了 `model.value` 上,在组件的属性 `value` 发生变化时,会自动同步到 `model.value` 中。\n\n通过一个 `map` 进行映射,还可以让「组件属性」和「模型的成员」使用不同名称,如下:\n\n```js\n@model({ value: 'demo' })\n@mapping({ content: 'value' })\nclass Demo extends React.Component {\n render () {\n return
{this.model.value}
;\n }\n}\n```\n\n上边的代码,将组件 demo 的 `content` 属性映射到了 `model.value` 上,那么这个组件就可以这样使用了\n\n```js\nfunction App(){\n return ;\n}\n```\n\n`Demo` 组件的 `content` 属性,将自动被赋值给 `model.value`,如果没有 `mapping`,通常我们就需要在 `componentDidMount` 和 `componentWillReceiveProps` 之类的生命周函数去处理。其实,`mapping` 就像是一个语法糖,使用它将不再需要手动处理 prop->model 的更新了。","filename":"markdowns/mapping.md","root":"/Users/Houfeng/my/dev/mota","result":"

属性映射

\n

在 React 中通常会将应用折分为多个组件重用它们,并在用时传递给它「属性」,Mota 提供了将「组件属性」映射到「模型数据」的能力,基于 model 编程会让「视图层」的编写更为方例,专注于 UI 的呈现,如下

\n
@model({ value: 'demo' })\n@mapping(['value'])\nclass Demo extends React.Component {\n  render () {\n    return <div>{this.model.value}</div>;\n  }\n}\n
\n

上边的代码通过 mappingDemo 这个组件的 value 属性映射到了 model.value 上,在组件的属性 value 发生变化时,会自动同步到 model.value 中。

\n

通过一个 map 进行映射,还可以让「组件属性」和「模型的成员」使用不同名称,如下:

\n
@model({ value: 'demo' })\n@mapping({ content: 'value' })\nclass Demo extends React.Component {\n  render () {\n    return <div>{this.model.value}</div>;\n  }\n}\n
\n

上边的代码,将组件 demo 的 content 属性映射到了 model.value 上,那么这个组件就可以这样使用了

\n
function App(){\n  return <Demo content={'yyyy'} />;\n}\n
\n

Demo 组件的 content 属性,将自动被赋值给 model.value,如果没有 mapping,通常我们就需要在 componentDidMountcomponentWillReceiveProps 之类的生命周函数去处理。其实,mapping 就像是一个语法糖,使用它将不再需要手动处理 prop->model 的更新了。

\n"},{"group":"guide","name":"autorun","title":"自执行函数","index":4,"source":"# 自执行函数\n\nMota 中提供了一个 `autorun` 函数,可用于装饰 React 组件的成员方法,被装饰的「成员方法」将会在组件挂载后自动执行一次,Mota 将「收集方法中依赖的模型数据」,在依赖的模型数据发生变化时会「自动重新执行」对应的组件方法。\n\n示例\n\n```js\nimport { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n @autorun\n test() {\n console.log(this.model.name);\n }\n\n}\n```\n\n上边的示例代码中,组件在被挂载后将会自动执行 `test` 方法,同时 mota 会发现方法中依赖了 `model.name`,那么,在 `model.name` 发生变化时,就会重新执行 `test` 方法。","filename":"markdowns/autorun.md","root":"/Users/Houfeng/my/dev/mota","result":"

自执行函数

\n

Mota 中提供了一个 autorun 函数,可用于装饰 React 组件的成员方法,被装饰的「成员方法」将会在组件挂载后自动执行一次,Mota 将「收集方法中依赖的模型数据」,在依赖的模型数据发生变化时会「自动重新执行」对应的组件方法。

\n

示例

\n
import { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n  @autorun\n  test() {\n    console.log(this.model.name);\n  }\n\n}\n
\n

上边的示例代码中,组件在被挂载后将会自动执行 test 方法,同时 mota 会发现方法中依赖了 model.name,那么,在 model.name 发生变化时,就会重新执行 test 方法。

\n"},{"group":"guide","name":"watch","title":"监听模型变化","index":5,"source":"# 监听模型变化\n\n\nMota 中提供了一个 `watch` 函数,可用于装饰 React 组件的成员方法,`watch` 可以指定要观察的「模型数据」,在模型数据发变化时,就会自动执行「被装饰的组件方法」,`watch` 还可以像 `autorun` 一样自动执行一次,但它和 `autorun` 还是不尽相同,主要有如下区别\n\n- `autorun` 会自动收集依赖,而 `watch` 不会关心组件方法中有何依赖,需要手动指定依赖的模型数据\n- `watch` 默认不会「自动执行」,需显式的指定「立即执行参数为 true」,才会自动执行首次。\n- `autorun` 依赖的是「模型数据」本身,而 `watch` 依赖的是「计算函数」每次的「计算结果」\n \n示例\n\n```js\nimport { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n @watch(model=>model.name)\n test() {\n console.log('name 发生了变化');\n }\n\n}\n```\n\n上边的代码,通过 `watch` 装饰了 `test` 方法,并指定了观察的模型数据 `model.name`,那么每当 `model.name` 发生变\b化时,都会打印 `name 发生了变化`.\n\n`watch` 是否重新执行,取决于 `watch` 的作为第一个参数传给它的「计算函数」的计算结果,每当依赖的模型数据发生变化时 `watch` 都会重执行计算函数,当计算结果有变化时,才会执行被装饰的「组件方法」,示例\n\n```js\nexport default Demo extends Component {\n\n @watch(model=>model.name+model.age)\n test() {\n console.log('name 发生变化');\n }\n\n}\n```\n\n有时,我们希望 `watch` 能首先自动执行一次,那么可通过向第二个参数传一个 `true` 声明这个 `watch` 要自动执行一次。\n\n```js\nexport default Demo extends Component {\n\n @watch(model=>model.name,true)\n test() {\n console.log('name 发生变化');\n }\n\n}\n```\n\n上边的 `test` 方法,将会在「组件挂载之后自动执行」,之后在 `model.name` 发生变化时也将自动重新执行。","filename":"markdowns/watch.md","root":"/Users/Houfeng/my/dev/mota","result":"

监听模型变化

\n

Mota 中提供了一个 watch 函数,可用于装饰 React 组件的成员方法,watch 可以指定要观察的「模型数据」,在模型数据发变化时,就会自动执行「被装饰的组件方法」,watch 还可以像 autorun 一样自动执行一次,但它和 autorun 还是不尽相同,主要有如下区别

\n
    \n
  • autorun 会自动收集依赖,而 watch 不会关心组件方法中有何依赖,需要手动指定依赖的模型数据
  • \n
  • watch 默认不会「自动执行」,需显式的指定「立即执行参数为 true」,才会自动执行首次。
  • \n
  • autorun 依赖的是「模型数据」本身,而 watch 依赖的是「计算函数」每次的「计算结果」
  • \n
\n

示例

\n
import { Component } from 'react';\nimport { model, autorun } from 'mota';\nimport DemoModel from './models/demo';\n\n@model(DemoModel)\nexport default Demo extends Component {\n\n  @watch(model=>model.name)\n  test() {\n    console.log('name 发生了变化');\n  }\n\n}\n
\n

上边的代码,通过 watch 装饰了 test 方法,并指定了观察的模型数据 model.name,那么每当 model.name 发生变\b化时,都会打印 name 发生了变化.

\n

watch 是否重新执行,取决于 watch 的作为第一个参数传给它的「计算函数」的计算结果,每当依赖的模型数据发生变化时 watch 都会重执行计算函数,当计算结果有变化时,才会执行被装饰的「组件方法」,示例

\n
export default Demo extends Component {\n\n  @watch(model=>model.name+model.age)\n  test() {\n    console.log('name 发生变化');\n  }\n\n}\n
\n

有时,我们希望 watch 能首先自动执行一次,那么可通过向第二个参数传一个 true 声明这个 watch 要自动执行一次。

\n
export default Demo extends Component {\n\n  @watch(model=>model.name,true)\n  test() {\n    console.log('name 发生变化');\n  }\n\n}\n
\n

上边的 test 方法,将会在「组件挂载之后自动执行」,之后在 model.name 发生变化时也将自动重新执行。

\n"},{"group":"guide","name":"binding","title":"数据绑定","index":6,"source":"# 数据绑定\n\n\n### 基本用法\n\n不要惊诧,就是「双向绑定」。Mota 不排斥「双向绑定」,使用 Mota 能够实现类似 `ng` 或 `vue` 的绑定效果。还是前边小节中的模型,我们来稍微改动一下组件的代码\n\n```js\nimport { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\n@binding\nclass App extends React.Component {\n render(){\n const { fullName, firstName, popup } = this.model;\n return
\n

{fullName}

\n

\n \n \n

\n
;\n }\n}\nReactDOM.render(, mountNode);\n```\n\n~~其中的「关键」就是 `@binding`,使用 `@binding` 后~~ (>=1.2.0 的版本将会自动处理,不必显示的启用),组件便具备了「双向绑定」的能力,在 `jsx` 中便可以通过名为 `data-bind` 的自定义 `attribute` 进行绑定了,`data-bind` 的值是一个「绑定表达式字符串」,绑定表达式执行的 `scope` 是 `model` 而不是 `this`,也就是只能与 `模型的成员` 进行绑定。\n\n会有一种情况是当要绑定的数据是一个循环变量时,「绑定表达式」写起会较麻烦也稍显长,比如\n\n```js\n@model(userModel)\n@binding\nclass App extends React.Component {\n render(){\n const { userList } = this.model;\n return
    \n {userList.map((user,index)=>(\n
  • \n \n {user.name}\n
  • \n ))}\n
;\n }\n}\n```\n\n因为「绑定表达式」的执行 `scope` 默认是 `this.model`,以及「表达式是个字符串」,看一下 `userList[${index}].selected` 这并不友好,为此 Mota 还提供了一个名为 `data-scope` 的 `attribute`,通过它能改变要绑定的 `scope`,参考如下示例\n\n```js\n@model(userModel)\n@binding\nclass App extends React.Component {\n render(){\n const { userList } = this.model;\n return
    \n {userList.map(user=>(\n
  • \n \n {user.name}\n
  • \n ))}\n
;\n }\n}\n```\n\n通过 `data-scope` 将 `input` 的绑定上下文对象声明为当前循环变量 `user`,这样就可以用 `data-bind` 直接绑定到对应 `user` 的属性上了。\n\n### 原生表单控件\n\n所有的原生表单控件,比如「普通 input、checkbox、radio、textarea、select」都可以直接进行绑定。其中,「普通 input 和 textrea」比较简单,将一个字符类型的模型数据与控件绑定就行了,而对于「checkbox 和 radio」 有多种不同的绑定形式。\n\n将「checkbox 或 radio」绑定到一个 `boolean` 值,此时会将 checkbox 或 radio 的 `checked` 属性和模型数据建立绑定,`checked` 反应了 `boolean` 变量的值,参考如下示例\n\n```js\n@model({ selected:false })\n@binding\nclass App extends React.Component {\n render(){\n return
\n \n \n
;\n }\n}\n```\n\n如上示例通过 `this.model.selected` 就能拿到当前 checkbox 或 radio 的选中状态。\n\n\n将 checkbox 绑定到一个「数组」,通常是多个 checkbox 绑定同一个数组变量上,此时和数据建立绑定的是 checkbox 的 value,数据中会包含当前选中的 checkbox 的 value,如下\n\n```js\n@model({ selected:[] })\n@binding\nclass App extends React.Component {\n render(){\n return
\n \n \n
;\n }\n}\n```\n\n如上示例,通过 `this.selected` 就能知道当前有哪些 checkbox 被选中了,并拿到所有选中的 value\n\n\n将多个 radio 绑定我到一个「字符类型的变量」,此时和数据建立绑定的是 raido 的 value,因为 radio 是单选的,所以对应的数据是当前选中的 radio 的 value,如下\n\n```js\n@model({ selected:'' })\n@binding\nclass App extends React.Component {\n render(){\n return
\n \n \n
;\n }\n}\n```\n通过 `this.model.selected` 就能拿到当前选中的 radio 的 `value`\n\n\n### 自定义组件\n\n但是对于一些「组件库」中的「部分表单组件」不能直接绑定,因为 Mota 并没有什么依据可以判断这是一个什么组件。所以 Mota 提供了一个名为 `bindable` 的函数,用将任意组件包装成「可绑定组件」。\n\nbindable 有两种个参数,用于分别指定「原始组件」和「包装选项」\n\n```js\n//可以这样\nconst MyComponent = bindable(opts, Component);\n//也可这样\nconst MyCompoent = bindable(Component, opts);\n```\n\n关建是 `bindable` 需要的 `opts`,通过 `opts` 我们可以造诉 Mota 如何绑定这个组件,`opts` 中有两个重要的成员,它的结构如下\n\n```js\n{\n value: ['value 对应的属性名'],\n event: ['value 改变的事件名']\n}\n```\n\n所以,我们可以这样包装一个自定义文本输入框\n\n```js\nconst MyInput = bindable(Input,{\n value: ['value'],\n event: ['onChange']\n});\n```\n\n对这种「value 不需要转换,`change` 能通过 `event` 或 `event.target.value` 拿到值」的组件,通过如上的代码就能完成包装了。\n\n对于有 `onChange` 和 `value` 的这类文本输入组件,因为 opts 的默认值就是\n\n```js\n{\n value: ['value'],\n event: ['onChange']\n}\n```\n\n所以,可以更简单,这样就行,\n```js\nconst MyInput = bindable(Input);\n```\n\n而对于 checkbox 和 radio 来讲,如上边讲到的它「根据不同的数据型有不同的绑定形式」,这就需要指定处理函数了,如下\n\n```js\nconst radioOpts = {\n prop: ['checked', (ctx, props) => {\n const mValue = ctx.getValue();\n if (typeof mValue == 'boolean') {\n return !!mValue;\n } else {\n return mValue == props.value;\n }\n }],\n event: ['onChange', (ctx, event) => {\n const { value, checked } = event.target;\n const mValue = ctx.getValue();\n if (typeof mValue == 'boolean') {\n ctx.setValue(checked);\n } else if (checked) ctx.setValue(value);\n }]\n};\n```\n\n通过 `prop` 的第二个值,能指定「属性处理函数」,`event` 的第二个值能指取「事件处理函数」,处理函数的 `ctx` 是个特殊的对象 \n\n- `ctx.getValue` 能获取「当前绑定的模型数据」\n- `ctx.setValue` 能设置「当前绑定的模型数据」\n\n上边是 `radio` 的配置,首先,在「属性处理函数」中通过绑定的「模型数据的类型」决定 `checked` 最终的状态是什么,并在函数中返回。再次,在「事件处理函数」中通过绑定的「模型数据的类型」决定将什么值回写到模型中。\n\n通过「属性处理函数」和「事件处理函数」几乎就能将任意的自定义组件转换为「可绑定组件」了。\n\n另外,对于常见的 `CheckBox` 和 `Radio` 类型的组件 Mota 也提供了内建的 `opts` 配置支持,如果一个自定义组件拥有和「原生 checkbox 一致的属性和事件模型」,那边可以直接用简单的方式去包装,如下\n\n```js\nconst MyCheckBox = bindable('checkbox',CheckBox);\nconst MyRadio = bindable('radio',Radio);\n```\n\n好了,关于绑定就这些了。","filename":"markdowns/binding.md","root":"/Users/Houfeng/my/dev/mota","result":"

数据绑定

\n

基本用法

\n

不要惊诧,就是「双向绑定」。Mota 不排斥「双向绑定」,使用 Mota 能够实现类似 ngvue 的绑定效果。还是前边小节中的模型,我们来稍微改动一下组件的代码

\n
import { model,binding } from 'mota';\nimport React from 'react';\nimport ReactDOM from 'react-dom';\nimport User from './models/user';\n\n@model(User)\n@binding\nclass App extends React.Component {\n  render(){\n    const { fullName, firstName, popup } = this.model;\n    return <div>\n      <p>{fullName}</p>\n      <p>\n        <input data-bind=\"firstName\"/>\n        <button onClick={popup}> click me </button>\n      </p>\n    </div>;\n  }\n}\nReactDOM.render(<App/>, mountNode);\n
\n

其中的「关键」就是 @binding,使用 @binding (>=1.2.0 的版本将会自动处理,不必显示的启用),组件便具备了「双向绑定」的能力,在 jsx 中便可以通过名为 data-bind 的自定义 attribute 进行绑定了,data-bind 的值是一个「绑定表达式字符串」,绑定表达式执行的 scopemodel 而不是 this,也就是只能与 模型的成员 进行绑定。

\n

会有一种情况是当要绑定的数据是一个循环变量时,「绑定表达式」写起会较麻烦也稍显长,比如

\n
@model(userModel)\n@binding\nclass App extends React.Component {\n  render(){\n    const { userList } = this.model;\n    return <ul>\n     {userList.map((user,index)=>(\n       <li key={user.id}>\n         <input type=\"checkobx\" data-bind={`userList[${index}].selected`}>\n         {user.name}\n       </li>\n     ))}\n    </ul>;\n  }\n}\n
\n

因为「绑定表达式」的执行 scope 默认是 this.model,以及「表达式是个字符串」,看一下 userList[${index}].selected 这并不友好,为此 Mota 还提供了一个名为 data-scopeattribute,通过它能改变要绑定的 scope,参考如下示例

\n
@model(userModel)\n@binding\nclass App extends React.Component {\n  render(){\n    const { userList } = this.model;\n    return <ul>\n     {userList.map(user=>(\n       <li key={user.id}>\n         <input type=\"checkobx\" data-scope={user} data-bind=\"selected\">\n         {user.name}\n       </li>\n     ))}\n    </ul>;\n  }\n}\n
\n

通过 data-scopeinput 的绑定上下文对象声明为当前循环变量 user,这样就可以用 data-bind 直接绑定到对应 user 的属性上了。

\n

原生表单控件

\n

所有的原生表单控件,比如「普通 input、checkbox、radio、textarea、select」都可以直接进行绑定。其中,「普通 input 和 textrea」比较简单,将一个字符类型的模型数据与控件绑定就行了,而对于「checkbox 和 radio」 有多种不同的绑定形式。

\n

将「checkbox 或 radio」绑定到一个 boolean 值,此时会将 checkbox 或 radio 的 checked 属性和模型数据建立绑定,checked 反应了 boolean 变量的值,参考如下示例

\n
@model({ selected:false })\n@binding\nclass App extends React.Component {\n  render(){\n    return <div>\n      <input type=\"checkbox\" data-bind=\"selected\"/>\n      <input type=\"radio\" data-bind=\"selected\"/>\n    </div>;\n  }\n}\n
\n

如上示例通过 this.model.selected 就能拿到当前 checkbox 或 radio 的选中状态。

\n

将 checkbox 绑定到一个「数组」,通常是多个 checkbox 绑定同一个数组变量上,此时和数据建立绑定的是 checkbox 的 value,数据中会包含当前选中的 checkbox 的 value,如下

\n
@model({ selected:[] })\n@binding\nclass App extends React.Component {\n  render(){\n    return <div>\n      <input type=\"checkbox\" data-bind=\"selected\" value=\"1\"/>\n      <input type=\"checkbox\" data-bind=\"selected\" value=\"2\"/>\n    </div>;\n  }\n}\n
\n

如上示例,通过 this.selected 就能知道当前有哪些 checkbox 被选中了,并拿到所有选中的 value

\n

将多个 radio 绑定我到一个「字符类型的变量」,此时和数据建立绑定的是 raido 的 value,因为 radio 是单选的,所以对应的数据是当前选中的 radio 的 value,如下

\n
@model({ selected:'' })\n@binding\nclass App extends React.Component {\n  render(){\n    return <div>\n      <input type=\"radio\" data-bind=\"selected\" value=\"1\"/>\n      <input type=\"radio\" data-bind=\"selected\" value=\"2\"/>\n    </div>;\n  }\n}\n
\n

通过 this.model.selected 就能拿到当前选中的 radio 的 value

\n

自定义组件

\n

但是对于一些「组件库」中的「部分表单组件」不能直接绑定,因为 Mota 并没有什么依据可以判断这是一个什么组件。所以 Mota 提供了一个名为 bindable 的函数,用将任意组件包装成「可绑定组件」。

\n

bindable 有两种个参数,用于分别指定「原始组件」和「包装选项」

\n
//可以这样\nconst MyComponent = bindable(opts, Component);\n//也可这样\nconst MyCompoent = bindable(Component, opts);\n
\n

关建是 bindable 需要的 opts,通过 opts 我们可以造诉 Mota 如何绑定这个组件,opts 中有两个重要的成员,它的结构如下

\n
{\n  value: ['value 对应的属性名'],\n  event: ['value 改变的事件名']\n}\n
\n

所以,我们可以这样包装一个自定义文本输入框

\n
const MyInput = bindable(Input,{\n  value: ['value'],\n  event: ['onChange']\n});\n
\n

对这种「value 不需要转换,change 能通过 eventevent.target.value 拿到值」的组件,通过如上的代码就能完成包装了。

\n

对于有 onChangevalue 的这类文本输入组件,因为 opts 的默认值就是

\n
{\n  value: ['value'],\n  event: ['onChange']\n}\n
\n

所以,可以更简单,这样就行,

\n
const MyInput = bindable(Input);\n
\n

而对于 checkbox 和 radio 来讲,如上边讲到的它「根据不同的数据型有不同的绑定形式」,这就需要指定处理函数了,如下

\n
const radioOpts = {\n  prop: ['checked', (ctx, props) => {\n    const mValue = ctx.getValue();\n    if (typeof mValue == 'boolean') {\n      return !!mValue;\n    } else {\n      return mValue == props.value;\n    }\n  }],\n  event: ['onChange', (ctx, event) => {\n    const { value, checked } = event.target;\n    const mValue = ctx.getValue();\n    if (typeof mValue == 'boolean') {\n      ctx.setValue(checked);\n    } else if (checked) ctx.setValue(value);\n  }]\n};\n
\n

通过 prop 的第二个值,能指定「属性处理函数」,event 的第二个值能指取「事件处理函数」,处理函数的 ctx 是个特殊的对象

\n
    \n
  • ctx.getValue 能获取「当前绑定的模型数据」
  • \n
  • ctx.setValue 能设置「当前绑定的模型数据」
  • \n
\n

上边是 radio 的配置,首先,在「属性处理函数」中通过绑定的「模型数据的类型」决定 checked 最终的状态是什么,并在函数中返回。再次,在「事件处理函数」中通过绑定的「模型数据的类型」决定将什么值回写到模型中。

\n

通过「属性处理函数」和「事件处理函数」几乎就能将任意的自定义组件转换为「可绑定组件」了。

\n

另外,对于常见的 CheckBoxRadio 类型的组件 Mota 也提供了内建的 opts 配置支持,如果一个自定义组件拥有和「原生 checkbox 一致的属性和事件模型」,那边可以直接用简单的方式去包装,如下

\n
const MyCheckBox = bindable('checkbox',CheckBox);\nconst MyRadio = bindable('radio',Radio);\n
\n

好了,关于绑定就这些了。

\n"},{"group":"guide","name":"use_model","title":"使用 Hook API","index":7,"source":"# 使用 Hook API\n\n### 简单介绍\n在 React 发布包含 `Hooks` 的 `alpah` 版后,Mota 也在 `next` 版本中新增支持了 Hook 风格的 API,随着 React `v16.8` 版本的发布带来了稳定版的 `Hooks` 支持。\n\n目前,Mota \b已经在稳定版中,提供了 `Hook API` 的支持,利用 React 的 `Hooks` 可以让你在不编写类的情况下使用 `state` 和 React 的其他功能。而使用 Mota 极少的 `Hook API` 将给应用带来 Hook 风格可响应的全局状态管理支持。\n\n### 基本用法\n\n```js\nimport React from 'react';\nimport { render } from 'react-dom';\n\nfunction App(){\n //通过 useModel 拿到一个可响应的 model\n const model = useModel({ count:0 });\n //定义累加按钮事件\n const onClick = useCallback(()=>model.count++);\n //--\n return
\n
{model.count}
\n \n
;\n}\n\nrender(, document.getElementId('root'));\n```\n\n仅有一个新增 API `useModel`,通过 `useModel` 可在一个 `Function Component` 中使用 `model`,如同在 `Class Component` 中的 `@model`,此时的 `model` 依然是可响应的,执行时会对组件进行「依赖收集」,当操作 `model` 的成员时(比如 `model.count=1` 的赋值操作),Mota 会自动发现组件依赖的数据发生了变化并通知组件进行更新。\n\n### 进阶说明\n\n在基本用法中提到了一个关键词「依赖收集」,通过 `useModel` 拿到了可响应的 `model`,默认情况下只有被组件依赖的模型数据发生了变化组件才会更新,比如下边的示例代码中,只有在 `model.a` 发生变化时,组件才会重新渲染。\n\n\n```js\nfunction Demo(){\n const model = useModel({ a:0, b:1 });\n ...\n return
{model.a}
\n}\n```\n\n实际开发过程中有时「组件依赖了模型上的某个对象,但希望这个对象的子成员发生变化时,组件也要重新渲染」\n\n```js\nfunction Demo(){\n const { info } = useModel({ info: { name: 'test'} });\n ...\n return \n}\n```\n\n因为对于 `Demo` 来说只依赖了 `info`,而 `info` 的引用是一直没有变化的,所以在 `info.name` 发生变化时 `Demo` 并不会重新渲染。那这样 `Info` 组件会一直显示旧的数据。\n\n如何处理这个问题?\n\n一个方法是让 `Info` 也通过 `useModel` 有自已的 `model`,那 `Info` 的依赖会被独立解析,比如\n\n```js\nfunction Info(props){\n const info = useModel(props.data);\n return
{info.name}
\n}\n```\n\n这个虽然 `Demo` 不会重新渲染,但 Mota 会发现 `Info` 依赖了 `info.name`,但发现数据变化时,`Info` 会自动更新。\n\n还有一个方法是,在更新 `info.name` 时换一个写法\n\n```js\n//通常的直接给 name 赋值\nmodel.info.name = 'test';\n//如下的给 info 赋值的写法,会让 Demo 发现 info 的变化\nmodel.info = {...model.info, name: 'test'};\n```\n\n除了上述的两个方法,还有一个方法就是通过 `useModel` 的第二个参数显示的声明额外的依赖,第二个参数可是一个数组,数组中是显式声明的依赖,格式为子成员的路径,如下\n\n```js\nfunction Demo(){\n const { info } = useModel({ \n info: { name: 'test'} \n }, ['info.name']);\n ...\n return \n}\n```\n\n但有时时模型数据是一个数组,我们无法直接指定每个子元素的路径,这时第二个参数还可以是一个函数,函数的参数是「变化的模型数据的路径」,可参函数中返回 `boolean` 值决定是否需要更新组件,如下\n\n```js\nfunction Demo(){\n const { info } = useModel({ \n info: [{name: 'test1'}] \n }, p=> p.endsWith('.name'));\n ...\n return \n}\n```\n\n示例中通过用 `endsWith` 路径是不是 `*.name` 结尾的决定要不要更新,当然也可以用更多的判断方法决定要不要更新。","filename":"markdowns/use_model.md","root":"/Users/Houfeng/my/dev/mota","result":"

使用 Hook API

\n

简单介绍

\n

在 React 发布包含 Hooksalpah 版后,Mota 也在 next 版本中新增支持了 Hook 风格的 API,随着 React v16.8 版本的发布带来了稳定版的 Hooks 支持。

\n

目前,Mota \b已经在稳定版中,提供了 Hook API 的支持,利用 React 的 Hooks 可以让你在不编写类的情况下使用 state 和 React 的其他功能。而使用 Mota 极少的 Hook API 将给应用带来 Hook 风格可响应的全局状态管理支持。

\n

基本用法

\n
import React from 'react';\nimport { render } from 'react-dom';\n\nfunction App(){\n  //通过 useModel 拿到一个可响应的 model\n  const model = useModel({ count:0 });\n  //定义累加按钮事件\n  const onClick = useCallback(()=>model.count++);\n  //--\n  return <div>\n    <div>{model.count}</div>\n    <button onClick={onClick}>click me</button>\n  </div>;\n}\n\nrender(<App/>, document.getElementId('root'));\n
\n

仅有一个新增 API useModel,通过 useModel 可在一个 Function Component 中使用 model,如同在 Class Component 中的 @model,此时的 model 依然是可响应的,执行时会对组件进行「依赖收集」,当操作 model 的成员时(比如 model.count=1 的赋值操作),Mota 会自动发现组件依赖的数据发生了变化并通知组件进行更新。

\n

进阶说明

\n

在基本用法中提到了一个关键词「依赖收集」,通过 useModel 拿到了可响应的 model,默认情况下只有被组件依赖的模型数据发生了变化组件才会更新,比如下边的示例代码中,只有在 model.a 发生变化时,组件才会重新渲染。

\n
function Demo(){\n  const model = useModel({ a:0, b:1 });\n  ...\n  return <div>{model.a}</div>\n}\n
\n

实际开发过程中有时「组件依赖了模型上的某个对象,但希望这个对象的子成员发生变化时,组件也要重新渲染」

\n
function Demo(){\n  const { info } = useModel({ info: { name: 'test'} });\n  ...\n  return <Info data={info}/>\n}\n
\n

因为对于 Demo 来说只依赖了 info,而 info 的引用是一直没有变化的,所以在 info.name 发生变化时 Demo 并不会重新渲染。那这样 Info 组件会一直显示旧的数据。

\n

如何处理这个问题?

\n

一个方法是让 Info 也通过 useModel 有自已的 model,那 Info 的依赖会被独立解析,比如

\n
function Info(props){\n  const info = useModel(props.data);\n  return <div>{info.name}</div>\n}\n
\n

这个虽然 Demo 不会重新渲染,但 Mota 会发现 Info 依赖了 info.name,但发现数据变化时,Info 会自动更新。

\n

还有一个方法是,在更新 info.name 时换一个写法

\n
//通常的直接给 name 赋值\nmodel.info.name = 'test';\n//如下的给 info 赋值的写法,会让 Demo 发现 info 的变化\nmodel.info = {...model.info, name: 'test'};\n
\n

除了上述的两个方法,还有一个方法就是通过 useModel 的第二个参数显示的声明额外的依赖,第二个参数可是一个数组,数组中是显式声明的依赖,格式为子成员的路径,如下

\n
function Demo(){\n  const { info } = useModel({ \n    info: { name: 'test'} \n  }, ['info.name']);\n  ...\n  return <Info data={info}/>\n}\n
\n

但有时时模型数据是一个数组,我们无法直接指定每个子元素的路径,这时第二个参数还可以是一个函数,函数的参数是「变化的模型数据的路径」,可参函数中返回 boolean 值决定是否需要更新组件,如下

\n
function Demo(){\n  const { info } = useModel({ \n    info: [{name: 'test1'}] \n  }, p=> p.endsWith('.name'));\n  ...\n  return <Info data={info}/>\n}\n
\n

示例中通过用 endsWith 路径是不是 *.name 结尾的决定要不要更新,当然也可以用更多的判断方法决定要不要更新。

\n"},{"group":"guide","name":"hook_model","title":"面向 Hook 的模型","index":8,"source":"# 面向 Hook 的模型\n\n### 一些说明\n\n针对 Mota 的 `useModel` 的模型本质上和针对 `@model` 的模型并无本质区别,用以往的风格编写的「模型类」或「单例的普通 Object」,除了能用于 `@model` 也是能用于 `useModel` 的。\n\n既然用了 `Hook API`,是不是可在编写模型也避免再写「类」或「单例的 Object」?为此 `useModel` 还提供了用 `ES Module` 作为模型的支持,如用其他风格的模型一样,`ES Module` 风格的模型也不需要引用额外的依赖,仅用 `ES` 原生语法即可。\n\n### 用 ES Module 编写模型\n\n通过 `ES Module` 直接作为 `model` 的优点时「简单、直接」,同时由于为了保证「可响应」不被破坏,需要一点点约束,就是「必须导出一个 state 对象」,如下\n\n```js\nexport const state = {\n name: 'test'\n}\n\nexport function setName(name){\n state.name = name\n}\n```\n\n上边的代码是一个最简单的可当作 `model` 的 `ES Module`,一个包含「state 和一组件函数」的 `ES Module` 就是一个可被 `useModel` 使用的 `model`,参考如下代码\n\n```js\nimport * as demo from './models/demo';\n\nfunction App(){\n const { name } = useModel(demo);\n return
{name}
;\n}\n```\n\n注意:必须 export 一个 state 的约束只针对于 `ES Module`,用 `class` 或 `object` 的风格编写的 `model` 无任何约束。","filename":"markdowns/hook_model.md","root":"/Users/Houfeng/my/dev/mota","result":"

面向 Hook 的模型

\n

一些说明

\n

针对 Mota 的 useModel 的模型本质上和针对 @model 的模型并无本质区别,用以往的风格编写的「模型类」或「单例的普通 Object」,除了能用于 @model 也是能用于 useModel 的。

\n

既然用了 Hook API,是不是可在编写模型也避免再写「类」或「单例的 Object」?为此 useModel 还提供了用 ES Module 作为模型的支持,如用其他风格的模型一样,ES Module 风格的模型也不需要引用额外的依赖,仅用 ES 原生语法即可。

\n

用 ES Module 编写模型

\n

通过 ES Module 直接作为 model 的优点时「简单、直接」,同时由于为了保证「可响应」不被破坏,需要一点点约束,就是「必须导出一个 state 对象」,如下

\n
export const state = {\n  name: 'test'\n}\n\nexport function setName(name){\n  state.name = name\n}\n
\n

上边的代码是一个最简单的可当作 modelES Module,一个包含「state 和一组件函数」的 ES Module 就是一个可被 useModel 使用的 model,参考如下代码

\n
import * as demo from './models/demo';\n\nfunction App(){\n  const { name } = useModel(demo);\n  return <div>{name}</div>;\n}\n
\n

注意:必须 export 一个 state 的约束只针对于 ES Module,用 classobject 的风格编写的 model 无任何约束。

\n"},{"group":"guide","name":"typescript","title":"在 TS 中使用","index":8,"source":"# 在 TS 中使用 Mota\n\nMota 的 `Package` 中自带了「类型定义文件」,无论使用 `Class + Decorator` 风格的 API 或使用 `Hooks` 风格的 API,都能愉快的使用 TypeScript,下边有两个小提示。\n\n### 提示一:使用 @model \n\n在通过 `@model` 为组件关联了一个 `model` 后,需要声明 `this.model` 的类型,参考如下代码\n\n```js\nimport * as React from \"react\";\nimport { model, watch, mapping } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\n@model(DemoModel)\nexport class Demo extends React.Component {\n\n //需要声明 model 的类型\n model: DemoModel;\n\n render() {\n //便能让 this.model 具备完整的类型提示了\n const { name } = this.model;\n return
\n {name} \n
;\n }\n}\n```\n\n### 提示一:使用 useModel\n\n完整的 `useModel` 的定义为 `useModel(model:T)=>T`,但使用 `useModel` 时一般不需做特别的声明,默认情况下 `TS` 就能完成类型推导\n\n```js\nimport * as React from \"react\";\nimport { useModel } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\nexport function Demo {\n const { name } = useModel(DemoModel);\n return
\n {name} \n
;\n}\n```","filename":"markdowns/typescript.md","root":"/Users/Houfeng/my/dev/mota","result":"

在 TS 中使用 Mota

\n

Mota 的 Package 中自带了「类型定义文件」,无论使用 Class + Decorator 风格的 API 或使用 Hooks 风格的 API,都能愉快的使用 TypeScript,下边有两个小提示。

\n

提示一:使用 @model

\n

在通过 @model 为组件关联了一个 model 后,需要声明 this.model 的类型,参考如下代码

\n
import * as React from \"react\";\nimport { model, watch, mapping } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\n@model(DemoModel)\nexport class Demo extends React.Component {\n\n  //需要声明 model 的类型\n  model: DemoModel;\n\n  render() {\n    //便能让 this.model 具备完整的类型提示了\n    const { name } = this.model;\n    return <div className=\"demo\">  \n      {name}  \n    </div>;\n  }\n}\n
\n

提示一:使用 useModel

\n

完整的 useModel 的定义为 useModel<T>(model:T)=>T,但使用 useModel 时一般不需做特别的声明,默认情况下 TS 就能完成类型推导

\n
import * as React from \"react\";\nimport { useModel } from \"mota\";\nimport { DemoModel } from \"./DemoModel\";\n\nexport function Demo {\n  const { name } = useModel(DemoModel);\n  return <div className=\"demo\">  \n    {name}  \n  </div>;\n}\n
\n"}]}],"links":[{"text":"状态管理 (Mota)","url":"//houfeng.net/mota/"},{"text":"数据验证 (Validation)","url":"//houfeng.net/mota-validation/"},{"text":"表单组件 (Form)","url":"//houfeng.net/mota-form/"},{"text":"源码 (GitHub)","url":"//github.com/Houfeng/mota"}]}],"plugins":[{"name":"doczilla-place","options":{}},{"name":"doczilla-include","options":{}},{"name":"doczilla-container","options":{}},{"name":"doczilla-highlight","options":{}},{"name":"doczilla-details","options":{}},{"name":"doczilla-card","options":{}},{"name":"doczilla-anchor","options":{}}],"mode":"hash","baseUri":"","extname":""}; \ No newline at end of file diff --git a/docs/index.html b/docs/index.html index b251c3c..47ba01e 100644 --- a/docs/index.html +++ b/docs/index.html @@ -1,24 +1,16 @@ - + - Mota - - - - - - - + +
- - - + \ No newline at end of file diff --git a/docs/logo.jpg b/docs/logo.jpg deleted file mode 100644 index d24f88add1f815a889977cb0e6b2ba6edb9d34b2..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 25969 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 )AB|H0s&9J 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 T5m-lH9f5TO{#OVXuMGSFD8P%b diff --git a/docs/zh/guide/autorun.html b/docs/zh/guide/autorun.html deleted file mode 100644 index 5ad6f60..0000000 --- a/docs/zh/guide/autorun.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 自执行函数 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/binding.html b/docs/zh/guide/binding.html deleted file mode 100644 index 63f7e18..0000000 --- a/docs/zh/guide/binding.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 数据绑定 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/hook_model.html b/docs/zh/guide/hook_model.html deleted file mode 100644 index 0aff211..0000000 --- a/docs/zh/guide/hook_model.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 面向 Hook 的模型 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/mapping.html b/docs/zh/guide/mapping.html deleted file mode 100644 index b6d425c..0000000 --- a/docs/zh/guide/mapping.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 属性映射 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/model.html b/docs/zh/guide/model.html deleted file mode 100644 index da7dae8..0000000 --- a/docs/zh/guide/model.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 编写业务模型 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/quick.html b/docs/zh/guide/quick.html deleted file mode 100644 index 04e8747..0000000 --- a/docs/zh/guide/quick.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 快速开始 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/typescript.html b/docs/zh/guide/typescript.html deleted file mode 100644 index 710f790..0000000 --- a/docs/zh/guide/typescript.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 在 TS 中使用 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/use_model.html b/docs/zh/guide/use_model.html deleted file mode 100644 index 65ac113..0000000 --- a/docs/zh/guide/use_model.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 使用 Hook API - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/docs/zh/guide/watch.html b/docs/zh/guide/watch.html deleted file mode 100644 index 8ce19f5..0000000 --- a/docs/zh/guide/watch.html +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - 监听模型变化 - Mota - - - - - - - - - -
- - - - - - \ No newline at end of file diff --git a/markdowns/quick.md b/markdowns/quick.md index 72d7c12..b752832 100644 --- a/markdowns/quick.md +++ b/markdowns/quick.md @@ -18,6 +18,8 @@ Mota 是一个响应式的 React 应用状态管理库,基于 Mota 你可以 [在线 TodoList 示例](http://houfeng.net/dn-template-mota/example/) ([示例源码](https://github.com/Houfeng/dn-template-mota)) + + ## 安装 通过 npm 安装,如下