接口多态的扩展

(chameleon-tool@0.4.0以上开始支持)

接口多态一节讲解了接口多态的使用,这一节讲解接口多态的一些扩展语法,更适用于扩展新端使用。

1 扩展语法

1.1 include标签

当前interface文件用<include>标签可以引入其他.interface文件,代表继承该文件的接口定义和实现。然后在当前文件中去实现引入的接口定义,可以覆写引入的某一端实现,也可以去扩展新端的实现。 语法是<include src="${引用路径}.interface"></include>,有如下规则:

  • src属性指向的文件路径不支持webpack别名,只能是相对路径或者npm包名开始。

例如扩展新端API(以头条小程序为例,假设端扩展标识为:toutiao):

// 引入官方标准interface文件
<include src="chameleon-api/src/interfaces/alert/index.interface"></include>

// 扩展实现新端(以头条小程序为例,假设端扩展标识为:toutiao)
<script cml-type="toutiao">
class Method implements uiInterface {
  alert(opt, successCallBack, failCallBack) {
    // 根据头条小程序实现alert弹窗
    let { message, confirmTitle} = opt;
    tt.showModal({
      showCancel: false,
      title: '',
      content: message,
      confirmText: confirmTitle,
      success() {
        successCallBack(confirmTitle);
      },
      fail() {
        failCallBack(confirmTitle);
      }
    });
  }
}

export default new Method();
</script>

// 想覆写某已有端的方法实现(以微信小程序为例)
<script cml-type="wx">
class Method implements uiInterface {
  alert(opt, successCallBack, failCallBack) {
    // 按你的想法重新实现
  }
}
export default new Method();
</script>

1.2 script src属性

<script></script>标签可以通过指定src的方式引用外部js文件为某一平台或某几个的实现或者接口定义。 有如下规则:

  • src属性指向的文件路径不支持webpack别名,只能是相对路径或者npm包名开始。
  • cml-type属性为字符串,如果多个端可以用英文逗号 , 进行分割,例如 cml-type="web,wx".

例如:

<script cml-type="interface">
interface FirstInterface {
  getMsg(msg: String): String;
}

</script>

<script cml-type="web" src="./web.js"></script>

<script cml-type="wx,alipay" src="./miniapp.js"></script>

web.js文件内容如下:

class Method implements FirstInterface {
  getMsg(msg) {
    return 'web:' + msg;
  }
}

export default new Method();

2 多态API查找优先级

没有扩展语法之前,每个端的实现都在同一个文件中,没有优先级的问题,但是有了扩展语法之后,就会有优先级问题。查找优先级如下:

  • 文件内部的定义, 包括采用src的形式指定其他文件
  • <include>src属性指向的interface文件中继续查找该部分

例如有如下两个interface文件:

├── first
│   └── first.interface
└── second
    └── second.interface

first.interface 包括接口定义,web和weex端的实现,内容如下:

<script cml-type="interface">
interface FirstInterface {
  getMsg(msg: String): String;
}
</script>
<script cml-type="web">
class Method implements FirstInterface {
  getMsg(msg) {
    return 'first web:' + msg;
  }
}
export default new Method();
</script>
<script cml-type="weex">
class Method implements FirstInterface {
  getMsg(msg) {
    return 'first weex:' + msg;
  }
}
export default new Method();
</script>

second.interface 包括对first.interfaceinclude weex端和wx端的实现,内容如下:

<include src="../first/first.interface"></include>
<script cml-type="weex">
class Method implements FirstInterface {
  getMsg(msg) {
    return 'second weex:' + msg;
  }
}
export default new Method();
</script>
<script cml-type="wx">
class Method implements FirstInterface {
  getMsg(msg) {
    return 'second wx:' + msg;
  }
}
export default new Method();
</script>

当外部引用second.interface文件并调用getMsg方法时 各端编译获取方法如下:

  • web端,因为second.interface中没有web端实现 所以查找到first.interface中web端getMsg方法
  • weex端,因为second.interface中有weex端实现 所以使用second.interface中weex端getMsg方法
  • wx端,因为second.interface中有wx端实现 所以使用second.interface中wx端getMsg方法

注意 cml-type interface部分必须是唯一的

例1:

// 引入chameleon-api interface文件
<include src="chameleon-api/src/interfaces/alert/index.interface"></include>

// 错误实现interface部分
<script cml-type="interface">
......
</script>

// 扩展实现新端
<script cml-type="demo">
......
</script>

<include>chameleon-api/src/interfaces/alert/index.interface文件中已经有cml-type="interface"的定义,所以当前文件中<script cml-type="interface"></script>定义部分是错误的。

例2:

// 引入cml-tt-api interface文件
<include src="cml-tt-api/src/interfaces/alert/index.interface"></include>

// 引入cml-quickapp-api interface文件
<include src="cml-quickapp-api/src/interfaces/alert/index.interface"></include>

// 扩展实现新端
<script cml-type="demo">
......
</script>

文件中引入了两个interface文件,但是他们内部找到的<script cml-type="interface"></script>定义部分都在chameleon-api/src/interfaces/alert/index.interface中,是同一文件。所以不认为是错误的。

results matching ""

    No results matching ""