mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-13 15:34:01 +00:00
Bug 336408 - Allow the caret to be positioned at the end of trimmed whitespace, as if the whitespace wasn't trimmed [p=roc r=smontagu sr=mrbkap a=blocking1.9+]
This commit is contained in:
parent
2c03a2b214
commit
788e2903bf
@ -68,6 +68,7 @@
|
|||||||
#include "nsISelectionController.h"
|
#include "nsISelectionController.h"
|
||||||
#include "nsDisplayList.h"
|
#include "nsDisplayList.h"
|
||||||
#include "nsCaret.h"
|
#include "nsCaret.h"
|
||||||
|
#include "nsTextFrame.h"
|
||||||
|
|
||||||
// The bidi indicator hangs off the caret to one side, to show which
|
// The bidi indicator hangs off the caret to one side, to show which
|
||||||
// direction the typing is in. It needs to be at least 2x2 to avoid looking like
|
// direction the typing is in. It needs to be at least 2x2 to avoid looking like
|
||||||
@ -608,6 +609,54 @@ nsCaret::DrawAtPositionWithHint(nsIDOMNode* aNode,
|
|||||||
return PR_TRUE;
|
return PR_TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Find the first frame in an in-order traversal of the frame subtree rooted
|
||||||
|
* at aFrame which is either a text frame logically at the end of a line,
|
||||||
|
* or which is aStopAtFrame. Return null if no such frame is found. We don't
|
||||||
|
* descend into the children of non-eLineParticipant frames.
|
||||||
|
*/
|
||||||
|
static nsIFrame*
|
||||||
|
CheckForTrailingTextFrameRecursive(nsIFrame* aFrame, nsIFrame* aStopAtFrame)
|
||||||
|
{
|
||||||
|
if (aFrame == aStopAtFrame ||
|
||||||
|
((aFrame->GetType() == nsGkAtoms::textFrame &&
|
||||||
|
(static_cast<nsTextFrame*>(aFrame))->IsAtEndOfLine())))
|
||||||
|
return aFrame;
|
||||||
|
if (!aFrame->IsFrameOfType(nsIFrame::eLineParticipant))
|
||||||
|
return nsnull;
|
||||||
|
|
||||||
|
for (nsIFrame* f = aFrame->GetFirstChild(nsnull); f; f = f->GetNextSibling())
|
||||||
|
{
|
||||||
|
nsIFrame* r = CheckForTrailingTextFrameRecursive(f, aStopAtFrame);
|
||||||
|
if (r)
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
return nsnull;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
AdjustCaretFrameForLineEnd(nsIFrame** aFrame, PRInt32* aOffset)
|
||||||
|
{
|
||||||
|
nsBlockFrame* block = nsLayoutUtils::FindNearestBlockAncestor(*aFrame);
|
||||||
|
if (!block)
|
||||||
|
return;
|
||||||
|
nsLineBox* line = block->FindLineFor(nsLayoutUtils::FindChildContainingDescendant(block, *aFrame));
|
||||||
|
PRInt32 count = line->GetChildCount();
|
||||||
|
for (nsIFrame* f = line->mFirstChild; count > 0; --count, f = f->GetNextSibling())
|
||||||
|
{
|
||||||
|
nsIFrame* r = CheckForTrailingTextFrameRecursive(f, *aFrame);
|
||||||
|
if (r == *aFrame)
|
||||||
|
return;
|
||||||
|
if (r)
|
||||||
|
{
|
||||||
|
*aFrame = r;
|
||||||
|
NS_ASSERTION(r->GetType() == nsGkAtoms::textFrame, "Expected text frame");
|
||||||
|
*aOffset = (static_cast<nsTextFrame*>(r))->GetContentEnd();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
NS_IMETHODIMP
|
NS_IMETHODIMP
|
||||||
nsCaret::GetCaretFrameForNodeOffset(nsIContent* aContentNode,
|
nsCaret::GetCaretFrameForNodeOffset(nsIContent* aContentNode,
|
||||||
PRInt32 aOffset,
|
PRInt32 aOffset,
|
||||||
@ -634,6 +683,12 @@ nsCaret::GetCaretFrameForNodeOffset(nsIContent* aContentNode,
|
|||||||
if (!theFrame)
|
if (!theFrame)
|
||||||
return NS_ERROR_FAILURE;
|
return NS_ERROR_FAILURE;
|
||||||
|
|
||||||
|
// if theFrame is after a text frame that's logically at the end of the line
|
||||||
|
// (e.g. if theFrame is a <br> frame), then put the caret at the end of
|
||||||
|
// that text frame instead. This way, the caret will be positioned as if
|
||||||
|
// trailing whitespace was not trimmed.
|
||||||
|
AdjustCaretFrameForLineEnd(&theFrame, &theFrameOffset);
|
||||||
|
|
||||||
// Mamdouh : modification of the caret to work at rtl and ltr with Bidi
|
// Mamdouh : modification of the caret to work at rtl and ltr with Bidi
|
||||||
//
|
//
|
||||||
// Direction Style from this->GetStyleData()
|
// Direction Style from this->GetStyleData()
|
||||||
|
@ -180,6 +180,12 @@ public:
|
|||||||
* should return PR_FALSE if this is not a text frame.
|
* should return PR_FALSE if this is not a text frame.
|
||||||
*/
|
*/
|
||||||
virtual PRBool HasTerminalNewline() const;
|
virtual PRBool HasTerminalNewline() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true if this text frame is logically adjacent to the end of the
|
||||||
|
* line.
|
||||||
|
*/
|
||||||
|
PRBool IsAtEndOfLine() const;
|
||||||
|
|
||||||
#ifdef ACCESSIBILITY
|
#ifdef ACCESSIBILITY
|
||||||
NS_IMETHOD GetAccessible(nsIAccessible** aAccessible);
|
NS_IMETHOD GetAccessible(nsIAccessible** aAccessible);
|
||||||
|
@ -4457,7 +4457,7 @@ nsTextFrame::PeekOffsetCharacter(PRBool aForward, PRInt32* aOffset)
|
|||||||
if (!mTextRun)
|
if (!mTextRun)
|
||||||
return PR_FALSE;
|
return PR_FALSE;
|
||||||
|
|
||||||
TrimmedOffsets trimmed = GetTrimmedOffsets(mContent->GetText(), PR_TRUE);
|
TrimmedOffsets trimmed = GetTrimmedOffsets(mContent->GetText(), PR_FALSE);
|
||||||
|
|
||||||
// A negative offset means "end of frame".
|
// A negative offset means "end of frame".
|
||||||
PRInt32 startOffset = GetContentOffset() + (*aOffset < 0 ? contentLength : *aOffset);
|
PRInt32 startOffset = GetContentOffset() + (*aOffset < 0 ? contentLength : *aOffset);
|
||||||
@ -5889,3 +5889,9 @@ nsTextFrame::HasTerminalNewline() const
|
|||||||
{
|
{
|
||||||
return ::HasTerminalNewline(this);
|
return ::HasTerminalNewline(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PRBool
|
||||||
|
nsTextFrame::IsAtEndOfLine() const
|
||||||
|
{
|
||||||
|
return (GetStateBits() & TEXT_END_OF_LINE) != 0;
|
||||||
|
}
|
||||||
|
@ -55,6 +55,7 @@ _TEST_FILES = test_bug323656.html \
|
|||||||
test_bug392923.html \
|
test_bug392923.html \
|
||||||
test_bug394173.html \
|
test_bug394173.html \
|
||||||
test_bug394239.html \
|
test_bug394239.html \
|
||||||
|
test_character_movement.html \
|
||||||
test_word_movement.html \
|
test_word_movement.html \
|
||||||
$(NULL)
|
$(NULL)
|
||||||
|
|
||||||
|
75
layout/generic/test/test_character_movement.html
Normal file
75
layout/generic/test/test_character_movement.html
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
<!DOCTYPE HTML>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>Test Character Movement (including nsTextFrame::PeekOffsetCharacter)</title>
|
||||||
|
<script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
|
||||||
|
<script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
|
||||||
|
<script type="application/javascript" src="/tests/SimpleTest/EventUtils.js"></script>
|
||||||
|
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<p id="display"></p>
|
||||||
|
<div id="content" style="display: block">
|
||||||
|
<div contentEditable id="editor"></div>
|
||||||
|
</div>
|
||||||
|
<pre id="test">
|
||||||
|
<script class="testbody" type="text/javascript;version=1.7">
|
||||||
|
|
||||||
|
SimpleTest.waitForExplicitFinish();
|
||||||
|
|
||||||
|
// This seems to be necessary because the selection is not set up properly otherwise
|
||||||
|
setTimeout(test, 0);
|
||||||
|
|
||||||
|
function test() {
|
||||||
|
var sel = window.getSelection();
|
||||||
|
var editor = document.getElementById("editor");
|
||||||
|
|
||||||
|
function testRight(node, offset) {
|
||||||
|
synthesizeKey("VK_RIGHT", {});
|
||||||
|
is(sel.anchorNode, node, "Right movement broken in " + editor.innerHTML);
|
||||||
|
is(sel.anchorOffset, offset, "Right movement broken in " + editor.innerHTML);
|
||||||
|
}
|
||||||
|
|
||||||
|
function testLeft(node, offset) {
|
||||||
|
synthesizeKey("VK_LEFT", {});
|
||||||
|
is(sel.anchorNode, node, "Left movement broken in " + editor.innerHTML);
|
||||||
|
is(sel.anchorOffset, offset, "Left movement broken in " + editor.innerHTML);
|
||||||
|
}
|
||||||
|
|
||||||
|
editor.innerHTML = "H K";
|
||||||
|
sel.collapse(editor.firstChild, 0);
|
||||||
|
testRight(editor.firstChild, 1);
|
||||||
|
testRight(editor.firstChild, 2);
|
||||||
|
testRight(editor.firstChild, 3);
|
||||||
|
testLeft(editor.firstChild, 2);
|
||||||
|
testLeft(editor.firstChild, 1);
|
||||||
|
testLeft(editor.firstChild, 0);
|
||||||
|
|
||||||
|
editor.innerHTML = "<b>H</b> K";
|
||||||
|
sel.collapse(editor.firstChild.firstChild, 0);
|
||||||
|
testRight(editor.firstChild.firstChild, 1);
|
||||||
|
testRight(editor.firstChild.nextSibling, 1);
|
||||||
|
testRight(editor.firstChild.nextSibling, 2);
|
||||||
|
testLeft(editor.firstChild.nextSibling, 1);
|
||||||
|
testLeft(editor.firstChild.nextSibling, 0);
|
||||||
|
testLeft(editor.firstChild.firstChild, 0);
|
||||||
|
|
||||||
|
editor.innerHTML = "H <br>K";
|
||||||
|
sel.collapse(editor.firstChild, 0);
|
||||||
|
testRight(editor.firstChild, 1);
|
||||||
|
testRight(editor.firstChild, 2);
|
||||||
|
testRight(editor.firstChild.nextSibling.nextSibling, 0);
|
||||||
|
testRight(editor.firstChild.nextSibling.nextSibling, 1);
|
||||||
|
testLeft(editor.firstChild.nextSibling.nextSibling, 0);
|
||||||
|
testLeft(editor, 1);
|
||||||
|
testLeft(editor.firstChild, 1);
|
||||||
|
testLeft(editor.firstChild, 0);
|
||||||
|
|
||||||
|
SimpleTest.finish();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
</script>
|
||||||
|
</pre>
|
||||||
|
</body>
|
||||||
|
</html>
|
Loading…
x
Reference in New Issue
Block a user