UNPKG

@jigx/mdk

Version:

Jigx Mobile Development Kit - SDK for building Jigx applications

363 lines 45.8 kB
import { application, INP } from '../src'; /** * Template expense tracking app demonstrating Jigx mobile patterns. * ADAPT THIS: Replace 'expense' with your domain (tasks, inventory, etc.) * Add/edit/remove entities, datasources, screens, tabs, etc * Explore Jigx SDK types and docs for more info */ // ======================================== // CONSTANTS - IDs and references // ======================================== // Screen Ids const SCREEN = { HOME: 'home', EXPENSE_LIST: 'expense-list', ADD_EXPENSE: 'add-expense', EDIT_EXPENSE: 'edit-expense' }; // Datasource Ids const DATASOURCE = { EXPENSES: 'data-expenses', EXPENSE_CATEGORIES: 'expense-categories', EXPENSE_STATS: 'expense-stats', RECENT_EXPENSES: 'recent-expenses', EXPENSES_SUMMARY: 'expenses-summary', CURRENT_EXPENSE: 'current-expense' }; // Database entities const ENTITY = { EXPENSES: 'default/expenses' }; // Tab Ids const TABS = { HOME: 'home' }; // ======================================== // APPLICATION - Core app configuration // ======================================== export const app = application('expense-tracker') // Unique programmatic Id .title('Expense Tracker') // Display name .category('personal'); // App store category (choices in doc-comments) // Global expressions - available everywhere app.expression('formatCurrency', `=function($amount) { '$' & $formatNumber($amount, '#,##0.00') }`); app.expression('formatDate', `=function($date) { $fromMillis($toMillis($date), '[D1o] [MNn] [Y]') }`); // ======================================== // DATABASE - SQLite table definitions // ======================================== /* Database tables - JSON docs in sqlite */ app.addDatabase .default // Only ONE db, always named 'default' .table('expenses'); // Declares [default/expenses] table in DB // ======================================== // SHARED DATASOURCES - Available to all screens // ======================================== /* Main data provider - queries all expenses */ app.addDatasource.sqlite(DATASOURCE.EXPENSES, 'dynamic') // 'dynamic' = offline sync .entity(ENTITY.EXPENSES) // Table(s) to query .query(/* sql */ ` SELECT exp.id, -- Always include row id -- Option 1: Return entire JSON doc (preferred) exp.data -- Option 2: Extract specific fields for expressions -- json_extract(exp.data, '$.title') AS title FROM [${ENTITY.EXPENSES}] AS exp ORDER BY -- Always extract for joins, sorting, etc json_extract(exp.data, '$.date') DESC `) // Decorate data as 'this is a json document' .jsonProperties('data'); /* Static data - no database needed, ideal for fixed lists */ app.addDatasource.static(DATASOURCE.EXPENSE_CATEGORIES, [ { id: 'food', name: 'Food & Dining' }, { id: 'transport', name: 'Transportation' }, { id: 'accommodation', name: 'Accommodation' }, { id: 'office', name: 'Office Supplies' }, { id: 'technology', name: 'Technology' }, { id: 'entertainment', name: 'Entertainment' }, { id: 'health', name: 'Health & Medical' }, { id: 'education', name: 'Education & Training' }, { id: 'other', name: 'Other' } ]); // ======================================== // HOME SCREEN - Main dashboard // ======================================== const homeScreen = app.addScreen .default(SCREEN.HOME, 'Expense Tracker'); { // Screen-specific datasource for aggregate stats homeScreen.addDatasource.sqlite(DATASOURCE.EXPENSE_STATS, 'dynamic') .entity(ENTITY.EXPENSES) .query(/* sql */ ` SELECT COUNT(1) AS count, SUM(json_extract(exp.data, '$.amount')) AS total_amount, AVG(json_extract(exp.data, '$.amount')) AS avg_amount, MAX(json_extract(exp.data, '$.amount')) AS max_amount FROM [${ENTITY.EXPENSES}] AS exp `); //.isDocument() // Return single document instead of array // Card component - container for related fields const card = homeScreen.addControl.card(); { card.addControl.list() .data(DATASOURCE.EXPENSE_STATS) .addControl.listItem() // Call global function to format currency .title('=$formatCurrency(@ctx.current.item.total_amount) & " (" & @ctx.current.item.count & " items)"'); } // Recent items (LOCAL DATASOURCE) homeScreen.addDatasource.sqlite(DATASOURCE.RECENT_EXPENSES, 'dynamic') .entity(ENTITY.EXPENSES) .query(/* sql */ ` SELECT exp.id, exp.data FROM [${ENTITY.EXPENSES}] AS exp ORDER BY json_extract(exp.data, '$.date') DESC LIMIT 5 `) .jsonProperties('data'); // List component - displays collection const list = homeScreen.addControl .list('recent-list') .data(DATASOURCE.RECENT_EXPENSES); // Bind to datasource list.addControl.listItem() .title('=@ctx.current.item.data.title') .subtitle('=@ctx.current.item.data.categoryId & " • " & $formatDate(@ctx.current.item.data.date)') .description('=$formatCurrency(@ctx.current.item.data.amount)') .onPress .goto(SCREEN.EDIT_EXPENSE) // Navigate to screen .parameter('id', '=@ctx.current.item.id') .parameter('expenseItem', '=@ctx.current.item.data'); // Buttons at bottom of screen const homeButtons = homeScreen.bottomPanel.buttons(2); // Show 2 of 3 max homeButtons.add.goto('Add Expense', SCREEN.ADD_EXPENSE); homeButtons.add.goto('View All', SCREEN.EXPENSE_LIST); // Show all expenses } // ======================================== // EXPENSE LIST SCREEN // ======================================== /* Expense list - all transactions */ const expenseListScreen = app.addScreen .default(SCREEN.EXPENSE_LIST, 'All Expenses'); { // LOCAL datasource (only available in this screen) expenseListScreen.addDatasource.sqlite(DATASOURCE.EXPENSES_SUMMARY, 'dynamic') .entity(ENTITY.EXPENSES) .query(/* sql */ ` SELECT json_extract(exp.data, '$.categoryId') AS categoryId, SUM(json_extract(exp.data, '$.amount')) AS total_amount, COUNT(1) AS count FROM [${ENTITY.EXPENSES}] AS exp GROUP BY categoryId ORDER BY total_amount DESC `) .isDocument(false); // In this case we want the array, since it is multiple categories // Summary card const summaryCard = expenseListScreen.addControl.card(); { summaryCard.addControl.textField() .label('Summary by Category') .value('Category Breakdown') .isDisabled(); summaryCard.addControl.list() .data(DATASOURCE.EXPENSES_SUMMARY) .addControl.listItem() .title('=@ctx.current.item.categoryId') .subtitle('=$formatCurrency(@ctx.current.item.total_amount) & " • " & @ctx.current.item.count & " expenses"'); } // Expense list const expenseList = expenseListScreen.addControl.list('expense-list') .data(DATASOURCE.EXPENSES); // Bind to global datasource expenseList.addControl.listItem() .title('=@ctx.current.item.data.title') .subtitle('=@ctx.current.item.data.categoryId & " • " & @ctx.current.item.data.date') .description('=$formatCurrency(@ctx.current.item.data.amount)') .onPress .goto(SCREEN.EDIT_EXPENSE) // Actual parameters .parameter('id', '=@ctx.current.item.id') .parameter('expenseItem', '=@ctx.current.item.data'); // Actions const expenseListButtons = expenseListScreen.bottomPanel.buttons(); expenseListButtons.add.goto('Add New', SCREEN.ADD_EXPENSE); } // ======================================== // ADD EXPENSE SCREEN // ======================================== /* Add expense form */ const addExpenseScreen = app.addScreen .default(SCREEN.ADD_EXPENSE, 'Add Expense'); { // Clear previous form state on screen focus (else will pre-populate using old values) const formInstanceId = 'add-expense-form'; addExpenseScreen.onFocus .clearState(`=@ctx.components.${formInstanceId}.state.value`); // Form container const form = addExpenseScreen.addControl .form({ instanceId: formInstanceId }); { // If set to true, will warn user on back navigation form.discardChangesAlert(false); // Form fields - Title form.addControl // Use instanceId for state tracking later, eg `=@ctx.components.title.state.value` .textField({ instanceId: 'title' }) .label('Title') .required(true) .isAutoFocused() .isAutoCorrected() .autoCapitalize('sentences'); // Amount form.addControl .numberField({ instanceId: 'amount' }) .label('Amount') .required(true) .format({ numberStyle: 'currency' }); // Format as currency // Category const category = form.addControl .dropdown({ isRequired: true, instanceId: 'categoryId' }) .label('Category') // Dropdown bound to static datasource .data(DATASOURCE.EXPENSE_CATEGORIES); { // Template for dropdown members (property names from datasource) category.item() .value('=@ctx.current.item.id') // Stored key .title('=@ctx.current.item.name'); // Display text } // Description form.addControl .textField({ instanceId: 'description' }) .label('Description') .required(false) .isMultiline(true) .autoCapitalize('sentences') .isOptionalLabelHidden(); // Hide "optional" label // Date form.addControl.datePicker({ isRequired: true, mode: 'date', instanceId: 'date' }) .label('Date') .initialValue('=$now()'); // Jsonata expression } // Save to database const addExpenseButtons = addExpenseScreen.bottomPanel.buttons(); addExpenseButtons.add.executeEntity('Save Expense') .dynamicData(ENTITY.EXPENSES, 'create') .data({ // Component state mapped via instanceId title: '=@ctx.components.title.state.value', amount: '=@ctx.components.amount.state.value', categoryId: '=@ctx.components.categoryId.state.value', date: '=@ctx.components.date.state.value', description: '=@ctx.components.description.state.value', timestamp: '=$now()' }) .goBack('previous'); // Navigate back after save (see discardChangesAlert above) } // ======================================== // EDIT EXPENSE SCREEN // ======================================== /* Edit expense form */ const editExpenseScreen = app.addScreen .default(SCREEN.EDIT_EXPENSE, 'Edit Expense'); { // Formal parameters editExpenseScreen .input(INP.string('id').required()) .input(INP.object('expenseItem').required()); const formInstanceId = 'edit-expense-form'; // Edit form - pre-populated const form = editExpenseScreen.addControl.form({ instanceId: formInstanceId }); { form.discardChangesAlert(false); // Form fields - Title form.addControl .textField({ instanceId: 'title' }) .label('Title') .required(true) .isAutoFocused() .isAutoCorrected() .autoCapitalize('sentences') // Pre-populate from inputs. By convention, instanceId matches data-field/state .initialValue(`=@ctx.inputs.expenseItem.title`); // Amount form.addControl .numberField({ instanceId: 'amount' }) .label('Amount') .required(true) .format({ numberStyle: 'currency' }) .initialValue(`=@ctx.inputs.expenseItem.amount`); // Category const category = form.addControl .dropdown({ isRequired: true, instanceId: 'categoryId' }) .label('Category') .data(DATASOURCE.EXPENSE_CATEGORIES) .initialValue(`=@ctx.inputs.expenseItem.categoryId`); { category.item() .value('=@ctx.current.item.id') .title('=@ctx.current.item.name'); } // Description form.addControl .textField({ instanceId: 'description' }) .label('Description') .required(false) .isMultiline(true) .autoCapitalize('sentences') .initialValue('=@ctx.inputs.expenseItem.description'); // Date form.addControl .datePicker({ isRequired: true, mode: 'date', instanceId: 'date' }) .label('Date') .initialValue(`=@ctx.inputs.expenseItem.date`); } // Buttons/actions const editButtons = editExpenseScreen.bottomPanel.buttons(2); // Button/action to UPDATE selected expense editButtons.add .executeEntity('Update') .dynamicData(ENTITY.EXPENSES, 'update') // 'update' = modify existing .data({ id: `=@ctx.inputs.id`, // Locator // Include all fields for update title: '=@ctx.components.title.state.value', amount: '=@ctx.components.amount.state.value', categoryId: '=@ctx.components.categoryId.state.value', date: '=@ctx.components.date.state.value', description: '=@ctx.components.description.state.value', timestamp: '=$now()' }) .goBack('previous'); // Navigate back after update (see discardChangesAlert above) // Add button to DELETE with confirmation modal const deleteButton = editButtons.add.confirm('Delete'); deleteButton.modal({ text: 'Delete Expense?' }) // User must confirm .confirmText('Delete') .cancelText('Cancel'); deleteButton.onConfirmed.executeEntity() .dynamicData(ENTITY.EXPENSES, 'delete') // 'delete' = remove record .data({ id: `=@ctx.inputs.id` // Locator }) .goBack('previous'); } // ======================================== // NAVIGATION - Tab bar configuration // ======================================== // Bottom navigation - entry point app.addTab(TABS.HOME, 'credit-card') // Icon name .label('Expenses'); // Tab text // ======================================== // BUILD - Generate output // ======================================== // CRITICAL: MUST call build() app.build(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"app.js","sourceRoot":"","sources":["../../template-app-1/app.ts"],"names":[],"mappings":"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","sourcesContent":["import { application, INP } from '../src'\n\n/**\n * Template expense tracking app demonstrating Jigx mobile patterns.\n * ADAPT THIS: Replace 'expense' with your domain (tasks, inventory, etc.)\n * Add/edit/remove entities, datasources, screens, tabs, etc\n * Explore Jigx SDK types and docs for more info\n */\n\n// ========================================\n// CONSTANTS - IDs and references\n// ========================================\n\n// Screen Ids\nconst SCREEN = {\n  HOME: 'home',\n  EXPENSE_LIST: 'expense-list',\n  ADD_EXPENSE: 'add-expense',\n  EDIT_EXPENSE: 'edit-expense'\n} as const\n\n// Datasource Ids\nconst DATASOURCE = {\n  EXPENSES: 'data-expenses',\n  EXPENSE_CATEGORIES: 'expense-categories',\n  EXPENSE_STATS: 'expense-stats',\n  RECENT_EXPENSES: 'recent-expenses',\n  EXPENSES_SUMMARY: 'expenses-summary',\n  CURRENT_EXPENSE: 'current-expense'\n} as const\n\n// Database entities\nconst ENTITY = {\n  EXPENSES: 'default/expenses'\n} as const\n\n// Tab Ids\nconst TABS = {\n  HOME: 'home'\n} as const\n\n// ========================================\n// APPLICATION - Core app configuration\n// ========================================\n\nexport const app = application('expense-tracker') // Unique programmatic Id\n  .title('Expense Tracker') // Display name\n  .category('personal') // App store category (choices in doc-comments)\n\n// Global expressions - available everywhere\napp.expression('formatCurrency', `=function($amount) {\n  '$' & $formatNumber($amount, '#,##0.00')\n}`)\napp.expression('formatDate', `=function($date) {\n  $fromMillis($toMillis($date), '[D1o] [MNn] [Y]')\n}`)\n\n// ========================================\n// DATABASE - SQLite table definitions\n// ========================================\n\n/* Database tables - JSON docs in sqlite */\napp.addDatabase\n  .default // Only ONE db, always named 'default'\n  .table('expenses') // Declares [default/expenses] table in DB \n\n// ========================================\n// SHARED DATASOURCES - Available to all screens\n// ========================================\n\n/* Main data provider - queries all expenses */\napp.addDatasource.sqlite(DATASOURCE.EXPENSES, 'dynamic') // 'dynamic' = offline sync\n  .entity(ENTITY.EXPENSES) // Table(s) to query\n  .query(/* sql */`\n      SELECT \n        exp.id, -- Always include row id\n        -- Option 1: Return entire JSON doc (preferred)\n        exp.data\n        -- Option 2: Extract specific fields for expressions\n        -- json_extract(exp.data, '$.title') AS title\n      FROM [${ENTITY.EXPENSES}] AS exp\n      ORDER BY\n        -- Always extract for joins, sorting, etc\n        json_extract(exp.data, '$.date') DESC\n    `)\n    // Decorate data as 'this is a json document'\n    .jsonProperties('data')\n\n/* Static data - no database needed, ideal for fixed lists */\napp.addDatasource.static(DATASOURCE.EXPENSE_CATEGORIES, [\n  { id: 'food', name: 'Food & Dining' },\n  { id: 'transport', name: 'Transportation' },\n  { id: 'accommodation', name: 'Accommodation' },\n  { id: 'office', name: 'Office Supplies' },\n  { id: 'technology', name: 'Technology' },\n  { id: 'entertainment', name: 'Entertainment' },\n  { id: 'health', name: 'Health & Medical' },\n  { id: 'education', name: 'Education & Training' },\n  { id: 'other', name: 'Other' }\n])\n\n// ========================================\n// HOME SCREEN - Main dashboard\n// ========================================\n\nconst homeScreen = app.addScreen\n  .default(SCREEN.HOME, 'Expense Tracker')\n{\n  // Screen-specific datasource for aggregate stats\n  homeScreen.addDatasource.sqlite(DATASOURCE.EXPENSE_STATS, 'dynamic')\n    .entity(ENTITY.EXPENSES)\n    .query(/* sql */`\n      SELECT \n        COUNT(1) AS count,\n        SUM(json_extract(exp.data, '$.amount')) AS total_amount,\n        AVG(json_extract(exp.data, '$.amount')) AS avg_amount,\n        MAX(json_extract(exp.data, '$.amount')) AS max_amount\n      FROM [${ENTITY.EXPENSES}] AS exp\n    `)\n    //.isDocument() // Return single document instead of array\n\n  // Card component - container for related fields\n  const card = homeScreen.addControl.card()\n  {\n    card.addControl.list()\n      .data(DATASOURCE.EXPENSE_STATS)\n      .addControl.listItem()\n      // Call global function to format currency\n      .title('=$formatCurrency(@ctx.current.item.total_amount) & \" (\" & @ctx.current.item.count & \" items)\"')\n  }\n\n  // Recent items (LOCAL DATASOURCE)\n  homeScreen.addDatasource.sqlite(DATASOURCE.RECENT_EXPENSES, 'dynamic')\n    .entity(ENTITY.EXPENSES)\n    .query(/* sql */`\n      SELECT \n        exp.id,\n        exp.data\n      FROM [${ENTITY.EXPENSES}] AS exp\n      ORDER BY\n        json_extract(exp.data, '$.date') DESC\n      LIMIT 5\n    `)\n    .jsonProperties('data')\n\n  // List component - displays collection\n  const list = homeScreen.addControl\n    .list('recent-list')\n    .data(DATASOURCE.RECENT_EXPENSES) // Bind to datasource\n\n  list.addControl.listItem()\n    .title('=@ctx.current.item.data.title')\n    .subtitle('=@ctx.current.item.data.categoryId & \" • \" & $formatDate(@ctx.current.item.data.date)')\n    .description('=$formatCurrency(@ctx.current.item.data.amount)')\n    .onPress\n      .goto(SCREEN.EDIT_EXPENSE) // Navigate to screen\n      .parameter('id', '=@ctx.current.item.id')\n      .parameter('expenseItem', '=@ctx.current.item.data')\n\n  // Buttons at bottom of screen\n  const homeButtons = homeScreen.bottomPanel.buttons(2) // Show 2 of 3 max\n  homeButtons.add.goto('Add Expense', SCREEN.ADD_EXPENSE)\n  homeButtons.add.goto('View All', SCREEN.EXPENSE_LIST) // Show all expenses\n}\n\n// ========================================\n// EXPENSE LIST SCREEN\n// ========================================\n\n/* Expense list - all transactions */\nconst expenseListScreen = app.addScreen\n  .default(SCREEN.EXPENSE_LIST, 'All Expenses')\n{\n  // LOCAL datasource (only available in this screen)\n  expenseListScreen.addDatasource.sqlite(DATASOURCE.EXPENSES_SUMMARY, 'dynamic')\n    .entity(ENTITY.EXPENSES)\n    .query(/* sql */`\n      SELECT \n        json_extract(exp.data, '$.categoryId') AS categoryId,\n        SUM(json_extract(exp.data, '$.amount')) AS total_amount,\n        COUNT(1) AS count\n      FROM [${ENTITY.EXPENSES}] AS exp\n      GROUP BY categoryId\n      ORDER BY total_amount DESC\n    `)\n    .isDocument(false) // In this case we want the array, since it is multiple categories\n\n  // Summary card\n  const summaryCard = expenseListScreen.addControl.card()\n  {\n    summaryCard.addControl.textField()\n      .label('Summary by Category')\n      .value('Category Breakdown')\n      .isDisabled()\n\n    summaryCard.addControl.list()\n      .data(DATASOURCE.EXPENSES_SUMMARY)\n      .addControl.listItem()\n      .title('=@ctx.current.item.categoryId')\n      .subtitle('=$formatCurrency(@ctx.current.item.total_amount) & \" • \" & @ctx.current.item.count & \" expenses\"')\n  }\n\n  // Expense list\n  const expenseList = expenseListScreen.addControl.list('expense-list')\n    .data(DATASOURCE.EXPENSES) // Bind to global datasource\n\n  expenseList.addControl.listItem()\n    .title('=@ctx.current.item.data.title')\n    .subtitle('=@ctx.current.item.data.categoryId & \" • \" & @ctx.current.item.data.date')\n    .description('=$formatCurrency(@ctx.current.item.data.amount)')\n    .onPress\n      .goto(SCREEN.EDIT_EXPENSE)\n      // Actual parameters\n      .parameter('id', '=@ctx.current.item.id')\n      .parameter('expenseItem', '=@ctx.current.item.data')\n\n  // Actions\n  const expenseListButtons = expenseListScreen.bottomPanel.buttons()\n  expenseListButtons.add.goto('Add New', SCREEN.ADD_EXPENSE)\n}\n\n// ========================================\n// ADD EXPENSE SCREEN\n// ========================================\n\n/* Add expense form */\nconst addExpenseScreen = app.addScreen\n  .default(SCREEN.ADD_EXPENSE, 'Add Expense')\n{\n  // Clear previous form state on screen focus (else will pre-populate using old values)\n  const formInstanceId = 'add-expense-form'\n  addExpenseScreen.onFocus\n    .clearState(`=@ctx.components.${formInstanceId}.state.value`)\n\n  // Form container\n  const form = addExpenseScreen.addControl\n    .form({ instanceId: formInstanceId })\n  {\n\n    // If set to true, will warn user on back navigation\n    form.discardChangesAlert(false)\n\n    // Form fields - Title\n    form.addControl\n      // Use instanceId for state tracking later, eg `=@ctx.components.title.state.value`\n      .textField({ instanceId: 'title' })\n      .label('Title')\n      .required(true)\n      .isAutoFocused()\n      .isAutoCorrected()\n      .autoCapitalize('sentences')\n\n    // Amount\n    form.addControl\n      .numberField({ instanceId: 'amount' })\n      .label('Amount')\n      .required(true)\n      .format({ numberStyle: 'currency' }) // Format as currency\n\n    // Category\n    const category = form.addControl\n      .dropdown({ isRequired: true, instanceId: 'categoryId' })\n      .label('Category')\n      // Dropdown bound to static datasource\n      .data(DATASOURCE.EXPENSE_CATEGORIES)\n    {\n      // Template for dropdown members (property names from datasource)\n      category.item()\n        .value('=@ctx.current.item.id') // Stored key\n        .title('=@ctx.current.item.name') // Display text\n    }\n\n    // Description\n    form.addControl\n      .textField({ instanceId: 'description' })\n      .label('Description')\n      .required(false)\n      .isMultiline(true)\n      .autoCapitalize('sentences')\n      .isOptionalLabelHidden() // Hide \"optional\" label\n\n    // Date\n    form.addControl.datePicker({ isRequired: true, mode: 'date', instanceId: 'date' })\n      .label('Date')\n      .initialValue('=$now()') // Jsonata expression\n  }\n\n  // Save to database\n  const addExpenseButtons = addExpenseScreen.bottomPanel.buttons()\n  addExpenseButtons.add.executeEntity('Save Expense')\n    .dynamicData(ENTITY.EXPENSES, 'create')\n    .data({\n      // Component state mapped via instanceId\n      title: '=@ctx.components.title.state.value',\n      amount: '=@ctx.components.amount.state.value',\n      categoryId: '=@ctx.components.categoryId.state.value',\n      date: '=@ctx.components.date.state.value',\n      description: '=@ctx.components.description.state.value',\n      timestamp: '=$now()'\n    })\n    .goBack('previous') // Navigate back after save (see discardChangesAlert above)\n}\n\n// ========================================\n// EDIT EXPENSE SCREEN  \n// ========================================\n\n/* Edit expense form */\nconst editExpenseScreen = app.addScreen\n  .default(SCREEN.EDIT_EXPENSE, 'Edit Expense')\n{\n  // Formal parameters\n  editExpenseScreen\n    .input(INP.string('id').required())\n    .input(INP.object('expenseItem').required())\n\n  const formInstanceId = 'edit-expense-form'\n\n  // Edit form - pre-populated\n  const form = editExpenseScreen.addControl.form({ instanceId: formInstanceId })\n  {\n    form.discardChangesAlert(false)\n\n    // Form fields - Title\n    form.addControl\n      .textField({ instanceId: 'title' })\n      .label('Title')\n      .required(true)\n      .isAutoFocused()\n      .isAutoCorrected()\n      .autoCapitalize('sentences')\n      // Pre-populate from inputs. By convention, instanceId matches data-field/state\n      .initialValue(`=@ctx.inputs.expenseItem.title`)\n\n    // Amount\n    form.addControl\n      .numberField({ instanceId: 'amount' })\n      .label('Amount')\n      .required(true)\n      .format({ numberStyle: 'currency' })\n      .initialValue(`=@ctx.inputs.expenseItem.amount`)\n\n    // Category\n    const category = form.addControl\n      .dropdown({ isRequired: true, instanceId: 'categoryId' })\n      .label('Category')\n      .data(DATASOURCE.EXPENSE_CATEGORIES)\n      .initialValue(`=@ctx.inputs.expenseItem.categoryId`)\n    {\n      category.item()\n        .value('=@ctx.current.item.id')\n        .title('=@ctx.current.item.name')\n    }\n\n    // Description\n    form.addControl\n      .textField({ instanceId: 'description' })\n      .label('Description')\n      .required(false)\n      .isMultiline(true)\n      .autoCapitalize('sentences')\n      .initialValue('=@ctx.inputs.expenseItem.description')\n\n    // Date\n    form.addControl\n      .datePicker({ isRequired: true, mode: 'date', instanceId: 'date' })\n      .label('Date')\n      .initialValue(`=@ctx.inputs.expenseItem.date`)\n  }\n\n  // Buttons/actions\n  const editButtons = editExpenseScreen.bottomPanel.buttons(2)\n\n  // Button/action to UPDATE selected expense\n  editButtons.add\n    .executeEntity('Update')\n    .dynamicData(ENTITY.EXPENSES, 'update') // 'update' = modify existing\n    .data({\n      id: `=@ctx.inputs.id`, // Locator\n      // Include all fields for update\n      title: '=@ctx.components.title.state.value',\n      amount: '=@ctx.components.amount.state.value',\n      categoryId: '=@ctx.components.categoryId.state.value',\n      date: '=@ctx.components.date.state.value',\n      description: '=@ctx.components.description.state.value',\n      timestamp: '=$now()'\n    })\n    .goBack('previous') // Navigate back after update (see discardChangesAlert above)\n\n  // Add button to DELETE with confirmation modal\n  const deleteButton = editButtons.add.confirm('Delete')\n  deleteButton.modal({ text: 'Delete Expense?' }) // User must confirm\n    .confirmText('Delete')\n    .cancelText('Cancel')\n  deleteButton.onConfirmed.executeEntity()\n    .dynamicData(ENTITY.EXPENSES, 'delete') // 'delete' = remove record\n    .data({\n      id: `=@ctx.inputs.id` // Locator\n    })\n    .goBack('previous')\n}\n\n// ========================================\n// NAVIGATION - Tab bar configuration\n// ========================================\n\n// Bottom navigation - entry point\napp.addTab(TABS.HOME, 'credit-card') // Icon name\n  .label('Expenses') // Tab text\n\n// ========================================\n// BUILD - Generate output\n// ========================================\n\n// CRITICAL: MUST call build()\napp.build()\n"]}