diff --git a/package.json b/package.json index 924b108b..f314b07e 100644 --- a/package.json +++ b/package.json @@ -193,6 +193,12 @@ "description": "Enable/disable autoclosing of XML tags. \n\nIMPORTANT: Turn off editor.autoClosingTags for this to work", "scope": "window" }, + "xml.mirrorCursorOnMatchingTag": { + "type": "boolean", + "scope": "resource", + "default": false, + "description": "Adds an additional cursor on the matching tag, allows for start/end tag editing." + }, "xml.codeLens.enabled": { "type": "boolean", "default": false, @@ -295,6 +301,14 @@ "fileMatch": "package.json", "url": "./schemas/package.schema.json" } + ], + "keybindings":[ + { + "command": "xml.toggleMatchingTagEdit", + "key": "ctrl+shift+f2", + "mac": "cmd+shift+f2", + "when": "editorFocus" + } ] } } diff --git a/src/extension.ts b/src/extension.ts index 817ebe83..ce604124 100644 --- a/src/extension.ts +++ b/src/extension.ts @@ -13,13 +13,14 @@ import { prepareExecutable } from './javaServerStarter'; import { LanguageClientOptions, RevealOutputChannelOn, LanguageClient, DidChangeConfigurationNotification, RequestType, TextDocumentPositionParams, ReferencesRequest } from 'vscode-languageclient'; import * as requirements from './requirements'; -import { languages, IndentAction, workspace, window, commands, ExtensionContext, TextDocument, Position, LanguageConfiguration, Uri, extensions } from "vscode"; +import { languages, IndentAction, workspace, window, commands, ExtensionContext, TextDocument, Position, LanguageConfiguration, Uri, extensions, WorkspaceConfiguration, ConfigurationTarget } from "vscode"; import * as path from 'path'; import * as os from 'os'; import { activateTagClosing, AutoCloseResult } from './tagClosing'; import { Commands } from './commands'; import { onConfigurationChange, subscribeJDKChangeConfiguration } from './settings'; import { collectXmlJavaExtensions, onExtensionChange } from './plugin'; +import { setupMirrorCursor } from './mirrorCursor'; export interface ScopeInfo { scope : "default" | "global" | "workspace" | "folder"; @@ -30,7 +31,9 @@ namespace TagCloseRequest { export const type: RequestType = new RequestType('xml/closeTag'); } - +namespace MatchingTagPositionRequest { + export const type: RequestType = new RequestType('xml/matchingTagPosition'); +} export function activate(context: ExtensionContext) { let storagePath = context.storagePath; @@ -122,6 +125,30 @@ export function activate(context: ExtensionContext) { disposable = activateTagClosing(tagProvider, { xml: true, xsl: true }, Commands.AUTO_CLOSE_TAGS); toDispose.push(disposable); + + //Setup mirrored tag rename request + const matchingTagPositionRequestor = (document: TextDocument, position: Position) => { + let param = languageClient.code2ProtocolConverter.asTextDocumentPositionParams(document, position); + return languageClient.sendRequest(MatchingTagPositionRequest.type, param); + }; + + toDispose.push(setupMirrorCursor(matchingTagPositionRequestor, ['xml', 'xsl'])); + + const matchingTagEditCommand = 'xml.toggleMatchingTagEdit'; + + const matchingTagEditHandler = async () => { + let xmlConfiguration: WorkspaceConfiguration; + if (window.activeTextEditor) { + xmlConfiguration = workspace.getConfiguration('xml', window.activeTextEditor.document.uri); + } else { + xmlConfiguration = workspace.getConfiguration('xml'); + } + const current = xmlConfiguration.mirrorCursorOnMatchingTag; + await updateConfig(xmlConfiguration, 'mirrorCursorOnMatchingTag', !current); + } + + toDispose.push(commands.registerCommand(matchingTagEditCommand, matchingTagEditHandler)); + }); languages.setLanguageConfiguration('xml', getIndentationRules()); languages.setLanguageConfiguration('xsl', getIndentationRules()); @@ -192,3 +219,24 @@ function getIndentationRules(): LanguageConfiguration { }; } +/** + * Update config with the following precedence: WorkspaceFolder -> Workspace -> Global + * @param config config containing the section to update + * @param section section to update + * @param value new value + */ +async function updateConfig(config: WorkspaceConfiguration, section: string, value: any): Promise { + try { + await config.update(section, value); + return; + } catch(e) { + // try ConfigurationTarget.Global + } + + try { + await config.update(section, value, ConfigurationTarget.Global); + return; + } catch(e) { + throw 'Failed to update config'; + } +} \ No newline at end of file diff --git a/src/mirrorCursor.ts b/src/mirrorCursor.ts new file mode 100644 index 00000000..c9b516ab --- /dev/null +++ b/src/mirrorCursor.ts @@ -0,0 +1,457 @@ +import { + window, + workspace, + ConfigurationChangeEvent, + Disposable, + TextDocument, + TextEditorSelectionChangeEvent, + Position, + TextEditor, + Selection, + Range, + WorkspaceEdit +} from 'vscode'; + +let mirrorCursorsUtils: MirrorCursorsUtils; +let mirrorCursors: MirrorCursors | undefined; + +export interface PositionInterface { + line: number; + character: number; +} + +export function setupMirrorCursor(matchingTagPositionProvider: (document: TextDocument, position: Position) => Thenable, supportedLanguages: string[]): Disposable { + mirrorCursorsUtils = new MirrorCursorsUtils(matchingTagPositionProvider, supportedLanguages); + + if (mirrorCursorsUtils.mirrorConfigOn()) { + mirrorCursors = new MirrorCursors(); + } + + return workspace.onDidChangeConfiguration( + (event: ConfigurationChangeEvent) => { + if (!event.affectsConfiguration('xml.mirrorCursorOnMatchingTag')) { + return; + } + + if (mirrorCursorsUtils.mirrorConfigOn()) { + if (mirrorCursors) { + throw 'Mirror cursors turned on but mirror cursors already on. This should never happen'; + } + mirrorCursors = new MirrorCursors(); + } else { + if (!mirrorCursors) { + throw 'Mirror cursors turned off but mirror cursors were already off. This should never happen'; + } + mirrorCursors.cleanup(); + mirrorCursors = undefined; + } + } + ); +} + +class MirrorCursorsUtils { + + private matchingTagPositionProvider: (document: TextDocument, position: Position) => Thenable; + private supportedLanguages; + + constructor(matchingTagPositionProvider: (document: TextDocument, position: Position) => Thenable, + supportedLanguages: string[]) { + + this.matchingTagPositionProvider = matchingTagPositionProvider; + this.supportedLanguages = supportedLanguages; + } + + public getMatchingTagPosition(document: TextDocument, position: Position): Thenable { + + // instantiate `Position` object from `PositionInterface` object + return this.matchingTagPositionProvider(document, position).then((res: PositionInterface | null) => { + if (!res) return null; + return new Position(res.line, res.character); + }); + } + + public isEditorSupportedLanguage(editor: TextEditor | undefined): boolean { + if (!editor) return false; + return this.supportedLanguages.includes(editor.document.languageId); + } + + public mirrorConfigOn(): boolean { + return workspace.getConfiguration('xml').get('mirrorCursorOnMatchingTag'); + } +} + +/** + * This class handles the instantiation and disposal of `CursorUpdater` + * + * Instantiates `CursorUpdater` if a the currently active file is a supported file. + * + * Should only be instantiated if the `xml.mirrorCursorOnMatchingTag` + * config is `true`. Otherwise, should be disposed. + */ +class MirrorCursors { + private currentFileIsSupported: boolean; + private cursorUpdater: CursorUpdater | undefined; + + private disposables: Disposable[]; + + constructor() { + this.currentFileIsSupported = mirrorCursorsUtils.isEditorSupportedLanguage( + window.activeTextEditor + ); + this.disposables = []; + this.setupSupportedFileListener(); + + if (this.currentFileIsSupported) { + this.cursorUpdater = new CursorUpdater(); + } + } + + /** + * Listener that listens to active text editor changes, so that + * we know if the user has currently opened a supported file. + */ + private setupSupportedFileListener(): void { + this.disposables.push(window.onDidChangeActiveTextEditor( + (editor: TextEditor | undefined) => { + this.currentFileIsSupported = mirrorCursorsUtils.isEditorSupportedLanguage(editor); + if (this.unsupportedFileToSupported()) { + this.cursorUpdater = new CursorUpdater(); + } else if (this.supportedFileToUnsupported()) { + this.cursorUpdater.cleanup(); + this.cursorUpdater = undefined; + } else if (this.supportedFileToSupported()) { + this.cursorUpdater.cleanup(); + this.cursorUpdater = new CursorUpdater(); + } + } + )); + } + + private supportedFileToUnsupported(): boolean { + return !this.currentFileIsSupported && !!this.cursorUpdater; + } + + private unsupportedFileToSupported(): boolean { + return this.currentFileIsSupported && !this.cursorUpdater; + } + + private supportedFileToSupported(): boolean { + return this.currentFileIsSupported && !!this.cursorUpdater; + } + + public cleanup(): void { + if (this.cursorUpdater) this.cursorUpdater.cleanup(true); + this.dispose(); + } + + private dispose(): void { + this.disposables.forEach((d: Disposable) => d.dispose()); + } +} + +/** + * This class handles updating the cursor(s) when inside a + * supported file. + * + * Should only be instantiated if the user's active editor + * is a supported file. + * + * Should only persist if the user's active editor is a + * supported file. + */ +class CursorUpdater { + private prevCursors: Selection[]; + private currCursors: Selection[]; + private disposables: Disposable[]; + + /** + * This is to distinguish the difference between being + * in mirror mode and having multiple cursors. + * Being in mirror mode means that there are multiple cursors. + * However, having multiple does not necessarily mean that you are in mirror mode + */ + private isMirrorCursorsOn: boolean; + + constructor() { + if (!window.activeTextEditor) { + throw 'Please do not instantiate this class if there is no active editor'; + } + this.prevCursors = []; + this.currCursors = []; + this.disposables = []; + this.isMirrorCursorsOn = false; + + this.disposables.push( + window.onDidChangeTextEditorSelection(this.updateCursors.bind(this)) + ); + this.updateCursors(); + } + + /** + * Updates cursors in current document + * @param event + */ + private async updateCursors(event?: TextEditorSelectionChangeEvent): Promise { + const editor: TextEditor = window.activeTextEditor; + const textEditor: TextEditor = event ? event.textEditor : editor; + + if (!mirrorCursorsUtils.isEditorSupportedLanguage(textEditor)) { + return; + } + + const selections: Selection[] = event + ? Object.assign([], event.selections) + : editor.selections; + + this.prevCursors = this.currCursors; + + if (this.existsRangedSelection(selections)) { + this.updateCursorsWhenRangedExists(textEditor.document, selections); + } else if (selections.length === 1 && selections[0].isEmpty) { + await this.trySetMirrorCursor(textEditor.document, selections); + } else if (selections.length === 2 && selections[0].isEmpty && selections[1].isEmpty) { + await this.updateCursorsWhenTwoExist(textEditor.document, selections); + } else { + this.currCursors = selections; + } + } + + private updateCursorsWhenRangedExists(document: TextDocument, selections: Selection[]): void { + if (selections.length === 2 && !selections[0].isEmpty && !selections[1].isEmpty) { + const charBeforeAndAfterAnchorPositionsRoughlyEqual = this.isCharBeforeAndAfterPositionsRoughlyEqual( + document, + selections[0].anchor, + selections[1].anchor + ); + + const charBeforeAndAfterActivePositionsRoughlyEqual = this.isCharBeforeAndAfterPositionsRoughlyEqual( + document, + selections[0].active, + selections[1].active + ); + + if (!charBeforeAndAfterAnchorPositionsRoughlyEqual || !charBeforeAndAfterActivePositionsRoughlyEqual) { + this.isMirrorCursorsOn = false; + this.setCurrCursors([selections[0]]); + } + } else { + this.isMirrorCursorsOn = false; + this.currCursors = selections; + } + } + + /** + * Updates cursors when mirror cursors are already on + * @param document + * @param selections + */ + private async updateCursorsWhenTwoExist(document: TextDocument, selections: Selection[]): Promise { + + if (this.isMirrorCursorsOn) { + if (!this.areMirrorCursorsWithinTagName(document, selections[0].anchor, selections[1].anchor)) { + this.setCurrCursors([selections[0]]); + this.isMirrorCursorsOn = false; + return; + } + + if (this.prevCursors.length === 2 && + this.onDifferentLines(selections[0], this.prevCursors[0]) && + this.onDifferentLines(selections[0], this.prevCursors[0]) && + this.onDifferentLines(selections[0], this.prevCursors[0]) && + this.onDifferentLines(selections[1], this.prevCursors[0]) + ) { + await this.trySetMirrorCursor(document, selections); + return; + } + + if ( + this.shouldDoCleanupForXmlAttributeInput( + document, + selections[0].anchor, + selections[1].anchor + ) + ) { + workspace.applyEdit(this.createCleanupForXmlAttributeInput(document, selections[1].anchor)); + this.setCurrCursors([selections[0]]); + this.isMirrorCursorsOn = false; + return; + } + } else { + const matchingTagPosition: Position | null = await mirrorCursorsUtils.getMatchingTagPosition(document, selections[0].active); + if (matchingTagPosition && selections[1].active.isEqual(matchingTagPosition)) { + this.isMirrorCursorsOn = true; + this.currCursors = selections; + return; + } + } + this.currCursors = selections; + } + + private existsRangedSelection(selections: Selection[]): boolean { + return selections.some((s: Selection) => !s.isEmpty); + } + + private onDifferentLines(first: Selection, second: Selection): boolean { + return first.anchor.line !== second.anchor.line; + } + + /** + * Tries to set a mirror cursor if matching tag is available + * + * Returns true if the mirror cursor was set. + * Returns false otherwise. + * @param document + * @param selections + */ + private async trySetMirrorCursor(document: TextDocument, selections: Selection[]): Promise { + const matchingTagPosition: Position | null = await mirrorCursorsUtils.getMatchingTagPosition(document, selections[0].active); + if (!matchingTagPosition) { + this.isMirrorCursorsOn = false; + } else if (this.areMirrorCursorsWithinTagName(document, selections[0].anchor, matchingTagPosition)) { + this.isMirrorCursorsOn = true; + const newCursor = new Selection(matchingTagPosition, matchingTagPosition); + this.setCurrCursors([selections[0], newCursor]); + } + } + + /** + * Returns true if `startCursor` and `endCursor` are mirror tags that are + * within tag names. + * @param document + * @param startCursor + * @param endCursor + */ + private areMirrorCursorsWithinTagName(document: TextDocument, startCursor: Position, endCursor: Position): boolean { + return !this.isPositionsOutsideTags(document, startCursor, endCursor) && + this.isCharBeforeAndAfterPositionsRoughlyEqual(document, startCursor, endCursor); + } + + private setCurrCursors(cursors: Selection[]): void { + window.activeTextEditor.selections = cursors; + this.currCursors = cursors; + } + + private isPositionsOutsideTags(document: TextDocument, firstPos: Position, secondPos: Position): boolean { + const charBeforeFirstPos: string = this.getCharBefore(document, firstPos); + const charAfterFirstPos: string = this.getCharAfter(document, firstPos); + const charBeforeSecondPos: string = this.getCharBefore(document, secondPos); + const charAfterSecondPos: string = this.getCharAfter(document, secondPos); + + /** + * Special case for exiting + * |
+ * |
+ */ + if ( + charBeforeFirstPos === ' ' && + charBeforeSecondPos === ' ' && + charAfterFirstPos === '<' && + charAfterSecondPos === '<' + ) { + return true; + } + /** + * Special case for exiting + * |
+ * |
+ */ + if (charBeforeFirstPos === '\n' && charBeforeSecondPos === '\n') { + return true; + } + /** + * Special case for exiting + *
| + *
| + */ + if (charAfterFirstPos === '\n' && charAfterSecondPos === '\n') { + return true; + } + + return false; + + } + + // Check if chars before and after the two positions are equal + // For the chars before, `<` and `/` are considered equal to handle the case of `<|>` + private isCharBeforeAndAfterPositionsRoughlyEqual(document: TextDocument, firstPos: Position, secondPos: Position): boolean { + const charBeforeFirstPos: string = this.getCharBefore(document, firstPos); + const charAfterFirstPos: string = this.getCharAfter(document, firstPos); + const charBeforeSecondPos: string = this.getCharBefore(document, secondPos); + const charAfterSecondPos: string = this.getCharAfter(document, secondPos); + + // Exit mirror mode when cursor position no longer mirror + // Unless it's in the case of `<|>` + const charBeforeBothPositionRoughlyEqual: boolean = + charBeforeFirstPos === charBeforeSecondPos || + (charBeforeFirstPos === '/' && charBeforeSecondPos === '<') || + (charBeforeSecondPos === '/' && charBeforeFirstPos === '<'); + const charAfterBothPositionRoughlyEqual: boolean = + charAfterFirstPos === charAfterSecondPos || + (charAfterFirstPos === ' ' && charAfterSecondPos === '>') || + (charAfterSecondPos === ' ' && charAfterFirstPos === '>'); + + return charBeforeBothPositionRoughlyEqual && charAfterBothPositionRoughlyEqual; + } + + private shouldDoCleanupForXmlAttributeInput(document: TextDocument, firstPos: Position, secondPos: Position): boolean { + // Need to cleanup in the case of
+ const charBeforeFirstPos: string = this.getCharBefore(document, firstPos); + const charAfterFirstPos: string = this.getCharAfter(document, firstPos); + const charBeforeSecondPos: string = this.getCharBefore(document, secondPos); + const charAfterSecondPos: string = this.getCharAfter(document, secondPos); + const firstBeforeSecond: boolean = document.offsetAt(firstPos) < document.offsetAt(secondPos); + + /** + * Check two cases + *
+ *
+ * Before 1st cursor: ` ` + * After 1st cursor: `>` or ` ` + * Before 2nd cursor: ` ` + * After 2nd cursor: `>` + */ + return ( + firstBeforeSecond && + charBeforeFirstPos === ' ' && + (charAfterFirstPos === '>' || charAfterFirstPos === ' ') && + charBeforeSecondPos === ' ' && + charAfterSecondPos === '>' + ); + } + + private createCleanupForXmlAttributeInput(document: TextDocument, endTagCursorPos: Position): WorkspaceEdit { + const cleanupEdit: WorkspaceEdit = new WorkspaceEdit(); + const cleanupRange: Range = new Range(endTagCursorPos.translate(0, -1), endTagCursorPos); + cleanupEdit.replace(document.uri, cleanupRange, ''); + return cleanupEdit; + } + + private getCharBefore(document: TextDocument, position: Position): string { + const offset: number = document.offsetAt(position); + if (offset === 0) { + return ''; + } + + return document.getText(new Range(document.positionAt(offset - 1), position)); + } + + private getCharAfter(document: TextDocument, position: Position): string { + const offset: number = document.offsetAt(position); + if (offset === document.getText().length) { + return ''; + } + + return document.getText(new Range(position, document.positionAt(offset + 1))); + } + + public cleanup(forceOneCursor = false): void { + if (forceOneCursor && window.activeTextEditor && window.activeTextEditor.selections.length > 0) { + window.activeTextEditor.selections = [window.activeTextEditor.selections[0]]; + } + this.dispose(); + } + + private dispose(): void { + this.disposables.forEach((d: Disposable) => d.dispose()); + } +} \ No newline at end of file