summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--starmath/inc/cursor.hxx28
-rw-r--r--starmath/source/cursor.cxx152
-rw-r--r--starmath/source/view.cxx5
3 files changed, 115 insertions, 70 deletions
diff --git a/starmath/inc/cursor.hxx b/starmath/inc/cursor.hxx
index 49f1e0c18a8c..b5fbd13e28cb 100644
--- a/starmath/inc/cursor.hxx
+++ b/starmath/inc/cursor.hxx
@@ -132,6 +132,12 @@ public:
/** Delete the current selection or do nothing */
void Delete();
+ /** Delete selection, previous element or merge lines
+ *
+ * This method implements the behaviour of backspace.
+ */
+ void DeletePrev(OutputDevice* pDev);
+
/** Insert text at the current position */
void InsertText(XubString aString);
@@ -270,6 +276,28 @@ private:
*/
static SmNodeList* LineToList(SmStructureNode* pLine, SmNodeList* pList = new SmNodeList());
+ /** Auxiliary function for calling LineToList on a node
+ *
+ * This method sets pNode = NULL and remove it from it's parent.
+ * (Assuming it has a parent, and is a child of it).
+ */
+ static SmNodeList* NodeToList(SmNode*& rpNode, SmNodeList* pList = new SmNodeList()){
+ //Remove from parent and NULL rpNode
+ SmNode* pNode = rpNode;
+ if(rpNode && rpNode->GetParent()){ //Don't remove this, correctness relies on it
+ int index = rpNode->GetParent()->IndexOfSubNode(rpNode);
+ if(index != -1)
+ rpNode->GetParent()->SetSubNode(index, NULL);
+ }
+ rpNode = NULL;
+ //Create line from node
+ if(pNode && IsLineCompositionNode(pNode))
+ return LineToList((SmStructureNode*)pNode, pList);
+ if(pNode)
+ pList->push_front(pNode);
+ return pList;
+ }
+
/** Clone a visual line to a list
*
* Doesn't clone SmErrorNode's these are ignored, as they are context dependent metadata.
diff --git a/starmath/source/cursor.cxx b/starmath/source/cursor.cxx
index 25c4522b67d1..c5303663f047 100644
--- a/starmath/source/cursor.cxx
+++ b/starmath/source/cursor.cxx
@@ -186,6 +186,76 @@ void SmCursor::Draw(OutputDevice& pDev, Point Offset, bool isCaretVisible){
SmCaretDrawingVisitor(pDev, GetPosition(), Offset, isCaretVisible);
}
+void SmCursor::DeletePrev(OutputDevice* pDev){
+ //Delete only a selection if there's a selection
+ if(HasSelection()){
+ Delete();
+ return;
+ }
+
+ SmNode* pLine = FindTopMostNodeInLine(position->CaretPos.pSelectedNode);
+ SmStructureNode* pLineParent = pLine->GetParent();
+ int nLineOffset = pLineParent->IndexOfSubNode(pLine);
+
+ //If we're in front of a node who's parent is a TABLE
+ if(pLineParent->GetType() == NTABLE && position->CaretPos.Index == 0 && nLineOffset > 0){
+ //Now we can merge with nLineOffset - 1
+ BeginEdit();
+ //Line to merge things into, so we can delete pLine
+ SmNode* pMergeLine = pLineParent->GetSubNode(nLineOffset-1);
+ j_assert(pMergeLine, "pMergeLine cannot be NULL!");
+ //Convert first line to list
+ SmNodeList *pLineList = NodeToList(pMergeLine);
+ //Find iterator to patch
+ SmNodeList::iterator patchPoint = pLineList->end();
+ patchPoint--;
+ //Convert second line to list
+ NodeToList(pLine, pLineList);
+ //Patch the line list
+ patchPoint++;
+ SmCaretPos PosAfterDelete = PatchLineList(pLineList, patchPoint);
+ //Parse the line
+ pLine = SmNodeListParser().Parse(pLineList);
+ delete pLineList;
+ pLineParent->SetSubNode(nLineOffset-1, pLine);
+ //Delete the removed line slot
+ SmNodeArray lines(pLineParent->GetNumSubNodes()-1);
+ for(int i = 0; i < pLineParent->GetNumSubNodes(); i++){
+ if(i < nLineOffset)
+ lines[i] = pLineParent->GetSubNode(i);
+ else if(i > nLineOffset)
+ lines[i-1] = pLineParent->GetSubNode(i);
+ }
+ pLineParent->SetSubNodes(lines);
+ //Rebuild graph
+ anchor = NULL;
+ position = NULL;
+ BuildGraph();
+ AnnotateSelection();
+ //Set caret position
+ if(!SetCaretPosition(PosAfterDelete, true))
+ SetCaretPosition(SmCaretPos(pLine, 0), true);
+ //Finish editing
+ EndEdit();
+
+ //TODO: If we're in an empty (sub/super/*) script
+ /*}else if(pLineParent->GetType() == NSUBSUP &&
+ nLineOffset != 0 &&
+ pLine->GetType() == NEXPRESSION &&
+ pLine->GetNumSubNodes() == 0){
+ //There's a (sub/super/*) script we can delete
+ //Consider selecting the entire script if GetNumSubNodes() != 0 or pLine->GetType() != NEXPRESSION
+ //TODO: Handle case where we delete a limit
+ */
+
+ //Else move select, and delete if not complex
+ }else{
+ this->Move(pDev, MoveLeft, false);
+ if(!this->HasComplexSelection())
+ Delete();
+ }
+}
+
void SmCursor::Delete(){
//Return if we don't have a selection to delete
if(!HasSelection())
@@ -214,13 +284,7 @@ void SmCursor::Delete(){
//Position after delete
SmCaretPos PosAfterDelete;
- SmNodeList* pLineList;
- if(IsLineCompositionNode(pLine))
- pLineList = LineToList((SmStructureNode*)pLine);
- else {
- pLineList = new SmNodeList();
- pLineList->push_back(pLine);
- }
+ SmNodeList* pLineList = NodeToList(pLine);
//Take the selected nodes and delete them...
SmNodeList::iterator patchIt = TakeSelectedNodesFromList(pLineList);
@@ -256,13 +320,7 @@ void SmCursor::InsertNodes(SmNodeList* pNewNodes){
j_assert(nParentIndex != -1, "pLine must be a subnode of pLineParent!");
//Convert line to list
- SmNodeList* pLineList;
- if(IsLineCompositionNode(pLine))
- pLineList = LineToList((SmStructureNode*)pLine);
- else {
- pLineList = new SmNodeList();
- pLineList->push_front(pLine);
- }
+ SmNodeList* pLineList = NodeToList(pLine);
//Find iterator for place to insert nodes
SmNodeList::iterator it = FindPositionInLineList(pLineList, pos);
@@ -465,13 +523,7 @@ void SmCursor::InsertSubSup(SmSubSup eSubSup) {
BeginEdit();
//Convert line to list
- SmNodeList* pLineList;
- if(IsLineCompositionNode(pLine))
- pLineList = LineToList((SmStructureNode*)pLine);
- else {
- pLineList = new SmNodeList();
- pLineList->push_front(pLine);
- }
+ SmNodeList* pLineList = NodeToList(pLine);
//Take the selection, and/or find iterator for current position
SmNodeList* pSelectedNodesList = new SmNodeList();
@@ -518,14 +570,7 @@ void SmCursor::InsertSubSup(SmSubSup eSubSup) {
//Convert existing, if any, sub-/superscript line to list
SmNode *pScriptLine = pSubSup->GetSubSup(eSubSup);
- SmNodeList* pScriptLineList;
- if(pScriptLine && IsLineCompositionNode(pScriptLine))
- pScriptLineList = LineToList((SmStructureNode*)pScriptLine);
- else{
- pScriptLineList = new SmNodeList();
- if(pScriptLine)
- pScriptLineList->push_front(pScriptLine);
- }
+ SmNodeList* pScriptLineList = NodeToList(pScriptLine);
//Add selection to pScriptLineList
unsigned int nOldSize = pScriptLineList->size();
@@ -601,13 +646,7 @@ bool SmCursor::InsertLimit(SmSubSup eSubSup, bool bMoveCaret) {
//If it's already there... let's move the caret
} else if(bMoveCaret){
pLine = pSubSup->GetSubSup(eSubSup);
- SmNodeList* pLineList;
- if(IsLineCompositionNode(pLine))
- pLineList = LineToList((SmStructureNode*)pLine);
- else {
- pLineList = new SmNodeList();
- pLineList->push_front(pLine);
- }
+ SmNodeList* pLineList = NodeToList(pLine);
if(pLineList->size() > 0)
PosAfterLimit = SmCaretPos::GetPosAfter(pLineList->back());
pLine = SmNodeListParser().Parse(pLineList);
@@ -649,13 +688,7 @@ void SmCursor::InsertBrackets(SmBracketType eBracketType) {
j_assert( nParentIndex != -1, "pLine must be a subnode of pLineParent!");
//Convert line to list
- SmNodeList *pLineList;
- if(IsLineCompositionNode(pLine))
- pLineList = LineToList((SmStructureNode*)pLine);
- else {
- pLineList = new SmNodeList();
- pLineList->push_front(pLine);
- }
+ SmNodeList *pLineList = NodeToList(pLine);
//Take the selection, and/or find iterator for current position
SmNodeList *pSelectedNodesList = new SmNodeList();
@@ -816,13 +849,7 @@ bool SmCursor::InsertRow() {
BeginEdit();
//Convert line to list
- SmNodeList *pLineList;
- if(IsLineCompositionNode(pLine))
- pLineList = LineToList((SmStructureNode*)pLine);
- else {
- pLineList = new SmNodeList();
- pLineList->push_front(pLine);
- }
+ SmNodeList *pLineList = NodeToList(pLine);
//Find position in line
SmNodeList::iterator it;
@@ -849,20 +876,21 @@ bool SmCursor::InsertRow() {
//Parse new line
SmNode *pNewLine = SmNodeListParser().Parse(pNewLineList);
delete pNewLineList;
- //Get position before we wrap in SmLineNode
- //NOTE: This should be done after, if SmLineNode ever becomes a line composition node
- PosAfterInsert = SmCaretPos(pNewLine, 0);
//Wrap pNewLine in SmLineNode if needed
if(pLineParent->GetType() == NLINE) {
SmLineNode *pNewLineNode = new SmLineNode(SmToken(TNEWLINE, '\0', "newline"));
pNewLineNode->SetSubNodes(pNewLine, NULL);
pNewLine = pNewLineNode;
}
+ //Get position
+ PosAfterInsert = SmCaretPos(pNewLine, 0);
//Move other nodes if needed
for( int i = pTable->GetNumSubNodes(); i > nTableIndex + 1; i--)
pTable->SetSubNode(i, pTable->GetSubNode(i-1));
+
//Insert new line
pTable->SetSubNode(nTableIndex + 1, pNewLine);
+
//Check if we need to change token type:
if(pTable->GetNumSubNodes() > 2 && pTable->GetToken().eType == TBINOM) {
SmToken tok = pTable->GetToken();
@@ -919,13 +947,7 @@ void SmCursor::InsertFraction() {
BeginEdit();
//Convert line to list
- SmNodeList* pLineList;
- if(IsLineCompositionNode(pLine))
- pLineList = LineToList((SmStructureNode*)pLine);
- else {
- pLineList = new SmNodeList();
- pLineList->push_front(pLine);
- }
+ SmNodeList* pLineList = NodeToList(pLine);
//Take the selection, and/or find iterator for current position
SmNodeList* pSelectedNodesList = new SmNodeList();
@@ -957,7 +979,6 @@ void SmCursor::InsertFraction() {
FinishEdit(pLineList, pLineParent, nParentIndex, SmCaretPos(pDenom, 1));
}
-
void SmCursor::InsertText(XubString aString){
BeginEdit();
@@ -1142,13 +1163,7 @@ void SmCursor::InsertCommandText(XubString aCommandText) {
pSubExpr->Prepare(pDocShell->GetFormat(), *pDocShell);
//Convert subtree to list
- SmNodeList* pLineList;
- if(IsLineCompositionNode(pSubExpr))
- pLineList = LineToList((SmStructureNode*)pSubExpr);
- else {
- pLineList = new SmNodeList();
- pLineList->push_front(pSubExpr);
- }
+ SmNodeList* pLineList = NodeToList(pSubExpr);
BeginEdit();
@@ -1219,7 +1234,6 @@ SmNodeList* SmCursor::CloneList(SmNodeList* pList){
return pClones;
}
-
void SmCursor::SetClipboard(SmNodeList* pList){
if(pClipboard){
//Delete all nodes on the clipboard
diff --git a/starmath/source/view.cxx b/starmath/source/view.cxx
index bfe0bc7b2cb9..aec233e52650 100644
--- a/starmath/source/view.cxx
+++ b/starmath/source/view.cxx
@@ -451,7 +451,6 @@ void SmGraphicWindow::KeyInput(const KeyEvent& rKEvt)
#endif /* DEBUG_ENABLE_DUMPASDOT */
}break;
case KEY_DELETE:
- case KEY_BACKSPACE:
{
if(!rCursor.HasSelection()){
rCursor.Move(this, nCode == KEY_DELETE ? MoveRight : MoveLeft, false);
@@ -459,6 +458,10 @@ void SmGraphicWindow::KeyInput(const KeyEvent& rKEvt)
}
rCursor.Delete();
}break;
+ case KEY_BACKSPACE:
+ {
+ rCursor.DeletePrev(this);
+ }break;
case KEY_ADD:
rCursor.InsertElement(PlusElement);
break;