A struct groups related values under one name. Instead of keeping a products name, price, and stock count in three separate variables that can drift out of sync, you bundle them together into a Product struct. Each piece of data inside the struct is called a field. You read and write fields using dot notation: product.price'. Here is a simple product record. All four fields are plain integers or text.
struct Product {
name: text,
price: integer,
stock: integer
}
Field Constraints
You can restrict what values a field may hold. limit(min, max) rejects any value outside that range at runtime. not null tells the field that zero is a real data value — without it, zero is treated as "no value" (null). Colour channels can all be zero (pure black is a valid colour), so all three need not null. Fields you omit in a constructor receive zero (or null for nullable fields) by default.
struct Colour {
r: integer limit(0, 255) not null,
g: integer limit(0, 255) not null,
b: integer limit(0, 255) not null
}
Methods
A method is a function whose first parameter is named self. Loft uses the type of self to decide which struct the method belongs to. Call it with dot notation: c.to_hex(). A method can read fields via self.field and return any type.
fn to_hex(self: Colour) -> integer {
self.r * 0x10000 + self.g * 0x100 + self.b
}
A method can also return a completely new struct value. Write -> StructName as the return type and construct the value in the body. The original struct is not modified — the caller gets a brand-new copy.
fn dimmed(self: Colour) -> Colour {
Colour {r: self.r / 2, g: self.g / 2, b: self.b / 2 }
}
Defaults and Computed Fields
Write = expression after the type to set a stored default, filled at construction. Use computed(expression) for a field that recalculates every time you read it — it takes no space in the record and always reflects the current state. Inside both, $ refers to the struct.
struct Item {
name: text,
name_length: integer = len($.name)
}
struct Circle {
radius: float,
area: float computed(3.14159265 * $.radius * $.radius)
}
Storing many structs in a vector
Area uses smaller integer types to save memory: u16 holds values 0–65535, and u8 holds values 0–255. This matters when you have millions of records (such as tiles in a game map) and memory usage counts.
struct Area {
height: u16,
terrain: u8,
water: u8,
direction: u8
}
fn main() {
Constructing a struct: name the fields you want to set. Fields you leave out default to zero (or null for nullable fields).
apple = Product {name: "Apple", price: 120, stock: 50 };
assert(apple.price == 120, "price field: {apple.price}");
assert(apple.name == "Apple", "name field: {apple.name}");
You can read and write individual fields after construction.
apple.stock -= 1;
assert(apple.stock == 49, "stock after one sale: {apple.stock}");
A field omitted from the constructor gets zero as its default.
col = Colour {r: 128, b: 128 };
assert(col.g == 0, "omitted green channel defaults to zero");
Formatting a struct shows all fields compactly.
assert("{col}" == "{{r:128,g:0,b:128}}", "Struct compact formatting");
:j produces JSON output — a text format used by many web APIs and config tools.
assert("{col:j}" == "{{\"r\":128,\"g\":0,\"b\":128}}", "JSON format");
Call a method on a variable using dot notation.
purple = Colour {r: 128, b: 128 };
assert("{purple.to_hex():x}" == "800080", "hex method result");
You can call a method directly on a constructor expression.
assert(Colour {r: 255, g: 0, b: 0 }.to_hex() == 0xff0000, "method on constructor");
dimmed returns a new Colour with each channel halved. The original variable is unchanged.
dark = purple.dimmed();
assert(dark.r == 64, "dimmed r: {dark.r}");
assert(dark.b == 64, "dimmed b: {dark.b}");
assert(dark.g == 0, "dimmed g: {dark.g}");
Stored defaults are filled once at construction time.
it = Item {name: "hello" };
assert(it.name_length == 5, "stored default name_length: {it.name_length}");
computed() fields recalculate on every access — changing radius updates area.
ci = Circle { radius: 5.0 };
assert(ci.area > 78.0, "computed area: {ci.area}");
ci.radius = 10.0;
assert(ci.area > 314.0, "recomputed area: {ci.area}");
Fill a vector with copies of the same struct using ; count syntax. This creates 16 Area records, all with the same initial values.
map =[Area {height: 0, terrain: 1, water: 1, direction: 1 };
16];
assert("{map[3]}" == "{{height:0,terrain:1,water:1,direction:1}}", "tile record");
map[3].height = 200;
assert(map[3].height == 200, "individual tile update");
sizeof
sizeof(Type) returns the packed byte size used when the type is stored as a struct field or vector element. Range-constrained integer types like u8 and u16 report their packed size, not the 4-byte stack slot size.
assert(sizeof(integer) == 4, "integer: 4 bytes");
assert(sizeof(u8) == 1, "u8: 1 byte (packed)");
assert(sizeof(u16) == 2, "u16: 2 bytes (packed)");
assert(sizeof(Colour) == 3, "Colour: 3 × u8 = 3 bytes");
assert(sizeof(Area) == 5, "Area: u16 + 3 × u8 = 5 bytes");
}