14 #include "xnodeconnector.h"
16 #include <QPushButton>
19 #include <QListWidget>
22 #include <QTableWidget>
23 #include <QHeaderView>
24 #include <QDoubleSpinBox>
27 #include <QTextBrowser>
31 #include <QToolButton>
32 #include <QFileDialog>
33 #include <QColorDialog>
35 #include <QMainWindow>
36 #include <QApplication>
40 #include "icons/icon.h"
41 #include "messagebox.h"
42 #include <type_traits>
45 #define UI_DISP_DELAY 10
47 static std::deque<shared_ptr<XStatusPrinter> > s_statusPrinterCreating;
48 static std::deque<shared_ptr<XQConnector> > s_conCreating;
49 static std::map<const QWidget*, weak_ptr<XNode> > s_widgetMap;
51 void sharedPtrQDeleter_(QObject *obj) {
58 XQConnectorHolder_::XQConnectorHolder_(
XQConnector *con) :
60 m_connector = s_conCreating.back();
61 s_conCreating.pop_back();
62 connect(con->m_pWidget, SIGNAL( destroyed() ),
this, SLOT( destroyed() ) );
63 assert(con->shared_from_this());
65 XQConnectorHolder_::~XQConnectorHolder_() {
67 disconnect(m_connector->m_pWidget, SIGNAL( destroyed() ),
this, SLOT( destroyed() ) );
70 XQConnectorHolder_::isAlive()
const {
75 XQConnectorHolder_::destroyed () {
76 disconnect(m_connector->m_pWidget, SIGNAL( destroyed() ),
this, SLOT( destroyed() ) );
77 std::map<const QWidget*, weak_ptr<XNode> >::iterator it = s_widgetMap.find(m_connector->m_pWidget);
78 assert(it != s_widgetMap.end());
79 s_widgetMap.erase(it);
80 m_connector->m_pWidget = 0L;
90 s_conCreating.push_back(shared_ptr<XQConnector>(
this));
93 m_lsnUIEnabled = tr[ *node].onUIFlagsChanged().connectWeakly(shared_from_this(), &XQConnector::onUIFlagsChanged,
94 XListener::FLAG_MAIN_THREAD_CALL | XListener::FLAG_AVOID_DUP);
96 XQConnector::onUIFlagsChanged(
Snapshot(*node), node.get());
97 dbgPrint(QString(
"connector %1 created., addr=0x%2, size=0x%3")
98 .arg(node->getLabel())
99 .arg((uintptr_t)
this, 0, 16)
102 auto ret = s_widgetMap.insert(std::pair<
const QWidget*, weak_ptr<XNode> >(item, node));
104 gErrPrint(
"Connection to Widget Duplicated!");
111 m_pWidget->setEnabled(
false);
112 dbgPrint(QString(
"connector %1 released., addr=0x%2").arg(objectName()).arg((uintptr_t)
this, 0, 16));
113 auto it = s_widgetMap.find(m_pWidget);
114 assert(it != s_widgetMap.end());
115 s_widgetMap.erase(it);
118 dbgPrint(QString(
"connector %1 & widget released., addr=0x%2").arg(objectName()).arg((uintptr_t)
this, 0, 16));
126 XQConnector::connectedNode(
const QWidget *item) {
127 auto it = s_widgetMap.find(item);
128 if(it == s_widgetMap.end())
129 return shared_ptr<XNode>();
130 return it->second.lock();
134 XQConnector::onUIFlagsChanged(
const Snapshot &shot,
XNode *node) {
135 m_pWidget->setEnabled(shot[ *node].isUIEnabled());
138 XQButtonConnector::XQButtonConnector(
const shared_ptr<XTouchableNode> &node,
139 QAbstractButton *item)
141 m_node(node), m_pItem(item) {
143 connect(item, SIGNAL( clicked() ),
this, SLOT( onClick() ) );
145 m_lsnTouch = tr[ *node].onTouch().connectWeakly
146 (shared_from_this(), &XQButtonConnector::onTouch, XListener::FLAG_MAIN_THREAD_CALL);
149 XQButtonConnector::~XQButtonConnector() {
152 XQButtonConnector::onClick() {
154 tr[ *m_node].touch();
155 tr.unmark(m_lsnTouch);
162 XValueQConnector::XValueQConnector(
const shared_ptr<XValueNodeBase> &node, QWidget *item)
165 m_lsnValueChanged = tr[ *node].onValueChanged().connectWeakly(
166 shared_from_this(), &XValueQConnector::onValueChanged,
167 XListener::FLAG_MAIN_THREAD_CALL | XListener::FLAG_AVOID_DUP);
170 XValueQConnector::~XValueQConnector() {
173 XQLineEditConnector::XQLineEditConnector(
174 const shared_ptr<XValueNodeBase> &node, QLineEdit *item,
bool forcereturn)
176 m_node(node), m_pItem(item), m_editing(false) {
177 connect(item, SIGNAL( returnPressed() ),
this, SLOT( onReturnPressed() ) );
179 connect(item, SIGNAL( editingFinished() ),
this, SLOT( onExit() ) );
180 connect(item, SIGNAL( textEdited(
const QString &) ),
181 this, SLOT( onTextChanged(
const QString &) ) );
184 connect(item, SIGNAL( textEdited(
const QString &) ),
185 this, SLOT( onTextChanged2(
const QString &) ) );
187 onValueChanged(
Snapshot( *node), node.get());
190 XQLineEditConnector::onTextChanged(
const QString &text) {
191 QPalette palette(m_pItem->palette());
192 palette.setColor(QPalette::Text, Qt::blue);
193 m_pItem->setPalette(palette);
197 XQLineEditConnector::onTextChanged2(
const QString &text) {
198 QPalette palette(m_pItem->palette());
201 tr[ *m_node].str(text);
202 tr.unmark(m_lsnValueChanged);
204 palette.setColor(QPalette::Text, Qt::black);
207 palette.setColor(QPalette::Text, Qt::red);
209 m_pItem->setPalette(palette);
212 XQLineEditConnector::onReturnPressed() {
213 QPalette palette(m_pItem->palette());
216 tr[ *m_node].str(m_pItem->text());
217 tr.unmark(m_lsnValueChanged);
219 palette.setColor(QPalette::Text, Qt::black);
223 palette.setColor(QPalette::Text, Qt::red);
225 m_pItem->setPalette(palette);
229 XQLineEditConnector::onExit() {
230 if( !m_editing)
return;
231 QPalette palette(m_pItem->palette());
232 palette.setColor(QPalette::Text, Qt::black);
233 m_pItem->setPalette(palette);
235 if(QString(shot[ *m_node].to_str()) != m_pItem->text()) {
236 m_pItem->blockSignals(
true);
237 m_pItem->setText(shot[ *m_node].to_str());
238 m_pItem->blockSignals(
false);
239 shared_ptr<XStatusPrinter> statusprinter = g_statusPrinter;
240 if(statusprinter) statusprinter->printMessage(i18n(
"Input canceled."),
true, 0, 0,
true);
245 m_pItem->blockSignals(
true);
246 m_pItem->setText(shot[ *node].to_str());
247 m_pItem->blockSignals(
false);
250 template <
class QN,
class XN,
class X>
252 QN *item, QSlider *slider)
258 if(std::is_integral<X>::value) {
259 slider->setRange(item->minimum(), item->maximum());
260 slider->setSingleStep(item->singleStep());
263 slider->setRange(0, 100);
264 slider->setSingleStep(5);
268 XQSpinBoxConnector::XQSpinBoxConnector(
const shared_ptr<XIntNode> &node,
269 QSpinBox *item, QSlider *slider)
271 connect(item, SIGNAL( valueChanged(
int) ),
this, SLOT( onChange(
int) ) );
273 connect(slider, SIGNAL( valueChanged(
int) ),
this, SLOT( onSliderChange(
int) ) );
275 onValueChanged(
Snapshot( *node), node.get());
277 XQSpinBoxUnsignedConnector::XQSpinBoxUnsignedConnector(
const shared_ptr<XUIntNode> &node,
278 QSpinBox *item, QSlider *slider)
280 connect(item, SIGNAL( valueChanged(
int) ),
this, SLOT( onChange(
int) ) );
282 connect(slider, SIGNAL( valueChanged(
int) ),
this, SLOT( onSliderChange(
int) ) );
284 onValueChanged(
Snapshot( *node), node.get());
286 XQDoubleSpinBoxConnector::XQDoubleSpinBoxConnector(
const shared_ptr<XDoubleNode> &node,
287 QDoubleSpinBox *item, QSlider *slider)
289 connect(item, SIGNAL( valueChanged(
double) ),
this, SLOT( onChange(
double) ) );
291 connect(slider, SIGNAL( valueChanged(
int) ),
this, SLOT( onSliderChange(
int) ) );
293 onValueChanged(
Snapshot( *node), node.get());
295 template <
class QN,
class XN,
class X>
299 m_pSlider->setEnabled(shot[ *node].isUIEnabled());
300 XQConnector::onUIFlagsChanged(shot, node);
303 template <
class QN,
class XN,
class X>
307 if( !std::is_integral<X>::value) {
308 double max = m_pItem->maximum();
309 double min = m_pItem->minimum();
310 var = lrint((val - min) / (max - min) * 100);
313 m_pSlider->blockSignals(
true);
314 m_pSlider->setValue(var);
315 m_pSlider->blockSignals(
false);
319 tr.unmark(m_lsnValueChanged);
322 template <
class QN,
class XN,
class X>
326 if( !std::is_integral<X>::value) {
327 double max = m_pItem->maximum();
328 double min = m_pItem->minimum();
329 var = val / 100.0 * (max - min) + min;
331 m_pItem->blockSignals(
true);
332 m_pItem->setValue(var);
333 m_pItem->blockSignals(
false);
336 tr.unmark(m_lsnValueChanged);
339 template <
class QN,
class XN,
class X>
342 X var = std::is_integral<X>::value ?
343 QString(shot[ *node].to_str()).toInt() :
344 QString(shot[ *node].to_str()).toDouble();
345 m_pItem->blockSignals(
true);
346 m_pItem->setValue(var);
347 m_pItem->blockSignals(
false);
350 if( !std::is_integral<X>::value) {
351 double max = m_pItem->maximum();
352 double min = m_pItem->minimum();
353 svar = lrint((var - min) / (max - min) * 100);
355 m_pSlider->blockSignals(
true);
356 m_pSlider->setValue(svar);
357 m_pSlider->blockSignals(
false);
365 XFilePathConnector::XFilePathConnector(
const shared_ptr<XStringNode> &node,
366 QLineEdit *edit, QAbstractButton *btn,
const char *filter,
bool saving)
368 m_pBtn(btn), m_filter(filter), m_saving(saving) {
369 connect(btn, SIGNAL( clicked ( ) ),
this, SLOT( onClick( ) ) );
372 XFilePathConnector::onClick() {
373 #if QT_VERSION < QT_VERSION_CHECK(5,0,0)
375 m_saving ? QFileDialog::
376 getSaveFileName(m_pItem, QString(), m_pItem->text(), m_filter)
378 getOpenFileName(m_pItem, QString(), m_pItem->text(), m_filter);
381 QFileDialog dialog(m_pItem);
382 dialog.setViewMode(QFileDialog::Detail);
383 dialog.setNameFilter(m_filter);
385 int perpos = m_filter.find_first_of(
'.');
386 assert(perpos != std::string::npos);
387 XString suf = m_filter.substr(perpos + 1, 3);
388 dialog.setDefaultSuffix(suf);
389 dialog.setDirectory(m_pItem->text());
390 dialog.setAcceptMode(m_saving ? QFileDialog::AcceptSave: QFileDialog::AcceptOpen);
393 QString str = dialog.selectedFiles().at(0);
396 m_pItem->blockSignals(
true);
397 m_pItem->setText(str);
398 m_pItem->blockSignals(
false);
401 tr[ *m_node].str(str);
402 tr.unmark(m_lsnValueChanged);
413 XQLineEditConnector::onValueChanged(shot, node);
416 XQLabelConnector::XQLabelConnector(
const shared_ptr<XValueNodeBase> &node, QLabel *item)
418 m_node(node), m_pItem(item) {
419 onValueChanged(
Snapshot( *node), node.get());
424 m_pItem->setText(shot[ *node].to_str());
427 XQTextBrowserConnector::XQTextBrowserConnector(
const shared_ptr<XValueNodeBase> &node, QTextBrowser *item)
429 m_node(node), m_pItem(item) {
430 onValueChanged(
Snapshot( *node), node.get());
434 m_pItem->setText(shot[ *node].to_str());
437 XQLCDNumberConnector::XQLCDNumberConnector(
const shared_ptr<XDoubleNode> &node, QLCDNumber *item)
439 m_node(node), m_pItem(item) {
440 onValueChanged(
Snapshot( *node), node.get());
445 QString buf(shot[ *node].to_str());
446 if((
int)buf.length() > m_pItem->digitCount())
447 m_pItem->setDigitCount(buf.length());
448 m_pItem->display(buf);
451 XQLedConnector::XQLedConnector(
const shared_ptr<XBoolNode> &node, QPushButton *item)
453 m_node(node), m_pItem(item) {
454 item->setCheckable(
false);
455 item->setAutoDefault(
false);
457 item->setFocusPolicy(Qt::NoFocus);
458 item->setIconSize(QSize(16, 16));
459 onValueChanged(
Snapshot( *node), node.get());
464 m_pItem->setIcon(shot[ *m_node] ?
465 *g_pIconLEDOn : *g_pIconLEDOff);
468 XQToggleButtonConnector::XQToggleButtonConnector(
const shared_ptr<XBoolNode> &node, QAbstractButton *item)
470 m_node(node), m_pItem(item) {
471 connect(item, SIGNAL( clicked() ),
this, SLOT( onClick() ) );
472 onValueChanged(
Snapshot( *node), node.get());
476 XQToggleButtonConnector::onClick() {
478 tr[ *m_node] = m_pItem->isChecked();
479 tr.unmark(m_lsnValueChanged);
485 if((shot[ *m_node]) ^ m_pItem->isChecked())
489 XListQConnector::XListQConnector(
const shared_ptr<XListNodeBase> &node, QTableWidget *item)
491 m_pItem(item), m_list(node) {
494 m_lsnMove = tr[ *node].onMove().connectWeakly(shared_from_this(),
495 &XListQConnector::onMove, XListener::FLAG_MAIN_THREAD_CALL);
496 m_lsnCatch = tr[ *node].onCatch().connectWeakly(shared_from_this(),
497 &XListQConnector::onCatch, XListener::FLAG_MAIN_THREAD_CALL);
498 m_lsnRelease = tr[ *node].onRelease().connectWeakly(shared_from_this(),
499 &XListQConnector::onRelease, XListener::FLAG_MAIN_THREAD_CALL);
501 QHeaderView *header = m_pItem->verticalHeader();
502 #if QT_VERSION < QT_VERSION_CHECK(5,0,0)
503 header->setMovable(
true);
504 header->setResizeMode(QHeaderView::ResizeToContents);
506 header->setSectionsMovable(
true);
507 header->setSectionResizeMode(QHeaderView::ResizeToContents);
509 connect(header, SIGNAL( sectionMoved(
int,
int,
int)),
510 this, SLOT( OnSectionMoved(
int,
int,
int)));
511 header->setToolTip(i18n(
"Use drag-n-drop to reorder."));
513 XListQConnector::~XListQConnector() {
515 QHeaderView *header = m_pItem->verticalHeader();
516 disconnect(header, NULL,
this, NULL );
517 m_pItem->clearSpans();
521 XListQConnector::OnSectionMoved(
int logicalIndex,
int oldVisualIndex,
int newVisualIndex) {
522 int fromIndex = oldVisualIndex;
523 int toIndex = newVisualIndex;
524 if(toIndex == fromIndex)
527 unsigned int src = fromIndex;
528 unsigned int dst = toIndex;
529 if( !tr.size() || src >= tr.size() || (dst >= tr.size())) {
533 for(
int i = src; i != dst;) {
534 int next = i + ((src < dst) ? 1: -1);
535 m_list->swap(tr, tr.list()->at(i), tr.list()->at(next));
538 tr.unmark(m_lsnMove);
543 QHeaderView *header = m_pItem->verticalHeader();
544 int dir = (e.src_idx - e.dst_idx > 0) ? 1 : -1;
545 for(
unsigned int idx = e.dst_idx; idx != e.src_idx; idx += dir) {
546 header->swapSections(idx, idx + dir);
550 XItemQConnector::XItemQConnector(
const shared_ptr<XItemNodeBase> &node, QWidget *item)
553 m_lsnListChanged = tr[ *node].onListChanged().connectWeakly(shared_from_this(),
554 &XItemQConnector::onListChanged,
555 XListener::FLAG_MAIN_THREAD_CALL | XListener::FLAG_AVOID_DUP);
558 XItemQConnector::~XItemQConnector() {
561 XQComboBoxConnector::XQComboBoxConnector(
const shared_ptr<XItemNodeBase> &node,
562 QComboBox *item,
const Snapshot &shot_of_list)
564 m_node(node), m_pItem(item) {
565 connect(item, SIGNAL( activated(
int) ),
this, SLOT( onSelect(
int) ) );
570 XQComboBoxConnector::onSelect(
int idx) {
573 if( (idx >= m_itemStrings.size()) || (idx < 0))
576 tr[ *m_node].str(m_itemStrings.at(idx).label);
577 tr.unmark(m_lsnValueChanged);
586 XQComboBoxConnector::findItem(
const QString &text) {
587 for(
int i = 0; i < m_pItem->count(); i++) {
588 if(text == m_pItem->itemText(i)) {
597 m_pItem->blockSignals(
true);
598 QString str = shot[ *node].to_str();
601 for(
auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
602 if(QString(it->label) == str) {
608 m_pItem->setCurrentIndex(idx);
609 if(m_node->autoSetAny()) {
610 int idx1 = findItem(i18n(
"(UNSEL)"));
612 m_pItem->removeItem(idx1);
617 if(m_node->autoSetAny()) {
618 int idx = findItem(i18n(
"(UNSEL)"));
620 m_pItem->addItem(i18n(
"(UNSEL)"));
623 int idx = findItem(i18n(
"(UNSEL)"));
625 m_pItem->setCurrentIndex(idx);
627 m_pItem->blockSignals(
false);
631 m_itemStrings = m_node->itemStrings(e.shot_of_list);
634 for(
auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
635 if(it->label.empty()) {
636 m_pItem->addItem(i18n(
"(NO NAME)"));
639 m_pItem->addItem(QString(it->label));
643 if( !m_node->autoSetAny())
644 m_pItem->addItem(i18n(
"(UNSEL)"));
645 onValueChanged(shot, e.emitter);
648 XQListWidgetConnector::XQListWidgetConnector(
const shared_ptr<XItemNodeBase> &node,
649 QListWidget *item,
const Snapshot &shot_of_list)
651 m_node(node), m_pItem(item) {
652 connect(item, SIGNAL( itemSelectionChanged() ),
653 this, SLOT( OnItemSelectionChanged() ) );
654 item->setMovement(QListView::Static);
655 item->setSelectionBehavior(QAbstractItemView::SelectRows);
656 item->setSelectionMode(QAbstractItemView::SingleSelection);
660 XQListWidgetConnector::~XQListWidgetConnector() {
666 XQListWidgetConnector::OnItemSelectionChanged() {
667 int idx = m_pItem->currentRow();
670 if((idx >= m_itemStrings.size()) || (idx < 0))
673 tr[ *m_node].str(m_itemStrings.at(idx).label);
674 tr.unmark(m_lsnValueChanged);
683 QString str = shot[ *node].to_str();
684 m_pItem->blockSignals(
true);
686 for(
auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
687 if(str == QString(it->label))
688 m_pItem->setCurrentRow(i);
691 m_pItem->blockSignals(
false);
695 m_itemStrings = m_node->itemStrings(e.shot_of_list);
697 for(
auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
698 new QListWidgetItem(it->label, m_pItem);
700 onValueChanged(shot, e.emitter);
703 XColorConnector::XColorConnector(
const shared_ptr<XHexNode> &node, QPushButton *item)
705 m_node(node), m_pItem(item) {
706 connect(item, SIGNAL( clicked() ),
this, SLOT( onClick() ) );
707 item->setAutoDefault(
false);
708 item->setDefault(
false);
709 onValueChanged(
Snapshot( *node), node.get());
712 XColorConnector::onClick() {
713 auto dialog = m_dialog;
715 dialog.reset(
new QColorDialog(m_pItem));
718 connect( &*dialog, SIGNAL( colorSelected(
const QColor &) ),
this, SLOT( OnColorSelected(
const QColor &) ) );
720 dialog->setCurrentColor(QColor((QRgb)(
unsigned int)shot[ *m_node]));
724 XColorConnector::OnColorSelected(
const QColor & color) {
726 tr[ *m_node] = color.rgb();
731 QColor color((QRgb)(
unsigned int)shot[ *m_node]);
732 auto dialog = m_dialog;
734 dialog->setCurrentColor(color);
735 QPixmap pixmap(m_pItem->size());
737 m_pItem->setIcon(pixmap);
740 XStatusPrinter::XStatusPrinter(QMainWindow *window) {
741 if( !window) window =
dynamic_cast<QMainWindow*
>(g_pFrmMain);
742 m_pWindow = (window);
743 m_pBar = (window->statusBar());
744 s_statusPrinterCreating.push_back(shared_ptr<XStatusPrinter>(
this));
746 m_lsn = m_tlkTalker.connectWeakly(
747 shared_from_this(), &XStatusPrinter::print,
748 XListener::FLAG_MAIN_THREAD_CALL);
750 XStatusPrinter::~XStatusPrinter() {
752 shared_ptr<XStatusPrinter>
753 XStatusPrinter::create(QMainWindow *window) {
755 shared_ptr<XStatusPrinter> ptr = s_statusPrinterCreating.back();
756 s_statusPrinterCreating.pop_back();
760 XStatusPrinter::printMessage(
const XString &str,
bool popup,
const char *file,
int line,
bool beep) {
764 if(file) status.tooltip = i18n_noncontext(
"Message emitted at %1:%2").arg(file).arg(line);
765 status.popup = popup;
767 status.type = tstatus::Normal;
768 m_tlkTalker.talk(std::move(status));
771 XStatusPrinter::printWarning(
const XString &str,
bool popup,
const char *file,
int line,
bool beep) {
774 status.str =
XString(i18n_noncontext(
"Warning: ")) + str;
775 if(file) status.tooltip = i18n_noncontext(
"Warning emitted at %1:%2").arg(file).arg(line);
776 status.popup = popup;
778 status.type = tstatus::Warning;
779 m_tlkTalker.talk(std::move(status));
782 XStatusPrinter::printError(
const XString &str,
bool popup,
const char *file,
int line,
bool beep) {
785 status.str =
XString(i18n_noncontext(
"Error: ")) + str;
786 if(file) status.tooltip = i18n_noncontext(
"Error emitted at %1:%2").arg(file).arg(line);
787 status.popup = popup;
789 status.type = tstatus::Error;
790 m_tlkTalker.talk(std::move(status));
793 XStatusPrinter::clear(
void) {
797 m_tlkTalker.talk(std::move(status));
801 XStatusPrinter::print(
const tstatus &status) {
802 bool popup = status.popup;
803 QString str = std::move(status.str);
806 m_pBar->showMessage(str, status.ms);
808 QApplication::beep();
812 m_pBar->clearMessage();
815 switch(status.type) {
816 case tstatus::Normal:
820 case tstatus::Warning:
828 XMessageBox::post(str, QIcon( *icon), popup, status.ms, status.tooltip);