亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > HTML > 正文

canvas里面如何基于隨機點繪制一個多邊形的方法

2024-08-26 00:20:44
字體:
來源:轉載
供稿:網友

起因

今天在學習《HTML5+Javascript動畫基礎》這本書的時候,在第八章的第三節講到如何用三個彈簧連接三個點來做拉伸運動。

在做完例子之后,就想到如果是四個點,五個點,怎么樣。

就改寫了一下代碼,把點的數目變量化。最終的效果是能實現各個點最終的拉伸運動到平衡,可是點之間的連線不是很好看,有些是交叉的。

canvas,多邊形

于是就想著能不能優化這一塊。

旋轉連線

前面例子里面的點,都是隨機位置,所以連線不可控。所以想先從這塊著手。

先以某一個點為參照點,獲得其他點相對于這個點的角度。

然后按照角度從小到大的去連接這些點,這樣就能畫出一個正常的多邊形了。

大致實現代碼如下:

let balls = [];let ballNum = 6;let firstBall = null;while(ballNum--) {  let ball = new Ball(20, parseColor(Math.random() * 0xffffff))  ball.x = Math.random() * width;  ball.y = Math.random() * height;  balls.push(ball)  if (!firstBall) {    firstBall = ball    ball.angle = 0  } else {    const dx = ball.x - firstBall.x,          dy = ball.y - firstBall.y;    ball.angle = Math.atan2(dy, dx);  }}// 嘗試讓球連線是一個正多邊形balls = balls.sort((ballA, ballB) => {  return ballA.angle - ballB.angle})

這樣在最后繪制連線的時候,遍歷數組就能按照角度從小到大來繪制了。

效果如下:

canvas,多邊形

這樣是能極大的減少交叉線的情況,可還是無法完全避免。

接下來,想嘗試優化這個方案,比如angle用Math.abs來取正,或者每一個點都找夾角最小的點來連線??墒墙Y果都不行,無法避免交叉線。

基于中心點旋轉

后面又想到一個思路,如果能確定多邊形的中心點,那么分別計算所有點相對于中心點的夾角,就能以順時針或者逆時針來連接這些點。

可是在網上找了半天,所有點算法里面,都是要求有一系列按某個時針順序排列的點。

可是如果我有這些點,就已經能繪制多邊形了。只好放棄

X軸兩極點分割

無奈之下只好找Google,然后就發現了知乎上的一個答案挺好的: 如何將平面上無序的一組點連成一個簡單多邊形?

具體算法描述,大家看那個答案就好,我就不贅述了。

不過在連接上鏈和下鏈的時候,其實只要保證上鏈是X軸降序連接,下鏈是X軸升序連接即可(以逆時針方向繪制)。至于X軸相同的點,不管是優先Y軸大的還是小的都可以。

實現的時候,是嚴格按照答案里面的算法實現的。

在判斷一個點是屬于上鏈還是下鏈的時候,一開始想的是基于兩點確定直線的函數方程,再引入點的坐標來計算。不過后面想到,所有的點都以最左邊的極點來計算斜角,然后根據角度大小來劃分,視覺上更好理解。

大致代碼如下:

let balls = [];let tempBalls = [];let ballNum = 6;let isDragingBall = false;while(ballNum--) {  let ball = new Ball(10, parseColor(Math.random() * 0xffffff))  ball.x = Math.random() * width;  ball.y = Math.random() * height;  tempBalls.push(ball)}// 讓點按X軸升序排序tempBalls = tempBalls.sort((ballA, ballB) => {  return ballA.x - ballB.x})// 找X軸左右極點let firstBall = tempBalls[0],    lastBall = tempBalls[tempBalls.length -1];let smallXBalls = tempBalls.filter(ball => ball.x === firstBall.x),    bigXBalls = tempBalls.filter(ball => ball.x === lastBall.x)// 處理左右極點有多個的情況if (smallXBalls.length > 1) {  smallXBalls.sort((ballA, ballB) => {    return ballB.y - ballA.y  })}if (bigXBalls.length > 1) {  bigXBalls.sort((ballA, ballB) => {    return ballB.y - ballA.y  })}firstBall = smallXBalls[0]lastBall = bigXBalls[0]// 獲得極點連線的角度let splitLineAngle = Math.atan2(lastBall.y - firstBall.y, lastBall.x - firstBall.x);let upperBalls = [],    lowerBalls = [];// 所有其他點跟firstBall計算角度// 大于splitLineAngle的都是下鏈// 其他是上鏈tempBalls.forEach(ball => {  if (ball === firstBall || ball === lastBall) {    return false  }  let angle = Math.atan2(ball.y - firstBall.y, ball.x - firstBall.x);  if (angle > splitLineAngle) {    lowerBalls.push(ball)  } else {    upperBalls.push(ball)  }})// 處理X軸相同情況的排序lowerBalls = lowerBalls.sort((ballA, ballB) => {  if (ballA.x !== ballB.x) {    return ballA.x - ballB.x  }  return ballB.y - ballA.y})upperBalls = upperBalls.sort((ballA, ballB) => {  if (ballA.x !== ballB.x) {    return ballB.x - ballA.x  }  return ballB.y - ballB.x})// 逆時針連接所有的點balls = [firstBall].concat(lowerBalls, [lastBall], upperBalls)balls = balls.map((ball, i) => {  ball.text = i + 1;  return ball})

最終返回的balls,就是按逆時針排序的多邊形的點了。

效果如下:

canvas,多邊形

各個球的內部狀態如下:

canvas,多邊形

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到HTML教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
免费99精品国产自在在线| 亚洲欧美日韩精品久久亚洲区| 国产亚洲欧美一区| 8x海外华人永久免费日韩内陆视频| 91免费福利视频| 欧美激情一区二区三区久久久| 午夜精品一区二区三区视频免费看| 国产日韩视频在线观看| 亚洲精品自拍偷拍| 日韩精品在线私人| 欧美二区在线播放| 久久久久久一区二区三区| 91精品国产99| 日本亚洲精品在线观看| 欧美激情欧美狂野欧美精品| 久久国产精品网站| 精品国产91久久久久久老师| 中文精品99久久国产香蕉| 92版电视剧仙鹤神针在线观看| 欧美一级淫片videoshd| 九九久久久久99精品| 日韩av在线天堂网| 国产精品高潮粉嫩av| 中文字幕欧美日韩va免费视频| 精品久久久久久久久久久| 亚洲天堂免费观看| 91精品国产高清久久久久久91| 日韩综合中文字幕| 亚洲国产精久久久久久| 国产精品美女久久久免费| 热re99久久精品国产66热| 欧美老少做受xxxx高潮| 亚洲欧洲一区二区三区在线观看| 日韩免费av在线| 精品久久久香蕉免费精品视频| 国产精品午夜国产小视频| 91精品国产综合久久男男| 成人黄色在线观看| 久久精品一区中文字幕| 国产99久久精品一区二区永久免费| 欧美精品aaa| 亚洲白拍色综合图区| 国产mv久久久| 欧美黑人一区二区三区| 久久99久国产精品黄毛片入口| 久久久精品在线| 国产成人高清激情视频在线观看| 日韩成人中文字幕在线观看| 亚洲欧美一区二区三区情侣bbw| 国产精品中文在线| 91久久久国产精品| 国产噜噜噜噜噜久久久久久久久| 神马久久久久久| 亚洲欧美国产精品va在线观看| 国产成人精品日本亚洲| 中文字幕视频在线免费欧美日韩综合在线看| 亚洲精品久久久久国产| 亚洲成人中文字幕| 在线视频欧美性高潮| 欧美寡妇偷汉性猛交| 日韩在线观看免费| 日韩视频免费观看| 欧美寡妇偷汉性猛交| 国产三级精品网站| 亚洲男人天堂久| 欧美国产在线视频| 日韩精品在线私人| 欧美人与性动交| 97精品在线视频| 亚洲va久久久噜噜噜| 亚洲免费电影一区| 国产精自产拍久久久久久蜜| 日韩av大片免费看| 亚洲97在线观看| 日韩精品在线私人| 亚洲级视频在线观看免费1级| 成人黄色大片在线免费观看| 国产日韩欧美在线| 欧美疯狂xxxx大交乱88av| 91精品视频在线播放| 国产精品嫩草影院一区二区| 日韩av免费在线看| 久久久噜噜噜久久| 国产一区二区视频在线观看| 久久久国产在线视频| 久久久久久久激情视频| 欧美二区乱c黑人| 久久久亚洲精品视频| 91亚洲永久免费精品| 国产视频亚洲精品| 国产精品福利在线观看网址| 尤物九九久久国产精品的分类| 亚洲国产精彩中文乱码av在线播放| 91精品在线观| 国产精品视频一区国模私拍| 精品久久久久久久久久| 欧美日韩成人在线播放| 亚洲综合在线中文字幕| 日韩亚洲欧美中文高清在线| 欧美在线影院在线视频| 日韩欧美在线观看视频| 91精品国产91久久久久| 国内精品在线一区| 成人做爰www免费看视频网站| 91中文在线观看| 久久色免费在线视频| 亚洲一区二区福利| 欧美在线日韩在线| 亚洲国产精彩中文乱码av在线播放| 亚洲品质视频自拍网| 欧美激情一区二区三区久久久| 亚洲国产福利在线| 亚洲精品成人网| 欧美色xxxx| 国产精品久久久久久av| 日韩欧美国产骚| 久久精品视频中文字幕| 欧美极品少妇xxxxⅹ裸体艺术| 国产精品一区=区| 欧美性jizz18性欧美| 日韩欧美中文第一页| 国产精品视频一| 亚洲一区免费网站| 亚洲综合在线中文字幕| 日韩在线观看免费高清| 不卡av在线网站| 国产精品网站入口| 久热精品视频在线| 亚洲资源在线看| 亚洲国产精品国自产拍av秋霞| 欧美老女人www| 成人欧美一区二区三区在线湿哒哒| 欧美午夜片欧美片在线观看| 国内精品小视频| 亚洲精品99久久久久中文字幕| 国产中文字幕亚洲| 欧美精品videossex性护士| 亚洲欧洲在线看| 国产在线视频91| 久久夜色撩人精品| 精品人伦一区二区三区蜜桃免费| 欧美天天综合色影久久精品| 欧美在线影院在线视频| 亚洲最大在线视频| 日韩电影视频免费| 久久婷婷国产麻豆91天堂| 欧美成在线观看| 欧美日韩一二三四五区| 亚洲美女性视频| 欧美午夜美女看片| 91久久精品美女高潮| 97精品久久久中文字幕免费| 国产在线98福利播放视频| 久久99久久亚洲国产| 海角国产乱辈乱精品视频| 91高潮在线观看| 色婷婷综合成人av| 国产精品入口福利| 国产日本欧美一区二区三区在线| 91精品在线国产| 色七七影院综合| 久久久久久有精品国产| 色噜噜狠狠狠综合曰曰曰88av| 欧美高清视频在线观看|