一个简单的可视化的A星自动寻路

一个简单的应用场景,流程图连线

源码:

    addExample("A星路径查找", function () {
            return {
                template: `<div><div ref="main"></div></div>`,
                data() { return {}; },
                computed: {},
                methods: {},
                mounted() {
                    var container = this.$refs.main;
                    var render = new CanvasShapeRender(container, {
                        width: 700,
                        height: 700,
                        background: '#efefef'
                    })
                    var group = render.addShape({ type: 'group', x: 50, y: 50 })
                    const start = Vector.create(2, 2)
                    const end = Vector.create(8, 6)
                    let dragObj

                    const tileMap = group.addShape({
                        type: 'tileMap',
                        visibleGrid: true,
                        onCreateTileData(col, row) {
                            if (row === start.y && col === start.x) {

                                return {
                                    type: 'rect',
                                    color: '#ff0000',
                                    value: 1,
                                    canMove: true
                                }
                            }
                            else if (row === end.y && col === end.x) {

                                return {
                                    type: 'rect',
                                    color: '#00ff00',
                                    value: 2,
                                    canMove: true
                                }
                            } else {
                                return {
                                    type: 'rect',
                                    color: '#ddd',
                                    value: 0
                                }
                            }

                        },
                        onAfterDrawMapCell(ctx, col, row, x, y, data) {
                            if (!gui.visibleDist || data.value !== 4) {
                                return
                            }
                            ctx.beginPath()
                            ctx.fillStyle = '#000'
                            ctx.font = '12px sans-serif'
                            ctx.textAlign = 'start'
                            ctx.textBaseline = 'base'
                            const getDist = distOps[gui.dist]

                            const startDist = Number(data.startDist.toFixed(2)) // Number(getDist({ col: start.x, row: start.y }, { col, row }).toFixed(2))
                            const endDist = Number(data.endDist.toFixed(2))// Number(getDist({ col, row }, { col: end.x, row: end.y }).toFixed(2))
                            const dist = Number(data.dist.toFixed(2))//Number((endDist + startDist).toFixed(2))

                            ctx.fillText('' + startDist, x, y + 10)
                            ctx.fillText('' + endDist, x, y + this.cellSize[1] - 5)
                            ctx.beginPath()
                            ctx.font = '14px sans-serif'
                            ctx.textAlign = 'center'
                            ctx.textBaseline = 'middle'
                            //  CanvasRenderingContext2D.prototype.textBaseline
                            ctx.fillText('' + dist, x + this.cellSize[0] / 2, y + this.cellSize[1] / 2)
                        },
                        cellSize: [50, 50],
                        mapSize: [10, 10],
                        mousedown(e) {
                            const downPoint = e.downPoint
                            const [x, y] = group.transformLocalCoord(downPoint.x, downPoint.y)
                            const [col, row] = this.getMapCoordinate(x, y)
                            const data = this.getCellData(col, row)
                            if (data && data.canMove) {
                                dragObj = {
                                    col,
                                    row,
                                    data
                                }
                            } else if (data && (data.value == 0 || data.value == 3)) {
                                dragObj = {
                                    col,
                                    row,
                                    data: {
                                        value: data.value
                                    }
                                }
                            }

                        },
                        drag(e) {
                            if (dragObj) {
                                const point = e.point
                                const [x, y] = group.transformLocalCoord(point.x, point.y)
                                let [col, row] = this.getMapCoordinate(x, y)
                                const data = this.getCellData(col, row)
                                if (dragObj.data.value === 0 && !data.canMove) {
                                    // 变成障碍
                                    data.value = 3
                                    data.color = '#666'
                                    this.setCellData(col, row, data)
                                    render.requestDraw()
                                } else if (dragObj.data.value === 3 && !data.canMove) {
                                    // 移除障碍
                                    data.value = 0
                                    data.color = '#ddd'
                                    this.setCellData(col, row, data)
                                    render.requestDraw()
                                }
                                else if (dragObj.data.canMove && data && data !== dragObj.data) {
                                    this.setCellData(dragObj.col, dragObj.row, data)
                                    this.setCellData(col, row, dragObj.data)
                                    if (dragObj.data.value === 1) {
                                        start.x = col
                                        start.y = row
                                    }
                                    if (dragObj.data.value === 2) {
                                        end.x = col
                                        end.y = row
                                    }
                                    dragObj.col = col
                                    dragObj.row = row
                                    render.requestDraw()
                                }
                            }
                        },
                        mouseup() {
                            dragObj = null
                        }
                    })
                    render.requestDraw()
                    const map = tileMap.map // 0 空 1 起点 2终点 3障碍

                    const clear = () => {
                        tileMap.visitMap(tileMap.map, (r, c, data) => {
                            if (data.value === 4) {
                                data.value = 0
                                data.color = '#ddd'
                            }
                        })
                        render.requestDraw()
                    }
                    const renderPaths = (paths, color, renderPath, duration = 1000) => {
                        if (paths.length <= 0) {
                            return
                        }

                        let start = performance.now()
                        let len = paths.length - 1
                        const animate = (time) => {
                            const d = performance.now() - start
                            const p = Math.min(d / duration, 1)
                            const index = Math.floor(p * len);
                            const data = paths[index]
                            if (renderPath || !data.isPath) {
                                tileMap.setCellData(data.col, data.row, {
                                    ...data,
                                    type: 'rect',
                                    value: 4,
                                    color: color
                                })
                            }
                            render.requestDraw()
                            if (p < 1) {
                                requestAnimationFrame(animate)
                            }
                        }
                        requestAnimationFrame(animate)
                    }
                    // 计算两个点的距离
                    //Manhattan Distance
                    const getManhattanDist = (a, b) => {
                        // 曼哈顿距离 
                        return Math.abs(a.col - b.col) + Math.abs(a.row - b.row)
                    }
                    // 欧几里得距离( Euclidean distance)也称欧氏距离
                    const getEuclideanDist = (a, b) => {
                        const x = a.col - b.col
                        const y = a.row - b.row
                        return Math.sqrt(x * x + y * y)
                    }
                    //,切比雪夫距离(Chebyshev distance)
                    const getChebyshevDist = (a, b) => {
                        const x = a.col - b.col
                        const y = a.row - b.row
                        return Math.max(Math.abs(x), Math.abs(y))
                    }
                    const distOps = {
                        getManhattanDist,
                        getEuclideanDist,
                        getChebyshevDist
                    }
                    // 返回最短路径
                    const findPath = function* (start, end, _map) {
                        // 创建图顶点信息
                        const map = _map.map((rd, row) => {
                            return rd.map((cd, col) => {
                                return {
                                    ...cd,
                                    row,
                                    col,
                                    value: cd.value,
                                    isPath: false,
                                    visited: false,// 是否访问过
                                    // 无有可走的路
                                    closed: false, // 已经查找过
                                    parent: null,
                                    startDist: 0, // 起点距离当前格子
                                    endDist: 0, // 当前距离终点
                                    dist: 0, // 总距离
                                    weight: 0, // 权重
                                    order: 0,
                                }
                            })
                        })
                        const rows = map.length, cols = map[0].length
                        const getNode = (col, row) => {
                            if (col < 0 || col >= cols || row < 0 || row >= rows) {
                                return null
                            }
                            return map[row][col]
                        }
                        // 找相邻的
                        const getAdjacent = (node) => {
                            const c = node.col
                            const r = node.row
                            const left = getNode(c - 1, r) // left
                            const top = getNode(c, r - 1) // top
                            const right = getNode(c + 1, r) // right
                            const bottom = getNode(c, r + 1) // bottom
                            return [left, top, right, bottom].filter(Boolean)
                        }


                        const getCost=()=>{
                            return 0.1
                        }
                        const findNearestDistance = (node) => {
                            const adjacent = getAdjacent(node)
                            let min = Infinity, minNode
                            // let resultNode;
                            adj:
                            for (let i = 0; i < adjacent.length; i++) {
                                const adj = adjacent[i]
                                // 如果已关闭或是障碍,不处理
                                if (adj.closed || adj.value === 3) {
                                    continue;
                                }
                                let startDist=node.startDist+getCost(adj,node) //getDist(adj,node)
                                // 如果还未访问
                                if (!adj.visited) {
                                    // g(n)表示从初始结点到任意结点n的代价,
                                    // h(n)表示从结点n到目标点的启发式评估代价(heuristic estimated cost)。
                                    // f=g(n)+h(n)
                                    // getDist(startNode, adj)
                                    adj.startDist = startDist
                                    adj.endDist = getDist(adj, endNode)
                                    adj.dist = adj.startDist + adj.endDist //getDist(adj, startNode)
                                    adj.parent = node
                                    adj.visited = true
                                    openList.push(adj)
                                    // 如果是空闲
                                    if (adj.value === 0) {
                                        visitedPaths.push(adj)
                                    }
                                }else{
                                    if(startDist<node.startDist){
                                        adj.parent = node
                                        adj.startDist = startDist
                                        adj.dist = adj.startDist + adj.endDist //getDist(adj, startNode)
                                    }
                                }
                                if (adj.value === 2) {
                                    return adj
                                }
                            }
                            openList.sort((a, b) => a.dist - b.dist)
                            // openList.sort((a, b) => a.dist === b.dist ? a.dist - b.dist : a.order - b.order)
                        }
                        const getDist = distOps[gui.dist]
                        // 查找邻居四个方位,上下左右

                        let current = null
                        let startNode = getNode(start[0], start[1])
                        let endNode = getNode(end[0], end[1])




                        let paths = []
                        let visitedPaths = []


                        let openList = []
                        openList.push(startNode)
                        let resultNode;
                        path:
                        while (openList.length) {
                            yield { visitedPaths, paths };
                            current = openList.shift()
                            current.closed = true;
                            if (current === endNode) {
                                resultNode = current
                                break
                            }
                            resultNode = findNearestDistance(current)
                            if (resultNode) {
                                break
                            }
                        }
                        current = resultNode ? resultNode.parent : null
                        while (current && current !== startNode) {
                            current.isPath = true;
                            paths.unshift(current)
                            current = current.parent
                        }

                        return {
                            paths: paths,
                            visitedPaths
                        }

                    }
                    const resultGenerator = (result) => {
                        let current;
                        do {
                            current = result.next();
                        } while (!current.done)
                        return current.value
                    }

                    const stepGenerator = (generatorFn, callback) => {


                        let result;
                        let isStart = false
                        const next = () => {
                            if (!isStart) {
                                isStart = true;
                                result = generatorFn()
                            }
                            let current = result.next();
                            callback(current.value)
                            if (current.done) {
                                isStart = false
                            }
                        }
                        return {
                            next,
                        }
                    }
                    const distList = Object.keys(distOps)
                    const step = stepGenerator(function* () {
                        return yield* findPath([start.x, start.y], [end.x, end.y], map)
                    }, ({ visitedPaths, paths }) => {
                        renderPaths(visitedPaths, '#aaa', false)
                        renderPaths(paths, '#ffff00', true)
                    })
                    const gui = addGuiScheme(this.$gui, {
                        source: {
                            dist: 'getEuclideanDist',
                            visibleDist: false,
                            start: () => {
                                const { paths, visitedPaths } = resultGenerator(findPath([start.x, start.y], [end.x, end.y], map))
                                renderPaths(visitedPaths, '#aaa', false)
                                renderPaths(paths, '#ffff00', true)
                            },
                            step: () => {
                                step.next()
                            },
                            clear() {
                                clear()
                            }
                        },
                        schemes: {
                            dist: { type: 'list', params: distList }
                        },
                        onChange() {
                            render.requestDraw()
                        }
                    })

                }
            }
        })

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/233473.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2-3、运算符

语雀原文链接 文章目录 1、算术运算符2、关系运算符3、逻辑运算符4、赋值运算符5、移位运算符6、位运算符(二进制位进行运算)7、条件运算符:三目运算符8、运算符的优先级 1、算术运算符 &#xff1a;加法-&#xff1a;减法*&#xff1a;乘法/&#xff1a;除法取商%&#xff1…

logback日志框架使用

依赖引入 <dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version>1.1.7</version> </dependency> 使用logback日志框架只需要引入以上即可&#xff0c;(我们平时使用较多的Slf4j…

Fall in love with English

Fall in love with English 爱上英语 Hiding behind the loose dusty curtain, a teenager packed up his overcoat into the suitcase. 躲藏在布满尘土的松软的窗帘后边&#xff0c;一个年轻人打包他的外套到行李箱中。 He planned to leave home at dusk though there was th…

ssh免密登录及scp/rsync免密传输文件的方式

在通过ssh登录其它电脑或通过scp/rsync同其它电脑之间传输文件时&#xff0c;每次都需要输入密码&#xff0c;如下图所示&#xff1a;在windows10上通过ssh登录虚拟机&#xff0c;每次登录都需要输入密码&#xff1b;若端口默认为22,可省略通过-p指定 可通过将本机上的公钥key存…

熔池处理Tecplot 360 和CFD-Post做出一样的效果

熔池处理Tecplot 360 和CFD-Post做出一样的效果 效果展示详细讲述Tecplot 360实现过程分析实现过程第一步实现过程第二步界面美化注意点效果展示 详细讲述Tecplot 360实现过程 分析 这里主要是将体积分数大于0.5的区域抽取出来,然后显示温度场,所以这里主要考虑下面连个思考…

Agent相关工作调研

API Bank 要解决两个问题&#xff1a; 1)目前的LLM在使用工具方面的效果如何&#xff1f; 2) LLM使用工具还存在哪些障碍&#xff1f; 理想的效果&#xff1a;&#xff1a;通过访问全球工具存储库&#xff0c;LLM可以通过概述实现需求所需的所有步骤来帮助人们规划需求。随后…

你知道Java中的BigInteger类和BigDecimal类吗?

BigInteger和BigDecimal&#xff1a; 我们在学习JavaSE基础的时候学习过int和double&#xff0c;前者是整形&#xff0c;后者是双精度浮点数&#xff0c;但它们是有最大值的&#xff0c;也就是说&#xff0c;他两并不支持无限大的数字。 其范围如下所示&#xff1a; 因此对于…

图论-并查集

并查集(Union-find Sets)是一种非常精巧而实用的数据结构,它主要用于处理一些不相交集合的合并问题.一些常见的用途有求连通子图,求最小生成树Kruskal算法和最近公共祖先(LCA)等. 并查集的基本操作主要有: .1.初始化 2.查询find 3.合并union 一般我们都会采用路径压缩 这样…

Java期末复习题之分支循环

点击返回标题->23年Java期末复习-CSDN博客 第1题. 编写一个模拟同时掷骰子的程序。要用Math.random()模拟产生两个骰子&#xff0c;将两个结果相加&#xff0c;相加的和等于7的可能性最大&#xff0c;等于2和12的可能性最小。程序模投掷3600次&#xff0c;判断求和的结果是否…

在IDEA中创建Maven项目时没有src文件、不自动配置文件

错误示例&#xff1a; 没有src文件&#xff0c;并且没有自动下载相关的配置文件 对我这中情况无效的解决办法&#xff1a; ①配置好下列图中圈出来的文件 ②在VM选项中输入&#xff1a;“-DarchetypeInternal” ③点击应用&#xff0c;再点击确定 ④还是不行 解决办法&#x…

20 套监控平台统一成 1 套 Flashcat,国泰君安监控选型提效之路

author:宋庆羽-国泰君安期货 运维工作最重要的就是维护系统的稳定性&#xff0c;其中监控是保证系统稳定性很重要的一环。通过监控可以了解系统的运行状态&#xff0c;及时发现问题和系统隐患&#xff0c;有助于一线人员快速解决问题&#xff0c;提高业务系统的可用时长。 作为…

class064 Dijkstra算法、分层图最短路【算法】

class064 Dijkstra算法、分层图最短路【算法】 算法讲解064【必备】Dijkstra算法、分层图最短路 code1 743. 网络延迟时间 // Dijkstra算法模版&#xff08;Leetcode&#xff09; // 网络延迟时间 // 有 n 个网络节点&#xff0c;标记为 1 到 n // 给你一个列表 times&…

机器学习基本概念介绍 2023

笔记来源于&#xff1a; https://www.youtube.com/watch?vphQK8xZpgoU&t172s https://www.youtube.com/watch?vXLyPFnephpY&t645s Machine/Deep Learning 机器学习概况来说&#xff0c;让机器具备自动找函式的能力 &#xff08;Machine Learning 约等于 Looking …

MongoDB的条件操作符

本文主要介绍MongoDB的条件操作符。 目录 MongoDB条件操作符1.比较操作符2.逻辑操作符3.元素操作符4.数组操作符5.文本搜索操作符 MongoDB条件操作符 MongoDB的条件操作符主要分为比较操作符、逻辑操作符、元素操作符、数组操作符、文本搜索操作符等几种类型。 以下是这些操作…

unity Mesh Simplify 1.10(模型优化工具:查看面数,降低面数灯)

提示&#xff1a;文章有错误的地方&#xff0c;还望诸位大神不吝指教&#xff01; 文章目录 前言一、面板参数详解说明二、使用方法总结 前言 有时候想对模型优化一下&#xff0c;奈何又不会建模方面的。虽然我感觉它的数值不大对&#xff0c;但是不影响我们优化顶点数嘛。 Me…

python 画条形图(柱状图)

目录 前言 基础介绍 月度开支的条形图 前言 条形图&#xff08;bar chart&#xff09;&#xff0c;也称为柱状图&#xff0c;是一种以长方形的长度为变量的统计图表&#xff0c;长方形的长度与它所对应的变量数值呈一定比例。 当使用 Python 画条形图时&#xff0c;通常会使…

0基础学java-day14-(集合)

一、集合 前面我们保存多个数据使用的是数组&#xff0c;那么数组有不足的地方&#xff0c;我们分析一下 1.数组 2 集合 数据类型也可以不一样 3.集合的框架体系 Java 的集合类很多&#xff0c;主要分为两大类&#xff0c;如图 &#xff1a;[背下来] package com.hspedu.c…

如何确认网站是否有漏洞,如何找出网站存在的漏洞,找到漏洞该如何处理

如何确认网站或者服务器是否有漏洞 判断一个网站是否是存在漏洞的方法&#xff1a; 1.可以借助德迅云安全漏洞扫描功能来检查漏洞。 2.打开德迅云安全首页&#xff0c;点击最上面导航栏中的“安全产品”。 3.滑到“漏洞扫描”&#xff0c;选择“产品价格”服务。 4.选择您需…

CleanMyMac2024破解版激活码许可证密钥

CleanMyMac X是一款颇受欢迎的专业清理软件&#xff0c;拥有十多项强大的功能&#xff0c;可以进行系统清理、清空废纸篓、清除大旧型文件、程序卸载、除恶意软件、系统维护等等&#xff0c;并且这款清理软件操作简易&#xff0c;非常好上手&#xff0c;特别适用于那些刚入手苹…

51单片机的内核架构组成 介绍

对于51单片机相信很多电子信息或者相关专业的朋友应该都不会感觉陌生&#xff0c;很多专业在大学课程中开设的单片机课程就是使用的51单片机进行授课和学习的。51单片机的内容相较于其他高性能复杂的单片机来说&#xff0c;架构相对简单一些&#xff0c;寄存器也少很多&#xf…