《QT实用小工具·五十八》模仿VSCode的可任意拖拽的Tab标签组

2024-05-06 23:20

本文主要是介绍《QT实用小工具·五十八》模仿VSCode的可任意拖拽的Tab标签组,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、概述
源码放在文章末尾

该项目实现了模仿VSCode的可任意拖拽的Tab标签组,包含如下功能:

拖拽标签页至新窗口
拖拽标签页合并控件
无限嵌套的横纵分割布局(类似Qt Creator的编辑框)
获取当前使用的标签组、标签页
自动向上合并标签组
左右拖拽排序(Qt自带)

下面是demo演示:
在这里插入图片描述
项目部分代码如下所示:


#ifndef DRAGABLETABGROUP_H
#define DRAGABLETABGROUP_H#include <QObject>
#include <QTabWidget>
#include <QMimeData>
#include <QDrag>
#include <QScreen>
#include <QApplication>
#include <QHBoxLayout>
#include <QTimer>
#include <QDesktopWidget>
#include <QStyle>
#include <QJsonObject>
#include <QJsonDocument>
#include <QJsonArray>
#include "dragabletabbar.h"#define DRAGABLE_TAB_WINDOW_MIME_KEY "DRAGABLE_TAB_WINDOW_MIME_KEY"
#define DRAGABLE_TAB_WIDGET_MIME_KEY "DRAGABLE_TAB_WIGET_MIME_KEY"
#define DRAGABLE_TAB_LABEL_MIME_KEY "DRAGABLE_TAB_LABEL_MIME_KEY"
#define DRAGABLE_TAB_ICON_MIME_KEY "DRAGABLE_TAB_ICON_MIME_KEY"#define WIN_FRAME_LEFE_OFFSET 8 // 移动时左边需要偏差的像素;可能是左边的阴影?#define DRAG_DEB if (0) qDebug()struct TabPageBean
{TabPageBean(QString label, QWidget* widget): label(label), widget(widget){}TabPageBean(QIcon icon, QString label, QWidget* widget): icon(icon), label(label), widget(widget){}QIcon icon;QString label;QWidget* widget;
};class DragableTabGroup : public QTabWidget
{Q_OBJECTfriend class DragableTabArea;
public:DragableTabGroup(QWidget* parent = nullptr);virtual void tabInserted(int index) override;void removeTab(int index);void split(QBoxLayout::Direction direction, bool copy = true);bool isFocusing();bool hasTab(QWidget* widget);void deleteTab(int index);void deleteAllWidget();void removeTabs(QList<int> indexes, bool del = false);void removeTabs(int start, int end, bool del = false);void deleteIfEmptyWindow();bool isInMain();bool isRestoreTabEnabled();bool canRestoreTab();virtual QJsonObject toJson();virtual QJsonObject widgetToJson(QWidget* widget);virtual void jsonToWidget(QJsonObject object);public slots:virtual void closeLeftTabs(int index = -1);virtual void closeRightTabs(int index = -1);virtual void closeOtherTabs(int index = -1);virtual void closeAllTabs();virtual void restoreClosedTab(int index = -1);protected:void dragEnterEvent(QDragEnterEvent *event) override;void dragMoveEvent(QDragMoveEvent *event) override;void dropEvent(QDropEvent *event) override;virtual void tabWidgetInserted(int index, QWidget *widget);virtual void tabWidgetRemoved(int index, QWidget *widget);virtual void tabWidgetFocused(int index, QWidget *widget);virtual DragableTabBar* newTabBar(QWidget* parent = nullptr);virtual DragableTabGroup* newTabGroup(QWidget* parent = nullptr);virtual void tabDraggingEvent(QWidget* widget);virtual void tabDragMergedEvent(QWidget* widget);virtual void tabDragWindowedEvent(DragableTabGroup* window, QWidget* widget);virtual void tabDropMergedEvent(QWidget* widget);protected slots:void moveTabInNewWindow(int index);void slotStartDrag(int index);DragableTabGroup *createDraggedNewWindow();bool mergeDroppedLabel(QDropEvent* event);signals:void signalNewTabWindowCreated(DragableTabGroup* window);void signalWidgetFocused(QWidget* widget);void signalSplitCurrentTab(QBoxLayout::Direction direction, bool copy);void signalTabDragged(QWidget* widget); // 开始拖拽void signalTabMerged(QWidget* widget); // 被拖至其他标签组void signalTabWindowed(DragableTabGroup* window, QWidget* widget); // 被拖出来形成新窗口void signalTabDropped(QWidget* widget); // 合并来自其他标签组的标签void signalJsonToWidget(QJsonObject object);protected:DragableTabBar* tab_bar;int dragging_index;QWidget* dragging_widget;static QString dragging_label;static QIcon dragging_icon;QPoint dragging_point_delta; // 拖拽的 鼠标-子窗口左上角bool _is_main; // 是不是主窗口static bool _drag_merged;bool frameless = false; // 使用无窗口边框static int closed_stack_max; // 已关闭标签页的最大数量static QList<TabPageBean> closed_stack; // 已关闭的标签页
};#endif // DRAGABLETABGROUP_H

#include "dragabletabgroup.h"bool DragableTabGroup::_drag_merged = false;
QString DragableTabGroup::dragging_label;
QIcon DragableTabGroup::dragging_icon;
int DragableTabGroup::closed_stack_max = 20;
QList<TabPageBean> DragableTabGroup::closed_stack;DragableTabGroup::DragableTabGroup(QWidget *parent): QTabWidget(parent), tab_bar(newTabBar(this)),dragging_index(0), dragging_widget(nullptr), _is_main(false)
{setAcceptDrops(true);setTabBar(tab_bar);setMovable(true);//    setTabsClosable(true);//    tab_bar->setAutoHide(true);// 无窗口,看起来更像浏览器的if (frameless){setWindowFlag(Qt::FramelessWindowHint, true);}connect(tab_bar, SIGNAL(signalStartDrag(int)), this, SLOT(slotStartDrag(int)));connect(tab_bar, SIGNAL(signalEndDrag()), this, SLOT(createDraggedNewWindow()));connect(qApp, &QApplication::focusChanged, this, [=](QWidget *, QWidget *now) {int index = indexOf(now);if (index > -1){tabWidgetFocused(index, now);}});
}void DragableTabGroup::tabInserted(int index)
{auto widget = QTabWidget::widget(index);if (widget)tabWidgetInserted(index, widget);
}/*** 替换自带的removeTab方法* 所有的移除都可以在这里监听到*/
void DragableTabGroup::removeTab(int index)
{if (index < 0 || index >= count())return;if (isRestoreTabEnabled()){closed_stack.append(TabPageBean(tabIcon(index),tabText(index),widget(index)));if (closed_stack.count() > closed_stack_max){closed_stack.takeFirst().widget->deleteLater();}}tabWidgetRemoved(index, this->widget(index));QTabWidget::removeTab(index);
}/*** 分割标签组* @param direction 方向* @param copy      是否复制(默认复制,保留旧tab)*/
void DragableTabGroup::split(QBoxLayout::Direction direction, bool copy)
{emit signalSplitCurrentTab(direction, copy);
}/*** 是否拥有焦点*/
bool DragableTabGroup::isFocusing()
{QWidget *widget = QApplication::focusWidget();return widget != nullptr && hasTab(widget);
}/*** 是否包含某一widget*/
bool DragableTabGroup::hasTab(QWidget *widget)
{return indexOf(widget) > -1;
}void DragableTabGroup::deleteTab(int index)
{QWidget *widget = this->widget(index);removeTab(index);widget->deleteLater();
}/*** 移除tab,并删除所有控件*/
void DragableTabGroup::deleteAllWidget()
{while (this->count()){auto widget = this->widget(0);widget->deleteLater();removeTab(0);}
}/*** 移除或删除多个标签页*/
void DragableTabGroup::removeTabs(QList<int> indexes, bool del)
{// 排好序std::sort(indexes.begin(), indexes.end(), [=](int a, int b) {return a < b;});for (int i = indexes.size() - 1; i >= 0; i--){if (del)removeTab(i);elsedeleteTab(i);}
}/*** 移除或者删除指定范围内的标签页*/
void DragableTabGroup::removeTabs(int start, int end, bool del)
{QList<int> indexes;if (start == -1)start = 0;else if (end == -1)end = this->count() - 1;for (int i = start; i <= end; i++){indexes << i;}removeTabs(indexes, del);
}/*** 判断自己的数量是不是空的* 如果是空窗口,则主动删除*/
void DragableTabGroup::deleteIfEmptyWindow()
{if (!_is_main && count() == 0)deleteLater();
}/*** 判断标签组是否在主窗口中*/
bool DragableTabGroup::isInMain()
{return parentWidget() != nullptr;
}bool DragableTabGroup::isRestoreTabEnabled()
{return closed_stack_max > 0;
}/*** 能否恢复已关闭的标签页*/
bool DragableTabGroup::canRestoreTab()
{return closed_stack.size() > 0;
}QJsonObject DragableTabGroup::toJson()
{QJsonObject object;object.insert("width", this->width());object.insert("height", this->height());QJsonArray array;for (int i = 0; i < this->count(); i++){QWidget *widget = this->widget(i);QJsonObject obj = widgetToJson(widget);array.append(obj);}object.insert("tabs", array);object.insert("current", this->currentIndex());return object;
}QJsonObject DragableTabGroup::widgetToJson(QWidget *widget)
{Q_UNUSED(widget)return QJsonObject();
}/*** 从JSON恢复标签组的控件* 有两种方式:* - TabGroup中继承,添加widget* - TabsArea中在group中添加widget*/
void DragableTabGroup::jsonToWidget(QJsonObject object)
{emit signalJsonToWidget(object);
}void DragableTabGroup::closeLeftTabs(int index)
{if (index == -1)index = currentIndex();removeTabs(-1, index - 1);
}void DragableTabGroup::closeRightTabs(int index)
{if (index == -1)index = currentIndex();removeTabs(index + 1, -1);
}void DragableTabGroup::closeOtherTabs(int index)
{if (index == -1)index = currentIndex();QList<int> indexes;for (int i = 0; i < this->count(); i++)indexes << i;indexes.removeOne(index);removeTabs(indexes);
}void DragableTabGroup::closeAllTabs()
{QList<int> indexes;for (int i = 0; i < this->count(); i++)indexes << i;removeTabs(indexes);deleteIfEmptyWindow();
}void DragableTabGroup::restoreClosedTab(int index)
{if (!closed_stack.size())return;TabPageBean page = closed_stack.takeLast();if (index == -1)addTab(page.widget, page.icon, page.label);elseinsertTab(index, page.widget, page.icon, page.label);setCurrentWidget(page.widget);
}void DragableTabGroup::dragEnterEvent(QDragEnterEvent *event)
{const QMimeData *mime = event->mimeData();if (mime->hasFormat(DRAGABLE_TAB_WIDGET_MIME_KEY)) // Tab拖拽{event->accept();}return QTabWidget::dragEnterEvent(event);
}void DragableTabGroup::dragMoveEvent(QDragMoveEvent *event)
{const QMimeData *mime = event->mimeData();if (count() == 0){event->accept();}else if (mime->hasFormat(DRAGABLE_TAB_WIDGET_MIME_KEY)) // 整行拖拽{QPoint pos = event->pos();if (pos.y() >= tab_bar->geometry().top() - qMax(tab_bar->height(), 32) && pos.y() <= tab_bar->geometry().bottom() + qMax(tab_bar->height(), 32)) // 只有 tabBar 的位置可拖拽{event->accept();}else{event->ignore();}}else{return QTabWidget::dragMoveEvent(event);}
}void DragableTabGroup::dropEvent(QDropEvent *event)
{if (mergeDroppedLabel(event))event->accept();return QTabWidget::dropEvent(event);
}/*** 任何Tab的添加都会触发这个方法* 包括拖动tab到此页面* 在插入之后调用*/
void DragableTabGroup::tabWidgetInserted(int index, QWidget *widget)
{
}/*** 任何tab的删除都会触发这个方法* 包括拖拽tab出去* 在删除之前调用*/
void DragableTabGroup::tabWidgetRemoved(int index, QWidget *widget)
{if (this->count() <= 1) // 全部移除完了,则删除当前的TabGroup{this->deleteLater();}
}/*** 标签控件获得焦点事件*/
void DragableTabGroup::tabWidgetFocused(int index, QWidget *widget)
{
}/*** 创建一个自定义TabBar* 注意:实际上,这个虚函数不会被覆盖* 因为是在构造函数中调用,此时不会调用子类重写的方法*/
DragableTabBar *DragableTabGroup::newTabBar(QWidget *parent)
{return new DragableTabBar(parent);
}/*** 创建一个标签组的新方法* 如果需要继承的话,则需要把所有创建的地方都覆盖掉*/
DragableTabGroup *DragableTabGroup::newTabGroup(QWidget *parent)
{return new DragableTabGroup(parent);
}/*** 标签被拖拽事件* widget所在的tab还在自己这儿*/
void DragableTabGroup::tabDraggingEvent(QWidget *widget)
{
}/*** 本标签组的标签被其他标签组合并* 此时的widget所在的tab已经不是自己的了* 如果是独立窗口的唯一标签被合并,那么合并后标签组自动删除,不会触发此方法* (另外……这个方法会导致崩溃,已经注释掉了,永远不会触发)*/
void DragableTabGroup::tabDragMergedEvent(QWidget *widget)
{
}/*** 本标签组的标签拖动至新的窗口* tab也已经不在自己这儿了*/
void DragableTabGroup::tabDragWindowedEvent(DragableTabGroup *window, QWidget *widget)
{
}/*** 标签拖到上面事件* 在合并到自己之后*/
void DragableTabGroup::tabDropMergedEvent(QWidget *widget)
{
}/*** 移动标签到新的窗口* 其实是相当于触发拖拽事件一样的*/
void DragableTabGroup::moveTabInNewWindow(int index)
{dragging_index = index;dragging_widget = this->widget(index);dragging_point_delta = QCursor::pos() - (parent() == nullptr ? tabBar()->mapToGlobal(tabBar()->pos()) : mapToGlobal(QPoint(0, 0)));createDraggedNewWindow();
}/*** 开始标签拖拽*/
void DragableTabGroup::slotStartDrag(int index)
{dragging_index = index;dragging_widget = this->widget(index);dragging_point_delta = QCursor::pos() - (parent() == nullptr ? tabBar()->mapToGlobal(tabBar()->pos()) : mapToGlobal(QPoint(0, 0)));tabDraggingEvent(dragging_widget);emit signalTabDragged(dragging_widget);_drag_merged = false;QPixmap pixmap(this->size());pixmap.fill(Qt::transparent);//    dragging_widget->render(&pixmap, dragging_widget->mapToGlobal(pos()) - this->mapToGlobal(pos()));this->render(&pixmap, this->mapToGlobal(pos()) - this->mapToGlobal(pos()));QMimeData *mime = new QMimeData;mime->setData(DRAGABLE_TAB_WINDOW_MIME_KEY, QString::number(reinterpret_cast<int>(this)).toUtf8());mime->setData(DRAGABLE_TAB_WIDGET_MIME_KEY, QString::number(reinterpret_cast<int>(dragging_widget)).toUtf8());mime->setData(DRAGABLE_TAB_LABEL_MIME_KEY, tab_bar->tabText(index).toLocal8Bit());dragging_label = tab_bar->tabText(index);dragging_icon = tab_bar->tabIcon(index);QDrag *drag = new QDrag(this);drag->setMimeData(mime);drag->setPixmap(pixmap);drag->setHotSpot(dragging_point_delta);bool is_one = (count() == 1 && !isInMain());connect(drag, &QDrag::destroyed, this, [=](QObject *) {DRAG_DEB << "destroyed";// 顺序:先触发 dropEvent,在 drag::destroyed// 判断有没有被合并到窗口if (_drag_merged){DRAG_DEB << "drag 合并标签";// 判断是不是唯一标签的窗口// 如果单窗口只有这一个标签,那么拖动结束后自己也删除了,后面的代码并没有什么意义if (!is_one){// tabDragMergedEvent(dragging_widget); // 这句话会导致莫名的崩溃emit signalTabMerged(dragging_widget);}return;}// 没有合并到其他窗口if (this->count() == 1 && !isInMain()){// 单个标签拖动,移动窗口int titlebar_height = style()->pixelMetric(QStyle::PM_TitleBarHeight);this->move(QCursor::pos() - dragging_point_delta - QPoint(WIN_FRAME_LEFE_OFFSET, titlebar_height));DRAG_DEB << "drag 移动窗口";}else{DRAG_DEB << "drag 创新窗口";// 多个标签拖出,创建新窗口createDraggedNewWindow();}});// 如果只有一个标签,则假装移动整个窗口if (this->count() == 1){QTimer::singleShot(0, [=] {//            this->hide();this->move(-30000, -30000); // 隐藏起来});}DRAG_DEB << "----------开始drag--------";// exec 操作会一直阻塞后面的代码,除非使用多线程或者信号槽drag->exec();
}/*** 自己的标签拖出到新窗口*/
DragableTabGroup *DragableTabGroup::createDraggedNewWindow()
{if (count() == 1) // 只有一个标签,直接移动窗口{// 会导致没有 update,第一次按下无法操作,已取消// move(QCursor::pos()-dragging_point_delta-QPoint(0,tab_bar->height()));// return ;}int titlebar_height = style()->pixelMetric(QStyle::PM_TitleBarHeight);DragableTabGroup *window = newTabGroup(nullptr /*_is_main ? this : this->parentWidget()*/);window->setAttribute(Qt::WA_DeleteOnClose, true);window->resize(this->size());window->move(QCursor::pos() - dragging_point_delta - QPoint(WIN_FRAME_LEFE_OFFSET, titlebar_height));window->show();QString label = tab_bar->tabText(dragging_index);QIcon icon = tab_bar->tabIcon(dragging_index);removeTab(dragging_index);if (!icon.isNull())window->addTab(dragging_widget, icon, label);elsewindow->addTab(dragging_widget, label);emit signalNewTabWindowCreated(window);if (!_is_main && count() == 0) // 标签拖完了deleteLater();window->raise();window->setFocus();dragging_widget->setFocus();dragging_widget->setFocus();QTimer::singleShot(0, dragging_widget, [=] {// 为啥要延迟……window->setFocus();});tabDragWindowedEvent(window, dragging_widget);emit signalTabWindowed(window, dragging_widget);return window;
}/*** 另一个窗口拖拽本窗口的tabbar,合并标签*/
bool DragableTabGroup::mergeDroppedLabel(QDropEvent *event)
{int insert_index = count();// 根据鼠标的位置判断插入的位置for (int i = count() - 1; i >= 0; i--){if (tab_bar->tabRect(i).center().x() + tab_bar->pos().x() >= event->pos().x())insert_index = i;}// 被拖拽的信息const QMimeData *mime = event->mimeData();DragableTabGroup *window = reinterpret_cast<DragableTabGroup *>(mime->data(DRAGABLE_TAB_WINDOW_MIME_KEY).toInt());QWidget *widget = reinterpret_cast<QWidget *>(mime->data(DRAGABLE_TAB_WIDGET_MIME_KEY).toInt());QString label = QString::fromLocal8Bit(mime->data(DRAGABLE_TAB_LABEL_MIME_KEY));QIcon icon = dragging_icon;if (window == this) // 被拖拽的就是自己{if (insert_index == currentIndex()) // 根本就没有被拖动return false;// 交换标签顺序removeTab(currentIndex());insertTab(insert_index, widget, label);setCurrentIndex(insert_index);return false;}// 可以拖拽合并_drag_merged = true;// 移除旧的window->removeTab(window->currentIndex());window->deleteIfEmptyWindow(); // 标签拖完了(标签移除事件中也会自己删除)// 插入新的if (icon.isNull()){if (insert_index >= count()) // 加到末尾addTab(widget, label);elseinsertTab(insert_index, widget, label);}else{if (insert_index >= count()) // 加到末尾addTab(widget, icon, label);elseinsertTab(insert_index, widget, icon, label);}setCurrentIndex(insert_index);//    this->raise(); // 如果用frameless,raise会一直生效,导致界面会被挡住……this->setFocus();QTimer::singleShot(0, this, [=] {widget->setFocus();});tabDropMergedEvent(widget);emit signalTabDropped(widget);return true;
}

源码下载

这篇关于《QT实用小工具·五十八》模仿VSCode的可任意拖拽的Tab标签组的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/965667

相关文章

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

高效录音转文字:2024年四大工具精选!

在快节奏的工作生活中,能够快速将录音转换成文字是一项非常实用的能力。特别是在需要记录会议纪要、讲座内容或者是采访素材的时候,一款优秀的在线录音转文字工具能派上大用场。以下推荐几个好用的录音转文字工具! 365在线转文字 直达链接:https://www.pdf365.cn/ 365在线转文字是一款提供在线录音转文字服务的工具,它以其高效、便捷的特点受到用户的青睐。用户无需下载安装任何软件,只

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

EMLOG程序单页友链和标签增加美化

单页友联效果图: 标签页面效果图: 源码介绍 EMLOG单页友情链接和TAG标签,友链单页文件代码main{width: 58%;是设置宽度 自己把设置成与您的网站宽度一样,如果自适应就填写100%,TAG文件不用修改 安装方法:把Links.php和tag.php上传到网站根目录即可,访问 域名/Links.php、域名/tag.php 所有模板适用,代码就不粘贴出来,已经打

vscode中文乱码问题,注释,终端,调试乱码一劳永逸版

忘记咋回事突然出现了乱码问题,很多方法都试了,注释乱码解决了,终端又乱码,调试窗口也乱码,最后经过本人不懈努力,终于全部解决了,现在分享给大家我的方法。 乱码的原因是各个地方用的编码格式不统一,所以把他们设成统一的utf8. 1.电脑的编码格式 开始-设置-时间和语言-语言和区域 管理语言设置-更改系统区域设置-勾选Bata版:使用utf8-确定-然后按指示重启 2.vscode

超强的截图工具:PixPin

你是否还在为寻找一款功能强大、操作简便的截图工具而烦恼?市面上那么多工具,常常让人无从选择。今天,想给大家安利一款神器——PixPin,一款真正解放双手的截图工具。 想象一下,你只需要按下快捷键就能轻松完成多种截图任务,还能快速编辑、标注甚至保存多种格式的图片。这款工具能满足这些需求吗? PixPin不仅支持全屏、窗口、区域截图等基础功能,它还可以进行延时截图,让你捕捉到每个关键画面。不仅如此

【QT】基础入门学习

文章目录 浅析Qt应用程序的主函数使用qDebug()函数常用快捷键Qt 编码风格信号槽连接模型实现方案 信号和槽的工作机制Qt对象树机制 浅析Qt应用程序的主函数 #include "mywindow.h"#include <QApplication>// 程序的入口int main(int argc, char *argv[]){// argc是命令行参数个数,argv是

PR曲线——一个更敏感的性能评估工具

在不均衡数据集的情况下,精确率-召回率(Precision-Recall, PR)曲线是一种非常有用的工具,因为它提供了比传统的ROC曲线更准确的性能评估。以下是PR曲线在不均衡数据情况下的一些作用: 关注少数类:在不均衡数据集中,少数类的样本数量远少于多数类。PR曲线通过关注少数类(通常是正类)的性能来弥补这一点,因为它直接评估模型在识别正类方面的能力。 精确率与召回率的平衡:精确率(Pr

Python QT实现A-star寻路算法

目录 1、界面使用方法 2、注意事项 3、补充说明 用Qt5搭建一个图形化测试寻路算法的测试环境。 1、界面使用方法 设定起点: 鼠标左键双击,设定红色的起点。左键双击设定起点,用红色标记。 设定终点: 鼠标右键双击,设定蓝色的终点。右键双击设定终点,用蓝色标记。 设置障碍点: 鼠标左键或者右键按着不放,拖动可以设置黑色的障碍点。按住左键或右键并拖动,设置一系列黑色障碍点