/// <summary> /// Applies right alignment to the text on the linebox /// </summary> /// <param name="g"></param> /// <param name="line"></param> private static void ApplyRightAlignment(IGraphics g, CssLineBox line) { if (line.Words.Count == 0) { return; } CssRect lastWord = line.Words[line.Words.Count - 1]; float right = line.OwnerBox.ActualRight - line.OwnerBox.ActualPaddingRight - line.OwnerBox.ActualBorderRightWidth; float diff = right - lastWord.Right - lastWord.OwnerBox.ActualBorderRightWidth - lastWord.OwnerBox.ActualPaddingRight; if (diff > 0) { foreach (CssRect word in line.Words) { word.Left += diff; } foreach (CssBox b in line.Rectangles.Keys) { RectangleF r = b.Rectangles[line]; b.Rectangles[line] = new RectangleF(r.X + diff, r.Y, r.Width, r.Height); } } }
/// <summary> /// Lets the linebox add the word an its box to their lists if necessary. /// </summary> /// <param name="word"></param> internal void ReportExistanceOf(CssRect word) { if (!Words.Contains(word)) { Words.Add(word); } if (!RelatedBoxes.Contains(word.OwnerBox)) { RelatedBoxes.Add(word.OwnerBox); } }
/// <summary> /// Check if the given word is the last selected word in the line.<br/> /// It can either be the last word in the line or the next word has no selection. /// </summary> /// <param name="word">the word to check</param> /// <returns></returns> public bool IsLastSelectedWord(CssRect word) { for (int i = 0; i < _words.Count - 1; i++) { if (_words[i] == word) { return(!_words[i + 1].Selected); } } return(true); }
/// <summary> /// Sets the baseline of the words of the specified box to certain height /// </summary> /// <param name="g">Device info</param> /// <param name="b">box to check words</param> /// <param name="baseline">baseline</param> internal void SetBaseLine(IGraphics g, CssBox b, float baseline) { //TODO: Aqui me quede, checar poniendo "by the" con un font-size de 3em List <CssRect> ws = WordsOf(b); if (!Rectangles.ContainsKey(b)) { return; } RectangleF r = Rectangles[b]; //Save top of words related to the top of rectangle float gap = 0f; if (ws.Count > 0) { gap = ws[0].Top - r.Top; } else { CssRect firstw = b.FirstWordOccourence(b, this); if (firstw != null) { gap = firstw.Top - r.Top; } } //New top that words will have //float newtop = baseline - (Height - OwnerBox.FontDescent - 3); //OLD float newtop = baseline;// -GetBaseLineHeight(b, g); //OLD if (b.ParentBox != null && b.ParentBox.Rectangles.ContainsKey(this) && r.Height < b.ParentBox.Rectangles[this].Height) { //Do this only if rectangle is shorter than parent's float recttop = newtop - gap; RectangleF newr = new RectangleF(r.X, recttop, r.Width, r.Height); Rectangles[b] = newr; b.OffsetRectangle(this, gap); } foreach (var word in ws) { if (!word.IsImage) { word.Top = newtop; } } }
/// <summary> /// Clear the current selection. /// </summary> private void ClearSelection() { // clear drag and drop _dragDropData = null; ClearSelection(_root); _selectionStartOffset = -1; _selectionStartIndex = -1; _selectionEndOffset = -1; _selectionEndIndex = -1; _selectionStartPoint = Point.Empty; _selectionStart = null; _selectionEnd = null; }
/// <summary> /// Handle html text selection by mouse move over the html with left mouse button pressed.<br/> /// Calculate the words in the selected range and set their selected property. /// </summary> /// <param name="control">the control hosting the html to invalidate</param> /// <param name="loc">the mouse location</param> /// <param name="allowPartialSelect">true - partial word selection allowed, false - only full words selection</param> private void HandleSelection(Control control, Point loc, bool allowPartialSelect) { // get the line under the mouse or nearest from the top var lineBox = DomUtils.GetCssLineBox(_root, loc); if (lineBox != null) { // get the word under the mouse var word = DomUtils.GetCssBoxWord(lineBox, loc); // if no word found under the mouse use the last or the first word in the line if (word == null && lineBox.Words.Count > 0) { if (loc.Y > lineBox.LineBottom) { // under the line word = lineBox.Words[lineBox.Words.Count - 1]; } else if (loc.X < lineBox.Words[0].Left) { // before the line word = lineBox.Words[0]; } else if (loc.X > lineBox.Words[lineBox.Words.Count - 1].Right) { // at the end of the line word = lineBox.Words[lineBox.Words.Count - 1]; } } // if there is matching word if (word != null) { if (_selectionStart == null) { // on start set the selection start word _selectionStartPoint = loc; _selectionStart = word; if (allowPartialSelect) CalculateWordCharIndexAndOffset(control, word, loc, true); } // always set selection end word _selectionEnd = word; if (allowPartialSelect) CalculateWordCharIndexAndOffset(control, word, loc, false); ClearSelection(_root); if (CheckNonEmptySelection(loc, allowPartialSelect)) { CheckSelectionDirection(); SelectWordsInRange(_root, _backwardSelection ? _selectionEnd : _selectionStart, _backwardSelection ? _selectionStart : _selectionEnd); } else { _selectionEnd = null; } _cursorChanged = true; control.Cursor = Cursors.IBeam; control.Invalidate(); } } }
/// <summary> /// Calculate the character index and offset by characters for the given word and given offset.<br/> /// If the location is below the word line then set the selection to the end.<br/> /// If the location is to the right of the word then set the selection to the end.<br/> /// If the offset is to the left of the word set the selection to the beginning.<br/> /// Otherwise calculate the width of each substring to find the char the location is on. /// </summary> /// <param name="control">used to create graphics to measure string</param> /// <param name="word">the word to calculate its index and offset</param> /// <param name="loc">the location to calculate for</param> /// <param name="selectionIndex">return the index of the char under the location</param> /// <param name="selectionOffset">return the offset of the char under the location</param> /// <param name="inclusive">is to include the first character in the calculation</param> private static void CalculateWordCharIndexAndOffset(Control control, CssRect word, Point loc, bool inclusive, out int selectionIndex, out float selectionOffset) { selectionIndex = 0; selectionOffset = 0f; var offset = loc.X - word.Left; if (word.Text == null) { // not a text word - set full selection selectionIndex = -1; selectionOffset = -1; } else if (offset > word.Width - word.OwnerBox.ActualWordSpacing || loc.Y > DomUtils.GetCssLineBoxByWord(word).LineBottom) { // mouse under the line, to the right of the word - set to the end of the word selectionIndex = word.Text.Length; selectionOffset = word.Width; } else if (offset > 0) { // calculate partial word selection var font = word.OwnerBox.ActualFont; using (var g = new WinGraphics(control.CreateGraphics())) { int charFit; int charFitWidth; var maxWidth = offset + ( inclusive ? 0 : 1.5f*word.LeftGlyphPadding ); g.MeasureString(word.Text, font, maxWidth, out charFit, out charFitWidth); selectionIndex = charFit; selectionOffset = charFitWidth; } } }
/// <summary> /// Calculate the character index and offset by characters for the given word and given offset.<br/> /// <seealso cref="CalculateWordCharIndexAndOffset(Control, CssRect, Point, bool, out int, out float)"/>. /// </summary> /// <param name="control">used to create graphics to measure string</param> /// <param name="word">the word to calculate its index and offset</param> /// <param name="loc">the location to calculate for</param> /// <param name="selectionStart">to set the starting or ending char and offset data</param> private void CalculateWordCharIndexAndOffset(Control control, CssRect word, Point loc, bool selectionStart) { int selectionIndex; float selectionOffset; CalculateWordCharIndexAndOffset(control, word, loc, selectionStart, out selectionIndex, out selectionOffset); if(selectionStart) { _selectionStartIndex = selectionIndex; _selectionStartOffset = selectionOffset; } else { _selectionEndIndex = selectionIndex; _selectionEndOffset = selectionOffset; } }
/// <summary> /// The selection start index if the first selected word is partially selected (-1 if not selected or fully selected)<br/> /// if the given word is not starting or ending selection word -1 is returned as full word selection is in place. /// </summary> /// <remarks> /// Handles backward selecting by returning the selection end data instead of start. /// </remarks> /// <param name="word">the word to return the selection start index for</param> /// <returns>data value or -1 if not aplicable</returns> public int GetSelectingStartIndex(CssRect word) { return word == (_backwardSelection ? _selectionEnd : _selectionStart) ? (_backwardSelection ? _selectionEndIndex : _selectionStartIndex) : -1; }
/// <summary> /// Select the word at the given location if found. /// </summary> /// <param name="control">the control hosting the html to invalidate</param> /// <param name="loc">the location to select word at</param> public void SelectWord(Control control, Point loc) { if (_root.HtmlContainer.IsSelectionEnabled) { var word = DomUtils.GetCssBoxWord(_root, loc); if (word != null) { word.Selection = this; _selectionStartPoint = loc; _selectionStart = _selectionEnd = word; control.Invalidate(); } } }
/// <summary> /// The selection end index if the last selected word is partially selected (-1 if not selected or fully selected)<br/> /// if the given word is not starting or ending selection word -1 is returned as full word selection is in place. /// </summary> /// <remarks> /// Handles backward selecting by returning the selection end data instead of start. /// </remarks> /// <param name="word">the word to return the selection end index for</param> public int GetSelectedEndIndexOffset(CssRect word) { return word == (_backwardSelection ? _selectionStart : _selectionEnd) ? (_backwardSelection ? _selectionStartIndex : _selectionEndIndex) : -1; }
/// <summary> /// The selection start offset if the first selected word is partially selected (-1 if not selected or fully selected)<br/> /// if the given word is not starting or ending selection word -1 is returned as full word selection is in place. /// </summary> /// <remarks> /// Handles backward selecting by returning the selection end data instead of start. /// </remarks> /// <param name="word">the word to return the selection start offset for</param> public float GetSelectedStartOffset(CssRect word) { return word == (_backwardSelection ? _selectionEnd : _selectionStart) ? (_backwardSelection ? _selectionEndOffset : _selectionStartOffset) : -1; }
/// <summary> /// Select all the words that are between <paramref name="selectionStart"/> word and <paramref name="selectionEnd"/> word in the DOM hierarchy.<br/> /// </summary> /// <param name="root">the root of the DOM sub-tree the selection is in</param> /// <param name="selectionStart">selection start word limit</param> /// <param name="selectionEnd">selection end word limit</param> private void SelectWordsInRange(CssBox root, CssRect selectionStart, CssRect selectionEnd) { bool inSelection = false; SelectWordsInRange(root, selectionStart, selectionEnd, ref inSelection); }
/// <summary> /// Gets the longest word (in width) inside the box, deeply. /// </summary> /// <param name="box"></param> /// <param name="maxWidth"> </param> /// <param name="maxWidthWord"> </param> /// <returns></returns> private static void GetMinimumWidth_LongestWord(CssBox box, ref float maxWidth, ref CssRect maxWidthWord) { if (box.Words.Count > 0) { foreach (CssRect cssRect in box.Words) { if (cssRect.Width > maxWidth) { maxWidth = cssRect.Width; maxWidthWord = cssRect; } } } else { foreach (CssBox childBox in box.Boxes) GetMinimumWidth_LongestWord(childBox, ref maxWidth, ref maxWidthWord); } }
/// <summary> /// Show context menu clicked on given rectangle. /// </summary> /// <param name="parent">the parent control to show the context menu on</param> /// <param name="rect">the rectangle that was clicked to show context menu</param> /// <param name="link">the link that was clicked to show context menu on</param> public void ShowContextMenu(Control parent, CssRect rect, CssBox link) { try { DisposeContextMenu(); _parentControl = parent; _currentRect = rect; _currentLink = link; _contextMenu = new ContextMenuStrip(); _contextMenu.ShowImageMargin = false; if(rect != null) { bool isVideo = false; if (link != null) { isVideo = link is CssBoxFrame && ((CssBoxFrame)link).IsVideo; var openLink = _contextMenu.Items.Add(isVideo ? _openVideo : _openLink, null, OnOpenLinkClick); if(_htmlContainer.IsSelectionEnabled) { var copyLink = _contextMenu.Items.Add(isVideo ? _copyVideoUrl : _copyLink, null, OnCopyLinkClick); copyLink.Enabled = !string.IsNullOrEmpty(link.HrefLink); } openLink.Enabled = !string.IsNullOrEmpty(link.HrefLink); _contextMenu.Items.Add("-"); } if (rect.IsImage && !isVideo) { var saveImage = _contextMenu.Items.Add(_saveImage, null, OnSaveImageClick); if(_htmlContainer.IsSelectionEnabled) { var copyImageUrl = _contextMenu.Items.Add(_copyImageLink, null, OnCopyImageLinkClick); var copyImage = _contextMenu.Items.Add(_copyImage, null, OnCopyImageClick); copyImageUrl.Enabled = !string.IsNullOrEmpty(_currentRect.OwnerBox.GetAttribute("src")); copyImage.Enabled = rect.Image != null; } saveImage.Enabled = rect.Image != null; _contextMenu.Items.Add("-"); } if(_htmlContainer.IsSelectionEnabled) { var copy = _contextMenu.Items.Add(_copy, null, OnCopyClick); copy.Enabled = rect.Selected; } } if(_htmlContainer.IsSelectionEnabled) { _contextMenu.Items.Add(_selectAll, null, OnSelectAllClick); } if(_contextMenu.Items.Count > 0) { if(_contextMenu.Items[_contextMenu.Items.Count-1].Text == string.Empty) _contextMenu.Items.RemoveAt(_contextMenu.Items.Count - 1); _contextMenu.Show(parent, parent.PointToClient(Control.MousePosition)); } } catch (Exception ex) { _htmlContainer.ReportError(HtmlRenderErrorType.ContextMenu, "Failed to show context menu", ex); } }
/// <summary> /// Get the selected word with respect to partial selected words. /// </summary> /// <param name="rect">the word to append</param> /// <param name="selectedText">is to get selected text or all the text in the word</param> private static string GetSelectedWord(CssRect rect, bool selectedText) { if (selectedText && rect.SelectedStartIndex > -1 && rect.SelectedEndIndexOffset > -1) { return rect.Text.Substring(rect.SelectedStartIndex, rect.SelectedEndIndexOffset - rect.SelectedStartIndex); } else if (selectedText && rect.SelectedStartIndex > -1) { return rect.Text.Substring(rect.SelectedStartIndex) + (rect.HasSpaceAfter ? " " : ""); } else if (selectedText && rect.SelectedEndIndexOffset > -1) { return rect.Text.Substring(0, rect.SelectedEndIndexOffset); } else { var whitespaceBefore = rect.OwnerBox.Words[0] == rect ? IsBoxHasWhitespace(rect.OwnerBox) : rect.HasSpaceBefore; return (whitespaceBefore ? " " : "") + rect.Text + (rect.HasSpaceAfter ? " " : ""); } }
/// <summary> /// Find the css line box that the given word is in. /// </summary> /// <param name="word">the word to search for it's line box</param> /// <returns>line box that the word is in</returns> public static CssLineBox GetCssLineBoxByWord(CssRect word) { var box = word.OwnerBox; while (box.LineBoxes.Count == 0) { box = box.ParentBox; } foreach (var lineBox in box.LineBoxes) { foreach (var lineWord in lineBox.Words) { if(lineWord == word) { return lineBox; } } } return box.LineBoxes[0]; }
/// <summary> /// Dispose of the last used context menu. /// </summary> private void DisposeContextMenu() { try { if (_contextMenu != null) _contextMenu.Dispose(); _contextMenu = null; _parentControl = null; _currentRect = null; _currentLink = null; } catch {} }
/// <summary> /// Select all the words that are between <paramref name="selectionStart"/> word and <paramref name="selectionEnd"/> word in the DOM hierarchy. /// </summary> /// <param name="box">the current traversal node</param> /// <param name="selectionStart">selection start word limit</param> /// <param name="selectionEnd">selection end word limit</param> /// <param name="inSelection">used to know the traversal is currently in selected range</param> /// <returns></returns> private bool SelectWordsInRange(CssBox box, CssRect selectionStart, CssRect selectionEnd, ref bool inSelection) { foreach (var boxWord in box.Words) { if (!inSelection && boxWord == selectionStart) { inSelection = true; } if (inSelection) { boxWord.Selection = this; if (selectionStart == selectionEnd || boxWord == selectionEnd) { return true; } } } foreach (var childBox in box.Boxes) { if (SelectWordsInRange(childBox, selectionStart, selectionEnd, ref inSelection)) { return true; } } return false; }
/// <summary> /// Check if the given word is the last selected word in the line.<br/> /// It can either be the last word in the line or the next word has no selection. /// </summary> /// <param name="word">the word to check</param> /// <returns></returns> public bool IsLastSelectedWord(CssRect word) { for(int i = 0; i < _words.Count-1; i++) { if( _words[i] == word ) { return !_words[i + 1].Selected; } } return true; }