js实现固定区域内的不重叠随机圆

作者:__Charming__ 时间:2024-05-13 09:18:40 

本文实例为大家分享了js实现固定区域内的不重叠随机圆,供大家参考,具体内容如下

关键词:js、固定区域、不重叠、随机圆,半径固定、半径随机

最近公司有一个需求就是在一个固定的区域(500X500)内显示10个圆,且半径固定,而且不重叠

因为圆的个数固定,而且半径固定,那么就有可能会没有解决方案。

不过其实也没有很难,处理好半径的最大值就好了。

效果图:

js实现固定区域内的不重叠随机圆

思路:

(固定半径)

step1:先在区域内生成一个随机的圆心坐标,
step2:然后拿一个固定半径(从大到小拿固定半径)
step3:判断圆心和半径是否合法(是否超边距,或者两个圆相交)
step4:如果不合法,重新执行step2和step3
step5:如果合法,记为一个新圆
step6:重复step1~5,直到生成10个圆

(随机半径)

step1:先在区域内生成一个随机的圆心坐标,
step2:根据圆心坐标,与其他圆比较,获取最短的圆心距减去比较圆的半径(圆心距-R n  RnR_n)的值,作为新圆的半径(这样就会生成一个相切的圆)
step3:判断圆心和半径是否合法(是否超边距)
step4:如果不合法,重新执行step2和step3
step5:如果合法,记为一个新圆
step6:重复step1~5,直到生成10个圆

代码:


// 参数
let obj = {
 id: string, // canvas 的id
 fix:boolean, // 是否固定半径,默认为false
 minMargin: Number, // 两个圆的最短距离,默认为10
 minRadius: Number, 最小的圆半径,默认为30
 radiuArr: Array, 圆的半径的数组,当fix为true时该值必须填
 total: Number ,圆的个数,默认为10
}

<!DOCTYPE html>
<html>
<body>

<canvas id="myCanvas" width="500" height="500" style="border:1px solid #d3d3d3;">
Your browser does not support the HTML5 canvas tag.
</canvas>

<script>
class Circle {
 constructor(x, y, r, color){
   this.x = x
   this.y = y
   this.r = r,
   this.c = color ? color : this.getRandomColor()
 }
 getRandomColor(){
   let r = Math.floor(Math.random()*100) + 155
   let g = Math.floor(Math.random()*100) + 155
   let b = Math.floor(Math.random()*100) + 155
   return `rgb(${r},${g},${b})`
 }
}

class RandomCircle {

constructor(obj) {
   this.c      = document.getElementById(obj.id);
   this.ctx     = this.c.getContext("2d");
   this.dWidth   = this.c.width;
   this.dHeight   = this.c.height

this.fix     = obj.fix || false;
   this.minMargin  = obj.minMargin || 10
   this.minRadius  = obj.minRadius || 30
   this.radiuArr  = obj.radiuArr || [80,70,60,50,45,40,40,35,35,30]
   this.total = obj.total || 10
   this.circleArray = []
   this.circleNumber = 1
 }

drawOneCircle(c) {
   let ctx = this.ctx;
   ctx.beginPath();
   ctx.strokeStyle = c.c;
   ctx.fillStyle=c.c;
   ctx.arc(c.x, c.y, c.r, 0, 2*Math.PI);
   ctx.stroke();
   ctx.fill();

ctx.fillStyle='black';
   ctx.fillText('No:'+this.circleNumber, c.x-10, c.y-5);
   ctx.fillText('R:'+c.r, c.x-10, c.y+5);
   this.circleNumber ++
 }

check(x,y,r) {
   return !(x+r > this.dWidth || x-r < 0 || y + r > this.dHeight || y-r < 0)
 }

// 获取一个新圆的半径,主要判断半径与最近的一个圆的距离
 getR(x,y) {
   if (this.circleArray.length === 0) return Math.floor(Math.random()*20 + 80)
   let lenArr = this.circleArray.map(c => {
     let xSpan = c.x-x
     let ySpan = c.y-y
     return Math.floor(Math.sqrt(Math.pow(xSpan,2) + Math.pow(ySpan,2))) - c.r
   })
   let minCircleLen = Math.min(...lenArr)
   let minC = this.circleArray[lenArr.indexOf(minCircleLen)]
   let tempR = this.fix ? this.radiuArr[this.circleArray.length] : minCircleLen - this.minMargin
   let bool = this.fix ? (tempR <= minCircleLen - minC.r) : (tempR >= this.minRadius)
   return bool ? tempR : false
 }

// 生成一个圆,随机生成圆心。
 // 如果连续生成200次半径都没有合适的话,终止进程
 createOneCircle(){
   let x,y,r;
   let createCircleTimes = 0
   while(true) {
     createCircleTimes ++
     x = Math.floor(Math.random()*this.dWidth)
     y = Math.floor(Math.random()*this.dHeight)
     let TR = this.getR(x,y)
     if (!TR) {
       continue;
     } else {
       r = TR
     }
     if (this.check(x,y,r) || createCircleTimes > 200) {
       break
     }

}
   this.check(x,y,r) && this.circleArray.push(new Circle(x, y, r))

}

// 如果生成100次新圆都失败的话,终止方案。
 // 如果生成100种方案都没有合适可用的话,终止进程。
 init() {
   let n = 0
   while(this.circleArray.length < this.total) {
     this.circleArray = []
     let i = 0;
     while (this.circleArray.length < this.total) {
       this.createOneCircle()
       i ++
       if (i >= 100) {
         break;
       }
     }
     n ++
     if (n > 100) {
       break;
     }
   }
   // 根据半径从大到小画圆。
   this.circleArray.sort( (a,b) => b.r-a.r).forEach(c => {
     this.drawOneCircle(c)
   })
 }
}

let p = new RandomCircle({id: 'myCanvas', total: 20})
p.init()

console.log(p.circleArray)

</script>
</body>
</html>

来源:https://blog.csdn.net/qq_41882147/article/details/81063160

标签:js,固定区域,随机圆
0
投稿

猜你喜欢

  • AJAX初体验之上手篇

    2007-08-23 08:50:00
  • golang设置http response响应头与填坑记录

    2024-05-21 10:22:24
  • Java基础之JDBC的数据库连接与基本操作

    2024-01-13 12:35:04
  • sql server中datetime字段去除时间的语句

    2024-01-23 17:47:57
  • MySQL中Innodb的事务隔离级别和锁的关系的讲解教程

    2024-01-13 18:20:21
  • Python语言实现二分法查找

    2021-12-01 18:39:49
  • 基于python 的Pygame最小开发框架

    2022-01-23 12:22:40
  • 加载 Javascript 最佳实践

    2011-01-16 18:29:00
  • python 打印对象的所有属性值的方法

    2023-11-05 04:06:00
  • MySQL数据库中删除重复记录的方法总结[推荐]

    2024-01-13 07:13:45
  • 最简单的tab切换实例代码

    2023-08-22 08:38:59
  • Go语言快速入门图文教程

    2023-07-23 10:30:16
  • python 实现dcmtk关联pacs功能推送下拉影像(推荐)

    2021-10-15 15:58:43
  • Python OpenCV图像指定区域裁剪的实现

    2021-12-30 01:41:35
  • Python装饰器实现方法及应用场景详解

    2022-04-30 22:57:49
  • Vue中keep-alive的两种应用方式

    2024-05-09 15:10:53
  • python实现固定尺寸图像拼接

    2021-12-16 20:27:01
  • vue中插件和组件的区别点及用法总结

    2024-05-09 09:30:07
  • Python调用ChatGPT API接口的用法详解

    2022-10-11 23:29:25
  • python ElementTree 基本读操作示例

    2022-10-23 07:27:25
  • asp之家 网络编程 m.aspxhome.com