Linux Audio

Check our new training course

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