[Открытый исходный код] Выпуск Tencent Omio — полная совместимость с IE8 и мобильными устройствами

Открытый исходный код Omi

Выпущен 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 and props.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>

Начать

→ Оми Гитхаб