2015-03-13 06:26:01 +01:00
|
|
|
#include "lc_global.h"
|
|
|
|
#include "lc_timelinewidget.h"
|
|
|
|
#include "lc_model.h"
|
|
|
|
#include "project.h"
|
|
|
|
#include "piece.h"
|
|
|
|
#include "pieceinf.h"
|
|
|
|
#include "lc_mainwindow.h"
|
|
|
|
|
|
|
|
lcTimelineWidget::lcTimelineWidget(QWidget* Parent)
|
|
|
|
: QTreeWidget(Parent)
|
|
|
|
{
|
|
|
|
mIgnoreUpdates = false;
|
|
|
|
|
|
|
|
setSelectionMode(QAbstractItemView::ExtendedSelection);
|
2015-03-08 01:27:11 +01:00
|
|
|
setDragEnabled(true);
|
|
|
|
setDragDropMode(QAbstractItemView::InternalMove);
|
|
|
|
setUniformRowHeights(true);
|
|
|
|
setHeaderHidden(true);
|
2015-03-13 06:26:01 +01:00
|
|
|
setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
|
2015-03-15 21:01:07 +01:00
|
|
|
invisibleRootItem()->setFlags(invisibleRootItem()->flags() & ~Qt::ItemIsDropEnabled);
|
|
|
|
|
2015-03-13 06:26:01 +01:00
|
|
|
connect(this, SIGNAL(itemSelectionChanged()), SLOT(ItemSelectionChanged()));
|
|
|
|
connect(this, SIGNAL(customContextMenuRequested(QPoint)), SLOT(CustomMenuRequested(QPoint)));
|
|
|
|
}
|
|
|
|
|
|
|
|
lcTimelineWidget::~lcTimelineWidget()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-03-14 20:07:07 +01:00
|
|
|
void lcTimelineWidget::CustomMenuRequested(QPoint Pos)
|
|
|
|
{
|
|
|
|
QMenu* Menu = new QMenu(this);
|
|
|
|
|
2015-03-17 05:18:28 +01:00
|
|
|
lcObject* FocusObject = lcGetActiveModel()->GetFocusObject();
|
|
|
|
|
|
|
|
if (FocusObject && FocusObject->IsPiece())
|
|
|
|
{
|
|
|
|
lcPiece* Piece = (lcPiece*)FocusObject;
|
|
|
|
|
|
|
|
if (Piece->mPieceInfo->IsModel())
|
|
|
|
{
|
|
|
|
Menu->addAction(gMainWindow->mActions[LC_MODEL_EDIT_FOCUS]);
|
|
|
|
Menu->addSeparator();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QAction* InsertStepAction = Menu->addAction(gMainWindow->mActions[LC_VIEW_TIME_INSERT]->text(), this, SLOT(InsertStep()));
|
|
|
|
InsertStepAction->setStatusTip(gMainWindow->mActions[LC_VIEW_TIME_INSERT]->statusTip());
|
|
|
|
QAction* RemoveStepAction = Menu->addAction(gMainWindow->mActions[LC_VIEW_TIME_DELETE]->text(), this, SLOT(RemoveStep()));
|
|
|
|
RemoveStepAction->setStatusTip(gMainWindow->mActions[LC_VIEW_TIME_DELETE]->statusTip());
|
2015-03-15 20:42:11 +01:00
|
|
|
Menu->addSeparator();
|
2015-03-17 05:18:28 +01:00
|
|
|
|
2015-03-14 20:07:07 +01:00
|
|
|
Menu->addAction(gMainWindow->mActions[LC_PIECE_HIDE_SELECTED]);
|
|
|
|
Menu->addAction(gMainWindow->mActions[LC_PIECE_HIDE_UNSELECTED]);
|
2015-03-15 21:01:07 +01:00
|
|
|
Menu->addAction(gMainWindow->mActions[LC_PIECE_UNHIDE_SELECTED]);
|
2015-03-14 20:07:07 +01:00
|
|
|
Menu->addAction(gMainWindow->mActions[LC_PIECE_UNHIDE_ALL]);
|
|
|
|
|
|
|
|
Menu->popup(viewport()->mapToGlobal(Pos));
|
|
|
|
}
|
2015-03-13 06:26:01 +01:00
|
|
|
|
2015-03-16 02:10:01 +01:00
|
|
|
void lcTimelineWidget::Update(bool Clear)
|
2015-03-13 06:26:01 +01:00
|
|
|
{
|
2015-03-14 22:09:10 +01:00
|
|
|
if (mIgnoreUpdates)
|
|
|
|
return;
|
|
|
|
|
2015-03-13 06:26:01 +01:00
|
|
|
lcModel* Model = lcGetActiveModel();
|
|
|
|
|
|
|
|
if (!Model)
|
|
|
|
{
|
|
|
|
mItems.clear();
|
|
|
|
clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Blocked = blockSignals(true);
|
|
|
|
|
2015-03-16 02:10:01 +01:00
|
|
|
if (Clear)
|
|
|
|
{
|
|
|
|
mItems.clear();
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
2015-03-15 20:42:11 +01:00
|
|
|
int Steps = lcMax(Model->GetLastStep(), Model->GetCurrentStep());
|
2015-03-13 06:26:01 +01:00
|
|
|
|
2015-03-14 20:07:07 +01:00
|
|
|
for (int TopLevelItemIdx = Steps; TopLevelItemIdx < topLevelItemCount(); )
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* StepItem = topLevelItem(TopLevelItemIdx);
|
|
|
|
|
|
|
|
while (StepItem->childCount())
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* PieceItem = StepItem->child(0);
|
|
|
|
lcPiece* Piece = (lcPiece*)PieceItem->data(0, Qt::UserRole).value<uintptr_t>();
|
|
|
|
mItems.remove(Piece);
|
|
|
|
delete PieceItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete StepItem;
|
|
|
|
}
|
2015-03-13 06:26:01 +01:00
|
|
|
|
2015-03-14 22:09:10 +01:00
|
|
|
for (int TopLevelItemIdx = topLevelItemCount(); TopLevelItemIdx < Steps; TopLevelItemIdx++)
|
2015-03-13 06:26:01 +01:00
|
|
|
{
|
2015-03-14 22:09:10 +01:00
|
|
|
QTreeWidgetItem* StepItem = new QTreeWidgetItem(this, QStringList(tr("Step %1").arg(TopLevelItemIdx + 1)));
|
2015-03-13 06:26:01 +01:00
|
|
|
StepItem->setFlags(Qt::ItemIsEnabled | Qt::ItemIsDropEnabled);
|
|
|
|
addTopLevelItem(StepItem);
|
|
|
|
StepItem->setExpanded(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
const lcArray<lcPiece*>& Pieces = Model->GetPieces();
|
2015-03-14 20:07:07 +01:00
|
|
|
QTreeWidgetItem* StepItem = NULL;
|
|
|
|
int PieceItemIndex = 0;
|
2015-03-14 22:09:10 +01:00
|
|
|
int Step = 0;
|
2015-03-13 06:26:01 +01:00
|
|
|
|
2015-03-14 20:07:07 +01:00
|
|
|
for (int PieceIdx = 0; PieceIdx != Pieces.GetSize(); PieceIdx++)
|
2015-03-13 06:26:01 +01:00
|
|
|
{
|
|
|
|
lcPiece* Piece = Pieces[PieceIdx];
|
|
|
|
|
2015-03-14 22:09:10 +01:00
|
|
|
while (Step != Piece->GetStepShow())
|
2015-03-13 06:26:01 +01:00
|
|
|
{
|
2015-03-14 20:07:07 +01:00
|
|
|
if (StepItem)
|
|
|
|
{
|
|
|
|
while (PieceItemIndex < StepItem->childCount())
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* PieceItem = StepItem->child(PieceItemIndex);
|
|
|
|
lcPiece* RemovePiece = (lcPiece*)PieceItem->data(0, Qt::UserRole).value<uintptr_t>();
|
|
|
|
|
|
|
|
if (Pieces.FindIndex(RemovePiece) == -1)
|
|
|
|
{
|
|
|
|
mItems.remove(RemovePiece);
|
|
|
|
delete PieceItem;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
PieceItem->parent()->removeChild(PieceItem);
|
|
|
|
topLevelItem(RemovePiece->GetStepShow() - 1)->addChild(PieceItem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-14 22:09:10 +01:00
|
|
|
Step++;
|
2015-03-14 20:07:07 +01:00
|
|
|
StepItem = topLevelItem(Step - 1);
|
|
|
|
PieceItemIndex = 0;
|
2015-03-13 06:26:01 +01:00
|
|
|
}
|
2015-03-14 20:07:07 +01:00
|
|
|
|
|
|
|
QTreeWidgetItem* PieceItem = mItems.value(Piece);
|
|
|
|
|
|
|
|
if (!PieceItem)
|
2015-03-13 06:26:01 +01:00
|
|
|
{
|
2015-03-14 20:07:07 +01:00
|
|
|
PieceItem = new QTreeWidgetItem(QStringList(Piece->mPieceInfo->m_strDescription));
|
2015-03-13 06:26:01 +01:00
|
|
|
PieceItem->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled | Qt::ItemIsDragEnabled);
|
|
|
|
PieceItem->setData(0, Qt::UserRole, qVariantFromValue<uintptr_t>((uintptr_t)Piece));
|
2015-03-14 20:07:07 +01:00
|
|
|
StepItem->insertChild(PieceItemIndex, PieceItem);
|
2015-03-13 06:26:01 +01:00
|
|
|
mItems[Piece] = PieceItem;
|
|
|
|
}
|
2015-03-14 20:07:07 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (PieceItemIndex >= StepItem->childCount() || PieceItem != StepItem->child(PieceItemIndex))
|
|
|
|
{
|
|
|
|
if (PieceItem->parent() == StepItem)
|
|
|
|
StepItem->removeChild(PieceItem);
|
|
|
|
|
|
|
|
StepItem->insertChild(PieceItemIndex, PieceItem);
|
|
|
|
}
|
|
|
|
}
|
2015-03-13 06:26:01 +01:00
|
|
|
|
2015-03-16 02:47:55 +01:00
|
|
|
QColor Color = palette().text().color();
|
|
|
|
if (Piece->IsHidden())
|
|
|
|
Color.setAlpha(128);
|
|
|
|
PieceItem->setTextColor(0, Color);
|
|
|
|
|
2015-03-13 06:26:01 +01:00
|
|
|
PieceItem->setSelected(Piece->IsSelected());
|
2015-03-14 20:07:07 +01:00
|
|
|
PieceItemIndex++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!StepItem)
|
|
|
|
StepItem = topLevelItem(0);
|
|
|
|
|
|
|
|
while (PieceItemIndex < StepItem->childCount())
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* PieceItem = StepItem->child(PieceItemIndex);
|
|
|
|
lcPiece* RemovePiece = (lcPiece*)PieceItem->data(0, Qt::UserRole).value<uintptr_t>();
|
|
|
|
|
|
|
|
mItems.remove(RemovePiece);
|
|
|
|
delete PieceItem;
|
2015-03-13 06:26:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
blockSignals(Blocked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void lcTimelineWidget::UpdateSelection()
|
|
|
|
{
|
|
|
|
if (mIgnoreUpdates)
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool Blocked = blockSignals(true);
|
|
|
|
|
|
|
|
for (int TopLevelItemIdx = 0; TopLevelItemIdx < topLevelItemCount(); TopLevelItemIdx++)
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* StepItem = topLevelItem(TopLevelItemIdx);
|
|
|
|
|
|
|
|
for (int PieceItemIdx = 0; PieceItemIdx < StepItem->childCount(); PieceItemIdx++)
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* PieceItem = StepItem->child(PieceItemIdx);
|
|
|
|
lcPiece* Piece = (lcPiece*)PieceItem->data(0, Qt::UserRole).value<uintptr_t>();
|
|
|
|
|
|
|
|
PieceItem->setSelected(Piece->IsSelected());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
blockSignals(Blocked);
|
|
|
|
}
|
|
|
|
|
2015-03-17 05:18:28 +01:00
|
|
|
void lcTimelineWidget::InsertStep()
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* CurrentItem = currentItem();
|
|
|
|
|
|
|
|
if (!CurrentItem)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (CurrentItem->parent())
|
|
|
|
CurrentItem = CurrentItem->parent();
|
|
|
|
|
|
|
|
int Step = indexOfTopLevelItem(CurrentItem);
|
|
|
|
|
|
|
|
if (Step == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
lcGetActiveModel()->InsertStep(Step + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void lcTimelineWidget::RemoveStep()
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* CurrentItem = currentItem();
|
|
|
|
|
|
|
|
if (!CurrentItem)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (CurrentItem->parent())
|
|
|
|
CurrentItem = CurrentItem->parent();
|
|
|
|
|
|
|
|
int Step = indexOfTopLevelItem(CurrentItem);
|
|
|
|
|
|
|
|
if (Step == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
lcGetActiveModel()->RemoveStep(Step + 1);
|
|
|
|
}
|
|
|
|
|
2015-03-13 06:26:01 +01:00
|
|
|
void lcTimelineWidget::ItemSelectionChanged()
|
|
|
|
{
|
|
|
|
lcArray<lcObject*> Selection;
|
2015-03-15 20:42:11 +01:00
|
|
|
lcStep LastStep = 1;
|
2015-03-13 06:26:01 +01:00
|
|
|
|
|
|
|
foreach (QTreeWidgetItem* PieceItem, selectedItems())
|
|
|
|
{
|
|
|
|
lcPiece* Piece = (lcPiece*)PieceItem->data(0, Qt::UserRole).value<uintptr_t>();
|
2015-03-15 20:42:11 +01:00
|
|
|
LastStep = lcMax(LastStep, Piece->GetStepShow());
|
2015-03-13 06:26:01 +01:00
|
|
|
Selection.Add(Piece);
|
|
|
|
}
|
|
|
|
|
2015-03-14 22:09:10 +01:00
|
|
|
lcPiece* CurrentPiece = NULL;
|
|
|
|
QTreeWidgetItem* CurrentItem = currentItem();
|
|
|
|
if (CurrentItem && CurrentItem->isSelected())
|
|
|
|
CurrentPiece = (lcPiece*)CurrentItem->data(0, Qt::UserRole).value<uintptr_t>();
|
|
|
|
|
2015-03-13 06:26:01 +01:00
|
|
|
bool Blocked = blockSignals(true);
|
|
|
|
mIgnoreUpdates = true;
|
2015-03-15 20:42:11 +01:00
|
|
|
lcModel* Model = lcGetActiveModel();
|
|
|
|
if (LastStep > Model->GetCurrentStep())
|
|
|
|
Model->SetCurrentStep(LastStep);
|
|
|
|
Model->SetSelectionAndFocus(Selection, CurrentPiece, LC_PIECE_SECTION_POSITION);
|
2015-03-13 06:26:01 +01:00
|
|
|
mIgnoreUpdates = false;
|
|
|
|
blockSignals(Blocked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void lcTimelineWidget::dropEvent(QDropEvent* Event)
|
|
|
|
{
|
|
|
|
QTreeWidget::dropEvent(Event);
|
|
|
|
|
|
|
|
QList<QPair<lcPiece*, int>> PieceSteps;
|
|
|
|
|
|
|
|
for (int TopLevelItemIdx = 0; TopLevelItemIdx < topLevelItemCount(); TopLevelItemIdx++)
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* StepItem = topLevelItem(TopLevelItemIdx);
|
|
|
|
|
|
|
|
for (int PieceItemIdx = 0; PieceItemIdx < StepItem->childCount(); PieceItemIdx++)
|
|
|
|
{
|
|
|
|
QTreeWidgetItem* PieceItem = StepItem->child(PieceItemIdx);
|
|
|
|
lcPiece* Piece = (lcPiece*)PieceItem->data(0, Qt::UserRole).value<uintptr_t>();
|
|
|
|
|
|
|
|
PieceSteps.append(QPair<lcPiece*, int>(Piece, TopLevelItemIdx + 1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-14 22:09:10 +01:00
|
|
|
mIgnoreUpdates = true;
|
2015-03-13 06:26:01 +01:00
|
|
|
lcGetActiveModel()->SetPieceSteps(PieceSteps);
|
2015-03-14 22:09:10 +01:00
|
|
|
mIgnoreUpdates = false;
|
2015-03-13 06:26:01 +01:00
|
|
|
}
|