diff --git a/data/themes/default/glue.png b/data/themes/default/glue.png new file mode 100644 index 00000000000..de611b4bc34 Binary files /dev/null and b/data/themes/default/glue.png differ diff --git a/data/themes/default/tool.png b/data/themes/default/tool.png new file mode 100644 index 00000000000..6d818a53efc Binary files /dev/null and b/data/themes/default/tool.png differ diff --git a/include/PianoRoll.h b/include/PianoRoll.h index e3e4b312085..a876e03a013 100644 --- a/include/PianoRoll.h +++ b/include/PianoRoll.h @@ -208,6 +208,7 @@ protected slots: void selectRegionFromPixels( int xStart, int xEnd ); void clearGhostPattern(); + void glueNotes(); signals: diff --git a/src/gui/editors/PianoRoll.cpp b/src/gui/editors/PianoRoll.cpp index 8f1a827a95e..27198ce8b19 100644 --- a/src/gui/editors/PianoRoll.cpp +++ b/src/gui/editors/PianoRoll.cpp @@ -37,6 +37,7 @@ #include #include #include +#include #ifndef __USE_XOPEN #define __USE_XOPEN @@ -644,6 +645,73 @@ void PianoRoll::clearGhostPattern() } +void PianoRoll::glueNotes() +{ + if (hasValidPattern()) + { + NoteVector selectedNotes = getSelectedNotes(); + if (selectedNotes.empty()) + { + TextFloat::displayMessage( tr( "Glue notes failed" ), + tr( "Please select notes to glue first." ), + embed::getIconPixmap( "glue", 24, 24 ), + 3000 ); + return; + } + + // Make undo possible + m_pattern->addJournalCheckPoint(); + + // Sort notes on key and then pos. + std::sort(selectedNotes.begin(), selectedNotes.end(), + [](const Note * note, const Note * compareNote) -> bool + { + if (note->key() == compareNote->key()) + { + return note->pos() < compareNote->pos(); + } + return note->key() < compareNote->key(); + }); + + QList noteToRemove; + + NoteVector::iterator note = selectedNotes.begin(); + auto nextNote = note+1; + NoteVector::iterator end = selectedNotes.end(); + + while (note != end && nextNote != end) + { + // key and position match for glue. The notes are already + // sorted so we don't need to test that nextNote is the same + // position or next in sequence. + if ((*note)->key() == (*nextNote)->key() + && (*nextNote)->pos() <= (*note)->pos() + + qMax(MidiTime(0), (*note)->length())) + { + (*note)->setLength(qMax((*note)->length(), + MidiTime((*nextNote)->endPos() - (*note)->pos()))); + noteToRemove.push_back(*nextNote); + ++nextNote; + } + // key or position doesn't match + else + { + note = nextNote; + nextNote = note+1; + } + } + + // Remove old notes + for (int i = 0; i < noteToRemove.count(); ++i) + { + m_pattern->removeNote(noteToRemove[i]); + } + + update(); + } +} + + void PianoRoll::loadMarkedSemiTones(const QDomElement & de) { // clear marked semitones to prevent leftover marks @@ -4361,6 +4429,21 @@ PianoRollWindow::PianoRollWindow() : DropToolBar *timeLineToolBar = addDropToolBarToTop( tr( "Timeline controls" ) ); m_editor->m_timeLine->addToolButtons( timeLineToolBar ); + // -- Note modifier tools + // Toolbar + QToolButton * noteToolsButton = new QToolButton(m_toolBar); + noteToolsButton->setIcon(embed::getIconPixmap("tool")); + noteToolsButton->setPopupMode(QToolButton::InstantPopup); + + // Glue + QAction * glueAction = new QAction(embed::getIconPixmap("glue"), + tr("Glue"), noteToolsButton); + connect(glueAction, SIGNAL(triggered()), m_editor, SLOT(glueNotes())); + glueAction->setShortcut( Qt::SHIFT | Qt::Key_G ); + + noteToolsButton->addAction(glueAction); + + notesActionsToolBar->addWidget(noteToolsButton); addToolBarBreak();