Preset steps can be added/removed
This commit is contained in:
@ -45,7 +45,7 @@ ColumnLayout {
|
|||||||
modal: true
|
modal: true
|
||||||
title: qsTr('Confirmation')
|
title: qsTr('Confirmation')
|
||||||
|
|
||||||
onAccepted: deviceTypesModel.removeRow(index)
|
onAccepted: if (!deviceTypesModel.removeRow(index)) console.warn('failed');
|
||||||
|
|
||||||
Label {
|
Label {
|
||||||
id: textContainer
|
id: textContainer
|
||||||
@ -62,6 +62,8 @@ ColumnLayout {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
Layout.preferredWidth: 300
|
||||||
|
Layout.maximumWidth: 300
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
|
|
||||||
enabled: deviceTypesListView.currentIndex !== -1
|
enabled: deviceTypesListView.currentIndex !== -1
|
||||||
@ -77,6 +79,7 @@ ColumnLayout {
|
|||||||
}
|
}
|
||||||
Label { text: qsTr("Name:") }
|
Label { text: qsTr("Name:") }
|
||||||
TextField {
|
TextField {
|
||||||
|
Layout.fillWidth: true
|
||||||
text: deviceTypesListView.currentData ? deviceTypesListView.currentData.name : ''
|
text: deviceTypesListView.currentData ? deviceTypesListView.currentData.name : ''
|
||||||
onTextEdited: if (deviceTypesListView.currentData) deviceTypesListView.currentData.name = text; else console.warn('discarded');
|
onTextEdited: if (deviceTypesListView.currentData) deviceTypesListView.currentData.name = text; else console.warn('discarded');
|
||||||
}
|
}
|
||||||
@ -109,34 +112,61 @@ ColumnLayout {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Pane {
|
EditableListView {
|
||||||
|
id: registersListView
|
||||||
|
|
||||||
Layout.preferredWidth: 300
|
Layout.preferredWidth: 300
|
||||||
|
Layout.maximumWidth: 300
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
|
|
||||||
enabled: deviceTypesListView.currentIndex !== -1
|
enabled: deviceTypesListView.currentIndex !== -1
|
||||||
|
|
||||||
Material.elevation: 6
|
textRole: 'registerTypeName'
|
||||||
|
|
||||||
EditableListView {
|
model: DeviceTypeRegistersModel {
|
||||||
id: registersListView
|
id: registersModel
|
||||||
anchors.fill: parent
|
controller: __controller
|
||||||
|
|
||||||
textRole: 'registerTypeName'
|
deviceTypeId: deviceTypesListView.currentData ? deviceTypesListView.currentData.id : -1
|
||||||
|
}
|
||||||
|
|
||||||
model: DeviceTypeRegistersModel {
|
onAddClicked: (index) => { const newIndex = index < 0 ? 0 : index + 1; if (registersModel.insertRow(newIndex)) currentIndex = newIndex; else console.warn('failed'); }
|
||||||
id: registersModel
|
onRemoveClicked: (index) => {
|
||||||
controller: __controller
|
const dialog = dialogComponent2.createObject(Overlay.overlay);
|
||||||
|
dialog.index = index;
|
||||||
|
dialog.open();
|
||||||
|
}
|
||||||
|
|
||||||
deviceTypeId: deviceTypesListView.currentData ? deviceTypesListView.currentData.id : -1
|
Component {
|
||||||
|
id: dialogComponent2
|
||||||
|
|
||||||
|
Dialog {
|
||||||
|
property int index
|
||||||
|
|
||||||
|
anchors.centerIn: parent
|
||||||
|
standardButtons: DialogButtonBox.Yes | DialogButtonBox.Cancel
|
||||||
|
modal: true
|
||||||
|
title: qsTr('Confirmation')
|
||||||
|
|
||||||
|
onAccepted: if (!registersModel.removeRow(index)) console.warn('failed');
|
||||||
|
|
||||||
|
Label {
|
||||||
|
id: textContainer
|
||||||
|
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
horizontalAlignment: Qt.AlignLeft
|
||||||
|
verticalAlignment: Qt.AlignTop
|
||||||
|
|
||||||
|
text: qsTr('Are you sure you want to remove row %0').arg(index)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
onAddClicked: (index) => { const newIndex = index < 0 ? 0 : index + 1; if (registersModel.insertRow(newIndex)) currentIndex = newIndex; else console.warn('failed'); }
|
|
||||||
onRemoveClicked: (index) => registersModel.removeRow(index)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
//Layout.fillWidth: true
|
Layout.preferredWidth: 300
|
||||||
|
Layout.maximumWidth: 300
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
|
|
||||||
enabled: registersListView.currentIndex >= 0
|
enabled: registersListView.currentIndex >= 0
|
||||||
@ -163,5 +193,9 @@ ColumnLayout {
|
|||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Item {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -48,7 +48,7 @@ ColumnLayout {
|
|||||||
modal: true
|
modal: true
|
||||||
title: qsTr('Confirmation')
|
title: qsTr('Confirmation')
|
||||||
|
|
||||||
onAccepted: model.removeRow(index)
|
onAccepted: if (!model.removeRow(index)) console.warn('failed');
|
||||||
|
|
||||||
Label {
|
Label {
|
||||||
id: textContainer
|
id: textContainer
|
||||||
@ -82,13 +82,13 @@ ColumnLayout {
|
|||||||
enabled: false
|
enabled: false
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
value: listView.currentData ? listView.currentData.id : -1
|
value: listView.currentData ? listView.currentData.id : -1
|
||||||
onValueModified: listView.currentData.id = value
|
onValueModified: if (listView.currentData) listView.currentData.id = value; else console.warn('discarded');
|
||||||
}
|
}
|
||||||
Label { text: qsTr("Name:") }
|
Label { text: qsTr("Name:") }
|
||||||
TextField {
|
TextField {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: listView.currentData ? listView.currentData.name : ''
|
text: listView.currentData ? listView.currentData.name : ''
|
||||||
onTextEdited: listView.currentData.name = text
|
onTextEdited: if (listView.currentData) listView.currentData.name = text; else console.warn('discarded');
|
||||||
}
|
}
|
||||||
Label { text: qsTr("DeviceType:") }
|
Label { text: qsTr("DeviceType:") }
|
||||||
IconComboBox {
|
IconComboBox {
|
||||||
@ -112,7 +112,7 @@ ColumnLayout {
|
|||||||
to: 512
|
to: 512
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
value: listView.currentData ? listView.currentData.address : -1
|
value: listView.currentData ? listView.currentData.address : -1
|
||||||
onValueModified: listView.currentData.address = value
|
onValueModified: if (listView.currentData) listView.currentData.address = value; else console.warn('discarded');
|
||||||
}
|
}
|
||||||
Label {
|
Label {
|
||||||
Layout.alignment: Qt.AlignTop | Qt.AlignLeft
|
Layout.alignment: Qt.AlignTop | Qt.AlignLeft
|
||||||
@ -121,7 +121,7 @@ ColumnLayout {
|
|||||||
Vector3DField {
|
Vector3DField {
|
||||||
id: positionField
|
id: positionField
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
onValueModified: listView.currentData.position = value;
|
onValueModified: if (listView.currentData) listView.currentData.position = value; else console.warn('discarded');
|
||||||
// TODO solve without onCurrentDataChanged
|
// TODO solve without onCurrentDataChanged
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -44,7 +44,7 @@ ColumnLayout {
|
|||||||
modal: true
|
modal: true
|
||||||
title: qsTr('Confirmation')
|
title: qsTr('Confirmation')
|
||||||
|
|
||||||
onAccepted: presetsModel.removeRow(index)
|
onAccepted: if (!presetsModel.removeRow(index)) console.warn('failed');
|
||||||
|
|
||||||
Label {
|
Label {
|
||||||
id: textContainer
|
id: textContainer
|
||||||
@ -61,6 +61,8 @@ ColumnLayout {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
Layout.preferredWidth: 300
|
||||||
|
Layout.maximumWidth: 300
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
|
|
||||||
enabled: presetsListView.currentIndex !== -1
|
enabled: presetsListView.currentIndex !== -1
|
||||||
@ -76,6 +78,7 @@ ColumnLayout {
|
|||||||
}
|
}
|
||||||
Label { text: qsTr("Name:") }
|
Label { text: qsTr("Name:") }
|
||||||
TextField {
|
TextField {
|
||||||
|
Layout.fillWidth: true
|
||||||
text: presetsListView.currentData ? presetsListView.currentData.name : ''
|
text: presetsListView.currentData ? presetsListView.currentData.name : ''
|
||||||
onTextEdited: if (presetsListView.currentData) presetsListView.currentData.name = text; else console.warn('discarded');
|
onTextEdited: if (presetsListView.currentData) presetsListView.currentData.name = text; else console.warn('discarded');
|
||||||
}
|
}
|
||||||
@ -86,24 +89,61 @@ ColumnLayout {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
PresetModel {
|
|
||||||
id: presetModel
|
|
||||||
controller: __controller
|
|
||||||
presetId: presetsListView.currentData ? presetsListView.currentData.id : -1
|
|
||||||
}
|
|
||||||
|
|
||||||
EditableListView {
|
EditableListView {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
|
||||||
enabled: presetsListView.currentIndex !== -1
|
enabled: presetsListView.currentIndex !== -1
|
||||||
|
|
||||||
model: PresetStepsModel {
|
model: PresetStepsModel {
|
||||||
|
id: presetStepsModel
|
||||||
|
|
||||||
controller: __controller
|
controller: __controller
|
||||||
presetId: presetsListView.currentData ? presetsListView.currentData.id : -1
|
presetId: presetsListView.currentData ? presetsListView.currentData.id : -1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onAddClicked: (index) => { const newIndex = index < 0 ? 0 : index + 1; if (presetStepsModel.insertRow(newIndex)) currentIndex = newIndex; else console.warn('failed'); }
|
||||||
|
onRemoveClicked: (index) => {
|
||||||
|
const dialog = dialogComponent2.createObject(Overlay.overlay);
|
||||||
|
dialog.index = index;
|
||||||
|
dialog.open();
|
||||||
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: dialogComponent2
|
||||||
|
|
||||||
|
Dialog {
|
||||||
|
property int index
|
||||||
|
|
||||||
|
anchors.centerIn: parent
|
||||||
|
standardButtons: DialogButtonBox.Yes | DialogButtonBox.Cancel
|
||||||
|
modal: true
|
||||||
|
title: qsTr('Confirmation')
|
||||||
|
|
||||||
|
onAccepted: if (!presetStepsModel.removeRow(index)) console.warn('failed');
|
||||||
|
|
||||||
|
Label {
|
||||||
|
id: textContainer
|
||||||
|
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
horizontalAlignment: Qt.AlignLeft
|
||||||
|
verticalAlignment: Qt.AlignTop
|
||||||
|
|
||||||
|
text: qsTr('Are you sure you want to remove row %0').arg(index)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
|
PresetModel {
|
||||||
|
id: presetModel
|
||||||
|
controller: __controller
|
||||||
|
presetId: presetsListView.currentData ? presetsListView.currentData.id : -1
|
||||||
|
}
|
||||||
|
|
||||||
RowLayout {
|
RowLayout {
|
||||||
Button {
|
Button {
|
||||||
text: qsTr('Auf Schieberegler\nunten kopieren');
|
text: qsTr('Auf Schieberegler\nunten kopieren');
|
||||||
|
@ -74,6 +74,9 @@ signals:
|
|||||||
void presetRemoved(int first, int last);
|
void presetRemoved(int first, int last);
|
||||||
void presetNameChanged(int row, const QString &name);
|
void presetNameChanged(int row, const QString &name);
|
||||||
|
|
||||||
|
void presetStepInserted(const PresetConfig &preset, int first, int last);
|
||||||
|
void presetStepRemoved(const PresetConfig &preset, int first, int last);
|
||||||
|
|
||||||
void sliderStatesChanged(const sliders_state_t &sliderStates);
|
void sliderStatesChanged(const sliders_state_t &sliderStates);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
@ -11,24 +11,24 @@ void PresetStepsModel::setController(DmxController *controller)
|
|||||||
|
|
||||||
if (m_controller)
|
if (m_controller)
|
||||||
{
|
{
|
||||||
// disconnect(m_controller, &DmxController::presetRegisterInserted,
|
disconnect(m_controller, &DmxController::presetStepInserted,
|
||||||
// this, &PresetStepsModel::otherPresetRegisterInserted);
|
this, &PresetStepsModel::otherPresetStepInserted);
|
||||||
// disconnect(m_controller, &DmxController::presetRegisterRemoved,
|
disconnect(m_controller, &DmxController::presetStepRemoved,
|
||||||
// this, &PresetStepsModel::otherPresetRegisterRemoved);
|
this, &PresetStepsModel::otherPresetStepRemoved);
|
||||||
// disconnect(m_controller, &DmxController::presetRegisterTypeChanged,
|
// disconnect(m_controller, &DmxController::presetStepTypeChanged,
|
||||||
// this, &PresetStepsModel::otherPresetRegisterTypeChanged);
|
// this, &PresetStepsModel::otherPresetStepTypeChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
m_controller = controller;
|
m_controller = controller;
|
||||||
|
|
||||||
if (m_controller)
|
if (m_controller)
|
||||||
{
|
{
|
||||||
// connect(m_controller, &DmxController::presetRegisterInserted,
|
connect(m_controller, &DmxController::presetStepInserted,
|
||||||
// this, &PresetStepsModel::otherPresetRegisterInserted);
|
this, &PresetStepsModel::otherPresetStepInserted);
|
||||||
// connect(m_controller, &DmxController::presetRegisterRemoved,
|
connect(m_controller, &DmxController::presetStepRemoved,
|
||||||
// this, &PresetStepsModel::otherPresetRegisterRemoved);
|
this, &PresetStepsModel::otherPresetStepRemoved);
|
||||||
// connect(m_controller, &DmxController::presetRegisterTypeChanged,
|
// connect(m_controller, &DmxController::presetStepTypeChanged,
|
||||||
// this, &PresetStepsModel::otherPresetRegisterTypeChanged);
|
// this, &PresetStepsModel::otherPresetStepTypeChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
endResetModel();
|
endResetModel();
|
||||||
@ -184,3 +184,169 @@ QHash<int, QByteArray> PresetStepsModel::roleNames() const
|
|||||||
{ Qt::DisplayRole, "name" },
|
{ Qt::DisplayRole, "name" },
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool PresetStepsModel::insertRows(int row, int count, const QModelIndex &parent)
|
||||||
|
{
|
||||||
|
if (parent.isValid())
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!m_controller)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_presetId == -1)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto &presets = m_controller->lightProject().presets;
|
||||||
|
auto presetPtr = presets.findById(m_presetId);
|
||||||
|
if (!presetPtr)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto &preset = *presetPtr;
|
||||||
|
|
||||||
|
if (row < 0)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto &steps = preset.steps;
|
||||||
|
|
||||||
|
beginInsertRows({}, row, row+count-1);
|
||||||
|
{
|
||||||
|
QMutexLocker locker{&m_controller->mutex()};
|
||||||
|
auto iter = std::begin(steps) + row;
|
||||||
|
for (auto i = 0; i < count; i++)
|
||||||
|
iter = steps.insert(iter, PresetStepConfig{}) + 1;
|
||||||
|
}
|
||||||
|
endInsertRows();
|
||||||
|
|
||||||
|
disconnect(m_controller, &DmxController::presetStepInserted,
|
||||||
|
this, &PresetStepsModel::otherPresetStepInserted);
|
||||||
|
emit m_controller->presetStepInserted(preset, row, row+count-1);
|
||||||
|
connect(m_controller, &DmxController::presetStepInserted,
|
||||||
|
this, &PresetStepsModel::otherPresetStepInserted);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PresetStepsModel::removeRows(int row, int count, const QModelIndex &parent)
|
||||||
|
{
|
||||||
|
if (parent.isValid())
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!m_controller)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_presetId == -1)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto &presets = m_controller->lightProject().presets;
|
||||||
|
auto presetPtr = presets.findById(m_presetId);
|
||||||
|
if (!presetPtr)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto &preset = *presetPtr;
|
||||||
|
|
||||||
|
if (row < 0)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto &steps = preset.steps;
|
||||||
|
|
||||||
|
if (row >= steps.size())
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (row + count > steps.size())
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
beginRemoveRows({}, row, row+count-1);
|
||||||
|
{
|
||||||
|
QMutexLocker locker{&m_controller->mutex()};
|
||||||
|
auto begin = std::begin(steps) + row;
|
||||||
|
auto end = begin + count;
|
||||||
|
steps.erase(begin, end);
|
||||||
|
}
|
||||||
|
endRemoveRows();
|
||||||
|
|
||||||
|
disconnect(m_controller, &DmxController::presetStepRemoved,
|
||||||
|
this, &PresetStepsModel::otherPresetStepRemoved);
|
||||||
|
emit m_controller->presetStepRemoved(preset, row, row+count-1);
|
||||||
|
connect(m_controller, &DmxController::presetStepRemoved,
|
||||||
|
this, &PresetStepsModel::otherPresetStepRemoved);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PresetStepsModel::otherPresetStepInserted(const PresetConfig &preset, int first, int last)
|
||||||
|
{
|
||||||
|
if (!m_controller)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_presetId == -1)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_presetId != preset.id)
|
||||||
|
return;
|
||||||
|
|
||||||
|
beginInsertRows({}, first, last);
|
||||||
|
endInsertRows();
|
||||||
|
}
|
||||||
|
|
||||||
|
void PresetStepsModel::otherPresetStepRemoved(const PresetConfig &preset, int first, int last)
|
||||||
|
{
|
||||||
|
if (!m_controller)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_presetId == -1)
|
||||||
|
{
|
||||||
|
qWarning() << "hilfe" << __LINE__;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_presetId != preset.id)
|
||||||
|
return;
|
||||||
|
|
||||||
|
beginRemoveRows({}, first, last);
|
||||||
|
endRemoveRows();
|
||||||
|
}
|
||||||
|
@ -27,10 +27,17 @@ public:
|
|||||||
QMap<int, QVariant> itemData(const QModelIndex &index) const override;
|
QMap<int, QVariant> itemData(const QModelIndex &index) const override;
|
||||||
QHash<int, QByteArray> roleNames() const override;
|
QHash<int, QByteArray> roleNames() const override;
|
||||||
|
|
||||||
|
bool insertRows(int row, int count, const QModelIndex &parent) override;
|
||||||
|
bool removeRows(int row, int count, const QModelIndex &parent) override;
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void controllerChanged(DmxController *controller);
|
void controllerChanged(DmxController *controller);
|
||||||
void presetIdChanged(int presetId);
|
void presetIdChanged(int presetId);
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void otherPresetStepInserted(const PresetConfig &preset, int first, int last);
|
||||||
|
void otherPresetStepRemoved(const PresetConfig &preset, int first, int last);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
DmxController *m_controller{};
|
DmxController *m_controller{};
|
||||||
int m_presetId{-1};
|
int m_presetId{-1};
|
||||||
|
Reference in New Issue
Block a user