#include "document.hpp" #include void CSMDoc::Document::load (const std::vector::const_iterator& begin, const std::vector::const_iterator& end, bool lastAsModified) { assert (begin!=end); std::vector::const_iterator end2 (end); if (lastAsModified) --end2; for (std::vector::const_iterator iter (begin); iter!=end2; ++iter) getData().loadFile (*iter, true); if (lastAsModified) getData().loadFile (*end2, false); } void CSMDoc::Document::addOptionalGmsts() { static const char *sFloats[] = { "fCombatDistanceWerewolfMod", "fFleeDistance", "fWereWolfAcrobatics", "fWereWolfAgility", "fWereWolfAlchemy", "fWereWolfAlteration", "fWereWolfArmorer", "fWereWolfAthletics", "fWereWolfAxe", "fWereWolfBlock", "fWereWolfBluntWeapon", "fWereWolfConjuration", "fWereWolfDestruction", "fWereWolfEnchant", "fWereWolfEndurance", "fWereWolfFatigue", "fWereWolfHandtoHand", "fWereWolfHealth", "fWereWolfHeavyArmor", "fWereWolfIllusion", "fWereWolfIntellegence", "fWereWolfLightArmor", "fWereWolfLongBlade", "fWereWolfLuck", "fWereWolfMagicka", "fWereWolfMarksman", "fWereWolfMediumArmor", "fWereWolfMerchantile", "fWereWolfMysticism", "fWereWolfPersonality", "fWereWolfRestoration", "fWereWolfRunMult", "fWereWolfSecurity", "fWereWolfShortBlade", "fWereWolfSilverWeaponDamageMult", "fWereWolfSneak", "fWereWolfSpear", "fWereWolfSpeechcraft", "fWereWolfSpeed", "fWereWolfStrength", "fWereWolfUnarmored", "fWereWolfWillPower", 0 }; static const char *sIntegers[] = { "iWereWolfBounty", "iWereWolfFightMod", "iWereWolfFleeMod", "iWereWolfLevelToAttack", 0 }; static const char *sStrings[] = { "sCompanionShare", "sCompanionWarningButtonOne", "sCompanionWarningButtonTwo", "sCompanionWarningMessage", "sDeleteNote", "sEditNote", "sEffectSummonCreature01", "sEffectSummonCreature02", "sEffectSummonCreature03", "sEffectSummonCreature04", "sEffectSummonCreature05", "sEffectSummonFabricant", "sLevitateDisabled", "sMagicCreature01ID", "sMagicCreature02ID", "sMagicCreature03ID", "sMagicCreature04ID", "sMagicCreature05ID", "sMagicFabricantID", "sMaxSale", "sProfitValue", "sTeleportDisabled", "sWerewolfAlarmMessage", "sWerewolfPopup", "sWerewolfRefusal", "sWerewolfRestMessage", 0 }; for (int i=0; sFloats[i]; ++i) { ESM::GameSetting gmst; gmst.mId = sFloats[i]; gmst.mValue.setType (ESM::VT_Float); addOptionalGmst (gmst); } for (int i=0; sIntegers[i]; ++i) { ESM::GameSetting gmst; gmst.mId = sIntegers[i]; gmst.mValue.setType (ESM::VT_Int); addOptionalGmst (gmst); } for (int i=0; sStrings[i]; ++i) { ESM::GameSetting gmst; gmst.mId = sStrings[i]; gmst.mValue.setType (ESM::VT_String); gmst.mValue.setString (""); addOptionalGmst (gmst); } } void CSMDoc::Document::addOptionalGlobals() { static const char *sGlobals[] = { "dayspassed", "pcwerewolf", "pcyear", 0 }; for (int i=0; sGlobals[i]; ++i) { ESM::Global global; global.mId = sGlobals[i]; global.mValue.setType (ESM::VT_Long); if (i==0) global.mValue.setInteger (1); // dayspassed starts counting at 1 addOptionalGlobal (global); } } void CSMDoc::Document::addOptionalGmst (const ESM::GameSetting& gmst) { if (getData().getGmsts().searchId (gmst.mId)==-1) { CSMWorld::Record record; record.mBase = gmst; record.mState = CSMWorld::RecordBase::State_BaseOnly; getData().getGmsts().appendRecord (record); } } void CSMDoc::Document::addOptionalGlobal (const ESM::Global& global) { if (getData().getGlobals().searchId (global.mId)==-1) { CSMWorld::Record record; record.mBase = global; record.mState = CSMWorld::RecordBase::State_BaseOnly; getData().getGlobals().appendRecord (record); } } void CSMDoc::Document::createBase() { static const char *sGlobals[] = { "Day", "DaysPassed", "GameHour", "Month", "PCRace", "PCVampire", "PCWerewolf", "PCYear", 0 }; for (int i=0; sGlobals[i]; ++i) { ESM::Global record; record.mId = sGlobals[i]; record.mValue.setType (i==2 ? ESM::VT_Float : ESM::VT_Long); if (i==0 || i==1) record.mValue.setInteger (1); getData().getGlobals().add (record); } /// \todo add GMSTs for (int i=0; i<27; ++i) { ESM::Skill record; record.mIndex = i; record.mId = ESM::Skill::getIndexToId (record.mIndex); record.blank(); getData().getSkills().add (record); } } CSMDoc::Document::Document (const std::vector& files, bool new_) : mTools (mData) { if (files.empty()) throw std::runtime_error ("Empty content file sequence"); /// \todo adjust last file name: /// \li make sure it is located in the data-local directory (adjust path if necessary) /// \li make sure the extension matches the new scheme (change it if necesarry) mName = files.back().filename().string(); if (new_ && files.size()==1) createBase(); else { std::vector::const_iterator end = files.end(); if (new_) --end; load (files.begin(), end, !new_); } addOptionalGmsts(); addOptionalGlobals(); connect (&mUndoStack, SIGNAL (cleanChanged (bool)), this, SLOT (modificationStateChanged (bool))); connect (&mTools, SIGNAL (progress (int, int, int)), this, SLOT (progress (int, int, int))); connect (&mTools, SIGNAL (done (int)), this, SLOT (operationDone (int))); // dummy implementation -> remove when proper save is implemented. mSaveCount = 0; connect (&mSaveTimer, SIGNAL(timeout()), this, SLOT (saving())); } CSMDoc::Document::~Document() {} QUndoStack& CSMDoc::Document::getUndoStack() { return mUndoStack; } int CSMDoc::Document::getState() const { int state = 0; if (!mUndoStack.isClean()) state |= State_Modified; if (mSaveCount) state |= State_Locked | State_Saving | State_Operation; if (int operations = mTools.getRunningOperations()) state |= State_Locked | State_Operation | operations; return state; } const std::string& CSMDoc::Document::getName() const { return mName; } void CSMDoc::Document::save() { mSaveCount = 1; mSaveTimer.start (500); emit stateChanged (getState(), this); emit progress (1, 16, State_Saving, 1, this); } CSMWorld::UniversalId CSMDoc::Document::verify() { CSMWorld::UniversalId id = mTools.runVerifier(); emit stateChanged (getState(), this); return id; } void CSMDoc::Document::abortOperation (int type) { mTools.abortOperation (type); if (type==State_Saving) { mSaveCount=0; mSaveTimer.stop(); emit stateChanged (getState(), this); } } void CSMDoc::Document::modificationStateChanged (bool clean) { emit stateChanged (getState(), this); } void CSMDoc::Document::operationDone (int type) { emit stateChanged (getState(), this); } void CSMDoc::Document::saving() { ++mSaveCount; emit progress (mSaveCount, 16, State_Saving, 1, this); if (mSaveCount>15) { //clear the stack before resetting the save state //to avoid emitting incorrect states mUndoStack.setClean(); mSaveCount = 0; mSaveTimer.stop(); emit stateChanged (getState(), this); } } const CSMWorld::Data& CSMDoc::Document::getData() const { return mData; } CSMWorld::Data& CSMDoc::Document::getData() { return mData; } CSMTools::ReportModel *CSMDoc::Document::getReport (const CSMWorld::UniversalId& id) { return mTools.getReport (id); } void CSMDoc::Document::progress (int current, int max, int type) { emit progress (current, max, type, 1, this); }