欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

跟混亂的頁面彈窗說再見

 更新時間:2019年04月11日 11:15:16   作者:清夜  
這篇文章主要介紹了跟混亂的頁面彈窗說再見,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友們下面隨著小編來一起學(xué)習學(xué)習吧

對于一些快速迭代的產(chǎn)品來說,特別是移動端 C端產(chǎn)品,基于用戶運營的目的,在 app首頁給用戶展示各種各樣的彈窗是很常見的事情,在產(chǎn)品初期,由于迭代版本和運營策略變化地還不是太大,所以可能覺得沒什么,但當產(chǎn)品運營到后期,各種八竿子打不著的運營策略輪番上陣,彈窗的樣式、邏輯等都變了不知道多少遍的時候,問題就出來了

由于前期沒有做好規(guī)劃,首頁的彈窗組件可能放了十多個甚至更多,不僅是首頁有,首頁內(nèi)又引入了十多個個子組件,這些子組件內(nèi)也有,搞不好這些子組件內(nèi)還有子組件,子組件的子組件同樣還有彈窗,每個彈窗都有對應(yīng)的一組控制顯隱邏輯,分散在多個組件多個方法中,但是首頁只有一個頁面,你不可能讓所有符合顯示條件的彈窗,全都一下子彈出來,反正我是沒見過這么做的 app,那么如何管理這些彈窗就成了頭等大事

而往往當你意識到這一點的時候,很可能也正是局勢發(fā)展到無法控制的時候 治不了,等死吧

場景:A彈窗和 B彈窗位于主組件內(nèi),C彈窗位于主組件的子組件 C中,D彈窗位于主組件的子組件 B中,E彈窗位于主組件的子組件F的子組件G中

 

PM:我希望剛進入這個頁面的時候,只有當 A彈窗 和 B彈窗以及 C彈窗,都不展示的時候,才展示 D彈窗,如果 D彈窗展示過了,除非 B彈窗之后又展示了一遍,否則無論什么情況下都不展示 E彈窗
FE:???

 稍加思考一下,其實這件事情并不難辦,交給后端通過接口控制所有彈窗的顯隱就行了 主要是架構(gòu)的提前規(guī)劃,以及低耦合的代碼邏輯

彈窗的配置化

先確定一個大體思路,首先,必須要明確地知道當前頁面共有哪些彈窗組件,包括頁面的子組件以及子組件的子組件內(nèi)的彈窗組件,這是必須的,否則你連有哪些組件都不知道怎么精確控制?
所以,還是上面那句話,提前規(guī)劃,防患于未然是很重要的,不然等頁面迭代了幾十版,當初寫代碼的人都不在了你才想到去統(tǒng)計一下頁面上到底有多少個彈窗,那真是夠你喝一壺的

那么就需要在一個地方統(tǒng)一把這些彈窗全記錄下來,方便管理,于是可以得到下面這種數(shù)據(jù)結(jié)構(gòu):

// modalMap.js
export default {
 // 記錄首頁 index頁面內(nèi)的彈窗項
 index: {
  modalList: [{
   name: 'modal_1',
   level: 10,
   show: true
  }, {
   name: 'modal_2',
   level: 22,
   show: true
  }, {
   name: 'modal_3',
   level: 70,
   show: true
  }],
  children: {
   child1: {
    modalList: [{
     name: 'modal_1_1',
     level: 8,
     show: true
    }, {
     name: 'modal_1_2',
     level: 62,
     show: true
    }],
    children: {
     child1_1: {
      modalList: [{
       name: 'modal_1_1_1',
       level: 8,
       show: true
      }, {
       name: 'modal_1_1_2',
       level: 60,
       show: true
      }]
     }
    }
   }
  }
 }
 // ...還可以繼續(xù)記錄其他頁面的彈窗結(jié)構(gòu)
}

modalMap.js文件記錄每個頁面內(nèi)所有的彈窗項,例如,首頁 index內(nèi)的彈窗項都在屬性名index對于的值數(shù)據(jù)結(jié)構(gòu)中,index這個頁面主組件內(nèi)存在兩個 modal,可以分別命名為 modal_1和 modal_2,如果 index這個頁面主組件的子組件內(nèi)也有 modal,則繼續(xù)嵌套,例如,index主組件的子組件 child1中也有 modal,那么就把 child1放到 index的 children中繼續(xù)記錄,以此類推

這種結(jié)構(gòu)看起來比較清晰,主組件及主組件內(nèi)的子組件內(nèi)的 modal都很清晰,一目了然,當然,你可以不用這種結(jié)構(gòu),完全取決于你,這里就暫時這么定義

每個 modal除了 name之外,還有 level 和 show屬性

level 用于標識當前 modal的層級,每個頁面正常只能同時展示一個 modal,但如果有多個 modal都同一時間都滿足展示的條件,則對比它們的 level值,哪個大就優(yōu)先展示哪個,其余的忽略掉,杜絕一個頁面可能提示展示多個彈窗的情況;

show屬性則是在 modal內(nèi)部來決定 modal最終是否展示,這樣一來就可以無視外界條件,很輕松地通過配置來禁止掉彈窗的顯示

通過發(fā)布/訂閱模式來管理彈窗

彈窗的配置結(jié)構(gòu)已經(jīng)確定了,下一步就是對這些配置的管理了

一般情況下,多個頁面同時滿足條件需要進行展示的場景,大多數(shù)都是發(fā)生在剛進入頁面,頁面發(fā)出多個請求,這些請求的返回結(jié)果分別控制對應(yīng)的一個彈窗的展示

因為發(fā)出去的這些請求很可能分屬于不同的業(yè)務(wù)線或部門管轄,相互獨立,所以說如果把彈窗的控制權(quán)交給后端來做,其實是有點困難的,再加上請求是異步的,前端想要用意大利面條式代碼來保證彈窗之間的互斥性也不太容易,綜合起來,也就導(dǎo)致了當頁面上迭代出了數(shù)十個以上彈窗的時候,如果沒有提前規(guī)劃好,還是很容易出現(xiàn)彈窗同時展示的問題的

這里暫時就以剛進入頁面的情況為例,進行邏輯梳理

首先,我需要知道頁面上有哪些彈窗可能會在剛進入頁面的時候彈出來(即通過接口控制單個彈窗的展現(xiàn)與否),然后在所有彈窗的數(shù)據(jù)都拿到了的時候(即跟彈窗相關(guān)的接口都已經(jīng)返回數(shù)據(jù)),才進行彈窗的展示

這種情況比較適合使用發(fā)布/訂閱者模式,單個接口的數(shù)據(jù)返回就是一個訂閱,當所有接口都訂閱之后,就進行發(fā)布,也就是彈窗展示

// modalManage.js
class ModalManage {
 constructor (modalList) {
  this.modalFlatMap = {}
  this.modalList = modalList
 }
 // ... 
}

通過 ModalManage類來管理彈窗,此類在初始化時接收一個參數(shù) modalList,這個參數(shù)其實就是剛進入頁面時,頁面上所有可能展示的彈窗(包括子組件的彈窗)的名稱集合,也就是必須要知道頁面上到底有多少個可能同時展示的彈窗,以上述示例代碼 modalMap.js為例, index頁面的 modalList值就是 ['modal_1', 'modal_2', 'modal_3', 'modal_1_1', 'modal_1_2', 'modal_1_1_1', 'modal_1_1_2']

這里其實直接傳彈窗數(shù)量就行了,index中有 7個彈窗可能同時展示,所以可以直接傳 7,我這里之所以要傳名稱進去,實際上是為了方便調(diào)試,如果代碼出問題了,比如頁面上實際有 5個接口可以控制 5個彈窗的展示,但你卻只訂閱了 4次,如果只傳數(shù)字,你就需要一個個找過去看是哪一個忘記訂閱了,但如果傳名稱,你一下子就能調(diào)試出來,也就是代碼的可維護性會好一點

當頁面上任意一個彈窗的狀態(tài)(即是否滿足展示的條件)確定下來后,就進行訂閱操作:

// modalManage.js
add (name, dataInfo) {
 // level, handler
 if (this.modalList.indexOf(name) !== -1) {
  if (!this.modalFlatMap[name]) {
   this.modalFlatMap[name] = dataInfo
   this.notify()
  } else {
   console.log('重復(fù)訂閱')
  }
 } else {
  console.log('無效訂閱')
 }
}

this.modalFlatMap是為了記錄訂閱列表,當訂閱列表的長度和 modalList相同時,說明所有的彈窗狀態(tài)都已經(jīng)準備就緒,可以根據(jù)這些彈窗的優(yōu)先級進行展示了,也就是 notify方法要做的事情

notify方法中,先排除掉屬性 show為 false的彈窗項,再對比剩下的彈窗的 level,只展示 level最大的那個彈窗:

// modalManage.js
notify () {
 if (Object.keys(this.modalFlatMap).length === this.modalList.length) {
  const highLevelModal = Object.keys(this.modalFlatMap).filter(key => this.modalFlatMap[key].show).reduce((t, c) => {
   return this.modalFlatMap[c].level > t.level ? this.modalFlatMap[c] : t
   // 這個 { level: -1 } 只是為了給 reduce函數(shù)一個 initialValue,modal項的 level都應(yīng)該大于這個 initialValue的 level值,即 -1
  }, { level: -1 })
  highLevelModal.handler()
 }
}

使用單例模式管理嵌套組件以及多個頁面的彈窗

上述的 ModalManage類已經(jīng)足以管理彈窗了,但還有個問題,如果一個頁面上的彈窗,分散位于頁面主組件及其子組件,甚至是子組件的子組件內(nèi),怎么辦?

這個時候就需要使用單例了

// 單例管理
const manageTypeMap = {}
// 獲取單例
function createModalManage (type) {
 if (!manageTypeMap[type]) {
  manageTypeMap[type] = new ModalManage(getAllModalList(modalMap[type]))
 }
 return manageTypeMap[type]
}

通過 createModalManage這個方法來創(chuàng)建 ModalManage實例,根據(jù)傳入的 type來決定是否創(chuàng)建新的實例,如果單例管理對象 manageTypeMap中不存在 type對于的實例,則 new一個 ModalManage實例,存入 manageTypeMap中,并返回這個新實例,否則就返回 manageTypeMap中已經(jīng)創(chuàng)建好了的實例

這樣一來,無論彈窗分散在多少個組件內(nèi),無論這些組件嵌套得有多深,都能夠在保證代碼低耦合的前提下,順利地訂閱/發(fā)布事件

這里的 getAllModalList方法是個工具方法,用于從 modalMap中獲取頁面對應(yīng)的彈窗數(shù)據(jù)結(jié)構(gòu):

// util.js
const getAllModalList = modalInfo => {
 let currentList = []
 if (modalInfo.modalList) {
  currentList = currentList.concat(
   modalInfo.modalList.reduce((t, c) => t.concat(c.name), [])
  )
 }
 if (modalInfo.children) {
  currentList = currentList.concat(
   Object.keys(modalInfo.children).reduce((t, c) => {
    return t.concat(getAllModalList(modalInfo.children[c]))
   }, [])
  )
 }
 return currentList
}

至于 createModalManage的參數(shù)type,其值可以就是一個字符串,例如如果需要管理首頁 index上可能同時展示的所有的彈窗,則可以將 type 的值指定為 index,在 index主組件以及其包含彈窗的子組件內(nèi),都通過這個字段來獲取 ModalManage單例對象:

const modalManage = createModalManage('index')

這樣做同時也解決了另外一個問題,就是多個頁面的彈窗管理問題,index頁面通過 index創(chuàng)建 ModalManage單例,詳情頁就可以通過 detail來創(chuàng)建 ModalManage單例,雙方互不干擾
總結(jié)

上述所有示例代碼已經(jīng)上傳到 github,有興趣地可以看下

本文只是對彈窗這么一種具體的案例進行分析,實際上應(yīng)用于其他場景,例如頁面同一個位置的懸浮掛件管理等都是可行的
無論是彈窗的管理還是掛件的管理,放在 mvvm框架中,都是數(shù)據(jù)的管理,主流前端框架對于復(fù)雜的數(shù)據(jù)管理,都已經(jīng)有對應(yīng)的解決方案,例如 vuex 和 redux等,這些解決方案當然也能夠解決上面的問題

本文主要是對這種理念的探討,探討出一種通用的解決方案,無論你用的是 vue、react、angular還是jquery一把梭,亦或是微信小程序、支付寶小程序、快應(yīng)用等,都可以低成本地輕松套入使用

以上所述是小編給大家介紹的跟混亂的頁面彈窗說再見詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!

相關(guān)文章

  • vue中三種不同插槽使用小結(jié)

    vue中三種不同插槽使用小結(jié)

    本文主要介紹了vue中三種不同插槽使用小結(jié),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友們下面隨著小編來一起學(xué)習學(xué)習吧
    2023-07-07
  • 基于vue-video-player自定義播放器的方法

    基于vue-video-player自定義播放器的方法

    這篇文章主要介紹了基于vue-video-player自定義播放器的方法,主要是基于video.js開發(fā)的vue-video-player的使用,以及如何操作video.js中的api。需要的朋友可以參考下
    2018-03-03
  • Cookbook組件形式:優(yōu)化 Vue 組件的運行時性能

    Cookbook組件形式:優(yōu)化 Vue 組件的運行時性能

    本文仿照Vue Cookbook 組織形式,對優(yōu)化 Vue 組件的運行時性能進行闡述。通過基本的示例代碼給大家講解,需要的朋友參考下
    2018-11-11
  • vue中使用router全局守衛(wèi)實現(xiàn)頁面攔截的示例

    vue中使用router全局守衛(wèi)實現(xiàn)頁面攔截的示例

    這篇文章主要介紹了vue中使用router全局守衛(wèi)實現(xiàn)頁面攔截的示例,幫助大家維護自己的項目,感興趣的朋友可以了解下
    2020-10-10
  • 詳解vue如何獲取當前系統(tǒng)時間

    詳解vue如何獲取當前系統(tǒng)時間

    這篇文章主要詳細介紹了vue如何獲取當前系統(tǒng)時間,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2024-01-01
  • VUE+Element環(huán)境搭建與安裝的方法步驟

    VUE+Element環(huán)境搭建與安裝的方法步驟

    這篇文章主要介紹了VUE+Element環(huán)境搭建與安裝的方法步驟,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-01-01
  • 淺析vue插槽和作用域插槽的理解

    淺析vue插槽和作用域插槽的理解

    插槽,也就是slot,是組件的一塊HTML模板,這塊模板顯示不現(xiàn)實、以及怎樣顯示由父組件來決定。這篇文章主要介紹了淺析vue插槽和作用域插槽的理解,需要的朋友可以參考下
    2019-04-04
  • 使用vue組件封裝共用的組件

    使用vue組件封裝共用的組件

    這篇文章主要介紹了使用vue組件封裝共用的組件,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • Vue頁面跳轉(zhuǎn)動畫效果的實現(xiàn)方法

    Vue頁面跳轉(zhuǎn)動畫效果的實現(xiàn)方法

    百度了好久都沒辦法實現(xiàn)vue中一個頁面跳到另一個頁面,甚至到google上搜索也是沒辦法的,最終還是找了高人親自指導(dǎo),所以下面這篇文章主要給大家介紹了關(guān)于Vue頁面跳轉(zhuǎn)動畫效果的實現(xiàn)方法,需要的朋友可以參考下
    2018-09-09
  • Vue中mintui的field實現(xiàn)blur和focus事件的方法

    Vue中mintui的field實現(xiàn)blur和focus事件的方法

    今天小編就為大家分享一篇Vue中mintui的field實現(xiàn)blur和focus事件的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-08-08

最新評論