Lewati ke konten

Model Data

Wails secara otomatis menghasilkan class JavaScript/TypeScript dari struct Go, memberikan type safety penuh saat meneruskan data kompleks antara backend dan frontend. Tulis struct Go, hasilkan binding, dan dapatkan model frontend bertipe penuh lengkap dengan constructor, anotasi tipe, dan komentar JSDoc.

Struct Go:

type User struct {
ID int `json:"id"`
Name string `json:"name"`
Email string `json:"email"`
CreatedAt time.Time `json:"createdAt"`
}
func (s *UserService) GetUser(id int) (*User, error) {
// Return user
}

Hasilkan:

Terminal window
wails3 generate bindings

JavaScript:

import { GetUser } from './bindings/changeme/userservice'
import { User } from './bindings/changeme/models'
const user = await GetUser(1)
console.log(user.Name) // Type-safe!

Itu saja! Type safety penuh melintasi bridge.

type Person struct {
Name string
Age int
}

JavaScript yang Dihasilkan:

export class Person {
/** @type {string} */
Name = ""
/** @type {number} */
Age = 0
constructor(source = {}) {
Object.assign(this, source)
}
static createFrom(source = {}) {
return new Person(source)
}
}
type User struct {
ID int `json:"id"`
Name string `json:"name"`
Email string `json:"email"`
CreatedAt time.Time `json:"createdAt"`
}

JavaScript yang Dihasilkan:

export class User {
/** @type {number} */
id = 0
/** @type {string} */
name = ""
/** @type {string} */
email = ""
/** @type {Date} */
createdAt = new Date()
constructor(source = {}) {
Object.assign(this, source)
}
}

Tag JSON mengontrol nama field di JavaScript.

// User represents an application user
type User struct {
// Unique identifier
ID int `json:"id"`
// Full name of the user
Name string `json:"name"`
// Email address (must be unique)
Email string `json:"email"`
}

JavaScript yang Dihasilkan:

/**
* User represents an application user
*/
export class User {
/**
* Unique identifier
* @type {number}
*/
id = 0
/**
* Full name of the user
* @type {string}
*/
name = ""
/**
* Email address (must be unique)
* @type {string}
*/
email = ""
}

Komentar menjadi JSDoc! IDE Anda menampilkannya.

type Address struct {
Street string `json:"street"`
City string `json:"city"`
Country string `json:"country"`
}
type User struct {
ID int `json:"id"`
Name string `json:"name"`
Address Address `json:"address"`
}

JavaScript yang Dihasilkan:

export class Address {
/** @type {string} */
street = ""
/** @type {string} */
city = ""
/** @type {string} */
country = ""
}
export class User {
/** @type {number} */
id = 0
/** @type {string} */
name = ""
/** @type {Address} */
address = new Address()
}

Penggunaan:

const user = new User({
id: 1,
name: "Alice",
address: new Address({
street: "123 Main St",
city: "Springfield",
country: "USA"
})
})
type Team struct {
Name string `json:"name"`
Members []string `json:"members"`
}

JavaScript yang Dihasilkan:

export class Team {
/** @type {string} */
name = ""
/** @type {string[]} */
members = []
}

Penggunaan:

const team = new Team({
name: "Engineering",
members: ["Alice", "Bob", "Charlie"]
})
type Config struct {
Settings map[string]string `json:"settings"`
}

JavaScript yang Dihasilkan:

export class Config {
/** @type {Record<string, string>} */
settings = {}
}

Penggunaan:

const config = new Config({
settings: {
theme: "dark",
language: "en"
}
})
Tipe GoJavaScript/TypeScript
stringstring
boolboolean
int, int8, int16, int32, int64number
uint, uint8, uint16, uint32, uint64number
float32, float64number
bytenumber
runenumber
Tipe GoJavaScript/TypeScriptCatatan
[]TT[]-
[N]TT[]-
map[string]T{ [_: string]: T }map dengan key string
map[K]V{ [_ in K]?: V }K non-string dirender sebagai mapped type, bukan JS Map
[]bytestringbase64-encoded
structclass / interfacedengan field
time.Timeanydiserialisasi sebagai string RFC3339Nano di runtime
*TT | nullpointer berarti nullable
any / interface{}any-
errorany / ExceptionException jika sebagai nilai return, selain itu any
  • chan T (channel)
  • func() (fungsi)
  • Interface kompleks (kecuali interface{})
  • Field tidak diekspor (lowercase)
import { User } from './bindings/changeme/models'
// Empty instance
const user1 = new User()
// With data
const user2 = new User({
id: 1,
name: "Alice",
})
// From JSON string
const user3 = User.createFrom('{"id":1,"name":"Alice"}')
// From object
const user4 = User.createFrom({ id: 1, name: "Alice" })
import { CreateUser } from './bindings/changeme/userservice'
import { User } from './bindings/changeme/models'
const user = new User({
name: "Bob",
})
const created = await CreateUser(user)
console.log("Created user:", created.id)
import { GetUser } from './bindings/changeme/userservice'
const user = await GetUser(1)
// user is already a User instance
console.log(user.name)
console.log(user.email)
console.log(user.createdAt.toISOString())
import { GetUser, UpdateUser } from './bindings/changeme/userservice'
// Get user
const user = await GetUser(1)
// Modify
user.name = "Alice Smith"
user.email = "[email protected]"
// Save
await UpdateUser(user)
Terminal window
wails3 generate bindings -ts

Yang Dihasilkan:

/**
* User represents an application user
*/
export class User {
/**
* Unique identifier
*/
id: number = 0
/**
* Full name of the user
*/
name: string = ""
/**
* Email address (must be unique)
*/
email: string = ""
/**
* Account creation date
*/
createdAt: Date = new Date()
constructor(source: Partial<User> = {}) {
Object.assign(this, source)
}
static createFrom(source: string | Partial<User> = {}): User {
const parsedSource = typeof source === 'string'
? JSON.parse(source)
: source
return new User(parsedSource)
}
}
import { GetUser, CreateUser } from './bindings/changeme/userservice'
import { User } from './bindings/changeme/models'
async function example() {
// Create user
const newUser = new User({
name: "Alice",
})
const created: User = await CreateUser(newUser)
// Get user
const user: User = await GetUser(created.id)
// Type-safe access
console.log(user.name.toUpperCase()) // ✅ string method
console.log(user.id + 1) // ✅ number operation
console.log(user.createdAt.getTime()) // ✅ Date method
}
type User struct {
ID int `json:"id"`
Name string `json:"name"`
Nickname *string `json:"nickname,omitempty"`
}

JavaScript:

const user = new User({
id: 1,
name: "Alice",
nickname: "Ally" // Optional
})
// Check if set
if (user.nickname) {
console.log("Nickname:", user.nickname)
}

Generator binding secara otomatis mendeteksi tipe bernama Go dengan konstanta dan menghasilkan enum TypeScript atau const object JavaScript — termasuk anggota $zero untuk nilai nol Go dan preservasi JSDoc penuh.

type UserRole string
const (
RoleAdmin UserRole = "admin"
RoleUser UserRole = "user"
RoleGuest UserRole = "guest"
)

TypeScript yang Dihasilkan:

export enum UserRole {
$zero = "",
RoleAdmin = "admin",
RoleUser = "user",
RoleGuest = "guest",
}

Penggunaan:

import { User, UserRole } from './bindings/changeme/models'
const admin = new User({
name: "Admin",
role: UserRole.RoleAdmin
})

Untuk cakupan lengkap enum string, enum integer, type alias, enum paket yang diimpor, dan keterbatasan, lihat halaman khusus Enum.

class User {
validate() {
if (!this.name) {
throw new Error("Name is required")
}
if (!this.email.includes('@')) {
throw new Error("Invalid email")
}
return true
}
}
// Use
const user = new User({ name: "Alice", email: "[email protected]" })
user.validate() // ✅
const invalid = new User({ name: "", email: "invalid" })
invalid.validate() // ❌ Throws
// To JSON
const json = JSON.stringify(user)
// From JSON
const user = User.createFrom(json)
// To plain object
const obj = { ...user }
// From plain object
const user2 = new User(obj)

Go:

package main
import (
"fmt"
"time"
"github.com/wailsapp/wails/v3/pkg/application"
)
type Address struct {
Street string `json:"street"`
City string `json:"city"`
Country string `json:"country"`
}
type User struct {
ID int `json:"id"`
Name string `json:"name"`
Email string `json:"email"`
Address Address `json:"address"`
CreatedAt time.Time `json:"createdAt"`
}
type UserService struct {
users []User
}
func (s *UserService) GetAll() []User {
return s.users
}
func (s *UserService) GetByID(id int) (*User, error) {
for _, user := range s.users {
if user.ID == id {
return &user, nil
}
}
return nil, fmt.Errorf("user %d not found", id)
}
func (s *UserService) Create(user User) User {
user.ID = len(s.users) + 1
user.CreatedAt = time.Now()
s.users = append(s.users, user)
return user
}
func (s *UserService) Update(user User) error {
for i, u := range s.users {
if u.ID == user.ID {
s.users[i] = user
return nil
}
}
return fmt.Errorf("user %d not found", user.ID)
}
func main() {
app := application.New(application.Options{
Services: []application.Service{
application.NewService(&UserService{}),
},
})
app.Window.New()
app.Run()
}

JavaScript:

import { GetAll, GetByID, Create, Update } from './bindings/changeme/userservice'
import { User, Address } from './bindings/changeme/models'
class UserManager {
async loadUsers() {
const users = await GetAll()
this.renderUsers(users)
}
async createUser(name, email, address) {
const user = new User({
name,
email,
address: new Address(address)
})
try {
const created = await Create(user)
console.log("Created user:", created.id)
this.loadUsers()
} catch (error) {
console.error("Failed to create user:", error)
}
}
async updateUser(id, updates) {
try {
const user = await GetByID(id)
Object.assign(user, updates)
await Update(user)
this.loadUsers()
} catch (error) {
console.error("Failed to update user:", error)
}
}
renderUsers(users) {
const list = document.getElementById('users')
list.innerHTML = users.map(user => `
<div class="user">
<h3>${user.name}</h3>
<p>${user.email}</p>
<p>${user.address.city}, ${user.address.country}</p>
<small>Created: ${user.createdAt.toLocaleDateString()}</small>
</div>
`).join('')
}
}
const manager = new UserManager()
manager.loadUsers()
  • Gunakan tag JSON - Kontrol nama field
  • Tambahkan komentar - Menjadi JSDoc
  • Gunakan time.Time - Dikonversi ke Date
  • Validasi di sisi Go - Jangan percaya frontend
  • Jaga model sederhana - Hanya sebagai container data
  • Gunakan pointer untuk opsional - *string untuk nullable
  • Jangan tambahkan metode ke struct Go - Jaga sebagai data
  • Jangan gunakan field tidak diekspor - Tidak akan ter-bind
  • Jangan gunakan interface kompleks - Tidak didukung
  • Jangan lupa tag JSON - Nama field penting
  • Jangan nest terlalu dalam - Jaga tetap sederhana

Ada pertanyaan? Tanyakan di Discord atau lihat contoh binding.