活用 async/await ,让 Vue 变得更好用的装饰器!

开发 前端
下文三个装饰器,都是利用了async/await把异步变成同步的特性实现的。

[[420903]]

下文三个装饰器,都是利用了async/await把异步变成同步的特性实现的。

要求被装饰的方法必须写成async/await,用起来十分方便,实现彻底被隐藏在了装饰器内部。

前两个都是用在ts环境下class写法的vue里的。不过看清楚逻辑后,很容易修改成可以用在js环境中的vue组件上。

1. 给vue添加一个指示初始化完成的变量。

指业务相关的初始化逻辑都完成了 比如搜索功能:搜索中显示loading,结果为空时显示暂无数据。但是第一次打开页面时,搜索还没完成,但显示暂无数据又不合适 这个时候就需要一个这样的变量处理边界情况 用于ts环境下的vue。

通过装饰器添加这个属性,并包装vue的created, mounted和beforeDestroy方法。当created或者mounted里发出的请求完成后,就把pageIsReady设为true。使用这个装饰器时,在业务代码中完全无感,没有任何心智负担。 

  1. import { Constructor } from"vue/types/options";  
  2. export type WrapReadyProperty<T> = T & { pageIsReady?: boolean }  
  3. /**  
  4.  * 在@compontent 之后使用这个装饰器,  
  5.  * 组件就会被注入属性 pageIsReady,  
  6.  * 当created和mounted都执行完成时 pageIsReady 变成true, 
  7.  * 要求mounted或created是async/await。(取决于在哪个方法中发请求初始化组件)  
  8.  * 然后可以在template中直接使用。  
  9.  * 在script中使用调用isPageReady.call(this)方法;  
  10.     */  
  11. exportdefaultfunction PageReadyStatus() {  
  12.     let createdDone = false 
  13.     let mountedDone = false 
  14.     function isCreatedMountedAllDone() {  
  15.         return createdDone && mountedDone;  
  16.     }  
  17.     returnfunction pageReadyEnhancement<T extends Constructor>(target: T) {  
  18.         const oldMounted = target.prototype.mounted || function() { }  
  19.         const oldCreated = target.prototype.created || function() { }  
  20.         const oldBeforeDestroy = target.prototype.beforeDestroy || function() { }  
  21.         target.prototype.pageIsReady = false 
  22.         target.prototype.created = asyncfunction(...params: any[]) {  
  23.             await oldCreated.apply(this, params);  
  24.             createdDone = true 
  25.             this.pageIsReady = isCreatedMountedAllDone()  
  26.         }  
  27.         target.prototype.mounted = asyncfunction(...params: any[]) {  
  28.             await oldMounted.apply(this, params); 
  29.             mountedDone = true 
  30.             this.pageIsReady = isCreatedMountedAllDone()  
  31.         }  
  32.         target.prototype.beforeDestroy = asyncfunction(...params: any[]) {  
  33.             await oldBeforeDestroy.apply(this, params);  
  34.             mountedDone = false 
  35.             createdDone = false 
  36.             this.pageIsReady = false 
  37.         }  
  38.         return target  
  39.     };  
  40.  
  41. exportfunction isPageReady(this: WrapReadyProperty<Vue>) {  
  42.     returnthis.pageIsReady  

2. 给事件回调函数和按钮Dom添加防抖与loading样式

用于ts环境下的vue。

通过装饰器包装被装饰的方法。要求被包装的方式是async/await的。这样装饰器内只需要用一个await就可以得知被包装的方法是否执行完成。同时,可以从事件对象中拿到被点击的dom元素并修改它。 

  1. /*  
  2.  * 请保证被包装的方法的参数列表最后一个是点击事件的参数  
  3.  */  
  4. exportdefaultfunction buttonThrottle() {  
  5.     let pending = false 
  6.     returnfunction(target: any, name: string): any {  
  7.         const btnClickFunc = target[name];  
  8.         const newFunc = asyncfunction(this: Vue, ...params: any[]) {  
  9.             if (pending) {  
  10.                 return;  
  11.             }  
  12.             const event:Event = params[params.length - 1];  
  13.             let btn = event.target as HTMLElement  
  14.             pending = true 
  15.             const recoverCursor = changeCursor(btn);  
  16.             try {  
  17.                 await btnClickFunc.apply(this, params);  
  18.             } catch (error) {  
  19.                 console.error(error);  
  20.             }  
  21.             recoverCursor();  
  22.             pending = false 
  23.         };  
  24.         target[name] = newFunc;  
  25.         return target;  
  26.     };  
  27.  
  28. function changeCursor(btn?: HTMLElement) {  
  29.     if (btn == null) {  
  30.         return() => {};  
  31.     }  
  32.     const oldCursor = btn.style.cursor;  
  33.     btn.style.cursor = "wait" 
  34.     return() => {  
  35.         btn.style.cursor = oldCursor 
  36.     };  

用法: 在点击事件函数上使用这个装饰器。装饰器会自动检测该函数是否执行完成,并在执行过程中往按钮的Dom节点上添加point:wait属性。

  1. import { Component, Vue } from"vue-property-decorator";  
  2.     import buttonThrottle from"@/ui/view/utils/buttonThrottle";  
  3.     type Member = { account_no: string; name: string; warn?: string };  
  4.     @Component({ components: {} })  
  5.     exportdefaultclass AddMemberInput extends Vue {        @buttonThrottle()  
  6.         private async confirmAdd() {  
  7.             awaitthis.addMembers(this.getVaildMembers());  
  8.         }  
  9.     } 

3. mounted之前显示白屏

用于js的vue中包装vue的对象。

同上,通过async/await获得mounted或者created是否执行完成 再通过指向vue实力的this拿到组件根节点,然后按需修改它 以下代码只是将组件隐藏了,实际上可以写更复杂的逻辑,在加载过程中显示其他内容,毕竟拿到了Dom,想干嘛就干嘛。 

  1. function firstPaintControl(vueObj) {  
  2.     let oldMounted = vueObj.mounted || function() {};  
  3.     vueObj.mounted = asyncfunction(...params) {  
  4.       this.$el.style.visibility = 'hidden' 
  5.       await oldMounted.apply(this, params);  
  6.       this.$el.style.visibility = 'visible' 
  7.     };  
  8.     return vueObj;  
  9.   }  

 

责任编辑:庞桂玉 来源: 前端大全
相关推荐

2017-08-02 14:17:08

前端asyncawait

2022-07-11 10:32:35

Vue3await响应式

2024-10-17 08:58:31

2014-07-15 10:31:07

asyncawait

2016-11-22 11:08:34

asyncjavascript

2014-05-07 09:41:21

UbuntuTrusty Tahr

2022-06-15 10:24:13

Pytho装饰器代码

2023-10-08 10:21:11

JavaScriptAsync

2022-03-04 20:28:02

VueReact网页

2021-01-14 22:13:03

人工智能教育互联网

2018-01-02 12:13:39

浏览器火狐扩展

2021-07-20 10:26:12

JavaScriptasyncawait

2012-07-22 15:59:42

Silverlight

2024-05-10 12:33:06

flask装饰器

2017-04-10 15:57:10

AsyncAwaitPromise

2022-08-27 13:49:36

ES7promiseresolve

2023-07-28 07:31:52

JavaScriptasyncawait

2021-06-28 07:27:43

AwaitAsync语法

2017-09-02 14:29:01

路由器Wi-Fi无线网络

2019-02-22 21:53:22

习惯FabulousAPP
点赞
收藏

51CTO技术栈公众号