# 原生渐进迁移

已有项目 期望接入 Mpx,可根据项目或人力情况选择如何迁移,Mpx 不要求用户一次性用上框架的所有东西。

  1. 项目初始期:可以考虑一次性转为 Mpx,此时迁移成本比较低
  2. 项目成熟期:若人力有限,可选择逐步将原生小程序转为Mpx,而且 不需要对原有代码做全局重写。可参考此demo:Mpx渐进接入demo (opens new window)
    • 可以保持原有代码不变,新的组件、页面期望使用 Mpx 某些特性时才引入 Mpx。(推荐新模块引 Mpx,老模块逐步迁移 Mpx)
    • 用 Mpx 编写新的页面、组件,再局部导出对应的页面、组件,反向应用到现有的原生小程序项目中。见导出原生一节。(建议优先考虑老项目渐进改为 Mpx,而不是反向 Mpx 输出原生小程序的模式)

# 原生接入

有些时候,我们需要在Mpx工程中使用原生小程序组件:

  • 通过npm引用安装第三包
  • 将第三方包源码拷贝到本地src目录下

注:Mpx并不限制第三方包的格式。开发者可以自己参考小程序官方的开发第三方自定义组件 (opens new window)

# 原理

根据unsingComponents中设定的路径,Mpx会去查找包入口的js文件。然后提取入口文件所在的目录中的js json wxss wxml进行编译

编译带来的好处是,常规的拷贝操作,会造成组件内部的依赖缺失,以及冗余代码被打包。而执行了编译,使得Mpx可以精确的收集依赖,这表现在:

  • js文件中的依赖也会被打包,没有被加载的依赖库不会打包,减小体积
  • json文件的usingComponents会被解析,因此原生组件内部可以再引用其他原生组件,甚至是mpx组件
  • wxss中引用外部样式
  • wxml中的图片资源会被打包

例如:使用第三方组件库时,很多组件可能并未使用,如果按照官方给出的组件库使用方式,会将整个组件库放进项目。
而采用Mpx这种方式则只会引入使用了的组件,所以如果你喜欢vant的按钮,iview的输入框,ColorUI的布局,欢迎尝试mpx。
(本段内容具有时效性,未来微信可能会有优化,毕竟一开始微信连npm都不支持)

# 例子

文件目录

node_modules
|-- npm-a-wx-component // npm安装
|   --package.json
|   --src
|     --index.js
|     --index.json
|     --index.wxss
|     --index.wxml
|-- npm-b-wx-component // npm安装
|   --package.json
|   --src
|     --index.js
|     --index.json
|     --index.wxss
|     --index.wxml
component
│-- container.mpx 
│-- com-a.mpx 
|-- src-wx-component // 手动拷贝
|  --index.js
|  --index.json
|  --index.wxss
|  --index.wxml

container.mpx

<template>
  <view>
    <!-- mpx组件 -->
    <com-a></com-a>
    <!-- npm安装的原生组件 -->
    <npm-a-wx-component></npm-a-wx-component>
    <!-- 手动拷贝到工程的原生组件 -->
    <src-wx-component></src-wx-component>
  </view>
</template>

<script type="application/json">
  "usingComponents": {
    "com-a": "./com-a",
    "npm-a-wx-component": "npm-a-wx-component",
    "src-wx-component": "./src-wx-component"
  }
</script>

node_modules/npm-a-wx-component/src/index.wxml

<template>
  <view>
    <view>this is a native component</view>
    <!-- 原生组件内部使用原生组件 -->
    <npm-b-wx-component></npm-b-wx-component>
  </view>
</template>

node_modules/npm-a-wx-component/src/index.json

{
  "usingComponents": {
    "npm-b-wx-component": "npm-b-wx-component"
  }
}

# 原生page支持

原生自定义组件的支持已经基本能保证第三方 UI 库和 Mpx 的完美结合,但如果是用户存在已经开发好的小程序,在后续的迭代中发现了 Mpx 想使用,就需要用户手工将4个文件变成 Mpx 文件,这不够友好。

Mpx 提供了对原生页面的支持,允许项目中存在原生小程序文件(wxml,js,json,wxss)和 Mpx 文件,两者可以混合使用,通过 webpack 的构建将两者完美混合在一起生成最终的 dist。

使用方式和组件相似。

# 原生导出

通过导出原生能力,你可以将一个 Mpx 项目融回到原生小程序项目中。有两种做法:

  • 一是局部导出部分页面、组件
  • 二是完整导出一个 Mpx 项目。

# 导出部分页面/组件

使用 Mpx 开发的页面/组件也可以局部导出为纯粹的普通的原生小程序页面/组件,整合到已有的原生小程序中。

  1. 修改 webpack config 中 entry 一项,将 app 改为对应的页面/组件即可。

在路径后追加 ?isPage 来声明独立页面构建,构建产物为该页面的独立原生代码,在路径后追加 ?isComponent 来声明独立组件构建,构建产物为该组件的独立原生代码。

请参考下面的例子,注意resolve时候最后的query不可以省略,一定要按正确的类型声明这是一个组件or页面。

例子:

// 
module.exports = merge(baseWebpackConfig, {
  name: 'main-compile',
  // entry point of our application
  entry: {
    // 此处以mpx脚手架生成的项目为例
    
    // before
    // app: resolveSrc('app.mpx')
    
    // after,这里"pages/dindex"代表将原页面导出到output目录下的pages目录,文件名改为dindex.*
    'pages/dindex': resolveSrc('./pages/index.mpx?isPage'), // ?后标识导出类型
    'components/dlist': resolveSrc('./components/list.mpx?isComponent')
  }
})
  1. 执行 webpack 打包命令
  2. 拷贝打包后 dist 里所有文件到原生微信小程序项目根目录即可正常工作。

# 完整导出

举例:假如我们使用 Mpx 开发了一个完整的项目,这个项目可能包含多个页面,这些页面组合完成一个完整的功能。一般可能是公共需求,比如登录/用户中心等公共模块

如果其它接入方想复用这一公共模块,考虑有以下两种情形

  • 接入方也是 Mpx 框架开发的项目,直接迁移
  • 接入方是原生开发,这时我们希望能将整个项目完整导出成原生,并让接入方顺利使用。

其实观察下 Mpx 项目的打包结构,结构是非常简单的,页面/组件都很规矩地放在对应文件夹里的,所以删掉app.json/app.js/app.wxss/project.config.json几个文件后直接整个拷贝即可。

完整导出整个项目的做法可以是这样:

  1. 确认页面路径不要冲突,一般这种公共模块项目,路径上就不要占据/pages/index/index,页面路径Mpx是不会修改的,所以定一个/pages/{模块名}/{页面名}就好。

  2. app.*的内容都要删掉的,所以全局样式都应该写在独立的文件中(wxss),全局配置有什么特殊的要告知接入方(json),因为App.js会被舍弃,所以入口js要抽出来(js)。

  3. 如果有要导出的入口文件,需要给output增加配置:

// webpack.conf
module.exports = {
  entry: {
    app: resolveSrc('app.mpx'),
    index: resolveSrc('index.js') // 导出的入口文件,若没有可不写
  },
  output: {
    libraryTarget: 'commonjs2',
    libraryExport: 'default' // 若export default导出需要写这个,module.exports可省略
  },
  // ... 略
}
  1. 整个复制进接入方的项目里,注册对应的页面,然后就可以正常使用了!