Chris@45: /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ Chris@45: Chris@45: /* Chris@45: Sonic Visualiser Chris@45: An audio file viewer and annotation editor. Chris@45: Centre for Digital Music, Queen Mary, University of London. Chris@45: This file copyright 2006-2007 Chris Cannam and QMUL. Chris@45: Chris@45: This program is free software; you can redistribute it and/or Chris@45: modify it under the terms of the GNU General Public License as Chris@45: published by the Free Software Foundation; either version 2 of the Chris@45: License, or (at your option) any later version. See the file Chris@45: COPYING included with this distribution for more information. Chris@45: */ Chris@45: Chris@45: #include "MainWindowBase.h" Chris@46: #include "Document.h" Chris@45: Chris@45: #include "view/Pane.h" Chris@45: #include "view/PaneStack.h" Chris@479: #include "data/model/ReadOnlyWaveFileModel.h" Chris@477: #include "data/model/WritableWaveFileModel.h" Chris@45: #include "data/model/SparseOneDimensionalModel.h" Chris@45: #include "data/model/NoteModel.h" matthiasm@267: #include "data/model/FlexiNoteModel.h" Chris@45: #include "data/model/Labeller.h" Chris@124: #include "data/model/TabularModel.h" Chris@45: #include "view/ViewManager.h" Chris@45: Chris@45: #include "layer/WaveformLayer.h" Chris@45: #include "layer/TimeRulerLayer.h" Chris@45: #include "layer/TimeInstantLayer.h" Chris@45: #include "layer/TimeValueLayer.h" Chris@45: #include "layer/Colour3DPlotLayer.h" Chris@45: #include "layer/SliceLayer.h" Chris@45: #include "layer/SliceableLayer.h" Chris@45: #include "layer/ImageLayer.h" Chris@184: #include "layer/NoteLayer.h" matthiasm@267: #include "layer/FlexiNoteLayer.h" Chris@184: #include "layer/RegionLayer.h" Chris@45: Chris@45: #include "widgets/ListInputDialog.h" Chris@105: #include "widgets/CommandHistory.h" Chris@109: #include "widgets/ProgressDialog.h" Chris@109: #include "widgets/MIDIFileImportDialog.h" Chris@109: #include "widgets/CSVFormatDialog.h" Chris@123: #include "widgets/ModelDataTableDialog.h" Chris@341: #include "widgets/InteractiveFileFinder.h" Chris@45: Chris@468: #include "audio/AudioCallbackPlaySource.h" Chris@475: #include "audio/AudioRecordTarget.h" Chris@468: #include "audio/PlaySpeedRangeMapper.h" Chris@475: Chris@45: #include "data/fileio/DataFileReaderFactory.h" Chris@45: #include "data/fileio/PlaylistFileReader.h" Chris@45: #include "data/fileio/WavFileWriter.h" Chris@45: #include "data/fileio/MIDIFileWriter.h" Chris@45: #include "data/fileio/BZipFileDevice.h" Chris@45: #include "data/fileio/FileSource.h" Chris@152: #include "data/fileio/AudioFileReaderFactory.h" Chris@134: #include "rdf/RDFImporter.h" Chris@45: Chris@45: #include "base/RecentFiles.h" Chris@45: Chris@45: #include "base/PlayParameterRepository.h" Chris@45: #include "base/XmlExportable.h" Chris@45: #include "base/Profiler.h" Chris@45: #include "base/Preferences.h" Chris@217: #include "base/TempWriteFile.h" Chris@217: #include "base/Exceptions.h" Chris@223: #include "base/ResourceFinder.h" Chris@45: Chris@45: #include "data/osc/OSCQueue.h" Chris@157: #include "data/midi/MIDIInput.h" Chris@45: Chris@468: #include Chris@475: #include Chris@468: #include Chris@468: Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@432: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: #include Chris@168: #include Chris@354: #include Chris@45: Chris@45: #include Chris@45: #include Chris@45: #include Chris@45: Chris@45: using std::vector; Chris@45: using std::map; Chris@45: using std::set; Chris@45: Chris@255: #ifdef Q_WS_X11 Chris@255: #define Window X11Window Chris@255: #include Chris@255: #include Chris@255: #include Chris@255: #include Chris@255: Chris@255: static int handle_x11_error(Display *dpy, XErrorEvent *err) Chris@255: { Chris@255: char errstr[256]; Chris@255: XGetErrorText(dpy, err->error_code, errstr, 256); Chris@255: if (err->error_code != BadWindow) { Chris@293: cerr << "Sonic Visualiser: X Error: " Chris@255: << errstr << " " << int(err->error_code) Chris@255: << "\nin major opcode: " Chris@293: << int(err->request_code) << endl; Chris@255: } Chris@255: return 0; Chris@255: } Chris@255: #undef Window Chris@255: #endif Chris@45: Chris@475: MainWindowBase::MainWindowBase(SoundOptions options) : Chris@45: m_document(0), Chris@45: m_paneStack(0), Chris@45: m_viewManager(0), Chris@45: m_timeRulerLayer(0), Chris@475: m_soundOptions(options), Chris@45: m_playSource(0), Chris@475: m_recordTarget(0), Chris@45: m_playTarget(0), Chris@475: m_audioIO(0), Chris@113: m_oscQueue(0), Chris@113: m_oscQueueStarter(0), Chris@157: m_midiInput(0), Chris@45: m_recentFiles("RecentFiles", 20), Chris@54: m_recentTransforms("RecentTransforms", 20), Chris@45: m_documentModified(false), Chris@45: m_openingAudioFile(false), Chris@45: m_abandoning(false), Chris@121: m_labeller(0), Chris@357: m_lastPlayStatusSec(0), Chris@357: m_initialDarkBackground(false), Chris@378: m_defaultFfwdRwdStep(2, 0), Chris@483: m_audioRecordMode(RecordCreateAdditionalModel), Chris@390: m_statusLabel(0), Chris@390: m_menuShortcutMapper(0) Chris@45: { Chris@113: Profiler profiler("MainWindowBase::MainWindowBase"); Chris@113: Chris@475: if (options & WithAudioInput) { Chris@475: if (!(options & WithAudioOutput)) { Chris@475: cerr << "WARNING: MainWindowBase: WithAudioInput requires WithAudioOutput -- recording will not work" << endl; Chris@475: } Chris@475: } Chris@475: Chris@438: qRegisterMetaType("sv_frame_t"); Chris@438: qRegisterMetaType("sv_samplerate_t"); Chris@438: Chris@255: #ifdef Q_WS_X11 Chris@255: XSetErrorHandler(handle_x11_error); Chris@255: #endif Chris@255: Chris@452: connect(this, SIGNAL(hideSplash()), this, SLOT(emitHideSplash())); Chris@452: Chris@45: connect(CommandHistory::getInstance(), SIGNAL(commandExecuted()), Chris@45: this, SLOT(documentModified())); Chris@45: connect(CommandHistory::getInstance(), SIGNAL(documentRestored()), Chris@45: this, SLOT(documentRestored())); Chris@45: Chris@45: m_viewManager = new ViewManager(); Chris@45: connect(m_viewManager, SIGNAL(selectionChanged()), Chris@45: this, SLOT(updateMenuStates())); Chris@45: connect(m_viewManager, SIGNAL(inProgressSelectionChanged()), Chris@45: this, SLOT(inProgressSelectionChanged())); Chris@45: Chris@105: // set a sensible default font size for views -- cannot do this Chris@105: // in Preferences, which is in base and not supposed to use QtGui Chris@436: int viewFontSize = int(QApplication::font().pointSize() * 0.9); Chris@105: QSettings settings; Chris@105: settings.beginGroup("Preferences"); Chris@105: viewFontSize = settings.value("view-font-size", viewFontSize).toInt(); Chris@105: settings.setValue("view-font-size", viewFontSize); Chris@105: settings.endGroup(); Chris@105: Chris@45: Preferences::BackgroundMode mode = Chris@45: Preferences::getInstance()->getBackgroundMode(); Chris@45: m_initialDarkBackground = m_viewManager->getGlobalDarkBackground(); Chris@45: if (mode != Preferences::BackgroundFromTheme) { Chris@45: m_viewManager->setGlobalDarkBackground Chris@45: (mode == Preferences::DarkBackground); Chris@45: } Chris@45: Chris@45: m_paneStack = new PaneStack(0, m_viewManager); Chris@45: connect(m_paneStack, SIGNAL(currentPaneChanged(Pane *)), Chris@45: this, SLOT(currentPaneChanged(Pane *))); Chris@45: connect(m_paneStack, SIGNAL(currentLayerChanged(Pane *, Layer *)), Chris@45: this, SLOT(currentLayerChanged(Pane *, Layer *))); Chris@45: connect(m_paneStack, SIGNAL(rightButtonMenuRequested(Pane *, QPoint)), Chris@45: this, SLOT(rightButtonMenuRequested(Pane *, QPoint))); Chris@45: connect(m_paneStack, SIGNAL(contextHelpChanged(const QString &)), Chris@45: this, SLOT(contextHelpChanged(const QString &))); Chris@45: connect(m_paneStack, SIGNAL(paneAdded(Pane *)), Chris@45: this, SLOT(paneAdded(Pane *))); Chris@45: connect(m_paneStack, SIGNAL(paneHidden(Pane *)), Chris@45: this, SLOT(paneHidden(Pane *))); Chris@45: connect(m_paneStack, SIGNAL(paneAboutToBeDeleted(Pane *)), Chris@45: this, SLOT(paneAboutToBeDeleted(Pane *))); Chris@45: connect(m_paneStack, SIGNAL(dropAccepted(Pane *, QStringList)), Chris@45: this, SLOT(paneDropAccepted(Pane *, QStringList))); Chris@45: connect(m_paneStack, SIGNAL(dropAccepted(Pane *, QString)), Chris@45: this, SLOT(paneDropAccepted(Pane *, QString))); Chris@55: connect(m_paneStack, SIGNAL(paneDeleteButtonClicked(Pane *)), Chris@55: this, SLOT(paneDeleteButtonClicked(Pane *))); Chris@45: Chris@57: m_playSource = new AudioCallbackPlaySource(m_viewManager, Chris@57: QApplication::applicationName()); Chris@475: if (m_soundOptions & WithAudioInput) { Chris@475: m_recordTarget = new AudioRecordTarget(m_viewManager, Chris@475: QApplication::applicationName()); Chris@485: connect(m_recordTarget, SIGNAL(recordDurationChanged(sv_frame_t, sv_samplerate_t)), Chris@485: this, SLOT(recordDurationChanged(sv_frame_t, sv_samplerate_t))); Chris@475: } Chris@45: Chris@436: connect(m_playSource, SIGNAL(sampleRateMismatch(sv_samplerate_t, sv_samplerate_t, bool)), Chris@436: this, SLOT(sampleRateMismatch(sv_samplerate_t, sv_samplerate_t, bool))); Chris@45: connect(m_playSource, SIGNAL(audioOverloadPluginDisabled()), Chris@45: this, SLOT(audioOverloadPluginDisabled())); Chris@130: connect(m_playSource, SIGNAL(audioTimeStretchMultiChannelDisabled()), Chris@130: this, SLOT(audioTimeStretchMultiChannelDisabled())); Chris@45: Chris@45: connect(m_viewManager, SIGNAL(outputLevelsChanged(float, float)), Chris@45: this, SLOT(outputLevelsChanged(float, float))); Chris@45: Chris@435: connect(m_viewManager, SIGNAL(playbackFrameChanged(sv_frame_t)), Chris@435: this, SLOT(playbackFrameChanged(sv_frame_t))); Chris@435: Chris@435: connect(m_viewManager, SIGNAL(globalCentreFrameChanged(sv_frame_t)), Chris@435: this, SLOT(globalCentreFrameChanged(sv_frame_t))); Chris@435: Chris@435: connect(m_viewManager, SIGNAL(viewCentreFrameChanged(View *, sv_frame_t)), Chris@435: this, SLOT(viewCentreFrameChanged(View *, sv_frame_t))); Chris@366: Chris@366: connect(m_viewManager, SIGNAL(viewZoomLevelChanged(View *, int, bool)), Chris@366: this, SLOT(viewZoomLevelChanged(View *, int, bool))); Chris@45: Chris@45: connect(Preferences::getInstance(), Chris@45: SIGNAL(propertyChanged(PropertyContainer::PropertyName)), Chris@45: this, Chris@45: SLOT(preferenceChanged(PropertyContainer::PropertyName))); Chris@45: Chris@45: Labeller::ValueType labellerType = Labeller::ValueFromTwoLevelCounter; Chris@45: settings.beginGroup("MainWindow"); Chris@230: Chris@45: labellerType = (Labeller::ValueType) Chris@45: settings.value("labellertype", (int)labellerType).toInt(); Chris@45: int cycle = settings.value("labellercycle", 4).toInt(); Chris@230: Chris@45: settings.endGroup(); Chris@45: Chris@45: m_labeller = new Labeller(labellerType); Chris@45: m_labeller->setCounterCycleSize(cycle); Chris@113: Chris@475: if (m_soundOptions & WithMIDIInput) { Chris@161: m_midiInput = new MIDIInput(QApplication::applicationName(), this); Chris@161: } Chris@452: Chris@452: QTimer::singleShot(1500, this, SIGNAL(hideSplash())); Chris@45: } Chris@45: Chris@45: MainWindowBase::~MainWindowBase() Chris@45: { Chris@233: SVDEBUG << "MainWindowBase::~MainWindowBase" << endl; Chris@468: delete m_playTarget; Chris@45: delete m_playSource; Chris@475: delete m_audioIO; Chris@475: delete m_recordTarget; Chris@45: delete m_viewManager; Chris@45: delete m_oscQueue; Chris@304: delete m_oscQueueStarter; Chris@157: delete m_midiInput; Chris@45: Profiles::getInstance()->dump(); Chris@45: } Chris@45: Chris@113: void Chris@452: MainWindowBase::emitHideSplash() Chris@452: { Chris@452: emit hideSplash(this); Chris@452: } Chris@452: Chris@452: void Chris@354: MainWindowBase::finaliseMenus() Chris@354: { Chris@390: delete m_menuShortcutMapper; Chris@390: m_menuShortcutMapper = 0; Chris@390: Chris@391: foreach (QShortcut *sc, m_appShortcuts) { Chris@391: delete sc; Chris@391: } Chris@391: m_appShortcuts.clear(); Chris@391: Chris@354: QMenuBar *mb = menuBar(); Chris@394: Chris@396: // This used to find all children of QMenu type, and call Chris@396: // finaliseMenu on those. But it seems we are getting hold of some Chris@396: // menus that way that are not actually active in the menu bar and Chris@396: // are not returned in their parent menu's actions() list, and if Chris@396: // we finalise those, we end up with duplicate shortcuts in the Chris@396: // app shortcut mapper. So we should do this by descending the Chris@396: // menu tree through only those menus accessible via actions() Chris@396: // from their parents instead. Chris@396: Chris@394: QList menus = mb->findChildren Chris@394: (QString(), Qt::FindDirectChildrenOnly); Chris@394: Chris@354: foreach (QMenu *menu, menus) { Chris@354: if (menu) finaliseMenu(menu); Chris@354: } Chris@354: } Chris@354: Chris@354: void Chris@358: MainWindowBase::finaliseMenu(QMenu * Chris@358: #ifdef Q_OS_MAC Chris@358: menu Chris@358: #endif Chris@358: ) Chris@354: { Chris@354: #ifdef Q_OS_MAC Chris@354: // See https://bugreports.qt-project.org/browse/QTBUG-38256 and Chris@354: // our issue #890 http://code.soundsoftware.ac.uk/issues/890 -- Chris@354: // single-key shortcuts that are associated only with a menu Chris@384: // action (and not with a toolbar button) do not work with Qt 5.x Chris@384: // under OS/X. Chris@354: // Chris@354: // Apparently Cocoa never handled them as a matter of course, but Chris@354: // earlier versions of Qt picked them up as widget shortcuts and Chris@354: // handled them anyway. That behaviour was removed to fix a crash Chris@354: // when invoking a menu while its window was overridden by a modal Chris@354: // dialog (https://bugreports.qt-project.org/browse/QTBUG-30657). Chris@354: // Chris@354: // This workaround restores the single-key shortcut behaviour by Chris@384: // searching in menus for single-key shortcuts that are associated Chris@384: // only with the menu and not with a toolbar button, and Chris@384: // augmenting them with global application shortcuts that invoke Chris@384: // the relevant actions, testing whether the actions are enabled Chris@384: // on invocation. Chris@354: // Chris@384: // (Previously this acted on all single-key shortcuts in menus, Chris@384: // and it removed the shortcut from the action when it created Chris@384: // each new global one, in order to avoid an "ambiguous shortcut" Chris@384: // error in the case where the action was also associated with a Chris@384: // toolbar button. But that has the unwelcome side-effect of Chris@384: // removing the shortcut hint from the menu entry. So now we leave Chris@384: // the shortcut in the menu action as well as creating a global Chris@384: // one, and we only act on shortcuts that have no toolbar button, Chris@384: // i.e. that will not otherwise work. The downside is that if this Chris@384: // bug is fixed in a future Qt release, we will start getting Chris@384: // "ambiguous shortcut" errors from the menu entry actions and Chris@384: // will need to update the code.) Chris@354: Chris@443: // Update: The bug was fixed in Qt 5.4 for shortcuts with no Chris@443: // modifier, and I believe it is fixed in Qt 5.5 for shortcuts Chris@443: // with Shift modifiers. The below reflects that Chris@443: Chris@443: #if (QT_VERSION < QT_VERSION_CHECK(5, 5, 0)) Chris@443: Chris@390: if (!m_menuShortcutMapper) { Chris@390: m_menuShortcutMapper = new QSignalMapper(this); Chris@392: connect(m_menuShortcutMapper, SIGNAL(mapped(QObject *)), Chris@392: this, SLOT(menuActionMapperInvoked(QObject *))); Chris@390: } Chris@390: Chris@354: foreach (QAction *a, menu->actions()) { Chris@394: Chris@394: if (a->isSeparator()) { Chris@394: continue; Chris@394: } else if (a->menu()) { Chris@394: finaliseMenu(a->menu()); Chris@394: } else { Chris@394: Chris@394: QWidgetList ww = a->associatedWidgets(); Chris@394: bool hasButton = false; Chris@394: foreach (QWidget *w, ww) { Chris@394: if (qobject_cast(w)) { Chris@394: hasButton = true; Chris@394: break; Chris@394: } Chris@394: } Chris@394: if (hasButton) continue; Chris@394: QKeySequence sc = a->shortcut(); Chris@399: Chris@399: // Note that the set of "single-key shortcuts" that aren't Chris@399: // working and that we need to handle here includes those Chris@399: // with the Shift modifier mask as well as those with no Chris@399: // modifier at all Chris@443: #if (QT_VERSION >= QT_VERSION_CHECK(5, 5, 0)) Chris@443: // Nothing needed Chris@443: if (false) { Chris@443: #elif (QT_VERSION >= QT_VERSION_CHECK(5, 4, 0)) Chris@443: if (sc.count() == 1 && Chris@443: (sc[0] & Qt::KeyboardModifierMask) == Qt::ShiftModifier) { Chris@443: #else Chris@399: if (sc.count() == 1 && Chris@399: ((sc[0] & Qt::KeyboardModifierMask) == Qt::NoModifier || Chris@399: (sc[0] & Qt::KeyboardModifierMask) == Qt::ShiftModifier)) { Chris@443: #endif Chris@394: QShortcut *newSc = new QShortcut(sc, a->parentWidget()); Chris@394: QObject::connect(newSc, SIGNAL(activated()), Chris@394: m_menuShortcutMapper, SLOT(map())); Chris@394: m_menuShortcutMapper->setMapping(newSc, a); Chris@394: m_appShortcuts.push_back(newSc); Chris@384: } Chris@384: } Chris@354: } Chris@354: #endif Chris@443: #endif Chris@354: } Chris@354: Chris@354: void Chris@354: MainWindowBase::menuActionMapperInvoked(QObject *o) Chris@354: { Chris@354: QAction *a = qobject_cast(o); Chris@354: if (a && a->isEnabled()) { Chris@354: a->trigger(); Chris@354: } Chris@354: } Chris@354: Chris@354: void Chris@168: MainWindowBase::resizeConstrained(QSize size) Chris@168: { Chris@168: QDesktopWidget *desktop = QApplication::desktop(); Chris@168: QRect available = desktop->availableGeometry(); Chris@168: QSize actual(std::min(size.width(), available.width()), Chris@168: std::min(size.height(), available.height())); Chris@168: resize(actual); Chris@168: } Chris@168: Chris@168: void Chris@304: MainWindowBase::startOSCQueue() Chris@304: { Chris@304: m_oscQueueStarter = new OSCQueueStarter(this); Chris@304: connect(m_oscQueueStarter, SIGNAL(finished()), this, SLOT(oscReady())); Chris@304: m_oscQueueStarter->start(); Chris@304: } Chris@304: Chris@304: void Chris@113: MainWindowBase::oscReady() Chris@113: { Chris@113: if (m_oscQueue && m_oscQueue->isOK()) { Chris@113: connect(m_oscQueue, SIGNAL(messagesAvailable()), this, SLOT(pollOSC())); Chris@113: QTimer *oscTimer = new QTimer(this); Chris@113: connect(oscTimer, SIGNAL(timeout()), this, SLOT(pollOSC())); Chris@113: oscTimer->start(1000); Chris@293: cerr << "Finished setting up OSC interface" << endl; Chris@113: } Chris@113: } Chris@113: Chris@45: QString Chris@45: MainWindowBase::getOpenFileName(FileFinder::FileType type) Chris@45: { Chris@45: FileFinder *ff = FileFinder::getInstance(); Chris@358: Chris@358: if (type == FileFinder::AnyFile) { Chris@45: if (getMainModel() != 0 && Chris@45: m_paneStack != 0 && Chris@45: m_paneStack->getCurrentPane() != 0) { // can import a layer Chris@45: return ff->getOpenFileName(FileFinder::AnyFile, m_sessionFile); Chris@45: } else { Chris@45: return ff->getOpenFileName(FileFinder::SessionOrAudioFile, Chris@45: m_sessionFile); Chris@45: } Chris@358: } Chris@358: Chris@358: QString lastPath = m_sessionFile; Chris@358: Chris@358: if (type == FileFinder::AudioFile) { Chris@358: lastPath = m_audioFile; Chris@45: } Chris@358: Chris@358: return ff->getOpenFileName(type, lastPath); Chris@45: } Chris@45: Chris@45: QString Chris@45: MainWindowBase::getSaveFileName(FileFinder::FileType type) Chris@45: { Chris@358: QString lastPath = m_sessionFile; Chris@358: Chris@358: if (type == FileFinder::AudioFile) { Chris@358: lastPath = m_audioFile; Chris@358: } Chris@358: Chris@45: FileFinder *ff = FileFinder::getInstance(); Chris@358: return ff->getSaveFileName(type, lastPath); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::registerLastOpenedFilePath(FileFinder::FileType type, QString path) Chris@45: { Chris@45: FileFinder *ff = FileFinder::getInstance(); Chris@45: ff->registerLastOpenedFilePath(type, path); Chris@45: } Chris@45: Chris@222: QString Chris@222: MainWindowBase::getDefaultSessionTemplate() const Chris@222: { Chris@231: QSettings settings; Chris@231: settings.beginGroup("MainWindow"); Chris@231: QString templateName = settings.value("sessiontemplate", "").toString(); Chris@231: if (templateName == "") templateName = "default"; Chris@231: return templateName; Chris@222: } Chris@222: Chris@222: void Chris@251: MainWindowBase::setDefaultSessionTemplate(QString n) Chris@251: { Chris@251: QSettings settings; Chris@251: settings.beginGroup("MainWindow"); Chris@251: settings.setValue("sessiontemplate", n); Chris@251: } Chris@251: Chris@251: void Chris@45: MainWindowBase::updateMenuStates() Chris@45: { Chris@45: Pane *currentPane = 0; Chris@45: Layer *currentLayer = 0; Chris@45: Chris@45: if (m_paneStack) currentPane = m_paneStack->getCurrentPane(); Chris@45: if (currentPane) currentLayer = currentPane->getSelectedLayer(); Chris@45: Chris@73: bool havePrevPane = false, haveNextPane = false; Chris@73: bool havePrevLayer = false, haveNextLayer = false; Chris@73: Chris@73: if (currentPane) { Chris@73: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@73: if (m_paneStack->getPane(i) == currentPane) { Chris@73: if (i > 0) havePrevPane = true; Chris@73: if (i < m_paneStack->getPaneCount()-1) haveNextPane = true; Chris@73: break; Chris@73: } Chris@73: } Chris@403: // the prev/next layer commands actually include the pane Chris@403: // itself as one of the selectables -- so we always have a Chris@403: // prev and next layer, as long as we have a pane with at Chris@403: // least one layer in it Chris@403: if (currentPane->getLayerCount() > 0) { Chris@403: havePrevLayer = true; Chris@403: haveNextLayer = true; Chris@73: } Chris@73: } Chris@73: Chris@45: bool haveCurrentPane = Chris@45: (currentPane != 0); Chris@45: bool haveCurrentLayer = Chris@45: (haveCurrentPane && Chris@45: (currentLayer != 0)); Chris@45: bool haveMainModel = Chris@45: (getMainModel() != 0); Chris@45: bool havePlayTarget = Chris@475: (m_playTarget != 0 || m_audioIO != 0); Chris@45: bool haveSelection = Chris@45: (m_viewManager && Chris@45: !m_viewManager->getSelections().empty()); Chris@45: bool haveCurrentEditableLayer = Chris@45: (haveCurrentLayer && Chris@45: currentLayer->isLayerEditable()); Chris@45: bool haveCurrentTimeInstantsLayer = Chris@45: (haveCurrentLayer && Chris@45: dynamic_cast(currentLayer)); Chris@184: bool haveCurrentDurationLayer = Chris@184: (haveCurrentLayer && Chris@184: (dynamic_cast(currentLayer) || matthiasm@267: dynamic_cast(currentLayer) || Chris@184: dynamic_cast(currentLayer))); Chris@45: bool haveCurrentColour3DPlot = Chris@45: (haveCurrentLayer && Chris@45: dynamic_cast(currentLayer)); Chris@45: bool haveClipboardContents = Chris@45: (m_viewManager && Chris@45: !m_viewManager->getClipboard().empty()); Chris@146: bool haveTabularLayer = Chris@146: (haveCurrentLayer && Chris@146: dynamic_cast(currentLayer->getModel())); Chris@45: Chris@45: emit canAddPane(haveMainModel); Chris@45: emit canDeleteCurrentPane(haveCurrentPane); Chris@45: emit canZoom(haveMainModel && haveCurrentPane); Chris@45: emit canScroll(haveMainModel && haveCurrentPane); Chris@45: emit canAddLayer(haveMainModel && haveCurrentPane); Chris@45: emit canImportMoreAudio(haveMainModel); Chris@259: emit canReplaceMainAudio(haveMainModel); Chris@45: emit canImportLayer(haveMainModel && haveCurrentPane); Chris@45: emit canExportAudio(haveMainModel); Chris@289: emit canChangeSessionTemplate(haveMainModel); Chris@45: emit canExportLayer(haveMainModel && Chris@45: (haveCurrentEditableLayer || haveCurrentColour3DPlot)); Chris@45: emit canExportImage(haveMainModel && haveCurrentPane); Chris@45: emit canDeleteCurrentLayer(haveCurrentLayer); Chris@45: emit canRenameLayer(haveCurrentLayer); Chris@45: emit canEditLayer(haveCurrentEditableLayer); Chris@146: emit canEditLayerTabular(haveCurrentEditableLayer || haveTabularLayer); Chris@45: emit canMeasureLayer(haveCurrentLayer); Chris@45: emit canSelect(haveMainModel && haveCurrentPane); Chris@188: emit canPlay(haveMainModel && havePlayTarget); Chris@483: emit canRecord(m_recordTarget != 0); Chris@453: emit canFfwd(haveMainModel); Chris@453: emit canRewind(haveMainModel); Chris@87: emit canPaste(haveClipboardContents); Chris@45: emit canInsertInstant(haveCurrentPane); Chris@45: emit canInsertInstantsAtBoundaries(haveCurrentPane && haveSelection); Chris@184: emit canInsertItemAtSelection(haveCurrentPane && haveSelection && haveCurrentDurationLayer); Chris@45: emit canRenumberInstants(haveCurrentTimeInstantsLayer && haveSelection); Chris@45: emit canPlaySelection(haveMainModel && havePlayTarget && haveSelection); Chris@45: emit canClearSelection(haveSelection); Chris@45: emit canEditSelection(haveSelection && haveCurrentEditableLayer); Chris@45: emit canSave(m_sessionFile != "" && m_documentModified); Chris@359: emit canSaveAs(haveMainModel); Chris@73: emit canSelectPreviousPane(havePrevPane); Chris@73: emit canSelectNextPane(haveNextPane); Chris@73: emit canSelectPreviousLayer(havePrevLayer); Chris@73: emit canSelectNextLayer(haveNextLayer); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::documentModified() Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::documentModified" << endl; Chris@45: Chris@45: if (!m_documentModified) { Chris@45: //!!! this in subclass implementation? Chris@45: setWindowTitle(tr("%1 (modified)").arg(windowTitle())); Chris@45: } Chris@45: Chris@45: m_documentModified = true; Chris@45: updateMenuStates(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::documentRestored() Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::documentRestored" << endl; Chris@45: Chris@45: if (m_documentModified) { Chris@45: //!!! this in subclass implementation? Chris@45: QString wt(windowTitle()); Chris@45: wt.replace(tr(" (modified)"), ""); Chris@45: setWindowTitle(wt); Chris@45: } Chris@45: Chris@45: m_documentModified = false; Chris@45: updateMenuStates(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::playLoopToggled() Chris@45: { Chris@45: QAction *action = dynamic_cast(sender()); Chris@45: Chris@45: if (action) { Chris@45: m_viewManager->setPlayLoopMode(action->isChecked()); Chris@45: } else { Chris@45: m_viewManager->setPlayLoopMode(!m_viewManager->getPlayLoopMode()); Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::playSelectionToggled() Chris@45: { Chris@45: QAction *action = dynamic_cast(sender()); Chris@45: Chris@45: if (action) { Chris@45: m_viewManager->setPlaySelectionMode(action->isChecked()); Chris@45: } else { Chris@45: m_viewManager->setPlaySelectionMode(!m_viewManager->getPlaySelectionMode()); Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::playSoloToggled() Chris@45: { Chris@45: QAction *action = dynamic_cast(sender()); Chris@45: Chris@45: if (action) { Chris@45: m_viewManager->setPlaySoloMode(action->isChecked()); Chris@45: } else { Chris@45: m_viewManager->setPlaySoloMode(!m_viewManager->getPlaySoloMode()); Chris@45: } Chris@45: Chris@45: if (m_viewManager->getPlaySoloMode()) { Chris@45: currentPaneChanged(m_paneStack->getCurrentPane()); Chris@45: } else { Chris@45: m_viewManager->setPlaybackModel(0); Chris@45: if (m_playSource) { Chris@45: m_playSource->clearSoloModelSet(); Chris@45: } Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::currentPaneChanged(Pane *p) Chris@45: { Chris@45: updateMenuStates(); Chris@45: updateVisibleRangeDisplay(p); Chris@45: Chris@45: if (!p) return; Chris@45: Chris@45: if (!(m_viewManager && Chris@45: m_playSource && Chris@45: m_viewManager->getPlaySoloMode())) { Chris@45: if (m_viewManager) m_viewManager->setPlaybackModel(0); Chris@45: return; Chris@45: } Chris@45: Chris@45: Model *prevPlaybackModel = m_viewManager->getPlaybackModel(); Chris@60: Chris@93: // What we want here is not the currently playing frame (unless we Chris@93: // are about to clear out the audio playback buffers -- which may Chris@93: // or may not be possible, depending on the audio driver). What Chris@93: // we want is the frame that was last committed to the soundcard Chris@93: // buffers, as the audio driver will continue playing up to that Chris@93: // frame before switching to whichever one we decide we want to Chris@93: // switch to, regardless of our efforts. Chris@93: Chris@435: sv_frame_t frame = m_playSource->getCurrentBufferedFrame(); Chris@93: Chris@388: cerr << "currentPaneChanged: current frame (in ref model) = " << frame << endl; Chris@45: Chris@45: View::ModelSet soloModels = p->getModels(); Chris@45: Chris@57: View::ModelSet sources; Chris@57: for (View::ModelSet::iterator mi = soloModels.begin(); Chris@57: mi != soloModels.end(); ++mi) { Chris@190: // If a model in this pane is derived from something else, Chris@190: // then we want to play that model as well -- if the model Chris@190: // that's derived from it is not something that is itself Chris@190: // individually playable (e.g. a waveform) Chris@190: if (*mi && Chris@190: !dynamic_cast(*mi) && Chris@190: (*mi)->getSourceModel()) { Chris@57: sources.insert((*mi)->getSourceModel()); Chris@57: } Chris@57: } Chris@57: for (View::ModelSet::iterator mi = sources.begin(); Chris@57: mi != sources.end(); ++mi) { Chris@57: soloModels.insert(*mi); Chris@57: } Chris@57: Chris@60: //!!! Need an "atomic" way of telling the play source that the Chris@60: //playback model has changed, and changing it on ViewManager -- Chris@60: //the play source should be making the setPlaybackModel call to Chris@60: //ViewManager Chris@60: Chris@45: for (View::ModelSet::iterator mi = soloModels.begin(); Chris@45: mi != soloModels.end(); ++mi) { Chris@45: if (dynamic_cast(*mi)) { Chris@45: m_viewManager->setPlaybackModel(*mi); Chris@45: } Chris@45: } Chris@45: Chris@45: RangeSummarisableTimeValueModel *a = Chris@45: dynamic_cast(prevPlaybackModel); Chris@45: RangeSummarisableTimeValueModel *b = Chris@45: dynamic_cast(m_viewManager-> Chris@45: getPlaybackModel()); Chris@45: Chris@45: m_playSource->setSoloModelSet(soloModels); Chris@45: Chris@45: if (a && b && (a != b)) { Chris@60: if (m_playSource->isPlaying()) m_playSource->play(frame); Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::currentLayerChanged(Pane *p, Layer *) Chris@45: { Chris@45: updateMenuStates(); Chris@45: updateVisibleRangeDisplay(p); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::selectAll() Chris@45: { Chris@45: if (!getMainModel()) return; Chris@45: m_viewManager->setSelection(Selection(getMainModel()->getStartFrame(), Chris@45: getMainModel()->getEndFrame())); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::selectToStart() Chris@45: { Chris@45: if (!getMainModel()) return; Chris@45: m_viewManager->setSelection(Selection(getMainModel()->getStartFrame(), Chris@45: m_viewManager->getGlobalCentreFrame())); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::selectToEnd() Chris@45: { Chris@45: if (!getMainModel()) return; Chris@45: m_viewManager->setSelection(Selection(m_viewManager->getGlobalCentreFrame(), Chris@45: getMainModel()->getEndFrame())); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::selectVisible() Chris@45: { Chris@45: Model *model = getMainModel(); Chris@45: if (!model) return; Chris@45: Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (!currentPane) return; Chris@45: Chris@435: sv_frame_t startFrame, endFrame; Chris@45: Chris@45: if (currentPane->getStartFrame() < 0) startFrame = 0; Chris@45: else startFrame = currentPane->getStartFrame(); Chris@45: Chris@45: if (currentPane->getEndFrame() > model->getEndFrame()) endFrame = model->getEndFrame(); Chris@45: else endFrame = currentPane->getEndFrame(); Chris@45: Chris@45: m_viewManager->setSelection(Selection(startFrame, endFrame)); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::clearSelection() Chris@45: { Chris@45: m_viewManager->clearSelections(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::cut() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (!currentPane) return; Chris@45: Chris@45: Layer *layer = currentPane->getSelectedLayer(); Chris@45: if (!layer) return; Chris@45: Chris@45: Clipboard &clipboard = m_viewManager->getClipboard(); Chris@45: clipboard.clear(); Chris@45: Chris@45: MultiSelection::SelectionList selections = m_viewManager->getSelections(); Chris@45: Chris@45: CommandHistory::getInstance()->startCompoundOperation(tr("Cut"), true); Chris@45: Chris@45: for (MultiSelection::SelectionList::iterator i = selections.begin(); Chris@45: i != selections.end(); ++i) { Chris@86: layer->copy(currentPane, *i, clipboard); Chris@45: layer->deleteSelection(*i); Chris@45: } Chris@45: Chris@45: CommandHistory::getInstance()->endCompoundOperation(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::copy() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (!currentPane) return; Chris@45: Chris@45: Layer *layer = currentPane->getSelectedLayer(); Chris@45: if (!layer) return; Chris@45: Chris@45: Clipboard &clipboard = m_viewManager->getClipboard(); Chris@45: clipboard.clear(); Chris@45: Chris@45: MultiSelection::SelectionList selections = m_viewManager->getSelections(); Chris@45: Chris@45: for (MultiSelection::SelectionList::iterator i = selections.begin(); Chris@45: i != selections.end(); ++i) { Chris@86: layer->copy(currentPane, *i, clipboard); Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::paste() Chris@45: { Chris@215: pasteRelative(0); Chris@215: } Chris@215: Chris@215: void Chris@215: MainWindowBase::pasteAtPlaybackPosition() Chris@215: { Chris@435: sv_frame_t pos = getFrame(); Chris@215: Clipboard &clipboard = m_viewManager->getClipboard(); Chris@215: if (!clipboard.empty()) { Chris@435: sv_frame_t firstEventFrame = clipboard.getPoints()[0].getFrame(); Chris@435: sv_frame_t offset = 0; Chris@215: if (firstEventFrame < 0) { Chris@366: offset = pos - firstEventFrame; Chris@354: } else if (firstEventFrame < pos) { Chris@366: offset = pos - firstEventFrame; Chris@215: } else { Chris@366: offset = -(firstEventFrame - pos); Chris@215: } Chris@215: pasteRelative(offset); Chris@215: } Chris@215: } Chris@215: Chris@215: void Chris@435: MainWindowBase::pasteRelative(sv_frame_t offset) Chris@215: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (!currentPane) return; Chris@45: Chris@45: Layer *layer = currentPane->getSelectedLayer(); Chris@45: Chris@45: Clipboard &clipboard = m_viewManager->getClipboard(); Chris@87: Chris@98: bool inCompound = false; Chris@87: Chris@87: if (!layer || !layer->isLayerEditable()) { Chris@87: Chris@87: CommandHistory::getInstance()->startCompoundOperation Chris@87: (tr("Paste"), true); Chris@87: Chris@87: // no suitable current layer: create one of the most Chris@87: // appropriate sort Chris@87: LayerFactory::LayerType type = Chris@87: LayerFactory::getInstance()->getLayerTypeForClipboardContents(clipboard); Chris@87: layer = m_document->createEmptyLayer(type); Chris@87: Chris@87: if (!layer) { Chris@87: CommandHistory::getInstance()->endCompoundOperation(); Chris@87: return; Chris@45: } Chris@87: Chris@87: m_document->addLayerToView(currentPane, layer); Chris@87: m_paneStack->setCurrentLayer(currentPane, layer); Chris@87: Chris@87: inCompound = true; Chris@45: } Chris@45: Chris@215: layer->paste(currentPane, clipboard, offset, true); Chris@45: Chris@87: if (inCompound) CommandHistory::getInstance()->endCompoundOperation(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::deleteSelected() Chris@45: { Chris@45: if (m_paneStack->getCurrentPane() && Chris@45: m_paneStack->getCurrentPane()->getSelectedLayer()) { Chris@45: Chris@45: Layer *layer = m_paneStack->getCurrentPane()->getSelectedLayer(); Chris@45: Chris@409: if (m_viewManager) { Chris@409: Chris@409: if (m_viewManager->getToolMode() == ViewManager::MeasureMode) { Chris@409: Chris@409: layer->deleteCurrentMeasureRect(); Chris@45: Chris@409: } else { Chris@409: Chris@409: MultiSelection::SelectionList selections = Chris@409: m_viewManager->getSelections(); Chris@409: Chris@409: for (MultiSelection::SelectionList::iterator i = selections.begin(); Chris@409: i != selections.end(); ++i) { Chris@409: layer->deleteSelection(*i); Chris@409: } Chris@45: } Chris@45: } Chris@45: } Chris@45: } Chris@45: Chris@161: // FrameTimer method Chris@161: Chris@435: sv_frame_t Chris@161: MainWindowBase::getFrame() const Chris@161: { Chris@161: if (m_playSource && m_playSource->isPlaying()) { Chris@161: return m_playSource->getCurrentPlayingFrame(); Chris@161: } else { Chris@161: return m_viewManager->getPlaybackFrame(); Chris@161: } Chris@161: } Chris@161: Chris@45: void Chris@45: MainWindowBase::insertInstant() Chris@45: { Chris@161: insertInstantAt(getFrame()); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::insertInstantsAtBoundaries() Chris@45: { Chris@45: MultiSelection::SelectionList selections = m_viewManager->getSelections(); Chris@45: for (MultiSelection::SelectionList::iterator i = selections.begin(); Chris@45: i != selections.end(); ++i) { Chris@435: sv_frame_t start = i->getStartFrame(); Chris@435: sv_frame_t end = i->getEndFrame(); Chris@45: if (start != end) { Chris@184: insertInstantAt(start); Chris@184: insertInstantAt(end); Chris@45: } Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@435: MainWindowBase::insertInstantAt(sv_frame_t frame) Chris@45: { Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: if (!pane) { Chris@45: return; Chris@45: } Chris@45: Chris@74: frame = pane->alignFromReference(frame); Chris@74: Chris@45: Layer *layer = dynamic_cast Chris@45: (pane->getSelectedLayer()); Chris@45: Chris@45: if (!layer) { Chris@45: for (int i = pane->getLayerCount(); i > 0; --i) { Chris@45: layer = dynamic_cast(pane->getLayer(i - 1)); Chris@45: if (layer) break; Chris@45: } Chris@45: Chris@45: if (!layer) { Chris@45: CommandHistory::getInstance()->startCompoundOperation Chris@45: (tr("Add Point"), true); Chris@45: layer = m_document->createEmptyLayer(LayerFactory::TimeInstants); Chris@45: if (layer) { Chris@45: m_document->addLayerToView(pane, layer); Chris@45: m_paneStack->setCurrentLayer(pane, layer); Chris@45: } Chris@45: CommandHistory::getInstance()->endCompoundOperation(); Chris@45: } Chris@45: } Chris@45: Chris@45: if (layer) { Chris@45: Chris@45: Model *model = layer->getModel(); Chris@45: SparseOneDimensionalModel *sodm = dynamic_cast Chris@45: (model); Chris@45: Chris@45: if (sodm) { Chris@45: SparseOneDimensionalModel::Point point(frame, ""); Chris@45: Chris@45: SparseOneDimensionalModel::Point prevPoint(0); Chris@45: bool havePrevPoint = false; Chris@45: Chris@45: SparseOneDimensionalModel::EditCommand *command = Chris@45: new SparseOneDimensionalModel::EditCommand(sodm, tr("Add Point")); Chris@45: Chris@409: if (m_labeller) { Chris@409: Chris@409: if (m_labeller->requiresPrevPoint()) { Chris@409: Chris@409: SparseOneDimensionalModel::PointList prevPoints = Chris@409: sodm->getPreviousPoints(frame); Chris@409: Chris@409: if (!prevPoints.empty()) { Chris@409: prevPoint = *prevPoints.begin(); Chris@409: havePrevPoint = true; Chris@409: } Chris@45: } Chris@45: Chris@45: m_labeller->setSampleRate(sodm->getSampleRate()); Chris@45: Chris@352: if (m_labeller->actingOnPrevPoint() && havePrevPoint) { Chris@45: command->deletePoint(prevPoint); Chris@45: } Chris@45: Chris@45: m_labeller->label Chris@45: (point, havePrevPoint ? &prevPoint : 0); Chris@45: Chris@352: if (m_labeller->actingOnPrevPoint() && havePrevPoint) { Chris@45: command->addPoint(prevPoint); Chris@45: } Chris@45: } Chris@45: Chris@45: command->addPoint(point); Chris@45: Chris@45: command->setName(tr("Add Point at %1 s") Chris@45: .arg(RealTime::frame2RealTime Chris@45: (frame, Chris@45: sodm->getSampleRate()) Chris@45: .toText(false).c_str())); Chris@45: Chris@108: Command *c = command->finish(); Chris@108: if (c) CommandHistory::getInstance()->addCommand(c, false); Chris@45: } Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@184: MainWindowBase::insertItemAtSelection() Chris@184: { Chris@184: MultiSelection::SelectionList selections = m_viewManager->getSelections(); Chris@184: for (MultiSelection::SelectionList::iterator i = selections.begin(); Chris@184: i != selections.end(); ++i) { Chris@435: sv_frame_t start = i->getStartFrame(); Chris@435: sv_frame_t end = i->getEndFrame(); Chris@184: if (start < end) { Chris@184: insertItemAt(start, end - start); Chris@184: } Chris@184: } Chris@184: } Chris@184: Chris@184: void Chris@435: MainWindowBase::insertItemAt(sv_frame_t frame, sv_frame_t duration) Chris@184: { Chris@184: Pane *pane = m_paneStack->getCurrentPane(); Chris@184: if (!pane) { Chris@184: return; Chris@184: } Chris@184: Chris@184: // ugh! Chris@184: Chris@435: sv_frame_t alignedStart = pane->alignFromReference(frame); Chris@435: sv_frame_t alignedEnd = pane->alignFromReference(frame + duration); Chris@184: if (alignedStart >= alignedEnd) return; Chris@435: sv_frame_t alignedDuration = alignedEnd - alignedStart; Chris@184: Chris@184: Command *c = 0; Chris@184: Chris@184: QString name = tr("Add Item at %1 s") Chris@184: .arg(RealTime::frame2RealTime Chris@184: (alignedStart, Chris@184: getMainModel()->getSampleRate()) Chris@184: .toText(false).c_str()); Chris@184: Chris@184: Layer *layer = pane->getSelectedLayer(); Chris@184: if (!layer) return; Chris@184: Chris@184: RegionModel *rm = dynamic_cast(layer->getModel()); Chris@184: if (rm) { Chris@184: RegionModel::Point point(alignedStart, Chris@185: rm->getValueMaximum() + 1, Chris@184: alignedDuration, Chris@184: ""); Chris@184: RegionModel::EditCommand *command = Chris@184: new RegionModel::EditCommand(rm, tr("Add Point")); Chris@184: command->addPoint(point); Chris@184: command->setName(name); Chris@184: c = command->finish(); Chris@184: } Chris@184: Chris@184: if (c) { Chris@184: CommandHistory::getInstance()->addCommand(c, false); Chris@184: return; Chris@184: } Chris@184: Chris@184: NoteModel *nm = dynamic_cast(layer->getModel()); Chris@184: if (nm) { Chris@184: NoteModel::Point point(alignedStart, Chris@409: nm->getValueMinimum(), Chris@184: alignedDuration, Chris@184: 1.f, Chris@184: ""); Chris@184: NoteModel::EditCommand *command = Chris@184: new NoteModel::EditCommand(nm, tr("Add Point")); Chris@184: command->addPoint(point); Chris@184: command->setName(name); Chris@184: c = command->finish(); Chris@184: } Chris@184: Chris@184: if (c) { Chris@184: CommandHistory::getInstance()->addCommand(c, false); Chris@184: return; Chris@184: } matthiasm@267: matthiasm@268: FlexiNoteModel *fnm = dynamic_cast(layer->getModel()); matthiasm@268: if (fnm) { matthiasm@267: FlexiNoteModel::Point point(alignedStart, Chris@409: fnm->getValueMinimum(), Chris@409: alignedDuration, Chris@409: 1.f, Chris@409: ""); matthiasm@267: FlexiNoteModel::EditCommand *command = matthiasm@268: new FlexiNoteModel::EditCommand(fnm, tr("Add Point")); matthiasm@267: command->addPoint(point); matthiasm@267: command->setName(name); matthiasm@267: c = command->finish(); matthiasm@267: } Chris@409: matthiasm@267: if (c) { matthiasm@267: CommandHistory::getInstance()->addCommand(c, false); matthiasm@267: return; matthiasm@267: } Chris@184: } Chris@184: Chris@184: void Chris@45: MainWindowBase::renumberInstants() Chris@45: { Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: if (!pane) return; Chris@45: Chris@45: Layer *layer = dynamic_cast(pane->getSelectedLayer()); Chris@45: if (!layer) return; Chris@45: Chris@45: MultiSelection ms(m_viewManager->getSelection()); Chris@45: Chris@45: Model *model = layer->getModel(); Chris@45: SparseOneDimensionalModel *sodm = dynamic_cast Chris@45: (model); Chris@45: if (!sodm) return; Chris@45: Chris@45: if (!m_labeller) return; Chris@45: Chris@45: Labeller labeller(*m_labeller); Chris@45: labeller.setSampleRate(sodm->getSampleRate()); Chris@45: Chris@45: // This uses a command Chris@45: Chris@45: labeller.labelAll(*sodm, &ms); Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@373: MainWindowBase::openPath(QString fileOrUrl, AudioFileOpenMode mode) Chris@45: { Chris@134: ProgressDialog dialog(tr("Opening file or URL..."), true, 2000, this); Chris@134: connect(&dialog, SIGNAL(showing()), this, SIGNAL(hideSplash())); Chris@109: return open(FileSource(fileOrUrl, &dialog), mode); Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@45: MainWindowBase::open(FileSource source, AudioFileOpenMode mode) Chris@45: { Chris@45: FileOpenStatus status; Chris@45: Chris@45: if (!source.isAvailable()) return FileOpenFailed; Chris@45: source.waitForData(); Chris@45: Chris@45: bool canImportLayer = (getMainModel() != 0 && Chris@45: m_paneStack != 0 && Chris@45: m_paneStack->getCurrentPane() != 0); Chris@45: Chris@152: bool rdf = (source.getExtension().toLower() == "rdf" || Chris@152: source.getExtension().toLower() == "n3" || Chris@152: source.getExtension().toLower() == "ttl"); Chris@152: Chris@152: bool audio = AudioFileReaderFactory::getKnownExtensions().contains Chris@152: (source.getExtension().toLower()); Chris@145: Chris@145: bool rdfSession = false; Chris@145: if (rdf) { Chris@145: RDFImporter::RDFDocumentType rdfType = Chris@145: RDFImporter::identifyDocumentType Chris@145: (QUrl::fromLocalFile(source.getLocalFilename()).toString()); Chris@145: if (rdfType == RDFImporter::AudioRefAndAnnotations || Chris@145: rdfType == RDFImporter::AudioRef) { Chris@145: rdfSession = true; Chris@145: } else if (rdfType == RDFImporter::NotRDF) { Chris@145: rdf = false; Chris@145: } Chris@145: } Chris@145: Chris@145: if (rdf) { Chris@145: if (rdfSession) { Chris@147: bool cancel = false; Chris@147: if (!canImportLayer || shouldCreateNewSessionForRDFAudio(&cancel)) { Chris@145: return openSession(source); Chris@147: } else if (cancel) { Chris@147: return FileOpenCancelled; Chris@145: } else { Chris@145: return openLayer(source); Chris@145: } Chris@145: } else { Chris@145: if ((status = openSession(source)) != FileOpenFailed) { Chris@145: return status; Chris@145: } else if (!canImportLayer) { Chris@145: return FileOpenWrongMode; Chris@145: } else if ((status = openLayer(source)) != FileOpenFailed) { Chris@145: return status; Chris@145: } else { Chris@145: return FileOpenFailed; Chris@145: } Chris@145: } Chris@145: } Chris@145: Chris@152: if (audio && (status = openAudio(source, mode)) != FileOpenFailed) { Chris@45: return status; Chris@45: } else if ((status = openSession(source)) != FileOpenFailed) { Chris@45: return status; Chris@45: } else if ((status = openPlaylist(source, mode)) != FileOpenFailed) { Chris@45: return status; Chris@45: } else if (!canImportLayer) { Chris@45: return FileOpenWrongMode; Chris@45: } else if ((status = openImage(source)) != FileOpenFailed) { Chris@45: return status; Chris@45: } else if ((status = openLayer(source)) != FileOpenFailed) { Chris@45: return status; Chris@45: } else { Chris@45: return FileOpenFailed; Chris@45: } Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@227: MainWindowBase::openAudio(FileSource source, AudioFileOpenMode mode, Chris@227: QString templateName) Chris@45: { Chris@386: SVDEBUG << "MainWindowBase::openAudio(" << source.getLocation() << ") with mode " << mode << " and template " << templateName << endl; Chris@45: Chris@222: if (templateName == "") { Chris@231: templateName = getDefaultSessionTemplate(); Chris@222: } Chris@220: Chris@374: // cerr << "template is: \"" << templateName << "\"" << endl; Chris@223: Chris@413: if (!source.isAvailable()) { Chris@413: if (source.wasCancelled()) { Chris@413: return FileOpenCancelled; Chris@413: } else { Chris@413: return FileOpenFailed; Chris@413: } Chris@413: } Chris@413: Chris@45: source.waitForData(); Chris@45: Chris@45: m_openingAudioFile = true; Chris@45: Chris@435: sv_samplerate_t rate = 0; Chris@45: Chris@360: if (Preferences::getInstance()->getFixedSampleRate() != 0) { Chris@360: rate = Preferences::getInstance()->getFixedSampleRate(); Chris@360: } else if (Preferences::getInstance()->getResampleOnLoad()) { Chris@45: rate = m_playSource->getSourceSampleRate(); Chris@45: } Chris@45: Chris@479: ReadOnlyWaveFileModel *newModel = new ReadOnlyWaveFileModel(source, rate); Chris@45: Chris@45: if (!newModel->isOK()) { Chris@45: delete newModel; Chris@45: m_openingAudioFile = false; Chris@413: if (source.wasCancelled()) { Chris@413: return FileOpenCancelled; Chris@413: } else { Chris@413: return FileOpenFailed; Chris@413: } Chris@45: } Chris@45: Chris@293: // cerr << "mode = " << mode << endl; Chris@45: Chris@45: if (mode == AskUser) { Chris@45: if (getMainModel()) { Chris@45: Chris@147: QSettings settings; Chris@147: settings.beginGroup("MainWindow"); Chris@221: int lastMode = settings.value("lastaudioopenmode", 0).toBool(); Chris@147: settings.endGroup(); Chris@221: int imode = 0; Chris@45: Chris@45: QStringList items; Chris@221: items << tr("Close the current session and start a new one") Chris@221: << tr("Replace the main audio file in this session") Chris@221: << tr("Add the audio file to this session"); Chris@45: Chris@45: bool ok = false; Chris@45: QString item = ListInputDialog::getItem Chris@45: (this, tr("Select target for import"), Chris@221: tr("Select a target for import

You already have an audio file loaded.
What would you like to do with the new audio file?"), Chris@221: items, lastMode, &ok); Chris@45: Chris@45: if (!ok || item.isEmpty()) { Chris@45: delete newModel; Chris@45: m_openingAudioFile = false; Chris@45: return FileOpenCancelled; Chris@45: } Chris@45: Chris@221: for (int i = 0; i < items.size(); ++i) { Chris@221: if (item == items[i]) imode = i; Chris@221: } Chris@221: Chris@147: settings.beginGroup("MainWindow"); Chris@221: settings.setValue("lastaudioopenmode", imode); Chris@147: settings.endGroup(); Chris@45: Chris@221: mode = (AudioFileOpenMode)imode; Chris@45: Chris@45: } else { Chris@221: // no main model: make a new session Chris@221: mode = ReplaceSession; Chris@45: } Chris@45: } Chris@45: Chris@45: if (mode == ReplaceCurrentPane) { Chris@45: Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: if (pane) { Chris@45: if (getMainModel()) { Chris@45: View::ModelSet models(pane->getModels()); Chris@45: if (models.find(getMainModel()) != models.end()) { Chris@221: // Current pane contains main model: replace that Chris@45: mode = ReplaceMainModel; Chris@45: } Chris@221: // Otherwise the current pane has a non-default model, Chris@221: // which we will deal with later Chris@45: } else { Chris@221: // We have no main model, so start a new session with Chris@221: // optional template Chris@221: mode = ReplaceSession; Chris@45: } Chris@45: } else { Chris@221: // We seem to have no current pane! Oh well Chris@45: mode = CreateAdditionalModel; Chris@45: } Chris@45: } Chris@45: Chris@45: if (mode == CreateAdditionalModel && !getMainModel()) { Chris@386: SVDEBUG << "Mode is CreateAdditionalModel but we have no main model, switching to ReplaceSession mode" << endl; Chris@221: mode = ReplaceSession; Chris@221: } Chris@221: Chris@221: bool loadedTemplate = false; Chris@221: Chris@221: if (mode == ReplaceSession) { Chris@258: Chris@258: if (!checkSaveModified()) return FileOpenCancelled; Chris@258: Chris@386: SVDEBUG << "SV looking for template " << templateName << endl; Chris@230: if (templateName != "") { Chris@230: FileOpenStatus tplStatus = openSessionTemplate(templateName); Chris@258: if (tplStatus == FileOpenCancelled) { Chris@293: cerr << "Template load cancelled" << endl; Chris@258: return FileOpenCancelled; Chris@258: } Chris@230: if (tplStatus != FileOpenFailed) { Chris@293: cerr << "Template load succeeded" << endl; Chris@230: loadedTemplate = true; Chris@221: } Chris@221: } Chris@221: Chris@221: if (!loadedTemplate) { Chris@386: SVDEBUG << "No template found: closing session, creating new empty document" << endl; Chris@221: closeSession(); Chris@221: createDocument(); Chris@221: } Chris@221: Chris@386: SVDEBUG << "Now switching to ReplaceMainModel mode" << endl; Chris@45: mode = ReplaceMainModel; Chris@45: } Chris@45: Chris@164: emit activity(tr("Import audio file \"%1\"").arg(source.getLocation())); Chris@164: Chris@45: if (mode == ReplaceMainModel) { Chris@45: Chris@45: Model *prevMain = getMainModel(); Chris@45: if (prevMain) { Chris@45: m_playSource->removeModel(prevMain); Chris@108: PlayParameterRepository::getInstance()->removePlayable(prevMain); Chris@45: } Chris@108: PlayParameterRepository::getInstance()->addPlayable(newModel); Chris@45: Chris@248: SVDEBUG << "SV about to call setMainModel(" << newModel << "): prevMain is " << prevMain << endl; Chris@248: Chris@45: m_document->setMainModel(newModel); Chris@45: Chris@45: setupMenus(); Chris@45: dan@210: if (loadedTemplate || (m_sessionFile == "")) { Chris@45: //!!! shouldn't be dealing directly with title from here -- call a method Chris@57: setWindowTitle(tr("%1: %2") Chris@57: .arg(QApplication::applicationName()) Chris@45: .arg(source.getLocation())); Chris@45: CommandHistory::getInstance()->clear(); Chris@45: CommandHistory::getInstance()->documentSaved(); Chris@45: m_documentModified = false; Chris@45: } else { Chris@57: setWindowTitle(tr("%1: %2 [%3]") Chris@57: .arg(QApplication::applicationName()) Chris@45: .arg(QFileInfo(m_sessionFile).fileName()) Chris@45: .arg(source.getLocation())); Chris@45: if (m_documentModified) { Chris@45: m_documentModified = false; Chris@45: documentModified(); // so as to restore "(modified)" window title Chris@45: } Chris@45: } Chris@45: Chris@45: if (!source.isRemote()) m_audioFile = source.getLocalFilename(); Chris@45: Chris@45: } else if (mode == CreateAdditionalModel) { Chris@45: Chris@45: CommandHistory::getInstance()->startCompoundOperation Chris@219: (tr("Import \"%1\"").arg(source.getBasename()), true); Chris@45: Chris@45: m_document->addImportedModel(newModel); Chris@45: Chris@45: AddPaneCommand *command = new AddPaneCommand(this); Chris@45: CommandHistory::getInstance()->addCommand(command); Chris@45: Chris@45: Pane *pane = command->getPane(); Chris@45: Chris@47: if (m_timeRulerLayer) { Chris@47: m_document->addLayerToView(pane, m_timeRulerLayer); Chris@47: } Chris@45: Chris@45: Layer *newLayer = m_document->createImportedLayer(newModel); Chris@45: Chris@45: if (newLayer) { Chris@45: m_document->addLayerToView(pane, newLayer); Chris@45: } Chris@45: Chris@45: CommandHistory::getInstance()->endCompoundOperation(); Chris@45: Chris@45: } else if (mode == ReplaceCurrentPane) { Chris@45: Chris@45: // We know there is a current pane, otherwise we would have Chris@45: // reset the mode to CreateAdditionalModel above; and we know Chris@45: // the current pane does not contain the main model, otherwise Chris@45: // we would have reset it to ReplaceMainModel. But we don't Chris@45: // know whether the pane contains a waveform model at all. Chris@45: Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: Layer *replace = 0; Chris@45: Chris@45: for (int i = 0; i < pane->getLayerCount(); ++i) { Chris@45: Layer *layer = pane->getLayer(i); Chris@45: if (dynamic_cast(layer)) { Chris@45: replace = layer; Chris@45: break; Chris@45: } Chris@45: } Chris@45: Chris@45: CommandHistory::getInstance()->startCompoundOperation Chris@219: (tr("Import \"%1\"").arg(source.getBasename()), true); Chris@45: Chris@45: m_document->addImportedModel(newModel); Chris@45: Chris@45: if (replace) { Chris@45: m_document->removeLayerFromView(pane, replace); Chris@45: } Chris@45: Chris@45: Layer *newLayer = m_document->createImportedLayer(newModel); Chris@45: Chris@45: if (newLayer) { Chris@45: m_document->addLayerToView(pane, newLayer); Chris@45: } Chris@45: Chris@45: CommandHistory::getInstance()->endCompoundOperation(); Chris@45: } Chris@45: Chris@45: updateMenuStates(); Chris@45: m_recentFiles.addFile(source.getLocation()); Chris@45: if (!source.isRemote()) { Chris@45: // for file dialog Chris@45: registerLastOpenedFilePath(FileFinder::AudioFile, Chris@45: source.getLocalFilename()); Chris@45: } Chris@45: m_openingAudioFile = false; Chris@45: Chris@45: currentPaneChanged(m_paneStack->getCurrentPane()); Chris@45: Chris@342: emit audioFileLoaded(); Chris@342: Chris@45: return FileOpenSucceeded; Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@45: MainWindowBase::openPlaylist(FileSource source, AudioFileOpenMode mode) Chris@45: { Chris@233: SVDEBUG << "MainWindowBase::openPlaylist(" << source.getLocation() << ")" << endl; Chris@135: Chris@45: std::set extensions; Chris@45: PlaylistFileReader::getSupportedExtensions(extensions); Chris@152: QString extension = source.getExtension().toLower(); Chris@45: if (extensions.find(extension) == extensions.end()) return FileOpenFailed; Chris@45: Chris@45: if (!source.isAvailable()) return FileOpenFailed; Chris@45: source.waitForData(); Chris@45: Chris@45: PlaylistFileReader reader(source.getLocalFilename()); Chris@45: if (!reader.isOK()) return FileOpenFailed; Chris@45: Chris@45: PlaylistFileReader::Playlist playlist = reader.load(); Chris@45: Chris@45: bool someSuccess = false; Chris@45: Chris@45: for (PlaylistFileReader::Playlist::const_iterator i = playlist.begin(); Chris@45: i != playlist.end(); ++i) { Chris@45: Chris@134: ProgressDialog dialog(tr("Opening playlist..."), true, 2000, this); Chris@134: connect(&dialog, SIGNAL(showing()), this, SIGNAL(hideSplash())); Chris@109: FileOpenStatus status = openAudio(FileSource(*i, &dialog), mode); Chris@45: Chris@45: if (status == FileOpenCancelled) { Chris@45: return FileOpenCancelled; Chris@45: } Chris@45: Chris@45: if (status == FileOpenSucceeded) { Chris@45: someSuccess = true; Chris@45: mode = CreateAdditionalModel; Chris@45: } Chris@45: } Chris@45: Chris@45: if (someSuccess) return FileOpenSucceeded; Chris@45: else return FileOpenFailed; Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@45: MainWindowBase::openLayer(FileSource source) Chris@45: { Chris@233: SVDEBUG << "MainWindowBase::openLayer(" << source.getLocation() << ")" << endl; Chris@135: Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: Chris@45: if (!pane) { Chris@45: // shouldn't happen, as the menu action should have been disabled Chris@293: cerr << "WARNING: MainWindowBase::openLayer: no current pane" << endl; Chris@45: return FileOpenWrongMode; Chris@45: } Chris@45: Chris@45: if (!getMainModel()) { Chris@45: // shouldn't happen, as the menu action should have been disabled Chris@293: cerr << "WARNING: MainWindowBase::openLayer: No main model -- hence no default sample rate available" << endl; Chris@45: return FileOpenWrongMode; Chris@45: } Chris@45: Chris@45: if (!source.isAvailable()) return FileOpenFailed; Chris@45: source.waitForData(); Chris@45: Chris@45: QString path = source.getLocalFilename(); Chris@45: Chris@145: RDFImporter::RDFDocumentType rdfType = Chris@145: RDFImporter::identifyDocumentType(QUrl::fromLocalFile(path).toString()); Chris@145: Chris@293: // cerr << "RDF type: (in layer) " << (int) rdfType << endl; Chris@148: Chris@145: if (rdfType != RDFImporter::NotRDF) { Chris@145: Chris@145: return openLayersFromRDF(source); Chris@134: Chris@152: } else if (source.getExtension().toLower() == "svl" || Chris@152: (source.getExtension().toLower() == "xml" && Chris@140: (SVFileReader::identifyXmlFile(source.getLocalFilename()) Chris@140: == SVFileReader::SVLayerFile))) { Chris@45: Chris@45: PaneCallback callback(this); Chris@45: QFile file(path); Chris@45: Chris@45: if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { Chris@293: cerr << "ERROR: MainWindowBase::openLayer(" Chris@294: << source.getLocation() Chris@293: << "): Failed to open file for reading" << endl; Chris@45: return FileOpenFailed; Chris@45: } Chris@45: Chris@45: SVFileReader reader(m_document, callback, source.getLocation()); Chris@79: connect Chris@79: (&reader, SIGNAL(modelRegenerationFailed(QString, QString, QString)), Chris@79: this, SLOT(modelRegenerationFailed(QString, QString, QString))); Chris@79: connect Chris@79: (&reader, SIGNAL(modelRegenerationWarning(QString, QString, QString)), Chris@79: this, SLOT(modelRegenerationWarning(QString, QString, QString))); Chris@45: reader.setCurrentPane(pane); Chris@45: Chris@45: QXmlInputSource inputSource(&file); Chris@45: reader.parse(inputSource); Chris@45: Chris@45: if (!reader.isOK()) { Chris@293: cerr << "ERROR: MainWindowBase::openLayer(" Chris@294: << source.getLocation() Chris@45: << "): Failed to read XML file: " Chris@293: << reader.getErrorString() << endl; Chris@45: return FileOpenFailed; Chris@45: } Chris@45: Chris@164: emit activity(tr("Import layer XML file \"%1\"").arg(source.getLocation())); Chris@164: Chris@45: m_recentFiles.addFile(source.getLocation()); Chris@45: Chris@45: if (!source.isRemote()) { Chris@45: registerLastOpenedFilePath(FileFinder::LayerFile, path); // for file dialog Chris@45: } Chris@45: Chris@75: return FileOpenSucceeded; Chris@75: Chris@45: } else { Chris@45: Chris@45: try { Chris@45: Chris@109: MIDIFileImportDialog midiDlg(this); Chris@109: Chris@109: Model *model = DataFileReaderFactory::loadNonCSV Chris@109: (path, &midiDlg, getMainModel()->getSampleRate()); Chris@45: Chris@109: if (!model) { Chris@196: CSVFormat format(path); Chris@196: format.setSampleRate(getMainModel()->getSampleRate()); Chris@196: CSVFormatDialog *dialog = new CSVFormatDialog(this, format); Chris@109: if (dialog->exec() == QDialog::Accepted) { Chris@109: model = DataFileReaderFactory::loadCSV Chris@109: (path, dialog->getFormat(), Chris@109: getMainModel()->getSampleRate()); Chris@109: } Chris@109: } Chris@109: Chris@45: if (model) { Chris@45: Chris@233: SVDEBUG << "MainWindowBase::openLayer: Have model" << endl; Chris@45: Chris@164: emit activity(tr("Import MIDI file \"%1\"").arg(source.getLocation())); Chris@164: Chris@45: Layer *newLayer = m_document->createImportedLayer(model); Chris@45: Chris@45: if (newLayer) { Chris@45: Chris@45: m_document->addLayerToView(pane, newLayer); Chris@88: m_paneStack->setCurrentLayer(pane, newLayer); Chris@88: Chris@45: m_recentFiles.addFile(source.getLocation()); Chris@45: Chris@45: if (!source.isRemote()) { Chris@45: registerLastOpenedFilePath Chris@45: (FileFinder::LayerFile, Chris@45: path); // for file dialog Chris@45: } Chris@88: Chris@45: return FileOpenSucceeded; Chris@45: } Chris@45: } Chris@45: } catch (DataFileReaderFactory::Exception e) { Chris@45: if (e == DataFileReaderFactory::ImportCancelled) { Chris@45: return FileOpenCancelled; Chris@45: } Chris@45: } Chris@45: } Chris@45: Chris@45: return FileOpenFailed; Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@45: MainWindowBase::openImage(FileSource source) Chris@45: { Chris@233: SVDEBUG << "MainWindowBase::openImage(" << source.getLocation() << ")" << endl; Chris@135: Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: Chris@45: if (!pane) { Chris@45: // shouldn't happen, as the menu action should have been disabled Chris@293: cerr << "WARNING: MainWindowBase::openImage: no current pane" << endl; Chris@45: return FileOpenWrongMode; Chris@45: } Chris@45: Chris@45: if (!m_document->getMainModel()) { Chris@45: return FileOpenWrongMode; Chris@45: } Chris@45: Chris@45: bool newLayer = false; Chris@45: Chris@45: ImageLayer *il = dynamic_cast(pane->getSelectedLayer()); Chris@45: if (!il) { Chris@45: for (int i = pane->getLayerCount()-1; i >= 0; --i) { Chris@45: il = dynamic_cast(pane->getLayer(i)); Chris@45: if (il) break; Chris@45: } Chris@45: } Chris@45: if (!il) { Chris@45: il = dynamic_cast Chris@45: (m_document->createEmptyLayer(LayerFactory::Image)); Chris@45: if (!il) return FileOpenFailed; Chris@45: newLayer = true; Chris@45: } Chris@45: Chris@45: // We don't put the image file in Recent Files Chris@45: Chris@293: cerr << "openImage: trying location \"" << source.getLocation() << "\" in image layer" << endl; Chris@45: Chris@45: if (!il->addImage(m_viewManager->getGlobalCentreFrame(), source.getLocation())) { Chris@45: if (newLayer) { Chris@52: m_document->deleteLayer(il); // also releases its model Chris@45: } Chris@45: return FileOpenFailed; Chris@45: } else { Chris@45: if (newLayer) { Chris@45: m_document->addLayerToView(pane, il); Chris@45: } Chris@45: m_paneStack->setCurrentLayer(pane, il); Chris@45: } Chris@45: Chris@45: return FileOpenSucceeded; Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@373: MainWindowBase::openSessionPath(QString fileOrUrl) Chris@45: { Chris@134: ProgressDialog dialog(tr("Opening session..."), true, 2000, this); Chris@134: connect(&dialog, SIGNAL(showing()), this, SIGNAL(hideSplash())); Chris@109: return openSession(FileSource(fileOrUrl, &dialog)); Chris@45: } Chris@45: Chris@45: MainWindowBase::FileOpenStatus Chris@45: MainWindowBase::openSession(FileSource source) Chris@45: { Chris@233: SVDEBUG << "MainWindowBase::openSession(" << source.getLocation() << ")" << endl; Chris@135: Chris@45: if (!source.isAvailable()) return FileOpenFailed; Chris@145: source.waitForData(); Chris@141: Chris@341: QString sessionExt = Chris@341: InteractiveFileFinder::getInstance()->getApplicationSessionExtension(); Chris@341: Chris@341: if (source.getExtension().toLower() != sessionExt) { Chris@145: Chris@145: RDFImporter::RDFDocumentType rdfType = Chris@145: RDFImporter::identifyDocumentType Chris@145: (QUrl::fromLocalFile(source.getLocalFilename()).toString()); Chris@145: Chris@293: // cerr << "RDF type: " << (int)rdfType << endl; Chris@148: Chris@145: if (rdfType == RDFImporter::AudioRefAndAnnotations || Chris@145: rdfType == RDFImporter::AudioRef) { Chris@145: return openSessionFromRDF(source); Chris@145: } else if (rdfType != RDFImporter::NotRDF) { Chris@145: return FileOpenFailed; Chris@145: } Chris@145: Chris@152: if (source.getExtension().toLower() == "xml") { Chris@140: if (SVFileReader::identifyXmlFile(source.getLocalFilename()) == Chris@140: SVFileReader::SVSessionFile) { Chris@293: cerr << "This XML file looks like a session file, attempting to open it as a session" << endl; Chris@140: } else { Chris@140: return FileOpenFailed; Chris@140: } Chris@140: } else { Chris@140: return FileOpenFailed; Chris@140: } Chris@140: } Chris@45: Chris@140: QXmlInputSource *inputSource = 0; Chris@140: BZipFileDevice *bzFile = 0; Chris@140: QFile *rawFile = 0; Chris@140: Chris@341: if (source.getExtension().toLower() == sessionExt) { Chris@140: bzFile = new BZipFileDevice(source.getLocalFilename()); Chris@140: if (!bzFile->open(QIODevice::ReadOnly)) { Chris@140: delete bzFile; Chris@140: return FileOpenFailed; Chris@140: } Chris@140: inputSource = new QXmlInputSource(bzFile); Chris@140: } else { Chris@140: rawFile = new QFile(source.getLocalFilename()); Chris@140: inputSource = new QXmlInputSource(rawFile); Chris@140: } Chris@140: Chris@140: if (!checkSaveModified()) { Chris@140: if (bzFile) bzFile->close(); Chris@140: delete inputSource; Chris@140: delete bzFile; Chris@140: delete rawFile; Chris@140: return FileOpenCancelled; Chris@140: } Chris@45: Chris@45: QString error; Chris@45: closeSession(); Chris@45: createDocument(); Chris@45: Chris@45: PaneCallback callback(this); Chris@45: m_viewManager->clearSelections(); Chris@45: Chris@45: SVFileReader reader(m_document, callback, source.getLocation()); Chris@79: connect Chris@79: (&reader, SIGNAL(modelRegenerationFailed(QString, QString, QString)), Chris@79: this, SLOT(modelRegenerationFailed(QString, QString, QString))); Chris@79: connect Chris@79: (&reader, SIGNAL(modelRegenerationWarning(QString, QString, QString)), Chris@79: this, SLOT(modelRegenerationWarning(QString, QString, QString))); Chris@140: Chris@140: reader.parse(*inputSource); Chris@45: Chris@45: if (!reader.isOK()) { Chris@45: error = tr("SV XML file read error:\n%1").arg(reader.getErrorString()); Chris@45: } Chris@45: Chris@140: if (bzFile) bzFile->close(); Chris@140: Chris@140: delete inputSource; Chris@140: delete bzFile; Chris@140: delete rawFile; Chris@45: Chris@45: bool ok = (error == ""); Chris@45: Chris@45: if (ok) { Chris@45: Chris@164: emit activity(tr("Import session file \"%1\"").arg(source.getLocation())); Chris@164: Chris@57: setWindowTitle(tr("%1: %2") Chris@57: .arg(QApplication::applicationName()) Chris@45: .arg(source.getLocation())); Chris@45: Chris@45: if (!source.isRemote()) m_sessionFile = source.getLocalFilename(); Chris@45: Chris@45: setupMenus(); Chris@45: Chris@45: CommandHistory::getInstance()->clear(); Chris@45: CommandHistory::getInstance()->documentSaved(); Chris@45: m_documentModified = false; Chris@45: updateMenuStates(); Chris@45: Chris@227: m_recentFiles.addFile(source.getLocation()); Chris@45: Chris@45: if (!source.isRemote()) { Chris@45: // for file dialog Chris@45: registerLastOpenedFilePath(FileFinder::SessionFile, Chris@227: source.getLocalFilename()); Chris@45: } Chris@45: Chris@342: emit sessionLoaded(); Chris@342: Chris@45: } else { Chris@57: setWindowTitle(QApplication::applicationName()); Chris@45: } Chris@45: Chris@45: return ok ? FileOpenSucceeded : FileOpenFailed; Chris@45: } Chris@45: Chris@141: MainWindowBase::FileOpenStatus Chris@230: MainWindowBase::openSessionTemplate(QString templateName) Chris@230: { Chris@230: // Template in the user's template directory takes Chris@230: // priority over a bundled one; we don't unbundle, but Chris@230: // open directly from the bundled file (where applicable) Chris@230: ResourceFinder rf; Chris@230: QString tfile = rf.getResourcePath("templates", templateName + ".svt"); Chris@230: if (tfile != "") { Chris@294: cerr << "SV loading template file " << tfile << endl; Chris@230: return openSessionTemplate(FileSource("file:" + tfile)); Chris@230: } else { Chris@230: return FileOpenFailed; Chris@230: } Chris@230: } Chris@230: Chris@230: MainWindowBase::FileOpenStatus Chris@227: MainWindowBase::openSessionTemplate(FileSource source) Chris@227: { Chris@294: cerr << "MainWindowBase::openSessionTemplate(" << source.getLocation() << ")" << endl; Chris@227: Chris@227: if (!source.isAvailable()) return FileOpenFailed; Chris@227: source.waitForData(); Chris@227: Chris@227: QXmlInputSource *inputSource = 0; Chris@227: QFile *file = 0; Chris@227: Chris@227: file = new QFile(source.getLocalFilename()); Chris@227: inputSource = new QXmlInputSource(file); Chris@227: Chris@227: if (!checkSaveModified()) { Chris@227: delete inputSource; Chris@227: delete file; Chris@227: return FileOpenCancelled; Chris@227: } Chris@227: Chris@227: QString error; Chris@227: closeSession(); Chris@227: createDocument(); Chris@227: Chris@227: PaneCallback callback(this); Chris@227: m_viewManager->clearSelections(); Chris@227: Chris@227: SVFileReader reader(m_document, callback, source.getLocation()); Chris@227: connect Chris@227: (&reader, SIGNAL(modelRegenerationFailed(QString, QString, QString)), Chris@227: this, SLOT(modelRegenerationFailed(QString, QString, QString))); Chris@227: connect Chris@227: (&reader, SIGNAL(modelRegenerationWarning(QString, QString, QString)), Chris@227: this, SLOT(modelRegenerationWarning(QString, QString, QString))); Chris@227: Chris@227: reader.parse(*inputSource); Chris@227: Chris@227: if (!reader.isOK()) { Chris@227: error = tr("SV XML file read error:\n%1").arg(reader.getErrorString()); Chris@227: } Chris@227: Chris@227: delete inputSource; Chris@227: delete file; Chris@227: Chris@227: bool ok = (error == ""); Chris@227: Chris@227: setWindowTitle(QApplication::applicationName()); Chris@227: Chris@227: if (ok) { Chris@227: Chris@227: emit activity(tr("Open session template \"%1\"").arg(source.getLocation())); Chris@227: Chris@227: setupMenus(); Chris@227: Chris@227: CommandHistory::getInstance()->clear(); Chris@227: CommandHistory::getInstance()->documentSaved(); Chris@227: m_documentModified = false; Chris@227: updateMenuStates(); Chris@342: Chris@342: emit sessionLoaded(); Chris@227: } Chris@227: Chris@227: return ok ? FileOpenSucceeded : FileOpenFailed; Chris@227: } Chris@227: Chris@227: MainWindowBase::FileOpenStatus Chris@141: MainWindowBase::openSessionFromRDF(FileSource source) Chris@141: { Chris@233: SVDEBUG << "MainWindowBase::openSessionFromRDF(" << source.getLocation() << ")" << endl; Chris@141: Chris@141: if (!source.isAvailable()) return FileOpenFailed; Chris@141: source.waitForData(); Chris@141: Chris@145: if (!checkSaveModified()) { Chris@145: return FileOpenCancelled; Chris@141: } Chris@143: Chris@145: closeSession(); Chris@145: createDocument(); Chris@145: Chris@145: FileOpenStatus status = openLayersFromRDF(source); Chris@141: Chris@141: setupMenus(); Chris@141: Chris@141: setWindowTitle(tr("%1: %2") Chris@141: .arg(QApplication::applicationName()) Chris@141: .arg(source.getLocation())); Chris@141: CommandHistory::getInstance()->clear(); Chris@141: CommandHistory::getInstance()->documentSaved(); Chris@141: m_documentModified = false; Chris@145: Chris@342: emit sessionLoaded(); Chris@342: Chris@145: return status; Chris@145: } Chris@145: Chris@145: MainWindowBase::FileOpenStatus Chris@145: MainWindowBase::openLayersFromRDF(FileSource source) Chris@145: { Chris@435: sv_samplerate_t rate = 0; Chris@145: Chris@233: SVDEBUG << "MainWindowBase::openLayersFromRDF" << endl; Chris@186: Chris@145: ProgressDialog dialog(tr("Importing from RDF..."), true, 2000, this); Chris@145: connect(&dialog, SIGNAL(showing()), this, SIGNAL(hideSplash())); Chris@145: Chris@145: if (getMainModel()) { Chris@145: rate = getMainModel()->getSampleRate(); Chris@145: } else if (Preferences::getInstance()->getResampleOnLoad()) { Chris@145: rate = m_playSource->getSourceSampleRate(); Chris@145: } Chris@145: Chris@145: RDFImporter importer Chris@145: (QUrl::fromLocalFile(source.getLocalFilename()).toString(), rate); Chris@145: Chris@145: if (!importer.isOK()) { Chris@147: if (importer.getErrorString() != "") { Chris@147: QMessageBox::critical Chris@147: (this, tr("Failed to import RDF"), Chris@147: tr("Failed to import RDF

Importing data from RDF document at \"%1\" failed: %2

") Chris@147: .arg(source.getLocation()).arg(importer.getErrorString())); Chris@147: } Chris@145: return FileOpenFailed; Chris@145: } Chris@145: Chris@145: std::vector models = importer.getDataModels(&dialog); Chris@145: Chris@145: dialog.setMessage(tr("Importing from RDF...")); Chris@145: Chris@145: if (models.empty()) { Chris@186: QMessageBox::critical Chris@186: (this, tr("Failed to import RDF"), Chris@186: tr("Failed to import RDF

No suitable data models found for import from RDF document at \"%1\"

").arg(source.getLocation())); Chris@145: return FileOpenFailed; Chris@145: } Chris@145: Chris@164: emit activity(tr("Import RDF document \"%1\"").arg(source.getLocation())); Chris@164: Chris@145: std::set added; Chris@145: Chris@221: for (int i = 0; i < (int)models.size(); ++i) { Chris@145: Chris@145: Model *m = models[i]; Chris@145: WaveFileModel *w = dynamic_cast(m); Chris@145: Chris@145: if (w) { Chris@145: Chris@145: Pane *pane = addPaneToStack(); Chris@145: Layer *layer = 0; Chris@145: Chris@145: if (m_timeRulerLayer) { Chris@145: m_document->addLayerToView(pane, m_timeRulerLayer); Chris@145: } Chris@145: Chris@145: if (!getMainModel()) { Chris@145: m_document->setMainModel(w); Chris@145: layer = m_document->createMainModelLayer(LayerFactory::Waveform); Chris@145: } else { Chris@145: layer = m_document->createImportedLayer(w); Chris@145: } Chris@145: Chris@145: m_document->addLayerToView(pane, layer); Chris@145: Chris@145: added.insert(w); Chris@145: Chris@221: for (int j = 0; j < (int)models.size(); ++j) { Chris@145: Chris@145: Model *dm = models[j]; Chris@145: Chris@145: if (dm == m) continue; Chris@145: if (dm->getSourceModel() != m) continue; Chris@145: Chris@145: layer = m_document->createImportedLayer(dm); Chris@145: Chris@145: if (layer->isLayerOpaque() || Chris@145: dynamic_cast(layer)) { Chris@145: Chris@156: // these always go in a new pane, with nothing Chris@156: // else going in the same pane Chris@156: Chris@145: Pane *singleLayerPane = addPaneToStack(); Chris@145: if (m_timeRulerLayer) { Chris@145: m_document->addLayerToView(singleLayerPane, m_timeRulerLayer); Chris@145: } Chris@145: m_document->addLayerToView(singleLayerPane, layer); Chris@145: Chris@156: } else if (layer->getLayerColourSignificance() == Chris@156: Layer::ColourHasMeaningfulValue) { Chris@156: Chris@156: // these can go in a pane with something else, but Chris@156: // only if none of the something elses also have Chris@156: // this quality Chris@156: Chris@156: bool needNewPane = false; Chris@156: for (int i = 0; i < pane->getLayerCount(); ++i) { Chris@156: Layer *otherLayer = pane->getLayer(i); Chris@156: if (otherLayer && Chris@156: (otherLayer->getLayerColourSignificance() == Chris@156: Layer::ColourHasMeaningfulValue)) { Chris@156: needNewPane = true; Chris@156: break; Chris@156: } Chris@156: } Chris@156: if (needNewPane) { Chris@156: pane = addPaneToStack(); Chris@156: } Chris@156: Chris@156: m_document->addLayerToView(pane, layer); Chris@156: Chris@145: } else { Chris@145: Chris@145: if (pane->getLayerCount() > 4) { Chris@145: pane = addPaneToStack(); Chris@145: } Chris@145: Chris@145: m_document->addLayerToView(pane, layer); Chris@145: } Chris@145: Chris@145: added.insert(dm); Chris@145: } Chris@145: } Chris@145: } Chris@145: Chris@221: for (int i = 0; i < (int)models.size(); ++i) { Chris@145: Chris@145: Model *m = models[i]; Chris@145: Chris@145: if (added.find(m) == added.end()) { Chris@145: Chris@145: Layer *layer = m_document->createImportedLayer(m); Chris@145: if (!layer) return FileOpenFailed; Chris@145: Chris@145: Pane *singleLayerPane = addPaneToStack(); Chris@145: if (m_timeRulerLayer) { Chris@145: m_document->addLayerToView(singleLayerPane, m_timeRulerLayer); Chris@145: } Chris@145: m_document->addLayerToView(singleLayerPane, layer); Chris@145: } Chris@145: } Chris@145: Chris@145: m_recentFiles.addFile(source.getLocation()); Chris@145: return FileOpenSucceeded; Chris@141: } Chris@141: Chris@45: void Chris@475: MainWindowBase::createAudioIO() Chris@45: { Chris@475: if (m_playTarget || m_audioIO) return; Chris@475: Chris@475: if (!(m_soundOptions & WithAudioOutput)) return; Chris@45: Chris@468: //!!! how to handle preferences Chris@468: /* Chris@126: QSettings settings; Chris@126: settings.beginGroup("Preferences"); Chris@126: QString targetName = settings.value("audio-target", "").toString(); Chris@126: settings.endGroup(); Chris@126: AudioTargetFactory *factory = AudioTargetFactory::getInstance(); Chris@126: Chris@126: factory->setDefaultCallbackTarget(targetName); Chris@468: */ Chris@475: Chris@475: if (m_soundOptions & WithAudioInput) { Chris@475: m_audioIO = breakfastquay::AudioFactory:: Chris@475: createCallbackIO(m_recordTarget, m_playSource); Chris@475: m_playSource->setSystemPlaybackTarget(m_audioIO); Chris@475: } else { Chris@475: m_playTarget = breakfastquay::AudioFactory:: Chris@475: createCallbackPlayTarget(m_playSource); Chris@475: m_playSource->setSystemPlaybackTarget(m_playTarget); Chris@475: } Chris@475: Chris@475: if (!m_playTarget && !m_audioIO) { Chris@104: emit hideSplash(); Chris@126: Chris@468: // if (factory->isAutoCallbackTarget(targetName)) { Chris@128: QMessageBox::warning Chris@45: (this, tr("Couldn't open audio device"), Chris@126: tr("No audio available

Could not open an audio device for playback.

Automatic audio device detection failed. Audio playback will not be available during this session.

"), Chris@45: QMessageBox::Ok); Chris@468: /* Chris@126: } else { Chris@126: QMessageBox::warning Chris@126: (this, tr("Couldn't open audio device"), Chris@126: tr("No audio available

Failed to open your preferred audio device (\"%1\").

Audio playback will not be available during this session.

") Chris@126: .arg(factory->getCallbackTargetDescription(targetName)), Chris@126: QMessageBox::Ok); Chris@126: } Chris@468: */ Chris@475: return; Chris@45: } Chris@45: } Chris@45: Chris@45: WaveFileModel * Chris@45: MainWindowBase::getMainModel() Chris@45: { Chris@45: if (!m_document) return 0; Chris@45: return m_document->getMainModel(); Chris@45: } Chris@45: Chris@45: const WaveFileModel * Chris@45: MainWindowBase::getMainModel() const Chris@45: { Chris@45: if (!m_document) return 0; Chris@45: return m_document->getMainModel(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::createDocument() Chris@45: { Chris@45: m_document = new Document; Chris@45: Chris@45: connect(m_document, SIGNAL(layerAdded(Layer *)), Chris@45: this, SLOT(layerAdded(Layer *))); Chris@45: connect(m_document, SIGNAL(layerRemoved(Layer *)), Chris@45: this, SLOT(layerRemoved(Layer *))); Chris@45: connect(m_document, SIGNAL(layerAboutToBeDeleted(Layer *)), Chris@45: this, SLOT(layerAboutToBeDeleted(Layer *))); Chris@45: connect(m_document, SIGNAL(layerInAView(Layer *, bool)), Chris@45: this, SLOT(layerInAView(Layer *, bool))); Chris@45: Chris@45: connect(m_document, SIGNAL(modelAdded(Model *)), Chris@45: this, SLOT(modelAdded(Model *))); Chris@45: connect(m_document, SIGNAL(mainModelChanged(WaveFileModel *)), Chris@45: this, SLOT(mainModelChanged(WaveFileModel *))); Chris@45: connect(m_document, SIGNAL(modelAboutToBeDeleted(Model *)), Chris@45: this, SLOT(modelAboutToBeDeleted(Model *))); Chris@45: Chris@78: connect(m_document, SIGNAL(modelGenerationFailed(QString, QString)), Chris@78: this, SLOT(modelGenerationFailed(QString, QString))); Chris@78: connect(m_document, SIGNAL(modelRegenerationWarning(QString, QString, QString)), Chris@78: this, SLOT(modelRegenerationWarning(QString, QString, QString))); Chris@78: connect(m_document, SIGNAL(modelGenerationFailed(QString, QString)), Chris@78: this, SLOT(modelGenerationFailed(QString, QString))); Chris@78: connect(m_document, SIGNAL(modelRegenerationWarning(QString, QString, QString)), Chris@78: this, SLOT(modelRegenerationWarning(QString, QString, QString))); Chris@78: connect(m_document, SIGNAL(alignmentFailed(QString, QString)), Chris@78: this, SLOT(alignmentFailed(QString, QString))); Chris@160: Chris@160: emit replacedDocument(); Chris@45: } Chris@45: Chris@45: bool Chris@45: MainWindowBase::saveSessionFile(QString path) Chris@45: { Chris@217: try { Chris@217: Chris@217: TempWriteFile temp(path); Chris@217: Chris@217: BZipFileDevice bzFile(temp.getTemporaryFilename()); Chris@217: if (!bzFile.open(QIODevice::WriteOnly)) { Chris@293: cerr << "Failed to open session file \"" Chris@294: << temp.getTemporaryFilename() Chris@217: << "\" for writing: " Chris@293: << bzFile.errorString() << endl; Chris@217: return false; Chris@217: } Chris@217: Chris@217: QApplication::setOverrideCursor(QCursor(Qt::WaitCursor)); Chris@217: Chris@217: QTextStream out(&bzFile); Chris@432: out.setCodec(QTextCodec::codecForName("UTF-8")); Chris@226: toXml(out, false); Chris@217: out.flush(); Chris@217: Chris@217: QApplication::restoreOverrideCursor(); Chris@217: Chris@217: if (!bzFile.isOK()) { Chris@217: QMessageBox::critical(this, tr("Failed to write file"), Chris@217: tr("Save failed

Failed to write to file \"%1\": %2") Chris@217: .arg(path).arg(bzFile.errorString())); Chris@217: bzFile.close(); Chris@217: return false; Chris@217: } Chris@217: Chris@217: bzFile.close(); Chris@217: temp.moveToTarget(); Chris@217: return true; Chris@217: Chris@217: } catch (FileOperationFailed &f) { Chris@217: Chris@217: QMessageBox::critical(this, tr("Failed to write file"), Chris@217: tr("Save failed

Failed to write to file \"%1\": %2") Chris@217: .arg(path).arg(f.what())); Chris@45: return false; Chris@45: } Chris@45: } Chris@45: Chris@224: bool Chris@224: MainWindowBase::saveSessionTemplate(QString path) Chris@224: { Chris@224: try { Chris@224: Chris@224: TempWriteFile temp(path); Chris@224: Chris@224: QFile file(temp.getTemporaryFilename()); Chris@224: if (!file.open(QIODevice::WriteOnly)) { Chris@293: cerr << "Failed to open session template file \"" Chris@294: << temp.getTemporaryFilename() Chris@224: << "\" for writing: " Chris@294: << file.errorString() << endl; Chris@224: return false; Chris@224: } Chris@224: Chris@224: QApplication::setOverrideCursor(QCursor(Qt::WaitCursor)); Chris@224: Chris@224: QTextStream out(&file); Chris@432: out.setCodec(QTextCodec::codecForName("UTF-8")); Chris@226: toXml(out, true); Chris@224: out.flush(); Chris@224: Chris@224: QApplication::restoreOverrideCursor(); Chris@224: Chris@224: file.close(); Chris@224: temp.moveToTarget(); Chris@224: return true; Chris@224: Chris@224: } catch (FileOperationFailed &f) { Chris@224: Chris@224: QMessageBox::critical(this, tr("Failed to write file"), Chris@224: tr("Save failed

Failed to write to file \"%1\": %2") Chris@224: .arg(path).arg(f.what())); Chris@224: return false; Chris@224: } Chris@224: } Chris@224: Chris@45: void Chris@226: MainWindowBase::toXml(QTextStream &out, bool asTemplate) Chris@45: { Chris@45: QString indent(" "); Chris@45: Chris@45: out << "\n"; Chris@45: out << "\n"; Chris@45: out << "\n"; Chris@45: Chris@226: if (asTemplate) { Chris@226: m_document->toXmlAsTemplate(out, "", ""); Chris@226: } else { Chris@226: m_document->toXml(out, "", ""); Chris@226: } Chris@45: Chris@45: out << "\n"; Chris@45: Chris@45: out << QString(" \n") Chris@45: .arg(width()).arg(height()); Chris@45: Chris@45: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@45: Chris@45: Pane *pane = m_paneStack->getPane(i); Chris@45: Chris@45: if (pane) { Chris@45: pane->toXml(out, indent); Chris@45: } Chris@45: } Chris@45: Chris@45: out << "\n"; Chris@45: Chris@45: m_viewManager->getSelection().toXml(out); Chris@45: Chris@45: out << "\n"; Chris@45: } Chris@45: Chris@45: Pane * Chris@45: MainWindowBase::addPaneToStack() Chris@45: { Chris@342: cerr << "MainWindowBase::addPaneToStack()" << endl; Chris@45: AddPaneCommand *command = new AddPaneCommand(this); Chris@45: CommandHistory::getInstance()->addCommand(command); Chris@57: Pane *pane = command->getPane(); Chris@57: return pane; Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::zoomIn() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (currentPane) currentPane->zoom(true); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::zoomOut() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (currentPane) currentPane->zoom(false); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::zoomToFit() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (!currentPane) return; Chris@45: Chris@45: Model *model = getMainModel(); Chris@45: if (!model) return; Chris@45: Chris@434: sv_frame_t start = model->getStartFrame(); Chris@434: sv_frame_t end = model->getEndFrame(); Chris@60: if (m_playSource) end = std::max(end, m_playSource->getPlayEndFrame()); Chris@366: int pixels = currentPane->width(); Chris@366: Chris@366: int sw = currentPane->getVerticalScaleWidth(); Chris@45: if (pixels > sw * 2) pixels -= sw * 2; Chris@45: else pixels = 1; Chris@45: if (pixels > 4) pixels -= 4; Chris@45: Chris@436: int zoomLevel = int((end - start) / pixels); Chris@150: if (zoomLevel < 1) zoomLevel = 1; Chris@45: Chris@45: currentPane->setZoomLevel(zoomLevel); Chris@45: currentPane->setCentreFrame((start + end) / 2); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::zoomDefault() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@302: QSettings settings; Chris@302: settings.beginGroup("MainWindow"); Chris@302: int zoom = settings.value("zoom-default", 1024).toInt(); Chris@302: settings.endGroup(); Chris@302: if (currentPane) currentPane->setZoomLevel(zoom); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::scrollLeft() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (currentPane) currentPane->scroll(false, false); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::jumpLeft() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (currentPane) currentPane->scroll(false, true); Chris@45: } Chris@45: Chris@45: void Chris@162: MainWindowBase::peekLeft() Chris@162: { Chris@162: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@162: if (currentPane) currentPane->scroll(false, false, false); Chris@162: } Chris@162: Chris@162: void Chris@45: MainWindowBase::scrollRight() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (currentPane) currentPane->scroll(true, false); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::jumpRight() Chris@45: { Chris@45: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@45: if (currentPane) currentPane->scroll(true, true); Chris@45: } Chris@45: Chris@45: void Chris@162: MainWindowBase::peekRight() Chris@162: { Chris@162: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@162: if (currentPane) currentPane->scroll(true, false, false); Chris@162: } Chris@162: Chris@162: void Chris@45: MainWindowBase::showNoOverlays() Chris@45: { Chris@45: m_viewManager->setOverlayMode(ViewManager::NoOverlays); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::showMinimalOverlays() Chris@45: { Chris@335: m_viewManager->setOverlayMode(ViewManager::StandardOverlays); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::showAllOverlays() Chris@45: { Chris@45: m_viewManager->setOverlayMode(ViewManager::AllOverlays); Chris@45: } Chris@45: Chris@45: void Chris@211: MainWindowBase::toggleTimeRulers() Chris@211: { Chris@211: bool haveRulers = false; Chris@211: bool someHidden = false; Chris@211: Chris@211: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@211: Chris@211: Pane *pane = m_paneStack->getPane(i); Chris@211: if (!pane) continue; Chris@211: Chris@211: for (int j = 0; j < pane->getLayerCount(); ++j) { Chris@211: Chris@211: Layer *layer = pane->getLayer(j); Chris@211: if (!dynamic_cast(layer)) continue; Chris@211: Chris@211: haveRulers = true; Chris@211: if (layer->isLayerDormant(pane)) someHidden = true; Chris@211: } Chris@211: } Chris@211: Chris@211: if (haveRulers) { Chris@211: Chris@211: bool show = someHidden; Chris@211: Chris@211: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@211: Chris@211: Pane *pane = m_paneStack->getPane(i); Chris@211: if (!pane) continue; Chris@211: Chris@211: for (int j = 0; j < pane->getLayerCount(); ++j) { Chris@211: Chris@211: Layer *layer = pane->getLayer(j); Chris@211: if (!dynamic_cast(layer)) continue; Chris@211: Chris@211: layer->showLayer(pane, show); Chris@211: } Chris@211: } Chris@211: } Chris@211: } Chris@211: Chris@211: void Chris@45: MainWindowBase::toggleZoomWheels() Chris@45: { Chris@45: if (m_viewManager->getZoomWheelsEnabled()) { Chris@45: m_viewManager->setZoomWheelsEnabled(false); Chris@45: } else { Chris@45: m_viewManager->setZoomWheelsEnabled(true); Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::togglePropertyBoxes() Chris@45: { Chris@45: if (m_paneStack->getLayoutStyle() == PaneStack::NoPropertyStacks) { Chris@45: if (Preferences::getInstance()->getPropertyBoxLayout() == Chris@45: Preferences::VerticallyStacked) { Chris@45: m_paneStack->setLayoutStyle(PaneStack::PropertyStackPerPaneLayout); Chris@45: } else { Chris@45: m_paneStack->setLayoutStyle(PaneStack::SinglePropertyStackLayout); Chris@45: } Chris@45: } else { Chris@45: m_paneStack->setLayoutStyle(PaneStack::NoPropertyStacks); Chris@45: } Chris@45: } Chris@45: Chris@378: QLabel * Chris@378: MainWindowBase::getStatusLabel() const Chris@378: { Chris@378: if (!m_statusLabel) { Chris@378: m_statusLabel = new QLabel(); Chris@378: statusBar()->addWidget(m_statusLabel, 1); Chris@378: } Chris@379: Chris@379: QList frames = statusBar()->findChildren(); Chris@379: foreach (QFrame *f, frames) { Chris@379: f->setFrameStyle(QFrame::NoFrame); Chris@379: } Chris@379: Chris@378: return m_statusLabel; Chris@378: } Chris@378: Chris@45: void Chris@45: MainWindowBase::toggleStatusBar() Chris@45: { Chris@45: QSettings settings; Chris@45: settings.beginGroup("MainWindow"); Chris@45: bool sb = settings.value("showstatusbar", true).toBool(); Chris@45: Chris@45: if (sb) { Chris@45: statusBar()->hide(); Chris@45: } else { Chris@45: statusBar()->show(); Chris@45: } Chris@45: Chris@45: settings.setValue("showstatusbar", !sb); Chris@45: Chris@45: settings.endGroup(); Chris@45: } Chris@45: Chris@45: void Chris@256: MainWindowBase::toggleCentreLine() Chris@256: { Chris@256: if (m_viewManager->shouldShowCentreLine()) { Chris@256: m_viewManager->setShowCentreLine(false); Chris@256: } else { Chris@256: m_viewManager->setShowCentreLine(true); Chris@256: } Chris@256: } Chris@256: Chris@256: void Chris@45: MainWindowBase::preferenceChanged(PropertyContainer::PropertyName name) Chris@45: { Chris@45: if (name == "Property Box Layout") { Chris@45: if (m_paneStack->getLayoutStyle() != PaneStack::NoPropertyStacks) { Chris@45: if (Preferences::getInstance()->getPropertyBoxLayout() == Chris@45: Preferences::VerticallyStacked) { Chris@45: m_paneStack->setLayoutStyle(PaneStack::PropertyStackPerPaneLayout); Chris@45: } else { Chris@45: m_paneStack->setLayoutStyle(PaneStack::SinglePropertyStackLayout); Chris@45: } Chris@45: } Chris@45: } else if (name == "Background Mode" && m_viewManager) { Chris@45: Preferences::BackgroundMode mode = Chris@45: Preferences::getInstance()->getBackgroundMode(); Chris@45: if (mode == Preferences::BackgroundFromTheme) { Chris@45: m_viewManager->setGlobalDarkBackground(m_initialDarkBackground); Chris@45: } else if (mode == Preferences::DarkBackground) { Chris@45: m_viewManager->setGlobalDarkBackground(true); Chris@45: } else { Chris@45: m_viewManager->setGlobalDarkBackground(false); Chris@45: } Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::play() Chris@45: { Chris@477: if (m_recordTarget->isRecording() || m_playSource->isPlaying()) { Chris@45: stop(); Chris@479: QAction *action = qobject_cast(sender()); Chris@479: if (action) action->setChecked(false); Chris@45: } else { Chris@45: playbackFrameChanged(m_viewManager->getPlaybackFrame()); Chris@45: m_playSource->play(m_viewManager->getPlaybackFrame()); Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@477: MainWindowBase::record() Chris@477: { Chris@478: if (!(m_soundOptions & WithAudioInput)) { Chris@478: return; Chris@478: } Chris@478: Chris@477: if (!m_recordTarget) { Chris@477: //!!! report Chris@477: return; Chris@477: } Chris@477: Chris@478: if (!m_audioIO) { Chris@478: createAudioIO(); Chris@478: } Chris@478: Chris@477: if (m_recordTarget->isRecording()) { Chris@477: m_recordTarget->stopRecording(); Chris@479: emit audioFileLoaded(); Chris@477: return; Chris@477: } Chris@477: Chris@483: if (m_audioRecordMode == RecordReplaceSession) { Chris@483: if (!checkSaveModified()) return; Chris@483: } Chris@483: Chris@477: WritableWaveFileModel *model = m_recordTarget->startRecording(); Chris@477: if (!model) { Chris@477: cerr << "ERROR: MainWindowBase::record: Recording failed" << endl; Chris@477: //!!! report Chris@477: return; Chris@477: } Chris@477: Chris@477: if (!model->isOK()) { Chris@477: m_recordTarget->stopRecording(); Chris@477: delete model; Chris@477: //!!! ??? Chris@477: return; Chris@477: } Chris@477: Chris@478: PlayParameterRepository::getInstance()->addPlayable(model); Chris@483: Chris@483: if (m_audioRecordMode == RecordReplaceSession || !getMainModel()) { Chris@478: Chris@479: //!!! duplication with openAudio here Chris@479: Chris@479: QString templateName = getDefaultSessionTemplate(); Chris@479: bool loadedTemplate = false; Chris@479: Chris@479: if (templateName != "") { Chris@479: FileOpenStatus tplStatus = openSessionTemplate(templateName); Chris@479: if (tplStatus == FileOpenCancelled) { Chris@479: return; Chris@479: } Chris@479: if (tplStatus != FileOpenFailed) { Chris@479: loadedTemplate = true; Chris@479: } Chris@479: } Chris@479: Chris@479: if (!loadedTemplate) { Chris@479: closeSession(); Chris@479: createDocument(); Chris@479: } Chris@479: Chris@479: Model *prevMain = getMainModel(); Chris@479: if (prevMain) { Chris@479: m_playSource->removeModel(prevMain); Chris@479: PlayParameterRepository::getInstance()->removePlayable(prevMain); Chris@479: } Chris@479: Chris@478: m_document->setMainModel(model); Chris@478: setupMenus(); Chris@478: Chris@479: if (loadedTemplate || (m_sessionFile == "")) { Chris@479: //!!! shouldn't be dealing directly with title from here -- call a method Chris@479: setWindowTitle(tr("%1: %2") Chris@479: .arg(QApplication::applicationName()) Chris@479: .arg(model->getLocation())); Chris@479: CommandHistory::getInstance()->clear(); Chris@479: CommandHistory::getInstance()->documentSaved(); Chris@479: m_documentModified = false; Chris@479: } else { Chris@479: setWindowTitle(tr("%1: %2 [%3]") Chris@479: .arg(QApplication::applicationName()) Chris@479: .arg(QFileInfo(m_sessionFile).fileName()) Chris@479: .arg(model->getLocation())); Chris@479: if (m_documentModified) { Chris@479: m_documentModified = false; Chris@479: documentModified(); // so as to restore "(modified)" window title Chris@479: } Chris@479: } Chris@479: Chris@478: } else { Chris@478: Chris@478: CommandHistory::getInstance()->startCompoundOperation Chris@478: (tr("Import Recorded Audio"), true); Chris@478: Chris@478: m_document->addImportedModel(model); Chris@478: Chris@478: AddPaneCommand *command = new AddPaneCommand(this); Chris@478: CommandHistory::getInstance()->addCommand(command); Chris@478: Chris@478: Pane *pane = command->getPane(); Chris@478: Chris@478: if (m_timeRulerLayer) { Chris@478: m_document->addLayerToView(pane, m_timeRulerLayer); Chris@478: } Chris@478: Chris@478: Layer *newLayer = m_document->createImportedLayer(model); Chris@478: Chris@478: if (newLayer) { Chris@478: m_document->addLayerToView(pane, newLayer); Chris@478: } Chris@478: Chris@478: CommandHistory::getInstance()->endCompoundOperation(); Chris@477: } Chris@479: Chris@479: updateMenuStates(); Chris@479: m_recentFiles.addFile(model->getLocation()); Chris@479: currentPaneChanged(m_paneStack->getCurrentPane()); Chris@477: } Chris@477: Chris@477: void Chris@45: MainWindowBase::ffwd() Chris@45: { Chris@45: if (!getMainModel()) return; Chris@45: Chris@435: sv_frame_t frame = m_viewManager->getPlaybackFrame(); Chris@45: ++frame; Chris@45: Chris@85: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: Layer *layer = getSnapLayer(); Chris@435: sv_samplerate_t sr = getMainModel()->getSampleRate(); Chris@45: Chris@45: if (!layer) { Chris@45: Chris@45: frame = RealTime::realTime2Frame Chris@357: (RealTime::frame2RealTime(frame, sr) + m_defaultFfwdRwdStep, sr); Chris@435: if (frame > getMainModel()->getEndFrame()) { Chris@45: frame = getMainModel()->getEndFrame(); Chris@45: } Chris@45: Chris@45: } else { Chris@45: Chris@366: int resolution = 0; Chris@166: if (pane) frame = pane->alignFromReference(frame); Chris@85: if (layer->snapToFeatureFrame(m_paneStack->getCurrentPane(), Chris@85: frame, resolution, Layer::SnapRight)) { Chris@85: if (pane) frame = pane->alignToReference(frame); Chris@85: } else { Chris@45: frame = getMainModel()->getEndFrame(); Chris@45: } Chris@45: } Chris@45: Chris@45: if (frame < 0) frame = 0; Chris@45: Chris@45: if (m_viewManager->getPlaySelectionMode()) { Chris@435: frame = m_viewManager->constrainFrameToSelection(frame); Chris@45: } Chris@45: Chris@45: m_viewManager->setPlaybackFrame(frame); Chris@166: Chris@435: if (frame == getMainModel()->getEndFrame() && Chris@166: m_playSource && Chris@166: m_playSource->isPlaying() && Chris@166: !m_viewManager->getPlayLoopMode()) { Chris@166: stop(); Chris@166: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::ffwdEnd() Chris@45: { Chris@45: if (!getMainModel()) return; Chris@45: Chris@139: if (m_playSource && Chris@139: m_playSource->isPlaying() && Chris@139: !m_viewManager->getPlayLoopMode()) { Chris@139: stop(); Chris@139: } Chris@139: Chris@435: sv_frame_t frame = getMainModel()->getEndFrame(); Chris@45: Chris@45: if (m_viewManager->getPlaySelectionMode()) { Chris@45: frame = m_viewManager->constrainFrameToSelection(frame); Chris@45: } Chris@45: Chris@45: m_viewManager->setPlaybackFrame(frame); Chris@45: } Chris@45: Chris@45: void Chris@166: MainWindowBase::ffwdSimilar() Chris@166: { Chris@166: if (!getMainModel()) return; Chris@166: Chris@166: Layer *layer = getSnapLayer(); Chris@166: if (!layer) { ffwd(); return; } Chris@166: Chris@166: Pane *pane = m_paneStack->getCurrentPane(); Chris@435: sv_frame_t frame = m_viewManager->getPlaybackFrame(); Chris@166: Chris@366: int resolution = 0; Chris@166: if (pane) frame = pane->alignFromReference(frame); Chris@166: if (layer->snapToSimilarFeature(m_paneStack->getCurrentPane(), Chris@166: frame, resolution, Layer::SnapRight)) { Chris@166: if (pane) frame = pane->alignToReference(frame); Chris@166: } else { Chris@166: frame = getMainModel()->getEndFrame(); Chris@166: } Chris@166: Chris@166: if (frame < 0) frame = 0; Chris@166: Chris@166: if (m_viewManager->getPlaySelectionMode()) { Chris@435: frame = m_viewManager->constrainFrameToSelection(frame); Chris@166: } Chris@166: Chris@166: m_viewManager->setPlaybackFrame(frame); Chris@166: Chris@435: if (frame == getMainModel()->getEndFrame() && Chris@166: m_playSource && Chris@166: m_playSource->isPlaying() && Chris@166: !m_viewManager->getPlayLoopMode()) { Chris@166: stop(); Chris@166: } Chris@166: } Chris@166: Chris@166: void Chris@45: MainWindowBase::rewind() Chris@45: { Chris@45: if (!getMainModel()) return; Chris@45: Chris@435: sv_frame_t frame = m_viewManager->getPlaybackFrame(); Chris@45: if (frame > 0) --frame; Chris@45: Chris@85: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: Layer *layer = getSnapLayer(); Chris@435: sv_samplerate_t sr = getMainModel()->getSampleRate(); Chris@45: Chris@45: // when rewinding during playback, we want to allow a period Chris@45: // following a rewind target point at which the rewind will go to Chris@45: // the prior point instead of the immediately neighbouring one Chris@45: if (m_playSource && m_playSource->isPlaying()) { Chris@45: RealTime ct = RealTime::frame2RealTime(frame, sr); Chris@357: ct = ct - RealTime::fromSeconds(0.15); Chris@45: if (ct < RealTime::zeroTime) ct = RealTime::zeroTime; Chris@45: frame = RealTime::realTime2Frame(ct, sr); Chris@45: } Chris@45: Chris@45: if (!layer) { Chris@45: Chris@45: frame = RealTime::realTime2Frame Chris@357: (RealTime::frame2RealTime(frame, sr) - m_defaultFfwdRwdStep, sr); Chris@435: if (frame < getMainModel()->getStartFrame()) { Chris@45: frame = getMainModel()->getStartFrame(); Chris@45: } Chris@45: Chris@45: } else { Chris@45: Chris@366: int resolution = 0; Chris@166: if (pane) frame = pane->alignFromReference(frame); Chris@85: if (layer->snapToFeatureFrame(m_paneStack->getCurrentPane(), Chris@85: frame, resolution, Layer::SnapLeft)) { Chris@85: if (pane) frame = pane->alignToReference(frame); Chris@85: } else { Chris@45: frame = getMainModel()->getStartFrame(); Chris@45: } Chris@45: } Chris@45: Chris@45: if (frame < 0) frame = 0; Chris@45: Chris@45: if (m_viewManager->getPlaySelectionMode()) { Chris@435: frame = m_viewManager->constrainFrameToSelection(frame); Chris@45: } Chris@45: Chris@45: m_viewManager->setPlaybackFrame(frame); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::rewindStart() Chris@45: { Chris@45: if (!getMainModel()) return; Chris@45: Chris@435: sv_frame_t frame = getMainModel()->getStartFrame(); Chris@45: Chris@45: if (m_viewManager->getPlaySelectionMode()) { Chris@45: frame = m_viewManager->constrainFrameToSelection(frame); Chris@45: } Chris@45: Chris@45: m_viewManager->setPlaybackFrame(frame); Chris@45: } Chris@45: Chris@166: void Chris@166: MainWindowBase::rewindSimilar() Chris@166: { Chris@166: if (!getMainModel()) return; Chris@166: Chris@166: Layer *layer = getSnapLayer(); Chris@166: if (!layer) { rewind(); return; } Chris@166: Chris@166: Pane *pane = m_paneStack->getCurrentPane(); Chris@435: sv_frame_t frame = m_viewManager->getPlaybackFrame(); Chris@166: Chris@366: int resolution = 0; Chris@166: if (pane) frame = pane->alignFromReference(frame); Chris@166: if (layer->snapToSimilarFeature(m_paneStack->getCurrentPane(), Chris@166: frame, resolution, Layer::SnapLeft)) { Chris@166: if (pane) frame = pane->alignToReference(frame); Chris@166: } else { Chris@166: frame = getMainModel()->getStartFrame(); Chris@166: } Chris@166: Chris@166: if (frame < 0) frame = 0; Chris@166: Chris@166: if (m_viewManager->getPlaySelectionMode()) { Chris@435: frame = m_viewManager->constrainFrameToSelection(frame); Chris@166: } Chris@166: Chris@166: m_viewManager->setPlaybackFrame(frame); Chris@166: } Chris@166: Chris@45: Layer * Chris@45: MainWindowBase::getSnapLayer() const Chris@45: { Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: if (!pane) return 0; Chris@45: Chris@45: Layer *layer = pane->getSelectedLayer(); Chris@45: Chris@45: if (!dynamic_cast(layer) && Chris@45: !dynamic_cast(layer) && Chris@194: !dynamic_cast(layer) && Chris@45: !dynamic_cast(layer)) { Chris@45: Chris@45: layer = 0; Chris@45: Chris@45: for (int i = pane->getLayerCount(); i > 0; --i) { Chris@45: Layer *l = pane->getLayer(i-1); Chris@45: if (dynamic_cast(l)) { Chris@45: layer = l; Chris@45: break; Chris@45: } Chris@45: } Chris@45: } Chris@45: Chris@45: return layer; Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::stop() Chris@45: { Chris@477: if (m_recordTarget->isRecording()) { Chris@477: m_recordTarget->stopRecording(); Chris@479: emit audioFileLoaded(); Chris@477: } Chris@477: Chris@45: m_playSource->stop(); Chris@45: Chris@45: if (m_paneStack && m_paneStack->getCurrentPane()) { Chris@45: updateVisibleRangeDisplay(m_paneStack->getCurrentPane()); Chris@45: } else { Chris@45: m_myStatusMessage = ""; Chris@378: getStatusLabel()->setText(""); Chris@45: } Chris@45: } Chris@45: Chris@45: MainWindowBase::AddPaneCommand::AddPaneCommand(MainWindowBase *mw) : Chris@45: m_mw(mw), Chris@45: m_pane(0), Chris@45: m_prevCurrentPane(0), Chris@45: m_added(false) Chris@45: { Chris@45: } Chris@45: Chris@45: MainWindowBase::AddPaneCommand::~AddPaneCommand() Chris@45: { Chris@45: if (m_pane && !m_added) { Chris@45: m_mw->m_paneStack->deletePane(m_pane); Chris@45: } Chris@45: } Chris@45: Chris@45: QString Chris@45: MainWindowBase::AddPaneCommand::getName() const Chris@45: { Chris@45: return tr("Add Pane"); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::AddPaneCommand::execute() Chris@45: { Chris@45: if (!m_pane) { Chris@45: m_prevCurrentPane = m_mw->m_paneStack->getCurrentPane(); Chris@45: m_pane = m_mw->m_paneStack->addPane(); Chris@45: Chris@45: connect(m_pane, SIGNAL(contextHelpChanged(const QString &)), Chris@45: m_mw, SLOT(contextHelpChanged(const QString &))); Chris@45: } else { Chris@45: m_mw->m_paneStack->showPane(m_pane); Chris@45: } Chris@45: Chris@45: m_mw->m_paneStack->setCurrentPane(m_pane); Chris@45: m_added = true; Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::AddPaneCommand::unexecute() Chris@45: { Chris@45: m_mw->m_paneStack->hidePane(m_pane); Chris@45: m_mw->m_paneStack->setCurrentPane(m_prevCurrentPane); Chris@45: m_added = false; Chris@45: } Chris@45: Chris@45: MainWindowBase::RemovePaneCommand::RemovePaneCommand(MainWindowBase *mw, Pane *pane) : Chris@45: m_mw(mw), Chris@45: m_pane(pane), Chris@409: m_prevCurrentPane(0), Chris@45: m_added(true) Chris@45: { Chris@45: } Chris@45: Chris@45: MainWindowBase::RemovePaneCommand::~RemovePaneCommand() Chris@45: { Chris@45: if (m_pane && !m_added) { Chris@45: m_mw->m_paneStack->deletePane(m_pane); Chris@45: } Chris@45: } Chris@45: Chris@45: QString Chris@45: MainWindowBase::RemovePaneCommand::getName() const Chris@45: { Chris@45: return tr("Remove Pane"); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::RemovePaneCommand::execute() Chris@45: { Chris@45: m_prevCurrentPane = m_mw->m_paneStack->getCurrentPane(); Chris@45: m_mw->m_paneStack->hidePane(m_pane); Chris@45: m_added = false; Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::RemovePaneCommand::unexecute() Chris@45: { Chris@45: m_mw->m_paneStack->showPane(m_pane); Chris@45: m_mw->m_paneStack->setCurrentPane(m_prevCurrentPane); Chris@45: m_added = true; Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::deleteCurrentPane() Chris@45: { Chris@45: CommandHistory::getInstance()->startCompoundOperation Chris@45: (tr("Delete Pane"), true); Chris@45: Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: if (pane) { Chris@45: while (pane->getLayerCount() > 0) { Chris@45: Layer *layer = pane->getLayer(0); Chris@45: if (layer) { Chris@45: m_document->removeLayerFromView(pane, layer); Chris@45: } else { Chris@45: break; Chris@45: } Chris@45: } Chris@45: Chris@45: RemovePaneCommand *command = new RemovePaneCommand(this, pane); Chris@45: CommandHistory::getInstance()->addCommand(command); Chris@45: } Chris@45: Chris@45: CommandHistory::getInstance()->endCompoundOperation(); Chris@45: Chris@45: updateMenuStates(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::deleteCurrentLayer() Chris@45: { Chris@45: Pane *pane = m_paneStack->getCurrentPane(); Chris@45: if (pane) { Chris@45: Layer *layer = pane->getSelectedLayer(); Chris@45: if (layer) { Chris@45: m_document->removeLayerFromView(pane, layer); Chris@45: } Chris@45: } Chris@45: updateMenuStates(); Chris@45: } Chris@45: Chris@45: void Chris@123: MainWindowBase::editCurrentLayer() Chris@123: { Chris@123: Layer *layer = 0; Chris@123: Pane *pane = m_paneStack->getCurrentPane(); Chris@123: if (pane) layer = pane->getSelectedLayer(); Chris@123: if (!layer) return; Chris@123: Chris@123: Model *model = layer->getModel(); Chris@123: if (!model) return; Chris@123: Chris@124: TabularModel *tabular = dynamic_cast(model); Chris@124: if (!tabular) { Chris@124: //!!! how to prevent this function from being active if not Chris@124: //appropriate model type? or will we ultimately support Chris@124: //tabular display for all editable models? Chris@233: SVDEBUG << "NOTE: Not a tabular model" << endl; Chris@124: return; Chris@124: } Chris@124: Chris@123: if (m_layerDataDialogMap.find(layer) != m_layerDataDialogMap.end()) { Chris@126: if (!m_layerDataDialogMap[layer].isNull()) { Chris@126: m_layerDataDialogMap[layer]->show(); Chris@126: m_layerDataDialogMap[layer]->raise(); Chris@126: return; Chris@126: } Chris@123: } Chris@123: Chris@125: QString title = layer->getLayerPresentationName(); Chris@125: Chris@125: ModelDataTableDialog *dialog = new ModelDataTableDialog(tabular, title, this); Chris@128: dialog->setAttribute(Qt::WA_DeleteOnClose); Chris@128: Chris@128: connectLayerEditDialog(dialog); Chris@123: Chris@128: m_layerDataDialogMap[layer] = dialog; Chris@128: m_viewDataDialogMap[pane].insert(dialog); Chris@128: Chris@128: dialog->show(); Chris@128: } Chris@128: Chris@128: void Chris@128: MainWindowBase::connectLayerEditDialog(ModelDataTableDialog *dialog) Chris@128: { Chris@123: connect(m_viewManager, Chris@435: SIGNAL(globalCentreFrameChanged(sv_frame_t)), Chris@123: dialog, Chris@435: SLOT(userScrolledToFrame(sv_frame_t))); Chris@127: Chris@127: connect(m_viewManager, Chris@435: SIGNAL(playbackFrameChanged(sv_frame_t)), Chris@127: dialog, Chris@435: SLOT(playbackScrolledToFrame(sv_frame_t))); Chris@127: Chris@123: connect(dialog, Chris@435: SIGNAL(scrollToFrame(sv_frame_t)), Chris@123: m_viewManager, Chris@435: SLOT(setGlobalCentreFrame(sv_frame_t))); Chris@129: Chris@129: connect(dialog, Chris@435: SIGNAL(scrollToFrame(sv_frame_t)), Chris@129: m_viewManager, Chris@435: SLOT(setPlaybackFrame(sv_frame_t))); Chris@128: } Chris@123: Chris@123: void Chris@73: MainWindowBase::previousPane() Chris@73: { Chris@73: if (!m_paneStack) return; Chris@73: Chris@73: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@73: if (!currentPane) return; Chris@73: Chris@73: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@73: if (m_paneStack->getPane(i) == currentPane) { Chris@73: if (i == 0) return; Chris@73: m_paneStack->setCurrentPane(m_paneStack->getPane(i-1)); Chris@73: updateMenuStates(); Chris@73: return; Chris@73: } Chris@73: } Chris@73: } Chris@73: Chris@73: void Chris@73: MainWindowBase::nextPane() Chris@73: { Chris@73: if (!m_paneStack) return; Chris@73: Chris@73: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@73: if (!currentPane) return; Chris@73: Chris@73: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@73: if (m_paneStack->getPane(i) == currentPane) { Chris@73: if (i == m_paneStack->getPaneCount()-1) return; Chris@73: m_paneStack->setCurrentPane(m_paneStack->getPane(i+1)); Chris@73: updateMenuStates(); Chris@73: return; Chris@73: } Chris@73: } Chris@73: } Chris@73: Chris@73: void Chris@73: MainWindowBase::previousLayer() Chris@73: { Chris@73: if (!m_paneStack) return; Chris@73: Chris@73: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@73: if (!currentPane) return; Chris@73: Chris@403: int count = currentPane->getLayerCount(); Chris@403: if (count == 0) return; Chris@403: Chris@73: Layer *currentLayer = currentPane->getSelectedLayer(); Chris@403: Chris@403: if (!currentLayer) { Chris@403: // The pane itself is current Chris@403: m_paneStack->setCurrentLayer Chris@403: (currentPane, currentPane->getFixedOrderLayer(count-1)); Chris@403: } else { Chris@403: for (int i = 0; i < count; ++i) { Chris@403: if (currentPane->getFixedOrderLayer(i) == currentLayer) { Chris@403: if (i == 0) { Chris@403: m_paneStack->setCurrentLayer Chris@403: (currentPane, 0); // pane Chris@403: } else { Chris@403: m_paneStack->setCurrentLayer Chris@403: (currentPane, currentPane->getFixedOrderLayer(i-1)); Chris@403: } Chris@403: break; Chris@403: } Chris@73: } Chris@73: } Chris@403: Chris@403: updateMenuStates(); Chris@73: } Chris@73: Chris@73: void Chris@73: MainWindowBase::nextLayer() Chris@73: { Chris@73: if (!m_paneStack) return; Chris@73: Chris@73: Pane *currentPane = m_paneStack->getCurrentPane(); Chris@73: if (!currentPane) return; Chris@73: Chris@403: int count = currentPane->getLayerCount(); Chris@403: if (count == 0) return; Chris@403: Chris@73: Layer *currentLayer = currentPane->getSelectedLayer(); Chris@403: Chris@403: if (!currentLayer) { Chris@403: // The pane itself is current Chris@403: m_paneStack->setCurrentLayer Chris@403: (currentPane, currentPane->getFixedOrderLayer(0)); Chris@403: } else { Chris@403: for (int i = 0; i < count; ++i) { Chris@403: if (currentPane->getFixedOrderLayer(i) == currentLayer) { Chris@403: if (i == currentPane->getLayerCount()-1) { Chris@403: m_paneStack->setCurrentLayer Chris@403: (currentPane, 0); // pane Chris@403: } else { Chris@403: m_paneStack->setCurrentLayer Chris@403: (currentPane, currentPane->getFixedOrderLayer(i+1)); Chris@403: } Chris@403: break; Chris@403: } Chris@73: } Chris@73: } Chris@403: Chris@403: updateMenuStates(); Chris@73: } Chris@73: Chris@73: void Chris@435: MainWindowBase::playbackFrameChanged(sv_frame_t frame) Chris@45: { Chris@45: if (!(m_playSource && m_playSource->isPlaying()) || !getMainModel()) return; Chris@45: Chris@187: updatePositionStatusDisplays(); Chris@187: Chris@45: RealTime now = RealTime::frame2RealTime Chris@45: (frame, getMainModel()->getSampleRate()); Chris@45: Chris@45: if (now.sec == m_lastPlayStatusSec) return; Chris@45: Chris@45: RealTime then = RealTime::frame2RealTime Chris@45: (m_playSource->getPlayEndFrame(), getMainModel()->getSampleRate()); Chris@45: Chris@45: QString nowStr; Chris@45: QString thenStr; Chris@45: QString remainingStr; Chris@45: Chris@45: if (then.sec > 10) { Chris@45: nowStr = now.toSecText().c_str(); Chris@45: thenStr = then.toSecText().c_str(); Chris@45: remainingStr = (then - now).toSecText().c_str(); Chris@45: m_lastPlayStatusSec = now.sec; Chris@45: } else { Chris@45: nowStr = now.toText(true).c_str(); Chris@45: thenStr = then.toText(true).c_str(); Chris@45: remainingStr = (then - now).toText(true).c_str(); Chris@45: } Chris@45: Chris@45: m_myStatusMessage = tr("Playing: %1 of %2 (%3 remaining)") Chris@45: .arg(nowStr).arg(thenStr).arg(remainingStr); Chris@45: Chris@378: getStatusLabel()->setText(m_myStatusMessage); Chris@45: } Chris@45: Chris@45: void Chris@485: MainWindowBase::recordDurationChanged(sv_frame_t frame, sv_samplerate_t rate) Chris@485: { Chris@485: RealTime duration = RealTime::frame2RealTime(frame, rate); Chris@485: QString durStr = duration.toSecText().c_str(); Chris@485: Chris@485: m_myStatusMessage = tr("Recording: %1").arg(durStr); Chris@485: Chris@485: getStatusLabel()->setText(m_myStatusMessage); Chris@485: } Chris@485: Chris@485: void Chris@435: MainWindowBase::globalCentreFrameChanged(sv_frame_t ) Chris@45: { Chris@45: if ((m_playSource && m_playSource->isPlaying()) || !getMainModel()) return; Chris@45: Pane *p = 0; Chris@45: if (!m_paneStack || !(p = m_paneStack->getCurrentPane())) return; Chris@45: if (!p->getFollowGlobalPan()) return; Chris@45: updateVisibleRangeDisplay(p); Chris@45: } Chris@45: Chris@45: void Chris@435: MainWindowBase::viewCentreFrameChanged(View *v, sv_frame_t frame) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::viewCentreFrameChanged(" << v << "," << frame << ")" << endl; Chris@123: Chris@123: if (m_viewDataDialogMap.find(v) != m_viewDataDialogMap.end()) { Chris@123: for (DataDialogSet::iterator i = m_viewDataDialogMap[v].begin(); Chris@123: i != m_viewDataDialogMap[v].end(); ++i) { Chris@127: (*i)->userScrolledToFrame(frame); Chris@123: } Chris@123: } Chris@45: if ((m_playSource && m_playSource->isPlaying()) || !getMainModel()) return; Chris@45: Pane *p = 0; Chris@45: if (!m_paneStack || !(p = m_paneStack->getCurrentPane())) return; Chris@45: if (v == p) updateVisibleRangeDisplay(p); Chris@45: } Chris@45: Chris@45: void Chris@366: MainWindowBase::viewZoomLevelChanged(View *v, int , bool ) Chris@45: { Chris@45: if ((m_playSource && m_playSource->isPlaying()) || !getMainModel()) return; Chris@45: Pane *p = 0; Chris@45: if (!m_paneStack || !(p = m_paneStack->getCurrentPane())) return; Chris@45: if (v == p) updateVisibleRangeDisplay(p); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::layerAdded(Layer *) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::layerAdded(" << layer << ")" << endl; Chris@45: updateMenuStates(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::layerRemoved(Layer *) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::layerRemoved(" << layer << ")" << endl; Chris@45: updateMenuStates(); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::layerAboutToBeDeleted(Layer *layer) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::layerAboutToBeDeleted(" << layer << ")" << endl; Chris@123: Chris@128: removeLayerEditDialog(layer); Chris@123: Chris@47: if (m_timeRulerLayer && (layer == m_timeRulerLayer)) { Chris@293: // cerr << "(this is the time ruler layer)" << endl; Chris@45: m_timeRulerLayer = 0; Chris@45: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::layerInAView(Layer *layer, bool inAView) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::layerInAView(" << layer << "," << inAView << ")" << endl; Chris@128: Chris@128: if (!inAView) removeLayerEditDialog(layer); Chris@45: Chris@45: // Check whether we need to add or remove model from play source Chris@45: Model *model = layer->getModel(); Chris@45: if (model) { Chris@45: if (inAView) { Chris@45: m_playSource->addModel(model); Chris@45: } else { Chris@45: bool found = false; Chris@45: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@45: Pane *pane = m_paneStack->getPane(i); Chris@45: if (!pane) continue; Chris@45: for (int j = 0; j < pane->getLayerCount(); ++j) { Chris@45: Layer *pl = pane->getLayer(j); Chris@183: if (pl && Chris@183: !dynamic_cast(pl) && Chris@183: (pl->getModel() == model)) { Chris@45: found = true; Chris@45: break; Chris@45: } Chris@45: } Chris@45: if (found) break; Chris@45: } Chris@173: if (!found) { Chris@173: m_playSource->removeModel(model); Chris@173: } Chris@45: } Chris@45: } Chris@45: Chris@45: updateMenuStates(); Chris@45: } Chris@45: Chris@45: void Chris@128: MainWindowBase::removeLayerEditDialog(Layer *layer) Chris@128: { Chris@128: if (m_layerDataDialogMap.find(layer) != m_layerDataDialogMap.end()) { Chris@128: Chris@128: ModelDataTableDialog *dialog = m_layerDataDialogMap[layer]; Chris@128: Chris@128: for (ViewDataDialogMap::iterator vi = m_viewDataDialogMap.begin(); Chris@128: vi != m_viewDataDialogMap.end(); ++vi) { Chris@128: vi->second.erase(dialog); Chris@128: } Chris@128: Chris@128: m_layerDataDialogMap.erase(layer); Chris@128: delete dialog; Chris@128: } Chris@128: } Chris@128: Chris@128: void Chris@45: MainWindowBase::modelAdded(Model *model) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::modelAdded(" << model << ")" << endl; gyorgyf@273: std::cerr << "\nAdding model " << model->getTypeName() << " to playsource " << std::endl; Chris@45: m_playSource->addModel(model); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::mainModelChanged(WaveFileModel *model) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::mainModelChanged(" << model << ")" << endl; Chris@45: updateDescriptionLabel(); Chris@45: if (model) m_viewManager->setMainModelSampleRate(model->getSampleRate()); Chris@475: if (model && !(m_playTarget || m_audioIO) && Chris@475: (m_soundOptions & WithAudioOutput)) { Chris@475: createAudioIO(); Chris@360: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::modelAboutToBeDeleted(Model *model) Chris@45: { Chris@233: // SVDEBUG << "MainWindowBase::modelAboutToBeDeleted(" << model << ")" << endl; Chris@45: if (model == m_viewManager->getPlaybackModel()) { Chris@45: m_viewManager->setPlaybackModel(0); Chris@45: } Chris@45: m_playSource->removeModel(model); Chris@45: } Chris@45: Chris@45: void Chris@55: MainWindowBase::paneDeleteButtonClicked(Pane *pane) Chris@55: { Chris@55: bool found = false; Chris@55: for (int i = 0; i < m_paneStack->getPaneCount(); ++i) { Chris@55: if (m_paneStack->getPane(i) == pane) { Chris@55: found = true; Chris@55: break; Chris@55: } Chris@55: } Chris@55: if (!found) { Chris@233: SVDEBUG << "MainWindowBase::paneDeleteButtonClicked: Unknown pane " Chris@229: << pane << endl; Chris@55: return; Chris@55: } Chris@55: Chris@55: CommandHistory::getInstance()->startCompoundOperation Chris@55: (tr("Delete Pane"), true); Chris@55: Chris@55: while (pane->getLayerCount() > 0) { Chris@55: Layer *layer = pane->getLayer(0); Chris@55: if (layer) { Chris@55: m_document->removeLayerFromView(pane, layer); Chris@55: } else { Chris@55: break; Chris@55: } Chris@55: } Chris@55: Chris@55: RemovePaneCommand *command = new RemovePaneCommand(this, pane); Chris@55: CommandHistory::getInstance()->addCommand(command); Chris@55: Chris@55: CommandHistory::getInstance()->endCompoundOperation(); Chris@55: Chris@55: updateMenuStates(); Chris@55: } Chris@55: Chris@55: void Chris@45: MainWindowBase::pollOSC() Chris@45: { Chris@45: if (!m_oscQueue || m_oscQueue->isEmpty()) return; Chris@233: SVDEBUG << "MainWindowBase::pollOSC: have " << m_oscQueue->getMessagesAvailable() << " messages" << endl; Chris@45: Chris@45: if (m_openingAudioFile) return; Chris@45: Chris@45: OSCMessage message = m_oscQueue->readMessage(); Chris@45: Chris@45: if (message.getTarget() != 0) { Chris@45: return; //!!! for now -- this class is target 0, others not handled yet Chris@45: } Chris@45: Chris@45: handleOSCMessage(message); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::inProgressSelectionChanged() Chris@45: { Chris@45: Pane *currentPane = 0; Chris@45: if (m_paneStack) currentPane = m_paneStack->getCurrentPane(); justin@331: if (currentPane) { justin@331: //cerr << "JTEST: mouse event on selection pane" << endl; justin@331: updateVisibleRangeDisplay(currentPane); justin@331: } Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::contextHelpChanged(const QString &s) Chris@45: { Chris@378: QLabel *lab = getStatusLabel(); Chris@375: Chris@45: if (s == "" && m_myStatusMessage != "") { Chris@378: if (lab->text() != m_myStatusMessage) { Chris@378: lab->setText(m_myStatusMessage); Chris@375: } Chris@45: return; Chris@45: } Chris@375: Chris@378: lab->setText(s); Chris@45: } Chris@45: Chris@45: void Chris@45: MainWindowBase::openHelpUrl(QString url) Chris@45: { Chris@45: // This method mostly lifted from Qt Assistant source code Chris@45: Chris@45: QProcess *process = new QProcess(this); Chris@45: connect(process, SIGNAL(finished(int)), process, SLOT(deleteLater())); Chris@45: Chris@45: QStringList args; Chris@45: Chris@45: #ifdef Q_OS_MAC Chris@45: args.append(url); Chris@45: process->start("open", args); Chris@45: #else Chris@45: #ifdef Q_OS_WIN32 Chris@358: QString pf(getenv("ProgramFiles")); Chris@358: QString command = pf + QString("\\Internet Explorer\\IEXPLORE.EXE"); Chris@358: Chris@358: args.append(url); Chris@358: process->start(command, args); Chris@45: #else Chris@45: if (!qgetenv("KDE_FULL_SESSION").isEmpty()) { Chris@45: args.append("exec"); Chris@45: args.append(url); Chris@45: process->start("kfmclient", args); Chris@45: } else if (!qgetenv("BROWSER").isEmpty()) { Chris@45: args.append(url); Chris@45: process->start(qgetenv("BROWSER"), args); Chris@45: } else { Chris@45: args.append(url); Chris@45: process->start("firefox", args); Chris@45: } Chris@45: #endif Chris@45: #endif Chris@45: } Chris@45: Chris@483: void Chris@483: MainWindowBase::openLocalFolder(QString path) Chris@483: { Chris@483: QDir d(path); Chris@483: if (d.exists()) { Chris@483: QStringList args; Chris@483: QString path = d.canonicalPath(); Chris@483: #if defined Q_OS_WIN32 Chris@483: // Although the Win32 API is quite happy to have Chris@483: // forward slashes as directory separators, Windows Chris@483: // Explorer is not Chris@483: path = path.replace('/', '\\'); Chris@483: args << path; Chris@483: QProcess::execute("c:/windows/explorer.exe", args); Chris@483: #else Chris@483: args << path; Chris@483: QProcess::execute( Chris@483: #if defined Q_OS_MAC Chris@483: "/usr/bin/open", Chris@483: #else Chris@483: "/usr/bin/xdg-open", Chris@483: #endif Chris@483: args); Chris@483: #endif Chris@483: } Chris@483: } Chris@483: