5
0
mirror of https://github.com/wailsapp/wails.git synced 2025-05-06 22:29:31 +08:00
wails/v3/internal/generator/render/renderer.go
Fabio Massaioli f01b4b9a21
[v3] Fix binding generator bugs (#4001)
* Add some clarifying comments

* Remove special handling of window parameters

* Improve internal method exclusion

* Add test for internal method exclusion

* Remove useless blank field from app options

This is a leftover from an older version of the static analyser. It should have been removed long ago.

* Remove redundant godebug setting

gotypesalias=1 is the default starting with go1.23

* Use new range for syntax to simplify code

* Remove generator dependency on github.com/samber/lo

* Ensure generator testing tasks do not use the test cache

* Rename cyclic types test

* Test for cyclic imports

* Fix import cycle between model files

* Sort class aliases after their aliased class

* Test class aliases

* Fix length of default value for array types

* Test array initialization

* Add changelog

* Update changelog

* Fix contrived marking technique in model sorting algorithm

* Update binding example

* Update test data

---------

Co-authored-by: Lea Anthony <lea.anthony@gmail.com>
2025-01-17 18:56:07 +11:00

161 lines
4.3 KiB
Go

package render
import (
"go/types"
"io"
"slices"
"strings"
"text/template"
"github.com/wailsapp/wails/v3/internal/flags"
"github.com/wailsapp/wails/v3/internal/generator/collect"
)
// Renderer holds the template set for a given configuration.
// It provides methods for rendering various output modules.
type Renderer struct {
options *flags.GenerateBindingsOptions
collector *collect.Collector
ext string
service *template.Template
typedefs *template.Template
}
// NewRenderer initialises a code renderer
// for the given configuration and data collector.
func NewRenderer(options *flags.GenerateBindingsOptions, collector *collect.Collector) *Renderer {
ext := ".js"
if options.TS {
ext = ".ts"
}
return &Renderer{
options: options,
collector: collector,
ext: ext,
service: tmplService[tmplLanguage(options.TS)],
typedefs: tmplTypedefs[tmplLanguage(options.TS)],
}
}
// ServiceFile returns the standard name of a service file
// for the given struct name, with the appropriate extension.
func (renderer *Renderer) ServiceFile(name string) string {
return strings.ToLower(name) + renderer.ext
}
// ModelsFile returns the standard name of a models file
// with the appropriate extension.
func (renderer *Renderer) ModelsFile() string {
return renderer.options.ModelsFilename + renderer.ext
}
// InternalFile returns the standard name of an internal model file
// with the appropriate extension.
func (renderer *Renderer) InternalFile() string {
return renderer.options.InternalFilename + renderer.ext
}
// IndexFile returns the standard name of a package index file
// with the appropriate extension.
func (renderer *Renderer) IndexFile() string {
return renderer.options.IndexFilename + renderer.ext
}
// Service renders binding code for the given service type to w.
func (renderer *Renderer) Service(w io.Writer, info *collect.ServiceInfo) error {
return renderer.service.Execute(w, &struct {
module
Service *collect.ServiceInfo
}{
module{
Renderer: renderer,
GenerateBindingsOptions: renderer.options,
Imports: info.Imports,
},
info,
})
}
// Typedefs renders type definitions for the given list of models.
func (renderer *Renderer) Typedefs(w io.Writer, imports *collect.ImportMap, models []*collect.ModelInfo) error {
if !renderer.options.UseInterfaces {
// Sort class aliases after the class they alias.
// Works in amortized linear time thanks to an auxiliary map.
// Track postponed class aliases and their dependencies.
aliases := make(map[types.Object][]*collect.ModelInfo, len(models))
models = slices.Clone(models)
for i, j := 0, 0; i < len(models); i++ {
if models[i].Type != nil && collect.IsClass(models[i].Type) {
// models[i] is a class alias:
// models[i].Type is guaranteed to be
// either an alias or a named type
obj := models[i].Type.(interface{ Obj() *types.TypeName }).Obj()
if obj.Pkg().Path() == imports.Self {
// models[i] aliases a type from the current module.
if a, ok := aliases[obj]; !ok || len(a) > 0 {
// The aliased type has not been visited already, postpone.
aliases[obj] = append(a, models[i])
continue
}
}
}
// Append models[i].
models[j] = models[i]
j++
// Keep appending aliases whose aliased type has been just appended.
for k := j - 1; k < j; k++ {
a := aliases[models[k].Object()]
aliases[models[k].Object()] = nil // Mark aliased model as visited
j += copy(models[j:], a)
}
}
}
return renderer.typedefs.Execute(w, &struct {
module
Models []*collect.ModelInfo
}{
module{
Renderer: renderer,
GenerateBindingsOptions: renderer.options,
Imports: imports,
},
models,
})
}
// Models renders exported models for the given package index to w.
func (renderer *Renderer) Models(w io.Writer, index *collect.PackageIndex) error {
return tmplModels.Execute(w, &struct {
*collect.PackageIndex
*Renderer
*flags.GenerateBindingsOptions
}{
index,
renderer,
renderer.options,
})
}
// Index renders the given package index to w.
func (renderer *Renderer) Index(w io.Writer, index *collect.PackageIndex) error {
return tmplIndex.Execute(w, &struct {
*collect.PackageIndex
*Renderer
*flags.GenerateBindingsOptions
}{
index,
renderer,
renderer.options,
})
}