18bcc9a6f2
- Asynchronously load source map from sourceMappingURL - Once loaded, annotate source positions with their original position - Update script panel tooltip to include link to original source - For the above, make DOM.element a slightly more flexible API, allowing defining attributes and children - Also fix ToolTipEvent handling to support nodes. - Shuffle around some code to make createScriptNode async, in case we want to load the source map when building the script node itself. - Drive-by: make source markers a simple backgroundColor when there is only one group. Change-Id: I0926807761cbfe8b6dd8ff5154815a7e5ccb39bf Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2972827 Commit-Queue: Leszek Swirski <leszeks@chromium.org> Commit-Queue: Camillo Bruni <cbruni@chromium.org> Auto-Submit: Leszek Swirski <leszeks@chromium.org> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Cr-Commit-Position: refs/heads/master@{#75302}
133 lines
4.0 KiB
JavaScript
133 lines
4.0 KiB
JavaScript
// Copyright 2020 the V8 project authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
import {DOM, V8CustomElement} from './helper.mjs';
|
|
|
|
DOM.defineCustomElement(
|
|
'view/tool-tip', (templateText) => class Tooltip extends V8CustomElement {
|
|
_targetNode;
|
|
_content;
|
|
_isHidden = true;
|
|
|
|
constructor() {
|
|
super(templateText);
|
|
this._intersectionObserver = new IntersectionObserver((entries) => {
|
|
if (entries[0].intersectionRatio <= 0) {
|
|
this.hide();
|
|
} else {
|
|
this.show();
|
|
this.requestUpdate(true);
|
|
}
|
|
});
|
|
document.addEventListener('click', (event) => {
|
|
// Only hide the tooltip if we click anywhere outside of it.
|
|
let target = event.target;
|
|
while (target) {
|
|
if (target == this) return;
|
|
target = target.parentNode;
|
|
}
|
|
this.hide()
|
|
});
|
|
}
|
|
|
|
_update() {
|
|
if (!this._targetNode || this._isHidden) return;
|
|
const rect = this._targetNode.getBoundingClientRect();
|
|
rect.x += rect.width / 2;
|
|
let atRight = this._useRight(rect.x);
|
|
let atBottom = this._useBottom(rect.y);
|
|
if (atBottom) rect.y += rect.height;
|
|
this._setPosition(rect, atRight, atBottom);
|
|
this.requestUpdate(true);
|
|
}
|
|
|
|
set positionOrTargetNode(positionOrTargetNode) {
|
|
if (positionOrTargetNode.nodeType === undefined) {
|
|
this.position = positionOrTargetNode;
|
|
} else {
|
|
this.targetNode = positionOrTargetNode;
|
|
}
|
|
}
|
|
|
|
set targetNode(targetNode) {
|
|
this._intersectionObserver.disconnect();
|
|
this._targetNode = targetNode;
|
|
if (targetNode === undefined) return;
|
|
if (!(targetNode instanceof SVGElement)) {
|
|
this._intersectionObserver.observe(targetNode);
|
|
}
|
|
this.requestUpdate(true);
|
|
}
|
|
|
|
set position(position) {
|
|
this._targetNode = undefined;
|
|
this._setPosition(
|
|
position, this._useRight(position.x), this._useBottom(position.y));
|
|
}
|
|
|
|
_setPosition(viewportPosition, atRight, atBottom) {
|
|
const horizontalMode = atRight ? 'right' : 'left';
|
|
const verticalMode = atBottom ? 'bottom' : 'top';
|
|
this.bodyNode.className = horizontalMode + ' ' + verticalMode;
|
|
const pageX = viewportPosition.x + window.scrollX;
|
|
this.style.left = `${pageX}px`;
|
|
const pageY = viewportPosition.y + window.scrollY;
|
|
this.style.top = `${pageY}px`;
|
|
}
|
|
|
|
_useBottom(viewportY) {
|
|
return viewportY <= 400;
|
|
}
|
|
|
|
_useRight(viewportX) {
|
|
return viewportX < document.documentElement.clientWidth / 2;
|
|
}
|
|
|
|
set content(content) {
|
|
if (!content) return this.hide();
|
|
this.show();
|
|
if (typeof content === 'string') {
|
|
this.contentNode.innerHTML = content;
|
|
this.contentNode.className = 'textContent';
|
|
} else if (content?.nodeType && content?.nodeName) {
|
|
this._setContentNode(content);
|
|
} else {
|
|
if (this.contentNode.firstChild?.localName == 'property-link-table') {
|
|
this.contentNode.firstChild.propertyDict = content;
|
|
} else {
|
|
const node = DOM.element('property-link-table');
|
|
node.instanceLinkButtons = true;
|
|
node.propertyDict = content;
|
|
this._setContentNode(node);
|
|
}
|
|
}
|
|
}
|
|
|
|
_setContentNode(content) {
|
|
const newContent = DOM.div();
|
|
newContent.appendChild(content);
|
|
this.contentNode.replaceWith(newContent);
|
|
newContent.id = 'content';
|
|
}
|
|
|
|
hide() {
|
|
this._isHidden = true;
|
|
this.bodyNode.style.display = 'none';
|
|
this.targetNode = undefined;
|
|
}
|
|
|
|
show() {
|
|
this.bodyNode.style.display = 'block';
|
|
this._isHidden = false;
|
|
}
|
|
|
|
get bodyNode() {
|
|
return this.$('#body');
|
|
}
|
|
|
|
get contentNode() {
|
|
return this.$('#content');
|
|
}
|
|
});
|