Linux Audio

Check our new training course

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