Skip to content

Commit c664d52

Browse files
authoredOct 1, 2024··
Fix HTML element close tag auto-insertion inside Liquid branches (#516)
There was a bug with our findCurrentNode algorithm that made it so we'd go in the wrong node when a branch had child nodes. The fix was to have more information on the AST and flag LiquidBranch as "unclosed" when their blockEndPosition is -1, -1. It's kind of a nasty bug tbh. Fixes #515

File tree

9 files changed

+50
-16
lines changed

9 files changed

+50
-16
lines changed
 

‎.changeset/green-otters-smoke.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@shopify/theme-language-server-common': patch
3+
---
4+
5+
Fix HTML element close tag auto-insertion inside Liquid branches

‎.changeset/moody-books-remember.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@shopify/liquid-html-parser': patch
3+
---
4+
5+
LiquidBranch nodes should always report a blockEndPosition of 0 length

‎packages/liquid-html-parser/src/stage-2-ast.spec.ts

+18
Original file line numberDiff line numberDiff line change
@@ -681,6 +681,12 @@ describe('Unit: Stage 2 (AST)', () => {
681681
expectPath(ast, 'children.0.children.0.blockStartPosition.end').to.equal(
682682
source.indexOf(branchA),
683683
);
684+
expectPath(ast, 'children.0.children.0.blockEndPosition.start').to.equal(
685+
source.indexOf('{% elsif b %}'),
686+
);
687+
expectPath(ast, 'children.0.children.0.blockEndPosition.end').to.equal(
688+
source.indexOf('{% elsif b %}'),
689+
);
684690

685691
expectPath(ast, 'children.0.children.1.type').to.equal('LiquidBranch');
686692
expectPath(ast, 'children.0.children.1.position.start').to.equal(
@@ -695,6 +701,12 @@ describe('Unit: Stage 2 (AST)', () => {
695701
expectPath(ast, 'children.0.children.1.blockStartPosition.end').to.equal(
696702
source.indexOf('{% elsif b %}') + '{% elsif b %}'.length,
697703
);
704+
expectPath(ast, 'children.0.children.1.blockEndPosition.start').to.equal(
705+
source.indexOf('{% else %}'),
706+
);
707+
expectPath(ast, 'children.0.children.1.blockEndPosition.end').to.equal(
708+
source.indexOf('{% else %}'),
709+
);
698710

699711
expectPath(ast, 'children.0.children.2.type').to.equal('LiquidBranch');
700712
expectPath(ast, 'children.0.children.2.position.start').to.equal(
@@ -709,6 +721,12 @@ describe('Unit: Stage 2 (AST)', () => {
709721
expectPath(ast, 'children.0.children.2.blockStartPosition.end').to.equal(
710722
source.indexOf('{% else %}') + '{% else %}'.length,
711723
);
724+
expectPath(ast, 'children.0.children.2.blockEndPosition.start').to.equal(
725+
source.indexOf('{% endif %}', source.indexOf(branchC) + branchC.length),
726+
);
727+
expectPath(ast, 'children.0.children.2.blockEndPosition.end').to.equal(
728+
source.indexOf('{% endif %}', source.indexOf(branchC) + branchC.length),
729+
);
712730
}
713731
});
714732

‎packages/liquid-html-parser/src/stage-2-ast.ts

+10-9
Original file line numberDiff line numberDiff line change
@@ -441,6 +441,8 @@ interface LiquidBranchNode<Name, Markup> extends ASTNode<NodeTypes.LiquidBranch>
441441
whitespaceEnd: '-' | '';
442442
/** Range of the LiquidTag that delimits the branch (does not include children) */
443443
blockStartPosition: Position;
444+
/** 0-size range that delimits where the branch ends, (-1, -1) when unclosed */
445+
blockEndPosition: Position;
444446
}
445447

446448
/**
@@ -577,7 +579,7 @@ export interface HtmlElement extends HtmlNodeBase<NodeTypes.HtmlElement> {
577579
* The node used to represent close tags without its matching opening tag.
578580
*
579581
* Typically found inside if statements.
580-
*
582+
581583
* ```
582584
* {% if cond %}
583585
* </wrapper>
@@ -890,6 +892,7 @@ class ASTBuilder {
890892
if (node.type === NodeTypes.LiquidBranch) {
891893
const previousBranch = this.findCloseableParentBranch(node);
892894
if (previousBranch) {
895+
previousBranch.blockEndPosition = { start: node.position.start, end: node.position.start };
893896
// close dangling open HTML nodes
894897
while (
895898
this.parent &&
@@ -911,6 +914,7 @@ class ASTBuilder {
911914

912915
close(node: ConcreteCloseNode, nodeType: NodeTypes.LiquidTag | NodeTypes.HtmlElement) {
913916
if (isLiquidBranch(this.parent)) {
917+
this.parent.blockEndPosition = { start: node.locStart, end: node.locStart };
914918
this.closeParentWith(node);
915919
}
916920

@@ -1328,6 +1332,7 @@ function liquidBranchBaseAttributes(
13281332
whitespaceStart: node.whitespaceStart ?? '',
13291333
whitespaceEnd: node.whitespaceEnd ?? '',
13301334
blockStartPosition: position(node),
1335+
blockEndPosition: { start: -1, end: -1 },
13311336
source: node.source,
13321337
};
13331338
}
@@ -1514,6 +1519,7 @@ function toNamedLiquidBranchBaseCase(node: ConcreteLiquidTagBaseCase): LiquidBra
15141519
position: { start: node.locStart, end: node.locEnd },
15151520
children: [],
15161521
blockStartPosition: { start: node.locStart, end: node.locEnd },
1522+
blockEndPosition: { start: -1, end: -1 },
15171523
whitespaceStart: node.whitespaceStart ?? '',
15181524
whitespaceEnd: node.whitespaceEnd ?? '',
15191525
source: node.source,
@@ -1525,14 +1531,9 @@ function toUnnamedLiquidBranch(parentNode: LiquidHtmlNode): LiquidBranchUnnamed
15251531
type: NodeTypes.LiquidBranch,
15261532
name: null,
15271533
markup: '',
1528-
position: {
1529-
start: parentNode.position.end,
1530-
end: parentNode.position.end, // tmp value
1531-
},
1532-
blockStartPosition: {
1533-
start: parentNode.position.end,
1534-
end: parentNode.position.end,
1535-
},
1534+
position: { start: parentNode.position.end, end: parentNode.position.end },
1535+
blockStartPosition: { start: parentNode.position.end, end: parentNode.position.end },
1536+
blockEndPosition: { start: -1, end: -1 },
15361537
children: [],
15371538
whitespaceStart: '',
15381539
whitespaceEnd: '',

‎packages/theme-language-server-common/src/formatting/providers/HtmlElementAutoclosingOnTypeFormattingProvider.spec.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -129,7 +129,11 @@ describe('Module: HtmlElementAutoclosingOnTypeFormattingProvider', () => {
129129
const scenarios = [
130130
'{% if cond %}<div>█{% endif %}',
131131
'{% if cond %}{% else %}<div>█{% endif %}',
132+
'{% if cond %}<div></div>{% else %}<div>█{% endif %}',
133+
'{% if cond %}<div></div>{% elsif cond %}<div>█{% else %}<div>{% endif %}',
134+
'{% if cond %}<div></div>{% elsif cond %}<div>█ {% else %}<div>{% endif %}',
132135
'{% unless cond %}<div>█{% endunless %}',
136+
'{% unless cond %}<div><div>█{% endunless %}',
133137
'{% case thing %}{% when thing %}<div>█{% endif %}',
134138
];
135139
for (const source of scenarios) {
@@ -144,7 +148,7 @@ describe('Module: HtmlElementAutoclosingOnTypeFormattingProvider', () => {
144148
};
145149
assert(document);
146150
const result = await onTypeFormattingProvider.onTypeFormatting(params);
147-
assert(result);
151+
assert(result, 'expected results for source:\n' + source);
148152
expect(TextDocument.applyEdits(document, result)).to.equal(source.replace(CURSOR, '</div>'));
149153

150154
vi.advanceTimersByTime(10);

‎packages/theme-language-server-common/src/hover/providers/LiquidTagHoverProvider.spec.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -18,16 +18,17 @@ describe('Module: LiquidTagHoverProvider', async () => {
1818
});
1919

2020
it('should return the hover description of the correct tag', async () => {
21+
// cursor always points at character before █
2122
await expect(provider).to.hover(`{%█ if cond %}{% endif %}`, expect.stringContaining('if'));
2223
await expect(provider).to.hover(`{% i█f cond %}{% endif %}`, expect.stringContaining('if'));
2324
await expect(provider).to.hover(`{% if█ cond %}{% endif %}`, expect.stringContaining('if'));
25+
await expect(provider).to.hover(`{% if █cond %}{% endif %}`, expect.stringContaining('if'));
2426
await expect(provider).to.hover(`{% if cond █%}{% endif %}`, expect.stringContaining('if'));
2527
await expect(provider).to.hover(`{% if cond %}{% █ endif %}`, expect.stringContaining('if'));
2628
await expect(provider).to.hover(`{% echo█ 'hi' %}`, expect.stringContaining('echo'));
2729
});
2830

2931
it('should not return the tag hover description when hovering over anything else in the tag', async () => {
30-
await expect(provider).to.not.hover(`{% if █cond %}{% endif %}`, expect.stringContaining('if'));
3132
await expect(provider).to.not.hover(`{% if c█ond %}{% endif %}`, expect.stringContaining('if'));
3233
await expect(provider).to.not.hover(
3334
`{% if cond %} █ {% endif %}`,

‎packages/theme-language-server-common/src/linkedEditingRanges/providers/HtmlTagNameLinkedRangesProvider.spec.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ describe('Module: HtmlTagNameLinkedRangesProvider', () => {
2929
it('should return linked editing ranges for HTML tag names', async () => {
3030
params = {
3131
textDocument: { uri },
32-
position: document.positionAt(1), // position within the opening div#main tag name
32+
position: document.positionAt(2), // position within the opening div#main tag name
3333
};
3434

3535
const result = await provider.linkedEditingRanges(params);

‎packages/theme-language-server-common/src/rename/providers/HtmlTagNameRenameProvider.spec.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ describe('HtmlTagNameRenameProvider', () => {
2727
it('returns the correct workspace edit when renaming an HTML tag name', async () => {
2828
const params = {
2929
textDocument: { uri: 'file:///path/to/document.liquid' },
30-
position: Position.create(0, 1),
30+
position: Position.create(0, 2),
3131
newName: 'new-name',
3232
};
3333
documentManager.open(params.textDocument.uri, '<old><old></old></old>', 1);
@@ -53,7 +53,7 @@ describe('HtmlTagNameRenameProvider', () => {
5353
it('also works on complext liquid + text html tag names', async () => {
5454
const params = {
5555
textDocument: { uri: 'file:///path/to/document.liquid' },
56-
position: Position.create(0, 1),
56+
position: Position.create(0, 2),
5757
newName: 'new-name',
5858
};
5959
documentManager.open(params.textDocument.uri, '<web--{{ comp }}>text</web--{{ comp }}>', 1);

‎packages/theme-language-server-common/src/visitor.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,8 @@ import {
22
AST,
33
LiquidHtmlNode,
44
NodeOfType,
5-
NodeTypes,
65
SourceCodeType,
6+
NodeTypes,
77
} from '@shopify/theme-check-common';
88

99
export type VisitorMethod<S extends SourceCodeType, T, R> = (
@@ -108,7 +108,7 @@ export function findCurrentNode(
108108
}
109109

110110
function isCovered(node: LiquidHtmlNode, offset: number): boolean {
111-
return node.position.start <= offset && offset <= node.position.end;
111+
return node.position.start < offset && offset <= node.position.end;
112112
}
113113

114114
function size(node: LiquidHtmlNode): number {

0 commit comments

Comments
 (0)
Please sign in to comment.