/* * ProjectJournal.cpp - implementation of ProjectJournal * * Copyright (c) 2006-2014 Tobias Doerffel * * This file is part of LMMS - https://lmms.io * * 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 2 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 (see COPYING); if not, write to the * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301 USA. * */ #include #include "ProjectJournal.h" #include "Engine.h" #include "JournallingObject.h" #include "Song.h" //! Avoid clashes between loaded IDs (have the bit cleared) //! and newly created IDs (have the bit set) static const int EO_ID_MSB = 1 << 23; const int ProjectJournal::MAX_UNDO_STATES = 100; // TODO: make this configurable in settings ProjectJournal::ProjectJournal() : m_joIDs(), m_undoCheckPoints(), m_redoCheckPoints(), m_journalling( false ) { } ProjectJournal::~ProjectJournal() { } void ProjectJournal::undo() { while( !m_undoCheckPoints.isEmpty() ) { CheckPoint c = m_undoCheckPoints.pop(); JournallingObject *jo = m_joIDs[c.joID]; if( jo ) { DataFile curState( DataFile::JournalData ); jo->saveState( curState, curState.content() ); m_redoCheckPoints.push( CheckPoint( c.joID, curState ) ); bool prev = isJournalling(); setJournalling( false ); jo->restoreState( c.data.content().firstChildElement() ); setJournalling( prev ); Engine::getSong()->setModified(); break; } } } void ProjectJournal::redo() { while( !m_redoCheckPoints.isEmpty() ) { CheckPoint c = m_redoCheckPoints.pop(); JournallingObject *jo = m_joIDs[c.joID]; if( jo ) { DataFile curState( DataFile::JournalData ); jo->saveState( curState, curState.content() ); m_undoCheckPoints.push( CheckPoint( c.joID, curState ) ); bool prev = isJournalling(); setJournalling( false ); jo->restoreState( c.data.content().firstChildElement() ); setJournalling( prev ); Engine::getSong()->setModified(); break; } } } bool ProjectJournal::canUndo() const { return !m_undoCheckPoints.isEmpty(); } bool ProjectJournal::canRedo() const { return !m_redoCheckPoints.isEmpty(); } void ProjectJournal::addJournalCheckPoint( JournallingObject *jo ) { if( isJournalling() ) { m_redoCheckPoints.clear(); DataFile dataFile( DataFile::JournalData ); jo->saveState( dataFile, dataFile.content() ); m_undoCheckPoints.push( CheckPoint( jo->id(), dataFile ) ); if( m_undoCheckPoints.size() > MAX_UNDO_STATES ) { m_undoCheckPoints.remove( 0, m_undoCheckPoints.size() - MAX_UNDO_STATES ); } } } jo_id_t ProjectJournal::allocID( JournallingObject * _obj ) { jo_id_t id; for( jo_id_t tid = rand(); m_joIDs.contains( id = tid % EO_ID_MSB | EO_ID_MSB ); tid++ ) { } m_joIDs[id] = _obj; //printf("new id: %d\n", id ); return id; } void ProjectJournal::reallocID( const jo_id_t _id, JournallingObject * _obj ) { //printf("realloc %d %d\n", _id, _obj ); // if( m_joIDs.contains( _id ) ) { m_joIDs[_id] = _obj; } } jo_id_t ProjectJournal::idToSave( jo_id_t id ) { return id & ~EO_ID_MSB; } jo_id_t ProjectJournal::idFromSave( jo_id_t id ) { return id | EO_ID_MSB; } void ProjectJournal::clearJournal() { m_undoCheckPoints.clear(); m_redoCheckPoints.clear(); for( JoIdMap::Iterator it = m_joIDs.begin(); it != m_joIDs.end(); ) { if( it.value() == NULL ) { it = m_joIDs.erase( it ); } else { ++it; } } } void ProjectJournal::stopAllJournalling() { for( JoIdMap::Iterator it = m_joIDs.begin(); it != m_joIDs.end(); ++it) { if( it.value() != NULL ) { it.value()->setJournalling(false); } } setJournalling(false); }