xnodeconnector.cpp
1 /***************************************************************************
2  Copyright (C) 2002-2015 Kentaro Kitagawa
3  kitagawa@phys.s.u-tokyo.ac.jp
4 
5  This program is free software; you can redistribute it and/or
6  modify it under the terms of the GNU Library General Public
7  License as published by the Free Software Foundation; either
8  version 2 of the License, or (at your option) any later version.
9 
10  You should have received a copy of the GNU Library General
11  Public License and a list of authors along with this program;
12  see the files COPYING and AUTHORS.
13 ***************************************************************************/
14 #include "xnodeconnector.h"
15 #include <deque>
16 #include <QPushButton>
17 #include <QLineEdit>
18 #include <QCheckBox>
19 #include <QListWidget>
20 #include <QComboBox>
21 #include <QLabel>
22 #include <QTableWidget>
23 #include <QHeaderView>
24 #include <QDoubleSpinBox>
25 #include <QSlider>
26 #include <QLCDNumber>
27 #include <QTextBrowser>
28 #include <QToolTip>
29 #include <QStatusBar>
30 #include <QSlider>
31 #include <QToolButton>
32 #include <QFileDialog>
33 #include <QColorDialog>
34 #include <QPainter>
35 #include <QMainWindow>
36 #include <QApplication>
37 
38 #include <map>
39 #include "measure.h"
40 #include "icons/icon.h"
41 #include "messagebox.h"
42 #include <type_traits>
43 
44 //ms
45 #define UI_DISP_DELAY 10
46 
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;
50 
51 void sharedPtrQDeleter_(QObject *obj) {
52  if(isMainThread())
53  delete obj;
54  else
55  obj->deleteLater();
56 }
57 
58 XQConnectorHolder_::XQConnectorHolder_(XQConnector *con) :
59  QObject(0L) {
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());
64 }
65 XQConnectorHolder_::~XQConnectorHolder_() {
66  if(m_connector)
67  disconnect(m_connector->m_pWidget, SIGNAL( destroyed() ), this, SLOT( destroyed() ) );
68 }
69 bool
70 XQConnectorHolder_::isAlive() const {
71  return !!m_connector;
72 }
73 
74 void
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;
81  m_connector.reset();
82 }
83 
84 XQConnector::XQConnector(const shared_ptr<XNode> &node, QWidget *item)
85  : QObject(),
86  m_pWidget(item) {
87 
88  assert(node);
89  assert(item);
90  s_conCreating.push_back(shared_ptr<XQConnector>(this));
91 
92  node->iterate_commit([=](Transaction &tr){
93  m_lsnUIEnabled = tr[ *node].onUIFlagsChanged().connectWeakly(shared_from_this(), &XQConnector::onUIFlagsChanged,
94  XListener::FLAG_MAIN_THREAD_CALL | XListener::FLAG_AVOID_DUP);
95  });
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)
100  .arg((uintptr_t)sizeof(XQConnector), 0, 16));
101 
102  auto ret = s_widgetMap.insert(std::pair<const QWidget*, weak_ptr<XNode> >(item, node));
103  if( !ret.second)
104  gErrPrint("Connection to Widget Duplicated!");
105 #ifdef HAVE_LIBGCCPP
106  GC_gcollect();
107 #endif
108 }
110  if(isItemAlive()) {
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);
116  }
117  else {
118  dbgPrint(QString("connector %1 & widget released., addr=0x%2").arg(objectName()).arg((uintptr_t)this, 0, 16));
119  }
120 
121 #ifdef HAVE_LIBGCCPP
122  GC_gcollect();
123 #endif
124 }
125 shared_ptr<XNode>
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();
131 }
132 
133 void
134 XQConnector::onUIFlagsChanged(const Snapshot &shot, XNode *node) {
135  m_pWidget->setEnabled(shot[ *node].isUIEnabled());
136 }
137 
138 XQButtonConnector::XQButtonConnector(const shared_ptr<XTouchableNode> &node,
139  QAbstractButton *item)
140  : XQConnector(node, item),
141  m_node(node), m_pItem(item) {
142 
143  connect(item, SIGNAL( clicked() ), this, SLOT( onClick() ) );
144  node->iterate_commit([=](Transaction &tr){
145  m_lsnTouch = tr[ *node].onTouch().connectWeakly
146  (shared_from_this(), &XQButtonConnector::onTouch, XListener::FLAG_MAIN_THREAD_CALL);
147  });
148 }
149 XQButtonConnector::~XQButtonConnector() {
150 }
151 void
152 XQButtonConnector::onClick() {
153  m_node->iterate_commit([=](Transaction &tr){
154  tr[ *m_node].touch();
155  tr.unmark(m_lsnTouch);
156  });
157 }
158 void
159 XQButtonConnector::onTouch(const Snapshot &shot, XTouchableNode *node) {
160 }
161 
162 XValueQConnector::XValueQConnector(const shared_ptr<XValueNodeBase> &node, QWidget *item)
163  : XQConnector(node, item) {
164  node->iterate_commit([=](Transaction &tr){
165  m_lsnValueChanged = tr[ *node].onValueChanged().connectWeakly(
166  shared_from_this(), &XValueQConnector::onValueChanged,
167  XListener::FLAG_MAIN_THREAD_CALL | XListener::FLAG_AVOID_DUP);
168  });
169 }
170 XValueQConnector::~XValueQConnector() {
171 }
172 
173 XQLineEditConnector::XQLineEditConnector(
174  const shared_ptr<XValueNodeBase> &node, QLineEdit *item, bool forcereturn)
175  : XValueQConnector(node, item),
176  m_node(node), m_pItem(item), m_editing(false) {
177  connect(item, SIGNAL( returnPressed() ), this, SLOT( onReturnPressed() ) );
178  if(forcereturn) {
179  connect(item, SIGNAL( editingFinished() ), this, SLOT( onExit() ) );
180  connect(item, SIGNAL( textEdited( const QString &) ),
181  this, SLOT( onTextChanged(const QString &) ) );
182  }
183  else {
184  connect(item, SIGNAL( textEdited( const QString &) ),
185  this, SLOT( onTextChanged2(const QString &) ) );
186  }
187  onValueChanged(Snapshot( *node), node.get());
188 }
189 void
190 XQLineEditConnector::onTextChanged(const QString &text) {
191  QPalette palette(m_pItem->palette());
192  palette.setColor(QPalette::Text, Qt::blue);
193  m_pItem->setPalette(palette);
194  m_editing = true;
195 }
196 void
197 XQLineEditConnector::onTextChanged2(const QString &text) {
198  QPalette palette(m_pItem->palette());
199  try {
200  m_node->iterate_commit([=](Transaction &tr){
201  tr[ *m_node].str(text);
202  tr.unmark(m_lsnValueChanged);
203  });
204  palette.setColor(QPalette::Text, Qt::black);
205  }
206  catch (XKameError &e) {
207  palette.setColor(QPalette::Text, Qt::red);
208  }
209  m_pItem->setPalette(palette);
210 }
211 void
212 XQLineEditConnector::onReturnPressed() {
213  QPalette palette(m_pItem->palette());
214  try {
215  m_node->iterate_commit([=](Transaction &tr){
216  tr[ *m_node].str(m_pItem->text());
217  tr.unmark(m_lsnValueChanged);
218  });
219  palette.setColor(QPalette::Text, Qt::black);
220  }
221  catch (XKameError &e) {
222  e.print();
223  palette.setColor(QPalette::Text, Qt::red);
224  }
225  m_pItem->setPalette(palette);
226  m_editing = false;
227 }
228 void
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);
234  Snapshot shot( *m_node);
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);
241  }
242 }
243 void
244 XQLineEditConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
245  m_pItem->blockSignals(true);
246  m_pItem->setText(shot[ *node].to_str());
247  m_pItem->blockSignals(false);
248 }
249 
250 template <class QN, class XN, class X>
252  QN *item, QSlider *slider)
253  : XValueQConnector(node, item),
254  m_node(node),
255  m_pItem(item),
256  m_pSlider(slider) {
257  if(slider) {
258  if(std::is_integral<X>::value) {
259  slider->setRange(item->minimum(), item->maximum());
260  slider->setSingleStep(item->singleStep());
261  }
262  else {
263  slider->setRange(0, 100);
264  slider->setSingleStep(5);
265  }
266  }
267 }
268 XQSpinBoxConnector::XQSpinBoxConnector(const shared_ptr<XIntNode> &node,
269  QSpinBox *item, QSlider *slider)
270  : XQSpinBoxConnectorTMPL<QSpinBox, XIntNode, int>(node, item, slider) {
271  connect(item, SIGNAL( valueChanged(int) ), this, SLOT( onChange(int) ) );
272  if(slider) {
273  connect(slider, SIGNAL( valueChanged(int) ), this, SLOT( onSliderChange(int) ) );
274  }
275  onValueChanged(Snapshot( *node), node.get());
276 }
277 XQSpinBoxUnsignedConnector::XQSpinBoxUnsignedConnector(const shared_ptr<XUIntNode> &node,
278  QSpinBox *item, QSlider *slider)
279  : XQSpinBoxConnectorTMPL<QSpinBox, XUIntNode, int>(node, item, slider) {
280  connect(item, SIGNAL( valueChanged(int) ), this, SLOT( onChange(int) ) );
281  if(slider) {
282  connect(slider, SIGNAL( valueChanged(int) ), this, SLOT( onSliderChange(int) ) );
283  }
284  onValueChanged(Snapshot( *node), node.get());
285 }
286 XQDoubleSpinBoxConnector::XQDoubleSpinBoxConnector(const shared_ptr<XDoubleNode> &node,
287  QDoubleSpinBox *item, QSlider *slider)
288  : XQSpinBoxConnectorTMPL<QDoubleSpinBox, XDoubleNode, double>(node, item, slider) {
289  connect(item, SIGNAL( valueChanged(double) ), this, SLOT( onChange(double) ) );
290  if(slider) {
291  connect(slider, SIGNAL( valueChanged(int) ), this, SLOT( onSliderChange(int) ) );
292  }
293  onValueChanged(Snapshot( *node), node.get());
294 }
295 template <class QN, class XN, class X>
296 void
298  if(m_pSlider)
299  m_pSlider->setEnabled(shot[ *node].isUIEnabled());
300  XQConnector::onUIFlagsChanged(shot, node);
301 }
302 
303 template <class QN, class XN, class X>
304 void
306  int var = val;
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);
311  }
312  if(m_pSlider) {
313  m_pSlider->blockSignals(true);
314  m_pSlider->setValue(var);
315  m_pSlider->blockSignals(false);
316  }
317  m_node->iterate_commit([=](Transaction &tr){
318  tr[ *m_node] = val;
319  tr.unmark(m_lsnValueChanged);
320  });
321 }
322 template <class QN, class XN, class X>
323 void
325  X var = val;
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;
330  }
331  m_pItem->blockSignals(true);
332  m_pItem->setValue(var);
333  m_pItem->blockSignals(false);
334  m_node->iterate_commit([=](Transaction &tr){
335  tr[ *m_node] = var;
336  tr.unmark(m_lsnValueChanged);
337  });
338 }
339 template <class QN, class XN, class X>
340 void
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);
348  if(m_pSlider) {
349  int svar = var;
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);
354  }
355  m_pSlider->blockSignals(true);
356  m_pSlider->setValue(svar);
357  m_pSlider->blockSignals(false);
358  }
359 }
360 
364 
365 XFilePathConnector::XFilePathConnector(const shared_ptr<XStringNode> &node,
366  QLineEdit *edit, QAbstractButton *btn, const char *filter, bool saving)
367  : XQLineEditConnector(node, edit),
368  m_pBtn(btn), m_filter(filter), m_saving(saving) {
369  connect(btn, SIGNAL( clicked ( ) ), this, SLOT( onClick( ) ) );
370 }
371 void
372 XFilePathConnector::onClick() {
373 #if QT_VERSION < QT_VERSION_CHECK(5,0,0)
374  XString str =
375  m_saving ? QFileDialog::
376  getSaveFileName(m_pItem, QString(), m_pItem->text(), m_filter)
377  : QFileDialog::
378  getOpenFileName(m_pItem, QString(), m_pItem->text(), m_filter);
379 #else
380  //old qt cannot make native dialog in this mode.
381  QFileDialog dialog(m_pItem);
382  dialog.setViewMode(QFileDialog::Detail);
383  dialog.setNameFilter(m_filter);
384 // dialog.setConfirmOverwrite(false);
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);
391  if( !dialog.exec())
392  return;
393  QString str = dialog.selectedFiles().at(0);
394 #endif
395  if(str.length()) {
396  m_pItem->blockSignals(true);
397  m_pItem->setText(str);
398  m_pItem->blockSignals(false);
399  try {
400  m_node->iterate_commit([=](Transaction &tr){
401  tr[ *m_node].str(str);
402  tr.unmark(m_lsnValueChanged);
403  });
404  }
405  catch (XKameError &e) {
406  e.print();
407  }
408  }
409 }
410 
411 void
412 XFilePathConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
413  XQLineEditConnector::onValueChanged(shot, node);
414 }
415 
416 XQLabelConnector::XQLabelConnector(const shared_ptr<XValueNodeBase> &node, QLabel *item)
417  : XValueQConnector(node, item),
418  m_node(node), m_pItem(item) {
419  onValueChanged(Snapshot( *node), node.get());
420 }
421 
422 void
423 XQLabelConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
424  m_pItem->setText(shot[ *node].to_str());
425 }
426 
427 XQTextBrowserConnector::XQTextBrowserConnector(const shared_ptr<XValueNodeBase> &node, QTextBrowser *item)
428  : XValueQConnector(node, item),
429  m_node(node), m_pItem(item) {
430  onValueChanged(Snapshot( *node), node.get());
431 }
432 void
433 XQTextBrowserConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
434  m_pItem->setText(shot[ *node].to_str());
435 }
436 
437 XQLCDNumberConnector::XQLCDNumberConnector(const shared_ptr<XDoubleNode> &node, QLCDNumber *item)
438  : XValueQConnector(node, item),
439  m_node(node), m_pItem(item) {
440  onValueChanged(Snapshot( *node), node.get());
441 }
442 
443 void
444 XQLCDNumberConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
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);
449 }
450 
451 XQLedConnector::XQLedConnector(const shared_ptr<XBoolNode> &node, QPushButton *item)
452  : XValueQConnector(node, item),
453  m_node(node), m_pItem(item) {
454  item->setCheckable(false);
455  item->setAutoDefault(false);
456  item->setFlat(true);
457  item->setFocusPolicy(Qt::NoFocus);
458  item->setIconSize(QSize(16, 16));
459  onValueChanged(Snapshot( *node), node.get());
460 }
461 
462 void
463 XQLedConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
464  m_pItem->setIcon(shot[ *m_node] ?
465  *g_pIconLEDOn : *g_pIconLEDOff);
466 }
467 
468 XQToggleButtonConnector::XQToggleButtonConnector(const shared_ptr<XBoolNode> &node, QAbstractButton *item)
469  : XValueQConnector(node, item),
470  m_node(node), m_pItem(item) {
471  connect(item, SIGNAL( clicked() ), this, SLOT( onClick() ) );
472  onValueChanged(Snapshot( *node), node.get());
473 }
474 
475 void
476 XQToggleButtonConnector::onClick() {
477  m_node->iterate_commit([=](Transaction &tr){
478  tr[ *m_node] = m_pItem->isChecked();
479  tr.unmark(m_lsnValueChanged);
480  });
481 }
482 
483 void
484 XQToggleButtonConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
485  if((shot[ *m_node]) ^ m_pItem->isChecked())
486  m_pItem->toggle();
487 }
488 
489 XListQConnector::XListQConnector(const shared_ptr<XListNodeBase> &node, QTableWidget *item)
490  : XQConnector(node, item),
491  m_pItem(item), m_list(node) {
492 
493  node->iterate_commit([=](Transaction &tr){
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);
500  });
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); //QHeaderView::Fixed
505 #else
506  header->setSectionsMovable(true);
507  header->setSectionResizeMode(QHeaderView::ResizeToContents);
508 #endif
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."));
512 }
513 XListQConnector::~XListQConnector() {
514  if(isItemAlive()) {
515  QHeaderView *header = m_pItem->verticalHeader();
516  disconnect(header, NULL, this, NULL );
517  m_pItem->clearSpans();
518  }
519 }
520 void
521 XListQConnector::OnSectionMoved(int logicalIndex, int oldVisualIndex, int newVisualIndex) {
522  int fromIndex = oldVisualIndex;
523  int toIndex = newVisualIndex;
524  if(toIndex == fromIndex)
525  toIndex++;
526  m_list->iterate_commit([=](Transaction &tr){
527  unsigned int src = fromIndex;
528  unsigned int dst = toIndex;
529  if( !tr.size() || src >= tr.size() || (dst >= tr.size())) {
530 // gErrPrint(i18n("Invalid range of selections."));
531  return;
532  }
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));
536  i = next;
537  }
538  tr.unmark(m_lsnMove);
539  });
540 }
541 void
542 XListQConnector::onMove(const Snapshot &shot, const XListNodeBase::Payload::MoveEvent &e) {
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);
547  }
548 }
549 
550 XItemQConnector::XItemQConnector(const shared_ptr<XItemNodeBase> &node, QWidget *item)
551  : XValueQConnector(node, item) {
552  node->iterate_commit([=](Transaction &tr){
553  m_lsnListChanged = tr[ *node].onListChanged().connectWeakly(shared_from_this(),
554  &XItemQConnector::onListChanged,
555  XListener::FLAG_MAIN_THREAD_CALL | XListener::FLAG_AVOID_DUP);
556  });
557 }
558 XItemQConnector::~XItemQConnector() {
559 }
560 
561 XQComboBoxConnector::XQComboBoxConnector(const shared_ptr<XItemNodeBase> &node,
562  QComboBox *item, const Snapshot &shot_of_list)
563  : XItemQConnector(node, item),
564  m_node(node), m_pItem(item) {
565  connect(item, SIGNAL( activated(int) ), this, SLOT( onSelect(int) ) );
566  XItemNodeBase::Payload::ListChangeEvent e(shot_of_list, node.get());
567  onListChanged(Snapshot( *node), e);
568 }
569 void
570 XQComboBoxConnector::onSelect(int idx) {
571  try {
572  m_node->iterate_commit([=](Transaction &tr){
573  if( (idx >= m_itemStrings.size()) || (idx < 0))
574  tr[ *m_node].str(XString());
575  else
576  tr[ *m_node].str(m_itemStrings.at(idx).label);
577  tr.unmark(m_lsnValueChanged);
578  });
579  }
580  catch (XKameError &e) {
581  e.print();
582  }
583 }
584 
585 int
586 XQComboBoxConnector::findItem(const QString &text) {
587  for(int i = 0; i < m_pItem->count(); i++) {
588  if(text == m_pItem->itemText(i)) {
589  return i;
590  }
591  }
592  return -1;
593 }
594 
595 void
596 XQComboBoxConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
597  m_pItem->blockSignals(true);
598  QString str = shot[ *node].to_str();
599  int idx = -1;
600  int i = 0;
601  for(auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
602  if(QString(it->label) == str) {
603  idx = i;
604  }
605  i++;
606  }
607  if(idx >= 0) {
608  m_pItem->setCurrentIndex(idx);
609  if(m_node->autoSetAny()) {
610  int idx1 = findItem(i18n("(UNSEL)"));
611  if(idx1 >= 0) {
612  m_pItem->removeItem(idx1);
613  }
614  }
615  }
616  else {
617  if(m_node->autoSetAny()) {
618  int idx = findItem(i18n("(UNSEL)"));
619  if(idx < 0) {
620  m_pItem->addItem(i18n("(UNSEL)"));
621  }
622  }
623  int idx = findItem(i18n("(UNSEL)"));
624  assert(idx >= 0);
625  m_pItem->setCurrentIndex(idx);
626  }
627  m_pItem->blockSignals(false);
628 }
629 void
630 XQComboBoxConnector::onListChanged(const Snapshot &shot, const XItemNodeBase::Payload::ListChangeEvent &e) {
631  m_itemStrings = m_node->itemStrings(e.shot_of_list);
632  m_pItem->clear();
633  bool exist = false;
634  for(auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
635  if(it->label.empty()) {
636  m_pItem->addItem(i18n("(NO NAME)"));
637  }
638  else {
639  m_pItem->addItem(QString(it->label));
640  exist = true;
641  }
642  }
643  if( !m_node->autoSetAny())
644  m_pItem->addItem(i18n("(UNSEL)"));
645  onValueChanged(shot, e.emitter);
646 }
647 
648 XQListWidgetConnector::XQListWidgetConnector(const shared_ptr<XItemNodeBase> &node,
649  QListWidget *item, const Snapshot &shot_of_list)
650  : XItemQConnector(node, item),
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);
657  XItemNodeBase::Payload::ListChangeEvent e(shot_of_list, node.get());
658  onListChanged(Snapshot( *node), e);
659 }
660 XQListWidgetConnector::~XQListWidgetConnector() {
661  if(isItemAlive()) {
662  m_pItem->clear();
663  }
664 }
665 void
666 XQListWidgetConnector::OnItemSelectionChanged() {
667  int idx = m_pItem->currentRow();
668  try {
669  m_node->iterate_commit([=](Transaction &tr){
670  if((idx >= m_itemStrings.size()) || (idx < 0))
671  tr[ *m_node].str(XString());
672  else
673  tr[ *m_node].str(m_itemStrings.at(idx).label);
674  tr.unmark(m_lsnValueChanged);
675  });
676  }
677  catch (XKameError &e) {
678  e.print();
679  }
680 }
681 void
682 XQListWidgetConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *node) {
683  QString str = shot[ *node].to_str();
684  m_pItem->blockSignals(true);
685  unsigned int i = 0;
686  for(auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
687  if(str == QString(it->label))
688  m_pItem->setCurrentRow(i);
689  i++;
690  }
691  m_pItem->blockSignals(false);
692 }
693 void
694 XQListWidgetConnector::onListChanged(const Snapshot &shot, const XItemNodeBase::Payload::ListChangeEvent &e) {
695  m_itemStrings = m_node->itemStrings(e.shot_of_list);
696  m_pItem->clear();
697  for(auto it = m_itemStrings.begin(); it != m_itemStrings.end(); it++) {
698  new QListWidgetItem(it->label, m_pItem);
699  }
700  onValueChanged(shot, e.emitter);
701 }
702 
703 XColorConnector::XColorConnector(const shared_ptr<XHexNode> &node, QPushButton *item)
704  : XValueQConnector(node, 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());
710 }
711 void
712 XColorConnector::onClick() {
713  auto dialog = m_dialog;
714  if( !dialog) {
715  dialog.reset(new QColorDialog(m_pItem));
716  m_dialog = dialog;
717  }
718  connect( &*dialog, SIGNAL( colorSelected(const QColor &) ), this, SLOT( OnColorSelected(const QColor &) ) );
719  Snapshot shot( *m_node);
720  dialog->setCurrentColor(QColor((QRgb)(unsigned int)shot[ *m_node]));
721  dialog->open();
722 }
723 void
724 XColorConnector::OnColorSelected(const QColor & color) {
725  m_node->iterate_commit([=](Transaction &tr){
726  tr[ *m_node] = color.rgb();
727  });
728 }
729 void
730 XColorConnector::onValueChanged(const Snapshot &shot, XValueNodeBase *) {
731  QColor color((QRgb)(unsigned int)shot[ *m_node]);
732  auto dialog = m_dialog;
733  if(dialog)
734  dialog->setCurrentColor(color);
735  QPixmap pixmap(m_pItem->size());
736  pixmap.fill(color);
737  m_pItem->setIcon(pixmap);
738 }
739 
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));
745  m_pBar->hide();
746  m_lsn = m_tlkTalker.connectWeakly(
747  shared_from_this(), &XStatusPrinter::print,
748  XListener::FLAG_MAIN_THREAD_CALL);
749 }
750 XStatusPrinter::~XStatusPrinter() {
751 }
752 shared_ptr<XStatusPrinter>
753 XStatusPrinter::create(QMainWindow *window) {
754  new XStatusPrinter(window);
755  shared_ptr<XStatusPrinter> ptr = s_statusPrinterCreating.back();
756  s_statusPrinterCreating.pop_back();
757  return ptr;
758 }
759 void
760 XStatusPrinter::printMessage(const XString &str, bool popup, const char *file, int line, bool beep) {
761  tstatus status;
762  status.ms = 3000;
763  status.str = str;
764  if(file) status.tooltip = i18n_noncontext("Message emitted at %1:%2").arg(file).arg(line);
765  status.popup = popup;
766  status.beep = beep;
767  status.type = tstatus::Normal;
768  m_tlkTalker.talk(std::move(status));
769 }
770 void
771 XStatusPrinter::printWarning(const XString &str, bool popup, const char *file, int line, bool beep) {
772  tstatus status;
773  status.ms = 3000;
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;
777  status.beep = beep;
778  status.type = tstatus::Warning;
779  m_tlkTalker.talk(std::move(status));
780 }
781 void
782 XStatusPrinter::printError(const XString &str, bool popup, const char *file, int line, bool beep) {
783  tstatus status;
784  status.ms = 5000;
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;
788  status.beep = beep;
789  status.type = tstatus::Error;
790  m_tlkTalker.talk(std::move(status));
791 }
792 void
793 XStatusPrinter::clear(void) {
794  tstatus status;
795  status.ms = 0;
796  status.str = "";
797  m_tlkTalker.talk(std::move(status));
798 }
799 
800 void
801 XStatusPrinter::print(const tstatus &status) {
802  bool popup = status.popup;
803  QString str = std::move(status.str);
804  if(status.ms) {
805  m_pBar->show();
806  m_pBar->showMessage(str, status.ms);
807  if(status.beep)
808  QApplication::beep();
809  }
810  else {
811  m_pBar->hide();
812  m_pBar->clearMessage();
813  }
814  QPixmap *icon;
815  switch(status.type) {
816  case tstatus::Normal:
817  default:
818  icon = g_pIconInfo;
819  break;
820  case tstatus::Warning:
821  icon = g_pIconWarn;
822  break;
823  case tstatus::Error:
824  icon = g_pIconError;
825  break;
826  }
827  if(popup)
828  XMessageBox::post(str, QIcon( *icon), popup, status.ms, status.tooltip);
829 }

Generated for KAME4 by  doxygen 1.8.3