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