From 11ad707b9509b709e3043a6aa1f19a2871e3bf91 Mon Sep 17 00:00:00 2001 From: arch1t3cht Date: Sun, 9 Apr 2023 02:14:40 +0200 Subject: [PATCH] Add basic undo/redo functionality This works by simply keeping backups the entire relevant state of the Wobbly project after each action and rolling back to such states on undo/redo. This is feasible since Wobbly's internal state is not very memory-intensive. A drawback is that all list or table models have to be repopulated from scratch, so selections in dialogs are cleared on every undo/redo. This could be improved later on by saving the UI state before a rollback and restoring it after it is completed. Fixes dubhater/Wobbly#15 . --- src/shared/BookmarksModel.cpp | 8 ++ src/shared/BookmarksModel.h | 4 +- src/shared/CombedFramesModel.h | 2 +- src/shared/CustomListsModel.cpp | 9 ++ src/shared/CustomListsModel.h | 4 +- src/shared/FrameRangesModel.h | 2 +- src/shared/FrozenFramesModel.cpp | 8 ++ src/shared/FrozenFramesModel.h | 4 +- src/shared/PresetsModel.cpp | 8 ++ src/shared/PresetsModel.h | 4 +- src/shared/SectionsModel.cpp | 6 ++ src/shared/SectionsModel.h | 4 +- src/shared/WobblyProject.cpp | 126 ++++++++++++++++++++++++-- src/shared/WobblyProject.h | 32 +++++++ src/wobbly/WobblyWindow.cpp | 147 ++++++++++++++++++++++++++++++- src/wobbly/WobblyWindow.h | 9 ++ 16 files changed, 361 insertions(+), 16 deletions(-) diff --git a/src/shared/BookmarksModel.cpp b/src/shared/BookmarksModel.cpp index f87a72a..73381c6 100644 --- a/src/shared/BookmarksModel.cpp +++ b/src/shared/BookmarksModel.cpp @@ -133,3 +133,11 @@ void BookmarksModel::erase(int frame) { endRemoveRows(); } + +void BookmarksModel::clear() { + if (!size()) return; + + beginRemoveRows(QModelIndex(), 0, size() - 1); + BookmarkMap::clear(); + endRemoveRows(); +} \ No newline at end of file diff --git a/src/shared/BookmarksModel.h b/src/shared/BookmarksModel.h index 428b10e..5d2178a 100644 --- a/src/shared/BookmarksModel.h +++ b/src/shared/BookmarksModel.h @@ -26,7 +26,7 @@ SOFTWARE. #include "WobblyTypes.h" -class BookmarksModel : public QAbstractTableModel, private BookmarkMap { +class BookmarksModel : public QAbstractTableModel, public BookmarkMap { Q_OBJECT public: @@ -62,6 +62,8 @@ class BookmarksModel : public QAbstractTableModel, private BookmarkMap { void insert(const value_type &bookmark); void erase(int frame); + + void clear(); }; #endif // BOOKMARKSMODEL_H diff --git a/src/shared/CombedFramesModel.h b/src/shared/CombedFramesModel.h index 2a2ebf1..35093e5 100644 --- a/src/shared/CombedFramesModel.h +++ b/src/shared/CombedFramesModel.h @@ -26,7 +26,7 @@ SOFTWARE. #include -class CombedFramesModel : public QAbstractListModel, private std::set { +class CombedFramesModel : public QAbstractListModel, public std::set { Q_OBJECT public: diff --git a/src/shared/CustomListsModel.cpp b/src/shared/CustomListsModel.cpp index e937f92..177f07d 100644 --- a/src/shared/CustomListsModel.cpp +++ b/src/shared/CustomListsModel.cpp @@ -105,6 +105,15 @@ void CustomListsModel::erase(int list_index) { } +void CustomListsModel::clear() { + if (!size()) return; + + beginRemoveRows(QModelIndex(), 0, size() - 1); + CustomListVector::clear(); + endRemoveRows(); +} + + void CustomListsModel::moveCustomListUp(int list_index) { if (beginMoveRows(QModelIndex(), list_index, list_index, QModelIndex(), list_index - 1)) { std::swap(at(list_index - 1), at(list_index)); diff --git a/src/shared/CustomListsModel.h b/src/shared/CustomListsModel.h index 0b11300..6cc4515 100644 --- a/src/shared/CustomListsModel.h +++ b/src/shared/CustomListsModel.h @@ -26,7 +26,7 @@ SOFTWARE. #include "WobblyTypes.h" -class CustomListsModel : public QAbstractTableModel, private CustomListVector { +class CustomListsModel : public QAbstractTableModel, public CustomListVector { Q_OBJECT enum Columns { @@ -61,6 +61,8 @@ class CustomListsModel : public QAbstractTableModel, private CustomListVector { void erase(int list_index); + void clear(); + void moveCustomListUp(int list_index); void moveCustomListDown(int list_index); diff --git a/src/shared/FrameRangesModel.h b/src/shared/FrameRangesModel.h index 1949b24..188268f 100644 --- a/src/shared/FrameRangesModel.h +++ b/src/shared/FrameRangesModel.h @@ -30,7 +30,7 @@ struct FrameRange { }; -class FrameRangesModel : public QAbstractTableModel, private std::map { +class FrameRangesModel : public QAbstractTableModel, public std::map { Q_OBJECT enum Columns { diff --git a/src/shared/FrozenFramesModel.cpp b/src/shared/FrozenFramesModel.cpp index b30991c..1429df9 100644 --- a/src/shared/FrozenFramesModel.cpp +++ b/src/shared/FrozenFramesModel.cpp @@ -111,3 +111,11 @@ void FrozenFramesModel::erase(int freeze_frame) { endRemoveRows(); } + +void FrozenFramesModel::clear() { + if (!size()) return; + + beginRemoveRows(QModelIndex(), 0, size() - 1); + FreezeFrameMap::clear(); + endRemoveRows(); +} \ No newline at end of file diff --git a/src/shared/FrozenFramesModel.h b/src/shared/FrozenFramesModel.h index 6af8a14..a4570f5 100644 --- a/src/shared/FrozenFramesModel.h +++ b/src/shared/FrozenFramesModel.h @@ -28,7 +28,7 @@ SOFTWARE. #include "WobblyTypes.h" -class FrozenFramesModel : public QAbstractTableModel, private FreezeFrameMap { +class FrozenFramesModel : public QAbstractTableModel, public FreezeFrameMap { Q_OBJECT enum Columns { @@ -57,6 +57,8 @@ class FrozenFramesModel : public QAbstractTableModel, private FreezeFrameMap { void insert(const value_type &freeze_frame); void erase(int freeze_frame); + + void clear(); }; #endif // FROZENFRAMESMODEL_H diff --git a/src/shared/PresetsModel.cpp b/src/shared/PresetsModel.cpp index c48a85e..074b83a 100644 --- a/src/shared/PresetsModel.cpp +++ b/src/shared/PresetsModel.cpp @@ -91,3 +91,11 @@ void PresetsModel::erase(const std::string &preset_name) { endRemoveRows(); } + +void PresetsModel::clear() { + if (!size()) return; + + beginRemoveRows(QModelIndex(), 0, size() - 1); + PresetMap::clear(); + endRemoveRows(); +} \ No newline at end of file diff --git a/src/shared/PresetsModel.h b/src/shared/PresetsModel.h index c2ab36b..729fe8b 100644 --- a/src/shared/PresetsModel.h +++ b/src/shared/PresetsModel.h @@ -26,7 +26,7 @@ SOFTWARE. #include "WobblyTypes.h" -class PresetsModel : public QAbstractListModel, private PresetMap { +class PresetsModel : public QAbstractListModel, public PresetMap { Q_OBJECT public: @@ -46,6 +46,8 @@ class PresetsModel : public QAbstractListModel, private PresetMap { void insert(const value_type &preset); void erase(const std::string &preset_name); + + void clear(); }; #endif // PRESETSMODEL_H diff --git a/src/shared/SectionsModel.cpp b/src/shared/SectionsModel.cpp index 6cd32dd..fbbbd0d 100644 --- a/src/shared/SectionsModel.cpp +++ b/src/shared/SectionsModel.cpp @@ -117,6 +117,12 @@ void SectionsModel::erase(int section_start) { endRemoveRows(); } +void SectionsModel::clear() { + beginRemoveRows(QModelIndex(), 0, size()); + SectionMap::clear(); + endRemoveRows(); +} + void SectionsModel::setSectionPresetName(int section_start, size_t preset_index, const std::string &preset_name) { SectionMap::iterator it = find(section_start); diff --git a/src/shared/SectionsModel.h b/src/shared/SectionsModel.h index 08d0c61..874d233 100644 --- a/src/shared/SectionsModel.h +++ b/src/shared/SectionsModel.h @@ -26,7 +26,7 @@ SOFTWARE. #include "WobblyTypes.h" -class SectionsModel : public QAbstractTableModel, private SectionMap { +class SectionsModel : public QAbstractTableModel, public SectionMap { Q_OBJECT public: @@ -55,6 +55,8 @@ class SectionsModel : public QAbstractTableModel, private SectionMap { void erase(int section_start); + void clear(); + void setSectionPresetName(int section_start, size_t preset_index, const std::string &preset_name); void appendSectionPreset(int section_start, const std::string &preset_name); diff --git a/src/shared/WobblyProject.cpp b/src/shared/WobblyProject.cpp index 528e37c..f2da8ac 100644 --- a/src/shared/WobblyProject.cpp +++ b/src/shared/WobblyProject.cpp @@ -2589,6 +2589,18 @@ std::map WobblyProject::getCMatchSequences(int minimum) const { } +void WobblyProject::updateOrphanFrames() { + // Find the ends manually so this is not O(#sections^2) + auto it = sections->cbegin(); + while (it != sections->cend()) { + int section_start = it->second.start; + it++; + int section_end = it == sections->cend() ? getNumFrames(PostSource) : it->second.start; + + updateSectionOrphanFrames(section_start, section_end); + } +} + void WobblyProject::updateSectionOrphanFrames(int section_start, int section_end) { if (getMatch(section_start) == 'n') addOrphanFrame({ section_start, 'n' }); @@ -2809,6 +2821,108 @@ void WobblyProject::setModified(bool modified) { } +std::string WobblyProject::getUndoDescription() { + if (undo_stack.size() <= 1) + return ""; + return undo_stack.back().description; +} + +std::string WobblyProject::getRedoDescription() { + if (redo_stack.empty()) + return ""; + return redo_stack.back().description; +} + +void WobblyProject::restoreState(UndoStep state) { + matches = state.matches; + decimated_frames = state.decimated_frames; + pattern_guessing = state.pattern_guessing; + + presets->clear(); + for (auto const& p : state.presets) + presets->insert(p); + + custom_lists->clear(); + for (auto const& c : state.custom_lists) { + custom_lists->push_back(c); + custom_lists->back().ranges = std::make_shared(); + for (auto const& r : *c.ranges) + custom_lists->back().ranges->insert(r); + } + + combed_frames->clear(); + for (auto const& c : state.combed_frames) + combed_frames->insert(c); + + frozen_frames->clear(); + for (auto const& f : state.frozen_frames) + frozen_frames->insert(f); + + sections->clear(); + for (auto const& s : state.sections) + sections->insert(s); + + bookmarks->clear(); + for (auto const& b : state.bookmarks) + bookmarks->insert(b); +} + +void WobblyProject::commit(std::string description) { + UndoStep step = { + .description = description, + .matches = matches, + .decimated_frames = decimated_frames, + .pattern_guessing = pattern_guessing, + + .presets = *presets, + .custom_lists = *custom_lists, + .combed_frames = *combed_frames, + .frozen_frames = *frozen_frames, + .sections = *sections, + .bookmarks = *bookmarks, + }; + for (auto &cl : step.custom_lists) { + std::shared_ptr oldranges = cl.ranges; + cl.ranges = std::make_shared(); + + for (auto const& r : *oldranges) + cl.ranges->insert(r); + } + + undo_stack.push_back(step); + + redo_stack.clear(); + + while (undo_stack.size() > undo_steps) + undo_stack.pop_front(); +} + +void WobblyProject::undo() { + if (undo_stack.size() <= 1) return; + redo_stack.push_back(undo_stack.back()); + undo_stack.pop_back(); + restoreState(undo_stack.back()); +} + +void WobblyProject::redo() { + if (redo_stack.empty()) return; + restoreState(redo_stack.back()); + undo_stack.push_back(redo_stack.back()); + redo_stack.pop_back(); +} + +void WobblyProject::setUndoSteps(size_t steps) { + undo_steps = steps; + if (undo_steps < redo_stack.size()) { + undo_stack.clear(); + while (undo_steps < redo_stack.size()) + redo_stack.pop_front(); + } + while (undo_steps < undo_stack.size() + redo_stack.size()) + undo_stack.pop_front(); +} + + int WobblyProject::getZoom() const { return zoom; } @@ -3495,8 +3609,7 @@ void WobblyProject::guessProjectPatternsFromMics(int minimum_length, int edge_cu for (auto it = sections->cbegin(); it != sections->cend(); it++) guessSectionPatternsFromMics(it->second.start, minimum_length, edge_cutoff, use_patterns, drop_duplicate); - for (auto it = sections->cbegin(); it != sections->cend(); it++) - updateSectionOrphanFrames(it->second.start, getSectionEnd(it->second.start)); + updateOrphanFrames(); pattern_guessing.method = PatternGuessingFromMics; pattern_guessing.minimum_length = minimum_length; @@ -3514,8 +3627,7 @@ void WobblyProject::guessProjectPatternsFromDMetrics(int minimum_length, int edg for (auto it = sections->cbegin(); it != sections->cend(); it++) guessSectionPatternsFromDMetrics(it->second.start, minimum_length, edge_cutoff, use_patterns, drop_duplicate); - for (auto it = sections->cbegin(); it != sections->cend(); it++) - updateSectionOrphanFrames(it->second.start, getSectionEnd(it->second.start)); + updateOrphanFrames(); pattern_guessing.method = PatternGuessingFromDMetrics; pattern_guessing.minimum_length = minimum_length; @@ -3532,8 +3644,7 @@ void WobblyProject::guessProjectPatternsFromMicsAndDMetrics(int minimum_length, for (auto it = sections->cbegin(); it != sections->cend(); it++) guessSectionPatternsFromMicsAndDMetrics(it->second.start, minimum_length, edge_cutoff, use_patterns, drop_duplicate); - for (auto it = sections->cbegin(); it != sections->cend(); it++) - updateSectionOrphanFrames(it->second.start, getSectionEnd(it->second.start)); + updateOrphanFrames(); pattern_guessing.method = PatternGuessingFromMicsAndDMetrics; pattern_guessing.minimum_length = minimum_length; @@ -3668,8 +3779,7 @@ void WobblyProject::guessProjectPatternsFromMatches(int minimum_length, int edge for (auto it = sections->cbegin(); it != sections->cend(); it++) guessSectionPatternsFromMatches(it->second.start, minimum_length, edge_cutoff, use_third_n_match, drop_duplicate); - for (auto it = sections->cbegin(); it != sections->cend(); it++) - updateSectionOrphanFrames(it->second.start, getSectionEnd(it->second.start)); + updateOrphanFrames(); pattern_guessing.method = PatternGuessingFromMatches; pattern_guessing.minimum_length = minimum_length; diff --git a/src/shared/WobblyProject.h b/src/shared/WobblyProject.h index 670f045..95299b0 100644 --- a/src/shared/WobblyProject.h +++ b/src/shared/WobblyProject.h @@ -81,6 +81,21 @@ static inline uint8_t matchCharToIndexDMetrics(char match) { return 255; } +struct UndoStep { + std::string description; + + std::vector matches; + std::vector > decimated_frames; + PatternGuessing pattern_guessing; + + PresetMap presets; + CustomListVector custom_lists; + std::set combed_frames; + FreezeFrameMap frozen_frames; + SectionMap sections; + BookmarkMap bookmarks; +}; + class WobblyProject : public QObject { Q_OBJECT @@ -141,6 +156,10 @@ class WobblyProject : public QObject { bool is_modified = false; + std::list undo_stack; + std::list redo_stack; + size_t undo_steps; + // Only functions below. static bool isValidMatchChar(char match); @@ -151,6 +170,8 @@ class WobblyProject : public QObject { void applyPatternGuessingDecimation(const int section_start, const int section_end, const int first_duplicate, int drop_duplicate); + void restoreState(UndoStep state); + public: WobblyProject(bool _is_wobbly); WobblyProject(bool _is_wobbly, const std::string &_input_file, const std::string &_source_filter, int64_t _fps_num, int64_t _fps_den, int _width, int _height, int _num_frames); @@ -263,6 +284,7 @@ class WobblyProject : public QObject { std::map getCMatchSequences(int minimum) const; + void updateOrphanFrames(); void updateSectionOrphanFrames(int section_start, int section_end); CombedFramesModel *getCombedFramesModel(); @@ -307,6 +329,16 @@ class WobblyProject : public QObject { void setModified(bool modified); + // If these are the empty string, there is no undo/redo action available + std::string getUndoDescription(); + std::string getRedoDescription(); + + void commit(std::string description); + void undo(); + void redo(); + void setUndoSteps(size_t steps); + + int getZoom() const; void setZoom(int ratio); diff --git a/src/wobbly/WobblyWindow.cpp b/src/wobbly/WobblyWindow.cpp index bae89bf..1f48cb6 100644 --- a/src/wobbly/WobblyWindow.cpp +++ b/src/wobbly/WobblyWindow.cpp @@ -64,6 +64,7 @@ SOFTWARE. #define KEY_COLORMATRIX QStringLiteral("user_interface/colormatrix") #define KEY_MAXIMUM_CACHE_SIZE QStringLiteral("user_interface/maximum_cache_size") #define KEY_PRINT_DETAILS_ON_VIDEO QStringLiteral("user_interface/print_details_on_video") +#define KEY_UNDO_STEPS QStringLiteral("user_interface/undo_steps") #define KEY_NUMBER_OF_THUMBNAILS QStringLiteral("user_interface/number_of_thumbnails") #define KEY_THUMBNAIL_SIZE QStringLiteral("user_interface/thumbnail_size") #define KEY_LAST_DIR QStringLiteral("user_interface/last_dir") @@ -179,6 +180,8 @@ void WobblyWindow::readSettings() { settings_print_details_check->setChecked(settings.value(KEY_PRINT_DETAILS_ON_VIDEO, true).toBool()); + settings_undo_steps_spin->setValue(settings.value(KEY_UNDO_STEPS, 50).toInt()); + settings_num_thumbnails_spin->setValue(settings.value(KEY_NUMBER_OF_THUMBNAILS, 5).toInt()); settings_thumbnail_size_dspin->setValue(settings.value(KEY_THUMBNAIL_SIZE, 15).toDouble()); @@ -311,6 +314,18 @@ void WobblyWindow::createMenu() { QMenu *p = bar->addMenu("&Project"); + undo_action = new QAction("Undo", this); + undo_action->setEnabled(false); + redo_action = new QAction("Redo", this); + redo_action->setEnabled(false); + + connect(undo_action, &QAction::triggered, this, &WobblyWindow::undo); + connect(redo_action, &QAction::triggered, this, &WobblyWindow::redo); + + p->addAction(undo_action); + p->addAction(redo_action); + p->addSeparator(); + struct Menu { const char *name; void (WobblyWindow::* func)(); @@ -409,6 +424,8 @@ void WobblyWindow::createShortcuts() { { "", "", "Save timecodes as", &WobblyWindow::saveTimecodesAs }, { "", "", "Save screenshot", &WobblyWindow::saveScreenshot }, { "", "", "Import from project", &WobblyWindow::importFromProject }, + { "", "Ctrl+Z", "Undo", &WobblyWindow::undo }, + { "", "Ctrl+Y", "Redo", &WobblyWindow::redo }, { "", "", "Quit", &WobblyWindow::quit }, { "", "", "Show or hide frame details", &WobblyWindow::showHideFrameDetails }, @@ -882,6 +899,7 @@ void WobblyWindow::createPresetEditor() { selected = preset_combo->itemText(selected_index); project->addPreset(preset_name.toStdString()); + commit("Add preset"); if (selected_index > -1) setSelectedPreset(preset_combo->findText(selected)); @@ -916,6 +934,7 @@ void WobblyWindow::createPresetEditor() { if (!preset_name.isEmpty() && preset_name != preset_combo->currentText()) { try { project->renamePreset(preset_combo->currentText().toStdString(), preset_name.toStdString()); + commit("Rename preset"); int index = preset_combo->findText(preset_name); setSelectedPreset(index); @@ -952,6 +971,7 @@ void WobblyWindow::createPresetEditor() { } project->deletePreset(preset); + commit("Delete preset"); setSelectedPreset(selected_preset); @@ -1116,6 +1136,7 @@ void WobblyWindow::createSectionsEditor() { for (size_t i = 0; i < sections.size(); i++) if (sections[i].start != 0) project->deleteSection(sections[i].start); + commit("Delete section(s)"); bool update_needed = false; @@ -1199,6 +1220,7 @@ void WobblyWindow::createSectionsEditor() { for (size_t i = 0; i < preset_indexes.size(); i++) project->moveSectionPresetUp(section_start, preset_indexes[i]); + commit("Move section preset"); for (size_t i = 0; i < preset_indexes.size(); i++) section_presets_list->item(preset_indexes[i] - 1)->setSelected(true); @@ -1243,6 +1265,7 @@ void WobblyWindow::createSectionsEditor() { for (int i = preset_indexes.size() - 1; i >= 0; i--) project->moveSectionPresetDown(section_start, preset_indexes[i]); + commit("Move section preset"); for (size_t i = 0; i < preset_indexes.size(); i++) section_presets_list->item(preset_indexes[i] + 1)->setSelected(true); @@ -1284,6 +1307,7 @@ void WobblyWindow::createSectionsEditor() { for (int i = preset_indexes.size() - 1; i >= 0; i--) project->deleteSectionPreset(section_start, preset_indexes[i]); + commit("Remove preset from section"); if (preview) { try { @@ -1313,6 +1337,7 @@ void WobblyWindow::createSectionsEditor() { if (ok) project->setSectionPreset(section_start, preset->data().toString().toStdString()); } + commit("Add preset to section"); if (selected_sections.size()) { if (preview) { @@ -1455,6 +1480,7 @@ void WobblyWindow::createCustomListsEditor() { if (!cl_name.isEmpty()) { try { project->addCustomList(cl_name.toStdString()); + commit("Add custom list"); ok = true; } catch (WobblyException &e) { @@ -1493,6 +1519,7 @@ void WobblyWindow::createCustomListsEditor() { if (!new_name.isEmpty()) { try { project->renameCustomList(old_name.toStdString(), new_name.toStdString()); + commit("Rename custom list"); if (cl_index == getSelectedCustomList()) setSelectedCustomList(cl_index); @@ -1536,6 +1563,7 @@ void WobblyWindow::createCustomListsEditor() { else if (indexes[i] < selected_custom_list) selected_custom_list--; } + commit("Delete custom list(s)"); if (cl_view->model()->rowCount()) cl_view->selectRow(cl_view->currentIndex().row()); @@ -1582,6 +1610,7 @@ void WobblyWindow::createCustomListsEditor() { if (indexes[i] == selected_custom_list + 1) selected_custom_list++; } + commit("Move custom list(s)"); if (preview && update_needed) { try { @@ -1625,6 +1654,7 @@ void WobblyWindow::createCustomListsEditor() { if (indexes[i] == selected_custom_list - 1) selected_custom_list--; } + commit("Move custom list(s)"); if (preview && update_needed) { try { @@ -1652,6 +1682,7 @@ void WobblyWindow::createCustomListsEditor() { bool update_needed = project->isCustomListInUse(cl_index) && project->getCustomListPreset(cl_index) != text.toStdString(); project->setCustomListPreset(cl_index, text.toStdString()); + commit("Set custom list preset"); if (preview && update_needed) { try { @@ -1679,6 +1710,7 @@ void WobblyWindow::createCustomListsEditor() { bool update_needed = project->isCustomListInUse(cl_index) && new_position != project->getCustomListPosition(cl_index); project->setCustomListPosition(cl_index, new_position); + commit("Set custom list position"); if (preview && update_needed) { try { @@ -1733,6 +1765,7 @@ void WobblyWindow::createCustomListsEditor() { for (size_t i = 0; i < frames.size(); i++) project->deleteCustomListRange(cl_index, frames[i]); + commit("Delete frames from custom list"); if (cl_ranges_view->model()->rowCount()) cl_ranges_view->selectRow(cl_ranges_view->currentIndex().row()); @@ -1793,6 +1826,7 @@ void WobblyWindow::createCustomListsEditor() { const FrameRange *range = project->findCustomListRange(cl_src_index, selection[i].data().toInt()); project->addCustomListRange(cl_dst_index, range->first, range->last); } + commit("Copy frames to custom list"); // This implementation copies the frames and then deletes them, which creates two undo events, but it's not really worth the effort to fix... // An update is only needed if the source custom list wasn't in use before deleting the ranges // and the destination custom list is in use after adding the ranges. @@ -1836,6 +1870,7 @@ void WobblyWindow::createCustomListsEditor() { const FrameRange *range = project->findCustomListRange(cl_src_index, selection[i].data().toInt()); project->addCustomListRange(cl_dst_index, range->first, range->last); } + commit("Copy frames to custom list"); bool update_needed = project->isCustomListInUse(cl_dst_index); @@ -2009,6 +2044,7 @@ void WobblyWindow::createFrozenFramesViewer() { for (size_t i = 0 ; i < frames.size(); i++) project->deleteFreezeFrame(frames[i]); + commit("Delete frozen frames"); if (frozen_frames_view->model()->rowCount()) frozen_frames_view->selectRow(frozen_frames_view->currentIndex().row()); @@ -2519,6 +2555,7 @@ void WobblyWindow::createCombedFramesWindow() { for (size_t i = 0 ; i < frames.size(); i++) project->deleteCombedFrame(frames[i]); + commit("Delete combed frame(s)"); if (combed_view->model()->rowCount()) combed_view->selectRow(combed_view->currentIndex().row()); @@ -2569,6 +2606,7 @@ void WobblyWindow::createCombedFramesWindow() { for (auto it = combed_frames.cbegin(); it != combed_frames.cend(); it++) project->addCombedFrame(project->frameNumberBeforeDecimation(*it)); + commit("Find combed frames"); updateFrameDetails(); }); @@ -2683,6 +2721,7 @@ void WobblyWindow::createBookmarksWindow() { for (size_t i = 0 ; i < frames.size(); i++) project->deleteBookmark(frames[i]); + commit("Delete bookmark(s)"); if (bookmarks_view->model()->rowCount()) bookmarks_view->selectRow(bookmarks_view->currentIndex().row()); @@ -2748,6 +2787,9 @@ void WobblyWindow::createSettingsWindow() { settings_cache_spin->setValue(4096); settings_cache_spin->setSuffix(QStringLiteral(" MiB")); + settings_undo_steps_spin = new SpinBox; + settings_undo_steps_spin->setRange(0, 1000); + settings_num_thumbnails_spin = new SpinBox; settings_num_thumbnails_spin->setRange(-1, 21); settings_num_thumbnails_spin->setSingleStep(2); @@ -2828,6 +2870,12 @@ void WobblyWindow::createSettingsWindow() { settings.setValue(KEY_MAXIMUM_CACHE_SIZE, value); }); + connect(settings_undo_steps_spin, static_cast(&SpinBox::valueChanged), [this] (int value) { + if (project) + project->setUndoSteps(size_t(value)); + settings.setValue(KEY_UNDO_STEPS, value); + }); + connect(settings_num_thumbnails_spin, static_cast(&SpinBox::valueChanged), [this] (int num_thumbnails) { settings.setValue(KEY_NUMBER_OF_THUMBNAILS, num_thumbnails); @@ -2944,6 +2992,7 @@ void WobblyWindow::createSettingsWindow() { form->addRow(QStringLiteral("Application style"), application_style_combo); form->addRow(QStringLiteral("Colormatrix"), settings_colormatrix_combo); form->addRow(QStringLiteral("Maximum cache size"), settings_cache_spin); + form->addRow(QStringLiteral("Maximum undo steps"), settings_undo_steps_spin); form->addRow(QStringLiteral("Number of thumbnails"), settings_num_thumbnails_spin); form->addRow(QStringLiteral("Thumbnail size"), settings_thumbnail_size_dspin); @@ -3684,6 +3733,7 @@ void WobblyWindow::initialiseBookmarksWindow() { bookmarks_view->setModel(project->getBookmarksModel()); connect(project->getBookmarksModel(), &BookmarksModel::dataChanged, [this] (const QModelIndex &topLeft, const QModelIndex &bottomRight) { + commit("Rename bookmark"); if (topLeft == bottomRight) { int frame = bookmarks_view->model()->index(topLeft.row(), BookmarksModel::FrameColumn).data().toInt(); @@ -3707,6 +3757,9 @@ void WobblyWindow::initialiseUIFromProject() { updateGeometry(); + project->setUndoSteps(size_t(settings.value(KEY_UNDO_STEPS, 50).toInt())); + project->updateOrphanFrames(); + initialiseCropAssistant(); initialisePresetEditor(); initialiseSectionsEditor(); @@ -3746,6 +3799,7 @@ void WobblyWindow::realOpenProject(const QString &path) { current_frame = project->getLastVisitedFrame(); initialiseUIFromProject(); + project->commit("Initial"); vssapi->evaluateBuffer(vsscript, "vs.clear_output(1)", "wobbly.cleanup"); @@ -3856,6 +3910,7 @@ void WobblyWindow::realOpenVideo(const QString &path) { project_path.clear(); initialiseUIFromProject(); + project->commit("Initial"); vssapi->evaluateBuffer(vsscript, "vs.clear_output(1)", "wobbly.cleanup"); @@ -4976,6 +5031,7 @@ void WobblyWindow::cycleMatchBCN() { return; project->cycleMatchBCN(current_frame); + commit("Cycle frame's match"); updateSectionOrphanFrames(current_frame); @@ -4998,6 +5054,7 @@ void WobblyWindow::freezeForward() { try { project->addFreezeFrame(current_frame, current_frame, current_frame + 1); + commit("Add freeze frame"); evaluateScript(preview); } catch (WobblyException &e) { @@ -5016,6 +5073,7 @@ void WobblyWindow::freezeBackward() { try { project->addFreezeFrame(current_frame, current_frame, current_frame - 1); + commit("Add freeze frame"); evaluateScript(preview); } catch (WobblyException &e) { @@ -5049,6 +5107,7 @@ void WobblyWindow::freezeRange() { ff.replacement = current_frame; try { project->addFreezeFrame(ff.first, ff.last, ff.replacement); + commit("Freeze range"); evaluateScript(preview); } catch (WobblyException &e) { @@ -5070,6 +5129,7 @@ void WobblyWindow::deleteFreezeFrame() { const FreezeFrame *ff = project->findFreezeFrame(current_frame); if (ff) { project->deleteFreezeFrame(ff->first); + commit("Delete freeze frame"); try { evaluateScript(preview); @@ -5104,6 +5164,7 @@ void WobblyWindow::toggleDecimation() { project->deleteDecimatedFrame(current_frame); else project->addDecimatedFrame(current_frame); + commit("Toggle decimation"); if (preview) { try { @@ -5149,6 +5210,8 @@ void WobblyWindow::toggleCombed() { for (int i = start; i <= end; i++) project->addCombedFrame(i); + commit("Toggle combed"); + /// Uncomment if combed frames ever get filtered // if (preview) { // try { @@ -5180,6 +5243,7 @@ void WobblyWindow::toggleBookmark() { if (ok) project->addBookmark(current_frame, description.toStdString()); } + commit("Toggle bookmark"); updateFrameDetails(); } @@ -5192,6 +5256,7 @@ void WobblyWindow::addSection() { const Section *section = project->findSection(current_frame); if (section->start != current_frame) { project->addSection(current_frame); + commit("Add section"); if (preview && section->presets.size()) { try { @@ -5224,6 +5289,7 @@ void WobblyWindow::deleteSection() { } project->deleteSection(section->start); + commit("Delete section"); if (update_needed) { try { @@ -5258,7 +5324,12 @@ void WobblyWindow::presetEdited() { if (preset_combo->currentIndex() == -1) return; - project->setPresetContents(preset_combo->currentText().toStdString(), preset_edit->toPlainText().toStdString()); + std::string name = preset_combo->currentText().toStdString(); + std::string contents = preset_edit->toPlainText().toStdString(); + if (contents != project->getPresetContents(name)) { + project->setPresetContents(name, contents); + commit("Edit preset"); + } } @@ -5282,6 +5353,7 @@ void WobblyWindow::resetMatch() { project->resetRangeMatches(start, end); + commit("Reset match(es)"); updateSectionOrphanFrames(current_frame); @@ -5302,6 +5374,7 @@ void WobblyWindow::resetSection() { const Section *section = project->findSection(current_frame); project->resetSectionMatches(section->start); + commit("Reset section"); updateSectionOrphanFrames(section); @@ -5333,6 +5406,7 @@ void WobblyWindow::rotateAndSetPatterns() { project->setSectionMatchesFromPattern(section->start, match_pattern.toStdString()); project->setSectionDecimationFromPattern(section->start, decimation_pattern.toStdString()); + commit("Rotate section pattern"); updateSectionOrphanFrames(section); @@ -5358,6 +5432,7 @@ void WobblyWindow::setMatchPattern() { finishRange(); project->setRangeMatchesFromPattern(range_start, range_end, match_pattern.toStdString()); + commit("Set match pattern to range"); cancelRange(); @@ -5383,6 +5458,7 @@ void WobblyWindow::setDecimationPattern() { finishRange(); project->setRangeDecimationFromPattern(range_start, range_end, decimation_pattern.toStdString()); + commit("Set decimation pattern to range"); cancelRange(); @@ -5407,6 +5483,7 @@ void WobblyWindow::setMatchAndDecimationPatterns() { project->setRangeMatchesFromPattern(range_start, range_end, match_pattern.toStdString()); project->setRangeDecimationFromPattern(range_start, range_end, decimation_pattern.toStdString()); + commit("Set match and decimation patterns to range"); cancelRange(); @@ -5460,6 +5537,7 @@ void WobblyWindow::guessCurrentSectionPatternsFromMics() { try { success = project->guessSectionPatternsFromMics(section_start, pg_length_spin->value(), pg_edge_cutoff->value(), use_patterns, pg_decimate_buttons->checkedId()); + commit("Guess section patterns from mics"); } catch (WobblyException &e) { QApplication::restoreOverrideCursor(); @@ -5505,6 +5583,7 @@ void WobblyWindow::guessCurrentSectionPatternsFromDMetrics() { try { success = project->guessSectionPatternsFromDMetrics(section_start, pg_length_spin->value(), pg_edge_cutoff->value(), use_patterns, pg_decimate_buttons->checkedId()); + commit("Guess section patterns from dmetrics"); } catch (WobblyException &e) { QApplication::restoreOverrideCursor(); @@ -5550,6 +5629,7 @@ void WobblyWindow::guessCurrentSectionPatternsFromMicsAndDMetrics() { try { success = project->guessSectionPatternsFromMicsAndDMetrics(section_start, pg_length_spin->value(), pg_edge_cutoff->value(), use_patterns, pg_decimate_buttons->checkedId()); + commit("Guess section patterns from mics and dmetrics"); } catch (WobblyException &e) { QApplication::restoreOverrideCursor(); @@ -5593,6 +5673,7 @@ void WobblyWindow::guessProjectPatternsFromMics() { project->clearOrphanFrames(); project->guessProjectPatternsFromMics(pg_length_spin->value(), pg_edge_cutoff->value(), use_patterns, pg_decimate_buttons->checkedId()); + commit("Guess project patterns from mics"); QApplication::restoreOverrideCursor(); @@ -5627,6 +5708,7 @@ void WobblyWindow::guessProjectPatternsFromDMetrics() { project->clearOrphanFrames(); project->guessProjectPatternsFromDMetrics(pg_length_spin->value(), pg_edge_cutoff->value(), use_patterns, pg_decimate_buttons->checkedId()); + commit("Guess section patterns from dmetrics"); QApplication::restoreOverrideCursor(); @@ -5661,6 +5743,7 @@ void WobblyWindow::guessProjectPatternsFromMicsAndDMetrics() { project->clearOrphanFrames(); project->guessProjectPatternsFromMicsAndDMetrics(pg_length_spin->value(), pg_edge_cutoff->value(), use_patterns, pg_decimate_buttons->checkedId()); + commit("Guess project patterns from mics and dmetrics"); QApplication::restoreOverrideCursor(); @@ -5688,6 +5771,7 @@ void WobblyWindow::guessCurrentSectionPatternsFromMatches() { int section_start = project->findSection(current_frame)->start; bool success = project->guessSectionPatternsFromMatches(section_start, pg_length_spin->value(), pg_edge_cutoff->value(), pg_n_match_buttons->checkedId(), pg_decimate_buttons->checkedId()); + commit("Guess section patterns from matches"); updatePatternGuessingWindow(); @@ -5716,6 +5800,7 @@ void WobblyWindow::guessProjectPatternsFromMatches() { project->clearOrphanFrames(); project->guessProjectPatternsFromMatches(pg_length_spin->value(), pg_edge_cutoff->value(), pg_n_match_buttons->checkedId(), pg_decimate_buttons->checkedId()); + commit("Guess project patterns from matches"); updatePatternGuessingWindow(); @@ -5760,6 +5845,64 @@ void WobblyWindow::togglePreview() { } } +void WobblyWindow::undo() { + if (!project) return; + project->undo(); + updateAfterUndo(); +} + +void WobblyWindow::redo() { + if (!project) return; + project->redo(); + updateAfterUndo(); +} + +void WobblyWindow::commit(std::string message) { + if (!project) return; + project->commit(message); + + updateUndoActions(); +} + +void WobblyWindow::updateUndoActions() { + if (!project) { + undo_action->setEnabled(false); + redo_action->setEnabled(false); + } else { + std::string undo_text = project->getUndoDescription(); + std::string redo_text = project->getRedoDescription(); + + if (undo_text.empty()) { + undo_action->setEnabled(false); + undo_action->setText("Undo"); + } else { + undo_action->setEnabled(true); + undo_action->setText(QStringLiteral("Undo %1").arg(QString::fromStdString(undo_text))); + } + + if (redo_text.empty()) { + redo_action->setEnabled(false); + redo_action->setText("Redo"); + } else { + redo_action->setEnabled(true); + redo_action->setText(QStringLiteral("Redo %1").arg(QString::fromStdString(redo_text))); + } + } +} + +void WobblyWindow::updateAfterUndo() { + updateUndoActions(); + + project->updateOrphanFrames(); + updateFrameRatesViewer(); + updatePatternGuessingWindow(); + updateCMatchSequencesWindow(); + updateFadesWindow(); + presetChanged(preset_combo->currentText()); + + evaluateMainDisplayScript(); +} + void WobblyWindow::zoom(bool in) { if (!project) @@ -5973,6 +6116,7 @@ void WobblyWindow::assignSelectedPresetToCurrentSection() { int section_start = project->findSection(current_frame)->start; project->setSectionPreset(section_start, presets_model->data(presets_model->index(selected_preset)).toString().toStdString()); + commit("Assign preset"); if (preview) { try { @@ -6011,6 +6155,7 @@ void WobblyWindow::addRangeToSelectedCustomList() { try { project->addCustomListRange(selected_custom_list, start, end); + commit("Add range to custom list"); updateFrameDetails(); } catch (WobblyException &e) { diff --git a/src/wobbly/WobblyWindow.h b/src/wobbly/WobblyWindow.h index 3d29237..c183979 100644 --- a/src/wobbly/WobblyWindow.h +++ b/src/wobbly/WobblyWindow.h @@ -70,6 +70,8 @@ class WobblyWindow : public QMainWindow { QMenu *recent_menu; + QAction *undo_action; + QAction *redo_action; // Widgets. @@ -180,6 +182,7 @@ class WobblyWindow : public QMainWindow { QCheckBox *settings_use_relative_paths_check; QComboBox *settings_colormatrix_combo; QSpinBox *settings_cache_spin; + SpinBox *settings_undo_steps_spin; QCheckBox *settings_print_details_check; QCheckBox *settings_bookmark_description_check; SpinBox *settings_num_thumbnails_spin; @@ -460,6 +463,12 @@ public slots: void zoomIn(); void zoomOut(); + void undo(); + void redo(); + void commit(std::string message); + void updateUndoActions(); + void updateAfterUndo(); + void vsLogPopup(int msgType, const QString &msg); void frameDone(void *framev, int n, bool preview_node, const QString &errorMsg); };