- Delete old Vite+Svelte frontend - Initialize new SvelteKit project with TypeScript - Configure Tailwind CSS v4 + DaisyUI - Implement JWT authentication with auto-refresh - Create login page with form validation (Zod) - Add protected route guards - Update Docker configuration for single-stage build - Add E2E tests with Playwright (6/11 passing) - Fix Svelte 5 reactivity with $state() runes Known issues: - 5 E2E tests failing (timing/async issues) - Token refresh implementation needs debugging - Validation error display timing
163 lines
6.5 KiB
JavaScript
163 lines
6.5 KiB
JavaScript
import { createRule } from '../utils/index.js';
|
|
import { parseStyleAttributeValue } from '../utils/css-utils/index.js';
|
|
import { isHTMLElementLike } from '../utils/ast-utils.js';
|
|
/** Checks wether the given node is string literal or not */
|
|
function isStringLiteral(node) {
|
|
return node.type === 'Literal' && typeof node.value === 'string';
|
|
}
|
|
export default createRule('prefer-style-directive', {
|
|
meta: {
|
|
docs: {
|
|
description: 'require style directives instead of style attribute',
|
|
category: 'Stylistic Issues',
|
|
recommended: false,
|
|
conflictWithPrettier: false
|
|
},
|
|
fixable: 'code',
|
|
schema: [],
|
|
messages: {
|
|
unexpected: 'Can use style directives instead.'
|
|
},
|
|
type: 'suggestion'
|
|
},
|
|
create(context) {
|
|
const sourceCode = context.sourceCode;
|
|
/**
|
|
* Process for `style=" ... "`
|
|
*/
|
|
function processStyleValue(node, root) {
|
|
for (const child of root.nodes) {
|
|
if (child.type === 'decl') {
|
|
processDeclaration(node, root, child);
|
|
}
|
|
else if (child.type === 'inline') {
|
|
processInline(node, root, child);
|
|
}
|
|
}
|
|
}
|
|
/**
|
|
* Process for declaration
|
|
*/
|
|
function processDeclaration(attrNode, root, decl) {
|
|
if (decl.important || decl.unknownInterpolations.length || decl.prop.interpolations.length)
|
|
return;
|
|
if (attrNode.parent.attributes.some((attr) => attr.type === 'SvelteStyleDirective' && attr.key.name.name === decl.prop.name)) {
|
|
// has style directive
|
|
return;
|
|
}
|
|
context.report({
|
|
node: attrNode,
|
|
loc: decl.loc,
|
|
messageId: 'unexpected',
|
|
*fix(fixer) {
|
|
const styleDirective = `style:${decl.prop.name}="${sourceCode.text.slice(...decl.value.range)}"`;
|
|
if (root.nodes.length === 1 && root.nodes[0] === decl) {
|
|
yield fixer.replaceTextRange(attrNode.range, styleDirective);
|
|
}
|
|
else {
|
|
yield removeStyle(fixer, root, decl);
|
|
if (root.nodes[0] === decl) {
|
|
yield fixer.insertTextBeforeRange(attrNode.range, `${styleDirective} `);
|
|
}
|
|
else {
|
|
yield fixer.insertTextAfterRange(attrNode.range, ` ${styleDirective}`);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
/**
|
|
* Process for inline
|
|
*/
|
|
function processInline(attrNode, root, inline) {
|
|
const node = inline.node.expression;
|
|
if (node.type !== 'ConditionalExpression') {
|
|
return;
|
|
}
|
|
if (!isStringLiteral(node.consequent) || !isStringLiteral(node.alternate)) {
|
|
return;
|
|
}
|
|
if (node.consequent.value && node.alternate.value) {
|
|
// e.g. t ? 'top: 20px' : 'left: 30px'
|
|
return;
|
|
}
|
|
const positive = !node.alternate.value;
|
|
const inlineRoot = inline.getInlineStyle(positive ? node.consequent : node.alternate);
|
|
if (!inlineRoot || inlineRoot.nodes.length !== 1) {
|
|
return;
|
|
}
|
|
const decl = inlineRoot.nodes[0];
|
|
if (decl.type !== 'decl') {
|
|
return;
|
|
}
|
|
if (attrNode.parent.attributes.some((attr) => attr.type === 'SvelteStyleDirective' && attr.key.name.name === decl.prop.name)) {
|
|
// has style directive
|
|
return;
|
|
}
|
|
context.report({
|
|
node,
|
|
messageId: 'unexpected',
|
|
*fix(fixer) {
|
|
let valueText = sourceCode.text.slice(node.test.range[0], node.consequent.range[0]);
|
|
if (positive) {
|
|
valueText +=
|
|
sourceCode.text[node.consequent.range[0]] +
|
|
decl.value.value +
|
|
sourceCode.text[node.consequent.range[1] - 1];
|
|
}
|
|
else {
|
|
valueText += 'null';
|
|
}
|
|
valueText += sourceCode.text.slice(node.consequent.range[1], node.alternate.range[0]);
|
|
if (positive) {
|
|
valueText += 'null';
|
|
}
|
|
else {
|
|
valueText +=
|
|
sourceCode.text[node.alternate.range[0]] +
|
|
decl.value.value +
|
|
sourceCode.text[node.alternate.range[1] - 1];
|
|
}
|
|
const styleDirective = `style:${decl.prop.name}={${valueText}}`;
|
|
if (root.nodes.length === 1 && root.nodes[0] === inline) {
|
|
yield fixer.replaceTextRange(attrNode.range, styleDirective);
|
|
}
|
|
else {
|
|
yield removeStyle(fixer, root, inline);
|
|
if (root.nodes[0] === inline) {
|
|
yield fixer.insertTextBeforeRange(attrNode.range, `${styleDirective} `);
|
|
}
|
|
else {
|
|
yield fixer.insertTextAfterRange(attrNode.range, ` ${styleDirective}`);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
/** Remove style */
|
|
function removeStyle(fixer, root, node) {
|
|
const index = root.nodes.indexOf(node);
|
|
const after = root.nodes[index + 1];
|
|
if (after) {
|
|
return fixer.removeRange([node.range[0], after.range[0]]);
|
|
}
|
|
const before = root.nodes[index - 1];
|
|
if (before) {
|
|
return fixer.removeRange([before.range[1], node.range[1]]);
|
|
}
|
|
return fixer.removeRange(node.range);
|
|
}
|
|
return {
|
|
'SvelteStartTag > SvelteAttribute'(node) {
|
|
if (!isHTMLElementLike(node.parent.parent) || node.key.name !== 'style') {
|
|
return;
|
|
}
|
|
const root = parseStyleAttributeValue(node, context);
|
|
if (root) {
|
|
processStyleValue(node, root);
|
|
}
|
|
}
|
|
};
|
|
}
|
|
});
|