zoukankan      html  css  js  c++  java
  • Qt之股票组件-股票检索--支持搜索结果预览、鼠标、键盘操作

    原文链接:Qt之股票组件-股票检索--支持搜索结果预览、鼠标、键盘操作

    一、感慨一下

    之前做过一款炒股软件,个人觉着是我职业生涯里做过的效果最好的一款产品,而且速度也不慢,效果可以参考财联社-产品展示这篇文章,当然这篇文章只能显示有限的内容,其中整个代码的结构、一些好的方法和设计模式是没有机会展示的。

    最近听到一个不好的消息,我们的产品夭折了。刚听到这个消息时心理还挺不是滋味的,毕竟这个产品我是从头参与到尾,后来因为种种原因离开了,产品功能也就此终结,但回想起那段开发的日子,真的是收获满满。更确切的说,这个产品应该是换了一种语言重新开始做。

    不爽归不爽,可整个产品的代码还是不错的,因此 后续有时间我会慢慢的把一些好的代码抽离出来,编译成一个个可以单独运行的demo,方便有需要的朋友使用。

    如果有需要的朋友可以加我好友,有偿提供源码、或者也可以进一步提供功能定制

    封装的控件,或者demo都是没有样式的,所以看着会比较丑一些,不过加样式也是分分钟。。。这里咱可以先看功能,需要即可定制

    本篇文章我们首先介绍的就是股票,该控件支持常用的股票检索功能,支持模糊匹配,键盘上下键切换当前检索项等

    右键菜单包括复制、粘贴、剪贴、全选等

    本篇文章中不包括的功能也可以提供定制,需求合理即可。

    下面来具体说一说这个功能的实现思路,会公开大多数核心代码,有需要的同学可以根据思路自行完善整个代码。

    二、效果展示

    如下效果图所示,是自选股使用上的一个展示效果,具有如下功能

    1. 搜索编辑框,支持股票代码和股票名称搜索
    2. 搜索预览框支持鼠标hover,并且可以使用键盘上下键进行当前项切换,单机时支持切换自选股
    3. 自选股列表,支持拖拽,拖拽时会有拖拽项映像,并示意将要拖拽到哪个位置
    4. 支持右键菜单,可以对某一项进行移动,删除等操作

    如果觉着demo比较丑的话,可以看财联社-产品展示这篇文章中的效果图

    三、搜索编辑框

    首先出场的是搜索编辑框,如gif图中展示所示,搜索框支持预览数据,当我们输入了字符串后,就会出现过滤后的预览数据。这里由于我们的股票数据是我自己模拟的,因此只显示了5条数据。

    实现搜索编辑框,有2个小的模块需要讲解,一个是编辑框本身,它用于输入文本的能力,并且支持复制、粘贴等交互操作;另一个就是预览框了,他会动态的展示当前搜索的内容。

    1、编辑框

    Qt已经帮我们实现了一种编辑框,但是他自带了很多菜单项,如果产品这个时候说,菜单项我需要自己定制,多余的项不要。那么我们是不是得重写这个控件呢?答案是肯定的

    下面我们就来讲解这个控件的重写步骤

    重写一个Qt控件还是很简单的,使用Qt超过半年的同学都会重写大量各种各样的控件,而我们的编辑框重写就会像下面这样,是一个简单的头文件展示

    ///***********************************///
    /// 描述:自定义编辑框,重写鼠标右键事件
    ///***********************************///
    class SearchEdit : public QLineEdit
    {
    public:
    	SearchEdit(QWidget * parent = nullptr);
    	~SearchEdit(){}
    
    protected:
    	virtual void contextMenuEvent(QContextMenuEvent * event) override;
    
    private:
    	void InitMenu();
    
    private:
    	QMenu * m_PopMenu = nullptr;
    };
    

    这里我们主要是针对右键菜单进行了重写,Qt窗体实现右键菜单的方式多种多样,具体可以参考我很早以前写的Qt之自定义QLineEdit右键菜单这篇文章,今天我们也使用其中的一种方式来实现右键菜单,那就是实现默认的contextMenuEvent函数,这个函数之所以会响应,也是有一定条件的,Qt之自定义QLineEdit右键菜单这篇文章中讲解的也很清楚,那就是contextMenuPolicy的值必须为默认的Qt::DefaultContextMenu属性。

    至于菜单重写实现函数,这里就不展示了,就是比较常规的使用QMenu增加QAction的操作

    2、预览框

    大家仔细想一想,预览框是什么时候出现的?他显示的数据有什么样的特征?接下来我们来一一做以分析

    首先是出现时机

    预览框主要是展示我们模糊搜索后的股票数据,那么结论就很明显了。预览的出现时机就是搜索内容发现变化的时候,并且当编辑框失去焦点时,我们应该主动关闭预览框

    编辑框内容发现变化时,显示预览框

    connect(d_ptr->m_pSearchLineEdit, &QLineEdit::textChanged, this, &SelfStocksWidget::TextChanged);
    

    处理预览框数据,主要是使用了FilterModel来进行过滤所有股票后选项,注意我们过滤的条件就是搜索框中输入的内容

    void SelfStocksWidget::TextChanged(const QString & text)
    {
    	if (d_ptr->m_pFilterModel)
    	{
    		d_ptr->m_pFilterModel->SetFilterContext(text);
    	}
    	if (d_ptr->m_pStockPreviewWidget)
    	{
    		if (text.isEmpty())
    		{
    			d_ptr->m_pStockPreviewWidget->hide();
    			d_ptr->m_pPreviewError->hide();
    			d_ptr->m_pCloseButton->setIcon(QIcon());
    		}
    		else
    		{
    			d_ptr->m_pCloseButton->setIcon(QIcon(":/optional/Resources/optional/sotck_search_close_normal.png"));
    			d_ptr->m_pStockPreviewWidget->move(d_ptr->m_pTitleWidget->mapToGlobal(QPoint(0, d_ptr->m_pTitleWidget->height())));
    			int rowHeight = d_ptr->m_pStockPreview->rowHeight(0);
    			int rowCount = d_ptr->m_pFilterModel->rowCount();
    
    			...
    		}
    	}
    }
    

    当编辑框失去焦点时,关闭预览框
    这里我们取了一个巧,接收了该App的原生Win32消息,当我们发现一些影响窗口焦点的事件被触发时,我们去判断是否需要关闭预览框。

    具体可以参考我很早之前写的qt捕获全局windows消息这篇文章

    bool SelfStocksWidget::nativeEventFilter(const QByteArray & eventType, void * message, long * result)
    {
    	if (eventType == "windows_generic_MSG" || eventType == "windows_dispatcher_MSG")
    	{
    		MSG * pMsg = reinterpret_cast<MSG *>(message);
    
    		if (pMsg->message == WM_MOVE)
    		{
    			NativeParentWindowMove();
    		}
    		else if (pMsg->message == WM_ACTIVATEAPP)
    		{
    			if (bool(pMsg->wParam) == false)
    			{
    				if (!d_ptr->m_pStockPreviewWidget->rect().contains(d_ptr->m_pStockPreview->mapFromGlobal(QPoint(pMsg->pt.x, pMsg->pt.y))))
    				{
    					d_ptr->m_pStockPreviewWidget->hide();
    				}
    				if (!d_ptr->m_pPreviewError->rect().contains(d_ptr->m_pPreviewError->mapFromGlobal(QPoint(pMsg->pt.x, pMsg->pt.y))))
    				{
    					d_ptr->m_pPreviewError->hide();
    				}
    			}
    		}
    		else if (pMsg->message == WM_NCMBUTTONDOWN
    			|| pMsg->message == WM_LBUTTONDOWN
    			|| pMsg->message == WM_RBUTTONDOWN
    			|| pMsg->message == WM_NCLBUTTONDOWN
    			|| pMsg->message == WM_NCRBUTTONDOWN
    			|| pMsg->message == WM_MBUTTONDOWN)
    		{
    			同上...
    		}
    

    下面就是一个比较负责预览数据环节了,几千只股票,要准、要快,我们应该怎么技术选型呢?

    预览框到底怎么显示数据的?他显示的都是哪些数据?

    Qt提供了QListView、QTableView和QTreeView这3种视图模式,然后搭配Mode数据源,可以完成高效的大量数据展示,得知这个内容后是不是还有些小兴奋呢!

    乍一看,QListView和QTableView都可以作为我们的预览框窗口,毕竟每一个Item项都是可以去重新定制的,看起来QListView还是更简单一些,而且速度也会更快一些,但是仔细想想,好像不是这么回事,我们既然要支持股票代码和名称都进行搜索,那么自然不是一列数据就可以进行过滤的,方便起见我们还是使用QTableView作为我们的视图窗口

    既然视图窗口选定了,接下来就是一堆的事件定制了

    a、重写QTableView

    重写QTableView时,我们得考虑一个很重要的事情,那就是鼠标hover事件了,鼠标移动时我们需要把当前行设置为鼠标hover状态,为了实现这个效果,我可谓是费劲脑汁,想出了一个办法,写了一个IView接口类,让QTableView去继承,当鼠标hover时,去调用这个接口类告知QTableView当前hover项。

    class IView
    {
    public:
    	virtual void SetMouseHover(int, bool forceChanged = false) = 0;
    };
    

    上边的代码是不是看着很简单呢,就一个接口,就是当鼠标hover时告知表格当前hover项,那么什么实际通知合适呢?我这里是重写了QStyledItemDelegate绘图代理类,在paint函数中通知表格的,其他同学有好的办法也可以留言。

    预览框的头文件大致是下面这样的,这里我只把公有的接口放出来了,其他的一些私有接口和成员变量没有公开(放出来估计大家也不看)

    ///***********************************///
    /// 描述:搜索预览框
    ///***********************************///
    class StockTableView : public QTableView, public IView
    {
    	Q_OBJECT
    
    signals :
    	void RowClicked(const QString & code);
    	void RowDbClicked(const QString & code);
    
    public:
    	StockTableView(QStandardItemModel * model, QWidget * parent = 0);
    
    public:
    	void SetMouseHover(int, bool forceChanged = false);
    	void SetMouseChecked(int);
    	void SetDbClickedEnable(bool enable);
    
    	void SetHoverColor(const QColor & color);
    	void SetCheckedColor(const QColor & color);
    
    	void CheckedMoveUp();
    	void CheckedMoveDown();
    	void EnterPressed();
    
    protected:
        ...
    private:
        ...
    };
    

    代码中的接口都比较好理解,看名字应该都知道是干嘛的,这里就不做过多解释。

    b、表格初始化

    表格的数据内容在m_pListModel中存放,但是表格直接接收数据的是m_pFilterModel对象。

    m_pFilterModel对象可以理解为是一个映像数据源,他没有真正的去存储数据,他的数据都是来自m_pListModel类。

    //初始化搜索个股列表
    d_ptr->m_pStockPreview = new StockTableView(d_ptr->m_pListModel);
    d_ptr->m_pFilterModel = new StockSortFilterProxyModel;
    
    d_ptr->m_pPreviewError->setAlignment(Qt::AlignLeft | Qt::AlignVCenter);
    d_ptr->m_pPreviewError->setText(QStringLiteral("未搜索到相关股票"));
    
    d_ptr->m_pStockPreview->horizontalHeader()->setVisible(false);
    d_ptr->m_pStockPreview->verticalHeader()->setVisible(false);
    d_ptr->m_pStockPreview->setShowGrid(false);
    d_ptr->m_pStockPreview->horizontalHeader()->setStretchLastSection(true);
    d_ptr->m_pStockPreview->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
    
    d_ptr->m_pStockPreview->setMouseTracking(true);
    
    previewLayout->addWidget(d_ptr->m_pStockPreview);
    d_ptr->m_pStockPreviewWidget->setLayout(previewLayout);
    
    StockItemDelegate * itemDelegate = new StockItemDelegate(d_ptr->m_pStockPreview);
    d_ptr->m_pStockPreview->setItemDelegate(itemDelegate);
    itemDelegate->setView(d_ptr->m_pStockPreview);
    
    d_ptr->m_pPreviewError->setWindowFlags(Qt::WindowStaysOnTopHint | Qt::Tool | Qt::FramelessWindowHint);
    d_ptr->m_pStockPreviewWidget->setWindowFlags(Qt::WindowStaysOnTopHint | Qt::Tool | Qt::FramelessWindowHint);
    
    d_ptr->m_pFilterModel->setSourceModel(d_ptr->m_pListModel);
    d_ptr->m_pStockPreview->setModel(d_ptr->m_pFilterModel);
    d_ptr->m_pStockPreview->setColumnHidden(2, true);
    d_ptr->m_pStockPreview->setSortingEnabled(true);
    
    d_ptr->m_pPreviewError->setFixedSize(DropWidgetMaxWidth, 26);
    d_ptr->m_pStockPreviewWidget->setFixedWidth(DropWidgetMaxWidth);
    

    c、表格填充数据
    正常来说数据应该是网络上拉取的,但是这里作为测试,我直接添加了5行模拟数据

    void SelfStocksWidget::InitiAStock()
    {
    	std::vector<BaseStockInfoItem> sotckLists;
    	
    	BaseStockInfoItem item;
    	for (int i = 1; i <= 5; ++i)
    	{
    		item.wstrSymbol = QString("0h000%1").arg(i).toStdWString();
    		item.wstrName = QString("%1%1%1").arg(i).toStdWString();
    		item.wstrSymbol = QString("pingyin%1").arg(i).toStdWString();
    		sotckLists.push_back(item);
    	}
    
    	for each (BaseStockInfoItem stock in sotckLists)
    	{
    		QList<QStandardItem *> rows;
    		QStandardItem * symbol = new QStandardItem(QString::fromStdWString(stock.wstrSymbol).toUpper());
    		symbol->setData(QColor(28, 30, 34), Qt::BackgroundRole);
    		symbol->setData(QColor(204, 204, 204), Qt::ForegroundRole);
    		symbol->setSelectable(false);
    		rows << symbol;
    
    		QStandardItem * name = new QStandardItem(QString::fromStdWString(stock.wstrName));
    		name->setData(QColor(28, 30, 34), Qt::BackgroundRole);
    		name->setData(QColor(204, 204, 204), Qt::ForegroundRole);
    		name->setSelectable(false);
    		rows << name;
    
    		QStandardItem * pinyin = new QStandardItem(QString::fromStdWString(stock.wstrShortPinYin));
    		pinyin->setData(QColor(28, 30, 34), Qt::BackgroundRole);
    		pinyin->setData(QColor(204, 204, 204), Qt::ForegroundRole);
    		pinyin->setSelectable(false);
    		rows << pinyin;
    
    		//QStandardItem * type = new QStandardItem(QString::number(stock.m_stockType));
    		//type->setData(QColor(28, 30, 34), Qt::BackgroundRole);
    		//type->setData(QColor(204, 204, 204), Qt::ForegroundRole);
    		//type->setSelectable(false);
    		//rows << type;
    
    		d_ptr->m_pListModel->appendRow(rows);
    	}
    }
    

    最终的数据被填充到了m_pListModel数据源中。

    d、键盘操作

    文章开始的地方也说过了,我们的搜索预览框是支持键盘上下键来切换当前股票的,这个又是怎么完成的呢!

    预览框显示时,编辑框一直处于鼠标输入状态,并且具有键盘有限处理权限。

    因此里我们是取了个巧,把编辑框的事件挂载在了他的父窗体上,当键盘按下时,父窗口拿到键盘按下事件,首先转发给了预览框,让预览框去换一个最新的当前股票,并选中。

    代码如下所示,是不是也很简单。

    bool SelfStocksWidget::eventFilter(QObject * watched, QEvent * event)
    {
    	if (d_ptr->m_pSearchLineEdit == watched)
    	{
    		if (event->type() == QEvent::KeyPress)
    		{
    			if (QKeyEvent * keyEvent = static_cast<QKeyEvent *>(event))
    			{
    				switch (keyEvent->key())
    				{
    				case Qt::Key_Up:
    					d_ptr->m_pStockPreview->CheckedMoveUp();
    					break;
    				case Qt::Key_Down:
    					d_ptr->m_pStockPreview->CheckedMoveDown();
    					break;
    				case Qt::Key_Enter:
    				case Qt::Key_Return:
    					d_ptr->m_pStockPreview->EnterPressed();
    					break;
    				default:
    					break;
    				}
    			}
    		}
    	}
    	return __super::eventFilter(watched, event);
    }
    

    e、过滤

    前边也讲述过了,我们表格数据都是来自m_pFilterModel对象的,数据源中的数据m_pListModel基本没有发生变化过,及时我们现实的内容变化了,那也仅仅是m_pFilterModel对象过滤到的内容发生了变化。

    过滤接口Qt已经帮我们写好了,我们只需要实现其中的过滤方式即可。

    bool StockSortFilterProxyModel::filterAcceptsRow(int source_row
    											  , const QModelIndex & source_parent) const
    {
    	QRegExp regExp = filterRegExp();
    
    	if (regExp.isEmpty())
    	{
    		return true;
    	}
    
    	bool result = false;
    	for (int i = 0; i < sortColumn; ++i)
    	{
    		QModelIndex index = sourceModel()->index(source_row, i, source_parent);
    		QString context = sourceModel()->data(index).toString();
    
    		QString regExpStr = regExp.pattern();
    		result = regExp.exactMatch(context);
    
    		if (result)
    		{
    			break;
    		}
    	}
    
    	return result;
    }
    

    以上就是搜索股票编辑框的大致内容了,至于一些细微的设置,大家自行去完善即可。

    比如说预览框的窗口属性应该是这样的:

    setWindowFlags(Qt::WindowStaysOnTopHint | Qt::Tool | Qt::FramelessWindowHint);
    

    未输入任务内容时,编辑框的holderText应该是这样的:

    setPlaceholderText(QStringLiteral("搜索股票代码/名称"));
    

    由于篇幅原因,本篇文章就只先说搜索编辑框吧,本来想把自选股列表页一起加上,不过觉着内容太多,也不利于大家吸收,下一篇文章补上吧。。。

    写的手都酸了,其他内容自行脑补吧。。。

    四、相关文章

    财联社-产品展示

    Qt之自定义QLineEdit右键菜单

    qt捕获全局windows消息

    高仿富途牛牛-组件化(一)-支持页签拖拽、增删、小工具

    高仿富途牛牛-组件化(二)-磁力吸附

    高仿富途牛牛-组件化(三)-界面美化

    高仿富途牛牛-组件化(四)-优秀的时钟

    高仿富途牛牛-组件化(五)-如何去管理炒鸡多的小窗口

    高仿富途牛牛-组件化(六)-炒鸡牛逼的布局记忆功能(序列化和反序列化)


    如果您觉得文章不错,不妨给个打赏,写作不易,感谢各位的支持。您的支持是我最大的动力,谢谢!!!




    很重要--转载声明

    1. 本站文章无特别说明,皆为原创,版权所有,转载时请用链接的方式,给出原文出处。同时写上原作者:朝十晚八 or Twowords

    2. 如要转载,请原文转载,如在转载时修改本文,请事先告知,谢绝在转载时通过修改本文达到有利于转载者的目的。


  • 相关阅读:
    远程监控基础知识和故障排除
    感慨颇多
    物联网操作系统Hello China移植mile stone之一:移植基础版本V1.76发布
    iOS5系统API和5个开源库的JSON解析速度测试
    (译)iPhone: 用公开API创建带小数点的数字键盘 (OS 3.0, OS 4.0)
    [工具]Mac平台开发几个网络抓包工具(sniffer)
    [工具]Mac下非常好用的快捷终端Dterm
    【IOS】在SDK中打开其他接入应用的解决方案
    [开源]在iOS上实现Android风格的控件Toast
    [技巧]使用Xcode集成的HeaderDoc自动生成注释和开发文档
  • 原文地址:https://www.cnblogs.com/swarmbees/p/11154821.html
Copyright © 2011-2022 走看看