Implemented keyboard selection and better cell navigation for tables
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@76440 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
parent
6b12d12fe9
commit
5b8e5e81b6
@ -1977,6 +1977,16 @@ public:
|
|||||||
*/
|
*/
|
||||||
virtual bool ExtendSelection(long oldPosition, long newPosition, int flags);
|
virtual bool ExtendSelection(long oldPosition, long newPosition, int flags);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Extends a table selection in the given direction.
|
||||||
|
*/
|
||||||
|
virtual bool ExtendCellSelection(wxRichTextTable* table, int noRowSteps, int noColSteps);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Starts selecting table cells.
|
||||||
|
*/
|
||||||
|
virtual bool StartCellSelection(wxRichTextTable* table, wxRichTextParagraphLayoutBox* newCell);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Scrolls @a position into view. This function takes a caret position.
|
Scrolls @a position into view. This function takes a caret position.
|
||||||
*/
|
*/
|
||||||
|
@ -1929,6 +1929,16 @@ public:
|
|||||||
*/
|
*/
|
||||||
virtual bool ExtendSelection(long oldPosition, long newPosition, int flags);
|
virtual bool ExtendSelection(long oldPosition, long newPosition, int flags);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Extends a table selection in the given direction.
|
||||||
|
*/
|
||||||
|
virtual bool ExtendCellSelection(wxRichTextTable* table, int noRowSteps, int noColSteps);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Starts selecting table cells.
|
||||||
|
*/
|
||||||
|
virtual bool StartCellSelection(wxRichTextTable* table, wxRichTextParagraphLayoutBox* newCell);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Scrolls @a position into view. This function takes a caret position.
|
Scrolls @a position into view. This function takes a caret position.
|
||||||
*/
|
*/
|
||||||
|
@ -2069,9 +2069,95 @@ void wxRichTextCtrl::MoveCaretBack(long oldPosition)
|
|||||||
/// Move right
|
/// Move right
|
||||||
bool wxRichTextCtrl::MoveRight(int noPositions, int flags)
|
bool wxRichTextCtrl::MoveRight(int noPositions, int flags)
|
||||||
{
|
{
|
||||||
|
// Test for continuing table selection
|
||||||
|
if (flags && wxRICHTEXT_SHIFT_DOWN)
|
||||||
|
{
|
||||||
|
if (m_selection.GetContainer() && m_selection.GetContainer()->IsKindOf(CLASSINFO(wxRichTextTable)))
|
||||||
|
{
|
||||||
|
wxRichTextTable* table = wxDynamicCast(m_selection.GetContainer(), wxRichTextTable);
|
||||||
|
if (GetFocusObject() && GetFocusObject()->GetParent() == m_selection.GetContainer())
|
||||||
|
{
|
||||||
|
ExtendCellSelection(table, 0, noPositions);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
long startPos = -1;
|
||||||
long endPos = GetFocusObject()->GetOwnRange().GetEnd();
|
long endPos = GetFocusObject()->GetOwnRange().GetEnd();
|
||||||
|
|
||||||
if (m_caretPosition + noPositions < endPos)
|
bool beyondBottom = (noPositions > 0 && (m_caretPosition + noPositions >= endPos));
|
||||||
|
bool beyondTop = (noPositions < 0 && (m_caretPosition <= startPos + noPositions + 1));
|
||||||
|
|
||||||
|
if (beyondBottom || beyondTop)
|
||||||
|
{
|
||||||
|
wxPoint pt = GetLogicalPoint(GetCaret()->GetPosition());
|
||||||
|
|
||||||
|
int hitTestFlags = wxRICHTEXT_HITTEST_NO_FLOATING_OBJECTS|wxRICHTEXT_HITTEST_HONOUR_ATOMIC;
|
||||||
|
|
||||||
|
if (beyondBottom)
|
||||||
|
pt.x = GetFocusObject()->GetPosition().x + GetFocusObject()->GetCachedSize().x + 2;
|
||||||
|
else
|
||||||
|
pt.x = GetFocusObject()->GetPosition().x - 2;
|
||||||
|
|
||||||
|
pt.y += 2;
|
||||||
|
|
||||||
|
long newPos = 0;
|
||||||
|
wxClientDC dc(this);
|
||||||
|
PrepareDC(dc);
|
||||||
|
dc.SetFont(GetFont());
|
||||||
|
|
||||||
|
wxRichTextObject* hitObj = NULL;
|
||||||
|
wxRichTextObject* contextObj = NULL;
|
||||||
|
wxRichTextDrawingContext context(& GetBuffer());
|
||||||
|
int hitTest = GetBuffer().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'
|
||||||
|
)
|
||||||
|
{
|
||||||
|
wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox);
|
||||||
|
if (actualContainer && actualContainer != GetFocusObject() && actualContainer->AcceptsFocus() && actualContainer->IsShown())
|
||||||
|
{
|
||||||
|
if ((flags && wxRICHTEXT_SHIFT_DOWN) &&
|
||||||
|
GetFocusObject()->IsKindOf(CLASSINFO(wxRichTextCell)) &&
|
||||||
|
actualContainer->IsKindOf(CLASSINFO(wxRichTextCell)) &&
|
||||||
|
GetFocusObject()->GetParent() == actualContainer->GetParent())
|
||||||
|
{
|
||||||
|
// Start selecting cells in a table
|
||||||
|
wxRichTextTable* table = wxDynamicCast(actualContainer->GetParent(), wxRichTextTable);
|
||||||
|
if (table)
|
||||||
|
{
|
||||||
|
StartCellSelection(table, actualContainer);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the new container is a cell, go to the top or bottom of it.
|
||||||
|
if (actualContainer->IsKindOf(CLASSINFO(wxRichTextCell)))
|
||||||
|
{
|
||||||
|
if (beyondBottom)
|
||||||
|
newPos = 0;
|
||||||
|
else
|
||||||
|
newPos = actualContainer->GetOwnRange().GetEnd()-1;
|
||||||
|
}
|
||||||
|
|
||||||
|
SetFocusObject(actualContainer, false /* don't set caret position yet */);
|
||||||
|
bool caretLineStart = true;
|
||||||
|
long caretPosition = FindCaretPositionForCharacterPosition(newPos, hitTest, actualContainer, caretLineStart);
|
||||||
|
|
||||||
|
SelectNone();
|
||||||
|
|
||||||
|
SetCaretPosition(caretPosition, caretLineStart);
|
||||||
|
PositionCaret();
|
||||||
|
SetDefaultStyleToCursorStyle();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (!beyondTop && !beyondBottom)
|
||||||
{
|
{
|
||||||
long oldPos = m_caretPosition;
|
long oldPos = m_caretPosition;
|
||||||
long newPos = m_caretPosition + noPositions;
|
long newPos = m_caretPosition + noPositions;
|
||||||
@ -2087,32 +2173,7 @@ bool wxRichTextCtrl::MoveRight(int noPositions, int flags)
|
|||||||
// line.
|
// line.
|
||||||
if (noPositions == 1)
|
if (noPositions == 1)
|
||||||
MoveCaretForward(oldPos);
|
MoveCaretForward(oldPos);
|
||||||
else
|
else if (noPositions == -1)
|
||||||
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)
|
|
||||||
MoveCaretBack(oldPos);
|
MoveCaretBack(oldPos);
|
||||||
else
|
else
|
||||||
SetCaretPosition(newPos);
|
SetCaretPosition(newPos);
|
||||||
@ -2122,8 +2183,13 @@ bool wxRichTextCtrl::MoveLeft(int noPositions, int flags)
|
|||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else
|
return false;
|
||||||
return false;
|
}
|
||||||
|
|
||||||
|
/// Move left
|
||||||
|
bool wxRichTextCtrl::MoveLeft(int noPositions, int flags)
|
||||||
|
{
|
||||||
|
return MoveRight(-noPositions, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find the caret position for the combination of hit-test flags and character position.
|
// Find the caret position for the combination of hit-test flags and character position.
|
||||||
@ -2160,20 +2226,28 @@ long wxRichTextCtrl::FindCaretPositionForCharacterPosition(long position, int hi
|
|||||||
return caretPosition;
|
return caretPosition;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Move up
|
|
||||||
bool wxRichTextCtrl::MoveUp(int noLines, int flags)
|
|
||||||
{
|
|
||||||
return MoveDown(- noLines, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Move up
|
/// Move up
|
||||||
bool wxRichTextCtrl::MoveDown(int noLines, int flags)
|
bool wxRichTextCtrl::MoveDown(int noLines, int flags)
|
||||||
{
|
{
|
||||||
if (!GetCaret())
|
if (!GetCaret())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
// Test for continuing table selection
|
||||||
|
if (flags && wxRICHTEXT_SHIFT_DOWN)
|
||||||
|
{
|
||||||
|
if (m_selection.GetContainer() && m_selection.GetContainer()->IsKindOf(CLASSINFO(wxRichTextTable)))
|
||||||
|
{
|
||||||
|
wxRichTextTable* table = wxDynamicCast(m_selection.GetContainer(), wxRichTextTable);
|
||||||
|
if (GetFocusObject() && GetFocusObject()->GetParent() == m_selection.GetContainer())
|
||||||
|
{
|
||||||
|
ExtendCellSelection(table, noLines, 0);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
long lineNumber = GetFocusObject()->GetVisibleLineNumber(m_caretPosition, true, m_caretAtLineStart);
|
long lineNumber = GetFocusObject()->GetVisibleLineNumber(m_caretPosition, true, m_caretAtLineStart);
|
||||||
wxPoint pt = GetCaret()->GetPosition();
|
wxPoint pt = GetLogicalPoint(GetCaret()->GetPosition());
|
||||||
long newLine = lineNumber + noLines;
|
long newLine = lineNumber + noLines;
|
||||||
bool notInThisObject = false;
|
bool notInThisObject = false;
|
||||||
|
|
||||||
@ -2245,6 +2319,20 @@ bool wxRichTextCtrl::MoveDown(int noLines, int flags)
|
|||||||
wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox);
|
wxRichTextParagraphLayoutBox* actualContainer = wxDynamicCast(contextObj, wxRichTextParagraphLayoutBox);
|
||||||
if (actualContainer && actualContainer != GetFocusObject() && actualContainer->AcceptsFocus())
|
if (actualContainer && actualContainer != GetFocusObject() && actualContainer->AcceptsFocus())
|
||||||
{
|
{
|
||||||
|
if ((flags && wxRICHTEXT_SHIFT_DOWN) &&
|
||||||
|
GetFocusObject()->IsKindOf(CLASSINFO(wxRichTextCell)) &&
|
||||||
|
actualContainer->IsKindOf(CLASSINFO(wxRichTextCell)) &&
|
||||||
|
GetFocusObject()->GetParent() == actualContainer->GetParent())
|
||||||
|
{
|
||||||
|
// Start selecting cells in a table
|
||||||
|
wxRichTextTable* table = wxDynamicCast(actualContainer->GetParent(), wxRichTextTable);
|
||||||
|
if (table)
|
||||||
|
{
|
||||||
|
StartCellSelection(table, actualContainer);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
SetFocusObject(actualContainer, false /* don't set caret position yet */);
|
SetFocusObject(actualContainer, false /* don't set caret position yet */);
|
||||||
|
|
||||||
container = actualContainer;
|
container = actualContainer;
|
||||||
@ -2283,6 +2371,122 @@ bool wxRichTextCtrl::MoveDown(int noLines, int flags)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Extend a table selection in the given direction
|
||||||
|
bool wxRichTextCtrl::ExtendCellSelection(wxRichTextTable* table, int noRowSteps, int noColSteps)
|
||||||
|
{
|
||||||
|
int thisRow = -1;
|
||||||
|
int thisCol = -1;
|
||||||
|
int r, c;
|
||||||
|
for (r = 0; r < table->GetRowCount(); r ++)
|
||||||
|
{
|
||||||
|
for (c = 0; c < table->GetColumnCount(); c++)
|
||||||
|
{
|
||||||
|
wxRichTextCell* cell = table->GetCell(r, c);
|
||||||
|
if (cell == GetFocusObject())
|
||||||
|
{
|
||||||
|
thisRow = r;
|
||||||
|
thisCol = c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (thisRow != -1)
|
||||||
|
{
|
||||||
|
int newRow = wxMax(0, wxMin((thisRow + noRowSteps), table->GetRowCount()-1));
|
||||||
|
int newCol = wxMax(0, wxMin((thisCol + noColSteps), table->GetColumnCount()-1));
|
||||||
|
|
||||||
|
if (newRow != thisRow || newCol != thisCol)
|
||||||
|
{
|
||||||
|
// Make sure we're on a visible row or column
|
||||||
|
r = newRow;
|
||||||
|
c = newCol;
|
||||||
|
int rowInc = noRowSteps > 0 ? 1 : -1;
|
||||||
|
int colInc = noColSteps > 0 ? 1 : -1;
|
||||||
|
bool visibleRow = false;
|
||||||
|
bool visibleCol = false;
|
||||||
|
if (noRowSteps != 0)
|
||||||
|
{
|
||||||
|
while (r >= 0 && r < table->GetRowCount())
|
||||||
|
{
|
||||||
|
wxRichTextCell* cell = table->GetCell(r, newCol);
|
||||||
|
if (cell->IsShown())
|
||||||
|
{
|
||||||
|
newRow = r;
|
||||||
|
visibleRow = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
r += rowInc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// No change if the cell would not be visible
|
||||||
|
if (!visibleRow)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (noColSteps != 0)
|
||||||
|
{
|
||||||
|
while (c >= 0 && c < table->GetColumnCount())
|
||||||
|
{
|
||||||
|
wxRichTextCell* cell = table->GetCell(newRow, c);
|
||||||
|
if (cell->IsShown())
|
||||||
|
{
|
||||||
|
newCol = c;
|
||||||
|
visibleCol = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
c += colInc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// No change if the cell would not be visible
|
||||||
|
if (!visibleCol)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
wxRichTextCell* newCell = table->GetCell(newRow, newCol);
|
||||||
|
if (newCell)
|
||||||
|
{
|
||||||
|
m_selection = table->GetSelection(m_selectionAnchor, newCell->GetRange().GetStart());
|
||||||
|
Refresh();
|
||||||
|
if (newCell->AcceptsFocus())
|
||||||
|
SetFocusObject(newCell, false);
|
||||||
|
MoveCaret(-1, false);
|
||||||
|
SetDefaultStyleToCursorStyle();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Start selecting cells
|
||||||
|
bool wxRichTextCtrl::StartCellSelection(wxRichTextTable* table, wxRichTextParagraphLayoutBox* newCell)
|
||||||
|
{
|
||||||
|
// Start selecting cells in a table
|
||||||
|
m_selectionState = wxRichTextCtrlSelectionState_CommonAncestor;
|
||||||
|
m_selectionAnchorObject = GetFocusObject();
|
||||||
|
m_selectionAnchor = GetFocusObject()->GetRange().GetStart();
|
||||||
|
|
||||||
|
// The common ancestor, such as a table, returns the cell selection
|
||||||
|
// between the anchor and current position.
|
||||||
|
m_selection = table->GetSelection(m_selectionAnchor, newCell->GetRange().GetStart());
|
||||||
|
Refresh();
|
||||||
|
|
||||||
|
if (newCell->AcceptsFocus())
|
||||||
|
SetFocusObject(newCell, false /* don't set caret and clear selection */);
|
||||||
|
MoveCaret(-1, false);
|
||||||
|
SetDefaultStyleToCursorStyle();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Move up
|
||||||
|
bool wxRichTextCtrl::MoveUp(int noLines, int flags)
|
||||||
|
{
|
||||||
|
return MoveDown(- noLines, flags);
|
||||||
|
}
|
||||||
|
|
||||||
/// Move to the end of the paragraph
|
/// Move to the end of the paragraph
|
||||||
bool wxRichTextCtrl::MoveToParagraphEnd(int flags)
|
bool wxRichTextCtrl::MoveToParagraphEnd(int flags)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user