virtual-dom


Keywords
mcore, mcore3
License
ISC
Install
npm install mcore3@3.2.5

Documentation

mcore

  • MVC 框架 [Based on MVC Architecture]
  • 模板渲染基于 virtual-dom diff 实现, 只更新需要变更的 DOM [Template rendering is based on Virtual-Dom diff. Only update the DOM that needs to be updated]
  • 模板不需要像 react 那样用 JSX; 跟原来一样, 写 html 就行了 [Unlike ReactJS, templates don't need JSX. We can just write html as usual]
  • 模板中的 DIY tag 就是一个 Components, 实现组件化非常方便 [The DIY tag in template is a Components, so it's easy to achieve modular]
  • 每一个 DIY tag 是一个 “独立空间”, 子节点不参于 diff. 所以,你可以很方便的, 在 DIY tag 里,使用 jQuery 的插件 [Each DIY tag is like a independent space, child node doesn't involve in diff. So, you can easily use jQuery plugin in the DIY tag]

Getting started

Install mcore-cli

npm i -g yarn
yarn global add mcore-cli
mkdir mcore-app
cd mcore-app
mcore-cli --init ./
yarn
npm run dev

MOVING PARTS

Binders

Describe how your underlying data is reflected in the DOM and vice-versa. All binders create a binding that is automatically updated when model data changes or when the user interacts with the DOM.

html

Sets the element's HTML content.

<section mc-html="scope.item.summary"></section>

show

Shows the element when the value evaluates to true and hides the element when the value evaluates to false.

<button mc-show="scope.user.admin">Remove</button>
<button mc-show="scope.user.ix > 0">Remove</button>

hide

Hides the element when the value evaluates to true and shows the element when the value evaluates to false.

<section mc-hide="scope.feature.disabled"></section>

if

Inserts and binds the element as well as it's child nodes into the DOM when the value evaluates to true and removes the element when the value evaluates to false.

<section mc-if="scope.item.editable"></section>
<section mc-if="scope.item.editable || scope.item.ix > 0"></section>
<section mc-if="scope.item.editable && scope.item.ix != 0"></section>
<section mc-if="(scope.item.editable && scope.item.ix != 0) || scope.show"></section>

for

Each Array or Object

<ul>
    <li mc-for="v in scope.arrayData">
        {v}
    </li>
    <li mc-for="v, i in scope.arrayData">
        {v} index: {i}
    </li>

    <li mc-for="v of scope.objectData">
        {v}
    </li>

    <li mc-for="v, key of scope.objectData">
        {v} key: {key}
    </li>

    <li mc-for="v in (scope.arrayData || scope.otherData)">
        {v}
    </li>
</ul>

on-[event]

Binds an event listener on the element using the event specified in [event]

注: callback 至小有二个参数, 参数1: event 参数2: element (跟 jQuery 的事件回调参数一样)

<button mc-on-click="edit">Remove</button>

在事件中,传递上下文

<div mc-for="v, i in scope.list">
    <span mc-on-click="show(v, i)">{v.title}</span>
</div>
class Edit extends mcore.Veiw {
    show(event, el, v, i){
        console.log(v, i);
    }
}

class-[classname]

version added: 3.0

Adds a class (whatever value is in place of [classname]) on the element when the value evaluates to true and removes that class if the value evaluates to false.

<li mc-class-completed="scope.todo.done">{ scope.todo.name }</li>

[attribute]

Sets the value of an attribute (whatever value is in place of [attribute]) on the element.

If your binding declaration does not match any of the above routines, it will fallback to use this binding.

<input type="text" mc-placeholder="scope.field.placeholder">

Formatters

Formatters are functions that mutate the incoming and/or outgoing value of a binding. You can use them to format dates, numbers, currencies, etc. and because they work in a similar fashion to the Unix pipeline, the output of each feeds directly as input to the next one, so you can stack as many of them together as you like.

mcore.Template.formatters.date = (value)=>{
    return moment(value).format('YYYY-MM-DD');
};

Formatters are applied by piping them to binding declarations using | as a delimiter.

<input mc-value="scope.startDate | date" type="text" />

<span>{scope.startDate | date}</span>

Formatter arguments

Formatters can accept any number of arguments in the form of keypaths or primitives. Keypath arguments get observed and will recompute the binding when any intermediary key changes. A primitive can be a string, number, boolean, null or undefined.

<span>{ scope.alarm.time | time scope.user.timezone 'hh:mm' }</span>

The value of each argument in the binding declaration will be evaluated and passed into the formatter function as an additional argument.

mcore.Template.formatters.time = (value, timezone, format)=>{
    return moment(value).tz(timezone).format(format);
};

Components

Components let you define reusable views that can be used within any of your templates. For some perspective on where components fit into your templates in relation to binders; binders define custom attributes, while components define custom elements.

<!-- time.html -->
<span>{scope.time}</span>
export default class Time extends mcore.Component {
    init(){
        this.render(require('time.html'),{
            time: new Date()
        });

        setInterval(()=>{
            this.scope.time = new Date();
        }, 1000);
    }
}
mcore.Template.components.time = Time;

Use Components

<time></time>

View

extends Components

<!-- home.html -->
<h1> Hello {scope.name} !</h1>
export default class Home extends mcore.View {
    run(){
        this.render(require('home.html'), {
            name: 'freddy'
        });
    }
}

Route