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

* 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>
137 lines
3.8 KiB
Go
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)
|
|
}
|