leocad/common/lc_partselectionwidget.cpp
2017-01-24 19:49:49 -08:00

521 lines
13 KiB
C++

#include "lc_global.h"
#include "lc_partselectionwidget.h"
#include "lc_profile.h"
#include "lc_application.h"
#include "lc_library.h"
#include "lc_model.h"
#include "project.h"
#include "pieceinf.h"
static int lcPartSortFunc(PieceInfo* const& a, PieceInfo* const& b)
{
return strcmp(a->m_strDescription, b->m_strDescription);
}
lcPartSelectionFilterModel::lcPartSelectionFilterModel(QObject* Parent)
: QSortFilterProxyModel(Parent)
{
}
void lcPartSelectionFilterModel::SetFilter(const QString& Filter)
{
mFilter = Filter.toLatin1();
invalidateFilter();
}
bool lcPartSelectionFilterModel::filterAcceptsRow(int SourceRow, const QModelIndex& SourceParent) const
{
Q_UNUSED(SourceParent);
if (mFilter.isEmpty())
return true;
lcPartSelectionListModel* SourceModel = (lcPartSelectionListModel*)sourceModel();
PieceInfo* Info = SourceModel->GetPieceInfo(SourceRow);
char Description[sizeof(Info->m_strDescription)];
char* Src = Info->m_strDescription;
char* Dst = Description;
for (;;)
{
*Dst = *Src;
if (*Src == ' ' && *(Src + 1) == ' ')
Src++;
else if (*Src == 0)
break;
Src++;
Dst++;
}
return strcasestr(Description, mFilter) || strcasestr(Info->m_strName, mFilter);
}
void lcPartSelectionItemDelegate::paint(QPainter* Painter, const QStyleOptionViewItem& Option, const QModelIndex& Index) const
{
mListModel->RequestPreview(mFilterModel->mapToSource(Index).row());
QStyledItemDelegate::paint(Painter, Option, Index);
}
QSize lcPartSelectionItemDelegate::sizeHint(const QStyleOptionViewItem& Option, const QModelIndex& Index) const
{
return QStyledItemDelegate::sizeHint(Option, Index);
}
lcPartSelectionListModel::lcPartSelectionListModel(QObject* Parent)
: QAbstractListModel(Parent)
{
mListView = (lcPartSelectionListView*)Parent;
mIconSize = 0;
connect(lcGetPiecesLibrary(), SIGNAL(PartLoaded(PieceInfo*)), this, SLOT(PartLoaded(PieceInfo*)));
}
lcPartSelectionListModel::~lcPartSelectionListModel()
{
ClearRequests();
}
void lcPartSelectionListModel::ClearRequests()
{
lcPiecesLibrary* Library = lcGetPiecesLibrary();
foreach(int RequestIdx, mRequestedPreviews)
{
PieceInfo* Info = mParts[RequestIdx].first;
Library->ReleasePieceInfo(Info);
}
mRequestedPreviews.clear();
}
void lcPartSelectionListModel::Redraw()
{
ClearRequests();
beginResetModel();
for (int PartIdx = 0; PartIdx < mParts.size(); PartIdx++)
mParts[PartIdx].second = QPixmap();
endResetModel();
}
void lcPartSelectionListModel::SetCategory(int CategoryIndex)
{
ClearRequests();
beginResetModel();
lcPiecesLibrary* Library = lcGetPiecesLibrary();
lcArray<PieceInfo*> SingleParts, GroupedParts;
Library->GetCategoryEntries(CategoryIndex, false, SingleParts, GroupedParts);
SingleParts.Sort(lcPartSortFunc);
mParts.resize(SingleParts.GetSize());
for (int PartIdx = 0; PartIdx < SingleParts.GetSize(); PartIdx++)
mParts[PartIdx] = QPair<PieceInfo*, QPixmap>(SingleParts[PartIdx], QPixmap());
endResetModel();
}
void lcPartSelectionListModel::SetModelsCategory()
{
ClearRequests();
beginResetModel();
mParts.clear();
const lcArray<lcModel*>& Models = lcGetActiveProject()->GetModels();
lcModel* CurrentModel = lcGetActiveModel();
for (int ModelIdx = 0; ModelIdx < Models.GetSize(); ModelIdx++)
{
lcModel* Model = Models[ModelIdx];
if (!Model->IncludesModel(CurrentModel))
mParts.append(QPair<PieceInfo*, QPixmap>(Model->GetPieceInfo(), QPixmap()));
}
endResetModel();
}
int lcPartSelectionListModel::rowCount(const QModelIndex& Parent) const
{
Q_UNUSED(Parent);
return mParts.size();
}
QVariant lcPartSelectionListModel::data(const QModelIndex& Index, int Role) const
{
int InfoIndex = Index.row();
if (Index.isValid() && InfoIndex < mParts.size())
{
PieceInfo* Info = mParts[InfoIndex].first;
switch (Role)
{
case Qt::DisplayRole:
if (!mIconSize)
{
PieceInfo* Info = mParts[InfoIndex].first;
return QVariant(QString::fromLatin1(Info->m_strDescription));
}
break;
case Qt::ToolTipRole:
return QVariant(QString("%1 (%2)").arg(QString::fromLatin1(Info->m_strDescription), QString::fromLatin1(Info->m_strName)));
case Qt::DecorationRole:
if (!mParts[InfoIndex].second.isNull())
return QVariant(mParts[InfoIndex].second);
else
return QVariant(QColor(0, 0, 0, 0));
default:
break;
}
}
return QVariant();
}
QVariant lcPartSelectionListModel::headerData(int Section, Qt::Orientation Orientation, int Role) const
{
Q_UNUSED(Section);
Q_UNUSED(Orientation);
return Role == Qt::DisplayRole ? QVariant(QLatin1String("Image")) : QVariant();
}
Qt::ItemFlags lcPartSelectionListModel::flags(const QModelIndex& Index) const
{
Qt::ItemFlags DefaultFlags = QAbstractListModel::flags(Index);
if (Index.isValid())
return Qt::ItemIsDragEnabled | DefaultFlags;
else
return DefaultFlags;
}
#include "lc_mainwindow.h"
#include "view.h"
void lcPartSelectionListModel::RequestPreview(int InfoIndex)
{
if (!mIconSize || !mParts[InfoIndex].second.isNull())
return;
if (mRequestedPreviews.indexOf(InfoIndex) != -1)
return;
PieceInfo* Info = mParts[InfoIndex].first;
lcGetPiecesLibrary()->LoadPieceInfo(Info, false, false);
if (Info->mState == LC_PIECEINFO_LOADED)
DrawPreview(InfoIndex);
else
mRequestedPreviews.append(InfoIndex);
}
void lcPartSelectionListModel::PartLoaded(PieceInfo* Info)
{
for (int PartIdx = 0; PartIdx < mParts.size(); PartIdx++)
{
if (mParts[PartIdx].first == Info)
{
if (mRequestedPreviews.removeOne(PartIdx))
DrawPreview(PartIdx);
break;
}
}
}
void lcPartSelectionListModel::DrawPreview(int InfoIndex)
{
/*
QRegion VisibleRegion = mListView->viewport()->visibleRegion();
int InfoIndex;
for (;;)
{
if (mRequestedPreviews.isEmpty())
return;
InfoIndex = mRequestedPreviews.takeFirst();
QRect ItemRect = mListView->visualRect(mListView->GetFilterModel()->mapFromSource(index(InfoIndex, 0)));
if (VisibleRegion.contains(ItemRect) || VisibleRegion.intersects(ItemRect))
break;
}
*/
View* View = gMainWindow->GetActiveView();
View->MakeCurrent();
lcContext* Context = View->mContext;
int Width = 128;
int Height = 128;
if (!Context->BeginRenderToTexture(Width, Height))
return;
float Aspect = (float)Width / (float)Height;
Context->SetViewport(0, 0, Width, Height);
lcMatrix44 ProjectionMatrix = lcMatrix44Perspective(20.0f, Aspect, 1.0f, 12500.0f);
lcMatrix44 ViewMatrix;
Context->SetDefaultState();
Context->SetProjectionMatrix(ProjectionMatrix);
Context->SetProgram(LC_PROGRAM_SIMPLE);
lcPiecesLibrary* Library = lcGetPiecesLibrary();
PieceInfo* Info = mParts[InfoIndex].first;
glClearColor(1.0f, 1.0f, 1.0f, 0.0f);
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
lcVector3 CameraPosition(-100.0f, -100.0f, 75.0f);
Info->ZoomExtents(ProjectionMatrix, ViewMatrix, CameraPosition);
lcScene Scene;
Scene.Begin(ViewMatrix);
Info->AddRenderMeshes(Scene, lcMatrix44Identity(), gMainWindow->mColorIndex, false, false);
Scene.End();
Context->SetViewMatrix(ViewMatrix);
Context->DrawOpaqueMeshes(Scene.mOpaqueMeshes);
Context->DrawTranslucentMeshes(Scene.mTranslucentMeshes);
Context->UnbindMesh(); // context remove
Library->ReleasePieceInfo(Info);
mParts[InfoIndex].second = QPixmap::fromImage(Context->GetRenderToTextureImage(Width, Height)).scaled(mIconSize, mIconSize, Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
Context->EndRenderToTexture();
#if (QT_VERSION >= QT_VERSION_CHECK(5, 2, 0))
emit dataChanged(index(InfoIndex, 0), index(InfoIndex, 0), QVector<int>() << Qt::DecorationRole);
#else
emit dataChanged(index(InfoIndex, 0), index(InfoIndex, 0));
#endif
}
void lcPartSelectionListModel::SetIconSize(int Size)
{
if (Size == mIconSize)
return;
mIconSize = Size;
beginResetModel();
for (int PartIdx = 0; PartIdx < mParts.size(); PartIdx++)
mParts[PartIdx].second = QPixmap();
endResetModel();
}
lcPartSelectionListView::lcPartSelectionListView(QWidget* Parent)
: QListView(Parent)
{
setUniformItemSizes(true);
setResizeMode(QListView::Adjust);
setDragEnabled(true);
setContextMenuPolicy(Qt::CustomContextMenu);
mListModel = new lcPartSelectionListModel(this);
mFilterModel = new lcPartSelectionFilterModel(this);
mFilterModel->setSourceModel(mListModel);
setModel(mFilterModel);
lcPartSelectionItemDelegate* ItemDelegate = new lcPartSelectionItemDelegate(this, mListModel, mFilterModel);
setItemDelegate(ItemDelegate);
connect(this, SIGNAL(customContextMenuRequested(QPoint)), SLOT(CustomContextMenuRequested(QPoint)));
SetIconSize(lcGetProfileInt(LC_PROFILE_PARTS_LIST_ICONS));
}
void lcPartSelectionListView::CustomContextMenuRequested(QPoint Pos)
{
QMenu* Menu = new QMenu(this);
Menu->addAction("Small Icons", this, SLOT(SetSmallIcons()));
Menu->addAction("Medium Icons", this, SLOT(SetMediumIcons()));
Menu->addAction("Large Icons", this, SLOT(SetLargeIcons()));
Menu->addAction("Text", this, SLOT(SetText()));
Menu->popup(viewport()->mapToGlobal(Pos));
}
void lcPartSelectionListView::SetSmallIcons()
{
SetIconSize(32);
}
void lcPartSelectionListView::SetMediumIcons()
{
SetIconSize(64);
}
void lcPartSelectionListView::SetLargeIcons()
{
SetIconSize(96);
}
void lcPartSelectionListView::SetText()
{
SetIconSize(0);
}
void lcPartSelectionListView::SetIconSize(int Size)
{
setViewMode(Size ? QListView::IconMode : QListView::ListMode);
setIconSize(QSize(Size, Size));
lcSetProfileInt(LC_PROFILE_PARTS_LIST_ICONS, Size);
mListModel->SetIconSize(Size);
}
void lcPartSelectionListView::startDrag(Qt::DropActions SupportedActions)
{
Q_UNUSED(SupportedActions);
PieceInfo* Info = GetCurrentPart();
if (!Info)
return;
QByteArray ItemData;
QDataStream DataStream(&ItemData, QIODevice::WriteOnly);
DataStream << QString(Info->m_strName);
QMimeData* MimeData = new QMimeData;
MimeData->setData("application/vnd.leocad-part", ItemData);
QDrag* Drag = new QDrag(this);
Drag->setMimeData(MimeData);
Drag->exec(Qt::CopyAction);
}
lcPartSelectionWidget::lcPartSelectionWidget(QWidget* Parent)
: QWidget(Parent)
{
mSplitter = new QSplitter(this);
mCategoriesWidget = new QTreeWidget(mSplitter);
mCategoriesWidget->setHeaderHidden(true);
mCategoriesWidget->setUniformRowHeights(true);
mCategoriesWidget->setRootIsDecorated(false);
QSizePolicy PartsWidgetPolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
PartsWidgetPolicy.setHorizontalStretch(1);
PartsWidgetPolicy.setVerticalStretch(1);
QWidget* PartsGroupWidget = new QWidget(mSplitter);
PartsGroupWidget->setSizePolicy(PartsWidgetPolicy);
QVBoxLayout* PartsLayout = new QVBoxLayout();
PartsLayout->setContentsMargins(0, 0, 0, 0);
PartsGroupWidget->setLayout(PartsLayout);
mFilterWidget = new QLineEdit(PartsGroupWidget);
mFilterWidget->setPlaceholderText(tr("Search Parts"));
#if (QT_VERSION >= QT_VERSION_CHECK(5, 2, 0))
mFilterWidget->addAction(QIcon(":/resources/parts_search.png"), QLineEdit::TrailingPosition);
#endif
PartsLayout->addWidget(mFilterWidget);
mPartsWidget = new lcPartSelectionListView(PartsGroupWidget);
PartsLayout->addWidget(mPartsWidget);
QHBoxLayout* Layout = new QHBoxLayout(this);
Layout->setContentsMargins(0, 0, 0, 0);
Layout->addWidget(mSplitter);
setLayout(Layout);
connect(mPartsWidget->selectionModel(), SIGNAL(currentChanged(const QModelIndex&, const QModelIndex&)), this, SLOT(PartChanged(const QModelIndex&, const QModelIndex&)));
connect(mFilterWidget, SIGNAL(textEdited(const QString&)), this, SLOT(FilterChanged(const QString&)));
connect(mCategoriesWidget, SIGNAL(currentItemChanged(QTreeWidgetItem*, QTreeWidgetItem*)), this, SLOT(CategoryChanged(QTreeWidgetItem*, QTreeWidgetItem*)));
UpdateCategories();
}
void lcPartSelectionWidget::resizeEvent(QResizeEvent* Event)
{
if (Event->size().width() > Event->size().height())
mSplitter->setOrientation(Qt::Horizontal);
else
mSplitter->setOrientation(Qt::Vertical);
QWidget::resizeEvent(Event);
}
void lcPartSelectionWidget::FilterChanged(const QString& Text)
{
mPartsWidget->GetFilterModel()->SetFilter(Text);
}
void lcPartSelectionWidget::CategoryChanged(QTreeWidgetItem* Current, QTreeWidgetItem* Previous)
{
Q_UNUSED(Previous);
if (Current != mModelsCategoryItem)
mPartsWidget->GetListModel()->SetCategory(mCategoriesWidget->indexOfTopLevelItem(Current));
else
mPartsWidget->GetListModel()->SetModelsCategory();
mPartsWidget->setCurrentIndex(mPartsWidget->GetFilterModel()->index(0, 0));
}
void lcPartSelectionWidget::PartChanged(const QModelIndex& Current, const QModelIndex& Previous)
{
Q_UNUSED(Current);
Q_UNUSED(Previous);
gMainWindow->SetCurrentPieceInfo(mPartsWidget->GetCurrentPart());
}
void lcPartSelectionWidget::Redraw()
{
mPartsWidget->GetListModel()->Redraw();
}
void lcPartSelectionWidget::SetDefaultPart()
{
for (int CategoryIdx = 0; CategoryIdx < mCategoriesWidget->topLevelItemCount(); CategoryIdx++)
{
QTreeWidgetItem* CategoryItem = mCategoriesWidget->topLevelItem(CategoryIdx);
if (CategoryItem->text(0) == "Brick")
{
mCategoriesWidget->setCurrentItem(CategoryItem);
break;
}
}
}
void lcPartSelectionWidget::UpdateCategories()
{
mCategoriesWidget->clear();
for (int CategoryIdx = 0; CategoryIdx < gCategories.GetSize(); CategoryIdx++)
new QTreeWidgetItem(mCategoriesWidget, QStringList((const char*)gCategories[CategoryIdx].Name));
mModelsCategoryItem = new QTreeWidgetItem(mCategoriesWidget, QStringList(tr("Models")));
}
void lcPartSelectionWidget::UpdateModels()
{
if (mCategoriesWidget->currentItem() == mModelsCategoryItem)
mPartsWidget->GetListModel()->SetModelsCategory();
}