5
0
mirror of https://github.com/wailsapp/wails.git synced 2025-05-04 23:23:48 +08:00
wails/v3/internal/generator/render/doc.go
Fabio Massaioli 37673eb24d
[v3] Fix binding generator bugs and prepare for Go 1.24 (#4045)
* Rename predicates source file

* Overhaul and document type predicates

* Fix model collection logic for named types

* Fix map key type rendering

* Fix map creation code

* Fix rendering of structs that implement marshaler interfaces

* Fix type cycle detection to take type args into account

* Fix enum and typeparam field initialisation

* Improve unsupported type warnings

* Remove internal models file

* Deduplicate template code

* Accept generic aliases in static analyser

* Support new `encoding/json` flag `omitzero`

* Handle special cases when rendering generic aliases

* Update npm test dependencies

* Test class aliases and implicit private dependencies

* Test marshaler combinations

* Test map key types

* Remove bad map keys from unrelated tests

* Test service discovery through generic aliases

* Test generic aliases

* Test warning messages

* Disable go1.24 tests

* Update changelog

* Restore rendering of injected lines in index file

* Test directives

* Add wails:ignore directive

* Fix typo

* Move injections to the bottom of service files

* Handle errors from closing files

* Do not emit messages when services define only lifecycle methods

* Add internal directive for services and models

* Update changelog

* Fix error in service templates

* Test internal directive on services/models

* Fix error in index template

* Base testdata updates

* Testdata for class aliases and implicit private dependencies

* Testdata for marshaler combinations

* Testdata for map key types

* Testdata for bad map key fixes

* Add weakly typed enums aka alias constants

* Testdata for enum and typeparam field fixes

* Testdata for generic aliases

* Testdata for warning messages

* Testdata for directives

* Testdata for weakly typed enums

* Update binding example

* Update services example

* Remove go1.24 testdata

* Update cli doc

* Fix analyser tests

* Fix windows tests... hopefully

* go mod tidy on examples

* Update bindings guide

---------

Co-authored-by: Lea Anthony <lea.anthony@gmail.com>
2025-02-09 09:44:34 +11:00

137 lines
3.8 KiB
Go

package render
import (
"bufio"
"bytes"
"go/ast"
"strings"
"unicode"
"github.com/wailsapp/wails/v3/internal/generator/collect"
)
// hasdoc checks whether the given comment group contains actual doc comments.
func hasdoc(group *ast.CommentGroup) bool {
if group == nil {
return false
}
// TODO: this is horrible, make it more efficient?
return strings.ContainsFunc(group.Text(), func(r rune) bool { return !unicode.IsSpace(r) })
}
var commentTerminator = []byte("*/")
// jsdoc splits the given comment into lines and rewrites it as follows:
// - first, line terminators are stripped;
// - then a line terminator, the indent string and ' * '
// are prepended to each line;
// - occurrences of the comment terminator '*/' are replaced with '* /'
// to avoid accidentally terminating the surrounding comment.
//
// All lines thus modified are joined back together.
//
// The returned string can be inserted in a multiline JSDoc comment
// with the given indentation.
func jsdoc(comment string, indent string) string {
var builder strings.Builder
prefix := []byte(Newline + indent + " * ")
scanner := bufio.NewScanner(bytes.NewReader([]byte(comment)))
for scanner.Scan() {
line := scanner.Bytes()
// Prepend prefix.
builder.Write(prefix)
// Escape comment terminators.
for t := bytes.Index(line, commentTerminator); t >= 0; t = bytes.Index(line, commentTerminator) {
builder.Write(line[:t+1])
builder.WriteRune(' ')
line = line[t+1:]
}
builder.Write(line)
}
return builder.String()
}
// jsdocline removes all newlines in the given comment
// and escapes comment terminators using the same strategy as jsdoc.
func jsdocline(comment string) string {
var builder strings.Builder
scanner := bufio.NewScanner(bytes.NewReader([]byte(comment)))
for scanner.Scan() {
line := bytes.TrimSpace(scanner.Bytes())
if len(line) == 0 {
// Skip empty lines.
continue
}
// Prepend space to separate lines.
builder.WriteRune(' ')
// Escape comment terminators.
for t := bytes.Index(line, commentTerminator); t >= 0; t = bytes.Index(line, commentTerminator) {
builder.Write(line[:t+1])
builder.WriteRune(' ')
line = line[t+1:]
}
builder.Write(line)
}
// Return resulting string, but skip initial space.
return builder.String()[1:]
}
// isjsdocid returns true if the given string is a valid ECMAScript identifier,
// excluding unicode escape sequences. This is the property name format supported by JSDoc.
func isjsdocid(name string) bool {
for i, r := range name {
if i == 0 && !id_start(r) && r != '$' && r != '_' {
return false
} else if i > 0 && !id_continue(r) && r != '$' {
return false
}
}
return true
}
// isjsdocobj returns true if all field names in the given model
// are valid jsdoc property names.
func isjsdocobj(model *collect.ModelInfo) bool {
if len(model.Fields) == 0 {
return false
}
for _, decl := range model.Fields {
for _, field := range decl {
if !isjsdocid(field.JsonName) {
return false
}
}
}
return true
}
// id_start returns true if the given rune is in the ID_Start category
// according to UAX#31 (https://unicode.org/reports/tr31/).
func id_start(r rune) bool {
return (unicode.IsLetter(r) ||
unicode.Is(unicode.Nl, r) ||
unicode.Is(unicode.Other_ID_Start, r)) && !unicode.Is(unicode.Pattern_Syntax, r) && !unicode.Is(unicode.Pattern_White_Space, r)
}
// id_continue returns true if the given rune is in the ID_Continue category
// according to UAX#31 (https://unicode.org/reports/tr31/).
func id_continue(r rune) bool {
return (id_start(r) ||
unicode.Is(unicode.Mn, r) ||
unicode.Is(unicode.Mc, r) ||
unicode.Is(unicode.Nd, r) ||
unicode.Is(unicode.Pc, r) ||
unicode.Is(unicode.Other_ID_Continue, r)) && !unicode.Is(unicode.Pattern_Syntax, r) && !unicode.Is(unicode.Pattern_White_Space, r)
}