From patchwork Tue Apr 23 13:27:38 2019 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Yordan Karadzhov X-Patchwork-Id: 10912961 Return-Path: Received: from mail.wl.linuxfoundation.org (pdx-wl-mail.web.codeaurora.org [172.30.200.125]) by pdx-korg-patchwork-2.web.codeaurora.org (Postfix) with ESMTP id 264B6112C for ; Tue, 23 Apr 2019 13:28:06 +0000 (UTC) Received: from mail.wl.linuxfoundation.org (localhost [127.0.0.1]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 15C77286F2 for ; Tue, 23 Apr 2019 13:28:06 +0000 (UTC) Received: by mail.wl.linuxfoundation.org (Postfix, from userid 486) id 0A0E628715; Tue, 23 Apr 2019 13:28:06 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 3.3.1 (2010-03-16) on pdx-wl-mail.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-7.9 required=2.0 tests=BAYES_00,MAILING_LIST_MULTI, RCVD_IN_DNSWL_HI autolearn=ham version=3.3.1 Received: from vger.kernel.org (vger.kernel.org [209.132.180.67]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 17DFB286F2 for ; Tue, 23 Apr 2019 13:28:05 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727829AbfDWN2E (ORCPT ); Tue, 23 Apr 2019 09:28:04 -0400 Received: from mail-wr1-f66.google.com ([209.85.221.66]:33992 "EHLO mail-wr1-f66.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727827AbfDWN2E (ORCPT ); Tue, 23 Apr 2019 09:28:04 -0400 Received: by mail-wr1-f66.google.com with SMTP id c6so14534807wrm.1 for ; Tue, 23 Apr 2019 06:28:03 -0700 (PDT) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=ehWXD5xz2Th45I7P9D3ZpvQ5qdm3wj/0biwIanxs6Qg=; b=tjz7Wlt40sIAJO9MzB9lbJE0ACIvmoMKdK46+s1Ms46ixlrlzbiTuahJvpcoGbd+lY DHLqzI9H4g8Q2qkvOt0MlGJwXgjov7SwNEaVM65Bj+ZNsqZurTyRTrUBgRv9zpD5j0JX TF+QxMJ60SnyQzPeRXkymgQrt//Y21rSHk0RxwqZYKaQ+15NQvlRLlbaoTq1h4ClwtKs lb9w3BdV81lfgGkfgm58ff5AC5S4a/LYzgRRmDdd1Irp3VCg+aWIPJ69+ives9U3y3ws dZLC9l7/jhaL5ep5obyQspQGrMBlL+qT2LUUTnfn56XcGhgoBd8zLyP0rZIR9iYyHypo E8Mw== X-Gm-Message-State: APjAAAXiAXVMZChTA5YJ5SAa6TGDq9MWPwSkPwEHB7lH/94HtFAfrsLM oMYYueG5ikgPaLrRvJ3IFRk= X-Google-Smtp-Source: APXvYqy9cC4e8rhdE38g22YOfAZ1bduHfY13bBBmwuQPfOlOiIJGjSfNznTjRgYEDc43VIR5qmfWhg== X-Received: by 2002:adf:8051:: with SMTP id 75mr18118148wrk.2.1556026082745; Tue, 23 Apr 2019 06:28:02 -0700 (PDT) Received: from mamba.eng.vmware.com ([146.247.46.5]) by smtp.gmail.com with ESMTPSA id w11sm31222231wre.15.2019.04.23.06.28.01 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Tue, 23 Apr 2019 06:28:02 -0700 (PDT) From: Yordan Karadzhov To: rostedt@goodmis.org Cc: linux-trace-devel@vger.kernel.org, y.karadz@gmail.com, Yordan Karadzhov Subject: [PATCH 2/5] kernel-shark: Add logic for the initial path of Open-File dialogs Date: Tue, 23 Apr 2019 16:27:38 +0300 Message-Id: <20190423132741.17864-3-ykaradzhov@vmware.com> X-Mailer: git-send-email 2.20.1 In-Reply-To: <20190423132741.17864-1-ykaradzhov@vmware.com> References: <20190423132741.17864-1-ykaradzhov@vmware.com> MIME-Version: 1.0 Sender: linux-trace-devel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-trace-devel@vger.kernel.org X-Virus-Scanned: ClamAV using ClamSMTP If the application has been started from its installation location, all Open File dialogs will start at ${HOME}. Otherwise the dialogs will start at ${PWD}. If a given dialog has been used already to select a file, next time the dialog will start in the directory of this file. Suggested-by: Steven Rostedt (VMware) Signed-off-by: Yordan Karadzhov --- kernel-shark/src/KsCaptureDialog.cpp | 34 +++----- kernel-shark/src/KsCaptureDialog.hpp | 2 + kernel-shark/src/KsMainWindow.cpp | 70 ++++++---------- kernel-shark/src/KsMainWindow.hpp | 4 +- kernel-shark/src/KsUtils.cpp | 121 +++++++++++++++++++++++++++ kernel-shark/src/KsUtils.hpp | 17 ++++ 6 files changed, 183 insertions(+), 65 deletions(-) diff --git a/kernel-shark/src/KsCaptureDialog.cpp b/kernel-shark/src/KsCaptureDialog.cpp index 1272c2e..2976a3b 100644 --- a/kernel-shark/src/KsCaptureDialog.cpp +++ b/kernel-shark/src/KsCaptureDialog.cpp @@ -206,10 +206,9 @@ void KsCaptureControl::_importSettings() events = tep_list_events(_localTEP, TEP_EVENT_SORT_SYSTEM); /* Get the configuration document. */ - fileName = QFileDialog::getOpenFileName(this, - "Import from Filter", - KS_DIR, - "Kernel Shark Config files (*.json);;"); + fileName = KsUtils::getFile(this, "Import from Filter", + "Kernel Shark Config files (*.json);;", + _lastFilePath); if (fileName.isEmpty()) return; @@ -256,23 +255,16 @@ void KsCaptureControl::_exportSettings() json_object *jplugin; QString plugin, out, comm; QVector ids; - QString fileName = - QFileDialog::getSaveFileName(this, - "Export to File", - KS_DIR, - "Kernel Shark Config files (*.json);;"); + QString fileName; + + fileName = KsUtils::getSaveFile(this, "Export to File", + "Kernel Shark Config files (*.json);;", + ".json", + _lastFilePath); if (fileName.isEmpty()) return; - if (!fileName.endsWith(".json")) { - fileName += ".json"; - if (QFileInfo(fileName).exists()) { - if (!KsWidgetsLib::fileExistsDialog(fileName)) - return; - } - } - /* Create a configuration document. */ conf = kshark_record_config_new(KS_CONFIG_JSON); events = kshark_filter_config_new(KS_CONFIG_JSON); @@ -312,10 +304,10 @@ void KsCaptureControl::_exportSettings() void KsCaptureControl::_browse() { QString fileName = - QFileDialog::getSaveFileName(this, - "Save File", - KS_DIR, - "trace-cmd files (*.dat);;All files (*)"); + KsUtils::getSaveFile(this, "Save File", + "trace-cmd files (*.dat);;All files (*)", + ".dat", + _lastFilePath); if (!fileName.isEmpty()) _outputLineEdit.setText(fileName); diff --git a/kernel-shark/src/KsCaptureDialog.hpp b/kernel-shark/src/KsCaptureDialog.hpp index d65f475..2265704 100644 --- a/kernel-shark/src/KsCaptureDialog.hpp +++ b/kernel-shark/src/KsCaptureDialog.hpp @@ -61,6 +61,8 @@ private: QPushButton _outputBrowseButton; + QString _lastFilePath; + QStringList _getPlugins(); void _importSettings(); diff --git a/kernel-shark/src/KsMainWindow.cpp b/kernel-shark/src/KsMainWindow.cpp index c839aca..748bacd 100644 --- a/kernel-shark/src/KsMainWindow.cpp +++ b/kernel-shark/src/KsMainWindow.cpp @@ -360,11 +360,11 @@ void KsMainWindow::_createMenus() void KsMainWindow::_open() { - QString fileName = - QFileDialog::getOpenFileName(this, - "Open File", - KS_DIR, - "trace-cmd files (*.dat);;All files (*)"); + QString fileName; + + fileName = KsUtils::getFile(this, "Open File", + "trace-cmd files (*.dat);;All files (*)", + _lastDataFilePath); if (!fileName.isEmpty()) loadDataFile(fileName); @@ -429,11 +429,11 @@ void KsMainWindow::_restoreSession() void KsMainWindow::_importSession() { - QString fileName = - QFileDialog::getOpenFileName(this, - "Import Session", - KS_DIR, - "Kernel Shark Config files (*.json);;"); + QString fileName; + + fileName = KsUtils::getFile(this, "Import Session", + "Kernel Shark Config files (*.json);;", + _lastConfFilePath); if (fileName.isEmpty()) return; @@ -460,23 +460,15 @@ void KsMainWindow::_updateSession() void KsMainWindow::_exportSession() { - QString fileName = - QFileDialog::getSaveFileName(this, - "Export Filter", - KS_DIR, - "Kernel Shark Config files (*.json);;"); + QString fileName; + fileName = KsUtils::getSaveFile(this, "Export Filter", + "Kernel Shark Config files (*.json);;", + ".json", + _lastConfFilePath); if (fileName.isEmpty()) return; - if (!fileName.endsWith(".json")) { - fileName += ".json"; - if (QFileInfo(fileName).exists()) { - if (!KsWidgetsLib::fileExistsDialog(fileName)) - return; - } - } - _updateSession(); _session.exportToFile(fileName); } @@ -512,8 +504,9 @@ void KsMainWindow::_importFilter() if (!kshark_instance(&kshark_ctx)) return; - fileName = QFileDialog::getOpenFileName(this, "Import Filter", KS_DIR, - "Kernel Shark Config files (*.json);;"); + fileName = KsUtils::getFile(this, "Import Filter", + "Kernel Shark Config files (*.json);;", + _lastConfFilePath); if (fileName.isEmpty()) return; @@ -540,20 +533,14 @@ void KsMainWindow::_exportFilter() if (!kshark_instance(&kshark_ctx)) return; - fileName = QFileDialog::getSaveFileName(this, "Export Filter", KS_DIR, - "Kernel Shark Config files (*.json);;"); + fileName = KsUtils::getSaveFile(this, "Export Filter", + "Kernel Shark Config files (*.json);;", + ".json", + _lastConfFilePath); if (fileName.isEmpty()) return; - if (!fileName.endsWith(".json")) { - fileName += ".json"; - if (QFileInfo(fileName).exists()) { - if (!KsWidgetsLib::fileExistsDialog(fileName)) - return; - } - } - kshark_export_all_event_filters(kshark_ctx, &conf); kshark_save_config_file(fileName.toStdString().c_str(), conf); kshark_free_config_doc(conf); @@ -859,15 +846,12 @@ void KsMainWindow::_pluginAdd() { QStringList fileNames; - fileNames = - QFileDialog::getOpenFileNames(this, "Add KernelShark plugins", - KS_DIR, - "KernelShark Plugins (*.so);;"); - - if (fileNames.isEmpty()) - return; + fileNames = KsUtils::getFiles(this, "Add KernelShark plugins", + "KernelShark Plugins (*.so);;", + _lastPluginFilePath); - _plugins.addPlugins(fileNames); + if (!fileNames.isEmpty()) + _plugins.addPlugins(fileNames); } void KsMainWindow::_record() diff --git a/kernel-shark/src/KsMainWindow.hpp b/kernel-shark/src/KsMainWindow.hpp index ec6506e..2bf3285 100644 --- a/kernel-shark/src/KsMainWindow.hpp +++ b/kernel-shark/src/KsMainWindow.hpp @@ -151,7 +151,9 @@ private: QAction _contentsAction; - QShortcut _deselectShortcut; + QShortcut _deselectShortcut; + + QString _lastDataFilePath, _lastConfFilePath, _lastPluginFilePath; void _open(); diff --git a/kernel-shark/src/KsUtils.cpp b/kernel-shark/src/KsUtils.cpp index 6af0c66..8c42206 100644 --- a/kernel-shark/src/KsUtils.cpp +++ b/kernel-shark/src/KsUtils.cpp @@ -11,6 +11,7 @@ // KernelShark #include "KsUtils.hpp" +#include "KsWidgetsLib.hpp" namespace KsUtils { @@ -136,6 +137,126 @@ bool matchCPUVisible(struct kshark_context *kshark_ctx, return (e->cpu == cpu && (e->visible & KS_GRAPH_VIEW_FILTER_MASK)); } +/** + * @brief Check if the application runs from its installation location. + */ +bool isInstalled() +{ + QString appPath = QCoreApplication::applicationFilePath(); + QString installPath(_INSTALL_PREFIX); + + installPath += "/bin/kernelshark"; + installPath = QDir::cleanPath(installPath); + + return appPath == installPath; +} + +static QString getFileDialog(QWidget *parent, + const QString &windowName, + const QString &filter, + QString &lastFilePath, + bool forSave) +{ + QString fileName; + + if (lastFilePath.isEmpty()) { + lastFilePath = isInstalled() ? QDir::homePath() : + QDir::currentPath(); + } + + if (forSave) { + fileName = QFileDialog::getSaveFileName(parent, + windowName, + lastFilePath, + filter); + } else { + fileName = QFileDialog::getOpenFileName(parent, + windowName, + lastFilePath, + filter); + } + + if (!fileName.isEmpty()) + lastFilePath = QFileInfo(fileName).path(); + + return fileName; +} + +static QStringList getFilesDialog(QWidget *parent, + const QString &windowName, + const QString &filter, + QString &lastFilePath) +{ + QStringList fileNames; + + if (lastFilePath.isEmpty()) { + lastFilePath = isInstalled() ? QDir::homePath() : + QDir::currentPath(); + } + + fileNames = QFileDialog::getOpenFileNames(parent, + windowName, + lastFilePath, + filter); + + if (!fileNames.isEmpty()) + lastFilePath = QFileInfo(fileNames[0]).path(); + + return fileNames; +} + +/** + * @brief Open a standard Qt getFileName dialog and return the name of the + * selected file. Only one file can be selected. + */ +QString getFile(QWidget *parent, + const QString &windowName, + const QString &filter, + QString &lastFilePath) +{ + return getFileDialog(parent, windowName, filter, lastFilePath, false); +} + +/** + * @brief Open a standard Qt getFileName dialog and return the names of the + * selected files. Multiple files can be selected. + */ +QStringList getFiles(QWidget *parent, + const QString &windowName, + const QString &filter, + QString &lastFilePath) +{ + return getFilesDialog(parent, windowName, filter, lastFilePath); +} + +/** + * @brief Open a standard Qt getFileName dialog and return the name of the + * selected file. Only one file can be selected. + */ +QString getSaveFile(QWidget *parent, + const QString &windowName, + const QString &filter, + const QString &extension, + QString &lastFilePath) +{ + QString fileName = getFileDialog(parent, + windowName, + filter, + lastFilePath, + true); + + if (!fileName.isEmpty() && !fileName.endsWith(extension)) { + fileName += extension; + + if (QFileInfo(fileName).exists()) { + if (!KsWidgetsLib::fileExistsDialog(fileName)) + fileName.clear(); + } + } + + return fileName; +} + }; // KsUtils /** A stream operator for converting QColor into KsPlot::Color. */ diff --git a/kernel-shark/src/KsUtils.hpp b/kernel-shark/src/KsUtils.hpp index c8b5e88..7b80b21 100644 --- a/kernel-shark/src/KsUtils.hpp +++ b/kernel-shark/src/KsUtils.hpp @@ -111,6 +111,23 @@ inline QString Ts2String(int64_t ts, int prec) bool matchCPUVisible(struct kshark_context *kshark_ctx, struct kshark_entry *e, int cpu); + +QString getFile(QWidget *parent, + const QString &windowName, + const QString &filter, + QString &lastFilePath); + +QStringList getFiles(QWidget *parent, + const QString &windowName, + const QString &filter, + QString &lastFilePath); + +QString getSaveFile(QWidget *parent, + const QString &windowName, + const QString &filter, + const QString &extension, + QString &lastFilePath); + }; // KsUtils /** Identifier of the Dual Marker active state. */