/* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */ /* Rosegarden A MIDI and audio sequencer and musical notation editor. This program is Copyright 2000-2008 Guillaume Laurent , Chris Cannam , Richard Bown The moral rights of Guillaume Laurent, Chris Cannam, and Richard Bown to claim authorship of this work have been asserted. Other copyrights also apply to some parts of this work. Please see the AUTHORS file and individual file headers for details. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. See the file COPYING included with this distribution for more information. */ #include "RosegardenGUIView.h" #include #include "sound/Midi.h" #include "gui/editors/segment/TrackButtons.h" #include #include "misc/Debug.h" #include "misc/Strings.h" #include "document/ConfigGroups.h" #include "gui/application/RosegardenDCOP.h" #include "base/AudioLevel.h" #include "base/Composition.h" #include "base/Instrument.h" #include "base/MidiDevice.h" #include "base/MidiProgram.h" #include "base/NotationTypes.h" #include "base/RealTime.h" #include "base/RulerScale.h" #include "base/Segment.h" #include "base/Selection.h" #include "base/Studio.h" #include "base/Track.h" #include "commands/segment/AudioSegmentAutoSplitCommand.h" #include "commands/segment/AudioSegmentInsertCommand.h" #include "commands/segment/SegmentSingleRepeatToCopyCommand.h" #include "document/MultiViewCommandHistory.h" #include "document/RosegardenGUIDoc.h" #include "RosegardenApplication.h" #include "gui/configuration/GeneralConfigurationPage.h" #include "gui/configuration/AudioConfigurationPage.h" #include "gui/dialogs/AudioSplitDialog.h" #include "gui/dialogs/AudioManagerDialog.h" #include "gui/dialogs/DocumentConfigureDialog.h" #include "gui/dialogs/TempoDialog.h" #include "gui/editors/eventlist/EventView.h" #include "gui/editors/matrix/MatrixView.h" #include "gui/editors/notation/NotationView.h" #include "gui/editors/parameters/InstrumentParameterBox.h" #include "gui/editors/parameters/SegmentParameterBox.h" #include "gui/editors/parameters/TrackParameterBox.h" #include "gui/editors/segment/segmentcanvas/CompositionView.h" #include "gui/editors/segment/segmentcanvas/SegmentSelector.h" #include "gui/editors/segment/TrackEditor.h" #include "gui/seqmanager/SequenceManager.h" #include "gui/seqmanager/SequencerMapper.h" #include "gui/rulers/ChordNameRuler.h" #include "gui/rulers/LoopRuler.h" #include "gui/rulers/TempoRuler.h" #include "gui/rulers/StandardRuler.h" #include "gui/widgets/ProgressDialog.h" #include "gui/widgets/CurrentProgressDialog.h" #include "RosegardenGUIApp.h" #include "SetWaitCursor.h" #include "sound/AudioFile.h" #include "sound/AudioFileManager.h" #include "sound/MappedEvent.h" #include #include #include #include #include #include #include #include #include #include #include #include namespace Rosegarden { // Use this to define the basic unit of the main QCanvas size. // // This apparently arbitrary figure is what we think is an // appropriate width in pixels for a 4/4 bar. Beware of making it // too narrow, as shorter bars will be proportionally smaller -- // the visual difference between 2/4 and 4/4 is perhaps greater // than it sounds. // static double barWidth44 = 100.0; const QWidget *RosegardenGUIView::m_lastActiveMainWindow = 0; // This is the maximum number of matrix, event view or percussion // matrix editors to open in a single operation (not the maximum that // can be open at a time -- there isn't one) // static int maxEditorsToOpen = 8; RosegardenGUIView::RosegardenGUIView(bool showTrackLabels, SegmentParameterBox* segmentParameterBox, InstrumentParameterBox* instrumentParameterBox, TrackParameterBox* trackParameterBox, QWidget *parent, const char* /*name*/) : QVBox(parent), m_rulerScale(0), m_trackEditor(0), m_segmentParameterBox(segmentParameterBox), m_instrumentParameterBox(instrumentParameterBox), m_trackParameterBox(trackParameterBox) { RosegardenGUIDoc* doc = getDocument(); Composition *comp = &doc->getComposition(); double unitsPerPixel = TimeSignature(4, 4).getBarDuration() / barWidth44; m_rulerScale = new SimpleRulerScale(comp, 0, unitsPerPixel); // Construct the trackEditor first so we can then // query it for placement information // m_trackEditor = new TrackEditor(doc, this, m_rulerScale, showTrackLabels, unitsPerPixel, this /*hbox*/); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(editSegment(Segment*)), SLOT(slotEditSegment(Segment*))); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(editSegmentNotation(Segment*)), SLOT(slotEditSegmentNotation(Segment*))); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(editSegmentMatrix(Segment*)), SLOT(slotEditSegmentMatrix(Segment*))); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(editSegmentAudio(Segment*)), SLOT(slotEditSegmentAudio(Segment*))); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(audioSegmentAutoSplit(Segment*)), SLOT(slotSegmentAutoSplit(Segment*))); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(editSegmentEventList(Segment*)), SLOT(slotEditSegmentEventList(Segment*))); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(editRepeat(Segment*, timeT)), SLOT(slotEditRepeat(Segment*, timeT))); connect(m_trackEditor->getSegmentCanvas(), SIGNAL(setPointerPosition(timeT)), doc, SLOT(slotSetPointerPosition(timeT))); connect(m_trackEditor, SIGNAL(droppedDocument(QString)), parent, SLOT(slotOpenDroppedURL(QString))); connect(m_trackEditor, SIGNAL(droppedAudio(QString)), this, SLOT(slotDroppedAudio(QString))); connect(m_trackEditor, SIGNAL(droppedNewAudio(QString)), this, SLOT(slotDroppedNewAudio(QString))); connect(m_instrumentParameterBox, SIGNAL(changeInstrumentLabel(InstrumentId, QString)), this, SLOT(slotChangeInstrumentLabel(InstrumentId, QString))); connect(m_instrumentParameterBox, SIGNAL(changeInstrumentLabel(InstrumentId, QString)), m_trackParameterBox, SLOT(slotInstrumentLabelChanged(InstrumentId, QString))); connect(m_trackEditor->getTrackButtons(), SIGNAL(nameChanged()), m_trackParameterBox, SLOT(slotSelectedTrackNameChanged())); connect(m_trackEditor->getTrackButtons(), SIGNAL(instrumentSelected(int)), m_trackParameterBox, SLOT(slotUpdateControls(int))); connect(m_trackParameterBox, SIGNAL(instrumentSelected(TrackId, int)), m_trackEditor->getTrackButtons(), SLOT(slotTrackInstrumentSelection(TrackId, int))); connect(this, SIGNAL(controllerDeviceEventReceived(MappedEvent *, const void *)), this, SLOT(slotControllerDeviceEventReceived(MappedEvent *, const void *))); if (doc) { /* signal no longer exists connect(doc, SIGNAL(recordingSegmentUpdated(Segment *, timeT)), this, SLOT(slotUpdateRecordingSegment(Segment *, timeT))); */ QObject::connect (getCommandHistory(), SIGNAL(commandExecuted()), m_trackEditor->getSegmentCanvas(), SLOT(slotUpdateSegmentsDrawBuffer())); } } RosegardenGUIView::~RosegardenGUIView() { RG_DEBUG << "~RosegardenGUIView()" << endl; delete m_rulerScale; } RosegardenGUIDoc* RosegardenGUIView::getDocument() const { return RosegardenGUIApp::self()->getDocument(); } void RosegardenGUIView::print(Composition* p, bool previewOnly) { SetWaitCursor waitCursor; std::vector segmentsToEdit; for (Composition::iterator i = p->begin(); i != p->end(); ++i) { if ((*i)->getType() != Segment::Audio) { segmentsToEdit.push_back(*i); } } if (segmentsToEdit.empty()) { KMessageBox::sorry(this, i18n("No non-audio segments in composition")); return ; } NotationView *notationView = new NotationView(getDocument(), segmentsToEdit, this, (NotationView *)0); if (!notationView->isOK()) { RG_DEBUG << "RosegardenGUIView::print : operation cancelled" << endl; delete notationView; return ; } notationView->print(previewOnly); delete notationView; } void RosegardenGUIView::selectTool(const QString toolName) { m_trackEditor->getSegmentCanvas()->slotSetTool(toolName); } bool RosegardenGUIView::haveSelection() { return m_trackEditor->getSegmentCanvas()->haveSelection(); } SegmentSelection RosegardenGUIView::getSelection() { return m_trackEditor->getSegmentCanvas()->getSelectedSegments(); } void RosegardenGUIView::updateSelectionContents() { m_trackEditor->getSegmentCanvas()->updateSelectionContents(); } /* hjj: WHAT DO DO WITH THIS ? void RosegardenGUIView::slotEditMetadata(QString name) { const QWidget *ww = dynamic_cast(sender()); QWidget *w = const_cast(ww); DocumentConfigureDialog *configDlg = new DocumentConfigureDialog(getDocument(), w ? w : this); configDlg->selectMetadata(name); configDlg->show(); } */ void RosegardenGUIView::slotEditSegment(Segment* segment) { Segment::SegmentType type = Segment::Internal; if (segment) { type = segment->getType(); } else { if (haveSelection()) { bool haveType = false; SegmentSelection selection = getSelection(); for (SegmentSelection::iterator i = selection.begin(); i != selection.end(); ++i) { Segment::SegmentType myType = (*i)->getType(); if (haveType) { if (myType != type) { KMessageBox::sorry(this, i18n("Selection must contain only audio or non-audio segments")); return ; } } else { type = myType; haveType = true; segment = *i; } } } else return ; } if (type == Segment::Audio) { slotEditSegmentAudio(segment); } else { KConfig* config = kapp->config(); config->setGroup(GeneralOptionsConfigGroup); GeneralConfigurationPage::DoubleClickClient client = (GeneralConfigurationPage::DoubleClickClient) (config->readUnsignedNumEntry("doubleclickclient", (unsigned int)GeneralConfigurationPage::NotationView)); if (client == GeneralConfigurationPage::MatrixView) { bool isPercussion = false; Track *track = getDocument()->getComposition().getTrackById (segment->getTrack()); if (track) { InstrumentId iid = track->getInstrument(); Instrument *instrument = getDocument()->getStudio().getInstrumentById(iid); if (instrument && instrument->isPercussion()) isPercussion = true; } if (isPercussion) { slotEditSegmentPercussionMatrix(segment); } else { slotEditSegmentMatrix(segment); } } else if (client == GeneralConfigurationPage::EventView) { slotEditSegmentEventList(segment); } else { slotEditSegmentNotation(segment); } } } void RosegardenGUIView::slotEditRepeat(Segment *segment, timeT time) { SegmentSingleRepeatToCopyCommand *command = new SegmentSingleRepeatToCopyCommand(segment, time); slotAddCommandToHistory(command); } void RosegardenGUIView::slotEditSegmentNotation(Segment* p) { SetWaitCursor waitCursor; std::vector segmentsToEdit; RG_DEBUG << "\n\n\n\nRosegardenGUIView::slotEditSegmentNotation: p is " << p << endl; // The logic here is: If we're calling for this operation to // happen on a particular segment, then open that segment and if // it's part of a selection open all other selected segments too. // If we're not calling for any particular segment, then open all // selected segments if there are any. if (haveSelection()) { SegmentSelection selection = getSelection(); if (!p || (selection.find(p) != selection.end())) { for (SegmentSelection::iterator i = selection.begin(); i != selection.end(); ++i) { if ((*i)->getType() != Segment::Audio) { segmentsToEdit.push_back(*i); } } } else { if (p->getType() != Segment::Audio) { segmentsToEdit.push_back(p); } } } else if (p) { if (p->getType() != Segment::Audio) { segmentsToEdit.push_back(p); } } else { return ; } if (segmentsToEdit.empty()) { KMessageBox::sorry(this, i18n("No non-audio segments selected")); return ; } slotEditSegmentsNotation(segmentsToEdit); } void RosegardenGUIView::slotEditSegmentsNotation(std::vector segmentsToEdit) { NotationView *view = createNotationView(segmentsToEdit); if (view) view->show(); } NotationView * RosegardenGUIView::createNotationView(std::vector segmentsToEdit) { NotationView *notationView = new NotationView(getDocument(), segmentsToEdit, this, true); if (!notationView->isOK()) { RG_DEBUG << "slotEditSegmentNotation : operation cancelled" << endl; delete notationView; return 0; } // For tempo changes (ugh -- it'd be nicer to make a tempo change // command that could interpret all this stuff from the dialog) // connect(notationView, SIGNAL(changeTempo(timeT, tempoT, tempoT, TempoDialog::TempoDialogAction)), RosegardenGUIApp::self(), SLOT(slotChangeTempo(timeT, tempoT, tempoT, TempoDialog::TempoDialogAction))); connect(notationView, SIGNAL(windowActivated()), this, SLOT(slotActiveMainWindowChanged())); connect(notationView, SIGNAL(selectTrack(int)), this, SLOT(slotSelectTrackSegments(int))); connect(notationView, SIGNAL(play()), RosegardenGUIApp::self(), SLOT(slotPlay())); connect(notationView, SIGNAL(stop()), RosegardenGUIApp::self(), SLOT(slotStop())); connect(notationView, SIGNAL(fastForwardPlayback()), RosegardenGUIApp::self(), SLOT(slotFastforward())); connect(notationView, SIGNAL(rewindPlayback()), RosegardenGUIApp::self(), SLOT(slotRewind())); connect(notationView, SIGNAL(fastForwardPlaybackToEnd()), RosegardenGUIApp::self(), SLOT(slotFastForwardToEnd())); connect(notationView, SIGNAL(rewindPlaybackToBeginning()), RosegardenGUIApp::self(), SLOT(slotRewindToBeginning())); connect(notationView, SIGNAL(panic()), RosegardenGUIApp::self(), SLOT(slotPanic())); connect(notationView, SIGNAL(saveFile()), RosegardenGUIApp::self(), SLOT(slotFileSave())); connect(notationView, SIGNAL(jumpPlaybackTo(timeT)), getDocument(), SLOT(slotSetPointerPosition(timeT))); connect(notationView, SIGNAL(openInNotation(std::vector)), this, SLOT(slotEditSegmentsNotation(std::vector))); connect(notationView, SIGNAL(openInMatrix(std::vector)), this, SLOT(slotEditSegmentsMatrix(std::vector))); connect(notationView, SIGNAL(openInPercussionMatrix(std::vector)), this, SLOT(slotEditSegmentsPercussionMatrix(std::vector))); connect(notationView, SIGNAL(openInEventList(std::vector)), this, SLOT(slotEditSegmentsEventList(std::vector))); /* hjj: WHAT DO DO WITH THIS ? connect(notationView, SIGNAL(editMetadata(QString)), this, SLOT(slotEditMetadata(QString))); */ connect(notationView, SIGNAL(editTriggerSegment(int)), this, SLOT(slotEditTriggerSegment(int))); connect(notationView, SIGNAL(staffLabelChanged(TrackId, QString)), this, SLOT(slotChangeTrackLabel(TrackId, QString))); connect(notationView, SIGNAL(toggleSolo(bool)), RosegardenGUIApp::self(), SLOT(slotToggleSolo(bool))); connect(notationView, SIGNAL(editTimeSignature(timeT)), RosegardenGUIApp::self(), SLOT(slotEditTempos(timeT))); SequenceManager *sM = getDocument()->getSequenceManager(); connect(sM, SIGNAL(insertableNoteOnReceived(int, int)), notationView, SLOT(slotInsertableNoteOnReceived(int, int))); connect(sM, SIGNAL(insertableNoteOffReceived(int, int)), notationView, SLOT(slotInsertableNoteOffReceived(int, int))); connect(notationView, SIGNAL(stepByStepTargetRequested(QObject *)), this, SIGNAL(stepByStepTargetRequested(QObject *))); connect(this, SIGNAL(stepByStepTargetRequested(QObject *)), notationView, SLOT(slotStepByStepTargetRequested(QObject *))); connect(RosegardenGUIApp::self(), SIGNAL(compositionStateUpdate()), notationView, SLOT(slotCompositionStateUpdate())); connect(this, SIGNAL(compositionStateUpdate()), notationView, SLOT(slotCompositionStateUpdate())); // Encourage the notation view window to open to the same // interval as the current segment view if (m_trackEditor->getSegmentCanvas()->horizontalScrollBar()->value() > 1) { // don't scroll unless we need to // first find the time at the center of the visible segment canvas int centerX = (int)(m_trackEditor->getSegmentCanvas()->contentsX() + m_trackEditor->getSegmentCanvas()->visibleWidth() / 2); timeT centerSegmentView = m_trackEditor->getRulerScale()->getTimeForX(centerX); // then scroll the notation view to that time, "localized" for the current segment notationView->scrollToTime(centerSegmentView); notationView->updateView(); } return notationView; } void RosegardenGUIView::slotEditSegmentMatrix(Segment* p) { SetWaitCursor waitCursor; std::vector segmentsToEdit; // unlike notation, if we're calling for this on a particular // segment we don't open all the other selected segments as well // (fine in notation because they're in a single window) if (p) { if (p->getType() != Segment::Audio) { segmentsToEdit.push_back(p); } } else { int count = 0; SegmentSelection selection = getSelection(); for (SegmentSelection::iterator i = selection.begin(); i != selection.end(); ++i) { if ((*i)->getType() != Segment::Audio) { slotEditSegmentMatrix(*i); if (++count == maxEditorsToOpen) break; } } return ; } if (segmentsToEdit.empty()) { KMessageBox::sorry(this, i18n("No non-audio segments selected")); return ; } slotEditSegmentsMatrix(segmentsToEdit); } void RosegardenGUIView::slotEditSegmentPercussionMatrix(Segment* p) { SetWaitCursor waitCursor; std::vector segmentsToEdit; // unlike notation, if we're calling for this on a particular // segment we don't open all the other selected segments as well // (fine in notation because they're in a single window) if (p) { if (p->getType() != Segment::Audio) { segmentsToEdit.push_back(p); } } else { int count = 0; SegmentSelection selection = getSelection(); for (SegmentSelection::iterator i = selection.begin(); i != selection.end(); ++i) { if ((*i)->getType() != Segment::Audio) { slotEditSegmentPercussionMatrix(*i); if (++count == maxEditorsToOpen) break; } } return ; } if (segmentsToEdit.empty()) { KMessageBox::sorry(this, i18n("No non-audio segments selected")); return ; } slotEditSegmentsPercussionMatrix(segmentsToEdit); } void RosegardenGUIView::slotEditSegmentsMatrix(std::vector segmentsToEdit) { int count = 0; for (std::vector::iterator i = segmentsToEdit.begin(); i != segmentsToEdit.end(); ++i) { std::vector tmpvec; tmpvec.push_back(*i); MatrixView *view = createMatrixView(tmpvec, false); if (view) { view->show(); if (++count == maxEditorsToOpen) break; } } } void RosegardenGUIView::slotEditSegmentsPercussionMatrix(std::vector segmentsToEdit) { int count = 0; for (std::vector::iterator i = segmentsToEdit.begin(); i != segmentsToEdit.end(); ++i) { std::vector tmpvec; tmpvec.push_back(*i); MatrixView *view = createMatrixView(tmpvec, true); if (view) { view->show(); if (++count == maxEditorsToOpen) break; } } } MatrixView * RosegardenGUIView::createMatrixView(std::vector segmentsToEdit, bool drumMode) { MatrixView *matrixView = new MatrixView(getDocument(), segmentsToEdit, this, drumMode); // For tempo changes (ugh -- it'd be nicer to make a tempo change // command that could interpret all this stuff from the dialog) // connect(matrixView, SIGNAL(changeTempo(timeT, tempoT, tempoT, TempoDialog::TempoDialogAction)), RosegardenGUIApp::self(), SLOT(slotChangeTempo(timeT, tempoT, tempoT, TempoDialog::TempoDialogAction))); connect(matrixView, SIGNAL(windowActivated()), this, SLOT(slotActiveMainWindowChanged())); connect(matrixView, SIGNAL(selectTrack(int)), this, SLOT(slotSelectTrackSegments(int))); connect(matrixView, SIGNAL(play()), RosegardenGUIApp::self(), SLOT(slotPlay())); connect(matrixView, SIGNAL(stop()), RosegardenGUIApp::self(), SLOT(slotStop())); connect(matrixView, SIGNAL(fastForwardPlayback()), RosegardenGUIApp::self(), SLOT(slotFastforward())); connect(matrixView, SIGNAL(rewindPlayback()), RosegardenGUIApp::self(), SLOT(slotRewind())); connect(matrixView, SIGNAL(fastForwardPlaybackToEnd()), RosegardenGUIApp::self(), SLOT(slotFastForwardToEnd())); connect(matrixView, SIGNAL(rewindPlaybackToBeginning()), RosegardenGUIApp::self(), SLOT(slotRewindToBeginning())); connect(matrixView, SIGNAL(panic()), RosegardenGUIApp::self(), SLOT(slotPanic())); connect(matrixView, SIGNAL(saveFile()), RosegardenGUIApp::self(), SLOT(slotFileSave())); connect(matrixView, SIGNAL(jumpPlaybackTo(timeT)), getDocument(), SLOT(slotSetPointerPosition(timeT))); connect(matrixView, SIGNAL(openInNotation(std::vector)), this, SLOT(slotEditSegmentsNotation(std::vector))); connect(matrixView, SIGNAL(openInMatrix(std::vector)), this, SLOT(slotEditSegmentsMatrix(std::vector))); connect(matrixView, SIGNAL(openInEventList(std::vector)), this, SLOT(slotEditSegmentsEventList(std::vector))); connect(matrixView, SIGNAL(editTriggerSegment(int)), this, SLOT(slotEditTriggerSegment(int))); connect(matrixView, SIGNAL(toggleSolo(bool)), RosegardenGUIApp::self(), SLOT(slotToggleSolo(bool))); connect(matrixView, SIGNAL(editTimeSignature(timeT)), RosegardenGUIApp::self(), SLOT(slotEditTempos(timeT))); SequenceManager *sM = getDocument()->getSequenceManager(); connect(sM, SIGNAL(insertableNoteOnReceived(int, int)), matrixView, SLOT(slotInsertableNoteOnReceived(int, int))); connect(sM, SIGNAL(insertableNoteOffReceived(int, int)), matrixView, SLOT(slotInsertableNoteOffReceived(int, int))); connect(matrixView, SIGNAL(stepByStepTargetRequested(QObject *)), this, SIGNAL(stepByStepTargetRequested(QObject *))); connect(this, SIGNAL(stepByStepTargetRequested(QObject *)), matrixView, SLOT(slotStepByStepTargetRequested(QObject *))); connect(RosegardenGUIApp::self(), SIGNAL(compositionStateUpdate()), matrixView, SLOT(slotCompositionStateUpdate())); connect(this, SIGNAL(compositionStateUpdate()), matrixView, SLOT(slotCompositionStateUpdate())); connect(this, SIGNAL(instrumentLevelsChanged(InstrumentId, const LevelInfo &)), matrixView, SLOT(slotInstrumentLevelsChanged(InstrumentId, const LevelInfo &))); // Encourage the matrix view window to open to the same // interval as the current segment view if (m_trackEditor->getSegmentCanvas()->horizontalScrollBar()->value() > 1) { // don't scroll unless we need to // first find the time at the center of the visible segment canvas int centerX = (int)(m_trackEditor->getSegmentCanvas()->contentsX()); // Seems to work better for matrix view to scroll to left side // + m_trackEditor->getSegmentCanvas()->visibleWidth() / 2); timeT centerSegmentView = m_trackEditor->getRulerScale()->getTimeForX(centerX); // then scroll the notation view to that time, "localized" for the current segment matrixView->scrollToTime(centerSegmentView); matrixView->updateView(); } return matrixView; } void RosegardenGUIView::slotEditSegmentEventList(Segment *p) { SetWaitCursor waitCursor; std::vector segmentsToEdit; // unlike notation, if we're calling for this on a particular // segment we don't open all the other selected segments as well // (fine in notation because they're in a single window) if (p) { if (p->getType() != Segment::Audio) { segmentsToEdit.push_back(p); } } else { int count = 0; SegmentSelection selection = getSelection(); for (SegmentSelection::iterator i = selection.begin(); i != selection.end(); ++i) { if ((*i)->getType() != Segment::Audio) { slotEditSegmentEventList(*i); if (++count == maxEditorsToOpen) break; } } return ; } if (segmentsToEdit.empty()) { KMessageBox::sorry(this, i18n("No non-audio segments selected")); return ; } slotEditSegmentsEventList(segmentsToEdit); } void RosegardenGUIView::slotEditSegmentsEventList(std::vector segmentsToEdit) { int count = 0; for (std::vector::iterator i = segmentsToEdit.begin(); i != segmentsToEdit.end(); ++i) { std::vector tmpvec; tmpvec.push_back(*i); EventView *view = createEventView(tmpvec); if (view) { view->show(); if (++count == maxEditorsToOpen) break; } } } void RosegardenGUIView::slotEditTriggerSegment(int id) { SetWaitCursor waitCursor; std::vector segmentsToEdit; Segment *s = getDocument()->getComposition().getTriggerSegment(id); if (s) { segmentsToEdit.push_back(s); } else { return ; } slotEditSegmentsEventList(segmentsToEdit); } void RosegardenGUIView::slotSegmentAutoSplit(Segment *segment) { AudioSplitDialog aSD(this, segment, getDocument()); if (aSD.exec() == QDialog::Accepted) { KCommand *command = new AudioSegmentAutoSplitCommand(getDocument(), segment, aSD.getThreshold()); slotAddCommandToHistory(command); } } void RosegardenGUIView::slotEditSegmentAudio(Segment *segment) { std::cout << "RosegardenGUIView::slotEditSegmentAudio() - " << "starting external audio editor" << std::endl; KConfig* config = kapp->config(); config->setGroup(GeneralOptionsConfigGroup); QString application = config->readEntry("externalaudioeditor", ""); if (application == "") { application = AudioConfigurationPage::getBestAvailableAudioEditor(); } QStringList splitCommand = QStringList::split(" ", application); if (splitCommand.size() == 0) { std::cerr << "RosegardenGUIView::slotEditSegmentAudio() - " << "external editor \"" << application.data() << "\" not found" << std::endl; KMessageBox::sorry(this, i18n("You've not yet defined an audio editor for Rosegarden to use.\nSee Settings -> Configure Rosegarden -> Audio.")); return ; } QFileInfo *appInfo = new QFileInfo(splitCommand[0]); if (appInfo->exists() == false || appInfo->isExecutable() == false) { std::cerr << "RosegardenGUIView::slotEditSegmentAudio() - " << "can't execute \"" << splitCommand[0] << "\"" << std::endl; return; } AudioFile *aF = getDocument()->getAudioFileManager(). getAudioFile(segment->getAudioFileId()); if (aF == 0) { std::cerr << "RosegardenGUIView::slotEditSegmentAudio() - " << "can't find audio file" << std::endl; return ; } // wait cursor QApplication::setOverrideCursor(QCursor(Qt::waitCursor)); // Prepare the process // KProcess *process = new KProcess(); (*process) << splitCommand; (*process) << QString(aF->getFilename().c_str()); // Start it // if (!process->start()) { std::cerr << "RosegardenGUIView::slotEditSegmentAudio() - " << "can't start external editor" << std::endl; } // restore cursor QApplication::restoreOverrideCursor(); } void RosegardenGUIView::setZoomSize(double size) { m_rulerScale->setUnitsPerPixel(size); double duration44 = TimeSignature(4, 4).getBarDuration(); double xScale = duration44 / (size * barWidth44); RG_DEBUG << "RosegardenGUIView::setZoomSize - xScale = " << xScale << endl; m_trackEditor->slotSetPointerPosition (getDocument()->getComposition().getPosition()); m_trackEditor->getSegmentCanvas()->clearSegmentRectsCache(true); m_trackEditor->getSegmentCanvas()->slotUpdateSize(); m_trackEditor->getSegmentCanvas()->slotUpdateSegmentsDrawBuffer(); if (m_trackEditor->getTempoRuler()) { m_trackEditor->getTempoRuler()->repaint(); } if (m_trackEditor->getChordNameRuler()) { m_trackEditor->getChordNameRuler()->repaint(); } if (m_trackEditor->getTopStandardRuler()) { m_trackEditor->getTopStandardRuler()->repaint(); } if (m_trackEditor->getBottomStandardRuler()) { m_trackEditor->getBottomStandardRuler()->repaint(); } } void RosegardenGUIView::slotSelectTrackSegments(int trackId) { // update the instrument parameter box Composition &comp = getDocument()->getComposition(); Track *track = comp.getTrackById(trackId); if (track == 0) return ; // Show the selection on the track buttons. Find the position. // m_trackEditor->getTrackButtons()->selectLabel(track->getPosition()); m_trackEditor->slotScrollToTrack(track->getPosition()); SegmentSelection segments; for (Composition::iterator i = getDocument()->getComposition().begin(); i != getDocument()->getComposition().end(); i++) { if (((int)(*i)->getTrack()) == trackId) segments.insert(*i); } // Store the selected Track in the Composition // comp.setSelectedTrack(trackId); m_trackParameterBox->slotSelectedTrackChanged(); slotUpdateInstrumentParameterBox(comp.getTrackById(trackId)-> getInstrument()); slotPropagateSegmentSelection(segments); // inform emit segmentsSelected(segments); emit compositionStateUpdate(); } void RosegardenGUIView::slotPropagateSegmentSelection(const SegmentSelection &segments) { // Send this signal to the GUI to activate the correct tool // on the toolbar so that we have a SegmentSelector object // to write the Segments into // if (!segments.empty()) { emit activateTool(SegmentSelector::ToolName); } // Send the segment list even if it's empty as we // use that to clear any current selection // m_trackEditor->getSegmentCanvas()->slotSelectSegments(segments); // update the segment parameter box m_segmentParameterBox->useSegments(segments); if (!segments.empty()) { emit stateChange("have_selection", true); if (!segments.hasNonAudioSegment()) { emit stateChange("audio_segment_selected", true); } } else { emit stateChange("have_selection", false); } } void RosegardenGUIView::slotSelectAllSegments() { SegmentSelection segments; InstrumentId instrument = 0; bool haveInstrument = false; bool multipleInstruments = false; Composition &comp = getDocument()->getComposition(); for (Composition::iterator i = comp.begin(); i != comp.end(); ++i) { InstrumentId myInstrument = comp.getTrackById((*i)->getTrack())->getInstrument(); if (haveInstrument) { if (myInstrument != instrument) { multipleInstruments = true; } } else { instrument = myInstrument; haveInstrument = true; } segments.insert(*i); } // Send this signal to the GUI to activate the correct tool // on the toolbar so that we have a SegmentSelector object // to write the Segments into // if (!segments.empty()) { emit activateTool(SegmentSelector::ToolName); } // Send the segment list even if it's empty as we // use that to clear any current selection // m_trackEditor->getSegmentCanvas()->slotSelectSegments(segments); // update the segment parameter box m_segmentParameterBox->useSegments(segments); // update the instrument parameter box if (haveInstrument && !multipleInstruments) { slotUpdateInstrumentParameterBox(instrument); } else { m_instrumentParameterBox->useInstrument(0); } //!!! similarly, how to set no selected track? //comp.setSelectedTrack(trackId); if (!segments.empty()) { emit stateChange("have_selection", true); if (!segments.hasNonAudioSegment()) { emit stateChange("audio_segment_selected", true); } } else { emit stateChange("have_selection", false); } // inform //!!! inform what? is this signal actually used? emit segmentsSelected(segments); } void RosegardenGUIView::slotUpdateInstrumentParameterBox(int id) { Studio &studio = getDocument()->getStudio(); Instrument *instrument = studio.getInstrumentById(id); Composition &comp = getDocument()->getComposition(); Track *track = comp.getTrackById(comp.getSelectedTrack()); // Reset the instrument // m_instrumentParameterBox->useInstrument(instrument); // Then do this instrument/track fiddling // /* if (track && instrument && instrument->getType() == Instrument::Audio) { // Set the mute status m_instrumentParameterBox->setMute(track->isMuted()); // Set the record track m_instrumentParameterBox->setRecord( track->getId() == comp.getRecordTrack()); // Set solo m_instrumentParameterBox->setSolo( comp.isSolo() && (track->getId() == comp.getSelectedTrack())); } */ emit checkTrackAssignments(); } void RosegardenGUIView::showVisuals(const MappedEvent *mE) { double valueLeft = ((double)mE->getData1()) / 127.0; double valueRight = ((double)mE->getData2()) / 127.0; if (mE->getType() == MappedEvent::AudioLevel) { // Send to the high sensitivity instrument parameter box // (if any) // if (m_instrumentParameterBox->getSelectedInstrument() && mE->getInstrument() == m_instrumentParameterBox->getSelectedInstrument()->getId()) { float dBleft = AudioLevel::fader_to_dB (mE->getData1(), 127, AudioLevel::LongFader); float dBright = AudioLevel::fader_to_dB (mE->getData2(), 127, AudioLevel::LongFader); m_instrumentParameterBox->setAudioMeter(dBleft, dBright, AudioLevel::DB_FLOOR, AudioLevel::DB_FLOOR); } // Don't always send all audio levels so we don't // get vu meter flickering on track meters // if (valueLeft < 0.05 && valueRight < 0.05) return ; } else if (mE->getType() != MappedEvent::MidiNote) return ; m_trackEditor->getTrackButtons()-> slotSetMetersByInstrument((valueLeft + valueRight) / 2, mE->getInstrument()); } void RosegardenGUIView::updateMeters(SequencerMapper *mapper) { const int unknownState = 0, oldState = 1, newState = 2; typedef std::map StateMap; static StateMap states; static StateMap recStates; typedef std::map LevelMap; static LevelMap levels; static LevelMap recLevels; for (StateMap::iterator i = states.begin(); i != states.end(); ++i) { i->second = unknownState; } for (StateMap::iterator i = recStates.begin(); i != recStates.end(); ++i) { i->second = unknownState; } for (Composition::trackcontainer::iterator i = getDocument()->getComposition().getTracks().begin(); i != getDocument()->getComposition().getTracks().end(); ++i) { Track *track = i->second; if (!track) continue; InstrumentId instrumentId = track->getInstrument(); if (states[instrumentId] == unknownState) { bool isNew = mapper->getInstrumentLevel(instrumentId, levels[instrumentId]); states[instrumentId] = (isNew ? newState : oldState); } if (recStates[instrumentId] == unknownState) { bool isNew = mapper->getInstrumentRecordLevel(instrumentId, recLevels[instrumentId]); recStates[instrumentId] = (isNew ? newState : oldState); } if (states[instrumentId] == oldState && recStates[instrumentId] == oldState) continue; Instrument *instrument = getDocument()->getStudio().getInstrumentById(instrumentId); if (!instrument) continue; // This records the level of this instrument, not neccessarily // caused by notes on this particular track. LevelInfo &info = levels[instrumentId]; LevelInfo &recInfo = recLevels[instrumentId]; if (instrument->getType() == Instrument::Audio || instrument->getType() == Instrument::SoftSynth) { float dBleft = AudioLevel::DB_FLOOR; float dBright = AudioLevel::DB_FLOOR; float recDBleft = AudioLevel::DB_FLOOR; float recDBright = AudioLevel::DB_FLOOR; bool toSet = false; if (states[instrumentId] == newState && (getDocument()->getSequenceManager()->getTransportStatus() != STOPPED)) { if (info.level != 0 || info.levelRight != 0) { dBleft = AudioLevel::fader_to_dB (info.level, 127, AudioLevel::LongFader); dBright = AudioLevel::fader_to_dB (info.levelRight, 127, AudioLevel::LongFader); } toSet = true; m_trackEditor->getTrackButtons()->slotSetTrackMeter ((info.level + info.levelRight) / 254.0, track->getPosition()); } if (recStates[instrumentId] == newState && instrument->getType() == Instrument::Audio && (getDocument()->getSequenceManager()->getTransportStatus() != PLAYING)) { if (recInfo.level != 0 || recInfo.levelRight != 0) { recDBleft = AudioLevel::fader_to_dB (recInfo.level, 127, AudioLevel::LongFader); recDBright = AudioLevel::fader_to_dB (recInfo.levelRight, 127, AudioLevel::LongFader); } toSet = true; } if (toSet && m_instrumentParameterBox->getSelectedInstrument() && instrument->getId() == m_instrumentParameterBox->getSelectedInstrument()->getId()) { m_instrumentParameterBox->setAudioMeter(dBleft, dBright, recDBleft, recDBright); } } else { // Not audio or softsynth if (info.level == 0) continue; if (getDocument()->getSequenceManager()->getTransportStatus() != STOPPED) { // The information in 'info' is specific for this instrument, not // for this track. //m_trackEditor->getTrackButtons()->slotSetTrackMeter // (info.level / 127.0, track->getPosition()); m_trackEditor->getTrackButtons()->slotSetMetersByInstrument (info.level / 127.0, instrumentId); } } } for (StateMap::iterator i = states.begin(); i != states.end(); ++i) { if (i->second == newState) { emit instrumentLevelsChanged(i->first, levels[i->first]); } } } void RosegardenGUIView::updateMonitorMeters(SequencerMapper *mapper) { Instrument *instrument = m_instrumentParameterBox->getSelectedInstrument(); if (!instrument || (instrument->getType() != Instrument::Audio)) return ; LevelInfo level; if (!mapper->getInstrumentRecordLevel(instrument->getId(), level)) return ; float dBleft = AudioLevel::fader_to_dB (level.level, 127, AudioLevel::LongFader); float dBright = AudioLevel::fader_to_dB (level.levelRight, 127, AudioLevel::LongFader); m_instrumentParameterBox->setAudioMeter (AudioLevel::DB_FLOOR, AudioLevel::DB_FLOOR, dBleft, dBright); } void RosegardenGUIView::slotSelectedSegments(const SegmentSelection &segments) { // update the segment parameter box m_segmentParameterBox->useSegments(segments); if (!segments.empty()) { emit stateChange("have_selection", true); if (!segments.hasNonAudioSegment()) emit stateChange("audio_segment_selected", true); } else { emit stateChange("have_selection", false); } emit segmentsSelected(segments); } void RosegardenGUIView::slotShowRulers(bool v) { if (v) { m_trackEditor->getTopStandardRuler()->getLoopRuler()->show(); m_trackEditor->getBottomStandardRuler()->getLoopRuler()->show(); } else { m_trackEditor->getTopStandardRuler()->getLoopRuler()->hide(); m_trackEditor->getBottomStandardRuler()->getLoopRuler()->hide(); } } void RosegardenGUIView::slotShowTempoRuler(bool v) { if (v) { m_trackEditor->getTempoRuler()->show(); } else { m_trackEditor->getTempoRuler()->hide(); } } void RosegardenGUIView::slotShowChordNameRuler(bool v) { if (v) { m_trackEditor->getChordNameRuler()->setStudio(&getDocument()->getStudio()); m_trackEditor->getChordNameRuler()->show(); } else { m_trackEditor->getChordNameRuler()->hide(); } } void RosegardenGUIView::slotShowPreviews(bool v) { m_trackEditor->getSegmentCanvas()->setShowPreviews(v); m_trackEditor->getSegmentCanvas()->slotUpdateSegmentsDrawBuffer(); } void RosegardenGUIView::slotShowSegmentLabels(bool v) { m_trackEditor->getSegmentCanvas()->setShowSegmentLabels(v); m_trackEditor->getSegmentCanvas()->slotUpdateSegmentsDrawBuffer(); } void RosegardenGUIView::slotAddTracks(unsigned int nbTracks, InstrumentId id, int pos) { RG_DEBUG << "RosegardenGUIView::slotAddTracks(" << nbTracks << ", " << pos << ")" << endl; m_trackEditor->slotAddTracks(nbTracks, id, pos); } void RosegardenGUIView::slotDeleteTracks( std::vector tracks) { RG_DEBUG << "RosegardenGUIView::slotDeleteTracks - " << "deleting " << tracks.size() << " tracks" << endl; m_trackEditor->slotDeleteTracks(tracks); } MultiViewCommandHistory* RosegardenGUIView::getCommandHistory() { return getDocument()->getCommandHistory(); } void RosegardenGUIView::slotAddCommandToHistory(KCommand *command) { getCommandHistory()->addCommand(command); } void RosegardenGUIView::slotChangeInstrumentLabel(InstrumentId id, QString label) { m_trackEditor->getTrackButtons()->changeInstrumentLabel(id, label); } void RosegardenGUIView::slotChangeTrackLabel(TrackId id, QString label) { m_trackEditor->getTrackButtons()->changeTrackLabel(id, label); } void RosegardenGUIView::slotAddAudioSegment(AudioFileId audioId, TrackId trackId, timeT position, const RealTime &startTime, const RealTime &endTime) { AudioSegmentInsertCommand *command = new AudioSegmentInsertCommand(getDocument(), trackId, position, audioId, startTime, endTime); slotAddCommandToHistory(command); Segment *newSegment = command->getNewSegment(); if (newSegment) { SegmentSelection selection; selection.insert(newSegment); slotPropagateSegmentSelection(selection); emit segmentsSelected(selection); } } void RosegardenGUIView::slotAddAudioSegmentCurrentPosition(AudioFileId audioFileId, const RealTime &startTime, const RealTime &endTime) { Composition &comp = getDocument()->getComposition(); AudioSegmentInsertCommand *command = new AudioSegmentInsertCommand(getDocument(), comp.getSelectedTrack(), comp.getPosition(), audioFileId, startTime, endTime); slotAddCommandToHistory(command); Segment *newSegment = command->getNewSegment(); if (newSegment) { SegmentSelection selection; selection.insert(newSegment); slotPropagateSegmentSelection(selection); emit segmentsSelected(selection); } } void RosegardenGUIView::slotAddAudioSegmentDefaultPosition(AudioFileId audioFileId, const RealTime &startTime, const RealTime &endTime) { // Add at current track if it's an audio track, otherwise at first // empty audio track if there is one, otherwise at first audio track. // This behaviour should be of no interest to proficient users (who // should have selected the right track already, or be using drag- // and-drop) but it should save beginners from inserting an audio // segment and being quite unable to work out why it won't play Composition &comp = getDocument()->getComposition(); Studio &studio = getDocument()->getStudio(); TrackId currentTrackId = comp.getSelectedTrack(); Track *track = comp.getTrackById(currentTrackId); if (track) { InstrumentId ii = track->getInstrument(); Instrument *instrument = studio.getInstrumentById(ii); if (instrument && instrument->getType() == Instrument::Audio) { slotAddAudioSegment(audioFileId, currentTrackId, comp.getPosition(), startTime, endTime); return ; } } // current track is not an audio track, find a more suitable one TrackId bestSoFar = currentTrackId; for (Composition::trackcontainer::const_iterator ti = comp.getTracks().begin(); ti != comp.getTracks().end(); ++ti) { InstrumentId ii = ti->second->getInstrument(); Instrument *instrument = studio.getInstrumentById(ii); if (instrument && instrument->getType() == Instrument::Audio) { if (bestSoFar == currentTrackId) bestSoFar = ti->first; bool haveSegment = false; for (Composition::segmentcontainer::const_iterator si = comp.getSegments().begin(); si != comp.getSegments().end(); ++si) { if ((*si)->getTrack() == ti->first) { // there's a segment on this track haveSegment = true; break; } } if (!haveSegment) { // perfect slotAddAudioSegment(audioFileId, ti->first, comp.getPosition(), startTime, endTime); return ; } } } slotAddAudioSegment(audioFileId, bestSoFar, comp.getPosition(), startTime, endTime); return ; } void RosegardenGUIView::slotDroppedNewAudio(QString audioDesc) { QTextIStream s(&audioDesc); QString url; int trackId; timeT time; url = s.readLine(); s >> trackId; s >> time; std::cerr << "RosegardenGUIView::slotDroppedNewAudio: url " << url << ", trackId " << trackId << ", time " << time << std::endl; RosegardenGUIApp *app = RosegardenGUIApp::self(); AudioFileManager &aFM = getDocument()->getAudioFileManager(); AudioFileId audioFileId = 0; int sampleRate = 0; if (getDocument()->getSequenceManager()) { sampleRate = getDocument()->getSequenceManager()->getSampleRate(); } KURL kurl(url); if (!kurl.isLocalFile()) { if (!RosegardenGUIApp::self()->testAudioPath("importing a remote audio file")) return; } else if (aFM.fileNeedsConversion(qstrtostr(kurl.path()), sampleRate)) { if (!RosegardenGUIApp::self()->testAudioPath("importing an audio file that needs to be converted or resampled")) return; } ProgressDialog progressDlg(i18n("Adding audio file..."), 100, this); CurrentProgressDialog::set(&progressDlg); progressDlg.progressBar()->hide(); progressDlg.show(); // Connect the progress dialog // connect(&aFM, SIGNAL(setProgress(int)), progressDlg.progressBar(), SLOT(setValue(int))); connect(&aFM, SIGNAL(setOperationName(QString)), &progressDlg, SLOT(slotSetOperationName(QString))); connect(&progressDlg, SIGNAL(cancelClicked()), &aFM, SLOT(slotStopImport())); try { audioFileId = aFM.importURL(kurl, sampleRate); } catch (AudioFileManager::BadAudioPathException e) { CurrentProgressDialog::freeze(); QString errorString = i18n("Can't add dropped file. ") + strtoqstr(e.getMessage()); KMessageBox::sorry(this, errorString); return ; } catch (SoundFile::BadSoundFileException e) { CurrentProgressDialog::freeze(); QString errorString = i18n("Can't add dropped file. ") + strtoqstr(e.getMessage()); KMessageBox::sorry(this, errorString); return ; } disconnect(&progressDlg, SIGNAL(cancelClicked()), &aFM, SLOT(slotStopImport())); connect(&progressDlg, SIGNAL(cancelClicked()), &aFM, SLOT(slotStopPreview())); progressDlg.progressBar()->show(); progressDlg.slotSetOperationName(i18n("Generating audio preview...")); try { aFM.generatePreview(audioFileId); } catch (Exception e) { CurrentProgressDialog::freeze(); QString message = strtoqstr(e.getMessage()) + "\n\n" + i18n("Try copying this file to a directory where you have write permission and re-add it"); KMessageBox::information(this, message); //return false; } disconnect(&progressDlg, SIGNAL(cancelClicked()), &aFM, SLOT(slotStopPreview())); // add the file at the sequencer emit addAudioFile(audioFileId); // Now fetch file details // AudioFile *aF = aFM.getAudioFile(audioFileId); if (aF) { slotAddAudioSegment(audioFileId, trackId, time, RealTime(0, 0), aF->getLength()); RG_DEBUG << "RosegardenGUIView::slotDroppedNewAudio(" << "file = " << url << ", trackid = " << trackId << ", time = " << time << endl; } } void RosegardenGUIView::slotDroppedAudio(QString audioDesc) { QTextIStream s(&audioDesc); AudioFileId audioFileId; TrackId trackId; timeT position; RealTime startTime, endTime; // read the audio info s >> audioFileId; s >> trackId; s >> position; s >> startTime.sec; s >> startTime.nsec; s >> endTime.sec; s >> endTime.nsec; RG_DEBUG << "RosegardenGUIView::slotDroppedAudio(" //<< audioDesc << ") : audioFileId = " << audioFileId << " - trackId = " << trackId << " - position = " << position << " - startTime.sec = " << startTime.sec << " - startTime.nsec = " << startTime.nsec << " - endTime.sec = " << endTime.sec << " - endTime.nsec = " << endTime.nsec << endl; slotAddAudioSegment(audioFileId, trackId, position, startTime, endTime); } void RosegardenGUIView::slotSetMuteButton(TrackId track, bool value) { RG_DEBUG << "RosegardenGUIView::slotSetMuteButton - track id = " << track << ", value = " << value << endl; m_trackEditor->getTrackButtons()->setMuteButton(track, value); Track *trackObj = getDocument()-> getComposition().getTrackById(track); /* // to fix 739544 if (m_instrumentParameterBox->getSelectedInstrument() && m_instrumentParameterBox->getSelectedInstrument()->getId() == trackObj->getInstrument()) { m_instrumentParameterBox->setMute(value); } */ // set the value in the composition trackObj->setMuted(value); getDocument()->slotDocumentModified(); // set the modification flag } void RosegardenGUIView::slotSetMute(InstrumentId id, bool value) { RG_DEBUG << "RosegardenGUIView::slotSetMute - " << "id = " << id << ",value = " << value << endl; Composition &comp = getDocument()->getComposition(); Composition::trackcontainer &tracks = comp.getTracks(); Composition::trackiterator it; for (it = tracks.begin(); it != tracks.end(); ++it) { if ((*it).second->getInstrument() == id) slotSetMuteButton((*it).second->getId(), value); } } void RosegardenGUIView::slotSetRecord(InstrumentId id, bool value) { RG_DEBUG << "RosegardenGUIView::slotSetRecord - " << "id = " << id << ",value = " << value << endl; /* // IPB // m_instrumentParameterBox->setRecord(value); */ Composition &comp = getDocument()->getComposition(); Composition::trackcontainer &tracks = comp.getTracks(); Composition::trackiterator it; #ifdef NOT_DEFINED for (it = tracks.begin(); it != tracks.end(); ++it) { if (comp.getSelectedTrack() == (*it).second->getId()) { //!!! MTR m_trackEditor->getTrackButtons()-> // setRecordTrack((*it).second->getPosition()); //!!! MTR is this needed? I think probably not slotUpdateInstrumentParameterBox((*it).second->getInstrument()); } } #endif Studio &studio = getDocument()->getStudio(); Instrument *instr = studio.getInstrumentById(id); } void RosegardenGUIView::slotSetSolo(InstrumentId id, bool value) { RG_DEBUG << "RosegardenGUIView::slotSetSolo - " << "id = " << id << ",value = " << value << endl; emit toggleSolo(value); } void RosegardenGUIView::slotUpdateRecordingSegment(Segment *segment, timeT ) { // We're only interested in this on the first call per recording segment, // when we possibly create a view for it static Segment *lastRecordingSegment = 0; if (segment == lastRecordingSegment) return ; lastRecordingSegment = segment; KConfig* config = kapp->config(); config->setGroup(GeneralOptionsConfigGroup); int tracking = config->readUnsignedNumEntry("recordtracking", 0); if (tracking != 1) return ; RG_DEBUG << "RosegardenGUIView::slotUpdateRecordingSegment: segment is " << segment << ", lastRecordingSegment is " << lastRecordingSegment << ", opening a new view" << endl; std::vector segments; segments.push_back(segment); NotationView *view = createNotationView(segments); if (!view) return ; /* signal no longer exists QObject::connect (getDocument(), SIGNAL(recordingSegmentUpdated(Segment *, timeT)), view, SLOT(slotUpdateRecordingSegment(Segment *, timeT))); */ view->show(); } void RosegardenGUIView::slotSynchroniseWithComposition() { // Track buttons // m_trackEditor->getTrackButtons()->slotSynchroniseWithComposition(); // Update all IPBs // Composition &comp = getDocument()->getComposition(); Track *track = comp.getTrackById(comp.getSelectedTrack()); slotUpdateInstrumentParameterBox(track->getInstrument()); m_instrumentParameterBox->slotUpdateAllBoxes(); } void RosegardenGUIView::windowActivationChange(bool) { if (isActiveWindow()) { slotActiveMainWindowChanged(this); } } void RosegardenGUIView::slotActiveMainWindowChanged(const QWidget *w) { m_lastActiveMainWindow = w; } void RosegardenGUIView::slotActiveMainWindowChanged() { const QWidget *w = dynamic_cast(sender()); if (w) slotActiveMainWindowChanged(w); } void RosegardenGUIView::slotControllerDeviceEventReceived(MappedEvent *e) { RG_DEBUG << "Controller device event received - send to " << (void *)m_lastActiveMainWindow << " (I am " << this << ")" << endl; //!!! So, what _should_ we do with these? // -- external controller that sends e.g. volume control for each // of a number of channels -> if mixer present, use control to adjust // tracks on mixer // -- external controller that sends e.g. separate controllers on // the same channel for adjusting various parameters -> if IPB // visible, adjust it. Should we use the channel to select the // track? maybe as an option // do we actually need the last active main window for either of // these? -- yes, to determine whether to send to mixer or to IPB // in the first place. Send to audio mixer if active, midi mixer // if active, plugin dialog if active, otherwise keep it for // ourselves for the IPB. But, we'll do that by having the edit // views pass it back to us. // -- then we need to send back out to device. //!!! special cases: controller 81 received by any window -> // select window 0->main, 1->audio mix, 2->midi mix //!!! controller 82 received by main window -> select track //!!! these obviously should be configurable if (e->getType() == MappedEvent::MidiController) { if (e->getData1() == 81) { // select window int window = e->getData2(); if (window < 10) { // me show(); raise(); setActiveWindow(); } else if (window < 20) { RosegardenGUIApp::self()->slotOpenAudioMixer(); } else if (window < 30) { RosegardenGUIApp::self()->slotOpenMidiMixer(); } } } emit controllerDeviceEventReceived(e, m_lastActiveMainWindow); } void RosegardenGUIView::slotControllerDeviceEventReceived(MappedEvent *e, const void *preferredCustomer) { if (preferredCustomer != this) return ; RG_DEBUG << "RosegardenGUIView::slotControllerDeviceEventReceived: this one's for me" << endl; raise(); RG_DEBUG << "Event is type: " << int(e->getType()) << ", channel " << int(e->getRecordedChannel()) << ", data1 " << int(e->getData1()) << ", data2 " << int(e->getData2()) << endl; Composition &comp = getDocument()->getComposition(); Studio &studio = getDocument()->getStudio(); TrackId currentTrackId = comp.getSelectedTrack(); Track *track = comp.getTrackById(currentTrackId); // If the event is a control change on channel n, then (if // follow-channel is on) switch to the nth track of the same type // as the current track -- or the first track of the given // channel?, and set the control appropriately. Any controls in // IPB are supported for a MIDI device plus program and bank; only // volume and pan are supported for audio/synth devices. //!!! complete this if (e->getType() != MappedEvent::MidiController) { if (e->getType() == MappedEvent::MidiProgramChange) { int program = e->getData1(); if (!track) return ; InstrumentId ii = track->getInstrument(); Instrument *instrument = studio.getInstrumentById(ii); if (!instrument) return ; instrument->setProgramChange(program); emit instrumentParametersChanged(ii); } return ; } unsigned int channel = e->getRecordedChannel(); MidiByte controller = e->getData1(); MidiByte value = e->getData2(); if (controller == 82) { //!!! magic select-track controller int tracks = comp.getNbTracks(); Track *track = comp.getTrackByPosition(value * tracks / 127); if (track) { slotSelectTrackSegments(track->getId()); } return ; } if (!track) return ; InstrumentId ii = track->getInstrument(); Instrument *instrument = studio.getInstrumentById(ii); if (!instrument) return ; switch (instrument->getType()) { case Instrument::Midi: { MidiDevice *md = dynamic_cast (instrument->getDevice()); if (!md) { std::cerr << "WARNING: MIDI instrument has no MIDI device in slotControllerDeviceEventReceived" << std::endl; return ; } //!!! we need a central clearing house for these changes, // for a proper mvc structure. reqd for automation post-1.2. // in the mean time this duplicates much of // MIDIInstrumentParameterPanel::slotControllerChanged etc switch (controller) { case MIDI_CONTROLLER_VOLUME: RG_DEBUG << "Setting volume for instrument " << instrument->getId() << " to " << value << endl; instrument->setVolume(value); break; case MIDI_CONTROLLER_PAN: RG_DEBUG << "Setting pan for instrument " << instrument->getId() << " to " << value << endl; instrument->setPan(value); break; default: { ControlList cl = md->getIPBControlParameters(); for (ControlList::const_iterator i = cl.begin(); i != cl.end(); ++i) { if ((*i).getControllerValue() == controller) { RG_DEBUG << "Setting controller " << controller << " for instrument " << instrument->getId() << " to " << value << endl; instrument->setControllerValue(controller, value); break; } } break; } } break; } case Instrument::SoftSynth: case Instrument::Audio: switch (controller) { case MIDI_CONTROLLER_VOLUME: RG_DEBUG << "Setting volume for instrument " << instrument->getId() << " to " << value << endl; instrument->setLevel(AudioLevel::fader_to_dB (value, 127, AudioLevel::ShortFader)); break; case MIDI_CONTROLLER_PAN: RG_DEBUG << "Setting pan for instrument " << instrument->getId() << " to " << value << endl; instrument->setPan(MidiByte((value / 64.0) * 100.0 + 0.01)); break; default: break; } break; } emit instrumentParametersChanged(instrument->getId()); //!!! send out updates via MIDI } void RosegardenGUIView::initChordNameRuler() { getTrackEditor()->getChordNameRuler()->setReady(); } EventView * RosegardenGUIView::createEventView(std::vector segmentsToEdit) { EventView *eventView = new EventView(getDocument(), segmentsToEdit, this); connect(eventView, SIGNAL(windowActivated()), this, SLOT(slotActiveMainWindowChanged())); connect(eventView, SIGNAL(selectTrack(int)), this, SLOT(slotSelectTrackSegments(int))); connect(eventView, SIGNAL(saveFile()), RosegardenGUIApp::self(), SLOT(slotFileSave())); connect(eventView, SIGNAL(openInNotation(std::vector)), this, SLOT(slotEditSegmentsNotation(std::vector))); connect(eventView, SIGNAL(openInMatrix(std::vector)), this, SLOT(slotEditSegmentsMatrix(std::vector))); connect(eventView, SIGNAL(openInPercussionMatrix(std::vector)), this, SLOT(slotEditSegmentsPercussionMatrix(std::vector))); connect(eventView, SIGNAL(openInEventList(std::vector)), this, SLOT(slotEditSegmentsEventList(std::vector))); connect(eventView, SIGNAL(editTriggerSegment(int)), this, SLOT(slotEditTriggerSegment(int))); connect(this, SIGNAL(compositionStateUpdate()), eventView, SLOT(slotCompositionStateUpdate())); connect(RosegardenGUIApp::self(), SIGNAL(compositionStateUpdate()), eventView, SLOT(slotCompositionStateUpdate())); connect(eventView, SIGNAL(toggleSolo(bool)), RosegardenGUIApp::self(), SLOT(slotToggleSolo(bool))); // create keyboard accelerators on view // RosegardenGUIApp *par = dynamic_cast(parent()); if (par) { par->plugAccelerators(eventView, eventView->getAccelerators()); } return eventView; } } #include "RosegardenGUIView.moc"