X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/d2d0adc7765dfe904f0f7e97d85a2638c934a7d6..76c67131022c84eb8c424adea124620da6b0d03e:/src/richtext/richtextctrl.cpp diff --git a/src/richtext/richtextctrl.cpp b/src/richtext/richtextctrl.cpp index c59cbfb851..e8eff5d801 100644 --- a/src/richtext/richtextctrl.cpp +++ b/src/richtext/richtextctrl.cpp @@ -1,5 +1,5 @@ ///////////////////////////////////////////////////////////////////////////// -// Name: src/richtext/richeditctrl.cpp +// Name: src/richtext/richtextctrl.cpp // Purpose: A rich edit control // Author: Julian Smart // Modified by: @@ -26,32 +26,135 @@ #include "wx/settings.h" #endif +#include "wx/timer.h" #include "wx/textfile.h" #include "wx/ffile.h" #include "wx/filename.h" #include "wx/dcbuffer.h" #include "wx/arrimpl.cpp" #include "wx/fontenum.h" +#include "wx/accel.h" + +#if defined (__WXGTK__) || defined(__WXX11__) || defined(__WXMOTIF__) +#define wxHAVE_PRIMARY_SELECTION 1 +#else +#define wxHAVE_PRIMARY_SELECTION 0 +#endif + +#if wxUSE_CLIPBOARD && wxHAVE_PRIMARY_SELECTION +#include "wx/clipbrd.h" +#endif // DLL options compatibility check: #include "wx/app.h" WX_CHECK_BUILD_OPTIONS("wxRichTextCtrl") -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_ITEM_SELECTED) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_ITEM_DESELECTED) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_LEFT_CLICK) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_MIDDLE_CLICK) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_RIGHT_CLICK) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_LEFT_DCLICK) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_RETURN) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_STYLESHEET_REPLACING) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_STYLESHEET_REPLACED) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_STYLESHEET_CHANGING) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_RICHTEXT_STYLESHEET_CHANGED) +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_LEFT_CLICK, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_MIDDLE_CLICK, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_RIGHT_CLICK, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_LEFT_DCLICK, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_RETURN, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_CHARACTER, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_DELETE, wxRichTextEvent ); + +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_STYLESHEET_REPLACING, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_STYLESHEET_REPLACED, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_STYLESHEET_CHANGING, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_STYLESHEET_CHANGED, wxRichTextEvent ); + +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_CONTENT_INSERTED, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_STYLE_CHANGED, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_PROPERTIES_CHANGED, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_SELECTION_CHANGED, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_BUFFER_RESET, wxRichTextEvent ); +wxDEFINE_EVENT( wxEVT_COMMAND_RICHTEXT_FOCUS_OBJECT_CHANGED, wxRichTextEvent ); + +#if wxRICHTEXT_USE_OWN_CARET + +/*! + * wxRichTextCaret + * + * This implements a non-flashing cursor in case there + * are platform-specific problems with the generic caret. + * wxRICHTEXT_USE_OWN_CARET is set in richtextbuffer.h. + */ + +class wxRichTextCaret; +class wxRichTextCaretTimer: public wxTimer +{ + public: + wxRichTextCaretTimer(wxRichTextCaret* caret) + { + m_caret = caret; + } + virtual void Notify(); + wxRichTextCaret* m_caret; +}; + +class wxRichTextCaret: public wxCaret +{ +public: + // ctors + // ----- + // default - use Create() + wxRichTextCaret(): m_timer(this) { Init(); } + // creates a block caret associated with the given window + wxRichTextCaret(wxRichTextCtrl *window, int width, int height) + : wxCaret(window, width, height), m_timer(this) { Init(); m_richTextCtrl = window; } + wxRichTextCaret(wxRichTextCtrl *window, const wxSize& size) + : wxCaret(window, size), m_timer(this) { Init(); m_richTextCtrl = window; } + + virtual ~wxRichTextCaret(); + + // implementation + // -------------- + + // called by wxWindow (not using the event tables) + virtual void OnSetFocus(); + virtual void OnKillFocus(); + + // draw the caret on the given DC + void DoDraw(wxDC *dc); + + // get the visible count + int GetVisibleCount() const { return m_countVisible; } + + // delay repositioning + bool GetNeedsUpdate() const { return m_needsUpdate; } + void SetNeedsUpdate(bool needsUpdate = true ) { m_needsUpdate = needsUpdate; } + + void Notify(); + + bool GetRefreshEnabled() const { return m_refreshEnabled; } + void EnableRefresh(bool b) { m_refreshEnabled = b; } + +protected: + virtual void DoShow(); + virtual void DoHide(); + virtual void DoMove(); + virtual void DoSize(); + + // refresh the caret + void Refresh(); + +private: + void Init(); + + int m_xOld, + m_yOld; + bool m_hasFocus; // true => our window has focus + bool m_needsUpdate; // must be repositioned + bool m_flashOn; + wxRichTextCaretTimer m_timer; + wxRichTextCtrl* m_richTextCtrl; + bool m_refreshEnabled; +}; +#endif -IMPLEMENT_CLASS( wxRichTextCtrl, wxControl ) +IMPLEMENT_DYNAMIC_CLASS( wxRichTextCtrl, wxControl ) -IMPLEMENT_CLASS( wxRichTextEvent, wxNotifyEvent ) +IMPLEMENT_DYNAMIC_CLASS( wxRichTextEvent, wxNotifyEvent ) BEGIN_EVENT_TABLE( wxRichTextCtrl, wxControl ) EVT_PAINT(wxRichTextCtrl::OnPaint) @@ -65,10 +168,13 @@ BEGIN_EVENT_TABLE( wxRichTextCtrl, wxControl ) EVT_MIDDLE_DOWN(wxRichTextCtrl::OnMiddleClick) EVT_LEFT_DCLICK(wxRichTextCtrl::OnLeftDClick) EVT_CHAR(wxRichTextCtrl::OnChar) + EVT_KEY_DOWN(wxRichTextCtrl::OnChar) EVT_SIZE(wxRichTextCtrl::OnSize) EVT_SET_FOCUS(wxRichTextCtrl::OnSetFocus) EVT_KILL_FOCUS(wxRichTextCtrl::OnKillFocus) + EVT_MOUSE_CAPTURE_LOST(wxRichTextCtrl::OnCaptureLost) EVT_CONTEXT_MENU(wxRichTextCtrl::OnContextMenu) + EVT_SYS_COLOUR_CHANGED(wxRichTextCtrl::OnSysColourChanged) EVT_MENU(wxID_UNDO, wxRichTextCtrl::OnUndo) EVT_UPDATE_UI(wxID_UNDO, wxRichTextCtrl::OnUpdateUndo) @@ -90,6 +196,16 @@ BEGIN_EVENT_TABLE( wxRichTextCtrl, wxControl ) EVT_MENU(wxID_SELECTALL, wxRichTextCtrl::OnSelectAll) EVT_UPDATE_UI(wxID_SELECTALL, wxRichTextCtrl::OnUpdateSelectAll) + + EVT_MENU(wxID_RICHTEXT_PROPERTIES1, wxRichTextCtrl::OnProperties) + EVT_UPDATE_UI(wxID_RICHTEXT_PROPERTIES1, wxRichTextCtrl::OnUpdateProperties) + + EVT_MENU(wxID_RICHTEXT_PROPERTIES2, wxRichTextCtrl::OnProperties) + EVT_UPDATE_UI(wxID_RICHTEXT_PROPERTIES2, wxRichTextCtrl::OnUpdateProperties) + + EVT_MENU(wxID_RICHTEXT_PROPERTIES3, wxRichTextCtrl::OnProperties) + EVT_UPDATE_UI(wxID_RICHTEXT_PROPERTIES3, wxRichTextCtrl::OnUpdateProperties) + END_EVENT_TABLE() /*! @@ -109,34 +225,41 @@ wxRichTextCtrl::wxRichTextCtrl(wxWindow* parent, const wxString& value, const wxPoint& pos, const wxSize& size, - long style) + long style, + const wxValidator& validator, + const wxString& name) : wxScrollHelper(this) { Init(); - Create(parent, id, value, pos, size, style); + Create(parent, id, value, pos, size, style, validator, name); } /// Creation -bool wxRichTextCtrl::Create( wxWindow* parent, wxWindowID id, const wxString& value, const wxPoint& pos, const wxSize& size, long style) +bool wxRichTextCtrl::Create( wxWindow* parent, wxWindowID id, const wxString& value, const wxPoint& pos, const wxSize& size, long style, + const wxValidator& validator, const wxString& name) { - if (!wxTextCtrlBase::Create(parent, id, pos, size, - style|wxFULL_REPAINT_ON_RESIZE)) + style |= wxVSCROLL; + + if (!wxControl::Create(parent, id, pos, size, + style|wxFULL_REPAINT_ON_RESIZE, + validator, name)) return false; - if (!GetFont().Ok()) + if (!GetFont().IsOk()) { SetFont(wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT)); } - GetBuffer().SetRichTextCtrl(this); + // No physical scrolling, so we can preserve margins + EnableScrolling(false, false); if (style & wxTE_READONLY) SetEditable(false); - wxTextAttrEx attributes; + // The base attributes must all have default values + wxRichTextAttr attributes; attributes.SetFont(GetFont()); - attributes.SetTextColour(*wxBLACK); - attributes.SetBackgroundColour(*wxWHITE); + attributes.SetTextColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT)); attributes.SetAlignment(wxTEXT_ALIGNMENT_LEFT); attributes.SetLineSpacing(10); attributes.SetParagraphSpacingAfter(10); @@ -144,16 +267,28 @@ bool wxRichTextCtrl::Create( wxWindow* parent, wxWindowID id, const wxString& va SetBasicStyle(attributes); + int margin = 5; + SetMargins(margin, margin); + // The default attributes will be merged with base attributes, so // can be empty to begin with - wxTextAttrEx defaultAttributes; + wxRichTextAttr defaultAttributes; SetDefaultStyle(defaultAttributes); - SetBackgroundColour(*wxWHITE); + SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); SetBackgroundStyle(wxBG_STYLE_CUSTOM); + GetBuffer().Reset(); + GetBuffer().SetRichTextCtrl(this); + +#if wxRICHTEXT_USE_OWN_CARET + SetCaret(new wxRichTextCaret(this, wxRICHTEXT_DEFAULT_CARET_WIDTH, 16)); +#else + SetCaret(new wxCaret(this, wxRICHTEXT_DEFAULT_CARET_WIDTH, 16)); +#endif + // Tell the sizers to use the given or best size - SetBestFittingSize(size); + SetInitialSize(size); #if wxRICHTEXT_BUFFERED_PAINTING // Create a buffer @@ -170,78 +305,123 @@ bool wxRichTextCtrl::Create( wxWindow* parent, wxWindowID id, const wxString& va GetBuffer().AddEventHandler(this); + // Accelerators + wxAcceleratorEntry entries[6]; + + entries[0].Set(wxACCEL_CTRL, (int) 'C', wxID_COPY); + entries[1].Set(wxACCEL_CTRL, (int) 'X', wxID_CUT); + entries[2].Set(wxACCEL_CTRL, (int) 'V', wxID_PASTE); + entries[3].Set(wxACCEL_CTRL, (int) 'A', wxID_SELECTALL); + entries[4].Set(wxACCEL_CTRL, (int) 'Z', wxID_UNDO); + entries[5].Set(wxACCEL_CTRL, (int) 'Y', wxID_REDO); + + wxAcceleratorTable accel(6, entries); + SetAcceleratorTable(accel); + + m_contextMenu = new wxMenu; + m_contextMenu->Append(wxID_UNDO, _("&Undo")); + m_contextMenu->Append(wxID_REDO, _("&Redo")); + m_contextMenu->AppendSeparator(); + m_contextMenu->Append(wxID_CUT, _("Cu&t")); + m_contextMenu->Append(wxID_COPY, _("&Copy")); + m_contextMenu->Append(wxID_PASTE, _("&Paste")); + m_contextMenu->Append(wxID_CLEAR, _("&Delete")); + m_contextMenu->AppendSeparator(); + m_contextMenu->Append(wxID_SELECTALL, _("Select &All")); + m_contextMenu->AppendSeparator(); + m_contextMenu->Append(wxID_RICHTEXT_PROPERTIES1, _("&Properties")); + +#if wxUSE_DRAG_AND_DROP + SetDropTarget(new wxRichTextDropTarget(this)); +#endif + return true; } wxRichTextCtrl::~wxRichTextCtrl() { + SetFocusObject(& GetBuffer(), false); GetBuffer().RemoveEventHandler(this); - + delete m_contextMenu; } /// Member initialisation void wxRichTextCtrl::Init() { - m_freezeCount = 0; m_contextMenu = NULL; m_caret = NULL; m_caretPosition = -1; - m_selectionRange.SetRange(-2, -2); m_selectionAnchor = -2; + m_selectionAnchorObject = NULL; + m_selectionState = wxRichTextCtrlSelectionState_Normal; m_editable = true; + m_verticalScrollbarEnabled = true; m_caretAtLineStart = false; m_dragging = false; +#if wxUSE_DRAG_AND_DROP + m_preDrag = false; +#endif m_fullLayoutRequired = false; m_fullLayoutTime = 0; m_fullLayoutSavedPosition = 0; m_delayedLayoutThreshold = wxRICHTEXT_DEFAULT_DELAYED_LAYOUT_THRESHOLD; m_caretPositionForDefaultStyle = -2; + m_focusObject = & m_buffer; } -/// Call Freeze to prevent refresh -void wxRichTextCtrl::Freeze() -{ - m_freezeCount ++; -} - -/// Call Thaw to refresh -void wxRichTextCtrl::Thaw() +void wxRichTextCtrl::DoThaw() { - m_freezeCount --; - - if (m_freezeCount == 0) - { + if (GetBuffer().IsDirty()) + LayoutContent(); + else SetupScrollbars(); - Refresh(false); - } + + wxWindow::DoThaw(); } /// Clear all text void wxRichTextCtrl::Clear() { - m_buffer.Reset(); - m_buffer.SetDirty(true); + if (GetFocusObject() == & GetBuffer()) + { + m_buffer.ResetAndClearCommands(); + m_buffer.Invalidate(wxRICHTEXT_ALL); + } + else + { + GetFocusObject()->Reset(); + } + m_caretPosition = -1; m_caretPositionForDefaultStyle = -2; m_caretAtLineStart = false; - m_selectionRange.SetRange(-2, -2); + m_selection.Reset(); + m_selectionState = wxRichTextCtrlSelectionState_Normal; - SetScrollbars(0, 0, 0, 0, 0, 0); + Scroll(0,0); - if (m_freezeCount == 0) + if (!IsFrozen()) { - SetupScrollbars(); + LayoutContent(); Refresh(false); } - SendTextUpdatedEvent(); + + wxTextCtrl::SendTextUpdatedEvent(this); } /// Painting void wxRichTextCtrl::OnPaint(wxPaintEvent& WXUNUSED(event)) { - if (GetCaret()) +#if !wxRICHTEXT_USE_OWN_CARET + if (GetCaret() && !IsFrozen()) GetCaret()->Hide(); +#else + // Stop the caret refreshing the control from within the + // paint handler + if (GetCaret()) + ((wxRichTextCaret*) GetCaret())->EnableRefresh(false); +#endif { #if wxRICHTEXT_BUFFERED_PAINTING @@ -249,32 +429,67 @@ void wxRichTextCtrl::OnPaint(wxPaintEvent& WXUNUSED(event)) #else wxPaintDC dc(this); #endif - PrepareDC(dc); - if (m_freezeCount > 0) + if (IsFrozen()) return; + PrepareDC(dc); + dc.SetFont(GetFont()); // Paint the background PaintBackground(dc); - wxRect drawingArea(GetLogicalPoint(wxPoint(0, 0)), GetClientSize()); + // wxRect drawingArea(GetLogicalPoint(wxPoint(0, 0)), GetClientSize()); + + wxRect drawingArea(GetUpdateRegion().GetBox()); + drawingArea.SetPosition(GetLogicalPoint(drawingArea.GetPosition())); + wxRect availableSpace(GetClientSize()); - if (GetBuffer().GetDirty()) + wxRichTextDrawingContext context(& GetBuffer()); + if (GetBuffer().IsDirty()) { - GetBuffer().Layout(dc, availableSpace, wxRICHTEXT_FIXED_WIDTH|wxRICHTEXT_VARIABLE_HEIGHT); - GetBuffer().SetDirty(false); + GetBuffer().Layout(dc, context, availableSpace, availableSpace, wxRICHTEXT_FIXED_WIDTH|wxRICHTEXT_VARIABLE_HEIGHT); + GetBuffer().Invalidate(wxRICHTEXT_NONE); SetupScrollbars(); } - GetBuffer().Draw(dc, GetBuffer().GetRange(), GetInternalSelectionRange(), drawingArea, 0 /* descent */, 0 /* flags */); + wxRect clipRect(availableSpace); + clipRect.x += GetBuffer().GetLeftMargin(); + clipRect.y += GetBuffer().GetTopMargin(); + clipRect.width -= (GetBuffer().GetLeftMargin() + GetBuffer().GetRightMargin()); + clipRect.height -= (GetBuffer().GetTopMargin() + GetBuffer().GetBottomMargin()); + clipRect.SetPosition(GetLogicalPoint(clipRect.GetPosition())); + dc.SetClippingRegion(clipRect); + + int flags = 0; + if ((GetExtraStyle() & wxRICHTEXT_EX_NO_GUIDELINES) == 0) + flags |= wxRICHTEXT_DRAW_GUIDELINES; + + GetBuffer().Draw(dc, context, GetBuffer().GetOwnRange(), GetSelection(), drawingArea, 0 /* descent */, flags); + + dc.DestroyClippingRegion(); + + // Other user defined painting after everything else (i.e. all text) is painted + PaintAboveContent(dc); + +#if wxRICHTEXT_USE_OWN_CARET + if (GetCaret()->IsVisible()) + { + PositionCaret(); + ((wxRichTextCaret*) GetCaret())->DoDraw(& dc); + } +#endif } +#if !wxRICHTEXT_USE_OWN_CARET if (GetCaret()) GetCaret()->Show(); - PositionCaret(); +#else + if (GetCaret()) + ((wxRichTextCaret*) GetCaret())->EnableRefresh(true); +#endif } // Empty implementation, to prevent flicker @@ -284,21 +499,67 @@ void wxRichTextCtrl::OnEraseBackground(wxEraseEvent& WXUNUSED(event)) void wxRichTextCtrl::OnSetFocus(wxFocusEvent& WXUNUSED(event)) { - wxCaret* caret = new wxCaret(this, wxRICHTEXT_DEFAULT_CARET_WIDTH, 16); - SetCaret(caret); - caret->Show(); - PositionCaret(); + if (GetCaret()) + { +#if !wxRICHTEXT_USE_OWN_CARET + PositionCaret(); +#endif + GetCaret()->Show(); + } +#if defined(__WXGTK__) && !wxRICHTEXT_USE_OWN_CARET + // Work around dropouts when control is focused if (!IsFrozen()) + { Refresh(false); + } +#endif } void wxRichTextCtrl::OnKillFocus(wxFocusEvent& WXUNUSED(event)) { - SetCaret(NULL); + if (GetCaret()) + GetCaret()->Hide(); +#if defined(__WXGTK__) && !wxRICHTEXT_USE_OWN_CARET + // Work around dropouts when control is focused if (!IsFrozen()) + { Refresh(false); + } +#endif +} + +void wxRichTextCtrl::OnCaptureLost(wxMouseCaptureLostEvent& WXUNUSED(event)) +{ + m_dragging = false; +} + +// Set up the caret for the given position and container, after a mouse click +bool wxRichTextCtrl::SetCaretPositionAfterClick(wxRichTextParagraphLayoutBox* container, long position, int hitTestFlags, bool extendSelection) +{ + bool caretAtLineStart = false; + + if (hitTestFlags & wxRICHTEXT_HITTEST_BEFORE) + { + // If we're at the start of a line (but not first in para) + // then we should keep the caret showing at the start of the line + // by showing the m_caretAtLineStart flag. + wxRichTextParagraph* para = container->GetParagraphAtPosition(position); + wxRichTextLine* line = container->GetLineAtPosition(position); + + if (line && para && line->GetAbsoluteRange().GetStart() == position && para->GetRange().GetStart() != position) + caretAtLineStart = true; + position --; + } + + if (extendSelection && (m_caretPosition != position)) + ExtendSelection(m_caretPosition, position, wxRICHTEXT_SHIFT_DOWN); + + MoveCaret(position, caretAtLineStart); + SetDefaultStyleToCursorStyle(); + + return true; } /// Left-click @@ -310,34 +571,54 @@ void wxRichTextCtrl::OnLeftClick(wxMouseEvent& event) PrepareDC(dc); dc.SetFont(GetFont()); + // TODO: detect change of focus object long position = 0; - int hit = GetBuffer().HitTest(dc, event.GetLogicalPosition(dc), position); - - if (hit != wxRICHTEXT_HITTEST_NONE) + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + wxRichTextDrawingContext context(& GetBuffer()); + int hit = GetBuffer().HitTest(dc, context, event.GetLogicalPosition(dc), position, & hitObj, & contextObj); + +#if wxUSE_DRAG_AND_DROP + // If there's no selection, or we're not inside it, this isn't an attempt to initiate Drag'n'Drop + if (IsEditable() && HasSelection() && GetSelectionRange().ToInternal().Contains(position)) { - m_dragStart = event.GetLogicalPosition(dc); - m_dragging = true; - CaptureMouse(); + // This might be an attempt at initiating Drag'n'Drop. So set the time & flags + m_preDrag = true; + m_dragStartPoint = event.GetPosition(); // No need to worry about logical positions etc, we only care about the distance from the original pt - SelectNone(); +#if wxUSE_DATETIME + m_dragStartTime = wxDateTime::UNow(); +#endif // wxUSE_DATETIME - bool caretAtLineStart = false; + // Preserve behaviour of clicking on an object within the selection + if (hit != wxRICHTEXT_HITTEST_NONE && hitObj) + m_dragging = true; - if (hit & wxRICHTEXT_HITTEST_BEFORE) - { - // If we're at the start of a line (but not first in para) - // then we should keep the caret showing at the start of the line - // by showing the m_caretAtLineStart flag. - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(position); - wxRichTextLine* line = GetBuffer().GetLineAtPosition(position); + return; // Don't skip the event, else the selection will be lost + } +#endif // wxUSE_DRAG_AND_DROP - if (line && para && line->GetAbsoluteRange().GetStart() == position && para->GetRange().GetStart() != position) - caretAtLineStart = true; - position --; + if (hit != wxRICHTEXT_HITTEST_NONE && hitObj) + { + wxRichTextParagraphLayoutBox* oldFocusObject = GetFocusObject(); + wxRichTextParagraphLayoutBox* container = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); + if (container && container != GetFocusObject() && container->AcceptsFocus()) + { + SetFocusObject(container, false /* don't set caret position yet */); } - MoveCaret(position, caretAtLineStart); - SetDefaultStyleToCursorStyle(); + m_dragging = true; + CaptureMouse(); + + long oldCaretPos = m_caretPosition; + + SetCaretPositionAfterClick(container, position, hit); + + // For now, don't handle shift-click when we're selecting multiple objects. + if (event.ShiftDown() && GetFocusObject() == oldFocusObject && m_selectionState == wxRichTextCtrlSelectionState_Normal) + ExtendSelection(oldCaretPos, m_caretPosition, wxRICHTEXT_SHIFT_DOWN); + else + SelectNone(); } event.Skip(); @@ -359,72 +640,207 @@ void wxRichTextCtrl::OnLeftUp(wxMouseEvent& event) long position = 0; wxPoint logicalPt = event.GetLogicalPosition(dc); - int hit = GetBuffer().HitTest(dc, logicalPt, position); - - if (hit != wxRICHTEXT_HITTEST_NONE) + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + wxRichTextDrawingContext context(& GetBuffer()); + // Only get objects at this level, not nested, because otherwise we couldn't swipe text at a single level. + int hit = GetFocusObject()->HitTest(dc, context, logicalPt, position, & hitObj, & contextObj, wxRICHTEXT_HITTEST_NO_NESTED_OBJECTS); + +#if wxUSE_DRAG_AND_DROP + if (m_preDrag) + { + // Preserve the behaviour that would have happened without drag-and-drop detection, in OnLeftClick + m_preDrag = false; // Tell DnD not to happen now: we are processing Left Up ourselves. + + // Do the actions that would have been done in OnLeftClick if we hadn't tried to drag + long position = 0; + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + int hit = GetBuffer().HitTest(dc, context, event.GetLogicalPosition(dc), position, & hitObj, & contextObj); + wxRichTextParagraphLayoutBox* oldFocusObject = GetFocusObject(); + wxRichTextParagraphLayoutBox* container = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); + if (container && container != GetFocusObject() && container->AcceptsFocus()) + { + SetFocusObject(container, false /* don't set caret position yet */); + } + + long oldCaretPos = m_caretPosition; + + SetCaretPositionAfterClick(container, position, hit); + + // For now, don't handle shift-click when we're selecting multiple objects. + if (event.ShiftDown() && GetFocusObject() == oldFocusObject && m_selectionState == wxRichTextCtrlSelectionState_Normal) + ExtendSelection(oldCaretPos, m_caretPosition, wxRICHTEXT_SHIFT_DOWN); + else + SelectNone(); + } +#endif + + if ((hit != wxRICHTEXT_HITTEST_NONE) && !(hit & wxRICHTEXT_HITTEST_OUTSIDE)) { - wxTextAttrEx attr; - if (GetStyle(position, attr)) + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_LEFT_CLICK, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetPosition(position); + if (hitObj) + cmdEvent.SetContainer(hitObj->GetContainer()); + + if (!GetEventHandler()->ProcessEvent(cmdEvent)) { - if (attr.HasFlag(wxTEXT_ATTR_URL)) + wxRichTextAttr attr; + if (GetStyle(position, attr)) { - wxString urlTarget = attr.GetURL(); - if (!urlTarget.IsEmpty()) + if (attr.HasFlag(wxTEXT_ATTR_URL)) { - wxMouseEvent mouseEvent(event); - - long startPos = 0, endPos = 0; - wxRichTextObject* obj = GetBuffer().GetLeafObjectAtPosition(position); - if (obj) + wxString urlTarget = attr.GetURL(); + if (!urlTarget.IsEmpty()) { - startPos = obj->GetRange().GetStart(); - endPos = obj->GetRange().GetEnd(); - } - - wxTextUrlEvent urlEvent(GetId(), mouseEvent, startPos, endPos); - InitCommandEvent(urlEvent); - - urlEvent.SetString(urlTarget); - - GetEventHandler()->ProcessEvent(urlEvent); + wxMouseEvent mouseEvent(event); + + long startPos = 0, endPos = 0; + wxRichTextObject* obj = GetFocusObject()->GetLeafObjectAtPosition(position); + if (obj) + { + startPos = obj->GetRange().GetStart(); + endPos = obj->GetRange().GetEnd(); + } + + wxTextUrlEvent urlEvent(GetId(), mouseEvent, startPos, endPos); + InitCommandEvent(urlEvent); + + urlEvent.SetString(urlTarget); + + GetEventHandler()->ProcessEvent(urlEvent); + } } } } } } + +#if wxUSE_DRAG_AND_DROP + m_preDrag = false; +#endif // wxUSE_DRAG_AND_DROP + +#if wxUSE_CLIPBOARD && wxUSE_DATAOBJ && wxHAVE_PRIMARY_SELECTION + if (HasSelection() && GetFocusObject() && GetFocusObject()->GetBuffer()) + { + // Put the selection in PRIMARY, if it exists + wxTheClipboard->UsePrimarySelection(true); + + wxRichTextRange range = GetInternalSelectionRange(); + GetFocusObject()->GetBuffer()->CopyToClipboard(range); + + wxTheClipboard->UsePrimarySelection(false); + } +#endif } -/// Left-click +/// Mouse-movements void wxRichTextCtrl::OnMoveMouse(wxMouseEvent& event) { +#if wxUSE_DRAG_AND_DROP + // See if we're starting Drag'n'Drop + if (m_preDrag) + { + int x = m_dragStartPoint.x - event.GetPosition().x; + int y = m_dragStartPoint.y - event.GetPosition().y; + size_t distance = abs(x) + abs(y); +#if wxUSE_DATETIME + wxTimeSpan diff = wxDateTime::UNow() - m_dragStartTime; +#endif + if ((distance > 10) +#if wxUSE_DATETIME + && (diff.GetMilliseconds() > 100) +#endif + ) + { + m_dragging = false; + + // Start drag'n'drop + wxRichTextRange range = GetInternalSelectionRange(); + if (range == wxRICHTEXT_NONE) + { + // Don't try to drag an empty range + m_preDrag = false; + return; + } + + // Cache the current situation, to be restored if Drag'n'Drop is cancelled + long oldPos = GetCaretPosition(); + wxRichTextParagraphLayoutBox* oldFocus = GetFocusObject(); + + wxDataObjectComposite* compositeObject = new wxDataObjectComposite(); + wxString text = GetFocusObject()->GetTextForRange(range); +#ifdef __WXMSW__ + text = wxTextFile::Translate(text, wxTextFileType_Dos); +#endif + compositeObject->Add(new wxTextDataObject(text), false /* not preferred */); + + wxRichTextBuffer* richTextBuf = new wxRichTextBuffer; + GetFocusObject()->CopyFragment(range, *richTextBuf); + compositeObject->Add(new wxRichTextBufferDataObject(richTextBuf), true /* preferred */); + + wxRichTextDropSource source(*compositeObject, this); + // Use wxDrag_DefaultMove, not because it's the likelier choice but because pressing Ctrl for Copy obeys the principle of least surprise + // The alternative, wxDrag_DefaultCopy, requires the user to know that Move needs the Shift key pressed + BeginBatchUndo(_("Drag")); + switch (source.DoDragDrop(wxDrag_AllowMove | wxDrag_DefaultMove)) + { + case wxDragMove: + case wxDragCopy: break; + + case wxDragError: + wxLogError(wxT("An error occurred during drag and drop operation")); + case wxDragNone: + case wxDragCancel: + Refresh(); // This is needed in wxMSW, otherwise resetting the position doesn't 'take' + SetCaretPosition(oldPos); + SetFocusObject(oldFocus, false); + default: break; + } + EndBatchUndo(); + + m_preDrag = false; + return; + } + } +#endif // wxUSE_DRAG_AND_DROP + wxClientDC dc(this); PrepareDC(dc); dc.SetFont(GetFont()); long position = 0; wxPoint logicalPt = event.GetLogicalPosition(dc); - int hit = GetBuffer().HitTest(dc, logicalPt, position); - + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + + int flags = 0; + + // If we're dragging, let's only consider positions at this level; otherwise + // selecting a range is not going to work. + wxRichTextParagraphLayoutBox* container = & GetBuffer(); + if (m_dragging) + { + flags = wxRICHTEXT_HITTEST_NO_NESTED_OBJECTS; + container = GetFocusObject(); + } + wxRichTextDrawingContext context(& GetBuffer()); + int hit = container->HitTest(dc, context, logicalPt, position, & hitObj, & contextObj, flags); + // See if we need to change the cursor - + { - if (hit != wxRICHTEXT_HITTEST_NONE) + if (hit != wxRICHTEXT_HITTEST_NONE && !(hit & wxRICHTEXT_HITTEST_OUTSIDE) && hitObj) { - wxTextAttrEx attr; - if (GetStyle(position, attr)) - { - if (attr.HasFlag(wxTEXT_ATTR_URL)) - { - if (GetCursor() != m_urlCursor) - SetCursor(m_urlCursor); - } - else if (!attr.HasFlag(wxTEXT_ATTR_URL)) - { - if (GetCursor() != m_textCursor) - SetCursor(m_textCursor); - } - } + wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); + if (actualContainer) + ProcessMouseMovement(actualContainer, hitObj, position, logicalPt); } + else + SetCursor(m_textCursor); } if (!event.Dragging()) @@ -433,56 +849,167 @@ void wxRichTextCtrl::OnMoveMouse(wxMouseEvent& event) return; } - if (m_dragging && hit != wxRICHTEXT_HITTEST_NONE) + if (m_dragging +#if wxUSE_DRAG_AND_DROP + && !m_preDrag +#endif + ) { - // TODO: test closeness - - bool caretAtLineStart = false; - - if (hit & wxRICHTEXT_HITTEST_BEFORE) + wxRichTextParagraphLayoutBox* commonAncestor = NULL; + wxRichTextParagraphLayoutBox* otherContainer = NULL; + wxRichTextParagraphLayoutBox* firstContainer = NULL; + + // Check for dragging across multiple containers + long position2 = 0; + wxRichTextObject* hitObj2 = NULL, *contextObj2 = NULL; + int hit2 = GetBuffer().HitTest(dc, context, logicalPt, position2, & hitObj2, & contextObj2, 0); + if (hit2 != wxRICHTEXT_HITTEST_NONE && !(hit2 & wxRICHTEXT_HITTEST_OUTSIDE) && hitObj2 && hitObj != hitObj2) { - // If we're at the start of a line (but not first in para) - // then we should keep the caret showing at the start of the line - // by showing the m_caretAtLineStart flag. - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(position); - wxRichTextLine* line = GetBuffer().GetLineAtPosition(position); + // See if we can find a common ancestor + if (m_selectionState == wxRichTextCtrlSelectionState_Normal) + { + firstContainer = GetFocusObject(); + commonAncestor = wxDynamicCast(firstContainer->GetParent(), wxRichTextParagraphLayoutBox); + } + else + { + firstContainer = wxDynamicCast(m_selectionAnchorObject, wxRichTextParagraphLayoutBox); + //commonAncestor = GetFocusObject(); // when the selection state is not normal, the focus object (e.g. table) + // is the common ancestor. + commonAncestor = wxDynamicCast(firstContainer->GetParent(), wxRichTextParagraphLayoutBox); + } - if (line && para && line->GetAbsoluteRange().GetStart() == position && para->GetRange().GetStart() != position) - caretAtLineStart = true; - position --; - } + if (commonAncestor && commonAncestor->HandlesChildSelections()) + { + wxRichTextObject* p = hitObj2; + while (p) + { + if (p->GetParent() == commonAncestor) + { + otherContainer = wxDynamicCast(p, wxRichTextParagraphLayoutBox); + break; + } + p = p->GetParent(); + } + } - if (m_caretPosition != position) - { - bool extendSel = ExtendSelection(m_caretPosition, position, wxRICHTEXT_SHIFT_DOWN); + if (commonAncestor && firstContainer && otherContainer) + { + // We have now got a second container that shares a parent with the current or anchor object. + if (m_selectionState == wxRichTextCtrlSelectionState_Normal) + { + // Don't go into common-ancestor selection mode if we still have the same + // container. + if (otherContainer != firstContainer) + { + m_selectionState = wxRichTextCtrlSelectionState_CommonAncestor; + m_selectionAnchorObject = firstContainer; + m_selectionAnchor = firstContainer->GetRange().GetStart(); - MoveCaret(position, caretAtLineStart); - SetDefaultStyleToCursorStyle(); + // The common ancestor, such as a table, returns the cell selection + // between the anchor and current position. + m_selection = commonAncestor->GetSelection(m_selectionAnchor, otherContainer->GetRange().GetStart()); + } + } + else + { + m_selection = commonAncestor->GetSelection(m_selectionAnchor, otherContainer->GetRange().GetStart()); + } - if (extendSel) - Refresh(false); + Refresh(); + + if (otherContainer->AcceptsFocus()) + SetFocusObject(otherContainer, false /* don't set caret and clear selection */); + MoveCaret(-1, false); + SetDefaultStyleToCursorStyle(); + } } } + + if (hitObj && m_dragging && hit != wxRICHTEXT_HITTEST_NONE && m_selectionState == wxRichTextCtrlSelectionState_Normal +#if wxUSE_DRAG_AND_DROP + && !m_preDrag +#endif + ) + { + // TODO: test closeness + SetCaretPositionAfterClick(container, position, hit, true /* extend selection */); + } } /// Right-click void wxRichTextCtrl::OnRightClick(wxMouseEvent& event) { SetFocus(); - event.Skip(); + + wxClientDC dc(this); + PrepareDC(dc); + dc.SetFont(GetFont()); + + long position = 0; + wxPoint logicalPt = event.GetLogicalPosition(dc); + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + wxRichTextDrawingContext context(& GetBuffer()); + int hit = GetFocusObject()->HitTest(dc, context, logicalPt, position, & hitObj, & contextObj); + + if (hitObj && hitObj->GetContainer() != GetFocusObject()) + { + wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); + if (actualContainer && actualContainer->AcceptsFocus()) + { + SetFocusObject(actualContainer, false /* don't set caret position yet */); + SetCaretPositionAfterClick(actualContainer, position, hit); + } + } + + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_RIGHT_CLICK, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetPosition(position); + if (hitObj) + cmdEvent.SetContainer(hitObj->GetContainer()); + + if (!GetEventHandler()->ProcessEvent(cmdEvent)) + event.Skip(); } /// Left-double-click -void wxRichTextCtrl::OnLeftDClick(wxMouseEvent& event) +void wxRichTextCtrl::OnLeftDClick(wxMouseEvent& WXUNUSED(event)) { - SelectWord(GetCaretPosition()+1); - event.Skip(); + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_LEFT_DCLICK, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetPosition(m_caretPosition+1); + cmdEvent.SetContainer(GetFocusObject()); + + if (!GetEventHandler()->ProcessEvent(cmdEvent)) + { + SelectWord(GetCaretPosition()+1); + } } /// Middle-click void wxRichTextCtrl::OnMiddleClick(wxMouseEvent& event) { - event.Skip(); + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_MIDDLE_CLICK, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetPosition(m_caretPosition+1); + cmdEvent.SetContainer(GetFocusObject()); + + if (!GetEventHandler()->ProcessEvent(cmdEvent)) + event.Skip(); + +#if wxUSE_CLIPBOARD && wxUSE_DATAOBJ && wxHAVE_PRIMARY_SELECTION + // Paste any PRIMARY selection, if it exists + wxTheClipboard->UsePrimarySelection(true); + Paste(); + wxTheClipboard->UsePrimarySelection(false); +#endif } /// Key press @@ -496,129 +1023,18 @@ void wxRichTextCtrl::OnChar(wxKeyEvent& event) if (event.AltDown()) flags |= wxRICHTEXT_ALT_DOWN; - if (event.GetKeyCode() == WXK_LEFT || - event.GetKeyCode() == WXK_RIGHT || - event.GetKeyCode() == WXK_UP || - event.GetKeyCode() == WXK_DOWN || - event.GetKeyCode() == WXK_HOME || - event.GetKeyCode() == WXK_PAGEUP || - event.GetKeyCode() == WXK_PAGEDOWN || - event.GetKeyCode() == WXK_END || - - event.GetKeyCode() == WXK_NUMPAD_LEFT || - event.GetKeyCode() == WXK_NUMPAD_RIGHT || - event.GetKeyCode() == WXK_NUMPAD_UP || - event.GetKeyCode() == WXK_NUMPAD_DOWN || - event.GetKeyCode() == WXK_NUMPAD_HOME || - event.GetKeyCode() == WXK_NUMPAD_PAGEUP || - event.GetKeyCode() == WXK_NUMPAD_PAGEDOWN || - event.GetKeyCode() == WXK_NUMPAD_END) - { - KeyboardNavigate(event.GetKeyCode(), flags); - return; - } - - // all the other keys modify the controls contents which shouldn't be - // possible if we're read-only - if ( !IsEditable() ) - { - event.Skip(); - return; - } - - if (event.GetKeyCode() == WXK_RETURN) + if (event.GetEventType() == wxEVT_KEY_DOWN) { - BeginBatchUndo(_("Insert Text")); - - long newPos = m_caretPosition; - - DeleteSelectedContent(& newPos); + if (event.IsKeyInCategory(WXK_CATEGORY_NAVIGATION)) + { + KeyboardNavigate(event.GetKeyCode(), flags); + return; + } - GetBuffer().InsertNewlineWithUndo(newPos+1, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE); - EndBatchUndo(); - SetDefaultStyleToCursorStyle(); - - ScrollIntoView(m_caretPosition, WXK_RIGHT); - - wxRichTextEvent cmdEvent( - wxEVT_COMMAND_RICHTEXT_RETURN, - GetId()); - cmdEvent.SetEventObject(this); - cmdEvent.SetFlags(flags); - if (!GetEventHandler()->ProcessEvent(cmdEvent)) - { - // Generate conventional event - wxCommandEvent textEvent(wxEVT_COMMAND_TEXT_ENTER, GetId()); - InitCommandEvent(textEvent); - - GetEventHandler()->ProcessEvent(textEvent); - } - } - else if (event.GetKeyCode() == WXK_BACK) - { - BeginBatchUndo(_("Delete Text")); - - // Submit range in character positions, which are greater than caret positions, - // so subtract 1 for deleted character and add 1 for conversion to character position. - if (m_caretPosition > -1 && !HasSelection()) - { - GetBuffer().DeleteRangeWithUndo(wxRichTextRange(m_caretPosition, m_caretPosition), - m_caretPosition, // Current caret position - m_caretPosition-1, // New caret position - this); - } - else - DeleteSelectedContent(); - - EndBatchUndo(); - - // Shouldn't this be in Do()? - if (GetLastPosition() == -1) - { - GetBuffer().Reset(); - - m_caretPosition = -1; - PositionCaret(); - SetDefaultStyleToCursorStyle(); - } - - ScrollIntoView(m_caretPosition, WXK_LEFT); - } - else if (event.GetKeyCode() == WXK_DELETE) - { - BeginBatchUndo(_("Delete Text")); - - // Submit range in character positions, which are greater than caret positions, - if (m_caretPosition < GetBuffer().GetRange().GetEnd()+1 && !HasSelection()) - { - GetBuffer().DeleteRangeWithUndo(wxRichTextRange(m_caretPosition+1, m_caretPosition+1), - m_caretPosition, // Current caret position - m_caretPosition+1, // New caret position - this); - } - else - DeleteSelectedContent(); - - EndBatchUndo(); - - // Shouldn't this be in Do()? - if (GetLastPosition() == -1) - { - GetBuffer().Reset(); - - m_caretPosition = -1; - PositionCaret(); - SetDefaultStyleToCursorStyle(); - } - } - else - { long keycode = event.GetKeyCode(); switch ( keycode ) { case WXK_ESCAPE: - // case WXK_SPACE: - case WXK_DELETE: case WXK_START: case WXK_LBUTTON: case WXK_RBUTTON: @@ -643,22 +1059,6 @@ void wxRichTextCtrl::OnChar(wxKeyEvent& event) case WXK_SNAPSHOT: case WXK_INSERT: case WXK_HELP: - case WXK_NUMPAD0: - case WXK_NUMPAD1: - case WXK_NUMPAD2: - case WXK_NUMPAD3: - case WXK_NUMPAD4: - case WXK_NUMPAD5: - case WXK_NUMPAD6: - case WXK_NUMPAD7: - case WXK_NUMPAD8: - case WXK_NUMPAD9: - case WXK_MULTIPLY: - case WXK_ADD: - case WXK_SEPARATOR: - case WXK_SUBTRACT: - case WXK_DECIMAL: - case WXK_DIVIDE: case WXK_F1: case WXK_F2: case WXK_F3: @@ -687,9 +1087,6 @@ void wxRichTextCtrl::OnChar(wxKeyEvent& event) case WXK_SCROLL: case WXK_PAGEUP: case WXK_PAGEDOWN: - case WXK_NUMPAD_SPACE: - case WXK_NUMPAD_TAB: - case WXK_NUMPAD_ENTER: case WXK_NUMPAD_F1: case WXK_NUMPAD_F2: case WXK_NUMPAD_F3: @@ -704,2330 +1101,3726 @@ void wxRichTextCtrl::OnChar(wxKeyEvent& event) case WXK_NUMPAD_END: case WXK_NUMPAD_BEGIN: case WXK_NUMPAD_INSERT: - case WXK_NUMPAD_DELETE: - case WXK_NUMPAD_EQUAL: - case WXK_NUMPAD_MULTIPLY: - case WXK_NUMPAD_ADD: - case WXK_NUMPAD_SEPARATOR: - case WXK_NUMPAD_SUBTRACT: - case WXK_NUMPAD_DECIMAL: + case WXK_WINDOWS_LEFT: { - event.Skip(); return; } - default: { - if (event.CmdDown() || event.AltDown()) + } + } + + // Must process this before translation, otherwise it's translated into a WXK_DELETE event. + if (event.CmdDown() && event.GetKeyCode() == WXK_BACK) + { + if (!IsEditable()) + { + return; + } + + if (HasSelection() && !CanDeleteRange(* GetFocusObject(), GetSelectionRange())) + { + return; + } + + BeginBatchUndo(_("Delete Text")); + + long newPos = m_caretPosition; + + bool processed = DeleteSelectedContent(& newPos); + + int deletions = 0; + if (processed) + deletions ++; + + // Submit range in character positions, which are greater than caret positions, + // so subtract 1 for deleted character and add 1 for conversion to character position. + if (newPos > -1) + { + if (event.CmdDown()) { - event.Skip(); - return; + long pos = wxRichTextCtrl::FindNextWordPosition(-1); + if (pos < newPos) + { + wxRichTextRange range(pos+1, newPos); + if (CanDeleteRange(* GetFocusObject(), range.FromInternal())) + { + GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer()); + deletions ++; + } + processed = true; + } } - if (keycode == wxT('\t')) + if (!processed) { - // See if we need to promote or demote the selection or paragraph at the cursor - // position, instead of inserting a tab. - long pos = GetAdjustedCaretPosition(GetCaretPosition()); - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(pos); - if (para && para->GetRange().GetStart() == pos && para->GetAttributes().HasListStyleName()) + wxRichTextRange range(newPos, newPos); + if (CanDeleteRange(* GetFocusObject(), range.FromInternal())) { - wxRichTextRange range; - if (HasSelection()) - range = GetSelectionRange(); - else - range = para->GetRange().FromInternal(); - - int promoteBy = event.ShiftDown() ? 1 : -1; - - PromoteList(promoteBy, range, NULL); - - return; + GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer()); + deletions ++; } } + } - BeginBatchUndo(_("Insert Text")); + EndBatchUndo(); - long newPos = m_caretPosition; - DeleteSelectedContent(& newPos); + if (GetLastPosition() == -1) + { + GetFocusObject()->Reset(); - wxString str = (wxChar) event.GetKeyCode(); - GetBuffer().InsertTextWithUndo(newPos+1, str, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE); + m_caretPosition = -1; + PositionCaret(); + SetDefaultStyleToCursorStyle(); + } - EndBatchUndo(); + ScrollIntoView(m_caretPosition, WXK_LEFT); - SetDefaultStyleToCursorStyle(); - ScrollIntoView(m_caretPosition, WXK_RIGHT); + // Always send this event; wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion. + //if (deletions > 0) + { + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_DELETE, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetFlags(flags); + cmdEvent.SetPosition(m_caretPosition+1); + cmdEvent.SetContainer(GetFocusObject()); + GetEventHandler()->ProcessEvent(cmdEvent); } + + Update(); } + else + event.Skip(); + + return; } -} -/// Delete content if there is a selection, e.g. when pressing a key. -bool wxRichTextCtrl::DeleteSelectedContent(long* newPos) -{ - if (HasSelection()) + // all the other keys modify the controls contents which shouldn't be + // possible if we're read-only + if ( !IsEditable() ) { - long pos = m_selectionRange.GetStart(); - GetBuffer().DeleteRangeWithUndo(m_selectionRange, - m_caretPosition, // Current caret position - pos, // New caret position - this); - m_selectionRange.SetRange(-2, -2); - - if (newPos) - *newPos = pos-1; - return true; + event.Skip(); + return; } - else - return false; -} -/// Keyboard navigation + if (event.GetKeyCode() == WXK_RETURN) + { + if (!CanInsertContent(* GetFocusObject(), m_caretPosition+1)) + return; -/* + long newPos = m_caretPosition; -Left: left one character -Right: right one character -Up: up one line -Down: down one line -Ctrl-Left: left one word -Ctrl-Right: right one word -Ctrl-Up: previous paragraph start -Ctrl-Down: next start of paragraph -Home: start of line -End: end of line -Ctrl-Home: start of document -Ctrl-End: end of document -Page-Up: Up a screen -Page-Down: Down a screen + if (HasSelection() && !CanDeleteRange(* GetFocusObject(), GetSelectionRange())) + { + return; + } -Maybe: + BeginBatchUndo(_("Insert Text")); -Ctrl-Alt-PgUp: Start of window -Ctrl-Alt-PgDn: End of window -F8: Start selection mode -Esc: End selection mode + DeleteSelectedContent(& newPos); -Adding Shift does the above but starts/extends selection. + if (event.ShiftDown()) + { + wxString text; + text = wxRichTextLineBreakChar; + GetFocusObject()->InsertTextWithUndo(& GetBuffer(), newPos+1, text, this); + m_caretAtLineStart = true; + PositionCaret(); + } + else + GetFocusObject()->InsertNewlineWithUndo(& GetBuffer(), newPos+1, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE|wxRICHTEXT_INSERT_INTERACTIVE); + EndBatchUndo(); + SetDefaultStyleToCursorStyle(); - */ + ScrollIntoView(m_caretPosition, WXK_RIGHT); -bool wxRichTextCtrl::KeyboardNavigate(int keyCode, int flags) -{ - bool success = false; + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_RETURN, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetFlags(flags); + cmdEvent.SetPosition(newPos+1); + cmdEvent.SetContainer(GetFocusObject()); - if (keyCode == WXK_RIGHT || keyCode == WXK_NUMPAD_RIGHT) - { - if (flags & wxRICHTEXT_CTRL_DOWN) - success = WordRight(1, flags); - else - success = MoveRight(1, flags); - } - else if (keyCode == WXK_LEFT || keyCode == WXK_NUMPAD_LEFT) - { - if (flags & wxRICHTEXT_CTRL_DOWN) - success = WordLeft(1, flags); - else - success = MoveLeft(1, flags); - } - else if (keyCode == WXK_UP || keyCode == WXK_NUMPAD_UP) - { - if (flags & wxRICHTEXT_CTRL_DOWN) - success = MoveToParagraphStart(flags); - else - success = MoveUp(1, flags); - } - else if (keyCode == WXK_DOWN || keyCode == WXK_NUMPAD_DOWN) - { - if (flags & wxRICHTEXT_CTRL_DOWN) - success = MoveToParagraphEnd(flags); - else - success = MoveDown(1, flags); - } - else if (keyCode == WXK_PAGEUP || keyCode == WXK_NUMPAD_PAGEUP) - { - success = PageUp(1, flags); - } - else if (keyCode == WXK_PAGEDOWN || keyCode == WXK_NUMPAD_PAGEDOWN) - { - success = PageDown(1, flags); - } - else if (keyCode == WXK_HOME || keyCode == WXK_NUMPAD_HOME) - { - if (flags & wxRICHTEXT_CTRL_DOWN) - success = MoveHome(flags); - else - success = MoveToLineStart(flags); - } - else if (keyCode == WXK_END || keyCode == WXK_NUMPAD_END) - { - if (flags & wxRICHTEXT_CTRL_DOWN) - success = MoveEnd(flags); - else - success = MoveToLineEnd(flags); - } + if (!GetEventHandler()->ProcessEvent(cmdEvent)) + { + // Generate conventional event + wxCommandEvent textEvent(wxEVT_COMMAND_TEXT_ENTER, GetId()); + InitCommandEvent(textEvent); - if (success) - { - ScrollIntoView(m_caretPosition, keyCode); - SetDefaultStyleToCursorStyle(); + GetEventHandler()->ProcessEvent(textEvent); + } + Update(); } - - return success; -} - -/// Extend the selection. Selections are in caret positions. -bool wxRichTextCtrl::ExtendSelection(long oldPos, long newPos, int flags) -{ - if (flags & wxRICHTEXT_SHIFT_DOWN) + else if (event.GetKeyCode() == WXK_BACK) { - // If not currently selecting, start selecting - if (m_selectionRange.GetStart() == -2) + long newPos = m_caretPosition; + + if (HasSelection() && !CanDeleteRange(* GetFocusObject(), GetSelectionRange())) { - m_selectionAnchor = oldPos; + return; + } - if (oldPos > newPos) - m_selectionRange.SetRange(newPos+1, oldPos); - else - m_selectionRange.SetRange(oldPos+1, newPos); - } - else - { - // Always ensure that the selection range start is greater than - // the end. - if (newPos > m_selectionAnchor) - m_selectionRange.SetRange(m_selectionAnchor+1, newPos); - else - m_selectionRange.SetRange(newPos+1, m_selectionAnchor); - } - - if (m_selectionRange.GetStart() > m_selectionRange.GetEnd()) - { - wxLogDebug(wxT("Strange selection range")); - } + BeginBatchUndo(_("Delete Text")); - return true; - } - else - return false; -} + bool processed = DeleteSelectedContent(& newPos); -/// Scroll into view, returning true if we scrolled. -/// This takes a _caret_ position. -bool wxRichTextCtrl::ScrollIntoView(long position, int keyCode) -{ - wxRichTextLine* line = GetVisibleLineForCaretPosition(position); + int deletions = 0; + if (processed) + deletions ++; - if (!line) - return false; + // Submit range in character positions, which are greater than caret positions, + // so subtract 1 for deleted character and add 1 for conversion to character position. + if (newPos > -1) + { + if (event.CmdDown()) + { + long pos = wxRichTextCtrl::FindNextWordPosition(-1); + if (pos < newPos) + { + wxRichTextRange range(pos+1, newPos); + if (CanDeleteRange(* GetFocusObject(), range.FromInternal())) + { + GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer()); + deletions ++; + } + processed = true; + } + } - int ppuX, ppuY; - GetScrollPixelsPerUnit(& ppuX, & ppuY); + if (!processed) + { + wxRichTextRange range(newPos, newPos); + if (CanDeleteRange(* GetFocusObject(), range.FromInternal())) + { + GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer()); + deletions ++; + } + } + } - int startXUnits, startYUnits; - GetViewStart(& startXUnits, & startYUnits); - int startY = startYUnits * ppuY; + EndBatchUndo(); - int sx = 0, sy = 0; - GetVirtualSize(& sx, & sy); - int sxUnits = 0; - int syUnits = 0; - if (ppuY != 0) - syUnits = sy/ppuY; + if (GetLastPosition() == -1) + { + GetFocusObject()->Reset(); - wxRect rect = line->GetRect(); + m_caretPosition = -1; + PositionCaret(); + SetDefaultStyleToCursorStyle(); + } - bool scrolled = false; + ScrollIntoView(m_caretPosition, WXK_LEFT); - wxSize clientSize = GetClientSize(); + // Always send this event; wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion. + //if (deletions > 0) + { + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_DELETE, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetFlags(flags); + cmdEvent.SetPosition(m_caretPosition+1); + cmdEvent.SetContainer(GetFocusObject()); + GetEventHandler()->ProcessEvent(cmdEvent); + } - // Going down - if (keyCode == WXK_DOWN || keyCode == WXK_NUMPAD_DOWN || - keyCode == WXK_RIGHT || keyCode == WXK_NUMPAD_DOWN || - keyCode == WXK_END || keyCode == WXK_NUMPAD_END || - keyCode == WXK_PAGEDOWN || keyCode == WXK_NUMPAD_PAGEDOWN) + Update(); + } + else if (event.GetKeyCode() == WXK_DELETE) { - if ((rect.y + rect.height) > (clientSize.y + startY)) + long newPos = m_caretPosition; + + if (HasSelection() && !CanDeleteRange(* GetFocusObject(), GetSelectionRange())) { - // Make it scroll so this item is at the bottom - // of the window - int y = rect.y - (clientSize.y - rect.height); - int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); + return; + } - // If we're still off the screen, scroll another line down - if ((rect.y + rect.height) > (clientSize.y + (yUnits*ppuY))) - yUnits ++; + BeginBatchUndo(_("Delete Text")); - if (startYUnits != yUnits) + bool processed = DeleteSelectedContent(& newPos); + + int deletions = 0; + if (processed) + deletions ++; + + // Submit range in character positions, which are greater than caret positions, + if (newPos < GetFocusObject()->GetOwnRange().GetEnd()+1) + { + if (event.CmdDown()) { - SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); - scrolled = true; + long pos = wxRichTextCtrl::FindNextWordPosition(1); + if (pos != -1 && (pos > newPos)) + { + wxRichTextRange range(newPos+1, pos); + if (CanDeleteRange(* GetFocusObject(), range.FromInternal())) + { + GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer()); + deletions ++; + } + processed = true; + } } - } - else if (rect.y < startY) - { - // Make it scroll so this item is at the top - // of the window - int y = rect.y ; - int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); - if (startYUnits != yUnits) + if (!processed && newPos < (GetLastPosition()-1)) { - SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); - scrolled = true; + wxRichTextRange range(newPos+1, newPos+1); + if (CanDeleteRange(* GetFocusObject(), range.FromInternal())) + { + GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer()); + deletions ++; + } } } - } - // Going up - else if (keyCode == WXK_UP || keyCode == WXK_NUMPAD_UP || - keyCode == WXK_LEFT || keyCode == WXK_NUMPAD_LEFT || - keyCode == WXK_HOME || keyCode == WXK_NUMPAD_HOME || - keyCode == WXK_PAGEUP || keyCode == WXK_NUMPAD_PAGEUP ) - { - if (rect.y < startY) + + EndBatchUndo(); + + if (GetLastPosition() == -1) { - // Make it scroll so this item is at the top - // of the window - int y = rect.y ; - int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); + GetFocusObject()->Reset(); - if (startYUnits != yUnits) - { - SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); - scrolled = true; - } + m_caretPosition = -1; + PositionCaret(); + SetDefaultStyleToCursorStyle(); } - else if ((rect.y + rect.height) > (clientSize.y + startY)) - { - // Make it scroll so this item is at the bottom - // of the window - int y = rect.y - (clientSize.y - rect.height); - int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); - // If we're still off the screen, scroll another line down - if ((rect.y + rect.height) > (clientSize.y + (yUnits*ppuY))) - yUnits ++; + ScrollIntoView(m_caretPosition, WXK_LEFT); - if (startYUnits != yUnits) - { - SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); - scrolled = true; - } + // Always send this event; wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion. + //if (deletions > 0) + { + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_DELETE, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetFlags(flags); + cmdEvent.SetPosition(m_caretPosition+1); + cmdEvent.SetContainer(GetFocusObject()); + GetEventHandler()->ProcessEvent(cmdEvent); } + + Update(); } - PositionCaret(); + else + { + long keycode = event.GetKeyCode(); + switch ( keycode ) + { + case WXK_ESCAPE: + { + event.Skip(); + return; + } - return scrolled; -} + default: + { +#ifdef __WXMAC__ + if (event.CmdDown()) +#else + // Fixes AltGr+key with European input languages on Windows + if ((event.CmdDown() && !event.AltDown()) || (event.AltDown() && !event.CmdDown())) +#endif + { + event.Skip(); + return; + } -/// Is the given position visible on the screen? -bool wxRichTextCtrl::IsPositionVisible(long pos) const -{ - wxRichTextLine* line = GetVisibleLineForCaretPosition(pos-1); + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_CHARACTER, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetFlags(flags); +#if wxUSE_UNICODE + cmdEvent.SetCharacter(event.GetUnicodeKey()); +#else + cmdEvent.SetCharacter((wxChar) keycode); +#endif + cmdEvent.SetPosition(m_caretPosition+1); + cmdEvent.SetContainer(GetFocusObject()); - if (!line) - return false; + if (keycode == wxT('\t')) + { + // See if we need to promote or demote the selection or paragraph at the cursor + // position, instead of inserting a tab. + long pos = GetAdjustedCaretPosition(GetCaretPosition()); + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(pos); + if (para && para->GetRange().GetStart() == pos && para->GetAttributes().HasListStyleName()) + { + wxRichTextRange range; + if (HasSelection()) + range = GetSelectionRange(); + else + range = para->GetRange().FromInternal(); - int ppuX, ppuY; - GetScrollPixelsPerUnit(& ppuX, & ppuY); + int promoteBy = event.ShiftDown() ? 1 : -1; - int startX, startY; - GetViewStart(& startX, & startY); - startX = 0; - startY = startY * ppuY; + PromoteList(promoteBy, range, NULL); - int sx = 0, sy = 0; - GetVirtualSize(& sx, & sy); - sx = 0; - if (ppuY != 0) - sy = sy/ppuY; + GetEventHandler()->ProcessEvent(cmdEvent); - wxRect rect = line->GetRect(); + return; + } + } - wxSize clientSize = GetClientSize(); + if (!CanInsertContent(* GetFocusObject(), m_caretPosition+1)) + return; - return !(((rect.y + rect.height) > (clientSize.y + startY)) || rect.y < startY); -} + if (HasSelection() && !CanDeleteRange(* GetFocusObject(), GetSelectionRange())) + return; -void wxRichTextCtrl::SetCaretPosition(long position, bool showAtLineStart) -{ - m_caretPosition = position; - m_caretAtLineStart = showAtLineStart; -} + BeginBatchUndo(_("Insert Text")); -/// Move caret one visual step forward: this may mean setting a flag -/// and keeping the same position if we're going from the end of one line -/// to the start of the next, which may be the exact same caret position. -void wxRichTextCtrl::MoveCaretForward(long oldPosition) -{ - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(oldPosition); + long newPos = m_caretPosition; + DeleteSelectedContent(& newPos); - // Only do the check if we're not at the end of the paragraph (where things work OK - // anyway) - if (para && (oldPosition != para->GetRange().GetEnd() - 1)) - { - wxRichTextLine* line = GetBuffer().GetLineAtPosition(oldPosition); +#if wxUSE_UNICODE + wxString str = event.GetUnicodeKey(); +#else + wxString str = (wxChar) event.GetKeyCode(); +#endif + GetFocusObject()->InsertTextWithUndo(& GetBuffer(), newPos+1, str, this, 0); - if (line) - { - wxRichTextRange lineRange = line->GetAbsoluteRange(); + EndBatchUndo(); - // We're at the end of a line. See whether we need to - // stay at the same actual caret position but change visual - // position, or not. - if (oldPosition == lineRange.GetEnd()) - { - if (m_caretAtLineStart) - { - // We're already at the start of the line, so actually move on now. - m_caretPosition = oldPosition + 1; - m_caretAtLineStart = false; - } - else - { - // We're showing at the end of the line, so keep to - // the same position but indicate that we're to show - // at the start of the next line. - m_caretPosition = oldPosition; - m_caretAtLineStart = true; - } SetDefaultStyleToCursorStyle(); - return; + ScrollIntoView(m_caretPosition, WXK_RIGHT); + + cmdEvent.SetPosition(m_caretPosition); + GetEventHandler()->ProcessEvent(cmdEvent); + + Update(); } } } - m_caretPosition ++; - SetDefaultStyleToCursorStyle(); } -/// Move caret one visual step backward: this may mean setting a flag -/// and keeping the same position if we're going from the end of one line -/// to the start of the next, which may be the exact same caret position. +bool wxRichTextCtrl::ProcessMouseMovement(wxRichTextParagraphLayoutBox* container, wxRichTextObject* WXUNUSED(obj), long position, const wxPoint& WXUNUSED(pos)) +{ + wxRichTextAttr attr; + if (container && GetStyle(position, attr, container)) + { + if (attr.HasFlag(wxTEXT_ATTR_URL)) + { + SetCursor(m_urlCursor); + } + else if (!attr.HasFlag(wxTEXT_ATTR_URL)) + { + SetCursor(m_textCursor); + } + return true; + } + else + return false; +} + +/// Delete content if there is a selection, e.g. when pressing a key. +bool wxRichTextCtrl::DeleteSelectedContent(long* newPos) +{ + if (HasSelection()) + { + long pos = m_selection.GetRange().GetStart(); + wxRichTextRange range = m_selection.GetRange(); + + // SelectAll causes more to be selected than doing it interactively, + // and causes a new paragraph to be inserted. So for multiline buffers, + // don't delete the final position. + if (range.GetEnd() == GetLastPosition() && GetNumberOfLines() > 0) + range.SetEnd(range.GetEnd()-1); + + GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer()); + m_selection.Reset(); + m_selectionState = wxRichTextCtrlSelectionState_Normal; + + if (newPos) + *newPos = pos-1; + return true; + } + else + return false; +} + +/// Keyboard navigation + +/* + +Left: left one character +Right: right one character +Up: up one line +Down: down one line +Ctrl-Left: left one word +Ctrl-Right: right one word +Ctrl-Up: previous paragraph start +Ctrl-Down: next start of paragraph +Home: start of line +End: end of line +Ctrl-Home: start of document +Ctrl-End: end of document +Page-Up: Up a screen +Page-Down: Down a screen + +Maybe: + +Ctrl-Alt-PgUp: Start of window +Ctrl-Alt-PgDn: End of window +F8: Start selection mode +Esc: End selection mode + +Adding Shift does the above but starts/extends selection. + + + */ + +bool wxRichTextCtrl::KeyboardNavigate(int keyCode, int flags) +{ + bool success = false; + + if (keyCode == WXK_RIGHT || keyCode == WXK_NUMPAD_RIGHT) + { + if (flags & wxRICHTEXT_CTRL_DOWN) + success = WordRight(1, flags); + else + success = MoveRight(1, flags); + } + else if (keyCode == WXK_LEFT || keyCode == WXK_NUMPAD_LEFT) + { + if (flags & wxRICHTEXT_CTRL_DOWN) + success = WordLeft(1, flags); + else + success = MoveLeft(1, flags); + } + else if (keyCode == WXK_UP || keyCode == WXK_NUMPAD_UP) + { + if (flags & wxRICHTEXT_CTRL_DOWN) + success = MoveToParagraphStart(flags); + else + success = MoveUp(1, flags); + } + else if (keyCode == WXK_DOWN || keyCode == WXK_NUMPAD_DOWN) + { + if (flags & wxRICHTEXT_CTRL_DOWN) + success = MoveToParagraphEnd(flags); + else + success = MoveDown(1, flags); + } + else if (keyCode == WXK_PAGEUP || keyCode == WXK_NUMPAD_PAGEUP) + { + success = PageUp(1, flags); + } + else if (keyCode == WXK_PAGEDOWN || keyCode == WXK_NUMPAD_PAGEDOWN) + { + success = PageDown(1, flags); + } + else if (keyCode == WXK_HOME || keyCode == WXK_NUMPAD_HOME) + { + if (flags & wxRICHTEXT_CTRL_DOWN) + success = MoveHome(flags); + else + success = MoveToLineStart(flags); + } + else if (keyCode == WXK_END || keyCode == WXK_NUMPAD_END) + { + if (flags & wxRICHTEXT_CTRL_DOWN) + success = MoveEnd(flags); + else + success = MoveToLineEnd(flags); + } + + if (success) + { + ScrollIntoView(m_caretPosition, keyCode); + SetDefaultStyleToCursorStyle(); + } + + return success; +} + +/// Extend the selection. Selections are in caret positions. +bool wxRichTextCtrl::ExtendSelection(long oldPos, long newPos, int flags) +{ + if (flags & wxRICHTEXT_SHIFT_DOWN) + { + if (oldPos == newPos) + return false; + + wxRichTextSelection oldSelection = m_selection; + + m_selection.SetContainer(GetFocusObject()); + + wxRichTextRange oldRange; + if (m_selection.IsValid()) + oldRange = m_selection.GetRange(); + else + oldRange = wxRICHTEXT_NO_SELECTION; + wxRichTextRange newRange; + + // If not currently selecting, start selecting + if (oldRange.GetStart() == -2) + { + m_selectionAnchor = oldPos; + + if (oldPos > newPos) + newRange.SetRange(newPos+1, oldPos); + else + newRange.SetRange(oldPos+1, newPos); + } + else + { + // Always ensure that the selection range start is greater than + // the end. + if (newPos > m_selectionAnchor) + newRange.SetRange(m_selectionAnchor+1, newPos); + else if (newPos == m_selectionAnchor) + newRange = wxRichTextRange(-2, -2); + else + newRange.SetRange(newPos+1, m_selectionAnchor); + } + + m_selection.SetRange(newRange); + + RefreshForSelectionChange(oldSelection, m_selection); + + if (newRange.GetStart() > newRange.GetEnd()) + { + wxLogDebug(wxT("Strange selection range")); + } + + return true; + } + else + return false; +} + +/// Scroll into view, returning true if we scrolled. +/// This takes a _caret_ position. +bool wxRichTextCtrl::ScrollIntoView(long position, int keyCode) +{ + wxRichTextLine* line = GetVisibleLineForCaretPosition(position); + + if (!line) + return false; + + int ppuX, ppuY; + GetScrollPixelsPerUnit(& ppuX, & ppuY); + + int startXUnits, startYUnits; + GetViewStart(& startXUnits, & startYUnits); + int startY = startYUnits * ppuY; + + int sx = 0, sy = 0; + GetVirtualSize(& sx, & sy); + int sxUnits = 0; + int syUnits = 0; + if (ppuY != 0) + syUnits = sy/ppuY; + + wxRect rect = line->GetRect(); + + bool scrolled = false; + + wxSize clientSize = GetClientSize(); + + int leftMargin, rightMargin, topMargin, bottomMargin; + + { + wxClientDC dc(this); + wxRichTextObject::GetTotalMargin(dc, & GetBuffer(), GetBuffer().GetAttributes(), leftMargin, rightMargin, + topMargin, bottomMargin); + } +// clientSize.y -= GetBuffer().GetBottomMargin(); + clientSize.y -= bottomMargin; + + if (GetWindowStyle() & wxRE_CENTRE_CARET) + { + int y = rect.y - GetClientSize().y/2; + int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); + if (y >= 0 && (y + clientSize.y) < GetBuffer().GetCachedSize().y) + { + if (startYUnits != yUnits) + { + SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); + scrolled = true; + } +#if !wxRICHTEXT_USE_OWN_CARET + if (scrolled) +#endif + PositionCaret(); + + return scrolled; + } + } + + // Going down + if (keyCode == WXK_DOWN || keyCode == WXK_NUMPAD_DOWN || + keyCode == WXK_RIGHT || keyCode == WXK_NUMPAD_RIGHT || + keyCode == WXK_END || keyCode == WXK_NUMPAD_END || + keyCode == WXK_PAGEDOWN || keyCode == WXK_NUMPAD_PAGEDOWN) + { + if ((rect.y + rect.height) > (clientSize.y + startY)) + { + // Make it scroll so this item is at the bottom + // of the window + int y = rect.y - (clientSize.y - rect.height); + int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); + + // If we're still off the screen, scroll another line down + if ((rect.y + rect.height) > (clientSize.y + (yUnits*ppuY))) + yUnits ++; + + if (startYUnits != yUnits) + { + SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); + scrolled = true; + } + } + else if (rect.y < (startY + GetBuffer().GetTopMargin())) + { + // Make it scroll so this item is at the top + // of the window + int y = rect.y - GetBuffer().GetTopMargin(); + int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); + + if (startYUnits != yUnits) + { + SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); + scrolled = true; + } + } + } + // Going up + else if (keyCode == WXK_UP || keyCode == WXK_NUMPAD_UP || + keyCode == WXK_LEFT || keyCode == WXK_NUMPAD_LEFT || + keyCode == WXK_HOME || keyCode == WXK_NUMPAD_HOME || + keyCode == WXK_PAGEUP || keyCode == WXK_NUMPAD_PAGEUP ) + { + if (rect.y < (startY + GetBuffer().GetBottomMargin())) + { + // Make it scroll so this item is at the top + // of the window + int y = rect.y - GetBuffer().GetTopMargin(); + int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); + + if (startYUnits != yUnits) + { + SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); + scrolled = true; + } + } + else if ((rect.y + rect.height) > (clientSize.y + startY)) + { + // Make it scroll so this item is at the bottom + // of the window + int y = rect.y - (clientSize.y - rect.height); + int yUnits = (int) (0.5 + ((float) y)/(float) ppuY); + + // If we're still off the screen, scroll another line down + if ((rect.y + rect.height) > (clientSize.y + (yUnits*ppuY))) + yUnits ++; + + if (startYUnits != yUnits) + { + SetScrollbars(ppuX, ppuY, sxUnits, syUnits, 0, yUnits); + scrolled = true; + } + } + } + +#if !wxRICHTEXT_USE_OWN_CARET + if (scrolled) +#endif + PositionCaret(); + + return scrolled; +} + +/// Is the given position visible on the screen? +bool wxRichTextCtrl::IsPositionVisible(long pos) const +{ + wxRichTextLine* line = GetVisibleLineForCaretPosition(pos-1); + + if (!line) + return false; + + int ppuX, ppuY; + GetScrollPixelsPerUnit(& ppuX, & ppuY); + + int startX, startY; + GetViewStart(& startX, & startY); + startX = 0; + startY = startY * ppuY; + + wxRect rect = line->GetRect(); + wxSize clientSize = GetClientSize(); + clientSize.y -= GetBuffer().GetBottomMargin(); + + return (rect.GetTop() >= (startY + GetBuffer().GetTopMargin())) && (rect.GetBottom() <= (startY + clientSize.y)); +} + +void wxRichTextCtrl::SetCaretPosition(long position, bool showAtLineStart) +{ + m_caretPosition = position; + m_caretAtLineStart = showAtLineStart; +} + +/// Move caret one visual step forward: this may mean setting a flag +/// and keeping the same position if we're going from the end of one line +/// to the start of the next, which may be the exact same caret position. +void wxRichTextCtrl::MoveCaretForward(long oldPosition) +{ + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(oldPosition); + + // Only do the check if we're not at the end of the paragraph (where things work OK + // anyway) + if (para && (oldPosition != para->GetRange().GetEnd() - 1)) + { + wxRichTextLine* line = GetFocusObject()->GetLineAtPosition(oldPosition); + + if (line) + { + wxRichTextRange lineRange = line->GetAbsoluteRange(); + + // We're at the end of a line. See whether we need to + // stay at the same actual caret position but change visual + // position, or not. + if (oldPosition == lineRange.GetEnd()) + { + if (m_caretAtLineStart) + { + // We're already at the start of the line, so actually move on now. + m_caretPosition = oldPosition + 1; + m_caretAtLineStart = false; + } + else + { + // We're showing at the end of the line, so keep to + // the same position but indicate that we're to show + // at the start of the next line. + m_caretPosition = oldPosition; + m_caretAtLineStart = true; + } + SetDefaultStyleToCursorStyle(); + return; + } + } + } + m_caretPosition ++; + SetDefaultStyleToCursorStyle(); +} + +/// Move caret one visual step backward: this may mean setting a flag +/// and keeping the same position if we're going from the end of one line +/// to the start of the next, which may be the exact same caret position. void wxRichTextCtrl::MoveCaretBack(long oldPosition) { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(oldPosition); + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(oldPosition); + + // Only do the check if we're not at the start of the paragraph (where things work OK + // anyway) + if (para && (oldPosition != para->GetRange().GetStart())) + { + wxRichTextLine* line = GetFocusObject()->GetLineAtPosition(oldPosition); + + if (line) + { + wxRichTextRange lineRange = line->GetAbsoluteRange(); + + // We're at the start of a line. See whether we need to + // stay at the same actual caret position but change visual + // position, or not. + if (oldPosition == lineRange.GetStart()) + { + m_caretPosition = oldPosition-1; + m_caretAtLineStart = true; + return; + } + else if (oldPosition == lineRange.GetEnd()) + { + if (m_caretAtLineStart) + { + // We're at the start of the line, so keep the same caret position + // but clear the start-of-line flag. + m_caretPosition = oldPosition; + m_caretAtLineStart = false; + } + else + { + // We're showing at the end of the line, so go back + // to the previous character position. + m_caretPosition = oldPosition - 1; + } + SetDefaultStyleToCursorStyle(); + return; + } + } + } + m_caretPosition --; + SetDefaultStyleToCursorStyle(); +} + +/// Move right +bool wxRichTextCtrl::MoveRight(int noPositions, int flags) +{ + long endPos = GetFocusObject()->GetOwnRange().GetEnd(); + + if (m_caretPosition + noPositions < endPos) + { + long oldPos = m_caretPosition; + long newPos = m_caretPosition + noPositions; + + bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); + if (!extendSel) + SelectNone(); + + // Determine by looking at oldPos and m_caretPosition whether + // we moved from the end of a line to the start of the next line, in which case + // we want to adjust the caret position such that it is positioned at the + // start of the next line, rather than jumping past the first character of the + // line. + if (noPositions == 1 && !extendSel) + MoveCaretForward(oldPos); + else + SetCaretPosition(newPos); + + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + else + return false; +} + +/// Move left +bool wxRichTextCtrl::MoveLeft(int noPositions, int flags) +{ + long startPos = -1; + + if (m_caretPosition > startPos - noPositions + 1) + { + long oldPos = m_caretPosition; + long newPos = m_caretPosition - noPositions; + bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); + if (!extendSel) + SelectNone(); + + if (noPositions == 1 && !extendSel) + MoveCaretBack(oldPos); + else + SetCaretPosition(newPos); + + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + else + return false; +} + +// Find the caret position for the combination of hit-test flags and character position. +// Returns the caret position and also an indication of where to place the caret (caretLineStart) +// since this is ambiguous (same position used for end of line and start of next). +long wxRichTextCtrl::FindCaretPositionForCharacterPosition(long position, int hitTestFlags, wxRichTextParagraphLayoutBox* container, + bool& caretLineStart) +{ + // If end of previous line, and hitTest is wxRICHTEXT_HITTEST_BEFORE, + // we want to be at the end of the last line but with m_caretAtLineStart set to true, + // so we view the caret at the start of the line. + caretLineStart = false; + long caretPosition = position; + + if (hitTestFlags & wxRICHTEXT_HITTEST_BEFORE) + { + wxRichTextLine* thisLine = container->GetLineAtPosition(position-1); + wxRichTextRange lineRange; + if (thisLine) + lineRange = thisLine->GetAbsoluteRange(); + + if (thisLine && (position-1) == lineRange.GetEnd()) + { + caretPosition --; + caretLineStart = true; + } + else + { + wxRichTextParagraph* para = container->GetParagraphAtPosition(position); + if (para && para->GetRange().GetStart() == position) + caretPosition --; + } + } + return caretPosition; +} + +/// Move up +bool wxRichTextCtrl::MoveUp(int noLines, int flags) +{ + return MoveDown(- noLines, flags); +} + +/// Move up +bool wxRichTextCtrl::MoveDown(int noLines, int flags) +{ + if (!GetCaret()) + return false; + + long lineNumber = GetFocusObject()->GetVisibleLineNumber(m_caretPosition, true, m_caretAtLineStart); + wxPoint pt = GetCaret()->GetPosition(); + long newLine = lineNumber + noLines; + bool notInThisObject = false; + + if (lineNumber != -1) + { + if (noLines > 0) + { + long lastLine = GetFocusObject()->GetVisibleLineNumber(GetFocusObject()->GetOwnRange().GetEnd()); + if (newLine > lastLine) + notInThisObject = true; + } + else + { + if (newLine < 0) + notInThisObject = true; + } + } + + wxRichTextParagraphLayoutBox* container = GetFocusObject(); + int hitTestFlags = wxRICHTEXT_HITTEST_NO_NESTED_OBJECTS|wxRICHTEXT_HITTEST_NO_FLOATING_OBJECTS; + + if (notInThisObject) + { + // If we know we're navigating out of the current object, + // try to find an object anywhere in the buffer at the new position (up or down a bit) + container = & GetBuffer(); + hitTestFlags &= ~wxRICHTEXT_HITTEST_NO_NESTED_OBJECTS; + + if (noLines > 0) // going down + { + pt.y = GetFocusObject()->GetPosition().y + GetFocusObject()->GetCachedSize().y + 2; + } + else // going up + { + pt.y = GetFocusObject()->GetPosition().y - 2; + } + } + else + { + wxRichTextLine* lineObj = GetFocusObject()->GetLineForVisibleLineNumber(newLine); + if (lineObj) + pt.y = lineObj->GetAbsolutePosition().y + 2; + else + return false; + } + + long newPos = 0; + wxClientDC dc(this); + PrepareDC(dc); + dc.SetFont(GetFont()); + + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + wxRichTextDrawingContext context(& GetBuffer()); + int hitTest = container->HitTest(dc, context, pt, newPos, & hitObj, & contextObj, hitTestFlags); + + if (hitObj && + ((hitTest & wxRICHTEXT_HITTEST_NONE) == 0) && + (! (hitObj == (& m_buffer) && ((hitTest & wxRICHTEXT_HITTEST_OUTSIDE) != 0))) // outside the buffer counts as 'do nothing' + ) + { + if (notInThisObject) + { + wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); + if (actualContainer && actualContainer != GetFocusObject() && actualContainer->AcceptsFocus()) + { + SetFocusObject(actualContainer, false /* don't set caret position yet */); + + container = actualContainer; + } + } + + bool caretLineStart = true; + long caretPosition = FindCaretPositionForCharacterPosition(newPos, hitTest, container, caretLineStart); + long newSelEnd = caretPosition; + bool extendSel; + + if (notInThisObject) + extendSel = false; + else + extendSel = ExtendSelection(m_caretPosition, newSelEnd, flags); + + if (!extendSel) + SelectNone(); + + SetCaretPosition(caretPosition, caretLineStart); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + + return false; +} + +/// Move to the end of the paragraph +bool wxRichTextCtrl::MoveToParagraphEnd(int flags) +{ + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(m_caretPosition, true); + if (para) + { + long newPos = para->GetRange().GetEnd() - 1; + bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); + if (!extendSel) + SelectNone(); - // Only do the check if we're not at the start of the paragraph (where things work OK - // anyway) - if (para && (oldPosition != para->GetRange().GetStart())) + SetCaretPosition(newPos); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + + return false; +} + +/// Move to the start of the paragraph +bool wxRichTextCtrl::MoveToParagraphStart(int flags) +{ + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(m_caretPosition, true); + if (para) + { + long newPos = para->GetRange().GetStart() - 1; + bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); + if (!extendSel) + SelectNone(); + + SetCaretPosition(newPos); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + + return false; +} + +/// Move to the end of the line +bool wxRichTextCtrl::MoveToLineEnd(int flags) +{ + wxRichTextLine* line = GetVisibleLineForCaretPosition(m_caretPosition); + + if (line) + { + wxRichTextRange lineRange = line->GetAbsoluteRange(); + long newPos = lineRange.GetEnd(); + bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); + if (!extendSel) + SelectNone(); + + SetCaretPosition(newPos); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + + return false; +} + +/// Move to the start of the line +bool wxRichTextCtrl::MoveToLineStart(int flags) +{ + wxRichTextLine* line = GetVisibleLineForCaretPosition(m_caretPosition); + if (line) + { + wxRichTextRange lineRange = line->GetAbsoluteRange(); + long newPos = lineRange.GetStart()-1; + + bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); + if (!extendSel) + SelectNone(); + + wxRichTextParagraph* para = GetFocusObject()->GetParagraphForLine(line); + + SetCaretPosition(newPos, para->GetRange().GetStart() != lineRange.GetStart()); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + + return false; +} + +/// Move to the start of the buffer +bool wxRichTextCtrl::MoveHome(int flags) +{ + if (m_caretPosition != -1) + { + bool extendSel = ExtendSelection(m_caretPosition, -1, flags); + if (!extendSel) + SelectNone(); + + SetCaretPosition(-1); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + else + return false; +} + +/// Move to the end of the buffer +bool wxRichTextCtrl::MoveEnd(int flags) +{ + long endPos = GetFocusObject()->GetOwnRange().GetEnd()-1; + + if (m_caretPosition != endPos) + { + bool extendSel = ExtendSelection(m_caretPosition, endPos, flags); + if (!extendSel) + SelectNone(); + + SetCaretPosition(endPos); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + else + return false; +} + +/// Move noPages pages up +bool wxRichTextCtrl::PageUp(int noPages, int flags) +{ + return PageDown(- noPages, flags); +} + +/// Move noPages pages down +bool wxRichTextCtrl::PageDown(int noPages, int flags) +{ + // Calculate which line occurs noPages * screen height further down. + wxRichTextLine* line = GetVisibleLineForCaretPosition(m_caretPosition); + if (line) + { + wxSize clientSize = GetClientSize(); + int newY = line->GetAbsolutePosition().y + noPages*clientSize.y; + + wxRichTextLine* newLine = GetFocusObject()->GetLineAtYPosition(newY); + if (newLine) + { + wxRichTextRange lineRange = newLine->GetAbsoluteRange(); + long pos = lineRange.GetStart()-1; + if (pos != m_caretPosition) + { + wxRichTextParagraph* para = GetFocusObject()->GetParagraphForLine(newLine); + + bool extendSel = ExtendSelection(m_caretPosition, pos, flags); + if (!extendSel) + SelectNone(); + + SetCaretPosition(pos, para->GetRange().GetStart() != lineRange.GetStart()); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + } + } + + return false; +} + +static bool wxRichTextCtrlIsWhitespace(const wxString& str) +{ + return str == wxT(" ") || str == wxT("\t"); +} + +// Finds the caret position for the next word +long wxRichTextCtrl::FindNextWordPosition(int direction) const +{ + long endPos = GetFocusObject()->GetOwnRange().GetEnd(); + + if (direction > 0) + { + long i = m_caretPosition+1+direction; // +1 for conversion to character pos + + // First skip current text to space + while (i < endPos && i > -1) + { + // i is in character, not caret positions + wxString text = GetFocusObject()->GetTextForRange(wxRichTextRange(i, i)); + wxRichTextLine* line = GetFocusObject()->GetLineAtPosition(i, false); + if (line && (i == line->GetAbsoluteRange().GetEnd())) + { + break; + } + else if (!wxRichTextCtrlIsWhitespace(text) && !text.empty()) + i += direction; + else + { + break; + } + } + while (i < endPos && i > -1) + { + // i is in character, not caret positions + wxString text = GetFocusObject()->GetTextForRange(wxRichTextRange(i, i)); + wxRichTextLine* line = GetFocusObject()->GetLineAtPosition(i, false); + if (line && (i == line->GetAbsoluteRange().GetEnd())) + return wxMax(-1, i); + + if (text.empty()) // End of paragraph, or maybe an image + return wxMax(-1, i - 1); + else if (wxRichTextCtrlIsWhitespace(text) || text.empty()) + i += direction; + else + { + // Convert to caret position + return wxMax(-1, i - 1); + } + } + if (i >= endPos) + return endPos-1; + return i-1; + } + else { - wxRichTextLine* line = GetBuffer().GetLineAtPosition(oldPosition); + long i = m_caretPosition; + + // First skip white space + while (i < endPos && i > -1) + { + // i is in character, not caret positions + wxString text = GetFocusObject()->GetTextForRange(wxRichTextRange(i, i)); + wxRichTextLine* line = GetFocusObject()->GetLineAtPosition(i, false); - if (line) + if (text.empty() || (line && (i == line->GetAbsoluteRange().GetStart()))) // End of paragraph, or maybe an image + break; + else if (wxRichTextCtrlIsWhitespace(text) || text.empty()) + i += direction; + else + break; + } + // Next skip current text to space + while (i < endPos && i > -1) { - wxRichTextRange lineRange = line->GetAbsoluteRange(); + // i is in character, not caret positions + wxString text = GetFocusObject()->GetTextForRange(wxRichTextRange(i, i)); + wxRichTextLine* line = GetFocusObject()->GetLineAtPosition(i, false); + if (line && line->GetAbsoluteRange().GetStart() == i) + return i-1; - // We're at the start of a line. See whether we need to - // stay at the same actual caret position but change visual - // position, or not. - if (oldPosition == lineRange.GetStart()) - { - m_caretPosition = oldPosition-1; - m_caretAtLineStart = true; - return; - } - else if (oldPosition == lineRange.GetEnd()) + if (!wxRichTextCtrlIsWhitespace(text) /* && !text.empty() */) + i += direction; + else { - if (m_caretAtLineStart) - { - // We're at the start of the line, so keep the same caret position - // but clear the start-of-line flag. - m_caretPosition = oldPosition; - m_caretAtLineStart = false; - } - else - { - // We're showing at the end of the line, so go back - // to the previous character position. - m_caretPosition = oldPosition - 1; - } - SetDefaultStyleToCursorStyle(); - return; + return i; } } + if (i < -1) + return -1; + return i; } - m_caretPosition --; - SetDefaultStyleToCursorStyle(); } -/// Move right -bool wxRichTextCtrl::MoveRight(int noPositions, int flags) +/// Move n words left +bool wxRichTextCtrl::WordLeft(int WXUNUSED(n), int flags) { - long endPos = GetBuffer().GetRange().GetEnd(); + long pos = FindNextWordPosition(-1); + if (pos != m_caretPosition) + { + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(pos, true); - if (m_caretPosition + noPositions < endPos) + bool extendSel = ExtendSelection(m_caretPosition, pos, flags); + if (!extendSel) + SelectNone(); + + SetCaretPosition(pos, para->GetRange().GetStart() != pos); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + + return false; +} + +/// Move n words right +bool wxRichTextCtrl::WordRight(int WXUNUSED(n), int flags) +{ + long pos = FindNextWordPosition(1); + if (pos != m_caretPosition) { - long oldPos = m_caretPosition; - long newPos = m_caretPosition + noPositions; + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(pos, true); - bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); + bool extendSel = ExtendSelection(m_caretPosition, pos, flags); if (!extendSel) SelectNone(); - // Determine by looking at oldPos and m_caretPosition whether - // we moved from the end of a line to the start of the next line, in which case - // we want to adjust the caret position such that it is positioned at the - // start of the next line, rather than jumping past the first character of the - // line. - if (noPositions == 1 && !extendSel) - MoveCaretForward(oldPos); - else - SetCaretPosition(newPos); + SetCaretPosition(pos, para->GetRange().GetStart() != pos); + PositionCaret(); + SetDefaultStyleToCursorStyle(); + + return true; + } + + return false; +} + +/// Sizing +void wxRichTextCtrl::OnSize(wxSizeEvent& event) +{ + // Only do sizing optimization for large buffers + if (GetBuffer().GetOwnRange().GetEnd() > m_delayedLayoutThreshold) + { + m_fullLayoutRequired = true; + m_fullLayoutTime = wxGetLocalTimeMillis(); + m_fullLayoutSavedPosition = GetFirstVisiblePosition(); + LayoutContent(true /* onlyVisibleRect */); + } + else + GetBuffer().Invalidate(wxRICHTEXT_ALL); + +#if wxRICHTEXT_BUFFERED_PAINTING + RecreateBuffer(); +#endif + + event.Skip(); +} + +// Force any pending layout due to large buffer +void wxRichTextCtrl::ForceDelayedLayout() +{ + if (m_fullLayoutRequired) + { + m_fullLayoutRequired = false; + m_fullLayoutTime = 0; + GetBuffer().Invalidate(wxRICHTEXT_ALL); + ShowPosition(m_fullLayoutSavedPosition); + Refresh(false); + Update(); + } +} + +/// Idle-time processing +void wxRichTextCtrl::OnIdle(wxIdleEvent& event) +{ +#if wxRICHTEXT_USE_OWN_CARET + if (((wxRichTextCaret*) GetCaret())->GetNeedsUpdate()) + { + ((wxRichTextCaret*) GetCaret())->SetNeedsUpdate(false); + PositionCaret(); + GetCaret()->Show(); + } +#endif + + const int layoutInterval = wxRICHTEXT_DEFAULT_LAYOUT_INTERVAL; + + if (m_fullLayoutRequired && (wxGetLocalTimeMillis() > (m_fullLayoutTime + layoutInterval))) + { + m_fullLayoutRequired = false; + m_fullLayoutTime = 0; + GetBuffer().Invalidate(wxRICHTEXT_ALL); + ShowPosition(m_fullLayoutSavedPosition); + Refresh(false); + } + + if (m_caretPositionForDefaultStyle != -2) + { + // If the caret position has changed, no longer reflect the default style + // in the UI. + if (GetCaretPosition() != m_caretPositionForDefaultStyle) + m_caretPositionForDefaultStyle = -2; + } + + event.Skip(); +} + +/// Scrolling +void wxRichTextCtrl::OnScroll(wxScrollWinEvent& event) +{ +#if wxRICHTEXT_USE_OWN_CARET + if (!((wxRichTextCaret*) GetCaret())->GetNeedsUpdate()) + { + GetCaret()->Hide(); + ((wxRichTextCaret*) GetCaret())->SetNeedsUpdate(); + } +#endif + + event.Skip(); +} + +/// Set up scrollbars, e.g. after a resize +void wxRichTextCtrl::SetupScrollbars(bool atTop) +{ + if (IsFrozen()) + return; + + if (GetBuffer().IsEmpty() || !m_verticalScrollbarEnabled) + { + SetScrollbars(0, 0, 0, 0, 0, 0); + return; + } + + // TODO: reimplement scrolling so we scroll by line, not by fixed number + // of pixels. See e.g. wxVScrolledWindow for ideas. + int pixelsPerUnit = 5; + wxSize clientSize = GetClientSize(); + + int maxHeight = GetBuffer().GetCachedSize().y + GetBuffer().GetTopMargin(); + + // Round up so we have at least maxHeight pixels + int unitsY = (int) (((float)maxHeight/(float)pixelsPerUnit) + 0.5); + + int startX = 0, startY = 0; + if (!atTop) + GetViewStart(& startX, & startY); + + int maxPositionX = 0; + int maxPositionY = (int) ((((float)(wxMax((unitsY*pixelsPerUnit) - clientSize.y, 0)))/((float)pixelsPerUnit)) + 0.5); + + int newStartX = wxMin(maxPositionX, startX); + int newStartY = wxMin(maxPositionY, startY); + + int oldPPUX, oldPPUY; + int oldStartX, oldStartY; + int oldVirtualSizeX = 0, oldVirtualSizeY = 0; + GetScrollPixelsPerUnit(& oldPPUX, & oldPPUY); + GetViewStart(& oldStartX, & oldStartY); + GetVirtualSize(& oldVirtualSizeX, & oldVirtualSizeY); + if (oldPPUY > 0) + oldVirtualSizeY /= oldPPUY; + + if (oldPPUX == 0 && oldPPUY == pixelsPerUnit && oldVirtualSizeY == unitsY && oldStartX == newStartX && oldStartY == newStartY) + return; + + // Don't set scrollbars if there were none before, and there will be none now. + if (oldPPUY != 0 && (oldVirtualSizeY*oldPPUY < clientSize.y) && (unitsY*pixelsPerUnit < clientSize.y)) + return; + + // Move to previous scroll position if + // possible + SetScrollbars(0, pixelsPerUnit, 0, unitsY, newStartX, newStartY); +} + +/// Paint the background +void wxRichTextCtrl::PaintBackground(wxDC& dc) +{ + wxColour backgroundColour = GetBackgroundColour(); + if (!backgroundColour.IsOk()) + backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE); + + // Clear the background + dc.SetBrush(wxBrush(backgroundColour)); + dc.SetPen(*wxTRANSPARENT_PEN); + wxRect windowRect(GetClientSize()); + windowRect.x -= 2; windowRect.y -= 2; + windowRect.width += 4; windowRect.height += 4; + + // We need to shift the rectangle to take into account + // scrolling. Converting device to logical coordinates. + CalcUnscrolledPosition(windowRect.x, windowRect.y, & windowRect.x, & windowRect.y); + dc.DrawRectangle(windowRect); +} + +#if wxRICHTEXT_BUFFERED_PAINTING +/// Recreate buffer bitmap if necessary +bool wxRichTextCtrl::RecreateBuffer(const wxSize& size) +{ + wxSize sz = size; + if (sz == wxDefaultSize) + sz = GetClientSize(); + + if (sz.x < 1 || sz.y < 1) + return false; + + if (!m_bufferBitmap.IsOk() || m_bufferBitmap.GetWidth() < sz.x || m_bufferBitmap.GetHeight() < sz.y) + m_bufferBitmap = wxBitmap(sz.x, sz.y); + return m_bufferBitmap.IsOk(); +} +#endif + +// ---------------------------------------------------------------------------- +// file IO functions +// ---------------------------------------------------------------------------- + +bool wxRichTextCtrl::DoLoadFile(const wxString& filename, int fileType) +{ + bool success = GetBuffer().LoadFile(filename, (wxRichTextFileType)fileType); + if (success) + m_filename = filename; + + DiscardEdits(); + SetInsertionPoint(0); + LayoutContent(); + PositionCaret(); + SetupScrollbars(true); + Refresh(false); + wxTextCtrl::SendTextUpdatedEvent(this); + + if (success) + return true; + else + { + wxLogError(_("File couldn't be loaded.")); - PositionCaret(); - SetDefaultStyleToCursorStyle(); + return false; + } +} + +bool wxRichTextCtrl::DoSaveFile(const wxString& filename, int fileType) +{ + if (GetBuffer().SaveFile(filename, (wxRichTextFileType)fileType)) + { + m_filename = filename; + + DiscardEdits(); - if (extendSel) - Refresh(false); return true; } - else - return false; + + wxLogError(_("The text couldn't be saved.")); + + return false; } -/// Move left -bool wxRichTextCtrl::MoveLeft(int noPositions, int flags) +// ---------------------------------------------------------------------------- +// wxRichTextCtrl specific functionality +// ---------------------------------------------------------------------------- + +/// Add a new paragraph of text to the end of the buffer +wxRichTextRange wxRichTextCtrl::AddParagraph(const wxString& text) { - long startPos = -1; + wxRichTextRange range = GetFocusObject()->AddParagraph(text); + GetBuffer().Invalidate(); + LayoutContent(); + return range; +} - if (m_caretPosition > startPos - noPositions + 1) - { - long oldPos = m_caretPosition; - long newPos = m_caretPosition - noPositions; - bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); - if (!extendSel) - SelectNone(); +/// Add an image +wxRichTextRange wxRichTextCtrl::AddImage(const wxImage& image) +{ + wxRichTextRange range = GetFocusObject()->AddImage(image); + GetBuffer().Invalidate(); + LayoutContent(); + return range; +} - if (noPositions == 1 && !extendSel) - MoveCaretBack(oldPos); - else - SetCaretPosition(newPos); +// ---------------------------------------------------------------------------- +// selection and ranges +// ---------------------------------------------------------------------------- - PositionCaret(); - SetDefaultStyleToCursorStyle(); +void wxRichTextCtrl::SelectAll() +{ + SetSelection(-1, -1); +} - if (extendSel) - Refresh(false); - return true; +/// Select none +void wxRichTextCtrl::SelectNone() +{ + if (m_selection.IsValid()) + { + wxRichTextSelection oldSelection = m_selection; + + m_selection.Reset(); + + RefreshForSelectionChange(oldSelection, m_selection); } - else - return false; + m_selectionAnchor = -2; + m_selectionAnchorObject = NULL; + m_selectionState = wxRichTextCtrlSelectionState_Normal; } -/// Move up -bool wxRichTextCtrl::MoveUp(int noLines, int flags) +static bool wxIsWordDelimiter(const wxString& text) { - return MoveDown(- noLines, flags); + return !text.IsEmpty() && !wxIsalnum(text[0]); } -/// Move up -bool wxRichTextCtrl::MoveDown(int noLines, int flags) +/// Select the word at the given character position +bool wxRichTextCtrl::SelectWord(long position) { - if (!GetCaret()) + if (position < 0 || position > GetFocusObject()->GetOwnRange().GetEnd()) return false; - long lineNumber = GetBuffer().GetVisibleLineNumber(m_caretPosition, true, m_caretAtLineStart); - wxPoint pt = GetCaret()->GetPosition(); - long newLine = lineNumber + noLines; + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(position); + if (!para) + return false; - if (lineNumber != -1) + if (position == para->GetRange().GetEnd()) + position --; + + long positionStart = position; + long positionEnd = position; + + for (positionStart = position; positionStart >= para->GetRange().GetStart(); positionStart --) { - if (noLines > 0) + wxString text = GetFocusObject()->GetTextForRange(wxRichTextRange(positionStart, positionStart)); + if (wxIsWordDelimiter(text)) { - long lastLine = GetBuffer().GetVisibleLineNumber(GetBuffer().GetRange().GetEnd()); - - if (newLine > lastLine) - return false; + positionStart ++; + break; } - else + } + if (positionStart < para->GetRange().GetStart()) + positionStart = para->GetRange().GetStart(); + + for (positionEnd = position; positionEnd < para->GetRange().GetEnd(); positionEnd ++) + { + wxString text = GetFocusObject()->GetTextForRange(wxRichTextRange(positionEnd, positionEnd)); + if (wxIsWordDelimiter(text)) { - if (newLine < 0) - return false; + positionEnd --; + break; } } + if (positionEnd >= para->GetRange().GetEnd()) + positionEnd = para->GetRange().GetEnd(); + + if (positionEnd < positionStart) + return false; + + SetSelection(positionStart, positionEnd+1); - wxRichTextLine* lineObj = GetBuffer().GetLineForVisibleLineNumber(newLine); - if (lineObj) + if (positionStart >= 0) { - pt.y = lineObj->GetAbsolutePosition().y + 2; + MoveCaret(positionStart-1, true); + SetDefaultStyleToCursorStyle(); } - else - return false; - long newPos = 0; + return true; +} + +wxString wxRichTextCtrl::GetStringSelection() const +{ + long from, to; + GetSelection(&from, &to); + + return GetRange(from, to); +} + +// ---------------------------------------------------------------------------- +// hit testing +// ---------------------------------------------------------------------------- + +wxTextCtrlHitTestResult +wxRichTextCtrl::HitTest(const wxPoint& pt, wxTextCoord *x, wxTextCoord *y) const +{ + // implement in terms of the other overload as the native ports typically + // can get the position and not (x, y) pair directly (although wxUniv + // directly gets x and y -- and so overrides this method as well) + long pos; + wxTextCtrlHitTestResult rc = HitTest(pt, &pos); + + if ( rc != wxTE_HT_UNKNOWN ) + { + PositionToXY(pos, x, y); + } + + return rc; +} + +wxTextCtrlHitTestResult +wxRichTextCtrl::HitTest(const wxPoint& pt, + long * pos) const +{ + wxClientDC dc((wxRichTextCtrl*) this); + ((wxRichTextCtrl*)this)->PrepareDC(dc); + + // Buffer uses logical position (relative to start of buffer) + // so convert + wxPoint pt2 = GetLogicalPoint(pt); + + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + wxRichTextDrawingContext context((wxRichTextBuffer*) & GetBuffer()); + int hit = ((wxRichTextCtrl*)this)->GetFocusObject()->HitTest(dc, context, pt2, *pos, & hitObj, & contextObj, wxRICHTEXT_HITTEST_NO_NESTED_OBJECTS); + + if ((hit & wxRICHTEXT_HITTEST_BEFORE) && (hit & wxRICHTEXT_HITTEST_OUTSIDE)) + return wxTE_HT_BEFORE; + else if ((hit & wxRICHTEXT_HITTEST_AFTER) && (hit & wxRICHTEXT_HITTEST_OUTSIDE)) + return wxTE_HT_BEYOND; + else if (hit & (wxRICHTEXT_HITTEST_BEFORE|wxRICHTEXT_HITTEST_AFTER)) + return wxTE_HT_ON_TEXT; + + return wxTE_HT_UNKNOWN; +} + +wxRichTextParagraphLayoutBox* +wxRichTextCtrl::FindContainerAtPoint(const wxPoint pt, long& position, int& hit, wxRichTextObject* hitObj, int flags/* = 0*/) +{ wxClientDC dc(this); PrepareDC(dc); dc.SetFont(GetFont()); - int hitTest = GetBuffer().HitTest(dc, pt, newPos); + wxPoint logicalPt = GetLogicalPoint(pt); - if (hitTest != wxRICHTEXT_HITTEST_NONE) - { - // If end of previous line, and hitTest is wxRICHTEXT_HITTEST_BEFORE, - // we want to be at the end of the last line but with m_caretAtLineStart set to true, - // so we view the caret at the start of the line. - bool caretLineStart = false; - if (hitTest == wxRICHTEXT_HITTEST_BEFORE) - { - wxRichTextLine* thisLine = GetBuffer().GetLineAtPosition(newPos-1); - wxRichTextRange lineRange; - if (thisLine) - lineRange = thisLine->GetAbsoluteRange(); - - if (thisLine && (newPos-1) == lineRange.GetEnd()) - { - newPos --; - caretLineStart = true; - } - else - { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(newPos); - if (para && para->GetRange().GetStart() == newPos) - newPos --; - } - } + wxRichTextObject* contextObj = NULL; + wxRichTextDrawingContext context(& GetBuffer()); + hit = GetBuffer().HitTest(dc, context, logicalPt, position, &hitObj, &contextObj, flags); + wxRichTextParagraphLayoutBox* container = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); - long newSelEnd = newPos; + return container; +} - bool extendSel = ExtendSelection(m_caretPosition, newSelEnd, flags); - if (!extendSel) - SelectNone(); - SetCaretPosition(newPos, caretLineStart); - PositionCaret(); - SetDefaultStyleToCursorStyle(); +// ---------------------------------------------------------------------------- +// set/get the controls text +// ---------------------------------------------------------------------------- - if (extendSel) - Refresh(false); - return true; - } +wxString wxRichTextCtrl::DoGetValue() const +{ + return GetBuffer().GetText(); +} - return false; +wxString wxRichTextCtrl::GetRange(long from, long to) const +{ + // Public API for range is different from internals + return GetFocusObject()->GetTextForRange(wxRichTextRange(from, to-1)); } -/// Move to the end of the paragraph -bool wxRichTextCtrl::MoveToParagraphEnd(int flags) +void wxRichTextCtrl::DoSetValue(const wxString& value, int flags) { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(m_caretPosition, true); - if (para) + // Don't call Clear here, since it always sends a text updated event + m_buffer.ResetAndClearCommands(); + m_buffer.Invalidate(wxRICHTEXT_ALL); + m_caretPosition = -1; + m_caretPositionForDefaultStyle = -2; + m_caretAtLineStart = false; + m_selection.Reset(); + m_selectionState = wxRichTextCtrlSelectionState_Normal; + + Scroll(0,0); + + if (!IsFrozen()) { - long newPos = para->GetRange().GetEnd() - 1; - bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); - if (!extendSel) - SelectNone(); + LayoutContent(); + Refresh(false); + } - SetCaretPosition(newPos); - PositionCaret(); - SetDefaultStyleToCursorStyle(); + if (!value.IsEmpty()) + { + // Remove empty paragraph + GetBuffer().Clear(); + DoWriteText(value, flags); - if (extendSel) - Refresh(false); - return true; + // for compatibility, don't move the cursor when doing SetValue() + SetInsertionPoint(0); + } + else + { + // still send an event for consistency + if (flags & SetValue_SendEvent) + wxTextCtrl::SendTextUpdatedEvent(this); } + DiscardEdits(); +} - return false; +void wxRichTextCtrl::WriteText(const wxString& value) +{ + DoWriteText(value); } -/// Move to the start of the paragraph -bool wxRichTextCtrl::MoveToParagraphStart(int flags) +void wxRichTextCtrl::DoWriteText(const wxString& value, int flags) { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(m_caretPosition, true); - if (para) - { - long newPos = para->GetRange().GetStart() - 1; - bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); - if (!extendSel) - SelectNone(); + wxString valueUnix = wxTextFile::Translate(value, wxTextFileType_Unix); - SetCaretPosition(newPos); - PositionCaret(); - SetDefaultStyleToCursorStyle(); + GetFocusObject()->InsertTextWithUndo(& GetBuffer(), m_caretPosition+1, valueUnix, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE); + + if ( flags & SetValue_SendEvent ) + wxTextCtrl::SendTextUpdatedEvent(this); +} + +void wxRichTextCtrl::AppendText(const wxString& text) +{ + SetInsertionPointEnd(); + + WriteText(text); +} + +/// Write an image at the current insertion point +bool wxRichTextCtrl::WriteImage(const wxImage& image, wxBitmapType bitmapType, const wxRichTextAttr& textAttr) +{ + wxRichTextImageBlock imageBlock; + + wxImage image2 = image; + if (imageBlock.MakeImageBlock(image2, bitmapType)) + return WriteImage(imageBlock, textAttr); + + return false; +} - if (extendSel) - Refresh(false); - return true; - } +bool wxRichTextCtrl::WriteImage(const wxString& filename, wxBitmapType bitmapType, const wxRichTextAttr& textAttr) +{ + wxRichTextImageBlock imageBlock; + + wxImage image; + if (imageBlock.MakeImageBlock(filename, bitmapType, image, false)) + return WriteImage(imageBlock, textAttr); return false; } -/// Move to the end of the line -bool wxRichTextCtrl::MoveToLineEnd(int flags) +bool wxRichTextCtrl::WriteImage(const wxRichTextImageBlock& imageBlock, const wxRichTextAttr& textAttr) { - wxRichTextLine* line = GetVisibleLineForCaretPosition(m_caretPosition); + return GetFocusObject()->InsertImageWithUndo(& GetBuffer(), m_caretPosition+1, imageBlock, this, 0, textAttr); +} - if (line) +bool wxRichTextCtrl::WriteImage(const wxBitmap& bitmap, wxBitmapType bitmapType, const wxRichTextAttr& textAttr) +{ + if (bitmap.IsOk()) { - wxRichTextRange lineRange = line->GetAbsoluteRange(); - long newPos = lineRange.GetEnd(); - bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); - if (!extendSel) - SelectNone(); - - SetCaretPosition(newPos); - PositionCaret(); - SetDefaultStyleToCursorStyle(); + wxRichTextImageBlock imageBlock; - if (extendSel) - Refresh(false); - return true; + wxImage image = bitmap.ConvertToImage(); + if (image.IsOk() && imageBlock.MakeImageBlock(image, bitmapType)) + return WriteImage(imageBlock, textAttr); } return false; } -/// Move to the start of the line -bool wxRichTextCtrl::MoveToLineStart(int flags) +// Write a text box at the current insertion point. +wxRichTextBox* wxRichTextCtrl::WriteTextBox(const wxRichTextAttr& textAttr) { - wxRichTextLine* line = GetVisibleLineForCaretPosition(m_caretPosition); - if (line) - { - wxRichTextRange lineRange = line->GetAbsoluteRange(); - long newPos = lineRange.GetStart()-1; - - bool extendSel = ExtendSelection(m_caretPosition, newPos, flags); - if (!extendSel) - SelectNone(); + wxRichTextBox* textBox = new wxRichTextBox; + textBox->SetAttributes(textAttr); + textBox->SetParent(& GetBuffer()); // set parent temporarily for AddParagraph to use correct style + textBox->AddParagraph(wxEmptyString); + textBox->SetParent(NULL); - wxRichTextParagraph* para = GetBuffer().GetParagraphForLine(line); + // The object returned is the one actually inserted into the buffer, + // while the original one is deleted. + wxRichTextObject* obj = GetFocusObject()->InsertObjectWithUndo(& GetBuffer(), m_caretPosition+1, textBox, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE); + wxRichTextBox* box = wxDynamicCast(obj, wxRichTextBox); + return box; +} - SetCaretPosition(newPos, para->GetRange().GetStart() != lineRange.GetStart()); - PositionCaret(); - SetDefaultStyleToCursorStyle(); +// Write a table at the current insertion point, returning the table. +wxRichTextTable* wxRichTextCtrl::WriteTable(int rows, int cols, const wxRichTextAttr& tableAttr, const wxRichTextAttr& cellAttr) +{ + wxASSERT(rows > 0 && cols > 0); - if (extendSel) - Refresh(false); - return true; - } + if (rows <= 0 || cols <= 0) + return NULL; - return false; -} + wxRichTextTable* table = new wxRichTextTable; + table->SetAttributes(tableAttr); + table->SetParent(& GetBuffer()); // set parent temporarily for AddParagraph to use correct style -/// Move to the start of the buffer -bool wxRichTextCtrl::MoveHome(int flags) -{ - if (m_caretPosition != -1) - { - bool extendSel = ExtendSelection(m_caretPosition, -1, flags); - if (!extendSel) - SelectNone(); + table->CreateTable(rows, cols); - SetCaretPosition(-1); - PositionCaret(); - SetDefaultStyleToCursorStyle(); + table->SetParent(NULL); - if (extendSel) - Refresh(false); - return true; + int i, j; + for (j = 0; j < rows; j++) + { + for (i = 0; i < cols; i++) + { + table->GetCell(j, i)->GetAttributes() = cellAttr; + } } - else - return false; + + // The object returned is the one actually inserted into the buffer, + // while the original one is deleted. + wxRichTextObject* obj = GetFocusObject()->InsertObjectWithUndo(& GetBuffer(), m_caretPosition+1, table, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE); + wxRichTextTable* tableResult = wxDynamicCast(obj, wxRichTextTable); + return tableResult; } -/// Move to the end of the buffer -bool wxRichTextCtrl::MoveEnd(int flags) + +/// Insert a newline (actually paragraph) at the current insertion point. +bool wxRichTextCtrl::Newline() { - long endPos = GetBuffer().GetRange().GetEnd()-1; + return GetFocusObject()->InsertNewlineWithUndo(& GetBuffer(), m_caretPosition+1, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE); +} - if (m_caretPosition != endPos) - { - bool extendSel = ExtendSelection(m_caretPosition, endPos, flags); - if (!extendSel) - SelectNone(); +/// Insert a line break at the current insertion point. +bool wxRichTextCtrl::LineBreak() +{ + wxString text; + text = wxRichTextLineBreakChar; + return GetFocusObject()->InsertTextWithUndo(& GetBuffer(), m_caretPosition+1, text, this); +} - SetCaretPosition(endPos); - PositionCaret(); - SetDefaultStyleToCursorStyle(); +// ---------------------------------------------------------------------------- +// Clipboard operations +// ---------------------------------------------------------------------------- - if (extendSel) - Refresh(false); - return true; +void wxRichTextCtrl::Copy() +{ + if (CanCopy()) + { + wxRichTextRange range = GetInternalSelectionRange(); + GetBuffer().CopyToClipboard(range); } - else - return false; } -/// Move noPages pages up -bool wxRichTextCtrl::PageUp(int noPages, int flags) +void wxRichTextCtrl::Cut() { - return PageDown(- noPages, flags); + if (CanCut()) + { + wxRichTextRange range = GetInternalSelectionRange(); + GetBuffer().CopyToClipboard(range); + + DeleteSelectedContent(); + LayoutContent(); + Refresh(false); + } } -/// Move noPages pages down -bool wxRichTextCtrl::PageDown(int noPages, int flags) +void wxRichTextCtrl::Paste() { - // Calculate which line occurs noPages * screen height further down. - wxRichTextLine* line = GetVisibleLineForCaretPosition(m_caretPosition); - if (line) + if (CanPaste()) { - wxSize clientSize = GetClientSize(); - int newY = line->GetAbsolutePosition().y + noPages*clientSize.y; + BeginBatchUndo(_("Paste")); - wxRichTextLine* newLine = GetBuffer().GetLineAtYPosition(newY); - if (newLine) - { - wxRichTextRange lineRange = newLine->GetAbsoluteRange(); - long pos = lineRange.GetStart()-1; - if (pos != m_caretPosition) - { - wxRichTextParagraph* para = GetBuffer().GetParagraphForLine(newLine); + long newPos = m_caretPosition; + DeleteSelectedContent(& newPos); - bool extendSel = ExtendSelection(m_caretPosition, pos, flags); - if (!extendSel) - SelectNone(); + GetBuffer().PasteFromClipboard(newPos); - SetCaretPosition(pos, para->GetRange().GetStart() != lineRange.GetStart()); - PositionCaret(); - SetDefaultStyleToCursorStyle(); + EndBatchUndo(); + } +} - if (extendSel) - Refresh(false); - return true; - } - } +void wxRichTextCtrl::DeleteSelection() +{ + if (CanDeleteSelection()) + { + DeleteSelectedContent(); } +} - return false; +bool wxRichTextCtrl::HasSelection() const +{ + return (m_selection.IsValid() && m_selection.GetContainer() == GetFocusObject()); } -// Finds the caret position for the next word -long wxRichTextCtrl::FindNextWordPosition(int direction) const +bool wxRichTextCtrl::HasUnfocusedSelection() const { - long endPos = GetBuffer().GetRange().GetEnd(); + return m_selection.IsValid(); +} - if (direction > 0) - { - long i = m_caretPosition+1+direction; // +1 for conversion to character pos +bool wxRichTextCtrl::CanCopy() const +{ + // Can copy if there's a selection + return HasSelection(); +} - // First skip current text to space - while (i < endPos && i > -1) - { - // i is in character, not caret positions - wxString text = GetBuffer().GetTextForRange(wxRichTextRange(i, i)); - if (text != wxT(" ") && !text.empty()) - i += direction; - else - { - break; - } - } - while (i < endPos && i > -1) - { - // i is in character, not caret positions - wxString text = GetBuffer().GetTextForRange(wxRichTextRange(i, i)); - if (text.empty()) // End of paragraph, or maybe an image - return wxMax(-1, i - 1); - else if (text == wxT(" ") || text.empty()) - i += direction; - else - { - // Convert to caret position - return wxMax(-1, i - 1); - } - } - if (i >= endPos) - return endPos-1; - return i-1; - } - else - { - long i = m_caretPosition; +bool wxRichTextCtrl::CanCut() const +{ + return CanDeleteSelection(); +} - // First skip white space - while (i < endPos && i > -1) - { - // i is in character, not caret positions - wxString text = GetBuffer().GetTextForRange(wxRichTextRange(i, i)); - if (text.empty()) // End of paragraph, or maybe an image - break; - else if (text == wxT(" ") || text.empty()) - i += direction; - else - break; - } - // Next skip current text to space - while (i < endPos && i > -1) - { - // i is in character, not caret positions - wxString text = GetBuffer().GetTextForRange(wxRichTextRange(i, i)); - if (text != wxT(" ") /* && !text.empty() */) - i += direction; - else - { - return i; - } - } - if (i < -1) - return -1; - return i; - } +bool wxRichTextCtrl::CanPaste() const +{ + if ( !IsEditable() || !GetFocusObject() || !CanInsertContent(* GetFocusObject(), m_caretPosition+1)) + return false; + + return GetBuffer().CanPasteFromClipboard(); } -/// Move n words left -bool wxRichTextCtrl::WordLeft(int WXUNUSED(n), int flags) +bool wxRichTextCtrl::CanDeleteSelection() const { - long pos = FindNextWordPosition(-1); - if (pos != m_caretPosition) - { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(pos, true); + return HasSelection() && IsEditable() && CanDeleteRange(* GetFocusObject(), GetSelectionRange()); +} - bool extendSel = ExtendSelection(m_caretPosition, pos, flags); - if (!extendSel) - SelectNone(); - SetCaretPosition(pos, para->GetRange().GetStart() != pos); - PositionCaret(); - SetDefaultStyleToCursorStyle(); +// ---------------------------------------------------------------------------- +// Accessors +// ---------------------------------------------------------------------------- - if (extendSel) - Refresh(false); - return true; - } +void wxRichTextCtrl::SetContextMenu(wxMenu* menu) +{ + if (m_contextMenu && m_contextMenu != menu) + delete m_contextMenu; + m_contextMenu = menu; +} - return false; +void wxRichTextCtrl::SetEditable(bool editable) +{ + m_editable = editable; } -/// Move n words right -bool wxRichTextCtrl::WordRight(int WXUNUSED(n), int flags) +void wxRichTextCtrl::SetInsertionPoint(long pos) { - long pos = FindNextWordPosition(1); - if (pos != m_caretPosition) - { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(pos, true); + SelectNone(); - bool extendSel = ExtendSelection(m_caretPosition, pos, flags); - if (!extendSel) - SelectNone(); + m_caretPosition = pos - 1; + + PositionCaret(); + + SetDefaultStyleToCursorStyle(); +} - SetCaretPosition(pos, para->GetRange().GetStart() != pos); - PositionCaret(); - SetDefaultStyleToCursorStyle(); +void wxRichTextCtrl::SetInsertionPointEnd() +{ + long pos = GetLastPosition(); + SetInsertionPoint(pos); +} - if (extendSel) - Refresh(false); - return true; - } +long wxRichTextCtrl::GetInsertionPoint() const +{ + return m_caretPosition+1; +} - return false; +wxTextPos wxRichTextCtrl::GetLastPosition() const +{ + return GetFocusObject()->GetOwnRange().GetEnd(); } -/// Sizing -void wxRichTextCtrl::OnSize(wxSizeEvent& event) +// If the return values from and to are the same, there is no +// selection. +void wxRichTextCtrl::GetSelection(long* from, long* to) const { - // Only do sizing optimization for large buffers - if (GetBuffer().GetRange().GetEnd() > m_delayedLayoutThreshold) + if (m_selection.IsValid()) { - m_fullLayoutRequired = true; - m_fullLayoutTime = wxGetLocalTimeMillis(); - m_fullLayoutSavedPosition = GetFirstVisiblePosition(); - LayoutContent(true /* onlyVisibleRect */); + *from = m_selection.GetRange().GetStart(); + *to = m_selection.GetRange().GetEnd(); + (*to) ++; } else - GetBuffer().Invalidate(wxRICHTEXT_ALL); - -#if wxRICHTEXT_BUFFERED_PAINTING - RecreateBuffer(); -#endif + { + *from = -2; + *to = -2; + } +} - event.Skip(); +bool wxRichTextCtrl::IsEditable() const +{ + return m_editable; } +// ---------------------------------------------------------------------------- +// selection +// ---------------------------------------------------------------------------- -/// Idle-time processing -void wxRichTextCtrl::OnIdle(wxIdleEvent& event) +void wxRichTextCtrl::SetSelection(long from, long to) { - const int layoutInterval = wxRICHTEXT_DEFAULT_LAYOUT_INTERVAL; - - if (m_fullLayoutRequired && (wxGetLocalTimeMillis() > (m_fullLayoutTime + layoutInterval))) + // if from and to are both -1, it means (in wxWidgets) that all text should + // be selected. + if ( (from == -1) && (to == -1) ) { - m_fullLayoutRequired = false; - m_fullLayoutTime = 0; - GetBuffer().Invalidate(wxRICHTEXT_ALL); - ShowPosition(m_fullLayoutSavedPosition); - Refresh(false); + from = 0; + to = GetLastPosition()+1; } - if (m_caretPositionForDefaultStyle != -2) + if (from == to) { - // If the caret position has changed, no longer reflect the default style - // in the UI. - if (GetCaretPosition() != m_caretPositionForDefaultStyle) - m_caretPositionForDefaultStyle = -2; + SelectNone(); } + else + { + wxRichTextSelection oldSelection = m_selection; - event.Skip(); -} + m_selectionAnchor = from-1; + m_selectionAnchorObject = NULL; + m_selection.Set(wxRichTextRange(from, to-1), GetFocusObject()); -/// Scrolling -void wxRichTextCtrl::OnScroll(wxScrollWinEvent& event) -{ - // Not used - event.Skip(); + m_caretPosition = wxMax(-1, to-1); + + RefreshForSelectionChange(oldSelection, m_selection); + PositionCaret(); + } } -/// Set up scrollbars, e.g. after a resize -void wxRichTextCtrl::SetupScrollbars(bool atTop) +// ---------------------------------------------------------------------------- +// Editing +// ---------------------------------------------------------------------------- + +void wxRichTextCtrl::Replace(long from, long to, + const wxString& value) { - if (m_freezeCount) - return; + BeginBatchUndo(_("Replace")); - if (GetBuffer().IsEmpty()) - { - SetScrollbars(0, 0, 0, 0, 0, 0); - return; - } + SetSelection(from, to); - // TODO: reimplement scrolling so we scroll by line, not by fixed number - // of pixels. See e.g. wxVScrolledWindow for ideas. - int pixelsPerUnit = 5; - wxSize clientSize = GetClientSize(); + wxRichTextAttr attr = GetDefaultStyle(); - int maxHeight = GetBuffer().GetCachedSize().y; + DeleteSelectedContent(); - // Round up so we have at least maxHeight pixels - int unitsY = (int) (((float)maxHeight/(float)pixelsPerUnit) + 0.5); + SetDefaultStyle(attr); - int startX = 0, startY = 0; - if (!atTop) - GetViewStart(& startX, & startY); + DoWriteText(value, SetValue_SelectionOnly); - int maxPositionX = 0; // wxMax(sz.x - clientSize.x, 0); - int maxPositionY = (int) ((((float)(wxMax((unitsY*pixelsPerUnit) - clientSize.y, 0)))/((float)pixelsPerUnit)) + 0.5); + EndBatchUndo(); +} - // Move to previous scroll position if - // possible - SetScrollbars(0, pixelsPerUnit, - 0, unitsY, - wxMin(maxPositionX, startX), wxMin(maxPositionY, startY)); +void wxRichTextCtrl::Remove(long from, long to) +{ + SelectNone(); + + GetFocusObject()->DeleteRangeWithUndo(wxRichTextRange(from, to-1), this, & GetBuffer()); + + LayoutContent(); + if (!IsFrozen()) + Refresh(false); } -/// Paint the background -void wxRichTextCtrl::PaintBackground(wxDC& dc) +bool wxRichTextCtrl::IsModified() const { - wxColour backgroundColour = GetBackgroundColour(); - if (!backgroundColour.Ok()) - backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE); + return m_buffer.IsModified(); +} - // Clear the background - dc.SetBrush(wxBrush(backgroundColour)); - dc.SetPen(*wxTRANSPARENT_PEN); - wxRect windowRect(GetClientSize()); - windowRect.x -= 2; windowRect.y -= 2; - windowRect.width += 4; windowRect.height += 4; +void wxRichTextCtrl::MarkDirty() +{ + m_buffer.Modify(true); +} - // We need to shift the rectangle to take into account - // scrolling. Converting device to logical coordinates. - CalcUnscrolledPosition(windowRect.x, windowRect.y, & windowRect.x, & windowRect.y); - dc.DrawRectangle(windowRect); +void wxRichTextCtrl::DiscardEdits() +{ + m_caretPositionForDefaultStyle = -2; + m_buffer.Modify(false); + m_buffer.GetCommandProcessor()->ClearCommands(); } -#if wxRICHTEXT_BUFFERED_PAINTING -/// Recreate buffer bitmap if necessary -bool wxRichTextCtrl::RecreateBuffer(const wxSize& size) +int wxRichTextCtrl::GetNumberOfLines() const { - wxSize sz = size; - if (sz == wxDefaultSize) - sz = GetClientSize(); + return GetFocusObject()->GetParagraphCount(); +} - if (sz.x < 1 || sz.y < 1) - return false; +// ---------------------------------------------------------------------------- +// Positions <-> coords +// ---------------------------------------------------------------------------- - if (!m_bufferBitmap.Ok() || m_bufferBitmap.GetWidth() < sz.x || m_bufferBitmap.GetHeight() < sz.y) - m_bufferBitmap = wxBitmap(sz.x, sz.y); - return m_bufferBitmap.Ok(); +long wxRichTextCtrl::XYToPosition(long x, long y) const +{ + return GetFocusObject()->XYToPosition(x, y); +} + +bool wxRichTextCtrl::PositionToXY(long pos, long *x, long *y) const +{ + return GetFocusObject()->PositionToXY(pos, x, y); } -#endif // ---------------------------------------------------------------------------- -// file IO functions +// // ---------------------------------------------------------------------------- -bool wxRichTextCtrl::DoLoadFile(const wxString& filename, int fileType) +void wxRichTextCtrl::ShowPosition(long pos) { - bool success = GetBuffer().LoadFile(filename, fileType); - if (success) - m_filename = filename; + if (!IsPositionVisible(pos)) + ScrollIntoView(pos-1, WXK_DOWN); +} - DiscardEdits(); - SetInsertionPoint(0); - LayoutContent(); - PositionCaret(); - SetupScrollbars(true); - Refresh(false); - SendTextUpdatedEvent(); +int wxRichTextCtrl::GetLineLength(long lineNo) const +{ + return GetFocusObject()->GetParagraphLength(lineNo); +} - if (success) - return true; - else - { - wxLogError(_("File couldn't be loaded.")); +wxString wxRichTextCtrl::GetLineText(long lineNo) const +{ + return GetFocusObject()->GetParagraphText(lineNo); +} - return false; +// ---------------------------------------------------------------------------- +// Undo/redo +// ---------------------------------------------------------------------------- + +void wxRichTextCtrl::Undo() +{ + if (CanUndo()) + { + GetCommandProcessor()->Undo(); } } -bool wxRichTextCtrl::DoSaveFile(const wxString& filename, int fileType) +void wxRichTextCtrl::Redo() { - if (GetBuffer().SaveFile(filename, fileType)) + if (CanRedo()) { - m_filename = filename; + GetCommandProcessor()->Redo(); + } +} - DiscardEdits(); +bool wxRichTextCtrl::CanUndo() const +{ + return GetCommandProcessor()->CanUndo() && IsEditable(); +} - return true; - } +bool wxRichTextCtrl::CanRedo() const +{ + return GetCommandProcessor()->CanRedo() && IsEditable(); +} - wxLogError(_("The text couldn't be saved.")); +// ---------------------------------------------------------------------------- +// implementation details +// ---------------------------------------------------------------------------- - return false; +void wxRichTextCtrl::Command(wxCommandEvent& event) +{ + SetValue(event.GetString()); + GetEventHandler()->ProcessEvent(event); +} + +void wxRichTextCtrl::OnDropFiles(wxDropFilesEvent& event) +{ + // By default, load the first file into the text window. + if (event.GetNumberOfFiles() > 0) + { + LoadFile(event.GetFiles()[0]); + } +} + +wxSize wxRichTextCtrl::DoGetBestSize() const +{ + return wxSize(10, 10); } // ---------------------------------------------------------------------------- -// wxRichTextCtrl specific functionality +// standard handlers for standard edit menu events // ---------------------------------------------------------------------------- -/// Add a new paragraph of text to the end of the buffer -wxRichTextRange wxRichTextCtrl::AddParagraph(const wxString& text) +void wxRichTextCtrl::OnCut(wxCommandEvent& WXUNUSED(event)) +{ + Cut(); +} + +void wxRichTextCtrl::OnClear(wxCommandEvent& WXUNUSED(event)) +{ + DeleteSelection(); +} + +void wxRichTextCtrl::OnCopy(wxCommandEvent& WXUNUSED(event)) +{ + Copy(); +} + +void wxRichTextCtrl::OnPaste(wxCommandEvent& WXUNUSED(event)) { - return GetBuffer().AddParagraph(text); + Paste(); } -/// Add an image -wxRichTextRange wxRichTextCtrl::AddImage(const wxImage& image) +void wxRichTextCtrl::OnUndo(wxCommandEvent& WXUNUSED(event)) { - return GetBuffer().AddImage(image); + Undo(); } -// ---------------------------------------------------------------------------- -// selection and ranges -// ---------------------------------------------------------------------------- - -void wxRichTextCtrl::SelectAll() +void wxRichTextCtrl::OnRedo(wxCommandEvent& WXUNUSED(event)) { - SetSelection(0, GetLastPosition()+1); - m_selectionAnchor = -1; + Redo(); } -/// Select none -void wxRichTextCtrl::SelectNone() +void wxRichTextCtrl::OnUpdateCut(wxUpdateUIEvent& event) { - if (!(GetSelectionRange() == wxRichTextRange(-2, -2))) - SetSelection(-2, -2); - m_selectionAnchor = -2; + event.Enable( CanCut() ); } -static bool wxIsWordDelimiter(const wxString& text) +void wxRichTextCtrl::OnUpdateCopy(wxUpdateUIEvent& event) { - return !text.IsEmpty() && !wxIsalnum(text[0]); + event.Enable( CanCopy() ); } -/// Select the word at the given character position -bool wxRichTextCtrl::SelectWord(long position) +void wxRichTextCtrl::OnUpdateClear(wxUpdateUIEvent& event) { - if (position < 0 || position > GetBuffer().GetRange().GetEnd()) - return false; - - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(position); - if (!para) - return false; - - long positionStart = position; - long positionEnd = position; - - for (positionStart = position; positionStart >= para->GetRange().GetStart(); positionStart --) - { - wxString text = GetBuffer().GetTextForRange(wxRichTextRange(positionStart, positionStart)); - if (wxIsWordDelimiter(text)) - { - positionStart ++; - break; - } - } - if (positionStart < para->GetRange().GetStart()) - positionStart = para->GetRange().GetStart(); - - for (positionEnd = position; positionEnd < para->GetRange().GetEnd(); positionEnd ++) - { - wxString text = GetBuffer().GetTextForRange(wxRichTextRange(positionEnd, positionEnd)); - if (wxIsWordDelimiter(text)) - { - positionEnd --; - break; - } - } - if (positionEnd >= para->GetRange().GetEnd()) - positionEnd = para->GetRange().GetEnd(); - - SetSelection(positionStart, positionEnd+1); - - if (positionStart >= 0) - { - MoveCaret(positionStart-1, true); - SetDefaultStyleToCursorStyle(); - } - - return true; + event.Enable( CanDeleteSelection() ); } -wxString wxRichTextCtrl::GetStringSelection() const +void wxRichTextCtrl::OnUpdatePaste(wxUpdateUIEvent& event) { - long from, to; - GetSelection(&from, &to); - - return GetRange(from, to); + event.Enable( CanPaste() ); } -// ---------------------------------------------------------------------------- -// hit testing -// ---------------------------------------------------------------------------- - -wxTextCtrlHitTestResult -wxRichTextCtrl::HitTest(const wxPoint& pt, wxTextCoord *x, wxTextCoord *y) const +void wxRichTextCtrl::OnUpdateUndo(wxUpdateUIEvent& event) { - // implement in terms of the other overload as the native ports typically - // can get the position and not (x, y) pair directly (although wxUniv - // directly gets x and y -- and so overrides this method as well) - long pos; - wxTextCtrlHitTestResult rc = HitTest(pt, &pos); - - if ( rc != wxTE_HT_UNKNOWN ) - { - PositionToXY(pos, x, y); - } - - return rc; + event.Enable( CanUndo() ); + event.SetText( GetCommandProcessor()->GetUndoMenuLabel() ); } -wxTextCtrlHitTestResult -wxRichTextCtrl::HitTest(const wxPoint& pt, - long * pos) const +void wxRichTextCtrl::OnUpdateRedo(wxUpdateUIEvent& event) { - wxClientDC dc((wxRichTextCtrl*) this); - ((wxRichTextCtrl*)this)->PrepareDC(dc); + event.Enable( CanRedo() ); + event.SetText( GetCommandProcessor()->GetRedoMenuLabel() ); +} - // Buffer uses logical position (relative to start of buffer) - // so convert - wxPoint pt2 = GetLogicalPoint(pt); +void wxRichTextCtrl::OnSelectAll(wxCommandEvent& WXUNUSED(event)) +{ + if (GetLastPosition() > 0) + SelectAll(); +} - int hit = ((wxRichTextCtrl*)this)->GetBuffer().HitTest(dc, pt2, *pos); +void wxRichTextCtrl::OnUpdateSelectAll(wxUpdateUIEvent& event) +{ + event.Enable(GetLastPosition() > 0); +} - switch ( hit ) +void wxRichTextCtrl::OnProperties(wxCommandEvent& event) +{ + int idx = event.GetId() - wxID_RICHTEXT_PROPERTIES1; + if (idx >= 0 && idx < m_contextMenuPropertiesInfo.GetCount()) { - case wxRICHTEXT_HITTEST_BEFORE: - return wxTE_HT_BEFORE; + wxRichTextObject* obj = m_contextMenuPropertiesInfo.GetObject(idx); + if (obj && CanEditProperties(obj)) + EditProperties(obj, this); - case wxRICHTEXT_HITTEST_AFTER: - return wxTE_HT_BEYOND; - - case wxRICHTEXT_HITTEST_ON: - return wxTE_HT_ON_TEXT; + m_contextMenuPropertiesInfo.Clear(); } - - return wxTE_HT_UNKNOWN; } -// ---------------------------------------------------------------------------- -// set/get the controls text -// ---------------------------------------------------------------------------- - -wxString wxRichTextCtrl::GetValue() const +void wxRichTextCtrl::OnUpdateProperties(wxUpdateUIEvent& event) { - return GetBuffer().GetText(); + int idx = event.GetId() - wxID_RICHTEXT_PROPERTIES1; + event.Enable(idx >= 0 && idx < m_contextMenuPropertiesInfo.GetCount()); } -wxString wxRichTextCtrl::GetRange(long from, long to) const +void wxRichTextCtrl::OnContextMenu(wxContextMenuEvent& event) { - // Public API for range is different from internals - return GetBuffer().GetTextForRange(wxRichTextRange(from, to-1)); + if (event.GetEventObject() != this) + { + event.Skip(); + return; + } + + ShowContextMenu(m_contextMenu, event.GetPosition()); } -void wxRichTextCtrl::DoSetValue(const wxString& value, int flags) +// Prepares the context menu, adding appropriate property-editing commands. +// Returns the number of property commands added. +int wxRichTextCtrl::PrepareContextMenu(wxMenu* menu, const wxPoint& pt, bool addPropertyCommands) { - Clear(); + wxClientDC dc(this); + PrepareDC(dc); + dc.SetFont(GetFont()); - // if the text is long enough, it's faster to just set it instead of first - // comparing it with the old one (chances are that it will be different - // anyhow, this comparison is there to avoid flicker for small single-line - // edit controls mostly) - if ( (value.length() > 0x400) || (value != GetValue()) ) + m_contextMenuPropertiesInfo.Clear(); + + long position = 0; + wxRichTextObject* hitObj = NULL; + wxRichTextObject* contextObj = NULL; + if (pt != wxDefaultPosition) { - DoWriteText(value); + wxPoint logicalPt = GetLogicalPoint(ScreenToClient(pt)); + wxRichTextDrawingContext context(& GetBuffer()); + int hit = GetBuffer().HitTest(dc, context, logicalPt, position, & hitObj, & contextObj); - // for compatibility, don't move the cursor when doing SetValue() - SetInsertionPoint(0); + if (hit == wxRICHTEXT_HITTEST_ON || hit == wxRICHTEXT_HITTEST_BEFORE || hit == wxRICHTEXT_HITTEST_AFTER) + { + wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); + if (hitObj && actualContainer) + { + if (actualContainer->AcceptsFocus()) + { + SetFocusObject(actualContainer, false /* don't set caret position yet */); + SetCaretPositionAfterClick(actualContainer, position, hit); + } + + if (addPropertyCommands) + m_contextMenuPropertiesInfo.AddItems(this, actualContainer, hitObj); + } + else + { + if (addPropertyCommands) + m_contextMenuPropertiesInfo.AddItems(this, GetFocusObject(), NULL); + } + } + else + { + if (addPropertyCommands) + m_contextMenuPropertiesInfo.AddItems(this, GetFocusObject(), NULL); + } } - else // same text + else { - if ( flags & SetValue_SendEvent ) + // Invoked from the keyboard, so don't set the caret position and don't use the event + // position + hitObj = GetFocusObject()->GetLeafObjectAtPosition(m_caretPosition+1); + if (hitObj) + contextObj = hitObj->GetParentContainer(); + else + contextObj = GetFocusObject(); + + wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox); + if (hitObj && actualContainer) { - // still send an event for consistency - SendTextUpdatedEvent(); + if (addPropertyCommands) + m_contextMenuPropertiesInfo.AddItems(this, actualContainer, hitObj); + } + else + { + if (addPropertyCommands) + m_contextMenuPropertiesInfo.AddItems(this, GetFocusObject(), NULL); } } - // we should reset the modified flag even if the value didn't really change - - // mark the control as being not dirty - we changed its text, not the - // user - DiscardEdits(); + if (menu) + { + if (addPropertyCommands) + m_contextMenuPropertiesInfo.AddMenuItems(menu); + return m_contextMenuPropertiesInfo.GetCount(); + } + else + return 0; } -void wxRichTextCtrl::WriteText(const wxString& value) +// Shows the context menu, adding appropriate property-editing commands +bool wxRichTextCtrl::ShowContextMenu(wxMenu* menu, const wxPoint& pt, bool addPropertyCommands) { - DoWriteText(value); + if (menu) + { + PrepareContextMenu(menu, pt, addPropertyCommands); + PopupMenu(menu); + return true; + } + else + return false; } -void wxRichTextCtrl::DoWriteText(const wxString& value, int flags) +bool wxRichTextCtrl::SetStyle(long start, long end, const wxTextAttr& style) { - wxString valueUnix = wxTextFile::Translate(value, wxTextFileType_Unix); - - GetBuffer().InsertTextWithUndo(m_caretPosition+1, valueUnix, this); - - if ( flags & SetValue_SendEvent ) - SendTextUpdatedEvent(); + return GetFocusObject()->SetStyle(wxRichTextRange(start, end-1), wxRichTextAttr(style)); } -void wxRichTextCtrl::AppendText(const wxString& text) +bool wxRichTextCtrl::SetStyle(long start, long end, const wxRichTextAttr& style) { - SetInsertionPointEnd(); - - WriteText(text); + return GetFocusObject()->SetStyle(wxRichTextRange(start, end-1), style); } -/// Write an image at the current insertion point -bool wxRichTextCtrl::WriteImage(const wxImage& image, int bitmapType) +bool wxRichTextCtrl::SetStyle(const wxRichTextRange& range, const wxTextAttr& style) { - wxRichTextImageBlock imageBlock; - - wxImage image2 = image; - if (imageBlock.MakeImageBlock(image2, bitmapType)) - return WriteImage(imageBlock); + return GetFocusObject()->SetStyle(range.ToInternal(), wxRichTextAttr(style)); +} - return false; +bool wxRichTextCtrl::SetStyle(const wxRichTextRange& range, const wxRichTextAttr& style) +{ + return GetFocusObject()->SetStyle(range.ToInternal(), style); } -bool wxRichTextCtrl::WriteImage(const wxString& filename, int bitmapType) +void wxRichTextCtrl::SetStyle(wxRichTextObject *obj, const wxRichTextAttr& textAttr) { - wxRichTextImageBlock imageBlock; + GetFocusObject()->SetStyle(obj, textAttr); +} - wxImage image; - if (imageBlock.MakeImageBlock(filename, bitmapType, image, false)) - return WriteImage(imageBlock); +// extended style setting operation with flags including: +// wxRICHTEXT_SETSTYLE_WITH_UNDO, wxRICHTEXT_SETSTYLE_OPTIMIZE, wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY. +// see richtextbuffer.h for more details. - return false; +bool wxRichTextCtrl::SetStyleEx(const wxRichTextRange& range, const wxRichTextAttr& style, int flags) +{ + return GetFocusObject()->SetStyle(range.ToInternal(), style, flags); } -bool wxRichTextCtrl::WriteImage(const wxRichTextImageBlock& imageBlock) +bool wxRichTextCtrl::SetDefaultStyle(const wxTextAttr& style) { - return GetBuffer().InsertImageWithUndo(m_caretPosition+1, imageBlock, this); + return GetBuffer().SetDefaultStyle(style); } -bool wxRichTextCtrl::WriteImage(const wxBitmap& bitmap, int bitmapType) +bool wxRichTextCtrl::SetDefaultStyle(const wxRichTextAttr& style) { - if (bitmap.Ok()) - { - wxRichTextImageBlock imageBlock; - - wxImage image = bitmap.ConvertToImage(); - if (image.Ok() && imageBlock.MakeImageBlock(image, bitmapType)) - return WriteImage(imageBlock); - } - - return false; + wxRichTextAttr attr1(style); + attr1.GetTextBoxAttr().Reset(); + return GetBuffer().SetDefaultStyle(attr1); } -/// Insert a newline (actually paragraph) at the current insertion point. -bool wxRichTextCtrl::Newline() +const wxRichTextAttr& wxRichTextCtrl::GetDefaultStyleEx() const { - return GetBuffer().InsertNewlineWithUndo(m_caretPosition+1, this); + return GetBuffer().GetDefaultStyle(); } - -// ---------------------------------------------------------------------------- -// Clipboard operations -// ---------------------------------------------------------------------------- - -void wxRichTextCtrl::Copy() +bool wxRichTextCtrl::GetStyle(long position, wxTextAttr& style) { - if (CanCopy()) + wxRichTextAttr attr; + if (GetFocusObject()->GetStyle(position, attr)) { - wxRichTextRange range = GetInternalSelectionRange(); - GetBuffer().CopyToClipboard(range); + style = attr; + return true; } + else + return false; } -void wxRichTextCtrl::Cut() +bool wxRichTextCtrl::GetStyle(long position, wxRichTextAttr& style) { - if (CanCut()) - { - wxRichTextRange range = GetInternalSelectionRange(); - GetBuffer().CopyToClipboard(range); - - DeleteSelectedContent(); - LayoutContent(); - Refresh(false); - } + return GetFocusObject()->GetStyle(position, style); } -void wxRichTextCtrl::Paste() +bool wxRichTextCtrl::GetStyle(long position, wxRichTextAttr& style, wxRichTextParagraphLayoutBox* container) { - if (CanPaste()) + wxRichTextAttr attr; + if (container->GetStyle(position, attr)) { - BeginBatchUndo(_("Paste")); - - long newPos = m_caretPosition; - DeleteSelectedContent(& newPos); - - GetBuffer().PasteFromClipboard(newPos); - - EndBatchUndo(); + style = attr; + return true; } + else + return false; } -void wxRichTextCtrl::DeleteSelection() +// get the common set of styles for the range +bool wxRichTextCtrl::GetStyleForRange(const wxRichTextRange& range, wxTextAttr& style) { - if (CanDeleteSelection()) + wxRichTextAttr attr; + if (GetFocusObject()->GetStyleForRange(range.ToInternal(), attr)) { - DeleteSelectedContent(); + style = attr; + return true; } + else + return false; } -bool wxRichTextCtrl::HasSelection() const +bool wxRichTextCtrl::GetStyleForRange(const wxRichTextRange& range, wxRichTextAttr& style) { - return m_selectionRange.GetStart() != -2 && m_selectionRange.GetEnd() != -2; + return GetFocusObject()->GetStyleForRange(range.ToInternal(), style); } -bool wxRichTextCtrl::CanCopy() const +bool wxRichTextCtrl::GetStyleForRange(const wxRichTextRange& range, wxRichTextAttr& style, wxRichTextParagraphLayoutBox* container) { - // Can copy if there's a selection - return HasSelection(); + return container->GetStyleForRange(range.ToInternal(), style); } -bool wxRichTextCtrl::CanCut() const +/// Get the content (uncombined) attributes for this position. +bool wxRichTextCtrl::GetUncombinedStyle(long position, wxRichTextAttr& style) { - return HasSelection() && IsEditable(); + return GetFocusObject()->GetUncombinedStyle(position, style); } -bool wxRichTextCtrl::CanPaste() const +/// Get the content (uncombined) attributes for this position. +bool wxRichTextCtrl::GetUncombinedStyle(long position, wxRichTextAttr& style, wxRichTextParagraphLayoutBox* container) { - if ( !IsEditable() ) - return false; - - return GetBuffer().CanPasteFromClipboard(); + return container->GetUncombinedStyle(position, style); } -bool wxRichTextCtrl::CanDeleteSelection() const +bool wxRichTextCtrl::SetProperties(const wxRichTextRange& range, const wxRichTextProperties& properties, int flags) { - return HasSelection() && IsEditable(); + return GetFocusObject()->SetProperties(range.ToInternal(), properties, flags); } +/// Set font, and also the buffer attributes +bool wxRichTextCtrl::SetFont(const wxFont& font) +{ + wxControl::SetFont(font); -// ---------------------------------------------------------------------------- -// Accessors -// ---------------------------------------------------------------------------- + wxRichTextAttr attr = GetBuffer().GetAttributes(); + attr.SetFont(font); + GetBuffer().SetBasicStyle(attr); -void wxRichTextCtrl::SetEditable(bool editable) -{ - m_editable = editable; + GetBuffer().Invalidate(wxRICHTEXT_ALL); + Refresh(false); + + return true; } -void wxRichTextCtrl::SetInsertionPoint(long pos) +/// Transform logical to physical +wxPoint wxRichTextCtrl::GetPhysicalPoint(const wxPoint& ptLogical) const { - SelectNone(); + wxPoint pt; + CalcScrolledPosition(ptLogical.x, ptLogical.y, & pt.x, & pt.y); - m_caretPosition = pos - 1; + return pt; } -void wxRichTextCtrl::SetInsertionPointEnd() +/// Transform physical to logical +wxPoint wxRichTextCtrl::GetLogicalPoint(const wxPoint& ptPhysical) const { - long pos = GetLastPosition(); - SetInsertionPoint(pos); -} + wxPoint pt; + CalcUnscrolledPosition(ptPhysical.x, ptPhysical.y, & pt.x, & pt.y); -long wxRichTextCtrl::GetInsertionPoint() const -{ - return m_caretPosition+1; + return pt; } -wxTextPos wxRichTextCtrl::GetLastPosition() const +/// Position the caret +void wxRichTextCtrl::PositionCaret(wxRichTextParagraphLayoutBox* container) { - return GetBuffer().GetRange().GetEnd(); -} + if (!GetCaret()) + return; -// If the return values from and to are the same, there is no -// selection. -void wxRichTextCtrl::GetSelection(long* from, long* to) const -{ - *from = m_selectionRange.GetStart(); - *to = m_selectionRange.GetEnd(); - if ((*to) != -1 && (*to) != -2) - (*to) ++; + //wxLogDebug(wxT("PositionCaret")); + + wxRect caretRect; + if (GetCaretPositionForIndex(GetCaretPosition(), caretRect, container)) + { + wxPoint newPt = caretRect.GetPosition(); + wxSize newSz = caretRect.GetSize(); + wxPoint pt = GetPhysicalPoint(newPt); + if (GetCaret()->GetPosition() != pt || GetCaret()->GetSize() != newSz) + { + GetCaret()->Hide(); + if (GetCaret()->GetSize() != newSz) + GetCaret()->SetSize(newSz); + + // Adjust size so the caret size and position doesn't appear in the margins + if (((pt.y + newSz.y) <= GetBuffer().GetTopMargin()) || (pt.y >= (GetClientSize().y - GetBuffer().GetBottomMargin()))) + { + pt.x = -200; + pt.y = -200; + } + else if (pt.y < GetBuffer().GetTopMargin() && (pt.y + newSz.y) > GetBuffer().GetTopMargin()) + { + newSz.y -= (GetBuffer().GetTopMargin() - pt.y); + if (newSz.y > 0) + { + pt.y = GetBuffer().GetTopMargin(); + GetCaret()->SetSize(newSz); + } + } + else if (pt.y < (GetClientSize().y - GetBuffer().GetBottomMargin()) && (pt.y + newSz.y) > (GetClientSize().y - GetBuffer().GetBottomMargin())) + { + newSz.y = GetClientSize().y - GetBuffer().GetBottomMargin() - pt.y; + GetCaret()->SetSize(newSz); + } + + GetCaret()->Move(pt); + GetCaret()->Show(); + } + } } -bool wxRichTextCtrl::IsEditable() const +/// Get the caret height and position for the given character position +bool wxRichTextCtrl::GetCaretPositionForIndex(long position, wxRect& rect, wxRichTextParagraphLayoutBox* container) { - return m_editable; -} + wxClientDC dc(this); + dc.SetFont(GetFont()); -// ---------------------------------------------------------------------------- -// selection -// ---------------------------------------------------------------------------- + PrepareDC(dc); -void wxRichTextCtrl::SetSelection(long from, long to) -{ - // if from and to are both -1, it means (in wxWidgets) that all text should - // be selected. - if ( (from == -1) && (to == -1) ) + wxPoint pt; + int height = 0; + + if (!container) + container = GetFocusObject(); + + wxRichTextDrawingContext context(& GetBuffer()); + if (container->FindPosition(dc, context, position, pt, & height, m_caretAtLineStart)) { - from = 0; - to = GetLastPosition()+1; + // Caret height can't be zero + if (height == 0) + height = dc.GetCharHeight(); + + rect = wxRect(pt, wxSize(wxRICHTEXT_DEFAULT_CARET_WIDTH, height)); + return true; } - DoSetSelection(from, to); + return false; } -void wxRichTextCtrl::DoSetSelection(long from, long to, bool WXUNUSED(scrollCaret)) +/// Gets the line for the visible caret position. If the caret is +/// shown at the very end of the line, it means the next character is actually +/// on the following line. So let's get the line we're expecting to find +/// if this is the case. +wxRichTextLine* wxRichTextCtrl::GetVisibleLineForCaretPosition(long caretPosition) const { - m_selectionAnchor = from; - m_selectionRange.SetRange(from, to-1); - - Refresh(false); - PositionCaret(); + wxRichTextLine* line = GetFocusObject()->GetLineAtPosition(caretPosition, true); + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(caretPosition, true); + if (line) + { + wxRichTextRange lineRange = line->GetAbsoluteRange(); + if (caretPosition == lineRange.GetStart()-1 && + (para->GetRange().GetStart() != lineRange.GetStart())) + { + if (!m_caretAtLineStart) + line = GetFocusObject()->GetLineAtPosition(caretPosition-1, true); + } + } + return line; } -// ---------------------------------------------------------------------------- -// Editing -// ---------------------------------------------------------------------------- -void wxRichTextCtrl::Replace(long WXUNUSED(from), long WXUNUSED(to), - const wxString& value) +/// Move the caret to the given character position +bool wxRichTextCtrl::MoveCaret(long pos, bool showAtLineStart, wxRichTextParagraphLayoutBox* container) { - BeginBatchUndo(_("Replace")); + if (GetBuffer().IsDirty()) + LayoutContent(); - DeleteSelectedContent(); + if (!container) + container = GetFocusObject(); - DoWriteText(value, SetValue_SelectionOnly); + if (pos <= container->GetOwnRange().GetEnd()) + { + SetCaretPosition(pos, showAtLineStart); - EndBatchUndo(); + PositionCaret(container); + + return true; + } + else + return false; } -void wxRichTextCtrl::Remove(long from, long to) +/// Layout the buffer: which we must do before certain operations, such as +/// setting the caret position. +bool wxRichTextCtrl::LayoutContent(bool onlyVisibleRect) { - SelectNone(); + if (GetBuffer().IsDirty() || onlyVisibleRect) + { + wxRect availableSpace(GetClientSize()); + if (availableSpace.width == 0) + availableSpace.width = 10; + if (availableSpace.height == 0) + availableSpace.height = 10; - GetBuffer().DeleteRangeWithUndo(wxRichTextRange(from, to), - m_caretPosition, // Current caret position - from, // New caret position - this); + int flags = wxRICHTEXT_FIXED_WIDTH|wxRICHTEXT_VARIABLE_HEIGHT; + if (onlyVisibleRect) + { + flags |= wxRICHTEXT_LAYOUT_SPECIFIED_RECT; + availableSpace.SetPosition(GetLogicalPoint(wxPoint(0, 0))); + } - LayoutContent(); - if (!IsFrozen()) - Refresh(false); -} + wxClientDC dc(this); + dc.SetFont(GetFont()); -bool wxRichTextCtrl::IsModified() const -{ - return m_buffer.IsModified(); -} + PrepareDC(dc); -void wxRichTextCtrl::MarkDirty() -{ - m_buffer.Modify(true); -} + wxRichTextDrawingContext context(& GetBuffer()); + GetBuffer().Defragment(); + GetBuffer().UpdateRanges(); // If items were deleted, ranges need recalculation + GetBuffer().Layout(dc, context, availableSpace, availableSpace, flags); + GetBuffer().Invalidate(wxRICHTEXT_NONE); -void wxRichTextCtrl::DiscardEdits() -{ - m_caretPositionForDefaultStyle = -2; - m_buffer.Modify(false); - m_buffer.GetCommandProcessor()->ClearCommands(); + if (!IsFrozen()) + SetupScrollbars(); + } + + return true; } -int wxRichTextCtrl::GetNumberOfLines() const +/// Is all of the selection, or the current caret position, bold? +bool wxRichTextCtrl::IsSelectionBold() { - return GetBuffer().GetParagraphCount(); -} + if (HasSelection()) + { + wxRichTextAttr attr; + wxRichTextRange range = GetSelectionRange(); + attr.SetFlags(wxTEXT_ATTR_FONT_WEIGHT); + attr.SetFontWeight(wxFONTWEIGHT_BOLD); -// ---------------------------------------------------------------------------- -// Positions <-> coords -// ---------------------------------------------------------------------------- + return HasCharacterAttributes(range, attr); + } + else + { + // If no selection, then we need to combine current style with default style + // to see what the effect would be if we started typing. + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_WEIGHT); -long wxRichTextCtrl::XYToPosition(long x, long y) const -{ - return GetBuffer().XYToPosition(x, y); + long pos = GetAdjustedCaretPosition(GetCaretPosition()); + if (GetStyle(pos, attr)) + { + if (IsDefaultStyleShowing()) + wxRichTextApplyStyle(attr, GetDefaultStyleEx()); + return attr.GetFontWeight() == wxFONTWEIGHT_BOLD; + } + } + return false; } -bool wxRichTextCtrl::PositionToXY(long pos, long *x, long *y) const +/// Is all of the selection, or the current caret position, italics? +bool wxRichTextCtrl::IsSelectionItalics() { - return GetBuffer().PositionToXY(pos, x, y); -} + if (HasSelection()) + { + wxRichTextRange range = GetSelectionRange(); + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_ITALIC); + attr.SetFontStyle(wxFONTSTYLE_ITALIC); -// ---------------------------------------------------------------------------- -// -// ---------------------------------------------------------------------------- + return HasCharacterAttributes(range, attr); + } + else + { + // If no selection, then we need to combine current style with default style + // to see what the effect would be if we started typing. + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_ITALIC); -void wxRichTextCtrl::ShowPosition(long pos) -{ - if (!IsPositionVisible(pos)) - ScrollIntoView(pos-1, WXK_DOWN); + long pos = GetAdjustedCaretPosition(GetCaretPosition()); + if (GetStyle(pos, attr)) + { + if (IsDefaultStyleShowing()) + wxRichTextApplyStyle(attr, GetDefaultStyleEx()); + return attr.GetFontStyle() == wxFONTSTYLE_ITALIC; + } + } + return false; } -int wxRichTextCtrl::GetLineLength(long lineNo) const -{ - return GetBuffer().GetParagraphLength(lineNo); +/// Is all of the selection, or the current caret position, underlined? +bool wxRichTextCtrl::IsSelectionUnderlined() +{ + if (HasSelection()) + { + wxRichTextRange range = GetSelectionRange(); + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_UNDERLINE); + attr.SetFontUnderlined(true); + + return HasCharacterAttributes(range, attr); + } + else + { + // If no selection, then we need to combine current style with default style + // to see what the effect would be if we started typing. + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_UNDERLINE); + long pos = GetAdjustedCaretPosition(GetCaretPosition()); + + if (GetStyle(pos, attr)) + { + if (IsDefaultStyleShowing()) + wxRichTextApplyStyle(attr, GetDefaultStyleEx()); + return attr.GetFontUnderlined(); + } + } + return false; } -wxString wxRichTextCtrl::GetLineText(long lineNo) const +/// Does all of the selection, or the current caret position, have this wxTextAttrEffects flag(s)? +bool wxRichTextCtrl::DoesSelectionHaveTextEffectFlag(int flag) { - return GetBuffer().GetParagraphText(lineNo); -} - -// ---------------------------------------------------------------------------- -// Undo/redo -// ---------------------------------------------------------------------------- + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_EFFECTS); + attr.SetTextEffectFlags(flag); + attr.SetTextEffects(flag); -void wxRichTextCtrl::Undo() -{ - if (CanUndo()) + if (HasSelection()) { - GetCommandProcessor()->Undo(); + return HasCharacterAttributes(GetSelectionRange(), attr); + } + else + { + // If no selection, then we need to combine current style with default style + // to see what the effect would be if we started typing. + long pos = GetAdjustedCaretPosition(GetCaretPosition()); + if (GetStyle(pos, attr)) + { + if (IsDefaultStyleShowing()) + wxRichTextApplyStyle(attr, GetDefaultStyleEx()); + return (attr.GetTextEffectFlags() & flag) != 0; + } } + return false; } -void wxRichTextCtrl::Redo() +/// Apply bold to the selection +bool wxRichTextCtrl::ApplyBoldToSelection() { - if (CanRedo()) + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_WEIGHT); + attr.SetFontWeight(IsSelectionBold() ? wxFONTWEIGHT_NORMAL : wxFONTWEIGHT_BOLD); + + if (HasSelection()) + return SetStyleEx(GetSelectionRange(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY); + else { - GetCommandProcessor()->Redo(); + wxRichTextAttr current = GetDefaultStyleEx(); + current.Apply(attr); + SetAndShowDefaultStyle(current); } + return true; } -bool wxRichTextCtrl::CanUndo() const +/// Apply italic to the selection +bool wxRichTextCtrl::ApplyItalicToSelection() { - return GetCommandProcessor()->CanUndo(); -} + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_ITALIC); + attr.SetFontStyle(IsSelectionItalics() ? wxFONTSTYLE_NORMAL : wxFONTSTYLE_ITALIC); -bool wxRichTextCtrl::CanRedo() const -{ - return GetCommandProcessor()->CanRedo(); + if (HasSelection()) + return SetStyleEx(GetSelectionRange(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY); + else + { + wxRichTextAttr current = GetDefaultStyleEx(); + current.Apply(attr); + SetAndShowDefaultStyle(current); + } + return true; } -// ---------------------------------------------------------------------------- -// implementation details -// ---------------------------------------------------------------------------- - -void wxRichTextCtrl::Command(wxCommandEvent& event) +/// Apply underline to the selection +bool wxRichTextCtrl::ApplyUnderlineToSelection() { - SetValue(event.GetString()); - GetEventHandler()->ProcessEvent(event); -} + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_FONT_UNDERLINE); + attr.SetFontUnderlined(!IsSelectionUnderlined()); -void wxRichTextCtrl::OnDropFiles(wxDropFilesEvent& event) -{ - // By default, load the first file into the text window. - if (event.GetNumberOfFiles() > 0) + if (HasSelection()) + return SetStyleEx(GetSelectionRange(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY); + else { - LoadFile(event.GetFiles()[0]); + wxRichTextAttr current = GetDefaultStyleEx(); + current.Apply(attr); + SetAndShowDefaultStyle(current); } + return true; } -wxSize wxRichTextCtrl::DoGetBestSize() const +/// Apply the wxTextAttrEffects flag(s) to the selection, or the current caret position if there's no selection +bool wxRichTextCtrl::ApplyTextEffectToSelection(int flags) { - return wxSize(10, 10); -} - -// ---------------------------------------------------------------------------- -// standard handlers for standard edit menu events -// ---------------------------------------------------------------------------- + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_EFFECTS); + attr.SetTextEffectFlags(flags); + if (!DoesSelectionHaveTextEffectFlag(flags)) + attr.SetTextEffects(flags); + else + attr.SetTextEffects(attr.GetTextEffectFlags() & ~flags); -void wxRichTextCtrl::OnCut(wxCommandEvent& WXUNUSED(event)) -{ - Cut(); + if (HasSelection()) + return SetStyleEx(GetSelectionRange(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY); + else + { + wxRichTextAttr current = GetDefaultStyleEx(); + current.Apply(attr); + SetAndShowDefaultStyle(current); + } + return true; } -void wxRichTextCtrl::OnClear(wxCommandEvent& WXUNUSED(event)) +/// Is all of the selection aligned according to the specified flag? +bool wxRichTextCtrl::IsSelectionAligned(wxTextAttrAlignment alignment) { - DeleteSelection(); -} + wxRichTextRange range; + if (HasSelection()) + range = GetSelectionRange(); + else + range = wxRichTextRange(GetCaretPosition()+1, GetCaretPosition()+2); -void wxRichTextCtrl::OnCopy(wxCommandEvent& WXUNUSED(event)) -{ - Copy(); -} + wxRichTextAttr attr; + attr.SetAlignment(alignment); -void wxRichTextCtrl::OnPaste(wxCommandEvent& WXUNUSED(event)) -{ - Paste(); + return HasParagraphAttributes(range, attr); } -void wxRichTextCtrl::OnUndo(wxCommandEvent& WXUNUSED(event)) +/// Apply alignment to the selection +bool wxRichTextCtrl::ApplyAlignmentToSelection(wxTextAttrAlignment alignment) { - Undo(); + wxRichTextAttr attr; + attr.SetAlignment(alignment); + if (HasSelection()) + return SetStyle(GetSelectionRange(), attr); + else + { + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(GetCaretPosition()+1); + if (para) + return SetStyleEx(para->GetRange().FromInternal(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY); + } + return true; } -void wxRichTextCtrl::OnRedo(wxCommandEvent& WXUNUSED(event)) +/// Apply a named style to the selection +bool wxRichTextCtrl::ApplyStyle(wxRichTextStyleDefinition* def) { - Redo(); -} + // Flags are defined within each definition, so only certain + // attributes are applied. + wxRichTextAttr attr(GetStyleSheet() ? def->GetStyleMergedWithBase(GetStyleSheet()) : def->GetStyle()); -void wxRichTextCtrl::OnUpdateCut(wxUpdateUIEvent& event) -{ - event.Enable( CanCut() ); -} + int flags = wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_RESET; -void wxRichTextCtrl::OnUpdateCopy(wxUpdateUIEvent& event) -{ - event.Enable( CanCopy() ); -} + if (def->IsKindOf(CLASSINFO(wxRichTextListStyleDefinition))) + { + flags |= wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY; -void wxRichTextCtrl::OnUpdateClear(wxUpdateUIEvent& event) -{ - event.Enable( CanDeleteSelection() ); -} + wxRichTextRange range; -void wxRichTextCtrl::OnUpdatePaste(wxUpdateUIEvent& event) -{ - event.Enable( CanPaste() ); -} + if (HasSelection()) + range = GetSelectionRange(); + else + { + long pos = GetAdjustedCaretPosition(GetCaretPosition()); + range = wxRichTextRange(pos, pos+1); + } -void wxRichTextCtrl::OnUpdateUndo(wxUpdateUIEvent& event) -{ - event.Enable( CanUndo() ); - event.SetText( GetCommandProcessor()->GetUndoMenuLabel() ); -} + return SetListStyle(range, (wxRichTextListStyleDefinition*) def, flags); + } -void wxRichTextCtrl::OnUpdateRedo(wxUpdateUIEvent& event) -{ - event.Enable( CanRedo() ); - event.SetText( GetCommandProcessor()->GetRedoMenuLabel() ); -} + bool isPara = false; -void wxRichTextCtrl::OnSelectAll(wxCommandEvent& WXUNUSED(event)) -{ - SelectAll(); -} + // Make sure the attr has the style name + if (def->IsKindOf(CLASSINFO(wxRichTextParagraphStyleDefinition))) + { + isPara = true; + attr.SetParagraphStyleName(def->GetName()); -void wxRichTextCtrl::OnUpdateSelectAll(wxUpdateUIEvent& event) -{ - event.Enable(GetLastPosition() > 0); -} + // If applying a paragraph style, we only want the paragraph nodes to adopt these + // attributes, and not the leaf nodes. This will allow the content (e.g. text) + // to change its style independently. + flags |= wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY; + } + else if (def->IsKindOf(CLASSINFO(wxRichTextCharacterStyleDefinition))) + attr.SetCharacterStyleName(def->GetName()); + else if (def->IsKindOf(CLASSINFO(wxRichTextBoxStyleDefinition))) + attr.GetTextBoxAttr().SetBoxStyleName(def->GetName()); -void wxRichTextCtrl::OnContextMenu(wxContextMenuEvent& WXUNUSED(event)) -{ - if (!m_contextMenu) + if (def->IsKindOf(CLASSINFO(wxRichTextBoxStyleDefinition))) { - m_contextMenu = new wxMenu; - m_contextMenu->Append(wxID_UNDO, _("&Undo")); - m_contextMenu->Append(wxID_REDO, _("&Redo")); - m_contextMenu->AppendSeparator(); - m_contextMenu->Append(wxID_CUT, _("Cu&t")); - m_contextMenu->Append(wxID_COPY, _("&Copy")); - m_contextMenu->Append(wxID_PASTE, _("&Paste")); - m_contextMenu->Append(wxID_CLEAR, _("&Delete")); - m_contextMenu->AppendSeparator(); - m_contextMenu->Append(wxID_SELECTALL, _("Select &All")); + if (GetFocusObject() && (GetFocusObject() != & GetBuffer())) + { + SetStyle(GetFocusObject(), attr); + return true; + } + else + return false; } - PopupMenu(m_contextMenu); - return; -} + else if (HasSelection()) + return SetStyleEx(GetSelectionRange(), attr, flags); + else + { + wxRichTextAttr current = GetDefaultStyleEx(); + wxRichTextAttr defaultStyle(attr); + if (isPara) + { + // Don't apply extra character styles since they are already implied + // in the paragraph style + defaultStyle.SetFlags(defaultStyle.GetFlags() & ~wxTEXT_ATTR_CHARACTER); + } + current.Apply(defaultStyle); + SetAndShowDefaultStyle(current); -bool wxRichTextCtrl::SetStyle(long start, long end, const wxTextAttrEx& style) -{ - return GetBuffer().SetStyle(wxRichTextRange(start, end-1), style); + // If it's a paragraph style, we want to apply the style to the + // current paragraph even if we didn't select any text. + if (isPara) + { + long pos = GetAdjustedCaretPosition(GetCaretPosition()); + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(pos); + if (para) + { + return SetStyleEx(para->GetRange().FromInternal(), attr, flags); + } + } + return true; + } } -bool wxRichTextCtrl::SetStyle(long start, long end, const wxTextAttr& style) +/// Apply the style sheet to the buffer, for example if the styles have changed. +bool wxRichTextCtrl::ApplyStyleSheet(wxRichTextStyleSheet* styleSheet) { - return GetBuffer().SetStyle(wxRichTextRange(start, end-1), wxTextAttrEx(style)); -} + if (!styleSheet) + styleSheet = GetBuffer().GetStyleSheet(); + if (!styleSheet) + return false; -bool wxRichTextCtrl::SetStyle(const wxRichTextRange& range, const wxRichTextAttr& style) -{ - return GetBuffer().SetStyle(range.ToInternal(), style); + if (GetBuffer().ApplyStyleSheet(styleSheet)) + { + GetBuffer().Invalidate(wxRICHTEXT_ALL); + Refresh(false); + return true; + } + else + return false; } -// extended style setting operation with flags including: -// wxRICHTEXT_SETSTYLE_WITH_UNDO, wxRICHTEXT_SETSTYLE_OPTIMIZE, wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY. -// see richtextbuffer.h for more details. -bool wxRichTextCtrl::SetStyleEx(long start, long end, const wxTextAttrEx& style, int flags) +/// Sets the default style to the style under the cursor +bool wxRichTextCtrl::SetDefaultStyleToCursorStyle() { - return GetBuffer().SetStyle(wxRichTextRange(start, end-1), style, flags); -} + wxRichTextAttr attr; + attr.SetFlags(wxTEXT_ATTR_CHARACTER); -bool wxRichTextCtrl::SetStyleEx(const wxRichTextRange& range, const wxTextAttrEx& style, int flags) -{ - return GetBuffer().SetStyle(range.ToInternal(), style, flags); -} + // If at the start of a paragraph, use the next position. + long pos = GetAdjustedCaretPosition(GetCaretPosition()); -bool wxRichTextCtrl::SetStyleEx(const wxRichTextRange& range, const wxRichTextAttr& style, int flags) -{ - return GetBuffer().SetStyle(range.ToInternal(), style, flags); -} + wxRichTextObject* obj = GetFocusObject()->GetLeafObjectAtPosition(pos); + if (obj && obj->IsTopLevel()) + { + // Don't use the attributes of a top-level object, since they might apply + // to content of the object, e.g. background colour. + SetDefaultStyle(wxRichTextAttr()); + return true; + } + else if (GetUncombinedStyle(pos, attr)) + { + SetDefaultStyle(attr); + return true; + } -bool wxRichTextCtrl::SetDefaultStyle(const wxTextAttrEx& style) -{ - return GetBuffer().SetDefaultStyle(style); + return false; } -bool wxRichTextCtrl::SetDefaultStyle(const wxTextAttr& style) +/// Returns the first visible position in the current view +long wxRichTextCtrl::GetFirstVisiblePosition() const { - return GetBuffer().SetDefaultStyle(wxTextAttrEx(style)); + wxRichTextLine* line = GetFocusObject()->GetLineAtYPosition(GetLogicalPoint(wxPoint(0, 0)).y); + if (line) + return line->GetAbsoluteRange().GetStart(); + else + return 0; } -const wxTextAttrEx& wxRichTextCtrl::GetDefaultStyleEx() const +/// Get the first visible point in the window +wxPoint wxRichTextCtrl::GetFirstVisiblePoint() const { - return GetBuffer().GetDefaultStyle(); + int ppuX, ppuY; + int startXUnits, startYUnits; + + GetScrollPixelsPerUnit(& ppuX, & ppuY); + GetViewStart(& startXUnits, & startYUnits); + + return wxPoint(startXUnits * ppuX, startYUnits * ppuY); } -const wxTextAttr& wxRichTextCtrl::GetDefaultStyle() const +/// The adjusted caret position is the character position adjusted to take +/// into account whether we're at the start of a paragraph, in which case +/// style information should be taken from the next position, not current one. +long wxRichTextCtrl::GetAdjustedCaretPosition(long caretPos) const { - return GetBuffer().GetDefaultStyle(); + wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(caretPos+1); + + if (para && (caretPos+1 == para->GetRange().GetStart())) + caretPos ++; + return caretPos; } -bool wxRichTextCtrl::GetStyle(long position, wxTextAttr& style) +/// Get/set the selection range in character positions. -1, -1 means no selection. +/// The range is in API convention, i.e. a single character selection is denoted +/// by (n, n+1) +wxRichTextRange wxRichTextCtrl::GetSelectionRange() const { - wxTextAttrEx attr(style); - if (GetBuffer().GetStyle(position, attr)) - { - style = attr; - return true; - } - else - return false; + wxRichTextRange range = GetInternalSelectionRange(); + if (range != wxRichTextRange(-2,-2) && range != wxRichTextRange(-1,-1)) + range.SetEnd(range.GetEnd() + 1); + return range; } -bool wxRichTextCtrl::GetStyle(long position, wxTextAttrEx& style) +void wxRichTextCtrl::SetSelectionRange(const wxRichTextRange& range) { - return GetBuffer().GetStyle(position, style); + SetSelection(range.GetStart(), range.GetEnd()); } -bool wxRichTextCtrl::GetStyle(long position, wxRichTextAttr& style) +/// Set list style +bool wxRichTextCtrl::SetListStyle(const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int startFrom, int specifiedLevel) { - return GetBuffer().GetStyle(position, style); + return GetFocusObject()->SetListStyle(range.ToInternal(), def, flags, startFrom, specifiedLevel); } -/// Get the content (uncombined) attributes for this position. - -bool wxRichTextCtrl::GetUncombinedStyle(long position, wxTextAttr& style) +bool wxRichTextCtrl::SetListStyle(const wxRichTextRange& range, const wxString& defName, int flags, int startFrom, int specifiedLevel) { - wxTextAttrEx attr(style); - if (GetBuffer().GetUncombinedStyle(position, attr)) - { - style = attr; - return true; - } - else - return false; + return GetFocusObject()->SetListStyle(range.ToInternal(), defName, flags, startFrom, specifiedLevel); } -bool wxRichTextCtrl::GetUncombinedStyle(long position, wxTextAttrEx& style) +/// Clear list for given range +bool wxRichTextCtrl::ClearListStyle(const wxRichTextRange& range, int flags) { - return GetBuffer().GetUncombinedStyle(position, style); + return GetFocusObject()->ClearListStyle(range.ToInternal(), flags); } -bool wxRichTextCtrl::GetUncombinedStyle(long position, wxRichTextAttr& style) +/// Number/renumber any list elements in the given range +bool wxRichTextCtrl::NumberList(const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int startFrom, int specifiedLevel) { - return GetBuffer().GetUncombinedStyle(position, style); + return GetFocusObject()->NumberList(range.ToInternal(), def, flags, startFrom, specifiedLevel); } -/// Set font, and also the buffer attributes -bool wxRichTextCtrl::SetFont(const wxFont& font) +bool wxRichTextCtrl::NumberList(const wxRichTextRange& range, const wxString& defName, int flags, int startFrom, int specifiedLevel) { - wxControl::SetFont(font); - - wxTextAttrEx attr = GetBuffer().GetAttributes(); - attr.SetFont(font); - GetBuffer().SetBasicStyle(attr); - - GetBuffer().Invalidate(wxRICHTEXT_ALL); - Refresh(false); - - return true; + return GetFocusObject()->NumberList(range.ToInternal(), defName, flags, startFrom, specifiedLevel); } -/// Transform logical to physical -wxPoint wxRichTextCtrl::GetPhysicalPoint(const wxPoint& ptLogical) const +/// Promote the list items within the given range. promoteBy can be a positive or negative number, e.g. 1 or -1 +bool wxRichTextCtrl::PromoteList(int promoteBy, const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int specifiedLevel) { - wxPoint pt; - CalcScrolledPosition(ptLogical.x, ptLogical.y, & pt.x, & pt.y); - - return pt; + return GetFocusObject()->PromoteList(promoteBy, range.ToInternal(), def, flags, specifiedLevel); } -/// Transform physical to logical -wxPoint wxRichTextCtrl::GetLogicalPoint(const wxPoint& ptPhysical) const +bool wxRichTextCtrl::PromoteList(int promoteBy, const wxRichTextRange& range, const wxString& defName, int flags, int specifiedLevel) { - wxPoint pt; - CalcUnscrolledPosition(ptPhysical.x, ptPhysical.y, & pt.x, & pt.y); - - return pt; + return GetFocusObject()->PromoteList(promoteBy, range.ToInternal(), defName, flags, specifiedLevel); } -/// Position the caret -void wxRichTextCtrl::PositionCaret() +/// Deletes the content in the given range +bool wxRichTextCtrl::Delete(const wxRichTextRange& range) { - if (!GetCaret()) - return; - - //wxLogDebug(wxT("PositionCaret")); - - wxRect caretRect; - if (GetCaretPositionForIndex(GetCaretPosition(), caretRect)) - { - wxPoint originalPt = caretRect.GetPosition(); - wxPoint pt = GetPhysicalPoint(originalPt); - if (GetCaret()->GetPosition() != pt) - { - GetCaret()->Move(pt); - GetCaret()->SetSize(caretRect.GetSize()); - } - } + return GetFocusObject()->DeleteRangeWithUndo(range.ToInternal(), this, & GetBuffer()); } -/// Get the caret height and position for the given character position -bool wxRichTextCtrl::GetCaretPositionForIndex(long position, wxRect& rect) +const wxArrayString& wxRichTextCtrl::GetAvailableFontNames() { - wxClientDC dc(this); - dc.SetFont(GetFont()); - - PrepareDC(dc); - - wxPoint pt; - int height = 0; - - if (GetBuffer().FindPosition(dc, position, pt, & height, m_caretAtLineStart)) + if (sm_availableFontNames.GetCount() == 0) { - rect = wxRect(pt, wxSize(wxRICHTEXT_DEFAULT_CARET_WIDTH, height)); - return true; + sm_availableFontNames = wxFontEnumerator::GetFacenames(); + sm_availableFontNames.Sort(); } - - return false; + return sm_availableFontNames; } -/// Gets the line for the visible caret position. If the caret is -/// shown at the very end of the line, it means the next character is actually -/// on the following line. So let's get the line we're expecting to find -/// if this is the case. -wxRichTextLine* wxRichTextCtrl::GetVisibleLineForCaretPosition(long caretPosition) const +void wxRichTextCtrl::ClearAvailableFontNames() { - wxRichTextLine* line = GetBuffer().GetLineAtPosition(caretPosition, true); - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(caretPosition, true); - if (line) - { - wxRichTextRange lineRange = line->GetAbsoluteRange(); - if (caretPosition == lineRange.GetStart()-1 && - (para->GetRange().GetStart() != lineRange.GetStart())) - { - if (!m_caretAtLineStart) - line = GetBuffer().GetLineAtPosition(caretPosition-1, true); - } - } - return line; + sm_availableFontNames.Clear(); } - -/// Move the caret to the given character position -bool wxRichTextCtrl::MoveCaret(long pos, bool showAtLineStart) +void wxRichTextCtrl::OnSysColourChanged(wxSysColourChangedEvent& WXUNUSED(event)) { - if (GetBuffer().GetDirty()) - LayoutContent(); + //wxLogDebug(wxT("wxRichTextCtrl::OnSysColourChanged")); - if (pos <= GetBuffer().GetRange().GetEnd()) - { - SetCaretPosition(pos, showAtLineStart); - - PositionCaret(); + wxTextAttrEx basicStyle = GetBasicStyle(); + basicStyle.SetTextColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT)); + SetBasicStyle(basicStyle); + SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); - return true; - } - else - return false; + Refresh(); } -/// Layout the buffer: which we must do before certain operations, such as -/// setting the caret position. -bool wxRichTextCtrl::LayoutContent(bool onlyVisibleRect) +// Refresh the area affected by a selection change +bool wxRichTextCtrl::RefreshForSelectionChange(const wxRichTextSelection& oldSelection, const wxRichTextSelection& newSelection) { - if (GetBuffer().GetDirty() || onlyVisibleRect) + // If the selection is not part of the focus object, or we have multiple ranges, then the chances are that + // the selection contains whole containers rather than just text, so refresh everything + // for now as it would be hard to compute the rectangle bounding all selections. + // TODO: improve on this. + if ((oldSelection.IsValid() && (oldSelection.GetContainer() != GetFocusObject() || oldSelection.GetCount() > 1)) || + (newSelection.IsValid() && (newSelection.GetContainer() != GetFocusObject() || newSelection.GetCount() > 1))) { - wxRect availableSpace(GetClientSize()); - if (availableSpace.width == 0) - availableSpace.width = 10; - if (availableSpace.height == 0) - availableSpace.height = 10; - - int flags = wxRICHTEXT_FIXED_WIDTH|wxRICHTEXT_VARIABLE_HEIGHT; - if (onlyVisibleRect) - { - flags |= wxRICHTEXT_LAYOUT_SPECIFIED_RECT; - availableSpace.SetPosition(GetLogicalPoint(wxPoint(0, 0))); - } - - wxClientDC dc(this); - dc.SetFont(GetFont()); - - PrepareDC(dc); - - GetBuffer().Defragment(); - GetBuffer().UpdateRanges(); // If items were deleted, ranges need recalculation - GetBuffer().Layout(dc, availableSpace, flags); - GetBuffer().SetDirty(false); - - if (!IsFrozen()) - SetupScrollbars(); + Refresh(false); + return true; } - return true; -} + wxRichTextRange oldRange, newRange; + if (oldSelection.IsValid()) + oldRange = oldSelection.GetRange(); + else + oldRange = wxRICHTEXT_NO_SELECTION; + if (newSelection.IsValid()) + newRange = newSelection.GetRange(); + else + newRange = wxRICHTEXT_NO_SELECTION; -/// Is all of the selection bold? -bool wxRichTextCtrl::IsSelectionBold() -{ - if (HasSelection()) + // Calculate the refresh rectangle - just the affected lines + long firstPos, lastPos; + if (oldRange.GetStart() == -2 && newRange.GetStart() != -2) { - wxRichTextAttr attr; - wxRichTextRange range = GetInternalSelectionRange(); - attr.SetFlags(wxTEXT_ATTR_FONT_WEIGHT); - attr.SetFontWeight(wxBOLD); - - return HasCharacterAttributes(range, attr); + firstPos = newRange.GetStart(); + lastPos = newRange.GetEnd(); } - else + else if (oldRange.GetStart() != -2 && newRange.GetStart() == -2) { - // If no selection, then we need to combine current style with default style - // to see what the effect would be if we started typing. - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_WEIGHT); - - long pos = GetAdjustedCaretPosition(GetCaretPosition()); - if (GetStyle(pos, attr)) - { - if (IsDefaultStyleShowing()) - wxRichTextApplyStyle(attr, GetDefaultStyleEx()); - return attr.GetFontWeight() == wxBOLD; - } + firstPos = oldRange.GetStart(); + lastPos = oldRange.GetEnd(); } - return false; -} - -/// Is all of the selection italics? -bool wxRichTextCtrl::IsSelectionItalics() -{ - if (HasSelection()) + else if (oldRange.GetStart() == -2 && newRange.GetStart() == -2) { - wxRichTextRange range = GetInternalSelectionRange(); - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_ITALIC); - attr.SetFontStyle(wxITALIC); - - return HasCharacterAttributes(range, attr); + return false; } else { - // If no selection, then we need to combine current style with default style - // to see what the effect would be if we started typing. - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_ITALIC); - - long pos = GetAdjustedCaretPosition(GetCaretPosition()); - if (GetStyle(pos, attr)) - { - if (IsDefaultStyleShowing()) - wxRichTextApplyStyle(attr, GetDefaultStyleEx()); - return attr.GetFontStyle() == wxITALIC; - } + firstPos = wxMin(oldRange.GetStart(), newRange.GetStart()); + lastPos = wxMax(oldRange.GetEnd(), newRange.GetEnd()); } - return false; -} -/// Is all of the selection underlined? -bool wxRichTextCtrl::IsSelectionUnderlined() -{ - if (HasSelection()) - { - wxRichTextRange range = GetInternalSelectionRange(); - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_UNDERLINE); - attr.SetFontUnderlined(true); + wxRichTextLine* firstLine = GetFocusObject()->GetLineAtPosition(firstPos); + wxRichTextLine* lastLine = GetFocusObject()->GetLineAtPosition(lastPos); - return HasCharacterAttributes(range, attr); - } - else + if (firstLine && lastLine) { - // If no selection, then we need to combine current style with default style - // to see what the effect would be if we started typing. - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_UNDERLINE); - long pos = GetAdjustedCaretPosition(GetCaretPosition()); - - if (GetStyle(pos, attr)) - { - if (IsDefaultStyleShowing()) - wxRichTextApplyStyle(attr, GetDefaultStyleEx()); - return attr.GetFontUnderlined(); - } - } - return false; -} - -/// Apply bold to the selection -bool wxRichTextCtrl::ApplyBoldToSelection() -{ - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_WEIGHT); - attr.SetFontWeight(IsSelectionBold() ? wxNORMAL : wxBOLD); + wxSize clientSize = GetClientSize(); + wxPoint pt1 = GetPhysicalPoint(firstLine->GetAbsolutePosition()); + wxPoint pt2 = GetPhysicalPoint(lastLine->GetAbsolutePosition()) + wxPoint(0, lastLine->GetSize().y); - if (HasSelection()) - return SetStyleEx(GetSelectionRange(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY); + pt1.x = 0; + pt1.y = wxMax(0, pt1.y); + pt2.x = 0; + pt2.y = wxMin(clientSize.y, pt2.y); + + wxRect rect(pt1, wxSize(clientSize.x, pt2.y - pt1.y)); + RefreshRect(rect, false); + } else - SetAndShowDefaultStyle(attr); + Refresh(false); + return true; } -/// Apply italic to the selection -bool wxRichTextCtrl::ApplyItalicToSelection() +// margins functions +bool wxRichTextCtrl::DoSetMargins(const wxPoint& pt) { - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_ITALIC); - attr.SetFontStyle(IsSelectionItalics() ? wxNORMAL : wxITALIC); + GetBuffer().GetAttributes().GetTextBoxAttr().GetMargins().GetLeft().SetValue(pt.x, wxTEXT_ATTR_UNITS_PIXELS); + GetBuffer().GetAttributes().GetTextBoxAttr().GetMargins().GetRight().SetValue(pt.x, wxTEXT_ATTR_UNITS_PIXELS); + GetBuffer().GetAttributes().GetTextBoxAttr().GetMargins().GetTop().SetValue(pt.y, wxTEXT_ATTR_UNITS_PIXELS); + GetBuffer().GetAttributes().GetTextBoxAttr().GetMargins().GetBottom().SetValue(pt.y, wxTEXT_ATTR_UNITS_PIXELS); - if (HasSelection()) - return SetStyleEx(GetSelectionRange(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY); - else - SetAndShowDefaultStyle(attr); return true; } -/// Apply underline to the selection -bool wxRichTextCtrl::ApplyUnderlineToSelection() +wxPoint wxRichTextCtrl::DoGetMargins() const { - wxRichTextAttr attr; - attr.SetFlags(wxTEXT_ATTR_FONT_UNDERLINE); - attr.SetFontUnderlined(!IsSelectionUnderlined()); - - if (HasSelection()) - return SetStyleEx(GetSelectionRange(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY); - else - SetAndShowDefaultStyle(attr); - return true; + return wxPoint(GetBuffer().GetAttributes().GetTextBoxAttr().GetMargins().GetLeft().GetValue(), + GetBuffer().GetAttributes().GetTextBoxAttr().GetMargins().GetTop().GetValue()); } -/// Is all of the selection aligned according to the specified flag? -bool wxRichTextCtrl::IsSelectionAligned(wxTextAttrAlignment alignment) +bool wxRichTextCtrl::SetFocusObject(wxRichTextParagraphLayoutBox* obj, bool setCaretPosition) { - wxRichTextRange range; - if (HasSelection()) - range = GetInternalSelectionRange(); - else - range = wxRichTextRange(GetCaretPosition()+1, GetCaretPosition()+1); + if (obj && !obj->AcceptsFocus()) + return false; - wxRichTextAttr attr; - attr.SetAlignment(alignment); + wxRichTextParagraphLayoutBox* oldContainer = GetFocusObject(); + bool changingContainer = (m_focusObject != obj); - return HasParagraphAttributes(range, attr); -} + if (changingContainer && HasSelection()) + SelectNone(); -/// Apply alignment to the selection -bool wxRichTextCtrl::ApplyAlignmentToSelection(wxTextAttrAlignment alignment) -{ - wxRichTextAttr attr; - attr.SetAlignment(alignment); - if (HasSelection()) - return SetStyle(GetSelectionRange(), attr); - else + m_focusObject = obj; + + if (!obj) + m_focusObject = & m_buffer; + + if (setCaretPosition && changingContainer) { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(GetCaretPosition()+1); - if (para) - return SetStyleEx(para->GetRange().FromInternal(), attr, wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE|wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY); + m_selection.Reset(); + m_selectionAnchor = -2; + m_selectionAnchorObject = NULL; + m_selectionState = wxRichTextCtrlSelectionState_Normal; + + long pos = -1; + + m_caretAtLineStart = false; + MoveCaret(pos, m_caretAtLineStart); + SetDefaultStyleToCursorStyle(); + + wxRichTextEvent cmdEvent( + wxEVT_COMMAND_RICHTEXT_FOCUS_OBJECT_CHANGED, + GetId()); + cmdEvent.SetEventObject(this); + cmdEvent.SetPosition(m_caretPosition+1); + cmdEvent.SetOldContainer(oldContainer); + cmdEvent.SetContainer(m_focusObject); + + GetEventHandler()->ProcessEvent(cmdEvent); } return true; } -/// Apply a named style to the selection -bool wxRichTextCtrl::ApplyStyle(wxRichTextStyleDefinition* def) +#if wxUSE_DRAG_AND_DROP +void wxRichTextCtrl::OnDrop(wxCoord WXUNUSED(x), wxCoord WXUNUSED(y), wxDragResult def, wxDataObject* DataObj) { - // Flags are defined within each definition, so only certain - // attributes are applied. - wxRichTextAttr attr(def->GetStyle()); + m_preDrag = false; - int flags = wxRICHTEXT_SETSTYLE_WITH_UNDO|wxRICHTEXT_SETSTYLE_OPTIMIZE; + if ((def != wxDragCopy) && (def != wxDragMove)) + { + return; + } - if (def->IsKindOf(CLASSINFO(wxRichTextListStyleDefinition))) + if (!GetSelection().IsValid()) { - flags |= wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY; + return; + } - wxRichTextRange range; + wxRichTextParagraphLayoutBox* originContainer = GetSelection().GetContainer(); + wxRichTextParagraphLayoutBox* destContainer = GetFocusObject(); // This will be the drop container, not necessarily the same as the origin one - if (HasSelection()) - range = GetSelectionRange(); - else + wxRichTextBuffer* richTextBuffer = ((wxRichTextBufferDataObject*)DataObj)->GetRichTextBuffer(); + if (richTextBuffer) + { + long position = GetCaretPosition(); + wxRichTextRange selectionrange = GetInternalSelectionRange(); + if (selectionrange.Contains(position) && (def == wxDragMove)) { - long pos = GetAdjustedCaretPosition(GetCaretPosition()); - range = wxRichTextRange(pos, pos+1); + // It doesn't make sense to move onto itself + return; } - return SetListStyle(range, (wxRichTextListStyleDefinition*) def, flags); - } + // If we're moving, and the data is being moved forward, we need to drop first, then delete the selection + // If moving backwards, we need to delete then drop. If we're copying (or doing nothing) we don't delete anyway + bool DeleteAfter = (def == wxDragMove) && (position > selectionrange.GetEnd()); + if ((def == wxDragMove) && !DeleteAfter) + { + // We can't use e.g. DeleteSelectedContent() as it uses the focus container + originContainer->DeleteRangeWithUndo(selectionrange, this, &GetBuffer()); + } - // Make sure the attr has the style name - if (def->IsKindOf(CLASSINFO(wxRichTextParagraphStyleDefinition))) - { - attr.SetParagraphStyleName(def->GetName()); + destContainer->InsertParagraphsWithUndo(&GetBuffer(), position+1, *richTextBuffer, this, 0); + ShowPosition(position + richTextBuffer->GetOwnRange().GetEnd()); - // If applying a paragraph style, we only want the paragraph nodes to adopt these - // attributes, and not the leaf nodes. This will allow the content (e.g. text) - // to change its style independently. - flags |= wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY; - } - else - attr.SetCharacterStyleName(def->GetName()); + delete richTextBuffer; - if (HasSelection()) - return SetStyleEx(GetSelectionRange(), attr, flags); - else - { - SetAndShowDefaultStyle(attr); - return true; + if (DeleteAfter) + { + // We can't use e.g. DeleteSelectedContent() as it uses the focus container + originContainer->DeleteRangeWithUndo(selectionrange, this, &GetBuffer()); + } + + SelectNone(); + Refresh(); } } +#endif // wxUSE_DRAG_AND_DROP -/// Apply the style sheet to the buffer, for example if the styles have changed. -bool wxRichTextCtrl::ApplyStyleSheet(wxRichTextStyleSheet* styleSheet) + +#if wxUSE_DRAG_AND_DROP +bool wxRichTextDropSource::GiveFeedback(wxDragResult WXUNUSED(effect)) { - if (!styleSheet) - styleSheet = GetBuffer().GetStyleSheet(); - if (!styleSheet) - return false; + wxCHECK_MSG(m_rtc, false, wxT("NULL m_rtc")); - if (GetBuffer().ApplyStyleSheet(styleSheet)) + long position = 0; + int hit = 0; + wxRichTextObject* hitObj = NULL; + wxRichTextParagraphLayoutBox* container = m_rtc->FindContainerAtPoint(m_rtc->ScreenToClient(wxGetMousePosition()), position, hit, hitObj); + + if (!(hit & wxRICHTEXT_HITTEST_NONE) && container && container->AcceptsFocus()) { - GetBuffer().Invalidate(wxRICHTEXT_ALL); - Refresh(false); - return true; + m_rtc->StoreFocusObject(container); + m_rtc->SetCaretPositionAfterClick(container, position, hit); } - else - return false; + + return false; // so that the base-class sets a cursor } +#endif // wxUSE_DRAG_AND_DROP -/// Sets the default style to the style under the cursor -bool wxRichTextCtrl::SetDefaultStyleToCursorStyle() +bool wxRichTextCtrl::CanDeleteRange(wxRichTextParagraphLayoutBox& WXUNUSED(container), const wxRichTextRange& WXUNUSED(range)) const { - wxTextAttrEx attr; - attr.SetFlags(wxTEXT_ATTR_CHARACTER); + return true; +} - // If at the start of a paragraph, use the next position. - long pos = GetAdjustedCaretPosition(GetCaretPosition()); +bool wxRichTextCtrl::CanInsertContent(wxRichTextParagraphLayoutBox& WXUNUSED(container), long WXUNUSED(pos)) const +{ + return true; +} -#if wxRICHTEXT_USE_DYNAMIC_STYLES - if (GetUncombinedStyle(pos, attr)) -#else - if (GetStyle(pos, attr)) -#endif - { - SetDefaultStyle(attr); - return true; - } +void wxRichTextCtrl::EnableVerticalScrollbar(bool enable) +{ + m_verticalScrollbarEnabled = enable; + SetupScrollbars(); +} - return false; + +#if wxRICHTEXT_USE_OWN_CARET + +// ---------------------------------------------------------------------------- +// initialization and destruction +// ---------------------------------------------------------------------------- + +void wxRichTextCaret::Init() +{ + m_hasFocus = true; + m_refreshEnabled = true; + + m_xOld = + m_yOld = -1; + m_richTextCtrl = NULL; + m_needsUpdate = false; + m_flashOn = true; } -/// Returns the first visible position in the current view -long wxRichTextCtrl::GetFirstVisiblePosition() const +wxRichTextCaret::~wxRichTextCaret() { - wxRichTextLine* line = GetBuffer().GetLineAtYPosition(GetLogicalPoint(wxPoint(0, 0)).y); - if (line) - return line->GetAbsoluteRange().GetStart(); - else - return 0; + if (m_timer.IsRunning()) + m_timer.Stop(); } -/// The adjusted caret position is the character position adjusted to take -/// into account whether we're at the start of a paragraph, in which case -/// style information should be taken from the next position, not current one. -long wxRichTextCtrl::GetAdjustedCaretPosition(long caretPos) const +// ---------------------------------------------------------------------------- +// showing/hiding/moving the caret (base class interface) +// ---------------------------------------------------------------------------- + +void wxRichTextCaret::DoShow() { - wxRichTextParagraph* para = GetBuffer().GetParagraphAtPosition(caretPos+1); + m_flashOn = true; - if (para && (caretPos+1 == para->GetRange().GetStart())) - caretPos ++; - return caretPos; + if (!m_timer.IsRunning()) + m_timer.Start(GetBlinkTime()); + + Refresh(); } -/// Get/set the selection range in character positions. -1, -1 means no selection. -/// The range is in API convention, i.e. a single character selection is denoted -/// by (n, n+1) -wxRichTextRange wxRichTextCtrl::GetSelectionRange() const +void wxRichTextCaret::DoHide() { - wxRichTextRange range = GetInternalSelectionRange(); - if (range != wxRichTextRange(-2,-2) && range != wxRichTextRange(-1,-1)) - range.SetEnd(range.GetEnd() + 1); - return range; + if (m_timer.IsRunning()) + m_timer.Stop(); + + Refresh(); } -void wxRichTextCtrl::SetSelectionRange(const wxRichTextRange& range) +void wxRichTextCaret::DoMove() { - wxRichTextRange range1(range); - if (range1 != wxRichTextRange(-2,-2) && range1 != wxRichTextRange(-1,-1) ) - range1.SetEnd(range1.GetEnd() - 1); + if (IsVisible()) + { + Refresh(); - wxASSERT( range1.GetStart() > range1.GetEnd() ); + if (m_xOld != -1 && m_yOld != -1) + { + if (m_richTextCtrl && m_refreshEnabled) + { + wxRect rect(GetPosition(), GetSize()); + m_richTextCtrl->RefreshRect(rect, false); + } + } + } - SetInternalSelectionRange(range1); + m_xOld = m_x; + m_yOld = m_y; } -/// Set list style -bool wxRichTextCtrl::SetListStyle(const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int startFrom, int specifiedLevel) +void wxRichTextCaret::DoSize() { - return GetBuffer().SetListStyle(range.ToInternal(), def, flags, startFrom, specifiedLevel); + int countVisible = m_countVisible; + if (countVisible > 0) + { + m_countVisible = 0; + DoHide(); + } + + if (countVisible > 0) + { + m_countVisible = countVisible; + DoShow(); + } } -bool wxRichTextCtrl::SetListStyle(const wxRichTextRange& range, const wxString& defName, int flags, int startFrom, int specifiedLevel) +// ---------------------------------------------------------------------------- +// handling the focus +// ---------------------------------------------------------------------------- + +void wxRichTextCaret::OnSetFocus() { - return GetBuffer().SetListStyle(range.ToInternal(), defName, flags, startFrom, specifiedLevel); + m_hasFocus = true; + + if ( IsVisible() ) + Refresh(); } -/// Clear list for given range -bool wxRichTextCtrl::ClearListStyle(const wxRichTextRange& range, int flags) +void wxRichTextCaret::OnKillFocus() { - return GetBuffer().ClearListStyle(range.ToInternal(), flags); + m_hasFocus = false; } -/// Number/renumber any list elements in the given range -bool wxRichTextCtrl::NumberList(const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int startFrom, int specifiedLevel) +// ---------------------------------------------------------------------------- +// drawing the caret +// ---------------------------------------------------------------------------- + +void wxRichTextCaret::Refresh() { - return GetBuffer().NumberList(range.ToInternal(), def, flags, startFrom, specifiedLevel); + if (m_richTextCtrl && m_refreshEnabled) + { + wxRect rect(GetPosition(), GetSize()); + m_richTextCtrl->RefreshRect(rect, false); + } } -bool wxRichTextCtrl::NumberList(const wxRichTextRange& range, const wxString& defName, int flags, int startFrom, int specifiedLevel) +void wxRichTextCaret::DoDraw(wxDC *dc) { - return GetBuffer().NumberList(range.ToInternal(), defName, flags, startFrom, specifiedLevel); + dc->SetPen( *wxBLACK_PEN ); + + dc->SetBrush(*(m_hasFocus ? wxBLACK_BRUSH : wxTRANSPARENT_BRUSH)); + dc->SetPen(*wxBLACK_PEN); + + wxPoint pt(m_x, m_y); + + if (m_richTextCtrl) + { + pt = m_richTextCtrl->GetLogicalPoint(pt); + } + if (IsVisible() && m_flashOn) + dc->DrawRectangle(pt.x, pt.y, m_width, m_height); } -/// Promote the list items within the given range. promoteBy can be a positive or negative number, e.g. 1 or -1 -bool wxRichTextCtrl::PromoteList(int promoteBy, const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int specifiedLevel) +void wxRichTextCaret::Notify() { - return GetBuffer().PromoteList(promoteBy, range.ToInternal(), def, flags, specifiedLevel); + m_flashOn = !m_flashOn; + Refresh(); } -bool wxRichTextCtrl::PromoteList(int promoteBy, const wxRichTextRange& range, const wxString& defName, int flags, int specifiedLevel) +void wxRichTextCaretTimer::Notify() { - return GetBuffer().PromoteList(promoteBy, range.ToInternal(), defName, flags, specifiedLevel); + m_caret->Notify(); } +#endif + // wxRICHTEXT_USE_OWN_CARET -const wxArrayString& wxRichTextCtrl::GetAvailableFontNames() +// Add an item +bool wxRichTextContextMenuPropertiesInfo::AddItem(const wxString& label, wxRichTextObject* obj) { - if (sm_availableFontNames.GetCount() == 0) + if (GetCount() < 3) { - sm_availableFontNames = wxFontEnumerator::GetFacenames(); - sm_availableFontNames.Sort(); + m_labels.Add(label); + m_objects.Add(obj); + return true; } - return sm_availableFontNames; + else + return false; } -void wxRichTextCtrl::ClearAvailableFontNames() +// Returns number of menu items were added. +int wxRichTextContextMenuPropertiesInfo::AddMenuItems(wxMenu* menu, int startCmd) const { - sm_availableFontNames.Clear(); + wxMenuItem* item = menu->FindItem(startCmd); + // If none of the standard properties identifiers are in the menu, add them if necessary. + // If no items to add, just set the text to something generic + if (GetCount() == 0) + { + if (item) + { + menu->SetLabel(startCmd, _("&Properties")); + + // Delete the others if necessary + int i; + for (i = startCmd+1; i < startCmd+3; i++) + { + if (menu->FindItem(i)) + { + menu->Delete(i); + } + } + } + } + else + { + int i; + int pos = -1; + // Find the position of the first properties item + for (i = 0; i < (int) menu->GetMenuItemCount(); i++) + { + wxMenuItem* item = menu->FindItemByPosition(i); + if (item && item->GetId() == startCmd) + { + pos = i; + break; + } + } + + if (pos != -1) + { + int insertBefore = pos+1; + for (i = startCmd; i < startCmd+GetCount(); i++) + { + if (menu->FindItem(i)) + { + menu->SetLabel(i, m_labels[i - startCmd]); + } + else + { + if (insertBefore >= (int) menu->GetMenuItemCount()) + menu->Append(i, m_labels[i - startCmd]); + else + menu->Insert(insertBefore, i, m_labels[i - startCmd]); + } + insertBefore ++; + } + + // Delete any old items still left on the menu + for (i = startCmd + GetCount(); i < startCmd+3; i++) + { + if (menu->FindItem(i)) + { + menu->Delete(i); + } + } + } + else + { + // No existing property identifiers were found, so append to the end of the menu. + menu->AppendSeparator(); + for (i = startCmd; i < startCmd+GetCount(); i++) + { + menu->Append(i, m_labels[i - startCmd]); + } + } + } + + return GetCount(); +} + +// Add appropriate menu items for the current container and clicked on object +// (and container's parent, if appropriate). +int wxRichTextContextMenuPropertiesInfo::AddItems(wxRichTextCtrl* ctrl, wxRichTextObject* container, wxRichTextObject* obj) +{ + Clear(); + if (obj && ctrl->CanEditProperties(obj)) + AddItem(ctrl->GetPropertiesMenuLabel(obj), obj); + + if (container && container != obj && ctrl->CanEditProperties(container) && m_labels.Index(ctrl->GetPropertiesMenuLabel(container)) == wxNOT_FOUND) + AddItem(ctrl->GetPropertiesMenuLabel(container), container); + + if (container && container->GetParent() && ctrl->CanEditProperties(container->GetParent()) && m_labels.Index(ctrl->GetPropertiesMenuLabel(container->GetParent())) == wxNOT_FOUND) + AddItem(ctrl->GetPropertiesMenuLabel(container->GetParent()), container->GetParent()); + + return GetCount(); } #endif