2nd fix for hot spots for fade in/out cursors
[ardour.git] / gtk2_ardour / editor.cc
index e8c39faade70e5c0c8525d91228be054923d9c8d..74006c66f15d938b0868e9fb073a91e5ff17a91c 100644 (file)
@@ -157,6 +157,7 @@ static const gchar *_snap_type_strings[] = {
        N_("Beats/5"),
        N_("Beats/4"),
        N_("Beats/3"),
+       N_("Beats/2"),
        N_("Beats"),
        N_("Bars"),
        N_("Marks"),
@@ -208,6 +209,10 @@ static const gchar *_rb_opt_strings[] = {
 Gdk::Cursor* Editor::cross_hair_cursor = 0;
 Gdk::Cursor* Editor::selector_cursor = 0;
 Gdk::Cursor* Editor::trimmer_cursor = 0;
+Gdk::Cursor* Editor::left_side_trim_cursor = 0;
+Gdk::Cursor* Editor::right_side_trim_cursor = 0;
+Gdk::Cursor* Editor::fade_in_cursor = 0;
+Gdk::Cursor* Editor::fade_out_cursor = 0;
 Gdk::Cursor* Editor::grabber_cursor = 0;
 Gdk::Cursor* Editor::grabber_edit_point_cursor = 0;
 Gdk::Cursor* Editor::zoom_cursor = 0;
@@ -398,6 +403,8 @@ Editor::Editor ()
 
        frames_per_unit = 2048; /* too early to use reset_zoom () */
 
+       _scroll_callbacks = 0;
+
        zoom_focus = ZoomFocusLeft;
        set_zoom_focus (ZoomFocusLeft);
        zoom_range_clock.ValueChanged.connect (sigc::mem_fun(*this, &Editor::zoom_adjustment_changed));
@@ -573,27 +580,32 @@ Editor::Editor ()
        the_notebook.show_all ();
        
        post_maximal_editor_width = 0;
-       post_maximal_pane_position = 0;
+       post_maximal_horizontal_pane_position = 0;
+       post_maximal_editor_height = 0;
+       post_maximal_vertical_pane_position = 0;
 
-       VPaned *editor_summary_pane = manage(new VPaned());
-       editor_summary_pane->pack1(edit_packer);
+       editor_summary_pane.pack1(edit_packer);
 
        Button* summary_arrows_left_left = manage (new Button);
        summary_arrows_left_left->add (*manage (new Arrow (ARROW_LEFT, SHADOW_NONE)));
-       summary_arrows_left_left->signal_clicked().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_left));
+       summary_arrows_left_left->signal_pressed().connect (sigc::hide_return (sigc::mem_fun (*this, &Editor::horizontal_scroll_left_press)));
+       summary_arrows_left_left->signal_released().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_left_release));
        Button* summary_arrows_left_right = manage (new Button);
        summary_arrows_left_right->add (*manage (new Arrow (ARROW_RIGHT, SHADOW_NONE)));
-       summary_arrows_left_right->signal_clicked().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_right));
+       summary_arrows_left_right->signal_pressed().connect (sigc::hide_return (sigc::mem_fun (*this, &Editor::horizontal_scroll_right_press)));
+       summary_arrows_left_right->signal_released().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_right_release));
        VBox* summary_arrows_left = manage (new VBox);
        summary_arrows_left->pack_start (*summary_arrows_left_left);
        summary_arrows_left->pack_start (*summary_arrows_left_right);
 
        Button* summary_arrows_right_left = manage (new Button);
        summary_arrows_right_left->add (*manage (new Arrow (ARROW_LEFT, SHADOW_NONE)));
-       summary_arrows_right_left->signal_clicked().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_left));
+       summary_arrows_right_left->signal_pressed().connect (sigc::hide_return (sigc::mem_fun (*this, &Editor::horizontal_scroll_left_press)));
+       summary_arrows_right_left->signal_released().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_left_release));
        Button* summary_arrows_right_right = manage (new Button);
        summary_arrows_right_right->add (*manage (new Arrow (ARROW_RIGHT, SHADOW_NONE)));
-       summary_arrows_right_right->signal_clicked().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_right));
+       summary_arrows_right_right->signal_pressed().connect (sigc::hide_return (sigc::mem_fun (*this, &Editor::horizontal_scroll_right_press)));
+       summary_arrows_right_right->signal_released().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_right_release));
        VBox* summary_arrows_right = manage (new VBox);
        summary_arrows_right->pack_start (*summary_arrows_right_left);
        summary_arrows_right->pack_start (*summary_arrows_right_right);
@@ -607,11 +619,15 @@ Editor::Editor ()
        _summary_hbox.pack_start (*summary_frame, true, true);
        _summary_hbox.pack_start (*summary_arrows_right, false, false);
        
-       editor_summary_pane->pack2 (_summary_hbox);
+       editor_summary_pane.pack2 (_summary_hbox);
 
-       edit_pane.pack1 (*editor_summary_pane, true, true);
+       edit_pane.pack1 (editor_summary_pane, true, true);
        edit_pane.pack2 (the_notebook, false, true);
 
+       editor_summary_pane.signal_size_allocate().connect (sigc::bind (sigc::mem_fun (*this, &Editor::pane_allocation_handler), static_cast<Paned*> (&editor_summary_pane)));
+
+       /* XXX: editor_summary_pane might need similar special OS X treatment to the edit_pane */
+
        edit_pane.signal_size_allocate().connect (sigc::bind (sigc::mem_fun(*this, &Editor::pane_allocation_handler), static_cast<Paned*> (&edit_pane)));
 #ifdef GTKOSX
         Glib::PropertyProxy<int> proxy = edit_pane.property_position();
@@ -1228,6 +1244,27 @@ Editor::build_cursors ()
 
        cross_hair_cursor = new Gdk::Cursor (CROSSHAIR);
        trimmer_cursor =  new Gdk::Cursor (SB_H_DOUBLE_ARROW);
+
+       {
+               Glib::RefPtr<Gdk::Pixbuf> apixbuf (::get_icon ("trim_left_cursor"));
+               left_side_trim_cursor = new Gdk::Cursor (Gdk::Display::get_default(), apixbuf, 5, 11);
+       }
+
+       {
+               Glib::RefPtr<Gdk::Pixbuf> apixbuf (::get_icon ("trim_right_cursor"));
+               right_side_trim_cursor = new Gdk::Cursor (Gdk::Display::get_default(), apixbuf, 23, 11);
+       }
+
+       {
+               Glib::RefPtr<Gdk::Pixbuf> apixbuf (::get_icon ("fade_in_cursor"));
+               fade_in_cursor = new Gdk::Cursor (Gdk::Display::get_default(), apixbuf, 0, 40);
+       }
+
+       {
+               Glib::RefPtr<Gdk::Pixbuf> apixbuf (::get_icon ("fade_out_cursor"));
+               fade_out_cursor = new Gdk::Cursor (Gdk::Display::get_default(), apixbuf, 27, 40);
+       }
+
        selector_cursor = new Gdk::Cursor (XTERM);
        time_fx_cursor = new Gdk::Cursor (SIZING);
        wait_cursor = new Gdk::Cursor  (WATCH);
@@ -1322,6 +1359,8 @@ Editor::popup_track_context_menu (int button, int32_t time, ItemType item_type,
        case RegionItem:
        case RegionViewName:
        case RegionViewNameHighlight:
+       case LeftFrameHandle:
+       case RightFrameHandle:
                if (with_selection) {
                        build_menu_function = &Editor::build_track_selection_context_menu;
                } else {
@@ -1363,6 +1402,8 @@ Editor::popup_track_context_menu (int button, int32_t time, ItemType item_type,
        case RegionItem:
        case RegionViewName:
        case RegionViewNameHighlight:
+        case LeftFrameHandle:
+        case RightFrameHandle:
                if (!with_selection) {
                        if (region_edit_menu_split_item) {
                                if (clicked_regionview && clicked_regionview->region()->covers (get_preferred_edit_position())) {
@@ -1669,11 +1710,10 @@ Editor::add_region_context_items (StreamView* sv, boost::shared_ptr<Region> regi
                        sigc::bind (sigc::mem_fun(*this, &Editor::set_selected_regionview_from_map_event), sv, boost::weak_ptr<Region>(region)));
 
                items.push_back (MenuElem (_("Rename..."), sigc::mem_fun(*this, &Editor::rename_region)));
-               if (mr && internal_editing()) {
-                       items.push_back (MenuElem (_("List editor..."), sigc::mem_fun(*this, &Editor::show_midi_list_editor)));
-               } else {
-                       items.push_back (MenuElem (_("Region Properties..."), sigc::mem_fun(*this, &Editor::edit_region)));
+               if (mr) {
+                       items.push_back (MenuElem (_("List Editor..."), sigc::mem_fun(*this, &Editor::show_midi_list_editor)));
                }
+               items.push_back (MenuElem (_("Region Properties..."), sigc::mem_fun(*this, &Editor::edit_region)));
        }
 
        items.push_back (MenuElem (_("Raise to Top Layer"), sigc::mem_fun(*this, &Editor::raise_region_to_top)));
@@ -1925,7 +1965,7 @@ Editor::add_selection_context_items (Menu_Helpers::MenuList& edit_items)
        edit_items.push_back (MenuElem (_("Consolidate Range With Processing"), sigc::bind (sigc::mem_fun(*this, &Editor::bounce_range_selection), true, true)));
        edit_items.push_back (MenuElem (_("Bounce Range to Region List"), sigc::bind (sigc::mem_fun(*this, &Editor::bounce_range_selection), false, false)));
        edit_items.push_back (MenuElem (_("Bounce Range to Region List With Processing"), sigc::bind (sigc::mem_fun(*this, &Editor::bounce_range_selection), false, true)));
-       edit_items.push_back (MenuElem (_("Export Range"), sigc::mem_fun(*this, &Editor::export_range)));
+       edit_items.push_back (MenuElem (_("Export Range"), sigc::mem_fun(*this, &Editor::export_selection)));
 }
 
 
@@ -2112,6 +2152,7 @@ Editor::set_snap_to (SnapType st)
        case SnapToBeatDiv5:
        case SnapToBeatDiv4:
        case SnapToBeatDiv3:
+       case SnapToBeatDiv2:
                compute_bbt_ruler_scale (leftmost_frame, leftmost_frame + current_page_frames());
                update_tempo_based_rulers ();
                break;
@@ -2126,7 +2167,9 @@ Editor::set_snap_to (SnapType st)
        default:
                /* relax */
                break;
-    }
+       }
+
+       SnapChanged (); /* EMIT SIGNAL */
 }
 
 void
@@ -2403,10 +2446,6 @@ Editor::set_state (const XMLNode& node, int /*version*/)
                the_notebook.set_current_page (atoi (prop->value ()));
        }
 
-       if ((prop = node.property (X_("editor-pane-position")))) {
-               edit_pane.set_position (atoi (prop->value ()));
-       }
-
        return 0;
 }
 
@@ -2442,7 +2481,9 @@ Editor::get_state ()
                snprintf(buf, sizeof(buf), "%d", yoff);
                geometry->add_property("y-off", string(buf));
                snprintf(buf,sizeof(buf), "%d",gtk_paned_get_position (static_cast<Paned*>(&edit_pane)->gobj()));
-               geometry->add_property("edit_pane_pos", string(buf));
+               geometry->add_property("edit-horizontal-pane-pos", string(buf));
+               snprintf(buf,sizeof(buf), "%d",gtk_paned_get_position (static_cast<Paned*>(&editor_summary_pane)->gobj()));
+               geometry->add_property("edit-vertical-pane-pos", string(buf));
 
                node->add_child_nocopy (*geometry);
        }
@@ -2491,9 +2532,6 @@ Editor::get_state ()
        snprintf (buf, sizeof (buf), "%d", the_notebook.get_current_page ());
        node->add_property (X_("editor-list-page"), buf);
 
-       snprintf (buf, sizeof (buf), "%d", edit_pane.get_position ());
-       node->add_property (X_("editor-pane-position"), buf);
-
        return *node;
 }
 
@@ -2659,12 +2697,17 @@ Editor::snap_to_internal (nframes64_t& start, int32_t direction, bool for_mark)
                start = _session->tempo_map().round_to_beat (start, direction);
                break;
 
+       case SnapToBeatDiv32:
+               start = _session->tempo_map().round_to_beat_subdivision (start, 32, direction);
+               break;
        case SnapToBeatDiv28:
                start = _session->tempo_map().round_to_beat_subdivision (start, 28, direction);
                break;
-
        case SnapToBeatDiv24:
-               start = _session->tempo_map().round_to_beat_subdivision (start, 32, direction);
+               start = _session->tempo_map().round_to_beat_subdivision (start, 24, direction);
+               break;
+       case SnapToBeatDiv16:
+               start = _session->tempo_map().round_to_beat_subdivision (start, 16, direction);
                break;
        case SnapToBeatDiv14:
                start = _session->tempo_map().round_to_beat_subdivision (start, 14, direction);
@@ -2675,6 +2718,9 @@ Editor::snap_to_internal (nframes64_t& start, int32_t direction, bool for_mark)
        case SnapToBeatDiv10:
                start = _session->tempo_map().round_to_beat_subdivision (start, 10, direction);
                break;
+       case SnapToBeatDiv8:
+               start = _session->tempo_map().round_to_beat_subdivision (start, 8, direction);
+               break;
        case SnapToBeatDiv7:
                start = _session->tempo_map().round_to_beat_subdivision (start, 7, direction);
                break;
@@ -2684,21 +2730,15 @@ Editor::snap_to_internal (nframes64_t& start, int32_t direction, bool for_mark)
        case SnapToBeatDiv5:
                start = _session->tempo_map().round_to_beat_subdivision (start, 5, direction);
                break;
-       case SnapToBeatDiv32:
-               start = _session->tempo_map().round_to_beat_subdivision (start, 32, direction);
-               break;
-       case SnapToBeatDiv16:
-               start = _session->tempo_map().round_to_beat_subdivision (start, 16, direction);
-               break;
-       case SnapToBeatDiv8:
-               start = _session->tempo_map().round_to_beat_subdivision (start, 8, direction);
-               break;
        case SnapToBeatDiv4:
                start = _session->tempo_map().round_to_beat_subdivision (start, 4, direction);
                break;
        case SnapToBeatDiv3:
                start = _session->tempo_map().round_to_beat_subdivision (start, 3, direction);
                break;
+       case SnapToBeatDiv2:
+               start = _session->tempo_map().round_to_beat_subdivision (start, 2, direction);
+               break;
 
        case SnapToMark:
                if (for_mark) {
@@ -3005,7 +3045,6 @@ void
 Editor::setup_tooltips ()
 {
        ARDOUR_UI::instance()->set_tip (mouse_move_button, _("Select/Move Objects"));
-       ARDOUR_UI::instance()->set_tip (mouse_select_button, _("Select/Move Ranges"));
        ARDOUR_UI::instance()->set_tip (mouse_gain_button, _("Draw Gain Automation"));
        ARDOUR_UI::instance()->set_tip (mouse_zoom_button, _("Select Zoom Range"));
        ARDOUR_UI::instance()->set_tip (mouse_timefx_button, _("Stretch/Shrink Regions"));
@@ -3457,32 +3496,34 @@ Editor::snap_type_selection_done ()
        string choice = snap_type_selector.get_active_text();
        SnapType snaptype = SnapToBeat;
 
-       if (choice == _("Beats/3")) {
+       if (choice == _("Beats/2")) {
+               snaptype = SnapToBeatDiv2;
+       } else if (choice == _("Beats/3")) {
                snaptype = SnapToBeatDiv3;
        } else if (choice == _("Beats/4")) {
                snaptype = SnapToBeatDiv4;
-       } else if (choice == _("Beats/8")) {
-               snaptype = SnapToBeatDiv8;
-       } else if (choice == _("Beats/16")) {
-               snaptype = SnapToBeatDiv16;
-       } else if (choice == _("Beats/32")) {
-               snaptype = SnapToBeatDiv32;
        } else if (choice == _("Beats/5")) {
                snaptype = SnapToBeatDiv5;
        } else if (choice == _("Beats/6")) {
                snaptype = SnapToBeatDiv6;
        } else if (choice == _("Beats/7")) {
                snaptype = SnapToBeatDiv7;
+       } else if (choice == _("Beats/8")) {
+               snaptype = SnapToBeatDiv8;
        } else if (choice == _("Beats/10")) {
                snaptype = SnapToBeatDiv10;
        } else if (choice == _("Beats/12")) {
                snaptype = SnapToBeatDiv12;
        } else if (choice == _("Beats/14")) {
                snaptype = SnapToBeatDiv14;
+       } else if (choice == _("Beats/16")) {
+               snaptype = SnapToBeatDiv16;
        } else if (choice == _("Beats/24")) {
                snaptype = SnapToBeatDiv24;
        } else if (choice == _("Beats/28")) {
                snaptype = SnapToBeatDiv28;
+       } else if (choice == _("Beats/32")) {
+               snaptype = SnapToBeatDiv32;
        } else if (choice == _("Beats")) {
                snaptype = SnapToBeat;
        } else if (choice == _("Bars")) {
@@ -3663,7 +3704,14 @@ Editor::pane_allocation_handler (Allocation &alloc, Paned* which)
        char buf[32];
        XMLNode* node = ARDOUR_UI::instance()->editor_settings();
        int width, height;
-       static int32_t done;
+
+       enum Pane {
+               Horizontal = 0x1,
+               Vertical = 0x2
+       };
+
+       static Pane done;
+       
        XMLNode* geometry;
 
        width = default_width;
@@ -3671,15 +3719,11 @@ Editor::pane_allocation_handler (Allocation &alloc, Paned* which)
 
        if ((geometry = find_named_node (*node, "geometry")) != 0) {
 
-               if ((prop = geometry->property ("x_size")) == 0) {
-                       prop = geometry->property ("x-size");
-               }
+               prop = geometry->property ("x-size");
                if (prop) {
                        width = atoi (prop->value());
                }
-               if ((prop = geometry->property ("y_size")) == 0) {
-                       prop = geometry->property ("y-size");
-               }
+               prop = geometry->property ("y-size");
                if (prop) {
                        height = atoi (prop->value());
                }
@@ -3687,11 +3731,11 @@ Editor::pane_allocation_handler (Allocation &alloc, Paned* which)
 
        if (which == static_cast<Paned*> (&edit_pane)) {
 
-               if (done) {
+               if (done & Horizontal) {
                        return;
                }
 
-               if (!geometry || (prop = geometry->property ("edit-pane-pos")) == 0) {
+               if (!geometry || (prop = geometry->property ("edit-horizontal-pane-pos")) == 0) {
                        /* initial allocation is 90% to canvas, 10% to notebook */
                        pos = (int) floor (alloc.get_width() * 0.90f);
                        snprintf (buf, sizeof(buf), "%d", pos);
@@ -3699,10 +3743,33 @@ Editor::pane_allocation_handler (Allocation &alloc, Paned* which)
                        pos = atoi (prop->value());
                }
 
-               if ((done = GTK_WIDGET(edit_pane.gobj())->allocation.width > pos)) {
+               if (GTK_WIDGET(edit_pane.gobj())->allocation.width > pos) {
                        edit_pane.set_position (pos);
-                       pre_maximal_pane_position = pos;
+                       pre_maximal_horizontal_pane_position = pos;
                }
+
+               done = (Pane) (done | Horizontal);
+               
+       } else if (which == static_cast<Paned*> (&editor_summary_pane)) {
+
+               if (done & Vertical) {
+                       return;
+               }
+
+               if (!geometry || (prop = geometry->property ("edit-vertical-pane-pos")) == 0) {
+                       /* initial allocation is 90% to canvas, 10% to summary */
+                       pos = (int) floor (alloc.get_height() * 0.90f);
+                       snprintf (buf, sizeof(buf), "%d", pos);
+               } else {
+                       pos = atoi (prop->value());
+               }
+
+               if (GTK_WIDGET(editor_summary_pane.gobj())->allocation.height > pos) {
+                       editor_summary_pane.set_position (pos);
+                       pre_maximal_vertical_pane_position = pos;
+               }
+
+               done = (Pane) (done | Vertical);
        }
 }
 
@@ -3843,12 +3910,18 @@ Editor::get_grid_type_as_beats (bool& success, nframes64_t position)
                return 1.0;
                break;
 
+       case SnapToBeatDiv32:
+               return 1.0/32.0;
+               break;
        case SnapToBeatDiv28:
                return 1.0/28.0;
                break;
        case SnapToBeatDiv24:
                return 1.0/24.0;
                break;
+       case SnapToBeatDiv16:
+               return 1.0/16.0;
+               break;
        case SnapToBeatDiv14:
                return 1.0/14.0;
                break;
@@ -3858,6 +3931,9 @@ Editor::get_grid_type_as_beats (bool& success, nframes64_t position)
        case SnapToBeatDiv10:
                return 1.0/10.0;
                break;
+       case SnapToBeatDiv8:
+               return 1.0/8.0;
+               break;
        case SnapToBeatDiv7:
                return 1.0/7.0;
                break;
@@ -3867,26 +3943,15 @@ Editor::get_grid_type_as_beats (bool& success, nframes64_t position)
        case SnapToBeatDiv5:
                return 1.0/5.0;
                break;
-
-       case SnapToBeatDiv32:
-               return 1.0/32.0;
-               break;
-
-       case SnapToBeatDiv16:
-               return 1.0/16.0;
-               break;
-
-       case SnapToBeatDiv8:
-               return 1.0/8.0;
-               break;
-
        case SnapToBeatDiv4:
                return 1.0/4.0;
                break;
-
        case SnapToBeatDiv3:
                return 1.0/3.0;
                break;
+       case SnapToBeatDiv2:
+               return 1.0/2.0;
+               break;
 
        case SnapToBar:
                if (_session) {
@@ -4049,39 +4114,58 @@ Editor::maximise_editing_space ()
        _mouse_mode_tearoff->set_visible (false);
        _tools_tearoff->set_visible (false);
 
-       pre_maximal_pane_position = edit_pane.get_position();
-       pre_maximal_editor_width = this->get_width();
+       pre_maximal_horizontal_pane_position = edit_pane.get_position ();
+       pre_maximal_vertical_pane_position = editor_summary_pane.get_position ();
+       pre_maximal_editor_width = this->get_width ();
+       pre_maximal_editor_height = this->get_height ();
 
-       if(post_maximal_pane_position == 0) {
-               post_maximal_pane_position = edit_pane.get_width();
+       if (post_maximal_horizontal_pane_position == 0) {
+               post_maximal_horizontal_pane_position = edit_pane.get_width();
        }
 
-       fullscreen();
+       if (post_maximal_vertical_pane_position == 0) {
+               post_maximal_vertical_pane_position = editor_summary_pane.get_height();
+       }
+       
+       fullscreen ();
 
-       if(post_maximal_editor_width) {
-               edit_pane.set_position (post_maximal_pane_position -
+       if (post_maximal_editor_width) {
+               edit_pane.set_position (post_maximal_horizontal_pane_position -
                        abs(post_maximal_editor_width - pre_maximal_editor_width));
        } else {
-               edit_pane.set_position (post_maximal_pane_position);
+               edit_pane.set_position (post_maximal_horizontal_pane_position);
+       }
+
+       if (post_maximal_editor_height) {
+               editor_summary_pane.set_position (post_maximal_vertical_pane_position -
+                       abs(post_maximal_editor_height - pre_maximal_editor_height));
+       } else {
+               editor_summary_pane.set_position (post_maximal_vertical_pane_position);
        }
 }
 
 void
 Editor::restore_editing_space ()
 {
-       // user changed width of pane during fullscreen
+       // user changed width/height of panes during fullscreen
 
-       if(post_maximal_pane_position != edit_pane.get_position()) {
-               post_maximal_pane_position = edit_pane.get_position();
+       if (post_maximal_horizontal_pane_position != edit_pane.get_position()) {
+               post_maximal_horizontal_pane_position = edit_pane.get_position();
        }
 
+       if (post_maximal_vertical_pane_position != editor_summary_pane.get_position()) {
+               post_maximal_vertical_pane_position = editor_summary_pane.get_position();
+       }
+       
        unfullscreen();
 
        _mouse_mode_tearoff->set_visible (true);
        _tools_tearoff->set_visible (true);
        post_maximal_editor_width = this->get_width();
+       post_maximal_editor_height = this->get_height();
 
-       edit_pane.set_position (pre_maximal_pane_position + abs(this->get_width() - pre_maximal_editor_width));
+       edit_pane.set_position (pre_maximal_horizontal_pane_position + abs(this->get_width() - pre_maximal_editor_width));
+       editor_summary_pane.set_position (pre_maximal_vertical_pane_position + abs(this->get_height() - pre_maximal_editor_height));
 }
 
 /**
@@ -5100,21 +5184,63 @@ Editor::check_step_edit ()
        return true; // do it again, till we stop
 }
 
-void
-Editor::horizontal_scroll_left ()
+bool
+Editor::horizontal_scroll_left_press ()
 {
+       ++_scroll_callbacks;
+       
+       if (_scroll_connection.connected() && _scroll_callbacks < 5) {
+               /* delay the first auto-repeat */
+               return true;
+       }
+               
        double x = leftmost_position() - current_page_frames() / 5;
        if (x < 0) {
                x = 0;
        }
        
        reset_x_origin (x);
+
+       /* do hacky auto-repeat */
+       if (!_scroll_connection.connected ()) {
+               _scroll_connection = Glib::signal_timeout().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_left_press), 100);
+               _scroll_callbacks = 0;
+       }
+
+       return true;
 }
 
 void
-Editor::horizontal_scroll_right ()
+Editor::horizontal_scroll_left_release ()
 {
+       _scroll_connection.disconnect ();
+}
+
+bool
+Editor::horizontal_scroll_right_press ()
+{
+       ++_scroll_callbacks;
+       
+       if (_scroll_connection.connected() && _scroll_callbacks < 5) {
+               /* delay the first auto-repeat */
+               return true;
+       }
+
        reset_x_origin (leftmost_position() + current_page_frames() / 5);
+
+       /* do hacky auto-repeat */
+       if (!_scroll_connection.connected ()) {
+               _scroll_connection = Glib::signal_timeout().connect (sigc::mem_fun (*this, &Editor::horizontal_scroll_right_press), 100);
+               _scroll_callbacks = 0;
+       }
+
+       return true;
+}
+
+void
+Editor::horizontal_scroll_right_release ()
+{
+       _scroll_connection.disconnect ();
 }
 
 /** Queue a change for the Editor viewport x origin to follow the playhead */