UNPKG

@shopify/theme-language-server-common

Version:

<h1 align="center" style="position: relative;" > <br> <img src="https://github.com/Shopify/theme-check-vscode/blob/main/images/shopify_glyph.png?raw=true" alt="logo" width="141" height="160"> <br> Theme Language Server </h1>

118 lines (101 loc) 4.33 kB
import { CompletionItem, CompletionItemKind, InsertTextFormat, Range, TextEdit, } from 'vscode-languageserver'; import { Attribute, HtmlData, renderHtmlEntry } from '../../docset'; import { AugmentedSourceCode, DocumentManager } from '../../documents'; import { findLast, getCompoundName, isAttrEmpty, isNamedHtmlElementNode, isTextNode, } from '../../utils'; import { CURSOR, LiquidCompletionParams } from '../params'; import { Provider, sortByName } from './common'; import { LiquidHtmlNode } from '@shopify/liquid-html-parser'; export class HtmlAttributeCompletionProvider implements Provider { constructor(private readonly documentManager: DocumentManager) {} async completions(params: LiquidCompletionParams): Promise<CompletionItem[]> { if (!params.completionContext) return []; const { node, ancestors } = params.completionContext; const parentNode = findLast(ancestors, isAttrEmpty); const grandParentNode = findLast(ancestors, isNamedHtmlElementNode); const document = this.documentManager.get(params.textDocument.uri); if (!node || !parentNode || !grandParentNode || !document) { return []; } if (!isTextNode(node) || !isAttrEmpty(parentNode) || !isNamedHtmlElementNode(grandParentNode)) { return []; } const grandParentNodeName = getCompoundName(grandParentNode); const name = node.value; const partial = name.replace(CURSOR, ''); const options = getOptions(partial, grandParentNodeName); const attributeTagRange = this.attributeTagRange(node, document); const hasExistingAttributeValue = this.hasExistingAttributeValue(attributeTagRange, document); const hasLiquidTag = this.hasLiquidTag(attributeTagRange, document); return options.sort(sortByName).map((tag) => { return toCompletionItem(tag, attributeTagRange, hasExistingAttributeValue, hasLiquidTag); }); } hasExistingAttributeValue(attributeTagRange: Range, document: AugmentedSourceCode): boolean { return /^\s*=/.test( document.source.slice(document.textDocument.offsetAt(attributeTagRange.end)), ); } hasLiquidTag(attributeTagRange: Range, document: AugmentedSourceCode): boolean { return /^(?:\{%|\{\{)/.test( document.source.slice(document.textDocument.offsetAt(attributeTagRange.end)), ); } // Find the range of the attribute partial. If the attribute contains any liquid code, the range // will end before the first character of the liquid block. attributeTagRange(node: LiquidHtmlNode, document: AugmentedSourceCode): Range { if (node.type === 'TextNode' && node.value === CURSOR) { // If you try to auto-complete with no provided attribute tag, // we will not try to override the subsequent character. // E.g. <a href="" █> return { start: document.textDocument.positionAt(node.position.start), end: document.textDocument.positionAt(node.position.start), }; } const sourcePartialPastCursor = document.source.slice(node.position.end); const attributeEndOffset = sourcePartialPastCursor.match(/[\s=]|\{%|\{\{|>/)?.index ?? sourcePartialPastCursor.length; return { start: document.textDocument.positionAt(node.position.start), end: document.textDocument.positionAt(node.position.end + attributeEndOffset), }; } } function getOptions(partial: string, parentNodeName: string): Attribute[] { const tag = HtmlData.tags.find((tag) => tag.name === parentNodeName); const parentAttributes = tag?.attributes ?? []; return [...parentAttributes, ...HtmlData.globalAttributes].filter((x) => x.name.startsWith(partial), ); } function toCompletionItem( tag: Attribute, attributeTagRange: Range, hasExistingAttributeValue: boolean, hasLiquidTag: boolean, ): CompletionItem { const attributeWithValue = !tag.valueSet || tag.valueSet !== 'v'; const insertSnippet = attributeWithValue && !hasExistingAttributeValue && !hasLiquidTag; return { label: tag.name, kind: CompletionItemKind.Value, insertTextFormat: insertSnippet ? InsertTextFormat.Snippet : InsertTextFormat.PlainText, textEdit: TextEdit.replace(attributeTagRange, insertSnippet ? `${tag.name}="$1"$0` : tag.name), documentation: { kind: 'markdown', value: renderHtmlEntry(tag), }, }; }