Agent Skills: Makepad 1.x to 2.0 Migration Skill

|

UncategorizedID: zhanghandong/makepad-skills/makepad-2.0-migration

Install this agent skill to your local

pnpm dlx add-skill https://github.com/ZhangHanDong/makepad-skills/tree/HEAD/skills/makepad-2.0-migration

Skill Files

Browse the full folder contents for makepad-2.0-migration.

Download Skill

Loading file tree…

skills/makepad-2.0-migration/SKILL.md

Skill Metadata

Name
makepad-2.0-migration
Description
|

Makepad 1.x to 2.0 Migration Skill

Version: makepad-widgets (dev branch) | Last Updated: 2026-03-03

Overview

Makepad 2.0 is a fundamental architecture shift from compile-time static DSL to runtime scripting. Migration involves syntax changes, derive macro updates, lifecycle method renames, and new patterns for state management.

Documentation

Refer to the local files for detailed documentation:

  • ./references/migration-guide.md - Complete migration reference with examples

Quick Syntax Mapping Table

| Makepad 1.x | Makepad 2.0 | Notes | |-------------|-------------|-------| | live_design! { ... } | script_mod! { ... } | Core macro change | | <Widget> { ... } | Widget{ ... } | No angle brackets | | Key = Value | Key: value | Colon, not equals | | (THEME_COLOR) | theme.color_* | Theme namespace | | live_body: { ... } | body +: { ... } | Merge operator | | #[derive(Live)] | #[derive(Script)] | Derive macro | | #[derive(LiveHook)] | #[derive(ScriptHook)] | Lifecycle hooks | | #[derive(Widget)] | #[derive(Widget)] | Unchanged | | before_apply() | on_before_apply() | Method rename | | after_apply() | on_after_apply() | Method rename | | apply_over!() | script_apply_eval!() | Runtime updates | | DefaultNone | Default | Enum default | | LiveRegister | WidgetRegister | Widget registration | | live_register() | register_widget(vm) | Registration method | | LiveId | LiveId | Unchanged |


Step-by-Step Migration

Step 1: Replace Macro

// OLD
live_design! {
    import makepad_widgets::base::*;
    import makepad_widgets::theme_desktop_dark::*;

    App = {{App}} {
        ui: <Root> { ... }
    }
}

// NEW
script_mod! {
    use mod.prelude.widgets.*

    startup() do #(App::script_component(vm)){
        ui: Root{
            // ... UI definition
        }
    }
}

Step 2: Update Derives

// OLD
#[derive(Live, LiveHook, Widget)]
pub struct MyWidget { ... }

// NEW
#[derive(Script, ScriptHook, Widget)]
pub struct MyWidget { ... }

Step 3: Update App::run

// OLD
impl LiveRegister for App {
    fn live_register(cx: &mut Cx) {
        makepad_widgets::live_design(cx);
    }
}

// NEW
impl App {
    fn run(vm: &mut ScriptVm) -> Self {
        crate::makepad_widgets::script_mod(vm);
        App::from_script_mod(vm, self::script_mod)
    }
}

Step 4: Rename Lifecycle Methods

// OLD
impl LiveHook for MyWidget {
    fn before_apply(&mut self, cx: &mut Cx, ...) { ... }
    fn after_apply(&mut self, cx: &mut Cx, ...) { ... }
}

// NEW
impl ScriptHook for MyWidget {
    fn on_before_apply(&mut self, cx: &mut Cx, ...) { ... }
    fn on_after_apply(&mut self, cx: &mut Cx, ...) { ... }
}

Step 5: Update DSL Syntax

// OLD - angle brackets, equals signs
<View> {
    width: Fill, height: Fill
    show_bg: true
    draw_bg: { color: (THEME_BG) }

    title = <Label> {
        text: "Hello"
        draw_text: { color: #fff }
    }
}

// NEW - curly braces, colons, theme namespace
View{
    width: Fill height: Fill
    show_bg: true
    draw_bg.color: theme.color_bg_app

    title := Label{
        text: "Hello"
        draw_text.color: #fff
    }
}

Step 6: Replace apply_over with script_eval

// OLD
self.label(id!(title)).apply_over(cx, live! {
    text: "New text"
});

// NEW
script_eval!(cx, {
    ui.title.set_text("New text")
});
// OR
script_apply_eval!(cx, self.ui, {
    title.text: "New text"
});

Key Breaking Changes

  1. No commas between properties (whitespace-delimited)
  2. No semicolons anywhere in Splash
  3. No angle brackets for widget types
  4. Theme constants use theme.* prefix, not (THEME_*) syntax
  5. Named children use := operator, not =
  6. Merge operator is +: not : for extending parent properties
  7. height: Fit is MANDATORY on containers (default is 0px, not auto)
  8. Registration happens in App::run, not live_register
  9. Field attribute #[source] links to script object (required on some widgets)
  10. Old old/ directory contains archived 1.x code for reference

Common Migration Mistakes

| Mistake | Symptom | Fix | |---------|---------|-----| | Still using live_design! | Compile error | Replace with script_mod! | | Using <Widget> syntax | Parse error | Use Widget{} | | Using Key = Value | Property not applied | Use Key: value | | Using (THEME_COLOR) | Unknown token | Use theme.color_* | | Missing height: Fit | Container invisible (0px) | Add height: Fit | | Using Live derive | Compile error | Use Script | | Using before_apply | Method not found | Use on_before_apply | | Using commas between props | Parse error | Remove commas |


Best Practices for Migration

  1. Start with examples - Study examples/counter and examples/todo for 2.0 patterns
  2. Migrate one widget at a time - Don't try to convert everything at once
  3. Check the old/ directory - Compare old vs new widget implementations
  4. Test height: Fit - Most invisible UI is caused by missing height
  5. Use theme variables - Replace all hardcoded theme colors with theme.*
  6. Add new_batch: true - Any View with show_bg and text children needs it
Makepad 1.x to 2.0 Migration Skill Skill | Agent Skills