mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-06 06:12:04 +08:00

* Support variadic arguments and slice, pointer types * Fix computation of type namespaces * Improve comments and general formatting * Set default values correctly for composite types * Add templates for bindings Additionally: * fixes generation of tuple return type * improves imports and namespacing in JS mode * general cleanup of generated code * Simplify import list construction * Refactor type generation code Improves support for unknown types (encoded as any) and maps (using Typescript index signatures) * Support slices with pointer elements * Match encoding/json behaviour in struct parser * Update tests and example * Add tests for complex method signatures and json tag parsing * Add test `function_multiple_files` * Attempt looking up idents with missing denotation * Update test data * fix quoted bool field * Test quoted booleans * Delete old parser code * Remove old test data * Update bindgen flags * Makes call by ID the default * Add package loading code * Add static analyser * Temporarily ignore binding generation code * Add complex slice expressions test * Fix variable reference analysis * Unwrap casts to interface types * Complete code comments * Refactor static analyser * Restrict options struct usage * Update tests * Fix method selector sink and source processing * Improve Set API * Add package info collector * Rename analyser package to analyse * Improve template functions * Add index file templates * Add glue code for binding generation * Refactor collection and rendering code * Implement binding generator * Implement global index generation * Improve marshaler and alias handling * Use package path in binding calls by name * Implement model collection and rendering * Fix wrong exit condition in analyser * Fix enum rendering * Generate shortcuts for all packages. * Implement generator tests * Ignore non-pointer bound types * Treat main package specially * Compute stats * Plug new API into generate command * Support all named types * Update JS runtime * Report dual role types * Remove go1.22 syntax * Fix type assertion in TS bindings * encoding/json compliance for arrays and slices * Ignore got files in testdata * Cleanup type rendering mechanism * Update JS runtime * Implement generic models * Add missing field in renderer initialisation * Improve generic creation code * Add generic model test * Add error reporting infrastructure * Support configurable file names * Detect file naming collisions * Print final error report * New shortcut file structure + collision detection * Update test layout and data * Autoconfiguration for analyser tests * Live progress reporting * Update code comments * Fix model doc rendering * Simplify name resolution * Add test for out of tree types * Fix generic creation code * Fix potential collisions between methods and models * Fix generic class alias rendering * Report model discovery in debug mode * Add interface mode for JS * Collect interface method comments * Add interface methods test * Unwrap generic instantiations in method receivers * Fix rendering of nullable types in interface mode * Fix rendering of class aliases * Expose promise cancel method to typescript * Update test data * Update binding example * Fix rendering of aliased quoted type params * Move to strongly typed bindings * Implement lightweight analyser * Update test cases * Update binding example * Add complex instantiation test * Load full dependency tree * Rewrite collector * Update renderer to match new collector * Update generator to match new collector * Update test data * Update binding example * Configure includes and injections by language * Improve system path resolution * Support rich conditions in inject/include directives * Fix error handling in Generator.Generate * Retrieve compiled go file paths from fileset * Do not rely on struct info in struct flattening algorithm * Fix doc comment for findDeclaraion * Fix bugs in embedded field handling * Fix bugs and comments in package collection * Remove useless fields from ServiceInfo * Fix empty line at the beginning of TS indexes * Remove global index and shortcuts * Remove generation tests for individual packages * Enforce lower-case file names * Update test data * Improve error reporting * Update binding example * Reintroduce go1.22 syntax * Improve relative import path computation * Improve alias support * Add alias test * Update test data * Remove no services error * Rename global analyser test * Add workaround and test for bug in typeutil.Map * Update test data * Do not split fully qualified names * Update typeutil package and remove workaround * Unify alias/named type handling * Fix rendering of generic named class aliases * Fix rendering of array types * Minor tweaks and cleanups * Rmove namespaced export construct * Update test data * Update binding example * Break type cycles * Fix typo in comment * Fix creation code for cyclic types * Fix type of variadic params in interface mode * Update test data * Fix bad whitespace * Refactor type assertions inside bound methods * Update test data * Rename field application.Options.Bind to Services * Rename parser package to generator * Update binding example * Update test data * Update generator readme * Add typescript test harness * Move test output to new subfolder * Fix code generation bugs * Use .js extensions in TS mode imports * Update test data * Revert default generator output dir to frontend/bindings * Bump runtime package version * Update templates * Update changelog * Improve newline handling --------- Co-authored-by: Andreas Bichinger <andreas.bichinger@gmail.com>
176 lines
4.6 KiB
Go
176 lines
4.6 KiB
Go
package render
|
|
|
|
import (
|
|
"fmt"
|
|
"go/types"
|
|
"strings"
|
|
"text/template"
|
|
|
|
"github.com/wailsapp/wails/v3/internal/generator/collect"
|
|
)
|
|
|
|
// JSDefault renders the Javascript representation
|
|
// of the zero value of the given type,
|
|
// using the receiver's import map to resolve dependencies.
|
|
//
|
|
// JSDefault's output may be incorrect
|
|
// if imports.AddType has not been called for the given type.
|
|
func (m *module) JSDefault(typ types.Type, quoted bool) (result string) {
|
|
switch t := typ.(type) {
|
|
case *types.Alias, *types.Named:
|
|
result, ok := m.renderNamedDefault(t.(aliasOrNamed), quoted)
|
|
if ok {
|
|
return result
|
|
}
|
|
|
|
case *types.Array, *types.Slice:
|
|
if types.Identical(typ, typeByteSlice) {
|
|
return `""`
|
|
} else {
|
|
return "[]"
|
|
}
|
|
|
|
case *types.Basic:
|
|
return m.renderBasicDefault(t, quoted)
|
|
|
|
case *types.Map:
|
|
return "{}"
|
|
|
|
case *types.Pointer:
|
|
return "null"
|
|
|
|
case *types.Struct:
|
|
return m.renderStructDefault(t)
|
|
}
|
|
|
|
// Fall back to null.
|
|
// encoding/json ignores null values so this is safe.
|
|
return "null"
|
|
}
|
|
|
|
// renderBasicDefault outputs the Javascript representation
|
|
// of the zero value for the given basic type.
|
|
func (*module) renderBasicDefault(typ *types.Basic, quoted bool) string {
|
|
switch {
|
|
case typ.Info()&types.IsBoolean != 0:
|
|
if quoted {
|
|
return `"false"`
|
|
} else {
|
|
return "false"
|
|
}
|
|
|
|
case typ.Info()&types.IsNumeric != 0 && typ.Info()&types.IsComplex == 0:
|
|
if quoted {
|
|
return `"0"`
|
|
} else {
|
|
return "0"
|
|
}
|
|
|
|
case typ.Info()&types.IsString != 0:
|
|
if quoted {
|
|
return `'""'`
|
|
} else {
|
|
return `""`
|
|
}
|
|
}
|
|
|
|
// Fall back to untyped mode.
|
|
if quoted {
|
|
return `""`
|
|
} else {
|
|
// encoding/json ignores null values so this is safe.
|
|
return "null"
|
|
}
|
|
}
|
|
|
|
// renderNamedDefault outputs the Javascript representation
|
|
// of the zero value for the given alias or named type.
|
|
// The result field named 'ok' is true when the resulting code is valid.
|
|
// If false, it must be discarded.
|
|
func (m *module) renderNamedDefault(typ aliasOrNamed, quoted bool) (result string, ok bool) {
|
|
if typ.Obj().Pkg() == nil {
|
|
// Builtin alias or named type: render underlying type.
|
|
return m.JSDefault(typ.Underlying(), quoted), true
|
|
}
|
|
|
|
if quoted {
|
|
// WARN: Do not test with IsString here!! We only want to catch marshalers.
|
|
if !collect.IsAny(typ) && !collect.MaybeTextMarshaler(typ) {
|
|
if basic, ok := typ.Underlying().(*types.Basic); ok {
|
|
// Quoted mode for basic alias/named type that is not a marshaler: delegate.
|
|
return m.renderBasicDefault(basic, quoted), true
|
|
}
|
|
// No need to handle typeparams: they are initialised to null anyways.
|
|
}
|
|
}
|
|
|
|
prefix := ""
|
|
if typ.Obj().Exported() && m.Imports.ImportModels {
|
|
prefix = "$models."
|
|
} else if !typ.Obj().Exported() && m.Imports.ImportInternal {
|
|
prefix = "$internal."
|
|
}
|
|
|
|
if collect.IsAny(typ) {
|
|
return "", false
|
|
} else if collect.MaybeTextMarshaler(typ) {
|
|
return `""`, true
|
|
} else if collect.IsClass(typ) {
|
|
if typ.Obj().Pkg().Path() == m.Imports.Self {
|
|
return fmt.Sprintf("(new %s%s())", prefix, jsid(typ.Obj().Name())), true
|
|
} else {
|
|
return fmt.Sprintf("(new %s.%s())", jsimport(m.Imports.External[typ.Obj().Pkg().Path()]), jsid(typ.Obj().Name())), true
|
|
}
|
|
} else if _, isAlias := typ.(*types.Alias); isAlias {
|
|
return m.JSDefault(types.Unalias(typ), quoted), true
|
|
} else {
|
|
// Inject a type assertion in case we are breaking an enum.
|
|
// Using the true Go zero value is preferrable to selecting an arbitrary enum value.
|
|
value := m.JSDefault(typ.Underlying(), quoted)
|
|
if typ.Obj().Pkg().Path() == m.Imports.Self {
|
|
if m.TS {
|
|
return fmt.Sprintf("(%s as %s%s)", value, prefix, jsid(typ.Obj().Name())), true
|
|
} else {
|
|
return fmt.Sprintf("(/** @type {%s%s} */(%s))", prefix, jsid(typ.Obj().Name()), value), true
|
|
}
|
|
} else {
|
|
if m.TS {
|
|
return fmt.Sprintf("(%s as %s.%s)", value, jsimport(m.Imports.External[typ.Obj().Pkg().Path()]), jsid(typ.Obj().Name())), true
|
|
} else {
|
|
return fmt.Sprintf("(/** @type {%s.%s} */(%s))", jsimport(m.Imports.External[typ.Obj().Pkg().Path()]), jsid(typ.Obj().Name()), value), true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// renderStructDefault outputs the Javascript representation
|
|
// of the zero value for the given struct type.
|
|
func (m *module) renderStructDefault(typ *types.Struct) string {
|
|
info := m.collector.Struct(typ)
|
|
info.Collect()
|
|
|
|
var builder strings.Builder
|
|
|
|
builder.WriteRune('{')
|
|
for i, field := range info.Fields {
|
|
if field.Optional {
|
|
continue
|
|
}
|
|
|
|
if i > 0 {
|
|
builder.WriteString(", ")
|
|
}
|
|
|
|
builder.WriteRune('"')
|
|
template.JSEscape(&builder, []byte(field.JsonName))
|
|
builder.WriteRune('"')
|
|
|
|
builder.WriteString(": ")
|
|
|
|
builder.WriteString(m.JSDefault(field.Type, field.Quoted))
|
|
}
|
|
builder.WriteRune('}')
|
|
|
|
return builder.String()
|
|
}
|