使用 Qt 实现自定义罗盘控件

用 Qt 编写一个简单的罗盘控件,该控件能够动态显示方向。该控件实现了一个带有北(N)和南(S)标记的圆形罗盘面盘,具有可以根据输入角度旋转的指针。
在这里插入图片描述

代码功能概述

该项目定义了一个 CompassWidget 类,继承自 QWidget,实现了如下功能:

  1. 绘制罗盘的背景和刻度线。
  2. 显示北(N)和南(S)方向标记。
  3. 根据输入角度旋转指针,显示指定方向。

在这里插入图片描述

核心代码结构

CompassWidget.h

首先,CompassWidget.h 定义了罗盘控件的接口,包括一个设置角度的方法和一个用于自定义绘制的 paintEvent 函数。

#ifndef COMPASSWIDGET_H
#define COMPASSWIDGET_H#include <QWidget>class CompassWidget : public QWidget
{Q_OBJECTpublic:explicit CompassWidget(QWidget *parent = nullptr);void setAngle(double newAngle); // 设置指针角度protected:void paintEvent(QPaintEvent *event) override; // 自定义绘制private:double angle; // 以度为单位的角度,正北为0度,顺时针增加
};#endif // COMPASSWIDGET_H

代码实现详解

初始化和设置角度

CompassWidget.cpp 中,我们定义了构造函数和设置角度的方法。构造函数设置了默认角度 0(指向正北),并设置控件的最小尺寸为 200x200setAngle 方法用于更新角度,并通过 update() 触发重绘。

#include "CompassWidget.h"
#include <QPainter>#ifndef M_PI
#define M_PI 3.14159265358979323846
#endifCompassWidget::CompassWidget(QWidget *parent) : QWidget(parent), angle(0)
{setMinimumSize(200, 200);
}void CompassWidget::setAngle(double newAngle)
{angle = newAngle;update(); // 重新绘制小部件
}
自定义绘制逻辑:paintEvent

paintEvent 是 Qt 中用于自定义绘制的事件处理函数。该函数中包含绘制罗盘的所有细节,包括背景、刻度线、方向标记和指针。

void CompassWidget::paintEvent(QPaintEvent *)
{QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing); // 开启抗锯齿int side = qMin(width(), height());// 保持图形居中painter.setViewport((width() - side) / 2, (height() - side) / 2, side, side);painter.setWindow(-100, -100, 200, 200);
  1. 设置绘制区域:确保罗盘始终居中显示,使用 setViewportsetWindow 设置绘图区域。

  2. 绘制背景

    painter.setBrush(Qt::blue);
    painter.setPen(Qt::NoPen);
    painter.drawEllipse(-100, -100, 200, 200);
    

    使用 drawEllipse 绘制一个蓝色圆形背景作为罗盘盘面。

  3. 绘制刻度线

    painter.setPen(QPen(Qt::white, 1));
    for (int i = 0; i < 360; i += 6) {if (i % 30 == 0) {painter.drawLine(90, 0, 100, 0);} else {painter.drawLine(95, 0, 100, 0);}painter.rotate(6);
    }
    

    每隔 6 度绘制一个刻度线,每隔 30 度绘制一个较长的刻度,表示主要方向。rotate(6) 用于在绘制每条刻度后旋转坐标系。

  4. 绘制“N”和“S”标记

    QFont font = painter.font();
    font.setBold(true);
    font.setPointSize(10);
    painter.setFont(font);
    painter.setPen(Qt::white);
    painter.drawText(-5, -75, "N");
    painter.drawText(-5, 85, "S");
    

    在罗盘上部和下部分别绘制 “N” 和 “S” 以指示北和南的方向。

  5. 绘制罗盘针

    painter.translate(0, 0);
    painter.rotate(angle); // 根据角度旋转画布painter.setBrush(Qt::black);
    painter.drawEllipse(-8, -8, 16, 16); // 中心小圆// 白色南针
    painter.setPen(Qt::NoPen);
    painter.setBrush(Qt::white);
    QPointF southNeedle[3] = {QPointF(-8, 10),QPointF(0, 80),QPointF(8, 10)
    };
    painter.drawConvexPolygon(southNeedle, 3);// 红色北针
    painter.setBrush(Qt::red);
    QPointF northNeedle[3] = {QPointF(-8, -10),QPointF(0, -80),QPointF(8, -10)
    };
    painter.drawConvexPolygon(northNeedle, 3);
    
    • 旋转画布:使用 rotate(angle) 方法根据输入的角度旋转画布。
    • 绘制指针:指针分为白色的南针部分和红色的北针部分,使用三角形绘制两个箭头形状的针头。
  6. 恢复坐标系

    painter.resetTransform();
    

    最后,重置坐标系,以确保下一次重绘时从初始状态开始。

总结

通过这段代码,我们成功创建了一个自定义罗盘控件,具备动态旋转指针的功能。此控件非常适合用于导航应用或其他需要展示方向的场景。Qt 的 QPainterQWidget 的灵活性,使得我们可以轻松绘制出各种形状和动态效果,从而创建出更复杂的图形控件。

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

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

相关文章

项目模块十四:HttpRequest模块

一、项目设计思路 存储HTTP请求要素&#xff0c;提供简单接口 二、成员变量 全部公有 string _method; // 请求方法 string _path; // 资源路径 string _version; // 协议版本 string _body; // 请求正文 smatch _matches; // 资源路径正则提取 …

NASA:全球无机气溶胶酸度的机载观测和模拟比较

目录 简介 摘要 代码 引用 网址推荐 0代码在线构建地图应用 机器学习 Airborne Observations and Modeling Comparison of Global Inorganic Aerosol Acidity 全球无机气溶胶酸度的机载观测和模拟比较 简介 该数据集提供了在2006年至2017年期间收集的十一项空中观测活…

汽车零部件展|2025 第十二届广州国际汽车零部件加工技术及汽车模具展览会邀您共赏汽车行业盛会

汽车是我国国民经济的重要支柱行业&#xff0c;在我国国民经济中起到举足轻重的作用&#xff0c;也是现代高端制造业的代表。改革开放以来&#xff0c;我国汽车产销量保持增长态势&#xff0c;至 2017 年实现汽车销量 2,887.89 万辆。受到国际经济形势、居民需求下滑、国民经济…

JavaEE-多线程初阶(1)

目录 1. 线程的概念 1.1 线程是是什么 1.2 为什么要有线程 1.3 线程和进程的区别 1.4 Java的线程和操作系统线程的关系 2. 第一个多线程程序 2.1 Hello Thread 2.2 使用jconsole观察线程 3. 创建线程 3.1 继承Thread类 3.2 实现Runnable接口 1. 线程的概念 1.1 线程…

[FE] React 初窥门径(四):React 组件的加载过程(render 阶段)

1. 回顾 前几篇文章中&#xff0c;我们采用了 VSCode 插件 CodeTour 来记录代码的执行过程&#xff0c; 并把相关的数据 .tour/ 放到了 github: thzt/react-tour 中。 截止到本文为之&#xff0c;我们总共记录了这些 code-tour&#xff0c; .tour/ ├── 2. 构建过程.tour ├─…

ASP .NET CORE 6 在项目中集成WatchDog开源项目

概念 WatchDog是一个开源的项目&#xff0c;可以实现对.Net 应用程序和API实现实时应用日志和性能监控平台。可以实现实时记录和查看应用程序中的消息、事件、HTTP请求和响应&#xff0c;以及运行时捕获的异常&#xff0c;有效帮助开发人员去排查应用异常&#xff0c;提升开发效…

Chrome浏览器音/视频无法自动播放

背景&#xff1a;由于google的一些制度&#xff0c;我们在写html项目时会发现刷新页面时无法自动播放audio和video&#xff0c;即使你添加了autoplay属性也无济于事&#xff0c; 但是IE和Edge浏览器是可以自动播放的。 解决方案&#xff1a; 本人在网上搜寻了很多方法&#xf…

WPF自定义日历控件Calendar 的方法

推荐下载地址 https://www.haolizi.net/example/view_2107.html <UserControl.Resources><local1:DayConverter x:Key"DayConverter"/><!--导入转换器--><Style x:Key"CalendarStyle1"TargetType"{x:Type Calendar}">&…

TOEIC 词汇专题:旅游计划篇

TOEIC 词汇专题&#xff1a;旅游计划篇 制定旅行计划时&#xff0c;尤其是跨国旅游&#xff0c;会涉及到很多独特的英语词汇。以下是与“旅游计划”相关的托业词汇&#xff0c;帮助你更加自如地规划行程。 1. 旅行服务和优惠 出发前了解一下与服务和优惠相关的常用词汇&#…

Java集合框架面试指南

Java集合框架面试指南 文章目录 Java集合框架面试指南ArrayList特点&#xff1a;LinkedList特点&#xff1a;ArrayDeque特点&#xff1a;PriorityQueue特点&#xff1a;HashMap特点&#xff1a;HashSet特点&#xff1a;LinkedHashMap特点LinkedHashMap经典用法 TreeMap特点Conc…

QEMU学习之路(4)— Xilinx开源项目systemctlm-cosim-demo安装与使用

QEMU学习之路&#xff08;4&#xff09;— Xilinx开源项目systemctlm-cosim-demo安装与使用 一、前言 项目说明&#xff1a;https://xilinx-wiki.atlassian.net/wiki/spaces/A/pages/862421112/Co-simulation 操作系统&#xff1a;Ubuntu 20.04.6 LTS gcc版本&#xff1a;9.4…

【Java知识】高性能网络应用框架Netty应知应会

文章目录 概述线程模型IO模型代码示例服务端代码示例客户端代码示例代码说明&#xff1a; 自定义协议实现自定义协议格式自定义编码器&#xff08;Encoder&#xff09;自定义解码器&#xff08;Decoder&#xff09;业务处理器&#xff08;Handler&#xff09;在Netty服务器管道…

AUTOSAR 规范中的设计模式:传感器执行器模式

在 AUTOSAR Adaptive Platform (AP) 规范中,传感器执行器模式是一种典型的设计模式,主要用于实时控制系统中,用来实现传感器数据的获取和执行器指令的发送。该模式通过分离传感器和执行器的实现,使其独立运行并且能够通过某种通信机制进行数据交换,以确保数据的实时性和系…

Linux:编辑器Vim和Makefile

✨✨所属专栏&#xff1a;Linux✨✨ ✨✨作者主页&#xff1a;嶔某✨✨ vim的三种常用模式 分别是命令模式&#xff08;command mode&#xff09;、插入模式&#xff08;Insert mode&#xff09;和底行模式&#xff08;last line mode&#xff09; 各模式的功能区分如下&…

【Linux】【进程控制】API汇总整理

进程控制是操作系统中一个非常重要的概念&#xff0c;它涉及到创建、管理和终止进程的能力。进程控制包括一系列操作&#xff0c;如创建新进程、等待进程结束、发送信号给进程等。下面是进程控制中一些常见的操作及其相关API&#xff1a; 进程控制概述 进程控制是指操作系统提…

HTML练习题:彼岸的花(web)

展示效果: 代码: <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>彼岸の花</title><style…

【安全性分析】正式安全分析与非正式安全分析

安全性分析-系列文章目录 第一章 【安全性分析】正式安全分析与非正式安全分析 第二章 【安全性分析】BAN逻辑 (BAN Logic) 文章目录 安全性分析-系列文章目录前言一、正式安全分析1. 理想化模型(如随机预言机模型)2. 标准模型(Standard Model)3. 形式化验证4. 数学证明二…

HTML小阶段二维表和思维导图

下面是对标签、元素、属性的对比二维表&#xff0c;通过对比3w1h&#xff08;what是什么、where用在哪、why为什么要用、how如何用&#xff09;来学习区分学习标签、元素、属性 标签 元素 属性 what &#xff08;Tags&#xff09;标签是用来标记内容块或标明元素内容意义 …

NIO 核心知识总结

在传统的 Java I/O 模型&#xff08;BIO&#xff09;中&#xff0c;I/O 操作是以阻塞的方式进行的。也就是说&#xff0c;当一个线程执行一个 I/O 操作时&#xff0c;它会被阻塞直到操作完成。这种阻塞模型在处理多个并发连接时可能会导致性能瓶颈&#xff0c;因为需要为每个连…

助力风力发电风机设备智能化巡检,基于YOLOv7全系列【tiny/l/x】参数模型开发构建无人机巡检场景下风机叶片缺陷问题智能化检测预警模型

在全球能源转型的大潮中&#xff0c;清洁环境能源的发展已成为各国关注的焦点。风力发电作为其中的佼佼者&#xff0c;以其可再生、无污染的特点&#xff0c;受到了广泛的青睐。然而&#xff0c;风力发电设施大多建于人迹罕至的地区&#xff0c;设备庞大且复杂&#xff0c;其维…