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

圖形編輯器開發:縮放和旋轉控制點

開發 前端
控制點是吸附在圖形上的一些小矩形和圓形點擊區域,在控制點上拖拽鼠標,能夠實時對被選中進行屬性的更新。

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

挺久沒寫圖形編輯器開發系列了,今天來講講控制點,它是圖形編輯器的不可缺少的基礎功能。

控制點是吸附在圖形上的一些小矩形和圓形點擊區域,在控制點上拖拽鼠標,能夠實時對被選中進行屬性的更新

比如使用旋轉控制點可以更新圖形的旋轉角度,使用縮放控制點調整圖形的寬高。

這兩個都是通用的控制點,此外還有給特定圖形使用的專有控制點,像是矩形的圓角控制點,可拖動調整圓角大小。這些比較特別。后面會專門出一篇文章講這個。

需求描述

選中圖形,會出現旋轉控制點和縮放控制點,然后操作控制點,調整圖形屬性。

控制點的類型和位置如下:

縮放控制點有 8 個。

首先是 西北(nw)、東北(ne)、東南(se)、西南(sw)縮放控制點。它們在選中圖形包圍盒的四個頂點上,拖拽可同時調整圖形的寬高。

接著是 東(e)、南(s)、西(w)、北(n)縮放控制點,拖拽它們只更新圖形的寬或高。

它們是不可見的,但 hover 上去光標會變成縮放的光標。這幾個控制點的點擊區域很大。

旋轉控制點有 4 個,對應四個角落,分別為:nwRotation、neRotation、seRotation、swRotation。

同樣它們是透明的,但 hover 上去光標會變成旋轉光標。

旋轉控制點有另外一種風格,就是只在圖形的某個方向(通常是正上方)有一個可見旋轉控制點。下面是 Canva 編輯器的效果:

我更喜歡第一種風格,畫面會更清爽一些。

實現思路

整體實現思路很簡單:

  • 根據圖形的包圍盒,計算這些控制點的位置,設置好寬高。
  • 渲染,設置為不可見的控制點跳過渲染。
  • hover 或點擊時,編輯器會做 圖形拾取,會和渲染順序相反的順序遍歷控制點,調用控制點圖形的 hitTest 方法找到第一個被點中的圖形,返回對應控制點的類型和光標。然后編輯器更新光標,并根據控制點類型進入對應邏輯。如果你是用 html/svg 的方案,圖形拾取可以不用自己做。

代碼設計

我們需要實現控制點管理類 ControlHandleManager 和控制點類 ControlHandle。

ControlHandle 類記錄以下信息:

  • graph:圖形對象,記錄控制點的左上角位置、寬高、顏色、是否可見,并帶了一個點擊區域方法。
  • cx / cy:控制點的中點位置。
  • getCursor():獲取光標方法,hover 時返回一個需要設置的光標值。

這里直接用圖形編輯器繪制圖形用到的圖形類。

通常你使用的渲染圖形庫是會有

創建 ControlHandle 對象。

我們需要創建的控制點對象為:

// 右下角(ns)的控制點  
const se = new ControlHandle({
  graph: new Rect({
    objectName: 'se', // 控制點類型標識,放其他地方也行
    cx: 0, // x 和 y 會根據選中圖形的包圍盒更新
    cy: 0,
    width: 6,
    height: 6,
    fill: 'white',
    stroke: 'blue',
    strokeWidth: 1,
  }),
  getCursor: (type, rotation) => {
    // ...
    return 'se-rezise'
  } ,
});

這個對象會保存到控制點管理類的 transformHandles 屬性中。

transformHandles 是一個映射表,類型標識字符串映射到控制點對象。

class ControlHandleManager {
  visible = false;
  transformHandles;

  constructor() {
    // 映射表 type -> 控制點
    this.transformHandles = {
      se: new ControlHandle(/* ... */),
      n: new ControlHandle(/* ... */),
      nwRoation: new ControlHandle(/* ... */),
      // ...
    }
  }
}

渲染

當我們選中圖形時,調用渲染方法。

此時會調用 ControlHandleManager 的 draw 渲染方法,渲染控制點。

根據包圍盒計算控制點的中點位置。這個包圍盒有 x、y、width、height、rotation 屬性。我們需要計算這個包圍盒的四個頂點的位置,包圍盒外擴一定距離后的四個頂點的位置,四條線段的中點的位置。

class ControlHandleManager {
  // ...
  
  /** 渲染控制點 */
  draw(rect: IRectWithRotation) {
  
  // calculate handle position
  const handlePoints = (() => {
    const cornerPoints = rectToPoints(rect);
    const cornerRotation = rectToPoints(offsetRect(rect, size / 2 / zoom));
    const midPoints = rectToMidPoints(rect);

    return {
      ...cornerPoints,
      ...midPoints,
      nwRotation: { ...cornerRotation.nw },
      neRotation: { ...cornerRotation.ne },
      seRotation: { ...cornerRotation.se },
      swRotation: { ...cornerRotation.sw },
    };
  })();
 }
}

遍歷控制點對象,賦值上對應的中點坐標:cx、cy。調整 n/s/w/e 的寬高,它們的寬高是跟隨。

// 整個順序是有意義的,是渲染順序
const types = [
  'n',
  'e',
  's',
  'w',
  'nwRotation',
  'neRotation',
  'seRotation',
  'swRotation',
  'nw',
  'ne',
  'se',
  'sw',
] as const;

// 更新 cx 和 cy
for (const type of types) {
  const point = handlePoints[type];
  const handle = this.transformHandles.get(type);
  handle.cx = point.x;
  handle.cy = point.y;
}

// n/s/w/e 比較特殊,n/s 的寬和包圍盒寬度相等,w/e 高等于包圍盒高。
const neswHandleWidth = 9;
const n = this.transformHandles.get('n')!;
const s = this.transformHandles.get('s')!;
const w = this.transformHandles.get('w')!;
const e = this.transformHandles.get('e')!;
n.graph.width = s.graph.width = rect.width * zoom;
n.graph.height = s.graph.height = neswHandleWidth;
w.graph.height = e.graph.height = rect.height * zoom;
w.graph.width = e.graph.width = neswHandleWidth;

接著就是遍歷 transformHandles,基于 cx 和 cy 更新圖形的 x/y,然后繪制。

this.transformHandles.forEach((handle) => {
  // 場景坐標轉視口坐標
  const { x, y } = this.editor.sceneCoordsToViewport(handle.cx, handle.cy);
  const graph = handle.graph;
  graph.x = x - graph.width / 2;
  graph.y = y - graph.height / 2;
  graph.rotation = rect.rotation;

  // 不可見的圖形不渲染(本地調試的時候可以讓它可見)
  if (!graph.getVisible()) {
    return;
  }

  graph.draw();
});

渲染邏輯到此結束。

控制點拾取

在選擇工具下,選中圖形,控制點出現。

接著 hover 到控制點上,更新光標。并且在按下鼠標時,能夠拿到對應的控制點類型,進行對應的旋轉或縮放操作。

這里我們需要判斷光標的位置是否在控制點上,即控制點拾取。

控制點拾取邏輯為:

以渲染順序相反的方向遍歷控制點,調用 hitTest 方法檢測光標是否在控制點的點擊區域上。

如果在,返回 type 和 cursor;否則返回 null。

class ControlHandleManager {
  // ...

  /** 獲取在光標位置的控制點的信息 */
  getHandleInfoByPoint(hitPoint: IPoint) {
    const hitPointVW = this.editor.sceneCoordsToViewport(
      hitPoint.x,
      hitPoint.y,
    );
    
    for (let i = types.length - 1; i >= 0; i--) {
      const type = types[i];
      const handle = this.transformHandles.get(type);
 
      // 是否點中當前控制點
      const isHit = handle.graph.hitTest(
        hitPointVW.x,
        hitPointVW.y,
        handleHitToleration,
      );

      if (isHit) {
        return {
          handleName: type, // 控制點類型
          cursor: handle.getCursor(type, rotation), // 光標
        };
      }
    }
  }  
}

反向很重要,應為可能會有控制點發生重疊,此時應該是在更上方的控制點,也就是后渲染的控制點優先被選中。

光標

getCursor 返回的光標值是動態的,會因為包圍盒的角度不同而變化,這里會有一個簡單的轉換。

const getResizeCursor = (type: string, rotation: number): ICursor => {
  let dDegree = 0;
  switch (type) {
    case 'se':
    case 'nw':
      dDegree = -45;
      break;
    case 'ne':
    case 'sw':
      dDegree = 45;
      break;
    case 'n':
    case 's':
      dDegree = 0;
      break;
    case 'e':
    case 'w':
      dDegree = 90;
      break;
    default:
      console.warn('unknown type', type);
  }

  const degree = rad2Deg(rotation) + dDegree;
  // 這個 degree 精度是很高的,
  // 設置光標時會做一個舍入,匹配一個合法的接近光標值,比如 ne-resize
  return { type: 'resize', degree };
}

旋轉光標同理。

此外,瀏覽器支持的 resize 光標值是有限的。

為了更好的效果是實現 resize0 ~ resize179 代表不同角度的一共 180 個自定義 resize 光標。

或者做一個 “四舍五入”,轉為瀏覽器支持的那幾種 resize 角度,但這樣光標效果不是很好,看起來光標并沒有和控制點垂直,算是一種妥協。

旋轉光標更是不存在了,我們要設計 rotation0 ~ rotation179 共 360 個自定義光標。當然我們可以讓精度降一下,比如只實現偶數值的旋轉角度的光標,比如 rotation0、rotation2、rotation4,也要 180 個。

關于自定義光標的實現方案,本文不深入講解,會單獨寫一篇文章討論。

坐標系

有個容易忽略的問題,就是控制點是繪制在哪個坐標系中的?

是場景坐標系,還是視口坐標系。

如果在場景坐標系中,圖形會隨畫布的縮放或移動 “放大縮小”,比如一根 2px 的線條,在 zoom 為 50% 的畫布下,顯示的效果是 1px。

控制點的寬高是不應該跟隨  zoom 而變化的。

如果你繪制在視口坐標系,寬高不需要考慮,只要轉換一下 x,y。如果在場景坐標中,x、y 不用轉換,但是寬高要除以 zoom。

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

2023-10-19 10:12:34

圖形編輯器開發縮放圖形

2023-07-07 13:56:01

圖形編輯器畫布縮放

2023-01-18 08:30:40

圖形編輯器元素

2023-09-07 08:24:35

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

2023-08-31 11:32:57

圖形編輯器contain

2023-02-06 16:59:57

Canvas編輯器

2023-09-26 07:39:21

2024-01-08 08:30:05

光標圖形編輯器開發游標

2023-09-11 09:02:31

圖形編輯器模塊間的通信

2023-10-10 16:04:30

圖形編輯器格式轉換

2023-10-08 08:11:40

圖形編輯器快捷鍵操作

2023-08-28 08:10:50

Hex圖形編輯器

2023-06-12 08:22:56

圖形編輯器工具

2021-01-16 12:21:59

react-dragareact-resia可視化

2023-02-02 14:07:00

圖形編輯器Canvas

2023-07-31 08:46:07

圖形編輯器圖形自動對齊

2023-04-07 08:02:30

圖形編輯器對齊功能

2023-02-01 09:21:59

圖形編輯器標尺

2011-01-10 16:17:49

2011-09-28 13:28:56

F5虛擬化云計算
點贊
收藏

51CTO技術棧公眾號

亚洲+变态+欧美+另类+精品| 国精产品一区一区三区mba下载| 免费在线看一区| 久色乳综合思思在线视频| 亚洲美女高潮久久久| av岛国在线| 欧美国产精品一区二区三区| 91免费的视频在线播放| 91久久国产视频| 久久中文字幕av一区二区不卡| 欧美xxxxx牲另类人与| 男女高潮又爽又黄又无遮挡| 黄色免费在线观看网站| 久久综合九色综合久久久精品综合| 成人av在线天堂| 免费看日韩毛片| 一本到12不卡视频在线dvd| 亚洲精品久久7777777| 天天久久综合网| 亚洲a∨精品一区二区三区导航| 亚洲综合一区二区三区| 亚洲一区三区电影在线观看| 亚洲 小说区 图片区 都市| 国产一区在线精品| 日韩免费在线播放| 色婷婷av国产精品| 好吊视频一区二区三区四区| 日韩有码在线视频| 永久免费毛片在线观看| 亲子伦视频一区二区三区| 欧美一卡2卡3卡4卡| 冲田杏梨av在线| 成人影院av| 亚洲成av人片在线| 人妻激情另类乱人伦人妻| h视频在线观看免费| 久久综合九色综合97婷婷| 国产精品国产亚洲精品看不卡15| 国产又黄又猛又爽| 久久精品99久久久| 国产日韩欧美中文| 中文字幕激情视频| 日韩精品一区第一页| 欧美在线视频观看| 久久免费激情视频| 国产情侣一区| 欧美亚洲国产成人精品| 色播视频在线播放| 亚洲精品裸体| 7m精品福利视频导航| 国产在线拍揄自揄拍| 国产综合久久| 久久久久久有精品国产| 国语对白一区二区| 影音先锋中文字幕一区| 欧美精品久久久久久久久| 永久免费看黄网站| 韩日欧美一区| 91精品国产91久久久久久最新| 动漫精品一区一码二码三码四码| 日韩亚洲国产精品| 26uuu日韩精品一区二区| 男人午夜免费视频| 久久天堂成人| 国产欧美一区二区三区久久人妖| 一级特黄aaa| 国产高清久久久| 成人三级视频在线观看一区二区| 99在线观看精品视频| 国产成人综合视频| 精品久久久久久中文字幕动漫| 天堂在线中文网| 久久久久亚洲综合| 亚洲一区综合| 欧美伦理免费在线| 欧美日韩国产综合视频在线观看中文| 黄色免费观看视频网站| 日韩成人影音| 欧美疯狂做受xxxx富婆| 亚洲少妇一区二区| 妖精视频一区二区三区免费观看| 中文字幕九色91在线| 加勒比婷婷色综合久久| 在线精品观看| 国产精品av电影| 国产精品久久无码一三区| 成人性视频免费网站| 蜜桃狠狠色伊人亚洲综合网站| 香蕉视频在线看| 亚洲一区二区视频在线观看| 免费无码av片在线观看| 亚洲香蕉久久| 精品亚洲aⅴ在线观看| 奇米网一区二区| 国模一区二区三区| 国产精品久久久久久av下载红粉 | 你懂的免费在线观看| 国产精品私人影院| 久久久久久www| 国产精品久久久久久久久久齐齐| 日韩一二三区视频| 一级片手机在线观看| 欧美激情五月| 国产精品激情av电影在线观看| 99精品人妻无码专区在线视频区| 99久久精品免费看国产免费软件| 亚洲蜜桃在线| 51漫画成人app入口| 欧美精品一卡二卡| 中文字幕丰满乱子伦无码专区| 亚洲精品中文字幕乱码| 日韩免费高清在线观看| 日本久久一级片| 亚洲欧美激情一区二区| 亚欧在线免费观看| 欧美日韩精品一区二区三区在线观看| 久久精品国产69国产精品亚洲 | 在线观看三级视频| 91国产福利在线| 亚洲精品国产成人av在线| 99re6这里只有精品| 欧美在线一区二区视频| 亚洲第一天堂影院| 亚洲人一二三区| 色婷婷狠狠18| 你懂的一区二区三区| 高清欧美一区二区三区| 国产偷拍一区二区| 自拍视频在线观看一区二区| 中文字幕第36页| 国产精品日韩精品中文字幕| 91精品国产91久久久久久久久| 性做久久久久久久久久| 日韩一区有码在线| 欧美婷婷精品激情| 久久爱www成人| 欧美重口另类videos人妖| 免费国产精品视频| 亚洲成人一二三| 女同性αv亚洲女同志| 欧美搞黄网站| 91免费看蜜桃| 欧洲一区二区三区| 亚洲第一福利网| 日韩免费黄色片| 不卡av电影在线播放| www.好吊操| 超碰成人在线观看| 97精品国产97久久久久久春色| 黄色美女一级片| 亚洲永久免费av| xxxwww国产| 亚洲激情精品| 久久精品一二三区| 欧美xnxx| 另类图片亚洲另类| 亚洲xxxx天美| 亚洲不卡在线观看| 女同毛片一区二区三区| 老司机精品视频网站| 亚洲精品久久久久久一区二区| 国产精品99| 久久视频在线免费观看| 亚洲第一色网站| 粉嫩av一区二区三区免费野| 91精品人妻一区二区三区| 日韩av在线发布| 国产又黄又爽免费视频| 亚洲超碰在线观看| 欧美怡春院一区二区三区| 国产三级视频在线| 91精品国产一区二区三区香蕉| 免费无码毛片一区二区app| 9久草视频在线视频精品| 中文字幕乱码人妻综合二区三区| 欧美一级本道电影免费专区| 亚洲va欧美va国产综合久久| av资源中文在线| 亚洲社区在线观看| 97久久人国产精品婷婷| 亚洲国产日韩a在线播放性色| 性欧美13一14内谢| 激情综合网天天干| 成人在线免费观看av| 三上亚洲一区二区| 国产乱码精品一区二区三区不卡| 成人亚洲欧美| 久久高清视频免费| 精品美女视频在线观看免费软件 | 日产精品久久久一区二区福利| 最近高清中文在线字幕在线观看| 日韩一二三区不卡| 波多野结衣视频观看| 亚洲伊人色欲综合网| 中文字幕 自拍| 国产99久久久国产精品免费看| 国产精品99久久免费黑人人妻| 久久精品欧美一区| 美女被啪啪一区二区| 亚洲精品一区二区三区中文字幕| 日韩av免费在线播放| 怡红院在线播放| 中文字幕亚洲二区| 日韩黄色影片| 日韩免费福利电影在线观看| 中文亚洲av片在线观看| 性做久久久久久久久| 色婷婷粉嫩av| 久久综合久久综合久久| 麻豆短视频在线观看| 精品一区二区三区影院在线午夜| 99热在线这里只有精品| 中文在线日韩| 亚洲蜜桃在线| 国产精品密蕾丝视频下载| 成人免费视频网站| 电影一区二区三区久久免费观看| 国产999在线观看| 男女视频在线| 久久这里只有精品99| 二区三区在线播放| 亚洲精品自拍偷拍| 婷婷色在线观看| 精品国产乱码久久久久久老虎| 国产一区二区三区视频免费观看| 在线免费一区三区| 天天操天天操天天操天天| 亚洲大片一区二区三区| 91成人福利视频| 亚洲日本青草视频在线怡红院 | 日韩精品xxx| 九九视频精品免费| 久热精品在线观看视频| 日韩av一区二| 任你操这里只有精品| 久久久成人网| 日韩久久一级片| 久久亚洲精品伦理| 北条麻妃av高潮尖叫在线观看| 亚洲制服av| 成年人视频网站免费观看| 在线亚洲国产精品网站| av黄色在线网站| 国产精品久久777777毛茸茸 | 日本在线观看一区二区| 美女毛片一区二区三区四区| 欧美日韩电影一区二区| 你微笑时很美电视剧整集高清不卡| 你懂的视频在线一区二区| 亚洲动漫在线观看| 欧美日韩一区二| 欧美日韩激情| 婷婷四房综合激情五月| 日韩欧美精品综合| 综合网五月天| 欧美三级网页| 欧美一区二区中文字幕| 久久久久久9| 中国黄色片免费看| 韩国理伦片一区二区三区在线播放| 国产999免费视频| 国产成人精品免费在线| 精品中文字幕在线播放| 久久久国产精华| 天堂av网手机版| 亚洲欧美乱综合| 国产精品二区一区二区aⅴ| 欧美日韩在线视频一区二区| 久草热在线观看| 日韩一区二区在线观看视频播放| 六月丁香综合网| 亚洲天堂成人在线| 浪潮av一区| 久久人91精品久久久久久不卡| japanese23hdxxxx日韩| 成人中文字幕在线观看| 风间由美性色一区二区三区四区 | 色婷婷狠狠综合| 一区二区视频播放| 欧美成人三级在线| 狠狠狠综合7777久夜色撩人| 日韩在线观看网站| heyzo在线播放| 国产精品九九久久久久久久| 欧美高清一级片| 欧洲高清一区二区| 国产在线日韩| 视色视频在线观看| jlzzjlzz亚洲日本少妇| 三级黄色片在线观看| 亚洲成在线观看| 国产精品毛片一区二区在线看舒淇| 欧美精品一区二区在线观看| 91欧美在线视频| 97成人在线视频| 国产成人免费视频网站视频社区| 久久久久久久久久码影片| 99re久久最新地址获取| 成年网站在线免费观看| 国产成a人无v码亚洲福利| 亚洲精品视频网址| 欧美日韩精品在线视频| 国产欧美熟妇另类久久久| 亚洲天堂网站在线观看视频| 免费av不卡在线观看| 国产又爽又黄的激情精品视频 | 国产精品高潮久久| 精品在线视频一区二区三区| 91精品啪在线观看国产18| 日韩在线xxx| 99久久久国产精品| 久久中文字幕无码| 欧美精品777| 超碰免费在线| 日本三级久久久| 欧美成人基地| 国产在线播放观看| 国产91精品免费| 五月综合色婷婷| 欧美日韩午夜精品| 九色视频在线观看免费播放| 久久人人爽人人爽人人片av高请| 欧美经典一区| 艳母动漫在线观看| 免费高清不卡av| 国产高潮呻吟久久| 色综合天天综合在线视频| 免费观看黄一级视频| 久久6精品影院| 国产视频网站一区二区三区| 在线精品日韩| 久久超碰97中文字幕| 丰满的亚洲女人毛茸茸| 在线免费观看视频一区| 欧美女子与性| 日韩免费av一区二区| 国精一区二区| 9久久婷婷国产综合精品性色 | 成人日韩精品| 欧美一级爱爱| 免费成人在线网站| 黄色一级片一级片| 欧美另类一区二区三区| 黄色在线免费| 99国精产品一二二线| 欧美午夜不卡| 国产伦精品一区二区三区精品| 午夜伦理一区二区| 亚洲 精品 综合 精品 自拍| 欧美伊久线香蕉线新在线| 国产剧情一区| 亚洲 欧美 日韩系列| 亚洲欧洲www| 性欧美8khd高清极品| 久久久久国产视频| 乱亲女h秽乱长久久久| 无遮挡又爽又刺激的视频 | 日韩电影大片中文字幕| 亚洲一区资源| 亚洲伊人婷婷| 福利一区二区在线观看| www.国产高清| 在线精品播放av| 精品亚洲二区| 青青草视频在线免费播放 | 亚洲电影成人av99爱色| 激情国产在线| 亚洲高清123| 国产精品1区2区3区| 日韩精品一卡二卡| 亚洲人成在线观看网站高清| 涩涩涩久久久成人精品| 久久这里只有精品8| 久久久www免费人成精品| 91 中文字幕| 午夜精品久久久久久久99热| 经典一区二区| 日本55丰满熟妇厨房伦| 红桃视频成人在线观看| 亚洲欧美视频一区二区| 99国产超薄丝袜足j在线观看| 在线亚洲免费| 婷婷激情四射网| 日韩精品在线免费观看| 国产精品久久久久久av公交车| 你真棒插曲来救救我在线观看| 久久精品一区四区| 亚洲不卡免费视频| 国产精品九九九| 亚洲人体大胆视频| 任我爽在线视频| 精品无码久久久久久国产| 91国产一区| 国产xxxxx视频| 亚洲国产日韩a在线播放| 欧美极品视频| 日本高清不卡一区二区三| 国产成人在线观看| 成人免费毛片视频| 久久久久国产精品免费网站|