Qt之QListWidget实战

介绍

QListWidget是一个与QListView类似的列表视图类。
作为Qt常用的视图类,它提供了很多简便的函数,当项大于窗口高度时,会自动弹出滑动条。方便对列表中的项进行展示和操作。


1、以下是QListWidget的

添加项、删除项、清空列表、
重命名,右键菜单弹出,双击事件

等功能的实现。

重命名效果

所有代码

头文件

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QListWidget>
#include <QLineEdit>

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();

public slots:
    void addListItem();
    void deleteListItem();
    void clearListItem();
    void changeListItemName();
    void finishEdit();
    void onCustomContextMenuRequested(const QPoint &pos);
    void onListWidgetItemDblClicked(QListWidgetItem *item);

private:
    int getCurrentIndexItem(QListWidgetItem * item);
    void startEdit(QListWidgetItem * item);
    bool checkItemSameName(const QString &name);

private:
    QListWidget *m_listWidget;
    QLineEdit *m_renameEdit;
    int m_cnt;
};
#endif // WIDGET_H

cpp文件

#include "widget.h"
#include <QMenu>
#include <QAction>
#include <QVBoxLayout>

Widget::Widget(QWidget *parent)
    : QWidget(parent)
{
    m_listWidget = new QListWidget;
    m_renameEdit = new QLineEdit(m_listWidget);
    m_renameEdit->hide();

    QVBoxLayout *vLayout = new QVBoxLayout;

    vLayout->setSpacing(0);
    vLayout->setMargin(0);
    m_listWidget->setContextMenuPolicy(Qt::CustomContextMenu);;
    vLayout->addWidget(m_listWidget);
    setLayout(vLayout);
    connect(m_listWidget, SIGNAL(itemDoubleClicked(QListWidgetItem *)),
            this, SLOT(onListWidgetItemDblClicked(QListWidgetItem *)));
    connect(m_renameEdit, SIGNAL(editingFinished()), this, SLOT(finishEdit()));
    connect(m_listWidget, SIGNAL(customContextMenuRequested(const QPoint &)),
            this, SLOT(onCustomContextMenuRequested(const QPoint &)));
    this->setMinimumSize(300,500);
}

Widget::~Widget()
{
}

void Widget::addListItem()
{
    QListWidgetItem *pItem = new QListWidgetItem;
    if (pItem) {
        m_cnt++;
        pItem->setText(QString::number(m_cnt));
         m_listWidget->addItem(pItem);
         m_listWidget->setCurrentItem(pItem);
    }
}
void Widget::deleteListItem()
{
    QListWidgetItem *curItem =  m_listWidget->currentItem();

    delete curItem;
    curItem = nullptr;
}
void Widget::clearListItem()
{
    m_cnt = 0;
    m_listWidget->clear();
     m_renameEdit->hide();
}
void Widget::changeListItemName()
{
    QListWidgetItem *curItem =  m_listWidget->currentItem();
    if (curItem)
    {
         m_renameEdit->setText(curItem->text());
    }
     m_renameEdit->show();
     m_renameEdit->selectAll();
     m_renameEdit->setFocus();
}
void Widget::finishEdit()
{
    QListWidgetItem *curItem =  m_listWidget->currentItem();
    QString oldName = curItem->text();
    if (!checkItemSameName(oldName))
    {
        curItem->setText( m_renameEdit->text());
         m_renameEdit->hide();
    }
    else if ( m_renameEdit->text() == oldName)
    {
         m_renameEdit->hide();
        return;
    }
    else
    {
        curItem->setText(oldName);
        if ( m_renameEdit->isVisible())
        {
             m_renameEdit->hide();
        }
    }
}
void Widget::onCustomContextMenuRequested(const QPoint &pos)
{
    QListWidgetItem *item =  m_listWidget->itemAt(pos);
    if (!item)
    {
        //action
        QMenu *menu = new QMenu(this);
        QAction *newAct = new QAction(tr("New"), this);
        menu->addAction(newAct);

        QAction *clearAct = new QAction(tr("Clear"), this);
        menu->addAction(clearAct);

        connect(clearAct, SIGNAL(triggered()), this, SLOT(clearListItem()));
        connect(newAct, SIGNAL(triggered()), this, SLOT(addListItem()));
        menu->exec(QCursor::pos());
        return;
    }

    startEdit(item);

    QMenu *menu = new QMenu(this);
    QAction *deleteAct = new QAction("Delete", this);
    menu->addAction(deleteAct);

    QAction *renameAct = new QAction("Rename", this);
    menu->addAction(renameAct);

    connect(deleteAct, SIGNAL(triggered()), this, SLOT(deleteListItem()));
    connect(renameAct, SIGNAL(triggered()), this, SLOT(changeListItemName()));

    menu->exec(QCursor::pos());
}

void Widget::onListWidgetItemDblClicked(QListWidgetItem *item)
{
    startEdit(item);
    changeListItemName();
}

int Widget::getCurrentIndexItem(QListWidgetItem * item)
{
    int curIndex = 0;
    for (int i = 0; i <  m_listWidget->count(); i++) {
        if (item == m_listWidget->item(i))
        {
            curIndex = i;
            break;
        }
    }
    return curIndex;
}
void Widget::startEdit(QListWidgetItem * item)
{
    if (nullptr == item) return;

    QRect rect =  m_listWidget->visualItemRect(item);
     m_renameEdit->setFixedHeight(rect.height());
     m_renameEdit->move(rect.x(), rect.y());
}
bool Widget::checkItemSameName(const QString &name)
{
    bool bIsSame = false;
    for (int i = 0; i <  m_listWidget->count(); i++)
    {
        if ( m_listWidget->item(i)->text() == name)
        {
            bIsSame = true;
            break;
        }
    }
    return bIsSame;
}

通过以上,可以扩展对应的需求,如在列表项贴图,显示视频,显示动图等。

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 如何在Qt的QListWidget上进行下移动? 在Qt的QListWidget上进行下移动,可以使用QListWidgetItem的下移方法,将选中的项目向下移动一个位置。具体实现可以参考以下代码: ``` QListWidgetItem* selectedItem = ui->listWidget->currentItem(); //获取当前选中的项目 int currentIndex = ui->listWidget->row(selectedItem); //获取当前选中项目的行号 if (currentIndex < ui->listWidget->count() - 1) { //如果当前项目不是最后一个 QListWidgetItem* nextItem = ui->listWidget->item(currentIndex + 1); //获取下一个项目 ui->listWidget->takeItem(currentIndex); //将当前项目从列表中移除 ui->listWidget->insertItem(currentIndex + 1, selectedItem); //将当前项目插入到下一个项目的位置 ui->listWidget->setCurrentItem(selectedItem); //将当前项目设为选中状态 } ``` 如果需要实现全部选中项目下移一个位置的功能,也可以使用QListWidget的selectedItems()方法获取当前选中的项目列表,逐个进行下移操作。 ### 回答2: Qt中QListWidget提供了一种更加灵活,方便的管理item的方式,可以在这个控件上增加、删除、重新排序各种item,本文将针对QListWidget中item的上下移动进行简要说明。 首先,QListWidget的item是可以被拖拽的,也就是可以通过鼠标左键的拖拽操作进行重新排序。但这只是一种视觉上的操作,不会真正影响列表中item的位置,如果需要影响item的位置,需要编写代码实现。 当我们需要通过程序实现上下移动item时,可以通过QListWidget的成员函数进行操作,其调用格式为:`QListWidgetItem *QListWidget::takeItem(int row)`,返回指定row的item,并将其从列表中移除,然后可通过`void QListWidget::insertItem(int row, QListWidgetItem *item)`插入到指定位置。 在这个基础上,我们可以再编写一些简单的代码实现item的上下移动。 例如,当用户按下“上移”按钮时,我们可以将被选中的item向上移动一个位置,具体实现如下: ```c++ QListWidgetItem *item = ui->listWidget->currentItem(); // 获取被选中item if(item) // 如果有选中item { int row = ui->listWidget->row(item); // 获取其位置 if(row > 0) // 如果不是第一个 { ui->listWidget->takeItem(row); // 移除item ui->listWidget->insertItem(row - 1, item); // 插入到前一个位置 ui->listWidget->setCurrentItem(item); // 选中移动后的item } } ``` 同样,我们也可以编写“下移”按钮的操作,只需将row - 1改为row + 1即可。 当然,这只是一种简单的实现方式,如果需要更加高效、流畅的操作,可以使用QListWidget的其他接口或自定义item来实现。 总的来说,Qt中QListWidget的上下移动操作并不难实现,只需要理解其基本原理和API接口即可。 ### 回答3: 在Qt的QListWidget中移动item的操作实现比较方便,可以使用QListWidget自带的API来完成。主要的方法有两个:setCurrentRow()和currentRow()。setCurrentRow()用于设置当前选中的item,currentRow()用于获取当前选中的item的行号。 1. 上下移动的实现 上下移动item一般有两种方式:通过鼠标拖拽或者通过键盘上下键。 通过鼠标拖拽移动item需要重载QListWidget的dragEnterEvent()、dragMoveEvent()和dropEvent()三个方法,这里不多做介绍。 通过键盘上下键移动item,可以响应键盘事件,然后根据按键移动当前选中的item即可。以下是一个简单的示例代码: ```cpp void Widget::keyPressEvent(QKeyEvent *event) { int row = listWidget->currentRow(); if (event->key() == Qt::Key_Up) { if (row > 0) { listWidget->setCurrentRow(row - 1); } } else if (event->key() == Qt::Key_Down) { if (row < listWidget->count() - 1) { listWidget->setCurrentRow(row + 1); } } } ``` 在这个示例中,通过判断按下的键是否是上下键,然后根据当前选中的item的行号计算出要移动到的item的行号,最后调用setCurrentRow()方法设置选中的item即可。 2. 移动item的效果 默认情况下,QListWidget中上下移动item只会改变item的位置,不会改变item的外观效果。如果需要给移动item添加一些动画效果,可以通过QPropertyAnimation类来实现。以下是一个简单的示例代码: ```cpp void Widget::keyPressEvent(QKeyEvent *event) { int row = listWidget->currentRow(); if (event->key() == Qt::Key_Up) { if (row > 0) { listWidget->setCurrentRow(row - 1); QPropertyAnimation *animation = new QPropertyAnimation(listWidget->item(row), "pos"); animation->setDuration(300); animation->setEasingCurve(QEasingCurve::OutQuad); animation->setStartValue(QPoint(0, (row + 1) * listWidget->sizeHintForRow(0))); animation->setEndValue(QPoint(0, row * listWidget->sizeHintForRow(0))); animation->start(QAbstractAnimation::DeleteWhenStopped); } } else if (event->key() == Qt::Key_Down) { if (row < listWidget->count() - 1) { listWidget->setCurrentRow(row + 1); QPropertyAnimation *animation = new QPropertyAnimation(listWidget->item(row), "pos"); animation->setDuration(300); animation->setEasingCurve(QEasingCurve::OutQuad); animation->setStartValue(QPoint(0, (row - 1) * listWidget->sizeHintForRow(0))); animation->setEndValue(QPoint(0, row * listWidget->sizeHintForRow(0))); animation->start(QAbstractAnimation::DeleteWhenStopped); } } } ``` 在这个示例中,通过QPropertyAnimation类对item的位置进行动画变化,使移动item的过程更加流畅。其中,使用了OutQuad的缓动函数,可以使移动速度呈现加速度效果。具体的实现过程可以根据需求进行调整。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值