UNPKG

debug-server-next

Version:

Dev server for hippy-core.

342 lines (341 loc) 15.2 kB
// Copyright 2017 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. /* eslint-disable rulesdir/no_underscored_properties */ import * as Common from '../../core/common/common.js'; import * as i18n from '../../core/i18n/i18n.js'; import * as Diff from '../../third_party/diff/diff.js'; import * as UI from '../../ui/legacy/legacy.js'; import * as WorkspaceDiff from '../../models/workspace_diff/workspace_diff.js'; import { ChangesSidebar } from './ChangesSidebar.js'; import { ChangesTextEditor } from './ChangesTextEditor.js'; const UIStrings = { /** *@description Screen-reader accessible name for the code editor in the Changes tool showing the user's changes. */ changesDiffViewer: 'Changes diff viewer', /** *@description Screen reader/tooltip label for a button in the Changes tool that reverts all changes to the currently open file. */ revertAllChangesToCurrentFile: 'Revert all changes to current file', /** *@description Text in Changes View of the Changes tab */ noChanges: 'No changes', /** *@description Text in Changes View of the Changes tab */ binaryData: 'Binary data', /** * @description Text in the Changes tab that indicates how many lines of code have changed in the * selected file. An insertion refers to an added line of code. The (+) is a visual cue to indicate * lines were added (not translatable). */ sInsertions: '{n, plural, =1 {# insertion (+)} other {# insertions (+)}}', /** * @description Text in the Changes tab that indicates how many lines of code have changed in the * selected file. A deletion refers to a removed line of code. The (-) is a visual cue to indicate * lines were removed (not translatable). */ sDeletions: '{n, plural, =1 {# deletion (-)} other {# deletions (-)}}', /** *@description Text in Changes View of the Changes tab *@example {2} PH1 */ SkippingDMatchingLines: '( … Skipping {PH1} matching lines … )', }; const str_ = i18n.i18n.registerUIStrings('panels/changes/ChangesView.ts', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); let changesViewInstance; export class ChangesView extends UI.Widget.VBox { _emptyWidget; _workspaceDiff; _changesSidebar; _selectedUISourceCode; _diffRows; _maxLineDigits; _editor; _toolbar; _diffStats; constructor() { super(true); this.registerRequiredCSS('panels/changes/changesView.css'); const splitWidget = new UI.SplitWidget.SplitWidget(true /* vertical */, false /* sidebar on left */); const mainWidget = new UI.Widget.Widget(); splitWidget.setMainWidget(mainWidget); splitWidget.show(this.contentElement); this._emptyWidget = new UI.EmptyWidget.EmptyWidget(''); this._emptyWidget.show(mainWidget.element); this._workspaceDiff = WorkspaceDiff.WorkspaceDiff.workspaceDiff(); this._changesSidebar = new ChangesSidebar(this._workspaceDiff); this._changesSidebar.addEventListener("SelectedUISourceCodeChanged" /* SelectedUISourceCodeChanged */, this._selectedUISourceCodeChanged, this); splitWidget.setSidebarWidget(this._changesSidebar); this._selectedUISourceCode = null; this._diffRows = []; this._maxLineDigits = 1; this._editor = new ChangesTextEditor({ bracketMatchingSetting: undefined, devtoolsAccessibleName: i18nString(UIStrings.changesDiffViewer), lineNumbers: true, lineWrapping: false, mimeType: undefined, autoHeight: undefined, padBottom: undefined, maxHighlightLength: Infinity, placeholder: undefined, lineWiseCopyCut: undefined, inputStyle: undefined, }); this._editor.setReadOnly(true); const editorContainer = mainWidget.element.createChild('div', 'editor-container'); UI.ARIAUtils.markAsTabpanel(editorContainer); this._editor.show(editorContainer); this._editor.hideWidget(); self.onInvokeElement(this._editor.element, this._click.bind(this)); this._toolbar = new UI.Toolbar.Toolbar('changes-toolbar', mainWidget.element); const revertButton = new UI.Toolbar.ToolbarButton(i18nString(UIStrings.revertAllChangesToCurrentFile), 'largeicon-undo'); revertButton.addEventListener(UI.Toolbar.ToolbarButton.Events.Click, this._revert.bind(this)); this._toolbar.appendToolbarItem(revertButton); this._diffStats = new UI.Toolbar.ToolbarText(''); this._toolbar.appendToolbarItem(this._diffStats); this._toolbar.setEnabled(false); this._hideDiff(i18nString(UIStrings.noChanges)); this._selectedUISourceCodeChanged(); } static instance(opts = { forceNew: null }) { const { forceNew } = opts; if (!changesViewInstance || forceNew) { changesViewInstance = new ChangesView(); } return changesViewInstance; } _selectedUISourceCodeChanged() { this._revealUISourceCode(this._changesSidebar.selectedUISourceCode()); } _revert() { const uiSourceCode = this._selectedUISourceCode; if (!uiSourceCode) { return; } this._workspaceDiff.revertToOriginal(uiSourceCode); } _click(event) { const selection = this._editor.selection(); if (!selection.isEmpty() || !this._selectedUISourceCode) { return; } const row = this._diffRows[selection.startLine]; Common.Revealer.reveal(this._selectedUISourceCode.uiLocation(row.currentLineNumber - 1, selection.startColumn), false); event.consume(true); } _revealUISourceCode(uiSourceCode) { if (this._selectedUISourceCode === uiSourceCode) { return; } if (this._selectedUISourceCode) { this._workspaceDiff.unsubscribeFromDiffChange(this._selectedUISourceCode, this._refreshDiff, this); } if (uiSourceCode && this.isShowing()) { this._workspaceDiff.subscribeToDiffChange(uiSourceCode, this._refreshDiff, this); } this._selectedUISourceCode = uiSourceCode; this._refreshDiff(); } wasShown() { this._refreshDiff(); } _refreshDiff() { if (!this.isShowing()) { return; } if (!this._selectedUISourceCode) { this._renderDiffRows(null); return; } const uiSourceCode = this._selectedUISourceCode; if (!uiSourceCode.contentType().isTextType()) { this._hideDiff(i18nString(UIStrings.binaryData)); return; } this._workspaceDiff.requestDiff(uiSourceCode).then((diff) => { if (this._selectedUISourceCode !== uiSourceCode) { return; } this._renderDiffRows(diff); }); } _hideDiff(message) { this._diffStats.setText(''); this._toolbar.setEnabled(false); this._editor.hideWidget(); this._emptyWidget.text = message; this._emptyWidget.showWidget(); } _renderDiffRows(diff) { this._diffRows = []; if (!diff || (diff.length === 1 && diff[0][0] === Diff.Diff.Operation.Equal)) { this._hideDiff(i18nString(UIStrings.noChanges)); return; } let insertions = 0; let deletions = 0; let currentLineNumber = 0; let baselineLineNumber = 0; const paddingLines = 3; const originalLines = []; const currentLines = []; for (let i = 0; i < diff.length; ++i) { const token = diff[i]; switch (token[0]) { case Diff.Diff.Operation.Equal: this._diffRows.push(...createEqualRows(token[1], i === 0, i === diff.length - 1)); originalLines.push(...token[1]); currentLines.push(...token[1]); break; case Diff.Diff.Operation.Insert: for (const line of token[1]) { this._diffRows.push(createRow(line, "addition" /* Addition */)); } insertions += token[1].length; currentLines.push(...token[1]); break; case Diff.Diff.Operation.Delete: deletions += token[1].length; originalLines.push(...token[1]); if (diff[i + 1] && diff[i + 1][0] === Diff.Diff.Operation.Insert) { i++; this._diffRows.push(...createModifyRows(token[1].join('\n'), diff[i][1].join('\n'))); insertions += diff[i][1].length; currentLines.push(...diff[i][1]); } else { for (const line of token[1]) { this._diffRows.push(createRow(line, "deletion" /* Deletion */)); } } break; } } this._maxLineDigits = Math.ceil(Math.log10(Math.max(currentLineNumber, baselineLineNumber))); const insertionText = i18nString(UIStrings.sInsertions, { n: insertions }); const deletionText = i18nString(UIStrings.sDeletions, { n: deletions }); this._diffStats.setText(`${insertionText}, ${deletionText}`); this._toolbar.setEnabled(true); this._emptyWidget.hideWidget(); this._editor.operation(() => { this._editor.showWidget(); this._editor.setHighlightMode({ name: 'devtools-diff', diffRows: this._diffRows, mimeType: /** @type {!Workspace.UISourceCode.UISourceCode} */ this._selectedUISourceCode .mimeType(), baselineLines: originalLines, currentLines: currentLines, }); this._editor.setText(this._diffRows .map((row) => row.tokens.map((t) => t.text).join('')) .join('\n')); this._editor.setLineNumberFormatter(this._lineFormatter.bind(this)); this._editor.updateDiffGutter(this._diffRows); }); function createEqualRows(lines, atStart, atEnd) { const equalRows = []; if (!atStart) { for (let i = 0; i < paddingLines && i < lines.length; i++) { equalRows.push(createRow(lines[i], "equal" /* Equal */)); } if (lines.length > paddingLines * 2 + 1 && !atEnd) { equalRows.push(createRow(i18nString(UIStrings.SkippingDMatchingLines, { PH1: (lines.length - paddingLines * 2) }), "spacer" /* Spacer */)); } } if (!atEnd) { const start = Math.max(lines.length - paddingLines - 1, atStart ? 0 : paddingLines); let skip = lines.length - paddingLines - 1; if (!atStart) { skip -= paddingLines; } if (skip > 0) { baselineLineNumber += skip; currentLineNumber += skip; } for (let i = start; i < lines.length; i++) { equalRows.push(createRow(lines[i], "equal" /* Equal */)); } } return equalRows; } function createModifyRows(before, after) { const internalDiff = Diff.Diff.DiffWrapper.charDiff(before, after, true /* cleanup diff */); const deletionRows = [createRow('', "deletion" /* Deletion */)]; const insertionRows = [createRow('', "addition" /* Addition */)]; for (const token of internalDiff) { const text = token[1]; const type = token[0]; const className = type === Diff.Diff.Operation.Equal ? '' : 'inner-diff'; const lines = text.split('\n'); for (let i = 0; i < lines.length; i++) { if (i > 0 && type !== Diff.Diff.Operation.Insert) { deletionRows.push(createRow('', "deletion" /* Deletion */)); } if (i > 0 && type !== Diff.Diff.Operation.Delete) { insertionRows.push(createRow('', "addition" /* Addition */)); } if (!lines[i]) { continue; } if (type !== Diff.Diff.Operation.Insert) { deletionRows[deletionRows.length - 1].tokens.push({ text: lines[i], className }); } if (type !== Diff.Diff.Operation.Delete) { insertionRows[insertionRows.length - 1].tokens.push({ text: lines[i], className }); } } } return deletionRows.concat(insertionRows); } function createRow(text, type) { if (type === "addition" /* Addition */) { currentLineNumber++; } if (type === "deletion" /* Deletion */) { baselineLineNumber++; } if (type === "equal" /* Equal */) { baselineLineNumber++; currentLineNumber++; } return { baselineLineNumber, currentLineNumber, tokens: text ? [{ text, className: 'inner-diff' }] : [], type }; } } _lineFormatter(lineNumber) { const row = this._diffRows[lineNumber - 1]; let showBaseNumber = row.type === "deletion" /* Deletion */; let showCurrentNumber = row.type === "addition" /* Addition */; if (row.type === "equal" /* Equal */) { showBaseNumber = true; showCurrentNumber = true; } const baseText = showBaseNumber ? String(row.baselineLineNumber) : ''; const base = baseText.padStart(this._maxLineDigits, '\xA0'); const currentText = showCurrentNumber ? String(row.currentLineNumber) : ''; const current = currentText.padStart(this._maxLineDigits, '\xA0'); return base + '\xA0' + current; } } let diffUILocationRevealerInstance; export class DiffUILocationRevealer { static instance(opts = { forceNew: false }) { const { forceNew } = opts; if (!diffUILocationRevealerInstance || forceNew) { diffUILocationRevealerInstance = new DiffUILocationRevealer(); } return diffUILocationRevealerInstance; } async reveal(diffUILocation, omitFocus) { if (!(diffUILocation instanceof WorkspaceDiff.WorkspaceDiff.DiffUILocation)) { throw new Error('Internal error: not a diff ui location'); } await UI.ViewManager.ViewManager.instance().showView('changes.changes'); ChangesView.instance()._changesSidebar.selectUISourceCode(diffUILocation.uiSourceCode, omitFocus); } }