#include "lc_global.h" #include "lc_qpropertiestree.h" #include "lc_colorpicker.h" #include "lc_application.h" #include "lc_model.h" #include "lc_mainwindow.h" #include "object.h" #include "piece.h" #include "camera.h" #include "light.h" #include "pieceinf.h" #include "lc_library.h" #include "lc_qutils.h" #include "lc_viewwidget.h" #include "lc_previewwidget.h" // Draw an icon indicating opened/closing branches static QIcon drawIndicatorIcon(const QPalette &palette, QStyle *style) { QPixmap pix(14, 14); pix.fill(Qt::transparent); QStyleOption branchOption; branchOption.rect = QRect(2, 2, 9, 9); // ### hardcoded in qcommonstyle.cpp branchOption.palette = palette; branchOption.state = QStyle::State_Children; QPainter p; // Draw closed state p.begin(&pix); style->drawPrimitive(QStyle::PE_IndicatorBranch, &branchOption, &p); p.end(); QIcon rc = pix; rc.addPixmap(pix, QIcon::Selected, QIcon::Off); // Draw opened state branchOption.state |= QStyle::State_Open; pix.fill(Qt::transparent); p.begin(&pix); style->drawPrimitive(QStyle::PE_IndicatorBranch, &branchOption, &p); p.end(); rc.addPixmap(pix, QIcon::Normal, QIcon::On); rc.addPixmap(pix, QIcon::Selected, QIcon::On); return rc; } static QIcon drawCheckBox(bool value) { QStyleOptionButton opt; opt.state |= value ? QStyle::State_On : QStyle::State_Off; opt.state |= QStyle::State_Enabled; const QStyle *style = QApplication::style(); // Figure out size of an indicator and make sure it is not scaled down in a list view item // by making the pixmap as big as a list view icon and centering the indicator in it. // (if it is smaller, it can't be helped) const int indicatorWidth = style->pixelMetric(QStyle::PM_IndicatorWidth, &opt); const int indicatorHeight = style->pixelMetric(QStyle::PM_IndicatorHeight, &opt); const int listViewIconSize = indicatorWidth; const int pixmapWidth = indicatorWidth; const int pixmapHeight = qMax(indicatorHeight, listViewIconSize); opt.rect = QRect(0, 0, indicatorWidth, indicatorHeight); QPixmap pixmap = QPixmap(pixmapWidth, pixmapHeight); pixmap.fill(Qt::transparent); { // Center? const int xoff = (pixmapWidth > indicatorWidth) ? (pixmapWidth - indicatorWidth) / 2 : 0; const int yoff = (pixmapHeight > indicatorHeight) ? (pixmapHeight - indicatorHeight) / 2 : 0; QPainter painter(&pixmap); painter.translate(xoff, yoff); style->drawPrimitive(QStyle::PE_IndicatorCheckBox, &opt, &painter); } return QIcon(pixmap); } int lcQPropertiesTreeDelegate::indentation(const QModelIndex &index) const { if (!m_treeWidget) return 0; QTreeWidgetItem *item = m_treeWidget->indexToItem(index); int indent = 0; while (item->parent()) { item = item->parent(); ++indent; } if (m_treeWidget->rootIsDecorated()) ++indent; return indent * m_treeWidget->indentation(); } void lcQPropertiesTreeDelegate::slotEditorDestroyed(QObject *object) { if (m_editedWidget == object) { m_editedWidget = nullptr; m_editedItem = nullptr; } } QWidget *lcQPropertiesTreeDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &style, const QModelIndex &index) const { Q_UNUSED(style); if (index.column() == 1 && m_treeWidget) { QTreeWidgetItem *item = m_treeWidget->indexToItem(index); if (item && (item->flags() & Qt::ItemIsEnabled)) { QWidget *editor = m_treeWidget->createEditor(parent, item); if (editor) { editor->setAutoFillBackground(true); editor->installEventFilter(const_cast(this)); m_editedItem = item; m_editedWidget = editor; connect(editor, SIGNAL(destroyed(QObject *)), this, SLOT(slotEditorDestroyed(QObject *))); return editor; } } } return nullptr; } void lcQPropertiesTreeDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const { Q_UNUSED(index) editor->setGeometry(option.rect.adjusted(0, 0, 0, -1)); } void lcQPropertiesTreeDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const { bool hasValue = true; if (m_treeWidget) hasValue = m_treeWidget->indexToItem(index)->data(0, lcQPropertiesTree::PropertyTypeRole).toInt() != lcQPropertiesTree::PropertyGroup; QStyleOptionViewItem opt = option; opt.state &= ~QStyle::State_HasFocus; if (index.column() == 1 && m_treeWidget) { QTreeWidgetItem *item = m_treeWidget->indexToItem(index); if (m_editedItem && m_editedItem == item) m_disablePainting = true; } QItemDelegate::paint(painter, opt, index); if (option.type) m_disablePainting = false; opt.palette.setCurrentColorGroup(QPalette::Active); QColor color = static_cast(QApplication::style()->styleHint(QStyle::SH_Table_GridLineColor, &opt)); painter->save(); painter->setPen(QPen(color)); if (!m_treeWidget || (!m_treeWidget->lastColumn(index.column()) && hasValue)) { int right = (option.direction == Qt::LeftToRight) ? option.rect.right() : option.rect.left(); painter->drawLine(right, option.rect.y(), right, option.rect.bottom()); } painter->restore(); } void lcQPropertiesTreeDelegate::drawDecoration(QPainter *painter, const QStyleOptionViewItem &option, const QRect &rect, const QPixmap &pixmap) const { if (m_disablePainting) return; QItemDelegate::drawDecoration(painter, option, rect, pixmap); } void lcQPropertiesTreeDelegate::drawDisplay(QPainter *painter, const QStyleOptionViewItem &option, const QRect &rect, const QString &text) const { if (m_disablePainting) return; QItemDelegate::drawDisplay(painter, option, rect, text); } QSize lcQPropertiesTreeDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const { return QItemDelegate::sizeHint(option, index) + QSize(3, 4); } bool lcQPropertiesTreeDelegate::eventFilter(QObject *object, QEvent *event) { if (event->type() == QEvent::FocusOut) { QFocusEvent *fe = static_cast(event); if (fe->reason() == Qt::ActiveWindowFocusReason) return false; } return QItemDelegate::eventFilter(object, event); } lcQPropertiesTree::lcQPropertiesTree(QWidget *parent) : QTreeWidget(parent) { setIconSize(QSize(18, 18)); setColumnCount(2); QStringList labels; labels.append(tr("Property")); labels.append(tr("Value")); setHeaderLabels(labels); header()->setSectionsMovable(false); header()->setSectionResizeMode(QHeaderView::ResizeToContents); header()->setVisible(false); setAlternatingRowColors(true); setRootIsDecorated(false); setEditTriggers(QAbstractItemView::EditKeyPressed); m_expandIcon = drawIndicatorIcon(palette(), style()); m_checkedIcon = drawCheckBox(true); m_uncheckedIcon = drawCheckBox(false); mDelegate = new lcQPropertiesTreeDelegate(parent); mDelegate->setTreeWidget(this); setItemDelegate(mDelegate); SetEmpty(); connect(header(), SIGNAL(sectionDoubleClicked(int)), this, SLOT(resizeColumnToContents(int))); } QSize lcQPropertiesTree::sizeHint() const { return QSize(200, -1); } void lcQPropertiesTree::drawRow(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const { QStyleOptionViewItem opt = option; QTreeWidgetItem *item = itemFromIndex(index); if (item->data(0, lcQPropertiesTree::PropertyTypeRole).toInt() == lcQPropertiesTree::PropertyGroup) { const QColor c = option.palette.color(QPalette::Dark); painter->fillRect(option.rect, c); opt.palette.setColor(QPalette::AlternateBase, c); } QTreeWidget::drawRow(painter, opt, index); QColor color = static_cast(QApplication::style()->styleHint(QStyle::SH_Table_GridLineColor, &opt)); painter->save(); painter->setPen(QPen(color)); painter->drawLine(opt.rect.x(), opt.rect.bottom(), opt.rect.right(), opt.rect.bottom()); painter->restore(); } void lcQPropertiesTree::keyPressEvent(QKeyEvent *event) { switch (event->key()) { case Qt::Key_Return: case Qt::Key_Enter: case Qt::Key_Space: // Trigger Edit if (!mDelegate->editedItem()) { if (const QTreeWidgetItem *item = currentItem()) { if (item->columnCount() >= 2 && ((item->flags() & (Qt::ItemIsEditable | Qt::ItemIsEnabled)) == (Qt::ItemIsEditable | Qt::ItemIsEnabled))) { event->accept(); // If the current position is at column 0, move to 1. QModelIndex index = currentIndex(); if (index.column() == 0) { index = index.sibling(index.row(), 1); setCurrentIndex(index); } edit(index); return; } } } break; default: break; } QTreeWidget::keyPressEvent(event); } void lcQPropertiesTree::mousePressEvent(QMouseEvent *event) { QTreeWidget::mousePressEvent(event); QTreeWidgetItem *item = itemAt(event->pos()); if (item) { if ((item != mDelegate->editedItem()) && (event->button() == Qt::LeftButton) && (header()->logicalIndexAt(event->pos().x()) == 1) && ((item->flags() & (Qt::ItemIsEditable | Qt::ItemIsEnabled)) == (Qt::ItemIsEditable | Qt::ItemIsEnabled))) editItem(item, 1); } } void lcQPropertiesTree::Update(const lcArray& Selection, lcObject* Focus) { lcPropertyWidgetMode Mode = LC_PROPERTY_WIDGET_EMPTY; if (Focus) { switch (Focus->GetType()) { case lcObjectType::Piece: Mode = LC_PROPERTY_WIDGET_PIECE; break; case lcObjectType::Camera: Mode = LC_PROPERTY_WIDGET_CAMERA; break; case lcObjectType::Light: Mode = LC_PROPERTY_WIDGET_LIGHT; break; } } else { for (int ObjectIdx = 0; ObjectIdx < Selection.GetSize(); ObjectIdx++) { switch (Selection[ObjectIdx]->GetType()) { case lcObjectType::Piece: if (Mode == LC_PROPERTY_WIDGET_EMPTY) Mode = LC_PROPERTY_WIDGET_PIECE; else if (Mode != LC_PROPERTY_WIDGET_PIECE) { Mode = LC_PROPERTY_WIDGET_MULTIPLE; ObjectIdx = Selection.GetSize(); } break; case lcObjectType::Camera: if (Mode != LC_PROPERTY_WIDGET_EMPTY) { Mode = LC_PROPERTY_WIDGET_MULTIPLE; ObjectIdx = Selection.GetSize(); } else { Mode = LC_PROPERTY_WIDGET_CAMERA; Focus = Selection[ObjectIdx]; } break; case lcObjectType::Light: if (Mode != LC_PROPERTY_WIDGET_EMPTY) { Mode = LC_PROPERTY_WIDGET_MULTIPLE; ObjectIdx = Selection.GetSize(); } else { Mode = LC_PROPERTY_WIDGET_LIGHT; Focus = Selection[ObjectIdx]; } break; } } } switch (Mode) { case LC_PROPERTY_WIDGET_EMPTY: SetEmpty(); break; case LC_PROPERTY_WIDGET_PIECE: SetPiece(Selection, Focus); break; case LC_PROPERTY_WIDGET_CAMERA: SetCamera(Focus); break; case LC_PROPERTY_WIDGET_LIGHT: SetLight(Focus); break; case LC_PROPERTY_WIDGET_MULTIPLE: SetMultiple(); break; } } class lcStepValidator : public QIntValidator { public: lcStepValidator(lcStep Min, lcStep Max, bool AllowEmpty) : QIntValidator(1, INT_MAX), mMin(Min), mMax(Max), mAllowEmpty(AllowEmpty) { } QValidator::State validate(QString& Input, int& Pos) const override { if (mAllowEmpty && Input.isEmpty()) return Acceptable; bool Ok; lcStep Step = Input.toUInt(&Ok); if (Ok) return (Step >= mMin && Step <= mMax) ? Acceptable : Invalid; return QIntValidator::validate(Input, Pos); } protected: lcStep mMin; lcStep mMax; bool mAllowEmpty; }; QWidget* lcQPropertiesTree::createEditor(QWidget* Parent, QTreeWidgetItem* Item) const { lcQPropertiesTree::PropertyType PropertyType = (lcQPropertiesTree::PropertyType)Item->data(0, lcQPropertiesTree::PropertyTypeRole).toInt(); switch (PropertyType) { case PropertyGroup: case PropertyBool: return nullptr; case PropertyFloat: { QLineEdit* Editor = new QLineEdit(Parent); float Value = Item->data(0, PropertyValueRole).toFloat(); QPointF Range = Item->data(0, PropertyRangeRole).toPointF(); Editor->setValidator(Range.isNull() ? new QDoubleValidator(Editor) : new QDoubleValidator(Range.x(), Range.y(), 1, Editor)); Editor->setText(lcFormatValueLocalized(Value)); connect(Editor, &QLineEdit::returnPressed, this, &lcQPropertiesTree::slotReturnPressed); return Editor; } case PropertyFloatReadOnly: return nullptr; case PropertyStep: { QLineEdit* Editor = new QLineEdit(Parent); lcStep Value = Item->data(0, PropertyValueRole).toUInt(); lcStep Show = partShow->data(0, PropertyValueRole).toUInt(); lcStep Hide = partHide->data(0, PropertyValueRole).toUInt(); if (Show && Hide) { if (Item == partShow) Editor->setValidator(new lcStepValidator(1, Hide - 1, false)); else Editor->setValidator(new lcStepValidator(Show + 1, LC_STEP_MAX, true)); } else Editor->setValidator(new lcStepValidator(1, LC_STEP_MAX, Item == partHide)); if (Item != partHide || Value != LC_STEP_MAX) Editor->setText(QString::number(Value)); connect(Editor, SIGNAL(returnPressed()), this, SLOT(slotReturnPressed())); return Editor; } case PropertyString: { QLineEdit *editor = new QLineEdit(Parent); QString value = Item->data(0, PropertyValueRole).toString(); editor->setText(value); connect(editor, SIGNAL(returnPressed()), this, SLOT(slotReturnPressed())); return editor; } case PropertyStringList: { QComboBox* editor = new QComboBox(Parent); if (Item == mCameraProjectionItem) { editor->addItems( { tr("Perspective"), tr("Orthographic") } ); } else if (Item == mLightTypeItem) { for (int LightTypeIndex = 0; LightTypeIndex < static_cast(lcLightType::Count); LightTypeIndex++) editor->addItem(lcLight::GetLightTypeString(static_cast(LightTypeIndex))); } else if (Item == mLightAreaShapeItem) { for (int LightAreaShapeIndex = 0; LightAreaShapeIndex < static_cast(lcLightAreaShape::Count); LightAreaShapeIndex++) editor->addItem(lcLight::GetAreaShapeString(static_cast(LightAreaShapeIndex))); } int value = Item->data(0, PropertyValueRole).toInt(); editor->setCurrentIndex(value); connect(editor, SIGNAL(currentIndexChanged(int)), this, SLOT(slotSetValue(int))); return editor; } case PropertyLightFormat: { QComboBox *editor = new QComboBox(Parent); editor->setSizeAdjustPolicy(QComboBox::AdjustToMinimumContentsLengthWithIcon); editor->setMinimumContentsLength(1); QStringList formats = { QLatin1String("Blender"), QLatin1String("POVRay") }; for (int i = 0; i < formats.size(); i++) editor->addItem(formats.at(i), QVariant::fromValue(i)); int value = Item->data(0, PropertyValueRole).toInt(); editor->setCurrentIndex(value); connect(editor, SIGNAL(currentIndexChanged(int)), this, SLOT(slotSetValue(int))); return editor; } case PropertyColor: { QPushButton *Editor = new QPushButton(Parent); QColor Value = Item->data(0, PropertyValueRole).value(); UpdateLightColorEditor(Editor, Value); connect(Editor, &QPushButton::clicked, this, &lcQPropertiesTree::LightColorButtonClicked); return Editor; } case PropertyPieceColor: { QPushButton *editor = new QPushButton(Parent); int value = Item->data(0, PropertyValueRole).toInt(); updateColorEditor(editor, value); connect(editor, SIGNAL(clicked()), this, SLOT(slotColorButtonClicked())); return editor; } case PropertyPart: { QComboBox *editor = new QComboBox(Parent); editor->setSizeAdjustPolicy(QComboBox::AdjustToMinimumContentsLengthWithIcon); editor->setMinimumContentsLength(1); lcPiecesLibrary* Library = lcGetPiecesLibrary(); std::vector SortedPieces; SortedPieces.reserve(Library->mPieces.size()); const lcModel* ActiveModel = gMainWindow->GetActiveModel(); for (const auto& PartIt : Library->mPieces) { PieceInfo* Info = PartIt.second; if (!Info->IsModel() || !Info->GetModel()->IncludesModel(ActiveModel)) SortedPieces.push_back(PartIt.second); } auto PieceCompare = [](PieceInfo* Info1, PieceInfo* Info2) { return strcmp(Info1->m_strDescription, Info2->m_strDescription) < 0; }; std::sort(SortedPieces.begin(), SortedPieces.end(), PieceCompare); for (PieceInfo* Info : SortedPieces) editor->addItem(Info->m_strDescription, QVariant::fromValue((void*)Info)); PieceInfo *info = (PieceInfo*)Item->data(0, PropertyValueRole).value(); editor->setCurrentIndex(editor->findData(QVariant::fromValue((void*)info))); connect(editor, SIGNAL(currentIndexChanged(int)), this, SLOT(slotSetValue(int))); return editor; } } return nullptr; } void lcQPropertiesTree::updateColorEditor(QPushButton *editor, int value) const { QImage img(12, 12, QImage::Format_ARGB32); img.fill(0); lcColor* color = &gColorList[value]; QPainter painter(&img); painter.setCompositionMode(QPainter::CompositionMode_Source); painter.setPen(Qt::darkGray); painter.setBrush(QColor::fromRgbF(color->Value[0], color->Value[1], color->Value[2])); painter.drawRect(0, 0, img.width() - 1, img.height() - 1); painter.end(); editor->setStyleSheet("Text-align:left"); editor->setIcon(QPixmap::fromImage(img)); editor->setText(color->Name); } void lcQPropertiesTree::UpdateLightColorEditor(QPushButton* Editor, QColor Color) const { QImage Image(12, 12, QImage::Format_ARGB32); Image.fill(0); QPainter Painter(&Image); Painter.setCompositionMode(QPainter::CompositionMode_Source); Painter.setPen(Qt::darkGray); Painter.setBrush(Color); Painter.drawRect(0, 0, Image.width() - 1, Image.height() - 1); Painter.end(); Editor->setStyleSheet("Text-align:left"); Editor->setIcon(QPixmap::fromImage(Image)); Editor->setText(Color.name().toUpper()); } void lcQPropertiesTree::slotToggled(bool Value) { QTreeWidgetItem* Item = mDelegate->editedItem(); lcModel* Model = gMainWindow->GetActiveModel(); lcObject* Focus = Model->GetFocusObject(); if (mWidgetMode == LC_PROPERTY_WIDGET_LIGHT) { lcLight* Light = (Focus && Focus->IsLight()) ? (lcLight*)Focus : nullptr; if (Light) { if (Item == lightEnableCutoff) { lcLightProperties Props = Light->GetLightProperties(); Props.mEnableCutoff = Value; Model->UpdateLight(Light, Props, LC_LIGHT_USE_CUTOFF); } else if (Item == mLightCastShadowItem) { Model->SetLightCastShadow(Light, Value); } } } } void lcQPropertiesTree::slotReturnPressed() { QLineEdit* Editor = (QLineEdit*)sender(); QTreeWidgetItem* Item = mDelegate->editedItem(); lcModel* Model = gMainWindow->GetActiveModel(); if (mWidgetMode == LC_PROPERTY_WIDGET_PIECE) { lcPiece* Piece = (mFocus && mFocus->IsPiece()) ? (lcPiece*)mFocus : nullptr; if (Item == partPositionX || Item == partPositionY || Item == partPositionZ) { lcVector3 Center; lcMatrix33 RelativeRotation; Model->GetMoveRotateTransform(Center, RelativeRotation); lcVector3 Position = Center; float Value = lcParseValueLocalized(Editor->text()); if (Item == partPositionX) Position[0] = Value; else if (Item == partPositionY) Position[1] = Value; else if (Item == partPositionZ) Position[2] = Value; lcVector3 Distance = Position - Center; Model->MoveSelectedObjects(Distance, Distance, false, false, true, true); } else if (Item == partRotationX || Item == partRotationY || Item == partRotationZ) { lcVector3 InitialRotation; if (Piece) InitialRotation = lcMatrix44ToEulerAngles(Piece->mModelWorld) * LC_RTOD; else InitialRotation = lcVector3(0.0f, 0.0f, 0.0f); lcVector3 Rotation = InitialRotation; float Value = lcParseValueLocalized(Editor->text()); if (Item == partRotationX) Rotation[0] = Value; else if (Item == partRotationY) Rotation[1] = Value; else if (Item == partRotationZ) Rotation[2] = Value; Model->RotateSelectedObjects(Rotation - InitialRotation, true, false, true, true); } else if (Item == partShow) { bool Ok = false; lcStep Step = Editor->text().toUInt(&Ok); if (Ok) Model->SetSelectedPiecesStepShow(Step); } else if (Item == partHide) { QString Text = Editor->text(); if (Text.isEmpty()) Model->SetSelectedPiecesStepHide(LC_STEP_MAX); else { bool Ok = false; lcStep Step = Text.toUInt(&Ok); if (Ok) Model->SetSelectedPiecesStepHide(Step); } } } else if (mWidgetMode == LC_PROPERTY_WIDGET_CAMERA) { lcCamera* Camera = (mFocus && mFocus->IsCamera()) ? (lcCamera*)mFocus : nullptr; if (Camera) { if (Item == cameraPositionX || Item == cameraPositionY || Item == cameraPositionZ) { lcVector3 Center = Camera->mPosition; lcVector3 Position = Center; float Value = lcParseValueLocalized(Editor->text()); if (Item == cameraPositionX) Position[0] = Value; else if (Item == cameraPositionY) Position[1] = Value; else if (Item == cameraPositionZ) Position[2] = Value; lcVector3 Distance = Position - Center; Model->MoveSelectedObjects(Distance, Distance, false, false, true, true); } else if (Item == cameraTargetX || Item == cameraTargetY || Item == cameraTargetZ) { lcVector3 Center = Camera->mTargetPosition; lcVector3 Position = Center; float Value = lcParseValueLocalized(Editor->text()); if (Item == cameraTargetX) Position[0] = Value; else if (Item == cameraTargetY) Position[1] = Value; else if (Item == cameraTargetZ) Position[2] = Value; lcVector3 Distance = Position - Center; Model->MoveSelectedObjects(Distance, Distance, false, false, true, true); } else if (Item == cameraUpX || Item == cameraUpY || Item == cameraUpZ) { lcVector3 Center = Camera->mUpVector; lcVector3 Position = Center; float Value = lcParseValueLocalized(Editor->text()); if (Item == cameraUpX) Position[0] = Value; else if (Item == cameraUpY) Position[1] = Value; else if (Item == cameraUpZ) Position[2] = Value; lcVector3 Distance = Position - Center; Model->MoveSelectedObjects(Distance, Distance, false, false, true, true); } else if (Item == cameraFOV) { float Value = lcParseValueLocalized(Editor->text()); Model->SetCameraFOV(Camera, Value); } else if (Item == cameraNear) { float Value = lcParseValueLocalized(Editor->text()); Model->SetCameraZNear(Camera, Value); } else if (Item == cameraFar) { float Value = lcParseValueLocalized(Editor->text()); Model->SetCameraZFar(Camera, Value); } else if (Item == mCameraNameItem) { QString Value = Editor->text(); Model->SetCameraName(Camera, Value); } } } else if (mWidgetMode == LC_PROPERTY_WIDGET_LIGHT) { lcLight* Light = (mFocus && mFocus->IsLight()) ? (lcLight*)mFocus : nullptr; if (Light) { lcLightProperties Props = Light->GetLightProperties(); QString Name = Light->GetName(); if (Item == lightPositionX || Item == lightPositionY || Item == lightPositionZ) { lcVector3 Center = Light->GetPosition(); lcVector3 Position = Center; float Value = lcParseValueLocalized(Editor->text()); if (Item == lightPositionX) Position[0] = Value; else if (Item == lightPositionY) Position[1] = Value; else if (Item == lightPositionZ) Position[2] = Value; lcVector3 Distance = Position - Center; Model->MoveSelectedObjects(Distance, Distance, false, false, true, true); } else if (Item == lightTargetX || Item == lightTargetY || Item == lightTargetZ) { // lcVector3 Center = Light->mTargetPosition; // lcVector3 Position = Center; // float Value = lcParseValueLocalized(Editor->text()); // if (Item == lightTargetX) // Position[0] = Value; // else if (Item == lightTargetY) // Position[1] = Value; // else if (Item == lightTargetZ) // Position[2] = Value; // // lcVector3 Distance = Position - Center; // // Model->MoveSelectedObjects(Distance, Distance, false, false, true, true); } else if (Item == mLightSpotConeAngleItem) { float Value = lcParseValueLocalized(Editor->text()); Model->SetSpotLightConeAngle(Light, Value); } else if (Item == mLightSpotPenumbraAngleItem) { float Value = lcParseValueLocalized(Editor->text()); Model->SetSpotLightPenumbraAngle(Light, Value); } else if (Item == mLightSpotTightnessItem) { float Value = lcParseValueLocalized(Editor->text()); Model->SetSpotLightTightness(Light, Value); } else if (Item == mLightSizeXItem) { lcVector2 Value = Light->GetSize(); Value[0] = lcParseValueLocalized(Editor->text()); Model->SetLightSize(Light, Value); } else if (Item == mLightSizeYItem) { lcVector2 Value = Light->GetSize(); Value[1] = lcParseValueLocalized(Editor->text()); Model->SetLightSize(Light, Value); } else if (Item == lightDiffuse) { Props.mLightDiffuse = lcParseValueLocalized(Editor->text()); Model->UpdateLight(Light, Props, LC_LIGHT_DIFFUSE); } else if (Item == lightSpecular) { Props.mLightSpecular = lcParseValueLocalized(Editor->text()); Model->UpdateLight(Light, Props, LC_LIGHT_SPECULAR); } else if (Item == lightExponent) { Props.mSpotExponent = lcParseValueLocalized(Editor->text()); Model->UpdateLight(Light, Props, LC_LIGHT_EXPONENT); } else if (Item == lightCutoff) { Props.mSpotCutoff = lcParseValueLocalized(Editor->text()); Model->UpdateLight(Light, Props, LC_LIGHT_CUTOFF); } else if (Item == lightAreaGridRows || Item == lightAreaGridColumns) { float Value = lcParseValueLocalized(Editor->text()); if (Item == lightAreaGridRows) Props.mAreaGrid[0] = Value; else if (Item == lightAreaGridColumns) Props.mAreaGrid[1] = Value; Model->UpdateLight(Light, Props, LC_LIGHT_AREA_GRID); } else if (Item == mLightNameItem) { QString Value = Editor->text(); Model->SetLightName(Light, Value); } } } } void lcQPropertiesTree::slotSetValue(int Value) { QTreeWidgetItem* Item = mDelegate->editedItem(); lcModel* Model = gMainWindow->GetActiveModel(); if (mWidgetMode == LC_PROPERTY_WIDGET_PIECE) { if (Item == mPieceColorItem) { Model->SetSelectedPiecesColorIndex(Value); QPushButton *editor = (QPushButton*)mDelegate->editor(); updateColorEditor(editor, Value); } else if (Item == mPieceIdItem) { QComboBox *editor = (QComboBox*)sender(); PieceInfo* Info = (PieceInfo*)editor->itemData(Value).value(); Model->SetSelectedPiecesPieceInfo(Info); int ColorIndex = gDefaultColor; lcObject* Focus = gMainWindow->GetActiveModel()->GetFocusObject(); if (Focus && Focus->IsPiece()) ColorIndex = ((lcPiece*)Focus)->GetColorIndex(); quint32 ColorCode = lcGetColorCode(ColorIndex); gMainWindow->PreviewPiece(Info->mFileName, ColorCode, false); } } else if (mWidgetMode == LC_PROPERTY_WIDGET_CAMERA) { lcObject* Focus = Model->GetFocusObject(); if (Focus && Focus->IsCamera()) { lcCamera* Camera = (lcCamera*)Focus; if (Item == mCameraProjectionItem) { Model->SetCameraOrthographic(Camera, Value == 1); } } } else if (mWidgetMode == LC_PROPERTY_WIDGET_LIGHT) { lcObject* Focus = Model->GetFocusObject(); lcLight* Light = (Focus && Focus->IsLight()) ? (lcLight*)Focus : nullptr; if (Light) { lcLightProperties Props = Light->GetLightProperties(); if (Item == mLightTypeItem) { Model->SetLightType(Light, static_cast(Value)); } else if (Item == mLightAreaShapeItem) { Model->SetAreaLightShape(Light, static_cast(Value)); } else if (Item == lightFormat) { Props.mPOVRayLight = Value; Model->UpdateLight(Light, Props, LC_LIGHT_POVRAY); } } } } void lcQPropertiesTree::slotColorButtonClicked() { int ColorIndex = gDefaultColor; lcObject* Focus = gMainWindow->GetActiveModel()->GetFocusObject(); if (Focus && Focus->IsPiece()) ColorIndex = ((lcPiece*)Focus)->GetColorIndex(); QWidget* Button = (QWidget*)sender(); if (!Button) return; lcColorPickerPopup* Popup = new lcColorPickerPopup(Button, ColorIndex); connect(Popup, &lcColorPickerPopup::Selected, this, &lcQPropertiesTree::slotSetValue); Popup->setMinimumSize(qMax(300, width()), qMax(200, static_cast(width() * 2 / 3))); #if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)) QScreen* Screen = Button->screen(); const QRect ScreenRect = Screen ? Screen->geometry() : QRect(); #elif (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)) QScreen* Screen = QGuiApplication::screenAt(Button->mapToGlobal(Button->rect().bottomLeft())); const QRect ScreenRect = Screen ? Screen->geometry() : QApplication::desktop()->geometry(); #else const QRect ScreenRect = QApplication::desktop()->geometry(); #endif int x = mapToGlobal(QPoint(0, 0)).x(); int y = Button->mapToGlobal(Button->rect().bottomLeft()).y(); if (x < ScreenRect.left()) x = ScreenRect.left(); if (y < ScreenRect.top()) y = ScreenRect.top(); if (x + Popup->width() > ScreenRect.right()) x = ScreenRect.right() - Popup->width(); if (y + Popup->height() > ScreenRect.bottom()) y = ScreenRect.bottom() - Popup->height(); Popup->move(QPoint(x, y)); Popup->setFocus(); Popup->show(); } void lcQPropertiesTree::LightColorButtonClicked() { lcModel* Model = gMainWindow->GetActiveModel(); lcObject* Focus = Model->GetFocusObject(); lcLight* Light = (Focus && Focus->IsLight()) ? (lcLight*)Focus : nullptr; if (!Light) return; QColor Color = QColorDialog::getColor(lcQColorFromVector3(Light->GetColor()), this, tr("Select Light Color")); if (!Color.isValid()) return; Model->SetLightColor(Light, lcVector3FromQColor(Color)); QPushButton* Editor = qobject_cast(mDelegate->editor()); if (Editor) UpdateLightColorEditor(Editor, Color); } QTreeWidgetItem *lcQPropertiesTree::addProperty(QTreeWidgetItem *parent, const QString& label, PropertyType propertyType) { QTreeWidgetItem *newItem; if (parent) newItem = new QTreeWidgetItem(parent, QStringList(label)); else newItem = new QTreeWidgetItem(this, QStringList(label)); newItem->setData(0, PropertyTypeRole, QVariant(propertyType)); if (propertyType != PropertyFloatReadOnly) newItem->setFlags(newItem->flags() | Qt::ItemIsEditable); newItem->setExpanded(true); if (propertyType == PropertyGroup) { newItem->setFirstColumnSpanned(true); newItem->setIcon(0, m_expandIcon); } return newItem; } void lcQPropertiesTree::SetEmpty() { clear(); mPieceAttributesItem = nullptr; partPosition = nullptr; partPositionX = nullptr; partPositionY = nullptr; partPositionZ = nullptr; partRotation = nullptr; partRotationX = nullptr; partRotationY = nullptr; partRotationZ = nullptr; partVisibility = nullptr; partShow = nullptr; partHide = nullptr; partAppearance = nullptr; mPieceColorItem = nullptr; mPieceIdItem = nullptr; cameraPosition = nullptr; cameraPositionX = nullptr; cameraPositionY = nullptr; cameraPositionZ = nullptr; cameraTarget = nullptr; cameraTargetX = nullptr; cameraTargetY = nullptr; cameraTargetZ = nullptr; cameraUp = nullptr; cameraUpX = nullptr; cameraUpY = nullptr; cameraUpZ = nullptr; mCameraAttributesItem = nullptr; mCameraProjectionItem = nullptr; cameraFOV = nullptr; cameraNear = nullptr; cameraFar = nullptr; mCameraNameItem = nullptr; lightConfiguration = nullptr; lightPosition = nullptr; lightPositionX = nullptr; lightPositionY = nullptr; lightPositionZ = nullptr; lightTarget = nullptr; lightTargetX = nullptr; lightTargetY = nullptr; lightTargetZ = nullptr; mLightColorItem = nullptr; mLightAttributesItem = nullptr; lightDiffuse = nullptr; lightSpecular = nullptr; lightCutoff = nullptr; lightEnableCutoff = nullptr; lightExponent = nullptr; mLightTypeItem = nullptr; mLightNameItem = nullptr; mLightSpotConeAngleItem = nullptr; mLightSpotPenumbraAngleItem = nullptr; mLightSpotTightnessItem = nullptr; mLightAreaShapeItem = nullptr; mLightSizeXItem = nullptr; mLightSizeYItem = nullptr; lightFormat = nullptr; mLightCastShadowItem = nullptr; lightAreaGridRows = nullptr; lightAreaGridColumns = nullptr; mWidgetMode = LC_PROPERTY_WIDGET_EMPTY; mFocus = nullptr; } void lcQPropertiesTree::SetPiece(const lcArray& Selection, lcObject* Focus) { if (mWidgetMode != LC_PROPERTY_WIDGET_PIECE) { SetEmpty(); mPieceAttributesItem = addProperty(nullptr, tr("Piece Attributes"), PropertyGroup); mPieceIdItem = addProperty(mPieceAttributesItem, tr("Part"), PropertyPart); mPieceColorItem = addProperty(mPieceAttributesItem, tr("Color"), PropertyPieceColor); partVisibility = addProperty(nullptr, tr("Visible Steps"), PropertyGroup); partShow = addProperty(partVisibility, tr("Show"), PropertyStep); partHide = addProperty(partVisibility, tr("Hide"), PropertyStep); partPosition = addProperty(nullptr, tr("Position"), PropertyGroup); partPositionX = addProperty(partPosition, tr("X"), PropertyFloat); partPositionY = addProperty(partPosition, tr("Y"), PropertyFloat); partPositionZ = addProperty(partPosition, tr("Z"), PropertyFloat); partRotation = addProperty(nullptr, tr("Rotation"), PropertyGroup); partRotationX = addProperty(partRotation, tr("X"), PropertyFloat); partRotationY = addProperty(partRotation, tr("Y"), PropertyFloat); partRotationZ = addProperty(partRotation, tr("Z"), PropertyFloat); mWidgetMode = LC_PROPERTY_WIDGET_PIECE; } lcModel* Model = gMainWindow->GetActiveModel(); lcPiece* Piece = (Focus && Focus->IsPiece()) ? (lcPiece*)Focus : nullptr; mFocus = Piece; lcVector3 Position; lcMatrix33 RelativeRotation; Model->GetMoveRotateTransform(Position, RelativeRotation); partPositionX->setText(1, lcFormatValueLocalized(Position[0])); partPositionX->setData(0, PropertyValueRole, Position[0]); partPositionY->setText(1, lcFormatValueLocalized(Position[1])); partPositionY->setData(0, PropertyValueRole, Position[1]); partPositionZ->setText(1, lcFormatValueLocalized(Position[2])); partPositionZ->setData(0, PropertyValueRole, Position[2]); lcVector3 Rotation; if (Piece) Rotation = lcMatrix44ToEulerAngles(Piece->mModelWorld) * LC_RTOD; else Rotation = lcVector3(0.0f, 0.0f, 0.0f); partRotationX->setText(1, lcFormatValueLocalized(Rotation[0])); partRotationX->setData(0, PropertyValueRole, Rotation[0]); partRotationY->setText(1, lcFormatValueLocalized(Rotation[1])); partRotationY->setData(0, PropertyValueRole, Rotation[1]); partRotationZ->setText(1, lcFormatValueLocalized(Rotation[2])); partRotationZ->setData(0, PropertyValueRole, Rotation[2]); lcStep Show = 0; lcStep Hide = 0; int ColorIndex = gDefaultColor; PieceInfo* Info = nullptr; if (Piece) { Show = Piece->GetStepShow(); Hide = Piece->GetStepHide(); ColorIndex = Piece->GetColorIndex(); Info = Piece->mPieceInfo; quint32 ColorCode = lcGetColorCode(ColorIndex); gMainWindow->PreviewPiece(Info->mFileName, ColorCode, false); } else { bool FirstPiece = true; for (int ObjectIdx = 0; ObjectIdx < Selection.GetSize(); ObjectIdx++) { lcObject* Object = Selection[ObjectIdx]; if (!Object->IsPiece()) continue; lcPiece* SelectedPiece = (lcPiece*)Object; if (FirstPiece) { Show = SelectedPiece->GetStepShow(); Hide = SelectedPiece->GetStepHide(); ColorIndex = SelectedPiece->GetColorIndex(); Info = SelectedPiece->mPieceInfo; FirstPiece = false; } else { if (SelectedPiece->GetStepShow() != Show) Show = 0; if (SelectedPiece->GetStepHide() != Hide) Hide = 0; if (SelectedPiece->GetColorIndex() != ColorIndex) ColorIndex = gDefaultColor; if (SelectedPiece->mPieceInfo != Info) Info = nullptr; } } } partShow->setText(1, QString::number(Show)); partShow->setData(0, PropertyValueRole, Show); partHide->setText(1, Hide == LC_STEP_MAX ? QString() : QString::number(Hide)); partHide->setData(0, PropertyValueRole, Hide); QImage img(16, 16, QImage::Format_ARGB32); img.fill(0); lcColor* color = &gColorList[ColorIndex]; QPainter painter(&img); painter.setCompositionMode(QPainter::CompositionMode_Source); painter.setPen(Qt::darkGray); painter.setBrush(QColor::fromRgbF(color->Value[0], color->Value[1], color->Value[2])); painter.drawRect(0, 0, img.width() - 1, img.height() - 1); painter.end(); mPieceColorItem->setIcon(1, QIcon(QPixmap::fromImage(img))); mPieceColorItem->setText(1, color->Name); mPieceColorItem->setData(0, PropertyValueRole, ColorIndex); QString text = Info ? Info->m_strDescription : QString(); mPieceIdItem->setText(1, text); mPieceIdItem->setToolTip(1, text); mPieceIdItem->setData(0, PropertyValueRole, QVariant::fromValue((void*)Info)); } void lcQPropertiesTree::SetCamera(lcObject* Focus) { if (mWidgetMode != LC_PROPERTY_WIDGET_CAMERA) { SetEmpty(); mCameraAttributesItem = addProperty(nullptr, tr("Camera Attributes"), PropertyGroup); mCameraNameItem = addProperty(mCameraAttributesItem, tr("Name"), PropertyString); mCameraProjectionItem = addProperty(mCameraAttributesItem, tr("Projection"), PropertyStringList); cameraFOV = addProperty(mCameraAttributesItem, tr("FOV"), PropertyFloat); cameraNear = addProperty(mCameraAttributesItem, tr("Near"), PropertyFloat); cameraFar = addProperty(mCameraAttributesItem, tr("Far"), PropertyFloat); cameraPosition = addProperty(nullptr, tr("Position"), PropertyGroup); cameraPositionX = addProperty(cameraPosition, tr("X"), PropertyFloat); cameraPositionY = addProperty(cameraPosition, tr("Y"), PropertyFloat); cameraPositionZ = addProperty(cameraPosition, tr("Z"), PropertyFloat); cameraTarget = addProperty(nullptr, tr("Target"), PropertyGroup); cameraTargetX = addProperty(cameraTarget, tr("X"), PropertyFloat); cameraTargetY = addProperty(cameraTarget, tr("Y"), PropertyFloat); cameraTargetZ = addProperty(cameraTarget, tr("Z"), PropertyFloat); cameraUp = addProperty(nullptr, tr("Up"), PropertyGroup); cameraUpX = addProperty(cameraUp, tr("X"), PropertyFloat); cameraUpY = addProperty(cameraUp, tr("Y"), PropertyFloat); cameraUpZ = addProperty(cameraUp, tr("Z"), PropertyFloat); mWidgetMode = LC_PROPERTY_WIDGET_CAMERA; } lcCamera* Camera = (Focus && Focus->IsCamera()) ? (lcCamera*)Focus : nullptr; mFocus = Camera; lcVector3 Position(0.0f, 0.0f, 0.0f); lcVector3 Target(0.0f, 0.0f, 0.0f); lcVector3 UpVector(0.0f, 0.0f, 0.0f); bool Ortho = false; float FoV = 60.0f; float ZNear = 1.0f; float ZFar = 100.0f; QString Name; if (Camera) { Position = Camera->mPosition; Target = Camera->mTargetPosition; UpVector = Camera->mUpVector; Ortho = Camera->IsOrtho(); FoV = Camera->m_fovy; ZNear = Camera->m_zNear; ZFar = Camera->m_zFar; Name = Camera->GetName(); } cameraPositionX->setText(1, lcFormatValueLocalized(Position[0])); cameraPositionX->setData(0, PropertyValueRole, Position[0]); cameraPositionY->setText(1, lcFormatValueLocalized(Position[1])); cameraPositionY->setData(0, PropertyValueRole, Position[1]); cameraPositionZ->setText(1, lcFormatValueLocalized(Position[2])); cameraPositionZ->setData(0, PropertyValueRole, Position[2]); cameraTargetX->setText(1, lcFormatValueLocalized(Target[0])); cameraTargetX->setData(0, PropertyValueRole, Target[0]); cameraTargetY->setText(1, lcFormatValueLocalized(Target[1])); cameraTargetY->setData(0, PropertyValueRole, Target[1]); cameraTargetZ->setText(1, lcFormatValueLocalized(Target[2])); cameraTargetZ->setData(0, PropertyValueRole, Target[2]); cameraUpX->setText(1, lcFormatValueLocalized(UpVector[0])); cameraUpX->setData(0, PropertyValueRole, UpVector[0]); cameraUpY->setText(1, lcFormatValueLocalized(UpVector[1])); cameraUpY->setData(0, PropertyValueRole, UpVector[1]); cameraUpZ->setText(1, lcFormatValueLocalized(UpVector[2])); cameraUpZ->setData(0, PropertyValueRole, UpVector[2]); mCameraProjectionItem->setText(1, Ortho ? tr("Orthographic") : tr("Perspective")); mCameraProjectionItem->setData(0, PropertyValueRole, Ortho); cameraFOV->setText(1, lcFormatValueLocalized(FoV)); cameraFOV->setData(0, PropertyValueRole, FoV); cameraNear->setText(1, lcFormatValueLocalized(ZNear)); cameraNear->setData(0, PropertyValueRole, ZNear); cameraFar->setText(1, lcFormatValueLocalized(ZFar)); cameraFar->setData(0, PropertyValueRole, ZFar); mCameraNameItem->setText(1, Name); mCameraNameItem->setData(0, PropertyValueRole, Name); } void lcQPropertiesTree::SetLight(lcObject* Focus) { lcLight* Light = (Focus && Focus->IsLight()) ? (lcLight*)Focus : nullptr; QString Name = tr("Light"); QString ExponentLabel = tr("Exponent"); QString FactorALabel = QLatin1String("FactorA"); QString Format, ExponentToolTip, FactorAToolTip, FactorBToolTip; lcLightType LightType = lcLightType::Point; lcLightAreaShape LightAreaShape = lcLightAreaShape::Rectangle; lcVector2 LightSize(0.0f, 0.0f); int FormatIndex = 0; float Diffuse = 0.0f; float Specular = 0.0f; float Cutoff = 0.0f; float Exponent = 0.0f; bool EnableCutoff = false; bool POVRayLight = false; bool CastShadow = true; PropertyType TargetProperty = PropertyFloat; lcVector3 Position(0.0f, 0.0f, 0.0f); lcVector3 Target(0.0f, 0.0f, 0.0f); QColor Color(Qt::white); float SpotConeAngle = 0.0f, SpotPenumbraAngle = 0.0f, SpotTightness = 0.0f; lcVector2 AreaGrid(0.0f, 0.0f); if (Light) { Name = Light->GetName(); POVRayLight = Light->mPOVRayLight; FormatIndex = static_cast(POVRayLight); Format = POVRayLight ? QLatin1String("POVRay") : QLatin1String("Blender"); CastShadow = Light->GetCastShadow(); Position = Light->GetPosition(); // Target = Light->mTargetPosition; Color = lcQColorFromVector3(Light->GetColor()); SpotConeAngle = Light->GetSpotConeAngle(); SpotPenumbraAngle = Light->GetSpotPenumbraAngle(); LightType = Light->GetLightType(); LightAreaShape = Light->GetAreaShape(); LightSize = Light->GetSize(); switch (LightType) { case lcLightType::Point: ExponentLabel = tr("Exponent"); break; case lcLightType::Spot: FactorBToolTip = tr("The softness of the spotlight edge."); ExponentLabel = tr("Power"); break; case lcLightType::Directional: ExponentLabel = tr("Strength"); break; case lcLightType::Area: ExponentLabel = tr("Power"); break; default: break; } Diffuse = Light->mLightDiffuse; Specular = Light->mLightSpecular; if (POVRayLight) { Exponent = Light->mPOVRayExponent; ExponentToolTip = tr("Intensity of the light with typical range of 0 to 1.0."); } else { Exponent = Light->mSpotExponent; ExponentToolTip = tr("Intensity of the light in watts."); } ExponentLabel = LightType == lcLightType::Directional ? tr("Strength") : tr("Power"); Cutoff = Light->mSpotCutoff; EnableCutoff = Light->mEnableCutoff; TargetProperty = LightType != lcLightType::Point ? PropertyFloat : PropertyFloatReadOnly; SpotTightness = Light->GetSpotTightness(); AreaGrid = Light->mAreaGrid; } if (mWidgetMode != LC_PROPERTY_WIDGET_LIGHT || mLightType != LightType || mLightAreaShape != LightAreaShape || mPOVRayLight != POVRayLight) { SetEmpty(); // Attributes mLightAttributesItem = addProperty(nullptr, tr("Light Attributes"), PropertyGroup); mLightNameItem = addProperty(mLightAttributesItem, tr("Name"), PropertyString); mLightTypeItem = addProperty(mLightAttributesItem, tr("Type"), PropertyStringList); mLightColorItem = addProperty(mLightAttributesItem, tr("Color"), PropertyColor); mLightCastShadowItem = addProperty(mLightAttributesItem, tr("Cast Shadows"), PropertyBool); switch (LightType) { case lcLightType::Point: mLightSizeXItem = addProperty(mLightAttributesItem, tr("Radius"), PropertyFloat); mLightSizeXItem->setToolTip(1, tr("Shadow soft size (Blender only).")); break; case lcLightType::Spot: mLightSpotConeAngleItem = addProperty(mLightAttributesItem, tr("Spot Cone Angle"), PropertyFloat); mLightSpotConeAngleItem->setToolTip(1, tr("The angle (in degrees) of the spot light's beam.")); mLightSpotPenumbraAngleItem = addProperty(mLightAttributesItem, tr("Spot Penumbra Angle"), PropertyFloat); mLightSpotPenumbraAngleItem->setToolTip(1, tr("The angle (in degrees) over which the intensity of the spot light falls off to zero.")); mLightSpotTightnessItem = addProperty(mLightAttributesItem, tr("Spot Tightness"), PropertyFloat); mLightSpotTightnessItem->setToolTip(1, tr("Additional exponential spot light edge softening (POV-Ray only).")); mLightSizeXItem = addProperty(mLightAttributesItem, tr("Radius"), PropertyFloat); mLightSizeXItem->setToolTip(1, tr("Shadow soft size (Blender only).")); break; case lcLightType::Directional: mLightSizeXItem = addProperty(mLightAttributesItem, tr("Angle"), PropertyFloat); mLightSizeXItem->setToolTip(1, tr("Angular diameter of the light (Blender only).")); break; case lcLightType::Area: mLightAreaShapeItem = addProperty(mLightAttributesItem, tr("Area Shape"), PropertyStringList); mLightAreaShapeItem->setToolTip(1, tr("The shape of the area light.")); switch (LightAreaShape) { case lcLightAreaShape::Rectangle: case lcLightAreaShape::Ellipse: mLightSizeXItem = addProperty(mLightAttributesItem, tr("Area Width"), PropertyFloat); mLightSizeXItem->setToolTip(1, tr("The width (X direction) of the area light.")); mLightSizeYItem = addProperty(mLightAttributesItem, tr("Area Height"), PropertyFloat); mLightSizeYItem->setToolTip(1, tr("The height (Y direction) of the area light.")); break; case lcLightAreaShape::Square: case lcLightAreaShape::Disk: mLightSizeXItem = addProperty(mLightAttributesItem, tr("Area Size"), PropertyFloat); mLightSizeXItem->setToolTip(1, tr("The size of the area light.")); mLightSizeYItem = nullptr; break; case lcLightAreaShape::Count: break; } if (POVRayLight) { lightAreaGridRows = addProperty(mLightAttributesItem, tr("Grid Rows"), PropertyFloat); lightAreaGridColumns = addProperty(mLightAttributesItem, tr("Grid Columns"), PropertyFloat); } break; case lcLightType::Count: break; } lightExponent = addProperty(mLightAttributesItem, ExponentLabel, PropertyFloat); if (!POVRayLight) { if (LightType != lcLightType::Directional) { lightEnableCutoff = addProperty(mLightAttributesItem, tr("Cutoff"), PropertyBool); lightCutoff = addProperty(mLightAttributesItem, tr("Cutoff Distance"), PropertyFloat); } lightDiffuse = addProperty(mLightAttributesItem, tr("Diffuse"), PropertyFloat); lightSpecular = addProperty(mLightAttributesItem, tr("Specular"), PropertyFloat); } // Configuration lightConfiguration = addProperty(nullptr, tr("Configuration"), PropertyGroup); lightFormat = addProperty(lightConfiguration, tr("Format"), PropertyLightFormat); // Position lightPosition = addProperty(nullptr, tr("Position"), PropertyGroup); lightPositionX = addProperty(lightPosition, tr("X"), PropertyFloat); lightPositionY = addProperty(lightPosition, tr("Y"), PropertyFloat); lightPositionZ = addProperty(lightPosition, tr("Z"), PropertyFloat); // Target Position if (LightType != lcLightType::Point && !(LightType == lcLightType::Area && POVRayLight)) { lightTarget = addProperty(nullptr, tr("Target"), PropertyGroup); lightTargetX = addProperty(lightTarget, tr("X"), TargetProperty); lightTargetY = addProperty(lightTarget, tr("Y"), TargetProperty); lightTargetZ = addProperty(lightTarget, tr("Z"), TargetProperty); } mWidgetMode = LC_PROPERTY_WIDGET_LIGHT; mLightType = LightType; mLightAreaShape = LightAreaShape; mPOVRayLight = POVRayLight; } mFocus = Light; lightPositionX->setText(1, lcFormatValueLocalized(Position[0])); lightPositionX->setData(0, PropertyValueRole, Position[0]); lightPositionY->setText(1, lcFormatValueLocalized(Position[1])); lightPositionY->setData(0, PropertyValueRole, Position[1]); lightPositionZ->setText(1, lcFormatValueLocalized(Position[2])); lightPositionZ->setData(0, PropertyValueRole, Position[2]); if (LightType != lcLightType::Point && !(LightType == lcLightType::Area && POVRayLight)) { lightTargetX->setText(1, lcFormatValueLocalized(Target[0])); lightTargetX->setData(0, PropertyValueRole, Target[0]); lightTargetY->setText(1, lcFormatValueLocalized(Target[1])); lightTargetY->setData(0, PropertyValueRole, Target[1]); lightTargetZ->setText(1, lcFormatValueLocalized(Target[2])); lightTargetZ->setData(0, PropertyValueRole, Target[2]); } QImage ColorImage(16, 16, QImage::Format_ARGB32); ColorImage.fill(0); QPainter painter(&ColorImage); painter.setCompositionMode(QPainter::CompositionMode_Source); painter.setPen(Qt::darkGray); painter.setBrush(Color); painter.drawRect(0, 0, ColorImage.width() - 1, ColorImage.height() - 1); painter.end(); mLightColorItem->setIcon(1, QIcon(QPixmap::fromImage(ColorImage))); mLightColorItem->setText(1, Color.name().toUpper()); mLightColorItem->setData(0, PropertyValueRole, Color); lightFormat->setText(1, Format); lightFormat->setData(0, PropertyValueRole, FormatIndex); mLightTypeItem->setText(1, lcLight::GetLightTypeString(LightType)); mLightTypeItem->setData(0, PropertyValueRole, static_cast(LightType)); mLightCastShadowItem->setCheckState(1, CastShadow ? Qt::Checked : Qt::Unchecked); mLightCastShadowItem->setData(0, PropertyValueRole, CastShadow); mLightSizeXItem->setText(1, lcFormatValueLocalized(LightSize[0])); mLightSizeXItem->setData(0, PropertyValueRole, LightSize[0]); if (mLightSizeYItem) { mLightSizeYItem->setText(1, lcFormatValueLocalized(LightSize[1])); mLightSizeYItem->setData(0, PropertyValueRole, LightSize[1]); } lightExponent->setText(1, lcFormatValueLocalized(Exponent)); lightExponent->setData(0, PropertyValueRole, Exponent); lightExponent->setToolTip(1, ExponentToolTip); switch (LightType) { case lcLightType::Point: break; case lcLightType::Spot: mLightSpotConeAngleItem->setText(1, lcFormatValueLocalized(SpotConeAngle)); mLightSpotConeAngleItem->setData(0, PropertyValueRole, SpotConeAngle); mLightSpotConeAngleItem->setData(0, PropertyRangeRole, QPointF(1.0, 180.0)); mLightSpotPenumbraAngleItem->setText(1, lcFormatValueLocalized(SpotPenumbraAngle)); mLightSpotPenumbraAngleItem->setData(0, PropertyValueRole, SpotPenumbraAngle); mLightSpotPenumbraAngleItem->setData(0, PropertyRangeRole, QPointF(0.0, 180.0)); mLightSpotTightnessItem->setText(1, lcFormatValueLocalized(SpotTightness)); mLightSpotTightnessItem->setData(0, PropertyValueRole, SpotTightness); mLightSpotTightnessItem->setData(0, PropertyRangeRole, QPointF(0.0, 100.0)); break; case lcLightType::Directional: break; case lcLightType::Area: mLightAreaShapeItem->setText(1, lcLight::GetAreaShapeString(LightAreaShape)); mLightAreaShapeItem->setData(0, PropertyValueRole, static_cast(LightAreaShape)); if (POVRayLight) { lightAreaGridRows->setText(1, lcFormatValueLocalized(AreaGrid[0])); lightAreaGridRows->setData(0, PropertyValueRole, AreaGrid[0]); lightAreaGridRows->setToolTip(1, tr("The number of sample rows in the area light.")); lightAreaGridColumns->setText(1, lcFormatValueLocalized(AreaGrid[1])); lightAreaGridColumns->setData(0, PropertyValueRole, AreaGrid[1]); lightAreaGridColumns->setToolTip(1, tr("The number of sample columns in the area light.")); } break; case lcLightType::Count: break; } if (!POVRayLight) { if (LightType != lcLightType::Directional) { lightEnableCutoff->setCheckState(1, EnableCutoff ? Qt::Checked : Qt::Unchecked); lightEnableCutoff->setData(0, PropertyValueRole, EnableCutoff); lightCutoff->setText(1, lcFormatValueLocalized(Cutoff)); lightCutoff->setData(0, PropertyValueRole, Cutoff); lightCutoff->setToolTip(1, tr("Distance at which the light influence will be set to 0.")); } lightDiffuse->setText(1, lcFormatValueLocalized(Diffuse)); lightDiffuse->setData(0, PropertyValueRole, Diffuse); lightDiffuse->setToolTip(1, tr("Diffuse reflection multiplier factor.")); lightSpecular->setText(1, lcFormatValueLocalized(Specular)); lightSpecular->setData(0, PropertyValueRole, Specular); lightSpecular->setToolTip(1, tr("Specular reflection multiplier factor.")); } mLightNameItem->setText(1, Name); mLightNameItem->setData(0, PropertyValueRole, QVariant::fromValue(Name)); } void lcQPropertiesTree::SetMultiple() { if (mWidgetMode != LC_PROPERTY_WIDGET_MULTIPLE) { SetEmpty(); addProperty(nullptr, tr("Multiple Objects Selected"), PropertyGroup); mWidgetMode = LC_PROPERTY_WIDGET_MULTIPLE; } mFocus = nullptr; } bool lcQPropertiesTree::lastColumn(int column) const { return header()->visualIndex(column) == columnCount() - 1; }