Loading...
1// SPDX-License-Identifier: GPL-2.0
2/*
3 * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>
4 * Copyright (C) 2015 Boris Barbulovski <bbarbulovski@gmail.com>
5 */
6
7#include <qglobal.h>
8
9#include <QMainWindow>
10#include <QList>
11#include <qtextbrowser.h>
12#include <QAction>
13#include <QFileDialog>
14#include <QMenu>
15
16#include <qapplication.h>
17#include <qdesktopwidget.h>
18#include <qtoolbar.h>
19#include <qlayout.h>
20#include <qsplitter.h>
21#include <qlineedit.h>
22#include <qlabel.h>
23#include <qpushbutton.h>
24#include <qmenubar.h>
25#include <qmessagebox.h>
26#include <qregexp.h>
27#include <qevent.h>
28
29#include <stdlib.h>
30
31#include "lkc.h"
32#include "qconf.h"
33
34#include "qconf.moc"
35#include "images.h"
36
37
38static QApplication *configApp;
39static ConfigSettings *configSettings;
40
41QAction *ConfigMainWindow::saveAction;
42
43static inline QString qgettext(const char* str)
44{
45 return QString::fromLocal8Bit(str);
46}
47
48ConfigSettings::ConfigSettings()
49 : QSettings("kernel.org", "qconf")
50{
51}
52
53/**
54 * Reads a list of integer values from the application settings.
55 */
56QList<int> ConfigSettings::readSizes(const QString& key, bool *ok)
57{
58 QList<int> result;
59
60 if (contains(key))
61 {
62 QStringList entryList = value(key).toStringList();
63 QStringList::Iterator it;
64
65 for (it = entryList.begin(); it != entryList.end(); ++it)
66 result.push_back((*it).toInt());
67
68 *ok = true;
69 }
70 else
71 *ok = false;
72
73 return result;
74}
75
76/**
77 * Writes a list of integer values to the application settings.
78 */
79bool ConfigSettings::writeSizes(const QString& key, const QList<int>& value)
80{
81 QStringList stringList;
82 QList<int>::ConstIterator it;
83
84 for (it = value.begin(); it != value.end(); ++it)
85 stringList.push_back(QString::number(*it));
86 setValue(key, stringList);
87
88 return true;
89}
90
91
92/*
93 * set the new data
94 * TODO check the value
95 */
96void ConfigItem::okRename(int col)
97{
98}
99
100/*
101 * update the displayed of a menu entry
102 */
103void ConfigItem::updateMenu(void)
104{
105 ConfigList* list;
106 struct symbol* sym;
107 struct property *prop;
108 QString prompt;
109 int type;
110 tristate expr;
111
112 list = listView();
113 if (goParent) {
114 setPixmap(promptColIdx, list->menuBackPix);
115 prompt = "..";
116 goto set_prompt;
117 }
118
119 sym = menu->sym;
120 prop = menu->prompt;
121 prompt = qgettext(menu_get_prompt(menu));
122
123 if (prop) switch (prop->type) {
124 case P_MENU:
125 if (list->mode == singleMode || list->mode == symbolMode) {
126 /* a menuconfig entry is displayed differently
127 * depending whether it's at the view root or a child.
128 */
129 if (sym && list->rootEntry == menu)
130 break;
131 setPixmap(promptColIdx, list->menuPix);
132 } else {
133 if (sym)
134 break;
135 setPixmap(promptColIdx, QIcon());
136 }
137 goto set_prompt;
138 case P_COMMENT:
139 setPixmap(promptColIdx, QIcon());
140 goto set_prompt;
141 default:
142 ;
143 }
144 if (!sym)
145 goto set_prompt;
146
147 setText(nameColIdx, QString::fromLocal8Bit(sym->name));
148
149 type = sym_get_type(sym);
150 switch (type) {
151 case S_BOOLEAN:
152 case S_TRISTATE:
153 char ch;
154
155 if (!sym_is_changeable(sym) && list->optMode == normalOpt) {
156 setPixmap(promptColIdx, QIcon());
157 setText(noColIdx, QString::null);
158 setText(modColIdx, QString::null);
159 setText(yesColIdx, QString::null);
160 break;
161 }
162 expr = sym_get_tristate_value(sym);
163 switch (expr) {
164 case yes:
165 if (sym_is_choice_value(sym) && type == S_BOOLEAN)
166 setPixmap(promptColIdx, list->choiceYesPix);
167 else
168 setPixmap(promptColIdx, list->symbolYesPix);
169 setText(yesColIdx, "Y");
170 ch = 'Y';
171 break;
172 case mod:
173 setPixmap(promptColIdx, list->symbolModPix);
174 setText(modColIdx, "M");
175 ch = 'M';
176 break;
177 default:
178 if (sym_is_choice_value(sym) && type == S_BOOLEAN)
179 setPixmap(promptColIdx, list->choiceNoPix);
180 else
181 setPixmap(promptColIdx, list->symbolNoPix);
182 setText(noColIdx, "N");
183 ch = 'N';
184 break;
185 }
186 if (expr != no)
187 setText(noColIdx, sym_tristate_within_range(sym, no) ? "_" : 0);
188 if (expr != mod)
189 setText(modColIdx, sym_tristate_within_range(sym, mod) ? "_" : 0);
190 if (expr != yes)
191 setText(yesColIdx, sym_tristate_within_range(sym, yes) ? "_" : 0);
192
193 setText(dataColIdx, QChar(ch));
194 break;
195 case S_INT:
196 case S_HEX:
197 case S_STRING:
198 const char* data;
199
200 data = sym_get_string_value(sym);
201
202 setText(dataColIdx, data);
203 if (type == S_STRING)
204 prompt = QString("%1: %2").arg(prompt).arg(data);
205 else
206 prompt = QString("(%2) %1").arg(prompt).arg(data);
207 break;
208 }
209 if (!sym_has_value(sym) && visible)
210 prompt += " (NEW)";
211set_prompt:
212 setText(promptColIdx, prompt);
213}
214
215void ConfigItem::testUpdateMenu(bool v)
216{
217 ConfigItem* i;
218
219 visible = v;
220 if (!menu)
221 return;
222
223 sym_calc_value(menu->sym);
224 if (menu->flags & MENU_CHANGED) {
225 /* the menu entry changed, so update all list items */
226 menu->flags &= ~MENU_CHANGED;
227 for (i = (ConfigItem*)menu->data; i; i = i->nextItem)
228 i->updateMenu();
229 } else if (listView()->updateAll)
230 updateMenu();
231}
232
233
234/*
235 * construct a menu entry
236 */
237void ConfigItem::init(void)
238{
239 if (menu) {
240 ConfigList* list = listView();
241 nextItem = (ConfigItem*)menu->data;
242 menu->data = this;
243
244 if (list->mode != fullMode)
245 setExpanded(true);
246 sym_calc_value(menu->sym);
247 }
248 updateMenu();
249}
250
251/*
252 * destruct a menu entry
253 */
254ConfigItem::~ConfigItem(void)
255{
256 if (menu) {
257 ConfigItem** ip = (ConfigItem**)&menu->data;
258 for (; *ip; ip = &(*ip)->nextItem) {
259 if (*ip == this) {
260 *ip = nextItem;
261 break;
262 }
263 }
264 }
265}
266
267ConfigLineEdit::ConfigLineEdit(ConfigView* parent)
268 : Parent(parent)
269{
270 connect(this, SIGNAL(editingFinished()), SLOT(hide()));
271}
272
273void ConfigLineEdit::show(ConfigItem* i)
274{
275 item = i;
276 if (sym_get_string_value(item->menu->sym))
277 setText(QString::fromLocal8Bit(sym_get_string_value(item->menu->sym)));
278 else
279 setText(QString::null);
280 Parent::show();
281 setFocus();
282}
283
284void ConfigLineEdit::keyPressEvent(QKeyEvent* e)
285{
286 switch (e->key()) {
287 case Qt::Key_Escape:
288 break;
289 case Qt::Key_Return:
290 case Qt::Key_Enter:
291 sym_set_string_value(item->menu->sym, text().toLatin1());
292 parent()->updateList(item);
293 break;
294 default:
295 Parent::keyPressEvent(e);
296 return;
297 }
298 e->accept();
299 parent()->list->setFocus();
300 hide();
301}
302
303ConfigList::ConfigList(ConfigView* p, const char *name)
304 : Parent(p),
305 updateAll(false),
306 symbolYesPix(xpm_symbol_yes), symbolModPix(xpm_symbol_mod), symbolNoPix(xpm_symbol_no),
307 choiceYesPix(xpm_choice_yes), choiceNoPix(xpm_choice_no),
308 menuPix(xpm_menu), menuInvPix(xpm_menu_inv), menuBackPix(xpm_menuback), voidPix(xpm_void),
309 showName(false), showRange(false), showData(false), mode(singleMode), optMode(normalOpt),
310 rootEntry(0), headerPopup(0)
311{
312 int i;
313
314 setObjectName(name);
315 setSortingEnabled(false);
316 setRootIsDecorated(true);
317
318 setVerticalScrollMode(ScrollPerPixel);
319 setHorizontalScrollMode(ScrollPerPixel);
320
321 setHeaderLabels(QStringList() << "Option" << "Name" << "N" << "M" << "Y" << "Value");
322
323 connect(this, SIGNAL(itemSelectionChanged(void)),
324 SLOT(updateSelection(void)));
325
326 if (name) {
327 configSettings->beginGroup(name);
328 showName = configSettings->value("/showName", false).toBool();
329 showRange = configSettings->value("/showRange", false).toBool();
330 showData = configSettings->value("/showData", false).toBool();
331 optMode = (enum optionMode)configSettings->value("/optionMode", 0).toInt();
332 configSettings->endGroup();
333 connect(configApp, SIGNAL(aboutToQuit()), SLOT(saveSettings()));
334 }
335
336 addColumn(promptColIdx);
337
338 reinit();
339}
340
341bool ConfigList::menuSkip(struct menu *menu)
342{
343 if (optMode == normalOpt && menu_is_visible(menu))
344 return false;
345 if (optMode == promptOpt && menu_has_prompt(menu))
346 return false;
347 if (optMode == allOpt)
348 return false;
349 return true;
350}
351
352void ConfigList::reinit(void)
353{
354 removeColumn(dataColIdx);
355 removeColumn(yesColIdx);
356 removeColumn(modColIdx);
357 removeColumn(noColIdx);
358 removeColumn(nameColIdx);
359
360 if (showName)
361 addColumn(nameColIdx);
362 if (showRange) {
363 addColumn(noColIdx);
364 addColumn(modColIdx);
365 addColumn(yesColIdx);
366 }
367 if (showData)
368 addColumn(dataColIdx);
369
370 updateListAll();
371}
372
373void ConfigList::saveSettings(void)
374{
375 if (!objectName().isEmpty()) {
376 configSettings->beginGroup(objectName());
377 configSettings->setValue("/showName", showName);
378 configSettings->setValue("/showRange", showRange);
379 configSettings->setValue("/showData", showData);
380 configSettings->setValue("/optionMode", (int)optMode);
381 configSettings->endGroup();
382 }
383}
384
385ConfigItem* ConfigList::findConfigItem(struct menu *menu)
386{
387 ConfigItem* item = (ConfigItem*)menu->data;
388
389 for (; item; item = item->nextItem) {
390 if (this == item->listView())
391 break;
392 }
393
394 return item;
395}
396
397void ConfigList::updateSelection(void)
398{
399 struct menu *menu;
400 enum prop_type type;
401
402 if (selectedItems().count() == 0)
403 return;
404
405 ConfigItem* item = (ConfigItem*)selectedItems().first();
406 if (!item)
407 return;
408
409 menu = item->menu;
410 emit menuChanged(menu);
411 if (!menu)
412 return;
413 type = menu->prompt ? menu->prompt->type : P_UNKNOWN;
414 if (mode == menuMode && type == P_MENU)
415 emit menuSelected(menu);
416}
417
418void ConfigList::updateList(ConfigItem* item)
419{
420 ConfigItem* last = 0;
421
422 if (!rootEntry) {
423 if (mode != listMode)
424 goto update;
425 QTreeWidgetItemIterator it(this);
426 ConfigItem* item;
427
428 while (*it) {
429 item = (ConfigItem*)(*it);
430 if (!item->menu)
431 continue;
432 item->testUpdateMenu(menu_is_visible(item->menu));
433
434 ++it;
435 }
436 return;
437 }
438
439 if (rootEntry != &rootmenu && (mode == singleMode ||
440 (mode == symbolMode && rootEntry->parent != &rootmenu))) {
441 item = (ConfigItem *)topLevelItem(0);
442 if (!item)
443 item = new ConfigItem(this, 0, true);
444 last = item;
445 }
446 if ((mode == singleMode || (mode == symbolMode && !(rootEntry->flags & MENU_ROOT))) &&
447 rootEntry->sym && rootEntry->prompt) {
448 item = last ? last->nextSibling() : firstChild();
449 if (!item)
450 item = new ConfigItem(this, last, rootEntry, true);
451 else
452 item->testUpdateMenu(true);
453
454 updateMenuList(item, rootEntry);
455 update();
456 resizeColumnToContents(0);
457 return;
458 }
459update:
460 updateMenuList(this, rootEntry);
461 update();
462 resizeColumnToContents(0);
463}
464
465void ConfigList::setValue(ConfigItem* item, tristate val)
466{
467 struct symbol* sym;
468 int type;
469 tristate oldval;
470
471 sym = item->menu ? item->menu->sym : 0;
472 if (!sym)
473 return;
474
475 type = sym_get_type(sym);
476 switch (type) {
477 case S_BOOLEAN:
478 case S_TRISTATE:
479 oldval = sym_get_tristate_value(sym);
480
481 if (!sym_set_tristate_value(sym, val))
482 return;
483 if (oldval == no && item->menu->list)
484 item->setExpanded(true);
485 parent()->updateList(item);
486 break;
487 }
488}
489
490void ConfigList::changeValue(ConfigItem* item)
491{
492 struct symbol* sym;
493 struct menu* menu;
494 int type, oldexpr, newexpr;
495
496 menu = item->menu;
497 if (!menu)
498 return;
499 sym = menu->sym;
500 if (!sym) {
501 if (item->menu->list)
502 item->setExpanded(!item->isExpanded());
503 return;
504 }
505
506 type = sym_get_type(sym);
507 switch (type) {
508 case S_BOOLEAN:
509 case S_TRISTATE:
510 oldexpr = sym_get_tristate_value(sym);
511 newexpr = sym_toggle_tristate_value(sym);
512 if (item->menu->list) {
513 if (oldexpr == newexpr)
514 item->setExpanded(!item->isExpanded());
515 else if (oldexpr == no)
516 item->setExpanded(true);
517 }
518 if (oldexpr != newexpr)
519 parent()->updateList(item);
520 break;
521 case S_INT:
522 case S_HEX:
523 case S_STRING:
524 parent()->lineEdit->show(item);
525 break;
526 }
527}
528
529void ConfigList::setRootMenu(struct menu *menu)
530{
531 enum prop_type type;
532
533 if (rootEntry == menu)
534 return;
535 type = menu && menu->prompt ? menu->prompt->type : P_UNKNOWN;
536 if (type != P_MENU)
537 return;
538 updateMenuList(this, 0);
539 rootEntry = menu;
540 updateListAll();
541 if (currentItem()) {
542 currentItem()->setSelected(hasFocus());
543 scrollToItem(currentItem());
544 }
545}
546
547void ConfigList::setParentMenu(void)
548{
549 ConfigItem* item;
550 struct menu *oldroot;
551
552 oldroot = rootEntry;
553 if (rootEntry == &rootmenu)
554 return;
555 setRootMenu(menu_get_parent_menu(rootEntry->parent));
556
557 QTreeWidgetItemIterator it(this);
558 while (*it) {
559 item = (ConfigItem *)(*it);
560 if (item->menu == oldroot) {
561 setCurrentItem(item);
562 scrollToItem(item);
563 break;
564 }
565
566 ++it;
567 }
568}
569
570/*
571 * update all the children of a menu entry
572 * removes/adds the entries from the parent widget as necessary
573 *
574 * parent: either the menu list widget or a menu entry widget
575 * menu: entry to be updated
576 */
577void ConfigList::updateMenuList(ConfigItem *parent, struct menu* menu)
578{
579 struct menu* child;
580 ConfigItem* item;
581 ConfigItem* last;
582 bool visible;
583 enum prop_type type;
584
585 if (!menu) {
586 while (parent->childCount() > 0)
587 {
588 delete parent->takeChild(0);
589 }
590
591 return;
592 }
593
594 last = parent->firstChild();
595 if (last && !last->goParent)
596 last = 0;
597 for (child = menu->list; child; child = child->next) {
598 item = last ? last->nextSibling() : parent->firstChild();
599 type = child->prompt ? child->prompt->type : P_UNKNOWN;
600
601 switch (mode) {
602 case menuMode:
603 if (!(child->flags & MENU_ROOT))
604 goto hide;
605 break;
606 case symbolMode:
607 if (child->flags & MENU_ROOT)
608 goto hide;
609 break;
610 default:
611 break;
612 }
613
614 visible = menu_is_visible(child);
615 if (!menuSkip(child)) {
616 if (!child->sym && !child->list && !child->prompt)
617 continue;
618 if (!item || item->menu != child)
619 item = new ConfigItem(parent, last, child, visible);
620 else
621 item->testUpdateMenu(visible);
622
623 if (mode == fullMode || mode == menuMode || type != P_MENU)
624 updateMenuList(item, child);
625 else
626 updateMenuList(item, 0);
627 last = item;
628 continue;
629 }
630 hide:
631 if (item && item->menu == child) {
632 last = parent->firstChild();
633 if (last == item)
634 last = 0;
635 else while (last->nextSibling() != item)
636 last = last->nextSibling();
637 delete item;
638 }
639 }
640}
641
642void ConfigList::updateMenuList(ConfigList *parent, struct menu* menu)
643{
644 struct menu* child;
645 ConfigItem* item;
646 ConfigItem* last;
647 bool visible;
648 enum prop_type type;
649
650 if (!menu) {
651 while (parent->topLevelItemCount() > 0)
652 {
653 delete parent->takeTopLevelItem(0);
654 }
655
656 return;
657 }
658
659 last = (ConfigItem*)parent->topLevelItem(0);
660 if (last && !last->goParent)
661 last = 0;
662 for (child = menu->list; child; child = child->next) {
663 item = last ? last->nextSibling() : (ConfigItem*)parent->topLevelItem(0);
664 type = child->prompt ? child->prompt->type : P_UNKNOWN;
665
666 switch (mode) {
667 case menuMode:
668 if (!(child->flags & MENU_ROOT))
669 goto hide;
670 break;
671 case symbolMode:
672 if (child->flags & MENU_ROOT)
673 goto hide;
674 break;
675 default:
676 break;
677 }
678
679 visible = menu_is_visible(child);
680 if (!menuSkip(child)) {
681 if (!child->sym && !child->list && !child->prompt)
682 continue;
683 if (!item || item->menu != child)
684 item = new ConfigItem(parent, last, child, visible);
685 else
686 item->testUpdateMenu(visible);
687
688 if (mode == fullMode || mode == menuMode || type != P_MENU)
689 updateMenuList(item, child);
690 else
691 updateMenuList(item, 0);
692 last = item;
693 continue;
694 }
695 hide:
696 if (item && item->menu == child) {
697 last = (ConfigItem*)parent->topLevelItem(0);
698 if (last == item)
699 last = 0;
700 else while (last->nextSibling() != item)
701 last = last->nextSibling();
702 delete item;
703 }
704 }
705}
706
707void ConfigList::keyPressEvent(QKeyEvent* ev)
708{
709 QTreeWidgetItem* i = currentItem();
710 ConfigItem* item;
711 struct menu *menu;
712 enum prop_type type;
713
714 if (ev->key() == Qt::Key_Escape && mode != fullMode && mode != listMode) {
715 emit parentSelected();
716 ev->accept();
717 return;
718 }
719
720 if (!i) {
721 Parent::keyPressEvent(ev);
722 return;
723 }
724 item = (ConfigItem*)i;
725
726 switch (ev->key()) {
727 case Qt::Key_Return:
728 case Qt::Key_Enter:
729 if (item->goParent) {
730 emit parentSelected();
731 break;
732 }
733 menu = item->menu;
734 if (!menu)
735 break;
736 type = menu->prompt ? menu->prompt->type : P_UNKNOWN;
737 if (type == P_MENU && rootEntry != menu &&
738 mode != fullMode && mode != menuMode) {
739 emit menuSelected(menu);
740 break;
741 }
742 case Qt::Key_Space:
743 changeValue(item);
744 break;
745 case Qt::Key_N:
746 setValue(item, no);
747 break;
748 case Qt::Key_M:
749 setValue(item, mod);
750 break;
751 case Qt::Key_Y:
752 setValue(item, yes);
753 break;
754 default:
755 Parent::keyPressEvent(ev);
756 return;
757 }
758 ev->accept();
759}
760
761void ConfigList::mousePressEvent(QMouseEvent* e)
762{
763 //QPoint p(contentsToViewport(e->pos()));
764 //printf("contentsMousePressEvent: %d,%d\n", p.x(), p.y());
765 Parent::mousePressEvent(e);
766}
767
768void ConfigList::mouseReleaseEvent(QMouseEvent* e)
769{
770 QPoint p = e->pos();
771 ConfigItem* item = (ConfigItem*)itemAt(p);
772 struct menu *menu;
773 enum prop_type ptype;
774 QIcon icon;
775 int idx, x;
776
777 if (!item)
778 goto skip;
779
780 menu = item->menu;
781 x = header()->offset() + p.x();
782 idx = header()->logicalIndexAt(x);
783 switch (idx) {
784 case promptColIdx:
785 icon = item->pixmap(promptColIdx);
786 if (!icon.isNull()) {
787 int off = header()->sectionPosition(0) + visualRect(indexAt(p)).x() + 4; // 4 is Hardcoded image offset. There might be a way to do it properly.
788 if (x >= off && x < off + icon.availableSizes().first().width()) {
789 if (item->goParent) {
790 emit parentSelected();
791 break;
792 } else if (!menu)
793 break;
794 ptype = menu->prompt ? menu->prompt->type : P_UNKNOWN;
795 if (ptype == P_MENU && rootEntry != menu &&
796 mode != fullMode && mode != menuMode)
797 emit menuSelected(menu);
798 else
799 changeValue(item);
800 }
801 }
802 break;
803 case noColIdx:
804 setValue(item, no);
805 break;
806 case modColIdx:
807 setValue(item, mod);
808 break;
809 case yesColIdx:
810 setValue(item, yes);
811 break;
812 case dataColIdx:
813 changeValue(item);
814 break;
815 }
816
817skip:
818 //printf("contentsMouseReleaseEvent: %d,%d\n", p.x(), p.y());
819 Parent::mouseReleaseEvent(e);
820}
821
822void ConfigList::mouseMoveEvent(QMouseEvent* e)
823{
824 //QPoint p(contentsToViewport(e->pos()));
825 //printf("contentsMouseMoveEvent: %d,%d\n", p.x(), p.y());
826 Parent::mouseMoveEvent(e);
827}
828
829void ConfigList::mouseDoubleClickEvent(QMouseEvent* e)
830{
831 QPoint p = e->pos(); // TODO: Check if this works(was contentsToViewport).
832 ConfigItem* item = (ConfigItem*)itemAt(p);
833 struct menu *menu;
834 enum prop_type ptype;
835
836 if (!item)
837 goto skip;
838 if (item->goParent) {
839 emit parentSelected();
840 goto skip;
841 }
842 menu = item->menu;
843 if (!menu)
844 goto skip;
845 ptype = menu->prompt ? menu->prompt->type : P_UNKNOWN;
846 if (ptype == P_MENU && (mode == singleMode || mode == symbolMode))
847 emit menuSelected(menu);
848 else if (menu->sym)
849 changeValue(item);
850
851skip:
852 //printf("contentsMouseDoubleClickEvent: %d,%d\n", p.x(), p.y());
853 Parent::mouseDoubleClickEvent(e);
854}
855
856void ConfigList::focusInEvent(QFocusEvent *e)
857{
858 struct menu *menu = NULL;
859
860 Parent::focusInEvent(e);
861
862 ConfigItem* item = (ConfigItem *)currentItem();
863 if (item) {
864 item->setSelected(true);
865 menu = item->menu;
866 }
867 emit gotFocus(menu);
868}
869
870void ConfigList::contextMenuEvent(QContextMenuEvent *e)
871{
872 if (e->y() <= header()->geometry().bottom()) {
873 if (!headerPopup) {
874 QAction *action;
875
876 headerPopup = new QMenu(this);
877 action = new QAction("Show Name", this);
878 action->setCheckable(true);
879 connect(action, SIGNAL(toggled(bool)),
880 parent(), SLOT(setShowName(bool)));
881 connect(parent(), SIGNAL(showNameChanged(bool)),
882 action, SLOT(setOn(bool)));
883 action->setChecked(showName);
884 headerPopup->addAction(action);
885 action = new QAction("Show Range", this);
886 action->setCheckable(true);
887 connect(action, SIGNAL(toggled(bool)),
888 parent(), SLOT(setShowRange(bool)));
889 connect(parent(), SIGNAL(showRangeChanged(bool)),
890 action, SLOT(setOn(bool)));
891 action->setChecked(showRange);
892 headerPopup->addAction(action);
893 action = new QAction("Show Data", this);
894 action->setCheckable(true);
895 connect(action, SIGNAL(toggled(bool)),
896 parent(), SLOT(setShowData(bool)));
897 connect(parent(), SIGNAL(showDataChanged(bool)),
898 action, SLOT(setOn(bool)));
899 action->setChecked(showData);
900 headerPopup->addAction(action);
901 }
902 headerPopup->exec(e->globalPos());
903 e->accept();
904 } else
905 e->ignore();
906}
907
908ConfigView*ConfigView::viewList;
909QAction *ConfigView::showNormalAction;
910QAction *ConfigView::showAllAction;
911QAction *ConfigView::showPromptAction;
912
913ConfigView::ConfigView(QWidget* parent, const char *name)
914 : Parent(parent)
915{
916 setObjectName(name);
917 QVBoxLayout *verticalLayout = new QVBoxLayout(this);
918 verticalLayout->setContentsMargins(0, 0, 0, 0);
919
920 list = new ConfigList(this);
921 verticalLayout->addWidget(list);
922 lineEdit = new ConfigLineEdit(this);
923 lineEdit->hide();
924 verticalLayout->addWidget(lineEdit);
925
926 this->nextView = viewList;
927 viewList = this;
928}
929
930ConfigView::~ConfigView(void)
931{
932 ConfigView** vp;
933
934 for (vp = &viewList; *vp; vp = &(*vp)->nextView) {
935 if (*vp == this) {
936 *vp = nextView;
937 break;
938 }
939 }
940}
941
942void ConfigView::setOptionMode(QAction *act)
943{
944 if (act == showNormalAction)
945 list->optMode = normalOpt;
946 else if (act == showAllAction)
947 list->optMode = allOpt;
948 else
949 list->optMode = promptOpt;
950
951 list->updateListAll();
952}
953
954void ConfigView::setShowName(bool b)
955{
956 if (list->showName != b) {
957 list->showName = b;
958 list->reinit();
959 emit showNameChanged(b);
960 }
961}
962
963void ConfigView::setShowRange(bool b)
964{
965 if (list->showRange != b) {
966 list->showRange = b;
967 list->reinit();
968 emit showRangeChanged(b);
969 }
970}
971
972void ConfigView::setShowData(bool b)
973{
974 if (list->showData != b) {
975 list->showData = b;
976 list->reinit();
977 emit showDataChanged(b);
978 }
979}
980
981void ConfigList::setAllOpen(bool open)
982{
983 QTreeWidgetItemIterator it(this);
984
985 while (*it) {
986 (*it)->setExpanded(open);
987
988 ++it;
989 }
990}
991
992void ConfigView::updateList(ConfigItem* item)
993{
994 ConfigView* v;
995
996 for (v = viewList; v; v = v->nextView)
997 v->list->updateList(item);
998}
999
1000void ConfigView::updateListAll(void)
1001{
1002 ConfigView* v;
1003
1004 for (v = viewList; v; v = v->nextView)
1005 v->list->updateListAll();
1006}
1007
1008ConfigInfoView::ConfigInfoView(QWidget* parent, const char *name)
1009 : Parent(parent), sym(0), _menu(0)
1010{
1011 setObjectName(name);
1012
1013
1014 if (!objectName().isEmpty()) {
1015 configSettings->beginGroup(objectName());
1016 setShowDebug(configSettings->value("/showDebug", false).toBool());
1017 configSettings->endGroup();
1018 connect(configApp, SIGNAL(aboutToQuit()), SLOT(saveSettings()));
1019 }
1020}
1021
1022void ConfigInfoView::saveSettings(void)
1023{
1024 if (!objectName().isEmpty()) {
1025 configSettings->beginGroup(objectName());
1026 configSettings->setValue("/showDebug", showDebug());
1027 configSettings->endGroup();
1028 }
1029}
1030
1031void ConfigInfoView::setShowDebug(bool b)
1032{
1033 if (_showDebug != b) {
1034 _showDebug = b;
1035 if (_menu)
1036 menuInfo();
1037 else if (sym)
1038 symbolInfo();
1039 emit showDebugChanged(b);
1040 }
1041}
1042
1043void ConfigInfoView::setInfo(struct menu *m)
1044{
1045 if (_menu == m)
1046 return;
1047 _menu = m;
1048 sym = NULL;
1049 if (!_menu)
1050 clear();
1051 else
1052 menuInfo();
1053}
1054
1055void ConfigInfoView::symbolInfo(void)
1056{
1057 QString str;
1058
1059 str += "<big>Symbol: <b>";
1060 str += print_filter(sym->name);
1061 str += "</b></big><br><br>value: ";
1062 str += print_filter(sym_get_string_value(sym));
1063 str += "<br>visibility: ";
1064 str += sym->visible == yes ? "y" : sym->visible == mod ? "m" : "n";
1065 str += "<br>";
1066 str += debug_info(sym);
1067
1068 setText(str);
1069}
1070
1071void ConfigInfoView::menuInfo(void)
1072{
1073 struct symbol* sym;
1074 QString head, debug, help;
1075
1076 sym = _menu->sym;
1077 if (sym) {
1078 if (_menu->prompt) {
1079 head += "<big><b>";
1080 head += print_filter(_menu->prompt->text);
1081 head += "</b></big>";
1082 if (sym->name) {
1083 head += " (";
1084 if (showDebug())
1085 head += QString().sprintf("<a href=\"s%p\">", sym);
1086 head += print_filter(sym->name);
1087 if (showDebug())
1088 head += "</a>";
1089 head += ")";
1090 }
1091 } else if (sym->name) {
1092 head += "<big><b>";
1093 if (showDebug())
1094 head += QString().sprintf("<a href=\"s%p\">", sym);
1095 head += print_filter(sym->name);
1096 if (showDebug())
1097 head += "</a>";
1098 head += "</b></big>";
1099 }
1100 head += "<br><br>";
1101
1102 if (showDebug())
1103 debug = debug_info(sym);
1104
1105 struct gstr help_gstr = str_new();
1106 menu_get_ext_help(_menu, &help_gstr);
1107 help = print_filter(str_get(&help_gstr));
1108 str_free(&help_gstr);
1109 } else if (_menu->prompt) {
1110 head += "<big><b>";
1111 head += print_filter(_menu->prompt->text);
1112 head += "</b></big><br><br>";
1113 if (showDebug()) {
1114 if (_menu->prompt->visible.expr) {
1115 debug += " dep: ";
1116 expr_print(_menu->prompt->visible.expr, expr_print_help, &debug, E_NONE);
1117 debug += "<br><br>";
1118 }
1119 }
1120 }
1121 if (showDebug())
1122 debug += QString().sprintf("defined at %s:%d<br><br>", _menu->file->name, _menu->lineno);
1123
1124 setText(head + debug + help);
1125}
1126
1127QString ConfigInfoView::debug_info(struct symbol *sym)
1128{
1129 QString debug;
1130
1131 debug += "type: ";
1132 debug += print_filter(sym_type_name(sym->type));
1133 if (sym_is_choice(sym))
1134 debug += " (choice)";
1135 debug += "<br>";
1136 if (sym->rev_dep.expr) {
1137 debug += "reverse dep: ";
1138 expr_print(sym->rev_dep.expr, expr_print_help, &debug, E_NONE);
1139 debug += "<br>";
1140 }
1141 for (struct property *prop = sym->prop; prop; prop = prop->next) {
1142 switch (prop->type) {
1143 case P_PROMPT:
1144 case P_MENU:
1145 debug += QString().sprintf("prompt: <a href=\"m%p\">", prop->menu);
1146 debug += print_filter(prop->text);
1147 debug += "</a><br>";
1148 break;
1149 case P_DEFAULT:
1150 case P_SELECT:
1151 case P_RANGE:
1152 debug += prop_get_type_name(prop->type);
1153 debug += ": ";
1154 expr_print(prop->expr, expr_print_help, &debug, E_NONE);
1155 debug += "<br>";
1156 break;
1157 case P_CHOICE:
1158 if (sym_is_choice(sym)) {
1159 debug += "choice: ";
1160 expr_print(prop->expr, expr_print_help, &debug, E_NONE);
1161 debug += "<br>";
1162 }
1163 break;
1164 default:
1165 debug += "unknown property: ";
1166 debug += prop_get_type_name(prop->type);
1167 debug += "<br>";
1168 }
1169 if (prop->visible.expr) {
1170 debug += " dep: ";
1171 expr_print(prop->visible.expr, expr_print_help, &debug, E_NONE);
1172 debug += "<br>";
1173 }
1174 }
1175 debug += "<br>";
1176
1177 return debug;
1178}
1179
1180QString ConfigInfoView::print_filter(const QString &str)
1181{
1182 QRegExp re("[<>&\"\\n]");
1183 QString res = str;
1184 for (int i = 0; (i = res.indexOf(re, i)) >= 0;) {
1185 switch (res[i].toLatin1()) {
1186 case '<':
1187 res.replace(i, 1, "<");
1188 i += 4;
1189 break;
1190 case '>':
1191 res.replace(i, 1, ">");
1192 i += 4;
1193 break;
1194 case '&':
1195 res.replace(i, 1, "&");
1196 i += 5;
1197 break;
1198 case '"':
1199 res.replace(i, 1, """);
1200 i += 6;
1201 break;
1202 case '\n':
1203 res.replace(i, 1, "<br>");
1204 i += 4;
1205 break;
1206 }
1207 }
1208 return res;
1209}
1210
1211void ConfigInfoView::expr_print_help(void *data, struct symbol *sym, const char *str)
1212{
1213 QString* text = reinterpret_cast<QString*>(data);
1214 QString str2 = print_filter(str);
1215
1216 if (sym && sym->name && !(sym->flags & SYMBOL_CONST)) {
1217 *text += QString().sprintf("<a href=\"s%p\">", sym);
1218 *text += str2;
1219 *text += "</a>";
1220 } else
1221 *text += str2;
1222}
1223
1224QMenu* ConfigInfoView::createStandardContextMenu(const QPoint & pos)
1225{
1226 QMenu* popup = Parent::createStandardContextMenu(pos);
1227 QAction* action = new QAction("Show Debug Info", popup);
1228 action->setCheckable(true);
1229 connect(action, SIGNAL(toggled(bool)), SLOT(setShowDebug(bool)));
1230 connect(this, SIGNAL(showDebugChanged(bool)), action, SLOT(setOn(bool)));
1231 action->setChecked(showDebug());
1232 popup->addSeparator();
1233 popup->addAction(action);
1234 return popup;
1235}
1236
1237void ConfigInfoView::contextMenuEvent(QContextMenuEvent *e)
1238{
1239 Parent::contextMenuEvent(e);
1240}
1241
1242ConfigSearchWindow::ConfigSearchWindow(ConfigMainWindow* parent, const char *name)
1243 : Parent(parent), result(NULL)
1244{
1245 setObjectName(name);
1246 setWindowTitle("Search Config");
1247
1248 QVBoxLayout* layout1 = new QVBoxLayout(this);
1249 layout1->setContentsMargins(11, 11, 11, 11);
1250 layout1->setSpacing(6);
1251 QHBoxLayout* layout2 = new QHBoxLayout(0);
1252 layout2->setContentsMargins(0, 0, 0, 0);
1253 layout2->setSpacing(6);
1254 layout2->addWidget(new QLabel("Find:", this));
1255 editField = new QLineEdit(this);
1256 connect(editField, SIGNAL(returnPressed()), SLOT(search()));
1257 layout2->addWidget(editField);
1258 searchButton = new QPushButton("Search", this);
1259 searchButton->setAutoDefault(false);
1260 connect(searchButton, SIGNAL(clicked()), SLOT(search()));
1261 layout2->addWidget(searchButton);
1262 layout1->addLayout(layout2);
1263
1264 split = new QSplitter(this);
1265 split->setOrientation(Qt::Vertical);
1266 list = new ConfigView(split, name);
1267 list->list->mode = listMode;
1268 info = new ConfigInfoView(split, name);
1269 connect(list->list, SIGNAL(menuChanged(struct menu *)),
1270 info, SLOT(setInfo(struct menu *)));
1271 connect(list->list, SIGNAL(menuChanged(struct menu *)),
1272 parent, SLOT(setMenuLink(struct menu *)));
1273
1274 layout1->addWidget(split);
1275
1276 if (name) {
1277 QVariant x, y;
1278 int width, height;
1279 bool ok;
1280
1281 configSettings->beginGroup(name);
1282 width = configSettings->value("/window width", parent->width() / 2).toInt();
1283 height = configSettings->value("/window height", parent->height() / 2).toInt();
1284 resize(width, height);
1285 x = configSettings->value("/window x");
1286 y = configSettings->value("/window y");
1287 if ((x.isValid())&&(y.isValid()))
1288 move(x.toInt(), y.toInt());
1289 QList<int> sizes = configSettings->readSizes("/split", &ok);
1290 if (ok)
1291 split->setSizes(sizes);
1292 configSettings->endGroup();
1293 connect(configApp, SIGNAL(aboutToQuit()), SLOT(saveSettings()));
1294 }
1295}
1296
1297void ConfigSearchWindow::saveSettings(void)
1298{
1299 if (!objectName().isEmpty()) {
1300 configSettings->beginGroup(objectName());
1301 configSettings->setValue("/window x", pos().x());
1302 configSettings->setValue("/window y", pos().y());
1303 configSettings->setValue("/window width", size().width());
1304 configSettings->setValue("/window height", size().height());
1305 configSettings->writeSizes("/split", split->sizes());
1306 configSettings->endGroup();
1307 }
1308}
1309
1310void ConfigSearchWindow::search(void)
1311{
1312 struct symbol **p;
1313 struct property *prop;
1314 ConfigItem *lastItem = NULL;
1315
1316 free(result);
1317 list->list->clear();
1318 info->clear();
1319
1320 result = sym_re_search(editField->text().toLatin1());
1321 if (!result)
1322 return;
1323 for (p = result; *p; p++) {
1324 for_all_prompts((*p), prop)
1325 lastItem = new ConfigItem(list->list, lastItem, prop->menu,
1326 menu_is_visible(prop->menu));
1327 }
1328}
1329
1330/*
1331 * Construct the complete config widget
1332 */
1333ConfigMainWindow::ConfigMainWindow(void)
1334 : searchWindow(0)
1335{
1336 QMenuBar* menu;
1337 bool ok = true;
1338 QVariant x, y;
1339 int width, height;
1340 char title[256];
1341
1342 QDesktopWidget *d = configApp->desktop();
1343 snprintf(title, sizeof(title), "%s%s",
1344 rootmenu.prompt->text,
1345 ""
1346 );
1347 setWindowTitle(title);
1348
1349 width = configSettings->value("/window width", d->width() - 64).toInt();
1350 height = configSettings->value("/window height", d->height() - 64).toInt();
1351 resize(width, height);
1352 x = configSettings->value("/window x");
1353 y = configSettings->value("/window y");
1354 if ((x.isValid())&&(y.isValid()))
1355 move(x.toInt(), y.toInt());
1356
1357 split1 = new QSplitter(this);
1358 split1->setOrientation(Qt::Horizontal);
1359 setCentralWidget(split1);
1360
1361 menuView = new ConfigView(split1, "menu");
1362 menuList = menuView->list;
1363
1364 split2 = new QSplitter(split1);
1365 split2->setOrientation(Qt::Vertical);
1366
1367 // create config tree
1368 configView = new ConfigView(split2, "config");
1369 configList = configView->list;
1370
1371 helpText = new ConfigInfoView(split2, "help");
1372
1373 setTabOrder(configList, helpText);
1374 configList->setFocus();
1375
1376 menu = menuBar();
1377 toolBar = new QToolBar("Tools", this);
1378 addToolBar(toolBar);
1379
1380 backAction = new QAction(QPixmap(xpm_back), "Back", this);
1381 connect(backAction, SIGNAL(triggered(bool)), SLOT(goBack()));
1382 backAction->setEnabled(false);
1383 QAction *quitAction = new QAction("&Quit", this);
1384 quitAction->setShortcut(Qt::CTRL + Qt::Key_Q);
1385 connect(quitAction, SIGNAL(triggered(bool)), SLOT(close()));
1386 QAction *loadAction = new QAction(QPixmap(xpm_load), "&Load", this);
1387 loadAction->setShortcut(Qt::CTRL + Qt::Key_L);
1388 connect(loadAction, SIGNAL(triggered(bool)), SLOT(loadConfig()));
1389 saveAction = new QAction(QPixmap(xpm_save), "&Save", this);
1390 saveAction->setShortcut(Qt::CTRL + Qt::Key_S);
1391 connect(saveAction, SIGNAL(triggered(bool)), SLOT(saveConfig()));
1392 conf_set_changed_callback(conf_changed);
1393 // Set saveAction's initial state
1394 conf_changed();
1395 configname = xstrdup(conf_get_configname());
1396
1397 QAction *saveAsAction = new QAction("Save &As...", this);
1398 connect(saveAsAction, SIGNAL(triggered(bool)), SLOT(saveConfigAs()));
1399 QAction *searchAction = new QAction("&Find", this);
1400 searchAction->setShortcut(Qt::CTRL + Qt::Key_F);
1401 connect(searchAction, SIGNAL(triggered(bool)), SLOT(searchConfig()));
1402 singleViewAction = new QAction(QPixmap(xpm_single_view), "Single View", this);
1403 singleViewAction->setCheckable(true);
1404 connect(singleViewAction, SIGNAL(triggered(bool)), SLOT(showSingleView()));
1405 splitViewAction = new QAction(QPixmap(xpm_split_view), "Split View", this);
1406 splitViewAction->setCheckable(true);
1407 connect(splitViewAction, SIGNAL(triggered(bool)), SLOT(showSplitView()));
1408 fullViewAction = new QAction(QPixmap(xpm_tree_view), "Full View", this);
1409 fullViewAction->setCheckable(true);
1410 connect(fullViewAction, SIGNAL(triggered(bool)), SLOT(showFullView()));
1411
1412 QAction *showNameAction = new QAction("Show Name", this);
1413 showNameAction->setCheckable(true);
1414 connect(showNameAction, SIGNAL(toggled(bool)), configView, SLOT(setShowName(bool)));
1415 showNameAction->setChecked(configView->showName());
1416 QAction *showRangeAction = new QAction("Show Range", this);
1417 showRangeAction->setCheckable(true);
1418 connect(showRangeAction, SIGNAL(toggled(bool)), configView, SLOT(setShowRange(bool)));
1419 QAction *showDataAction = new QAction("Show Data", this);
1420 showDataAction->setCheckable(true);
1421 connect(showDataAction, SIGNAL(toggled(bool)), configView, SLOT(setShowData(bool)));
1422
1423 QActionGroup *optGroup = new QActionGroup(this);
1424 optGroup->setExclusive(true);
1425 connect(optGroup, SIGNAL(triggered(QAction*)), configView,
1426 SLOT(setOptionMode(QAction *)));
1427 connect(optGroup, SIGNAL(triggered(QAction *)), menuView,
1428 SLOT(setOptionMode(QAction *)));
1429
1430 configView->showNormalAction = new QAction("Show Normal Options", optGroup);
1431 configView->showAllAction = new QAction("Show All Options", optGroup);
1432 configView->showPromptAction = new QAction("Show Prompt Options", optGroup);
1433 configView->showNormalAction->setCheckable(true);
1434 configView->showAllAction->setCheckable(true);
1435 configView->showPromptAction->setCheckable(true);
1436
1437 QAction *showDebugAction = new QAction("Show Debug Info", this);
1438 showDebugAction->setCheckable(true);
1439 connect(showDebugAction, SIGNAL(toggled(bool)), helpText, SLOT(setShowDebug(bool)));
1440 showDebugAction->setChecked(helpText->showDebug());
1441
1442 QAction *showIntroAction = new QAction("Introduction", this);
1443 connect(showIntroAction, SIGNAL(triggered(bool)), SLOT(showIntro()));
1444 QAction *showAboutAction = new QAction("About", this);
1445 connect(showAboutAction, SIGNAL(triggered(bool)), SLOT(showAbout()));
1446
1447 // init tool bar
1448 toolBar->addAction(backAction);
1449 toolBar->addSeparator();
1450 toolBar->addAction(loadAction);
1451 toolBar->addAction(saveAction);
1452 toolBar->addSeparator();
1453 toolBar->addAction(singleViewAction);
1454 toolBar->addAction(splitViewAction);
1455 toolBar->addAction(fullViewAction);
1456
1457 // create config menu
1458 QMenu* config = menu->addMenu("&File");
1459 config->addAction(loadAction);
1460 config->addAction(saveAction);
1461 config->addAction(saveAsAction);
1462 config->addSeparator();
1463 config->addAction(quitAction);
1464
1465 // create edit menu
1466 QMenu* editMenu = menu->addMenu("&Edit");
1467 editMenu->addAction(searchAction);
1468
1469 // create options menu
1470 QMenu* optionMenu = menu->addMenu("&Option");
1471 optionMenu->addAction(showNameAction);
1472 optionMenu->addAction(showRangeAction);
1473 optionMenu->addAction(showDataAction);
1474 optionMenu->addSeparator();
1475 optionMenu->addActions(optGroup->actions());
1476 optionMenu->addSeparator();
1477 optionMenu->addAction(showDebugAction);
1478
1479 // create help menu
1480 menu->addSeparator();
1481 QMenu* helpMenu = menu->addMenu("&Help");
1482 helpMenu->addAction(showIntroAction);
1483 helpMenu->addAction(showAboutAction);
1484
1485 connect(configList, SIGNAL(menuChanged(struct menu *)),
1486 helpText, SLOT(setInfo(struct menu *)));
1487 connect(configList, SIGNAL(menuSelected(struct menu *)),
1488 SLOT(changeMenu(struct menu *)));
1489 connect(configList, SIGNAL(parentSelected()),
1490 SLOT(goBack()));
1491 connect(menuList, SIGNAL(menuChanged(struct menu *)),
1492 helpText, SLOT(setInfo(struct menu *)));
1493 connect(menuList, SIGNAL(menuSelected(struct menu *)),
1494 SLOT(changeMenu(struct menu *)));
1495
1496 connect(configList, SIGNAL(gotFocus(struct menu *)),
1497 helpText, SLOT(setInfo(struct menu *)));
1498 connect(menuList, SIGNAL(gotFocus(struct menu *)),
1499 helpText, SLOT(setInfo(struct menu *)));
1500 connect(menuList, SIGNAL(gotFocus(struct menu *)),
1501 SLOT(listFocusChanged(void)));
1502 connect(helpText, SIGNAL(menuSelected(struct menu *)),
1503 SLOT(setMenuLink(struct menu *)));
1504
1505 QString listMode = configSettings->value("/listMode", "symbol").toString();
1506 if (listMode == "single")
1507 showSingleView();
1508 else if (listMode == "full")
1509 showFullView();
1510 else /*if (listMode == "split")*/
1511 showSplitView();
1512
1513 // UI setup done, restore splitter positions
1514 QList<int> sizes = configSettings->readSizes("/split1", &ok);
1515 if (ok)
1516 split1->setSizes(sizes);
1517
1518 sizes = configSettings->readSizes("/split2", &ok);
1519 if (ok)
1520 split2->setSizes(sizes);
1521}
1522
1523void ConfigMainWindow::loadConfig(void)
1524{
1525 QString str;
1526 QByteArray ba;
1527 const char *name;
1528
1529 str = QFileDialog::getOpenFileName(this, "", configname);
1530 if (str.isNull())
1531 return;
1532
1533 ba = str.toLocal8Bit();
1534 name = ba.data();
1535
1536 if (conf_read(name))
1537 QMessageBox::information(this, "qconf", "Unable to load configuration!");
1538
1539 free(configname);
1540 configname = xstrdup(name);
1541
1542 ConfigView::updateListAll();
1543}
1544
1545bool ConfigMainWindow::saveConfig(void)
1546{
1547 if (conf_write(configname)) {
1548 QMessageBox::information(this, "qconf", "Unable to save configuration!");
1549 return false;
1550 }
1551 conf_write_autoconf(0);
1552
1553 return true;
1554}
1555
1556void ConfigMainWindow::saveConfigAs(void)
1557{
1558 QString str;
1559 QByteArray ba;
1560 const char *name;
1561
1562 str = QFileDialog::getSaveFileName(this, "", configname);
1563 if (str.isNull())
1564 return;
1565
1566 ba = str.toLocal8Bit();
1567 name = ba.data();
1568
1569 if (conf_write(name)) {
1570 QMessageBox::information(this, "qconf", "Unable to save configuration!");
1571 }
1572 conf_write_autoconf(0);
1573
1574 free(configname);
1575 configname = xstrdup(name);
1576}
1577
1578void ConfigMainWindow::searchConfig(void)
1579{
1580 if (!searchWindow)
1581 searchWindow = new ConfigSearchWindow(this, "search");
1582 searchWindow->show();
1583}
1584
1585void ConfigMainWindow::changeMenu(struct menu *menu)
1586{
1587 configList->setRootMenu(menu);
1588 if (configList->rootEntry->parent == &rootmenu)
1589 backAction->setEnabled(false);
1590 else
1591 backAction->setEnabled(true);
1592}
1593
1594void ConfigMainWindow::setMenuLink(struct menu *menu)
1595{
1596 struct menu *parent;
1597 ConfigList* list = NULL;
1598 ConfigItem* item;
1599
1600 if (configList->menuSkip(menu))
1601 return;
1602
1603 switch (configList->mode) {
1604 case singleMode:
1605 list = configList;
1606 parent = menu_get_parent_menu(menu);
1607 if (!parent)
1608 return;
1609 list->setRootMenu(parent);
1610 break;
1611 case symbolMode:
1612 if (menu->flags & MENU_ROOT) {
1613 configList->setRootMenu(menu);
1614 configList->clearSelection();
1615 list = menuList;
1616 } else {
1617 list = configList;
1618 parent = menu_get_parent_menu(menu->parent);
1619 if (!parent)
1620 return;
1621 item = menuList->findConfigItem(parent);
1622 if (item) {
1623 item->setSelected(true);
1624 menuList->scrollToItem(item);
1625 }
1626 list->setRootMenu(parent);
1627 }
1628 break;
1629 case fullMode:
1630 list = configList;
1631 break;
1632 default:
1633 break;
1634 }
1635
1636 if (list) {
1637 item = list->findConfigItem(menu);
1638 if (item) {
1639 item->setSelected(true);
1640 list->scrollToItem(item);
1641 list->setFocus();
1642 }
1643 }
1644}
1645
1646void ConfigMainWindow::listFocusChanged(void)
1647{
1648 if (menuList->mode == menuMode)
1649 configList->clearSelection();
1650}
1651
1652void ConfigMainWindow::goBack(void)
1653{
1654 ConfigItem* item, *oldSelection;
1655
1656 configList->setParentMenu();
1657 if (configList->rootEntry == &rootmenu)
1658 backAction->setEnabled(false);
1659
1660 if (menuList->selectedItems().count() == 0)
1661 return;
1662
1663 item = (ConfigItem*)menuList->selectedItems().first();
1664 oldSelection = item;
1665 while (item) {
1666 if (item->menu == configList->rootEntry) {
1667 oldSelection->setSelected(false);
1668 item->setSelected(true);
1669 break;
1670 }
1671 item = (ConfigItem*)item->parent();
1672 }
1673}
1674
1675void ConfigMainWindow::showSingleView(void)
1676{
1677 singleViewAction->setEnabled(false);
1678 singleViewAction->setChecked(true);
1679 splitViewAction->setEnabled(true);
1680 splitViewAction->setChecked(false);
1681 fullViewAction->setEnabled(true);
1682 fullViewAction->setChecked(false);
1683
1684 menuView->hide();
1685 menuList->setRootMenu(0);
1686 configList->mode = singleMode;
1687 if (configList->rootEntry == &rootmenu)
1688 configList->updateListAll();
1689 else
1690 configList->setRootMenu(&rootmenu);
1691 configList->setFocus();
1692}
1693
1694void ConfigMainWindow::showSplitView(void)
1695{
1696 singleViewAction->setEnabled(true);
1697 singleViewAction->setChecked(false);
1698 splitViewAction->setEnabled(false);
1699 splitViewAction->setChecked(true);
1700 fullViewAction->setEnabled(true);
1701 fullViewAction->setChecked(false);
1702
1703 configList->mode = symbolMode;
1704 if (configList->rootEntry == &rootmenu)
1705 configList->updateListAll();
1706 else
1707 configList->setRootMenu(&rootmenu);
1708 configList->setAllOpen(true);
1709 configApp->processEvents();
1710 menuList->mode = menuMode;
1711 menuList->setRootMenu(&rootmenu);
1712 menuList->setAllOpen(true);
1713 menuView->show();
1714 menuList->setFocus();
1715}
1716
1717void ConfigMainWindow::showFullView(void)
1718{
1719 singleViewAction->setEnabled(true);
1720 singleViewAction->setChecked(false);
1721 splitViewAction->setEnabled(true);
1722 splitViewAction->setChecked(false);
1723 fullViewAction->setEnabled(false);
1724 fullViewAction->setChecked(true);
1725
1726 menuView->hide();
1727 menuList->setRootMenu(0);
1728 configList->mode = fullMode;
1729 if (configList->rootEntry == &rootmenu)
1730 configList->updateListAll();
1731 else
1732 configList->setRootMenu(&rootmenu);
1733 configList->setFocus();
1734}
1735
1736/*
1737 * ask for saving configuration before quitting
1738 * TODO ask only when something changed
1739 */
1740void ConfigMainWindow::closeEvent(QCloseEvent* e)
1741{
1742 if (!conf_get_changed()) {
1743 e->accept();
1744 return;
1745 }
1746 QMessageBox mb("qconf", "Save configuration?", QMessageBox::Warning,
1747 QMessageBox::Yes | QMessageBox::Default, QMessageBox::No, QMessageBox::Cancel | QMessageBox::Escape);
1748 mb.setButtonText(QMessageBox::Yes, "&Save Changes");
1749 mb.setButtonText(QMessageBox::No, "&Discard Changes");
1750 mb.setButtonText(QMessageBox::Cancel, "Cancel Exit");
1751 switch (mb.exec()) {
1752 case QMessageBox::Yes:
1753 if (saveConfig())
1754 e->accept();
1755 else
1756 e->ignore();
1757 break;
1758 case QMessageBox::No:
1759 e->accept();
1760 break;
1761 case QMessageBox::Cancel:
1762 e->ignore();
1763 break;
1764 }
1765}
1766
1767void ConfigMainWindow::showIntro(void)
1768{
1769 static const QString str = "Welcome to the qconf graphical configuration tool.\n\n"
1770 "For each option, a blank box indicates the feature is disabled, a check\n"
1771 "indicates it is enabled, and a dot indicates that it is to be compiled\n"
1772 "as a module. Clicking on the box will cycle through the three states.\n\n"
1773 "If you do not see an option (e.g., a device driver) that you believe\n"
1774 "should be present, try turning on Show All Options under the Options menu.\n"
1775 "Although there is no cross reference yet to help you figure out what other\n"
1776 "options must be enabled to support the option you are interested in, you can\n"
1777 "still view the help of a grayed-out option.\n\n"
1778 "Toggling Show Debug Info under the Options menu will show the dependencies,\n"
1779 "which you can then match by examining other options.\n\n";
1780
1781 QMessageBox::information(this, "qconf", str);
1782}
1783
1784void ConfigMainWindow::showAbout(void)
1785{
1786 static const QString str = "qconf is Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>.\n"
1787 "Copyright (C) 2015 Boris Barbulovski <bbarbulovski@gmail.com>.\n\n"
1788 "Bug reports and feature request can also be entered at http://bugzilla.kernel.org/\n";
1789
1790 QMessageBox::information(this, "qconf", str);
1791}
1792
1793void ConfigMainWindow::saveSettings(void)
1794{
1795 configSettings->setValue("/window x", pos().x());
1796 configSettings->setValue("/window y", pos().y());
1797 configSettings->setValue("/window width", size().width());
1798 configSettings->setValue("/window height", size().height());
1799
1800 QString entry;
1801 switch(configList->mode) {
1802 case singleMode :
1803 entry = "single";
1804 break;
1805
1806 case symbolMode :
1807 entry = "split";
1808 break;
1809
1810 case fullMode :
1811 entry = "full";
1812 break;
1813
1814 default:
1815 break;
1816 }
1817 configSettings->setValue("/listMode", entry);
1818
1819 configSettings->writeSizes("/split1", split1->sizes());
1820 configSettings->writeSizes("/split2", split2->sizes());
1821}
1822
1823void ConfigMainWindow::conf_changed(void)
1824{
1825 if (saveAction)
1826 saveAction->setEnabled(conf_get_changed());
1827}
1828
1829void fixup_rootmenu(struct menu *menu)
1830{
1831 struct menu *child;
1832 static int menu_cnt = 0;
1833
1834 menu->flags |= MENU_ROOT;
1835 for (child = menu->list; child; child = child->next) {
1836 if (child->prompt && child->prompt->type == P_MENU) {
1837 menu_cnt++;
1838 fixup_rootmenu(child);
1839 menu_cnt--;
1840 } else if (!menu_cnt)
1841 fixup_rootmenu(child);
1842 }
1843}
1844
1845static const char *progname;
1846
1847static void usage(void)
1848{
1849 printf("%s [-s] <config>\n", progname);
1850 exit(0);
1851}
1852
1853int main(int ac, char** av)
1854{
1855 ConfigMainWindow* v;
1856 const char *name;
1857
1858 progname = av[0];
1859 configApp = new QApplication(ac, av);
1860 if (ac > 1 && av[1][0] == '-') {
1861 switch (av[1][1]) {
1862 case 's':
1863 conf_set_message_callback(NULL);
1864 break;
1865 case 'h':
1866 case '?':
1867 usage();
1868 }
1869 name = av[2];
1870 } else
1871 name = av[1];
1872 if (!name)
1873 usage();
1874
1875 conf_parse(name);
1876 fixup_rootmenu(&rootmenu);
1877 conf_read(NULL);
1878 //zconfdump(stdout);
1879
1880 configSettings = new ConfigSettings();
1881 configSettings->beginGroup("/kconfig/qconf");
1882 v = new ConfigMainWindow();
1883
1884 //zconfdump(stdout);
1885 configApp->connect(configApp, SIGNAL(lastWindowClosed()), SLOT(quit()));
1886 configApp->connect(configApp, SIGNAL(aboutToQuit()), v, SLOT(saveSettings()));
1887 v->show();
1888 configApp->exec();
1889
1890 configSettings->endGroup();
1891 delete configSettings;
1892 delete v;
1893 delete configApp;
1894
1895 return 0;
1896}
1// SPDX-License-Identifier: GPL-2.0
2/*
3 * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>
4 * Copyright (C) 2015 Boris Barbulovski <bbarbulovski@gmail.com>
5 */
6
7#include <QAction>
8#include <QActionGroup>
9#include <QApplication>
10#include <QCloseEvent>
11#include <QDebug>
12#include <QFileDialog>
13#include <QLabel>
14#include <QLayout>
15#include <QList>
16#include <QMenu>
17#include <QMenuBar>
18#include <QMessageBox>
19#include <QRegularExpression>
20#include <QScreen>
21#include <QToolBar>
22
23#include <stdlib.h>
24
25#include <xalloc.h>
26#include "lkc.h"
27#include "qconf.h"
28
29#include "images.h"
30
31
32static QApplication *configApp;
33static ConfigSettings *configSettings;
34
35QAction *ConfigMainWindow::saveAction;
36
37ConfigSettings::ConfigSettings()
38 : QSettings("kernel.org", "qconf")
39{
40}
41
42/**
43 * Reads a list of integer values from the application settings.
44 */
45QList<int> ConfigSettings::readSizes(const QString& key, bool *ok)
46{
47 QList<int> result;
48
49 if (contains(key))
50 {
51 QStringList entryList = value(key).toStringList();
52 QStringList::Iterator it;
53
54 for (it = entryList.begin(); it != entryList.end(); ++it)
55 result.push_back((*it).toInt());
56
57 *ok = true;
58 }
59 else
60 *ok = false;
61
62 return result;
63}
64
65/**
66 * Writes a list of integer values to the application settings.
67 */
68bool ConfigSettings::writeSizes(const QString& key, const QList<int>& value)
69{
70 QStringList stringList;
71 QList<int>::ConstIterator it;
72
73 for (it = value.begin(); it != value.end(); ++it)
74 stringList.push_back(QString::number(*it));
75 setValue(key, stringList);
76
77 return true;
78}
79
80QIcon ConfigItem::symbolYesIcon;
81QIcon ConfigItem::symbolModIcon;
82QIcon ConfigItem::symbolNoIcon;
83QIcon ConfigItem::choiceYesIcon;
84QIcon ConfigItem::choiceNoIcon;
85QIcon ConfigItem::menuIcon;
86QIcon ConfigItem::menubackIcon;
87
88/*
89 * update the displayed of a menu entry
90 */
91void ConfigItem::updateMenu(void)
92{
93 ConfigList* list;
94 struct symbol* sym;
95 struct property *prop;
96 QString prompt;
97 int type;
98 tristate expr;
99
100 list = listView();
101 if (goParent) {
102 setIcon(promptColIdx, menubackIcon);
103 prompt = "..";
104 goto set_prompt;
105 }
106
107 sym = menu->sym;
108 prop = menu->prompt;
109 prompt = menu_get_prompt(menu);
110
111 if (prop) switch (prop->type) {
112 case P_MENU:
113 if (list->mode == singleMode) {
114 /* a menuconfig entry is displayed differently
115 * depending whether it's at the view root or a child.
116 */
117 if (sym && list->rootEntry == menu)
118 break;
119 setIcon(promptColIdx, menuIcon);
120 } else {
121 if (sym)
122 break;
123 setIcon(promptColIdx, QIcon());
124 }
125 goto set_prompt;
126 case P_COMMENT:
127 setIcon(promptColIdx, QIcon());
128 prompt = "*** " + prompt + " ***";
129 goto set_prompt;
130 default:
131 ;
132 }
133 if (!sym)
134 goto set_prompt;
135
136 setText(nameColIdx, sym->name);
137
138 type = sym_get_type(sym);
139 switch (type) {
140 case S_BOOLEAN:
141 case S_TRISTATE:
142 char ch;
143
144 if (!sym_is_changeable(sym) && list->optMode == normalOpt) {
145 setIcon(promptColIdx, QIcon());
146 break;
147 }
148 expr = sym_get_tristate_value(sym);
149 switch (expr) {
150 case yes:
151 if (sym_is_choice_value(sym))
152 setIcon(promptColIdx, choiceYesIcon);
153 else
154 setIcon(promptColIdx, symbolYesIcon);
155 ch = 'Y';
156 break;
157 case mod:
158 setIcon(promptColIdx, symbolModIcon);
159 ch = 'M';
160 break;
161 default:
162 if (sym_is_choice_value(sym))
163 setIcon(promptColIdx, choiceNoIcon);
164 else
165 setIcon(promptColIdx, symbolNoIcon);
166 ch = 'N';
167 break;
168 }
169
170 setText(dataColIdx, QChar(ch));
171 break;
172 case S_INT:
173 case S_HEX:
174 case S_STRING:
175 setText(dataColIdx, sym_get_string_value(sym));
176 break;
177 }
178 if (!sym_has_value(sym))
179 prompt += " (NEW)";
180set_prompt:
181 setText(promptColIdx, prompt);
182}
183
184void ConfigItem::testUpdateMenu(void)
185{
186 ConfigItem* i;
187
188 if (!menu)
189 return;
190
191 sym_calc_value(menu->sym);
192 if (menu->flags & MENU_CHANGED) {
193 /* the menu entry changed, so update all list items */
194 menu->flags &= ~MENU_CHANGED;
195 for (i = (ConfigItem*)menu->data; i; i = i->nextItem)
196 i->updateMenu();
197 } else if (listView()->updateAll)
198 updateMenu();
199}
200
201
202/*
203 * construct a menu entry
204 */
205void ConfigItem::init(void)
206{
207 if (menu) {
208 ConfigList* list = listView();
209 nextItem = (ConfigItem*)menu->data;
210 menu->data = this;
211
212 if (list->mode != fullMode)
213 setExpanded(true);
214 sym_calc_value(menu->sym);
215
216 if (menu->sym) {
217 enum symbol_type type = menu->sym->type;
218
219 // Allow to edit "int", "hex", and "string" in-place in
220 // the data column. Unfortunately, you cannot specify
221 // the flags per column. Set ItemIsEditable for all
222 // columns here, and check the column in createEditor().
223 if (type == S_INT || type == S_HEX || type == S_STRING)
224 setFlags(flags() | Qt::ItemIsEditable);
225 }
226 }
227 updateMenu();
228}
229
230/*
231 * destruct a menu entry
232 */
233ConfigItem::~ConfigItem(void)
234{
235 if (menu) {
236 ConfigItem** ip = (ConfigItem**)&menu->data;
237 for (; *ip; ip = &(*ip)->nextItem) {
238 if (*ip == this) {
239 *ip = nextItem;
240 break;
241 }
242 }
243 }
244}
245
246QWidget *ConfigItemDelegate::createEditor(QWidget *parent,
247 const QStyleOptionViewItem &option,
248 const QModelIndex &index) const
249{
250 ConfigItem *item;
251
252 // Only the data column is editable
253 if (index.column() != dataColIdx)
254 return nullptr;
255
256 // You cannot edit invisible menus
257 item = static_cast<ConfigItem *>(index.internalPointer());
258 if (!item || !item->menu || !menu_is_visible(item->menu))
259 return nullptr;
260
261 return QStyledItemDelegate::createEditor(parent, option, index);
262}
263
264void ConfigItemDelegate::setModelData(QWidget *editor,
265 QAbstractItemModel *model,
266 const QModelIndex &index) const
267{
268 QLineEdit *lineEdit;
269 ConfigItem *item;
270 struct symbol *sym;
271 bool success;
272
273 lineEdit = qobject_cast<QLineEdit *>(editor);
274 // If this is not a QLineEdit, use the parent's default.
275 // (does this happen?)
276 if (!lineEdit)
277 goto parent;
278
279 item = static_cast<ConfigItem *>(index.internalPointer());
280 if (!item || !item->menu)
281 goto parent;
282
283 sym = item->menu->sym;
284 if (!sym)
285 goto parent;
286
287 success = sym_set_string_value(sym, lineEdit->text().toUtf8().data());
288 if (success) {
289 ConfigList::updateListForAll();
290 } else {
291 QMessageBox::information(editor, "qconf",
292 "Cannot set the data (maybe due to out of range).\n"
293 "Setting the old value.");
294 lineEdit->setText(sym_get_string_value(sym));
295 }
296
297parent:
298 QStyledItemDelegate::setModelData(editor, model, index);
299}
300
301ConfigList::ConfigList(QWidget *parent, const char *name)
302 : QTreeWidget(parent),
303 updateAll(false),
304 showName(false), mode(singleMode), optMode(normalOpt),
305 rootEntry(0), headerPopup(0)
306{
307 setObjectName(name);
308 setSortingEnabled(false);
309
310 setVerticalScrollMode(ScrollPerPixel);
311 setHorizontalScrollMode(ScrollPerPixel);
312
313 setHeaderLabels(QStringList() << "Option" << "Name" << "Value");
314
315 connect(this, &ConfigList::itemSelectionChanged,
316 this, &ConfigList::updateSelection);
317
318 if (name) {
319 configSettings->beginGroup(name);
320 showName = configSettings->value("/showName", false).toBool();
321 optMode = (enum optionMode)configSettings->value("/optionMode", 0).toInt();
322 configSettings->endGroup();
323 connect(configApp, &QApplication::aboutToQuit,
324 this, &ConfigList::saveSettings);
325 }
326
327 showColumn(promptColIdx);
328
329 setItemDelegate(new ConfigItemDelegate(this));
330
331 allLists.append(this);
332
333 reinit();
334}
335
336ConfigList::~ConfigList()
337{
338 allLists.removeOne(this);
339}
340
341bool ConfigList::menuSkip(struct menu *menu)
342{
343 if (optMode == normalOpt && menu_is_visible(menu))
344 return false;
345 if (optMode == promptOpt && menu_has_prompt(menu))
346 return false;
347 if (optMode == allOpt)
348 return false;
349 return true;
350}
351
352void ConfigList::reinit(void)
353{
354 hideColumn(nameColIdx);
355
356 if (showName)
357 showColumn(nameColIdx);
358
359 updateListAll();
360}
361
362void ConfigList::setOptionMode(QAction *action)
363{
364 if (action == showNormalAction)
365 optMode = normalOpt;
366 else if (action == showAllAction)
367 optMode = allOpt;
368 else
369 optMode = promptOpt;
370
371 updateListAll();
372}
373
374void ConfigList::saveSettings(void)
375{
376 if (!objectName().isEmpty()) {
377 configSettings->beginGroup(objectName());
378 configSettings->setValue("/showName", showName);
379 configSettings->setValue("/optionMode", (int)optMode);
380 configSettings->endGroup();
381 }
382}
383
384ConfigItem* ConfigList::findConfigItem(struct menu *menu)
385{
386 ConfigItem* item = (ConfigItem*)menu->data;
387
388 for (; item; item = item->nextItem) {
389 if (this == item->listView())
390 break;
391 }
392
393 return item;
394}
395
396void ConfigList::updateSelection(void)
397{
398 struct menu *menu;
399 enum prop_type type;
400
401 if (selectedItems().count() == 0)
402 return;
403
404 ConfigItem* item = (ConfigItem*)selectedItems().first();
405 if (!item)
406 return;
407
408 menu = item->menu;
409 emit menuChanged(menu);
410 if (!menu)
411 return;
412 type = menu->prompt ? menu->prompt->type : P_UNKNOWN;
413 if (mode == menuMode && type == P_MENU)
414 emit menuSelected(menu);
415}
416
417void ConfigList::updateList()
418{
419 ConfigItem* last = 0;
420 ConfigItem *item;
421
422 if (!rootEntry) {
423 if (mode != listMode)
424 goto update;
425 QTreeWidgetItemIterator it(this);
426
427 while (*it) {
428 item = (ConfigItem*)(*it);
429 if (!item->menu)
430 continue;
431 item->testUpdateMenu();
432
433 ++it;
434 }
435 return;
436 }
437
438 if (rootEntry != &rootmenu && mode == singleMode) {
439 item = (ConfigItem *)topLevelItem(0);
440 if (!item)
441 item = new ConfigItem(this, 0);
442 last = item;
443 }
444 if ((mode == singleMode || (mode == symbolMode && !(rootEntry->flags & MENU_ROOT))) &&
445 rootEntry->sym && rootEntry->prompt) {
446 item = last ? last->nextSibling() : nullptr;
447 if (!item)
448 item = new ConfigItem(this, last, rootEntry);
449 else
450 item->testUpdateMenu();
451
452 updateMenuList(item, rootEntry);
453 update();
454 resizeColumnToContents(0);
455 return;
456 }
457update:
458 updateMenuList(rootEntry);
459 update();
460 resizeColumnToContents(0);
461}
462
463void ConfigList::updateListForAll()
464{
465 QListIterator<ConfigList *> it(allLists);
466
467 while (it.hasNext()) {
468 ConfigList *list = it.next();
469
470 list->updateList();
471 }
472}
473
474void ConfigList::updateListAllForAll()
475{
476 QListIterator<ConfigList *> it(allLists);
477
478 while (it.hasNext()) {
479 ConfigList *list = it.next();
480
481 list->updateList();
482 }
483}
484
485void ConfigList::setValue(ConfigItem* item, tristate val)
486{
487 struct symbol* sym;
488 int type;
489 tristate oldval;
490
491 sym = item->menu ? item->menu->sym : 0;
492 if (!sym)
493 return;
494
495 type = sym_get_type(sym);
496 switch (type) {
497 case S_BOOLEAN:
498 case S_TRISTATE:
499 oldval = sym_get_tristate_value(sym);
500
501 if (!sym_set_tristate_value(sym, val))
502 return;
503 if (oldval == no && item->menu->list)
504 item->setExpanded(true);
505 ConfigList::updateListForAll();
506 break;
507 }
508}
509
510void ConfigList::changeValue(ConfigItem* item)
511{
512 struct symbol* sym;
513 struct menu* menu;
514 int type, oldexpr, newexpr;
515
516 menu = item->menu;
517 if (!menu)
518 return;
519 sym = menu->sym;
520 if (!sym) {
521 if (item->menu->list)
522 item->setExpanded(!item->isExpanded());
523 return;
524 }
525
526 type = sym_get_type(sym);
527 switch (type) {
528 case S_BOOLEAN:
529 case S_TRISTATE:
530 oldexpr = sym_get_tristate_value(sym);
531 newexpr = sym_toggle_tristate_value(sym);
532 if (item->menu->list) {
533 if (oldexpr == newexpr)
534 item->setExpanded(!item->isExpanded());
535 else if (oldexpr == no)
536 item->setExpanded(true);
537 }
538 if (oldexpr != newexpr)
539 ConfigList::updateListForAll();
540 break;
541 default:
542 break;
543 }
544}
545
546void ConfigList::setRootMenu(struct menu *menu)
547{
548 enum prop_type type;
549
550 if (rootEntry == menu)
551 return;
552 type = menu && menu->prompt ? menu->prompt->type : P_UNKNOWN;
553 if (type != P_MENU)
554 return;
555 updateMenuList(0);
556 rootEntry = menu;
557 updateListAll();
558 if (currentItem()) {
559 setSelected(currentItem(), hasFocus());
560 scrollToItem(currentItem());
561 }
562}
563
564void ConfigList::setParentMenu(void)
565{
566 ConfigItem* item;
567 struct menu *oldroot;
568
569 oldroot = rootEntry;
570 if (rootEntry == &rootmenu)
571 return;
572 setRootMenu(menu_get_parent_menu(rootEntry->parent));
573
574 QTreeWidgetItemIterator it(this);
575 while (*it) {
576 item = (ConfigItem *)(*it);
577 if (item->menu == oldroot) {
578 setCurrentItem(item);
579 scrollToItem(item);
580 break;
581 }
582
583 ++it;
584 }
585}
586
587/*
588 * update all the children of a menu entry
589 * removes/adds the entries from the parent widget as necessary
590 *
591 * parent: either the menu list widget or a menu entry widget
592 * menu: entry to be updated
593 */
594void ConfigList::updateMenuList(ConfigItem *parent, struct menu* menu)
595{
596 struct menu* child;
597 ConfigItem* item;
598 ConfigItem* last;
599 enum prop_type type;
600
601 if (!menu) {
602 while (parent->childCount() > 0)
603 {
604 delete parent->takeChild(0);
605 }
606
607 return;
608 }
609
610 last = parent->firstChild();
611 if (last && !last->goParent)
612 last = 0;
613 for (child = menu->list; child; child = child->next) {
614 item = last ? last->nextSibling() : parent->firstChild();
615 type = child->prompt ? child->prompt->type : P_UNKNOWN;
616
617 switch (mode) {
618 case menuMode:
619 if (!(child->flags & MENU_ROOT))
620 goto hide;
621 break;
622 case symbolMode:
623 if (child->flags & MENU_ROOT)
624 goto hide;
625 break;
626 default:
627 break;
628 }
629
630 if (!menuSkip(child)) {
631 if (!child->sym && !child->list && !child->prompt)
632 continue;
633 if (!item || item->menu != child)
634 item = new ConfigItem(parent, last, child);
635 else
636 item->testUpdateMenu();
637
638 if (mode == fullMode || mode == menuMode || type != P_MENU)
639 updateMenuList(item, child);
640 else
641 updateMenuList(item, 0);
642 last = item;
643 continue;
644 }
645hide:
646 if (item && item->menu == child) {
647 last = parent->firstChild();
648 if (last == item)
649 last = 0;
650 else while (last->nextSibling() != item)
651 last = last->nextSibling();
652 delete item;
653 }
654 }
655}
656
657void ConfigList::updateMenuList(struct menu *menu)
658{
659 struct menu* child;
660 ConfigItem* item;
661 ConfigItem* last;
662 enum prop_type type;
663
664 if (!menu) {
665 while (topLevelItemCount() > 0)
666 {
667 delete takeTopLevelItem(0);
668 }
669
670 return;
671 }
672
673 last = (ConfigItem *)topLevelItem(0);
674 if (last && !last->goParent)
675 last = 0;
676 for (child = menu->list; child; child = child->next) {
677 item = last ? last->nextSibling() : (ConfigItem *)topLevelItem(0);
678 type = child->prompt ? child->prompt->type : P_UNKNOWN;
679
680 switch (mode) {
681 case menuMode:
682 if (!(child->flags & MENU_ROOT))
683 goto hide;
684 break;
685 case symbolMode:
686 if (child->flags & MENU_ROOT)
687 goto hide;
688 break;
689 default:
690 break;
691 }
692
693 if (!menuSkip(child)) {
694 if (!child->sym && !child->list && !child->prompt)
695 continue;
696 if (!item || item->menu != child)
697 item = new ConfigItem(this, last, child);
698 else
699 item->testUpdateMenu();
700
701 if (mode == fullMode || mode == menuMode || type != P_MENU)
702 updateMenuList(item, child);
703 else
704 updateMenuList(item, 0);
705 last = item;
706 continue;
707 }
708hide:
709 if (item && item->menu == child) {
710 last = (ConfigItem *)topLevelItem(0);
711 if (last == item)
712 last = 0;
713 else while (last->nextSibling() != item)
714 last = last->nextSibling();
715 delete item;
716 }
717 }
718}
719
720void ConfigList::keyPressEvent(QKeyEvent* ev)
721{
722 QTreeWidgetItem* i = currentItem();
723 ConfigItem* item;
724 struct menu *menu;
725 enum prop_type type;
726
727 if (ev->key() == Qt::Key_Escape && mode == singleMode) {
728 emit parentSelected();
729 ev->accept();
730 return;
731 }
732
733 if (!i) {
734 Parent::keyPressEvent(ev);
735 return;
736 }
737 item = (ConfigItem*)i;
738
739 switch (ev->key()) {
740 case Qt::Key_Return:
741 case Qt::Key_Enter:
742 if (item->goParent) {
743 emit parentSelected();
744 break;
745 }
746 menu = item->menu;
747 if (!menu)
748 break;
749 type = menu->prompt ? menu->prompt->type : P_UNKNOWN;
750 if (type == P_MENU && rootEntry != menu &&
751 mode != fullMode && mode != menuMode) {
752 if (mode == menuMode)
753 emit menuSelected(menu);
754 else
755 emit itemSelected(menu);
756 break;
757 }
758 case Qt::Key_Space:
759 changeValue(item);
760 break;
761 case Qt::Key_N:
762 setValue(item, no);
763 break;
764 case Qt::Key_M:
765 setValue(item, mod);
766 break;
767 case Qt::Key_Y:
768 setValue(item, yes);
769 break;
770 default:
771 Parent::keyPressEvent(ev);
772 return;
773 }
774 ev->accept();
775}
776
777void ConfigList::mouseReleaseEvent(QMouseEvent* e)
778{
779 QPoint p = e->pos();
780 ConfigItem* item = (ConfigItem*)itemAt(p);
781 struct menu *menu;
782 enum prop_type ptype;
783 QIcon icon;
784 int idx, x;
785
786 if (!item)
787 goto skip;
788
789 menu = item->menu;
790 x = header()->offset() + p.x();
791 idx = header()->logicalIndexAt(x);
792 switch (idx) {
793 case promptColIdx:
794 icon = item->icon(promptColIdx);
795 if (!icon.isNull()) {
796 int off = header()->sectionPosition(0) + visualRect(indexAt(p)).x() + 4; // 4 is Hardcoded image offset. There might be a way to do it properly.
797 if (x >= off && x < off + icon.availableSizes().first().width()) {
798 if (item->goParent) {
799 emit parentSelected();
800 break;
801 } else if (!menu)
802 break;
803 ptype = menu->prompt ? menu->prompt->type : P_UNKNOWN;
804 if (ptype == P_MENU && rootEntry != menu &&
805 mode != fullMode && mode != menuMode &&
806 mode != listMode)
807 emit menuSelected(menu);
808 else
809 changeValue(item);
810 }
811 }
812 break;
813 case dataColIdx:
814 changeValue(item);
815 break;
816 }
817
818skip:
819 //printf("contentsMouseReleaseEvent: %d,%d\n", p.x(), p.y());
820 Parent::mouseReleaseEvent(e);
821}
822
823void ConfigList::mouseDoubleClickEvent(QMouseEvent* e)
824{
825 QPoint p = e->pos();
826 ConfigItem* item = (ConfigItem*)itemAt(p);
827 struct menu *menu;
828 enum prop_type ptype;
829
830 if (!item)
831 goto skip;
832 if (item->goParent) {
833 emit parentSelected();
834 goto skip;
835 }
836 menu = item->menu;
837 if (!menu)
838 goto skip;
839 ptype = menu->prompt ? menu->prompt->type : P_UNKNOWN;
840 if (ptype == P_MENU && mode != listMode) {
841 if (mode == singleMode)
842 emit itemSelected(menu);
843 else if (mode == symbolMode)
844 emit menuSelected(menu);
845 } else if (menu->sym)
846 changeValue(item);
847
848skip:
849 //printf("contentsMouseDoubleClickEvent: %d,%d\n", p.x(), p.y());
850 Parent::mouseDoubleClickEvent(e);
851}
852
853void ConfigList::focusInEvent(QFocusEvent *e)
854{
855 struct menu *menu = NULL;
856
857 Parent::focusInEvent(e);
858
859 ConfigItem* item = (ConfigItem *)currentItem();
860 if (item) {
861 setSelected(item, true);
862 menu = item->menu;
863 }
864 emit gotFocus(menu);
865}
866
867void ConfigList::contextMenuEvent(QContextMenuEvent *e)
868{
869 if (!headerPopup) {
870 QAction *action;
871
872 headerPopup = new QMenu(this);
873 action = new QAction("Show Name", this);
874 action->setCheckable(true);
875 connect(action, &QAction::toggled,
876 this, &ConfigList::setShowName);
877 connect(this, &ConfigList::showNameChanged,
878 action, &QAction::setChecked);
879 action->setChecked(showName);
880 headerPopup->addAction(action);
881 }
882
883 headerPopup->exec(e->globalPos());
884 e->accept();
885}
886
887void ConfigList::setShowName(bool on)
888{
889 if (showName == on)
890 return;
891
892 showName = on;
893 reinit();
894 emit showNameChanged(on);
895}
896
897QList<ConfigList *> ConfigList::allLists;
898QAction *ConfigList::showNormalAction;
899QAction *ConfigList::showAllAction;
900QAction *ConfigList::showPromptAction;
901
902void ConfigList::setAllOpen(bool open)
903{
904 QTreeWidgetItemIterator it(this);
905
906 while (*it) {
907 (*it)->setExpanded(open);
908
909 ++it;
910 }
911}
912
913ConfigInfoView::ConfigInfoView(QWidget* parent, const char *name)
914 : Parent(parent), sym(0), _menu(0)
915{
916 setObjectName(name);
917 setOpenLinks(false);
918
919 if (!objectName().isEmpty()) {
920 configSettings->beginGroup(objectName());
921 setShowDebug(configSettings->value("/showDebug", false).toBool());
922 configSettings->endGroup();
923 connect(configApp, &QApplication::aboutToQuit,
924 this, &ConfigInfoView::saveSettings);
925 }
926
927 contextMenu = createStandardContextMenu();
928 QAction *action = new QAction("Show Debug Info", contextMenu);
929
930 action->setCheckable(true);
931 connect(action, &QAction::toggled,
932 this, &ConfigInfoView::setShowDebug);
933 connect(this, &ConfigInfoView::showDebugChanged,
934 action, &QAction::setChecked);
935 action->setChecked(showDebug());
936 contextMenu->addSeparator();
937 contextMenu->addAction(action);
938}
939
940void ConfigInfoView::saveSettings(void)
941{
942 if (!objectName().isEmpty()) {
943 configSettings->beginGroup(objectName());
944 configSettings->setValue("/showDebug", showDebug());
945 configSettings->endGroup();
946 }
947}
948
949void ConfigInfoView::setShowDebug(bool b)
950{
951 if (_showDebug != b) {
952 _showDebug = b;
953 if (_menu)
954 menuInfo();
955 else if (sym)
956 symbolInfo();
957 emit showDebugChanged(b);
958 }
959}
960
961void ConfigInfoView::setInfo(struct menu *m)
962{
963 if (_menu == m)
964 return;
965 _menu = m;
966 sym = NULL;
967 if (!_menu)
968 clear();
969 else
970 menuInfo();
971}
972
973void ConfigInfoView::symbolInfo(void)
974{
975 QString str;
976
977 str += "<big>Symbol: <b>";
978 str += print_filter(sym->name);
979 str += "</b></big><br><br>value: ";
980 str += print_filter(sym_get_string_value(sym));
981 str += "<br>visibility: ";
982 str += sym->visible == yes ? "y" : sym->visible == mod ? "m" : "n";
983 str += "<br>";
984 str += debug_info(sym);
985
986 setText(str);
987}
988
989void ConfigInfoView::menuInfo(void)
990{
991 struct symbol* sym;
992 QString info;
993 QTextStream stream(&info);
994
995 sym = _menu->sym;
996 if (sym) {
997 if (_menu->prompt) {
998 stream << "<big><b>";
999 stream << print_filter(_menu->prompt->text);
1000 stream << "</b></big>";
1001 if (sym->name) {
1002 stream << " (";
1003 if (showDebug())
1004 stream << "<a href=\"" << sym->name << "\">";
1005 stream << print_filter(sym->name);
1006 if (showDebug())
1007 stream << "</a>";
1008 stream << ")";
1009 }
1010 } else if (sym->name) {
1011 stream << "<big><b>";
1012 if (showDebug())
1013 stream << "<a href=\"" << sym->name << "\">";
1014 stream << print_filter(sym->name);
1015 if (showDebug())
1016 stream << "</a>";
1017 stream << "</b></big>";
1018 }
1019 stream << "<br><br>";
1020
1021 if (showDebug())
1022 stream << debug_info(sym);
1023
1024 struct gstr help_gstr = str_new();
1025
1026 menu_get_ext_help(_menu, &help_gstr);
1027 stream << print_filter(str_get(&help_gstr));
1028 str_free(&help_gstr);
1029 } else if (_menu->prompt) {
1030 stream << "<big><b>";
1031 stream << print_filter(_menu->prompt->text);
1032 stream << "</b></big><br><br>";
1033 if (showDebug()) {
1034 if (_menu->prompt->visible.expr) {
1035 stream << " dep: ";
1036 expr_print(_menu->prompt->visible.expr,
1037 expr_print_help, &stream, E_NONE);
1038 stream << "<br><br>";
1039 }
1040
1041 stream << "defined at " << _menu->filename << ":"
1042 << _menu->lineno << "<br><br>";
1043 }
1044 }
1045
1046 setText(info);
1047}
1048
1049QString ConfigInfoView::debug_info(struct symbol *sym)
1050{
1051 QString debug;
1052 QTextStream stream(&debug);
1053
1054 stream << "type: ";
1055 stream << print_filter(sym_type_name(sym->type));
1056 if (sym_is_choice(sym))
1057 stream << " (choice)";
1058 debug += "<br>";
1059 if (sym->rev_dep.expr) {
1060 stream << "reverse dep: ";
1061 expr_print(sym->rev_dep.expr, expr_print_help, &stream, E_NONE);
1062 stream << "<br>";
1063 }
1064 for (struct property *prop = sym->prop; prop; prop = prop->next) {
1065 switch (prop->type) {
1066 case P_PROMPT:
1067 case P_MENU:
1068 stream << "prompt: ";
1069 stream << print_filter(prop->text);
1070 stream << "<br>";
1071 break;
1072 case P_DEFAULT:
1073 case P_SELECT:
1074 case P_RANGE:
1075 case P_COMMENT:
1076 case P_IMPLY:
1077 stream << prop_get_type_name(prop->type);
1078 stream << ": ";
1079 expr_print(prop->expr, expr_print_help,
1080 &stream, E_NONE);
1081 stream << "<br>";
1082 break;
1083 default:
1084 stream << "unknown property: ";
1085 stream << prop_get_type_name(prop->type);
1086 stream << "<br>";
1087 }
1088 if (prop->visible.expr) {
1089 stream << " dep: ";
1090 expr_print(prop->visible.expr, expr_print_help,
1091 &stream, E_NONE);
1092 stream << "<br>";
1093 }
1094 }
1095 stream << "<br>";
1096
1097 return debug;
1098}
1099
1100QString ConfigInfoView::print_filter(const QString &str)
1101{
1102 QRegularExpression re("[<>&\"\\n]");
1103 QString res = str;
1104
1105 QHash<QChar, QString> patterns;
1106 patterns['<'] = "<";
1107 patterns['>'] = ">";
1108 patterns['&'] = "&";
1109 patterns['"'] = """;
1110 patterns['\n'] = "<br>";
1111
1112 for (int i = 0; (i = res.indexOf(re, i)) >= 0;) {
1113 const QString n = patterns.value(res[i], QString());
1114 if (!n.isEmpty()) {
1115 res.replace(i, 1, n);
1116 i += n.length();
1117 }
1118 }
1119 return res;
1120}
1121
1122void ConfigInfoView::expr_print_help(void *data, struct symbol *sym, const char *str)
1123{
1124 QTextStream *stream = reinterpret_cast<QTextStream *>(data);
1125
1126 if (sym && sym->name && !(sym->flags & SYMBOL_CONST)) {
1127 *stream << "<a href=\"" << sym->name << "\">";
1128 *stream << print_filter(str);
1129 *stream << "</a>";
1130 } else {
1131 *stream << print_filter(str);
1132 }
1133}
1134
1135void ConfigInfoView::clicked(const QUrl &url)
1136{
1137 struct menu *m;
1138
1139 sym = sym_find(url.toEncoded().constData());
1140
1141 m = sym_get_prompt_menu(sym);
1142 if (!m) {
1143 /* Symbol is not visible as a menu */
1144 symbolInfo();
1145 emit showDebugChanged(true);
1146 } else {
1147 emit menuSelected(m);
1148 }
1149}
1150
1151void ConfigInfoView::contextMenuEvent(QContextMenuEvent *event)
1152{
1153 contextMenu->popup(event->globalPos());
1154 event->accept();
1155}
1156
1157ConfigSearchWindow::ConfigSearchWindow(ConfigMainWindow *parent)
1158 : Parent(parent), result(NULL)
1159{
1160 setObjectName("search");
1161 setWindowTitle("Search Config");
1162
1163 QVBoxLayout* layout1 = new QVBoxLayout(this);
1164 layout1->setContentsMargins(11, 11, 11, 11);
1165 layout1->setSpacing(6);
1166
1167 QHBoxLayout* layout2 = new QHBoxLayout();
1168 layout2->setContentsMargins(0, 0, 0, 0);
1169 layout2->setSpacing(6);
1170 layout2->addWidget(new QLabel("Find:", this));
1171 editField = new QLineEdit(this);
1172 connect(editField, &QLineEdit::returnPressed,
1173 this, &ConfigSearchWindow::search);
1174 layout2->addWidget(editField);
1175 searchButton = new QPushButton("Search", this);
1176 searchButton->setAutoDefault(false);
1177 connect(searchButton, &QPushButton::clicked,
1178 this, &ConfigSearchWindow::search);
1179 layout2->addWidget(searchButton);
1180 layout1->addLayout(layout2);
1181
1182 split = new QSplitter(Qt::Vertical, this);
1183 list = new ConfigList(split, "search");
1184 list->mode = listMode;
1185 info = new ConfigInfoView(split, "search");
1186 connect(list, &ConfigList::menuChanged,
1187 info, &ConfigInfoView::setInfo);
1188 connect(list, &ConfigList::menuChanged,
1189 parent, &ConfigMainWindow::setMenuLink);
1190
1191 layout1->addWidget(split);
1192
1193 QVariant x, y;
1194 int width, height;
1195 bool ok;
1196
1197 configSettings->beginGroup("search");
1198 width = configSettings->value("/window width", parent->width() / 2).toInt();
1199 height = configSettings->value("/window height", parent->height() / 2).toInt();
1200 resize(width, height);
1201 x = configSettings->value("/window x");
1202 y = configSettings->value("/window y");
1203 if (x.isValid() && y.isValid())
1204 move(x.toInt(), y.toInt());
1205 QList<int> sizes = configSettings->readSizes("/split", &ok);
1206 if (ok)
1207 split->setSizes(sizes);
1208 configSettings->endGroup();
1209 connect(configApp, &QApplication::aboutToQuit,
1210 this, &ConfigSearchWindow::saveSettings);
1211}
1212
1213void ConfigSearchWindow::saveSettings(void)
1214{
1215 if (!objectName().isEmpty()) {
1216 configSettings->beginGroup(objectName());
1217 configSettings->setValue("/window x", pos().x());
1218 configSettings->setValue("/window y", pos().y());
1219 configSettings->setValue("/window width", size().width());
1220 configSettings->setValue("/window height", size().height());
1221 configSettings->writeSizes("/split", split->sizes());
1222 configSettings->endGroup();
1223 }
1224}
1225
1226void ConfigSearchWindow::search(void)
1227{
1228 struct symbol **p;
1229 struct property *prop;
1230 ConfigItem *lastItem = NULL;
1231
1232 free(result);
1233 list->clear();
1234 info->clear();
1235
1236 result = sym_re_search(editField->text().toLatin1());
1237 if (!result)
1238 return;
1239 for (p = result; *p; p++) {
1240 for_all_prompts((*p), prop)
1241 lastItem = new ConfigItem(list, lastItem, prop->menu);
1242 }
1243}
1244
1245/*
1246 * Construct the complete config widget
1247 */
1248ConfigMainWindow::ConfigMainWindow(void)
1249 : searchWindow(0)
1250{
1251 bool ok = true;
1252 QVariant x, y;
1253 int width, height;
1254 char title[256];
1255
1256 snprintf(title, sizeof(title), "%s%s",
1257 rootmenu.prompt->text,
1258 ""
1259 );
1260 setWindowTitle(title);
1261
1262 QRect g = configApp->primaryScreen()->geometry();
1263 width = configSettings->value("/window width", g.width() - 64).toInt();
1264 height = configSettings->value("/window height", g.height() - 64).toInt();
1265 resize(width, height);
1266 x = configSettings->value("/window x");
1267 y = configSettings->value("/window y");
1268 if ((x.isValid())&&(y.isValid()))
1269 move(x.toInt(), y.toInt());
1270
1271 // set up icons
1272 ConfigItem::symbolYesIcon = QIcon(QPixmap(xpm_symbol_yes));
1273 ConfigItem::symbolModIcon = QIcon(QPixmap(xpm_symbol_mod));
1274 ConfigItem::symbolNoIcon = QIcon(QPixmap(xpm_symbol_no));
1275 ConfigItem::choiceYesIcon = QIcon(QPixmap(xpm_choice_yes));
1276 ConfigItem::choiceNoIcon = QIcon(QPixmap(xpm_choice_no));
1277 ConfigItem::menuIcon = QIcon(QPixmap(xpm_menu));
1278 ConfigItem::menubackIcon = QIcon(QPixmap(xpm_menuback));
1279
1280 QWidget *widget = new QWidget(this);
1281 setCentralWidget(widget);
1282
1283 QVBoxLayout *layout = new QVBoxLayout(widget);
1284
1285 split2 = new QSplitter(Qt::Vertical, widget);
1286 layout->addWidget(split2);
1287 split2->setChildrenCollapsible(false);
1288
1289 split1 = new QSplitter(Qt::Horizontal, split2);
1290 split1->setChildrenCollapsible(false);
1291
1292 configList = new ConfigList(split1, "config");
1293
1294 menuList = new ConfigList(split1, "menu");
1295
1296 helpText = new ConfigInfoView(split2, "help");
1297 setTabOrder(configList, helpText);
1298
1299 configList->setFocus();
1300
1301 backAction = new QAction(QPixmap(xpm_back), "Back", this);
1302 backAction->setShortcut(QKeySequence::Back);
1303 connect(backAction, &QAction::triggered,
1304 this, &ConfigMainWindow::goBack);
1305
1306 QAction *quitAction = new QAction("&Quit", this);
1307 quitAction->setShortcut(QKeySequence::Quit);
1308 connect(quitAction, &QAction::triggered,
1309 this, &ConfigMainWindow::close);
1310
1311 QAction *loadAction = new QAction(QPixmap(xpm_load), "&Open", this);
1312 loadAction->setShortcut(QKeySequence::Open);
1313 connect(loadAction, &QAction::triggered,
1314 this, &ConfigMainWindow::loadConfig);
1315
1316 saveAction = new QAction(QPixmap(xpm_save), "&Save", this);
1317 saveAction->setShortcut(QKeySequence::Save);
1318 connect(saveAction, &QAction::triggered,
1319 this, &ConfigMainWindow::saveConfig);
1320
1321 conf_set_changed_callback(conf_changed);
1322
1323 configname = conf_get_configname();
1324
1325 QAction *saveAsAction = new QAction("Save &As...", this);
1326 saveAsAction->setShortcut(QKeySequence::SaveAs);
1327 connect(saveAsAction, &QAction::triggered,
1328 this, &ConfigMainWindow::saveConfigAs);
1329 QAction *searchAction = new QAction("&Find", this);
1330 searchAction->setShortcut(QKeySequence::Find);
1331 connect(searchAction, &QAction::triggered,
1332 this, &ConfigMainWindow::searchConfig);
1333 singleViewAction = new QAction(QPixmap(xpm_single_view), "Single View", this);
1334 singleViewAction->setCheckable(true);
1335 connect(singleViewAction, &QAction::triggered,
1336 this, &ConfigMainWindow::showSingleView);
1337 splitViewAction = new QAction(QPixmap(xpm_split_view), "Split View", this);
1338 splitViewAction->setCheckable(true);
1339 connect(splitViewAction, &QAction::triggered,
1340 this, &ConfigMainWindow::showSplitView);
1341 fullViewAction = new QAction(QPixmap(xpm_tree_view), "Full View", this);
1342 fullViewAction->setCheckable(true);
1343 connect(fullViewAction, &QAction::triggered,
1344 this, &ConfigMainWindow::showFullView);
1345
1346 QAction *showNameAction = new QAction("Show Name", this);
1347 showNameAction->setCheckable(true);
1348 connect(showNameAction, &QAction::toggled,
1349 configList, &ConfigList::setShowName);
1350 showNameAction->setChecked(configList->showName);
1351
1352 QActionGroup *optGroup = new QActionGroup(this);
1353 optGroup->setExclusive(true);
1354 connect(optGroup, &QActionGroup::triggered,
1355 configList, &ConfigList::setOptionMode);
1356 connect(optGroup, &QActionGroup::triggered,
1357 menuList, &ConfigList::setOptionMode);
1358
1359 ConfigList::showNormalAction = new QAction("Show Normal Options", optGroup);
1360 ConfigList::showNormalAction->setCheckable(true);
1361 ConfigList::showAllAction = new QAction("Show All Options", optGroup);
1362 ConfigList::showAllAction->setCheckable(true);
1363 ConfigList::showPromptAction = new QAction("Show Prompt Options", optGroup);
1364 ConfigList::showPromptAction->setCheckable(true);
1365
1366 QAction *showDebugAction = new QAction("Show Debug Info", this);
1367 showDebugAction->setCheckable(true);
1368 connect(showDebugAction, &QAction::toggled,
1369 helpText, &ConfigInfoView::setShowDebug);
1370 showDebugAction->setChecked(helpText->showDebug());
1371
1372 QAction *showIntroAction = new QAction("Introduction", this);
1373 connect(showIntroAction, &QAction::triggered,
1374 this, &ConfigMainWindow::showIntro);
1375 QAction *showAboutAction = new QAction("About", this);
1376 connect(showAboutAction, &QAction::triggered,
1377 this, &ConfigMainWindow::showAbout);
1378
1379 // init tool bar
1380 QToolBar *toolBar = addToolBar("Tools");
1381 toolBar->addAction(backAction);
1382 toolBar->addSeparator();
1383 toolBar->addAction(loadAction);
1384 toolBar->addAction(saveAction);
1385 toolBar->addSeparator();
1386 toolBar->addAction(singleViewAction);
1387 toolBar->addAction(splitViewAction);
1388 toolBar->addAction(fullViewAction);
1389
1390 // create file menu
1391 QMenu *menu = menuBar()->addMenu("&File");
1392 menu->addAction(loadAction);
1393 menu->addAction(saveAction);
1394 menu->addAction(saveAsAction);
1395 menu->addSeparator();
1396 menu->addAction(quitAction);
1397
1398 // create edit menu
1399 menu = menuBar()->addMenu("&Edit");
1400 menu->addAction(searchAction);
1401
1402 // create options menu
1403 menu = menuBar()->addMenu("&Option");
1404 menu->addAction(showNameAction);
1405 menu->addSeparator();
1406 menu->addActions(optGroup->actions());
1407 menu->addSeparator();
1408 menu->addAction(showDebugAction);
1409
1410 // create help menu
1411 menu = menuBar()->addMenu("&Help");
1412 menu->addAction(showIntroAction);
1413 menu->addAction(showAboutAction);
1414
1415 connect(helpText, &ConfigInfoView::anchorClicked,
1416 helpText, &ConfigInfoView::clicked);
1417
1418 connect(configList, &ConfigList::menuChanged,
1419 helpText, &ConfigInfoView::setInfo);
1420 connect(configList, &ConfigList::menuSelected,
1421 this, &ConfigMainWindow::changeMenu);
1422 connect(configList, &ConfigList::itemSelected,
1423 this, &ConfigMainWindow::changeItens);
1424 connect(configList, &ConfigList::parentSelected,
1425 this, &ConfigMainWindow::goBack);
1426 connect(menuList, &ConfigList::menuChanged,
1427 helpText, &ConfigInfoView::setInfo);
1428 connect(menuList, &ConfigList::menuSelected,
1429 this, &ConfigMainWindow::changeMenu);
1430
1431 connect(configList, &ConfigList::gotFocus,
1432 helpText, &ConfigInfoView::setInfo);
1433 connect(menuList, &ConfigList::gotFocus,
1434 helpText, &ConfigInfoView::setInfo);
1435 connect(menuList, &ConfigList::gotFocus,
1436 this, &ConfigMainWindow::listFocusChanged);
1437 connect(helpText, &ConfigInfoView::menuSelected,
1438 this, &ConfigMainWindow::setMenuLink);
1439
1440 connect(configApp, &QApplication::aboutToQuit,
1441 this, &ConfigMainWindow::saveSettings);
1442
1443 conf_read(NULL);
1444
1445 QString listMode = configSettings->value("/listMode", "symbol").toString();
1446 if (listMode == "single")
1447 showSingleView();
1448 else if (listMode == "full")
1449 showFullView();
1450 else /*if (listMode == "split")*/
1451 showSplitView();
1452
1453 // UI setup done, restore splitter positions
1454 QList<int> sizes = configSettings->readSizes("/split1", &ok);
1455 if (ok)
1456 split1->setSizes(sizes);
1457
1458 sizes = configSettings->readSizes("/split2", &ok);
1459 if (ok)
1460 split2->setSizes(sizes);
1461}
1462
1463void ConfigMainWindow::loadConfig(void)
1464{
1465 QString str;
1466
1467 str = QFileDialog::getOpenFileName(this, "", configname);
1468 if (str.isNull())
1469 return;
1470
1471 if (conf_read(str.toLocal8Bit().constData()))
1472 QMessageBox::information(this, "qconf", "Unable to load configuration!");
1473
1474 configname = str;
1475
1476 ConfigList::updateListAllForAll();
1477}
1478
1479bool ConfigMainWindow::saveConfig(void)
1480{
1481 if (conf_write(configname.toLocal8Bit().constData())) {
1482 QMessageBox::information(this, "qconf", "Unable to save configuration!");
1483 return false;
1484 }
1485 conf_write_autoconf(0);
1486
1487 return true;
1488}
1489
1490void ConfigMainWindow::saveConfigAs(void)
1491{
1492 QString str;
1493
1494 str = QFileDialog::getSaveFileName(this, "", configname);
1495 if (str.isNull())
1496 return;
1497
1498 if (conf_write(str.toLocal8Bit().constData())) {
1499 QMessageBox::information(this, "qconf", "Unable to save configuration!");
1500 }
1501 conf_write_autoconf(0);
1502
1503 configname = str;
1504}
1505
1506void ConfigMainWindow::searchConfig(void)
1507{
1508 if (!searchWindow)
1509 searchWindow = new ConfigSearchWindow(this);
1510 searchWindow->show();
1511}
1512
1513void ConfigMainWindow::changeItens(struct menu *menu)
1514{
1515 configList->setRootMenu(menu);
1516}
1517
1518void ConfigMainWindow::changeMenu(struct menu *menu)
1519{
1520 menuList->setRootMenu(menu);
1521}
1522
1523void ConfigMainWindow::setMenuLink(struct menu *menu)
1524{
1525 struct menu *parent;
1526 ConfigList* list = NULL;
1527 ConfigItem* item;
1528
1529 if (configList->menuSkip(menu))
1530 return;
1531
1532 switch (configList->mode) {
1533 case singleMode:
1534 list = configList;
1535 parent = menu_get_parent_menu(menu);
1536 if (!parent)
1537 return;
1538 list->setRootMenu(parent);
1539 break;
1540 case menuMode:
1541 if (menu->flags & MENU_ROOT) {
1542 menuList->setRootMenu(menu);
1543 configList->clearSelection();
1544 list = configList;
1545 } else {
1546 parent = menu_get_parent_menu(menu->parent);
1547 if (!parent)
1548 return;
1549
1550 /* Select the config view */
1551 item = configList->findConfigItem(parent);
1552 if (item) {
1553 configList->setSelected(item, true);
1554 configList->scrollToItem(item);
1555 }
1556
1557 menuList->setRootMenu(parent);
1558 menuList->clearSelection();
1559 list = menuList;
1560 }
1561 break;
1562 case fullMode:
1563 list = configList;
1564 break;
1565 default:
1566 break;
1567 }
1568
1569 if (list) {
1570 item = list->findConfigItem(menu);
1571 if (item) {
1572 list->setSelected(item, true);
1573 list->scrollToItem(item);
1574 list->setFocus();
1575 helpText->setInfo(menu);
1576 }
1577 }
1578}
1579
1580void ConfigMainWindow::listFocusChanged(void)
1581{
1582 if (menuList->mode == menuMode)
1583 configList->clearSelection();
1584}
1585
1586void ConfigMainWindow::goBack(void)
1587{
1588 configList->setParentMenu();
1589}
1590
1591void ConfigMainWindow::showSingleView(void)
1592{
1593 singleViewAction->setEnabled(false);
1594 singleViewAction->setChecked(true);
1595 splitViewAction->setEnabled(true);
1596 splitViewAction->setChecked(false);
1597 fullViewAction->setEnabled(true);
1598 fullViewAction->setChecked(false);
1599
1600 backAction->setEnabled(true);
1601
1602 menuList->hide();
1603 menuList->setRootMenu(0);
1604 configList->mode = singleMode;
1605 if (configList->rootEntry == &rootmenu)
1606 configList->updateListAll();
1607 else
1608 configList->setRootMenu(&rootmenu);
1609 configList->setFocus();
1610}
1611
1612void ConfigMainWindow::showSplitView(void)
1613{
1614 singleViewAction->setEnabled(true);
1615 singleViewAction->setChecked(false);
1616 splitViewAction->setEnabled(false);
1617 splitViewAction->setChecked(true);
1618 fullViewAction->setEnabled(true);
1619 fullViewAction->setChecked(false);
1620
1621 backAction->setEnabled(false);
1622
1623 configList->mode = menuMode;
1624 if (configList->rootEntry == &rootmenu)
1625 configList->updateListAll();
1626 else
1627 configList->setRootMenu(&rootmenu);
1628 configList->setAllOpen(true);
1629 configApp->processEvents();
1630 menuList->mode = symbolMode;
1631 menuList->setRootMenu(&rootmenu);
1632 menuList->setAllOpen(true);
1633 menuList->show();
1634 menuList->setFocus();
1635}
1636
1637void ConfigMainWindow::showFullView(void)
1638{
1639 singleViewAction->setEnabled(true);
1640 singleViewAction->setChecked(false);
1641 splitViewAction->setEnabled(true);
1642 splitViewAction->setChecked(false);
1643 fullViewAction->setEnabled(false);
1644 fullViewAction->setChecked(true);
1645
1646 backAction->setEnabled(false);
1647
1648 menuList->hide();
1649 menuList->setRootMenu(0);
1650 configList->mode = fullMode;
1651 if (configList->rootEntry == &rootmenu)
1652 configList->updateListAll();
1653 else
1654 configList->setRootMenu(&rootmenu);
1655 configList->setFocus();
1656}
1657
1658/*
1659 * ask for saving configuration before quitting
1660 */
1661void ConfigMainWindow::closeEvent(QCloseEvent* e)
1662{
1663 if (!conf_get_changed()) {
1664 e->accept();
1665 return;
1666 }
1667
1668 QMessageBox mb(QMessageBox::Icon::Warning, "qconf",
1669 "Save configuration?");
1670
1671 QPushButton *yb = mb.addButton(QMessageBox::Yes);
1672 QPushButton *db = mb.addButton(QMessageBox::No);
1673 QPushButton *cb = mb.addButton(QMessageBox::Cancel);
1674
1675 yb->setText("&Save Changes");
1676 db->setText("&Discard Changes");
1677 cb->setText("Cancel Exit");
1678
1679 mb.setDefaultButton(yb);
1680 mb.setEscapeButton(cb);
1681
1682 switch (mb.exec()) {
1683 case QMessageBox::Yes:
1684 if (saveConfig())
1685 e->accept();
1686 else
1687 e->ignore();
1688 break;
1689 case QMessageBox::No:
1690 e->accept();
1691 break;
1692 case QMessageBox::Cancel:
1693 e->ignore();
1694 break;
1695 }
1696}
1697
1698void ConfigMainWindow::showIntro(void)
1699{
1700 static const QString str =
1701 "Welcome to the qconf graphical configuration tool.\n"
1702 "\n"
1703 "For bool and tristate options, a blank box indicates the "
1704 "feature is disabled, a check indicates it is enabled, and a "
1705 "dot indicates that it is to be compiled as a module. Clicking "
1706 "on the box will cycle through the three states. For int, hex, "
1707 "and string options, double-clicking or pressing F2 on the "
1708 "Value cell will allow you to edit the value.\n"
1709 "\n"
1710 "If you do not see an option (e.g., a device driver) that you "
1711 "believe should be present, try turning on Show All Options "
1712 "under the Options menu. Enabling Show Debug Info will help you"
1713 "figure out what other options must be enabled to support the "
1714 "option you are interested in, and hyperlinks will navigate to "
1715 "them.\n"
1716 "\n"
1717 "Toggling Show Debug Info under the Options menu will show the "
1718 "dependencies, which you can then match by examining other "
1719 "options.\n";
1720
1721 QMessageBox::information(this, "qconf", str);
1722}
1723
1724void ConfigMainWindow::showAbout(void)
1725{
1726 static const QString str = "qconf is Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>.\n"
1727 "Copyright (C) 2015 Boris Barbulovski <bbarbulovski@gmail.com>.\n"
1728 "\n"
1729 "Bug reports and feature request can also be entered at http://bugzilla.kernel.org/\n"
1730 "\n"
1731 "Qt Version: ";
1732
1733 QMessageBox::information(this, "qconf", str + qVersion());
1734}
1735
1736void ConfigMainWindow::saveSettings(void)
1737{
1738 configSettings->setValue("/window x", pos().x());
1739 configSettings->setValue("/window y", pos().y());
1740 configSettings->setValue("/window width", size().width());
1741 configSettings->setValue("/window height", size().height());
1742
1743 QString entry;
1744 switch(configList->mode) {
1745 case singleMode :
1746 entry = "single";
1747 break;
1748
1749 case symbolMode :
1750 entry = "split";
1751 break;
1752
1753 case fullMode :
1754 entry = "full";
1755 break;
1756
1757 default:
1758 break;
1759 }
1760 configSettings->setValue("/listMode", entry);
1761
1762 configSettings->writeSizes("/split1", split1->sizes());
1763 configSettings->writeSizes("/split2", split2->sizes());
1764}
1765
1766void ConfigMainWindow::conf_changed(bool dirty)
1767{
1768 if (saveAction)
1769 saveAction->setEnabled(dirty);
1770}
1771
1772void fixup_rootmenu(struct menu *menu)
1773{
1774 struct menu *child;
1775 static int menu_cnt = 0;
1776
1777 menu->flags |= MENU_ROOT;
1778 for (child = menu->list; child; child = child->next) {
1779 if (child->prompt && child->prompt->type == P_MENU) {
1780 menu_cnt++;
1781 fixup_rootmenu(child);
1782 menu_cnt--;
1783 } else if (!menu_cnt)
1784 fixup_rootmenu(child);
1785 }
1786}
1787
1788static const char *progname;
1789
1790static void usage(void)
1791{
1792 printf("%s [-s] <config>\n", progname);
1793 exit(0);
1794}
1795
1796int main(int ac, char** av)
1797{
1798 ConfigMainWindow* v;
1799 const char *name;
1800
1801 progname = av[0];
1802 if (ac > 1 && av[1][0] == '-') {
1803 switch (av[1][1]) {
1804 case 's':
1805 conf_set_message_callback(NULL);
1806 break;
1807 case 'h':
1808 case '?':
1809 usage();
1810 }
1811 name = av[2];
1812 } else
1813 name = av[1];
1814 if (!name)
1815 usage();
1816
1817 conf_parse(name);
1818 fixup_rootmenu(&rootmenu);
1819 //zconfdump(stdout);
1820
1821 configApp = new QApplication(ac, av);
1822
1823 configSettings = new ConfigSettings();
1824 configSettings->beginGroup("/kconfig/qconf");
1825 v = new ConfigMainWindow();
1826
1827 //zconfdump(stdout);
1828
1829 v->show();
1830 configApp->exec();
1831
1832 configSettings->endGroup();
1833 delete configSettings;
1834 delete v;
1835 delete configApp;
1836
1837 return 0;
1838}