精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

圖形編輯器開發:模塊間如何通信?

開發 前端
本文簡單介紹了圖形編輯器架構中,如何進行模塊間的通信。對于某個模塊間,可以通過入口 Editor 對象,輕松主動訪問任何其他模塊。此外還可以用事件發布訂閱的方式綁定監聽器,在對應模塊狀態更新后被動地獲得通知。

大家好,我是前端西瓜哥。

圖形編輯器,隨著功能的增加,通常都會愈發復雜,良好的架構是保證圖形編輯器持續開發高效的重要技術。

根據功能拆分成一個一個的小模塊基本是家常便飯。那么模塊之間是如何配合以及進行數據傳輸的呢?

編輯器 github 地址:

https://github.com/F-star/suika

線上體驗:

https://blog.fstars.wang/app/suika/

注入 Editor 實例

首先我們有一個主模塊,也是入口模塊,叫做 Editor。

為了高內聚低耦合,其下會根據功能拆分出很多的子模塊。

這是為了讓我們要改造特定的功能時,只需要改對應模塊的小范圍代碼,不會被其他模塊代碼干擾,也不需要去理解它們。

子模塊會在 Editor 初始化的時候,將 Editor 實例對象注入(大概算是一種依賴注入)。

class Editor {
  sceneGraph: SceneGraph;
  setting: Setting;
  viewportManager: ViewportManager;
  toolManager: ToolManager;
  commandManager: CommandManager;
  zoomManager: ZoomManager;
  hostEventManager: HostEventManager;
  selectedElements: SelectedElements;

  // ...

  constructor(options: IEditorOptions) {
    // 也有些模塊不需要和其他模塊通信
    this.setting = new Setting();
    
    // 將 Editor 示例作為子模塊的構造參數
    this.sceneGraph = new SceneGraph(this);
    this.viewportManager = new ViewportManager(this);
    this.toolManager = new ToolManager(this);
    this.commandManager = new CommandManager(this);
    this.zoomManager = new ZoomManager(this);
    this.selectedElements = new SelectedElements(this);
    // ...
    this.hostEventManager = new HostEventManager(this);
    
    this.hostEventManager.bindHotkeys();
    this.zoomManager.zoomToFit(1);
  }
}

子模塊會將其保存為一個私有成員屬性。

以子模塊 ZoomManger 類為例,它大概是這樣的:

export class ZoomManager {
 private editor: Editor
  // ...

  constructor(editor: Editor) {
    // 將傳入的 Editor 對象保存為私有屬性
    this.editor = editor
    // ...
  }
  
  zoomIn(cx?: number, cy?: number) {
    // 通過 this.editor 訪問到其他模塊
    const zoomStep = this.editor.setting.get('zoomStep');
   // ...
}

子類的子類如果也要用 editor,我們就再傳,主打一個透傳,人手一份 Editor。

這樣所有的子模塊就都能拿到 Editor 對象,然后通過這個 Editor 對象去訪問其他的子類。

最小知識原則

其實這種做法并不滿足設計模式的 最小知識原則(或者叫迪米特法則)。

所謂最小知識原則,指的是每個模塊只和應該要用到的模塊要交流,不要和用不到的模塊發生關系。

甚至你可以抽一層接口或類繼承的方式,將細粒度達到被關聯模塊的某幾個需要用到的方法。

目前我的項目還處于早期階段,復雜度很低,所以沒必要這么做,之后會不斷添加功能中讓關聯模塊發生著變化。不應該過早優化。這是項目變得非常復雜,且開發人員非常多的時候才需要考慮優化。

事件發布訂閱

前面注入的方式,都是通過 主動的方式 去訪問其他模塊。

有時候我們需要用 被動的方式 去拿到其他模塊的數據,這時候我們常常會用 發布訂閱 模式。

發布訂閱模式,就是對象間存在一對多的依賴時,但一個對象改變狀態,所有的依賴對象會自動收到通知。

做法通常就是模塊加入的事件(event)的概念,并提供一些方法接受監聽器(函數),當這個模塊的某些狀態發生改變時,就會這些監聽器一一執行,并將最新狀態傳入。

這個其實我們并不陌生,像是定時器(setTimeout)、DOM 元素的事件(click、mouseover 等)都是用了這個設計模式。

Nodejs 也有個專門的 EventEmitter 類,來支持事件訂閱。

const { EventEmitter } = require('events');

// 創建事件觸發器實例
const emitter = new EventEmitter();

// 給 event-1 事件添加監聽器
emitter.on('event-1', (a, b) => {
  console.log('收到事件1消息,參數為:', a, b);
});

// 觸發事件,并提供參數。
emitter.emit('event-1', 3, 4);

// 移除指定監聽器
// emitter.off('event-1', handler);

可惜 Web 端并沒有這個輪子,得自己造或者找個輪子。

因為輪子實現并不復雜,我是更建議自己實現,方便修改和擴展。

通常我們只要實現 on、off、emit 三個方法就好了。

我們如果用 TypeScript 實現的話,需要用類型編程,讓事件名是類型安全的,即事件名對應的監聽器函數參數類型要匹配。

實現后的用法:

const ee = new EventEmitter<{
  // 指定事件和對應的函數類型
  update(newVal: string, prevVal: string): void;
  destroy(): void;
}>();
const handler = (newVal: string, prevVal: string) => {
  console.log(newVal, prevVal)
}
ee.on("update", handler);
ee.emit('update', '前端西瓜哥上班前的精神狀態', '前端西瓜哥上班后的精神狀態')
ee.off("update", handler);

// 編譯報錯(數字不匹配字符串類型)
// 'number' is not assignable to parameter of type 'string'
ee.emit('update', 1, 2)

// (val: number) => void' is not assignable to parameter of type '() => void
ee.on('destroy', (val: number) => {})

輪子的話我建議 mitt,同時這個輪子是 Vue3 官方推薦的(實現跨組件通信的一種方式),主要原因是它也是 類型安全 的。

這個輪子很簡單,高級方法也很少,源碼實現也就 100 多行,你完全可以拷貝過去自己改。

模塊如何使用事件

在 Nodejs 的內部模塊,是通過繼承的方式使用 EventEmitter 的,它的做法是:

class A extends EventEmitter {
  // ...
}

A.on('event-1', () => {})

但我更建議用 **組合 **而不是繼承的方式。

class A {
  emitter = new Emitter()
}

A.emitter.on('event-1', () => {})

繼承并不是好文明,不加限制可能導致復雜的多層繼承。我們應該多用組合,少用繼承。

這樣做的另一個次要好處是 EventEmitter 的方法不會污染 A 對象。

除了模塊間用發布訂閱方式通信,內核層(Editor對象)也常常利用它和 UI 層通信。

因為狀態源保存在 Editor 對象中,所以需要用發布訂閱的方式去同步狀態給 UI 層。

以畫布縮放的功能為例。

畫布縮放管理類的實現如下:

class ZoomManager {
  private zoom = 1;
  // 自己造的 EventEmitter 輪子
  private emitter = new EventEmitter<{
    zoomChange(zoom: number, prevZoom: number): void;
  }>();
  
  setZoom(zoom: number) {
    const prevZoom = this.zoom;
    this.zoom = zoom;

    // 觸發 “zoom改變” 事件
    this.emitter.emit('zoomChanged', zoom, prevZoom);
  }
}

對應的需要拿到 zoom 值的 React 組件,會在組件掛載時綁定監聽器(Vue 也是類似邏輯)。

const ZoomActions = () => {
  const editor = useContext(EditorContext);
  const [zoom, setZoom] = useState(1);
  
  // 組件掛載 hook
  useEffect(() => {
    if (editor) {
      // 初始化時要主動獲取 zoom 值
      setZoom(editor.zoomManager.getZoom());

      // 通過事件同步 core 層的狀態
      const handler = (zoom: number) => {
        setZoom(zoom);
      };
      editor.zoomManager.emitter.on('zoomChanged', handler);
  
      // 組件銷毀時解綁
      return () => {
        editor.zoomManager.emitter.off('zoomChanged', handler);
      };
    }
  }, []);
}

結尾

本文簡單介紹了圖形編輯器架構中,如何進行模塊間的通信。

對于某個模塊間,可以通過入口 Editor 對象,輕松主動訪問任何其他模塊。此外還可以用事件發布訂閱的方式綁定監聽器,在對應模塊狀態更新后被動地獲得通知。

責任編輯:姜華 來源: 前端西瓜哥
相關推薦

2023-10-19 10:12:34

圖形編輯器開發縮放圖形

2023-09-07 08:24:35

圖形編輯器開發繪制圖形工具

2023-08-31 11:32:57

圖形編輯器contain

2023-09-26 07:39:21

2024-01-08 08:30:05

光標圖形編輯器開發游標

2023-10-08 08:11:40

圖形編輯器快捷鍵操作

2023-10-10 16:04:30

圖形編輯器格式轉換

2023-08-28 08:10:50

Hex圖形編輯器

2023-07-07 13:56:01

圖形編輯器畫布縮放

2024-01-03 08:43:17

圖形編輯器旋轉控制點縮放控制點

2023-07-31 08:46:07

圖形編輯器圖形自動對齊

2023-02-06 16:59:57

Canvas編輯器

2023-04-07 08:02:30

圖形編輯器對齊功能

2023-01-18 08:30:40

圖形編輯器元素

2023-02-01 09:21:59

圖形編輯器標尺

2023-06-12 08:22:56

圖形編輯器工具

2023-05-09 08:15:32

圖形編輯器撤銷重做功能

2023-04-10 08:45:44

圖形編輯器排列移動功能

2023-02-09 07:02:30

圖形編輯器修改圖形

2022-12-02 07:24:46

點贊
收藏

51CTO技術棧公眾號

日本孕妇大胆孕交无码| 成人a v视频| 一区二区三区在线资源| 亚洲一区在线视频观看| 久久精品中文字幕一区二区三区| 亚洲大尺度在线观看| 亚洲一区在线| 精品亚洲一区二区三区在线观看 | 国产精国产精品| 在线免费看av网站| 首页亚洲中字| 91精品欧美福利在线观看| 男人添女人下部高潮视频在观看| 高清中文字幕一区二区三区| 国产成人综合视频| 国产精品久久精品| 日本va欧美va国产激情| 亚洲一区二区日韩| 尤物九九久久国产精品的分类 | 日韩精品一区二区三区中文字幕| 狠狠久久亚洲欧美专区| 潘金莲一级淫片aaaaa免费看| 人妻精品无码一区二区| 精品一区二区在线观看| 国产91久久婷婷一区二区| 九九在线观看视频| 久久激情电影| 亚洲人成在线一二| 色哟哟视频在线| 国产精一区二区| 欧美性猛片xxxx免费看久爱| 伊人成色综合网| 青草影视电视剧免费播放在线观看| 国产清纯在线一区二区www| 国产精品综合久久久久久| 国产女人18毛片18精品| 美女国产一区二区三区| 国产suv精品一区二区| 国产无遮挡aaa片爽爽| 综合天堂久久久久久久| 日韩视频免费中文字幕| 卡一卡二卡三在线观看| 亚洲第一二三区| 日韩电影中文字幕一区| 北条麻妃亚洲一区| 国产精选久久| 日韩一级片网站| 尤物网站在线看| 欧美一级在线| 欧美日韩国产天堂| 91热这里只有精品| av成人免费看| 欧美日韩一区在线观看| 三级a三级三级三级a十八发禁止| gogo亚洲高清大胆美女人体| 一本到不卡免费一区二区| 青青视频在线播放| 伊伊综合在线| 日本乱人伦aⅴ精品| 凹凸日日摸日日碰夜夜爽1| 小黄鸭精品aⅴ导航网站入口| 懂色av一区二区三区| 中国丰满人妻videoshd| 综合另类专区| 欧美色网一区二区| 欧美第一页浮力影院| 亚洲精品第一| 日韩一区二区三区四区| 色哟哟网站在线观看| 北条麻妃一区二区三区在线| 亚洲成人久久网| 美国黄色一级毛片| 精品国产一区二区三区小蝌蚪| 亚洲最新av在线| 亚洲色偷偷综合亚洲av伊人| 欧美午夜影院| 热久久免费视频精品| 日本成人一级片| 韩国欧美国产1区| 波多野结衣一区二区三区在线观看 | av在线免费播放网站| 中文字幕欧美一区| 久久久久久久香蕉| 欧美理论影院| 6080国产精品一区二区| 日本不卡视频一区| 欧美理论在线播放| 精品自拍视频在线观看| 久久99精品波多结衣一区| 蜜臀av一级做a爰片久久| 5566av亚洲| 精品美女视频在线观看免费软件| 国产精品久久久久精k8| 国产日本在线播放| 另类一区二区三区| 亚洲成人网av| 色欲AV无码精品一区二区久久| 中文在线播放一区二区| 欧美中文字幕精品| 99久久久久成人国产免费| 99re免费视频精品全部| 在线天堂一区av电影| av在线视屏| 91麻豆精品国产91久久久| 国产毛片毛片毛片毛片毛片毛片| 日韩一区自拍| 91高潮在线观看| 92久久精品一区二区| 99re8在线精品视频免费播放| 亚洲电影网站| 在线手机中文字幕| 日韩三级免费观看| www.日本高清视频| 亚洲精品美女| 91最新国产视频| 国产在线观看黄| 亚洲成年人影院| 激情在线观看视频| 欧美三级三级| 97在线视频精品| 国产99对白在线播放| 国产欧美一二三区| 亚洲色欲综合一区二区三区| 午夜视频在线观看精品中文| 色综合伊人色综合网站| 69亚洲精品久久久蜜桃小说| 成人亚洲一区二区一| 国产精品美女在线播放| 日本美女久久| 亚洲精品自拍偷拍| 日韩欧美中文字幕一区二区| 国产激情精品久久久第一区二区| 亚洲福利av在线| 天然素人一区二区视频| 亚洲精品影视在线观看| www.日本精品| 99久久久国产精品免费蜜臀| 日本国产中文字幕| 欧美片网站免费| 久久久91精品国产一区不卡| 一区二区日韩在线观看| 国产精品久久久久久一区二区三区| 妺妺窝人体色www在线小说| 久久99精品国产自在现线| 欧美激情啊啊啊| 亚洲精品一区二区三区四区| 亚洲老妇xxxxxx| 精品国产午夜福利在线观看| 综合久久久久| 国产精品久久久对白| 四虎亚洲精品| 精品久久国产老人久久综合| 欧美成人精品激情在线视频| 国产剧情一区二区| 亚洲中文字幕无码一区二区三区| 日韩中文字幕视频网| 欧美激情视频网站| 凸凹人妻人人澡人人添| 高跟丝袜欧美一区| 中文幕无线码中文字蜜桃| 麻豆精品网站| 午夜精品视频在线观看一区二区| 精品国产欧美日韩一区二区三区| 在线亚洲男人天堂| 亚洲一级黄色大片| 一区二区三区在线视频观看| 精品国产aⅴ一区二区三区东京热 久久久久99人妻一区二区三区 | 欧美日韩精品一区| 欧美成人性网| 中文字幕日韩欧美| jlzzjlzz亚洲女人18| 亚洲成人午夜电影| 91精品人妻一区二区三区| 日本伊人午夜精品| 少妇熟女一区二区| 国偷自产av一区二区三区| 日本不卡高字幕在线2019| 成人午夜影视| 日韩一卡二卡三卡四卡| 国产精品suv一区二区三区| 国产亚洲成aⅴ人片在线观看 | 国产一区二区成人| 国产一区二区女内射| 亚洲一区二区三区激情| 国产黄色网址在线观看| 另类欧美日韩国产在线| 真实国产乱子伦对白视频| 在线视频亚洲专区| 亚洲xxxxx性| 欧美aaaaa性bbbbb小妇| 最新的欧美黄色| 亚洲女人18毛片水真多| 在线观看日韩毛片| 欧美日韩三级在线观看| 久久久久久亚洲综合影院红桃| 日韩在线一区视频| 一本久久综合| 国产精品无码乱伦| 亚洲国产合集| 99热最新在线| 日韩成人精品一区二区三区| 午夜精品久久久久久99热软件| 国产小视频福利在线| 欧美va亚洲va国产综合| 亚洲天堂视频在线播放| 香蕉加勒比综合久久| 美国黄色片视频| 26uuu成人网一区二区三区| 天天干天天色天天干| 久久久xxx| 免费超爽大片黄| 国产精品国内免费一区二区三区| 国产一区在线观| 高清一区二区| 国产人妖伪娘一区91| 三级在线看中文字幕完整版| 不卡av电影在线观看| 不卡在线视频| 亚洲欧美日韩一区二区在线 | 一区二区三区黄色| 亚洲欧美自偷自拍| 欧美变态tickle挠乳网站| 一级α片免费看刺激高潮视频| 色综合久久综合网97色综合| 国产成人亚洲欧洲在线| 亚洲一区二区三区美女| 欧美成人国产精品高潮| 亚洲人一二三区| 女人裸体性做爰全过| 久久久久久久久蜜桃| 免费看毛片的网站| 国产成人8x视频一区二区| 毛片毛片毛片毛| 日韩 欧美一区二区三区| 国产精品秘入口18禁麻豆免会员| 在线播放一区| youjizz.com在线观看| 午夜国产精品视频免费体验区| 一区二区三区欧美成人| 日韩av有码| 亚洲成色www久久网站| 欧美一区电影| 日韩中文字幕一区二区| 精品国产一区探花在线观看| 日本一区二区在线视频| 欧美亚洲国产精品久久| 性欧美大战久久久久久久免费观看| 亚洲人成网亚洲欧洲无码| 欧美激情论坛| 奇米狠狠一区二区三区| 欧美一区2区三区4区公司二百| 欧美一区 二区| 蜜桃av噜噜一区二区三区| 女同久久另类99精品国产| 久久香蕉综合色| 国产videos久久| 图片区小说区区亚洲五月| 色狮一区二区三区四区视频| 亚洲欧洲一区二区福利| 亚洲女同一区| 欧美激情亚洲天堂| 日韩午夜高潮| 免费裸体美女网站| 九一久久久久久| 成人高清在线观看视频| 成人黄色a**站在线观看| www.超碰97| 中文字幕高清不卡| 成人涩涩小片视频日本| 亚洲第一福利视频在线| 人人草在线观看| 欧美日韩在线不卡| 成人高潮片免费视频| 亚洲国产精品99久久| 久久精品蜜桃| 久久夜色精品亚洲噜噜国产mv | 亚洲精品无码久久久| 日韩av影视在线| 成人福利在线| 欧美xxxx做受欧美| 伊人成综合网站| 国产欧美精品一区二区三区-老狼 国产欧美精品一区二区三区介绍 国产欧美精品一区二区 | 亚洲欧美综合一区| 欧美三级黄美女| 可以免费观看av毛片| 国产主播一区二区| 星空大象在线观看免费播放| 欧美韩日一区二区三区四区| 欧美老熟妇一区二区三区| 亚洲成人免费在线| 中文字幕在线网站| 精品日产卡一卡二卡麻豆| 久青青在线观看视频国产| 乱亲女秽乱长久久久| 成人免费无遮挡| 亚洲一区二区三区久久| 久久av电影| 国产情侣第一页| 蜜臀a∨国产成人精品| 国产婷婷在线观看| 综合网在线视频| 99久久精品国产亚洲| 日韩欧美视频在线| av在线三区| 91chinesevideo永久地址| 成人国产精品久久| 奇米视频888战线精品播放| 欧美阿v一级看视频| 天堂在线资源视频| 97久久超碰国产精品电影| 翔田千里88av中文字幕| 精品久久久久久亚洲国产300| av网站在线观看免费| 国产一区二区动漫| 中国色在线日|韩| 99re国产视频| 日韩精品91| 蜜臀久久99精品久久久酒店新书| 国产成人av电影在线观看| 天天爽天天爽天天爽| 色婷婷av一区二区三区之一色屋| 欧美在线精品一区二区三区| 久久福利视频网| 日韩成人一区| 亚洲精品白虎| 日韩av一区二| 国产av自拍一区| 色综合天天狠狠| 熟妇人妻系列aⅴ无码专区友真希| 欧美精品中文字幕一区| 中文字幕综合| 一区二区三区欧美成人| 美女视频黄a大片欧美| 久久国产柳州莫菁门| 日韩欧美国产一区二区| 日本中文字幕一区二区有码在线| 欧美激情综合色综合啪啪五月| 一级黄色a毛片| 一区二区三区国产在线| 九九九在线观看视频| 国产日产精品1区| 91久久国产综合久久91| 国产亚洲美女久久| 刘亦菲一区二区三区免费看| 精品视频高清无人区区二区三区| 在线电影一区| 亚洲av无码一区二区三区网址| 午夜精品一区二区三区三上悠亚| 韩国av免费在线观看| 91精品国产乱码久久久久久蜜臀| 国产精品自在线拍| 自慰无码一区二区三区| 26uuu精品一区二区三区四区在线| 麻豆久久久久久久久久| 亚洲欧美日韩网| 不卡亚洲精品| 麻豆md0077饥渴少妇| 国产精品系列在线观看| 五月天婷婷网站| 亚洲国产精品国自产拍av秋霞 | 国产午夜一区| 日韩一级片播放| 综合久久久久久久| 精品国产亚洲一区二区麻豆| 久久久久久成人| 日韩高清影视在线观看| 青青草精品视频在线观看| 国产精品福利电影一区二区三区四区| 97精品人妻一区二区三区| 欧美人成在线视频| 欧美jizz19性欧美| 天天操天天爽天天射| 亚洲人妖av一区二区| 欧美一区二不卡视频| 国产精品极品美女在线观看免费| 日韩在线视频精品| www日本在线观看| 欧美午夜电影在线| 欧美性videos| 精品久久sese| 精品一区二区三区免费毛片爱| 久久这里只有精品国产| 亚洲桃花岛网站| 成人午夜888| 人妻内射一区二区在线视频| 国产精品国产三级国产普通话蜜臀 | 成人黄视频在线观看| 国产精品视频福利| 奇米色777欧美一区二区| 久久国产精品波多野结衣av| 亚洲片在线观看| 18国产精品| 亚洲最大综合网| 亚洲成在人线在线播放| 欧美猛烈性xbxbxbxb| 国内一区在线| 韩国一区二区三区| wwwwww国产| 欧美富婆性猛交| 久久国产成人精品|