Skip to content

Commit

Permalink
fix: improve renderBody codegen
Browse files Browse the repository at this point in the history
  • Loading branch information
DylanPiercey committed Oct 8, 2024
1 parent 39ef7d1 commit abbb30a
Show file tree
Hide file tree
Showing 10 changed files with 63 additions and 67 deletions.
8 changes: 8 additions & 0 deletions .changeset/swift-pumas-dream.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
---
"@marko/language-server": patch
"@marko/language-tools": patch
"@marko/type-check": patch
"marko-vscode": patch
---

Align tag arguments type generation with Marko 6.
Original file line number Diff line number Diff line change
@@ -1,11 +1,20 @@
## Diagnostics
### Ln 3, Col 10
### Ln 2, Col 11
```marko
1 | <test-tag("hello!")/>
2 | <test-tag("hello!", 1)/>
1 | <test-tag({ value: "hello!" })/>
> 2 | <test-tag("hello!")/>
| ^^^^^^^^ Argument of type 'string' is not assignable to parameter of type 'Directives & Input'.
Type 'string' is not assignable to type 'Input'.
3 | <test-tag("hello!", 1, 2)/>
4 |
```

### Ln 3, Col 21
```marko
1 | <test-tag({ value: "hello!" })/>
2 | <test-tag("hello!")/>
> 3 | <test-tag("hello!", 1, 2)/>
| ^^^^^^^^^^^^^^^^ Type of computed property's value is '[string, number, number]', which is not assignable to type '[string, (number | undefined)?]'.
Source has 3 element(s) but target allows only 2.
| ^^^^ Expected 1 arguments, but got 3.
4 |
```

Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
export interface Input {
value: [string, number?]
value: string
}
<div/>
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
<test-tag({ value: "hello!" })/>
<test-tag("hello!")/>
<test-tag("hello!", 1)/>
<test-tag("hello!", 1, 2)/>
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,4 @@ export interface Input {
renderBody: Marko.Body<[string]>
}

<${input.renderBody} value=["hi"] />

<${input.renderBody}=["hi"] />

<${input.renderBody}("hi") />

Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,4 @@ export interface Input {
renderBody: Marko.Body<["a"], { value: "a" | "b" }>
}

<${input.renderBody}=["a"] />
<${input.renderBody}("a") />
Original file line number Diff line number Diff line change
@@ -1,22 +1,21 @@
## Diagnostics
### Ln 7, Col 28
### Ln 7, Col 27
```marko
5 | <${input.renderBody}=["a", "b"] />
5 | <${input.renderBody}("a", "b") />
6 |
> 7 | <${input.renderBody}=["a", "c"] />
| ^^^ Type '"c"' is not assignable to type '"b"'.
> 7 | <${input.renderBody}("a", "c") />
| ^^^ Argument of type '"c"' is not assignable to parameter of type '"b"'.
8 |
9 | <${input.renderBody}=["a", "b", "c"] />
9 | <${input.renderBody}("a", "b", "c") />
10 |
```

### Ln 9, Col 21
### Ln 9, Col 32
```marko
7 | <${input.renderBody}=["a", "c"] />
7 | <${input.renderBody}("a", "c") />
8 |
> 9 | <${input.renderBody}=["a", "b", "c"] />
| ^ Type '["a", "b", string]' is not assignable to type '["a", "b"]'.
Source has 3 element(s) but target allows only 2.
> 9 | <${input.renderBody}("a", "b", "c") />
| ^^^ Expected 2 arguments, but got 3.
10 |
```

Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,8 @@ export interface Input {
renderBody: Marko.Body<["a", "b"]>
}

<${input.renderBody}=["a", "b"] />
<${input.renderBody}("a", "b") />

<${input.renderBody}=["a", "c"] />
<${input.renderBody}("a", "c") />

<${input.renderBody}=["a", "b", "c"] />
<${input.renderBody}("a", "b", "c") />
35 changes: 12 additions & 23 deletions packages/language-tools/marko.internal.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -306,14 +306,10 @@ declare global {
? NativeTagRenderer<Name>
: [Name] extends [AnyMarkoBody]
? BodyRenderer<Name>
: [Name] extends [{ renderBody?: AnyMarkoBody }]
? [Name["renderBody"]] extends [AnyMarkoBody]
? BodyRenderer<Name["renderBody"]>
: BaseRenderer<
RenderBodyInput<
BodyParameters<Exclude<Name["renderBody"], void>>
>
>
: [Name] extends [
{ renderBody?: infer Name extends AnyMarkoBody },
]
? BodyRenderer<Name>
: DefaultRenderer;

export type TemplateRenderer<Template> = Template extends {
Expand Down Expand Up @@ -345,12 +341,8 @@ declare global {
export interface BodyRenderer<Body extends AnyMarkoBody> {
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-constraint
(): () => <__marko_internal_input extends unknown>(
input: Marko.Directives &
RenderBodyInput<BodyParameters<Body>> &
Relate<
__marko_internal_input,
Marko.Directives & RenderBodyInput<BodyParameters<Body>>
>,
...args: BodyParamsWithDefault<Body> &
Relate<__marko_internal_input, BodyParamsWithDefault<Body>>
) => ReturnAndScope<
Scopes<__marko_internal_input>,
BodyReturnType<Body>
Expand Down Expand Up @@ -397,15 +389,12 @@ type ReturnAndScope<Scope, Return> = {
scope: Scope;
};

type RenderBodyInput<Args extends readonly unknown[]> = Args extends {
length: infer Length;
}
? number extends Length
? { value?: Args }
: 0 extends Length
? { value?: [] }
: { value: Args }
: never;
type BodyParamsWithDefault<Body extends AnyMarkoBody> =
Body extends Marko.Body<infer Params, any>
? Params extends []
? [input?: Record<string, never>]
: Params
: never;

type Scopes<Input> = [0] extends [1 & Input]
? never
Expand Down
32 changes: 14 additions & 18 deletions packages/language-tools/src/extractors/script/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1188,33 +1188,27 @@ constructor(_?: Return) {}
#writeTagInputObject(tag: Node.ParentTag) {
if (!tag.params) this.#writeComments(tag);

const body = this.#processBody(tag);
let writeInputObj = true;
let hasInput = false;
this.#extractor.write("{\n");

if (tag.args) {
hasInput = true;
this.#extractor
.write("[")
.copy({
start: tag.args.start,
end: tag.args.start + 1,
})
.write('"value"')
.copy({
start: tag.args.end - 1,
end: tag.args.end,
})
.write(`]: ${varShared("tuple")}(`)
.copy(tag.args.value)
.write(")")
.write(",\n");
this.#extractor.copy(tag.args.value);

if (body || tag.attrs || tag.shorthandId || tag.shorthandClassNames) {
this.#extractor.write(",\n{\n");
} else {
writeInputObj = false;
}
} else {
this.#extractor.write("{\n");
}

if (this.#writeAttrs(tag)) {
hasInput = true;
}

const body = this.#processBody(tag);
let hasRenderBody = false;
if (body) {
hasInput = true;
Expand Down Expand Up @@ -1275,7 +1269,9 @@ constructor(_?: Return) {}
this.#writeTagNameComment(tag);
}

this.#extractor.write("\n}");
if (writeInputObj) {
this.#extractor.write("\n}");
}
}

#writeObjectKeys(keys: Iterable<string>) {
Expand Down

0 comments on commit abbb30a

Please sign in to comment.