Выпущен Tencent Omio - полностью совместим с IE8 и мобильными устройствами
В платежной системе WeChat мобильные QQ, Tencent TEG, Tencent IEG и другие команды смогли использовать Omi в большом количестве проектов и внутренних систем управления, чтобы добиться полного охвата Omi, совместимого с различными браузерными средами на в сторону c, так что есть Omio, имеет почти такой же синтаксис, что и Omi.
Совместимость с версиями старых браузеров Omi,→ Гитхаб
использовать немедленно
$ npm i omi-cli -g
$ omi init-o my-app
$ cd my-app
$ npm start
$ npm run build
Отличие от Оми
omio имеет тот же синтаксис, что и omi, но следует отметить некоторые отличия:
- поддержка омио
staticCss
, оми не поддерживается
css
а такжеstaticCss
Разница в том, например:
render() {
return (
<div>
<my-ele name={this.name}></my-ele>
<my-ele name={this.name}></my-ele>
<my-ele name={this.name}></my-ele>
</div>
)
}
Как и в приведенном выше примере,css
метод визуализируется три раза и вставляется в голову, аstaticCss
будет отображаться только один раз.
Когда вы обновляете компонент или setState,css
Метод выполнит рендеринг три раза и обновит стили в голове, соответствующие трем местам.staticCss
Не рендерить больше.
- Omio не поддерживает слоты, используйте
props.children
вместо этого, как реагировать - Omio поддерживает внедрение в хранилище, но не поддерживает обновление пути к хранилищу.
- Omio не поддерживает массив рендеринга, возможно, он будет поддерживаться в будущем.
- Омио не поддерживает
fire
Инициировать пользовательские события, которые можно использовать так же, как реагироватьprops.xxx()
Вызывать. Оми также поддерживаетfire
andprops.xxx()
Два пути.
Использование Omio в проекте Omi
Сначала установите:
npm i omio
Настроить псевдоним Webpack
Если вы хотите использовать omio в рамках существующего проекта omi, вы можете использовать следующую конфигурацию без каких-либо изменений кода:
module.exports = {
//...
resolve: {
alias: {
omi: 'omio'
}
}
};
Совместимость с IE, наступающим на яму
Яма 1 — Псевдомассив
Псевдомассив из querySelectorAll под IE, нет метода, связанного с массивом:
const codes = document.querySelectorAll('xxx')
//挂了
codesArr.forEach(() => {
})
Его нужно преобразовать в настоящий массив:
const codes = Array.prototype.slice.call(document.querySelectorAll('xxx'))
Вторая яма - теряются статические свойства
Вот исходный код Оми:
function define(name, ctor) {
if (ctor.is === 'WeElement') {
options.mapping[name] = ctor;
if (ctor.data && !ctor.pure) {
ctor.updatePath = getUpdatePath(ctor.data);
}
}
}
Но не могу ввести условие if в IE! ! В исходном коде Omi есть статические свойства:
class WeElement {
static is = 'WeElement'
constructor(props, store) {
...
}
...
...
render() { }
}
Почему он потерян? Вернемся к источнику:
использовать определение:
define('my-p', class extends WeElement {
render(props) {
return props.children[0]
}
})
Скомпилированный код:
define('my-p', function (_WeElement) {
_inherits(_class, _WeElement);
function _class() {
_classCallCheck(this, _class);
return _possibleConstructorReturn(this, _WeElement.apply(this, arguments));
}
_class.prototype.render = function render?1(props) {
return props.children[0];
};
return _class;
}(WeElement));
Тогда проблема кроется в статическом атрибуте в процессе _inheritsis
потерянный!
function _inherits(subClass, superClass) {
subClass.prototype = Object.create(superClass && superClass.prototype, {
constructor: {
value: subClass,
enumerable: false,
writable: true,
configurable: true
}
});
if (superClass) {
Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass;
}
}
Ну, так как это компиляция и внедрение кода, определения компонентов, которые поддерживают чистые функции, также могут потребоваться позже, так что это решается:
function define(name, ctor) {
//if (ctor.is === 'WeElement') {
options.mapping[name] = ctor;
if (ctor.data && !ctor.pure) {
ctor.updatePath = getUpdatePath(ctor.data);
}
//}
}
Третья яма — Object.assign IE не поддерживает
Поскольку Object.assign используется в исходном коде Omio, здесь требуется полифилл:
if (typeof Object.assign != 'function') {
// Must be writable: true, enumerable: false, configurable: true
Object.defineProperty(Object, "assign", {
value: function assign(target, varArgs) { // .length of function is 2
'use strict';
if (target == null) { // TypeError if undefined or null
throw new TypeError('Cannot convert undefined or null to object');
}
var to = Object(target);
for (var index = 1; index < arguments.length; index++) {
var nextSource = arguments[index];
if (nextSource != null) { // Skip over if undefined or null
for (var nextKey in nextSource) {
// Avoid bugs when hasOwnProperty is shadowed
if (Object.prototype.hasOwnProperty.call(nextSource, nextKey)) {
to[nextKey] = nextSource[nextKey];
}
}
}
}
return to;
},
writable: true,
configurable: true
});
}
Поскольку IE9 поддерживает ES5, es5, скомпилированный webpack, не нужно вводить.es5-shimбыть совместимым.
Четвертая яма - Прокси не поддерживает
Поскольку Omi необходимо отслеживать изменения данных, Omi использует прокси-сервер, поэтому здесь требуется решение для перехода на более раннюю версию —библиотека Обаа, слушайте любые изменения любого объекта.
Установка обааа
npm install obaa
использовать
observe object:
var obj = { a: 1 };
obaa(obj, function (name, value , old) {
console.log(name + "__" + value + "__" + old);
});
obj.a = 2; //a__2__1
observe array:
var arr = [1, 2, 3];
obaa(arr, function (name, value, old) {
console.log(name + "__" + value+"__"+old);
});
arr.push(4);//Array-push__[1,2,3,4]__[1,2,3]
arr[3] = 5;//3__5__4
observe class instance:
var User = function (name, age) {
this.name = name;
this.age = age;
//observe name only
obaa(this, ["name"], function (name, value, oldValue) {
console.log(name + "__" + value + "__" + oldValue);
});
}
var user = new User("lisi", 25);
user.name = "wangwu";//name__wangwu__lisi
user.age = 20; //nothing output
разное:
arr.push(111) //trigger observe callback
//every method of array has a pureXXX function
arr.purePush(111) //don't trigger observe callback
arr.size(2) //trigger observe callback
arr.length = 2 //don't trigger observe callback
//if obj.c is undefined
obaa.set(obj, 'c', 3)
obj.c = 4 //trigger observe callback
//if obj.c is undefined
obj.c = 3
obj.c = 4 //don't trigger observe callback
Пятая яма - mapjs MVVM не поддерживает
Mappingjs полностью использует прокси, поэтому представление будет автоматически обновляться в процессе сопоставления данных. Но после перехода на obaa обнаруживается, что представление обновления длины массива не будет обновляться, а представление увеличения массива не будет обновляться. Просмотрел mapjs и нашел:
- Mappingjs использует array.length для изменения длины массива
- Mappingjs использует массив [index] для добавления элементов
В obaa это запрещено, иначе нельзя будет отслеживать изменения.Obaa требует:
- Используйте array.size(len) для изменения длины массива
- Добавьте элементы, используя array.push
Так что естьmappingjs-omioТаким образом, Omio также может использовать настоящую архитектуру MVVM.
Омио в действии
md2siteПолностью построенный с omio, он имеет хороший опыт чтения и совместимость.
Совместимость с IE8
Первый карьер - ключевое слово как ключевое
const map = {
var: 'view',
switch: 'switch'
}
Для изменения на:
const map = {
'var': 'view',
'switch': 'switch'
}
Ключевые слова нельзя использовать в качестве ключей JSON.
Вторая яма — полифилл Object.assign недоступен
Использование полифилла Object.assignObject.defineProperty
, IE8 сообщает об ошибке, поэтому замените Object.assign наobject-assign
:
'use strict'
/* eslint-disable no-unused-vars */
var getOwnPropertySymbols = Object.getOwnPropertySymbols
var hasOwnProperty = Object.prototype.hasOwnProperty
var propIsEnumerable = Object.prototype.propertyIsEnumerable
function toObject(val) {
if (val === null || val === undefined) {
throw new TypeError('Object.assign cannot be called with null or undefined')
}
return Object(val)
}
export function assign(target, source) {
var from
var to = toObject(target)
var symbols
for (var s = 1; s < arguments.length; s++) {
from = Object(arguments[s])
for (var key in from) {
if (hasOwnProperty.call(from, key)) {
to[key] = from[key]
}
}
if (getOwnPropertySymbols) {
symbols = getOwnPropertySymbols(from)
for (var i = 0; i < symbols.length; i++) {
if (propIsEnumerable.call(from, symbols[i])) {
to[symbols[i]] = from[symbols[i]]
}
}
}
}
return to
}
Третья яма — Object.create недоступен
Используйте полифилл и закомментируйте код ниже! Потому что передача двух параметров не может быть полифиллом!
if (typeof Object.create !== 'function') {
Object.create = function(proto, propertiesObject) {
if (typeof proto !== 'object' && typeof proto !== 'function') {
throw new TypeError('Object prototype may only be an Object: ' + proto)
} else if (proto === null) {
throw new Error(
"This browser's implementation of Object.create is a shim and doesn't support 'null' as the first argument."
)
}
// if (typeof propertiesObject != 'undefined') {
// throw new Error("This browser's implementation of Object.create is a shim and doesn't support a second argument.");
// }
function F() {}
F.prototype = proto
return new F()
}
}
Четвертая яма - свойства настройки текстового узла
//ie8 error
try {
out[ATTR_KEY] = true
} catch (e) {}
Непосредственно оберните его в try catch, и его тестирование не повлияет на нормальное использование в настоящее время.
Пятая яма — addEventListener и removeEventListener
Полифил mdn здесь используется напрямую, а у других полифиллов есть ямки!
if (!Element.prototype.addEventListener) {
var oListeners = {};
function runListeners(oEvent) {
if (!oEvent) { oEvent = window.event; }
for (var iLstId = 0, iElId = 0, oEvtListeners = oListeners[oEvent.type]; iElId < oEvtListeners.aEls.length; iElId++) {
if (oEvtListeners.aEls[iElId] === this) {
for (iLstId; iLstId < oEvtListeners.aEvts[iElId].length; iLstId++) { oEvtListeners.aEvts[iElId][iLstId].call(this, oEvent); }
break;
}
}
}
Element.prototype.addEventListener = function (sEventType, fListener /*, useCapture (will be ignored!) */) {
if (oListeners.hasOwnProperty(sEventType)) {
var oEvtListeners = oListeners[sEventType];
for (var nElIdx = -1, iElId = 0; iElId < oEvtListeners.aEls.length; iElId++) {
if (oEvtListeners.aEls[iElId] === this) { nElIdx = iElId; break; }
}
if (nElIdx === -1) {
oEvtListeners.aEls.push(this);
oEvtListeners.aEvts.push([fListener]);
this["on" + sEventType] = runListeners;
} else {
var aElListeners = oEvtListeners.aEvts[nElIdx];
if (this["on" + sEventType] !== runListeners) {
aElListeners.splice(0);
this["on" + sEventType] = runListeners;
}
for (var iLstId = 0; iLstId < aElListeners.length; iLstId++) {
if (aElListeners[iLstId] === fListener) { return; }
}
aElListeners.push(fListener);
}
} else {
oListeners[sEventType] = { aEls: [this], aEvts: [[fListener]] };
this["on" + sEventType] = runListeners;
}
};
Element.prototype.removeEventListener = function (sEventType, fListener /*, useCapture (will be ignored!) */) {
if (!oListeners.hasOwnProperty(sEventType)) { return; }
var oEvtListeners = oListeners[sEventType];
for (var nElIdx = -1, iElId = 0; iElId < oEvtListeners.aEls.length; iElId++) {
if (oEvtListeners.aEls[iElId] === this) { nElIdx = iElId; break; }
}
if (nElIdx === -1) { return; }
for (var iLstId = 0, aElListeners = oEvtListeners.aEvts[nElIdx]; iLstId < aElListeners.length; iLstId++) {
if (aElListeners[iLstId] === fListener) { aElListeners.splice(iLstId, 1); }
}
};
}
Шестая яма - обрезка струн не поддерживается
if (!String.prototype.trim) {
String.prototype.trim = function () {
return this.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, '')
}
}
Седьмая яма - мониторинг данных
import { render, WeElement, define } from '../../src/omi'
define('my-counter', class extends WeElement {
//ie8 不能使用 observe
//static observe = true
data = {
count: 1
}
sub = () => {
this.data.count--
//手动 update
this.update()
}
add = () => {
this.data.count++
//手动 update
this.update()
}
render() {
return (
<div>
<button onClick={this.sub}>-</button>
<span>{this.data.count}</span>
<button onClick={this.add}>+</button>
</div>
)
}
})
render(<my-counter />, 'body')
Если вам не нужна совместимость с IE8, вы можете использоватьstatic observe = true
Выполните мониторинг данных для автоматического обновления представления.
Яма 8 — прокладка ES5
<script src="https://cdnjs.cloudflare.com/ajax/libs/es5-shim/4.5.7/es5-shim.min.js"></script>