mirror of
https://github.com/LMMS/lmms.git
synced 2026-01-22 21:38:12 -05:00
(Addresses #1421) **Behaviour description:** * Toggle step-recording mode using the dedicated icon. * This mode is mutually exclusive with other recoding modes (record/record accompany). * Step-Recording while song is playing is allowed (and fun! :) ). * When start recording, the start recording-position will be set where the timeline curser points (quantized backwards using PianoRoll's current quantization). If step-recording is started while the pattern is playing the start recording-position is set to the beginning of the pattern. * Step length is determined by the Piano Roll's current note-length (can be changed dynamically during step-recording). * The record-position can be moved forward/backward using the right/left keys. * When notes are pressed on keyboard/midi-device, they will be added temporarily ("recorded") with a length of a step. while still pressed, user can adjust the length by steps resolution using the arrow keys (e.g. moving right once will make the note's length 2-steps, another right press will make the length 3-steps etc.). * When all pressed-keys are released, the actual recording happen and the notes are added. * If the user press multiple notes, and release some of them for some time which indicates it is intentional i.e. he didn't want to do a full release to record the step but rather just change what will be recorded (I set the "intentional release threshold" to 70 milliseconds) - these note will be removed from current step-recording. e.g. * Added notes are not quantized, making the addition simpler and WYSIWYG * Similiarly to adding notes using mouse clicks, an undo-checkpoint is added per added step and not for the whole recording as in other record modes.
143 lines
3.1 KiB
C++
143 lines
3.1 KiB
C++
/*
|
|
* This file is part of LMMS - https://lmms.io
|
|
*
|
|
* 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.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public
|
|
* License along with this program (see COPYING); if not, write to the
|
|
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301 USA.
|
|
*
|
|
*/
|
|
|
|
#ifndef STEP_RECORDER_H
|
|
#define STEP_RECORDER_H
|
|
|
|
#include <QTime>
|
|
#include <QTimer>
|
|
#include <QObject>
|
|
#include <QKeyEvent>
|
|
|
|
#include "Note.h"
|
|
#include "lmms_basics.h"
|
|
#include "Pattern.h"
|
|
|
|
class PianoRoll;
|
|
class StepRecorderWidget;
|
|
|
|
class StepRecorder : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
StepRecorder(PianoRoll& pianoRoll, StepRecorderWidget& stepRecorderWidget);
|
|
|
|
void initialize();
|
|
void start(const MidiTime& currentPosition,const MidiTime& stepLength);
|
|
void stop();
|
|
void notePressed(const Note & n);
|
|
void noteReleased(const Note & n);
|
|
bool keyPressEvent(QKeyEvent* ke);
|
|
bool mousePressEvent(QMouseEvent* ke);
|
|
void setCurrentPattern(Pattern* newPattern);
|
|
void setStepsLength(const MidiTime& newLength);
|
|
|
|
QVector<Note*> getCurStepNotes();
|
|
|
|
bool isRecording() const
|
|
{
|
|
return m_isRecording;
|
|
}
|
|
|
|
QColor curStepNoteColor() const
|
|
{
|
|
return QColor(245,3,139); // radiant pink
|
|
}
|
|
|
|
private slots:
|
|
void removeNotesReleasedForTooLong();
|
|
|
|
private:
|
|
void stepForwards();
|
|
void stepBackwards();
|
|
|
|
void applyStep();
|
|
void dismissStep();
|
|
void prepareNewStep();
|
|
|
|
MidiTime getCurStepEndPos();
|
|
|
|
void updateCurStepNotes();
|
|
void updateWidget();
|
|
|
|
bool allCurStepNotesReleased();
|
|
|
|
PianoRoll& m_pianoRoll;
|
|
StepRecorderWidget& m_stepRecorderWidget;
|
|
|
|
bool m_isRecording = false;
|
|
MidiTime m_curStepStartPos = 0;
|
|
MidiTime m_curStepEndPos = 0;
|
|
|
|
MidiTime m_stepsLength;
|
|
MidiTime m_curStepLength; // current step length refers to the step currently recorded. it may defer from m_stepsLength
|
|
// since the user can make current step larger
|
|
|
|
QTimer m_updateReleasedTimer;
|
|
|
|
Pattern* m_pattern;
|
|
|
|
class StepNote
|
|
{
|
|
public:
|
|
StepNote(const Note & note) : m_note(note), m_pressed(true) {};
|
|
|
|
void setPressed()
|
|
{
|
|
m_pressed = true;
|
|
}
|
|
|
|
void setReleased()
|
|
{
|
|
m_pressed = false;
|
|
releasedTimer.start();
|
|
}
|
|
|
|
int timeSinceReleased()
|
|
{
|
|
return releasedTimer.elapsed();
|
|
}
|
|
|
|
bool isPressed() const
|
|
{
|
|
return m_pressed;
|
|
}
|
|
|
|
bool isReleased() const
|
|
{
|
|
return !m_pressed;
|
|
}
|
|
|
|
Note m_note;
|
|
|
|
private:
|
|
bool m_pressed;
|
|
QTime releasedTimer;
|
|
} ;
|
|
|
|
QVector<StepNote*> m_curStepNotes; // contains the current recorded step notes (i.e. while user still press the notes; before they are applied to the pattern)
|
|
|
|
StepNote* findCurStepNote(const int key);
|
|
|
|
bool m_isStepInProgress = false;
|
|
};
|
|
|
|
#endif //STEP_RECORDER_H
|