mirror of
https://github.com/wailsapp/wails.git
synced 2025-05-06 22:29:31 +08:00

* 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>
161 lines
4.3 KiB
Go
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,
|
|
})
|
|
}
|