diff options
Diffstat (limited to 'layout/generic/SelectionMovementUtils.h')
-rw-r--r-- | layout/generic/SelectionMovementUtils.h | 177 |
1 files changed, 177 insertions, 0 deletions
diff --git a/layout/generic/SelectionMovementUtils.h b/layout/generic/SelectionMovementUtils.h new file mode 100644 index 0000000000..5d81949488 --- /dev/null +++ b/layout/generic/SelectionMovementUtils.h @@ -0,0 +1,177 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#ifndef mozilla_SelectionMovementUtils_h +#define mozilla_SelectionMovementUtils_h + +#include "mozilla/intl/BidiEmbeddingLevel.h" +#include "mozilla/Attributes.h" +#include "mozilla/EnumSet.h" +#include "mozilla/Result.h" +#include "nsIFrame.h" + +struct nsPrevNextBidiLevels; + +namespace mozilla { + +class PresShell; +enum class PeekOffsetOption : uint16_t; + +namespace intl { +class BidiEmbeddingLevel; +} + +struct MOZ_STACK_CLASS PrimaryFrameData { + // The frame which should be used to layout the caret. + nsIFrame* mFrame = nullptr; + // The offset in content of mFrame. This is valid only when mFrame is not + // nullptr. + uint32_t mOffsetInFrameContent = 0; + // Whether the caret should be put before or after the point. This is valid + // only when mFrame is not nullptr. + CaretAssociationHint mHint{0}; // Before +}; + +struct MOZ_STACK_CLASS CaretFrameData : public PrimaryFrameData { + // The frame which is found only from a DOM point. This frame becomes + // different from mFrame when the point is around end of a line or + // at a bidi text boundary. + nsIFrame* mUnadjustedFrame = nullptr; +}; + +enum class ForceEditableRegion : bool { No, Yes }; + +class SelectionMovementUtils final { + public: + using PeekOffsetOptions = EnumSet<PeekOffsetOption>; + + /** + * Given a node and its child offset, return the nsIFrame and the offset into + * that frame. + * + * @param aNode input parameter for the node to look at + * TODO: Make this `const nsIContent*` for `ContentEventHandler`. + * @param aOffset offset into above node. + * @param aReturnOffset will contain offset into frame. + */ + static nsIFrame* GetFrameForNodeOffset(nsIContent* aNode, uint32_t aOffset, + CaretAssociationHint aHint, + uint32_t* aReturnOffset = nullptr); + + /** + * GetPrevNextBidiLevels will return the frames and associated Bidi levels of + * the characters logically before and after a (collapsed) selection. + * + * @param aNode is the node containing the selection + * @param aContentOffset is the offset of the selection in the node + * @param aJumpLines + * If true, look across line boundaries. + * If false, behave as if there were base-level frames at line edges. + * + * @return A struct holding the before/after frame and the before/after + * level. + * + * At the beginning and end of each line there is assumed to be a frame with + * Bidi level equal to the paragraph embedding level. + * + * In these cases the before frame and after frame respectively will be + * nullptr. + */ + static nsPrevNextBidiLevels GetPrevNextBidiLevels(nsIContent* aNode, + uint32_t aContentOffset, + CaretAssociationHint aHint, + bool aJumpLines); + + /** + * PeekOffsetForCaretMove() only peek offset for caret move from the specified + * point of the normal selection. I.e., won't change selection ranges nor + * bidi information. + */ + static Result<PeekOffsetStruct, nsresult> PeekOffsetForCaretMove( + nsIContent* aContent, uint32_t aOffset, nsDirection aDirection, + CaretAssociationHint aHint, intl::BidiEmbeddingLevel aCaretBidiLevel, + const nsSelectionAmount aAmount, const nsPoint& aDesiredCaretPos, + PeekOffsetOptions aOptions); + + /** + * IsIntraLineCaretMove() is a helper method for PeekOffsetForCaretMove() + * and CreateRangeExtendedToSomwhereFromNormalSelection(). This returns + * whether aAmount is intra line move or is crossing hard line break. + * This returns error if aMount is not supported by the methods. + */ + static Result<bool, nsresult> IsIntraLineCaretMove( + nsSelectionAmount aAmount) { + switch (aAmount) { + case eSelectCharacter: + case eSelectCluster: + case eSelectWord: + case eSelectWordNoSpace: + case eSelectBeginLine: + case eSelectEndLine: + return true; + case eSelectLine: + return false; + default: + return Err(NS_ERROR_FAILURE); + } + } + + /** + * Return a frame for considering caret geometry. + * + * @param aFrameSelection [optional] If this is specified and selection in + * aContent is not managed by the specified + * instance, return nullptr. + * @param aContentNode The content node where selection is collapsed. + * @param aOffset Collapsed position in aContentNode + * @param aFrameHint Caret association hint. + * @param aBidiLevel + * @param aForceEditableRegion Whether selection should be limited in + * editable region or not. + */ + static CaretFrameData GetCaretFrameForNodeOffset( + const nsFrameSelection* aFrameSelection, nsIContent* aContentNode, + uint32_t aOffset, CaretAssociationHint aFrameHint, + intl::BidiEmbeddingLevel aBidiLevel, + ForceEditableRegion aForceEditableRegion); + + static bool AdjustFrameForLineStart(nsIFrame*& aFrame, + uint32_t& aFrameOffset); + + /** + * Get primary frame and some other data for putting caret or extending + * selection at the point. + */ + static PrimaryFrameData GetPrimaryFrameForCaret( + nsIContent* aContent, uint32_t aOffset, bool aVisual, + CaretAssociationHint aHint, intl::BidiEmbeddingLevel aCaretBidiLevel); + + private: + /** + * GetFrameFromLevel will scan in a given direction + * until it finds a frame with a Bidi level less than or equal to a given + * level. It will return the last frame before this. + * + * @param aPresContext is the context to use + * @param aFrameIn is the frame to start from + * @param aDirection is the direction to scan + * @param aBidiLevel is the level to search for + */ + static Result<nsIFrame*, nsresult> GetFrameFromLevel( + nsIFrame* aFrameIn, nsDirection aDirection, + intl::BidiEmbeddingLevel aBidiLevel); + + // This is helper method for GetPrimaryFrameForCaret. + // If aVisual is true, this returns caret frame. + // If false, this returns primary frame. + static PrimaryFrameData GetPrimaryOrCaretFrameForNodeOffset( + nsIContent* aContent, uint32_t aOffset, bool aVisual, + CaretAssociationHint aHint, intl::BidiEmbeddingLevel aCaretBidiLevel); +}; + +} // namespace mozilla + +#endif // #ifndef mozilla_SelectionMovementUtils_h |