QT纯代码实现滑动开关控件

        开关按钮大家应该很熟悉,在设置里面经常遇到,切换时候的滑动效果比较帅气。通常说的开关按钮,有两个状态:on、off。大部分的开关按钮控件,基本上有两大类,第一类是纯代码绘制,这种对代码的掌控度要求比较高,但是灵活性比较好。第二类是贴图,专业的美工做好的各种状态的背景图片,只需要用代码将该图片画到界面上即可。下面,介绍一种实现纯代码绘制的开关按钮。想看利用贴图来实现的开关按钮请移步:https://blog.csdn.net/u012959478/article/details/140423375

一、简述

         QT纯代码实现滑动开关控件。

二、 设计思路

        通过继承QAbstractButton类,重写paintEvent函数来自定义绘制过程。        

        首先我们将滑动进度条的整体进行分块,先剖析整个按钮的实现包括的元素,分为两个外圆、一个矩形、一个内圆(白色)组成。接着我们通过QPaintEvent绘制出按钮的静止状态下的界面形状。接着,我们再剖析进度条的动态过程其动态过程为圆心在不同的时刻处在按钮中的不同位置。通过QPropertyAnimation来设置从左向右的运动实现滑动过程。

三、效果 

四、核心代码  
1、头文件
#ifndef SLIPBUTTON_H
#define SLIPBUTTON_H#include <QAbstractButton>struct SlipButtonPrivate;
class SlipButton : public QAbstractButton
{Q_OBJECTQ_PROPERTY(int offset READ offset WRITE setOffset)Q_PROPERTY(QColor checkedBackgroundColor READ checkedBackgroundColor WRITE setCheckedBackgroundColor)Q_PROPERTY(QColor uncheckedBackgroundColor READ uncheckedBackgroundColor WRITE setUncheckedBackgroundColor)
public:explicit SlipButton(QWidget* parent = nullptr);~SlipButton() override;QSize sizeHint() const override;QSize minimumSizeHint() const override;public slots:void setCheckedBackgroundColor(const QColor &color);QColor checkedBackgroundColor() const;void setUncheckedBackgroundColor(const QColor &color);QColor uncheckedBackgroundColor() const;protected:void paintEvent(QPaintEvent* event) override;void enterEvent(QEvent *event) override;void leaveEvent(QEvent* event) override;void resizeEvent(QResizeEvent* event) override;private slots:void onStartAnimation();int offset() const;void setOffset(int offset);private:inline double widthMargin() const;inline double heightMargin() const;QScopedPointer<SlipButtonPrivate> d;
};#endif // SLIPBUTTON_H
2、实现代码
#include "slipbutton.h"#include <QPropertyAnimation>
#include <QtWidgets>struct SlipButtonPrivate{int offset = 0;QColor uncheckBackgroundColor = QColor("#FFE0E0E0");QColor checkedBackgroundColor = QColor("#4da1ff");bool hover = false;QPropertyAnimation *animation;
};SlipButton::SlipButton(QWidget *parent): QAbstractButton(parent), d(new SlipButtonPrivate)
{d->animation = new QPropertyAnimation(this, "offset", this);setCheckable(true);setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);connect(this, &SlipButton::toggled, this, &SlipButton::onStartAnimation);
}SlipButton::~SlipButton()
{
}QSize SlipButton::sizeHint() const
{return QSize(100 * 1.8, 100);
}QSize SlipButton::minimumSizeHint() const
{return QSize(10 * 1.8, 10);
}void SlipButton::setCheckedBackgroundColor(const QColor& color)
{d->checkedBackgroundColor = color;update();
}QColor SlipButton::checkedBackgroundColor() const
{return d->checkedBackgroundColor;
}void SlipButton::setUncheckedBackgroundColor(const QColor &color)
{d->uncheckBackgroundColor = color;update();
}QColor SlipButton::uncheckedBackgroundColor() const
{return d->uncheckBackgroundColor;
}int SlipButton::offset() const
{return d->offset;
}void SlipButton::setOffset(int offset)
{d->offset = offset;update();
}double SlipButton::widthMargin() const
{return width() / 22.0;
}double SlipButton::heightMargin() const
{return height() / 22.0;
}void SlipButton::paintEvent(QPaintEvent* event)
{//qDebug() << offset();QWidget::paintEvent(event);QPainter painter(this);painter.setPen(Qt::NoPen);painter.setRenderHint(QPainter::Antialiasing);double w = width() - widthMargin() * 2;double h = height() - heightMargin() * 2;// 画背景QRectF rectSlot((width() - w) / 2, (height() - h) / 2, w, h);double slotRoundness = rectSlot.height() / 2;painter.setBrush(d->uncheckBackgroundColor);painter.drawRoundedRect(rectSlot, slotRoundness, slotRoundness);// 选中情况下,背景变蓝if(isEnabled() && isChecked()){QRectF rectSlotFill = rectSlot.adjusted(0, 0, offset() + h - (width() - widthMargin()), 0);painter.setBrush(d->checkedBackgroundColor);painter.drawRoundedRect(rectSlotFill, slotRoundness, slotRoundness);}QRectF rectThumb = QRectF(offset(), (height() - h) / 2, h, h);QColor colorThumbBorder = (d->hover) ? d->checkedBackgroundColor: QColor("#FFA8A8A8");painter.setBrush(colorThumbBorder);painter.drawEllipse(rectThumb);// 按钮圆点QColor colorThumb = isEnabled() ? QColor(Qt::white) : QColor("#FFE0E0E0");painter.setBrush(colorThumb);rectThumb.adjust(1.1, 1.1, -1.1, -1.1);painter.drawEllipse(rectThumb);
}void SlipButton::enterEvent(QEvent *event)
{QAbstractButton::enterEvent(event);setCursor(Qt::PointingHandCursor);d->hover = true;
}void SlipButton::leaveEvent(QEvent* event)
{QAbstractButton::leaveEvent(event);d->hover = false;
}void SlipButton::resizeEvent(QResizeEvent *event)
{QAbstractButton::resizeEvent(event);if(isChecked()){double h = height() - heightMargin() * 2;setOffset(width() - widthMargin() - h);}elsesetOffset(widthMargin());
}void SlipButton::onStartAnimation()
{double h = height() - heightMargin() * 2;double start = widthMargin();double end = width() - start - h;if(!isChecked())qSwap(start, end);// 改变参数值(offset = startValue, offset += interval_1...offset += interval_N)// 改变offset的同时,不断repaint(update)// 直到为目标值(offset = endValue)paint完成;d->animation->setStartValue(start);d->animation->setEndValue(end);d->animation->setDuration(120);d->animation->start();
}

        这个自定义控件SlipButton提供了基本的开关功能,通过鼠标点击可以在开启和关闭状态之间切换。你可以根据需要对控件进行自定义和扩展。

五、使用示例

以下是一个简单的示例代码,演示了如何在Qt中使用此控件:

#include "mainwindow.h"
#include "slipbutton.h"#include <QtWidgets>MainWindow::MainWindow(QWidget *parent): QMainWindow(parent)
{SlipButton *slipButton = new SlipButton(this);//slipButton->setFixedSize(50 * 1.8, 50);QWidget *widget = new QWidget(this);QVBoxLayout *layout = new QVBoxLayout(widget);layout->addWidget(slipButton);setCentralWidget(widget);resize(240, 140);
}MainWindow::~MainWindow()
{
}

        总结一下,笔者分享纯代码实现滑动开关控件的一种设计方法和流程,在此操作的基础上我们可以发挥想象开发出更多更有意思的控件,源码我放在此处以下地址。如有错误也请各位看官手下留情,欢迎评论区批评指正。

        谢谢你的关注和阅读,希望我的回答能帮到你。如果还有其他问题,欢迎随时向我提问。祝你一切顺利!

六、源代码下载

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

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

相关文章

dhtmlx-gantt甘特图数据展示

官网文档&#xff1a;甘特图文档 实现效果&#xff1a; 首先需要下载 dhtmlx-gantt组件 npm i dhtmlx-gantt //我项目中使用的是"dhtmlx-gantt": "^8.0.6" 这个版本&#xff0c;不同的版本api或是文档中存在的方法稍有差异 界面引用 <template>&l…

目标检测算法与应用算法 DS集成 接口相关_v0.1

目录 文章目录 目录0. 目标GPS信息、速度、加速度、航向角信息的输出1. 目标检测算法接口1.1 模型相关1.2 检测结果相关 2. 应用算法接口2.1 bool cross_line; //跨线&#xff08;变道压线检测&#xff09;2.2 bool break_in; //闯入&#xff08;目标闯入&#xff09;2.3 bool …

Linux HOOK机制与Netfilter HOOK

一. 什么是HOOK&#xff08;钩子&#xff09; 在计算机中&#xff0c;基本所有的软件程序都可以通过hook方式进行行为拦截&#xff0c;hook方式就是改变原始的执行流。 二. Linux常见的HOOK方式 1、修改函数指针。 2、用户态动态库拦截。 ①利用环境变量LD_PRELOAD和预装载机…

STM32自己从零开始实操:PCB全过程

一、PCB总体分布 以下只能让大家看到各个模块大致分布在板子的哪一块&#xff0c;只能说每个人画都有自己的理由&#xff1a; 电源&#xff1a;从外部接入电源&#xff0c;5V接到中间&#xff0c;向上变成4V供给无线&#xff0c;向下变成3V供给下面的接口&#xff08;也刻意放…

每日一练 - BGP 路由表中的团体属性

01 真题题目 下面一台路由器的输出信息&#xff0c;关于这段信息描述正确的是? A.目的网段 1.1.1.0/24 所携带的团体属性值是 NO-EXPORT&#xff0c; 表明该路由条目不能通告给任何 BGP 邻居 B.目的网段 1.1.1.0/24 所携带的图体属性值是 NO-EXPORT&#xff0c; 表明试路由…

Three.JS 使用RGBELoader和CubeTextureLoader 添加环境贴图

导入RGBELoader模块&#xff1a; import { RGBELoader } from "three/examples/jsm/loaders/RGBELoader.js"; 使用 addRGBEMappingk(environment, background,url) {rgbeLoader new RGBELoader();rgbeLoader.loadAsync(url).then((texture) > {//贴图模式 经纬…

git使用-命令行+VS Code结合使用

一、Git常用命令 // 显示当分支的状态。它会列出已修改、已暂存和未跟踪的文件 git status// 列出本地仓库中所有的分支&#xff0c;其中会特殊显示当前所在分支 git branch// 在当前分支的基础上创建一个新的分支&#xff0c;并切换到这个新的分支上 git checkout -b 新分支…

问题:向上对齐对象的快捷键是: #学习方法#笔记

问题&#xff1a;向上对齐对象的快捷键是: A、T B、L C、R D、W 参考答案如图所示

Leetcode 1302.层数最深子叶结点的和

大家好&#xff0c;今天我给大家分享一下我关于这个题的想法&#xff0c;我这个题过程比较复杂&#xff0c;但大家如果觉得好的话&#xff0c;就请给个免费的赞吧&#xff0c;谢谢了^ _ ^ 1.题目要求: 给你一棵二叉树的根节点 root &#xff0c;请你返回 层数最深的叶子节点的…

Go语言并发编程-Context上下文

Context上下文 Context概述 Go 1.7 标准库引入 context&#xff0c;译作“上下文”&#xff0c;准确说它是 goroutine 的上下文&#xff0c;包含 goroutine 的运行状态、环境、现场等信息。 context 主要用来在 goroutine 之间传递上下文信息&#xff0c;包括&#xff1a;取…

python爬虫实现简单的代理ip池

python爬虫实现简单的代理ip池 我们在普通的爬虫过程中经常遇到一些网站对ip进行封锁的 下面演示一下普通的爬虫程序 使用requests.get爬取数据 这段代码是爬取豆瓣排行榜的数据&#xff0c;使用f12来查看请求的url和数据格式 代码 def requestData():# 爬取数据的urlur…

数学基础【俗说矩阵】:矩阵相乘

矩阵乘法 矩阵乘法推导过程 一、两个线性方程复合代入 二、X1和X2合并同类项 三、复合后方程组结果 四、线性方程组矩阵表示 五、线性方程组矩阵映射表示 复合映射表示 六、矩阵乘法导出 矩阵乘法法则 1、规则一推导过程 左取行&#xff0c;右取列&#xff0c;对应相乘后…

【中项】系统集成项目管理工程师-第2章 信息技术发展-2.1信息技术及其发展-2.1.4信息安全

前言&#xff1a;系统集成项目管理工程师专业&#xff0c;现分享一些教材知识点。觉得文章还不错的喜欢点赞收藏的同时帮忙点点关注。 软考同样是国家人社部和工信部组织的国家级考试&#xff0c;全称为“全国计算机与软件专业技术资格&#xff08;水平&#xff09;考试”&…

「Vue组件化」封装i18n国际化

前言 在Vue应用中实现国际化(i18n),通常需要使用一些专门的库,比如vue-i18n。本文主要介绍自定义封装i18n,支持Vue、uniapp等版本。 设计思路 一、预期效果 二、核心代码 2.1 i18n.xlsx文件准备 2.2 脚本执行 根目录main.js根目录locali18n.xlsxnode main.jsmain.js 文件…

【ARM】MDK-解决CMSIS_DAP.DLL missing报错

【更多软件使用问题请点击亿道电子官方网站】 1、 文档目标 记录解决CMSIS_DAP.DLL missing的报错情况&#xff0c;对应相关报错信息&#xff0c;供后续客户参考&#xff0c;快速解决客户问题。 2、 问题场景 客户进行硬件调试时&#xff0c;发现Target设置内有CMSIS_DAP.DL…

全时守护,无死角监测:重点海域渔港视频AI智能监管方案

一、方案背景 随着海洋经济的快速发展和海洋资源的日益紧缺&#xff0c;对重点海域渔港进行有效监控和管理显得尤为重要。视频监控作为一种高效、实时的管理手段&#xff0c;已成为渔港管理中不可或缺的一部分。当前&#xff0c;我国海域面积广阔&#xff0c;渔港众多&#xf…

uni-app学习HBuilderX学习-微信开发者工具配置

HBuilderX官网&#xff1a;简介 - HBuilderX 文档 (dcloud.net.cn)https://hx.dcloud.net.cn/ uni-app官网&#xff1a; uni-app官网 (dcloud.net.cn)https://uniapp.dcloud.net.cn/quickstart-hx.htmlHBuilder下载安装&#xff1a;打开官网 uni-app项目的微信开发者工具配置…

41 QOS技术(服务质量)

1 QOS 产生背景 对于网络业务&#xff0c;影响服务质量的因素包括传输的带宽、传送的时延、数据的丢包率等。网络资源总是有限的&#xff0c;只要存在抢夺网络资源的情况&#xff0c;就会出现服务质量的要求网络总带宽固定的情况下&#xff0c;如果某类业务占用的带宽越多&am…

Mysql-错误处理: Found option without preceding group in config file

1、问题描述 安装MYSQL时&#xff0c;在cmd中“初始化”数据库时&#xff0c;输入命令&#xff1a; mysqld --initialize --consolecmd报错&#xff1a; D:\mysql-5.7.36-winx64\bin>mysql --initialize --console mysql: [ERROR] Found option without preceding group …

Web开发:ASP.NET CORE的后端小结(基础)

1.后端重定向到指定路由 public IActionResult Index(){return RedirectToAction("Index", "Main");//重定向>Main/Index} 【备注】如果在MainController的Index方法中return View();本质是 return View("Index")&#xff0c;返回和方法同名的…