C++(Qt)-GIS开发-QGraphicsView显示瓦片地图简单示例

news2024/9/23 5:34:01

C++(Qt)-GIS开发-QGraphicsView显示瓦片地图简单示例

文章目录

  • C++(Qt)-GIS开发-QGraphicsView显示瓦片地图简单示例
    • 1、概述
    • 2、实现效果
    • 3、主要代码
    • 4、源码地址

更多精彩内容
👉个人内容分类汇总 👈
👉GIS开发 👈

1、概述

  1. 支持多线程加载显示本地离线瓦片地图(墨卡托投影);
  2. 瓦片切片规则以左上角为原点(谷歌、高德、ArcGis等),不支持百度瓦片规则;
  3. 支持显示瓦片网格、编号信息。
  4. 支持鼠标滚轮缩放切换地图层级。
  5. 支持鼠标拖拽。
  6. 采用z/x/y层级瓦片存储格式。
  7. 在单文件中实现所有主要功能,简单便于理解。
  8. 以QGraphicsView原点为起始位置,将加载的第一张瓦片显示在原点,其它瓦片相对于第一张瓦片进行显示【相对像素坐标】。

开发环境说明

  • 系统:Windows11、Ubuntu20.04
  • Qt版本:Qt 5.14.2
  • 编译器:MSVC2017-64、GCC/G++64

2、实现效果

使用瓦片地图工具下载z/x/y存储格式的瓦片地图进行显示。

在这里插入图片描述

3、主要代码

  • mapgraphicsview.h文件

    #ifndef MAPGRAPHICSVIEW_H
    #define MAPGRAPHICSVIEW_H
    
    #include <QGraphicsView>
    #include <QGraphicsScene>
    #include <QFuture>
    
    class MapGraphicsView : public QGraphicsView
    {
        Q_OBJECT
    public:
        explicit MapGraphicsView(QWidget *parent = nullptr);
        ~MapGraphicsView();
    
        void setPath(const QString& path);
        void quit();
    
    protected:
        void wheelEvent(QWheelEvent *event) override;
    
    signals:
        void addImage(QPixmap img, QPoint pos);
    private:
        void getMapLevel();     // 获取路径中瓦片地图的层级
        void getTitle();        // 获取路径中瓦片地图编号
        void loatImage();       // 加载瓦片
        void clearReset();       // 清除重置所有内容
        int getKey();          // 获取当前显示的层级key值
        void on_addImage(QPixmap img, QPoint pos);
    
    private:
        QGraphicsScene* m_scene = nullptr;
        QString m_path;          // 瓦片地图文件路径
        QHash<int, QGraphicsItemGroup*> m_mapItemGroups;     // 存放地图图元组的数组,以瓦片层级为key
        QGraphicsItemGroup* m_mapitemGroup = nullptr;        // 当前显示层级图元
        QHash<int, QGraphicsItemGroup*> m_gridItemGroups;    // 存放地图网格图元组的数组,以瓦片层级为key
        QGraphicsItemGroup* m_griditemGroup = nullptr;       // 当前显示层级网格图元
        int m_keyIndex = 0;               // 当前显示的瓦片层级
        QVector<QPoint> m_imgTitle;       // 保存图片编号
        QFuture<void> m_future;
    };
    
    #endif // MAPGRAPHICSVIEW_H
    
    
  • mapgraphicsview.cpp文件

    #include "mapgraphicsview.h"
    
    #include <QDir>
    #include <QDebug>
    #include <QGraphicsItemGroup>
    #include <QtConcurrent>
    #include <QWheelEvent>
    
    MapGraphicsView* g_this = nullptr;
    MapGraphicsView::MapGraphicsView(QWidget *parent) : QGraphicsView(parent)
    {
        m_scene = new QGraphicsScene(this);
        this->setScene(m_scene);
        g_this = this;
        connect(this, &MapGraphicsView::addImage, this, &MapGraphicsView::on_addImage);
        this->setDragMode(QGraphicsView::ScrollHandDrag);      // 设置鼠标拖拽
    //    QThreadPool::globalInstance()->setMaxThreadCount(1);   // 可以设置线程池线程数
    }
    
    MapGraphicsView::~MapGraphicsView()
    {
        g_this = nullptr;
        quit();   // 如果程序退出时还在调用map就会报错,所以需要关闭
    }
    
    
    /**
     * @brief 退出多线程
     */
    void MapGraphicsView::quit()
    {
        if(m_future.isRunning())   // 判断是否在运行
        {
            m_future.cancel();               // 取消多线程
            m_future.waitForFinished();      // 等待退出
        }
    }
    
    
    /**
     * @brief       设置加载显示的瓦片地图路径
     * @param path
     */
    void MapGraphicsView::setPath(const QString &path)
    {
        if(path.isEmpty()) return;
        m_path = path;
        getMapLevel();      // 获取瓦片层级
        loatImage();        // 加载第一层瓦片
    }
    
    /**
     * @brief        鼠标缩放地图
     * @param event
     */
    void MapGraphicsView::wheelEvent(QWheelEvent *event)
    {
        QGraphicsView::wheelEvent(event);
    
        if(m_future.isRunning())   // 判断是否在运行
        {
            return;
        }
        if(event->angleDelta().y() > 0)   // 放大
        {
            if(m_keyIndex < m_mapItemGroups.count() -1)
            {
                m_keyIndex++;
            }
        }
        else
        {
            if(m_keyIndex > 0)
            {
                m_keyIndex--;
            }
        }
        loatImage();        // 加载新的层级瓦片
    }
    
    /**
     * @brief 计算瓦片层级
     */
    void MapGraphicsView::getMapLevel()
    {
        if(m_path.isEmpty()) return;
    
        clearReset();    // 加载新瓦片路径时将之前的内容清空
    
        QDir dir(m_path);
        dir.setFilter(QDir::Dirs | QDir::NoDotAndDotDot);    // 设置过滤类型为文件夹,且不包含隐藏文件夹
        dir.setSorting(QDir::Name);                          // 设置按文件夹名称排序
        QStringList dirs = dir.entryList();
        for(auto& strDir : dirs)
        {
            bool ok;
            int level = strDir.toInt(&ok);
            if(ok)
            {
                if(!m_mapItemGroups.contains(level))  // 如果不包含
                {
                    // 初始化加载所有瓦片层级到场景中,默认不显示
                    QGraphicsItemGroup* itemMap = new QGraphicsItemGroup();
                    m_scene->addItem(itemMap);
                    itemMap->setVisible(false);
                    m_mapItemGroups[level] = itemMap;
                    // 初始化加载所有瓦片层级网格到场景中,默认不显示
                    QGraphicsItemGroup* itemGrid = new QGraphicsItemGroup();
                    m_scene->addItem(itemGrid);
                    itemGrid->setVisible(false);
                    m_gridItemGroups[level] = itemGrid;
                }
            }
        }
    }
    
    /**
     * @brief 获取当前显示层级中所有瓦片的编号
     */
    void MapGraphicsView::getTitle()
    {
        QString path = m_path + QString("/%1").arg(getKey());    // z  第一层文件夹
        QDir dir(path);
        dir.setFilter(QDir::Dirs | QDir::NoDotAndDotDot);    // 设置过滤类型为文件夹,且不包含隐藏文件夹
        dir.setSorting(QDir::Name);                          // 设置按文件夹名称排序
        QStringList dirs = dir.entryList();
        QPoint point;
        for(auto& strDir : dirs)
        {
            bool ok;
            int x = strDir.toInt(&ok);                         // x层级 第二层文件夹
            if(ok)
            {
                point.setX(x);
                dir.setPath(path + QString("/%1").arg(x));
                dir.setFilter(QDir::Files | QDir::NoDotAndDotDot);    // 设置过滤类型为文件,且不包含隐藏文件
                dir.setSorting(QDir::Name);                           // 设置按文件夹名称排序
                QStringList files = dir.entryList();
                for(auto& file: files)
                {
                    int y = file.split('.').at(0).toInt(&ok);   // 去除后缀,以文件名为y
                    if(ok)
                    {
                        point.setY(y);
                        m_imgTitle.append(point);
                    }
                }
            }
        }
    }
    
    QString g_path;   // 保存当前层级路径
    /**
     * @brief       在多线程中加载图片
     * @param point
     */
    void readImg(const QPoint& point)
    {
        QString path = QString("%1/%2/%3.jpg").arg(g_path).arg(point.x()).arg(point.y());
        QPixmap image;
        if(image.load(path))
        {
            if(g_this)
            {
                emit g_this->addImage(image, point);   // 由于不能在子线程中访问ui,所以这里通过信号将图片传递到ui线程进行绘制
            }
        }
    //    QThread::msleep(50);     // 加载时加上延时可以更加清晰的看到加载过程
    }
    
    /**
     * @brief 加载显示瓦片图元
     */
    void MapGraphicsView::loatImage()
    {
        quit();                  // 加载新瓦片之前判断是否还有线程在运行
        m_imgTitle.clear();
        if(m_mapitemGroup)
        {
            m_mapitemGroup->setVisible(false);        // 隐藏图层
            m_griditemGroup->setVisible(false);       // 隐藏图层
        }
        m_mapitemGroup = m_mapItemGroups.value(getKey());
        m_griditemGroup = m_gridItemGroups.value(getKey());
        if(!m_mapitemGroup || !m_griditemGroup) return;
        if(m_mapitemGroup->boundingRect().isEmpty())   // 如果图元为空则加载图元显示
        {
            getTitle();      // 获取新层级的所有瓦片编号
            g_path = m_path + QString("/%1").arg(getKey());
            m_future = QtConcurrent::map(m_imgTitle, readImg);
        }
        m_mapitemGroup->setVisible(true);              // 显示新瓦片图层
        m_griditemGroup->setVisible(true);             // 显示新网格图层
        m_scene->setSceneRect(m_mapitemGroup->boundingRect());   // 根据图元大小自适应调整场景大小
    }
    
    /**
     * @brief 清除重置所有内容
     */
    void MapGraphicsView::clearReset()
    {
        if(m_mapItemGroups.isEmpty()) return;
        m_keyIndex = 0;
        m_mapitemGroup = nullptr;
        m_griditemGroup = nullptr;
        m_imgTitle.clear();
        QList<int>keys = m_mapItemGroups.keys();
        for(auto key : keys)
        {
            // 清除瓦片图元
            QGraphicsItemGroup* item = m_mapItemGroups.value(key);
            m_scene->removeItem(item);    // 从场景中移除图元
            delete item;
            m_mapItemGroups.remove(key);   // 从哈希表中移除图元
    
            // 清除网格
            item = m_gridItemGroups.value(key);
            m_scene->removeItem(item);     // 从场景中移除图元
            delete item;
            m_gridItemGroups.remove(key);   // 从哈希表中移除图元
        }
    }
    
    /**
     * @brief   获取当前层级的key值
     * @return  返回-1表示不存在
     */
    int MapGraphicsView::getKey()
    {
        if(m_mapItemGroups.isEmpty()) return -1;
    
        QList<int>keys = m_mapItemGroups.keys();
        std::sort(keys.begin(), keys.end());    // 由于keys不是升序的,所以需要进行排序
        if(m_keyIndex < 0 || m_keyIndex >= keys.count())
        {
            return -1;
        }
        return keys.at(m_keyIndex);
    }
    
    /**
     * @brief       绘制地图瓦片图元
     * @param img   显示的图片
     * @param pos   图片显示的位置
     */
    void MapGraphicsView::on_addImage(QPixmap img, QPoint pos)
    {
        if(!m_mapitemGroup || m_imgTitle.isEmpty())
        {
            return;
        }
    
        // 计算瓦片显示位置,默认为256*256的瓦片大小
        QPoint& begin = m_imgTitle.first();
        int x = (pos.x() - begin.x()) * 256;
        int y = (pos.y() - begin.y()) * 256;
        // 绘制瓦片
        QGraphicsPixmapItem* itemImg = new QGraphicsPixmapItem(img);
        itemImg->setPos(x, y);   // 以第一张瓦片为原点
        m_mapitemGroup->addToGroup(itemImg);
    
        // 绘制网格、
        QGraphicsRectItem* itemRect = new QGraphicsRectItem(x, y, 256, 256);
        m_griditemGroup->addToGroup(itemRect);
        itemRect->setPen(QPen(Qt::red));
        // 绘制编号
        QString text = QString("%1,%2,%3").arg(pos.x()).arg(pos.y()).arg(getKey());
        QGraphicsSimpleTextItem* itemText = new QGraphicsSimpleTextItem(text);
        QFont font;
        font.setPointSize(14);                           // 设置字体大小为12
        QFontMetrics metrics(font);
        qreal w = metrics.horizontalAdvance(text) / 2.0; // 计算字符串宽度
        qreal h = metrics.height() / 2.0;               // 字符串高度
        itemText->setPos(x + 128 - w, y + 128 - h);     // 编号居中显示
        itemText->setFont(font);
        itemText->setPen(QPen(Qt::red));
        m_griditemGroup->addToGroup(itemText);
    
        m_scene->setSceneRect(m_mapitemGroup->boundingRect());   // 根据图元大小自适应调整场景大小
    }
    
    

4、源码地址

  • github
  • gitee

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

2025湖北武汉智慧教育装备信息化展/智慧校园展/湖北高博会

2025武汉教育装备展,2025武汉智慧教育展,2025武汉智慧校园展,2025武汉教育信息化展,2025武汉智慧教室展,湖北智慧校园展,湖北智慧教室展,武汉教学设备展,湖北高教会,湖北高博会 2025湖北武汉智慧教育装备信息化展/智慧校园展/湖北高博会 2025第10届武汉国际教育装备及智慧校园…

uni-app组件 子组件onLoad、onReady事件无效

文章目录 导文解决方法 导文 突然发现在项目中&#xff0c;组件 子组件的onLoad、onReady事件无效 打印也出不来值 怎么处理呢&#xff1f; 解决方法 mounted() {console.log(onLoad, this.dateList);//有效// this.checkinDetails()},onReady() {console.log(onReady, this.da…

connect to github中personal access token生成token方法

一、问题 执行git push时弹出以下提示框 二、解决方法 去github官网生成Token&#xff0c;步骤如下 选择要授予此 令牌token 的 范围 或 权限 要使用 token 从命令行访问仓库&#xff0c;请选择 repo 。 要使用 token 从命令行删除仓库&#xff0c;请选择 delete_repo 其他根…

第9章 项目总结01:项目流程,每个模块的介绍

1 请介绍一下你的项目 学成在线项目是一个B2B2C的在线教育平台&#xff0c;本项目包括了用户端、机构端、运营端。 核心模块包括&#xff1a;内容管理、媒资管理、课程搜索、订单支付、选课管理、认证授权等。 下图是项目的功能模块图&#xff1a; 项目采用前后端分离的技…

使用Python绘制堆积柱形图

使用Python绘制堆积柱形图 堆积柱形图效果代码 堆积柱形图 堆积柱形图&#xff08;Stacked Bar Chart&#xff09;是一种数据可视化图表&#xff0c;用于显示不同类别的数值在某一变量上的累积情况。每一个柱状条显示多个子类别的数值&#xff0c;子类别的数值在柱状条上堆积在…

中金女员工离世悲剧:职场压力、心理健康与社会支持的深刻反思

中金女员工离世背后的深思 2024年7月1日,一则令人痛心的消息在金融界乃至整个网络迅速传播:中金公司一位年仅30岁的女员工郑某露,在降薪和房贷的双重压力下,不幸离世。这一事件不仅让她的家人和朋友陷入了深深的悲痛之中,也引发了社会各界对职场环境、个体心理健康以及社…

Android 集成OpenCV

记录自己在学习使用OpenCV的过程 我使用的是4.10.0 版本 Android 集成OpenCV 步骤 下载OpenCV新建工程依赖OpenCV初始化及逻辑处理 1、下载OpenCV 并解压到自己的电脑 官网 地址&#xff1a;https://opencv.org/releases/ 个人地址&#xff1a;https://pan.baidu.com/s/19f…

工作手机怎么做好业务员工作微信的监控管理

什么是工作手机管理系统&#xff1f; 工作手机管理系统是专为企业管理设计的员工微信管理&#xff0c;它通过监控通讯记录、保障数据安全、自动检测敏感行为、永久保留客户信息等功能&#xff0c;帮助企业提升销售效率、维护客户资源安全&#xff0c;并确保业务流程的合规性。…

51单片机嵌入式开发:3、STC89C52操作8八段式数码管原理

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 STC89C52操作8八段式数码管原理 1 8位数码管介绍1.1 8位数码管概述1.2 8位数码管原理1.3 应用场景 2 原理图图解2.1 74HC573原理2.2 74HC138原理2.3 数码管原理 3 数码管程序…

Linux系统的服务——以Centos7为例

一、Linux系统的服务简介 服务是向外部提供对应功能的进程&#xff0c;其运行在系统后台&#xff0c;能够7*24小时持续不断的提供外界随时发来的服务请求&#xff0c;且服务进程常驻在内存中&#xff0c;具有固定的端口号&#xff0c;通过端口号就能找到服务内容。 提供服务的一…

Linux 系统管理4——账号管理

一、用户账号管理 1、用户账号概述 &#xff08;1&#xff09;用户账号的常见分类&#xff1a; 1>超级用户&#xff1a;root uid0 gid0 权限最大。 2>普通用户&#xff1a;uid>500 做一般权限的系统管理&#xff0c;权限有限。 3>程序用户&#xff1a;1<uid&l…

OpenCV(绘图功能笔记)

目标 学习使用OpenCV绘制不同的几何形状 cv.line()&#xff0c;cv.circle()&#xff0c;cv.rectangle()&#xff0c;cv.ellipse()&#xff0c;cv.putText()等。 画线&#xff08;cv.line&#xff09; 要绘制一条线&#xff0c;需要传递线的开始和结束坐标。我们将创建一个黑…

【Oracle】Oracle常用函数

目录 聚合函数数字函数1. ABS函数&#xff1a;返回一个数的绝对值。2. CEIL函数&#xff1a;返回大于等于给定数的最小整数。3. FLOOR函数&#xff1a;返回小于等于给定数的最大整数。4. ROUND函数&#xff1a;将一个数四舍五入到指定的小数位。5. MOD函数&#xff1a;返回两个…

swin-unet编码端流程图

文章目录 1. PatchEmbed2. swinTransformerBlock2.1. window_partition2.2. WindowAttention2.3. Window_reverse2.4. MLP 3. PatchMerging完整流程图 1. PatchEmbed 2. swinTransformerBlock 2.1. window_partition 2.2. WindowAttention 2.3. Window_reverse 2.4. MLP 3. Pat…

集成学习(三)GBDT 梯度提升树

前面学习了&#xff1a;集成学习&#xff08;二&#xff09;Boosting-CSDN博客 梯度提升树&#xff1a;GBDT-Gradient Boosting Decision Tree 一、介绍 作为当代众多经典算法的基础&#xff0c;GBDT的求解过程可谓十分精妙&#xff0c;它不仅开创性地舍弃了使用原始标签进行…

模型训练之数据集

我们知道人工智能的四大要素&#xff1a;数据、算法、算力、场景。我们训练模型离不开数据 目标 一、数据集划分 定义 数据集&#xff1a;训练集是一组训练数据。 样本&#xff1a;一组数据中一个数据 特征&#xff1a;反映样本在某方面的表现、属性或性质事项 训练集&#…

昇思25天学习打卡营第15天|linchenfengxue

Pix2Pix实现图像转换 Pix2Pix概述 Pix2Pix是基于条件生成对抗网络&#xff08;cGAN, Condition Generative Adversarial Networks &#xff09;实现的一种深度学习图像转换模型&#xff0c;该模型是由Phillip Isola等作者在2017年CVPR上提出的&#xff0c;可以实现语义/标签到…

农资销售网站-计算机毕业设计源码54432

目录 摘要 Abstract 1绪论 1.1研究背景 1.2研究意义 1.3论文结构与章节安排 2农资销售网站系统分析 2.1可行性分析 2.1.1技术可行性分析 2.1.2经济可行性分析 2.1.3法律可行性分析 2.2系统功能分析 2.2.1功能性分析 2.2.2非功能性分析 2.3系统用例分析 2.4系统流…

业界数据架构的演变

目录 一、概述 二、业务处理-单体架构 三、业务处理-微服务架构 四、数据分析-大数据Lambda架构 五、数据分析-Kappa架构 六、数据分析-LambdaKappa混合架构 七、湖仓一体架构 一、概述 近年来随着越来越多的大数据技术被开源&#xff0c;例如&#xff1a;HDFS、Spark等…

数据库缓存管理

1. 简介 缓存管理器是数据库管理系统&#xff08;DBMS&#xff09;中负责管理内存中page并处理文件和索引管理器的page请求的组件。由于内存空间有限&#xff0c;我们不能将所有page存储在缓存池中。因此&#xff0c;缓存管理器需要制定替换策略&#xff0c;当空间填满时选择哪…