QT5+VS自定义IP地址控件生成及使用

本文使用环境为QT5.9.7+VS2015,其他环境未做测试,大同小异。

VS新建QT自定义控件类:

自动生成解决方案文件

.修改QIPLineEdit.h

#pragma once

#include <QLineEdit>
#include <QEvent>

class QIPLineEdit : public QLineEdit
{
	Q_OBJECT

public:
	QIPLineEdit(QWidget *parent = 0);
	~QIPLineEdit();

	void setText(const QString &strIP);
	QString text() const;
protected:
	void paintEvent(QPaintEvent *event);
	bool eventFilter(QObject *obj, QEvent *ev);

	int getIndex(QLineEdit *pEdit);
	bool isTextValid(const QString &strIP);
private:
	QLineEdit *m_lineEidt[4];
};

QIPLineEdit.cpp

#include "QIPLineEdit.h"
#include <QRegExpValidator>
#include <QPainter>
#include <QHBoxLayout>
#include <QKeyEvent>
#include <QMessageBox>
#include <QDebug>

QIPLineEdit::QIPLineEdit(QWidget *parent)
	: QLineEdit(parent)
{
	QRegExp rx("(2[0-5]{2}|2[0-4][0-9]|1?[0-9]{1,2})");
	QHBoxLayout *pHBox = new QHBoxLayout(this);
	pHBox->setSpacing(10);
	pHBox->setContentsMargins(0, 0, 0, 0);
	for (int i = 0; i < 4; i++)
	{
		m_lineEidt[i] = new QLineEdit(this);
		m_lineEidt[i]->setFrame(false);
		m_lineEidt[i]->setMaxLength(3);
		m_lineEidt[i]->setAlignment(Qt::AlignCenter);
		m_lineEidt[i]->installEventFilter(this);
		m_lineEidt[i]->setValidator(new QRegExpValidator(rx, this));
		m_lineEidt[i]->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
		pHBox->addWidget(m_lineEidt[i]);
	}
	this->setReadOnly(true);
}

QIPLineEdit::~QIPLineEdit()
{

}

void QIPLineEdit::paintEvent(QPaintEvent *event)
{
	__super::paintEvent(event);
	QPainter painter(this);
	QBrush brush;
	brush.setStyle(Qt::BrushStyle::SolidPattern);
	brush.setColor(Qt::black);
	painter.setBrush(brush);

	int width = 0;
	for (int i = 0; i < 3; i++)
	{
		width += m_lineEidt[i]->width() + (i == 0 ? 3 : 10);//布局的间隔
		painter.drawEllipse(width, height() / 2 - 2, 4, 4);
	}
}

int QIPLineEdit::getIndex(QLineEdit *pEdit)
{
	int index = -1;
	for (int i = 0; i < 4; i++)
	{
		if (pEdit == m_lineEidt[i])
			index = i;
	}
	return index;
}

bool QIPLineEdit::eventFilter(QObject *obj, QEvent *ev)
{
	if (children().contains(obj) && QEvent::KeyPress == ev->type())
	{
		QKeyEvent *keyEvent = dynamic_cast<QKeyEvent *>(ev);
		QLineEdit *pEdit = qobject_cast<QLineEdit *>(obj);
		switch (keyEvent->key())
		{
		case Qt::Key_0:
		case Qt::Key_1:
		case Qt::Key_2:
		case Qt::Key_3:
		case Qt::Key_4:
		case Qt::Key_5:
		case Qt::Key_6:
		case Qt::Key_7:
		case Qt::Key_8:
		case Qt::Key_9:
		{
			QString strText = pEdit->text();
			if (pEdit->selectedText().length())
			{
				pEdit->text().replace(pEdit->selectedText(), QChar(keyEvent->key()));
			}
			else if (strText.length() == 3 || strText.length() < 3 && strText.toInt() * 10 > 255)
			{
				int index = getIndex(pEdit);
				if (index != -1 && index != 3)
				{
					m_lineEidt[index + 1]->setFocus();
					m_lineEidt[index + 1]->selectAll();
				}
			}
			else if (strText.length() == 2 && strText.toInt() * 10 < 255)
			{
				if (Qt::Key_0 == keyEvent->key() && strText.toInt())
				{
					pEdit->setText(strText.insert(pEdit->cursorPosition(), QChar(Qt::Key_0)));
				}
			}
			return __super::eventFilter(obj, ev);
		}
		break;
		case Qt::Key_Backspace:
		{
			QString strText = pEdit->text();
			if (!strText.length() || strText.length() && !pEdit->cursorPosition())
			{
				int index = getIndex(pEdit);
				if (index != -1 && index != 0)
				{
					m_lineEidt[index - 1]->setFocus();
					int length = m_lineEidt[index - 1]->text().length();
					m_lineEidt[index - 1]->setCursorPosition(length ? length : 0);
				}
			}
			return __super::eventFilter(obj, ev);
		}
		case Qt::Key_Left:
		{
			if (!pEdit->cursorPosition())
			{
				int index = getIndex(pEdit);
				if (index != -1 && index != 0)
				{
					m_lineEidt[index - 1]->setFocus();
					int length = m_lineEidt[index - 1]->text().length();
					m_lineEidt[index - 1]->setCursorPosition(length ? length : 0);
				}
			}
			return __super::eventFilter(obj, ev);
		}
		case Qt::Key_Right:
		{
			if (pEdit->cursorPosition() == pEdit->text().length())
			{
				int index = getIndex(pEdit);
				if (index != -1 && index != 3)
				{
					m_lineEidt[index + 1]->setFocus();
					m_lineEidt[index + 1]->setCursorPosition(0);
				}
			}
			return __super::eventFilter(obj, ev);
		}
		default:
			break;
		}
	}
	return false;
}

void QIPLineEdit::setText(const QString &strIP)
{
	if (!isTextValid(strIP))
	{
		QMessageBox::warning(this, "Attention", "Your IP Address is Invalid!", QMessageBox::StandardButton::Ok);
		return;
	}
	else
	{
		int i = 0;
		QStringList ipList = strIP.split(".");
		for each (QString ip in ipList)
		{
			m_lineEidt[i]->setText(ip);
			i++;
		}
	}
}

bool QIPLineEdit::isTextValid(const QString &strIP)
{
	QRegExp rx2("\\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\b");
	if (!rx2.exactMatch(strIP))
		return false;
	return true;
}

QString QIPLineEdit::text() const
{
	QString strIP;
	for (int i = 0; i < 4; i++)
		strIP.append(m_lineEidt[i]->text());
	return strIP;
}

编译生成.dll及.lib,拷贝.dll到对应VS安装控件目录

拷贝QIPLineEdit.h及QIPLineEdit.cpp文件到使用自定义控件类的工程目录并添加

现在我们可以在QT类界面编辑器中看到我们的控件了,拖入直接使用。

到此OK,nice.

不过这样也不属于封装,只是方便而已,后续再研究~

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Qt是一种跨平台的C++开发框架,它提供了丰富的GUI组件库,使得开发者可以轻松地设计和创建各种控件和界面。自定义控件从一定程度上方便了程序的编写,让程序更加直观,使用得当的自定义控件可以让程序增添不少精彩细节。仪表盘是一种比较常见的自定义控件,它可以用来显示实时数据,例如速度,油量,温度等等。下面来简单介绍一下用Qt自定义一个仪表盘的一些步骤。 首先要构思好仪表盘的外观和功能,比如可以考虑盘的大小,采用什么颜色,显示哪些数据等等。 其次,需要用Qt中提供的基础控件(如QPainter、QPoint、QRect)来构建自定义控件的各个部分,包括盘表、指针、刻度等,然后为这些部分设置合适的属性(如颜色、位置、宽度等)。 接着,需要实现控件的数据传递和刷新。一般情况下,会使用定时器或者事件触发来更新控件显示的数据。考虑到仪表盘是一种实时显示数据的控件,所以在设计数据刷新时需要保证刷新频率足够高,否则会出现卡顿、显示延迟等问题。 最后,为了方便其他的开发者使用自定义控件,可以将其打包成独立的Qt插件,或者直接将自定义控件的源代码公开发布。 以上就是简单的Qt自定义控件仪表盘的一些步骤,开发者可以根据自己的需求进行相应更改和优化。总之自定义控件并不是一件简单的事情,但是如果能够掌握好最基本的知识,就能够创造出更加精美、实用的自定义控件。 ### 回答2: Qt自定义控件仪表盘可以用于需要展示数据的界面设计。通过自定义仪表盘,可以实现不同样式和功能的展示,并且能够满足不同场景下的需求。 在Qt中,仪表盘的设计可以通过绘图、圆弧、指针和动画的实现,使得界面更加直观、美观,也更加容易被用户理解和操作。可以通过Qt提供的QPainter绘图工具绘制圆弧,也可以通过QTimer控制指针的动画效果。 另外,仪表盘也是可以与其他控件进行绑定的。通过信号与槽的机制,可以将仪表盘的数值与其他控件进行绑定,实现更加复杂的界面功能。 需要特别注意的是,仪表盘的设计需要考虑到不同屏幕分辨率的适配性。通过使用Qt提供的屏幕适配机制,可以实现不同分辨率下的仪表盘展示效果。 总之,Qt自定义控件仪表盘是一个非常实用、优雅的设计元素,可以帮助开发者快速开发出漂亮的界面,并且提高用户体验。 ### 回答3: Qt是一个强大的C++跨平台框架,自带了很多常用的UI控件,但是有时候我们也需要自定义一些控件来满足我们的需求。今天我们来介绍一下如何使用Qt自定义一个仪表盘控件。 首先我们需要继承QWidget类,命名为Dashboard。然后我们需要在Dashboard的构造函数中初始化一些常量,比如外部圆弧的宽度、内部圆弧的半径等等。接着我们需要重写paintEvent函数,绘制仪表盘的外部圆弧、刻度、指针等等,具体绘制方式可以根据需求灵活设置。 为了使仪表盘可以在Qt Designer中拖拽使用,我们需要在Dashboard类中加入QIB_DESIGNER_EXPORT_WIDGETS宏。最后在.pro文件中添加如下代码: ``` QT += designer TARGET = Dashboard TEMPLATE = lib HEADERS += Dashboard.h SOURCES += Dashboard.cpp QIB_DESIGNER_EXPORT_WIDGETS(Dashboard) ``` 这样我们就可以在Qt Designer中使用Dashboard自定义控件了。当然,使用自定义控件也需要相应的信号槽机制来进行交互操作。 总的来说,自定义控件可以更好地满足我们的需求,同时也可以提高界面的美观度。在自定义控件的过程中,我们要考虑到控件的可扩展性、易用性以及代码的可维护性等方面,并在不断地改进和优化中,不断提高自己的技术水平。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值