/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "toolsettings.h" #include "dialogs/externaltoolconfig.h" #include "externaltool.h" #include "externaltoolmanager.h" #include "coreconstants.h" #include "icore.h" #include #include #include #include #include #include using namespace Core; using namespace Core::Internal; ToolSettings::ToolSettings(QObject *parent) : IOptionsPage(parent) { setId(Constants::SETTINGS_ID_TOOLS); setDisplayName(tr("External Tools")); setCategory(Constants::SETTINGS_CATEGORY_CORE); } QWidget *ToolSettings::widget() { if (!m_widget) { m_widget = new ExternalToolConfig; m_widget->setTools(ExternalToolManager::toolsByCategory()); } return m_widget; } static QString getUserFilePath(const QString &proposalFileName) { const QDir resourceDir(ICore::userResourcePath()); if (!resourceDir.exists(QLatin1String("externaltools"))) resourceDir.mkpath(QLatin1String("externaltools")); const QFileInfo fi(proposalFileName); const QString &suffix = QLatin1Char('.') + fi.completeSuffix(); const QString &newFilePath = ICore::userResourcePath() + QLatin1String("/externaltools/") + fi.baseName(); int count = 0; QString tryPath = newFilePath + suffix; while (QFile::exists(tryPath)) { if (++count > 15) return QString(); // add random number const int number = qrand() % 1000; tryPath = newFilePath + QString::number(number) + suffix; } return tryPath; } static QString idFromDisplayName(const QString &displayName) { QString id = displayName; id.remove(QRegExp(QLatin1String("&(?!&)"))); QChar *c = id.data(); while (!c->isNull()) { if (!c->isLetterOrNumber()) *c = QLatin1Char('_'); ++c; } return id; } static QString findUnusedId(const QString &proposal, const QMap > &tools) { int number = 0; QString result; bool found = false; do { result = proposal + (number > 0 ? QString::number(number) : QString::fromLatin1("")); ++number; found = false; QMapIterator > it(tools); while (!found && it.hasNext()) { it.next(); foreach (ExternalTool *tool, it.value()) { if (tool->id() == result) { found = true; break; } } } } while (found); return result; } void ToolSettings::apply() { if (!m_widget) return; m_widget->apply(); QMap originalTools = ExternalToolManager::toolsById(); QMap > newToolsMap = m_widget->tools(); QMap > resultMap; QMapIterator > it(newToolsMap); while (it.hasNext()) { it.next(); QList items; foreach (ExternalTool *tool, it.value()) { ExternalTool *toolToAdd = nullptr; if (ExternalTool *originalTool = originalTools.take(tool->id())) { // check if it has different category and is custom tool if (tool->displayCategory() != it.key() && !tool->preset()) tool->setDisplayCategory(it.key()); // check if the tool has changed if ((*originalTool) == (*tool)) { toolToAdd = originalTool; } else { // case 1: tool is changed preset if (tool->preset() && (*tool) != (*(tool->preset()))) { // check if we need to choose a new file name if (tool->preset()->fileName() == tool->fileName()) { const QString &fileName = Utils::FileName::fromString(tool->preset()->fileName()).fileName(); const QString &newFilePath = getUserFilePath(fileName); // TODO error handling if newFilePath.isEmpty() (i.e. failed to find a unused name) tool->setFileName(newFilePath); } // TODO error handling tool->save(); // case 2: tool is previously changed preset but now same as preset } else if (tool->preset() && (*tool) == (*(tool->preset()))) { // check if we need to delete the changed description if (originalTool->fileName() != tool->preset()->fileName() && QFile::exists(originalTool->fileName())) { // TODO error handling QFile::remove(originalTool->fileName()); } tool->setFileName(tool->preset()->fileName()); // no need to save, it's the same as the preset // case 3: tool is custom tool } else { // TODO error handling tool->save(); } // 'tool' is deleted by config page, 'originalTool' is deleted by setToolsByCategory toolToAdd = new ExternalTool(tool); } } else { // new tool. 'tool' is deleted by config page QString id = idFromDisplayName(tool->displayName()); id = findUnusedId(id, newToolsMap); tool->setId(id); // TODO error handling if newFilePath.isEmpty() (i.e. failed to find a unused name) tool->setFileName(getUserFilePath(id + QLatin1String(".xml"))); // TODO error handling tool->save(); toolToAdd = new ExternalTool(tool); } items.append(toolToAdd); } if (!items.isEmpty()) resultMap.insert(it.key(), items); } // Remove tools that have been deleted from the settings (and are no preset) foreach (ExternalTool *tool, originalTools) { QTC_ASSERT(!tool->preset(), continue); // TODO error handling QFile::remove(tool->fileName()); } ExternalToolManager::setToolsByCategory(resultMap); } void ToolSettings::finish() { delete m_widget; }