QGraphicsView 框架学习(四),Undo Redo

QGraphicsView 框架学习(四),Undo Redo

撤销,重做,任何编辑器的必备工具。还是Qt的提供的标准解决方案。

QUndoCommand;QUndoStack;QUndoView;

图形编辑的每一个动作都应该有相应的撤销和重做机制。我们为每个动作派生一个对象。

Qt demo里提供了两个例子,一个是使用QGraphic是View图形框架的,一个是Qt自己写的一个简单的图形编辑器。

这两个都可以参考。这里简单的描述一下我自己抄来的。

class MoveCommand : public QUndoCommand
{
public:
    MoveCommand(QGraphicsScene *graphicsScene, const QPointF & delta ,
                QUndoCommand * parent = 0);
    void undo() Q_DECL_OVERRIDE;
    void redo() Q_DECL_OVERRIDE;
private:
    QList<QGraphicsItem *> items;
    QGraphicsScene *myGraphicsScene;
    QPointF myDelta;
    bool bMoved;
};
MoveCommand::MoveCommand(QGraphicsScene *graphicsScene, const QPointF &delta, QUndoCommand *parent)
    : QUndoCommand(parent)
{
    items = graphicsScene->selectedItems();
    myDelta = delta;
    myGraphicsScene = graphicsScene;
    bMoved = true;
}
 
//! [2]
void MoveCommand::undo()
{
    foreach (QGraphicsItem *item, items) {
       item->moveBy(-myDelta.x(),-myDelta.y());
    }
    myGraphicsScene->update();
    setText(QObject::tr("Move %1,%2")
        .arg(-myDelta.x()).arg(-myDelta.y()));
    bMoved = false;
}
//! [2]
 
//! [3]
void MoveCommand::redo()
{
    if ( !bMoved ){
        foreach (QGraphicsItem *item, items) {
           item->moveBy(myDelta.x(),myDelta.y());
        }
        myGraphicsScene->update();
    }
    setText(QObject::tr("Move %1,%2")
        .arg(myDelta.x()).arg(myDelta.y()));
}

多个对象移动的动作是交给scene处理的,所以这里认为这个动作之前对象已经移动过了,在这个类里加了一个bMoved的标记, QUndoStack::push方法默认会调用redo方法,

所以第一次redo是不需要做的,除非做过undo后,再执行redo才能移动对象。
  我们可以创建一个QUndoView来观察 undoStack里的动作列表,不过发现如果在里面跳着执行,undostack会乱套,所以看看就好。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
SQLite3 是一个轻量级的数据库,它提供了一些基本的事务支持,但是并没有原生的 undo redo 功能。不过,我们可以通过自己编写代码来实现这些功能。 首先,我们需要创建一个表来保存历史记录,包括每个操作的 SQL 语句、操作时间等信息。例如: ```sql CREATE TABLE history ( id INTEGER PRIMARY KEY, sql TEXT, timestamp DATETIME DEFAULT CURRENT_TIMESTAMP ); ``` 然后,我们可以在每次执行 SQL 语句之前,将该语句插入到历史记录表中: ```python import sqlite3 conn = sqlite3.connect('example.db') cursor = conn.cursor() def execute(sql): cursor.execute(sql) conn.commit() cursor.execute("INSERT INTO history (sql) VALUES (?)", (sql,)) conn.commit() ``` 现在,我们可以实现 undoredo 功能。undo 操作可以通过删除最后一条历史记录,并执行该记录的相反操作来实现。redo 操作可以通过重新执行下一条历史记录来实现。例如: ```python def undo(): cursor.execute("SELECT * FROM history ORDER BY id DESC LIMIT 1") row = cursor.fetchone() if row: sql = row[1] cursor.execute(sql) cursor.execute("DELETE FROM history WHERE id=?", (row[0],)) conn.commit() def redo(): cursor.execute("SELECT * FROM history ORDER BY id ASC LIMIT 1") row = cursor.fetchone() if row: sql = row[1] cursor.execute(sql) cursor.execute("DELETE FROM history WHERE id=?", (row[0],)) conn.commit() ``` 这样,我们就可以实现基本的 undo redo 功能了。当然,这只是一个简单的示例,实际情况可能更加复杂,需要根据具体需求进行调整。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值