# UndoCommand.py -- Pamhyr # Copyright (C) 2023-2024 INRAE # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . # -*- coding: utf-8 -*- import logging from copy import deepcopy from tools import trace, timer from PyQt5.QtWidgets import ( QMessageBox, QUndoCommand, QUndoStack, ) from Model.BoundaryCondition.BoundaryCondition import BoundaryCondition logger = logging.getLogger() class SetDataCommand(QUndoCommand): def __init__(self, data, index, column, new_value): QUndoCommand.__init__(self) self._data = data self._index = index self._column = column self._old = self._data.get_i(self._index)[self._column] _type = self._data.get_type_column(self._column) self._new = _type(new_value) def undo(self): self._data._set_i_c_v(self._index, self._column, self._old) def redo(self): self._data._set_i_c_v(self._index, self._column, self._new) class SetMetaDataCommand(QUndoCommand): def __init__(self, data, column, new_value): QUndoCommand.__init__(self) self._data = data self._column = column if self._column == "d50": self._old = self._data.d50 elif self._column == "sigma": self._old = self._data.sigma self._new = float(new_value) def undo(self): if self._column == "d50": self._data.d50 = self._old elif self._column == "sigma": self._data.sigma = self._old def redo(self): if self._column == "d50": self._data.d50 = self._new elif self._column == "sigma": self._data.sigma = self._new class AddCommand(QUndoCommand): def __init__(self, data, index): QUndoCommand.__init__(self) self._data = data self._index = index self._new = None def undo(self): self._data.delete_i([self._index]) def redo(self): if self._new is None: self._new = self._data.add(self._index) else: self._data.insert(self._index, self._new) class DelCommand(QUndoCommand): def __init__(self, data, rows): QUndoCommand.__init__(self) self._data = data self._rows = rows self._bc = [] for row in rows: self._bc.append((row, self._data.get_i(row))) self._bc.sort() def undo(self): for row, el in self._bc: self._data.insert(row, el) def redo(self): self._data.delete_i(self._rows) class SortCommand(QUndoCommand): def __init__(self, data, _reverse): QUndoCommand.__init__(self) self._data = data self._reverse = _reverse self._old = self._data.data self._indexes = None def undo(self): ll = self._data.data self._data.sort( key=lambda x: self._indexes[ll.index(x)] ) def redo(self): self._data.sort( _reverse=self._reverse, key=lambda x: x[0] ) if self._indexes is None: self._indexes = list( map( lambda p: self._old.index(p), self._data.data ) ) self._old = None class MoveCommand(QUndoCommand): def __init__(self, data, up, i): QUndoCommand.__init__(self) self._data = data self._up = up == "up" self._i = i def undo(self): if self._up: self._data.move_up(self._i) else: self._data.move_down(self._i) def redo(self): if self._up: self._data.move_up(self._i) else: self._data.move_down(self._i) class PasteCommand(QUndoCommand): def __init__(self, data, row, bcs): QUndoCommand.__init__(self) self._data = data self._row = row self._bcs = bcs self._bcs.reverse() def undo(self): self._data.delete_i( range(self._row, self._row + len(self._bcs)) ) def redo(self): for bc in self._bcs: self._data.insert(self._row, bc) class ReplaceDataCommand(QUndoCommand): def __init__(self, data, data1, data2): QUndoCommand.__init__(self) self._data = data self._old_rows = len(data) self._data1 = data1 self._data2 = data2 self._rows = len(data1) self._old_bc = [] for row in range(self._old_rows): self._old_bc.append((row, self._data.get_i(row))) self._old_bc.sort() def undo(self): self._data.delete_i(list(range(self._rows))) for row, el in self._old_bc: self._data.insert(row, el) def redo(self): self._data.delete_i(list(range(self._old_rows))) for row in range(self._rows): self._data.add(row) self._data._set_i_c_v(row, 0, self._data1[row]) self._data._set_i_c_v(row, 1, self._data2[row])