// Author: Julian Smart
// Modified by:
// Created: 2005-09-30
-// RCS-ID: $Id$
// Copyright: (c) Julian Smart
// Licence: wxWindows licence
/////////////////////////////////////////////////////////////////////////////
#include "wx/app.h"
WX_CHECK_BUILD_OPTIONS("wxRichTextCtrl")
-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 );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_LEFT_CLICK, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_MIDDLE_CLICK, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_RIGHT_CLICK, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_LEFT_DCLICK, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_RETURN, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_CHARACTER, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_DELETE, wxRichTextEvent );
+
+wxDEFINE_EVENT( wxEVT_RICHTEXT_STYLESHEET_REPLACING, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_STYLESHEET_REPLACED, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_STYLESHEET_CHANGING, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_STYLESHEET_CHANGED, wxRichTextEvent );
+
+wxDEFINE_EVENT( wxEVT_RICHTEXT_CONTENT_INSERTED, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_CONTENT_DELETED, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_STYLE_CHANGED, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_PROPERTIES_CHANGED, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_SELECTION_CHANGED, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_BUFFER_RESET, wxRichTextEvent );
+wxDEFINE_EVENT( wxEVT_RICHTEXT_FOCUS_OBJECT_CHANGED, wxRichTextEvent );
#if wxRICHTEXT_USE_OWN_CARET
wxRichTextCaretTimer m_timer;
wxRichTextCtrl* m_richTextCtrl;
bool m_refreshEnabled;
+ wxPen m_caretPen;
+ wxBrush m_caretBrush;
};
#endif
m_selectionAnchorObject = NULL;
m_selectionState = wxRichTextCtrlSelectionState_Normal;
m_editable = true;
+ m_useVirtualAttributes = false;
m_verticalScrollbarEnabled = true;
m_caretAtLineStart = false;
m_dragging = false;
dc.SetFont(GetFont());
- // Paint the background
- PaintBackground(dc);
-
wxRect drawingArea(GetUpdateRegion().GetBox());
drawingArea.SetPosition(GetUnscaledPoint(GetLogicalPoint(drawingArea.GetPosition())));
drawingArea.SetSize(GetUnscaledSize(drawingArea.GetSize()));
SetupScrollbars();
}
+ // Paint the background
+ PaintBackground(dc);
+
wxRect clipRect(availableSpace);
clipRect.x += GetBuffer().GetLeftMargin();
clipRect.y += GetBuffer().GetTopMargin();
if ((hit != wxRICHTEXT_HITTEST_NONE) && !(hit & wxRICHTEXT_HITTEST_OUTSIDE))
{
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_LEFT_CLICK,
+ wxEVT_RICHTEXT_LEFT_CLICK,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetPosition(position);
/// Mouse-movements
void wxRichTextCtrl::OnMoveMouse(wxMouseEvent& event)
{
+ if (!event.Dragging() && m_dragging)
+ {
+ // We may have accidentally lost a mouse-up event, especially on Linux
+ m_dragging = false;
+ if (GetCapture() == this)
+ ReleaseMouse();
+ }
+
#if wxUSE_DRAG_AND_DROP
size_t distance = 0;
if (m_preDrag || m_dragging)
}
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_RIGHT_CLICK,
+ wxEVT_RICHTEXT_RIGHT_CLICK,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetPosition(position);
void wxRichTextCtrl::OnLeftDClick(wxMouseEvent& WXUNUSED(event))
{
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_LEFT_DCLICK,
+ wxEVT_RICHTEXT_LEFT_DCLICK,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetPosition(m_caretPosition+1);
void wxRichTextCtrl::OnMiddleClick(wxMouseEvent& event)
{
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_MIDDLE_CLICK,
+ wxEVT_RICHTEXT_MIDDLE_CLICK,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetPosition(m_caretPosition+1);
// Must process this before translation, otherwise it's translated into a WXK_DELETE event.
if (event.CmdDown() && event.GetKeyCode() == WXK_BACK)
{
- if (!IsEditable())
- {
+ if (!ProcessBackKey(event, flags))
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())
- {
- 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 (!processed)
- {
- wxRichTextRange range(newPos, newPos);
- if (CanDeleteRange(* GetFocusObject(), range.FromInternal()))
- {
- GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer());
- deletions ++;
- }
- }
- }
-
- EndBatchUndo();
-
- if (GetLastPosition() == -1)
- {
- GetFocusObject()->Reset();
-
- m_caretPosition = -1;
- PositionCaret();
- SetDefaultStyleToCursorStyle();
- }
-
- ScrollIntoView(m_caretPosition, WXK_LEFT);
-
- // Always send this event; wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion.
- {
- 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();
else
GetFocusObject()->InsertNewlineWithUndo(& GetBuffer(), newPos+1, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE|wxRICHTEXT_INSERT_INTERACTIVE);
+ // Automatically renumber list
+ bool isNumberedList = false;
+ wxRichTextRange numberedListRange = FindRangeForList(newPos+1, isNumberedList);
+ if (isNumberedList && numberedListRange != wxRichTextRange(-1, -1))
+ {
+ NumberList(numberedListRange, NULL, wxRICHTEXT_SETSTYLE_RENUMBER|wxRICHTEXT_SETSTYLE_WITH_UNDO);
+ }
+
EndBatchUndo();
SetDefaultStyleToCursorStyle();
ScrollIntoView(m_caretPosition, WXK_RIGHT);
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_RETURN,
+ wxEVT_RICHTEXT_RETURN,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetFlags(flags);
if (!GetEventHandler()->ProcessEvent(cmdEvent))
{
// Generate conventional event
- wxCommandEvent textEvent(wxEVT_COMMAND_TEXT_ENTER, GetId());
+ wxCommandEvent textEvent(wxEVT_TEXT_ENTER, GetId());
InitCommandEvent(textEvent);
GetEventHandler()->ProcessEvent(textEvent);
}
else if (event.GetKeyCode() == WXK_BACK)
{
- long newPos = m_caretPosition;
-
- if (HasSelection() && !CanDeleteRange(* GetFocusObject(), GetSelectionRange()))
- {
- return;
- }
-
- BeginBatchUndo(_("Delete Text"));
-
- 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())
- {
- 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 (!processed)
- {
- wxRichTextRange range(newPos, newPos);
- if (CanDeleteRange(* GetFocusObject(), range.FromInternal()))
- {
- GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer());
- deletions ++;
- }
- }
- }
-
- EndBatchUndo();
-
- if (GetLastPosition() == -1)
- {
- GetFocusObject()->Reset();
-
- m_caretPosition = -1;
- PositionCaret();
- SetDefaultStyleToCursorStyle();
- }
-
- ScrollIntoView(m_caretPosition, WXK_LEFT);
-
- // Always send this event; wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion.
- {
- 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();
+ ProcessBackKey(event, flags);
}
else if (event.GetKeyCode() == WXK_DELETE)
{
ScrollIntoView(m_caretPosition, WXK_LEFT);
- // Always send this event; wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion.
+ // Always send this event; wxEVT_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion.
{
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_DELETE,
+ wxEVT_RICHTEXT_DELETE,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetFlags(flags);
}
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_CHARACTER,
+ wxEVT_RICHTEXT_CHARACTER,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetFlags(flags);
return false;
}
+// Processes the back key
+bool wxRichTextCtrl::ProcessBackKey(wxKeyEvent& event, int flags)
+{
+ if (!IsEditable())
+ {
+ return false;
+ }
+
+ if (HasSelection() && !CanDeleteRange(* GetFocusObject(), GetSelectionRange()))
+ {
+ return false;
+ }
+
+ wxRichTextParagraph* para = GetFocusObject()->GetParagraphAtPosition(m_caretPosition, true);
+
+ // If we're at the start of a list item with a bullet, let's 'delete' the bullet, i.e.
+ // make it a continuation paragraph.
+ if (!HasSelection() && para && ((m_caretPosition+1) == para->GetRange().GetStart()) &&
+ para->GetAttributes().HasBulletStyle() && (para->GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_CONTINUATION) == 0)
+ {
+ wxRichTextParagraph* newPara = wxDynamicCast(para->Clone(), wxRichTextParagraph);
+ newPara->GetAttributes().SetBulletStyle(newPara->GetAttributes().GetBulletStyle() | wxTEXT_ATTR_BULLET_STYLE_CONTINUATION);
+
+ wxRichTextAction* action = new wxRichTextAction(NULL, _("Remove Bullet"), wxRICHTEXT_CHANGE_STYLE, & GetBuffer(), GetFocusObject(), this);
+ action->SetRange(newPara->GetRange());
+ action->SetPosition(GetCaretPosition());
+ action->GetNewParagraphs().AppendChild(newPara);
+ // Also store the old ones for Undo
+ action->GetOldParagraphs().AppendChild(new wxRichTextParagraph(*para));
+
+ GetBuffer().Invalidate(para->GetRange());
+ GetBuffer().SubmitAction(action);
+
+ // Automatically renumber list
+ bool isNumberedList = false;
+ wxRichTextRange numberedListRange = FindRangeForList(m_caretPosition, isNumberedList);
+ if (isNumberedList && numberedListRange != wxRichTextRange(-1, -1))
+ {
+ NumberList(numberedListRange, NULL, wxRICHTEXT_SETSTYLE_RENUMBER|wxRICHTEXT_SETSTYLE_WITH_UNDO);
+ }
+
+ Update();
+ }
+ else
+ {
+ 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())
+ {
+ 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 (!processed)
+ {
+ wxRichTextRange range(newPos, newPos);
+ if (CanDeleteRange(* GetFocusObject(), range.FromInternal()))
+ {
+ GetFocusObject()->DeleteRangeWithUndo(range, this, & GetBuffer());
+ deletions ++;
+ }
+ }
+ }
+
+ EndBatchUndo();
+
+ if (GetLastPosition() == -1)
+ {
+ GetFocusObject()->Reset();
+
+ m_caretPosition = -1;
+ PositionCaret();
+ SetDefaultStyleToCursorStyle();
+ }
+
+ ScrollIntoView(m_caretPosition, WXK_LEFT);
+
+ // Always send this event; wxEVT_RICHTEXT_CONTENT_DELETED will be sent only if there is an actual deletion.
+ {
+ wxRichTextEvent cmdEvent(
+ wxEVT_RICHTEXT_DELETE,
+ GetId());
+ cmdEvent.SetEventObject(this);
+ cmdEvent.SetFlags(flags);
+ cmdEvent.SetPosition(m_caretPosition+1);
+ cmdEvent.SetContainer(GetFocusObject());
+ GetEventHandler()->ProcessEvent(cmdEvent);
+ }
+
+ Update();
+ }
+
+ return true;
+}
+
/// Delete content if there is a selection, e.g. when pressing a key.
bool wxRichTextCtrl::DeleteSelectedContent(long* newPos)
{
// 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)
+ if (noPositions == 1)
MoveCaretForward(oldPos);
else
SetCaretPosition(newPos);
if (!extendSel)
SelectNone();
- if (noPositions == 1 && !extendSel)
+ if (noPositions == 1)
MoveCaretBack(oldPos);
else
SetCaretPosition(newPos);
wxRichTextParagraphLayoutBox* container = GetFocusObject();
int hitTestFlags = wxRICHTEXT_HITTEST_NO_NESTED_OBJECTS|wxRICHTEXT_HITTEST_NO_FLOATING_OBJECTS|wxRICHTEXT_HITTEST_HONOUR_ATOMIC;
+ bool lineIsEmpty = false;
if (notInThisObject)
{
// If we know we're navigating out of the current object,
{
wxRichTextLine* lineObj = GetFocusObject()->GetLineForVisibleLineNumber(newLine);
if (lineObj)
+ {
pt.y = lineObj->GetAbsolutePosition().y + 2;
+ if (lineObj->GetRange().GetStart() == lineObj->GetRange().GetEnd())
+ lineIsEmpty = true;
+ }
else
return false;
}
}
bool caretLineStart = true;
+
+ // If the line is empty, there is only one possible position for the caret,
+ // so force the 'before' state so FindCaretPositionForCharacterPosition doesn't
+ // just return the same position.
+ if (lineIsEmpty)
+ {
+ hitTest &= ~wxRICHTEXT_HITTEST_AFTER;
+ hitTest |= wxRICHTEXT_HITTEST_BEFORE;
+ }
long caretPosition = FindCaretPositionForCharacterPosition(newPos, hitTest, container, caretLineStart);
long newSelEnd = caretPosition;
bool extendSel;
if (!extendSel)
SelectNone();
- SetCaretPosition(newPos);
+ SetCaretPosition(newPos, true);
PositionCaret();
SetDefaultStyleToCursorStyle();
// ----------------------------------------------------------------------------
// file IO functions
// ----------------------------------------------------------------------------
-
+#if wxUSE_FFILE && wxUSE_STREAMS
bool wxRichTextCtrl::DoLoadFile(const wxString& filename, int fileType)
{
+ SetFocusObject(& GetBuffer(), true);
+
bool success = GetBuffer().LoadFile(filename, (wxRichTextFileType)fileType);
if (success)
m_filename = filename;
return false;
}
+#endif // wxUSE_FFILE && wxUSE_STREAMS
// ----------------------------------------------------------------------------
// wxRichTextCtrl specific functionality
// selection and ranges
// ----------------------------------------------------------------------------
-void wxRichTextCtrl::SelectAll()
-{
- SetSelection(-1, -1);
-}
-
/// Select none
void wxRichTextCtrl::SelectNone()
{
wxString valueUnix = wxTextFile::Translate(value, wxTextFileType_Unix);
GetFocusObject()->InsertTextWithUndo(& GetBuffer(), m_caretPosition+1, valueUnix, this, wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE);
- GetBuffer().Defragment();
+ wxRichTextDrawingContext context(& GetBuffer());
+ GetBuffer().Defragment(context);
if ( flags & SetValue_SendEvent )
wxTextCtrl::SendTextUpdatedEvent(this);
textBox->AddParagraph(wxEmptyString);
textBox->SetParent(NULL);
+ // If the box has an invalid foreground colour, its text will mimic any upstream value (see #15224)
+ if (!textBox->GetAttributes().GetTextColour().IsOk())
+ {
+ textBox->GetAttributes().SetTextColour(GetBasicStyle().GetTextColour());
+ }
+
// 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);
wxRichTextTable* table = new wxRichTextTable;
table->SetAttributes(tableAttr);
table->SetParent(& GetBuffer()); // set parent temporarily for AddParagraph to use correct style
+ table->SetBasicStyle(GetBasicStyle());
table->CreateTable(rows, cols);
table->SetParent(NULL);
+ // If cells have an invalid foreground colour, their text will mimic any upstream value (see #15224)
+ wxRichTextAttr attr = cellAttr;
+ if (!attr.GetTextColour().IsOk())
+ {
+ attr.SetTextColour(GetBasicStyle().GetTextColour());
+ }
+
int i, j;
for (j = 0; j < rows; j++)
{
for (i = 0; i < cols; i++)
{
- table->GetCell(j, i)->GetAttributes() = cellAttr;
+ table->GetCell(j, i)->GetAttributes() = attr;
}
}
SelectNone();
m_caretPosition = pos - 1;
+ m_caretAtLineStart = true;
PositionCaret();
SetSelection(from, to);
- wxRichTextAttr attr = GetDefaultStyle();
+ wxRichTextAttr attr(GetDefaultStyle());
DeleteSelectedContent();
SetDefaultStyle(attr);
- DoWriteText(value, SetValue_SelectionOnly);
+ if (!value.IsEmpty())
+ DoWriteText(value, SetValue_SelectionOnly);
EndBatchUndo();
}
wxRect caretRect;
if (GetCaretPositionForIndex(GetCaretPosition(), caretRect, container))
{
+#if !wxRICHTEXT_USE_OWN_CARET
caretRect = GetScaledRect(caretRect);
+#endif
int topMargin = (int) (0.5 + GetScale()*GetBuffer().GetTopMargin());
int bottomMargin = (int) (0.5 + GetScale()*GetBuffer().GetBottomMargin());
wxPoint newPt = caretRect.GetPosition();
PrepareDC(dc);
dc.SetFont(GetFont());
+ dc.SetUserScale(GetScale(), GetScale());
wxRichTextDrawingContext context(& GetBuffer());
- GetBuffer().Defragment();
+ GetBuffer().Defragment(context);
GetBuffer().UpdateRanges(); // If items were deleted, ranges need recalculation
GetBuffer().Layout(dc, context, availableSpace, availableSpace, flags);
GetBuffer().Invalidate(wxRICHTEXT_NONE);
- if (!IsFrozen())
+ dc.SetUserScale(1.0, 1.0);
+
+ if (!IsFrozen() && !onlyVisibleRect)
SetupScrollbars();
}
return GetFocusObject()->PromoteList(promoteBy, range.ToInternal(), defName, flags, specifiedLevel);
}
+// Given a character position at which there is a list style, find the range
+// encompassing the same list style by looking backwards and forwards.
+wxRichTextRange wxRichTextCtrl::FindRangeForList(long pos, bool& isNumberedList)
+{
+ wxRichTextParagraphLayoutBox* focusObject = GetFocusObject();
+ wxRichTextRange range = wxRichTextRange(-1, -1);
+ wxRichTextParagraph* para = focusObject->GetParagraphAtPosition(pos);
+ if (!para || !para->GetAttributes().HasListStyleName())
+ return range;
+ else
+ {
+ wxString listStyle = para->GetAttributes().GetListStyleName();
+ range = para->GetRange();
+
+ isNumberedList = para->GetAttributes().HasBulletNumber();
+
+ // Search back
+ wxRichTextObjectList::compatibility_iterator initialNode = focusObject->GetChildren().Find(para);
+ if (initialNode)
+ {
+ wxRichTextObjectList::compatibility_iterator startNode = initialNode->GetPrevious();
+ while (startNode)
+ {
+ wxRichTextParagraph* p = wxDynamicCast(startNode->GetData(), wxRichTextParagraph);
+ if (p)
+ {
+ if (!p->GetAttributes().HasListStyleName() || p->GetAttributes().GetListStyleName() != listStyle)
+ break;
+ else
+ range.SetStart(p->GetRange().GetStart());
+ }
+
+ startNode = startNode->GetPrevious();
+ }
+
+ // Search forward
+ wxRichTextObjectList::compatibility_iterator endNode = initialNode->GetNext();
+ while (endNode)
+ {
+ wxRichTextParagraph* p = wxDynamicCast(endNode->GetData(), wxRichTextParagraph);
+ if (p)
+ {
+ if (!p->GetAttributes().HasListStyleName() || p->GetAttributes().GetListStyleName() != listStyle)
+ break;
+ else
+ range.SetEnd(p->GetRange().GetEnd());
+ }
+
+ endNode = endNode->GetNext();
+ }
+ }
+ }
+ return range;
+}
+
/// Deletes the content in the given range
bool wxRichTextCtrl::Delete(const wxRichTextRange& range)
{
SetDefaultStyleToCursorStyle();
wxRichTextEvent cmdEvent(
- wxEVT_COMMAND_RICHTEXT_FOCUS_OBJECT_CHANGED,
+ wxEVT_RICHTEXT_FOCUS_OBJECT_CHANGED,
GetId());
cmdEvent.SetEventObject(this);
cmdEvent.SetPosition(m_caretPosition+1);
{
m_flashOn = true;
- if (!m_timer.IsRunning())
+ if (!m_timer.IsRunning() && GetBlinkTime() > 0)
m_timer.Start(GetBlinkTime());
Refresh();
{
if (m_richTextCtrl && m_refreshEnabled)
{
- wxRect rect(GetPosition(), GetSize());
- m_richTextCtrl->RefreshRect(rect, false);
+ wxRect rect(wxPoint(m_xOld, m_yOld), GetSize());
+ wxRect scaledRect = m_richTextCtrl->GetScaledRect(rect);
+ m_richTextCtrl->RefreshRect(scaledRect, false);
}
}
}
if (m_richTextCtrl && m_refreshEnabled)
{
wxRect rect(GetPosition(), GetSize());
- m_richTextCtrl->RefreshRect(rect, false);
+ wxRect rectScaled = m_richTextCtrl->GetScaledRect(rect);
+ m_richTextCtrl->RefreshRect(rectScaled, false);
}
}
void wxRichTextCaret::DoDraw(wxDC *dc)
{
- dc->SetPen( *wxBLACK_PEN );
-
- dc->SetBrush(*(m_hasFocus ? wxBLACK_BRUSH : wxTRANSPARENT_BRUSH));
- dc->SetPen(*wxBLACK_PEN);
+ wxBrush brush(m_caretBrush);
+ wxPen pen(m_caretPen);
+ if (m_richTextCtrl && m_richTextCtrl->GetBasicStyle().HasTextColour())
+ {
+ brush = wxBrush(m_richTextCtrl->GetBasicStyle().GetTextColour());
+ pen = wxPen(m_richTextCtrl->GetBasicStyle().GetTextColour());
+ }
+ dc->SetBrush((m_hasFocus ? brush : *wxTRANSPARENT_BRUSH));
+ dc->SetPen(pen);
wxPoint pt(m_x, m_y);
void wxRichTextCaret::Notify()
{
+#ifdef __WXMAC__
+ // Workaround for lack of kill focus event in wxOSX
+ if (m_richTextCtrl && !m_richTextCtrl->HasFocus())
+ {
+ Hide();
+ return;
+ }
+#endif
+
m_flashOn = !m_flashOn;
Refresh();
}