lsd-pi 1.1.10 → 1.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/onboarding.js +3 -3
- package/dist/resources/agents/scout.md +6 -0
- package/dist/resources/extensions/slash-commands/index.js +2 -0
- package/dist/resources/extensions/slash-commands/init.js +47 -0
- package/dist/resources/extensions/slash-commands/plan.js +238 -51
- package/dist/resources/extensions/slash-commands/tools.js +14 -27
- package/dist/resources/extensions/subagent/index.js +14 -11
- package/dist/resources/extensions/usage/index.js +87 -36
- package/package.json +1 -1
- package/packages/pi-ai/dist/models.generated.d.ts +222 -1
- package/packages/pi-ai/dist/models.generated.d.ts.map +1 -1
- package/packages/pi-ai/dist/models.generated.js +227 -6
- package/packages/pi-ai/dist/models.generated.js.map +1 -1
- package/packages/pi-ai/dist/models.test.js +28 -0
- package/packages/pi-ai/dist/models.test.js.map +1 -1
- package/packages/pi-ai/scripts/generate-models.ts +10 -100
- package/packages/pi-ai/src/models.generated.ts +227 -6
- package/packages/pi-ai/src/models.test.ts +30 -0
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/agent-session.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/agent-session.js +39 -5
- package/packages/pi-coding-agent/dist/core/agent-session.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resource-loader-lsd-md.test.js +59 -7
- package/packages/pi-coding-agent/dist/core/resource-loader-lsd-md.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/resource-loader.js +4 -4
- package/packages/pi-coding-agent/dist/core/resource-loader.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.d.ts +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.js +18 -7
- package/packages/pi-coding-agent/dist/core/sdk.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/sdk.test.js +80 -0
- package/packages/pi-coding-agent/dist/core/sdk.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts +12 -5
- package/packages/pi-coding-agent/dist/core/settings-manager.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/settings-manager.js +23 -9
- package/packages/pi-coding-agent/dist/core/settings-manager.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/system-prompt.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/system-prompt.js +10 -0
- package/packages/pi-coding-agent/dist/core/system-prompt.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts +8 -4
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js +32 -5
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +8 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +34 -25
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/tests/path-display.test.js +13 -0
- package/packages/pi-coding-agent/dist/tests/path-display.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/tests/settings-manager-scout-policy.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/tests/settings-manager-scout-policy.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/tests/settings-manager-scout-policy.test.js +15 -0
- package/packages/pi-coding-agent/dist/tests/settings-manager-scout-policy.test.js.map +1 -0
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/agent-session.ts +48 -5
- package/packages/pi-coding-agent/src/core/resource-loader-lsd-md.test.ts +67 -7
- package/packages/pi-coding-agent/src/core/resource-loader.ts +4 -4
- package/packages/pi-coding-agent/src/core/sdk.test.ts +100 -0
- package/packages/pi-coding-agent/src/core/sdk.ts +23 -8
- package/packages/pi-coding-agent/src/core/settings-manager.ts +36 -15
- package/packages/pi-coding-agent/src/core/system-prompt.ts +26 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/settings-selector.ts +41 -10
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +11 -0
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +43 -27
- package/packages/pi-coding-agent/src/tests/path-display.test.ts +14 -0
- package/pkg/package.json +1 -1
- package/src/resources/agents/scout.md +6 -0
- package/src/resources/extensions/slash-commands/index.ts +2 -0
- package/src/resources/extensions/slash-commands/init.ts +55 -0
- package/src/resources/extensions/slash-commands/plan.ts +279 -53
- package/src/resources/extensions/slash-commands/tools.ts +15 -29
- package/src/resources/extensions/subagent/index.ts +14 -11
- package/src/resources/extensions/usage/index.ts +92 -38
- package/dist/resources/agents/javascript-pro.md +0 -280
- package/dist/resources/agents/researcher.md +0 -29
- package/dist/resources/agents/teams-builder.md +0 -74
- package/dist/resources/agents/teams-reviewer.md +0 -113
- package/dist/resources/agents/typescript-pro.md +0 -255
- package/dist/resources/skills/accessibility/SKILL.md +0 -522
- package/dist/resources/skills/accessibility/references/WCAG.md +0 -162
- package/dist/resources/skills/core-web-vitals/SKILL.md +0 -441
- package/dist/resources/skills/core-web-vitals/references/LCP.md +0 -208
- package/dist/resources/skills/frontend-design/SKILL.md +0 -45
- package/dist/resources/skills/make-interfaces-feel-better/SKILL.md +0 -122
- package/dist/resources/skills/make-interfaces-feel-better/animations.md +0 -379
- package/dist/resources/skills/make-interfaces-feel-better/performance.md +0 -88
- package/dist/resources/skills/make-interfaces-feel-better/surfaces.md +0 -247
- package/dist/resources/skills/make-interfaces-feel-better/typography.md +0 -123
- package/dist/resources/skills/react-best-practices/README.md +0 -123
- package/dist/resources/skills/react-best-practices/SKILL.md +0 -136
- package/dist/resources/skills/react-best-practices/metadata.json +0 -15
- package/dist/resources/skills/react-best-practices/rules/_sections.md +0 -46
- package/dist/resources/skills/react-best-practices/rules/_template.md +0 -28
- package/dist/resources/skills/react-best-practices/rules/advanced-event-handler-refs.md +0 -55
- package/dist/resources/skills/react-best-practices/rules/advanced-init-once.md +0 -42
- package/dist/resources/skills/react-best-practices/rules/advanced-use-latest.md +0 -39
- package/dist/resources/skills/react-best-practices/rules/async-api-routes.md +0 -38
- package/dist/resources/skills/react-best-practices/rules/async-defer-await.md +0 -80
- package/dist/resources/skills/react-best-practices/rules/async-dependencies.md +0 -51
- package/dist/resources/skills/react-best-practices/rules/async-parallel.md +0 -28
- package/dist/resources/skills/react-best-practices/rules/async-suspense-boundaries.md +0 -99
- package/dist/resources/skills/react-best-practices/rules/bundle-barrel-imports.md +0 -59
- package/dist/resources/skills/react-best-practices/rules/bundle-conditional.md +0 -31
- package/dist/resources/skills/react-best-practices/rules/bundle-defer-third-party.md +0 -49
- package/dist/resources/skills/react-best-practices/rules/bundle-dynamic-imports.md +0 -35
- package/dist/resources/skills/react-best-practices/rules/bundle-preload.md +0 -50
- package/dist/resources/skills/react-best-practices/rules/client-event-listeners.md +0 -74
- package/dist/resources/skills/react-best-practices/rules/client-localstorage-schema.md +0 -71
- package/dist/resources/skills/react-best-practices/rules/client-passive-event-listeners.md +0 -48
- package/dist/resources/skills/react-best-practices/rules/client-swr-dedup.md +0 -56
- package/dist/resources/skills/react-best-practices/rules/js-batch-dom-css.md +0 -107
- package/dist/resources/skills/react-best-practices/rules/js-cache-function-results.md +0 -80
- package/dist/resources/skills/react-best-practices/rules/js-cache-property-access.md +0 -28
- package/dist/resources/skills/react-best-practices/rules/js-cache-storage.md +0 -70
- package/dist/resources/skills/react-best-practices/rules/js-combine-iterations.md +0 -32
- package/dist/resources/skills/react-best-practices/rules/js-early-exit.md +0 -50
- package/dist/resources/skills/react-best-practices/rules/js-hoist-regexp.md +0 -45
- package/dist/resources/skills/react-best-practices/rules/js-index-maps.md +0 -37
- package/dist/resources/skills/react-best-practices/rules/js-length-check-first.md +0 -49
- package/dist/resources/skills/react-best-practices/rules/js-min-max-loop.md +0 -82
- package/dist/resources/skills/react-best-practices/rules/js-set-map-lookups.md +0 -24
- package/dist/resources/skills/react-best-practices/rules/js-tosorted-immutable.md +0 -57
- package/dist/resources/skills/react-best-practices/rules/rendering-activity.md +0 -26
- package/dist/resources/skills/react-best-practices/rules/rendering-animate-svg-wrapper.md +0 -47
- package/dist/resources/skills/react-best-practices/rules/rendering-conditional-render.md +0 -40
- package/dist/resources/skills/react-best-practices/rules/rendering-content-visibility.md +0 -38
- package/dist/resources/skills/react-best-practices/rules/rendering-hoist-jsx.md +0 -46
- package/dist/resources/skills/react-best-practices/rules/rendering-hydration-no-flicker.md +0 -82
- package/dist/resources/skills/react-best-practices/rules/rendering-hydration-suppress-warning.md +0 -30
- package/dist/resources/skills/react-best-practices/rules/rendering-svg-precision.md +0 -28
- package/dist/resources/skills/react-best-practices/rules/rendering-usetransition-loading.md +0 -75
- package/dist/resources/skills/react-best-practices/rules/rerender-defer-reads.md +0 -39
- package/dist/resources/skills/react-best-practices/rules/rerender-dependencies.md +0 -45
- package/dist/resources/skills/react-best-practices/rules/rerender-derived-state-no-effect.md +0 -40
- package/dist/resources/skills/react-best-practices/rules/rerender-derived-state.md +0 -29
- package/dist/resources/skills/react-best-practices/rules/rerender-functional-setstate.md +0 -74
- package/dist/resources/skills/react-best-practices/rules/rerender-lazy-state-init.md +0 -58
- package/dist/resources/skills/react-best-practices/rules/rerender-memo-with-default-value.md +0 -38
- package/dist/resources/skills/react-best-practices/rules/rerender-memo.md +0 -44
- package/dist/resources/skills/react-best-practices/rules/rerender-move-effect-to-event.md +0 -45
- package/dist/resources/skills/react-best-practices/rules/rerender-simple-expression-in-memo.md +0 -35
- package/dist/resources/skills/react-best-practices/rules/rerender-transitions.md +0 -40
- package/dist/resources/skills/react-best-practices/rules/rerender-use-ref-transient-values.md +0 -73
- package/dist/resources/skills/react-best-practices/rules/server-after-nonblocking.md +0 -73
- package/dist/resources/skills/react-best-practices/rules/server-auth-actions.md +0 -96
- package/dist/resources/skills/react-best-practices/rules/server-cache-lru.md +0 -41
- package/dist/resources/skills/react-best-practices/rules/server-cache-react.md +0 -76
- package/dist/resources/skills/react-best-practices/rules/server-dedup-props.md +0 -65
- package/dist/resources/skills/react-best-practices/rules/server-parallel-fetching.md +0 -83
- package/dist/resources/skills/react-best-practices/rules/server-serialization.md +0 -38
- package/dist/resources/skills/userinterface-wiki/SKILL.md +0 -253
- package/dist/resources/skills/userinterface-wiki/rules/_sections.md +0 -66
- package/dist/resources/skills/userinterface-wiki/rules/_template.md +0 -24
- package/dist/resources/skills/userinterface-wiki/rules/a11y-reduced-motion-check.md +0 -30
- package/dist/resources/skills/userinterface-wiki/rules/a11y-toggle-setting.md +0 -30
- package/dist/resources/skills/userinterface-wiki/rules/a11y-visual-equivalent.md +0 -36
- package/dist/resources/skills/userinterface-wiki/rules/a11y-volume-control.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/appropriate-confirmations-only.md +0 -19
- package/dist/resources/skills/userinterface-wiki/rules/appropriate-errors-warnings.md +0 -18
- package/dist/resources/skills/userinterface-wiki/rules/appropriate-no-decorative.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/appropriate-no-high-frequency.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/appropriate-no-punishing.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/container-callback-ref.md +0 -31
- package/dist/resources/skills/userinterface-wiki/rules/container-guard-initial-zero.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/container-no-excessive-use.md +0 -13
- package/dist/resources/skills/userinterface-wiki/rules/container-overflow-hidden.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/container-transition-delay.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/container-two-div-pattern.md +0 -35
- package/dist/resources/skills/userinterface-wiki/rules/container-use-resize-observer.md +0 -48
- package/dist/resources/skills/userinterface-wiki/rules/context-cleanup-nodes.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/context-resume-suspended.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/context-reuse-single.md +0 -30
- package/dist/resources/skills/userinterface-wiki/rules/design-filter-for-character.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/design-noise-for-percussion.md +0 -26
- package/dist/resources/skills/userinterface-wiki/rules/design-oscillator-for-tonal.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/duration-max-300ms.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/duration-press-hover.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/duration-shorten-before-curve.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/duration-small-state.md +0 -15
- package/dist/resources/skills/userinterface-wiki/rules/easing-entrance-ease-out.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/easing-exit-ease-in.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/easing-for-state-change.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/easing-linear-only-progress.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/easing-natural-decay.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/easing-no-linear-motion.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/easing-transition-ease-in-out.md +0 -15
- package/dist/resources/skills/userinterface-wiki/rules/envelope-exponential-decay.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/envelope-no-zero-target.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/envelope-set-initial-value.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/exit-key-required.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/exit-matches-initial.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/exit-prop-required.md +0 -33
- package/dist/resources/skills/userinterface-wiki/rules/exit-requires-wrapper.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/impl-default-subtle.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/impl-preload-audio.md +0 -34
- package/dist/resources/skills/userinterface-wiki/rules/impl-reset-current-time.md +0 -26
- package/dist/resources/skills/userinterface-wiki/rules/mode-pop-layout-for-lists.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/mode-sync-layout-conflict.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/mode-wait-doubles-duration.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/morphing-aria-hidden.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/morphing-consistent-viewbox.md +0 -23
- package/dist/resources/skills/userinterface-wiki/rules/morphing-group-variants.md +0 -33
- package/dist/resources/skills/userinterface-wiki/rules/morphing-jump-non-grouped.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/morphing-reduced-motion.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/morphing-spring-rotation.md +0 -23
- package/dist/resources/skills/userinterface-wiki/rules/morphing-strokelinecap-round.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/morphing-three-lines.md +0 -32
- package/dist/resources/skills/userinterface-wiki/rules/morphing-use-collapsed.md +0 -33
- package/dist/resources/skills/userinterface-wiki/rules/native-backdrop-styling.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/native-placeholder-styling.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/native-selection-styling.md +0 -18
- package/dist/resources/skills/userinterface-wiki/rules/nested-consistent-timing.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/nested-propagate-required.md +0 -41
- package/dist/resources/skills/userinterface-wiki/rules/none-context-menu-entrance.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/none-high-frequency.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/none-keyboard-navigation.md +0 -32
- package/dist/resources/skills/userinterface-wiki/rules/param-click-duration.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/param-filter-frequency-range.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/param-q-value-range.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/param-reasonable-gain.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/physics-active-state.md +0 -23
- package/dist/resources/skills/userinterface-wiki/rules/physics-no-excessive-stagger.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/physics-spring-for-overshoot.md +0 -23
- package/dist/resources/skills/userinterface-wiki/rules/physics-subtle-deformation.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/prefetch-hit-slop.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/prefetch-keyboard-tab.md +0 -19
- package/dist/resources/skills/userinterface-wiki/rules/prefetch-not-everything.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/prefetch-touch-fallback.md +0 -34
- package/dist/resources/skills/userinterface-wiki/rules/prefetch-trajectory-over-hover.md +0 -32
- package/dist/resources/skills/userinterface-wiki/rules/prefetch-use-selectively.md +0 -13
- package/dist/resources/skills/userinterface-wiki/rules/presence-disable-interactions.md +0 -31
- package/dist/resources/skills/userinterface-wiki/rules/presence-hook-in-child.md +0 -31
- package/dist/resources/skills/userinterface-wiki/rules/presence-safe-to-remove.md +0 -37
- package/dist/resources/skills/userinterface-wiki/rules/pseudo-content-required.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/pseudo-first-line-styling.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/pseudo-hit-target-expansion.md +0 -31
- package/dist/resources/skills/userinterface-wiki/rules/pseudo-marker-styling.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/pseudo-over-dom-node.md +0 -32
- package/dist/resources/skills/userinterface-wiki/rules/pseudo-position-relative-parent.md +0 -33
- package/dist/resources/skills/userinterface-wiki/rules/pseudo-z-index-layering.md +0 -37
- package/dist/resources/skills/userinterface-wiki/rules/spring-for-gestures.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/spring-for-interruptible.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/spring-params-balanced.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/spring-preserves-velocity.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/staging-dim-background.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/staging-one-focal-point.md +0 -24
- package/dist/resources/skills/userinterface-wiki/rules/staging-z-index-hierarchy.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/timing-consistent.md +0 -24
- package/dist/resources/skills/userinterface-wiki/rules/timing-no-entrance-context-menu.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/timing-under-300ms.md +0 -22
- package/dist/resources/skills/userinterface-wiki/rules/transition-name-cleanup.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/transition-name-required.md +0 -27
- package/dist/resources/skills/userinterface-wiki/rules/transition-name-unique.md +0 -24
- package/dist/resources/skills/userinterface-wiki/rules/transition-over-js-library.md +0 -32
- package/dist/resources/skills/userinterface-wiki/rules/transition-style-pseudo-elements.md +0 -24
- package/dist/resources/skills/userinterface-wiki/rules/type-antialiased-on-retina.md +0 -18
- package/dist/resources/skills/userinterface-wiki/rules/type-disambiguation-stylistic-set.md +0 -15
- package/dist/resources/skills/userinterface-wiki/rules/type-font-display-swap.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/type-justify-with-hyphens.md +0 -24
- package/dist/resources/skills/userinterface-wiki/rules/type-letter-spacing-uppercase.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/type-no-font-synthesis.md +0 -18
- package/dist/resources/skills/userinterface-wiki/rules/type-oldstyle-nums-for-prose.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/type-opentype-contextual-alternates.md +0 -15
- package/dist/resources/skills/userinterface-wiki/rules/type-optical-sizing-auto.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/type-proper-fractions.md +0 -15
- package/dist/resources/skills/userinterface-wiki/rules/type-slashed-zero.md +0 -17
- package/dist/resources/skills/userinterface-wiki/rules/type-tabular-nums-for-data.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/type-text-wrap-balance-headings.md +0 -21
- package/dist/resources/skills/userinterface-wiki/rules/type-text-wrap-pretty.md +0 -16
- package/dist/resources/skills/userinterface-wiki/rules/type-underline-offset.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/type-variable-weight-continuous.md +0 -23
- package/dist/resources/skills/userinterface-wiki/rules/ux-aesthetic-usability.md +0 -32
- package/dist/resources/skills/userinterface-wiki/rules/ux-cognitive-load-reduce.md +0 -49
- package/dist/resources/skills/userinterface-wiki/rules/ux-common-region-boundaries.md +0 -50
- package/dist/resources/skills/userinterface-wiki/rules/ux-doherty-perceived-speed.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/ux-doherty-under-400ms.md +0 -30
- package/dist/resources/skills/userinterface-wiki/rules/ux-fitts-hit-area.md +0 -32
- package/dist/resources/skills/userinterface-wiki/rules/ux-fitts-target-size.md +0 -31
- package/dist/resources/skills/userinterface-wiki/rules/ux-goal-gradient-progress.md +0 -33
- package/dist/resources/skills/userinterface-wiki/rules/ux-hicks-minimize-choices.md +0 -45
- package/dist/resources/skills/userinterface-wiki/rules/ux-jakobs-familiar-patterns.md +0 -37
- package/dist/resources/skills/userinterface-wiki/rules/ux-millers-chunking.md +0 -23
- package/dist/resources/skills/userinterface-wiki/rules/ux-pareto-prioritize-features.md +0 -36
- package/dist/resources/skills/userinterface-wiki/rules/ux-peak-end-finish-strong.md +0 -35
- package/dist/resources/skills/userinterface-wiki/rules/ux-postels-accept-messy-input.md +0 -45
- package/dist/resources/skills/userinterface-wiki/rules/ux-pragnanz-simplify.md +0 -33
- package/dist/resources/skills/userinterface-wiki/rules/ux-progressive-disclosure.md +0 -41
- package/dist/resources/skills/userinterface-wiki/rules/ux-proximity-grouping.md +0 -38
- package/dist/resources/skills/userinterface-wiki/rules/ux-serial-position.md +0 -31
- package/dist/resources/skills/userinterface-wiki/rules/ux-similarity-consistency.md +0 -35
- package/dist/resources/skills/userinterface-wiki/rules/ux-teslers-complexity.md +0 -28
- package/dist/resources/skills/userinterface-wiki/rules/ux-uniform-connectedness.md +0 -43
- package/dist/resources/skills/userinterface-wiki/rules/ux-von-restorff-emphasis.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/ux-zeigarnik-show-incomplete.md +0 -36
- package/dist/resources/skills/userinterface-wiki/rules/visual-animate-shadow-pseudo.md +0 -49
- package/dist/resources/skills/userinterface-wiki/rules/visual-border-alpha-colors.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/visual-button-shadow-anatomy.md +0 -49
- package/dist/resources/skills/userinterface-wiki/rules/visual-concentric-radius.md +0 -40
- package/dist/resources/skills/userinterface-wiki/rules/visual-consistent-spacing-scale.md +0 -35
- package/dist/resources/skills/userinterface-wiki/rules/visual-layered-shadows.md +0 -30
- package/dist/resources/skills/userinterface-wiki/rules/visual-no-pure-black-shadow.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/visual-shadow-direction.md +0 -25
- package/dist/resources/skills/userinterface-wiki/rules/visual-shadow-matches-elevation.md +0 -23
- package/dist/resources/skills/userinterface-wiki/rules/weight-duration-matches-action.md +0 -29
- package/dist/resources/skills/userinterface-wiki/rules/weight-match-action.md +0 -32
- package/dist/resources/skills/web-design-guidelines/SKILL.md +0 -39
- package/dist/resources/skills/web-quality-audit/SKILL.md +0 -168
- package/dist/resources/skills/web-quality-audit/scripts/analyze.sh +0 -91
- package/src/resources/agents/javascript-pro.md +0 -280
- package/src/resources/agents/researcher.md +0 -29
- package/src/resources/agents/teams-builder.md +0 -74
- package/src/resources/agents/teams-reviewer.md +0 -113
- package/src/resources/agents/typescript-pro.md +0 -255
- package/src/resources/skills/accessibility/SKILL.md +0 -522
- package/src/resources/skills/accessibility/references/WCAG.md +0 -162
- package/src/resources/skills/core-web-vitals/SKILL.md +0 -441
- package/src/resources/skills/core-web-vitals/references/LCP.md +0 -208
- package/src/resources/skills/frontend-design/SKILL.md +0 -45
- package/src/resources/skills/make-interfaces-feel-better/SKILL.md +0 -122
- package/src/resources/skills/make-interfaces-feel-better/animations.md +0 -379
- package/src/resources/skills/make-interfaces-feel-better/performance.md +0 -88
- package/src/resources/skills/make-interfaces-feel-better/surfaces.md +0 -247
- package/src/resources/skills/make-interfaces-feel-better/typography.md +0 -123
- package/src/resources/skills/react-best-practices/README.md +0 -123
- package/src/resources/skills/react-best-practices/SKILL.md +0 -136
- package/src/resources/skills/react-best-practices/metadata.json +0 -15
- package/src/resources/skills/react-best-practices/rules/_sections.md +0 -46
- package/src/resources/skills/react-best-practices/rules/_template.md +0 -28
- package/src/resources/skills/react-best-practices/rules/advanced-event-handler-refs.md +0 -55
- package/src/resources/skills/react-best-practices/rules/advanced-init-once.md +0 -42
- package/src/resources/skills/react-best-practices/rules/advanced-use-latest.md +0 -39
- package/src/resources/skills/react-best-practices/rules/async-api-routes.md +0 -38
- package/src/resources/skills/react-best-practices/rules/async-defer-await.md +0 -80
- package/src/resources/skills/react-best-practices/rules/async-dependencies.md +0 -51
- package/src/resources/skills/react-best-practices/rules/async-parallel.md +0 -28
- package/src/resources/skills/react-best-practices/rules/async-suspense-boundaries.md +0 -99
- package/src/resources/skills/react-best-practices/rules/bundle-barrel-imports.md +0 -59
- package/src/resources/skills/react-best-practices/rules/bundle-conditional.md +0 -31
- package/src/resources/skills/react-best-practices/rules/bundle-defer-third-party.md +0 -49
- package/src/resources/skills/react-best-practices/rules/bundle-dynamic-imports.md +0 -35
- package/src/resources/skills/react-best-practices/rules/bundle-preload.md +0 -50
- package/src/resources/skills/react-best-practices/rules/client-event-listeners.md +0 -74
- package/src/resources/skills/react-best-practices/rules/client-localstorage-schema.md +0 -71
- package/src/resources/skills/react-best-practices/rules/client-passive-event-listeners.md +0 -48
- package/src/resources/skills/react-best-practices/rules/client-swr-dedup.md +0 -56
- package/src/resources/skills/react-best-practices/rules/js-batch-dom-css.md +0 -107
- package/src/resources/skills/react-best-practices/rules/js-cache-function-results.md +0 -80
- package/src/resources/skills/react-best-practices/rules/js-cache-property-access.md +0 -28
- package/src/resources/skills/react-best-practices/rules/js-cache-storage.md +0 -70
- package/src/resources/skills/react-best-practices/rules/js-combine-iterations.md +0 -32
- package/src/resources/skills/react-best-practices/rules/js-early-exit.md +0 -50
- package/src/resources/skills/react-best-practices/rules/js-hoist-regexp.md +0 -45
- package/src/resources/skills/react-best-practices/rules/js-index-maps.md +0 -37
- package/src/resources/skills/react-best-practices/rules/js-length-check-first.md +0 -49
- package/src/resources/skills/react-best-practices/rules/js-min-max-loop.md +0 -82
- package/src/resources/skills/react-best-practices/rules/js-set-map-lookups.md +0 -24
- package/src/resources/skills/react-best-practices/rules/js-tosorted-immutable.md +0 -57
- package/src/resources/skills/react-best-practices/rules/rendering-activity.md +0 -26
- package/src/resources/skills/react-best-practices/rules/rendering-animate-svg-wrapper.md +0 -47
- package/src/resources/skills/react-best-practices/rules/rendering-conditional-render.md +0 -40
- package/src/resources/skills/react-best-practices/rules/rendering-content-visibility.md +0 -38
- package/src/resources/skills/react-best-practices/rules/rendering-hoist-jsx.md +0 -46
- package/src/resources/skills/react-best-practices/rules/rendering-hydration-no-flicker.md +0 -82
- package/src/resources/skills/react-best-practices/rules/rendering-hydration-suppress-warning.md +0 -30
- package/src/resources/skills/react-best-practices/rules/rendering-svg-precision.md +0 -28
- package/src/resources/skills/react-best-practices/rules/rendering-usetransition-loading.md +0 -75
- package/src/resources/skills/react-best-practices/rules/rerender-defer-reads.md +0 -39
- package/src/resources/skills/react-best-practices/rules/rerender-dependencies.md +0 -45
- package/src/resources/skills/react-best-practices/rules/rerender-derived-state-no-effect.md +0 -40
- package/src/resources/skills/react-best-practices/rules/rerender-derived-state.md +0 -29
- package/src/resources/skills/react-best-practices/rules/rerender-functional-setstate.md +0 -74
- package/src/resources/skills/react-best-practices/rules/rerender-lazy-state-init.md +0 -58
- package/src/resources/skills/react-best-practices/rules/rerender-memo-with-default-value.md +0 -38
- package/src/resources/skills/react-best-practices/rules/rerender-memo.md +0 -44
- package/src/resources/skills/react-best-practices/rules/rerender-move-effect-to-event.md +0 -45
- package/src/resources/skills/react-best-practices/rules/rerender-simple-expression-in-memo.md +0 -35
- package/src/resources/skills/react-best-practices/rules/rerender-transitions.md +0 -40
- package/src/resources/skills/react-best-practices/rules/rerender-use-ref-transient-values.md +0 -73
- package/src/resources/skills/react-best-practices/rules/server-after-nonblocking.md +0 -73
- package/src/resources/skills/react-best-practices/rules/server-auth-actions.md +0 -96
- package/src/resources/skills/react-best-practices/rules/server-cache-lru.md +0 -41
- package/src/resources/skills/react-best-practices/rules/server-cache-react.md +0 -76
- package/src/resources/skills/react-best-practices/rules/server-dedup-props.md +0 -65
- package/src/resources/skills/react-best-practices/rules/server-parallel-fetching.md +0 -83
- package/src/resources/skills/react-best-practices/rules/server-serialization.md +0 -38
- package/src/resources/skills/userinterface-wiki/SKILL.md +0 -253
- package/src/resources/skills/userinterface-wiki/rules/_sections.md +0 -66
- package/src/resources/skills/userinterface-wiki/rules/_template.md +0 -24
- package/src/resources/skills/userinterface-wiki/rules/a11y-reduced-motion-check.md +0 -30
- package/src/resources/skills/userinterface-wiki/rules/a11y-toggle-setting.md +0 -30
- package/src/resources/skills/userinterface-wiki/rules/a11y-visual-equivalent.md +0 -36
- package/src/resources/skills/userinterface-wiki/rules/a11y-volume-control.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/appropriate-confirmations-only.md +0 -19
- package/src/resources/skills/userinterface-wiki/rules/appropriate-errors-warnings.md +0 -18
- package/src/resources/skills/userinterface-wiki/rules/appropriate-no-decorative.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/appropriate-no-high-frequency.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/appropriate-no-punishing.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/container-callback-ref.md +0 -31
- package/src/resources/skills/userinterface-wiki/rules/container-guard-initial-zero.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/container-no-excessive-use.md +0 -13
- package/src/resources/skills/userinterface-wiki/rules/container-overflow-hidden.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/container-transition-delay.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/container-two-div-pattern.md +0 -35
- package/src/resources/skills/userinterface-wiki/rules/container-use-resize-observer.md +0 -48
- package/src/resources/skills/userinterface-wiki/rules/context-cleanup-nodes.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/context-resume-suspended.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/context-reuse-single.md +0 -30
- package/src/resources/skills/userinterface-wiki/rules/design-filter-for-character.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/design-noise-for-percussion.md +0 -26
- package/src/resources/skills/userinterface-wiki/rules/design-oscillator-for-tonal.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/duration-max-300ms.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/duration-press-hover.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/duration-shorten-before-curve.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/duration-small-state.md +0 -15
- package/src/resources/skills/userinterface-wiki/rules/easing-entrance-ease-out.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/easing-exit-ease-in.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/easing-for-state-change.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/easing-linear-only-progress.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/easing-natural-decay.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/easing-no-linear-motion.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/easing-transition-ease-in-out.md +0 -15
- package/src/resources/skills/userinterface-wiki/rules/envelope-exponential-decay.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/envelope-no-zero-target.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/envelope-set-initial-value.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/exit-key-required.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/exit-matches-initial.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/exit-prop-required.md +0 -33
- package/src/resources/skills/userinterface-wiki/rules/exit-requires-wrapper.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/impl-default-subtle.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/impl-preload-audio.md +0 -34
- package/src/resources/skills/userinterface-wiki/rules/impl-reset-current-time.md +0 -26
- package/src/resources/skills/userinterface-wiki/rules/mode-pop-layout-for-lists.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/mode-sync-layout-conflict.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/mode-wait-doubles-duration.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/morphing-aria-hidden.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/morphing-consistent-viewbox.md +0 -23
- package/src/resources/skills/userinterface-wiki/rules/morphing-group-variants.md +0 -33
- package/src/resources/skills/userinterface-wiki/rules/morphing-jump-non-grouped.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/morphing-reduced-motion.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/morphing-spring-rotation.md +0 -23
- package/src/resources/skills/userinterface-wiki/rules/morphing-strokelinecap-round.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/morphing-three-lines.md +0 -32
- package/src/resources/skills/userinterface-wiki/rules/morphing-use-collapsed.md +0 -33
- package/src/resources/skills/userinterface-wiki/rules/native-backdrop-styling.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/native-placeholder-styling.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/native-selection-styling.md +0 -18
- package/src/resources/skills/userinterface-wiki/rules/nested-consistent-timing.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/nested-propagate-required.md +0 -41
- package/src/resources/skills/userinterface-wiki/rules/none-context-menu-entrance.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/none-high-frequency.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/none-keyboard-navigation.md +0 -32
- package/src/resources/skills/userinterface-wiki/rules/param-click-duration.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/param-filter-frequency-range.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/param-q-value-range.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/param-reasonable-gain.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/physics-active-state.md +0 -23
- package/src/resources/skills/userinterface-wiki/rules/physics-no-excessive-stagger.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/physics-spring-for-overshoot.md +0 -23
- package/src/resources/skills/userinterface-wiki/rules/physics-subtle-deformation.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/prefetch-hit-slop.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/prefetch-keyboard-tab.md +0 -19
- package/src/resources/skills/userinterface-wiki/rules/prefetch-not-everything.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/prefetch-touch-fallback.md +0 -34
- package/src/resources/skills/userinterface-wiki/rules/prefetch-trajectory-over-hover.md +0 -32
- package/src/resources/skills/userinterface-wiki/rules/prefetch-use-selectively.md +0 -13
- package/src/resources/skills/userinterface-wiki/rules/presence-disable-interactions.md +0 -31
- package/src/resources/skills/userinterface-wiki/rules/presence-hook-in-child.md +0 -31
- package/src/resources/skills/userinterface-wiki/rules/presence-safe-to-remove.md +0 -37
- package/src/resources/skills/userinterface-wiki/rules/pseudo-content-required.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/pseudo-first-line-styling.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/pseudo-hit-target-expansion.md +0 -31
- package/src/resources/skills/userinterface-wiki/rules/pseudo-marker-styling.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/pseudo-over-dom-node.md +0 -32
- package/src/resources/skills/userinterface-wiki/rules/pseudo-position-relative-parent.md +0 -33
- package/src/resources/skills/userinterface-wiki/rules/pseudo-z-index-layering.md +0 -37
- package/src/resources/skills/userinterface-wiki/rules/spring-for-gestures.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/spring-for-interruptible.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/spring-params-balanced.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/spring-preserves-velocity.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/staging-dim-background.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/staging-one-focal-point.md +0 -24
- package/src/resources/skills/userinterface-wiki/rules/staging-z-index-hierarchy.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/timing-consistent.md +0 -24
- package/src/resources/skills/userinterface-wiki/rules/timing-no-entrance-context-menu.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/timing-under-300ms.md +0 -22
- package/src/resources/skills/userinterface-wiki/rules/transition-name-cleanup.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/transition-name-required.md +0 -27
- package/src/resources/skills/userinterface-wiki/rules/transition-name-unique.md +0 -24
- package/src/resources/skills/userinterface-wiki/rules/transition-over-js-library.md +0 -32
- package/src/resources/skills/userinterface-wiki/rules/transition-style-pseudo-elements.md +0 -24
- package/src/resources/skills/userinterface-wiki/rules/type-antialiased-on-retina.md +0 -18
- package/src/resources/skills/userinterface-wiki/rules/type-disambiguation-stylistic-set.md +0 -15
- package/src/resources/skills/userinterface-wiki/rules/type-font-display-swap.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/type-justify-with-hyphens.md +0 -24
- package/src/resources/skills/userinterface-wiki/rules/type-letter-spacing-uppercase.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/type-no-font-synthesis.md +0 -18
- package/src/resources/skills/userinterface-wiki/rules/type-oldstyle-nums-for-prose.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/type-opentype-contextual-alternates.md +0 -15
- package/src/resources/skills/userinterface-wiki/rules/type-optical-sizing-auto.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/type-proper-fractions.md +0 -15
- package/src/resources/skills/userinterface-wiki/rules/type-slashed-zero.md +0 -17
- package/src/resources/skills/userinterface-wiki/rules/type-tabular-nums-for-data.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/type-text-wrap-balance-headings.md +0 -21
- package/src/resources/skills/userinterface-wiki/rules/type-text-wrap-pretty.md +0 -16
- package/src/resources/skills/userinterface-wiki/rules/type-underline-offset.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/type-variable-weight-continuous.md +0 -23
- package/src/resources/skills/userinterface-wiki/rules/ux-aesthetic-usability.md +0 -32
- package/src/resources/skills/userinterface-wiki/rules/ux-cognitive-load-reduce.md +0 -49
- package/src/resources/skills/userinterface-wiki/rules/ux-common-region-boundaries.md +0 -50
- package/src/resources/skills/userinterface-wiki/rules/ux-doherty-perceived-speed.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/ux-doherty-under-400ms.md +0 -30
- package/src/resources/skills/userinterface-wiki/rules/ux-fitts-hit-area.md +0 -32
- package/src/resources/skills/userinterface-wiki/rules/ux-fitts-target-size.md +0 -31
- package/src/resources/skills/userinterface-wiki/rules/ux-goal-gradient-progress.md +0 -33
- package/src/resources/skills/userinterface-wiki/rules/ux-hicks-minimize-choices.md +0 -45
- package/src/resources/skills/userinterface-wiki/rules/ux-jakobs-familiar-patterns.md +0 -37
- package/src/resources/skills/userinterface-wiki/rules/ux-millers-chunking.md +0 -23
- package/src/resources/skills/userinterface-wiki/rules/ux-pareto-prioritize-features.md +0 -36
- package/src/resources/skills/userinterface-wiki/rules/ux-peak-end-finish-strong.md +0 -35
- package/src/resources/skills/userinterface-wiki/rules/ux-postels-accept-messy-input.md +0 -45
- package/src/resources/skills/userinterface-wiki/rules/ux-pragnanz-simplify.md +0 -33
- package/src/resources/skills/userinterface-wiki/rules/ux-progressive-disclosure.md +0 -41
- package/src/resources/skills/userinterface-wiki/rules/ux-proximity-grouping.md +0 -38
- package/src/resources/skills/userinterface-wiki/rules/ux-serial-position.md +0 -31
- package/src/resources/skills/userinterface-wiki/rules/ux-similarity-consistency.md +0 -35
- package/src/resources/skills/userinterface-wiki/rules/ux-teslers-complexity.md +0 -28
- package/src/resources/skills/userinterface-wiki/rules/ux-uniform-connectedness.md +0 -43
- package/src/resources/skills/userinterface-wiki/rules/ux-von-restorff-emphasis.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/ux-zeigarnik-show-incomplete.md +0 -36
- package/src/resources/skills/userinterface-wiki/rules/visual-animate-shadow-pseudo.md +0 -49
- package/src/resources/skills/userinterface-wiki/rules/visual-border-alpha-colors.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/visual-button-shadow-anatomy.md +0 -49
- package/src/resources/skills/userinterface-wiki/rules/visual-concentric-radius.md +0 -40
- package/src/resources/skills/userinterface-wiki/rules/visual-consistent-spacing-scale.md +0 -35
- package/src/resources/skills/userinterface-wiki/rules/visual-layered-shadows.md +0 -30
- package/src/resources/skills/userinterface-wiki/rules/visual-no-pure-black-shadow.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/visual-shadow-direction.md +0 -25
- package/src/resources/skills/userinterface-wiki/rules/visual-shadow-matches-elevation.md +0 -23
- package/src/resources/skills/userinterface-wiki/rules/weight-duration-matches-action.md +0 -29
- package/src/resources/skills/userinterface-wiki/rules/weight-match-action.md +0 -32
- package/src/resources/skills/web-design-guidelines/SKILL.md +0 -39
- package/src/resources/skills/web-quality-audit/SKILL.md +0 -168
- package/src/resources/skills/web-quality-audit/scripts/analyze.sh +0 -91
|
@@ -1,74 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Deduplicate Global Event Listeners
|
|
3
|
-
impact: LOW
|
|
4
|
-
impactDescription: single listener for N components
|
|
5
|
-
tags: client, swr, event-listeners, subscription
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Deduplicate Global Event Listeners
|
|
9
|
-
|
|
10
|
-
Use `useSWRSubscription()` to share global event listeners across component instances.
|
|
11
|
-
|
|
12
|
-
**Incorrect (N instances = N listeners):**
|
|
13
|
-
|
|
14
|
-
```tsx
|
|
15
|
-
function useKeyboardShortcut(key: string, callback: () => void) {
|
|
16
|
-
useEffect(() => {
|
|
17
|
-
const handler = (e: KeyboardEvent) => {
|
|
18
|
-
if (e.metaKey && e.key === key) {
|
|
19
|
-
callback()
|
|
20
|
-
}
|
|
21
|
-
}
|
|
22
|
-
window.addEventListener('keydown', handler)
|
|
23
|
-
return () => window.removeEventListener('keydown', handler)
|
|
24
|
-
}, [key, callback])
|
|
25
|
-
}
|
|
26
|
-
```
|
|
27
|
-
|
|
28
|
-
When using the `useKeyboardShortcut` hook multiple times, each instance will register a new listener.
|
|
29
|
-
|
|
30
|
-
**Correct (N instances = 1 listener):**
|
|
31
|
-
|
|
32
|
-
```tsx
|
|
33
|
-
import useSWRSubscription from 'swr/subscription'
|
|
34
|
-
|
|
35
|
-
// Module-level Map to track callbacks per key
|
|
36
|
-
const keyCallbacks = new Map<string, Set<() => void>>()
|
|
37
|
-
|
|
38
|
-
function useKeyboardShortcut(key: string, callback: () => void) {
|
|
39
|
-
// Register this callback in the Map
|
|
40
|
-
useEffect(() => {
|
|
41
|
-
if (!keyCallbacks.has(key)) {
|
|
42
|
-
keyCallbacks.set(key, new Set())
|
|
43
|
-
}
|
|
44
|
-
keyCallbacks.get(key)!.add(callback)
|
|
45
|
-
|
|
46
|
-
return () => {
|
|
47
|
-
const set = keyCallbacks.get(key)
|
|
48
|
-
if (set) {
|
|
49
|
-
set.delete(callback)
|
|
50
|
-
if (set.size === 0) {
|
|
51
|
-
keyCallbacks.delete(key)
|
|
52
|
-
}
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
}, [key, callback])
|
|
56
|
-
|
|
57
|
-
useSWRSubscription('global-keydown', () => {
|
|
58
|
-
const handler = (e: KeyboardEvent) => {
|
|
59
|
-
if (e.metaKey && keyCallbacks.has(e.key)) {
|
|
60
|
-
keyCallbacks.get(e.key)!.forEach(cb => cb())
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
window.addEventListener('keydown', handler)
|
|
64
|
-
return () => window.removeEventListener('keydown', handler)
|
|
65
|
-
})
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
function Profile() {
|
|
69
|
-
// Multiple shortcuts will share the same listener
|
|
70
|
-
useKeyboardShortcut('p', () => { /* ... */ })
|
|
71
|
-
useKeyboardShortcut('k', () => { /* ... */ })
|
|
72
|
-
// ...
|
|
73
|
-
}
|
|
74
|
-
```
|
|
@@ -1,71 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Version and Minimize localStorage Data
|
|
3
|
-
impact: MEDIUM
|
|
4
|
-
impactDescription: prevents schema conflicts, reduces storage size
|
|
5
|
-
tags: client, localStorage, storage, versioning, data-minimization
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Version and Minimize localStorage Data
|
|
9
|
-
|
|
10
|
-
Add version prefix to keys and store only needed fields. Prevents schema conflicts and accidental storage of sensitive data.
|
|
11
|
-
|
|
12
|
-
**Incorrect:**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
// No version, stores everything, no error handling
|
|
16
|
-
localStorage.setItem('userConfig', JSON.stringify(fullUserObject))
|
|
17
|
-
const data = localStorage.getItem('userConfig')
|
|
18
|
-
```
|
|
19
|
-
|
|
20
|
-
**Correct:**
|
|
21
|
-
|
|
22
|
-
```typescript
|
|
23
|
-
const VERSION = 'v2'
|
|
24
|
-
|
|
25
|
-
function saveConfig(config: { theme: string; language: string }) {
|
|
26
|
-
try {
|
|
27
|
-
localStorage.setItem(`userConfig:${VERSION}`, JSON.stringify(config))
|
|
28
|
-
} catch {
|
|
29
|
-
// Throws in incognito/private browsing, quota exceeded, or disabled
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
function loadConfig() {
|
|
34
|
-
try {
|
|
35
|
-
const data = localStorage.getItem(`userConfig:${VERSION}`)
|
|
36
|
-
return data ? JSON.parse(data) : null
|
|
37
|
-
} catch {
|
|
38
|
-
return null
|
|
39
|
-
}
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
// Migration from v1 to v2
|
|
43
|
-
function migrate() {
|
|
44
|
-
try {
|
|
45
|
-
const v1 = localStorage.getItem('userConfig:v1')
|
|
46
|
-
if (v1) {
|
|
47
|
-
const old = JSON.parse(v1)
|
|
48
|
-
saveConfig({ theme: old.darkMode ? 'dark' : 'light', language: old.lang })
|
|
49
|
-
localStorage.removeItem('userConfig:v1')
|
|
50
|
-
}
|
|
51
|
-
} catch {}
|
|
52
|
-
}
|
|
53
|
-
```
|
|
54
|
-
|
|
55
|
-
**Store minimal fields from server responses:**
|
|
56
|
-
|
|
57
|
-
```typescript
|
|
58
|
-
// User object has 20+ fields, only store what UI needs
|
|
59
|
-
function cachePrefs(user: FullUser) {
|
|
60
|
-
try {
|
|
61
|
-
localStorage.setItem('prefs:v1', JSON.stringify({
|
|
62
|
-
theme: user.preferences.theme,
|
|
63
|
-
notifications: user.preferences.notifications
|
|
64
|
-
}))
|
|
65
|
-
} catch {}
|
|
66
|
-
}
|
|
67
|
-
```
|
|
68
|
-
|
|
69
|
-
**Always wrap in try-catch:** `getItem()` and `setItem()` throw in incognito/private browsing (Safari, Firefox), when quota exceeded, or when disabled.
|
|
70
|
-
|
|
71
|
-
**Benefits:** Schema evolution via versioning, reduced storage size, prevents storing tokens/PII/internal flags.
|
|
@@ -1,48 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Use Passive Event Listeners for Scrolling Performance
|
|
3
|
-
impact: MEDIUM
|
|
4
|
-
impactDescription: eliminates scroll delay caused by event listeners
|
|
5
|
-
tags: client, event-listeners, scrolling, performance, touch, wheel
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Use Passive Event Listeners for Scrolling Performance
|
|
9
|
-
|
|
10
|
-
Add `{ passive: true }` to touch and wheel event listeners to enable immediate scrolling. Browsers normally wait for listeners to finish to check if `preventDefault()` is called, causing scroll delay.
|
|
11
|
-
|
|
12
|
-
**Incorrect:**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
useEffect(() => {
|
|
16
|
-
const handleTouch = (e: TouchEvent) => console.log(e.touches[0].clientX)
|
|
17
|
-
const handleWheel = (e: WheelEvent) => console.log(e.deltaY)
|
|
18
|
-
|
|
19
|
-
document.addEventListener('touchstart', handleTouch)
|
|
20
|
-
document.addEventListener('wheel', handleWheel)
|
|
21
|
-
|
|
22
|
-
return () => {
|
|
23
|
-
document.removeEventListener('touchstart', handleTouch)
|
|
24
|
-
document.removeEventListener('wheel', handleWheel)
|
|
25
|
-
}
|
|
26
|
-
}, [])
|
|
27
|
-
```
|
|
28
|
-
|
|
29
|
-
**Correct:**
|
|
30
|
-
|
|
31
|
-
```typescript
|
|
32
|
-
useEffect(() => {
|
|
33
|
-
const handleTouch = (e: TouchEvent) => console.log(e.touches[0].clientX)
|
|
34
|
-
const handleWheel = (e: WheelEvent) => console.log(e.deltaY)
|
|
35
|
-
|
|
36
|
-
document.addEventListener('touchstart', handleTouch, { passive: true })
|
|
37
|
-
document.addEventListener('wheel', handleWheel, { passive: true })
|
|
38
|
-
|
|
39
|
-
return () => {
|
|
40
|
-
document.removeEventListener('touchstart', handleTouch)
|
|
41
|
-
document.removeEventListener('wheel', handleWheel)
|
|
42
|
-
}
|
|
43
|
-
}, [])
|
|
44
|
-
```
|
|
45
|
-
|
|
46
|
-
**Use passive when:** tracking/analytics, logging, any listener that doesn't call `preventDefault()`.
|
|
47
|
-
|
|
48
|
-
**Don't use passive when:** implementing custom swipe gestures, custom zoom controls, or any listener that needs `preventDefault()`.
|
|
@@ -1,56 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Use SWR for Automatic Deduplication
|
|
3
|
-
impact: MEDIUM-HIGH
|
|
4
|
-
impactDescription: automatic deduplication
|
|
5
|
-
tags: client, swr, deduplication, data-fetching
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Use SWR for Automatic Deduplication
|
|
9
|
-
|
|
10
|
-
SWR enables request deduplication, caching, and revalidation across component instances.
|
|
11
|
-
|
|
12
|
-
**Incorrect (no deduplication, each instance fetches):**
|
|
13
|
-
|
|
14
|
-
```tsx
|
|
15
|
-
function UserList() {
|
|
16
|
-
const [users, setUsers] = useState([])
|
|
17
|
-
useEffect(() => {
|
|
18
|
-
fetch('/api/users')
|
|
19
|
-
.then(r => r.json())
|
|
20
|
-
.then(setUsers)
|
|
21
|
-
}, [])
|
|
22
|
-
}
|
|
23
|
-
```
|
|
24
|
-
|
|
25
|
-
**Correct (multiple instances share one request):**
|
|
26
|
-
|
|
27
|
-
```tsx
|
|
28
|
-
import useSWR from 'swr'
|
|
29
|
-
|
|
30
|
-
function UserList() {
|
|
31
|
-
const { data: users } = useSWR('/api/users', fetcher)
|
|
32
|
-
}
|
|
33
|
-
```
|
|
34
|
-
|
|
35
|
-
**For immutable data:**
|
|
36
|
-
|
|
37
|
-
```tsx
|
|
38
|
-
import { useImmutableSWR } from '@/lib/swr'
|
|
39
|
-
|
|
40
|
-
function StaticContent() {
|
|
41
|
-
const { data } = useImmutableSWR('/api/config', fetcher)
|
|
42
|
-
}
|
|
43
|
-
```
|
|
44
|
-
|
|
45
|
-
**For mutations:**
|
|
46
|
-
|
|
47
|
-
```tsx
|
|
48
|
-
import { useSWRMutation } from 'swr/mutation'
|
|
49
|
-
|
|
50
|
-
function UpdateButton() {
|
|
51
|
-
const { trigger } = useSWRMutation('/api/user', updateUser)
|
|
52
|
-
return <button onClick={() => trigger()}>Update</button>
|
|
53
|
-
}
|
|
54
|
-
```
|
|
55
|
-
|
|
56
|
-
Reference: [https://swr.vercel.app](https://swr.vercel.app)
|
|
@@ -1,107 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Avoid Layout Thrashing
|
|
3
|
-
impact: MEDIUM
|
|
4
|
-
impactDescription: prevents forced synchronous layouts and reduces performance bottlenecks
|
|
5
|
-
tags: javascript, dom, css, performance, reflow, layout-thrashing
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Avoid Layout Thrashing
|
|
9
|
-
|
|
10
|
-
Avoid interleaving style writes with layout reads. When you read a layout property (like `offsetWidth`, `getBoundingClientRect()`, or `getComputedStyle()`) between style changes, the browser is forced to trigger a synchronous reflow.
|
|
11
|
-
|
|
12
|
-
**This is OK (browser batches style changes):**
|
|
13
|
-
```typescript
|
|
14
|
-
function updateElementStyles(element: HTMLElement) {
|
|
15
|
-
// Each line invalidates style, but browser batches the recalculation
|
|
16
|
-
element.style.width = '100px'
|
|
17
|
-
element.style.height = '200px'
|
|
18
|
-
element.style.backgroundColor = 'blue'
|
|
19
|
-
element.style.border = '1px solid black'
|
|
20
|
-
}
|
|
21
|
-
```
|
|
22
|
-
|
|
23
|
-
**Incorrect (interleaved reads and writes force reflows):**
|
|
24
|
-
```typescript
|
|
25
|
-
function layoutThrashing(element: HTMLElement) {
|
|
26
|
-
element.style.width = '100px'
|
|
27
|
-
const width = element.offsetWidth // Forces reflow
|
|
28
|
-
element.style.height = '200px'
|
|
29
|
-
const height = element.offsetHeight // Forces another reflow
|
|
30
|
-
}
|
|
31
|
-
```
|
|
32
|
-
|
|
33
|
-
**Correct (batch writes, then read once):**
|
|
34
|
-
```typescript
|
|
35
|
-
function updateElementStyles(element: HTMLElement) {
|
|
36
|
-
// Batch all writes together
|
|
37
|
-
element.style.width = '100px'
|
|
38
|
-
element.style.height = '200px'
|
|
39
|
-
element.style.backgroundColor = 'blue'
|
|
40
|
-
element.style.border = '1px solid black'
|
|
41
|
-
|
|
42
|
-
// Read after all writes are done (single reflow)
|
|
43
|
-
const { width, height } = element.getBoundingClientRect()
|
|
44
|
-
}
|
|
45
|
-
```
|
|
46
|
-
|
|
47
|
-
**Correct (batch reads, then writes):**
|
|
48
|
-
```typescript
|
|
49
|
-
function avoidThrashing(element: HTMLElement) {
|
|
50
|
-
// Read phase - all layout queries first
|
|
51
|
-
const rect1 = element.getBoundingClientRect()
|
|
52
|
-
const offsetWidth = element.offsetWidth
|
|
53
|
-
const offsetHeight = element.offsetHeight
|
|
54
|
-
|
|
55
|
-
// Write phase - all style changes after
|
|
56
|
-
element.style.width = '100px'
|
|
57
|
-
element.style.height = '200px'
|
|
58
|
-
}
|
|
59
|
-
```
|
|
60
|
-
|
|
61
|
-
**Better: use CSS classes**
|
|
62
|
-
```css
|
|
63
|
-
.highlighted-box {
|
|
64
|
-
width: 100px;
|
|
65
|
-
height: 200px;
|
|
66
|
-
background-color: blue;
|
|
67
|
-
border: 1px solid black;
|
|
68
|
-
}
|
|
69
|
-
```
|
|
70
|
-
```typescript
|
|
71
|
-
function updateElementStyles(element: HTMLElement) {
|
|
72
|
-
element.classList.add('highlighted-box')
|
|
73
|
-
|
|
74
|
-
const { width, height } = element.getBoundingClientRect()
|
|
75
|
-
}
|
|
76
|
-
```
|
|
77
|
-
|
|
78
|
-
**React example:**
|
|
79
|
-
```tsx
|
|
80
|
-
// Incorrect: interleaving style changes with layout queries
|
|
81
|
-
function Box({ isHighlighted }: { isHighlighted: boolean }) {
|
|
82
|
-
const ref = useRef<HTMLDivElement>(null)
|
|
83
|
-
|
|
84
|
-
useEffect(() => {
|
|
85
|
-
if (ref.current && isHighlighted) {
|
|
86
|
-
ref.current.style.width = '100px'
|
|
87
|
-
const width = ref.current.offsetWidth // Forces layout
|
|
88
|
-
ref.current.style.height = '200px'
|
|
89
|
-
}
|
|
90
|
-
}, [isHighlighted])
|
|
91
|
-
|
|
92
|
-
return <div ref={ref}>Content</div>
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
// Correct: toggle class
|
|
96
|
-
function Box({ isHighlighted }: { isHighlighted: boolean }) {
|
|
97
|
-
return (
|
|
98
|
-
<div className={isHighlighted ? 'highlighted-box' : ''}>
|
|
99
|
-
Content
|
|
100
|
-
</div>
|
|
101
|
-
)
|
|
102
|
-
}
|
|
103
|
-
```
|
|
104
|
-
|
|
105
|
-
Prefer CSS classes over inline styles when possible. CSS files are cached by the browser, and classes provide better separation of concerns and are easier to maintain.
|
|
106
|
-
|
|
107
|
-
See [this gist](https://gist.github.com/paulirish/5d52fb081b3570c81e3a) and [CSS Triggers](https://csstriggers.com/) for more information on layout-forcing operations.
|
|
@@ -1,80 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Cache Repeated Function Calls
|
|
3
|
-
impact: MEDIUM
|
|
4
|
-
impactDescription: avoid redundant computation
|
|
5
|
-
tags: javascript, cache, memoization, performance
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Cache Repeated Function Calls
|
|
9
|
-
|
|
10
|
-
Use a module-level Map to cache function results when the same function is called repeatedly with the same inputs during render.
|
|
11
|
-
|
|
12
|
-
**Incorrect (redundant computation):**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
function ProjectList({ projects }: { projects: Project[] }) {
|
|
16
|
-
return (
|
|
17
|
-
<div>
|
|
18
|
-
{projects.map(project => {
|
|
19
|
-
// slugify() called 100+ times for same project names
|
|
20
|
-
const slug = slugify(project.name)
|
|
21
|
-
|
|
22
|
-
return <ProjectCard key={project.id} slug={slug} />
|
|
23
|
-
})}
|
|
24
|
-
</div>
|
|
25
|
-
)
|
|
26
|
-
}
|
|
27
|
-
```
|
|
28
|
-
|
|
29
|
-
**Correct (cached results):**
|
|
30
|
-
|
|
31
|
-
```typescript
|
|
32
|
-
// Module-level cache
|
|
33
|
-
const slugifyCache = new Map<string, string>()
|
|
34
|
-
|
|
35
|
-
function cachedSlugify(text: string): string {
|
|
36
|
-
if (slugifyCache.has(text)) {
|
|
37
|
-
return slugifyCache.get(text)!
|
|
38
|
-
}
|
|
39
|
-
const result = slugify(text)
|
|
40
|
-
slugifyCache.set(text, result)
|
|
41
|
-
return result
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
function ProjectList({ projects }: { projects: Project[] }) {
|
|
45
|
-
return (
|
|
46
|
-
<div>
|
|
47
|
-
{projects.map(project => {
|
|
48
|
-
// Computed only once per unique project name
|
|
49
|
-
const slug = cachedSlugify(project.name)
|
|
50
|
-
|
|
51
|
-
return <ProjectCard key={project.id} slug={slug} />
|
|
52
|
-
})}
|
|
53
|
-
</div>
|
|
54
|
-
)
|
|
55
|
-
}
|
|
56
|
-
```
|
|
57
|
-
|
|
58
|
-
**Simpler pattern for single-value functions:**
|
|
59
|
-
|
|
60
|
-
```typescript
|
|
61
|
-
let isLoggedInCache: boolean | null = null
|
|
62
|
-
|
|
63
|
-
function isLoggedIn(): boolean {
|
|
64
|
-
if (isLoggedInCache !== null) {
|
|
65
|
-
return isLoggedInCache
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
isLoggedInCache = document.cookie.includes('auth=')
|
|
69
|
-
return isLoggedInCache
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
// Clear cache when auth changes
|
|
73
|
-
function onAuthChange() {
|
|
74
|
-
isLoggedInCache = null
|
|
75
|
-
}
|
|
76
|
-
```
|
|
77
|
-
|
|
78
|
-
Use a Map (not a hook) so it works everywhere: utilities, event handlers, not just React components.
|
|
79
|
-
|
|
80
|
-
Reference: [How we made the Vercel Dashboard twice as fast](https://vercel.com/blog/how-we-made-the-vercel-dashboard-twice-as-fast)
|
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Cache Property Access in Loops
|
|
3
|
-
impact: LOW-MEDIUM
|
|
4
|
-
impactDescription: reduces lookups
|
|
5
|
-
tags: javascript, loops, optimization, caching
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Cache Property Access in Loops
|
|
9
|
-
|
|
10
|
-
Cache object property lookups in hot paths.
|
|
11
|
-
|
|
12
|
-
**Incorrect (3 lookups × N iterations):**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
for (let i = 0; i < arr.length; i++) {
|
|
16
|
-
process(obj.config.settings.value)
|
|
17
|
-
}
|
|
18
|
-
```
|
|
19
|
-
|
|
20
|
-
**Correct (1 lookup total):**
|
|
21
|
-
|
|
22
|
-
```typescript
|
|
23
|
-
const value = obj.config.settings.value
|
|
24
|
-
const len = arr.length
|
|
25
|
-
for (let i = 0; i < len; i++) {
|
|
26
|
-
process(value)
|
|
27
|
-
}
|
|
28
|
-
```
|
|
@@ -1,70 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Cache Storage API Calls
|
|
3
|
-
impact: LOW-MEDIUM
|
|
4
|
-
impactDescription: reduces expensive I/O
|
|
5
|
-
tags: javascript, localStorage, storage, caching, performance
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Cache Storage API Calls
|
|
9
|
-
|
|
10
|
-
`localStorage`, `sessionStorage`, and `document.cookie` are synchronous and expensive. Cache reads in memory.
|
|
11
|
-
|
|
12
|
-
**Incorrect (reads storage on every call):**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
function getTheme() {
|
|
16
|
-
return localStorage.getItem('theme') ?? 'light'
|
|
17
|
-
}
|
|
18
|
-
// Called 10 times = 10 storage reads
|
|
19
|
-
```
|
|
20
|
-
|
|
21
|
-
**Correct (Map cache):**
|
|
22
|
-
|
|
23
|
-
```typescript
|
|
24
|
-
const storageCache = new Map<string, string | null>()
|
|
25
|
-
|
|
26
|
-
function getLocalStorage(key: string) {
|
|
27
|
-
if (!storageCache.has(key)) {
|
|
28
|
-
storageCache.set(key, localStorage.getItem(key))
|
|
29
|
-
}
|
|
30
|
-
return storageCache.get(key)
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
function setLocalStorage(key: string, value: string) {
|
|
34
|
-
localStorage.setItem(key, value)
|
|
35
|
-
storageCache.set(key, value) // keep cache in sync
|
|
36
|
-
}
|
|
37
|
-
```
|
|
38
|
-
|
|
39
|
-
Use a Map (not a hook) so it works everywhere: utilities, event handlers, not just React components.
|
|
40
|
-
|
|
41
|
-
**Cookie caching:**
|
|
42
|
-
|
|
43
|
-
```typescript
|
|
44
|
-
let cookieCache: Record<string, string> | null = null
|
|
45
|
-
|
|
46
|
-
function getCookie(name: string) {
|
|
47
|
-
if (!cookieCache) {
|
|
48
|
-
cookieCache = Object.fromEntries(
|
|
49
|
-
document.cookie.split('; ').map(c => c.split('='))
|
|
50
|
-
)
|
|
51
|
-
}
|
|
52
|
-
return cookieCache[name]
|
|
53
|
-
}
|
|
54
|
-
```
|
|
55
|
-
|
|
56
|
-
**Important (invalidate on external changes):**
|
|
57
|
-
|
|
58
|
-
If storage can change externally (another tab, server-set cookies), invalidate cache:
|
|
59
|
-
|
|
60
|
-
```typescript
|
|
61
|
-
window.addEventListener('storage', (e) => {
|
|
62
|
-
if (e.key) storageCache.delete(e.key)
|
|
63
|
-
})
|
|
64
|
-
|
|
65
|
-
document.addEventListener('visibilitychange', () => {
|
|
66
|
-
if (document.visibilityState === 'visible') {
|
|
67
|
-
storageCache.clear()
|
|
68
|
-
}
|
|
69
|
-
})
|
|
70
|
-
```
|
|
@@ -1,32 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Combine Multiple Array Iterations
|
|
3
|
-
impact: LOW-MEDIUM
|
|
4
|
-
impactDescription: reduces iterations
|
|
5
|
-
tags: javascript, arrays, loops, performance
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Combine Multiple Array Iterations
|
|
9
|
-
|
|
10
|
-
Multiple `.filter()` or `.map()` calls iterate the array multiple times. Combine into one loop.
|
|
11
|
-
|
|
12
|
-
**Incorrect (3 iterations):**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
const admins = users.filter(u => u.isAdmin)
|
|
16
|
-
const testers = users.filter(u => u.isTester)
|
|
17
|
-
const inactive = users.filter(u => !u.isActive)
|
|
18
|
-
```
|
|
19
|
-
|
|
20
|
-
**Correct (1 iteration):**
|
|
21
|
-
|
|
22
|
-
```typescript
|
|
23
|
-
const admins: User[] = []
|
|
24
|
-
const testers: User[] = []
|
|
25
|
-
const inactive: User[] = []
|
|
26
|
-
|
|
27
|
-
for (const user of users) {
|
|
28
|
-
if (user.isAdmin) admins.push(user)
|
|
29
|
-
if (user.isTester) testers.push(user)
|
|
30
|
-
if (!user.isActive) inactive.push(user)
|
|
31
|
-
}
|
|
32
|
-
```
|
|
@@ -1,50 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Early Return from Functions
|
|
3
|
-
impact: LOW-MEDIUM
|
|
4
|
-
impactDescription: avoids unnecessary computation
|
|
5
|
-
tags: javascript, functions, optimization, early-return
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Early Return from Functions
|
|
9
|
-
|
|
10
|
-
Return early when result is determined to skip unnecessary processing.
|
|
11
|
-
|
|
12
|
-
**Incorrect (processes all items even after finding answer):**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
function validateUsers(users: User[]) {
|
|
16
|
-
let hasError = false
|
|
17
|
-
let errorMessage = ''
|
|
18
|
-
|
|
19
|
-
for (const user of users) {
|
|
20
|
-
if (!user.email) {
|
|
21
|
-
hasError = true
|
|
22
|
-
errorMessage = 'Email required'
|
|
23
|
-
}
|
|
24
|
-
if (!user.name) {
|
|
25
|
-
hasError = true
|
|
26
|
-
errorMessage = 'Name required'
|
|
27
|
-
}
|
|
28
|
-
// Continues checking all users even after error found
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
return hasError ? { valid: false, error: errorMessage } : { valid: true }
|
|
32
|
-
}
|
|
33
|
-
```
|
|
34
|
-
|
|
35
|
-
**Correct (returns immediately on first error):**
|
|
36
|
-
|
|
37
|
-
```typescript
|
|
38
|
-
function validateUsers(users: User[]) {
|
|
39
|
-
for (const user of users) {
|
|
40
|
-
if (!user.email) {
|
|
41
|
-
return { valid: false, error: 'Email required' }
|
|
42
|
-
}
|
|
43
|
-
if (!user.name) {
|
|
44
|
-
return { valid: false, error: 'Name required' }
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
return { valid: true }
|
|
49
|
-
}
|
|
50
|
-
```
|
|
@@ -1,45 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Hoist RegExp Creation
|
|
3
|
-
impact: LOW-MEDIUM
|
|
4
|
-
impactDescription: avoids recreation
|
|
5
|
-
tags: javascript, regexp, optimization, memoization
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Hoist RegExp Creation
|
|
9
|
-
|
|
10
|
-
Don't create RegExp inside render. Hoist to module scope or memoize with `useMemo()`.
|
|
11
|
-
|
|
12
|
-
**Incorrect (new RegExp every render):**
|
|
13
|
-
|
|
14
|
-
```tsx
|
|
15
|
-
function Highlighter({ text, query }: Props) {
|
|
16
|
-
const regex = new RegExp(`(${query})`, 'gi')
|
|
17
|
-
const parts = text.split(regex)
|
|
18
|
-
return <>{parts.map((part, i) => ...)}</>
|
|
19
|
-
}
|
|
20
|
-
```
|
|
21
|
-
|
|
22
|
-
**Correct (memoize or hoist):**
|
|
23
|
-
|
|
24
|
-
```tsx
|
|
25
|
-
const EMAIL_REGEX = /^[^\s@]+@[^\s@]+\.[^\s@]+$/
|
|
26
|
-
|
|
27
|
-
function Highlighter({ text, query }: Props) {
|
|
28
|
-
const regex = useMemo(
|
|
29
|
-
() => new RegExp(`(${escapeRegex(query)})`, 'gi'),
|
|
30
|
-
[query]
|
|
31
|
-
)
|
|
32
|
-
const parts = text.split(regex)
|
|
33
|
-
return <>{parts.map((part, i) => ...)}</>
|
|
34
|
-
}
|
|
35
|
-
```
|
|
36
|
-
|
|
37
|
-
**Warning (global regex has mutable state):**
|
|
38
|
-
|
|
39
|
-
Global regex (`/g`) has mutable `lastIndex` state:
|
|
40
|
-
|
|
41
|
-
```typescript
|
|
42
|
-
const regex = /foo/g
|
|
43
|
-
regex.test('foo') // true, lastIndex = 3
|
|
44
|
-
regex.test('foo') // false, lastIndex = 0
|
|
45
|
-
```
|
|
@@ -1,37 +0,0 @@
|
|
|
1
|
-
---
|
|
2
|
-
title: Build Index Maps for Repeated Lookups
|
|
3
|
-
impact: LOW-MEDIUM
|
|
4
|
-
impactDescription: 1M ops to 2K ops
|
|
5
|
-
tags: javascript, map, indexing, optimization, performance
|
|
6
|
-
---
|
|
7
|
-
|
|
8
|
-
## Build Index Maps for Repeated Lookups
|
|
9
|
-
|
|
10
|
-
Multiple `.find()` calls by the same key should use a Map.
|
|
11
|
-
|
|
12
|
-
**Incorrect (O(n) per lookup):**
|
|
13
|
-
|
|
14
|
-
```typescript
|
|
15
|
-
function processOrders(orders: Order[], users: User[]) {
|
|
16
|
-
return orders.map(order => ({
|
|
17
|
-
...order,
|
|
18
|
-
user: users.find(u => u.id === order.userId)
|
|
19
|
-
}))
|
|
20
|
-
}
|
|
21
|
-
```
|
|
22
|
-
|
|
23
|
-
**Correct (O(1) per lookup):**
|
|
24
|
-
|
|
25
|
-
```typescript
|
|
26
|
-
function processOrders(orders: Order[], users: User[]) {
|
|
27
|
-
const userById = new Map(users.map(u => [u.id, u]))
|
|
28
|
-
|
|
29
|
-
return orders.map(order => ({
|
|
30
|
-
...order,
|
|
31
|
-
user: userById.get(order.userId)
|
|
32
|
-
}))
|
|
33
|
-
}
|
|
34
|
-
```
|
|
35
|
-
|
|
36
|
-
Build map once (O(n)), then all lookups are O(1).
|
|
37
|
-
For 1000 orders × 1000 users: 1M ops → 2K ops.
|