#include <memory>
#include "iocommand.h"
// Toonz includes
#include "menubarcommandids.h"
#include "filebrowserpopup.h"
#include "tapp.h"
#include "history.h"
#include "fileselection.h"
#include "previewer.h"
#include "previewfxmanager.h"
#include "tcontenthistory.h"
#include "filebrowsermodel.h"
#include "mainwindow.h"
#include "overwritepopup.h"
#include "cleanupsettingspopup.h"
#include "psdsettingspopup.h"
#include "filebrowser.h"
#include "versioncontrol.h"
#include "cachefxcommand.h"
// TnzTools includes
#include "tools/toolhandle.h"
// ToonzQt includes
#include "toonzqt/gutil.h"
#include "toonzqt/icongenerator.h"
#include "toonzqt/swatchviewer.h"
#include "toonzqt/tselectionhandle.h"
#include "toonzqt/dvdialog.h"
// ToonzLib includes
#include "toonz/palettecontroller.h"
#include "toonz/tscenehandle.h"
#include "toonz/tobjecthandle.h"
#include "toonz/tcolumnhandle.h"
#include "toonz/tframehandle.h"
#include "toonz/txsheethandle.h"
#include "toonz/txshlevelhandle.h"
#include "toonz/tpalettehandle.h"
#include "toonz/toonzscene.h"
#include "toonz/tproject.h"
#include "toonz/txshsimplelevel.h"
#include "toonz/txshchildlevel.h"
#include "toonz/sceneproperties.h"
#include "toonz/levelproperties.h"
#include "toonz/stage2.h"
#include "toonz/imagemanager.h"
#include "toonz/sceneresources.h"
#include "toonz/txshsoundlevel.h"
#include "toonz/txshpalettecolumn.h"
#include "toonz/txshpalettelevel.h"
#include "toonz/txshleveltypes.h"
#include "toonz/txshsoundtextcolumn.h"
#include "toonz/tstageobjecttree.h"
#include "toonz/levelset.h"
#include "toonz/namebuilder.h"
#include "toonz/fullcolorpalette.h"
#include "toonz/palettecmd.h"
#include "toonz/toonzimageutils.h"
#include "toonz/imagestyles.h"
#include "toutputproperties.h"
#include "toonz/studiopalette.h"
// TnzCore includes
#include "tofflinegl.h"
#include "tvectorimage.h"
#include "tvectorrenderdata.h"
#include "tropcm.h"
#include "tfiletype.h"
#include "tunit.h"
#include "tproperty.h"
#include "tlevel.h"
#include "tlevel_io.h"
// Qt includes
#include <QLabel>
#include <QApplication>
#include <QClipboard>
// boost includes
#include <boost/optional.hpp>
#include <boost/algorithm/cxx11/all_of.hpp>
#include <boost/utility/in_place_factory.hpp>
//#define USE_SQLITE_HDPOOL
using namespace DVGui;
namespace ba = boost::algorithm;
//-----------------------------------------------------------------------------
namespace
{
//-----------------------------------------------------------------------------
TXshLevel *getLevelByPath(ToonzScene *scene, const TFilePath &actualPath);
// forward declaration
class RenderingSuspender;
//===========================================================================
// class ResourceImportDialog
//---------------------------------------------------------------------------
class ResourceImportDialog : public ResourceImportStrategy
{
OverwriteDialog *m_dialog;
bool m_isLastResource;
bool m_importQuestionAsked;
bool m_aborted;
TFilePath m_dstFolder;
bool m_importEnabled;
std::map<TFilePath, TFilePath> m_importedFiles;
public:
enum Resolution { A_IMPORT,
A_LOAD,
A_CANCEL };
public:
ResourceImportDialog()
: ResourceImportStrategy(ResourceImportStrategy::IMPORT_AND_RENAME), m_dialog(0), m_isLastResource(false), m_importQuestionAsked(false), m_aborted(false), m_importEnabled(false)
{
m_dialog = new OverwriteDialog;
}
~ResourceImportDialog()
{
if (m_dialog)
m_dialog->deleteLater();
}
bool isImportEnabled() const { return m_importEnabled; }
void setImportEnabled(bool enabled)
{
m_importQuestionAsked = true, m_importEnabled = enabled;
}
void setDstFolder(const TFilePath &dstFolder) { m_dstFolder = dstFolder; }
TFilePath getDstFolder() const { return m_dstFolder; }
int askImportQuestion(const TFilePath &path)
{
if (m_importQuestionAsked)
return isImportEnabled() ? A_IMPORT : A_LOAD;
else {
m_importQuestionAsked = true;
QString label = QObject::tr("File %1 doesn't belong to the current project.\n"
"Do you want to import it or load it from its original location?")
.arg(QString::fromStdWString(path.getWideString()));
QStringList buttons;
buttons << QObject::tr("Import") << QObject::tr("Load") << QObject::tr("Cancel");
DVGui::Dialog *importDialog = DVGui::createMsgBox(DVGui::QUESTION,
label,
buttons,
0);
int ret = importDialog->exec();
importDialog->deleteLater();
if (ret == 0 || ret == 3) {
m_aborted = true;
return A_CANCEL;
}
m_importEnabled = (ret == 1);
return ret == 1 ? A_IMPORT : A_LOAD;
}
}
void setIsLastResource(bool isLastResource) { m_isLastResource = isLastResource; }
bool aborted() const { return m_aborted || m_dialog->cancelPressed(); }
//
// process the file 'srcPath' (possibly copying it)
// scrPath can be a coded path (related to srcScene); srcScene can be 0
// the method returns the processed codedPath (related to scene)
//
TFilePath process(ToonzScene *scene, ToonzScene *srcScene, TFilePath srcPath)
{
TFilePath actualSrcPath = srcPath;
if (srcScene)
actualSrcPath = srcScene->decodeFilePath(srcPath);
if (!isImportEnabled()) {
TFilePath path = scene->codeFilePath(actualSrcPath);
return path;
}
if ((!scene->isExternPath(actualSrcPath) && m_dstFolder.isEmpty()) || m_dialog->cancelPressed())
return srcPath;
// find the proper dstPath (coded)
TFilePath dstPath;
if (srcPath.getWideString().find(L'+') == 0) {
dstPath = srcPath;
// override the folder
if (m_dstFolder != TFilePath()) {
// +drawings/oldfolder/level.pli => +drawings/xsheetfolder/oldfolder/level.pli
wstring head;
TFilePath tail;
dstPath.split(head, tail);
dstPath = TFilePath(head) + m_dstFolder + tail;
}
} else {
dstPath = scene->getImportedLevelPath(srcPath);
// override the folder
if (m_dstFolder != TFilePath())
dstPath = dstPath.withParentDir(dstPath.getParentDir() + m_dstFolder);
}
// actual path
TFilePath actualDstPath = scene->decodeFilePath(dstPath);
assert(actualDstPath != TFilePath());
std::map<TFilePath, TFilePath>::iterator it = m_importedFiles.find(actualDstPath);
if (it != m_importedFiles.end())
return it->second;
m_importedFiles[actualDstPath] = dstPath;
// possibly, a level already exists
bool overwritten = false;
if (TSystem::doesExistFileOrLevel(actualDstPath)) {
std::wstring newName = m_dialog->execute(scene, dstPath, m_isLastResource == false);
if (m_dialog->cancelPressed())
return srcPath;
int importMode = m_dialog->getChoice();
if (importMode == OverwriteDialog::KEEP_OLD)
return dstPath;
else if (importMode == OverwriteDialog::OVERWRITE)
overwritten = true;
else {
dstPath = dstPath.withName(newName);
m_importedFiles[actualDstPath] = dstPath;
actualDstPath = actualDstPath.withName(newName);
}
}
// copy resources
try {
if (TSystem::doesExistFileOrLevel(actualDstPath))
TSystem::removeFileOrLevel(actualDstPath);
if (TSystem::doesExistFileOrLevel(actualSrcPath))
TXshSimpleLevel::copyFiles(actualDstPath, actualSrcPath);
} catch (TException &e) {
DVGui::Dialog *errorDialog = DVGui::createMsgBox(
DVGui::WARNING,
"Can't copy resources: " + QString::fromStdWString(e.getMessage()),
QStringList("OK"),
0);
errorDialog->exec();
errorDialog->deleteLater();
}
// notify
FileBrowser::refreshFolder(actualDstPath.getParentDir());
IconGenerator::instance()->invalidate(actualDstPath);
// refresh icons and level path
if (overwritten) {
TXshLevel *xl = getLevelByPath(scene, actualDstPath);
TXshSimpleLevel *sl = 0;
if (xl && 0 != (sl = xl->getSimpleLevel())) {
std::vector<TFrameId> fids;
sl->getFids(fids);
sl->setPath(sl->getPath(), false);
for (int i = 0; i < (int)fids.size(); i++)
IconGenerator::instance()->invalidate(sl, fids[i]);
}
}
return dstPath;
}
};
//===========================================================================
// getLevelByPath(scene, actualPath)
//---------------------------------------------------------------------------
TXshLevel *getLevelByPath(ToonzScene *scene, const TFilePath &actualPath)
{
TLevelSet *levelSet = scene->getLevelSet();
for (int i = 0; i < levelSet->getLevelCount(); i++) {
TXshLevel *xl = levelSet->getLevel(i);
if (!xl)
continue;
TFilePath fp = scene->decodeFilePath(xl->getPath());
if (fp == actualPath)
return xl;
}
return 0;
}
//===========================================================================
// getSimpleLevelByPath(scene, actualPath)
//---------------------------------------------------------------------------
TXshSimpleLevel *getSimpleLevelByPath(ToonzScene *scene, const TFilePath &actualPath)
{
TXshLevel *xl = getLevelByPath(scene, actualPath);
if (!xl)
return 0;
TXshSimpleLevel *sl = xl->getSimpleLevel();
return sl;
}
//===========================================================================
// beforeCellsInsert(xsh, row, col, rowCount)
//---------------------------------------------------------------------------
bool beforeCellsInsert(TXsheet *xsh, int row, int &col, int rowCount, int newLevelColumnType)
{
bool shiftColumn = false;
int i = 0;
TXshColumn *column = xsh->getColumn(col);
for (i = 0; i < rowCount && xsh->getCell(row + i, col).isEmpty(); i++) {
}
int type = column ? column->getColumnType() : newLevelColumnType;
//If some used cells in range or column type mismatch must insert a column.
if (i < rowCount || newLevelColumnType != type) {
col += 1;
TApp::instance()->getCurrentColumn()->setColumnIndex(col);
shiftColumn = true;
xsh->insertColumn(col);
} else {
// don't overlap
xsh->removeCells(row, col, rowCount);
}
return shiftColumn;
}
//===========================================================================
// getLevelType(actualLevelPath)
//---------------------------------------------------------------------------
int getLevelType(const TFilePath &actualPath)
{
TFileType::Type type = TFileType::getInfo(actualPath);
if (type == TFileType::RASTER_IMAGE ||
type == TFileType::RASTER_LEVEL ||
type == TFileType::CMAPPED_LEVEL) {
string ext = actualPath.getType();
if (ext == "tzp" || ext == "tzu" || ext == "tzl" || ext == "tlv")
return TZP_XSHLEVEL;
else
return OVL_XSHLEVEL;
} else if (type == TFileType::VECTOR_LEVEL) {
return PLI_XSHLEVEL;
} else
return UNKNOWN_XSHLEVEL;
}
//===========================================================================
// class LoadLevelUndo
//---------------------------------------------------------------------------
class LoadLevelUndo : public TUndo
{
TXshLevelP m_level;
TFilePath m_levelSetFolder;
int m_row, m_col, m_rowCount;
bool m_columnInserted;
std::vector<TXshCell> m_cells;
bool m_isFirstTime;
public:
LoadLevelUndo()
: m_level(), m_levelSetFolder(), m_row(0), m_col(0), m_rowCount(0), m_columnInserted(false), m_isFirstTime(true)
{
}
void setLevel(TXshLevel *xl) { m_level = xl; }
void setLevelSetFolder(const TFilePath &levelSetFolder) { m_levelSetFolder = levelSetFolder; }
void setIsFirstTime(bool flag) { m_isFirstTime = flag; }
void setCells(TXsheet *xsh, int row, int col, int rowCount)
{
m_row = row;
m_col = col;
m_rowCount = rowCount;
if (rowCount > 0) {
m_cells.resize(rowCount);
xsh->getCells(row, col, rowCount, &m_cells[0]);
} else
m_cells.clear();
}
void setColumnInserted(bool columnInserted) { m_columnInserted = columnInserted; }
void undo() const
{
TApp *app = TApp::instance();
/*- 最初にシーンに読み込んだ操作のUndoのとき、Castから除く -*/
if (m_level && m_isFirstTime)
app->getCurrentScene()->getScene()->getLevelSet()->removeLevel(m_level.getPointer());
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
if (m_columnInserted)
xsh->removeColumn(m_col);
else {
xsh->removeCells(m_row, m_col, m_rowCount);
xsh->insertCells(m_row, m_col, m_rowCount); // insert empty cells
//Remove sound or palette column if it is empty.
if (xsh->getColumn(m_col) && xsh->getColumn(m_col)->isEmpty()) {
TXshColumn::ColumnType columnType = xsh->getColumn(m_col)->getColumnType();
if (columnType != TXshColumn::eLevelType) {
TStageObjectTree *stageObjectTree = xsh->getStageObjectTree();
int lastIndex = stageObjectTree->getStageObjectCount();
stageObjectTree->getStageObject(TStageObjectId::ColumnId(lastIndex), true);
stageObjectTree->swapColumns(m_col, lastIndex);
xsh->removeColumn(m_col);
xsh->insertColumn(m_col);
stageObjectTree->swapColumns(lastIndex, m_col);
}
}
}
app->getCurrentXsheet()->notifyXsheetChanged();
if (app->getCurrentFrame()->isEditingLevel() && app->getCurrentLevel()->getLevel() == m_level.getPointer())
app->getCurrentLevel()->setLevel(0);
app->getCurrentScene()->notifyCastChange();
}
void redo() const
{
TApp *app = TApp::instance();
if (m_level)
app->getCurrentScene()->getScene()->getLevelSet()->insertLevel(m_level.getPointer());
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
if (m_columnInserted)
xsh->insertColumn(m_col);
if (m_rowCount > 0) {
xsh->removeCells(m_row, m_col, m_rowCount);
xsh->insertCells(m_row, m_col, m_rowCount);
xsh->setCells(m_row, m_col, m_rowCount, &m_cells[0]);
}
app->getCurrentXsheet()->notifyXsheetChanged();
if (app->getCurrentFrame()->isEditingLevel() && app->getCurrentLevel()->getLevel() == 0)
app->getCurrentLevel()->setLevel(m_level.getPointer());
app->getCurrentScene()->notifyCastChange();
}
int getSize() const
{
return sizeof(*this) + sizeof(TXshCell) * m_cells.size() + sizeof(TXshLevel);
}
QString getHistoryString()
{
return QObject::tr("Load Level %1").arg(QString::fromStdWString(m_level->getName()));
}
};
//===========================================================================
// class LoadLevelAndReplaceUndo
//---------------------------------------------------------------------------
class LoadAndReplaceLevelUndo : public TUndo
{
TXshSimpleLevelP m_level;
QMap<QPair<int, int>, QPair<TXshSimpleLevelP, TFrameId>> m_oldLevels;
int m_row0, m_col0, m_row1, m_col1;
public:
LoadAndReplaceLevelUndo(
const TXshSimpleLevelP &level,
int row0, int col0, int row1, int col1)
: m_level(level), m_row0(row0), m_row1(row1), m_col0(col0), m_col1(col1)
{
int c, r;
TXsheet *xsh = TApp::instance()->getCurrentXsheet()->getXsheet();
for (c = m_col0; c <= m_col1; c++)
for (r = m_row0; r <= m_row1; r++) {
TXshSimpleLevel *oldLevel = xsh->getCell(r, c).getSimpleLevel();
TFrameId fid = xsh->getCell(r, c).getFrameId();
QPair<int, int> cellId(r, c);
m_oldLevels[cellId] = QPair<TXshSimpleLevelP, TFrameId>(oldLevel, fid);
}
}
void undo() const
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
int c, r;
for (c = m_col0; c <= m_col1; c++)
for (r = m_row0; r <= m_row1; r++) {
QPair<int, int> cellId(r, c);
TXshSimpleLevel *oldLevel = m_oldLevels[cellId].first.getPointer();
TFrameId fid = m_oldLevels[cellId].second;
TXshCell cell = xsh->getCell(r, c);
cell.m_level = oldLevel;
cell.m_frameId = fid;
xsh->setCell(r, c, cell);
}
app->getCurrentXsheet()->notifyXsheetChanged();
app->getCurrentScene()->notifyCastChange();
}
void redo() const
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
int c, r;
for (c = m_col0; c <= m_col1; c++)
for (r = m_row0; r <= m_row1; r++) {
QPair<int, int> cellId(r, c);
TFrameId fid = m_oldLevels[cellId].second;
TXshCell cell = xsh->getCell(r, c);
cell.m_level = m_level.getPointer();
cell.m_frameId = fid;
xsh->setCell(r, c, cell);
}
app->getCurrentXsheet()->notifyXsheetChanged();
app->getCurrentScene()->notifyCastChange();
}
int getSize() const
{
return sizeof(*this);
}
QString getHistoryString()
{
return QObject::tr("Load and Replace Level %1").arg(QString::fromStdWString(m_level->getName()));
}
};
//===========================================================================
// loadPalette(scene, actualPath, row, col)
//---------------------------------------------------------------------------
TXshLevel *loadPalette(
ToonzScene *scene,
TFilePath actualPath,
TFilePath castFolder,
int row, int &col)
{
TFilePath palettePath = actualPath;
TXsheet *xsh = scene->getXsheet();
TXshPaletteColumn *column = new TXshPaletteColumn;
xsh->insertColumn(col, column);
TXshLevel *level = scene->createNewLevel(PLT_XSHLEVEL, palettePath.getWideName());
level->getPaletteLevel()->setPath(scene->codeFilePath(palettePath));
level->getPaletteLevel()->load();
TXshCell cell;
cell.m_level = level;
cell.m_frameId = TFrameId(1);
xsh->setCell(row, col, cell);
xsh->updateFrameCount();
//Undo
LoadLevelUndo *undo = new LoadLevelUndo();
undo->setLevel(level);
undo->setLevelSetFolder(castFolder);
undo->setCells(scene->getXsheet(), row, col, 1);
undo->setColumnInserted(true);
TUndoManager::manager()->add(undo);
return level;
}
//===========================================================================
// substituteLevel() ; TODO: move to a different file
//---------------------------------------------------------------------------
void substituteLevel(TXsheet *xsh, TXshLevel *srcLevel, TXshLevel *dstLevel)
{
std::set<TXshChildLevel *> substitutedSubs;
for (int c = 0; c < xsh->getColumnCount(); c++) {
int r0 = 0, r1 = -1;
xsh->getCellRange(c, r0, r1);
if (r0 > r1)
continue;
int rowCount = r1 - r0 + 1;
std::vector<TXshCell> cells(rowCount);
xsh->getCells(r0, c, rowCount, &cells[0]);
bool changed = false;
for (int i = 0; i < rowCount; i++) {
if (!cells[i].isEmpty()) {
if (cells[i].m_level.getPointer() == srcLevel) {
cells[i].m_level = dstLevel;
changed = true;
} else {
//Recursive on sub-xsheets
TXshChildLevel *childLevel = cells[i].m_level->getChildLevel();
if (childLevel)
if (substitutedSubs.find(childLevel) == substitutedSubs.end()) {
substituteLevel(childLevel->getXsheet(), srcLevel, dstLevel);
substitutedSubs.insert(childLevel);
}
}
}
}
if (changed)
xsh->setCells(r0, c, rowCount, &cells[0]);
}
}
//===========================================================================
// class ChildLevelResourceImporter
//---------------------------------------------------------------------------
class ChildLevelResourceImporter : public ResourceProcessor
{
ToonzScene *m_parentScene;
ToonzScene *m_childScene;
ResourceImportStrategy &m_importStrategy;
TFilePath m_levelSetFolder;
public:
ChildLevelResourceImporter(
ToonzScene *parentScene,
ToonzScene *childScene,
ResourceImportStrategy &importStrategy);
void setLevelSetFolder(const TFilePath &levelSetFolder) { m_levelSetFolder = levelSetFolder; }
void process(TXshSimpleLevel *sl);
void process(TXshPaletteLevel *sl);
void process(TXshSoundLevel *sl);
bool aborted() const { return m_importStrategy.aborted(); }
};
//---------------------------------------------------------------------------
ChildLevelResourceImporter::ChildLevelResourceImporter(
ToonzScene *parentScene,
ToonzScene *childScene,
ResourceImportStrategy &importStrategy)
: m_parentScene(parentScene), m_childScene(childScene), m_importStrategy(importStrategy), m_levelSetFolder()
{
}
//---------------------------------------------------------------------------
void ChildLevelResourceImporter::process(TXshSimpleLevel *sl)
{
TLevelSet *parentLevelSet = m_parentScene->getLevelSet();
if (parentLevelSet->hasLevel(sl->getName())) {
TXshSimpleLevel *other = parentLevelSet->getLevel(sl->getName())->getSimpleLevel();
if (other) // in case that the retrieved level from scene cast is subXSheetLevel
{
TFilePath otherActualPath = m_parentScene->decodeFilePath(other->getPath());
TFilePath slActualPath = m_childScene->decodeFilePath(sl->getPath());
if (otherActualPath == slActualPath &&
other->getProperties()->options() == sl->getProperties()->options()) {
substituteLevel(m_childScene->getXsheet(), sl, other);
return;
}
}
}
TFilePath slPath = sl->getPath();
string suffix = ResourceImporter::extractPsdSuffix(slPath);
TFilePath path = m_importStrategy.process(m_parentScene, m_childScene, slPath); // actualPath);
if (suffix != "")
path = ResourceImporter::buildPsd(path, suffix);
sl->setPath(path, false); // m_parentScene->codeFilePath(actualPath), false);
NameModifier nm(sl->getName());
wstring levelName;
for (;;) {
levelName = nm.getNext();
if (!parentLevelSet->hasLevel(levelName))
break;
}
assert(sl->getRefCount() > 0);
sl->addRef();
m_childScene->getLevelSet()->removeLevel(sl);
sl->setName(levelName);
bool ret = parentLevelSet->insertLevel(sl);
assert(ret);
if (m_levelSetFolder != TFilePath())
parentLevelSet->moveLevelToFolder(m_levelSetFolder, sl);
sl->setScene(m_parentScene);
try {
sl->load();
} catch (...) {
}
sl->release();
}
//---------------------------------------------------------------------------
void ChildLevelResourceImporter::process(TXshPaletteLevel *pl)
{
TLevelSet *parentLevelSet = m_parentScene->getLevelSet();
TFilePath plPath = pl->getPath();
TFilePath path = m_importStrategy.process(m_parentScene, m_childScene, plPath);
pl->setPath(path);
NameModifier nm(pl->getName());
wstring levelName;
for (;;) {
levelName = nm.getNext();
if (!parentLevelSet->hasLevel(levelName))
break;
}
assert(pl->getRefCount() > 0);
pl->addRef();
m_childScene->getLevelSet()->removeLevel(pl);
pl->setName(levelName);
bool ret = parentLevelSet->insertLevel(pl);
assert(ret);
if (m_levelSetFolder != TFilePath())
parentLevelSet->moveLevelToFolder(m_levelSetFolder, pl);
pl->setScene(m_parentScene);
try {
pl->load();
} catch (...) {
}
pl->release();
}
void ChildLevelResourceImporter::process(TXshSoundLevel *sl)
{
TLevelSet *parentLevelSet = m_parentScene->getLevelSet();
TFilePath path = m_importStrategy.process(m_parentScene, m_childScene, sl->getPath());
sl->setPath(path);
NameModifier nm(sl->getName());
wstring levelName;
for (;;) {
levelName = nm.getNext();
if (!parentLevelSet->hasLevel(levelName))
break;
}
assert(sl->getRefCount() > 0);
sl->addRef();
m_childScene->getLevelSet()->removeLevel(sl);
sl->setName(levelName);
bool ret = parentLevelSet->insertLevel(sl);
assert(ret);
if (m_levelSetFolder != TFilePath())
parentLevelSet->moveLevelToFolder(m_levelSetFolder, sl);
sl->setScene(m_parentScene);
try {
sl->loadSoundTrack();
} catch (...) {
}
sl->release();
}
//===========================================================================
// loadChildLevel(parentScene, actualPath, row, col, importStrategy)
//---------------------------------------------------------------------------
TXshLevel *loadChildLevel(ToonzScene *parentScene, TFilePath actualPath, int row, int &col, ResourceImportDialog &importStrategy)
{
TProjectP project = TProjectManager::instance()->loadSceneProject(actualPath);
// In Tab mode we don't need the project. Otherwise if it is not available
// we must exit
if (!project && !TProjectManager::instance()->isTabModeEnabled())
return 0;
// load the subxsheet
ToonzScene scene;
scene.loadTnzFile(actualPath);
scene.setProject(project.getPointer());
wstring subSceneName = actualPath.getWideName();
// camera settings. get the child camera ...
TXsheet *childXsh = scene.getXsheet();
TStageObjectId cameraId = childXsh->getStageObjectTree()->getCurrentCameraId();
TStageObject *childCameraObject = childXsh->getStageObject(cameraId);
TCamera *childCamera = childCameraObject->getCamera();
// ...and the parent camera...
TXsheet *parentXsh = parentScene->getXsheet();
cameraId = parentXsh->getStageObjectTree()->getCurrentCameraId();
TStageObject *cameraObject = parentXsh->getStageObject(cameraId);
TCamera *camera = cameraObject->getCamera();
// if the camera settings are different ask the user
const double eps = 0.00001;
if (fabs(camera->getSize().lx - childCamera->getSize().lx) > eps ||
fabs(camera->getSize().ly - childCamera->getSize().ly) > eps ||
camera->getRes() != childCamera->getRes()) {
QString question(QObject::tr("The camera settings of the scene you are loading as sub-xsheet are different from those of your current scene. What you want to do?"));
QList<QString> list;
list.append(QObject::tr("Keep the sub-xsheet original camera settings."));
list.append(QObject::tr("Apply the current scene camera settings to the sub-xsheet."));
int ret = DVGui::RadioButtonMsgBox(DVGui::WARNING, question, list);
if (ret == 0)
return 0;
if (ret == 2) {
childCamera->setRes(camera->getRes());
childCamera->setSize(camera->getSize());
}
}
ChildLevelResourceImporter childLevelResourceImporter(parentScene, &scene, importStrategy);
TFilePath dstFolder = importStrategy.getDstFolder();
if (dstFolder != TFilePath()) {
TLevelSet *levelSet = parentScene->getLevelSet();
TFilePath levelSetFolder = levelSet->createFolder(levelSet->getDefaultFolder() + dstFolder.getParentDir(), dstFolder.getWideName());
childLevelResourceImporter.setLevelSetFolder(levelSetFolder);
}
SceneResources resources(&scene, childXsh);
resources.accept(&childLevelResourceImporter);
scene.setProject(parentScene->getProject());
std::vector<TXshChildLevel *> childLevels;
for (int i = 0; i < scene.getLevelSet()->getLevelCount(); i++)
if (TXshChildLevel *cl = scene.getLevelSet()->getLevel(i)->getChildLevel())
childLevels.push_back(cl);
for (int i = 0; i < (int)childLevels.size(); i++) {
TXshChildLevel *cl = childLevels[i];
parentScene->getLevelSet()->insertLevel(cl);
// scene.getLevelSet()->removeLevel(cl);
}
TXshChildLevel *childLevel =
parentScene->createNewLevel(CHILD_XSHLEVEL, subSceneName)->getChildLevel();
int frameCount = scene.getXsheet()->getFrameCount();
childXsh->setScene(parentScene);
if (frameCount <= 0)
frameCount = 1;
int dummy, lastOldRow;
parentXsh->getCellRange(col, dummy, lastOldRow);
childLevel->setXsheet(childXsh);
childXsh->getStageObjectTree()->invalidateAll();
bool shiftColumn = beforeCellsInsert(parentXsh, row, col, frameCount, TXshColumn::eLevelType);
if (!shiftColumn)
parentXsh->insertCells(row, col, frameCount);
for (int i = 0; i < frameCount; i++)
parentXsh->setCell(row + i, col, TXshCell(childLevel, TFrameId(1 + i)));
for (int i = 0; i < parentScene->getLevelSet()->getLevelCount(); i++)
parentScene->getLevelSet()->getLevel(i)->setScene(parentScene);
//Inform the cache fx command that a scene was loaded
CacheFxCommand::instance()->onSceneLoaded();
return childLevel;
}
//===========================================================================
// loadLevel(scene, path, castFolder, row, col)
// (path can be coded path)
// for loading the all types of level other than scene/palette/sound
//---------------------------------------------------------------------------
TXshLevel *loadLevel(
ToonzScene *scene,
const IoCmd::LoadResourceArguments::ResourceData &rd,
const TFilePath &castFolder,
int row0, int &col0, int row1, int &col1,
bool expose,
std::vector<TFrameId> &fIds,
int xFrom = -1,
int xTo = -1,
std::wstring levelName = L"",
int step = -1,
int inc = -1,
int frameCount = -1,
bool doesFileActuallyExist = true)
{
TFilePath actualPath = scene->decodeFilePath(rd.m_path);
LoadLevelUndo *undo = 0;
LoadAndReplaceLevelUndo *replaceUndo = 0;
TXsheet *xsh = scene->getXsheet();
TFileType::Type type = TFileType::getInfo(actualPath);
// try to find the level with the same path in the Scene Cast. If found, reuse it
TXshLevel *xl = getLevelByPath(scene, actualPath);
bool isFirstTime = !xl;
std::wstring name = actualPath.getWideName();
IoCmd::ConvertingPopup *convertingPopup = new IoCmd::ConvertingPopup(
TApp::instance()->getMainWindow(),
QString::fromStdWString(name) + QString::fromStdString(actualPath.getDottedType()));
convertingPopup->hide(); // Should be unnecessary
if (!xl) {
try {
string format = actualPath.getType();
if (format == "tzp" || format == "tzu")
convertingPopup->show();
if (fIds.size() != 0 && doesFileActuallyExist)
xl = scene->loadLevel(actualPath, rd.m_options ? &*rd.m_options : 0, levelName, fIds);
else
xl = scene->loadLevel(actualPath, rd.m_options ? &*rd.m_options : 0);
if (!xl) {
error("Failed to create level " + toQString(actualPath) + " : this filetype is not supported.");
return 0;
}
TXshSimpleLevel *sl = xl->getSimpleLevel();
if (sl && sl->isReadOnly() &&
(sl->getType() == PLI_XSHLEVEL || sl->getType() == TZP_XSHLEVEL || sl->getType() == OVL_XSHLEVEL)) {
TLevelSet *levelSet = new TLevelSet;
levelSet->insertLevel(sl);
VersionControlManager::instance()->setFrameRange(levelSet, true);
}
if (convertingPopup->isVisible())
convertingPopup->hide();
} catch (TException &e) {
if (convertingPopup->isVisible())
convertingPopup->hide();
QString msg = QString::fromStdWString(e.getMessage());
if (msg == QString("Old 4.1 Palette"))
error("It is not possible to load the level " + toQString(actualPath) + " because its version is not supported.");
else
error(QString::fromStdWString(e.getMessage()));
return 0;
}
if (xl->getSimpleLevel() && xl->getSimpleLevel()->getProperties()->isForbidden()) {
error("It is not possible to load the level " + toQString(actualPath) + " because its version is not supported.");
scene->getLevelSet()->removeLevel(xl);
return 0;
}
if (castFolder != TFilePath())
scene->getLevelSet()->moveLevelToFolder(castFolder, xl);
History::instance()->addItem(actualPath);
if (row1 == -1 && col1 == -1) {
undo = new LoadLevelUndo();
undo->setLevel(xl);
undo->setLevelSetFolder(castFolder);
undo->setIsFirstTime(isFirstTime);
}
}
// if the level can be obtained (from scene cast or file)
if (xl) {
//placing in the xsheet
if (expose) {
// lo importo nell'xsheet
if (!undo) {
undo = new LoadLevelUndo();
undo->setLevel(xl);
undo->setIsFirstTime(isFirstTime);
}
int levelType = xl->getType();
TXshColumn::ColumnType newLevelColumnType = TXshColumn::toColumnType(levelType);
bool columnInserted = beforeCellsInsert(scene->getXsheet(), row0, col0, xl->getFrameCount(), newLevelColumnType);
scene->getXsheet()->exposeLevel(row0, col0, xl,
fIds,
xFrom,
xTo,
step,
inc,
frameCount,
doesFileActuallyExist);
if (frameCount > 0)
undo->setCells(scene->getXsheet(), row0, col0, frameCount);
else
undo->setCells(scene->getXsheet(), row0, col0, xl->getFrameCount());
undo->setColumnInserted(columnInserted);
}
if (row1 != -1 || col1 != -1)
replaceUndo = new LoadAndReplaceLevelUndo(xl->getSimpleLevel(), row0, col0, row1, col1);
}
if (undo)
TUndoManager::manager()->add(undo);
if (replaceUndo)
TUndoManager::manager()->add(replaceUndo);
return xl;
}
//===========================================================================
// loadResource(scene, path, castFolder, row, col, expose)
//---------------------------------------------------------------------------
TXshLevel *loadResource(
ToonzScene *scene,
const IoCmd::LoadResourceArguments::ResourceData &rd,
const TFilePath &castFolder,
int row0, int &col0, int row1, int &col1,
bool expose,
std::vector<TFrameId> fIds = std::vector<TFrameId>(),
int xFrom = -1,
int xTo = -1,
std::wstring levelName = L"",
int step = -1,
int inc = -1,
int frameCount = -1,
bool doesFileActuallyExist = true)
{
IoCmd::LoadResourceArguments::ResourceData actualRd(rd);
actualRd.m_path = scene->decodeFilePath(rd.m_path);
TFileType::Type type = TFileType::getInfo(actualRd.m_path);
return (type == TFileType::PALETTE_LEVEL) ? loadPalette(scene, actualRd.m_path, castFolder, row0, col0) : loadLevel(scene,
actualRd,
castFolder,
row0, col0, row1, col1,
expose,
fIds,
xFrom,
xTo,
levelName,
step,
inc,
frameCount,
doesFileActuallyExist);
}
//===========================================================================
enum ExposeType {
eOverWrite,
eShiftCells,
eNone
};
//===========================================================================
// class ExposeLevelUndo
//---------------------------------------------------------------------------
class ExposeLevelUndo : public TUndo
{
TXshSimpleLevelP m_sl;
std::vector<TXshCell> m_oldCells;
std::vector<TFrameId> m_fids;
int m_row;
int m_col;
int m_frameCount;
bool m_insertEmptyColumn;
ExposeType m_type;
public:
ExposeLevelUndo(TXshSimpleLevel *sl, int row, int col, int frameCount, bool insertEmptyColumn, ExposeType type = eNone)
: m_sl(sl), m_row(row), m_col(col), m_frameCount(frameCount), m_insertEmptyColumn(insertEmptyColumn), m_fids(), m_type(type)
{
if (type == eOverWrite) {
TXsheet *xsh = TApp::instance()->getCurrentXsheet()->getXsheet();
int r;
for (r = row; r < frameCount + row; r++)
m_oldCells.push_back(xsh->getCell(r, col));
}
}
void setFids(const std::vector<TFrameId> &fids)
{
m_fids = fids;
}
void undo() const
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
if (m_insertEmptyColumn)
xsh->removeColumn(m_col);
else {
xsh->removeCells(m_row, m_col, m_frameCount);
if (m_type == eNone)
xsh->insertCells(m_row, m_col, m_frameCount);
if (m_type == eOverWrite) {
xsh->insertCells(m_row, m_col, m_frameCount);
if (!m_oldCells.empty())
xsh->setCells(m_row, m_col, m_frameCount, &m_oldCells[0]);
}
app->getCurrentXsheet()->notifyXsheetChanged();
}
}
void redo() const
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
if (m_insertEmptyColumn)
xsh->insertColumn(m_col);
int frameCount = 0;
if (!m_fids.empty()) {
if (m_type == eShiftCells)
xsh->insertCells(m_row, m_col, m_frameCount);
frameCount = (int)m_fids.size();
std::vector<TFrameId>::const_iterator it;
int row = m_row;
for (it = m_fids.begin(); it != m_fids.end(); ++it) {
xsh->setCell(row, m_col, TXshCell(m_sl.getPointer(), *it));
++row;
}
} else {
xsh->removeCells(m_row, m_col, m_frameCount);
frameCount = xsh->exposeLevel(m_row, m_col, m_sl.getPointer());
}
app->getCurrentXsheet()->notifyXsheetChanged();
}
int getSize() const { return sizeof *this; }
QString getHistoryString()
{
return QObject::tr("Expose Level %1")
.arg(QString::fromStdWString(m_sl->getName()));
}
};
//===========================================================================
// class ExposeCommentUndo
//---------------------------------------------------------------------------
class ExposeCommentUndo : public TUndo
{
TXshSoundTextColumnP m_soundtextColumn;
int m_col;
bool m_insertEmptyColumn;
QString m_columnName;
public:
ExposeCommentUndo(TXshSoundTextColumn *soundtextColumn, int col, bool insertEmptyColumn, QString columnName)
: m_soundtextColumn(soundtextColumn), m_col(col), m_insertEmptyColumn(insertEmptyColumn), m_columnName(columnName)
{
}
void undo() const
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
xsh->removeColumn(m_col);
if (!m_insertEmptyColumn)
xsh->insertColumn(m_col);
app->getCurrentXsheet()->notifyXsheetChanged();
}
void redo() const
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
if (m_insertEmptyColumn)
xsh->insertColumn(m_col);
xsh->insertColumn(m_col, m_soundtextColumn.getPointer());
//Setto il nome di dafult della colonna al nome del file magpie.
TStageObject *obj = xsh->getStageObject(TStageObjectId::ColumnId(m_col));
std::string str = m_columnName.toStdString();
obj->setName(str);
app->getCurrentXsheet()->notifyXsheetChanged();
}
int getSize() const
{
return sizeof *this;
}
};
//=============================================================================
//This class is intended to serve as inhibitor for Previewer and SwatchViewer's activity
class RenderingSuspender
{
public:
RenderingSuspender()
{
Previewer::suspendRendering(true);
PreviewFxManager::suspendRendering(true);
SwatchViewer::suspendRendering(true);
}
~RenderingSuspender()
{
Previewer::suspendRendering(false);
PreviewFxManager::suspendRendering(false);
SwatchViewer::suspendRendering(false);
}
};
//=============================================================================
inline TPaletteP dirtyWhite(const TPaletteP &plt)
{
TPaletteP out = TPaletteP(plt->clone());
for (int i = 0; i < out->getStyleCount(); i++) {
if (out->getStyle(i)->getMainColor() == TPixel::White) {
out->getStyle(i)->setMainColor(TPixel(254, 254, 254, 255));
}
}
return out;
}
//---------------------------------------------------------------------------
} // namespace
//---------------------------------------------------------------------------
//Per ora e' usato solo per i formato "tzp" e "tzu".
IoCmd::ConvertingPopup::ConvertingPopup(QWidget *parent, QString fileName)
: QDialog(parent)
{
setModal(true);
setWindowFlags(Qt::Dialog | Qt::WindowTitleHint);
setMinimumSize(70, 50);
QVBoxLayout *mainLayout = new QVBoxLayout;
mainLayout->setMargin(5);
mainLayout->setSpacing(0);
QLabel *label = new QLabel(QString(QObject::tr("Converting %1 images to tlv format...").arg(fileName)));
mainLayout->addWidget(label);
setLayout(mainLayout);
}
IoCmd::ConvertingPopup::~ConvertingPopup() {}
//===========================================================================
// IoCmd::saveSceneIfNeeded(message)
//---------------------------------------------------------------------------
bool IoCmd::saveSceneIfNeeded(QString msg)
{
TApp *app = TApp::instance();
bool isLevelOrSceneIsDirty = false;
if (app->getCurrentScene()->getDirtyFlag()) {
#ifndef BRAVODEMO
QString question;
question = QObject::tr("%1: the current scene has been modified.\n"
"Do you want to save your changes?")
.arg(msg);
int ret = MsgBox(question, QObject::tr("Save"), QObject::tr("Discard"), QObject::tr("Cancel"), 0);
if (ret == 0 || ret == 3) {
// cancel (or closed message box window)
return false;
} else if (ret == 1) {
// save
if (!IoCmd::saveScene())
return false;
} else if (ret == 2)
#endif
{
}
isLevelOrSceneIsDirty = true;
}
ToonzScene *scene = app->getCurrentScene()->getScene();
if (scene) {
std::vector<QString> dirtyResources;
{
SceneResources resources(scene, 0);
resources.getDirtyResources(dirtyResources);
}
if (!dirtyResources.empty()) {
QString question;
question = msg + ":" + QObject::tr(" Following file(s) are modified.\n\n");
for (int i = 0; i < dirtyResources.size(); i++) {
question += " " + dirtyResources[i] + "\n";
}
question += QObject::tr("\nAre you sure to ") + msg + QObject::tr(" anyway ?");
int ret = MsgBox(question, QObject::tr("OK"), QObject::tr("Cancel"), 0);
if (ret == 0 || ret == 2) {
// cancel (or closed message box window)
return false;
} else if (ret == 1) {
// ok
}
isLevelOrSceneIsDirty = true;
}
//--- If both the level and scene is clean, then open the quit confirmation dialog
if (!isLevelOrSceneIsDirty && msg == "Quit") {
QString question("Are you sure ?");
int ret = MsgBox(question, QObject::tr("OK"), QObject::tr("Cancel"), 0);
if (ret == 0 || ret == 2) {
// cancel (or closed message box window)
return false;
} else if (ret == 1) {
// ok
}
}
RenderingSuspender suspender;
TFilePath scenePath(scene->getScenePath());
scene->clear(); // note: this (possibly) removes the "untitled" folder
app->getCurrentScene()->notifyCastChange();
//Si deve notificare anche il cambiamento che ha subito l'xsheet.
app->getCurrentXsheet()->notifyXsheetSwitched();
FileBrowser::refreshFolder(scenePath.getParentDir());
}
app->getCurrentScene()->setDirtyFlag(false);
return true;
}
//===========================================================================
// IoCmd::newScene()
//---------------------------------------------------------------------------
void IoCmd::newScene()
{
RenderingSuspender suspender;
TApp *app = TApp::instance();
double cameraDpi = 64.0;
if (!saveSceneIfNeeded(QApplication::tr("New Scene")))
return;
IconGenerator::instance()->clearRequests();
IconGenerator::instance()->clearSceneIcons();
ImageManager::instance()->clear();
FullColorPalette::instance()->clear();
CacheFxCommand::instance()->onNewScene();
CacheFxCommand::instance()->onSceneLoaded();
#ifdef USE_SQLITE_HDPOOL
// INTERMEDIATE CACHE RESULTS MANAGEMENT: Clear all resources not accessed in the last 2 sessions
TCacheResourcePool::instance()->clearAccessedAfterSessions(1); //0 would be after this session
#endif
ToonzScene *scene = new ToonzScene();
TImageStyle::setCurrentScene(scene);
TCamera *camera = scene->getCurrentCamera();
TDimension res(768, 576);
camera->setRes(res);
camera->setSize(TDimensionD((double)res.lx / cameraDpi, (double)res.ly / cameraDpi));
#ifndef STUDENT
scene->getProperties()->setBgColor(TPixel32::White);
#else
scene->getProperties()->setBgColor(TPixel32(255, 255, 255, 0));
#endif
TProjectManager::instance()->initializeScene(scene);
//Must set current scene after initializeScene!!
app->getCurrentScene()->setScene(scene);
//initializeScene() load project cleanup palette: set it to cleanup palette handle.
TPalette *palette = scene->getProperties()->getCleanupParameters()->m_cleanupPalette.getPointer();
PaletteController *paletteController = app->getPaletteController();
paletteController->getCurrentCleanupPalette()->setPalette(palette, -1);
paletteController->editLevelPalette();
TFilePath scenePath = scene->getScenePath();
DvDirModel::instance()->refreshFolder(scenePath.getParentDir());
Previewer::clearAll();
PreviewFxManager::instance()->reset();
app->getCurrentScene()->notifyNameSceneChange();
IconGenerator::instance()->invalidateSceneIcon();
ToolHandle *toolH = TApp::instance()->getCurrentTool();
if (toolH && toolH->getTool())
toolH->getTool()->reset();
CommandManager::instance()->execute("T_Hand");
CommandManager::instance()->enable(MI_SaveSubxsheetAs, false);
app->getCurrentScene()->notifySceneChanged();
app->getCurrentScene()->setDirtyFlag(false);
app->getCurrentObject()->setIsSpline(false);
app->getCurrentColumn()->setColumnIndex(0);
CleanupParameters *cp = scene->getProperties()->getCleanupParameters();
CleanupParameters::GlobalParameters.assign(cp);
//updateCleanupSettingsPopup();
CleanupPreviewCheck::instance()->setIsEnabled(false);
CameraTestCheck::instance()->setIsEnabled(false);
SetScanCropboxCheck::instance()->setIsEnabled(false);
if (!TApp::instance()->isApplicationStarting())
QApplication::clipboard()->clear();
TSelection::setCurrent(0);
TUndoManager::manager()->reset();
bool exist = TSystem::doesExistFileOrLevel(scene->decodeFilePath(scene->getScenePath()));
QAction *act = CommandManager::instance()->getAction(MI_RevertScene);
if (act)
act->setEnabled(exist);
}
//===========================================================================
// IoCmd::saveScene(scenePath, flags)
//---------------------------------------------------------------------------
bool IoCmd::saveScene(const TFilePath &path, int flags)
{
bool overwrite = (flags & SILENTLY_OVERWRITE) != 0;
bool saveSubxsheet = (flags & SAVE_SUBXSHEET) != 0;
TApp *app = TApp::instance();
#ifdef STUDENT
CommandManager::instance()->execute("MI_RemoveUnused");
#endif
assert(!path.isEmpty());
TFilePath scenePath = path;
if (scenePath.getType() == "")
scenePath = scenePath.withType("tnz");
if (scenePath.getType() != "tnz") {
error(QObject::tr("%1 has an invalid file extension.").arg(toQString(path)));
return false;
}
TFileStatus dirStatus(scenePath.getParentDir());
if (!(dirStatus.doesExist() && dirStatus.isWritable())) {
error(QObject::tr("%1 is an invalid path.").arg(toQString(scenePath.getParentDir())));
return false;
}
if (!overwrite && TFileStatus(scenePath).doesExist()) {
QString question;
question = QObject::tr("The scene %1 already exists.\nDo you want to overwrite it?").arg(QString::fromStdString(scenePath.getName()));
int ret = MsgBox(question, QObject::tr("Overwrite"), QObject::tr("Cancel"), 0);
if (ret == 2 || ret == 0)
return false;
}
QApplication::setOverrideCursor(Qt::WaitCursor);
ToonzScene *scene = TApp::instance()->getCurrentScene()->getScene();
TXsheet *xsheet = 0;
if (saveSubxsheet)
xsheet = TApp::instance()->getCurrentXsheet()->getXsheet();
if (app->getCurrentScene()->getDirtyFlag())
scene->getContentHistory(true)->modifiedNow();
if (scene->decodeFilePath(scene->getScenePath()) != scene->decodeFilePath(scenePath)) {
IconGenerator::instance()->clearRequests();
IconGenerator::instance()->clearSceneIcons();
#ifdef USE_SQLITE_HDPOOL
//Open the new cache resources HD pool
TCacheResourcePool::instance()->setPath(
QString::fromStdWString(ToonzFolder::getCacheRootFolder().getWideString()),
QString::fromStdWString(scene->getProject()->getName().getWideName()),
QString::fromStdWString(scene->getSceneName()));
#endif
}
CleanupParameters *cp = scene->getProperties()->getCleanupParameters();
CleanupParameters oldCP(*cp);
cp->assign(&CleanupParameters::GlobalParameters);
try {
scene->save(scenePath, xsheet);
TApp::instance()->getPaletteController()->getCurrentLevelPalette()->notifyPaletteChanged(); //non toglieva l'asterisco alla paletta...forse non va qua? vinz
} catch (const TSystemException &se) {
MsgBox(WARNING, QString::fromStdWString(se.getMessage()));
} catch (...) {
MsgBox(CRITICAL, QObject::tr("Couldn't save %1").arg(toQString(scenePath)));
}
cp->assign(&oldCP);
if (!overwrite)
app->getCurrentScene()->notifyNameSceneChange();
FileBrowser::refreshFolder(scenePath.getParentDir());
IconGenerator::instance()->invalidate(scenePath);
//Le seguenti notifiche non cambiano il dirty flag del livello o della paletta
//ma sono necessarie per aggiornare le titlebar dei pannelli.
app->getCurrentLevel()->notifyLevelTitleChange();
app->getCurrentPalette()->notifyPaletteTitleChanged();
app->getCurrentScene()->setDirtyFlag(false);
History::instance()->addItem(scenePath);
RecentFiles::instance()->addFilePath(toQString(scenePath), RecentFiles::Scene);
QApplication::restoreOverrideCursor();
bool exist = TSystem::doesExistFileOrLevel(scene->decodeFilePath(scene->getScenePath()));
QAction *act = CommandManager::instance()->getAction(MI_RevertScene);
if (act)
act->setEnabled(exist);
return true;
}
//===========================================================================
// IoCmd::saveScene()
//---------------------------------------------------------------------------
bool IoCmd::saveScene()
{
#ifdef BRAVODEMO
return false;
#else
TSelection *oldSelection = TApp::instance()->getCurrentSelection()->getSelection();
ToonzScene *scene = TApp::instance()->getCurrentScene()->getScene();
if (scene->isUntitled()) {
static SaveSceneAsPopup *popup = 0;
if (!popup)
popup = new SaveSceneAsPopup();
int ret = popup->exec();
if (ret == QDialog::Accepted) {
TApp::instance()->getCurrentScene()->setDirtyFlag(false);
return true;
} else {
TApp::instance()->getCurrentSelection()->setSelection(oldSelection);
return false;
}
} else {
TFilePath fp = scene->getScenePath();
// salva la scena con il nome fp. se fp esiste gia' lo sovrascrive
return saveScene(fp, SILENTLY_OVERWRITE);
}
#endif
}
//===========================================================================
// IoCmd::saveLevel(levelPath)
//---------------------------------------------------------------------------
bool IoCmd::saveLevel(const TFilePath &path)
{
#ifdef BRAVODEMO
return false;
#else
assert(!path.isEmpty());
TApp *app = TApp::instance();
TXshSimpleLevel *sl =
dynamic_cast<TXshSimpleLevel *>(app->getCurrentLevel()->getLevel());
if (!sl)
return false;
string ext = sl->getPath().getType();
string dotts = sl->getPath().getDots();
TFilePath realPath = path;
if (realPath.getType() == "")
realPath = TFilePath(realPath.getWideString() + toWideString(dotts + ext));
saveLevel(realPath, sl, false);
RecentFiles::instance()->addFilePath(toQString(realPath), RecentFiles::Level);
TApp::instance()->getPaletteController()->getCurrentLevelPalette()->notifyPaletteSwitched();
return true;
#endif
}
//===========================================================================
// IoCmd::saveLevel()
//---------------------------------------------------------------------------
bool IoCmd::saveLevel()
{
#ifdef BRAVODEMO
return false;
#else
TApp *app = TApp::instance();
TXshSimpleLevel *sl =
dynamic_cast<TXshSimpleLevel *>(app->getCurrentLevel()->getLevel());
if (!sl)
return false;
if (sl->getPath() == TFilePath())
return false;
TFilePath path = app->getCurrentScene()->getScene()->decodeFilePath(sl->getPath());
if (path == TFilePath())
return false;
if (path.getWideString()[0] == L'+')
return false;
if (!path.isAbsolute())
return false;
if (!saveLevel(path, sl, true))
return false;
sl->setDirtyFlag(false);
//for update title bar
app->getCurrentLevel()->notifyLevelChange();
return true;
#endif
}
//===========================================================================
// IoCmd::saveLevel(levelPath, simpleLevel, overwrite)
//---------------------------------------------------------------------------
bool IoCmd::saveLevel(const TFilePath &fp, TXshSimpleLevel *sl, bool overwrite)
{
#ifdef BRAVODEMO
return false;
#else
assert(sl);
bool fileDoesExist = TSystem::doesExistFileOrLevel(fp);
if (!overwrite && fileDoesExist) {
QString question;
question = QObject::tr("The level %1 already exists.\nDo you want to overwrite it?").arg(toQString(fp));
int ret = MsgBox(question, QObject::tr("Overwrite"), QObject::tr("Cancel"), 0);
if (ret == 2 || ret == 0)
return false;
}
bool overwritePalette = false;
// Confirmation of Overwrite palette
// open dialog IF 1) the level is dirty, and 2) confirmation of overwrite palette haven't been asked
// for PLI file, do nothing
if (sl->getPalette() &&
sl->getPalette()->getAskOverwriteFlag() &&
sl->getPath().getType() != "pli") {
/*-- ファイルが存在しない場合はパレットも必ず保存する --*/
if (!fileDoesExist)
overwritePalette = true;
else
{
QString question;
question = "Palette " + QString::fromStdWString(sl->getPalette()->getPaletteName()) + ".tpl has been modified. Do you want to overwrite palette as well ?";
int ret = MsgBox(question,
QObject::tr("Overwrite Palette") /*ret = 1*/, QObject::tr("Don't Overwrite Palette") /*ret = 2*/, 0);
if (ret == 1)
overwritePalette = true;
}
}
QApplication::setOverrideCursor(Qt::WaitCursor);
try {
sl->save(fp, TFilePath(), overwritePalette);
} catch (TSystemException se) {
QApplication::restoreOverrideCursor();
MsgBox(WARNING, QString::fromStdWString(se.getMessage()));
return false;
} catch (...) {
QApplication::restoreOverrideCursor();
MsgBox(CRITICAL, QObject::tr("Couldn't save %1").arg(toQString(fp)));
return false;
}
IconGenerator::instance()->invalidate(fp);
FileBrowser::refreshFolder(fp.getParentDir());
History::instance()->addItem(fp);
if (overwritePalette ||
sl->getPath().getType() == "pli")
sl->getPalette()->setDirtyFlag(false);
else // ask only once for save palette
sl->getPalette()->setAskOverwriteFlag(false);
RecentFiles::instance()->addFilePath(toQString(fp), RecentFiles::Level);
QApplication::restoreOverrideCursor();
TApp::instance()->getCurrentLevel()->notifyLevelTitleChange();
TApp::instance()->getPaletteController()->getCurrentLevelPalette()->notifyPaletteTitleChanged();
return true;
#endif
}
//===========================================================================
// IoCmd::saveLevel(simpleLevel)
//---------------------------------------------------------------------------
bool IoCmd::saveLevel(TXshSimpleLevel *sl)
{
#ifdef BRAVODEMO
return false;
#else
return saveLevel(sl->getPath(), sl, true);
#endif
}
//===========================================================================
// IoCmd::saveSound(soundPath, soundColumn, overwrite)
//---------------------------------------------------------------------------
bool IoCmd::saveSound(const TFilePath &fp, TXshSoundLevel *sl, bool overwrite)
{
if (!overwrite && TSystem::doesExistFileOrLevel(fp)) {
QString question;
question = QObject::tr("The soundtrack %1 already exists.\nDo you want to overwrite it?").arg(toQString(fp));
int ret = MsgBox(question, QObject::tr("Overwrite"), QObject::tr("Cancel"), 0);
if (ret == 2 || ret == 0)
return false;
}
try {
sl->save(fp);
} catch (...) {
MsgBox(CRITICAL, QObject::tr("Couldn't save %1").arg(toQString(fp)));
return false;
}
QApplication::setOverrideCursor(Qt::WaitCursor);
FileBrowser::refreshFolder(fp.getParentDir());
History::instance()->addItem(fp);
QApplication::restoreOverrideCursor();
return true;
}
//===========================================================================
// IoCmd::saveSound(soundColumn)
//---------------------------------------------------------------------------
bool IoCmd::saveSound(TXshSoundLevel *sl)
{
return saveSound(sl->getPath(), sl, true);
}
//===========================================================================
// IoCmd::loadColorModel(soundColumn)
//---------------------------------------------------------------------------
bool IoCmd::loadColorModel(const TFilePath &fp, int frame)
{
TPaletteHandle *paletteHandle = TApp::instance()->getPaletteController()->getCurrentPalette();
TPalette *palette = paletteHandle->getPalette();
if (!palette || palette->isCleanupPalette()) {
error(QObject::tr("Cannot load Color Model in current palette."));
return false;
}
ResourceImportDialog importDialog;
importDialog.setIsLastResource(true);
TFilePath path = fp;
if (!path.isLevelName())
path = TFilePath(path.getLevelNameW()).withParentDir(path.getParentDir());
if (!TSystem::doesExistFileOrLevel(path))
return false;
bool importFlag = false;
ToonzScene *scene = TApp::instance()->getCurrentScene()->getScene();
if (scene->isExternPath(path)) {
// extern resource: import or link?
int ret = importDialog.askImportQuestion(path);
if (ret == ResourceImportDialog::A_CANCEL)
return true;
importFlag = (ret == ResourceImportDialog::A_IMPORT);
}
QString question(QObject::tr("The color model palette is different from the destination palette.\nWhat do you want to do? "));
QList<QString> list;
list.append(QObject::tr("Overwrite the destination palette."));
list.append(QObject::tr("Keep the destination palette and apply it to the color model."));
int ret = RadioButtonMsgBox(DVGui::WARNING, question, list);
if (ret == 0)
return false;
bool replace = false;
if (ret == 2)
replace = true;
try {
path = importDialog.process(scene, 0, path);
} catch (string msg) {
error(QString::fromStdString(msg));
return true;
}
int isLoaded = PaletteCmd::loadReferenceImage(paletteHandle, replace, path, frame, scene);
if (isLoaded != 0)
return false;
TXshLevel *level = TApp::instance()->getCurrentLevel()->getLevel();
if (!level)
return true;
std::vector<TFrameId> fids;
level->getFids(fids);
invalidateIcons(level, fids);
return true;
}
//=========================================================================
// IoCmd::loadScene(scene, scenePath, import)
//---------------------------------------------------------------------------
bool IoCmd::loadScene(ToonzScene &scene, const TFilePath &scenePath, bool import)
{
if (!TSystem::doesExistFileOrLevel(scenePath))
return false;
scene.load(scenePath);
// import if needed
TProjectManager *pm = TProjectManager::instance();
TProjectP currentProject = pm->getCurrentProject();
if (!scene.getProject())
return false;
if (scene.getProject()->getProjectPath() != currentProject->getProjectPath()) {
ResourceImportDialog resourceLoader;
// resourceLoader.setImportEnabled(true);
ResourceImporter importer(&scene, currentProject.getPointer(), resourceLoader);
SceneResources resources(&scene, scene.getXsheet());
resources.accept(&importer);
scene.setScenePath(importer.getImportedScenePath());
scene.setProject(currentProject.getPointer());
}
return true;
}
//===========================================================================
// IoCmd::loadScene(scenePath)
//---------------------------------------------------------------------------
bool IoCmd::loadScene(const TFilePath &path, bool updateRecentFile, bool checkSaveOldScene)
{
RenderingSuspender suspender;
if (checkSaveOldScene)
if (!saveSceneIfNeeded(QApplication::tr("Load Scene")))
return false;
assert(!path.isEmpty());
TFilePath scenePath = path;
bool importScene = false;
if (scenePath.getType() == "")
scenePath = scenePath.withType("tnz");
if (scenePath.getType() != "tnz") {
QString msg;
msg = QObject::tr("File %1 doesn't look like a TOONZ Scene").arg(QString::fromStdWString(scenePath.getWideString()));
MsgBox(CRITICAL, msg);
return false;
}
if (!TSystem::doesExistFileOrLevel(scenePath))
return false;
TProjectManager *pm = TProjectManager::instance();
TProjectP sceneProject = pm->loadSceneProject(scenePath);
if (!sceneProject) {
QString msg;
msg = QObject::tr("It is not possible to load the scene %1 because it does not belong to any project.").arg(QString::fromStdWString(scenePath.getWideString()));
MsgBox(WARNING, msg);
}
if (sceneProject && !sceneProject->isCurrent()) {
QString currentProjectName =
QString::fromStdWString(pm->getCurrentProject()->getName().getWideString());
QString sceneProjectName =
QString::fromStdWString(sceneProject->getName().getWideString());
/*QString question = "The Scene '"
+ QString::fromStdWString(scenePath.getWideString())
+ "' belongs to project '" + sceneProjectName + "'.\n"
+ "What do you want to do?";*/
QString question = QObject::tr("The Scene '%1' belongs to project '%2'.\nWhat do you want to do?").arg(QString::fromStdWString(scenePath.getWideString())).arg(sceneProjectName);
QString importAnswer = QObject::tr("Import Scene");
QString switchProjectAnswer = QObject::tr("Change Project");
QString cancelAnswer = QObject::tr("Cancel");
int ret = MsgBox(question, importAnswer, switchProjectAnswer, cancelAnswer, 0);
if (ret == 3 || ret == 0) {
newScene();
return false;
}
if (ret == 2)
pm->setCurrentProjectPath(sceneProject->getProjectPath());
else
importScene = true;
}
QApplication::setOverrideCursor(Qt::WaitCursor);
TUndoManager::manager()->reset();
IconGenerator::instance()->clearRequests();
IconGenerator::instance()->clearSceneIcons();
ImageManager::instance()->clear();
CacheFxCommand::instance()->onNewScene();
#ifdef USE_SQLITE_HDPOOL
// INTERMEDIATE CACHE RESULTS MANAGEMENT: Clear all resources not accessed in the last 2 sessions
TCacheResourcePool::instance()->clearAccessedAfterSessions(1); //0 would be after this session
#endif
ToonzScene *scene = new ToonzScene();
TImageStyle::setCurrentScene(scene);
printf("%s:%s Progressing:\n", __FILE__, __FUNCTION__);
try {
/*-- プログレス表示を行いながらLoad --*/
scene->load(scenePath);
// import if needed
TProjectManager *pm = TProjectManager::instance();
TProjectP currentProject = pm->getCurrentProject();
if (!scene->getProject() || scene->getProject()->getProjectPath() != currentProject->getProjectPath()) {
ResourceImportDialog resourceLoader;
// resourceLoader.setImportEnabled(true);
ResourceImporter importer(scene, currentProject.getPointer(), resourceLoader);
SceneResources resources(scene, scene->getXsheet());
resources.accept(&importer);
scene->setScenePath(importer.getImportedScenePath());
scene->setProject(currentProject.getPointer());
}
VersionControlManager::instance()->setFrameRange(scene->getLevelSet());
}
#ifdef LINETEST
catch (TException &e) {
printf("%s:%s Exception:\n", __FILE__, __FUNCTION__);
QString msg;
if (e.getMessage() == L"File: " + scenePath.getWideString() + L":1\nCannot Load Toonz Scene in LineTest")
msg = QObject::tr("The scene %1 was created with Toonz and cannot be loaded in LineTest.").arg(QString::fromStdWString(scenePath.getWideString()));
else
msg = QObject::tr("There were problems loading the scene %1.\n Some files may be missing.").arg(QString::fromStdWString(scenePath.getWideString()));
MsgBox(WARNING, msg);
}
#endif
catch (...) {
printf("%s:%s Exception ...:\n", __FILE__, __FUNCTION__);
QString msg;
msg = QObject::tr("There were problems loading the scene %1.\n Some files may be missing.").arg(QString::fromStdWString(scenePath.getWideString()));
MsgBox(WARNING, msg);
}
printf("%s:%s end load:\n", __FILE__, __FUNCTION__);
TProject *project = scene->getProject();
if (!project) {
project = new TProject();
project->setFolder("project", scenePath);
scene->setProject(project);
}
TApp *app = TApp::instance();
app->getCurrentScene()->setScene(scene);
app->getCurrentScene()->notifyNameSceneChange();
app->getCurrentFrame()->setFrame(0);
app->getCurrentColumn()->setColumnIndex(0);
app->getCurrentXsheet()->notifyXsheetSoundChanged();
app->getCurrentObject()->setIsSpline(false);
Previewer::clearAll();
PreviewFxManager::instance()->reset();
//updateCleanupSettingsPopup();
/*- CleanupParameterの更新 -*/
CleanupParameters *cp = scene->getProperties()->getCleanupParameters();
CleanupParameters::GlobalParameters.assign(cp);
CacheFxCommand::instance()->onSceneLoaded();
#ifdef USE_SQLITE_HDPOOL
TCacheResourcePool::instance()->setPath(
QString::fromStdWString(ToonzFolder::getCacheRootFolder().getWideString()),
QString::fromStdWString(project->getName().getWideName()),
QString::fromStdWString(scene->getSceneName()));
#endif
UnitParameters::setFieldGuideAspectRatio(scene->getProperties()->getFieldGuideAspectRatio());
IconGenerator::instance()->invalidateSceneIcon();
DvDirModel::instance()->refreshFolder(scenePath.getParentDir());
TApp::instance()->getCurrentScene()->setDirtyFlag(false);
History::instance()->addItem(scenePath);
if (updateRecentFile)
RecentFiles::instance()->addFilePath(toQString(scenePath), RecentFiles::Scene);
QApplication::restoreOverrideCursor();
int forbiddenLevelCount = 0;
for (int i = 0; i < scene->getLevelSet()->getLevelCount(); i++) {
TXshLevel *xl = scene->getLevelSet()->getLevel(i);
if (xl && xl->getSimpleLevel() && xl->getSimpleLevel()->getProperties()->isForbidden())
forbiddenLevelCount++;
}
if (forbiddenLevelCount > 0) {
QString msg;
msg = QObject::tr("There were problems loading the scene %1.\nSome levels have not been loaded because their version is not supported").arg(QString::fromStdWString(scenePath.getWideString()));
MsgBox(WARNING, msg);
}
bool exist = TSystem::doesExistFileOrLevel(scene->decodeFilePath(scene->getScenePath()));
QAction *act = CommandManager::instance()->getAction(MI_RevertScene);
if (act)
act->setEnabled(exist);
printf("%s:%s loadScene() completed :\n", __FILE__, __FUNCTION__);
return true;
}
//===========================================================================
// IoCmd::loadScene()
//---------------------------------------------------------------------------
bool IoCmd::loadScene()
{
TSelection *oldSelection = TApp::instance()->getCurrentSelection()->getSelection();
FileSelection *fileSelection = dynamic_cast<FileSelection *>(TSelection::getCurrent());
if (fileSelection) {
std::vector<TFilePath> files;
fileSelection->getSelectedFiles(files);
if (files.size() == 1 && files[0] != TFilePath() && files[0].getType() == "tnz")
return loadScene(files[0]);
}
static LoadScenePopup *popup = 0;
if (!popup) {
popup = new LoadScenePopup();
popup->addFilterType("tnz");
}
int ret = popup->exec();
if (ret == QDialog::Accepted) {
TApp::instance()->getCurrentScene()->setDirtyFlag(false);
return true;
} else {
TApp::instance()->getCurrentSelection()->setSelection(oldSelection);
return false;
}
}
//===========================================================================
// IoCmd::loadSubScene()
//---------------------------------------------------------------------------
bool IoCmd::loadSubScene()
{
TSelection *oldSelection = TApp::instance()->getCurrentSelection()->getSelection();
FileSelection *fileSelection = dynamic_cast<FileSelection *>(TSelection::getCurrent());
if (fileSelection) {
IoCmd::LoadResourceArguments args;
{
std::vector<TFilePath> filePaths;
fileSelection->getSelectedFiles(filePaths);
args.resourceDatas.assign(filePaths.begin(), filePaths.end());
}
if (args.resourceDatas.size() == 1 && args.resourceDatas[0].m_path != TFilePath() && args.resourceDatas[0].m_path.getType() == "tnz") {
loadResources(args);
return true;
}
} else {
static LoadSubScenePopup *popup = 0;
if (!popup) {
popup = new LoadSubScenePopup();
popup->addFilterType("tnz");
}
int ret = popup->exec();
if (ret == QDialog::Accepted) {
TApp::instance()->getCurrentScene()->setDirtyFlag(false);
return true;
} else {
TApp::instance()->getCurrentSelection()->setSelection(oldSelection);
return false;
}
}
return false;
}
bool IoCmd::loadSubScene(const TFilePath &scenePath)
{
IoCmd::LoadResourceArguments args(scenePath);
if (args.resourceDatas.size() == 1 && args.resourceDatas[0].m_path != TFilePath() && args.resourceDatas[0].m_path.getType() == "tnz") {
loadResources(args);
return true;
}
return false;
}
std::vector<int> loadedPsdLevelIndex; // memorizza gli indici dei livelli già caricati
// serve per identificare i subfolder caricati
// Trovare un metodo alternativo.
//! Returns the number of actually loaded levels
int createSubXSheetFromPSDFolder(
IoCmd::LoadResourceArguments &args,
TXsheet *xsh, int &col0, int psdLevelIndex,
PsdSettingsPopup *popup)
{
assert(popup->isFolder(psdLevelIndex));
int row0 = 0;
int &row1 = args.row1,
&col1 = args.col1;
TApp *app = TApp::instance();
ToonzScene *scene = app->getCurrentScene()->getScene();
TXshLevel *cl = scene->createNewLevel(CHILD_XSHLEVEL);
assert(cl);
TXshChildLevel *childLevel = cl->getChildLevel();
assert(childLevel);
TXsheet *childXsh = childLevel->getXsheet();
int count = 0, subCol0 = 0;
for (int i = 0; i < popup->getFramesCount(psdLevelIndex); ++i) {
if (popup->isSubFolder(psdLevelIndex, i)) {
// se è un subfolder allora è un livello
int levelIndex = popup->getSubfolderLevelIndex(psdLevelIndex, i);
count += createSubXSheetFromPSDFolder(args, childXsh, subCol0, levelIndex, popup);
} else {
TFilePath psdpath = popup->getPsdFramePath(psdLevelIndex, i);
TXshLevel *xl = 0;
try {
xl = ::loadResource(scene, psdpath, args.castFolder, row0, col0, row1, col1, false);
} catch (TException &e) {
error(QString::fromStdWString(e.getMessage()));
}
if (xl) {
// lo importo nell'xsheet
childXsh->exposeLevel(0, subCol0, xl);
args.loadedLevels.push_back(xl);
++subCol0, ++count;
loadedPsdLevelIndex.push_back(psdLevelIndex);
}
}
}
if (childXsh) {
int rowCount = childXsh->getFrameCount();
int r;
for (r = 0; r < rowCount; r++)
xsh->setCell(r, col0, TXshCell(cl, TFrameId(r + 1)));
col0++;
}
return count;
}
// Load a psd file
//! Returns the number of actually loaded levels
int loadPSDResource(IoCmd::LoadResourceArguments &args, bool updateRecentFile, PsdSettingsPopup *popup)
{
int &row0 = args.row0;
int &col0 = args.col0;
int &row1 = args.row1;
int &col1 = args.col1;
int count = 0;
TApp *app = TApp::instance();
ToonzScene *scene = app->getCurrentScene()->getScene();
TXsheet *xsh = scene->getXsheet();
if (row0 == -1)
row0 = app->getCurrentFrame()->getFrameIndex();
if (col0 == -1)
col0 = app->getCurrentColumn()->getColumnIndex();
TXshLevel *cl = 0;
TXsheet *childXsh = 0;
// if the option "expose in sub-xsheet" is ON"
if (popup->subxsheet()) {
cl = scene->createNewLevel(CHILD_XSHLEVEL);
assert(cl);
TXshChildLevel *childLevel = cl->getChildLevel();
assert(childLevel);
childXsh = childLevel->getXsheet();
}
int subCol0 = args.col0;
loadedPsdLevelIndex.clear();
//for each layer in psd
for (int i = 0; i < popup->getPsdLevelCount(); i++) {
if (popup->isFolder(i) && popup->getFolderOption() == 1) {
if (find(loadedPsdLevelIndex.begin(), loadedPsdLevelIndex.end(), i) != loadedPsdLevelIndex.end())
continue;
if (childXsh)
count += createSubXSheetFromPSDFolder(args, childXsh, subCol0, i, popup);
else
count += createSubXSheetFromPSDFolder(args, xsh, subCol0, i, popup);
} else {
TFilePath psdpath = popup->getPsdPath(i);
TXshLevel *xl = 0;
try {
xl = ::loadResource(scene, psdpath, args.castFolder, row0, col0, row1, col1, !popup->subxsheet());
} catch (TException &e) {
error(QString::fromStdWString(e.getMessage()));
}
if (xl) {
// lo importo nell'xsheet
if (popup->subxsheet() && childXsh) {
childXsh->exposeLevel(0, subCol0, xl);
}
args.loadedLevels.push_back(xl);
subCol0++;
count++;
//move the current column to the right
col0++;
app->getCurrentColumn()->setColumnIndex(col0);
}
}
}
if (childXsh) {
int rowCount = childXsh->getFrameCount();
int r;
for (r = 0; r < rowCount; r++)
xsh->setCell(row0 + r, col0, TXshCell(cl, TFrameId(r + 1)));
}
return count;
}
//===========================================================================
// IoCmd::loadResources(actualPaths[], castFolder, row, col)
//---------------------------------------------------------------------------
typedef IoCmd::LoadResourceArguments::ScopedBlock LoadScopedBlock;
struct LoadScopedBlock::Data {
std::unique_ptr<DVGui::ProgressDialog>
m_progressDialog; //!< Progress dialog displayed on multiple paths.
int m_loadedCount; //!< Number of loaded levels.
bool m_hasSoundLevel; //!< Whether a sound level was loaded.
public:
Data() : m_loadedCount(), m_hasSoundLevel() {}
};
//-----------------------------------------------------------------------------
LoadScopedBlock::ScopedBlock()
: m_data(new Data)
{
}
//-----------------------------------------------------------------------------
LoadScopedBlock::~ScopedBlock()
{
if (m_data->m_loadedCount > 0) {
TApp *app = TApp::instance();
app->getCurrentXsheet()->notifyXsheetChanged();
if (m_data->m_hasSoundLevel)
app->getCurrentXsheet()->notifyXsheetSoundChanged();
app->getCurrentScene()->notifyCastChange();
app->getCurrentScene()->setDirtyFlag(true);
app->getCurrentTool()->onImageChanged((TImage::Type)app->getCurrentImageType());
}
}
//-----------------------------------------------------------------------------
DVGui::ProgressDialog &LoadScopedBlock::progressDialog() const
{
if (!m_data->m_progressDialog.get()) {
m_data->m_progressDialog.reset(new DVGui::ProgressDialog(
"", QObject::tr("Cancel"), 0, 0));
}
return *m_data->m_progressDialog;
}
//=============================================================================
int IoCmd::loadResources(LoadResourceArguments &args,
bool updateRecentFile,
LoadScopedBlock *sb,
int xFrom,
int xTo,
std::wstring levelName,
int step,
int inc,
int frameCount,
bool doesFileActuallyExist,
CacheTlvBehavior cachingBehavior)
{
struct locals {
static bool isDir(const LoadResourceArguments::ResourceData &rd)
{
return QFileInfo(rd.m_path.getQString()).isDir();
}
}; // locals
if (args.resourceDatas.empty())
return 0;
// Redirect to resource folders loading in case they're all dirs
if (ba::all_of(args.resourceDatas.begin(), args.resourceDatas.end(),
locals::isDir))
return loadResourceFolders(args, sb);
boost::optional<LoadScopedBlock> sb_;
if (!sb)
sb = (sb_ = boost::in_place()).get_ptr();
int &row0 = args.row0,
&col0 = args.col0,
&row1 = args.row1,
&col1 = args.col1;
// Setup local variables
TApp *app = TApp::instance();
ToonzScene *scene = app->getCurrentScene()->getScene();
TXsheet *xsh = scene->getXsheet();
// use current frame/column if row/col is not set
if (row0 == -1)
row0 = app->getCurrentFrame()->getFrameIndex();
if (col0 == -1)
col0 = app->getCurrentColumn()->getColumnIndex();
int rCount = args.resourceDatas.size(),
loadedCount = 0;
bool isSoundLevel = false;
// show wait cursor in case of caching all images because it is time consuming
if (cachingBehavior == ALL_ICONS_AND_IMAGES)
QApplication::setOverrideCursor(Qt::WaitCursor);
// Initialize progress dialog
DVGui::ProgressDialog *progressDialog = (rCount > 1) ? &sb->progressDialog() : 0;
if (progressDialog) {
progressDialog->setModal(true);
progressDialog->setMinimum(0);
progressDialog->setMaximum(rCount);
progressDialog->show();
}
// Initialize import dialog
ResourceImportDialog importDialog;
if (args.importPolicy != LoadResourceArguments::ASK_USER) {
importDialog.setImportEnabled(
args.importPolicy == LoadResourceArguments::IMPORT);
}
// Loop for all the resources to load
for (int r = 0; r != rCount; ++r) {
if (importDialog.aborted())
break;
LoadResourceArguments::ResourceData rd(args.resourceDatas[r]);
TFilePath &path = rd.m_path;
if (!rd.m_path.isLevelName())
path = TFilePath(path.getLevelNameW()).withParentDir(path.getParentDir());
if (progressDialog) {
if (progressDialog->wasCanceled())
break;
else {
progressDialog->setLabelText(DVGui::ProgressDialog::
tr("Loading \"%1\"...")
.arg(path.getQString()));
progressDialog->setValue(r);
QCoreApplication::processEvents();
}
}
bool isLastResource = (r == rCount - 1);
importDialog.setIsLastResource(isLastResource);
bool isScene = (TFileType::getInfo(path) == TFileType::TOONZSCENE);
if (scene->isExternPath(path) || isScene) {
// extern resource: import or link?
int ret = importDialog.askImportQuestion(path);
if (ret == ResourceImportDialog::A_CANCEL)
break;
}
// for the scene file
TXshLevel *xl = 0;
if (isScene) {
#ifndef STUDENT
TFilePath oldDstFolder = importDialog.getDstFolder();
TFilePath dstFolder = (Preferences::instance()->isSubsceneFolderEnabled())
? TFilePath(path.getName())
: TFilePath();
importDialog.setDstFolder(dstFolder);
importDialog.setIsLastResource(false);
//load the scene as subXsheet
try {
xl = loadChildLevel(scene, path, row0, col0, importDialog);
if (dstFolder != TFilePath())
app->getCurrentScene()->notifyCastFolderAdded(scene->getLevelSet()->getDefaultFolder() + dstFolder);
//increment the number of resources actually loaded
++loadedCount;
//move the column to the right
++col0;
//register the loaded level to args
args.loadedLevels.push_back(xl);
app->getCurrentXsheet()->notifyXsheetSoundChanged();
} catch (...) {
}
importDialog.setIsLastResource(isLastResource);
importDialog.setDstFolder(oldDstFolder);
app->getCurrentColumn()->setColumnIndex(col0);
#endif
continue;
}
// for other level files
else {
try {
path = importDialog.process(scene, 0, path);
// path = scene->decodeFilePath(codedPath);
} catch (string msg) {
error(QString::fromStdString(msg));
continue;
}
if (importDialog.aborted())
break;
}
if (path.getType() == "psd") {
static PsdSettingsPopup *popup = 0;
if (!popup) {
popup = new PsdSettingsPopup();
}
popup->setPath(path);
int ret = popup->exec();
if (ret == 0)
continue;
loadedCount += loadPSDResource(args, updateRecentFile, popup);
if (updateRecentFile)
RecentFiles::instance()->addFilePath(
toQString(scene->decodeFilePath(path)),
RecentFiles::Level);
} else {
// reuse TFrameIds retrieved by FileBrowser
std::vector<TFrameId> fIds;
if ((int)args.frameIdsSet.size() > r) // if there is fIds to be reused
{
fIds = args.frameIdsSet[r];
}
try {
xl = ::loadResource(scene,
rd,
args.castFolder,
row0, col0, row1, col1,
args.expose,
#if (__cplusplus > 199711L)
std::move(fIds),
#else
fIds,
#endif
xFrom,
xTo,
levelName,
step,
inc,
frameCount,
doesFileActuallyExist);
if (updateRecentFile) {
RecentFiles::instance()->addFilePath(
toQString(scene->decodeFilePath(path)),
RecentFiles::Level);
}
} catch (TException &e) {
error(QString::fromStdWString(e.getMessage()));
}
//if load success
if (xl) {
isSoundLevel = isSoundLevel || xl->getType() == SND_XSHLEVEL;
//register the loaded level to args
args.loadedLevels.push_back(xl);
//increment the number of loaded resources
++loadedCount;
//move the current column to right
col0++;
app->getCurrentColumn()->setColumnIndex(col0);
//load the image data of all frames to cache at the beginning
if (cachingBehavior != ON_DEMAND) {
TXshSimpleLevel *simpleLevel = xl->getSimpleLevel();
if (simpleLevel && simpleLevel->getType() == TZP_XSHLEVEL) {
bool cacheImagesAsWell = (cachingBehavior == ALL_ICONS_AND_IMAGES);
simpleLevel->loadAllIconsAndPutInCache(cacheImagesAsWell);
}
}
}
}
}
sb->data().m_loadedCount += loadedCount;
sb->data().m_hasSoundLevel = sb->data().m_hasSoundLevel || isSoundLevel;
//revert the cursor
if (cachingBehavior == ALL_ICONS_AND_IMAGES)
QApplication::restoreOverrideCursor();
return loadedCount;
}
//===========================================================================
// IoCmd::exposeLevel(simpleLevel, row, col)
//---------------------------------------------------------------------------
bool IoCmd::exposeLevel(TXshSimpleLevel *sl, int row, int col, bool insert, bool overWrite)
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
assert(xsh);
std::vector<TFrameId> fids;
sl->getFids(fids);
return exposeLevel(sl, row, col, fids, insert, overWrite);
}
//===========================================================================
// IoCmd::exposeLevel(simpleLevel, row, col)
//---------------------------------------------------------------------------
bool IoCmd::exposeLevel(TXshSimpleLevel *sl, int row, int col, const std::vector<TFrameId> &fids, bool insert, bool overWrite)
{
assert(sl);
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
assert(xsh);
int frameCount = fids.size();
bool insertEmptyColumn = false;
if (!insert && !overWrite)
insertEmptyColumn = beforeCellsInsert(xsh, row, col, fids.size(), TXshColumn::toColumnType(sl->getType()));
ExposeType type = eNone;
if (insert)
type = eShiftCells;
if (overWrite)
type = eOverWrite;
ExposeLevelUndo *undo = new ExposeLevelUndo(sl, row, col, frameCount, insertEmptyColumn, type);
xsh->exposeLevel(row, col, sl, fids, overWrite);
undo->setFids(fids);
TUndoManager::manager()->add(undo);
app->getCurrentXsheet()->notifyXsheetChanged();
return true;
}
//===========================================================================
// exposeComment
//---------------------------------------------------------------------------
bool IoCmd::exposeComment(int row, int &col, QList<QString> commentList, QString fileName)
{
TApp *app = TApp::instance();
TXsheet *xsh = app->getCurrentXsheet()->getXsheet();
TXshSoundTextColumn *textSoundCol = new TXshSoundTextColumn();
textSoundCol->setXsheet(xsh);
textSoundCol->createSoundTextLevel(row, commentList);
bool columnInserted = beforeCellsInsert(xsh, row, col, commentList.size(), TXshColumn::eSoundTextType);
xsh->insertColumn(col, textSoundCol);
//Setto il nome di dafult della colonna al nome del file magpie.
TStageObject *obj = xsh->getStageObject(TStageObjectId::ColumnId(col));
std::string str = fileName.toStdString();
obj->setName(str);
TUndoManager::manager()->add(new ExposeCommentUndo(textSoundCol, col, columnInserted, fileName));
return true;
}
//===========================================================================
// importLipSync
//---------------------------------------------------------------------------
bool IoCmd::importLipSync(TFilePath levelPath, QList<TFrameId> frameList, QList<QString> commentList, QString fileName)
{
TApp *app = TApp::instance();
int col = app->getCurrentColumn()->getColumnIndex();
int row = app->getCurrentFrame()->getFrameIndex();
IoCmd::LoadResourceArguments args;
IoCmd::LoadResourceArguments::ScopedBlock sb;
// Create text column
IoCmd::exposeComment(row, col, commentList, fileName);
// Load Level
args.resourceDatas.push_back(levelPath);
args.expose = false;
if (!IoCmd::loadResources(args, false, &sb)) {
DVGui::error(QObject::tr("It is not possible to load the level %1").arg(toQString(levelPath)));
return false;
}
// Expose Level in xsheet
assert(args.loadedLevels.size() == 1);
TXshLevel *loadedLevel = args.loadedLevels.at(0);
std::vector<TFrameId> fids;
for (int i = 0; i < frameList.size(); i++)
fids.push_back(frameList.at(i));
if (!IoCmd::exposeLevel(loadedLevel->getSimpleLevel(), row, col, fids)) {
DVGui::error(QObject::tr("It is not possible to load the level %1").arg(toQString(levelPath)));
return false;
}
return true;
}
//===========================================================================
// Commands
//---------------------------------------------------------------------------
class SaveSceneCommandHandler : public MenuItemHandler
{
public:
SaveSceneCommandHandler() : MenuItemHandler(MI_SaveScene) {}
void execute()
{
#ifdef BRAVODEMO
DVGui::featureNotAvelaible();
#else
IoCmd::saveScene();
#endif
}
} saveSceneCommandHandler;
//---------------------------------------------------------------------------
class SaveLevelCommandHandler : public MenuItemHandler
{
public:
SaveLevelCommandHandler() : MenuItemHandler(MI_SaveLevel) {}
void execute()
{
#ifdef BRAVODEMO
DVGui::featureNotAvelaible();
#else
TXshSimpleLevel *sl = TApp::instance()->getCurrentLevel()->getSimpleLevel();
if (!sl) {
MsgBox(WARNING, QObject::tr("No Current Level"));
return;
}
if (!sl->getPalette()) {
MsgBox(WARNING, QObject::tr("Toonz cannot Save this Level"));
return;
}
ToonzScene *scene = TApp::instance()->getCurrentScene()->getScene();
if (!scene) {
MsgBox(WARNING, QObject::tr("No Current Scene"));
return; // non dovrebbe succedere mai
}
TFilePath levelPath = sl->getPath();
levelPath = scene->decodeFilePath(levelPath);
QString str = QString::fromStdWString(levelPath.getWideString());
if (!(sl->getPath().isAbsolute() || !scene->isUntitled() ||
(!sl->getPath().isAbsolute() && !str.contains("untitled")))) {
error(QObject::tr("Save the scene first"));
return;
}
//reset the undo before save level
//TODO: この仕様、Preferencesでオプション化する
TUndoManager::manager()->reset();
if (!IoCmd::saveLevel())
error(QObject::tr("Save level Failed"));
#endif
}
} saveLevelCommandHandler;
//-----------------------------------------------------------------------------
class SaveProjectTemplate : public MenuItemHandler
{
public:
SaveProjectTemplate() : MenuItemHandler(MI_SaveDefaultSettings) {}
void execute()
{
QString question;
question = QObject::tr("Are you sure you want to save the Default Settings?");
int ret = MsgBox(question, QObject::tr("Save"), QObject::tr("Cancel"), 0);
if (ret == 2 || ret == 0)
return;
ToonzScene *scene = TApp::instance()->getCurrentScene()->getScene();
try {
TProjectManager::instance()->saveTemplate(scene);
} catch (TSystemException se) {
MsgBox(WARNING, QString::fromStdWString(se.getMessage()));
return;
}
}
} saveProjectTemplate;
//-----------------------------------------------------------------------------
class OpenRecentSceneFileCommandHandler : public MenuItemHandler
{
public:
OpenRecentSceneFileCommandHandler() : MenuItemHandler(MI_OpenRecentScene) {}
void execute()
{
QAction *act = CommandManager::instance()->getAction(MI_OpenRecentScene);
DVMenuAction *menu = dynamic_cast<DVMenuAction *>(act->menu());
int index = menu->getTriggeredActionIndex();
QString path = RecentFiles::instance()->getFilePath(index, RecentFiles::Scene);
IoCmd::loadScene(TFilePath(path.toStdWString()), false);
RecentFiles::instance()->moveFilePath(index, 0, RecentFiles::Scene);
}
} openRecentSceneFileCommandHandler;
//-----------------------------------------------------------------------------
class OpenRecentLevelFileCommandHandler : public MenuItemHandler
{
public:
OpenRecentLevelFileCommandHandler() : MenuItemHandler(MI_OpenRecentLevel) {}
void execute()
{
QAction *act = CommandManager::instance()->getAction(MI_OpenRecentLevel);
DVMenuAction *menu = dynamic_cast<DVMenuAction *>(act->menu());
int index = menu->getTriggeredActionIndex();
QString path = RecentFiles::instance()->getFilePath(index, RecentFiles::Level);
IoCmd::LoadResourceArguments args(TFilePath(path.toStdWString()));
IoCmd::loadResources(args, false);
RecentFiles::instance()->moveFilePath(index, 0, RecentFiles::Level);
if (args.loadedLevels.empty()) {
QString msg;
msg = QObject::tr("It is not possible to load the %1 level.").arg(path);
MsgBox(CRITICAL, msg);
}
}
} openRecentLevelFileCommandHandler;
//-----------------------------------------------------------------------------
class ClearRecentSceneFileListCommandHandler : public MenuItemHandler
{
public:
ClearRecentSceneFileListCommandHandler() : MenuItemHandler(MI_ClearRecentScene) {}
void execute()
{
RecentFiles::instance()->clearRecentFilesList(RecentFiles::Scene);
}
} clearRecentSceneFileListCommandHandler;
//-----------------------------------------------------------------------------
class ClearRecentLevelFileListCommandHandler : public MenuItemHandler
{
public:
ClearRecentLevelFileListCommandHandler() : MenuItemHandler(MI_ClearRecentLevel) {}
void execute()
{
RecentFiles::instance()->clearRecentFilesList(RecentFiles::Level);
}
} clearRecentLevelFileListCommandHandler;
//-----------------------------------------------------------------------------
class RevertScene : public MenuItemHandler
{
public:
RevertScene() : MenuItemHandler(MI_RevertScene) {}
void execute()
{
TSceneHandle *sceneHandle = TApp::instance()->getCurrentScene();
ToonzScene *scene = sceneHandle->getScene();
assert(scene);
TFilePath path = scene->getScenePath();
TFilePath decodePath = scene->decodeFilePath(scene->getScenePath());
if (!TSystem::doesExistFileOrLevel(decodePath)) {
MsgBox(DVGui::WARNING, QObject::tr("The scene %1 doesn't exist.").arg(toQString(decodePath)));
return;
}
if (sceneHandle->getDirtyFlag()) {
int ret = MsgBox(QString(QObject::tr("Revert: the current scene has been modified.\nAre you sure you want to revert to previous version?")),
QString(QObject::tr("Revert")), QString(QObject::tr("Cancel")));
if (ret == 2 || ret == 0)
return;
}
IoCmd::loadScene(path, false, false);
}
} RevertScene;
//=============================================================================
// Overwrite palette
//-----------------------------------------------------------------------------
class OverwritePaletteCommandHandler : public MenuItemHandler
{
public:
OverwritePaletteCommandHandler() : MenuItemHandler(MI_OverwritePalette) {}
void execute()
{
TXshLevel *level = TApp::instance()->getCurrentLevel()->getLevel();
if (!level) {
MsgBox(WARNING, "No current level.");
return;
}
TXshSimpleLevel *sl = level->getSimpleLevel();
TXshPaletteLevel *pl = level->getPaletteLevel();
if (!sl && !pl) {
MsgBox(WARNING, "Current level has no palette.");
return;
}
/*-- SimpleLevel/PaletteLevelの場合毎にパレット/パスの取得の仕方を変える --*/
TPalette *palette;
TFilePath palettePath;
/*- SimpleLevelの場合 -*/
if (sl) {
palette = sl->getPalette();
if (!palette) {
MsgBox(WARNING, "No current palette");
return;
}
if (sl->getPath().getType() == "pli")
palettePath = sl->getPath();
else
palettePath = sl->getPath().withType("tpl");
}
/*- PaletteLevelの場合 -*/
else if (pl) {
palette = pl->getPalette();
if (!palette) {
MsgBox(WARNING, "No current palette");
return;
}
palettePath = pl->getPath();
} else {
MsgBox(WARNING, "This level is not SimpleLevel or PaletteLevel");
return;
}
QString question;
int ret;
if (sl && sl->getPath().getType() == "pli") {
question = "Saving " + toQString(palettePath) + "\nThis command will ovewrite the level data as well. Are you sure ?";
ret = MsgBox(question, QObject::tr("OK"), QObject::tr("Cancel"), 0);
} else {
question = "Do you want to overwrite current palette to " + toQString(palettePath) + " ?";
ret = MsgBox(question, QObject::tr("Overwrite"), QObject::tr("Don't Overwrite"), 0);
}
if (ret == 2 || ret == 0)
return;
ToonzScene *scene = TApp::instance()->getCurrentScene()->getScene();
if (!palettePath.isAbsolute() && scene) {
palettePath = scene->decodeFilePath(palettePath);
}
if (sl && sl->getPath().getType() == "pli")
sl->save(palettePath, TFilePath(), true);
else
StudioPalette::instance()->save(palettePath, palette);
/*- Dirtyフラグの変更 -*/
if (sl)
sl->getPalette()->setDirtyFlag(false);
else if (pl)
pl->getPalette()->setDirtyFlag(false);
/*- Undoをリセット。 TODO:この挙動、Preferencesでオプション化 -*/
TUndoManager::manager()->reset();
TApp::instance()->getPaletteController()->getCurrentLevelPalette()->notifyPaletteDirtyFlagChanged();
}
} overwritePaletteCommandHandler;