diff --git a/packages/language-core/lib/codegen/template/element.ts b/packages/language-core/lib/codegen/template/element.ts
index cacbbd9572..da1ca30b8c 100644
--- a/packages/language-core/lib/codegen/template/element.ts
+++ b/packages/language-core/lib/codegen/template/element.ts
@@ -1,11 +1,9 @@
import * as CompilerDOM from '@vue/compiler-dom';
import { camelize, capitalize } from '@vue/shared';
-import type * as ts from 'typescript';
-import { getNodeText } from '../../parsers/scriptSetupRanges';
import type { Code, VueCodeInformation } from '../../types';
import { hyphenateTag } from '../../utils/shared';
import { createVBindShorthandInlayHintInfo } from '../inlayHints';
-import { collectVars, createTsAst, endOfLine, newLine, variableNameRegex, wrapWith } from '../utils';
+import { collectVars, createTsAst, endOfLine, newLine, normalizeAttributeValue, variableNameRegex, wrapWith } from '../utils';
import { generateCamelized } from '../utils/camelized';
import type { TemplateCodegenContext } from './context';
import { generateElementChildren } from './elementChildren';
@@ -16,6 +14,7 @@ import type { TemplateCodegenOptions } from './index';
import { generateInterpolation } from './interpolation';
import { generateObjectProperty } from './objectProperty';
import { generatePropertyAccess } from './propertyAccess';
+import { collectStyleScopedClassReferences } from './styleScopedClasses';
import { generateTemplateChild } from './templateChild';
const colonReg = /:/g;
@@ -420,7 +419,7 @@ function* generateVScope(
yield* generateElementDirectives(options, ctx, node);
const [refName, offset] = yield* generateReferencesForElements(options, ctx, node); //
- yield* generateReferencesForScopedCssClasses(options, ctx, node);
+ collectStyleScopedClassReferences(options, ctx, node);
if (inScope) {
yield `}${newLine}`;
@@ -618,142 +617,6 @@ function* generateReferencesForElements(
return [];
}
-function* generateReferencesForScopedCssClasses(
- options: TemplateCodegenOptions,
- ctx: TemplateCodegenContext,
- node: CompilerDOM.ElementNode
-): Generator {
- for (const prop of node.props) {
- if (
- prop.type === CompilerDOM.NodeTypes.ATTRIBUTE
- && prop.name === 'class'
- && prop.value
- ) {
- if (options.template.lang === 'pug') {
- const getClassOffset = Reflect.get(prop.value.loc.start, 'getClassOffset') as (offset: number) => number;
- const content = prop.value.loc.source.slice(1, -1);
-
- let startOffset = 1;
- for (const className of content.split(' ')) {
- if (className) {
- ctx.scopedClasses.push({
- source: 'template',
- className,
- offset: getClassOffset(startOffset),
- });
- }
- startOffset += className.length + 1;
- }
- }
- else {
- let isWrapped = false;
- const [content, startOffset] = normalizeAttributeValue(prop.value);
- if (content) {
- const classes = collectClasses(content, startOffset + (isWrapped ? 1 : 0));
- ctx.scopedClasses.push(...classes);
- }
- else {
- ctx.emptyClassOffsets.push(startOffset);
- }
- }
- }
- else if (
- prop.type === CompilerDOM.NodeTypes.DIRECTIVE
- && prop.arg?.type === CompilerDOM.NodeTypes.SIMPLE_EXPRESSION
- && prop.exp?.type === CompilerDOM.NodeTypes.SIMPLE_EXPRESSION
- && prop.arg.content === 'class'
- ) {
- const content = '`${' + prop.exp.content + '}`';
- const startOffset = prop.exp.loc.start.offset - 3;
-
- const { ts } = options;
- const ast = ts.createSourceFile('', content, 99 satisfies ts.ScriptTarget.Latest);
- const literals: ts.StringLiteralLike[] = [];
-
- ts.forEachChild(ast, node => {
- if (
- !ts.isExpressionStatement(node) ||
- !isTemplateExpression(node.expression)
- ) {
- return;
- }
-
- const expression = node.expression.templateSpans[0].expression;
-
- if (ts.isStringLiteralLike(expression)) {
- literals.push(expression);
- }
-
- if (ts.isArrayLiteralExpression(expression)) {
- walkArrayLiteral(expression);
- }
-
- if (ts.isObjectLiteralExpression(expression)) {
- walkObjectLiteral(expression);
- }
- });
-
- for (const literal of literals) {
- if (literal.text) {
- const classes = collectClasses(
- literal.text,
- literal.end - literal.text.length - 1 + startOffset
- );
- ctx.scopedClasses.push(...classes);
- }
- else {
- ctx.emptyClassOffsets.push(literal.end - 1 + startOffset);
- }
- }
-
- function walkArrayLiteral(node: ts.ArrayLiteralExpression) {
- const { elements } = node;
- for (const element of elements) {
- if (ts.isStringLiteralLike(element)) {
- literals.push(element);
- }
- else if (ts.isObjectLiteralExpression(element)) {
- walkObjectLiteral(element);
- }
- }
- }
-
- function walkObjectLiteral(node: ts.ObjectLiteralExpression) {
- const { properties } = node;
- for (const property of properties) {
- if (ts.isPropertyAssignment(property)) {
- const { name } = property;
- if (ts.isIdentifier(name)) {
- walkIdentifier(name);
- }
- else if (ts.isStringLiteral(name)) {
- literals.push(name);
- }
- else if (ts.isComputedPropertyName(name)) {
- const { expression } = name;
- if (ts.isStringLiteralLike(expression)) {
- literals.push(expression);
- }
- }
- }
- else if (ts.isShorthandPropertyAssignment(property)) {
- walkIdentifier(property.name);
- }
- }
- }
-
- function walkIdentifier(node: ts.Identifier) {
- const text = getNodeText(ts, node, ast);
- ctx.scopedClasses.push({
- source: 'template',
- className: text,
- offset: node.end - text.length + startOffset
- });
- }
- }
- }
-}
-
function camelizeComponentName(newName: string) {
return camelize('-' + newName);
}
@@ -761,50 +624,3 @@ function camelizeComponentName(newName: string) {
function getTagRenameApply(oldName: string) {
return oldName === hyphenateTag(oldName) ? hyphenateTag : undefined;
}
-
-function normalizeAttributeValue(node: CompilerDOM.TextNode): [string, number] {
- let offset = node.loc.start.offset;
- let content = node.loc.source;
- if (
- (content.startsWith(`'`) && content.endsWith(`'`))
- || (content.startsWith(`"`) && content.endsWith(`"`))
- ) {
- offset++;
- content = content.slice(1, -1);
- }
- return [content, offset];
-}
-
-function collectClasses(content: string, startOffset = 0) {
- const classes: {
- source: string;
- className: string;
- offset: number;
- }[] = [];
-
- let currentClassName = '';
- let offset = 0;
- for (const char of (content + ' ')) {
- if (char.trim() === '') {
- if (currentClassName !== '') {
- classes.push({
- source: 'template',
- className: currentClassName,
- offset: offset + startOffset
- });
- offset += currentClassName.length;
- currentClassName = '';
- }
- offset += char.length;
- }
- else {
- currentClassName += char;
- }
- }
- return classes;
-}
-
-// isTemplateExpression is missing in tsc
-function isTemplateExpression(node: ts.Node): node is ts.TemplateExpression {
- return node.kind === 228 satisfies ts.SyntaxKind.TemplateExpression;
-}
\ No newline at end of file
diff --git a/packages/language-core/lib/codegen/template/styleScopedClasses.ts b/packages/language-core/lib/codegen/template/styleScopedClasses.ts
index d7f8cb5bd8..43f2a4ce9b 100644
--- a/packages/language-core/lib/codegen/template/styleScopedClasses.ts
+++ b/packages/language-core/lib/codegen/template/styleScopedClasses.ts
@@ -1,6 +1,10 @@
+import * as CompilerDOM from '@vue/compiler-dom';
+import type * as ts from 'typescript';
+import { getNodeText } from '../../parsers/scriptSetupRanges';
import type { Code } from '../../types';
-import { endOfLine } from '../utils';
+import { endOfLine, normalizeAttributeValue } from '../utils';
import type { TemplateCodegenContext } from './context';
+import type { TemplateCodegenOptions } from './index';
export function* generateStyleScopedClassReferences(
ctx: TemplateCodegenContext,
@@ -81,4 +85,174 @@ export function* generateStyleScopedClassReferences(
}
}
}
-}
\ No newline at end of file
+}
+
+export function collectStyleScopedClassReferences(
+ options: TemplateCodegenOptions,
+ ctx: TemplateCodegenContext,
+ node: CompilerDOM.ElementNode
+) {
+ for (const prop of node.props) {
+ if (
+ prop.type === CompilerDOM.NodeTypes.ATTRIBUTE
+ && prop.name === 'class'
+ && prop.value
+ ) {
+ if (options.template.lang === 'pug') {
+ const getClassOffset = Reflect.get(prop.value.loc.start, 'getClassOffset') as (offset: number) => number;
+ const content = prop.value.loc.source.slice(1, -1);
+
+ let startOffset = 1;
+ for (const className of content.split(' ')) {
+ if (className) {
+ ctx.scopedClasses.push({
+ source: 'template',
+ className,
+ offset: getClassOffset(startOffset),
+ });
+ }
+ startOffset += className.length + 1;
+ }
+ }
+ else {
+ let isWrapped = false;
+ const [content, startOffset] = normalizeAttributeValue(prop.value);
+ if (content) {
+ const classes = collectClasses(content, startOffset + (isWrapped ? 1 : 0));
+ ctx.scopedClasses.push(...classes);
+ }
+ else {
+ ctx.emptyClassOffsets.push(startOffset);
+ }
+ }
+ }
+ else if (
+ prop.type === CompilerDOM.NodeTypes.DIRECTIVE
+ && prop.arg?.type === CompilerDOM.NodeTypes.SIMPLE_EXPRESSION
+ && prop.exp?.type === CompilerDOM.NodeTypes.SIMPLE_EXPRESSION
+ && prop.arg.content === 'class'
+ ) {
+ const content = '`${' + prop.exp.content + '}`';
+ const startOffset = prop.exp.loc.start.offset - 3;
+
+ const { ts } = options;
+ const ast = ts.createSourceFile('', content, 99 satisfies ts.ScriptTarget.Latest);
+ const literals: ts.StringLiteralLike[] = [];
+
+ ts.forEachChild(ast, node => {
+ if (
+ !ts.isExpressionStatement(node) ||
+ !isTemplateExpression(node.expression)
+ ) {
+ return;
+ }
+
+ const expression = node.expression.templateSpans[0].expression;
+
+ if (ts.isStringLiteralLike(expression)) {
+ literals.push(expression);
+ }
+
+ if (ts.isArrayLiteralExpression(expression)) {
+ walkArrayLiteral(expression);
+ }
+
+ if (ts.isObjectLiteralExpression(expression)) {
+ walkObjectLiteral(expression);
+ }
+ });
+
+ for (const literal of literals) {
+ if (literal.text) {
+ const classes = collectClasses(
+ literal.text,
+ literal.end - literal.text.length - 1 + startOffset
+ );
+ ctx.scopedClasses.push(...classes);
+ }
+ else {
+ ctx.emptyClassOffsets.push(literal.end - 1 + startOffset);
+ }
+ }
+
+ function walkArrayLiteral(node: ts.ArrayLiteralExpression) {
+ const { elements } = node;
+ for (const element of elements) {
+ if (ts.isStringLiteralLike(element)) {
+ literals.push(element);
+ }
+ else if (ts.isObjectLiteralExpression(element)) {
+ walkObjectLiteral(element);
+ }
+ }
+ }
+
+ function walkObjectLiteral(node: ts.ObjectLiteralExpression) {
+ const { properties } = node;
+ for (const property of properties) {
+ if (ts.isPropertyAssignment(property)) {
+ const { name } = property;
+ if (ts.isIdentifier(name)) {
+ walkIdentifier(name);
+ }
+ else if (ts.isStringLiteral(name)) {
+ literals.push(name);
+ }
+ else if (ts.isComputedPropertyName(name)) {
+ const { expression } = name;
+ if (ts.isStringLiteralLike(expression)) {
+ literals.push(expression);
+ }
+ }
+ }
+ else if (ts.isShorthandPropertyAssignment(property)) {
+ walkIdentifier(property.name);
+ }
+ }
+ }
+
+ function walkIdentifier(node: ts.Identifier) {
+ const text = getNodeText(ts, node, ast);
+ ctx.scopedClasses.push({
+ source: 'template',
+ className: text,
+ offset: node.end - text.length + startOffset
+ });
+ }
+ }
+ }
+}
+
+function collectClasses(content: string, startOffset = 0) {
+ const classes: {
+ source: string;
+ className: string;
+ offset: number;
+ }[] = [];
+
+ let currentClassName = '';
+ let offset = 0;
+ for (const char of (content + ' ')) {
+ if (char.trim() === '') {
+ if (currentClassName !== '') {
+ classes.push({
+ source: 'template',
+ className: currentClassName,
+ offset: offset + startOffset
+ });
+ offset += currentClassName.length;
+ currentClassName = '';
+ }
+ offset += char.length;
+ }
+ else {
+ currentClassName += char;
+ }
+ }
+ return classes;
+}
+
+// isTemplateExpression is missing in tsc
+function isTemplateExpression(node: ts.Node): node is ts.TemplateExpression {
+ return node.kind === 228 satisfies ts.SyntaxKind.TemplateExpression;
+}
diff --git a/packages/language-core/lib/codegen/utils/index.ts b/packages/language-core/lib/codegen/utils/index.ts
index e2f28894da..a39915e108 100644
--- a/packages/language-core/lib/codegen/utils/index.ts
+++ b/packages/language-core/lib/codegen/utils/index.ts
@@ -1,3 +1,4 @@
+import * as CompilerDOM from '@vue/compiler-dom';
import type * as ts from 'typescript';
import { getNodeText } from '../../parsers/scriptSetupRanges';
import type { Code, SfcBlock, VueCodeInformation } from '../../types';
@@ -64,6 +65,19 @@ export function collectIdentifiers(
return results;
}
+export function normalizeAttributeValue(node: CompilerDOM.TextNode): [string, number] {
+ let offset = node.loc.start.offset;
+ let content = node.loc.source;
+ if (
+ (content.startsWith(`'`) && content.endsWith(`'`))
+ || (content.startsWith(`"`) && content.endsWith(`"`))
+ ) {
+ offset++;
+ content = content.slice(1, -1);
+ }
+ return [content, offset];
+}
+
export function createTsAst(ts: typeof import('typescript'), astHolder: any, text: string) {
if (astHolder.__volar_ast_text !== text) {
astHolder.__volar_ast_text = text;