blob: b345b656d654a33ab2e77be1236152e0bb06f3ac [file] [log] [blame]
/*
* Copyright (C) 2013-2017 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
WI.SearchResultTreeElement = class SearchResultTreeElement extends WI.GeneralTreeElement
{
constructor(representedObject)
{
console.assert(representedObject instanceof WI.DOMSearchMatchObject || representedObject instanceof WI.SourceCodeSearchMatchObject);
var title = WI.SearchResultTreeElement.truncateAndHighlightTitle(representedObject.title, representedObject.searchTerm, representedObject.sourceCodeTextRange);
const subtitle = null;
super(representedObject.className, title, subtitle, representedObject);
}
// Static
static truncateAndHighlightTitle(title, searchTerm, sourceCodeTextRange)
{
let isRTL = WI.resolvedLayoutDirection() === WI.LayoutDirection.RTL;
const charactersToShowBeforeSearchMatch = isRTL ? 20 : 15;
const charactersToShowAfterSearchMatch = isRTL ? 15 : 50;
// Use the original location, since those line/column offsets match the line text in title.
var textRange = sourceCodeTextRange.textRange;
let searchTermIndex = textRange.startColumn;
let searchTermLength = textRange.endColumn - textRange.startColumn;
// We should only have one line text ranges, so make sure that is the case.
console.assert(textRange.startLine === textRange.endLine);
// Show some characters before the matching text (if there are enough) for context. TreeOutline takes care of the truncating
// at the end of the string.
var modifiedTitle = null;
if (searchTermIndex > charactersToShowBeforeSearchMatch) {
modifiedTitle = ellipsis + title.substring(searchTermIndex - charactersToShowBeforeSearchMatch);
searchTermIndex = charactersToShowBeforeSearchMatch + 1;
} else
modifiedTitle = title;
modifiedTitle = modifiedTitle.truncateEnd(searchTermIndex + searchTermLength + charactersToShowAfterSearchMatch);
var highlightedTitle = document.createDocumentFragment();
highlightedTitle.append(modifiedTitle.substring(0, searchTermIndex));
var highlightSpan = document.createElement("span");
highlightSpan.className = "highlighted";
highlightSpan.append(modifiedTitle.substring(searchTermIndex, searchTermIndex + searchTermLength));
highlightedTitle.appendChild(highlightSpan);
highlightedTitle.append(modifiedTitle.substring(searchTermIndex + searchTermLength));
return highlightedTitle;
}
// Public
get filterableData()
{
return {text: [this.representedObject.title]};
}
get synthesizedTextValue()
{
return this.representedObject.sourceCodeTextRange.synthesizedTextValue + ":" + this.representedObject.title;
}
// Protected
populateContextMenu(contextMenu, event)
{
if (this.representedObject instanceof WI.DOMSearchMatchObject) {
contextMenu.appendItem(WI.UIString("Reveal in Elements Tab"), () => {
WI.showMainFrameDOMTree(this.representedObject.domNode, {
ignoreSearchTab: true,
});
});
} else if (this.representedObject instanceof WI.SourceCodeSearchMatchObject) {
contextMenu.appendItem(WI.UIString("Reveal in Sources Tab"), () => {
WI.showOriginalOrFormattedSourceCodeTextRange(this.representedObject.sourceCodeTextRange, {
ignoreNetworkTab: true,
ignoreSearchTab: true,
});
});
}
super.populateContextMenu(contextMenu, event);
}
};