#
tokens: 48080/50000 1/1626 files (page 133/141)
lines: off (toggle) GitHub
raw markdown copy
This is page 133 of 141. Use http://codebase.md/xmlui-org/xmlui/tools/vscode/resources/main.tsx?lines=false&page={x} to view the full context.

# Directory Structure

```
├── .changeset
│   └── config.json
├── .eslintrc.cjs
├── .github
│   ├── build-checklist.png
│   ├── ISSUE_TEMPLATE
│   │   ├── bug_report.md
│   │   └── feature_request.md
│   └── workflows
│       ├── deploy-blog.yml
│       ├── deploy-docs-optimized.yml
│       ├── deploy-docs.yml
│       ├── prepare-versions.yml
│       ├── release-packages.yml
│       ├── run-all-tests.yml
│       └── run-smoke-tests.yml
├── .gitignore
├── .prettierrc.js
├── .vscode
│   ├── launch.json
│   └── settings.json
├── blog
│   ├── .gitignore
│   ├── .gitkeep
│   ├── CHANGELOG.md
│   ├── extensions.ts
│   ├── index.html
│   ├── index.ts
│   ├── layout-changes.md
│   ├── package.json
│   ├── public
│   │   ├── blog
│   │   │   ├── images
│   │   │   │   ├── blog-page-component.png
│   │   │   │   ├── blog-scrabble.png
│   │   │   │   ├── integrated-blog-search.png
│   │   │   │   └── lorem-ipsum.png
│   │   │   ├── lorem-ipsum.md
│   │   │   ├── newest-post.md
│   │   │   ├── older-post.md
│   │   │   └── welcome-to-the-xmlui-blog.md
│   │   ├── mockServiceWorker.js
│   │   ├── resources
│   │   │   ├── favicon.ico
│   │   │   ├── files
│   │   │   │   └── for-download
│   │   │   │       └── xmlui
│   │   │   │           └── xmlui-standalone.umd.js
│   │   │   ├── github.svg
│   │   │   ├── llms.txt
│   │   │   ├── logo-dark.svg
│   │   │   ├── logo.svg
│   │   │   ├── pg-popout.svg
│   │   │   ├── rss.svg
│   │   │   └── xmlui-logo.svg
│   │   ├── serve.json
│   │   └── web.config
│   ├── scripts
│   │   ├── download-latest-xmlui.js
│   │   ├── generate-rss.js
│   │   ├── get-releases.js
│   │   └── utils.js
│   ├── src
│   │   ├── components
│   │   │   ├── BlogOverview.xmlui
│   │   │   ├── BlogPage.xmlui
│   │   │   └── PageNotFound.xmlui
│   │   ├── config.ts
│   │   ├── Main.xmlui
│   │   └── themes
│   │       └── blog-theme.ts
│   └── tsconfig.json
├── CONTRIBUTING.md
├── docs
│   ├── .gitignore
│   ├── CHANGELOG.md
│   ├── ComponentRefLinks.txt
│   ├── content
│   │   ├── _meta.json
│   │   ├── components
│   │   │   ├── _meta.json
│   │   │   ├── _overview.md
│   │   │   ├── APICall.md
│   │   │   ├── App.md
│   │   │   ├── AppHeader.md
│   │   │   ├── AppState.md
│   │   │   ├── AutoComplete.md
│   │   │   ├── Avatar.md
│   │   │   ├── Backdrop.md
│   │   │   ├── Badge.md
│   │   │   ├── BarChart.md
│   │   │   ├── Bookmark.md
│   │   │   ├── Breakout.md
│   │   │   ├── Button.md
│   │   │   ├── Card.md
│   │   │   ├── Carousel.md
│   │   │   ├── ChangeListener.md
│   │   │   ├── Checkbox.md
│   │   │   ├── CHStack.md
│   │   │   ├── ColorPicker.md
│   │   │   ├── Column.md
│   │   │   ├── ContentSeparator.md
│   │   │   ├── CVStack.md
│   │   │   ├── DataSource.md
│   │   │   ├── DateInput.md
│   │   │   ├── DatePicker.md
│   │   │   ├── DonutChart.md
│   │   │   ├── DropdownMenu.md
│   │   │   ├── EmojiSelector.md
│   │   │   ├── ExpandableItem.md
│   │   │   ├── FileInput.md
│   │   │   ├── FileUploadDropZone.md
│   │   │   ├── FlowLayout.md
│   │   │   ├── Footer.md
│   │   │   ├── Form.md
│   │   │   ├── FormItem.md
│   │   │   ├── FormSection.md
│   │   │   ├── Fragment.md
│   │   │   ├── H1.md
│   │   │   ├── H2.md
│   │   │   ├── H3.md
│   │   │   ├── H4.md
│   │   │   ├── H5.md
│   │   │   ├── H6.md
│   │   │   ├── Heading.md
│   │   │   ├── HSplitter.md
│   │   │   ├── HStack.md
│   │   │   ├── Icon.md
│   │   │   ├── IFrame.md
│   │   │   ├── Image.md
│   │   │   ├── Items.md
│   │   │   ├── LabelList.md
│   │   │   ├── Legend.md
│   │   │   ├── LineChart.md
│   │   │   ├── Link.md
│   │   │   ├── List.md
│   │   │   ├── Logo.md
│   │   │   ├── Markdown.md
│   │   │   ├── MenuItem.md
│   │   │   ├── MenuSeparator.md
│   │   │   ├── ModalDialog.md
│   │   │   ├── NavGroup.md
│   │   │   ├── NavLink.md
│   │   │   ├── NavPanel.md
│   │   │   ├── NoResult.md
│   │   │   ├── NumberBox.md
│   │   │   ├── Option.md
│   │   │   ├── Page.md
│   │   │   ├── PageMetaTitle.md
│   │   │   ├── Pages.md
│   │   │   ├── Pagination.md
│   │   │   ├── PasswordInput.md
│   │   │   ├── PieChart.md
│   │   │   ├── ProgressBar.md
│   │   │   ├── Queue.md
│   │   │   ├── RadioGroup.md
│   │   │   ├── RealTimeAdapter.md
│   │   │   ├── Redirect.md
│   │   │   ├── Select.md
│   │   │   ├── Slider.md
│   │   │   ├── Slot.md
│   │   │   ├── SpaceFiller.md
│   │   │   ├── Spinner.md
│   │   │   ├── Splitter.md
│   │   │   ├── Stack.md
│   │   │   ├── StickyBox.md
│   │   │   ├── SubMenuItem.md
│   │   │   ├── Switch.md
│   │   │   ├── TabItem.md
│   │   │   ├── Table.md
│   │   │   ├── TableOfContents.md
│   │   │   ├── Tabs.md
│   │   │   ├── Text.md
│   │   │   ├── TextArea.md
│   │   │   ├── TextBox.md
│   │   │   ├── Theme.md
│   │   │   ├── TimeInput.md
│   │   │   ├── Timer.md
│   │   │   ├── ToneChangerButton.md
│   │   │   ├── ToneSwitch.md
│   │   │   ├── Tooltip.md
│   │   │   ├── Tree.md
│   │   │   ├── VSplitter.md
│   │   │   ├── VStack.md
│   │   │   ├── xmlui-animations
│   │   │   │   ├── _meta.json
│   │   │   │   ├── _overview.md
│   │   │   │   ├── Animation.md
│   │   │   │   ├── FadeAnimation.md
│   │   │   │   ├── FadeInAnimation.md
│   │   │   │   ├── FadeOutAnimation.md
│   │   │   │   ├── ScaleAnimation.md
│   │   │   │   └── SlideInAnimation.md
│   │   │   ├── xmlui-pdf
│   │   │   │   ├── _meta.json
│   │   │   │   ├── _overview.md
│   │   │   │   └── Pdf.md
│   │   │   ├── xmlui-spreadsheet
│   │   │   │   ├── _meta.json
│   │   │   │   ├── _overview.md
│   │   │   │   └── Spreadsheet.md
│   │   │   └── xmlui-website-blocks
│   │   │       ├── _meta.json
│   │   │       ├── _overview.md
│   │   │       ├── Carousel.md
│   │   │       ├── HelloMd.md
│   │   │       ├── HeroSection.md
│   │   │       └── ScrollToTop.md
│   │   └── extensions
│   │       ├── _meta.json
│   │       ├── xmlui-animations
│   │       │   ├── _meta.json
│   │       │   ├── _overview.md
│   │       │   ├── Animation.md
│   │       │   ├── FadeAnimation.md
│   │       │   ├── FadeInAnimation.md
│   │       │   ├── FadeOutAnimation.md
│   │       │   ├── ScaleAnimation.md
│   │       │   └── SlideInAnimation.md
│   │       └── xmlui-website-blocks
│   │           ├── _meta.json
│   │           ├── _overview.md
│   │           ├── Carousel.md
│   │           ├── HelloMd.md
│   │           ├── HeroSection.md
│   │           └── ScrollToTop.md
│   ├── extensions.ts
│   ├── index.html
│   ├── index.ts
│   ├── package.json
│   ├── public
│   │   ├── feed.rss
│   │   ├── mockServiceWorker.js
│   │   ├── pages
│   │   │   ├── _meta.json
│   │   │   ├── app-structure.md
│   │   │   ├── build-editor-component.md
│   │   │   ├── build-hello-world-component.md
│   │   │   ├── components-intro.md
│   │   │   ├── context-variables.md
│   │   │   ├── forms.md
│   │   │   ├── globals.md
│   │   │   ├── glossary.md
│   │   │   ├── helper-tags.md
│   │   │   ├── hosted-deployment.md
│   │   │   ├── howto
│   │   │   │   ├── assign-a-complex-json-literal-to-a-component-variable.md
│   │   │   │   ├── chain-a-refetch.md
│   │   │   │   ├── debug-a-component.md
│   │   │   │   ├── delay-a-datasource-until-another-datasource-is-ready.md
│   │   │   │   ├── delegate-a-method.md
│   │   │   │   ├── do-custom-form-validation.md
│   │   │   │   ├── expose-a-method-from-a-component.md
│   │   │   │   ├── filter-and-transform-data-from-an-api.md
│   │   │   │   ├── group-items-in-list-by-a-property.md
│   │   │   │   ├── handle-background-operations.md
│   │   │   │   ├── hide-an-element-until-its-datasource-is-ready.md
│   │   │   │   ├── make-a-set-of-equal-width-cards.md
│   │   │   │   ├── make-a-table-responsive.md
│   │   │   │   ├── make-navpanel-width-responsive.md
│   │   │   │   ├── modify-a-value-reported-in-a-column.md
│   │   │   │   ├── paginate-a-list.md
│   │   │   │   ├── pass-data-to-a-modal-dialog.md
│   │   │   │   ├── react-to-button-click-not-keystrokes.md
│   │   │   │   ├── set-the-initial-value-of-a-select-from-fetched-data.md
│   │   │   │   ├── share-a-modaldialog-across-components.md
│   │   │   │   ├── sync-selections-between-table-and-list-views.md
│   │   │   │   ├── update-ui-optimistically.md
│   │   │   │   ├── use-built-in-form-validation.md
│   │   │   │   └── use-the-same-modaldialog-to-add-or-edit.md
│   │   │   ├── howto.md
│   │   │   ├── intro.md
│   │   │   ├── layout.md
│   │   │   ├── markup.md
│   │   │   ├── mcp.md
│   │   │   ├── modal-dialogs.md
│   │   │   ├── news-and-reviews.md
│   │   │   ├── reactive-intro.md
│   │   │   ├── refactoring.md
│   │   │   ├── routing-and-links.md
│   │   │   ├── samples
│   │   │   │   ├── color-palette.xmlui
│   │   │   │   ├── color-values.xmlui
│   │   │   │   ├── shadow-sizes.xmlui
│   │   │   │   ├── spacing-sizes.xmlui
│   │   │   │   ├── swatch.xmlui
│   │   │   │   ├── theme-gallery-brief.xmlui
│   │   │   │   └── theme-gallery.xmlui
│   │   │   ├── scoping.md
│   │   │   ├── scripting.md
│   │   │   ├── styles-and-themes
│   │   │   │   ├── common-units.md
│   │   │   │   ├── layout-props.md
│   │   │   │   ├── theme-variable-defaults.md
│   │   │   │   ├── theme-variables.md
│   │   │   │   └── themes.md
│   │   │   ├── template-properties.md
│   │   │   ├── test.md
│   │   │   ├── tutorial-01.md
│   │   │   ├── tutorial-02.md
│   │   │   ├── tutorial-03.md
│   │   │   ├── tutorial-04.md
│   │   │   ├── tutorial-05.md
│   │   │   ├── tutorial-06.md
│   │   │   ├── tutorial-07.md
│   │   │   ├── tutorial-08.md
│   │   │   ├── tutorial-09.md
│   │   │   ├── tutorial-10.md
│   │   │   ├── tutorial-11.md
│   │   │   ├── tutorial-12.md
│   │   │   ├── universal-properties.md
│   │   │   ├── user-defined-components.md
│   │   │   ├── vscode.md
│   │   │   ├── working-with-markdown.md
│   │   │   ├── working-with-text.md
│   │   │   ├── xmlui-animations
│   │   │   │   ├── _meta.json
│   │   │   │   ├── _overview.md
│   │   │   │   ├── Animation.md
│   │   │   │   ├── FadeAnimation.md
│   │   │   │   ├── FadeInAnimation.md
│   │   │   │   ├── FadeOutAnimation.md
│   │   │   │   ├── ScaleAnimation.md
│   │   │   │   └── SlideInAnimation.md
│   │   │   ├── xmlui-charts
│   │   │   │   ├── _meta.json
│   │   │   │   ├── _overview.md
│   │   │   │   ├── BarChart.md
│   │   │   │   ├── DonutChart.md
│   │   │   │   ├── LabelList.md
│   │   │   │   ├── Legend.md
│   │   │   │   ├── LineChart.md
│   │   │   │   └── PieChart.md
│   │   │   ├── xmlui-pdf
│   │   │   │   ├── _meta.json
│   │   │   │   ├── _overview.md
│   │   │   │   └── Pdf.md
│   │   │   └── xmlui-spreadsheet
│   │   │       ├── _meta.json
│   │   │       ├── _overview.md
│   │   │       └── Spreadsheet.md
│   │   ├── resources
│   │   │   ├── devdocs
│   │   │   │   ├── debug-proxy-object-2.png
│   │   │   │   ├── debug-proxy-object.png
│   │   │   │   ├── table_editor_01.png
│   │   │   │   ├── table_editor_02.png
│   │   │   │   ├── table_editor_03.png
│   │   │   │   ├── table_editor_04.png
│   │   │   │   ├── table_editor_05.png
│   │   │   │   ├── table_editor_06.png
│   │   │   │   ├── table_editor_07.png
│   │   │   │   ├── table_editor_08.png
│   │   │   │   ├── table_editor_09.png
│   │   │   │   ├── table_editor_10.png
│   │   │   │   ├── table_editor_11.png
│   │   │   │   ├── table-editor-01.png
│   │   │   │   ├── table-editor-02.png
│   │   │   │   ├── table-editor-03.png
│   │   │   │   ├── table-editor-04.png
│   │   │   │   ├── table-editor-06.png
│   │   │   │   ├── table-editor-07.png
│   │   │   │   ├── table-editor-08.png
│   │   │   │   ├── table-editor-09.png
│   │   │   │   └── xmlui-rendering-of-tiptap-markdown.png
│   │   │   ├── favicon.ico
│   │   │   ├── files
│   │   │   │   ├── clients.json
│   │   │   │   ├── daily-revenue.json
│   │   │   │   ├── dashboard-stats.json
│   │   │   │   ├── demo.xmlui
│   │   │   │   ├── demo.xmlui.xs
│   │   │   │   ├── downloads
│   │   │   │   │   └── downloads.json
│   │   │   │   ├── for-download
│   │   │   │   │   ├── index-with-api.html
│   │   │   │   │   ├── index.html
│   │   │   │   │   ├── mockApi.js
│   │   │   │   │   ├── start-darwin.sh
│   │   │   │   │   ├── start-linux.sh
│   │   │   │   │   ├── start.bat
│   │   │   │   │   └── xmlui
│   │   │   │   │       └── xmlui-standalone.umd.js
│   │   │   │   ├── getting-started
│   │   │   │   │   ├── cl-tutorial-final.zip
│   │   │   │   │   ├── cl-tutorial.zip
│   │   │   │   │   ├── cl-tutorial2.zip
│   │   │   │   │   ├── cl-tutorial3.zip
│   │   │   │   │   ├── cl-tutorial4.zip
│   │   │   │   │   ├── cl-tutorial5.zip
│   │   │   │   │   ├── cl-tutorial6.zip
│   │   │   │   │   ├── getting-started.zip
│   │   │   │   │   ├── hello-xmlui.zip
│   │   │   │   │   ├── xmlui-empty.zip
│   │   │   │   │   └── xmlui-starter.zip
│   │   │   │   ├── howto
│   │   │   │   │   └── component-icons
│   │   │   │   │       └── up-arrow.svg
│   │   │   │   ├── invoices.json
│   │   │   │   ├── monthly-status.json
│   │   │   │   ├── news-and-reviews.json
│   │   │   │   ├── products.json
│   │   │   │   ├── releases.json
│   │   │   │   ├── tutorials
│   │   │   │   │   ├── datasource
│   │   │   │   │   │   └── api.ts
│   │   │   │   │   └── p2do
│   │   │   │   │       ├── api.ts
│   │   │   │   │       └── todo-logo.svg
│   │   │   │   └── xmlui.json
│   │   │   ├── github.svg
│   │   │   ├── images
│   │   │   │   ├── apiaction-tutorial
│   │   │   │   │   ├── add-success.png
│   │   │   │   │   ├── apiaction-param.png
│   │   │   │   │   ├── change-completed.png
│   │   │   │   │   ├── change-in-progress.png
│   │   │   │   │   ├── confirm-delete.png
│   │   │   │   │   ├── data-error.png
│   │   │   │   │   ├── data-progress.png
│   │   │   │   │   ├── data-success.png
│   │   │   │   │   ├── display-1.png
│   │   │   │   │   ├── item-deleted.png
│   │   │   │   │   ├── item-updated.png
│   │   │   │   │   ├── missing-api-key.png
│   │   │   │   │   ├── new-item-added.png
│   │   │   │   │   └── test-message.png
│   │   │   │   ├── chat-api
│   │   │   │   │   └── domain-model.svg
│   │   │   │   ├── components
│   │   │   │   │   ├── image
│   │   │   │   │   │   └── breakfast.jpg
│   │   │   │   │   ├── markdown
│   │   │   │   │   │   └── colors.png
│   │   │   │   │   └── modal
│   │   │   │   │       ├── deep_link_dialog_1.jpg
│   │   │   │   │       └── deep_link_dialog_2.jpg
│   │   │   │   ├── create-apps
│   │   │   │   │   ├── collapsed-vertical.png
│   │   │   │   │   ├── using-forms-warning-dialog.png
│   │   │   │   │   └── using-forms.png
│   │   │   │   ├── datasource-tutorial
│   │   │   │   │   ├── data-with-header.png
│   │   │   │   │   ├── filtered-data.png
│   │   │   │   │   ├── filtered-items.png
│   │   │   │   │   ├── initial-page-items.png
│   │   │   │   │   ├── list-items.png
│   │   │   │   │   ├── next-page-items.png
│   │   │   │   │   ├── no-data.png
│   │   │   │   │   ├── pagination-1.jpg
│   │   │   │   │   ├── pagination-1.png
│   │   │   │   │   ├── polling-1.png
│   │   │   │   │   ├── refetch-data.png
│   │   │   │   │   ├── slow-loading.png
│   │   │   │   │   ├── test-message.png
│   │   │   │   │   ├── Thumbs.db
│   │   │   │   │   ├── unconventional-data.png
│   │   │   │   │   └── unfiltered-items.png
│   │   │   │   ├── flower.jpg
│   │   │   │   ├── get-started
│   │   │   │   │   ├── add-new-contact.png
│   │   │   │   │   ├── app-modified.png
│   │   │   │   │   ├── app-start.png
│   │   │   │   │   ├── app-with-boxes.png
│   │   │   │   │   ├── app-with-toast.png
│   │   │   │   │   ├── boilerplate-structure.png
│   │   │   │   │   ├── cl-initial.png
│   │   │   │   │   ├── cl-start.png
│   │   │   │   │   ├── contact-counts.png
│   │   │   │   │   ├── contact-dialog-title.png
│   │   │   │   │   ├── contact-dialog.png
│   │   │   │   │   ├── contact-menus.png
│   │   │   │   │   ├── contact-predicates.png
│   │   │   │   │   ├── context-menu.png
│   │   │   │   │   ├── dashboard-numbers.png
│   │   │   │   │   ├── default-contact-list.png
│   │   │   │   │   ├── delete-contact.png
│   │   │   │   │   ├── delete-task.png
│   │   │   │   │   ├── detailed-template.png
│   │   │   │   │   ├── edit-contact-details.png
│   │   │   │   │   ├── edited-contact-saved.png
│   │   │   │   │   ├── empty-sections.png
│   │   │   │   │   ├── filter-completed.png
│   │   │   │   │   ├── fullwidth-desktop.png
│   │   │   │   │   ├── fullwidth-mobile.png
│   │   │   │   │   ├── initial-table.png
│   │   │   │   │   ├── items-and-badges.png
│   │   │   │   │   ├── loading-message.png
│   │   │   │   │   ├── new-contact-button.png
│   │   │   │   │   ├── new-contact-saved.png
│   │   │   │   │   ├── no-empty-sections.png
│   │   │   │   │   ├── personal-todo-initial.png
│   │   │   │   │   ├── piechart.png
│   │   │   │   │   ├── review-today.png
│   │   │   │   │   ├── rudimentary-dashboard.png
│   │   │   │   │   ├── section-collapsed.png
│   │   │   │   │   ├── sectioned-items.png
│   │   │   │   │   ├── sections-ordered.png
│   │   │   │   │   ├── spacex-list-with-links.png
│   │   │   │   │   ├── spacex-list.png
│   │   │   │   │   ├── start-personal-todo-1.png
│   │   │   │   │   ├── submit-new-contact.png
│   │   │   │   │   ├── submit-new-task.png
│   │   │   │   │   ├── syntax-highlighting.png
│   │   │   │   │   ├── table-with-badge.png
│   │   │   │   │   ├── template-with-card.png
│   │   │   │   │   ├── test-emulated-api.png
│   │   │   │   │   ├── Thumbs.db
│   │   │   │   │   ├── todo-logo.png
│   │   │   │   │   └── xmlui-tools.png
│   │   │   │   ├── HelloApp.png
│   │   │   │   ├── HelloApp2.png
│   │   │   │   ├── logos
│   │   │   │   │   ├── xmlui1.svg
│   │   │   │   │   ├── xmlui2.svg
│   │   │   │   │   ├── xmlui3.svg
│   │   │   │   │   ├── xmlui4.svg
│   │   │   │   │   ├── xmlui5.svg
│   │   │   │   │   ├── xmlui6.svg
│   │   │   │   │   └── xmlui7.svg
│   │   │   │   ├── pdf
│   │   │   │   │   └── dummy-pdf.jpg
│   │   │   │   ├── rendering-engine
│   │   │   │   │   ├── AppEngine-flow.svg
│   │   │   │   │   ├── Component.svg
│   │   │   │   │   ├── CompoundComponent.svg
│   │   │   │   │   ├── RootComponent.svg
│   │   │   │   │   └── tree-with-containers.svg
│   │   │   │   ├── reviewers-guide
│   │   │   │   │   ├── AppEngine-flow.svg
│   │   │   │   │   └── incbutton-in-action.png
│   │   │   │   ├── tools
│   │   │   │   │   └── boilerplate-structure.png
│   │   │   │   ├── try.svg
│   │   │   │   ├── tutorial
│   │   │   │   │   ├── app-chat-history.png
│   │   │   │   │   ├── app-content-placeholder.png
│   │   │   │   │   ├── app-header-and-content.png
│   │   │   │   │   ├── app-links-channel-selected.png
│   │   │   │   │   ├── app-links-click.png
│   │   │   │   │   ├── app-navigation.png
│   │   │   │   │   ├── finished-ex01.png
│   │   │   │   │   ├── finished-ex02.png
│   │   │   │   │   ├── hello.png
│   │   │   │   │   ├── splash-screen-advanced.png
│   │   │   │   │   ├── splash-screen-after-click.png
│   │   │   │   │   ├── splash-screen-centered.png
│   │   │   │   │   ├── splash-screen-events.png
│   │   │   │   │   ├── splash-screen-expression.png
│   │   │   │   │   ├── splash-screen-reuse-after.png
│   │   │   │   │   ├── splash-screen-reuse-before.png
│   │   │   │   │   └── splash-screen.png
│   │   │   │   └── tutorial-01.png
│   │   │   ├── llms.txt
│   │   │   ├── logo-dark.svg
│   │   │   ├── logo.svg
│   │   │   ├── pg-popout.svg
│   │   │   └── xmlui-logo.svg
│   │   ├── serve.json
│   │   └── web.config
│   ├── scripts
│   │   ├── download-latest-xmlui.js
│   │   ├── generate-rss.js
│   │   ├── get-releases.js
│   │   └── utils.js
│   ├── src
│   │   ├── components
│   │   │   ├── BlogOverview.xmlui
│   │   │   ├── BlogPage.xmlui
│   │   │   ├── Boxes.xmlui
│   │   │   ├── Breadcrumb.xmlui
│   │   │   ├── ChangeLog.xmlui
│   │   │   ├── ColorPalette.xmlui
│   │   │   ├── DocumentLinks.xmlui
│   │   │   ├── DocumentPage.xmlui
│   │   │   ├── DocumentPageNoTOC.xmlui
│   │   │   ├── Icons.xmlui
│   │   │   ├── IncButton.xmlui
│   │   │   ├── IncButton2.xmlui
│   │   │   ├── NameValue.xmlui
│   │   │   ├── PageNotFound.xmlui
│   │   │   ├── PaletteItem.xmlui
│   │   │   ├── Palettes.xmlui
│   │   │   ├── SectionHeader.xmlui
│   │   │   ├── TBD.xmlui
│   │   │   ├── Test.xmlui
│   │   │   ├── ThemesIntro.xmlui
│   │   │   ├── ThousandThemes.xmlui
│   │   │   ├── TubeStops.xmlui
│   │   │   ├── TubeStops.xmlui.xs
│   │   │   └── TwoColumnCode.xmlui
│   │   ├── config.ts
│   │   ├── Main.xmlui
│   │   └── themes
│   │       ├── docs-theme.ts
│   │       ├── earthtone.ts
│   │       ├── xmlui-gray-on-default.ts
│   │       ├── xmlui-green-on-default.ts
│   │       └── xmlui-orange-on-default.ts
│   └── tsconfig.json
├── LICENSE
├── package-lock.json
├── package.json
├── packages
│   ├── xmlui-animations
│   │   ├── .gitignore
│   │   ├── CHANGELOG.md
│   │   ├── demo
│   │   │   └── Main.xmlui
│   │   ├── index.html
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── Animation.tsx
│   │   │   ├── AnimationNative.tsx
│   │   │   ├── FadeAnimation.tsx
│   │   │   ├── FadeInAnimation.tsx
│   │   │   ├── FadeOutAnimation.tsx
│   │   │   ├── index.tsx
│   │   │   ├── ScaleAnimation.tsx
│   │   │   └── SlideInAnimation.tsx
│   │   └── tsconfig.json
│   ├── xmlui-devtools
│   │   ├── .gitignore
│   │   ├── CHANGELOG.md
│   │   ├── demo
│   │   │   └── Main.xmlui
│   │   ├── index.html
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── devtools
│   │   │   │   ├── DevTools.tsx
│   │   │   │   ├── DevToolsNative.module.scss
│   │   │   │   ├── DevToolsNative.tsx
│   │   │   │   ├── ModalDialog.module.scss
│   │   │   │   ├── ModalDialog.tsx
│   │   │   │   ├── ModalVisibilityContext.tsx
│   │   │   │   ├── Tooltip.module.scss
│   │   │   │   ├── Tooltip.tsx
│   │   │   │   └── utils.ts
│   │   │   ├── editor
│   │   │   │   └── Editor.tsx
│   │   │   └── index.tsx
│   │   ├── tsconfig.json
│   │   └── vite.config-overrides.ts
│   ├── xmlui-hello-world
│   │   ├── .gitignore
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── HelloWorld.module.scss
│   │   │   ├── HelloWorld.tsx
│   │   │   ├── HelloWorldNative.tsx
│   │   │   └── index.tsx
│   │   └── tsconfig.json
│   ├── xmlui-os-frames
│   │   ├── .gitignore
│   │   ├── demo
│   │   │   └── Main.xmlui
│   │   ├── index.html
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── index.tsx
│   │   │   ├── IPhoneFrame.module.scss
│   │   │   ├── IPhoneFrame.tsx
│   │   │   ├── MacOSAppFrame.module.scss
│   │   │   ├── MacOSAppFrame.tsx
│   │   │   ├── WindowsAppFrame.module.scss
│   │   │   └── WindowsAppFrame.tsx
│   │   └── tsconfig.json
│   ├── xmlui-pdf
│   │   ├── .gitignore
│   │   ├── CHANGELOG.md
│   │   ├── demo
│   │   │   ├── components
│   │   │   │   └── Pdf.xmlui
│   │   │   └── Main.xmlui
│   │   ├── index.html
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── index.tsx
│   │   │   ├── LazyPdfNative.tsx
│   │   │   ├── Pdf.module.scss
│   │   │   └── Pdf.tsx
│   │   └── tsconfig.json
│   ├── xmlui-playground
│   │   ├── .gitignore
│   │   ├── CHANGELOG.md
│   │   ├── demo
│   │   │   └── Main.xmlui
│   │   ├── index.html
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── hooks
│   │   │   │   ├── usePlayground.ts
│   │   │   │   └── useToast.ts
│   │   │   ├── index.tsx
│   │   │   ├── playground
│   │   │   │   ├── Box.module.scss
│   │   │   │   ├── Box.tsx
│   │   │   │   ├── CodeSelector.tsx
│   │   │   │   ├── ConfirmationDialog.module.scss
│   │   │   │   ├── ConfirmationDialog.tsx
│   │   │   │   ├── Editor.tsx
│   │   │   │   ├── Header.module.scss
│   │   │   │   ├── Header.tsx
│   │   │   │   ├── Playground.tsx
│   │   │   │   ├── PlaygroundContent.module.scss
│   │   │   │   ├── PlaygroundContent.tsx
│   │   │   │   ├── PlaygroundNative.module.scss
│   │   │   │   ├── PlaygroundNative.tsx
│   │   │   │   ├── Preview.module.scss
│   │   │   │   ├── Preview.tsx
│   │   │   │   ├── Select.module.scss
│   │   │   │   ├── StandalonePlayground.tsx
│   │   │   │   ├── StandalonePlaygroundNative.module.scss
│   │   │   │   ├── StandalonePlaygroundNative.tsx
│   │   │   │   ├── ThemeSwitcher.module.scss
│   │   │   │   ├── ThemeSwitcher.tsx
│   │   │   │   ├── ToneSwitcher.tsx
│   │   │   │   ├── Tooltip.module.scss
│   │   │   │   ├── Tooltip.tsx
│   │   │   │   └── utils.ts
│   │   │   ├── providers
│   │   │   │   ├── Toast.module.scss
│   │   │   │   └── ToastProvider.tsx
│   │   │   ├── state
│   │   │   │   └── store.ts
│   │   │   ├── themes
│   │   │   │   └── theme.ts
│   │   │   └── utils
│   │   │       └── helpers.ts
│   │   └── tsconfig.json
│   ├── xmlui-search
│   │   ├── .gitignore
│   │   ├── CHANGELOG.md
│   │   ├── demo
│   │   │   └── Main.xmlui
│   │   ├── index.html
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── index.tsx
│   │   │   ├── Search.module.scss
│   │   │   └── Search.tsx
│   │   └── tsconfig.json
│   ├── xmlui-spreadsheet
│   │   ├── .gitignore
│   │   ├── demo
│   │   │   └── Main.xmlui
│   │   ├── index.html
│   │   ├── index.ts
│   │   ├── meta
│   │   │   └── componentsMetadata.ts
│   │   ├── package.json
│   │   ├── src
│   │   │   ├── index.tsx
│   │   │   ├── Spreadsheet.tsx
│   │   │   └── SpreadsheetNative.tsx
│   │   └── tsconfig.json
│   └── xmlui-website-blocks
│       ├── .gitignore
│       ├── CHANGELOG.md
│       ├── demo
│       │   ├── components
│       │   │   ├── HeroBackgroundBreakoutPage.xmlui
│       │   │   ├── HeroBackgroundsPage.xmlui
│       │   │   ├── HeroContentsPage.xmlui
│       │   │   ├── HeroTextAlignPage.xmlui
│       │   │   ├── HeroTextPage.xmlui
│       │   │   └── HeroTonesPage.xmlui
│       │   ├── Main.xmlui
│       │   └── themes
│       │       └── default.ts
│       ├── index.html
│       ├── index.ts
│       ├── meta
│       │   └── componentsMetadata.ts
│       ├── package.json
│       ├── public
│       │   └── resources
│       │       ├── building.jpg
│       │       └── xmlui-logo.svg
│       ├── src
│       │   ├── Carousel
│       │   │   ├── Carousel.module.scss
│       │   │   ├── Carousel.tsx
│       │   │   ├── CarouselContext.tsx
│       │   │   └── CarouselNative.tsx
│       │   ├── FancyButton
│       │   │   ├── FancyButton.module.scss
│       │   │   ├── FancyButton.tsx
│       │   │   └── FancyButton.xmlui
│       │   ├── Hello
│       │   │   ├── Hello.tsx
│       │   │   ├── Hello.xmlui
│       │   │   └── Hello.xmlui.xs
│       │   ├── HeroSection
│       │   │   ├── HeroSection.module.scss
│       │   │   ├── HeroSection.tsx
│       │   │   └── HeroSectionNative.tsx
│       │   ├── index.tsx
│       │   ├── ScrollToTop
│       │   │   ├── ScrollToTop.module.scss
│       │   │   ├── ScrollToTop.tsx
│       │   │   └── ScrollToTopNative.tsx
│       │   └── vite-env.d.ts
│       └── tsconfig.json
├── README.md
├── tools
│   ├── codefence
│   │   └── xmlui-code-fence-docs.md
│   ├── create-app
│   │   ├── .gitignore
│   │   ├── CHANGELOG.md
│   │   ├── create-app.ts
│   │   ├── helpers
│   │   │   ├── copy.ts
│   │   │   ├── get-pkg-manager.ts
│   │   │   ├── git.ts
│   │   │   ├── install.ts
│   │   │   ├── is-folder-empty.ts
│   │   │   ├── is-writeable.ts
│   │   │   ├── make-dir.ts
│   │   │   └── validate-pkg.ts
│   │   ├── index.ts
│   │   ├── package.json
│   │   ├── templates
│   │   │   ├── default
│   │   │   │   └── ts
│   │   │   │       ├── gitignore
│   │   │   │       ├── index.html
│   │   │   │       ├── index.ts
│   │   │   │       ├── public
│   │   │   │       │   ├── mockServiceWorker.js
│   │   │   │       │   ├── resources
│   │   │   │       │   │   ├── favicon.ico
│   │   │   │       │   │   └── xmlui-logo.svg
│   │   │   │       │   └── serve.json
│   │   │   │       └── src
│   │   │   │           ├── components
│   │   │   │           │   ├── ApiAware.xmlui
│   │   │   │           │   ├── Home.xmlui
│   │   │   │           │   ├── IncButton.xmlui
│   │   │   │           │   └── PagePanel.xmlui
│   │   │   │           ├── config.ts
│   │   │   │           └── Main.xmlui
│   │   │   ├── index.ts
│   │   │   └── types.ts
│   │   └── tsconfig.json
│   ├── create-xmlui-hello-world
│   │   ├── index.js
│   │   └── package.json
│   └── vscode
│       ├── .gitignore
│       ├── .vscode
│       │   ├── launch.json
│       │   └── tasks.json
│       ├── .vscodeignore
│       ├── build.sh
│       ├── CHANGELOG.md
│       ├── esbuild.js
│       ├── eslint.config.mjs
│       ├── formatter-docs.md
│       ├── generate-test-sample.sh
│       ├── LICENSE.md
│       ├── package-lock.json
│       ├── package.json
│       ├── README.md
│       ├── resources
│       │   ├── xmlui-logo.png
│       │   └── xmlui-markup-syntax-highlighting.png
│       ├── src
│       │   ├── extension.ts
│       │   └── server.ts
│       ├── syntaxes
│       │   └── xmlui.tmLanguage.json
│       ├── test-samples
│       │   └── sample.xmlui
│       ├── tsconfig.json
│       └── tsconfig.tsbuildinfo
├── turbo.json
└── xmlui
    ├── .gitignore
    ├── bin
    │   ├── bootstrap.js
    │   ├── build-lib.ts
    │   ├── build.ts
    │   ├── index.ts
    │   ├── preview.ts
    │   ├── start.ts
    │   ├── vite-xmlui-plugin.ts
    │   └── viteConfig.ts
    ├── CHANGELOG.md
    ├── conventions
    │   ├── component-qa-checklist.md
    │   ├── copilot-conventions.md
    │   ├── create-xmlui-components.md
    │   ├── mermaid.md
    │   ├── testing-conventions.md
    │   └── xmlui-in-a-nutshell.md
    ├── dev-docs
    │   ├── accessibility.md
    │   ├── build-system.md
    │   ├── build-xmlui.md
    │   ├── component-behaviors.md
    │   ├── components-with-options.md
    │   ├── containers.md
    │   ├── data-operations.md
    │   ├── glossary.md
    │   ├── index.md
    │   ├── next
    │   │   ├── component-dev-guide.md
    │   │   ├── configuration-management-enhancement-summary.md
    │   │   ├── documentation-scripts-refactoring-complete-summary.md
    │   │   ├── documentation-scripts-refactoring-plan.md
    │   │   ├── duplicate-pattern-extraction-summary.md
    │   │   ├── error-handling-standardization-summary.md
    │   │   ├── generating-component-reference.md
    │   │   ├── index.md
    │   │   ├── logging-consistency-implementation-summary.md
    │   │   ├── project-build.md
    │   │   ├── project-structure.md
    │   │   ├── theme-context.md
    │   │   ├── tiptap-design-considerations.md
    │   │   ├── working-with-code.md
    │   │   ├── xmlui-runtime-architecture
    │   │   └── xmlui-wcag-accessibility-report.md
    │   ├── react-fundamentals.md
    │   ├── release-method.md
    │   ├── standalone-app.md
    │   ├── ud-components.md
    │   └── xmlui-repo.md
    ├── package.json
    ├── playwright.config.ts
    ├── scripts
    │   ├── coverage-only.js
    │   ├── e2e-test-summary.js
    │   ├── generate-docs
    │   │   ├── build-downloads-map.mjs
    │   │   ├── build-pages-map.mjs
    │   │   ├── components-config.json
    │   │   ├── configuration-management.mjs
    │   │   ├── constants.mjs
    │   │   ├── create-theme-files.mjs
    │   │   ├── DocsGenerator.mjs
    │   │   ├── error-handling.mjs
    │   │   ├── extensions-config.json
    │   │   ├── folders.mjs
    │   │   ├── generate-summary-files.mjs
    │   │   ├── get-docs.mjs
    │   │   ├── input-handler.mjs
    │   │   ├── logger.mjs
    │   │   ├── logging-standards.mjs
    │   │   ├── MetadataProcessor.mjs
    │   │   ├── pattern-utilities.mjs
    │   │   └── utils.mjs
    │   ├── get-langserver-metadata.mjs
    │   ├── inline-links.mjs
    │   └── README-e2e-summary.md
    ├── src
    │   ├── abstractions
    │   │   ├── _conventions.md
    │   │   ├── ActionDefs.ts
    │   │   ├── AppContextDefs.ts
    │   │   ├── ComponentDefs.ts
    │   │   ├── ContainerDefs.ts
    │   │   ├── ExtensionDefs.ts
    │   │   ├── FunctionDefs.ts
    │   │   ├── RendererDefs.ts
    │   │   ├── scripting
    │   │   │   ├── BlockScope.ts
    │   │   │   ├── Compilation.ts
    │   │   │   ├── LogicalThread.ts
    │   │   │   ├── LoopScope.ts
    │   │   │   ├── modules.ts
    │   │   │   ├── ScriptParserError.ts
    │   │   │   ├── Token.ts
    │   │   │   ├── TryScope.ts
    │   │   │   └── TryScopeExp.ts
    │   │   └── ThemingDefs.ts
    │   ├── components
    │   │   ├── _conventions.md
    │   │   ├── abstractions.ts
    │   │   ├── Accordion
    │   │   │   ├── Accordion.md
    │   │   │   ├── Accordion.module.scss
    │   │   │   ├── Accordion.spec.ts
    │   │   │   ├── Accordion.tsx
    │   │   │   ├── AccordionContext.tsx
    │   │   │   ├── AccordionItem.tsx
    │   │   │   ├── AccordionItemNative.tsx
    │   │   │   └── AccordionNative.tsx
    │   │   ├── Animation
    │   │   │   └── AnimationNative.tsx
    │   │   ├── APICall
    │   │   │   ├── APICall.md
    │   │   │   ├── APICall.spec.ts
    │   │   │   ├── APICall.tsx
    │   │   │   └── APICallNative.tsx
    │   │   ├── App
    │   │   │   ├── App.md
    │   │   │   ├── App.module.scss
    │   │   │   ├── App.spec.ts
    │   │   │   ├── App.tsx
    │   │   │   ├── AppLayoutContext.ts
    │   │   │   ├── AppNative.tsx
    │   │   │   ├── AppStateContext.ts
    │   │   │   ├── doc-resources
    │   │   │   │   ├── condensed-sticky.xmlui
    │   │   │   │   ├── condensed.xmlui
    │   │   │   │   ├── horizontal-sticky.xmlui
    │   │   │   │   ├── horizontal.xmlui
    │   │   │   │   ├── vertical-full-header.xmlui
    │   │   │   │   ├── vertical-sticky.xmlui
    │   │   │   │   └── vertical.xmlui
    │   │   │   ├── IndexerContext.ts
    │   │   │   ├── LinkInfoContext.ts
    │   │   │   ├── SearchContext.tsx
    │   │   │   ├── Sheet.module.scss
    │   │   │   └── Sheet.tsx
    │   │   ├── AppHeader
    │   │   │   ├── AppHeader.md
    │   │   │   ├── AppHeader.module.scss
    │   │   │   ├── AppHeader.spec.ts
    │   │   │   ├── AppHeader.tsx
    │   │   │   └── AppHeaderNative.tsx
    │   │   ├── AppState
    │   │   │   ├── AppState.md
    │   │   │   ├── AppState.spec.ts
    │   │   │   ├── AppState.tsx
    │   │   │   └── AppStateNative.tsx
    │   │   ├── AutoComplete
    │   │   │   ├── AutoComplete.md
    │   │   │   ├── AutoComplete.module.scss
    │   │   │   ├── AutoComplete.spec.ts
    │   │   │   ├── AutoComplete.tsx
    │   │   │   ├── AutoCompleteContext.tsx
    │   │   │   └── AutoCompleteNative.tsx
    │   │   ├── Avatar
    │   │   │   ├── Avatar.md
    │   │   │   ├── Avatar.module.scss
    │   │   │   ├── Avatar.spec.ts
    │   │   │   ├── Avatar.tsx
    │   │   │   └── AvatarNative.tsx
    │   │   ├── Backdrop
    │   │   │   ├── Backdrop.md
    │   │   │   ├── Backdrop.module.scss
    │   │   │   ├── Backdrop.spec.ts
    │   │   │   ├── Backdrop.tsx
    │   │   │   └── BackdropNative.tsx
    │   │   ├── Badge
    │   │   │   ├── Badge.md
    │   │   │   ├── Badge.module.scss
    │   │   │   ├── Badge.spec.ts
    │   │   │   ├── Badge.tsx
    │   │   │   └── BadgeNative.tsx
    │   │   ├── Bookmark
    │   │   │   ├── Bookmark.md
    │   │   │   ├── Bookmark.module.scss
    │   │   │   ├── Bookmark.spec.ts
    │   │   │   ├── Bookmark.tsx
    │   │   │   └── BookmarkNative.tsx
    │   │   ├── Breakout
    │   │   │   ├── Breakout.module.scss
    │   │   │   ├── Breakout.spec.ts
    │   │   │   ├── Breakout.tsx
    │   │   │   └── BreakoutNative.tsx
    │   │   ├── Button
    │   │   │   ├── Button-style.spec.ts
    │   │   │   ├── Button.md
    │   │   │   ├── Button.module.scss
    │   │   │   ├── Button.spec.ts
    │   │   │   ├── Button.tsx
    │   │   │   └── ButtonNative.tsx
    │   │   ├── Card
    │   │   │   ├── Card.md
    │   │   │   ├── Card.module.scss
    │   │   │   ├── Card.spec.ts
    │   │   │   ├── Card.tsx
    │   │   │   └── CardNative.tsx
    │   │   ├── Carousel
    │   │   │   ├── Carousel.md
    │   │   │   ├── Carousel.module.scss
    │   │   │   ├── Carousel.spec.ts
    │   │   │   ├── Carousel.tsx
    │   │   │   ├── CarouselContext.tsx
    │   │   │   ├── CarouselItem.tsx
    │   │   │   ├── CarouselItemNative.tsx
    │   │   │   └── CarouselNative.tsx
    │   │   ├── ChangeListener
    │   │   │   ├── ChangeListener.md
    │   │   │   ├── ChangeListener.spec.ts
    │   │   │   ├── ChangeListener.tsx
    │   │   │   └── ChangeListenerNative.tsx
    │   │   ├── chart-color-schemes.ts
    │   │   ├── Charts
    │   │   │   ├── AreaChart
    │   │   │   │   ├── AreaChart.md
    │   │   │   │   ├── AreaChart.spec.ts
    │   │   │   │   ├── AreaChart.tsx
    │   │   │   │   └── AreaChartNative.tsx
    │   │   │   ├── BarChart
    │   │   │   │   ├── BarChart.md
    │   │   │   │   ├── BarChart.module.scss
    │   │   │   │   ├── BarChart.spec.ts
    │   │   │   │   ├── BarChart.tsx
    │   │   │   │   └── BarChartNative.tsx
    │   │   │   ├── DonutChart
    │   │   │   │   ├── DonutChart.spec.ts
    │   │   │   │   └── DonutChart.tsx
    │   │   │   ├── LabelList
    │   │   │   │   ├── LabelList.spec.ts
    │   │   │   │   ├── LabelList.tsx
    │   │   │   │   ├── LabelListNative.module.scss
    │   │   │   │   └── LabelListNative.tsx
    │   │   │   ├── Legend
    │   │   │   │   ├── Legend.spec.ts
    │   │   │   │   ├── Legend.tsx
    │   │   │   │   └── LegendNative.tsx
    │   │   │   ├── LineChart
    │   │   │   │   ├── LineChart.md
    │   │   │   │   ├── LineChart.module.scss
    │   │   │   │   ├── LineChart.spec.ts
    │   │   │   │   ├── LineChart.tsx
    │   │   │   │   └── LineChartNative.tsx
    │   │   │   ├── PieChart
    │   │   │   │   ├── PieChart.md
    │   │   │   │   ├── PieChart.spec.ts
    │   │   │   │   ├── PieChart.tsx
    │   │   │   │   ├── PieChartNative.module.scss
    │   │   │   │   └── PieChartNative.tsx
    │   │   │   ├── RadarChart
    │   │   │   │   ├── RadarChart.md
    │   │   │   │   ├── RadarChart.spec.ts
    │   │   │   │   ├── RadarChart.tsx
    │   │   │   │   └── RadarChartNative.tsx
    │   │   │   ├── Tooltip
    │   │   │   │   ├── TooltipContent.module.scss
    │   │   │   │   ├── TooltipContent.spec.ts
    │   │   │   │   └── TooltipContent.tsx
    │   │   │   └── utils
    │   │   │       ├── abstractions.ts
    │   │   │       └── ChartProvider.tsx
    │   │   ├── Checkbox
    │   │   │   ├── Checkbox.md
    │   │   │   ├── Checkbox.spec.ts
    │   │   │   └── Checkbox.tsx
    │   │   ├── CodeBlock
    │   │   │   ├── CodeBlock.module.scss
    │   │   │   ├── CodeBlock.spec.ts
    │   │   │   ├── CodeBlock.tsx
    │   │   │   ├── CodeBlockNative.tsx
    │   │   │   └── highlight-code.ts
    │   │   ├── collectedComponentMetadata.ts
    │   │   ├── ColorPicker
    │   │   │   ├── ColorPicker.md
    │   │   │   ├── ColorPicker.module.scss
    │   │   │   ├── ColorPicker.spec.ts
    │   │   │   ├── ColorPicker.tsx
    │   │   │   └── ColorPickerNative.tsx
    │   │   ├── Column
    │   │   │   ├── Column.md
    │   │   │   ├── Column.tsx
    │   │   │   ├── ColumnNative.tsx
    │   │   │   ├── doc-resources
    │   │   │   │   └── list-component-data.js
    │   │   │   └── TableContext.tsx
    │   │   ├── component-utils.ts
    │   │   ├── ComponentProvider.tsx
    │   │   ├── ComponentRegistryContext.tsx
    │   │   ├── container-helpers.tsx
    │   │   ├── ContentSeparator
    │   │   │   ├── ContentSeparator.md
    │   │   │   ├── ContentSeparator.module.scss
    │   │   │   ├── ContentSeparator.spec.ts
    │   │   │   ├── ContentSeparator.tsx
    │   │   │   └── ContentSeparatorNative.tsx
    │   │   ├── DataSource
    │   │   │   ├── DataSource.md
    │   │   │   └── DataSource.tsx
    │   │   ├── DateInput
    │   │   │   ├── DateInput.md
    │   │   │   ├── DateInput.module.scss
    │   │   │   ├── DateInput.spec.ts
    │   │   │   ├── DateInput.tsx
    │   │   │   └── DateInputNative.tsx
    │   │   ├── DatePicker
    │   │   │   ├── DatePicker.md
    │   │   │   ├── DatePicker.module.scss
    │   │   │   ├── DatePicker.spec.ts
    │   │   │   ├── DatePicker.tsx
    │   │   │   └── DatePickerNative.tsx
    │   │   ├── DropdownMenu
    │   │   │   ├── DropdownMenu.md
    │   │   │   ├── DropdownMenu.module.scss
    │   │   │   ├── DropdownMenu.spec.ts
    │   │   │   ├── DropdownMenu.tsx
    │   │   │   ├── DropdownMenuNative.tsx
    │   │   │   ├── MenuItem.md
    │   │   │   └── SubMenuItem.md
    │   │   ├── EmojiSelector
    │   │   │   ├── EmojiSelector.md
    │   │   │   ├── EmojiSelector.spec.ts
    │   │   │   ├── EmojiSelector.tsx
    │   │   │   └── EmojiSelectorNative.tsx
    │   │   ├── ExpandableItem
    │   │   │   ├── ExpandableItem.module.scss
    │   │   │   ├── ExpandableItem.spec.ts
    │   │   │   ├── ExpandableItem.tsx
    │   │   │   └── ExpandableItemNative.tsx
    │   │   ├── FileInput
    │   │   │   ├── FileInput.md
    │   │   │   ├── FileInput.module.scss
    │   │   │   ├── FileInput.spec.ts
    │   │   │   ├── FileInput.tsx
    │   │   │   └── FileInputNative.tsx
    │   │   ├── FileUploadDropZone
    │   │   │   ├── FileUploadDropZone.md
    │   │   │   ├── FileUploadDropZone.module.scss
    │   │   │   ├── FileUploadDropZone.spec.ts
    │   │   │   ├── FileUploadDropZone.tsx
    │   │   │   └── FileUploadDropZoneNative.tsx
    │   │   ├── FlowLayout
    │   │   │   ├── FlowLayout.md
    │   │   │   ├── FlowLayout.module.scss
    │   │   │   ├── FlowLayout.spec.ts
    │   │   │   ├── FlowLayout.spec.ts-snapshots
    │   │   │   │   └── Edge-cases-boxShadow-is-not-clipped-1-non-smoke-darwin.png
    │   │   │   ├── FlowLayout.tsx
    │   │   │   └── FlowLayoutNative.tsx
    │   │   ├── Footer
    │   │   │   ├── Footer.md
    │   │   │   ├── Footer.module.scss
    │   │   │   ├── Footer.spec.ts
    │   │   │   ├── Footer.tsx
    │   │   │   └── FooterNative.tsx
    │   │   ├── Form
    │   │   │   ├── Form.md
    │   │   │   ├── Form.module.scss
    │   │   │   ├── Form.spec.ts
    │   │   │   ├── Form.tsx
    │   │   │   ├── formActions.ts
    │   │   │   ├── FormContext.ts
    │   │   │   └── FormNative.tsx
    │   │   ├── FormItem
    │   │   │   ├── FormItem.md
    │   │   │   ├── FormItem.module.scss
    │   │   │   ├── FormItem.spec.ts
    │   │   │   ├── FormItem.tsx
    │   │   │   ├── FormItemNative.tsx
    │   │   │   ├── HelperText.module.scss
    │   │   │   ├── HelperText.tsx
    │   │   │   ├── ItemWithLabel.tsx
    │   │   │   └── Validations.ts
    │   │   ├── FormSection
    │   │   │   ├── FormSection.md
    │   │   │   ├── FormSection.ts
    │   │   │   └── FormSection.xmlui
    │   │   ├── Fragment
    │   │   │   ├── Fragment.spec.ts
    │   │   │   └── Fragment.tsx
    │   │   ├── Heading
    │   │   │   ├── abstractions.ts
    │   │   │   ├── H1.md
    │   │   │   ├── H1.spec.ts
    │   │   │   ├── H2.md
    │   │   │   ├── H2.spec.ts
    │   │   │   ├── H3.md
    │   │   │   ├── H3.spec.ts
    │   │   │   ├── H4.md
    │   │   │   ├── H4.spec.ts
    │   │   │   ├── H5.md
    │   │   │   ├── H5.spec.ts
    │   │   │   ├── H6.md
    │   │   │   ├── H6.spec.ts
    │   │   │   ├── Heading.md
    │   │   │   ├── Heading.module.scss
    │   │   │   ├── Heading.spec.ts
    │   │   │   ├── Heading.tsx
    │   │   │   └── HeadingNative.tsx
    │   │   ├── HoverCard
    │   │   │   ├── HoverCard.tsx
    │   │   │   └── HovercardNative.tsx
    │   │   ├── HtmlTags
    │   │   │   ├── HtmlTags.module.scss
    │   │   │   ├── HtmlTags.spec.ts
    │   │   │   └── HtmlTags.tsx
    │   │   ├── Icon
    │   │   │   ├── AdmonitionDanger.tsx
    │   │   │   ├── AdmonitionInfo.tsx
    │   │   │   ├── AdmonitionNote.tsx
    │   │   │   ├── AdmonitionTip.tsx
    │   │   │   ├── AdmonitionWarning.tsx
    │   │   │   ├── ApiIcon.tsx
    │   │   │   ├── ArrowDropDown.module.scss
    │   │   │   ├── ArrowDropDown.tsx
    │   │   │   ├── ArrowDropUp.module.scss
    │   │   │   ├── ArrowDropUp.tsx
    │   │   │   ├── ArrowLeft.module.scss
    │   │   │   ├── ArrowLeft.tsx
    │   │   │   ├── ArrowRight.module.scss
    │   │   │   ├── ArrowRight.tsx
    │   │   │   ├── Attach.tsx
    │   │   │   ├── Binding.module.scss
    │   │   │   ├── Binding.tsx
    │   │   │   ├── BoardIcon.tsx
    │   │   │   ├── BoxIcon.tsx
    │   │   │   ├── CheckIcon.tsx
    │   │   │   ├── ChevronDownIcon.tsx
    │   │   │   ├── ChevronLeft.tsx
    │   │   │   ├── ChevronRight.tsx
    │   │   │   ├── ChevronUpIcon.tsx
    │   │   │   ├── CodeFileIcon.tsx
    │   │   │   ├── CodeSandbox.tsx
    │   │   │   ├── CompactListIcon.tsx
    │   │   │   ├── ContentCopyIcon.tsx
    │   │   │   ├── DarkToLightIcon.tsx
    │   │   │   ├── DatabaseIcon.module.scss
    │   │   │   ├── DatabaseIcon.tsx
    │   │   │   ├── DocFileIcon.tsx
    │   │   │   ├── DocIcon.tsx
    │   │   │   ├── DotMenuHorizontalIcon.tsx
    │   │   │   ├── DotMenuIcon.tsx
    │   │   │   ├── EmailIcon.tsx
    │   │   │   ├── EmptyFolderIcon.tsx
    │   │   │   ├── ErrorIcon.tsx
    │   │   │   ├── ExpressionIcon.tsx
    │   │   │   ├── FillPlusCricleIcon.tsx
    │   │   │   ├── FilterIcon.tsx
    │   │   │   ├── FolderIcon.tsx
    │   │   │   ├── GlobeIcon.tsx
    │   │   │   ├── HomeIcon.tsx
    │   │   │   ├── HyperLinkIcon.tsx
    │   │   │   ├── Icon.md
    │   │   │   ├── Icon.module.scss
    │   │   │   ├── Icon.spec.ts
    │   │   │   ├── Icon.tsx
    │   │   │   ├── IconNative.tsx
    │   │   │   ├── ImageFileIcon.tsx
    │   │   │   ├── Inspect.tsx
    │   │   │   ├── LightToDark.tsx
    │   │   │   ├── LinkIcon.tsx
    │   │   │   ├── ListIcon.tsx
    │   │   │   ├── LooseListIcon.tsx
    │   │   │   ├── MoonIcon.tsx
    │   │   │   ├── MoreOptionsIcon.tsx
    │   │   │   ├── NoSortIcon.tsx
    │   │   │   ├── PDFIcon.tsx
    │   │   │   ├── PenIcon.tsx
    │   │   │   ├── PhoneIcon.tsx
    │   │   │   ├── PhotoIcon.tsx
    │   │   │   ├── PlusIcon.tsx
    │   │   │   ├── SearchIcon.tsx
    │   │   │   ├── ShareIcon.tsx
    │   │   │   ├── SortAscendingIcon.tsx
    │   │   │   ├── SortDescendingIcon.tsx
    │   │   │   ├── StarsIcon.tsx
    │   │   │   ├── SunIcon.tsx
    │   │   │   ├── svg
    │   │   │   │   ├── admonition_danger.svg
    │   │   │   │   ├── admonition_info.svg
    │   │   │   │   ├── admonition_note.svg
    │   │   │   │   ├── admonition_tip.svg
    │   │   │   │   ├── admonition_warning.svg
    │   │   │   │   ├── api.svg
    │   │   │   │   ├── arrow-dropdown.svg
    │   │   │   │   ├── arrow-left.svg
    │   │   │   │   ├── arrow-right.svg
    │   │   │   │   ├── arrow-up.svg
    │   │   │   │   ├── attach.svg
    │   │   │   │   ├── binding.svg
    │   │   │   │   ├── box.svg
    │   │   │   │   ├── bulb.svg
    │   │   │   │   ├── code-file.svg
    │   │   │   │   ├── code-sandbox.svg
    │   │   │   │   ├── dark_to_light.svg
    │   │   │   │   ├── database.svg
    │   │   │   │   ├── doc.svg
    │   │   │   │   ├── empty-folder.svg
    │   │   │   │   ├── expression.svg
    │   │   │   │   ├── eye-closed.svg
    │   │   │   │   ├── eye-dark.svg
    │   │   │   │   ├── eye.svg
    │   │   │   │   ├── file-text.svg
    │   │   │   │   ├── filter.svg
    │   │   │   │   ├── folder.svg
    │   │   │   │   ├── img.svg
    │   │   │   │   ├── inspect.svg
    │   │   │   │   ├── light_to_dark.svg
    │   │   │   │   ├── moon.svg
    │   │   │   │   ├── pdf.svg
    │   │   │   │   ├── photo.svg
    │   │   │   │   ├── share.svg
    │   │   │   │   ├── stars.svg
    │   │   │   │   ├── sun.svg
    │   │   │   │   ├── trending-down.svg
    │   │   │   │   ├── trending-level.svg
    │   │   │   │   ├── trending-up.svg
    │   │   │   │   ├── txt.svg
    │   │   │   │   ├── unknown-file.svg
    │   │   │   │   ├── unlink.svg
    │   │   │   │   └── xls.svg
    │   │   │   ├── TableDeleteColumnIcon.tsx
    │   │   │   ├── TableDeleteRowIcon.tsx
    │   │   │   ├── TableInsertColumnIcon.tsx
    │   │   │   ├── TableInsertRowIcon.tsx
    │   │   │   ├── TrashIcon.tsx
    │   │   │   ├── TrendingDownIcon.tsx
    │   │   │   ├── TrendingLevelIcon.tsx
    │   │   │   ├── TrendingUpIcon.tsx
    │   │   │   ├── TxtIcon.tsx
    │   │   │   ├── UnknownFileIcon.tsx
    │   │   │   ├── UnlinkIcon.tsx
    │   │   │   ├── UserIcon.tsx
    │   │   │   ├── WarningIcon.tsx
    │   │   │   └── XlsIcon.tsx
    │   │   ├── IconProvider.tsx
    │   │   ├── IconRegistryContext.tsx
    │   │   ├── IFrame
    │   │   │   ├── IFrame.md
    │   │   │   ├── IFrame.module.scss
    │   │   │   ├── IFrame.spec.ts
    │   │   │   ├── IFrame.tsx
    │   │   │   └── IFrameNative.tsx
    │   │   ├── Image
    │   │   │   ├── Image.md
    │   │   │   ├── Image.module.scss
    │   │   │   ├── Image.spec.ts
    │   │   │   ├── Image.tsx
    │   │   │   └── ImageNative.tsx
    │   │   ├── Input
    │   │   │   ├── index.ts
    │   │   │   ├── InputAdornment.module.scss
    │   │   │   ├── InputAdornment.tsx
    │   │   │   ├── InputDivider.module.scss
    │   │   │   ├── InputDivider.tsx
    │   │   │   ├── InputLabel.module.scss
    │   │   │   ├── InputLabel.tsx
    │   │   │   ├── PartialInput.module.scss
    │   │   │   └── PartialInput.tsx
    │   │   ├── InspectButton
    │   │   │   ├── InspectButton.module.scss
    │   │   │   └── InspectButton.tsx
    │   │   ├── Items
    │   │   │   ├── Items.md
    │   │   │   ├── Items.spec.ts
    │   │   │   ├── Items.tsx
    │   │   │   └── ItemsNative.tsx
    │   │   ├── Link
    │   │   │   ├── Link.md
    │   │   │   ├── Link.module.scss
    │   │   │   ├── Link.spec.ts
    │   │   │   ├── Link.tsx
    │   │   │   └── LinkNative.tsx
    │   │   ├── List
    │   │   │   ├── doc-resources
    │   │   │   │   └── list-component-data.js
    │   │   │   ├── List.md
    │   │   │   ├── List.module.scss
    │   │   │   ├── List.spec.ts
    │   │   │   ├── List.tsx
    │   │   │   └── ListNative.tsx
    │   │   ├── Logo
    │   │   │   ├── doc-resources
    │   │   │   │   └── xmlui-logo.svg
    │   │   │   ├── Logo.md
    │   │   │   ├── Logo.tsx
    │   │   │   └── LogoNative.tsx
    │   │   ├── Markdown
    │   │   │   ├── CodeText.module.scss
    │   │   │   ├── CodeText.tsx
    │   │   │   ├── Markdown.md
    │   │   │   ├── Markdown.module.scss
    │   │   │   ├── Markdown.spec.ts
    │   │   │   ├── Markdown.tsx
    │   │   │   ├── MarkdownNative.tsx
    │   │   │   ├── parse-binding-expr.ts
    │   │   │   └── utils.ts
    │   │   ├── metadata-helpers.ts
    │   │   ├── ModalDialog
    │   │   │   ├── ConfirmationModalContextProvider.tsx
    │   │   │   ├── Dialog.module.scss
    │   │   │   ├── Dialog.tsx
    │   │   │   ├── ModalDialog.md
    │   │   │   ├── ModalDialog.module.scss
    │   │   │   ├── ModalDialog.spec.ts
    │   │   │   ├── ModalDialog.tsx
    │   │   │   ├── ModalDialogNative.tsx
    │   │   │   └── ModalVisibilityContext.tsx
    │   │   ├── NavGroup
    │   │   │   ├── NavGroup.md
    │   │   │   ├── NavGroup.module.scss
    │   │   │   ├── NavGroup.spec.ts
    │   │   │   ├── NavGroup.tsx
    │   │   │   ├── NavGroupContext.ts
    │   │   │   └── NavGroupNative.tsx
    │   │   ├── NavLink
    │   │   │   ├── NavLink.md
    │   │   │   ├── NavLink.module.scss
    │   │   │   ├── NavLink.spec.ts
    │   │   │   ├── NavLink.tsx
    │   │   │   └── NavLinkNative.tsx
    │   │   ├── NavPanel
    │   │   │   ├── NavPanel.md
    │   │   │   ├── NavPanel.module.scss
    │   │   │   ├── NavPanel.spec.ts
    │   │   │   ├── NavPanel.tsx
    │   │   │   └── NavPanelNative.tsx
    │   │   ├── NestedApp
    │   │   │   ├── AppWithCodeView.module.scss
    │   │   │   ├── AppWithCodeView.tsx
    │   │   │   ├── AppWithCodeViewNative.tsx
    │   │   │   ├── defaultProps.tsx
    │   │   │   ├── logo.svg
    │   │   │   ├── NestedApp.module.scss
    │   │   │   ├── NestedApp.tsx
    │   │   │   ├── NestedAppNative.tsx
    │   │   │   ├── Tooltip.module.scss
    │   │   │   ├── Tooltip.tsx
    │   │   │   └── utils.ts
    │   │   ├── NoResult
    │   │   │   ├── NoResult.md
    │   │   │   ├── NoResult.module.scss
    │   │   │   ├── NoResult.spec.ts
    │   │   │   ├── NoResult.tsx
    │   │   │   └── NoResultNative.tsx
    │   │   ├── NumberBox
    │   │   │   ├── numberbox-abstractions.ts
    │   │   │   ├── NumberBox.md
    │   │   │   ├── NumberBox.module.scss
    │   │   │   ├── NumberBox.spec.ts
    │   │   │   ├── NumberBox.tsx
    │   │   │   └── NumberBoxNative.tsx
    │   │   ├── Option
    │   │   │   ├── Option.md
    │   │   │   ├── Option.spec.ts
    │   │   │   ├── Option.tsx
    │   │   │   ├── OptionNative.tsx
    │   │   │   └── OptionTypeProvider.tsx
    │   │   ├── PageMetaTitle
    │   │   │   ├── PageMetaTilteNative.tsx
    │   │   │   ├── PageMetaTitle.md
    │   │   │   ├── PageMetaTitle.spec.ts
    │   │   │   └── PageMetaTitle.tsx
    │   │   ├── Pages
    │   │   │   ├── Page.md
    │   │   │   ├── Pages.md
    │   │   │   ├── Pages.module.scss
    │   │   │   ├── Pages.tsx
    │   │   │   └── PagesNative.tsx
    │   │   ├── Pagination
    │   │   │   ├── Pagination.md
    │   │   │   ├── Pagination.module.scss
    │   │   │   ├── Pagination.spec.ts
    │   │   │   ├── Pagination.tsx
    │   │   │   └── PaginationNative.tsx
    │   │   ├── PositionedContainer
    │   │   │   ├── PositionedContainer.module.scss
    │   │   │   ├── PositionedContainer.tsx
    │   │   │   └── PositionedContainerNative.tsx
    │   │   ├── ProfileMenu
    │   │   │   ├── ProfileMenu.module.scss
    │   │   │   └── ProfileMenu.tsx
    │   │   ├── ProgressBar
    │   │   │   ├── ProgressBar.md
    │   │   │   ├── ProgressBar.module.scss
    │   │   │   ├── ProgressBar.spec.ts
    │   │   │   ├── ProgressBar.tsx
    │   │   │   └── ProgressBarNative.tsx
    │   │   ├── Queue
    │   │   │   ├── Queue.md
    │   │   │   ├── Queue.spec.ts
    │   │   │   ├── Queue.tsx
    │   │   │   ├── queueActions.ts
    │   │   │   └── QueueNative.tsx
    │   │   ├── RadioGroup
    │   │   │   ├── RadioGroup.md
    │   │   │   ├── RadioGroup.module.scss
    │   │   │   ├── RadioGroup.spec.ts
    │   │   │   ├── RadioGroup.tsx
    │   │   │   ├── RadioGroupNative.tsx
    │   │   │   ├── RadioItem.tsx
    │   │   │   └── RadioItemNative.tsx
    │   │   ├── RealTimeAdapter
    │   │   │   ├── RealTimeAdapter.tsx
    │   │   │   └── RealTimeAdapterNative.tsx
    │   │   ├── Redirect
    │   │   │   ├── Redirect.md
    │   │   │   ├── Redirect.spec.ts
    │   │   │   └── Redirect.tsx
    │   │   ├── ResponsiveBar
    │   │   │   ├── README.md
    │   │   │   ├── ResponsiveBar.md
    │   │   │   ├── ResponsiveBar.module.scss
    │   │   │   ├── ResponsiveBar.spec.ts
    │   │   │   ├── ResponsiveBar.tsx
    │   │   │   └── ResponsiveBarNative.tsx
    │   │   ├── Select
    │   │   │   ├── HiddenOption.tsx
    │   │   │   ├── OptionContext.ts
    │   │   │   ├── Select.md
    │   │   │   ├── Select.module.scss
    │   │   │   ├── Select.spec.ts
    │   │   │   ├── Select.tsx
    │   │   │   ├── SelectContext.tsx
    │   │   │   └── SelectNative.tsx
    │   │   ├── SelectionStore
    │   │   │   ├── SelectionStore.md
    │   │   │   ├── SelectionStore.tsx
    │   │   │   └── SelectionStoreNative.tsx
    │   │   ├── Slider
    │   │   │   ├── Slider.md
    │   │   │   ├── Slider.module.scss
    │   │   │   ├── Slider.spec.ts
    │   │   │   ├── Slider.tsx
    │   │   │   └── SliderNative.tsx
    │   │   ├── Slot
    │   │   │   ├── Slot.md
    │   │   │   ├── Slot.spec.ts
    │   │   │   └── Slot.ts
    │   │   ├── SlotItem.tsx
    │   │   ├── SpaceFiller
    │   │   │   ├── SpaceFiller.md
    │   │   │   ├── SpaceFiller.module.scss
    │   │   │   ├── SpaceFiller.spec.ts
    │   │   │   ├── SpaceFiller.tsx
    │   │   │   └── SpaceFillerNative.tsx
    │   │   ├── Spinner
    │   │   │   ├── Spinner.md
    │   │   │   ├── Spinner.module.scss
    │   │   │   ├── Spinner.spec.ts
    │   │   │   ├── Spinner.tsx
    │   │   │   └── SpinnerNative.tsx
    │   │   ├── Splitter
    │   │   │   ├── HSplitter.md
    │   │   │   ├── HSplitter.spec.ts
    │   │   │   ├── Splitter.md
    │   │   │   ├── Splitter.module.scss
    │   │   │   ├── Splitter.spec.ts
    │   │   │   ├── Splitter.tsx
    │   │   │   ├── SplitterNative.tsx
    │   │   │   ├── utils.ts
    │   │   │   ├── VSplitter.md
    │   │   │   └── VSplitter.spec.ts
    │   │   ├── Stack
    │   │   │   ├── CHStack.md
    │   │   │   ├── CHStack.spec.ts
    │   │   │   ├── CVStack.md
    │   │   │   ├── CVStack.spec.ts
    │   │   │   ├── HStack.md
    │   │   │   ├── HStack.spec.ts
    │   │   │   ├── Stack.md
    │   │   │   ├── Stack.module.scss
    │   │   │   ├── Stack.spec.ts
    │   │   │   ├── Stack.tsx
    │   │   │   ├── StackNative.tsx
    │   │   │   ├── VStack.md
    │   │   │   └── VStack.spec.ts
    │   │   ├── StickyBox
    │   │   │   ├── StickyBox.md
    │   │   │   ├── StickyBox.module.scss
    │   │   │   ├── StickyBox.tsx
    │   │   │   └── StickyBoxNative.tsx
    │   │   ├── Switch
    │   │   │   ├── Switch.md
    │   │   │   ├── Switch.spec.ts
    │   │   │   └── Switch.tsx
    │   │   ├── Table
    │   │   │   ├── doc-resources
    │   │   │   │   └── list-component-data.js
    │   │   │   ├── react-table-config.d.ts
    │   │   │   ├── Table.md
    │   │   │   ├── Table.module.scss
    │   │   │   ├── Table.spec.ts
    │   │   │   ├── Table.tsx
    │   │   │   ├── TableNative.tsx
    │   │   │   └── useRowSelection.tsx
    │   │   ├── TableOfContents
    │   │   │   ├── TableOfContents.module.scss
    │   │   │   ├── TableOfContents.spec.ts
    │   │   │   ├── TableOfContents.tsx
    │   │   │   └── TableOfContentsNative.tsx
    │   │   ├── Tabs
    │   │   │   ├── TabContext.tsx
    │   │   │   ├── TabItem.md
    │   │   │   ├── TabItem.tsx
    │   │   │   ├── TabItemNative.tsx
    │   │   │   ├── Tabs.md
    │   │   │   ├── Tabs.module.scss
    │   │   │   ├── Tabs.spec.ts
    │   │   │   ├── Tabs.tsx
    │   │   │   └── TabsNative.tsx
    │   │   ├── Text
    │   │   │   ├── Text.md
    │   │   │   ├── Text.module.scss
    │   │   │   ├── Text.spec.ts
    │   │   │   ├── Text.tsx
    │   │   │   └── TextNative.tsx
    │   │   ├── TextArea
    │   │   │   ├── TextArea.md
    │   │   │   ├── TextArea.module.scss
    │   │   │   ├── TextArea.spec.ts
    │   │   │   ├── TextArea.tsx
    │   │   │   ├── TextAreaNative.tsx
    │   │   │   ├── TextAreaResizable.tsx
    │   │   │   └── useComposedRef.ts
    │   │   ├── TextBox
    │   │   │   ├── TextBox.md
    │   │   │   ├── TextBox.module.scss
    │   │   │   ├── TextBox.spec.ts
    │   │   │   ├── TextBox.tsx
    │   │   │   └── TextBoxNative.tsx
    │   │   ├── Theme
    │   │   │   ├── NotificationToast.tsx
    │   │   │   ├── Theme.md
    │   │   │   ├── Theme.module.scss
    │   │   │   ├── Theme.spec.ts
    │   │   │   ├── Theme.tsx
    │   │   │   └── ThemeNative.tsx
    │   │   ├── TimeInput
    │   │   │   ├── TimeInput.md
    │   │   │   ├── TimeInput.module.scss
    │   │   │   ├── TimeInput.spec.ts
    │   │   │   ├── TimeInput.tsx
    │   │   │   ├── TimeInputNative.tsx
    │   │   │   └── utils.ts
    │   │   ├── Timer
    │   │   │   ├── Timer.md
    │   │   │   ├── Timer.spec.ts
    │   │   │   ├── Timer.tsx
    │   │   │   └── TimerNative.tsx
    │   │   ├── Toggle
    │   │   │   ├── Toggle.module.scss
    │   │   │   └── Toggle.tsx
    │   │   ├── ToneChangerButton
    │   │   │   ├── ToneChangerButton.md
    │   │   │   ├── ToneChangerButton.spec.ts
    │   │   │   └── ToneChangerButton.tsx
    │   │   ├── ToneSwitch
    │   │   │   ├── ToneSwitch.md
    │   │   │   ├── ToneSwitch.module.scss
    │   │   │   ├── ToneSwitch.spec.ts
    │   │   │   ├── ToneSwitch.tsx
    │   │   │   └── ToneSwitchNative.tsx
    │   │   ├── Tooltip
    │   │   │   ├── Tooltip.md
    │   │   │   ├── Tooltip.module.scss
    │   │   │   ├── Tooltip.spec.ts
    │   │   │   ├── Tooltip.tsx
    │   │   │   └── TooltipNative.tsx
    │   │   ├── Tree
    │   │   │   ├── testData.ts
    │   │   │   ├── Tree-dynamic.spec.ts
    │   │   │   ├── Tree-icons.spec.ts
    │   │   │   ├── Tree.md
    │   │   │   ├── Tree.spec.ts
    │   │   │   ├── TreeComponent.module.scss
    │   │   │   ├── TreeComponent.tsx
    │   │   │   └── TreeNative.tsx
    │   │   ├── TreeDisplay
    │   │   │   ├── TreeDisplay.md
    │   │   │   ├── TreeDisplay.module.scss
    │   │   │   ├── TreeDisplay.tsx
    │   │   │   └── TreeDisplayNative.tsx
    │   │   ├── ValidationSummary
    │   │   │   ├── ValidationSummary.module.scss
    │   │   │   └── ValidationSummary.tsx
    │   │   └── VisuallyHidden.tsx
    │   ├── components-core
    │   │   ├── abstractions
    │   │   │   ├── ComponentRenderer.ts
    │   │   │   ├── LoaderRenderer.ts
    │   │   │   ├── standalone.ts
    │   │   │   └── treeAbstractions.ts
    │   │   ├── action
    │   │   │   ├── actions.ts
    │   │   │   ├── APICall.tsx
    │   │   │   ├── FileDownloadAction.tsx
    │   │   │   ├── FileUploadAction.tsx
    │   │   │   ├── NavigateAction.tsx
    │   │   │   └── TimedAction.tsx
    │   │   ├── ApiBoundComponent.tsx
    │   │   ├── appContext
    │   │   │   ├── date-functions.ts
    │   │   │   ├── math-function.ts
    │   │   │   └── misc-utils.ts
    │   │   ├── AppContext.tsx
    │   │   ├── behaviors
    │   │   │   ├── Behavior.tsx
    │   │   │   └── CoreBehaviors.tsx
    │   │   ├── component-hooks.ts
    │   │   ├── ComponentDecorator.tsx
    │   │   ├── ComponentViewer.tsx
    │   │   ├── CompoundComponent.tsx
    │   │   ├── constants.ts
    │   │   ├── DebugViewProvider.tsx
    │   │   ├── descriptorHelper.ts
    │   │   ├── devtools
    │   │   │   ├── InspectorDialog.module.scss
    │   │   │   ├── InspectorDialog.tsx
    │   │   │   └── InspectorDialogVisibilityContext.tsx
    │   │   ├── EngineError.ts
    │   │   ├── event-handlers.ts
    │   │   ├── InspectorButton.module.scss
    │   │   ├── InspectorContext.tsx
    │   │   ├── interception
    │   │   │   ├── abstractions.ts
    │   │   │   ├── ApiInterceptor.ts
    │   │   │   ├── ApiInterceptorProvider.tsx
    │   │   │   ├── apiInterceptorWorker.ts
    │   │   │   ├── Backend.ts
    │   │   │   ├── Errors.ts
    │   │   │   ├── IndexedDb.ts
    │   │   │   ├── initMock.ts
    │   │   │   ├── InMemoryDb.ts
    │   │   │   ├── ReadonlyCollection.ts
    │   │   │   └── useApiInterceptorContext.tsx
    │   │   ├── loader
    │   │   │   ├── ApiLoader.tsx
    │   │   │   ├── DataLoader.tsx
    │   │   │   ├── ExternalDataLoader.tsx
    │   │   │   ├── Loader.tsx
    │   │   │   ├── MockLoaderRenderer.tsx
    │   │   │   └── PageableLoader.tsx
    │   │   ├── LoaderComponent.tsx
    │   │   ├── markup-check.ts
    │   │   ├── parts.ts
    │   │   ├── renderers.ts
    │   │   ├── rendering
    │   │   │   ├── AppContent.tsx
    │   │   │   ├── AppRoot.tsx
    │   │   │   ├── AppWrapper.tsx
    │   │   │   ├── buildProxy.ts
    │   │   │   ├── collectFnVarDeps.ts
    │   │   │   ├── ComponentAdapter.tsx
    │   │   │   ├── ComponentWrapper.tsx
    │   │   │   ├── Container.tsx
    │   │   │   ├── containers.ts
    │   │   │   ├── ContainerWrapper.tsx
    │   │   │   ├── ErrorBoundary.module.scss
    │   │   │   ├── ErrorBoundary.tsx
    │   │   │   ├── InvalidComponent.module.scss
    │   │   │   ├── InvalidComponent.tsx
    │   │   │   ├── nodeUtils.ts
    │   │   │   ├── reducer.ts
    │   │   │   ├── renderChild.tsx
    │   │   │   ├── StandaloneComponent.tsx
    │   │   │   ├── StateContainer.tsx
    │   │   │   ├── UnknownComponent.module.scss
    │   │   │   ├── UnknownComponent.tsx
    │   │   │   └── valueExtractor.ts
    │   │   ├── reportEngineError.ts
    │   │   ├── RestApiProxy.ts
    │   │   ├── script-runner
    │   │   │   ├── asyncProxy.ts
    │   │   │   ├── AttributeValueParser.ts
    │   │   │   ├── bannedFunctions.ts
    │   │   │   ├── BindingTreeEvaluationContext.ts
    │   │   │   ├── eval-tree-async.ts
    │   │   │   ├── eval-tree-common.ts
    │   │   │   ├── eval-tree-sync.ts
    │   │   │   ├── ParameterParser.ts
    │   │   │   ├── process-statement-async.ts
    │   │   │   ├── process-statement-common.ts
    │   │   │   ├── process-statement-sync.ts
    │   │   │   ├── ScriptingSourceTree.ts
    │   │   │   ├── simplify-expression.ts
    │   │   │   ├── statement-queue.ts
    │   │   │   └── visitors.ts
    │   │   ├── StandaloneApp.tsx
    │   │   ├── StandaloneExtensionManager.ts
    │   │   ├── TableOfContentsContext.tsx
    │   │   ├── theming
    │   │   │   ├── _themes.scss
    │   │   │   ├── component-layout-resolver.ts
    │   │   │   ├── extendThemeUtils.ts
    │   │   │   ├── hvar.ts
    │   │   │   ├── layout-resolver.ts
    │   │   │   ├── parse-layout-props.ts
    │   │   │   ├── StyleContext.tsx
    │   │   │   ├── StyleRegistry.ts
    │   │   │   ├── ThemeContext.tsx
    │   │   │   ├── ThemeProvider.tsx
    │   │   │   ├── themes
    │   │   │   │   ├── base-utils.ts
    │   │   │   │   ├── palette.ts
    │   │   │   │   ├── root.ts
    │   │   │   │   ├── solid.ts
    │   │   │   │   ├── theme-colors.ts
    │   │   │   │   └── xmlui.ts
    │   │   │   ├── themeVars.module.scss
    │   │   │   ├── themeVars.ts
    │   │   │   ├── transformThemeVars.ts
    │   │   │   └── utils.ts
    │   │   ├── utils
    │   │   │   ├── actionUtils.ts
    │   │   │   ├── audio-utils.ts
    │   │   │   ├── base64-utils.ts
    │   │   │   ├── compound-utils.ts
    │   │   │   ├── css-utils.ts
    │   │   │   ├── DataLoaderQueryKeyGenerator.ts
    │   │   │   ├── date-utils.ts
    │   │   │   ├── extractParam.ts
    │   │   │   ├── hooks.tsx
    │   │   │   ├── LruCache.ts
    │   │   │   ├── mergeProps.ts
    │   │   │   ├── misc.ts
    │   │   │   ├── request-params.ts
    │   │   │   ├── statementUtils.ts
    │   │   │   └── treeUtils.ts
    │   │   └── xmlui-parser.ts
    │   ├── index-standalone.ts
    │   ├── index.scss
    │   ├── index.ts
    │   ├── language-server
    │   │   ├── server-common.ts
    │   │   ├── server-web-worker.ts
    │   │   ├── server.ts
    │   │   ├── services
    │   │   │   ├── common
    │   │   │   │   ├── docs-generation.ts
    │   │   │   │   ├── lsp-utils.ts
    │   │   │   │   ├── metadata-utils.ts
    │   │   │   │   └── syntax-node-utilities.ts
    │   │   │   ├── completion.ts
    │   │   │   ├── diagnostic.ts
    │   │   │   ├── format.ts
    │   │   │   └── hover.ts
    │   │   └── xmlui-metadata-generated.mjs
    │   ├── logging
    │   │   ├── LoggerContext.tsx
    │   │   ├── LoggerInitializer.tsx
    │   │   ├── LoggerService.ts
    │   │   └── xmlui.ts
    │   ├── logo.svg
    │   ├── parsers
    │   │   ├── common
    │   │   │   ├── GenericToken.ts
    │   │   │   ├── InputStream.ts
    │   │   │   └── utils.ts
    │   │   ├── scripting
    │   │   │   ├── code-behind-collect.ts
    │   │   │   ├── Lexer.ts
    │   │   │   ├── modules.ts
    │   │   │   ├── Parser.ts
    │   │   │   ├── ParserError.ts
    │   │   │   ├── ScriptingNodeTypes.ts
    │   │   │   ├── TokenTrait.ts
    │   │   │   ├── TokenType.ts
    │   │   │   └── tree-visitor.ts
    │   │   ├── style-parser
    │   │   │   ├── errors.ts
    │   │   │   ├── source-tree.ts
    │   │   │   ├── StyleInputStream.ts
    │   │   │   ├── StyleLexer.ts
    │   │   │   ├── StyleParser.ts
    │   │   │   └── tokens.ts
    │   │   └── xmlui-parser
    │   │       ├── CharacterCodes.ts
    │   │       ├── diagnostics.ts
    │   │       ├── fileExtensions.ts
    │   │       ├── index.ts
    │   │       ├── lint.ts
    │   │       ├── parser.ts
    │   │       ├── ParserError.ts
    │   │       ├── scanner.ts
    │   │       ├── syntax-kind.ts
    │   │       ├── syntax-node.ts
    │   │       ├── transform.ts
    │   │       ├── utils.ts
    │   │       ├── xmlui-serializer.ts
    │   │       └── xmlui-tree.ts
    │   ├── react-app-env.d.ts
    │   ├── syntax
    │   │   ├── monaco
    │   │   │   ├── grammar.monacoLanguage.ts
    │   │   │   ├── index.ts
    │   │   │   ├── xmlui-dark.ts
    │   │   │   ├── xmlui-light.ts
    │   │   │   └── xmluiscript.monacoLanguage.ts
    │   │   └── textMate
    │   │       ├── index.ts
    │   │       ├── xmlui-dark.json
    │   │       ├── xmlui-light.json
    │   │       ├── xmlui.json
    │   │       └── xmlui.tmLanguage.json
    │   ├── testing
    │   │   ├── assertions.ts
    │   │   ├── component-test-helpers.ts
    │   │   ├── ComponentDrivers.ts
    │   │   ├── drivers
    │   │   │   ├── DateInputDriver.ts
    │   │   │   ├── ModalDialogDriver.ts
    │   │   │   ├── NumberBoxDriver.ts
    │   │   │   ├── TextBoxDriver.ts
    │   │   │   ├── TimeInputDriver.ts
    │   │   │   ├── TimerDriver.ts
    │   │   │   └── TreeDriver.ts
    │   │   ├── fixtures.ts
    │   │   ├── infrastructure
    │   │   │   ├── index.html
    │   │   │   ├── main.tsx
    │   │   │   ├── public
    │   │   │   │   ├── mockServiceWorker.js
    │   │   │   │   ├── resources
    │   │   │   │   │   ├── bell.svg
    │   │   │   │   │   ├── box.svg
    │   │   │   │   │   ├── doc.svg
    │   │   │   │   │   ├── eye.svg
    │   │   │   │   │   ├── flower-640x480.jpg
    │   │   │   │   │   ├── sun.svg
    │   │   │   │   │   ├── test-image-100x100.jpg
    │   │   │   │   │   └── txt.svg
    │   │   │   │   └── serve.json
    │   │   │   └── TestBed.tsx
    │   │   └── themed-app-test-helpers.ts
    │   └── vite-env.d.ts
    ├── tests
    │   ├── components
    │   │   ├── CodeBlock
    │   │   │   └── hightlight-code.test.ts
    │   │   ├── playground-pattern.test.ts
    │   │   └── Tree
    │   │       └── Tree-states.test.ts
    │   ├── components-core
    │   │   ├── abstractions
    │   │   │   └── treeAbstractions.test.ts
    │   │   ├── container
    │   │   │   └── buildProxy.test.ts
    │   │   ├── interception
    │   │   │   ├── orderBy.test.ts
    │   │   │   ├── ReadOnlyCollection.test.ts
    │   │   │   └── request-param-converter.test.ts
    │   │   ├── scripts-runner
    │   │   │   ├── AttributeValueParser.test.ts
    │   │   │   ├── eval-tree-arrow-async.test.ts
    │   │   │   ├── eval-tree-arrow.test.ts
    │   │   │   ├── eval-tree-func-decl-async.test.ts
    │   │   │   ├── eval-tree-func-decl.test.ts
    │   │   │   ├── eval-tree-pre-post.test.ts
    │   │   │   ├── eval-tree-regression.test.ts
    │   │   │   ├── eval-tree.test.ts
    │   │   │   ├── function-proxy.test.ts
    │   │   │   ├── parser-regression.test.ts
    │   │   │   ├── process-event.test.ts
    │   │   │   ├── process-function.test.ts
    │   │   │   ├── process-implicit-context.test.ts
    │   │   │   ├── process-statement-asgn.test.ts
    │   │   │   ├── process-statement-destruct.test.ts
    │   │   │   ├── process-statement-regs.test.ts
    │   │   │   ├── process-statement-sync.test.ts
    │   │   │   ├── process-statement.test.ts
    │   │   │   ├── process-switch-sync.test.ts
    │   │   │   ├── process-switch.test.ts
    │   │   │   ├── process-try-sync.test.ts
    │   │   │   ├── process-try.test.ts
    │   │   │   └── test-helpers.ts
    │   │   ├── test-metadata-handler.ts
    │   │   ├── theming
    │   │   │   ├── border-segments.test.ts
    │   │   │   ├── component-layout.resolver.test.ts
    │   │   │   ├── layout-property-parser.test.ts
    │   │   │   ├── layout-resolver.test.ts
    │   │   │   ├── layout-resolver2.test.ts
    │   │   │   ├── layout-vp-override.test.ts
    │   │   │   └── padding-segments.test.ts
    │   │   └── utils
    │   │       ├── date-utils.test.ts
    │   │       ├── format-human-elapsed-time.test.ts
    │   │       └── LruCache.test.ts
    │   ├── language-server
    │   │   ├── completion.test.ts
    │   │   ├── format.test.ts
    │   │   ├── hover.test.ts
    │   │   └── mockData.ts
    │   └── parsers
    │       ├── common
    │       │   └── input-stream.test.ts
    │       ├── markdown
    │       │   └── parse-binding-expression.test.ts
    │       ├── parameter-parser.test.ts
    │       ├── paremeter-parser.test.ts
    │       ├── scripting
    │       │   ├── eval-tree-arrow.test.ts
    │       │   ├── eval-tree-pre-post.test.ts
    │       │   ├── eval-tree.test.ts
    │       │   ├── function-proxy.test.ts
    │       │   ├── lexer-literals.test.ts
    │       │   ├── lexer-misc.test.ts
    │       │   ├── module-parse.test.ts
    │       │   ├── parser-arrow.test.ts
    │       │   ├── parser-assignments.test.ts
    │       │   ├── parser-binary.test.ts
    │       │   ├── parser-destructuring.test.ts
    │       │   ├── parser-errors.test.ts
    │       │   ├── parser-expressions.test.ts
    │       │   ├── parser-function.test.ts
    │       │   ├── parser-literals.test.ts
    │       │   ├── parser-primary.test.ts
    │       │   ├── parser-regex.test.ts
    │       │   ├── parser-statements.test.ts
    │       │   ├── parser-unary.test.ts
    │       │   ├── process-event.test.ts
    │       │   ├── process-implicit-context.test.ts
    │       │   ├── process-statement-asgn.test.ts
    │       │   ├── process-statement-destruct.test.ts
    │       │   ├── process-statement-regs.test.ts
    │       │   ├── process-statement-sync.test.ts
    │       │   ├── process-statement.test.ts
    │       │   ├── process-switch-sync.test.ts
    │       │   ├── process-switch.test.ts
    │       │   ├── process-try-sync.test.ts
    │       │   ├── process-try.test.ts
    │       │   ├── simplify-expression.test.ts
    │       │   ├── statement-hooks.test.ts
    │       │   └── test-helpers.ts
    │       ├── style-parser
    │       │   ├── generateHvarChain.test.ts
    │       │   ├── parseHVar.test.ts
    │       │   ├── parser.test.ts
    │       │   └── tokens.test.ts
    │       └── xmlui
    │           ├── lint.test.ts
    │           ├── parser.test.ts
    │           ├── scanner.test.ts
    │           ├── transform.attr.test.ts
    │           ├── transform.circular.test.ts
    │           ├── transform.element.test.ts
    │           ├── transform.errors.test.ts
    │           ├── transform.escape.test.ts
    │           ├── transform.regression.test.ts
    │           ├── transform.script.test.ts
    │           ├── transform.test.ts
    │           └── xmlui.ts
    ├── tests-e2e
    │   ├── api-bound-component-regression.spec.ts
    │   ├── api-call-as-extracted-component.spec.ts
    │   ├── assign-to-object-or-array-regression.spec.ts
    │   ├── binding-regression.spec.ts
    │   ├── children-as-template-context-vars.spec.ts
    │   ├── compound-component.spec.ts
    │   ├── context-vars-regression.spec.ts
    │   ├── data-bindings.spec.ts
    │   ├── datasource-and-api-usage-in-var.spec.ts
    │   ├── datasource-direct-binding.spec.ts
    │   ├── datasource-onLoaded-regression.spec.ts
    │   ├── modify-array-item-regression.spec.ts
    │   ├── namespaces.spec.ts
    │   ├── push-to-array-regression.spec.ts
    │   ├── screen-breakpoints.spec.ts
    │   ├── scripting.spec.ts
    │   ├── state-scope-in-pages.spec.ts
    │   └── state-var-scopes.spec.ts
    ├── tsconfig.bin.json
    ├── tsconfig.json
    ├── tsconfig.node.json
    ├── vite.config.ts
    └── vitest.config.ts
```

# Files

--------------------------------------------------------------------------------
/xmlui/dev-docs/react-fundamentals.md:
--------------------------------------------------------------------------------

```markdown
# React Fundamentals

This document is a concise reference for React patterns and hooks used in XMLUI. It assumes you're familiar with [React basics](https://react.dev/learn) and provides practical guidance for reading and maintaining XMLUI source code.

## React Hooks: Quick Overview

**What are hooks?** Functions that let you "hook into" React features from function components. They always start with `use` (e.g., `useState`, `useEffect`).

**Core hooks in XMLUI:**
- **`useState`** - Add local state to components
- **`useEffect`** - Run side effects (data fetching, subscriptions, DOM updates)
- **`useRef`** - Store mutable values that don't trigger re-renders
- **`useMemo`** - Cache expensive calculations
- **`useCallback`** - Cache function definitions
- **`useReducer`** - Manage complex state with reducer pattern
- **`useContext`** - Access context values from providers

**Two critical rules:**
1. Only call hooks at the top level (not in loops, conditions, or nested functions)
2. Only call hooks from React functions (components or custom hooks)

**Why?** React tracks hooks by call order. Breaking these rules causes state mismatches and bugs.

## Component Rendering Lifecycle

React components re-render whenever their state or props change. Understanding this cycle prevents performance issues and unexpected behavior.

**The 5-Phase Render Cycle:**

1. **Trigger** → Something requests a render:
   - Parent component re-renders
   - Props change
   - State changes (`useState`, `useReducer`)

2. **Render** → React calls your component function, which returns JSX

3. **Reconciliation** → React's diffing algorithm determines what changed in the virtual DOM

4. **Commit** → React updates the actual DOM with minimal changes

5. **Effects** → React runs effects in order:
   - `useLayoutEffect` (synchronous, blocks paint)
   - Browser paints the screen
   - `useEffect` (asynchronous, after paint)

**Key insight:** Re-rendering is cheap (just a function call), but DOM updates are expensive. React optimizes by batching and minimizing DOM changes.

**Common performance pitfalls:**
```tsx
// ❌ WRONG - Parent re-renders cause all children to re-render
function Parent() {
  const [count, setCount] = useState(0);
  return (
    <div>
      <ExpensiveChild data={data} />  {/* Re-renders unnecessarily */}
      <button onClick={() => setCount(c => c + 1)}>Update</button>
    </div>
  );
}

// ✅ CORRECT - Memoize to prevent unnecessary re-renders
const MemoizedChild = React.memo(ExpensiveChild);

function Parent() {
  const [count, setCount] = useState(0);
  const data = useMemo(() => computeData(), []); // Stable reference
  return (
    <div>
      <MemoizedChild data={data} />  {/* Only re-renders when data changes */}
      <button onClick={() => setCount(c => c + 1)}>Update</button>
    </div>
  );
}
```

## Rules of Hooks

These rules are enforced by React and ESLint. Violating them causes hard-to-debug errors.

**Rule 1: Call hooks at the top level**
```tsx
// ❌ WRONG - Conditional hook
function Bad({ show }: Props) {
  if (show) {
    const [value, setValue] = useState(""); // Hook order changes!
  }
  return <div>...</div>;
}

// ✅ CORRECT - Hook always called
function Good({ show }: Props) {
  const [value, setValue] = useState("");
  if (!show) return null;
  return <div>{value}</div>;
}
```

**Rule 2: Only call from React functions**
```tsx
// ❌ WRONG - Hook in regular function
function getUser() {
  const [user, setUser] = useState(null); // Not allowed!
  return user;
}

// ✅ CORRECT - Hook in component or custom hook
function useUser() {
  const [user, setUser] = useState(null);
  return user;
}

function Component() {
  const user = useUser(); // OK
  return <div>{user?.name}</div>;
}
```

**Why these rules exist:** React stores hook state in a sequential array tied to each component instance. The array index depends on call order. Conditional hooks break this indexing, causing state to be assigned to the wrong hooks.

---

## React State Management Patterns

Fundamental patterns for managing state in React, from local component state to shared state across component trees.

### `useState` - Local State

**Syntax:** `const [state, setState] = useState(initialValue)`

```tsx
// Basic
const [count, setCount] = useState(0);

// Functional update (when new state depends on old)
setCount(prev => prev + 1);

// Lazy initialization (expensive initial state)
const [data, setData] = useState(() => expensiveComputation());

// Immutable updates
setUser(prev => ({ ...prev, name }));
setItems(prev => [...prev, newItem]);
```

**Use when:** State is local, updates are simple, no complex transitions.  
**Consider alternatives:** Multiple components → Context, complex logic → useReducer.

---

### `useReducer` - Complex State Logic

**Syntax:** `const [state, dispatch] = useReducer(reducer, initialState)`

```tsx
type Action = { type: 'increment' } | { type: 'decrement' } | { type: 'reset' };

function reducer(state: State, action: Action): State {
  switch (action.type) {
    case 'increment': return { count: state.count + 1 };
    case 'decrement': return { count: state.count - 1 };
    case 'reset': return { count: 0 };
    default: return state;
  }
}

const [state, dispatch] = useReducer(reducer, { count: 0 });
dispatch({ type: 'increment' });
```

**With Immer (XMLUI pattern):**
```tsx
import produce from 'immer';

const reducer = produce((draft, action) => {
  switch (action.type) {
    case 'ADD_TODO':
      draft.todos.push(action.payload); // Direct mutation
      break;
  }
});
```

**Use when:** Multiple related state values, complex transitions, want to separate state logic.  
**Use useState when:** Simple independent values, straightforward updates.

---

### Context API - Avoid Prop Drilling

**Purpose:** Share state across component tree without passing props through every level.

**Pattern:** Create context → Provider component → Custom hook → Consume

```tsx
// 1. Create context
const AuthContext = createContext<AuthContext | null>(null);

// 2. Custom hook with validation
function useAuth() {
  const context = useContext(AuthContext);
  if (!context) throw new Error('useAuth must be used within AuthProvider');
  return context;
}

// 3. Provider component
function AuthProvider({ children }: Props) {
  const [user, setUser] = useState<User | null>(null);
  
  const value = useMemo(() => ({
    user,
    login: async (creds: Credentials) => { /* ... */ },
    logout: () => setUser(null),
  }), [user]);
  
  return <AuthContext.Provider value={value}>{children}</AuthContext.Provider>;
}

// 4. Consume anywhere in tree
function Component() {
  const { user, logout } = useAuth(); // No prop drilling!
  return <button onClick={logout}>{user.name}</button>;
}
```

**XMLUI Example:**
```tsx
// AppLayoutContext - Provides layout state to nested navigation
const AppLayoutContext = createContext<IAppLayoutContext | null>(null);

export const App = forwardRef(function App(props, ref) {
  const layoutContextValue = useMemo(() => ({
    layout,
    navPanelVisible,
    toggleDrawer,
  }), [layout, navPanelVisible, toggleDrawer]);
  
  return (
    <AppLayoutContext.Provider value={layoutContextValue}>
      {content}
    </AppLayoutContext.Provider>
  );
});

// NavLink accesses layout directly
export const NavLink = forwardRef(function NavLink(props, ref) {
  const { layout } = useAppLayoutContext();
  // No prop drilling!
});
```

**Use when:** Many nested components need access, >3 levels of prop drilling, global state (theme, auth).  
**Don't use when:** 1-2 levels of nesting (props fine), high-frequency updates (re-renders all consumers).

**Performance tip:** Split contexts by update frequency - separate user/theme/settings contexts instead of one combined context.

---

### State Lifting

**Pattern:** Move state to common ancestor to share between siblings.

```tsx
// ❌ WRONG - Siblings can't communicate
<Parent><InputA /><InputB /></Parent>

// ✅ CORRECT - Lift state to parent
function Parent() {
  const [value, setValue] = useState('');
  return (
    <>
      <InputA value={value} onChange={setValue} />
      <InputB value={value} />
    </>
  );
}
```

**XMLUI Pattern:** Container-based state flows down automatically:
```tsx
<Stack var.selectedId="{null}">
  <Button onClick={"{() => selectedId = 'item1'}" />
  <Display value="{selectedId}" />
</Stack>
```

**Use when:** Siblings need to coordinate, parent orchestrates behavior.  
**Don't use when:** State only used by one component, excessive prop drilling (use context).

---

### Controlled vs Uncontrolled Components

**Controlled:** Parent manages state via `value` prop.
```tsx
function Controlled({ value, onChange }: Props) {
  return <input value={value} onChange={e => onChange(e.target.value)} />;
}
```

**Uncontrolled:** Component manages own state via `initialValue`.
```tsx
function Uncontrolled({ initialValue = '', onDidChange }: Props) {
  const [value, setValue] = useState(initialValue);
  return <input value={value} onChange={e => { setValue(e.target.value); onDidChange?.(e.target.value); }} />;
}
```

**Hybrid (XMLUI pattern):** Support both modes.
```tsx
function Flexible({ value, initialValue = '', onDidChange }: Props) {
  const [localValue, setLocalValue] = useState(initialValue);
  
  useEffect(() => {
    if (value !== undefined) setLocalValue(value);
  }, [value]);
  
  const handleChange = (e) => {
    setLocalValue(e.target.value);
    onDidChange?.(e.target.value);
  };
  
  return <input value={localValue} onChange={handleChange} />;
}
```

**Use controlled:** Validate/format input, value affects other UI, programmatic changes.  
**Use uncontrolled:** Simple forms (read on submit), performance critical.  
**Use hybrid:** Reusable component libraries.

---

### Compound Components

**Purpose:** Components work together as cohesive unit, sharing state via context.

```tsx
const TabsContext = createContext<{
  activeTab: string;
  setActiveTab: (id: string) => void;
} | null>(null);

function Tabs({ children }: Props) {
  const [activeTab, setActiveTab] = useState('tab1');
  return (
    <TabsContext.Provider value={{ activeTab, setActiveTab }}>
      <div className="tabs">{children}</div>
    </TabsContext.Provider>
  );
}

function Tab({ id, children }: Props) {
  const { activeTab, setActiveTab } = useContext(TabsContext)!;
  return (
    <button
      className={activeTab === id ? 'active' : ''}
      onClick={() => setActiveTab(id)}
    >
      {children}
    </button>
  );
}

Tabs.Tab = Tab;
Tabs.Panel = TabPanel;

// Usage - Flexible composition
<Tabs>
  <Tabs.Tab id="tab1">First</Tabs.Tab>
  <Tabs.Tab id="tab2">Second</Tabs.Tab>
  <Tabs.Panel id="tab1">Content 1</Tabs.Panel>
  <Tabs.Panel id="tab2">Content 2</Tabs.Panel>
</Tabs>
```

**Use when:** Tightly coupled components (tabs, accordion), need flexible composition, building libraries.  
**Don't use when:** Simple parent-child, no shared state, prop drilling is simple.

---

### Advanced Patterns

**Provider Composition:**
```tsx
function AppProviders({ children }: Props) {
  return (
    <ThemeProvider>
      <AuthProvider>
        <RouterProvider>
          {children}
        </RouterProvider>
      </AuthProvider>
    </ThemeProvider>
  );
}
```

**Async Initialization:**
```tsx
function AuthProvider({ children }: Props) {
  const [user, setUser] = useState(null);
  const [loading, setLoading] = useState(true);
  
  useEffect(() => {
    checkAuth().then(user => { setUser(user); setLoading(false); });
  }, []);
  
  if (loading) return <LoadingScreen />;
  return <AuthContext.Provider value={{ user }}>{children}</AuthContext.Provider>;
}
```

**Reducer with Immer (XMLUI):**
```tsx
// StateContainer reducer
export function createContainerReducer(debugView: IDebugViewContext) {
  return produce((state: ContainerState, action: ContainerAction) => {
    switch (action.type) {
      case ContainerActionKind.COMPONENT_STATE_CHANGED:
        state[uid] = { ...state[uid], ...action.payload.state };
        break;
    }
  });
}
```

---

## React Performance Optimization Patterns

This section covers React's performance optimization tools and patterns. **Always profile before optimizing**—premature optimization adds complexity without real benefits.

### Core Optimization Hooks

#### `useMemo` - Computation Caching

Cache expensive calculations between renders.

```tsx
const filtered = useMemo(() => 
  items.filter(item => item.includes(filter)), 
  [items, filter]
);
```

**Use when:** Computation is expensive (>10ms), creating objects/arrays for memoized children, or calculations in dependency arrays.  
**Don't use when:** Computation is cheap (<1ms), result used only once, or component rarely re-renders.

#### `useCallback` - Function Caching

Cache function definitions to prevent child re-renders.

```tsx
const handleClick = useCallback(() => {
  doSomething(value);
}, [value]);
```

**Use when:** Passing callbacks to memoized children or to dependency arrays.  
**Don't use when:** Function isn't passed to memoized components or deps arrays.

**Note:** `useCallback(fn, deps)` is equivalent to `useMemo(() => fn, deps)`.

#### `useTransition` - Non-Urgent Updates

Mark state updates as low-priority to keep UI responsive.

```tsx
const [isPending, startTransition] = useTransition();

startTransition(() => {
  setExpensiveState(newValue); // Won't block UI
});
```

**Use when:** Updating expensive state that doesn't need immediate feedback (filtering large lists, complex calculations).

#### `memo` - Component Memoization

Prevent re-renders when props haven't changed.

```tsx
const MemoChild = memo(function MemoChild({ data }: Props) {
  return <div>{data.value}</div>;
});
```

**Use when:** Component renders frequently with same props, has expensive rendering, or is in large lists.  
**Don't use when:** Component rarely re-renders, props change every render, or rendering is cheap.

**Important:** `memo` only works if props are stable. Use `useMemo`/`useCallback` for object/function props.

---

### Memoization Strategy Pattern

**Principle:** `memo` + `useMemo` + `useCallback` work together. `memo` prevents re-renders, but only if props stay stable. Use `useMemo`/`useCallback` to keep props stable.

#### The Memoization Cascade

```tsx
// ❌ ANTI-PATTERN - memo without stable props
const Child = memo(({ data, onClick }: Props) => <div onClick={onClick}>{data.value}</div>);

function Parent() {
  return <Child data={{ value: 123 }} onClick={() => {}} />; // New refs every render!
}

// ✅ CORRECT - memo with stable props
function Parent() {
  const data = useMemo(() => ({ value: 123 }), []);
  const onClick = useCallback(() => console.log('clicked'), []);
  return <Child data={data} onClick={onClick} />;
}
```

#### Decision Tree

1. **Performance problem?** No → Don't optimize. Yes → Step 2.
2. **What's the cause?**
   - Parent re-renders often → Use `memo()` on child
   - Expensive computation → Use `useMemo()` on calculation  
   - New function props → Use `useCallback()` on function
3. **Passing objects/arrays/functions to memoized component?** Yes → Memoize those too.

#### Common Patterns

```tsx
// Pattern 1: Context values
const contextValue = useMemo(() => ({
  state,
  setState,
  isLoading: state.status === 'loading',
}), [state]);

// Pattern 2: Event handlers with deps
const handleSearch = useCallback(() => {
  if (query.length >= minLength) onSearch(query);
}, [query, minLength, onSearch]);

// Pattern 3: Expensive selectors
const filteredData = useMemo(() => {
  return data.filter(item => item.name.includes(filter)).sort((a, b) => a.name.localeCompare(b.name));
}, [data, filter]);

// Pattern 4: Derived state
const summary = useMemo(() => ({
  total: items.reduce((sum, item) => sum + item.price * item.quantity, 0),
  itemCount: items.reduce((sum, item) => sum + item.quantity, 0),
}), [items]);
```

#### Anti-Patterns

```tsx
// ❌ Over-memoization
const greeting = useMemo(() => `Hello, ${name}`, [name]); // Too simple!

// ❌ Incomplete chain
<MemoChild config={{ theme: 'dark' }} />; // New object defeats memo

// ❌ Unstable dependencies
useMemo(() => formatUser(user), [user]); // user object recreated every render
// ✅ Fix: useMemo(() => formatUser(user), [user.id, user.name]);
```

#### Checklist

**✅ DO memoize:**
- Components rendering frequently with same props
- Expensive computations (>10ms)
- Objects/arrays/functions passed to memoized children
- Context values

**❌ DON'T memoize:**
- Cheap operations (<1ms)
- Values that change every render
- Without profiling first

---

### Virtualization Pattern

**Purpose:** Render only visible items in large lists by using "windowing." Instead of rendering 10,000 items, render only ~10 visible items.

**Libraries:** XMLUI uses two based on component needs:
- **virtua** (Tree, List) - Chat interfaces, reverse scrolling, auto-sizing, fixed-size lists
- **@tanstack/react-virtual** (Table) - Dynamic measurements, flexible

**Library Comparison:**

| Feature | virtua | @tanstack/react-virtual |
|---------|--------|------------------------|
| **Bundle Size** | ~6KB | ~4KB |
| **API** | Render props | Hooks |
| **Dynamic heights** | Automatic | Automatic |
| **Reverse scroll** | ✅ Built-in | Manual |
| **Auto-sizing** | ✅ Built-in | Manual |
| **XMLUI Usage** | Tree, List | Table |

**virtua Example (XMLUI List):**

```tsx
import { Virtualizer } from 'virtua';

function ChatList({ messages }: Props) {
  return (
    <Virtualizer count={messages.length}>
      {(index) => {
        const msg = messages[index];
        return (
          <div key={msg.id}>
            <div>{msg.author}</div>
            <div>{msg.content}</div>
          </div>
        );
      }}
    </Virtualizer>
  );
}
```

**@tanstack/react-virtual Example:**

```tsx
import { useVirtualizer } from '@tanstack/react-virtual';

function DataTable({ rows }: Props) {
  const tableRef = useRef<HTMLDivElement>(null);
  
  const rowVirtualizer = useVirtualizer({
    count: rows.length,
    getScrollElement: () => tableRef.current,
    estimateSize: () => 30,
    overscan: 5,
  });
  
  return (
    <div ref={tableRef} style={{ height: '400px', overflow: 'auto' }}>
      {rowVirtualizer.getVirtualItems().map((virtualRow) => (
        <div
          key={virtualRow.index}
          ref={(el) => rowVirtualizer.measureElement(el)}
          style={{ transform: `translateY(${virtualRow.start}px)` }}
        >
          {rows[virtualRow.index].content}
        </div>
      ))}
    </div>
  );
}
```

**Critical Rules:**
1. **Memoize row components** - Use `React.memo()`
2. **Apply transform/style** - Required for positioning (@tanstack)
3. **Memoize data** - Prevent row re-renders
4. **Handle scroll container** - Each library handles sizing differently

**Performance Impact:**

| Items | Normal | Virtualized | Improvement |
|-------|--------|-------------|-------------|
| 100 | 50ms | 10ms | 5x faster |
| 1,000 | 500ms | 10ms | 50x faster |
| 10,000 | 5s | 10ms | 500x faster |

**When to Use:**
- ✅ >100 items
- ✅ Uniform item sizes
- ✅ Scrollable datasets
- ❌ <100 items (overhead not worth it)
- ❌ Already paginated
- ❌ Complex/unpredictable heights

---

### Rate Limiting: Debouncing and Throttling

**Purpose:** Control the frequency of expensive operations during high-frequency events (user input, scrolling, resizing).

**Key Difference:**
- **Debouncing**: Wait until activity **stops** (search, autosave)
- **Throttling**: Execute at **regular intervals** during activity (scroll, mousemove)

```tsx
// User types "search" continuously

// DEBOUNCING: Executes ONCE after user stops typing
// Timeline: [type...type...type...STOP] → Execute

// THROTTLING: Executes EVERY 200ms while typing  
// Timeline: Execute → [200ms] → Execute → [200ms] → Execute...
```

#### Debouncing Solutions

**1. useDeferredValue (React 18+) - Recommended**

```tsx
function Search() {
  const [query, setQuery] = useState('');
  const deferredQuery = useDeferredValue(query);
  
  const results = useMemo(() => {
    if (deferredQuery.length < 2) return [];
    return performSearch(deferredQuery);
  }, [deferredQuery]);
  
  return (
    <>
      <input value={query} onChange={e => setQuery(e.target.value)} />
      <ResultsList results={results} />
    </>
  );
}
```

**2. Custom useDebounce Hook**

```tsx
function useDebounce<T>(value: T, delay: number = 500): T {
  const [debouncedValue, setDebouncedValue] = useState<T>(value);
  
  useEffect(() => {
    const timer = setTimeout(() => setDebouncedValue(value), delay);
    return () => clearTimeout(timer);
  }, [value, delay]);
  
  return debouncedValue;
}

// Usage
const debouncedQuery = useDebounce(query, 300);
```

**3. Lodash debounce**

```tsx
const debouncedSave = useMemo(
  () => debounce((text: string) => saveToServer(text), 1000),
  []
);

useEffect(() => {
  return () => debouncedSave.cancel();
}, [debouncedSave]);
```

#### Throttling Solutions

**1. Custom useThrottle Hook**

```tsx
function useThrottle<T extends (...args: any[]) => any>(
  callback: T,
  delay: number = 200
): T {
  const lastRun = useRef(Date.now());
  
  return useCallback((...args: Parameters<T>) => {
    const now = Date.now();
    if (now - lastRun.current >= delay) {
      lastRun.current = now;
      return callback(...args);
    }
  }, [callback, delay]) as T;
}

// Usage
const handleScroll = useThrottle(() => {
  setScrollPos(window.scrollY);
}, 200);
```

**2. Lodash throttle**

```tsx
const throttledScroll = useMemo(
  () => throttle(() => {
    updateScrollPosition();
  }, 200, {
    leading: true,   // Execute on first call
    trailing: true   // Execute after interval ends
  }),
  []
);

useEffect(() => {
  return () => throttledScroll.cancel();
}, [throttledScroll]);
```

#### XMLUI Examples

**Debounced Search (Search.tsx):**
```tsx
function Search({ data, limit }: Props) {
  const [inputValue, setInputValue] = useState("");
  const debouncedValue = useDeferredValue(inputValue);
  
  const results = useMemo(() => {
    if (debouncedValue.length <= 1) return [];
    return fuse.search(debouncedValue, { limit });
  }, [debouncedValue, limit]);
  
  return (
    <>
      <input value={inputValue} onChange={e => setInputValue(e.target.value)} />
      <SearchResults results={results} />
    </>
  );
}
```

**Throttled Change Listener (ChangeListenerNative.tsx):**
```tsx
function ChangeListener({ listenTo, onChange, throttleWaitInMs = 0 }: Props) {
  const throttledOnChange = useMemo(() => {
    if (throttleWaitInMs !== 0 && onChange) {
      return throttle(onChange, throttleWaitInMs, { leading: true });
    }
    return onChange;
  }, [onChange, throttleWaitInMs]);
  
  useEffect(() => {
    if (throttledOnChange) {
      throttledOnChange({ prevValue, newValue: listenTo });
    }
  }, [listenTo, throttledOnChange]);
}
```

**Async Throttle for Validation (misc.ts):**
```tsx
function asyncThrottle<F extends (...args: any[]) => Promise<any>>(
  func: F,
  wait?: number,
  options?: ThrottleSettings
) {
  const throttled = throttle(
    (resolve, reject, args: Parameters<F>) => {
      void func(...args).then(resolve).catch(reject);
    },
    wait,
    options
  );
  
  return (...args: Parameters<F>): ReturnType<F> =>
    new Promise((resolve, reject) => {
      throttled(resolve, reject, args);
    }) as ReturnType<F>;
}
```

#### Decision Guide

| Scenario | Solution | Timing |
|----------|----------|--------|
| Search input | Debounce | 300ms |
| Form validation | Debounce | 500ms |
| Autosave | Debounce | 1000ms |
| Scroll position | Throttle | 100-200ms |
| Window resize | Throttle | 200-300ms |
| Mouse tracking | Throttle | 50-100ms |
| API rate limiting | Throttle | 500-1000ms |

#### Performance Impact

| Operation | Without | With (300ms) | Improvement |
|-----------|---------|--------------|-------------|
| Search (6 chars typed) | 6 API calls | 1 API call | 83% reduction |
| Scroll (1s) | ~60 events | 5 events | 92% reduction |
| Window resize | ~30 events | 5 events | 83% reduction |

#### Critical Rules

**1. Always memoize** rate-limited functions:
```tsx
// ❌ WRONG - Creates new function every render
const handle = debounce(() => search(), 500);

// ✅ CORRECT - Memoized
const handle = useMemo(() => debounce(() => search(), 500), []);
```

**2. Always cleanup**:
```tsx
useEffect(() => {
  return () => debouncedFn.cancel();
}, [debouncedFn]);
```

**3. Don't rate-limit UI state** - only side effects:
```tsx
// ❌ WRONG - UI lags
const handleChange = debounce((e) => setValue(e.target.value), 300);

// ✅ CORRECT - Immediate UI, debounced side effect
const handleChange = (e) => {
  setValue(e.target.value); // Instant
  debouncedSearch(e.target.value); // Delayed
};
```

**4. Choose appropriate delays**:
- Search: 300ms
- Autosave: 1000ms  
- Scroll/resize: 100-200ms
- Mousemove: 50ms

#### When to Use

**Debouncing:**
- ✅ Search, autosave, validation
- ✅ Wait for user to finish action
- ❌ Don't use for immediate feedback

**Throttling:**
- ✅ Scroll, resize, mousemove
- ✅ Execute during continuous activity
- ❌ Don't use when only final value matters

#### Resources

- [useDeferredValue](https://react.dev/reference/react/useDeferredValue) - React 18 docs
- [lodash.debounce](https://lodash.com/docs/#debounce) - Debounce docs
- [lodash.throttle](https://lodash.com/docs/#throttle) - Throttle docs
- [XMLUI Search](packages/xmlui-search/src/Search.tsx) - Production example

---

## React Event Handling Patterns

Patterns for handling user interactions efficiently and correctly in React applications.

### Event Delegation Pattern

**Purpose:** Handle events for multiple children at parent level instead of attaching handlers to each child.

**Benefits:**
- Fewer event listeners (better memory usage)
- Works with dynamically added/removed children
- Simplifies event handler management

```tsx
// ❌ ANTI-PATTERN - Handler on every item
function List({ items }: Props) {
  return (
    <ul>
      {items.map(item => (
        <li key={item.id} onClick={() => handleClick(item.id)}>
          {item.name}
        </li>
      ))}
    </ul>
  );
}

// ✅ CORRECT - Single handler on parent
function List({ items }: Props) {
  const handleClick = (e: React.MouseEvent<HTMLUListElement>) => {
    const target = e.target as HTMLElement;
    const li = target.closest('li');
    if (li) {
      const itemId = li.dataset.id;
      console.log('Clicked item:', itemId);
    }
  };
  
  return (
    <ul onClick={handleClick}>
      {items.map(item => (
        <li key={item.id} data-id={item.id}>
          {item.name}
        </li>
      ))}
    </ul>
  );
}
```

**XMLUI Example (Tree component):**
```tsx
function Tree({ items }: Props) {
  // Single click handler for entire tree
  const handleTreeClick = useCallback((e: React.MouseEvent) => {
    const target = e.target as HTMLElement;
    const treeItem = target.closest('[data-tree-item]');
    
    if (treeItem) {
      const itemId = treeItem.getAttribute('data-item-id');
      const action = target.getAttribute('data-action');
      
      if (action === 'expand') {
        toggleExpand(itemId);
      } else if (action === 'select') {
        selectItem(itemId);
      }
    }
  }, [toggleExpand, selectItem]);
  
  return (
    <div className="tree" onClick={handleTreeClick}>
      {renderTreeItems(items)}
    </div>
  );
}
```

**When to use:**
- Lists with many items (>50)
- Dynamic children (added/removed frequently)
- Multiple event types on same children
- Performance-critical rendering

**When NOT to use:**
- Few items (<10) - overhead not worth it
- Need precise event target info
- Event handler logic is complex per-item

---

### Synthetic Event Pattern

**Purpose:** React wraps native browser events in `SyntheticEvent` for cross-browser consistency.

**Key differences from native events:**

| Feature | Native Event | Synthetic Event |
|---------|-------------|-----------------|
| **Type** | Browser-specific | Unified React type |
| **Pooling (React 16)** | No | Yes (reused) |
| **Pooling (React 17+)** | No | No (deprecated) |
| **Properties** | Browser-specific | Normalized |
| **Access after handler** | ✅ Always available | ⚠️ Nullified (React 16 only) |

**Basic usage:**
```tsx
function Input() {
  const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
    // e is SyntheticEvent, not native Event
    console.log(e.target.value); // ✅ Works
    console.log(e.currentTarget); // ✅ Works
    
    // Access native event if needed
    const nativeEvent = e.nativeEvent;
  };
  
  return <input onChange={handleChange} />;
}
```

**Event pooling (React 16 only):**
```tsx
// ❌ WRONG - Async access (React 16)
function Bad() {
  const handleClick = (e: React.MouseEvent) => {
    setTimeout(() => {
      console.log(e.target); // null in React 16!
    }, 1000);
  };
  return <button onClick={handleClick}>Click</button>;
}

// ✅ CORRECT - Persist event (React 16)
function Good() {
  const handleClick = (e: React.MouseEvent) => {
    e.persist(); // Keep event alive
    setTimeout(() => {
      console.log(e.target); // ✅ Works
    }, 1000);
  };
  return <button onClick={handleClick}>Click</button>;
}

// ✅ BETTER - Extract values (React 16 & 17+)
function Better() {
  const handleClick = (e: React.MouseEvent) => {
    const target = e.target; // Capture immediately
    setTimeout(() => {
      console.log(target); // ✅ Works in all versions
    }, 1000);
  };
  return <button onClick={handleClick}>Click</button>;
}
```

**Note:** React 17+ removed event pooling, so `e.persist()` is no longer needed.

**Common event types:**
```tsx
// Mouse events
const handleClick = (e: React.MouseEvent<HTMLButtonElement>) => {};
const handleDoubleClick = (e: React.MouseEvent) => {};

// Keyboard events
const handleKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => {
  if (e.key === 'Enter') submitForm();
};

// Form events
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {};
const handleSubmit = (e: React.FormEvent<HTMLFormElement>) => {
  e.preventDefault();
};

// Focus events
const handleFocus = (e: React.FocusEvent<HTMLInputElement>) => {};
const handleBlur = (e: React.FocusEvent<HTMLInputElement>) => {};

// Clipboard events
const handleCopy = (e: React.ClipboardEvent) => {};
const handlePaste = (e: React.ClipboardEvent) => {
  const text = e.clipboardData.getData('text');
};

// Drag events
const handleDragStart = (e: React.DragEvent) => {};
const handleDrop = (e: React.DragEvent) => {
  e.preventDefault();
  const data = e.dataTransfer.getData('text');
};
```

**Accessing native event:**
```tsx
function Component() {
  const handleClick = (e: React.MouseEvent) => {
    // React synthetic event
    console.log(e.currentTarget); // Element handler is attached to
    console.log(e.target); // Element that triggered event
    
    // Native browser event
    const nativeEvent = e.nativeEvent;
    console.log(nativeEvent); // MouseEvent object
  };
  
  return <button onClick={handleClick}>Click</button>;
}
```

---

### Event Callback Composition Pattern

**Purpose:** Combine multiple event handlers into a single handler, useful for library components that accept user callbacks.

**Pattern 1: Sequential execution**
```tsx
function composeHandlers<E>(...handlers: Array<((e: E) => void) | undefined>) {
  return (event: E) => {
    handlers.forEach(handler => {
      if (handler) {
        handler(event);
      }
    });
  };
}

// Usage
function Button({ onClick, onClickInternal }: Props) {
  const handleClick = composeHandlers(onClickInternal, onClick);
  return <button onClick={handleClick}>Click</button>;
}
```

**Pattern 2: Conditional execution (stop on preventDefault)**
```tsx
function composeEventHandlers<E extends React.SyntheticEvent>(
  internalHandler?: (e: E) => void,
  externalHandler?: (e: E) => void
) {
  return (event: E) => {
    internalHandler?.(event);
    
    // If internal handler called preventDefault, stop
    if (!event.defaultPrevented) {
      externalHandler?.(event);
    }
  };
}

// Usage - XMLUI pattern
function Select({ onChange, onDidChange }: Props) {
  const handleInternalChange = (e: React.ChangeEvent<HTMLSelectElement>) => {
    // Internal logic (validation, state updates)
    validateValue(e.target.value);
    
    // Prevent external handler if validation fails
    if (!isValid) {
      e.preventDefault();
    }
  };
  
  const handleChange = composeEventHandlers(handleInternalChange, onChange);
  
  return <select onChange={handleChange}>...</select>;
}
```

**Pattern 3: Merge handlers from props**
```tsx
function mergeEventHandlers<T extends React.SyntheticEvent>(
  ours: ((e: T) => void) | undefined,
  theirs: ((e: T) => void) | undefined
): ((e: T) => void) | undefined {
  if (!ours) return theirs;
  if (!theirs) return ours;
  
  return (event: T) => {
    ours(event);
    if (!event.defaultPrevented) {
      theirs(event);
    }
  };
}

// Usage - Wrapper component
function Wrapper({ children, onClick }: Props) {
  const internalClick = (e: React.MouseEvent) => {
    console.log('Wrapper clicked');
  };
  
  return cloneElement(children, {
    onClick: mergeEventHandlers(internalClick, children.props.onClick),
  });
}
```

**XMLUI Example (Container component):**
```tsx
function Container({ children, ...props }: Props, ref: Ref<HTMLElement>) {
  const renderedChild = renderChild(children);
  
  if (ref && renderedChild && isValidElement(renderedChild)) {
    // Merge event handlers from both Container and child
    const mergedProps = {
      ...renderedChild.props,
      onClick: composeEventHandlers(props.onClick, renderedChild.props.onClick),
      onKeyDown: composeEventHandlers(props.onKeyDown, renderedChild.props.onKeyDown),
      ref: composeRefs(ref, (renderedChild as any).ref),
    };
    
    return cloneElement(renderedChild, mergedProps);
  }
  
  return renderedChild;
}
```

**Pattern 4: Callback with additional args**
```tsx
function withArgs<E, T>(
  handler: ((e: E, ...args: T[]) => void) | undefined,
  ...args: T[]
) {
  if (!handler) return undefined;
  
  return (event: E) => {
    handler(event, ...args);
  };
}

// Usage
function List({ items, onItemClick }: Props) {
  return (
    <ul>
      {items.map(item => (
        <li key={item.id} onClick={withArgs(onItemClick, item.id)}>
          {item.name}
        </li>
      ))}
    </ul>
  );
}
```

**When to compose handlers:**
- Building reusable component libraries
- Wrapper components that add behavior
- Components with internal + external handlers
- Need to call parent handler conditionally

**Best practices:**
- Always check if handler exists before calling
- Respect `preventDefault()` and `stopPropagation()`
- Execute internal handlers first
- Document composition order clearly

---

## React Lifecycle and Effect Patterns

Patterns for managing side effects, synchronization, and component lifecycle in React applications.

### `useEffect` - Side Effects and Lifecycle

**Purpose:** Run side effects after render (data fetching, subscriptions, DOM manipulation).

**Syntax:** `useEffect(() => { /* effect */ return () => { /* cleanup */ } }, [dependencies])`

**Basic usage:**
```tsx
function UserProfile({ userId }: Props) {
  const [user, setUser] = useState(null);
  
  useEffect(() => {
    // Effect runs after render
    fetch(`/api/users/${userId}`)
      .then(res => res.json())
      .then(data => setUser(data));
  }, [userId]); // Re-run when userId changes
  
  return <div>{user?.name}</div>;
}
```

**With async/await:**
```tsx
useEffect(() => {
  // Can't make callback async directly, use IIFE
  (async () => {
    const res = await fetch(`/api/users/${userId}`);
    const data = await res.json();
    setUser(data);
  })();
}, [userId]);
```

**Use when:** Data fetching, subscriptions, event listeners, DOM manipulation, integrating with non-React libraries.  
**Avoid when:** Computing derived values (use `useMemo`), handling events (use handlers), initializing state (use initializer).

---

### Effect Cleanup Pattern

**Purpose:** Properly clean up subscriptions, timers, and event listeners to prevent memory leaks.

**Pattern: Always return cleanup function for subscriptions**
```tsx
function Chat({ roomId }: Props) {
  useEffect(() => {
    const connection = createConnection(roomId);
    connection.connect();
    
    // ✅ Cleanup runs before next effect and on unmount
    return () => {
      connection.disconnect();
    };
  }, [roomId]);
  
  return <div>Connected to {roomId}</div>;
}
```

**Pattern: Cancel async operations**
```tsx
function DataComponent({ url }: Props) {
  const [data, setData] = useState(null);
  
  useEffect(() => {
    let cancelled = false;
    
    fetch(url)
      .then(res => res.json())
      .then(data => {
        if (!cancelled) setData(data);
      });
    
    // ✅ Prevent state updates after unmount
    return () => { cancelled = true; };
  }, [url]);
  
  return <div>{JSON.stringify(data)}</div>;
}
```

**Pattern: Remove event listeners**
```tsx
function WindowSize() {
  const [size, setSize] = useState({ width: 0, height: 0 });
  
  useEffect(() => {
    const handleResize = () => {
      setSize({ width: window.innerWidth, height: window.innerHeight });
    };
    
    window.addEventListener('resize', handleResize);
    handleResize(); // Initial size
    
    // ✅ Always remove listeners
    return () => window.removeEventListener('resize', handleResize);
  }, []);
  
  return <div>{size.width} x {size.height}</div>;
}
```

**Pattern: Clear timers and intervals**
```tsx
function Timer() {
  const [count, setCount] = useState(0);
  
  useEffect(() => {
    const interval = setInterval(() => {
      setCount(c => c + 1);
    }, 1000);
    
    // ✅ Clear interval on unmount
    return () => clearInterval(interval);
  }, []);
  
  return <div>{count}s</div>;
}
```

**Pattern: Unsubscribe from external stores**
```tsx
function ExternalStore({ store }: Props) {
  const [value, setValue] = useState(store.getValue());
  
  useEffect(() => {
    const unsubscribe = store.subscribe(newValue => {
      setValue(newValue);
    });
    
    // ✅ Unsubscribe when component unmounts
    return unsubscribe;
  }, [store]);
  
  return <div>{value}</div>;
}
```

**Critical rules:**
- Return cleanup for subscriptions, listeners, timers
- Use cancellation flags for async operations
- Cleanup runs before next effect and on unmount
- Don't forget to remove event listeners

---

### Effect Dependencies Pattern

**Purpose:** Correctly manage dependency arrays to avoid stale closures and unnecessary re-runs.

**Anti-pattern: Missing dependencies (stale closure bug)**
```tsx
// ❌ WRONG - Stale closure
function Counter() {
  const [count, setCount] = useState(0);
  
  useEffect(() => {
    setInterval(() => {
      console.log(count); // Always logs 0!
    }, 1000);
  }, []); // Missing count
  
  return <button onClick={() => setCount(count + 1)}>Increment</button>;
}

// ✅ CORRECT - Use functional update
useEffect(() => {
  setInterval(() => {
    setCount(c => c + 1); // Uses latest value
  }, 1000);
}, []); // No dependencies needed
```

**Anti-pattern: Object/array in dependencies**
```tsx
// ❌ WRONG - Object recreated every render
function Component({ config }: Props) {
  useEffect(() => {
    fetchData(config);
  }, [config]); // Runs every render if config is new object
}

// ✅ CORRECT - Destructure primitive values
useEffect(() => {
  fetchData(config);
}, [config.id, config.filter]); // Only re-run when these change
```

**Pattern: Empty array = run once on mount**
```tsx
useEffect(() => {
  // Initialization logic
  initializeApp();
  
  return () => {
    // Cleanup on unmount
    cleanupApp();
  };
}, []); // Runs once on mount, cleanup on unmount
```

**Pattern: No array = run after every render**
```tsx
useEffect(() => {
  // Runs after every render (rarely needed)
  updateDocumentTitle(`Page - ${count}`);
}); // No dependency array
```

**Pattern: Avoid callback dependencies with useRef**
```tsx
function Component({ callback }: Props) {
  const callbackRef = useRef(callback);
  
  // Keep ref updated
  useEffect(() => {
    callbackRef.current = callback;
  }, [callback]);
  
  useEffect(() => {
    const interval = setInterval(() => {
      // ✅ Always uses latest callback
      callbackRef.current();
    }, 1000);
    
    return () => clearInterval(interval);
  }, []); // No callback in deps
}
```

**Common mistakes:**
```tsx
// ❌ Wrong - Missing dependencies
useEffect(() => {
  doSomething(prop); // prop not in deps
}, []);

// ❌ Wrong - Object identity
useEffect(() => {
  fetchData(user);
}, [user]); // user object changes every render

// ❌ Wrong - Function identity
useEffect(() => {
  callback();
}, [callback]); // callback recreated every render

// ✅ Correct - Destructure objects
useEffect(() => {
  fetchData({ id: user.id, name: user.name });
}, [user.id, user.name]);

// ✅ Correct - Memoize callbacks
const memoizedCallback = useCallback(callback, [dep]);
useEffect(() => {
  memoizedCallback();
}, [memoizedCallback]);
```

**ESLint rule:** Always enable `react-hooks/exhaustive-deps` to catch dependency issues.

---

### Layout Effect Pattern

**Purpose:** Run effects synchronously after DOM mutations but before browser paint to prevent visual flickering.

**When to use `useLayoutEffect`:**

**Pattern: DOM measurements before paint**
```tsx
function Tooltip() {
  const [position, setPosition] = useState({ x: 0, y: 0 });
  const ref = useRef<HTMLDivElement>(null);
  
  // ✅ CORRECT - Measure before paint
  useLayoutEffect(() => {
    if (ref.current) {
      const rect = ref.current.getBoundingClientRect();
      setPosition({
        x: rect.left + rect.width / 2,
        y: rect.top - 10,
      });
    }
  }, []);
  
  return <div ref={ref} style={{ left: position.x, top: position.y }}>Tooltip</div>;
}

// ❌ WRONG - useEffect causes visible flicker
useEffect(() => {
  // DOM measurements happen AFTER paint
  // User sees element jump from old to new position
}, []);
```

**Pattern: Synchronize scroll position**
```tsx
function ScrollSync({ targetRef }: Props) {
  useLayoutEffect(() => {
    if (targetRef.current) {
      // ✅ Scroll before paint, no flicker
      targetRef.current.scrollTop = savedPosition;
    }
  }, [targetRef, savedPosition]);
}
```

**Pattern: Prevent layout shift**
```tsx
function AutoResizeTextarea({ value }: Props) {
  const ref = useRef<HTMLTextAreaElement>(null);
  
  useLayoutEffect(() => {
    if (ref.current) {
      // ✅ Adjust height before paint
      ref.current.style.height = 'auto';
      ref.current.style.height = `${ref.current.scrollHeight}px`;
    }
  }, [value]);
  
  return <textarea ref={ref} value={value} />;
}
```

**Pattern: Third-party DOM library integration**
```tsx
function Chart({ data }: Props) {
  const containerRef = useRef<HTMLDivElement>(null);
  
  useLayoutEffect(() => {
    if (containerRef.current) {
      // ✅ Initialize library before paint
      const chart = new ChartLibrary(containerRef.current);
      chart.render(data);
      
      return () => chart.destroy();
    }
  }, [data]);
  
  return <div ref={containerRef} />;
}
```

**Comparison: useEffect vs useLayoutEffect**

| Aspect | `useEffect` | `useLayoutEffect` |
|--------|------------|-------------------|
| **Timing** | After paint (async) | Before paint (sync) |
| **Blocks rendering** | ❌ No | ✅ Yes |
| **Use for** | Data fetching, subscriptions | DOM measurements, preventing flicker |
| **Performance** | Better (non-blocking) | Worse (blocks paint) |
| **SSR** | ✅ Works | ⚠️ Warning (no DOM on server) |

**When to use each:**
- **useEffect**: 99% of cases - data fetching, subscriptions, analytics
- **useLayoutEffect**: DOM measurements, scroll sync, preventing visual flicker

**Warning:** `useLayoutEffect` blocks visual updates. Only use when you need synchronous DOM access before paint.

**SSR consideration:**
```tsx
// ⚠️ useLayoutEffect doesn't run on server
useLayoutEffect(() => {
  // This code only runs in browser
  measureDOM();
}, []);

// ✅ Better: Use useEffect for SSR-compatible code
useEffect(() => {
  measureDOM();
}, []);
```

---

### Insertion Effect Pattern

**Purpose:** Insert styles into DOM before layout effects run. Used by CSS-in-JS libraries.

**Syntax:** `useInsertionEffect(() => { /* insert styles */ }, [dependencies])`

**Effect execution order:**
1. `useInsertionEffect` - Insert styles
2. `useLayoutEffect` - Measure layout (reads styles)
3. Browser paints
4. `useEffect` - Other side effects

**Pattern: CSS-in-JS style injection**
```tsx
function useCSS(rule: string) {
  useInsertionEffect(() => {
    // ✅ Inject styles before layout reads
    const style = document.createElement('style');
    style.textContent = rule;
    document.head.appendChild(style);
    
    return () => document.head.removeChild(style);
  }, [rule]);
}

// Usage
function Button({ color }: Props) {
  useCSS(`
    .button-${color} {
      background: ${color};
      border: 1px solid ${darken(color)};
    }
  `);
  
  return <button className={`button-${color}`}>Click</button>;
}
```

**Pattern: Dynamic theme injection (XMLUI)**
```tsx
function ThemeProvider({ theme, children }: Props) {
  useInsertionEffect(() => {
    // ✅ Insert theme CSS before components measure
    const styleElement = document.createElement('style');
    styleElement.id = 'theme-styles';
    styleElement.textContent = generateThemeCSS(theme);
    document.head.appendChild(styleElement);
    
    return () => {
      document.getElementById('theme-styles')?.remove();
    };
  }, [theme]);
  
  return children;
}
```

**Pattern: Critical CSS injection**
```tsx
function useCriticalCSS(css: string) {
  useInsertionEffect(() => {
    // ✅ Inject before any layout calculations
    const style = document.createElement('style');
    style.setAttribute('data-critical', 'true');
    style.textContent = css;
    document.head.insertBefore(style, document.head.firstChild);
    
    return () => style.remove();
  }, [css]);
}
```

**When to use:**
- Building CSS-in-JS libraries
- Dynamic style generation
- Theme system implementation
- Critical CSS injection

**When NOT to use:**
- Regular application code (use `useEffect`)
- Static stylesheets (use `<link>` tags)
- Non-style DOM manipulation

**Note:** Rarely used directly in application code. Primarily for library authors. XMLUI uses this in `StyleContext` for theme style injection.

**Comparison with other effects:**

```tsx
// ❌ WRONG - useEffect runs too late
useEffect(() => {
  injectStyles(); // Styles added after layout measured
}, []);

// ❌ WRONG - useLayoutEffect causes double layout
useLayoutEffect(() => {
  injectStyles(); // Layout measured, then styles added, then re-measured
}, []);

// ✅ CORRECT - useInsertionEffect runs first
useInsertionEffect(() => {
  injectStyles(); // Styles ready before any layout measurement
}, []);
```

---

### Effect Best Practices Summary

**1. Always clean up:**
```tsx
useEffect(() => {
  const subscription = subscribe();
  return () => subscription.unsubscribe(); // ✅ Cleanup
}, []);
```

**2. Handle dependencies correctly:**
```tsx
// ✅ Include all dependencies
useEffect(() => {
  doSomething(prop, state);
}, [prop, state]);

// ✅ Or use functional updates
useEffect(() => {
  setState(prev => prev + 1);
}, []); // No state dependency needed
```

**3. Choose the right effect hook:**
- `useEffect` - Default choice (async, after paint)
- `useLayoutEffect` - DOM measurements, prevent flicker (sync, before paint)
- `useInsertionEffect` - CSS-in-JS only (before layout)

**4. Avoid common pitfalls:**
```tsx
// ❌ Don't use objects in deps
useEffect(() => {}, [config]); // Runs every render

// ✅ Destructure primitive values
useEffect(() => {}, [config.id, config.name]);

// ❌ Don't make effect callback async
useEffect(async () => {}, []); // Type error

// ✅ Use IIFE for async
useEffect(() => {
  (async () => await fetch())();
}, []);
```

**5. Profile before optimizing:**
- Most effects are cheap
- Don't prematurely optimize with `useLayoutEffect`
- Measure actual performance impact

---

## `useRef` - Persistent Mutable References

**Purpose:** Store mutable values that persist across renders without triggering re-renders.

**Syntax:** `const ref = useRef(initialValue)`

### DOM References

```tsx
function TextInput() {
  const inputRef = useRef<HTMLInputElement>(null);
  
  const focusInput = () => {
    inputRef.current?.focus();
  };
  
  return (
    <div>
      <input ref={inputRef} />
      <button onClick={focusInput}>Focus</button>
    </div>
  );
}
```

### Storing Mutable Values

```tsx
function Timer() {
  const [count, setCount] = useState(0);
  const intervalRef = useRef<NodeJS.Timeout>();
  
  useEffect(() => {
    intervalRef.current = setInterval(() => {
      setCount(c => c + 1);
    }, 1000);
    
    return () => clearInterval(intervalRef.current);
  }, []);
  
  const stop = () => {
    clearInterval(intervalRef.current);
  };
  
  return (
    <div>
      {count} seconds
      <button onClick={stop}>Stop</button>
    </div>
  );
}
```

### Avoiding Stale Closures

```tsx
function Component({ callback }: Props) {
  const callbackRef = useRef(callback);
  
  // Keep ref updated with latest callback
  useEffect(() => {
    callbackRef.current = callback;
  }, [callback]);
  
  useEffect(() => {
    const interval = setInterval(() => {
      // Always uses latest callback
      callbackRef.current();
    }, 1000);
    
    return () => clearInterval(interval);
  }, []); // No callback dependency needed
  
  return <div>Running...</div>;
}
```

### Common Patterns in XMLUI

**Previous Value Tracking:**
```tsx
function usePrevious<T>(value: T): T | undefined {
  const ref = useRef<T>();
  
  useEffect(() => {
    ref.current = value;
  }, [value]);
  
  return ref.current;
}

function Component({ count }: Props) {
  const prevCount = usePrevious(count);
  
  return <div>Now: {count}, Before: {prevCount}</div>;
}
```

### Key Differences: useState vs useRef

| Feature | `useState` | `useRef` |
|---------|-----------|---------|
| Triggers re-render | ✅ Yes | ❌ No |
| Persists across renders | ✅ Yes | ✅ Yes |
| Use for UI state | ✅ Yes | ❌ No |
| Use for DOM access | ❌ No | ✅ Yes |
| Use for mutable timers/intervals | ❌ No | ✅ Yes |

---

## `useId` - Unique ID Generation

**Purpose:** Generate stable unique IDs for accessibility attributes.

**Syntax:** `const id = useId()`

### Basic Usage

```tsx
function FormField({ label }: Props) {
  const id = useId();
  
  return (
    <div>
      <label htmlFor={id}>{label}</label>
      <input id={id} />
    </div>
  );
}
```

### Multiple IDs

```tsx
function ComplexForm() {
  const id = useId();
  
  return (
    <div>
      <label htmlFor={`${id}-name`}>Name</label>
      <input id={`${id}-name`} aria-describedby={`${id}-name-hint`} />
      <span id={`${id}-name-hint`}>Enter your full name</span>
      
      <label htmlFor={`${id}-email`}>Email</label>
      <input id={`${id}-email`} />
    </div>
  );
}
```

**Why not just use a counter?** `useId` generates IDs that are stable across server and client rendering, preventing hydration mismatches.

---

## `forwardRef` - Ref Forwarding to Child Components

**Purpose:** Allow parent components to access DOM nodes or component instances of child components by forwarding refs through component boundaries.

**Syntax:** `const Component = forwardRef((props, ref) => { ... })`

### Basic Usage

```tsx
const TextInput = forwardRef<HTMLInputElement, Props>(
  function TextInput({ label, ...props }, forwardedRef) {
    return (
      <div>
        <label>{label}</label>
        <input ref={forwardedRef} {...props} />
      </div>
    );
  }
);

// Parent can now access the input element
function Form() {
  const inputRef = useRef<HTMLInputElement>(null);
  
  const focusInput = () => {
    inputRef.current?.focus();
  };
  
  return (
    <div>
      <TextInput ref={inputRef} label="Name" />
      <button onClick={focusInput}>Focus Input</button>
    </div>
  );
}
```

### TypeScript Generic Syntax

```tsx
// Explicitly type both the ref and props
const Component = forwardRef<RefType, PropsType>(
  function Component(props, ref) {
    return <div ref={ref}>...</div>;
  }
);

// Example with HTMLDivElement
const Card = forwardRef<HTMLDivElement, CardProps>(
  function Card({ children, className }, ref) {
    return (
      <div ref={ref} className={className}>
        {children}
      </div>
    );
  }
);
```

**Why explicit typing matters:**

Without generic syntax, TypeScript infers types from the function signature, which can lead to several issues:

```tsx
// ❌ WRONG - Without explicit generics
const Input = forwardRef(function Input(props: Props, ref) {
  // TypeScript infers ref as: ForwardedRef<unknown>
  // This means:
  // 1. No autocomplete for ref.current properties
  // 2. No type checking when assigning ref to JSX elements
  // 3. Parent components can pass wrong ref type without errors
  return <input ref={ref} />; // Type error: ref might not be compatible!
});

// Usage - TypeScript won't catch this error:
const divRef = useRef<HTMLDivElement>(null);
<Input ref={divRef} /> // Should error but doesn't - expecting HTMLInputElement!

// ✅ CORRECT - With explicit generics
const Input = forwardRef<HTMLInputElement, Props>(
  function Input(props, ref) {
    // TypeScript knows ref is: ForwardedRef<HTMLInputElement>
    // Benefits:
    // 1. Autocomplete works: ref.current?.focus()
    // 2. Type checking ensures ref matches JSX element
    // 3. Parent must pass correct ref type
    return <input ref={ref} />; // Type safe!
  }
);

// Usage - TypeScript catches the error:
const divRef = useRef<HTMLDivElement>(null);
<Input ref={divRef} /> // ❌ Type error: expected RefObject<HTMLInputElement>
```

**Key problems without explicit generics:**
1. **Loss of type safety** - Parent can pass incompatible ref types
2. **No IntelliSense** - No autocomplete for `ref.current` properties
3. **Runtime errors** - Type mismatches only discovered at runtime
4. **Harder refactoring** - Changes to ref type don't propagate to consumers

**Best practice:** Always specify both generic parameters explicitly in XMLUI components.

### Composing Multiple Refs

**The Problem:** Components often need to manage multiple refs pointing to the same DOM element:
1. **Internal ref** - Component's own logic (measurements, animations, focus)
2. **Forwarded ref** - Parent needs access to the DOM element
3. **Third-party refs** - Integration with libraries (Popper, Radix UI, etc.)

**The Solution:** Use `composeRefs` from `@radix-ui/react-compose-refs` to merge multiple refs into one.

**Why you need to compose refs:**

| Use Case | Example | Reason |
|----------|---------|--------|
| **Internal logic + parent access** | Auto-resize textarea | Component measures scrollHeight, parent needs focus() |
| **Library integration** | Popover/Tooltip | Popper needs ref for positioning, parent needs ref for control |
| **Wrapper components** | Container with single child | Parent ref applies to child, child has own ref |
| **Multiple behaviors** | Draggable element | Drag library needs ref, resize observer needs ref, parent needs ref |

**Key differences: Inner vs Forwarded refs:**

| Aspect | Inner Ref | Forwarded Ref |
|--------|-----------|---------------|
| **Created by** | Component itself with `useRef()` | Parent component |
| **Purpose** | Internal component logic | Parent needs DOM access |
| **Type** | Always `RefObject<T>` | Can be `RefObject<T>`, `RefCallback<T>`, or `null` |
| **Guaranteed to exist** | Yes - always has `.current` property | No - parent might not pass a ref |
| **When to use** | Component needs DOM access for its own behavior | Expose DOM element to parent |

**Example 1: Internal + Forwarded (most common in XMLUI):**

```tsx
import { composeRefs } from "@radix-ui/react-compose-refs";

function TextArea({ value, onChange }: Props, forwardedRef: Ref<HTMLTextAreaElement>) {
  // Inner ref: Component creates and owns this for auto-resize logic
  const innerRef = useRef<HTMLTextAreaElement>(null);
  
  // Compose both refs - textarea element needs both
  const composedRef = forwardedRef 
    ? composeRefs(innerRef, forwardedRef) 
    : innerRef;
  
  useEffect(() => {
    // ✅ CORRECT: Use innerRef for internal logic
    // It's guaranteed to exist and have .current property
    if (innerRef.current) {
      innerRef.current.style.height = 'auto';
      innerRef.current.style.height = `${innerRef.current.scrollHeight}px`;
    }
    
    // ❌ WRONG: Don't use forwardedRef directly
    // if (forwardedRef?.current) { ... } // Type error: Ref<T> might be a callback!
  }, [value]);
  
  return <textarea ref={composedRef} value={value} onChange={onChange} />;
}

export const AutoResizeTextArea = forwardRef(TextArea);

// Parent usage:
function Form() {
  const textareaRef = useRef<HTMLTextAreaElement>(null);
  
  const focusTextarea = () => {
    textareaRef.current?.focus(); // Parent can access via forwarded ref
  };
  
  return <AutoResizeTextArea ref={textareaRef} />; // Component auto-resizes via inner ref
}
```

**Example 2: Library Integration (Popper + Forwarded):**

```tsx
function Select({ options }: Props, forwardedRef: Ref<HTMLButtonElement>) {
  // Popper library needs a ref for positioning
  const [referenceElement, setReferenceElement] = useState<HTMLButtonElement | null>(null);
  
  // Compose library ref setter with forwarded ref
  const composedRef = forwardedRef 
    ? composeRefs(setReferenceElement, forwardedRef) 
    : setReferenceElement;
  
  return (
    <>
      <button ref={composedRef}>Select</button>
      <Popper referenceElement={referenceElement}>
        {/* Dropdown content */}
      </Popper>
    </>
  );
}

export const SelectComponent = forwardRef(Select);
```

**Example 3: Wrapper Component (Parent + Child refs):**

```tsx
function Container({ children }: Props, ref: Ref<HTMLElement>) {
  const renderedChild = renderChild(children);
  
  // If single child, compose parent's ref with child's existing ref
  if (isValidElement(renderedChild)) {
    return cloneElement(renderedChild, {
      ref: composeRefs(ref, (renderedChild as any).ref),
    });
  }
  
  return renderedChild;
}

export const ContainerComponent = forwardRef(Container);
```

**Example 4: Multiple Behaviors (Drag + Resize + Forward):**

```tsx
function DraggablePanel(props: Props, forwardedRef: Ref<HTMLDivElement>) {
  const dragRef = useRef<HTMLDivElement>(null);
  const resizeObserverRef = useRef<HTMLDivElement>(null);
  
  // Compose all three refs
  const composedRef = composeRefs(
    dragRef,
    resizeObserverRef,
    forwardedRef || null
  );
  
  useDragLogic(dragRef);
  useResizeObserver(resizeObserverRef);
  
  return <div ref={composedRef}>Draggable and resizable</div>;
}
```

**When to compose refs:**
- Component needs internal ref AND parent needs access
- Integrating with libraries that require refs (Popper, React DnD, etc.)
- Wrapping components that need to forward refs to children
- Multiple hooks/effects need refs to the same element

**How `composeRefs` works:**
- Accepts multiple refs (RefObjects, callbacks, or null)
- Returns a single callback ref that updates all provided refs
- Handles both RefObject (sets `.current`) and callback refs (calls function)
- Safely ignores `null`/`undefined` refs

### When to Use forwardRef

**Use `forwardRef` when:**
- Building reusable components that wrap DOM elements
- Parent needs direct DOM access (focus, scroll, measurements)
- Integrating with third-party libraries requiring refs
- Creating form components that need imperative control

**Don't use `forwardRef` when:**
- Component doesn't wrap a single DOM element
- Refs aren't needed by parent components
- You can solve the problem with callbacks/props instead

### Common Mistakes

```tsx
// ❌ WRONG - Forgetting to attach ref to DOM element
const Bad = forwardRef((props, ref) => {
  return <div>{props.children}</div>; // ref is ignored!
});

// ✅ CORRECT - Always attach ref to actual DOM element
const Good = forwardRef((props, ref) => {
  return <div ref={ref}>{props.children}</div>;
});

// ❌ WRONG - Attaching ref to component (won't work)
const AlsoBad = forwardRef((props, ref) => {
  return <CustomComponent ref={ref} />; // CustomComponent must also use forwardRef
});

// ✅ CORRECT - Forward through nested components
const CustomComponent = forwardRef((props, ref) => {
  return <div ref={ref}>...</div>;
});

const AlsoGood = forwardRef((props, ref) => {
  return <CustomComponent ref={ref} />; // Works because CustomComponent forwards
});
```

---

## `createPortal` - Render Outside Hierarchy

**Purpose:** Render children into a DOM node outside the parent component's hierarchy.

**Syntax:** `createPortal(children, domNode, key?)`

### Basic Usage

```tsx
import { createPortal } from 'react-dom';

function Modal({ isOpen, children }: Props) {
  if (!isOpen) return null;
  
  // Render into document.body instead of parent component
  return createPortal(
    <div className="modal-overlay">
      <div className="modal-content">
        {children}
      </div>
    </div>,
    document.body
  );
}

// Usage
function App() {
  return (
    <div className="app">
      <Modal isOpen={true}>
        <h1>This renders in document.body, not .app!</h1>
      </Modal>
    </div>
  );
}
```

### Common Use Cases

**1. Tooltips/Popovers (avoid z-index issues):**
```tsx
function Tooltip({ targetRef, content }: Props) {
  return createPortal(
    <div className="tooltip" style={calculatePosition(targetRef)}>
      {content}
    </div>,
    document.body
  );
}
```

**2. Notifications/Toasts:**
```tsx
function NotificationToast() {
  const [shouldRender, setShouldRender] = useState(false);
  
  useEffect(() => {
    setShouldRender(true);
  }, []);
  
  if (!shouldRender) return null;
  
  return createPortal(
    <Toaster position="top-right">
      {(t) => <ToastBar toast={t} />}
    </Toaster>,
    document.body
  );
}
```

**3. Modal Dialogs:**
```tsx
function ModalDialog({ isOpen, children }: Props) {
  if (!isOpen) return null;
  
  return createPortal(
    <div className="modal-backdrop">
      <div className="modal-dialog">
        {children}
      </div>
    </div>,
    document.getElementById('modal-root') || document.body
  );
}
```

**4. Full-Screen Overlays:**
```tsx
function FullScreenOverlay({ show, children }: Props) {
  if (!show) return null;
  
  return createPortal(
    <div className="fullscreen-overlay">
      {children}
    </div>,
    document.body
  );
}
```

### Event Bubbling Still Works

```tsx
// Event bubbling works despite DOM hierarchy
function Parent() {
  const handleClick = () => {
    console.log('Clicked!'); // This fires even though button is portaled
  };
  
  return (
    <div onClick={handleClick}>
      <PortaledButton />
    </div>
  );
}

function PortaledButton() {
  return createPortal(
    <button>Click me</button>,
    document.body
  );
}
```

### Common Pattern in XMLUI

```tsx
// App component portals theme styles
function App({ children }: Props) {
  return (
    <>
      {children}
      {createPortal(
        <style>{themeCSS}</style>,
        document.head
      )}
    </>
  );
}

// Inspector portals debugging UI
function Inspector() {
  return createPortal(
    <div className="inspector-panel">
      {/* Debug tools */}
    </div>,
    document.body
  );
}
```

### When to Use createPortal

**Use `createPortal` when:**
- Modals, dialogs, and overlays
- Tooltips and popovers
- Notifications and toasts
- Avoiding parent overflow/z-index issues
- Rendering into different parts of DOM (head, body)

**Don't use when:**
- Normal component rendering is sufficient
- No CSS stacking or overflow issues
- Adds unnecessary complexity

---

## `Fragment` - Grouping Without DOM Nodes

**Purpose:** Group multiple elements without adding extra nodes to the DOM.

**Syntax:** `<Fragment>...</Fragment>` or `<>...</>`

### Basic Usage

```tsx
// ❌ WRONG - Adds unnecessary div wrapper
function List() {
  return (
    <div>
      <li>Item 1</li>
      <li>Item 2</li>
    </div>
  );
}

// ✅ CORRECT - No extra DOM node
function List() {
  return (
    <>
      <li>Item 1</li>
      <li>Item 2</li>
    </>
  );
}
```

### Short vs Long Syntax

```tsx
// Short syntax <> - Use for most cases
function Component() {
  return (
    <>
      <Header />
      <Content />
    </>
  );
}

// Long syntax <Fragment> - Required when you need a key
function List({ items }: Props) {
  return (
    <ul>
      {items.map(item => (
        <Fragment key={item.id}>
          <li>{item.name}</li>
          <li>{item.description}</li>
        </Fragment>
      ))}
    </ul>
  );
}
```

**Limitations of short syntax:**
- ❌ Cannot add `key` prop (use `<Fragment key={...}>` instead)
- ❌ Cannot add any other props (only `key` is allowed on Fragment)
- ✅ Use short syntax everywhere else (cleaner, less verbose)

### Common Use Cases

**1. Returning Multiple Elements:**
```tsx
function Header() {
  return (
    <>
      <h1>Title</h1>
      <nav>Navigation</nav>
    </>
  );
}
```

**2. Conditional Rendering:**
```tsx
function Component({ showExtra }: Props) {
  return (
    <div>
      <h1>Always shown</h1>
      {showExtra && (
        <>
          <p>Extra content</p>
          <button>Extra button</button>
        </>
      )}
    </div>
  );
}
```

**3. Table Rows:**
```tsx
function TableRows({ data }: Props) {
  return (
    <>
      {data.map(row => (
        <Fragment key={row.id}>
          <tr>
            <td>{row.name}</td>
            <td>{row.value}</td>
          </tr>
          {row.hasDetails && (
            <tr>
              <td colSpan={2}>{row.details}</td>
            </tr>
          )}
        </Fragment>
      ))}
    </>
  );
}
```

**4. Avoiding Invalid HTML:**
```tsx
// ❌ WRONG - div inside p is invalid HTML
function Text() {
  return (
    <p>
      <div>This is invalid!</div>
    </p>
  );
}

// ✅ CORRECT - Fragment doesn't create DOM node
function Text() {
  return (
    <p>
      <>
        <span>This is valid!</span>
      </>
    </p>
  );
}
```

### When to Use Fragment

**Use `Fragment` when:**
- Component must return multiple elements
- Avoiding wrapper divs that break CSS (flexbox, grid)
- Keeping HTML semantically valid
- Conditional rendering of multiple elements

**Don't use when:**
- Single element (no need to wrap)
- Wrapper div doesn't cause issues
- Need to attach events or refs (Fragment can't have them)

---

## `cloneElement` - Clone and Modify React Elements

**Purpose:** Clone a React element and override its props, refs, or children.

**Syntax:** `cloneElement(element, props?, ...children?)`

### Basic Usage

```tsx
import { cloneElement, isValidElement } from 'react';

function Container({ children }: Props) {
  if (!isValidElement(children)) {
    return children;
  }
  
  // Clone child and add extra props
  return cloneElement(children, {
    className: 'container-child',
    style: { padding: '10px' },
  });
}

// Usage
<Container>
  <div>Original</div> {/* Becomes <div className="container-child" style={{padding: '10px'}}>Original</div> */}
</Container>
```

### Adding Props to Children

```tsx
function Animation({ children, duration = 300 }: Props) {
  if (!isValidElement(children)) {
    return children;
  }
  
  // Add animation props to child
  return cloneElement(children, {
    style: {
      ...children.props.style,
      transition: `all ${duration}ms`,
    },
  });
}
```

### Forwarding Refs Through Clone

```tsx
function Wrapper({ children, ...rest }: Props, forwardedRef: Ref<any>) {
  if (!isValidElement(children)) {
    return children;
  }
  
  // Clone and forward ref + other props
  return cloneElement(children, {
    ...rest,
    ref: forwardedRef,
  });
}

export const WrapperComponent = forwardRef(Wrapper);
```

### Common Pattern in XMLUI

**Container with single child ref forwarding:**
```tsx
function Container({ children }: Props, ref: Ref<HTMLElement>) {
  const renderedChild = renderChild(children);
  
  // If single valid child, compose refs and merge props
  if (ref && renderedChild && isValidElement(renderedChild)) {
    return cloneElement(renderedChild, {
      ref: composeRefs(ref, (renderedChild as any).ref),
      ...mergeProps(renderedChild.props, rest),
    });
  }
  
  return renderedChild;
}
```

**Form field with label integration:**
```tsx
function ItemWithLabel({ children, label }: Props) {
  const id = useId();
  
  return (
    <div>
      <label htmlFor={id}>{label}</label>
      {cloneElement(children as ReactElement, {
        id,
        'aria-labelledby': id,
      })}
    </div>
  );
}
```

### When to Use cloneElement

**Use `cloneElement` when:**
- Wrapping components need to add props to children
- Forwarding refs through wrapper components
- Adding common behavior to arbitrary children
- Integrating with child elements you don't control

**Don't use when:**
- You can pass props directly (prefer explicit props)
- You need to modify deeply nested children (use context instead)
- Children are not React elements (check with `isValidElement` first)

### Common Mistakes

```tsx
// ❌ WRONG - Not checking if child is valid element
function Bad({ children }: Props) {
  return cloneElement(children, { className: 'bad' }); // Crashes if children is string/number
}

// ✅ CORRECT - Always validate first (see isValidElement section)
function Good({ children }: Props) {
  if (!isValidElement(children)) {
    return children;
  }
  return cloneElement(children, { className: 'good' });
}

// ❌ WRONG - Overriding all existing props
return cloneElement(child, { style: { color: 'red' } }); // Loses child's existing style

// ✅ CORRECT - Merge with existing props
return cloneElement(child, {
  style: { ...child.props.style, color: 'red' },
});
```

---

## `isValidElement` - Type Check for React Elements

**Purpose:** Check if a value is a valid React element (created with JSX or `createElement`). Always use before `cloneElement`.

**Syntax:** `isValidElement(value)`

### Basic Usage

```tsx
import { isValidElement } from 'react';

function processChild(child: React.ReactNode) {
  // child could be anything: string, number, element, array, etc.
  
  if (isValidElement(child)) {
    // TypeScript now knows child is ReactElement
    console.log(child.props); // ✅ OK - access props safely
    console.log(child.type);  // ✅ OK - access type safely
    return child;
  }
  
  // Not an element - return as is
  return child;
}
```

### Common Pattern in XMLUI

**Conditional element wrapping:**
```tsx
function ConditionalWrapper({ condition, children }: Props) {
  if (!condition) {
    return children;
  }
  
  // Only wrap if child is valid element
  return isValidElement(children) 
    ? <div className="wrapper">{children}</div>
    : children;
}
```

### What isValidElement Checks

```tsx
isValidElement(<div />);              // ✅ true - JSX element
isValidElement(React.createElement('div')); // ✅ true - created element
isValidElement(<Component />);        // ✅ true - component element
isValidElement('hello');              // ❌ false - string
isValidElement(123);                  // ❌ false - number
isValidElement(null);                 // ❌ false - null
isValidElement(undefined);            // ❌ false - undefined
isValidElement([<div key="1" />]);    // ❌ false - array of elements
```

### When to Use isValidElement

**Use `isValidElement` when:**
- Before calling `cloneElement` (required to avoid crashes)
- Type narrowing for TypeScript (ReactNode → ReactElement)
- Validating `children` prop type
- Conditional element manipulation

**Note:** See `cloneElement` section for examples of using these two functions together.

---

## `flushSync` - Synchronous State Updates

**Purpose:** Force React to flush state updates synchronously, bypassing automatic batching.

**Syntax:** `flushSync(() => { /* state updates */ })`

**Warning:** Use sparingly - breaks React's batching optimization and can hurt performance.

### Basic Usage

```tsx
import { flushSync } from 'react-dom';

function Form() {
  const [value, setValue] = useState('');
  
  const handleSubmit = () => {
    // Normal: state updates are batched
    setValue('');
    setError(null);
    // Both updates happen together
    
    // With flushSync: update happens immediately
    flushSync(() => {
      setValue('');
    });
    // DOM is updated here, before next line
    inputRef.current?.focus();
  };
}
```

### When DOM Must Update Immediately

```tsx
function Table({ data }: Props) {
  const [selectedRow, setSelectedRow] = useState(0);
  const rowRef = useRef<HTMLTableRowElement>(null);
  
  const selectRow = (index: number) => {
    // Must update DOM before scrolling
    flushSync(() => {
      setSelectedRow(index);
    });
    
    // DOM is updated, can now scroll
    rowRef.current?.scrollIntoView();
  };
}
```

### Common Pattern in XMLUI

**Form reset with focus:**
```tsx
function Form({ onSubmit }: Props) {
  const doReset = () => {
    // Reset all fields
  };
  
  const handleSuccess = () => {
    const prevFocused = document.activeElement;
    
    // Force synchronous reset before restoring focus
    flushSync(() => {
      doReset();
    });
    
    // DOM is reset, restore focus
    if (prevFocused && typeof (prevFocused as HTMLElement).focus === 'function') {
      (prevFocused as HTMLElement).focus();
    }
  };
}
```

**Table with immediate scroll:**
```tsx
function DataTable({ data }: Props) {
  const handleSort = (column: string) => {
    // Update sort synchronously before scrolling
    flushSync(() => {
      setSortColumn(column);
      setSortedData(sortData(data, column));
    });
    
    // Table is re-rendered, can scroll to top
    tableRef.current?.scrollTo(0, 0);
  };
}
```

### Why flushSync Exists

```tsx
// ❌ Problem: Without flushSync
function Component() {
  const [text, setText] = useState('');
  
  const update = () => {
    setText('new value');
    // DOM not updated yet!
    inputRef.current?.focus(); // Focuses old state
  };
}

// ✅ Solution: With flushSync
function Component() {
  const [text, setText] = useState('');
  
  const update = () => {
    flushSync(() => {
      setText('new value');
    });
    // DOM is updated
    inputRef.current?.focus(); // Focuses new state
  };
}
```

### When to Use flushSync

**Use `flushSync` when:**
- Need DOM measurements after state change
- Synchronizing with third-party libraries
- Scrolling after state update
- Focus management after state change

**Don't use when:**
- Normal state updates (let React batch)
- Performance-critical code paths
- You can solve it with `useLayoutEffect`
- Inside render (not allowed)

**Performance impact:**
```tsx
// ❌ BAD - Multiple flushSync calls
data.forEach(item => {
  flushSync(() => {
    processItem(item); // Forces re-render each time
  });
});

// ✅ GOOD - Single batch update
const processedItems = data.map(processItem);
flushSync(() => {
  setItems(processedItems); // Single re-render
});
```

---

## `createRoot` - React 18 Root API

**Purpose:** Create a root to render React components into a DOM container (React 18+).

**Syntax:** `const root = createRoot(container); root.render(<App />)`

### Basic Usage

```tsx
import { createRoot } from 'react-dom/client';

// Old way (React 17)
ReactDOM.render(<App />, document.getElementById('root'));

// New way (React 18+)
const root = createRoot(document.getElementById('root')!);
root.render(<App />);
```

### With TypeScript

```tsx
import { createRoot } from 'react-dom/client';

const container = document.getElementById('root');
if (!container) {
  throw new Error('Root element not found');
}

const root = createRoot(container);
root.render(<App />);
```

### Unmounting

```tsx
const root = createRoot(container);
root.render(<App />);

// Later: unmount
root.unmount();
```

### Common Pattern in XMLUI

**Standalone app rendering:**
```tsx
function renderStandaloneApp(rootElement: HTMLElement) {
  let contentRoot: Root;
  
  if (!contentRoot) {
    contentRoot = createRoot(rootElement);
  }
  
  contentRoot.render(
    <StrictMode>
      <App />
    </StrictMode>
  );
  
  return contentRoot;
}
```

**Shadow DOM rendering:**
```tsx
function NestedApp({ children }: Props) {
  const shadowRef = useRef<ShadowRoot>(null);
  const contentRootRef = useRef<Root | null>(null);
  
  useEffect(() => {
    if (shadowRef.current && !contentRootRef.current) {
      // Create root in shadow DOM
      contentRootRef.current = createRoot(shadowRef.current);
      contentRootRef.current.render(<NestedContent />);
    }
    
    return () => {
      contentRootRef.current?.unmount();
    };
  }, []);
}
```

### Benefits of createRoot (React 18)

1. **Automatic batching** - All updates batched, even in promises/setTimeout
2. **Concurrent features** - Enables `useTransition`, `useDeferredValue`, etc.
3. **Improved hydration** - Better SSR support
4. **Suspense improvements** - Better streaming SSR

### When to Use createRoot

**Use `createRoot` when:**
- Starting a new React 18+ application
- Rendering React into a DOM container
- Creating multiple roots in one app
- Rendering into shadow DOM

**Migration from React 17:**
```tsx
// React 17
import ReactDOM from 'react-dom';
ReactDOM.render(<App />, container);
ReactDOM.unmountComponentAtNode(container);

// React 18
import { createRoot } from 'react-dom/client';
const root = createRoot(container);
root.render(<App />);
root.unmount();
```

---

## React Accessibility Patterns

### ARIA Attributes Pattern

**Key ARIA attributes:** `role`, `aria-label`, `aria-labelledby`, `aria-describedby`, `aria-hidden`, `aria-live`, `aria-expanded`, `aria-selected`, `aria-disabled`, `aria-current`

**Common patterns:**

```tsx
// Icon button with accessible label
<button onClick={onClick} aria-label="Close dialog">
  <Icon name="close" aria-hidden="true" />
</button>

// Form field with error/help text
function FormField({ label, error, helpText }: Props) {
  const id = useId();
  return (
    <>
      <label htmlFor={id}>{label}</label>
      <input id={id} aria-describedby={`${id}-desc`} aria-invalid={!!error} />
      <span id={`${id}-desc`} role={error ? "alert" : undefined}>
        {error || helpText}
      </span>
    </>
  );
}

// Accordion/expandable section
<button aria-expanded={isOpen} aria-controls={contentId}>
  {title}
</button>
<div id={contentId} hidden={!isOpen} role="region">
  {children}
</div>

// Live region for announcements
<div role="status" aria-live="polite" aria-atomic="true">
  {message}
</div>

// Modal dialog
<div role="dialog" aria-modal="true" aria-labelledby={titleId}>
  <h2 id={titleId}>{title}</h2>
  {children}
</div>

// Tab navigation
<div role="tablist">
  <button role="tab" aria-selected={isActive} aria-controls={panelId}>
    {label}
  </button>
</div>
<div role="tabpanel" id={panelId} aria-labelledby={tabId}>
  {content}
</div>
```

**Rules:** Use semantic HTML first, add ARIA only when needed, keep attributes in sync with state, test with screen readers.

---

### Focus Management Pattern

**Common scenarios:** Auto-focus on mount, focus traps in modals, focus restoration, roving tab index.

```tsx
// Auto-focus first element in dialog
function Dialog({ isOpen }: Props) {
  const buttonRef = useRef<HTMLButtonElement>(null);
  
  useEffect(() => {
    if (isOpen) buttonRef.current?.focus();
  }, [isOpen]);
  
  return <button ref={buttonRef}>Close</button>;
}

// Focus trap + restoration in modal
function Modal({ isOpen, onClose, children }: Props) {
  const modalRef = useRef<HTMLDivElement>(null);
  const restoreFocusRef = useRef<HTMLElement | null>(null);
  
  useEffect(() => {
    if (!isOpen) return;
    
    restoreFocusRef.current = document.activeElement as HTMLElement;
    modalRef.current?.querySelector<HTMLElement>('button')?.focus();
    
    return () => restoreFocusRef.current?.focus();
  }, [isOpen]);
  
  const handleKeyDown = (e: React.KeyboardEvent) => {
    if (e.key === 'Escape') onClose();
    
    // Trap Tab key
    if (e.key === 'Tab') {
      const focusable = modalRef.current?.querySelectorAll<HTMLElement>(
        'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
      );
      if (!focusable?.length) return;
      
      const first = focusable[0];
      const last = focusable[focusable.length - 1];
      
      if (e.shiftKey && document.activeElement === first) {
        e.preventDefault();
        last.focus();
      } else if (!e.shiftKey && document.activeElement === last) {
        e.preventDefault();
        first.focus();
      }
    }
  };
  
  return (
    <div ref={modalRef} role="dialog" aria-modal="true" onKeyDown={handleKeyDown}>
      {children}
    </div>
  );
}

// Focus after delete action
function DeleteButton({ itemId, onDelete }: Props) {
  const handleDelete = () => {
    const current = document.getElementById(`item-${itemId}`);
    const next = (current?.nextElementSibling || current?.previousElementSibling)
      ?.querySelector('button') as HTMLElement;
    
    onDelete(itemId);
    setTimeout(() => next?.focus(), 0);
  };
  
  return <button onClick={handleDelete}>Delete</button>;
}

// Roving tab index for lists
function RadioGroup({ options, value, onChange }: Props) {
  const [focusedIndex, setFocusedIndex] = useState(0);
  
  const handleKeyDown = (e: React.KeyboardEvent, index: number) => {
    let newIndex = index;
    if (e.key === 'ArrowDown') newIndex = (index + 1) % options.length;
    if (e.key === 'ArrowUp') newIndex = index === 0 ? options.length - 1 : index - 1;
    if (e.key === 'Home') newIndex = 0;
    if (e.key === 'End') newIndex = options.length - 1;
    
    if (newIndex !== index) {
      e.preventDefault();
      setFocusedIndex(newIndex);
    }
  };
  
  return (
    <div role="radiogroup">
      {options.map((opt, i) => (
        <div
          key={opt.id}
          role="radio"
          aria-checked={value === opt.id}
          tabIndex={focusedIndex === i ? 0 : -1}
          onClick={() => onChange(opt.id)}
          onKeyDown={(e) => handleKeyDown(e, i)}
          onFocus={() => setFocusedIndex(i)}
        >
          {opt.label}
        </div>
      ))}
    </div>
  );
}
```

**Rules:** Always restore focus when closing modals, trap focus within modal contexts, use `focus-visible` for keyboard-only indicators, test thoroughly.

---

### Keyboard Navigation Pattern

**Standard keyboard shortcuts:** Escape (close), Tab/Shift+Tab (navigate), Arrow keys (move focus), Enter/Space (activate), Home/End (first/last).

```tsx
// Dropdown with full keyboard support
function Dropdown({ trigger, items, onSelect }: Props) {
  const [isOpen, setIsOpen] = useState(false);
  const [focusedIndex, setFocusedIndex] = useState(0);
  const itemsRef = useRef<(HTMLButtonElement | null)[]>([]);
  
  const handleKeyDown = (e: React.KeyboardEvent) => {
    switch (e.key) {
      case 'ArrowDown':
        e.preventDefault();
        if (!isOpen) {
          setIsOpen(true);
        } else {
          const next = (focusedIndex + 1) % items.length;
          setFocusedIndex(next);
          itemsRef.current[next]?.focus();
        }
        break;
      case 'ArrowUp':
        e.preventDefault();
        if (isOpen) {
          const prev = focusedIndex === 0 ? items.length - 1 : focusedIndex - 1;
          setFocusedIndex(prev);
          itemsRef.current[prev]?.focus();
        }
        break;
      case 'Escape':
        e.preventDefault();
        setIsOpen(false);
        break;
      case 'Enter':
      case ' ':
        e.preventDefault();
        if (isOpen) {
          onSelect(items[focusedIndex]);
          setIsOpen(false);
        } else {
          setIsOpen(true);
        }
        break;
    }
  };
  
  return (
    <div onKeyDown={handleKeyDown}>
      <button onClick={() => setIsOpen(!isOpen)} aria-expanded={isOpen}>
        {trigger}
      </button>
      {isOpen && (
        <ul role="menu">
          {items.map((item, i) => (
            <li key={item.id} role="none">
              <button
                ref={el => itemsRef.current[i] = el}
                role="menuitem"
                onClick={() => { onSelect(item); setIsOpen(false); }}
                onFocus={() => setFocusedIndex(i)}
              >
                {item.label}
              </button>
            </li>
          ))}
        </ul>
      )}
    </div>
  );
}

// Global keyboard shortcuts hook
function useKeyboardShortcuts(shortcuts: Record<string, () => void>) {
  useEffect(() => {
    const handleKeyDown = (e: KeyboardEvent) => {
      const keys: string[] = [];
      if (e.ctrlKey || e.metaKey) keys.push('Ctrl');
      if (e.shiftKey) keys.push('Shift');
      if (e.altKey) keys.push('Alt');
      keys.push(e.key.toUpperCase());
      
      const handler = shortcuts[keys.join('+')];
      if (handler) {
        e.preventDefault();
        handler();
      }
    };
    
    document.addEventListener('keydown', handleKeyDown);
    return () => document.removeEventListener('keydown', handleKeyDown);
  }, [shortcuts]);
}

// Usage: Editor with shortcuts
function Editor() {
  useKeyboardShortcuts({
    'Ctrl+S': handleSave,
    'Ctrl+Z': handleUndo,
    'Ctrl+Shift+Z': handleRedo,
  });
  return <div>Editor</div>;
}

// Data table with arrow key navigation
function DataTable({ columns, rows }: Props) {
  const [focusedCell, setFocusedCell] = useState({ row: 0, col: 0 });
  const cellRefs = useRef<(HTMLTableCellElement | null)[][]>([]);
  
  const handleKeyDown = (e: React.KeyboardEvent, rowIndex: number, colIndex: number) => {
    let newRow = rowIndex, newCol = colIndex;
    
    if (e.key === 'ArrowUp') newRow = Math.max(0, rowIndex - 1);
    if (e.key === 'ArrowDown') newRow = Math.min(rows.length - 1, rowIndex + 1);
    if (e.key === 'ArrowLeft') newCol = Math.max(0, colIndex - 1);
    if (e.key === 'ArrowRight') newCol = Math.min(columns.length - 1, colIndex + 1);
    if (e.key === 'Home') newCol = 0;
    if (e.key === 'End') newCol = columns.length - 1;
    
    if (newRow !== rowIndex || newCol !== colIndex) {
      e.preventDefault();
      setFocusedCell({ row: newRow, col: newCol });
      cellRefs.current[newRow]?.[newCol]?.focus();
    }
  };
  
  return (
    <table>
      <tbody>
        {rows.map((row, ri) => (
          <tr key={row.id}>
            {columns.map((col, ci) => (
              <td
                key={col.id}
                ref={el => {
                  if (!cellRefs.current[ri]) cellRefs.current[ri] = [];
                  cellRefs.current[ri][ci] = el;
                }}
                tabIndex={focusedCell.row === ri && focusedCell.col === ci ? 0 : -1}
                onKeyDown={e => handleKeyDown(e, ri, ci)}
                onFocus={() => setFocusedCell({ row: ri, col: ci })}
              >
                {row[col.id]}
              </td>
            ))}
          </tr>
        ))}
      </tbody>
    </table>
  );
}
```

**Rules:** Support standard shortcuts (Escape, Tab, Arrows), don't override browser/OS shortcuts, provide visible focus feedback, test keyboard-only navigation.

---

### Accessibility Best Practices Summary

**Key principles:**
- Use semantic HTML first (`<button>`, `<nav>`, `<main>`)
- Add ARIA only when semantic HTML isn't enough
- All interactive elements must be keyboard accessible
- Provide visible focus indicators (use `:focus-visible`)
- Keep ARIA attributes in sync with visual state
- Restore focus after closing modals/dialogs
- Test with screen readers and keyboard only

**Testing checklist:**
- [ ] Navigate entire app with keyboard only
- [ ] Focus indicators visible and high contrast
- [ ] Screen reader announces all content correctly
- [ ] Color not the only state indicator
- [ ] Text contrast ≥ 4.5:1 for normal text
- [ ] Interactive elements have accessible names
- [ ] Form fields have associated labels
- [ ] Error messages are announced

**Tools:** [axe DevTools](https://www.deque.com/axe/devtools/), [WAVE](https://wave.webaim.org/), [Lighthouse](https://developers.google.com/web/tools/lighthouse), screen readers (NVDA, JAWS, VoiceOver)

```
Page 133/141FirstPrevNextLast