_editor->selection->clear_points ();
_editor->selection->clear_lines ();
}
+
+NoteCreateDrag::NoteCreateDrag (Editor* e, ArdourCanvas::Item* i, MidiRegionView* rv)
+ : Drag (e, i)
+ , _region_view (rv)
+ , _drag_rect (0)
+{
+
+}
+
+NoteCreateDrag::~NoteCreateDrag ()
+{
+ delete _drag_rect;
+}
+
+void
+NoteCreateDrag::start_grab (GdkEvent* event, Gdk::Cursor* cursor)
+{
+ _drag_rect = new ArdourCanvas::SimpleRect (*_region_view->get_canvas_group ());
+
+ framepos_t pf = _drags->current_pointer_frame ();
+
+ bool success;
+ Evoral::MusicalTime grid_beats = _editor->get_grid_type_as_beats (success, pf);
+ if (!success) {
+ grid_beats = 1;
+ }
+
+ framecnt_t grid_frames = _region_view->region_beats_to_region_frames (grid_beats);
+
+ /* Hack so that we always snap to the note that we are over, instead of snapping
+ to the next one if we're more than halfway through the one we're over.
+ */
+ if (_editor->snap_mode() == SnapNormal && pf > grid_frames / 2) {
+ pf -= grid_frames / 2;
+ }
+
+ _note[0] = adjusted_frame (pf, event) - _region_view->region()->position ();
+
+ MidiStreamView* sv = _region_view->midi_stream_view ();
+ double const x = _editor->frame_to_pixel (_note[0]);
+ double const y = sv->note_to_y (sv->y_to_note (y_to_region (event->button.y)));
+
+ _drag_rect->property_x1() = x;
+ _drag_rect->property_y1() = y;
+ _drag_rect->property_x2() = x;
+ _drag_rect->property_y2() = y + floor (_region_view->midi_stream_view()->note_height ());
+
+ _drag_rect->property_outline_what() = 0xff;
+ _drag_rect->property_outline_color_rgba() = 0xffffff99;
+ _drag_rect->property_fill_color_rgba() = 0xffffff66;
+}
+
+void
+NoteCreateDrag::motion (GdkEvent* event, bool)
+{
+ _note[1] = adjusted_current_frame (event) - _region_view->region()->position ();
+ double const x = _editor->frame_to_pixel (_note[1]);
+ if (_note[1] > _note[0]) {
+ _drag_rect->property_x2() = x;
+ } else {
+ _drag_rect->property_x1() = x;
+ }
+}
+
+void
+NoteCreateDrag::finished (GdkEvent* event, bool)
+{
+ if (_drag_rect->property_x2() < _drag_rect->property_x1() + 2) {
+ abort ();
+ }
+
+ framepos_t const start = min (_note[0], _note[1]);
+ framecnt_t const length = abs (_note[0] - _note[1]);
+
+ _region_view->create_note_at (start, _drag_rect->property_y1(), _region_view->region_frames_to_region_beats (length), true, false);
+}
+
+double
+NoteCreateDrag::y_to_region (double y) const
+{
+ double x = 0;
+ _region_view->get_canvas_group()->w2i (x, y);
+ return y;
+}
+
+void
+NoteCreateDrag::aborted (bool)
+{
+
+}
, _note_group(new ArdourCanvas::Group(*group))
, _note_diff_command (0)
, _ghost_note(0)
- , _drag_rect (0)
, _step_edit_cursor (0)
, _step_edit_cursor_width (1.0)
, _step_edit_cursor_position (0.0)
, _note_group(new ArdourCanvas::Group(*parent))
, _note_diff_command (0)
, _ghost_note(0)
- , _drag_rect (0)
, _step_edit_cursor (0)
, _step_edit_cursor_width (1.0)
, _step_edit_cursor_position (0.0)
, _note_group(new ArdourCanvas::Group(*get_canvas_group()))
, _note_diff_command (0)
, _ghost_note(0)
- , _drag_rect (0)
, _step_edit_cursor (0)
, _step_edit_cursor_width (1.0)
, _step_edit_cursor_position (0.0)
, _note_group(new ArdourCanvas::Group(*get_canvas_group()))
, _note_diff_command (0)
, _ghost_note(0)
- , _drag_rect (0)
, _step_edit_cursor (0)
, _step_edit_cursor_width (1.0)
, _step_edit_cursor_position (0.0)
_mouse_mode_connection, invalidator (*this), ui_bind (&MidiRegionView::mouse_mode_changed, this), gui_context ()
);
- if (trackview.editor().current_mouse_mode() == MouseRange) {
+ if (trackview.editor().current_mouse_mode() == MouseRange && _mouse_state != AddDragging) {
create_ghost_note (ev->x, ev->y);
}
return false;
}
- _last_x = ev->x;
- _last_y = ev->y;
-
- group->w2i (_last_x, _last_y);
-
if (_mouse_state != SelectTouchDragging) {
_pressed_button = ev->button;
beats = 1;
}
- create_note_at (event_x, event_y, beats, true, true);
+ create_note_at (editor.pixel_to_frame (event_x), event_y, beats, true, true);
}
break;
beats = 1;
}
- create_note_at (event_x, event_y, beats, true, true);
+ create_note_at (editor.pixel_to_frame (event_x), event_y, beats, true, true);
break;
}
_mouse_state = None;
break;
- case SelectRectDragging: // Select drag done
+ case SelectRectDragging:
+ case AddDragging:
editor.drags()->end_grab ((GdkEvent *) ev);
_mouse_state = None;
+ create_ghost_note (ev->x, ev->y);
break;
- case AddDragging: // Add drag done
-
- _mouse_state = None;
-
- if (Keyboard::is_insert_note_event(ev) || trackview.editor().current_mouse_mode() == MouseRange) {
-
- if (_drag_rect->property_x2() > _drag_rect->property_x1() + 2) {
-
- const double x = _drag_rect->property_x1();
- const double length = trackview.editor().pixel_to_frame (_drag_rect->property_x2() - _drag_rect->property_x1());
-
- create_note_at (x, _drag_rect->property_y1(), region_frames_to_region_beats(length), true, false);
- }
- }
-
- delete _drag_rect;
- _drag_rect = 0;
-
- create_ghost_note (ev->x, ev->y);
default:
break;
bool
MidiRegionView::motion (GdkEventMotion* ev)
{
- double event_x, event_y;
- framepos_t event_frame = 0;
-
- event_x = ev->x;
- event_y = ev->y;
- group->w2i(event_x, event_y);
-
PublicEditor& editor = trackview.editor ();
-
- // convert event_x to global frame
- framecnt_t grid_frames;
- event_frame = snap_frame_to_grid_underneath (editor.pixel_to_frame (event_x), grid_frames);
if (!_ghost_note && editor.current_mouse_mode() != MouseRange
&& Keyboard::modifier_state_contains (ev->state, Keyboard::insert_note_modifier())
update_ghost_note (ev->x, ev->y);
} else if (_ghost_note && editor.current_mouse_mode() != MouseRange) {
- delete _ghost_note;
- _ghost_note = 0;
+ remove_ghost_note ();
editor.verbose_cursor()->hide ();
} else if (_ghost_note && editor.current_mouse_mode() == MouseRange) {
}
switch (_mouse_state) {
- case Pressed: // Maybe start a drag, if we've moved a bit
-
- if (fabs (event_x - _last_x) < 1 && fabs (event_y - _last_y) < 1) {
- /* no appreciable movement since the button was pressed */
- return false;
- }
+ case Pressed:
if (_pressed_button == 1 && editor.current_mouse_mode() == MouseObject
&& !Keyboard::modifier_state_contains (ev->state, Keyboard::insert_note_modifier())) {
return true;
} else if (editor.internal_editing()) {
- // Add note drag start
-
- group->grab(GDK_POINTER_MOTION_MASK | GDK_BUTTON_RELEASE_MASK,
- Gdk::Cursor(Gdk::FLEUR), ev->time);
-
- _last_x = event_x;
- _last_y = event_y;
- _drag_start_x = event_x;
- _drag_start_y = event_y;
-
- _drag_rect = new ArdourCanvas::SimpleRect(*group);
- _drag_rect->property_x1() = editor.frame_to_pixel(event_frame);
-
- _drag_rect->property_y1() = midi_stream_view()->note_to_y(
- midi_stream_view()->y_to_note(event_y));
- _drag_rect->property_x2() = editor.frame_to_pixel(event_frame);
-
- _drag_rect->property_y2() = _drag_rect->property_y1()
- + floor(midi_stream_view()->note_height());
- _drag_rect->property_outline_what() = 0xFF;
- _drag_rect->property_outline_color_rgba() = 0xFFFFFF99;
- _drag_rect->property_fill_color_rgba() = 0xFFFFFF66;
+ editor.drags()->set (new NoteCreateDrag (dynamic_cast<Editor *> (&editor), group, this), (GdkEvent *) ev);
_mouse_state = AddDragging;
-
- delete _ghost_note;
- _ghost_note = 0;
+
+ remove_ghost_note ();
editor.verbose_cursor()->hide ();
return false;
case SelectRectDragging:
+ case AddDragging:
editor.drags()->motion_handler ((GdkEvent *) ev, false);
break;
- case AddDragging: // Add note drag motion
-
- if (ev->is_hint) {
- int t_x;
- int t_y;
- GdkModifierType state;
- gdk_window_get_pointer(ev->window, &t_x, &t_y, &state);
- event_x = t_x;
- event_y = t_y;
- }
-
- if (_mouse_state == AddDragging) {
- event_x = editor.frame_to_pixel(event_frame);
-
- if (editor.snap_mode() == SnapNormal) {
- /* event_frame will have been snapped to the start of the note we are under;
- it's more intuitive if we use the end of that note here
- */
- event_x = editor.frame_to_pixel (event_frame + grid_frames);
- } else {
- event_x = editor.frame_to_pixel (event_frame);
- }
-
- }
-
- if (_drag_rect) {
-
- if (event_x > _drag_start_x) {
- _drag_rect->property_x2() = event_x;
- }
- else {
- _drag_rect->property_x1() = event_x;
- }
- }
-
- _last_x = event_x;
- _last_y = event_y;
-
case SelectTouchDragging:
return false;
}
/** Add a note to the model, and the view, at a canvas (click) coordinate.
- * \param x horizontal position in pixels
+ * \param t time in frames relative to the position of the region
* \param y vertical position in pixels
* \param length duration of the note in beats, which will be snapped to the grid
* \param sh true to make the note 1 frame shorter than the snapped version of \a length.
* \param snap_x true to snap x to the grid, otherwise false.
*/
void
-MidiRegionView::create_note_at (double x, double y, double length, bool sh, bool snap_x)
+MidiRegionView::create_note_at (framepos_t t, double y, double length, bool sh, bool snap_x)
{
MidiTimeAxisView* const mtv = dynamic_cast<MidiTimeAxisView*>(&trackview);
MidiStreamView* const view = mtv->midi_view();
assert(note <= 127.0);
// Start of note in frames relative to region start
- framepos_t start_frames = trackview.editor().pixel_to_frame (x);
if (snap_x) {
framecnt_t grid_frames;
- start_frames = snap_frame_to_grid_underneath (start_frames, grid_frames);
+ t = snap_frame_to_grid_underneath (t, grid_frames);
}
- assert(start_frames >= 0);
+ assert (t >= 0);
// Snap length
length = region_frames_to_region_beats(
- snap_frame_to_frame (start_frames + region_beats_to_region_frames(length)) - start_frames);
+ snap_frame_to_frame (t + region_beats_to_region_frames(length)) - t
+ );
assert (length != 0);
}
const boost::shared_ptr<NoteType> new_note (new NoteType (mtv->get_channel_for_add (),
- region_frames_to_region_beats(start_frames + _region->start()), length,
+ region_frames_to_region_beats(t + _region->start()), length,
(uint8_t)note, 0x40));
if (_model->contains (new_note)) {
void
MidiRegionView::create_ghost_note (double x, double y)
{
- delete _ghost_note;
- _ghost_note = 0;
+ remove_ghost_note ();
boost::shared_ptr<NoteType> g (new NoteType);
_ghost_note = new NoEventCanvasNote (*this, *_note_group, g);