mirror of
				git://git.openwrt.org/openwrt/openwrt.git
				synced 2025-10-25 11:04:28 -04:00 
			
		
		
		
	The main goal here is to keep this close to upstream. Changes include: - allow symbols implied by y to become m - make 'imply' obey the direct dependency - allow only 'config', 'comment', and 'if' inside 'choice' - qconf: make search fully work again on split mode - qconf: navigate menus on hyperlinks - remove '---help---' support - qconf: allow to edit "int", "hex", "string" menus in-place - qconf: drop Qt4 support - nconf: fix core dump when searching in empty menu - nconf: stop endless search loops - Create links to main menu items in search - fix segmentation fault in menuconfig search - nconf: Add search jump feature - port qconf to work with Qt6 in addition to Qt5 - fix possible buffer overflow - fix memory leak from range properties Signed-off-by: Eneas U de Queiroz <cotequeiroz@gmail.com>
		
			
				
	
	
		
			1929 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1929 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| /*
 | |
|  * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>
 | |
|  * Copyright (C) 2015 Boris Barbulovski <bbarbulovski@gmail.com>
 | |
|  */
 | |
| 
 | |
| #include <QAction>
 | |
| #include <QActionGroup>
 | |
| #include <QApplication>
 | |
| #include <QCloseEvent>
 | |
| #include <QDebug>
 | |
| #include <QFileDialog>
 | |
| #include <QLabel>
 | |
| #include <QLayout>
 | |
| #include <QList>
 | |
| #include <QMenu>
 | |
| #include <QMenuBar>
 | |
| #include <QMessageBox>
 | |
| #include <QRegularExpression>
 | |
| #include <QScreen>
 | |
| #include <QToolBar>
 | |
| 
 | |
| #include <stdlib.h>
 | |
| 
 | |
| #include "lkc.h"
 | |
| #include "qconf.h"
 | |
| 
 | |
| #include "images.h"
 | |
| 
 | |
| 
 | |
| static QApplication *configApp;
 | |
| static ConfigSettings *configSettings;
 | |
| 
 | |
| QAction *ConfigMainWindow::saveAction;
 | |
| 
 | |
| ConfigSettings::ConfigSettings()
 | |
| 	: QSettings("kernel.org", "qconf")
 | |
| {
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Reads a list of integer values from the application settings.
 | |
|  */
 | |
| QList<int> ConfigSettings::readSizes(const QString& key, bool *ok)
 | |
| {
 | |
| 	QList<int> result;
 | |
| 
 | |
| 	if (contains(key))
 | |
| 	{
 | |
| 		QStringList entryList = value(key).toStringList();
 | |
| 		QStringList::Iterator it;
 | |
| 
 | |
| 		for (it = entryList.begin(); it != entryList.end(); ++it)
 | |
| 			result.push_back((*it).toInt());
 | |
| 
 | |
| 		*ok = true;
 | |
| 	}
 | |
| 	else
 | |
| 		*ok = false;
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Writes a list of integer values to the application settings.
 | |
|  */
 | |
| bool ConfigSettings::writeSizes(const QString& key, const QList<int>& value)
 | |
| {
 | |
| 	QStringList stringList;
 | |
| 	QList<int>::ConstIterator it;
 | |
| 
 | |
| 	for (it = value.begin(); it != value.end(); ++it)
 | |
| 		stringList.push_back(QString::number(*it));
 | |
| 	setValue(key, stringList);
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| QIcon ConfigItem::symbolYesIcon;
 | |
| QIcon ConfigItem::symbolModIcon;
 | |
| QIcon ConfigItem::symbolNoIcon;
 | |
| QIcon ConfigItem::choiceYesIcon;
 | |
| QIcon ConfigItem::choiceNoIcon;
 | |
| QIcon ConfigItem::menuIcon;
 | |
| QIcon ConfigItem::menubackIcon;
 | |
| 
 | |
| /*
 | |
|  * update the displayed of a menu entry
 | |
|  */
 | |
| void ConfigItem::updateMenu(void)
 | |
| {
 | |
| 	ConfigList* list;
 | |
| 	struct symbol* sym;
 | |
| 	struct property *prop;
 | |
| 	QString prompt;
 | |
| 	int type;
 | |
| 	tristate expr;
 | |
| 
 | |
| 	list = listView();
 | |
| 	if (goParent) {
 | |
| 		setIcon(promptColIdx, menubackIcon);
 | |
| 		prompt = "..";
 | |
| 		goto set_prompt;
 | |
| 	}
 | |
| 
 | |
| 	sym = menu->sym;
 | |
| 	prop = menu->prompt;
 | |
| 	prompt = menu_get_prompt(menu);
 | |
| 
 | |
| 	if (prop) switch (prop->type) {
 | |
| 	case P_MENU:
 | |
| 		if (list->mode == singleMode || list->mode == symbolMode) {
 | |
| 			/* a menuconfig entry is displayed differently
 | |
| 			 * depending whether it's at the view root or a child.
 | |
| 			 */
 | |
| 			if (sym && list->rootEntry == menu)
 | |
| 				break;
 | |
| 			setIcon(promptColIdx, menuIcon);
 | |
| 		} else {
 | |
| 			if (sym)
 | |
| 				break;
 | |
| 			setIcon(promptColIdx, QIcon());
 | |
| 		}
 | |
| 		goto set_prompt;
 | |
| 	case P_COMMENT:
 | |
| 		setIcon(promptColIdx, QIcon());
 | |
| 		prompt = "*** " + prompt + " ***";
 | |
| 		goto set_prompt;
 | |
| 	default:
 | |
| 		;
 | |
| 	}
 | |
| 	if (!sym)
 | |
| 		goto set_prompt;
 | |
| 
 | |
| 	setText(nameColIdx, sym->name);
 | |
| 
 | |
| 	type = sym_get_type(sym);
 | |
| 	switch (type) {
 | |
| 	case S_BOOLEAN:
 | |
| 	case S_TRISTATE:
 | |
| 		char ch;
 | |
| 
 | |
| 		if (!sym_is_changeable(sym) && list->optMode == normalOpt) {
 | |
| 			setIcon(promptColIdx, QIcon());
 | |
| 			break;
 | |
| 		}
 | |
| 		expr = sym_get_tristate_value(sym);
 | |
| 		switch (expr) {
 | |
| 		case yes:
 | |
| 			if (sym_is_choice_value(sym) && type == S_BOOLEAN)
 | |
| 				setIcon(promptColIdx, choiceYesIcon);
 | |
| 			else
 | |
| 				setIcon(promptColIdx, symbolYesIcon);
 | |
| 			ch = 'Y';
 | |
| 			break;
 | |
| 		case mod:
 | |
| 			setIcon(promptColIdx, symbolModIcon);
 | |
| 			ch = 'M';
 | |
| 			break;
 | |
| 		default:
 | |
| 			if (sym_is_choice_value(sym) && type == S_BOOLEAN)
 | |
| 				setIcon(promptColIdx, choiceNoIcon);
 | |
| 			else
 | |
| 				setIcon(promptColIdx, symbolNoIcon);
 | |
| 			ch = 'N';
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		setText(dataColIdx, QChar(ch));
 | |
| 		break;
 | |
| 	case S_INT:
 | |
| 	case S_HEX:
 | |
| 	case S_STRING:
 | |
| 		setText(dataColIdx, sym_get_string_value(sym));
 | |
| 		break;
 | |
| 	}
 | |
| 	if (!sym_has_value(sym) && visible)
 | |
| 		prompt += " (NEW)";
 | |
| set_prompt:
 | |
| 	setText(promptColIdx, prompt);
 | |
| }
 | |
| 
 | |
| void ConfigItem::testUpdateMenu(bool v)
 | |
| {
 | |
| 	ConfigItem* i;
 | |
| 
 | |
| 	visible = v;
 | |
| 	if (!menu)
 | |
| 		return;
 | |
| 
 | |
| 	sym_calc_value(menu->sym);
 | |
| 	if (menu->flags & MENU_CHANGED) {
 | |
| 		/* the menu entry changed, so update all list items */
 | |
| 		menu->flags &= ~MENU_CHANGED;
 | |
| 		for (i = (ConfigItem*)menu->data; i; i = i->nextItem)
 | |
| 			i->updateMenu();
 | |
| 	} else if (listView()->updateAll)
 | |
| 		updateMenu();
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * construct a menu entry
 | |
|  */
 | |
| void ConfigItem::init(void)
 | |
| {
 | |
| 	if (menu) {
 | |
| 		ConfigList* list = listView();
 | |
| 		nextItem = (ConfigItem*)menu->data;
 | |
| 		menu->data = this;
 | |
| 
 | |
| 		if (list->mode != fullMode)
 | |
| 			setExpanded(true);
 | |
| 		sym_calc_value(menu->sym);
 | |
| 
 | |
| 		if (menu->sym) {
 | |
| 			enum symbol_type type = menu->sym->type;
 | |
| 
 | |
| 			// Allow to edit "int", "hex", and "string" in-place in
 | |
| 			// the data column. Unfortunately, you cannot specify
 | |
| 			// the flags per column. Set ItemIsEditable for all
 | |
| 			// columns here, and check the column in createEditor().
 | |
| 			if (type == S_INT || type == S_HEX || type == S_STRING)
 | |
| 				setFlags(flags() | Qt::ItemIsEditable);
 | |
| 		}
 | |
| 	}
 | |
| 	updateMenu();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * destruct a menu entry
 | |
|  */
 | |
| ConfigItem::~ConfigItem(void)
 | |
| {
 | |
| 	if (menu) {
 | |
| 		ConfigItem** ip = (ConfigItem**)&menu->data;
 | |
| 		for (; *ip; ip = &(*ip)->nextItem) {
 | |
| 			if (*ip == this) {
 | |
| 				*ip = nextItem;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QWidget *ConfigItemDelegate::createEditor(QWidget *parent,
 | |
| 					  const QStyleOptionViewItem &option,
 | |
| 					  const QModelIndex &index) const
 | |
| {
 | |
| 	ConfigItem *item;
 | |
| 
 | |
| 	// Only the data column is editable
 | |
| 	if (index.column() != dataColIdx)
 | |
| 		return nullptr;
 | |
| 
 | |
| 	// You cannot edit invisible menus
 | |
| 	item = static_cast<ConfigItem *>(index.internalPointer());
 | |
| 	if (!item || !item->menu || !menu_is_visible(item->menu))
 | |
| 		return nullptr;
 | |
| 
 | |
| 	return QStyledItemDelegate::createEditor(parent, option, index);
 | |
| }
 | |
| 
 | |
| void ConfigItemDelegate::setModelData(QWidget *editor,
 | |
| 				      QAbstractItemModel *model,
 | |
| 				      const QModelIndex &index) const
 | |
| {
 | |
| 	QLineEdit *lineEdit;
 | |
| 	ConfigItem *item;
 | |
| 	struct symbol *sym;
 | |
| 	bool success;
 | |
| 
 | |
| 	lineEdit = qobject_cast<QLineEdit *>(editor);
 | |
| 	// If this is not a QLineEdit, use the parent's default.
 | |
| 	// (does this happen?)
 | |
| 	if (!lineEdit)
 | |
| 		goto parent;
 | |
| 
 | |
| 	item = static_cast<ConfigItem *>(index.internalPointer());
 | |
| 	if (!item || !item->menu)
 | |
| 		goto parent;
 | |
| 
 | |
| 	sym = item->menu->sym;
 | |
| 	if (!sym)
 | |
| 		goto parent;
 | |
| 
 | |
| 	success = sym_set_string_value(sym, lineEdit->text().toUtf8().data());
 | |
| 	if (success) {
 | |
| 		ConfigList::updateListForAll();
 | |
| 	} else {
 | |
| 		QMessageBox::information(editor, "qconf",
 | |
| 			"Cannot set the data (maybe due to out of range).\n"
 | |
| 			"Setting the old value.");
 | |
| 		lineEdit->setText(sym_get_string_value(sym));
 | |
| 	}
 | |
| 
 | |
| parent:
 | |
| 	QStyledItemDelegate::setModelData(editor, model, index);
 | |
| }
 | |
| 
 | |
| ConfigList::ConfigList(QWidget *parent, const char *name)
 | |
| 	: QTreeWidget(parent),
 | |
| 	  updateAll(false),
 | |
| 	  showName(false), mode(singleMode), optMode(normalOpt),
 | |
| 	  rootEntry(0), headerPopup(0)
 | |
| {
 | |
| 	setObjectName(name);
 | |
| 	setSortingEnabled(false);
 | |
| 	setRootIsDecorated(true);
 | |
| 
 | |
| 	setVerticalScrollMode(ScrollPerPixel);
 | |
| 	setHorizontalScrollMode(ScrollPerPixel);
 | |
| 
 | |
| 	setHeaderLabels(QStringList() << "Option" << "Name" << "Value");
 | |
| 
 | |
| 	connect(this, &ConfigList::itemSelectionChanged,
 | |
| 		this, &ConfigList::updateSelection);
 | |
| 
 | |
| 	if (name) {
 | |
| 		configSettings->beginGroup(name);
 | |
| 		showName = configSettings->value("/showName", false).toBool();
 | |
| 		optMode = (enum optionMode)configSettings->value("/optionMode", 0).toInt();
 | |
| 		configSettings->endGroup();
 | |
| 		connect(configApp, &QApplication::aboutToQuit,
 | |
| 			this, &ConfigList::saveSettings);
 | |
| 	}
 | |
| 
 | |
| 	showColumn(promptColIdx);
 | |
| 
 | |
| 	setItemDelegate(new ConfigItemDelegate(this));
 | |
| 
 | |
| 	allLists.append(this);
 | |
| 
 | |
| 	reinit();
 | |
| }
 | |
| 
 | |
| ConfigList::~ConfigList()
 | |
| {
 | |
| 	allLists.removeOne(this);
 | |
| }
 | |
| 
 | |
| bool ConfigList::menuSkip(struct menu *menu)
 | |
| {
 | |
| 	if (optMode == normalOpt && menu_is_visible(menu))
 | |
| 		return false;
 | |
| 	if (optMode == promptOpt && menu_has_prompt(menu))
 | |
| 		return false;
 | |
| 	if (optMode == allOpt)
 | |
| 		return false;
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void ConfigList::reinit(void)
 | |
| {
 | |
| 	hideColumn(nameColIdx);
 | |
| 
 | |
| 	if (showName)
 | |
| 		showColumn(nameColIdx);
 | |
| 
 | |
| 	updateListAll();
 | |
| }
 | |
| 
 | |
| void ConfigList::setOptionMode(QAction *action)
 | |
| {
 | |
| 	if (action == showNormalAction)
 | |
| 		optMode = normalOpt;
 | |
| 	else if (action == showAllAction)
 | |
| 		optMode = allOpt;
 | |
| 	else
 | |
| 		optMode = promptOpt;
 | |
| 
 | |
| 	updateListAll();
 | |
| }
 | |
| 
 | |
| void ConfigList::saveSettings(void)
 | |
| {
 | |
| 	if (!objectName().isEmpty()) {
 | |
| 		configSettings->beginGroup(objectName());
 | |
| 		configSettings->setValue("/showName", showName);
 | |
| 		configSettings->setValue("/optionMode", (int)optMode);
 | |
| 		configSettings->endGroup();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ConfigItem* ConfigList::findConfigItem(struct menu *menu)
 | |
| {
 | |
| 	ConfigItem* item = (ConfigItem*)menu->data;
 | |
| 
 | |
| 	for (; item; item = item->nextItem) {
 | |
| 		if (this == item->listView())
 | |
| 			break;
 | |
| 	}
 | |
| 
 | |
| 	return item;
 | |
| }
 | |
| 
 | |
| void ConfigList::updateSelection(void)
 | |
| {
 | |
| 	struct menu *menu;
 | |
| 	enum prop_type type;
 | |
| 
 | |
| 	if (selectedItems().count() == 0)
 | |
| 		return;
 | |
| 
 | |
| 	ConfigItem* item = (ConfigItem*)selectedItems().first();
 | |
| 	if (!item)
 | |
| 		return;
 | |
| 
 | |
| 	menu = item->menu;
 | |
| 	emit menuChanged(menu);
 | |
| 	if (!menu)
 | |
| 		return;
 | |
| 	type = menu->prompt ? menu->prompt->type : P_UNKNOWN;
 | |
| 	if (mode == menuMode && type == P_MENU)
 | |
| 		emit menuSelected(menu);
 | |
| }
 | |
| 
 | |
| void ConfigList::updateList()
 | |
| {
 | |
| 	ConfigItem* last = 0;
 | |
| 	ConfigItem *item;
 | |
| 
 | |
| 	if (!rootEntry) {
 | |
| 		if (mode != listMode)
 | |
| 			goto update;
 | |
| 		QTreeWidgetItemIterator it(this);
 | |
| 
 | |
| 		while (*it) {
 | |
| 			item = (ConfigItem*)(*it);
 | |
| 			if (!item->menu)
 | |
| 				continue;
 | |
| 			item->testUpdateMenu(menu_is_visible(item->menu));
 | |
| 
 | |
| 			++it;
 | |
| 		}
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (rootEntry != &rootmenu && (mode == singleMode ||
 | |
| 	    (mode == symbolMode && rootEntry->parent != &rootmenu))) {
 | |
| 		item = (ConfigItem *)topLevelItem(0);
 | |
| 		if (!item)
 | |
| 			item = new ConfigItem(this, 0, true);
 | |
| 		last = item;
 | |
| 	}
 | |
| 	if ((mode == singleMode || (mode == symbolMode && !(rootEntry->flags & MENU_ROOT))) &&
 | |
| 	    rootEntry->sym && rootEntry->prompt) {
 | |
| 		item = last ? last->nextSibling() : nullptr;
 | |
| 		if (!item)
 | |
| 			item = new ConfigItem(this, last, rootEntry, true);
 | |
| 		else
 | |
| 			item->testUpdateMenu(true);
 | |
| 
 | |
| 		updateMenuList(item, rootEntry);
 | |
| 		update();
 | |
| 		resizeColumnToContents(0);
 | |
| 		return;
 | |
| 	}
 | |
| update:
 | |
| 	updateMenuList(rootEntry);
 | |
| 	update();
 | |
| 	resizeColumnToContents(0);
 | |
| }
 | |
| 
 | |
| void ConfigList::updateListForAll()
 | |
| {
 | |
| 	QListIterator<ConfigList *> it(allLists);
 | |
| 
 | |
| 	while (it.hasNext()) {
 | |
| 		ConfigList *list = it.next();
 | |
| 
 | |
| 		list->updateList();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigList::updateListAllForAll()
 | |
| {
 | |
| 	QListIterator<ConfigList *> it(allLists);
 | |
| 
 | |
| 	while (it.hasNext()) {
 | |
| 		ConfigList *list = it.next();
 | |
| 
 | |
| 		list->updateList();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigList::setValue(ConfigItem* item, tristate val)
 | |
| {
 | |
| 	struct symbol* sym;
 | |
| 	int type;
 | |
| 	tristate oldval;
 | |
| 
 | |
| 	sym = item->menu ? item->menu->sym : 0;
 | |
| 	if (!sym)
 | |
| 		return;
 | |
| 
 | |
| 	type = sym_get_type(sym);
 | |
| 	switch (type) {
 | |
| 	case S_BOOLEAN:
 | |
| 	case S_TRISTATE:
 | |
| 		oldval = sym_get_tristate_value(sym);
 | |
| 
 | |
| 		if (!sym_set_tristate_value(sym, val))
 | |
| 			return;
 | |
| 		if (oldval == no && item->menu->list)
 | |
| 			item->setExpanded(true);
 | |
| 		ConfigList::updateListForAll();
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigList::changeValue(ConfigItem* item)
 | |
| {
 | |
| 	struct symbol* sym;
 | |
| 	struct menu* menu;
 | |
| 	int type, oldexpr, newexpr;
 | |
| 
 | |
| 	menu = item->menu;
 | |
| 	if (!menu)
 | |
| 		return;
 | |
| 	sym = menu->sym;
 | |
| 	if (!sym) {
 | |
| 		if (item->menu->list)
 | |
| 			item->setExpanded(!item->isExpanded());
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	type = sym_get_type(sym);
 | |
| 	switch (type) {
 | |
| 	case S_BOOLEAN:
 | |
| 	case S_TRISTATE:
 | |
| 		oldexpr = sym_get_tristate_value(sym);
 | |
| 		newexpr = sym_toggle_tristate_value(sym);
 | |
| 		if (item->menu->list) {
 | |
| 			if (oldexpr == newexpr)
 | |
| 				item->setExpanded(!item->isExpanded());
 | |
| 			else if (oldexpr == no)
 | |
| 				item->setExpanded(true);
 | |
| 		}
 | |
| 		if (oldexpr != newexpr)
 | |
| 			ConfigList::updateListForAll();
 | |
| 		break;
 | |
| 	default:
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigList::setRootMenu(struct menu *menu)
 | |
| {
 | |
| 	enum prop_type type;
 | |
| 
 | |
| 	if (rootEntry == menu)
 | |
| 		return;
 | |
| 	type = menu && menu->prompt ? menu->prompt->type : P_UNKNOWN;
 | |
| 	if (type != P_MENU)
 | |
| 		return;
 | |
| 	updateMenuList(0);
 | |
| 	rootEntry = menu;
 | |
| 	updateListAll();
 | |
| 	if (currentItem()) {
 | |
| 		setSelected(currentItem(), hasFocus());
 | |
| 		scrollToItem(currentItem());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigList::setParentMenu(void)
 | |
| {
 | |
| 	ConfigItem* item;
 | |
| 	struct menu *oldroot;
 | |
| 
 | |
| 	oldroot = rootEntry;
 | |
| 	if (rootEntry == &rootmenu)
 | |
| 		return;
 | |
| 	setRootMenu(menu_get_parent_menu(rootEntry->parent));
 | |
| 
 | |
| 	QTreeWidgetItemIterator it(this);
 | |
| 	while (*it) {
 | |
| 		item = (ConfigItem *)(*it);
 | |
| 		if (item->menu == oldroot) {
 | |
| 			setCurrentItem(item);
 | |
| 			scrollToItem(item);
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		++it;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * update all the children of a menu entry
 | |
|  *   removes/adds the entries from the parent widget as necessary
 | |
|  *
 | |
|  * parent: either the menu list widget or a menu entry widget
 | |
|  * menu: entry to be updated
 | |
|  */
 | |
| void ConfigList::updateMenuList(ConfigItem *parent, struct menu* menu)
 | |
| {
 | |
| 	struct menu* child;
 | |
| 	ConfigItem* item;
 | |
| 	ConfigItem* last;
 | |
| 	bool visible;
 | |
| 	enum prop_type type;
 | |
| 
 | |
| 	if (!menu) {
 | |
| 		while (parent->childCount() > 0)
 | |
| 		{
 | |
| 			delete parent->takeChild(0);
 | |
| 		}
 | |
| 
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	last = parent->firstChild();
 | |
| 	if (last && !last->goParent)
 | |
| 		last = 0;
 | |
| 	for (child = menu->list; child; child = child->next) {
 | |
| 		item = last ? last->nextSibling() : parent->firstChild();
 | |
| 		type = child->prompt ? child->prompt->type : P_UNKNOWN;
 | |
| 
 | |
| 		switch (mode) {
 | |
| 		case menuMode:
 | |
| 			if (!(child->flags & MENU_ROOT))
 | |
| 				goto hide;
 | |
| 			break;
 | |
| 		case symbolMode:
 | |
| 			if (child->flags & MENU_ROOT)
 | |
| 				goto hide;
 | |
| 			break;
 | |
| 		default:
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		visible = menu_is_visible(child);
 | |
| 		if (!menuSkip(child)) {
 | |
| 			if (!child->sym && !child->list && !child->prompt)
 | |
| 				continue;
 | |
| 			if (!item || item->menu != child)
 | |
| 				item = new ConfigItem(parent, last, child, visible);
 | |
| 			else
 | |
| 				item->testUpdateMenu(visible);
 | |
| 
 | |
| 			if (mode == fullMode || mode == menuMode || type != P_MENU)
 | |
| 				updateMenuList(item, child);
 | |
| 			else
 | |
| 				updateMenuList(item, 0);
 | |
| 			last = item;
 | |
| 			continue;
 | |
| 		}
 | |
| hide:
 | |
| 		if (item && item->menu == child) {
 | |
| 			last = parent->firstChild();
 | |
| 			if (last == item)
 | |
| 				last = 0;
 | |
| 			else while (last->nextSibling() != item)
 | |
| 				last = last->nextSibling();
 | |
| 			delete item;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigList::updateMenuList(struct menu *menu)
 | |
| {
 | |
| 	struct menu* child;
 | |
| 	ConfigItem* item;
 | |
| 	ConfigItem* last;
 | |
| 	bool visible;
 | |
| 	enum prop_type type;
 | |
| 
 | |
| 	if (!menu) {
 | |
| 		while (topLevelItemCount() > 0)
 | |
| 		{
 | |
| 			delete takeTopLevelItem(0);
 | |
| 		}
 | |
| 
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	last = (ConfigItem *)topLevelItem(0);
 | |
| 	if (last && !last->goParent)
 | |
| 		last = 0;
 | |
| 	for (child = menu->list; child; child = child->next) {
 | |
| 		item = last ? last->nextSibling() : (ConfigItem *)topLevelItem(0);
 | |
| 		type = child->prompt ? child->prompt->type : P_UNKNOWN;
 | |
| 
 | |
| 		switch (mode) {
 | |
| 		case menuMode:
 | |
| 			if (!(child->flags & MENU_ROOT))
 | |
| 				goto hide;
 | |
| 			break;
 | |
| 		case symbolMode:
 | |
| 			if (child->flags & MENU_ROOT)
 | |
| 				goto hide;
 | |
| 			break;
 | |
| 		default:
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		visible = menu_is_visible(child);
 | |
| 		if (!menuSkip(child)) {
 | |
| 			if (!child->sym && !child->list && !child->prompt)
 | |
| 				continue;
 | |
| 			if (!item || item->menu != child)
 | |
| 				item = new ConfigItem(this, last, child, visible);
 | |
| 			else
 | |
| 				item->testUpdateMenu(visible);
 | |
| 
 | |
| 			if (mode == fullMode || mode == menuMode || type != P_MENU)
 | |
| 				updateMenuList(item, child);
 | |
| 			else
 | |
| 				updateMenuList(item, 0);
 | |
| 			last = item;
 | |
| 			continue;
 | |
| 		}
 | |
| hide:
 | |
| 		if (item && item->menu == child) {
 | |
| 			last = (ConfigItem *)topLevelItem(0);
 | |
| 			if (last == item)
 | |
| 				last = 0;
 | |
| 			else while (last->nextSibling() != item)
 | |
| 				last = last->nextSibling();
 | |
| 			delete item;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigList::keyPressEvent(QKeyEvent* ev)
 | |
| {
 | |
| 	QTreeWidgetItem* i = currentItem();
 | |
| 	ConfigItem* item;
 | |
| 	struct menu *menu;
 | |
| 	enum prop_type type;
 | |
| 
 | |
| 	if (ev->key() == Qt::Key_Escape && mode != fullMode && mode != listMode) {
 | |
| 		emit parentSelected();
 | |
| 		ev->accept();
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	if (!i) {
 | |
| 		Parent::keyPressEvent(ev);
 | |
| 		return;
 | |
| 	}
 | |
| 	item = (ConfigItem*)i;
 | |
| 
 | |
| 	switch (ev->key()) {
 | |
| 	case Qt::Key_Return:
 | |
| 	case Qt::Key_Enter:
 | |
| 		if (item->goParent) {
 | |
| 			emit parentSelected();
 | |
| 			break;
 | |
| 		}
 | |
| 		menu = item->menu;
 | |
| 		if (!menu)
 | |
| 			break;
 | |
| 		type = menu->prompt ? menu->prompt->type : P_UNKNOWN;
 | |
| 		if (type == P_MENU && rootEntry != menu &&
 | |
| 		    mode != fullMode && mode != menuMode) {
 | |
| 			if (mode == menuMode)
 | |
| 				emit menuSelected(menu);
 | |
| 			else
 | |
| 				emit itemSelected(menu);
 | |
| 			break;
 | |
| 		}
 | |
| 	case Qt::Key_Space:
 | |
| 		changeValue(item);
 | |
| 		break;
 | |
| 	case Qt::Key_N:
 | |
| 		setValue(item, no);
 | |
| 		break;
 | |
| 	case Qt::Key_M:
 | |
| 		setValue(item, mod);
 | |
| 		break;
 | |
| 	case Qt::Key_Y:
 | |
| 		setValue(item, yes);
 | |
| 		break;
 | |
| 	default:
 | |
| 		Parent::keyPressEvent(ev);
 | |
| 		return;
 | |
| 	}
 | |
| 	ev->accept();
 | |
| }
 | |
| 
 | |
| void ConfigList::mousePressEvent(QMouseEvent* e)
 | |
| {
 | |
| 	//QPoint p(contentsToViewport(e->pos()));
 | |
| 	//printf("contentsMousePressEvent: %d,%d\n", p.x(), p.y());
 | |
| 	Parent::mousePressEvent(e);
 | |
| }
 | |
| 
 | |
| void ConfigList::mouseReleaseEvent(QMouseEvent* e)
 | |
| {
 | |
| 	QPoint p = e->pos();
 | |
| 	ConfigItem* item = (ConfigItem*)itemAt(p);
 | |
| 	struct menu *menu;
 | |
| 	enum prop_type ptype;
 | |
| 	QIcon icon;
 | |
| 	int idx, x;
 | |
| 
 | |
| 	if (!item)
 | |
| 		goto skip;
 | |
| 
 | |
| 	menu = item->menu;
 | |
| 	x = header()->offset() + p.x();
 | |
| 	idx = header()->logicalIndexAt(x);
 | |
| 	switch (idx) {
 | |
| 	case promptColIdx:
 | |
| 		icon = item->icon(promptColIdx);
 | |
| 		if (!icon.isNull()) {
 | |
| 			int off = header()->sectionPosition(0) + visualRect(indexAt(p)).x() + 4; // 4 is Hardcoded image offset. There might be a way to do it properly.
 | |
| 			if (x >= off && x < off + icon.availableSizes().first().width()) {
 | |
| 				if (item->goParent) {
 | |
| 					emit parentSelected();
 | |
| 					break;
 | |
| 				} else if (!menu)
 | |
| 					break;
 | |
| 				ptype = menu->prompt ? menu->prompt->type : P_UNKNOWN;
 | |
| 				if (ptype == P_MENU && rootEntry != menu &&
 | |
| 				    mode != fullMode && mode != menuMode &&
 | |
|                                     mode != listMode)
 | |
| 					emit menuSelected(menu);
 | |
| 				else
 | |
| 					changeValue(item);
 | |
| 			}
 | |
| 		}
 | |
| 		break;
 | |
| 	case dataColIdx:
 | |
| 		changeValue(item);
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| skip:
 | |
| 	//printf("contentsMouseReleaseEvent: %d,%d\n", p.x(), p.y());
 | |
| 	Parent::mouseReleaseEvent(e);
 | |
| }
 | |
| 
 | |
| void ConfigList::mouseMoveEvent(QMouseEvent* e)
 | |
| {
 | |
| 	//QPoint p(contentsToViewport(e->pos()));
 | |
| 	//printf("contentsMouseMoveEvent: %d,%d\n", p.x(), p.y());
 | |
| 	Parent::mouseMoveEvent(e);
 | |
| }
 | |
| 
 | |
| void ConfigList::mouseDoubleClickEvent(QMouseEvent* e)
 | |
| {
 | |
| 	QPoint p = e->pos();
 | |
| 	ConfigItem* item = (ConfigItem*)itemAt(p);
 | |
| 	struct menu *menu;
 | |
| 	enum prop_type ptype;
 | |
| 
 | |
| 	if (!item)
 | |
| 		goto skip;
 | |
| 	if (item->goParent) {
 | |
| 		emit parentSelected();
 | |
| 		goto skip;
 | |
| 	}
 | |
| 	menu = item->menu;
 | |
| 	if (!menu)
 | |
| 		goto skip;
 | |
| 	ptype = menu->prompt ? menu->prompt->type : P_UNKNOWN;
 | |
| 	if (ptype == P_MENU && mode != listMode) {
 | |
| 		if (mode == singleMode)
 | |
| 			emit itemSelected(menu);
 | |
| 		else if (mode == symbolMode)
 | |
| 			emit menuSelected(menu);
 | |
| 	} else if (menu->sym)
 | |
| 		changeValue(item);
 | |
| 
 | |
| skip:
 | |
| 	//printf("contentsMouseDoubleClickEvent: %d,%d\n", p.x(), p.y());
 | |
| 	Parent::mouseDoubleClickEvent(e);
 | |
| }
 | |
| 
 | |
| void ConfigList::focusInEvent(QFocusEvent *e)
 | |
| {
 | |
| 	struct menu *menu = NULL;
 | |
| 
 | |
| 	Parent::focusInEvent(e);
 | |
| 
 | |
| 	ConfigItem* item = (ConfigItem *)currentItem();
 | |
| 	if (item) {
 | |
| 		setSelected(item, true);
 | |
| 		menu = item->menu;
 | |
| 	}
 | |
| 	emit gotFocus(menu);
 | |
| }
 | |
| 
 | |
| void ConfigList::contextMenuEvent(QContextMenuEvent *e)
 | |
| {
 | |
| 	if (!headerPopup) {
 | |
| 		QAction *action;
 | |
| 
 | |
| 		headerPopup = new QMenu(this);
 | |
| 		action = new QAction("Show Name", this);
 | |
| 		action->setCheckable(true);
 | |
| 		connect(action, &QAction::toggled,
 | |
| 			this, &ConfigList::setShowName);
 | |
| 		connect(this, &ConfigList::showNameChanged,
 | |
| 			action, &QAction::setChecked);
 | |
| 		action->setChecked(showName);
 | |
| 		headerPopup->addAction(action);
 | |
| 	}
 | |
| 
 | |
| 	headerPopup->exec(e->globalPos());
 | |
| 	e->accept();
 | |
| }
 | |
| 
 | |
| void ConfigList::setShowName(bool on)
 | |
| {
 | |
| 	if (showName == on)
 | |
| 		return;
 | |
| 
 | |
| 	showName = on;
 | |
| 	reinit();
 | |
| 	emit showNameChanged(on);
 | |
| }
 | |
| 
 | |
| QList<ConfigList *> ConfigList::allLists;
 | |
| QAction *ConfigList::showNormalAction;
 | |
| QAction *ConfigList::showAllAction;
 | |
| QAction *ConfigList::showPromptAction;
 | |
| 
 | |
| void ConfigList::setAllOpen(bool open)
 | |
| {
 | |
| 	QTreeWidgetItemIterator it(this);
 | |
| 
 | |
| 	while (*it) {
 | |
| 		(*it)->setExpanded(open);
 | |
| 
 | |
| 		++it;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ConfigInfoView::ConfigInfoView(QWidget* parent, const char *name)
 | |
| 	: Parent(parent), sym(0), _menu(0)
 | |
| {
 | |
| 	setObjectName(name);
 | |
| 	setOpenLinks(false);
 | |
| 
 | |
| 	if (!objectName().isEmpty()) {
 | |
| 		configSettings->beginGroup(objectName());
 | |
| 		setShowDebug(configSettings->value("/showDebug", false).toBool());
 | |
| 		configSettings->endGroup();
 | |
| 		connect(configApp, &QApplication::aboutToQuit,
 | |
| 			this, &ConfigInfoView::saveSettings);
 | |
| 	}
 | |
| 
 | |
| 	contextMenu = createStandardContextMenu();
 | |
| 	QAction *action = new QAction("Show Debug Info", contextMenu);
 | |
| 
 | |
| 	action->setCheckable(true);
 | |
| 	connect(action, &QAction::toggled,
 | |
| 		this, &ConfigInfoView::setShowDebug);
 | |
| 	connect(this, &ConfigInfoView::showDebugChanged,
 | |
| 		action, &QAction::setChecked);
 | |
| 	action->setChecked(showDebug());
 | |
| 	contextMenu->addSeparator();
 | |
| 	contextMenu->addAction(action);
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::saveSettings(void)
 | |
| {
 | |
| 	if (!objectName().isEmpty()) {
 | |
| 		configSettings->beginGroup(objectName());
 | |
| 		configSettings->setValue("/showDebug", showDebug());
 | |
| 		configSettings->endGroup();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::setShowDebug(bool b)
 | |
| {
 | |
| 	if (_showDebug != b) {
 | |
| 		_showDebug = b;
 | |
| 		if (_menu)
 | |
| 			menuInfo();
 | |
| 		else if (sym)
 | |
| 			symbolInfo();
 | |
| 		emit showDebugChanged(b);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::setInfo(struct menu *m)
 | |
| {
 | |
| 	if (_menu == m)
 | |
| 		return;
 | |
| 	_menu = m;
 | |
| 	sym = NULL;
 | |
| 	if (!_menu)
 | |
| 		clear();
 | |
| 	else
 | |
| 		menuInfo();
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::symbolInfo(void)
 | |
| {
 | |
| 	QString str;
 | |
| 
 | |
| 	str += "<big>Symbol: <b>";
 | |
| 	str += print_filter(sym->name);
 | |
| 	str += "</b></big><br><br>value: ";
 | |
| 	str += print_filter(sym_get_string_value(sym));
 | |
| 	str += "<br>visibility: ";
 | |
| 	str += sym->visible == yes ? "y" : sym->visible == mod ? "m" : "n";
 | |
| 	str += "<br>";
 | |
| 	str += debug_info(sym);
 | |
| 
 | |
| 	setText(str);
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::menuInfo(void)
 | |
| {
 | |
| 	struct symbol* sym;
 | |
| 	QString info;
 | |
| 	QTextStream stream(&info);
 | |
| 
 | |
| 	sym = _menu->sym;
 | |
| 	if (sym) {
 | |
| 		if (_menu->prompt) {
 | |
| 			stream << "<big><b>";
 | |
| 			stream << print_filter(_menu->prompt->text);
 | |
| 			stream << "</b></big>";
 | |
| 			if (sym->name) {
 | |
| 				stream << " (";
 | |
| 				if (showDebug())
 | |
| 					stream << "<a href=\"s" << sym->name << "\">";
 | |
| 				stream << print_filter(sym->name);
 | |
| 				if (showDebug())
 | |
| 					stream << "</a>";
 | |
| 				stream << ")";
 | |
| 			}
 | |
| 		} else if (sym->name) {
 | |
| 			stream << "<big><b>";
 | |
| 			if (showDebug())
 | |
| 				stream << "<a href=\"s" << sym->name << "\">";
 | |
| 			stream << print_filter(sym->name);
 | |
| 			if (showDebug())
 | |
| 				stream << "</a>";
 | |
| 			stream << "</b></big>";
 | |
| 		}
 | |
| 		stream << "<br><br>";
 | |
| 
 | |
| 		if (showDebug())
 | |
| 			stream << debug_info(sym);
 | |
| 
 | |
| 		struct gstr help_gstr = str_new();
 | |
| 
 | |
| 		menu_get_ext_help(_menu, &help_gstr);
 | |
| 		stream << print_filter(str_get(&help_gstr));
 | |
| 		str_free(&help_gstr);
 | |
| 	} else if (_menu->prompt) {
 | |
| 		stream << "<big><b>";
 | |
| 		stream << print_filter(_menu->prompt->text);
 | |
| 		stream << "</b></big><br><br>";
 | |
| 		if (showDebug()) {
 | |
| 			if (_menu->prompt->visible.expr) {
 | |
| 				stream << "  dep: ";
 | |
| 				expr_print(_menu->prompt->visible.expr,
 | |
| 					   expr_print_help, &stream, E_NONE);
 | |
| 				stream << "<br><br>";
 | |
| 			}
 | |
| 
 | |
| 			stream << "defined at " << _menu->file->name << ":"
 | |
| 			       << _menu->lineno << "<br><br>";
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	setText(info);
 | |
| }
 | |
| 
 | |
| QString ConfigInfoView::debug_info(struct symbol *sym)
 | |
| {
 | |
| 	QString debug;
 | |
| 	QTextStream stream(&debug);
 | |
| 
 | |
| 	stream << "type: ";
 | |
| 	stream << print_filter(sym_type_name(sym->type));
 | |
| 	if (sym_is_choice(sym))
 | |
| 		stream << " (choice)";
 | |
| 	debug += "<br>";
 | |
| 	if (sym->rev_dep.expr) {
 | |
| 		stream << "reverse dep: ";
 | |
| 		expr_print(sym->rev_dep.expr, expr_print_help, &stream, E_NONE);
 | |
| 		stream << "<br>";
 | |
| 	}
 | |
| 	for (struct property *prop = sym->prop; prop; prop = prop->next) {
 | |
| 		switch (prop->type) {
 | |
| 		case P_PROMPT:
 | |
| 		case P_MENU:
 | |
| 			stream << "prompt: <a href=\"m" << sym->name << "\">";
 | |
| 			stream << print_filter(prop->text);
 | |
| 			stream << "</a><br>";
 | |
| 			break;
 | |
| 		case P_DEFAULT:
 | |
| 		case P_SELECT:
 | |
| 		case P_RANGE:
 | |
| 		case P_COMMENT:
 | |
| 		case P_IMPLY:
 | |
| 		case P_SYMBOL:
 | |
| 			stream << prop_get_type_name(prop->type);
 | |
| 			stream << ": ";
 | |
| 			expr_print(prop->expr, expr_print_help,
 | |
| 				   &stream, E_NONE);
 | |
| 			stream << "<br>";
 | |
| 			break;
 | |
| 		case P_CHOICE:
 | |
| 			if (sym_is_choice(sym)) {
 | |
| 				stream << "choice: ";
 | |
| 				expr_print(prop->expr, expr_print_help,
 | |
| 					   &stream, E_NONE);
 | |
| 				stream << "<br>";
 | |
| 			}
 | |
| 			break;
 | |
| 		default:
 | |
| 			stream << "unknown property: ";
 | |
| 			stream << prop_get_type_name(prop->type);
 | |
| 			stream << "<br>";
 | |
| 		}
 | |
| 		if (prop->visible.expr) {
 | |
| 			stream << "    dep: ";
 | |
| 			expr_print(prop->visible.expr, expr_print_help,
 | |
| 				   &stream, E_NONE);
 | |
| 			stream << "<br>";
 | |
| 		}
 | |
| 	}
 | |
| 	stream << "<br>";
 | |
| 
 | |
| 	return debug;
 | |
| }
 | |
| 
 | |
| QString ConfigInfoView::print_filter(const QString &str)
 | |
| {
 | |
| 	QRegularExpression re("[<>&\"\\n]");
 | |
| 	QString res = str;
 | |
| 	for (int i = 0; (i = res.indexOf(re, i)) >= 0;) {
 | |
| 		switch (res[i].toLatin1()) {
 | |
| 		case '<':
 | |
| 			res.replace(i, 1, "<");
 | |
| 			i += 4;
 | |
| 			break;
 | |
| 		case '>':
 | |
| 			res.replace(i, 1, ">");
 | |
| 			i += 4;
 | |
| 			break;
 | |
| 		case '&':
 | |
| 			res.replace(i, 1, "&");
 | |
| 			i += 5;
 | |
| 			break;
 | |
| 		case '"':
 | |
| 			res.replace(i, 1, """);
 | |
| 			i += 6;
 | |
| 			break;
 | |
| 		case '\n':
 | |
| 			res.replace(i, 1, "<br>");
 | |
| 			i += 4;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	return res;
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::expr_print_help(void *data, struct symbol *sym, const char *str)
 | |
| {
 | |
| 	QTextStream *stream = reinterpret_cast<QTextStream *>(data);
 | |
| 
 | |
| 	if (sym && sym->name && !(sym->flags & SYMBOL_CONST)) {
 | |
| 		*stream << "<a href=\"s" << sym->name << "\">";
 | |
| 		*stream << print_filter(str);
 | |
| 		*stream << "</a>";
 | |
| 	} else {
 | |
| 		*stream << print_filter(str);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::clicked(const QUrl &url)
 | |
| {
 | |
| 	QByteArray str = url.toEncoded();
 | |
| 	const std::size_t count = str.size();
 | |
| 	char *data = new char[count + 1];
 | |
| 	struct symbol **result;
 | |
| 	struct menu *m = NULL;
 | |
| 
 | |
| 	if (count < 1) {
 | |
| 		delete[] data;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	memcpy(data, str.constData(), count);
 | |
| 	data[count] = '\0';
 | |
| 
 | |
| 	/* Seek for exact match */
 | |
| 	data[0] = '^';
 | |
| 	strcat(data, "$");
 | |
| 	result = sym_re_search(data);
 | |
| 	if (!result) {
 | |
| 		delete[] data;
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	sym = *result;
 | |
| 
 | |
| 	/* Seek for the menu which holds the symbol */
 | |
| 	for (struct property *prop = sym->prop; prop; prop = prop->next) {
 | |
| 		    if (prop->type != P_PROMPT && prop->type != P_MENU)
 | |
| 			    continue;
 | |
| 		    m = prop->menu;
 | |
| 		    break;
 | |
| 	}
 | |
| 
 | |
| 	if (!m) {
 | |
| 		/* Symbol is not visible as a menu */
 | |
| 		symbolInfo();
 | |
| 		emit showDebugChanged(true);
 | |
| 	} else {
 | |
| 		emit menuSelected(m);
 | |
| 	}
 | |
| 
 | |
| 	free(result);
 | |
| 	delete[] data;
 | |
| }
 | |
| 
 | |
| void ConfigInfoView::contextMenuEvent(QContextMenuEvent *event)
 | |
| {
 | |
| 	contextMenu->popup(event->globalPos());
 | |
| 	event->accept();
 | |
| }
 | |
| 
 | |
| ConfigSearchWindow::ConfigSearchWindow(ConfigMainWindow *parent)
 | |
| 	: Parent(parent), result(NULL)
 | |
| {
 | |
| 	setObjectName("search");
 | |
| 	setWindowTitle("Search Config");
 | |
| 
 | |
| 	QVBoxLayout* layout1 = new QVBoxLayout(this);
 | |
| 	layout1->setContentsMargins(11, 11, 11, 11);
 | |
| 	layout1->setSpacing(6);
 | |
| 
 | |
| 	QHBoxLayout* layout2 = new QHBoxLayout();
 | |
| 	layout2->setContentsMargins(0, 0, 0, 0);
 | |
| 	layout2->setSpacing(6);
 | |
| 	layout2->addWidget(new QLabel("Find:", this));
 | |
| 	editField = new QLineEdit(this);
 | |
| 	connect(editField, &QLineEdit::returnPressed,
 | |
| 		this, &ConfigSearchWindow::search);
 | |
| 	layout2->addWidget(editField);
 | |
| 	searchButton = new QPushButton("Search", this);
 | |
| 	searchButton->setAutoDefault(false);
 | |
| 	connect(searchButton, &QPushButton::clicked,
 | |
| 		this, &ConfigSearchWindow::search);
 | |
| 	layout2->addWidget(searchButton);
 | |
| 	layout1->addLayout(layout2);
 | |
| 
 | |
| 	split = new QSplitter(this);
 | |
| 	split->setOrientation(Qt::Vertical);
 | |
| 	list = new ConfigList(split, "search");
 | |
| 	list->mode = listMode;
 | |
| 	info = new ConfigInfoView(split, "search");
 | |
| 	connect(list, &ConfigList::menuChanged,
 | |
| 		info, &ConfigInfoView::setInfo);
 | |
| 	connect(list, &ConfigList::menuChanged,
 | |
| 		parent, &ConfigMainWindow::setMenuLink);
 | |
| 
 | |
| 	layout1->addWidget(split);
 | |
| 
 | |
| 	QVariant x, y;
 | |
| 	int width, height;
 | |
| 	bool ok;
 | |
| 
 | |
| 	configSettings->beginGroup("search");
 | |
| 	width = configSettings->value("/window width", parent->width() / 2).toInt();
 | |
| 	height = configSettings->value("/window height", parent->height() / 2).toInt();
 | |
| 	resize(width, height);
 | |
| 	x = configSettings->value("/window x");
 | |
| 	y = configSettings->value("/window y");
 | |
| 	if (x.isValid() && y.isValid())
 | |
| 		move(x.toInt(), y.toInt());
 | |
| 	QList<int> sizes = configSettings->readSizes("/split", &ok);
 | |
| 	if (ok)
 | |
| 		split->setSizes(sizes);
 | |
| 	configSettings->endGroup();
 | |
| 	connect(configApp, &QApplication::aboutToQuit,
 | |
| 		this, &ConfigSearchWindow::saveSettings);
 | |
| }
 | |
| 
 | |
| void ConfigSearchWindow::saveSettings(void)
 | |
| {
 | |
| 	if (!objectName().isEmpty()) {
 | |
| 		configSettings->beginGroup(objectName());
 | |
| 		configSettings->setValue("/window x", pos().x());
 | |
| 		configSettings->setValue("/window y", pos().y());
 | |
| 		configSettings->setValue("/window width", size().width());
 | |
| 		configSettings->setValue("/window height", size().height());
 | |
| 		configSettings->writeSizes("/split", split->sizes());
 | |
| 		configSettings->endGroup();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigSearchWindow::search(void)
 | |
| {
 | |
| 	struct symbol **p;
 | |
| 	struct property *prop;
 | |
| 	ConfigItem *lastItem = NULL;
 | |
| 
 | |
| 	free(result);
 | |
| 	list->clear();
 | |
| 	info->clear();
 | |
| 
 | |
| 	result = sym_re_search(editField->text().toLatin1());
 | |
| 	if (!result)
 | |
| 		return;
 | |
| 	for (p = result; *p; p++) {
 | |
| 		for_all_prompts((*p), prop)
 | |
| 			lastItem = new ConfigItem(list, lastItem, prop->menu,
 | |
| 						  menu_is_visible(prop->menu));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Construct the complete config widget
 | |
|  */
 | |
| ConfigMainWindow::ConfigMainWindow(void)
 | |
| 	: searchWindow(0)
 | |
| {
 | |
| 	bool ok = true;
 | |
| 	QVariant x, y;
 | |
| 	int width, height;
 | |
| 	char title[256];
 | |
| 
 | |
| 	snprintf(title, sizeof(title), "%s%s",
 | |
| 		rootmenu.prompt->text,
 | |
| 		""
 | |
| 		);
 | |
| 	setWindowTitle(title);
 | |
| 
 | |
| 	QRect g = configApp->primaryScreen()->geometry();
 | |
| 	width = configSettings->value("/window width", g.width() - 64).toInt();
 | |
| 	height = configSettings->value("/window height", g.height() - 64).toInt();
 | |
| 	resize(width, height);
 | |
| 	x = configSettings->value("/window x");
 | |
| 	y = configSettings->value("/window y");
 | |
| 	if ((x.isValid())&&(y.isValid()))
 | |
| 		move(x.toInt(), y.toInt());
 | |
| 
 | |
| 	// set up icons
 | |
| 	ConfigItem::symbolYesIcon = QIcon(QPixmap(xpm_symbol_yes));
 | |
| 	ConfigItem::symbolModIcon = QIcon(QPixmap(xpm_symbol_mod));
 | |
| 	ConfigItem::symbolNoIcon = QIcon(QPixmap(xpm_symbol_no));
 | |
| 	ConfigItem::choiceYesIcon = QIcon(QPixmap(xpm_choice_yes));
 | |
| 	ConfigItem::choiceNoIcon = QIcon(QPixmap(xpm_choice_no));
 | |
| 	ConfigItem::menuIcon = QIcon(QPixmap(xpm_menu));
 | |
| 	ConfigItem::menubackIcon = QIcon(QPixmap(xpm_menuback));
 | |
| 
 | |
| 	QWidget *widget = new QWidget(this);
 | |
| 	QVBoxLayout *layout = new QVBoxLayout(widget);
 | |
| 	setCentralWidget(widget);
 | |
| 
 | |
| 	split1 = new QSplitter(widget);
 | |
| 	split1->setOrientation(Qt::Horizontal);
 | |
| 	split1->setChildrenCollapsible(false);
 | |
| 
 | |
| 	menuList = new ConfigList(widget, "menu");
 | |
| 
 | |
| 	split2 = new QSplitter(widget);
 | |
| 	split2->setChildrenCollapsible(false);
 | |
| 	split2->setOrientation(Qt::Vertical);
 | |
| 
 | |
| 	// create config tree
 | |
| 	configList = new ConfigList(widget, "config");
 | |
| 
 | |
| 	helpText = new ConfigInfoView(widget, "help");
 | |
| 
 | |
| 	layout->addWidget(split2);
 | |
| 	split2->addWidget(split1);
 | |
| 	split1->addWidget(configList);
 | |
| 	split1->addWidget(menuList);
 | |
| 	split2->addWidget(helpText);
 | |
| 
 | |
| 	setTabOrder(configList, helpText);
 | |
| 	configList->setFocus();
 | |
| 
 | |
| 	backAction = new QAction(QPixmap(xpm_back), "Back", this);
 | |
| 	connect(backAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::goBack);
 | |
| 
 | |
| 	QAction *quitAction = new QAction("&Quit", this);
 | |
| 	quitAction->setShortcut(Qt::CTRL | Qt::Key_Q);
 | |
| 	connect(quitAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::close);
 | |
| 
 | |
| 	QAction *loadAction = new QAction(QPixmap(xpm_load), "&Load", this);
 | |
| 	loadAction->setShortcut(Qt::CTRL | Qt::Key_L);
 | |
| 	connect(loadAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::loadConfig);
 | |
| 
 | |
| 	saveAction = new QAction(QPixmap(xpm_save), "&Save", this);
 | |
| 	saveAction->setShortcut(Qt::CTRL | Qt::Key_S);
 | |
| 	connect(saveAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::saveConfig);
 | |
| 
 | |
| 	conf_set_changed_callback(conf_changed);
 | |
| 
 | |
| 	// Set saveAction's initial state
 | |
| 	conf_changed();
 | |
| 	configname = xstrdup(conf_get_configname());
 | |
| 
 | |
| 	QAction *saveAsAction = new QAction("Save &As...", this);
 | |
| 	connect(saveAsAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::saveConfigAs);
 | |
| 	QAction *searchAction = new QAction("&Find", this);
 | |
| 	searchAction->setShortcut(Qt::CTRL | Qt::Key_F);
 | |
| 	connect(searchAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::searchConfig);
 | |
| 	singleViewAction = new QAction(QPixmap(xpm_single_view), "Single View", this);
 | |
| 	singleViewAction->setCheckable(true);
 | |
| 	connect(singleViewAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::showSingleView);
 | |
| 	splitViewAction = new QAction(QPixmap(xpm_split_view), "Split View", this);
 | |
| 	splitViewAction->setCheckable(true);
 | |
| 	connect(splitViewAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::showSplitView);
 | |
| 	fullViewAction = new QAction(QPixmap(xpm_tree_view), "Full View", this);
 | |
| 	fullViewAction->setCheckable(true);
 | |
| 	connect(fullViewAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::showFullView);
 | |
| 
 | |
| 	QAction *showNameAction = new QAction("Show Name", this);
 | |
| 	  showNameAction->setCheckable(true);
 | |
| 	connect(showNameAction, &QAction::toggled,
 | |
| 		configList, &ConfigList::setShowName);
 | |
| 	showNameAction->setChecked(configList->showName);
 | |
| 
 | |
| 	QActionGroup *optGroup = new QActionGroup(this);
 | |
| 	optGroup->setExclusive(true);
 | |
| 	connect(optGroup, &QActionGroup::triggered,
 | |
| 		configList, &ConfigList::setOptionMode);
 | |
| 	connect(optGroup, &QActionGroup::triggered,
 | |
| 		menuList, &ConfigList::setOptionMode);
 | |
| 
 | |
| 	ConfigList::showNormalAction = new QAction("Show Normal Options", optGroup);
 | |
| 	ConfigList::showNormalAction->setCheckable(true);
 | |
| 	ConfigList::showAllAction = new QAction("Show All Options", optGroup);
 | |
| 	ConfigList::showAllAction->setCheckable(true);
 | |
| 	ConfigList::showPromptAction = new QAction("Show Prompt Options", optGroup);
 | |
| 	ConfigList::showPromptAction->setCheckable(true);
 | |
| 
 | |
| 	QAction *showDebugAction = new QAction("Show Debug Info", this);
 | |
| 	  showDebugAction->setCheckable(true);
 | |
| 	connect(showDebugAction, &QAction::toggled,
 | |
| 		helpText, &ConfigInfoView::setShowDebug);
 | |
| 	  showDebugAction->setChecked(helpText->showDebug());
 | |
| 
 | |
| 	QAction *showIntroAction = new QAction("Introduction", this);
 | |
| 	connect(showIntroAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::showIntro);
 | |
| 	QAction *showAboutAction = new QAction("About", this);
 | |
| 	connect(showAboutAction, &QAction::triggered,
 | |
| 		this, &ConfigMainWindow::showAbout);
 | |
| 
 | |
| 	// init tool bar
 | |
| 	QToolBar *toolBar = addToolBar("Tools");
 | |
| 	toolBar->addAction(backAction);
 | |
| 	toolBar->addSeparator();
 | |
| 	toolBar->addAction(loadAction);
 | |
| 	toolBar->addAction(saveAction);
 | |
| 	toolBar->addSeparator();
 | |
| 	toolBar->addAction(singleViewAction);
 | |
| 	toolBar->addAction(splitViewAction);
 | |
| 	toolBar->addAction(fullViewAction);
 | |
| 
 | |
| 	// create file menu
 | |
| 	QMenu *menu = menuBar()->addMenu("&File");
 | |
| 	menu->addAction(loadAction);
 | |
| 	menu->addAction(saveAction);
 | |
| 	menu->addAction(saveAsAction);
 | |
| 	menu->addSeparator();
 | |
| 	menu->addAction(quitAction);
 | |
| 
 | |
| 	// create edit menu
 | |
| 	menu = menuBar()->addMenu("&Edit");
 | |
| 	menu->addAction(searchAction);
 | |
| 
 | |
| 	// create options menu
 | |
| 	menu = menuBar()->addMenu("&Option");
 | |
| 	menu->addAction(showNameAction);
 | |
| 	menu->addSeparator();
 | |
| 	menu->addActions(optGroup->actions());
 | |
| 	menu->addSeparator();
 | |
| 	menu->addAction(showDebugAction);
 | |
| 
 | |
| 	// create help menu
 | |
| 	menu = menuBar()->addMenu("&Help");
 | |
| 	menu->addAction(showIntroAction);
 | |
| 	menu->addAction(showAboutAction);
 | |
| 
 | |
| 	connect(helpText, &ConfigInfoView::anchorClicked,
 | |
| 		helpText, &ConfigInfoView::clicked);
 | |
| 
 | |
| 	connect(configList, &ConfigList::menuChanged,
 | |
| 		helpText, &ConfigInfoView::setInfo);
 | |
| 	connect(configList, &ConfigList::menuSelected,
 | |
| 		this, &ConfigMainWindow::changeMenu);
 | |
| 	connect(configList, &ConfigList::itemSelected,
 | |
| 		this, &ConfigMainWindow::changeItens);
 | |
| 	connect(configList, &ConfigList::parentSelected,
 | |
| 		this, &ConfigMainWindow::goBack);
 | |
| 	connect(menuList, &ConfigList::menuChanged,
 | |
| 		helpText, &ConfigInfoView::setInfo);
 | |
| 	connect(menuList, &ConfigList::menuSelected,
 | |
| 		this, &ConfigMainWindow::changeMenu);
 | |
| 
 | |
| 	connect(configList, &ConfigList::gotFocus,
 | |
| 		helpText, &ConfigInfoView::setInfo);
 | |
| 	connect(menuList, &ConfigList::gotFocus,
 | |
| 		helpText, &ConfigInfoView::setInfo);
 | |
| 	connect(menuList, &ConfigList::gotFocus,
 | |
| 		this, &ConfigMainWindow::listFocusChanged);
 | |
| 	connect(helpText, &ConfigInfoView::menuSelected,
 | |
| 		this, &ConfigMainWindow::setMenuLink);
 | |
| 
 | |
| 	QString listMode = configSettings->value("/listMode", "symbol").toString();
 | |
| 	if (listMode == "single")
 | |
| 		showSingleView();
 | |
| 	else if (listMode == "full")
 | |
| 		showFullView();
 | |
| 	else /*if (listMode == "split")*/
 | |
| 		showSplitView();
 | |
| 
 | |
| 	// UI setup done, restore splitter positions
 | |
| 	QList<int> sizes = configSettings->readSizes("/split1", &ok);
 | |
| 	if (ok)
 | |
| 		split1->setSizes(sizes);
 | |
| 
 | |
| 	sizes = configSettings->readSizes("/split2", &ok);
 | |
| 	if (ok)
 | |
| 		split2->setSizes(sizes);
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::loadConfig(void)
 | |
| {
 | |
| 	QString str;
 | |
| 	QByteArray ba;
 | |
| 	const char *name;
 | |
| 
 | |
| 	str = QFileDialog::getOpenFileName(this, "", configname);
 | |
| 	if (str.isNull())
 | |
| 		return;
 | |
| 
 | |
| 	ba = str.toLocal8Bit();
 | |
| 	name = ba.data();
 | |
| 
 | |
| 	if (conf_read(name))
 | |
| 		QMessageBox::information(this, "qconf", "Unable to load configuration!");
 | |
| 
 | |
| 	free(configname);
 | |
| 	configname = xstrdup(name);
 | |
| 
 | |
| 	ConfigList::updateListAllForAll();
 | |
| }
 | |
| 
 | |
| bool ConfigMainWindow::saveConfig(void)
 | |
| {
 | |
| 	if (conf_write(configname)) {
 | |
| 		QMessageBox::information(this, "qconf", "Unable to save configuration!");
 | |
| 		return false;
 | |
| 	}
 | |
| 	conf_write_autoconf(0);
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::saveConfigAs(void)
 | |
| {
 | |
| 	QString str;
 | |
| 	QByteArray ba;
 | |
| 	const char *name;
 | |
| 
 | |
| 	str = QFileDialog::getSaveFileName(this, "", configname);
 | |
| 	if (str.isNull())
 | |
| 		return;
 | |
| 
 | |
| 	ba = str.toLocal8Bit();
 | |
| 	name = ba.data();
 | |
| 
 | |
| 	if (conf_write(name)) {
 | |
| 		QMessageBox::information(this, "qconf", "Unable to save configuration!");
 | |
| 	}
 | |
| 	conf_write_autoconf(0);
 | |
| 
 | |
| 	free(configname);
 | |
| 	configname = xstrdup(name);
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::searchConfig(void)
 | |
| {
 | |
| 	if (!searchWindow)
 | |
| 		searchWindow = new ConfigSearchWindow(this);
 | |
| 	searchWindow->show();
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::changeItens(struct menu *menu)
 | |
| {
 | |
| 	configList->setRootMenu(menu);
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::changeMenu(struct menu *menu)
 | |
| {
 | |
| 	menuList->setRootMenu(menu);
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::setMenuLink(struct menu *menu)
 | |
| {
 | |
| 	struct menu *parent;
 | |
| 	ConfigList* list = NULL;
 | |
| 	ConfigItem* item;
 | |
| 
 | |
| 	if (configList->menuSkip(menu))
 | |
| 		return;
 | |
| 
 | |
| 	switch (configList->mode) {
 | |
| 	case singleMode:
 | |
| 		list = configList;
 | |
| 		parent = menu_get_parent_menu(menu);
 | |
| 		if (!parent)
 | |
| 			return;
 | |
| 		list->setRootMenu(parent);
 | |
| 		break;
 | |
| 	case menuMode:
 | |
| 		if (menu->flags & MENU_ROOT) {
 | |
| 			menuList->setRootMenu(menu);
 | |
| 			configList->clearSelection();
 | |
| 			list = configList;
 | |
| 		} else {
 | |
| 			parent = menu_get_parent_menu(menu->parent);
 | |
| 			if (!parent)
 | |
| 				return;
 | |
| 
 | |
| 			/* Select the config view */
 | |
| 			item = configList->findConfigItem(parent);
 | |
| 			if (item) {
 | |
| 				configList->setSelected(item, true);
 | |
| 				configList->scrollToItem(item);
 | |
| 			}
 | |
| 
 | |
| 			menuList->setRootMenu(parent);
 | |
| 			menuList->clearSelection();
 | |
| 			list = menuList;
 | |
| 		}
 | |
| 		break;
 | |
| 	case fullMode:
 | |
| 		list = configList;
 | |
| 		break;
 | |
| 	default:
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	if (list) {
 | |
| 		item = list->findConfigItem(menu);
 | |
| 		if (item) {
 | |
| 			list->setSelected(item, true);
 | |
| 			list->scrollToItem(item);
 | |
| 			list->setFocus();
 | |
| 			helpText->setInfo(menu);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::listFocusChanged(void)
 | |
| {
 | |
| 	if (menuList->mode == menuMode)
 | |
| 		configList->clearSelection();
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::goBack(void)
 | |
| {
 | |
| 	if (configList->rootEntry == &rootmenu)
 | |
| 		return;
 | |
| 
 | |
| 	configList->setParentMenu();
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::showSingleView(void)
 | |
| {
 | |
| 	singleViewAction->setEnabled(false);
 | |
| 	singleViewAction->setChecked(true);
 | |
| 	splitViewAction->setEnabled(true);
 | |
| 	splitViewAction->setChecked(false);
 | |
| 	fullViewAction->setEnabled(true);
 | |
| 	fullViewAction->setChecked(false);
 | |
| 
 | |
| 	backAction->setEnabled(true);
 | |
| 
 | |
| 	menuList->hide();
 | |
| 	menuList->setRootMenu(0);
 | |
| 	configList->mode = singleMode;
 | |
| 	if (configList->rootEntry == &rootmenu)
 | |
| 		configList->updateListAll();
 | |
| 	else
 | |
| 		configList->setRootMenu(&rootmenu);
 | |
| 	configList->setFocus();
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::showSplitView(void)
 | |
| {
 | |
| 	singleViewAction->setEnabled(true);
 | |
| 	singleViewAction->setChecked(false);
 | |
| 	splitViewAction->setEnabled(false);
 | |
| 	splitViewAction->setChecked(true);
 | |
| 	fullViewAction->setEnabled(true);
 | |
| 	fullViewAction->setChecked(false);
 | |
| 
 | |
| 	backAction->setEnabled(false);
 | |
| 
 | |
| 	configList->mode = menuMode;
 | |
| 	if (configList->rootEntry == &rootmenu)
 | |
| 		configList->updateListAll();
 | |
| 	else
 | |
| 		configList->setRootMenu(&rootmenu);
 | |
| 	configList->setAllOpen(true);
 | |
| 	configApp->processEvents();
 | |
| 	menuList->mode = symbolMode;
 | |
| 	menuList->setRootMenu(&rootmenu);
 | |
| 	menuList->setAllOpen(true);
 | |
| 	menuList->show();
 | |
| 	menuList->setFocus();
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::showFullView(void)
 | |
| {
 | |
| 	singleViewAction->setEnabled(true);
 | |
| 	singleViewAction->setChecked(false);
 | |
| 	splitViewAction->setEnabled(true);
 | |
| 	splitViewAction->setChecked(false);
 | |
| 	fullViewAction->setEnabled(false);
 | |
| 	fullViewAction->setChecked(true);
 | |
| 
 | |
| 	backAction->setEnabled(false);
 | |
| 
 | |
| 	menuList->hide();
 | |
| 	menuList->setRootMenu(0);
 | |
| 	configList->mode = fullMode;
 | |
| 	if (configList->rootEntry == &rootmenu)
 | |
| 		configList->updateListAll();
 | |
| 	else
 | |
| 		configList->setRootMenu(&rootmenu);
 | |
| 	configList->setFocus();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * ask for saving configuration before quitting
 | |
|  */
 | |
| void ConfigMainWindow::closeEvent(QCloseEvent* e)
 | |
| {
 | |
| 	if (!conf_get_changed()) {
 | |
| 		e->accept();
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	QMessageBox mb(QMessageBox::Icon::Warning, "qconf",
 | |
| 		       "Save configuration?");
 | |
| 
 | |
| 	QPushButton *yb = mb.addButton(QMessageBox::Yes);
 | |
| 	QPushButton *db = mb.addButton(QMessageBox::No);
 | |
| 	QPushButton *cb = mb.addButton(QMessageBox::Cancel);
 | |
| 
 | |
| 	yb->setText("&Save Changes");
 | |
| 	db->setText("&Discard Changes");
 | |
| 	cb->setText("Cancel Exit");
 | |
| 
 | |
| 	mb.setDefaultButton(yb);
 | |
| 	mb.setEscapeButton(cb);
 | |
| 
 | |
| 	switch (mb.exec()) {
 | |
| 	case QMessageBox::Yes:
 | |
| 		if (saveConfig())
 | |
| 			e->accept();
 | |
| 		else
 | |
| 			e->ignore();
 | |
| 		break;
 | |
| 	case QMessageBox::No:
 | |
| 		e->accept();
 | |
| 		break;
 | |
| 	case QMessageBox::Cancel:
 | |
| 		e->ignore();
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::showIntro(void)
 | |
| {
 | |
| 	static const QString str =
 | |
| 		"Welcome to the qconf graphical configuration tool.\n"
 | |
| 		"\n"
 | |
| 		"For bool and tristate options, a blank box indicates the "
 | |
| 		"feature is disabled, a check indicates it is enabled, and a "
 | |
| 		"dot indicates that it is to be compiled as a module. Clicking "
 | |
| 		"on the box will cycle through the three states. For int, hex, "
 | |
| 		"and string options, double-clicking or pressing F2 on the "
 | |
| 		"Value cell will allow you to edit the value.\n"
 | |
| 		"\n"
 | |
| 		"If you do not see an option (e.g., a device driver) that you "
 | |
| 		"believe should be present, try turning on Show All Options "
 | |
| 		"under the Options menu. Enabling Show Debug Info will help you"
 | |
| 		"figure out what other options must be enabled to support the "
 | |
| 		"option you are interested in, and hyperlinks will navigate to "
 | |
| 		"them.\n"
 | |
| 		"\n"
 | |
| 		"Toggling Show Debug Info under the Options menu will show the "
 | |
| 		"dependencies, which you can then match by examining other "
 | |
| 		"options.\n";
 | |
| 
 | |
| 	QMessageBox::information(this, "qconf", str);
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::showAbout(void)
 | |
| {
 | |
| 	static const QString str = "qconf is Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>.\n"
 | |
| 		"Copyright (C) 2015 Boris Barbulovski <bbarbulovski@gmail.com>.\n"
 | |
| 		"\n"
 | |
| 		"Bug reports and feature request can also be entered at http://bugzilla.kernel.org/\n"
 | |
| 		"\n"
 | |
| 		"Qt Version: ";
 | |
| 
 | |
| 	QMessageBox::information(this, "qconf", str + qVersion());
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::saveSettings(void)
 | |
| {
 | |
| 	configSettings->setValue("/window x", pos().x());
 | |
| 	configSettings->setValue("/window y", pos().y());
 | |
| 	configSettings->setValue("/window width", size().width());
 | |
| 	configSettings->setValue("/window height", size().height());
 | |
| 
 | |
| 	QString entry;
 | |
| 	switch(configList->mode) {
 | |
| 	case singleMode :
 | |
| 		entry = "single";
 | |
| 		break;
 | |
| 
 | |
| 	case symbolMode :
 | |
| 		entry = "split";
 | |
| 		break;
 | |
| 
 | |
| 	case fullMode :
 | |
| 		entry = "full";
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		break;
 | |
| 	}
 | |
| 	configSettings->setValue("/listMode", entry);
 | |
| 
 | |
| 	configSettings->writeSizes("/split1", split1->sizes());
 | |
| 	configSettings->writeSizes("/split2", split2->sizes());
 | |
| }
 | |
| 
 | |
| void ConfigMainWindow::conf_changed(void)
 | |
| {
 | |
| 	if (saveAction)
 | |
| 		saveAction->setEnabled(conf_get_changed());
 | |
| }
 | |
| 
 | |
| void fixup_rootmenu(struct menu *menu)
 | |
| {
 | |
| 	struct menu *child;
 | |
| 	static int menu_cnt = 0;
 | |
| 
 | |
| 	menu->flags |= MENU_ROOT;
 | |
| 	for (child = menu->list; child; child = child->next) {
 | |
| 		if (child->prompt && child->prompt->type == P_MENU) {
 | |
| 			menu_cnt++;
 | |
| 			fixup_rootmenu(child);
 | |
| 			menu_cnt--;
 | |
| 		} else if (!menu_cnt)
 | |
| 			fixup_rootmenu(child);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static const char *progname;
 | |
| 
 | |
| static void usage(void)
 | |
| {
 | |
| 	printf("%s [-s] <config>\n", progname);
 | |
| 	exit(0);
 | |
| }
 | |
| 
 | |
| int main(int ac, char** av)
 | |
| {
 | |
| 	ConfigMainWindow* v;
 | |
| 	const char *name;
 | |
| 
 | |
| 	progname = av[0];
 | |
| 	if (ac > 1 && av[1][0] == '-') {
 | |
| 		switch (av[1][1]) {
 | |
| 		case 's':
 | |
| 			conf_set_message_callback(NULL);
 | |
| 			break;
 | |
| 		case 'h':
 | |
| 		case '?':
 | |
| 			usage();
 | |
| 		}
 | |
| 		name = av[2];
 | |
| 	} else
 | |
| 		name = av[1];
 | |
| 	if (!name)
 | |
| 		usage();
 | |
| 
 | |
| 	conf_parse(name);
 | |
| 	fixup_rootmenu(&rootmenu);
 | |
| 	conf_read(NULL);
 | |
| 	//zconfdump(stdout);
 | |
| 
 | |
| 	configApp = new QApplication(ac, av);
 | |
| 
 | |
| 	configSettings = new ConfigSettings();
 | |
| 	configSettings->beginGroup("/kconfig/qconf");
 | |
| 	v = new ConfigMainWindow();
 | |
| 
 | |
| 	//zconfdump(stdout);
 | |
| 	configApp->connect(configApp, SIGNAL(lastWindowClosed()), SLOT(quit()));
 | |
| 	configApp->connect(configApp, SIGNAL(aboutToQuit()), v, SLOT(saveSettings()));
 | |
| 	v->show();
 | |
| 	configApp->exec();
 | |
| 
 | |
| 	configSettings->endGroup();
 | |
| 	delete configSettings;
 | |
| 	delete v;
 | |
| 	delete configApp;
 | |
| 
 | |
| 	return 0;
 | |
| }
 |