百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

Qt编写自定义控件23-广告轮播控件

bigegpt 2024-08-21 12:21 3 浏览

一、前言

广告轮播这个控件做的比较早,是很早以前定制一个电信客户端时候用到的,该客户端需要在首页展示轮播预先设定好的图片,图片的路径可以自由设定,然后轮播的间隔速度可以自由控制,同时该控件还需要提供两种指示器的风格,一种是迷你型的样式,一种是数字型的样式。

本控件很早就做好了,由于当时的QPainter功力不足,还不是很熟悉QPainter,采用的是效率比较低的直接用现有控件堆积而成,比如指示器采用的QLabel,用样式表来控制对应的形状,指示器所在的底部放一个widget,采用左右布局,然后右侧放一个弹簧把指示器label全部顶在左边,至于图片的显示,采用的是样式表中的border-image来设置,开个定时器,到了时间则设置成不同的border-image即可。这种方法虽然效率低了点,但是初学者很容易理解接收,甚至可以做出更多的效果,只要项目对CPU要求不高,也不失为一种还行的办法。

二、实现的功能

* 1:可设置显示的图像

* 2:可添加多个广告

* 3:可设置指示器样式 迷你型样式 数字型样式

* 4:可设置指示器大小

* 5:可设置切换间隔

三、效果图

四、完整代码

#pragma execution_character_set("utf-8")
#include "adswidget.h"
#include "qevent.h"
#include "qlabel.h"
#include "qlayout.h"
#include "qtimer.h"
#include "qdebug.h"
AdsWidget::AdsWidget(QWidget *parent) : QWidget(parent)
{
 this->initWidget();
 this->initForm();
}
AdsWidget::~AdsWidget()
{
 if (timer->isActive()) {
 timer->stop();
 }
}
bool AdsWidget::eventFilter(QObject *obj, QEvent *event)
{
 if (event->type() == QEvent::MouseButtonPress) {
 if (obj->inherits("QLabel")) {
 //先停止定时器,防止按下切换的时候短时间内再度切换
 timer->stop();
 changedAds((QLabel *)obj);
 timer->start(interval);
 }
 }
 return QWidget::eventFilter(obj, event);
}
void AdsWidget::initWidget()
{
 QVBoxLayout *verticalLayout = new QVBoxLayout(this);
 verticalLayout->setSpacing(0);
 verticalLayout->setContentsMargins(0, 0, 0, 0);
 widgetBg = new QWidget(this);
 widgetBg->setObjectName(QString::fromUtf8("widgetBg"));
 QGridLayout *gridLayout = new QGridLayout(widgetBg);
 gridLayout->setSpacing(0);
 gridLayout->setContentsMargins(0, 0, 0, 0);
 QSpacerItem *verticalSpacer = new QSpacerItem(10, 10, QSizePolicy::Minimum, QSizePolicy::Expanding);
 gridLayout->addItem(verticalSpacer, 0, 0, 1, 1);
 widgetBanner = new QWidget(widgetBg);
 widgetBanner->setObjectName(QString::fromUtf8("widgetBanner"));
 QHBoxLayout *horizontalLayout = new QHBoxLayout(widgetBanner);
 horizontalLayout->setSpacing(3);
 gridLayout->addWidget(widgetBanner, 1, 0, 1, 1);
 QSpacerItem *horizontalSpacer = new QSpacerItem(10, 10, QSizePolicy::Expanding, QSizePolicy::Minimum);
 gridLayout->addItem(horizontalSpacer, 1, 1, 1, 1);
 verticalLayout->addWidget(widgetBg);
}
void AdsWidget::initForm()
{
 interval = 3000;
 bannerFixedSize = QSize(20, 20);
 bannerStyle = BannerStyle_Num;
 imageNames.clear();
 currentIndex = 0;
 timer = new QTimer(this);
 timer->setInterval(interval);
 connect(timer, SIGNAL(timeout()), this, SLOT(changedAds()));
 timer->start();
}
void AdsWidget::changedAds()
{
 if (names.count() == 0) {
 return;
 }
 if (currentIndex < names.count() - 1) {
 currentIndex++;
 } else {
 currentIndex = 0;
 }
 changedAds(labs.at(currentIndex));
}
void AdsWidget::changedAds(QLabel *lab)
{
 //这里采用样式改变背景颜色的方式,也可以改成贴背景图的方式
 QString qss;
 QString qssCurrent;
 if (bannerStyle == BannerStyle_Min) {
 qss = "QLabel{background:#4380A8;}";
 qssCurrent = "QLabel{background:#084279;}";
 } else if (bannerStyle == BannerStyle_Num) {
 qss = "QLabel{color:#FFFFFF;background:rgba(0,0,0,40);}";
 qssCurrent = "QLabel{color:#FFFFFF;background:#0C7FC8;}";
 }
 //将当前广告指示器突出显示
 foreach (QLabel *currentLab, labs) {
 if (currentLab == lab) {
 currentLab->setStyleSheet(qssCurrent);
 } else {
 currentLab->setStyleSheet(qss);
 }
 }
 //更新索引和图片
 currentIndex = labs.indexOf(lab);
 widgetBg->setStyleSheet(QString("QWidget#widgetBg{border-image:url(%1);}").arg(names.at(currentIndex)));
}
int AdsWidget::getInterval() const
{
 return this->interval;
}
QSize AdsWidget::getBannerFixedSize() const
{
 return this->bannerFixedSize;
}
AdsWidget::BannerStyle AdsWidget::getBannerStyle() const
{
 return this->bannerStyle;
}
QString AdsWidget::getImageNames() const
{
 return this->imageNames;
}
QSize AdsWidget::sizeHint() const
{
 return QSize(200, 150);
}
QSize AdsWidget::minimumSizeHint() const
{
 return QSize(20, 15);
}
void AdsWidget::setInterval(int interval)
{
 if (this->interval != interval) {
 this->interval = interval;
 timer->setInterval(interval);
 }
}
void AdsWidget::setBannerFixedSize(const QSize &bannerFixedSize)
{
 if (this->bannerFixedSize != bannerFixedSize) {
 this->bannerFixedSize = bannerFixedSize;
 foreach (QLabel *lab, labs) {
 lab->setFixedSize(bannerFixedSize);
 }
 }
}
void AdsWidget::setBannerStyle(const AdsWidget::BannerStyle &bannerStyle)
{
 if (this->bannerStyle != bannerStyle) {
 this->bannerStyle = bannerStyle;
 foreach (QLabel *lab, labs) {
 if (bannerStyle == BannerStyle_Min) {
 lab->setText("");
 } else if (bannerStyle == BannerStyle_Num) {
 lab->setText(lab->text());
 }
 }
 }
}
void AdsWidget::setImageNames(const QString &imageNames)
{
 if (this->imageNames != imageNames) {
 this->imageNames = imageNames;
 //先清空原有所有指示器
 qDeleteAll(labs);
 labs.clear();
 //根据图片链表自动生成导航指示器和图片链表
 names = this->imageNames.split(";");
 for (int i = 0; i < names.count(); i++) {
 QLabel *lab = new QLabel;
 widgetBanner->layout()->addWidget(lab);
 lab->setFixedSize(bannerFixedSize);
 lab->setAlignment(Qt::AlignCenter);
 lab->installEventFilter(this);
 if (bannerStyle == BannerStyle_Num) {
 lab->setText(QString::number(i + 1));
 }
 labs.append(lab);
 }
 //立即显示第一张
 changedAds();
 }
}

六、控件介绍

1. 超过146个精美控件,涵盖了各种仪表盘、进度条、进度球、指南针、曲线图、标尺、温度计、导航条、导航栏,flatui、高亮按钮、滑动选择器、农历等。远超qwt集成的控件数量。

2. 每个类都可以独立成一个单独的控件,零耦合,每个控件一个头文件和一个实现文件,不依赖其他文件,方便单个控件以源码形式集成到项目中,较少代码量。qwt的控件类环环相扣,高度耦合,想要使用其中一个控件,必须包含所有的代码。

3. 全部纯Qt编写,QWidget+QPainter绘制,支持Qt4.6到Qt5.12的任何Qt版本,支持mingw、msvc、gcc等编译器,不乱码,可直接集成到Qt Creator中,和自带的控件一样使用,大部分效果只要设置几个属性即可,极为方便。

4. 每个控件都有一个对应的单独的包含该控件源码的DEMO,方便参考使用。同时还提供一个所有控件使用的集成的DEMO。

5. 每个控件的源代码都有详细中文注释,都按照统一设计规范编写,方便学习自定义控件的编写。

6. 每个控件默认配色和demo对应的配色都非常精美。

7. 超过130个可见控件,6个不可见控件。

8. 部分控件提供多种样式风格选择,多种指示器样式选择。

9. 所有控件自适应窗体拉伸变化。

10. 集成自定义控件属性设计器,支持拖曳设计,所见即所得,支持导入导出xml格式。

11. 自带activex控件demo,所有控件可以直接运行在ie浏览器中。

12. 集成fontawesome图形字体+阿里巴巴iconfont收藏的几百个图形字体,享受图形字体带来的乐趣。

13. 所有控件最后生成一个dll动态库文件,可以直接集成到qtcreator中拖曳设计使用。

七、SDK下载

- 下载链接中包含了各个版本的动态库文件,所有控件的头文件,使用demo。

- 自定义控件插件开放动态库dll使用(永久免费),无任何后门和限制,请放心使用。

- 目前已提供26个版本的dll,其中包括了qt5.12.3 msvc2017 32+64 mingw 32+64 的。

- 不定期增加控件和完善控件,不定期更新SDK,欢迎各位提出建议,谢谢!

相关推荐

悠悠万事,吃饭为大(悠悠万事吃饭为大,什么意思)

新媒体编辑:杜岷赵蕾初审:程秀娟审核:汤小俊审签:周星...

高铁扒门事件升级版!婚宴上‘冲喜’老人团:我们抢的是社会资源

凌晨两点改方案时,突然收到婚庆团队发来的视频——胶东某酒店宴会厅,三个穿大红棉袄的中年妇女跟敢死队似的往前冲,眼瞅着就要扑到新娘的高额钻石项链上。要不是门口小伙及时阻拦,这婚礼造型团队熬了三个月的方案...

微服务架构实战:商家管理后台与sso设计,SSO客户端设计

SSO客户端设计下面通过模块merchant-security对SSO客户端安全认证部分的实现进行封装,以便各个接入SSO的客户端应用进行引用。安全认证的项目管理配置SSO客户端安全认证的项目管理使...

还在为 Spring Boot 配置类加载机制困惑?一文为你彻底解惑

在当今微服务架构盛行、项目复杂度不断攀升的开发环境下,SpringBoot作为Java后端开发的主流框架,无疑是我们手中的得力武器。然而,当我们在享受其自动配置带来的便捷时,是否曾被配置类加载...

Seata源码—6.Seata AT模式的数据源代理二

大纲1.Seata的Resource资源接口源码2.Seata数据源连接池代理的实现源码3.Client向Server发起注册RM的源码4.Client向Server注册RM时的交互源码5.数据源连接...

30分钟了解K8S(30分钟了解微积分)

微服务演进方向o面向分布式设计(Distribution):容器、微服务、API驱动的开发;o面向配置设计(Configuration):一个镜像,多个环境配置;o面向韧性设计(Resista...

SpringBoot条件化配置(@Conditional)全面解析与实战指南

一、条件化配置基础概念1.1什么是条件化配置条件化配置是Spring框架提供的一种基于特定条件来决定是否注册Bean或加载配置的机制。在SpringBoot中,这一机制通过@Conditional...

一招解决所有依赖冲突(克服依赖)

背景介绍最近遇到了这样一个问题,我们有一个jar包common-tool,作为基础工具包,被各个项目在引用。突然某一天发现日志很多报错。一看是NoSuchMethodError,意思是Dis...

你读过Mybatis的源码?说说它用到了几种设计模式

学习设计模式时,很多人都有类似的困扰——明明概念背得滚瓜烂熟,一到写代码就完全想不起来怎么用。就像学了一堆游泳技巧,却从没下过水实践,很难真正掌握。其实理解一个知识点,就像看立体模型,单角度观察总...

golang对接阿里云私有Bucket上传图片、授权访问图片

1、为什么要设置私有bucket公共读写:互联网上任何用户都可以对该Bucket内的文件进行访问,并且向该Bucket写入数据。这有可能造成您数据的外泄以及费用激增,若被人恶意写入违法信息还可...

spring中的资源的加载(spring加载原理)

最近在网上看到有人问@ContextConfiguration("classpath:/bean.xml")中除了classpath这种还有其他的写法么,看他的意思是想从本地文件...

Android资源使用(android资源文件)

Android资源管理机制在Android的开发中,需要使用到各式各样的资源,这些资源往往是一些静态资源,比如位图,颜色,布局定义,用户界面使用到的字符串,动画等。这些资源统统放在项目的res/独立子...

如何深度理解mybatis?(如何深度理解康乐服务质量管理的5个维度)

深度自定义mybatis回顾mybatis的操作的核心步骤编写核心类SqlSessionFacotryBuild进行解析配置文件深度分析解析SqlSessionFacotryBuild干的核心工作编写...

@Autowired与@Resource原理知识点详解

springIOCAOP的不多做赘述了,说下IOC:SpringIOC解决的是对象管理和对象依赖的问题,IOC容器可以理解为一个对象工厂,我们都把该对象交给工厂,工厂管理这些对象的创建以及依赖关系...

java的redis连接工具篇(java redis client)

在Java里,有不少用于连接Redis的工具,下面为你介绍一些主流的工具及其特点:JedisJedis是Redis官方推荐的Java连接工具,它提供了全面的Redis命令支持,且...