js多邊形(不規則的形狀)之合併

来源:https://www.cnblogs.com/weihexinCode/archive/2022/06/10/16363434.html
-Advertisement-
Play Games

1 /* Polygon 多邊形 2 3 parameter: 4 path: Array[x, y]; 5 6 attribute: 7 8 //只讀屬性 9 path: Array[x, y]; 10 11 method: 12 add(x, y): this; //x,y添加至path; 13 ...


 

  1 /* Polygon 多邊形
  2 
  3 parameter: 
  4     path: Array[x, y];
  5 
  6 attribute:
  7 
  8     //只讀屬性
  9     path: Array[x, y]; 
 10 
 11 method:
 12     add(x, y): this;             //x,y添加至path;
 13     containsPoint(x, y): Bool;    //x,y是否在多邊形的內部(註意: 在路徑上也返回 true)
 14     merge(polygon): this;         //合併; 假設polygon與this存在重疊的部分
 15     
 16 */
 17 class Polygon{
 18 
 19     #position = null;
 20     #path2D = null;
 21 
 22     get path(){
 23         
 24         return this.#position;
 25 
 26     }
 27 
 28     constructor(path = []){
 29         this.#position = path;
 30 
 31         this.#path2D = new Path2D();
 32         
 33         var len = path.length;
 34         if(len >= 2){
 35             if(len % 2 !== 0){
 36                 len -= 1;
 37                 path.splice(len, 1);
 38             }
 39 
 40             const con = this.#path2D;
 41             con.moveTo(path[0], path[1]);
 42             for(let k = 2; k < len; k+=2) con.lineTo(path[k], path[k+1]);
 43             
 44         }
 45 
 46     }
 47 
 48     add(x, y){
 49         this.#position.push(x, y);
 50         this.#path2D.lineTo(x, y);
 51         return this;
 52     }
 53 
 54     containsPoint(x, y){
 55         
 56         return UTILS.emptyContext.isPointInPath(this.#path2D, x, y);
 57 
 58     }
 59 
 60     toPoints(){
 61         const path = this.path, len = path.length, result = [];
 62         
 63         for(let k = 0; k < len; k += 2) result.push(new Point(path[k], path[k+1]));
 64 
 65         return result;
 66     }
 67 
 68     toLines(){
 69         const path = this.path, len = path.length, result = [];
 70         
 71         for(let k = 0, x = NaN, y; k < len; k += 2){
 72 
 73             if(isNaN(x)){
 74                 x = path[k];
 75                 y = path[k+1];
 76                 continue;
 77             }
 78 
 79             const line = new Line(x, y, path[k], path[k+1]);
 80             
 81             x = line.x1;
 82             y = line.y1;
 83 
 84             result.push(line);
 85 
 86         }
 87 
 88         return result;
 89     }
 90 
 91     merge(polygon){
 92 
 93         const linesA = this.toLines(), linesB = polygon.toLines(), nodes = [], newLines = [];
 94 
 95         //收集所有的交點 (保存至 line)
 96         for(let k = 0, lenB = linesB.length, lenA = linesA.length, point = new Point(), pointB = new Point(); k < lenA; k++){
 97             const lineA = linesA[k];
 98             lineA.nodes = [];
 99 
100             for(let i = 0; i < lenB; i++){
101                 const lineB = linesB[i];
102                 if(lineB.nodes === undefined) lineB.nodes = [];
103                 if(lineA.intersectPoint(lineB, point) === point){
104                     const node = {
105                         lineA: lineA, 
106                         lineB: lineB, 
107                         point: point.clone(),
108                         disA: point.distanceCompare(pointB.set(lineA.x, lineA.y)),
109                         disB: point.distanceCompare(pointB.set(lineB.x, lineB.y)),
110                     }
111                     lineA.nodes.push(node);
112                     lineB.nodes.push(node);
113                     nodes.push(node);
114                 }
115 
116             }
117 
118         }
119 
120         //獲取介入點
121         var startLine = null;
122         for(let k = 0, len = nodes.length, node; k < len; k++){
123             node = nodes[k];
124             if(node.lineA.nodes.length === 1 || node.lineA.nodes.length === 2 && 
125             node.lineB.nodes.length === 1 || node.lineB.nodes.length === 2){
126                 startLine = node.lineA;
127                 break;
128             }
129         }
130 
131         if(startLine === null){
132             console.warn('Polygon: 找不到介入點, 終止了合併');
133             return newLines;
134         }
135 
136         //交點以原點為目標排序
137         for(let k = 0, sotr = function (a,b){return a.disA - b.disA;}, countA = linesA.length; k < countA; k++) linesA[k].nodes.sort(sotr);
138         for(let k = 0, sotr = function (a,b){return a.disB - b.disB;}, countB = linesB.length; k < countB; k++) linesB[k].nodes.sort(sotr);
139 
140         const result_loop = {
141             lines: linesA,
142             loopType: 'next',
143             line: startLine,
144             count: startLine.nodes.length,
145             indexed: linesA.indexOf(startLine),
146         },
147         
148         //遍歷某條線
149         loop = (lines, index, loopType) => {
150             const length = lines.length, indexed = index, model = lines === linesA ? polygon : this;
151 
152             var line = lines[index];
153 
154             while(true){
155                 if(loopType === 'back' && newLines.includes(line) === false && model.containsPoint(line.x, line.y) === false) newLines.push(line);
156                 if(loopType === 'next') index = index === length - 1 ? 0 : index + 1;
157                 else if(loopType === 'back') index = index === 0 ? length - 1 : index - 1;
158 
159                 line = lines[index];
160                 if(loopType === 'next' && newLines.includes(line) === false && model.containsPoint(line.x, line.y) === false) newLines.push(line);
161 
162                 result_loop.count = line.nodes.length;
163                 if(result_loop.count !== 0){
164                     result_loop.lines = lines;
165                     result_loop.loopType = loopType;
166                     result_loop.line = line;
167                     result_loop.indexed = index;
168                     return result_loop;
169                 }
170                 
171                 if(indexed === index) break;
172 
173             }
174 
175         },
176 
177         //更新或創建交點的索引
178         setNodeIndex = (lines, index, loopType) => {
179             const line = lines[index], count = line.nodes.length;
180             if(loopType === undefined) loopType = line.loopType;
181             else if(line.loopType === undefined) line.loopType = loopType;
182 
183             if(loopType === undefined) return;
184 
185             if(line.nodeIndex === undefined){
186                 line.nodeIndex = loopType === 'next' ? 0 : count - 1;
187                 line.nodeState = count === 1 ? 'end' : 'start';
188                 line._loopType = loopType;
189             }
190 
191             else{
192                 if(line.nodeState === 'end' || line.nodeState === ''){
193                     line.nodeState = '';
194                     return;
195                 }
196 
197                 if(line._loopType === 'next'){
198                     line.nodeIndex += 1;
199 
200                     if(line.nodeIndex === count - 1) line.nodeState = 'end';
201                     else line.nodeState = 'run';
202                 }
203 
204                 else if(line._loopType === 'back'){
205                     line.nodeIndex -= 1;
206 
207                     if(line.nodeIndex === 0) line.nodeState = 'end';
208                     else line.nodeState = 'run';
209                 }
210 
211             }
212 
213         },
214 
215         //只有在跳線的時候才執行此方法, 如果跳線的話: 某條線上的交點必然在兩端;
216         getLoopType = (lines, index, nodePoint) => {
217             const line = lines[index], lineNext = lines[index === lines.length - 1 ? 0 : index + 1],
218 
219             model = lines === linesA ? polygon : this,
220             isLineBack = newLines.includes(line) === false && model.containsPoint(line.x, line.y) === false,
221             isLineNext = newLines.includes(lineNext) === false && model.containsPoint(lineNext.x, lineNext.y) === false;
222             
223             if(isLineBack && isLineNext){
224                 const len = line.nodes.length;
225                 if(len >= 2){
226                     if(line.nodes[len - 1].point.equals(nodePoint)) return 'next';
227                     else if(line.nodes[0].point.equals(nodePoint)) return 'back';
228                 }
229                 
230                 else console.warn('路徑複雜', line);
231                 
232             }
233 
234             else if(isLineNext){
235                 return 'next';
236             }
237 
238             else if(isLineBack){
239                 return 'back';
240             }
241 
242             return '';
243         },
244 
245         //處理擁有交點的線
246         computeNodes = v => {
247             if(v === undefined) return;
248 
249             setNodeIndex(v.lines, v.lines.indexOf(v.line), v.loopType);
250             
251             //
252             const node = v.line.nodes[v.line.nodeIndex], model = v.lines === linesA ? polygon : this;
253             if(newLines.includes(node.point) === false) newLines.push(node.point);
254 
255             var lines, index, loopType, line;
256 
257             //
258             const lineR = v.lines === linesA ? node.lineB : node.lineA, 
259             linesR = lineR === node.lineA ? linesA : linesB;
260             setNodeIndex(linesR, linesR.indexOf(lineR));
261 
262             const nodeState = lineR.nodeState !== undefined ? lineR.nodeState : v.line.nodeState;
263 
264             switch(nodeState){
265 
266                 //不跳線
267                 case 'run': 
268                     lines = v.lines; 
269                     index = v.indexed; 
270                     loopType = v.loopType;
271                     computeNodes(loop(lines, index, loopType));
272                     break;
273 
274                 //跳線
275                 case 'start':
276                 case 'end': 
277                     lines = model === polygon ? linesB : linesA;
278                     line = lines === linesA ? node.lineA : node.lineB;
279                     index = lines.indexOf(line); 
280                     loopType = getLoopType(lines, index, node.point);
281                     
282                     if(loopType !== ''){
283                         line.loopType = loopType;
284                         //if(loopType === 'back' && newLines.includes(line) === false && model.containsPoint(line.x, line.y) === false) newLines.push(line);
285                         computeNodes(loop(lines, index, loopType));
286                     }
287                     break;
288                 
289 
290                 default: break;
291 
292             }
293 
294         }
295 
296         computeNodes(result_loop);
297     
298         return newLines;
299     }
300 
301 }
部分代碼

 

註意: Polygon 目前並不完美

 

parameter:

    path: Array[x, y];
attribute:
    //只讀屬性     path: Array[x, y];
method:     add(x, y): this;            //x,y添加至path;     containsPoint(x, y): Bool;  //x,y是否在多邊形的內部(註意: 在路徑上也返回 true)     merge(polygon): this;       //合併; 假設polygon與this存在重疊的部分     這個時專門用於測試bug的ui控制項:
  1 /* PolygonTest 調試類 Polygon (可視化調試: 多邊形之合併)
  2 
  3     無任何屬性, new就完事了!
  4 
  5 */
  6 class PolygonTest{
  7 
  8     constructor(){
  9         const path = [], box = new Box(),
 10 
 11         car = new CanvasAnimateRender({
 12             width: window.innerWidth, 
 13             height: window.innerHeight,
 14         }).render(), 
 15     
 16         cab = car.add(new CanvasAnimateBox()).size(car.box.w, car.box.h, true);
 17         
 18         function draw(obj){
 19             console.log(obj);
 20     
 21             target._redraw();
 22     
 23             obj.forEach((v, i) => {
 24                 const str = String(i), 
 25                 size = cab.textWidth(str, 20), 
 26                 width = size < 20 ? 20 : size;
 27                 
 28                 cab.rect(2, box.set(v.x, v.y, width, width), 2).fill('#fff');
 29                 cab.text(str, '#0000ff', box.pos(v.x + (width - size)/2, v.y));
 30     
 31                 if(v.x1 === undefined) cab.stroke('#ffff00');
 32                 
 33             });
 34     
 35             car.redraw();
 36         }
 37     
 38     
 39         const target = {
 40             title: '測試多邊形之合併(此類並不完美)',
 41             polygonA: null,
 42             polygonB: null,
 43             centerA: new Box(),
 44             centerB: new Box(),
 45             length: 0,
 46             mergeType: 0,
 47     
 48             merge(){
 49                 if(this.polygonA === null || this.polygonB === null) return console.warn('必須已生成兩個多邊形');
 50                 if(this.mergeType === 0) draw(this.polygonA.merge(this.polygonB));
 51                 else draw(this.polygonB.merge(this.polygonA));
 52                 
 53             },
 54     
 55             setPolygonA(){
 56                 this.polygonA = new Polygon(path.concat());
 57                 this.centerA.setFromPolygon(this.polygonA, false);
 58                 update_offsetNum();
 59             },
 60     
 61             setPolygonB(){
 62                 this.polygonB = new Polygon(path.concat());
 63                 this.centerB.setFromPolygon(this.polygonB, false);
 64                 update_offsetNum();
 65             },
 66     
 67             generate(){
 68                 if(path.length < 6) return console.warn('路徑至少需要3個點');
 69     
 70                 if(this.polygonA !== null && this.polygonB !== null){
 71                     const _path = path.concat();
 72                     this.clear();
 73                     _path.forEach(v => path.push(v));
 74                     this.generate();
 75                     return;
 76                 }
 77     
 78                 else{
 79                     cab.path(path, true);
 80     
 81                     if(this.polygonA === null) this.setPolygonA();
 82                     else this.setPolygonB();
 83                     path.length = 0;
 84                     
 85                 }
 86                 
 87                 this._redraw();
 88                 car.redraw();
 89             },
 90     
 91             clear(){
 92                 path.length = 0;
 93                 this.polygonA = this.polygonB = null;
 94                 this.centerA.set(0,0,0,0);
 95                 this.centerB.set(0,0,0,0);
 96                 cab.clear();
 97                 car.clear();
 98             },
 99     
100             clearPath(){
101                 path.length = 0;
102                 cab.clear();
103                 this._redraw();
104                 car.redraw();
105             },
106     
107             _redraw(){
108                 if(this.polygonA !== null){
109                     cab.clear().path(this.polygonA.path).stroke('#00ff00', 1);
110                     this.sign(this.polygonA.path, 'rgba(0,255,0,1)', 'rgba(0,255,0,0.6)');
111                     if(this.polygonB !== null){
112                         cab.path(this.polygonB.path).stroke('#ff0000', 1);
113                         this.sign(this.polygonB.path, 'rgba(255,0,0,1)', 'rgba(255,0,0,0.6)');
114                     }
115                     
116                 }
117                 
118             },
119     
120             //標記線的第一和第二個點
121             sign(path, color1, color2){
122                 cab.rect(5, box.set(path[0]-5, path[1]-5, 10, 10), 2).fill(color1);
123                 cab.rect(5, box.set(path[2]-5, path[3]-5, 10, 10), 2).fill(color2);
124             },
125     
126             //偏移路徑
127             offsetTimer: new Timer(()=>target.offset(), 300, 1, false),
128             offsetNum: new Point(0, 0),
129             offsetTarget: 'polygonA',
130             offsetVN: 'x',
131             offsetPoint: new Point(),
132             offset(){
133                 if(this[this.offsetTarget] === null) return;
134                 path.length = 0;
135                 
136                 const pathNew = path, pathOld = this[this.offsetTarget].path, point = this.offsetPoint, 
137                 center = this.offsetTarget === 'polygonA' ? 'centerA' : 'centerB',
138                 x = this.offsetNum[this.offsetVN] * car.box[this.offsetVN === 'x' ? 'w' : 'h'];
139                 
140                 for(let k = 0, len = pathOld.length; k < len; k+=2){
141                     point.set(pathOld[k], pathOld[k+1]);
142                     point[this.offsetVN] = point[this.offsetVN] - this[center][this.offsetVN] + x;
143                     pathNew.push(point.x, point.y);
144                 }
145     
146                 this[this.offsetTarget === 'polygonA' ? 'setPolygonA' : 'setPolygonB']();
147                 path.length = 0;
	   

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 不僅 SQL,對所有的編程語言來說,函數都起著至關重要的作用。函數就像是編程語言的“道具箱”,每種編程語言都準備了非常多的函數。 使用函數,我們可以實現計算、字元串操作、日期計算等各種各樣的運算。 本文重點 根據用途,函數可以大致分為算術函數、字元串函數、日期函數、轉換函聚合函數。 函數的種類很多, ...
  • 導讀: 大家好,今天主要分享數據分析平臺的平臺演進以及我們在上面沉澱的一些數據分析方法是如何應用的。 具體分以下四部分: Part1:主要介紹下我所在的部門,數據平臺部主要是做什麼的,大概涉及到哪些業務,在整個數據流程當中數據平臺部負責哪些東西; Part2:既然我們講數據分析平臺,那麼數據分析是什 ...
  • 前幾篇我們一起學習了 SQL 如何對錶進行創建、更新和刪除操作、SQL SELECT WHERE 語句如何指定一個或多個查詢條件 和 SQL 如何插入、刪除和更新數據 等資料庫的基本操作方法。 從本文開始,我們將會在這些基本方法的基礎上,學習一些實際應用中的方法。 本文將以此前學過的 SELECT ...
  • 華為運動健康服務(HUAWEI Health Kit)為三方生態應用提供了REST API介面,通過其介面可訪問資料庫,為用戶提供運動健康類數據服務。在實際的集成過程中,開發者們可能會遇到各種問題,這裡我們將典型問題進行分享和總結,希望為其他遇到類似問題的開發者提供參考。 1. 註冊訂閱通知能力後, ...
  • springboot+layui 整合百度富文本編輯器ueditor入門使用教程(踩過的坑) 寫在前面: ​ 富文本編輯器,Multi-function Text Editor, 簡稱 MTE, 是一種可內嵌於瀏覽器,所見即所得的文本編輯器。 ​ UEditor 是由百度「FEX前端研發團隊」開發的 ...
  • 基於Linux環境下的個人網站搭建 一.下載工具 遠程主機: 1.jdk 下載地址 下拉網頁看到Java SE 8u221,選擇伺服器 JRE 選擇linux.tar.gz文件下載 o 2.tomcat 下載地址 選擇.tar.gz文件下載 本地: 1.下載putty:使用Windows遠程連接Li ...
  • 註意:編程式導航(push|replace)才會有這種情況的異常,聲明式導航是沒有這種問題,因為聲明式導航內部已經解決這種問題。 這種異常,對於程式沒有任何影響的。 為什麼會出現這種現象: 由於vue-router最新版本3.5.2,引入了promise,當傳遞參數多次且重覆,會拋出異常,因此出現上 ...
  • 什麼是生命周期? 從出生到成長,最後到死亡,這個過程的時間可以理解為生命周期。 React中的組件也是這麼一個過程。 React的生命周期分為三個階段:掛載期(也叫實例化期)、更新期(也叫存在期)、卸載期(也叫銷毀期)。 在每個周期中React都提供了一些鉤子函數。 生命周期的描述如下: 掛載期:一 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...