@supersoniks/concorde 4.6.0 → 4.7.3

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.
Files changed (182) hide show
  1. package/.gitlab-ci.yml +23 -0
  2. package/README.md +106 -55
  3. package/ai/AGENTS.md +52 -0
  4. package/ai/README.md +30 -0
  5. package/ai/cursor/rules/concorde-menu.mdc +15 -0
  6. package/ai/cursor/rules/concorde-scope.mdc +14 -0
  7. package/ai/cursor/rules/concorde-theme.mdc +13 -0
  8. package/ai/cursor/rules/concorde.mdc +49 -0
  9. package/ai/jetbrains/rules/concorde.md +39 -0
  10. package/ai/skills/concorde/SKILL.md +273 -0
  11. package/ai/skills/concorde-get-set-dp/SKILL.md +194 -0
  12. package/ai/skills/concorde-imports/SKILL.md +78 -0
  13. package/ai/skills/concorde-menu/SKILL.md +74 -0
  14. package/ai/skills/concorde-scope/SKILL.md +70 -0
  15. package/ai/skills/concorde-theme/SKILL.md +46 -0
  16. package/build-infos.json +1 -1
  17. package/concorde-core.bundle.js +152 -152
  18. package/concorde-core.es.js +1853 -1689
  19. package/dist/altcha-widget.js +2662 -0
  20. package/dist/concorde-core.bundle.js +152 -152
  21. package/dist/concorde-core.es.js +1853 -1689
  22. package/dist/docs-mock-api-sw.js +589 -0
  23. package/dist/docs-mock-api-sw.js.map +7 -0
  24. package/docs/altcha-widget.js +2662 -0
  25. package/docs/assets/index-D9pxaQYK.js +7508 -0
  26. package/docs/assets/index-t0-i22oI.css +1 -0
  27. package/docs/docs-mock-api-sw.js +589 -0
  28. package/docs/docs-mock-api-sw.js.map +7 -0
  29. package/docs/index.html +2 -2
  30. package/docs/src/core/components/functional/fetch/fetch.md +13 -11
  31. package/docs/src/core/components/functional/if/if.md +4 -11
  32. package/docs/src/core/components/functional/list/list.md +60 -194
  33. package/docs/src/core/components/functional/queue/queue.md +70 -85
  34. package/docs/src/core/components/functional/router/router.md +62 -97
  35. package/docs/src/core/components/functional/states/states.md +2 -2
  36. package/docs/src/core/components/functional/submit/submit.md +86 -55
  37. package/docs/src/core/components/ui/captcha/captcha.md +2 -2
  38. package/docs/src/core/components/ui/card/card.md +1 -1
  39. package/docs/src/core/components/ui/form/checkbox/checkbox.md +5 -32
  40. package/docs/src/core/components/ui/form/input/input.md +5 -30
  41. package/docs/src/core/components/ui/form/input-autocomplete/input-autocomplete.md +6 -4
  42. package/docs/src/core/components/ui/form/radio/radio.md +5 -32
  43. package/docs/src/core/components/ui/form/select/select.md +5 -31
  44. package/docs/src/core/components/ui/form/switch/switch.md +5 -32
  45. package/docs/src/core/components/ui/loader/loader.md +1 -13
  46. package/docs/src/core/components/ui/table/table.md +3 -3
  47. package/docs/src/docs/_core-concept/dataFlow.md +73 -0
  48. package/docs/src/docs/_core-concept/subscriber.md +9 -10
  49. package/docs/src/docs/_decorators/ancestor-attribute.md +4 -3
  50. package/docs/src/docs/_decorators/auto-subscribe.md +19 -16
  51. package/docs/src/docs/_decorators/bind.md +20 -17
  52. package/docs/src/docs/_decorators/get.md +7 -4
  53. package/docs/src/docs/_decorators/handle.md +171 -0
  54. package/docs/src/docs/_decorators/on-assign.md +99 -73
  55. package/docs/src/docs/_decorators/publish.md +2 -1
  56. package/docs/src/docs/_decorators/subscribe.md +70 -9
  57. package/docs/src/docs/_decorators/wait-for-ancestors.md +13 -10
  58. package/docs/src/docs/_directives/sub.md +91 -0
  59. package/docs/src/docs/_getting-started/ai-agents.md +56 -0
  60. package/docs/src/docs/_getting-started/concorde-manual-install.md +133 -0
  61. package/docs/src/docs/_getting-started/concorde-outside.md +13 -123
  62. package/docs/src/docs/_getting-started/create-a-component.md +2 -0
  63. package/docs/src/docs/_getting-started/my-first-component.md +236 -0
  64. package/docs/src/docs/_getting-started/my-first-subscriber.md +29 -83
  65. package/docs/src/docs/_getting-started/pubsub.md +21 -134
  66. package/docs/src/docs/_getting-started/start.md +26 -18
  67. package/docs/src/docs/_misc/api-configuration.md +79 -0
  68. package/docs/src/docs/_misc/dataProviderKey.md +38 -5
  69. package/docs/src/docs/_misc/docs-mock-api.md +60 -0
  70. package/docs/src/docs/_misc/endpoint.md +2 -1
  71. package/docs/src/docs/_misc/html-integration.md +13 -0
  72. package/docs/src/docs/search/docs-search.json +4163 -873
  73. package/docs/src/tsconfig.json +380 -317
  74. package/gitlab/job_tests.sh +55 -0
  75. package/package.json +34 -3
  76. package/public/altcha-widget.js +2662 -0
  77. package/public/docs-mock-api-sw.js +589 -0
  78. package/public/docs-mock-api-sw.js.map +7 -0
  79. package/scripts/ai-init.mjs +167 -0
  80. package/scripts/docs-mock-api-vite-plugin.ts +116 -0
  81. package/scripts/docs-open-in-editor-plugin.ts +130 -0
  82. package/scripts/pre-publish.mjs +2 -1
  83. package/src/core/components/functional/example/example.ts +1 -1
  84. package/src/core/components/functional/fetch/fetch.md +13 -11
  85. package/src/core/components/functional/if/if.md +4 -11
  86. package/src/core/components/functional/list/list.demo.ts +4 -4
  87. package/src/core/components/functional/list/list.md +60 -194
  88. package/src/core/components/functional/list/list.ts +8 -7
  89. package/src/core/components/functional/queue/queue.demo.ts +1 -1
  90. package/src/core/components/functional/queue/queue.md +70 -85
  91. package/src/core/components/functional/queue/queue.ts +4 -4
  92. package/src/core/components/functional/router/router.md +62 -97
  93. package/src/core/components/functional/router/router.ts +1 -1
  94. package/src/core/components/functional/states/states.md +2 -2
  95. package/src/core/components/functional/submit/submit.md +86 -55
  96. package/src/core/components/functional/submit/submit.ts +10 -3
  97. package/src/core/components/ui/captcha/captcha.md +2 -2
  98. package/src/core/components/ui/card/card.md +1 -1
  99. package/src/core/components/ui/form/checkbox/checkbox.md +5 -32
  100. package/src/core/components/ui/form/input/input.md +5 -30
  101. package/src/core/components/ui/form/input-autocomplete/input-autocomplete.md +6 -4
  102. package/src/core/components/ui/form/radio/radio.md +5 -32
  103. package/src/core/components/ui/form/select/select.md +5 -31
  104. package/src/core/components/ui/form/switch/switch.md +5 -32
  105. package/src/core/components/ui/loader/loader.md +1 -13
  106. package/src/core/components/ui/table/table.md +3 -3
  107. package/src/core/decorators/api.spec.ts +8 -1
  108. package/src/core/decorators/api.ts +126 -15
  109. package/src/core/directives/DataProvider.sub.spec.ts +96 -0
  110. package/src/core/directives/DataProvider.ts +109 -40
  111. package/src/core/utils/HTML.ts +42 -10
  112. package/src/core/utils/PublisherProxy.ts +33 -18
  113. package/src/core/utils/dataProviderKey.ts +23 -0
  114. package/src/core/utils/publisherPathKey.spec.ts +58 -0
  115. package/src/docs/_core-concept/dataFlow.md +73 -0
  116. package/src/docs/_core-concept/subscriber.md +9 -10
  117. package/src/docs/_decorators/ancestor-attribute.md +4 -3
  118. package/src/docs/_decorators/auto-subscribe.md +19 -16
  119. package/src/docs/_decorators/bind.md +19 -16
  120. package/src/docs/_decorators/get.md +7 -4
  121. package/src/docs/_decorators/handle.md +15 -13
  122. package/src/docs/_decorators/on-assign.md +53 -53
  123. package/src/docs/_decorators/publish.md +2 -1
  124. package/src/docs/_decorators/subscribe.md +70 -9
  125. package/src/docs/_decorators/wait-for-ancestors.md +13 -10
  126. package/src/docs/_directives/sub.md +91 -0
  127. package/src/docs/_getting-started/ai-agents.md +56 -0
  128. package/src/docs/_getting-started/concorde-manual-install.md +133 -0
  129. package/src/docs/_getting-started/concorde-outside.md +13 -123
  130. package/src/docs/_getting-started/create-a-component.md +2 -0
  131. package/src/docs/_getting-started/my-first-component.md +236 -0
  132. package/src/docs/_getting-started/my-first-subscriber.md +29 -83
  133. package/src/docs/_getting-started/pubsub.md +21 -134
  134. package/src/docs/_getting-started/start.md +26 -18
  135. package/src/docs/_misc/api-configuration.md +79 -0
  136. package/src/docs/_misc/dataProviderKey.md +34 -1
  137. package/src/docs/_misc/docs-mock-api.md +60 -0
  138. package/src/docs/_misc/endpoint.md +2 -1
  139. package/src/docs/_misc/html-integration.md +13 -0
  140. package/src/docs/code.ts +58 -12
  141. package/src/docs/components/docs-demo-sources.ts +397 -0
  142. package/src/docs/components/docs-lit-demo-raw.ts +28 -0
  143. package/src/docs/components/docs-lit-demo.ts +166 -0
  144. package/src/docs/components/docs-source-link.ts +72 -0
  145. package/src/docs/docs-location.ts +54 -0
  146. package/src/docs/docs.ts +12 -0
  147. package/src/docs/example/decorators-demo-bind-demos.ts +41 -46
  148. package/src/docs/example/decorators-demo-geo.ts +16 -11
  149. package/src/docs/example/decorators-demo-init.ts +2 -228
  150. package/src/docs/example/decorators-demo-subscribe-publish-get-demos.ts +54 -14
  151. package/src/docs/example/decorators-demo.ts +71 -70
  152. package/src/docs/example/docs-api-config-demos.ts +234 -0
  153. package/src/docs/example/docs-joke-demos.ts +297 -0
  154. package/src/docs/example/docs-list-demos.ts +179 -0
  155. package/src/docs/example/docs-provider-keys.ts +315 -0
  156. package/src/docs/example/docs-queue-demos.ts +114 -0
  157. package/src/docs/example/docs-router-demos.ts +89 -0
  158. package/src/docs/example/docs-submit-demos.ts +455 -0
  159. package/src/docs/example/docs-toggle-demos.ts +73 -0
  160. package/src/docs/example/docs-user-two-scopes.ts +37 -0
  161. package/src/docs/example/docs-users-list.ts +71 -0
  162. package/src/docs/example/users.ts +41 -24
  163. package/src/docs/mock-api/api-config-mock.ts +152 -0
  164. package/src/docs/mock-api/fixtures.ts +377 -0
  165. package/src/docs/mock-api/register.ts +25 -0
  166. package/src/docs/mock-api/router.ts +234 -0
  167. package/src/docs/mock-api/service-worker.ts +23 -0
  168. package/src/docs/mock-api/urls.ts +11 -0
  169. package/src/docs/navigation/navigation.ts +39 -7
  170. package/src/docs/search/docs-search.json +4021 -936
  171. package/src/docs/search/markdown-renderer.ts +7 -3
  172. package/src/docs/search/page.ts +11 -14
  173. package/src/docs/search/sonic-code-markdown.spec.ts +29 -0
  174. package/src/docs/search/sonic-code-markdown.ts +28 -0
  175. package/src/docs.ts +4 -0
  176. package/src/tsconfig.json +87 -0
  177. package/src/tsconfig.tsbuildinfo +1 -1
  178. package/vite.config.mts +8 -0
  179. package/docs/assets/index-CaysOMFz.js +0 -5046
  180. package/docs/assets/index-D8mGoXzF.css +0 -1
  181. package/docs/src/docs/_misc/templates-demo.md +0 -19
  182. package/src/docs/_misc/templates-demo.md +0 -19
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../src/docs/mock-api/urls.ts", "../src/docs/mock-api/api-config-mock.ts", "../src/docs/mock-api/fixtures.ts", "../src/docs/mock-api/router.ts", "../src/docs/mock-api/service-worker.ts"],
4
+ "sourcesContent": ["/** Base same-origin pour les d\u00E9mos doc (Service Worker + middleware Vite). */\nexport const DOCS_MOCK_API_BASE = \"/docs-mock-api\";\n\n/** Remplace https://reqres.in \u2014 chemins `api/users`, `api/users/2`, etc. */\nexport const DOCS_MOCK_REQRES_SERVICE = DOCS_MOCK_API_BASE;\n\n/** Remplace https://geo.api.gouv.fr/ \u2014 chemins `communes?\u2026`. */\nexport const DOCS_MOCK_GEO_SERVICE = `${DOCS_MOCK_API_BASE}/geo/`;\n\n/** Remplace https://v2.jokeapi.dev \u2014 chemins `joke/Any?\u2026`. */\nexport const DOCS_MOCK_JOKES_SERVICE = `${DOCS_MOCK_API_BASE}/jokes`;\n", "import { DOCS_MOCK_API_BASE } from \"./urls\";\n\n/** Bearer accept\u00E9 par GET /api/config/protected */\nexport const DOCS_MOCK_TOKEN_VALID = \"docs-mock-valid-token\";\n/** D\u00E9clenche HTTP 498 \u2192 renouvellement via tokenProvider */\nexport const DOCS_MOCK_TOKEN_STALE = \"docs-mock-stale-token\";\n/** \u00C9mis par GET /auth/token apr\u00E8s Basic auth ou eventsApiToken */\nexport const DOCS_MOCK_TOKEN_FRESH = \"docs-mock-fresh-token\";\n\nconst DOCS_MOCK_BASIC_USER = \"demo\";\nconst DOCS_MOCK_BASIC_PASSWORD = \"secret\";\n/** Valeur de l\u2019attribut anc\u00EAtre `eventsApiToken` (Bearer pour /auth/token) */\nexport const DOCS_MOCK_EVENTS_API_TOKEN = \"docs-mock-events-token\";\n\nconst WORDING_CATALOG: Record<string, Record<string, string>> = {\n \"api-config.greeting\": { fr: \"Bonjour\", en: \"Hello\" },\n \"api-config.farewell\": { fr: \"Au revoir\", en: \"Goodbye\" },\n \"api-config.hint\": {\n fr: \"Libell\u00E9s servis par le mock /wording/labels\",\n en: \"Labels served by mock /wording/labels\",\n },\n};\n\nfunction json(body: unknown, init: ResponseInit = {}): Response {\n const headers = new Headers(init.headers);\n headers.set(\"Content-Type\", \"application/json; charset=utf-8\");\n headers.set(\"Access-Control-Allow-Origin\", \"*\");\n headers.set(\"X-Docs-Mock-Api\", \"concorde-api-config\");\n return new Response(JSON.stringify(body), { ...init, headers });\n}\n\ntype ParsedAuth =\n | { kind: \"none\" }\n | { kind: \"basic\"; user: string; password: string }\n | { kind: \"bearer\"; token: string };\n\nfunction parseAuthorization(request: Request): ParsedAuth {\n const header = request.headers.get(\"authorization\");\n if (!header) return { kind: \"none\" };\n const basic = header.match(/^Basic\\s+(.+)$/i);\n if (basic) {\n try {\n const decoded = atob(basic[1]);\n const sep = decoded.indexOf(\":\");\n if (sep < 0) return { kind: \"none\" };\n return {\n kind: \"basic\",\n user: decoded.slice(0, sep),\n password: decoded.slice(sep + 1),\n };\n } catch {\n return { kind: \"none\" };\n }\n }\n const bearer = header.match(/^Bearer\\s+(.+)$/i);\n if (bearer) return { kind: \"bearer\", token: bearer[1].trim() };\n return { kind: \"none\" };\n}\n\nfunction isValidBasic(auth: ParsedAuth): boolean {\n return (\n auth.kind === \"basic\" &&\n auth.user === DOCS_MOCK_BASIC_USER &&\n auth.password === DOCS_MOCK_BASIC_PASSWORD\n );\n}\n\nfunction wordingLabels(url: URL): Response {\n const lang = url.searchParams.get(\"lang\") || \"fr\";\n const labels = url.searchParams.getAll(\"labels[]\").length\n ? url.searchParams.getAll(\"labels[]\")\n : url.searchParams.getAll(\"labels\");\n const result: Record<string, string> = {};\n for (const key of labels) {\n const entry = WORDING_CATALOG[key];\n result[key] =\n entry?.[lang] ?? entry?.en ?? `[missing:${key}]`;\n }\n return json({ lang, labels: result, ...result });\n}\n\n/**\n * Routes mock pour la page doc APIConfiguration (token, 498, wording).\n */\nexport function handleApiConfigMockRequest(\n request: Request,\n subPath: string,\n method: string,\n): Response | null {\n if (subPath === \"/auth/token\" && method === \"GET\") {\n const auth = parseAuthorization(request);\n if (isValidBasic(auth)) {\n return json({\n token: DOCS_MOCK_TOKEN_FRESH,\n source: \"basic\",\n serviceHost: new URL(request.url).searchParams.get(\"serviceHost\"),\n });\n }\n if (\n auth.kind === \"bearer\" &&\n auth.token === DOCS_MOCK_EVENTS_API_TOKEN\n ) {\n return json({\n token: DOCS_MOCK_TOKEN_FRESH,\n source: \"eventsApiToken\",\n });\n }\n return json({ error: \"Unauthorized\" }, { status: 401 });\n }\n\n if (subPath === \"/api/config/protected\" && method === \"GET\") {\n const auth = parseAuthorization(request);\n if (auth.kind === \"bearer\") {\n if (auth.token === DOCS_MOCK_TOKEN_STALE) {\n return json({ error: \"Token expired\", code: \"token_stale\" }, { status: 498 });\n }\n if (\n auth.token === DOCS_MOCK_TOKEN_VALID ||\n auth.token === DOCS_MOCK_TOKEN_FRESH\n ) {\n return json({\n ok: true,\n message: \"Protected resource\",\n tokenUsed: auth.token,\n });\n }\n }\n if (isValidBasic(auth)) {\n return json({\n ok: true,\n message: \"Protected resource (basic)\",\n tokenUsed: null,\n });\n }\n return json({ error: \"Unauthorized\" }, { status: 401 });\n }\n\n if (\n (subPath === \"/wording/labels\" || subPath.startsWith(\"/wording/labels\")) &&\n method === \"GET\"\n ) {\n return wordingLabels(new URL(request.url));\n }\n\n return null;\n}\n\nexport const API_CONFIG_MOCK_PATHS = [\n `${DOCS_MOCK_API_BASE}/auth/token`,\n `${DOCS_MOCK_API_BASE}/api/config/protected`,\n `${DOCS_MOCK_API_BASE}/wording/labels`,\n] as const;\n", "/** Donn\u00E9es type ReqRes pour les d\u00E9mos doc (sans appel externe). */\n\nexport type ReqresUser = {\n id: number;\n email: string;\n first_name: string;\n last_name: string;\n avatar: string;\n};\n\nconst reqresAvatar = (seed: string) =>\n `https://i.pravatar.cc/150?u=${encodeURIComponent(seed)}`;\n\nconst REQRES_EXTRA_NAMES: [string, string][] = [\n [\"Michael\", \"Lawson\"],\n [\"Tobias\", \"Funke\"],\n [\"Lindsay\", \"Funke\"],\n [\"Maeby\", \"Funke\"],\n [\"George\", \"Michael\"],\n [\"Annye\", \"Vang\"],\n [\"Rachel\", \"Howe\"],\n [\"Terry\", \"Medhurst\"],\n [\"Bette\", \"Haag\"],\n [\"Lester\", \"Solomon\"],\n [\"April\", \"Douglas\"],\n [\"Marian\", \"Roberts\"],\n [\"Kathryn\", \"Murphy\"],\n [\"Gloria\", \"Armstrong\"],\n [\"Adam\", \"Bradley\"],\n [\"Angela\", \"Hicks\"],\n [\"Rose\", \"Wheeler\"],\n [\"Sara\", \"Berry\"],\n [\"Glen\", \"Kuhn\"],\n [\"Roger\", \"Cox\"],\n];\n\nfunction buildReqresUser(\n id: number,\n first_name: string,\n last_name: string,\n): ReqresUser {\n const slug = `${first_name}-${last_name}`.toLowerCase();\n return {\n id,\n email: `${first_name.toLowerCase()}.${last_name.toLowerCase()}@example.com`,\n first_name,\n last_name,\n avatar: reqresAvatar(slug),\n };\n}\n\n/** Jeu \u00E9tendu pour d\u00E9mos queue / lazyload (pagination par offset). */\nexport const DOCS_REQRES_USERS: ReqresUser[] = [\n buildReqresUser(1, \"George\", \"Bluth\"),\n buildReqresUser(2, \"Janet\", \"Weaver\"),\n buildReqresUser(3, \"Emma\", \"Wong\"),\n buildReqresUser(4, \"Eve\", \"Holt\"),\n ...REQRES_EXTRA_NAMES.map(([first_name, last_name], index) =>\n buildReqresUser(5 + index, first_name, last_name),\n ),\n];\n\nexport type ReqresUserResponse = { data: ReqresUser };\nexport type ReqresUsersListResponse = {\n page: number;\n per_page: number;\n total: number;\n total_pages: number;\n data: ReqresUser[];\n};\n\n/** Filter users for mock search (`q` \u2014 first name, last name, email). Same logic as the TS starter. */\nexport function filterDocsUsers(\n users: ReqresUser[],\n query: string | null | undefined,\n): ReqresUser[] {\n const q = query?.trim().toLowerCase();\n if (!q) return users;\n return users.filter((user) => {\n const haystack =\n `${user.first_name} ${user.last_name} ${user.email}`.toLowerCase();\n return haystack.includes(q);\n });\n}\n\nexport type GeoCommuneRow = { nom: string; code: string };\n\nexport const DOCS_GEO_COMMUNES: GeoCommuneRow[] = [\n { nom: \"Paris\", code: \"75056\" },\n { nom: \"Lyon\", code: \"69123\" },\n { nom: \"Marseille\", code: \"13055\" },\n { nom: \"Toulouse\", code: \"31555\" },\n { nom: \"Lille\", code: \"59350\" },\n { nom: \"Bordeaux\", code: \"33063\" },\n { nom: \"Nantes\", code: \"44109\" },\n { nom: \"Strasbourg\", code: \"67482\" },\n { nom: \"Montpellier\", code: \"34172\" },\n { nom: \"Rennes\", code: \"35238\" },\n];\n\n/** JokeAPI-style content flags (checkbox / radio / switch blacklist demos). */\nexport type DocsJokeFlagKey =\n | \"nsfw\"\n | \"religious\"\n | \"political\"\n | \"racist\"\n | \"sexist\"\n | \"explicit\";\n\n/** Forme proche de JokeAPI v2 (`key=\"jokes\"` sur sonic-queue / list). */\nexport type DocsJokeItem = {\n categories: string[];\n createdAt: string;\n id: number;\n joke: string;\n lang: \"fr\" | \"en\";\n flags: Record<DocsJokeFlagKey, boolean>;\n safe: boolean;\n type: string;\n setup?: string;\n delivery?: string;\n};\n\nconst jokeFlags = (\n partial: Partial<Record<DocsJokeFlagKey, boolean>> = {},\n): Record<DocsJokeFlagKey, boolean> => ({\n nsfw: false,\n religious: false,\n political: false,\n racist: false,\n sexist: false,\n explicit: false,\n ...partial,\n});\n\nexport type DocsJokeApiResponse = {\n error: boolean;\n amount: number;\n jokes: DocsJokeItem[];\n};\n\nconst SAMPLE_JOKES: DocsJokeItem[] = [\n {\n categories: [\"Programming\"],\n createdAt: \"2020-01-01 00:00:00\",\n id: 1,\n lang: \"en\",\n joke: \"Why do programmers prefer dark mode? Because light attracts bugs.\",\n flags: jokeFlags(),\n safe: true,\n type: \"single\",\n },\n {\n categories: [\"Misc\"],\n createdAt: \"2020-01-02 00:00:00\",\n id: 2,\n lang: \"en\",\n joke: \"I told my computer I needed a break \u2014 it said: no problem, I'll go to sleep.\",\n flags: jokeFlags({ nsfw: true }),\n safe: false,\n type: \"single\",\n },\n {\n categories: [\"Pun\"],\n createdAt: \"2020-01-03 00:00:00\",\n id: 3,\n lang: \"en\",\n joke: \"Concorde docs run offline: even the API is a local joke.\",\n flags: jokeFlags(),\n safe: true,\n type: \"single\",\n },\n {\n categories: [\"Animals\"],\n createdAt: \"2020-01-04 00:00:00\",\n id: 4,\n lang: \"en\",\n joke: \"The dog ate my homework and my pull request.\",\n flags: jokeFlags({ political: true }),\n safe: true,\n type: \"single\",\n },\n {\n categories: [\"Animals\", \"Pun\"],\n createdAt: \"2020-01-05 00:00:00\",\n id: 5,\n lang: \"fr\",\n joke: \"Quel est le chien pr\u00E9f\u00E9r\u00E9 des d\u00E9veloppeurs ? Le labrador Retriever.\",\n flags: jokeFlags(),\n safe: true,\n type: \"single\",\n },\n {\n categories: [\"Animals\"],\n createdAt: \"2020-01-06 00:00:00\",\n id: 6,\n lang: \"fr\",\n joke: \"Un chat et un chien discutent de TypeScript : le chien dit \u00AB woof \u00BB, le chat compile.\",\n flags: jokeFlags({ racist: true }),\n safe: false,\n type: \"single\",\n },\n {\n categories: [\"Misc\"],\n createdAt: \"2020-01-07 00:00:00\",\n id: 7,\n lang: \"fr\",\n joke: \"Pourquoi les blagues offline sont-elles fiables ? Parce qu'elles passent par le Service Worker.\",\n flags: jokeFlags({ religious: true }),\n safe: true,\n type: \"single\",\n },\n {\n categories: [\"Programming\"],\n createdAt: \"2020-01-08 00:00:00\",\n id: 8,\n lang: \"fr\",\n joke: \"Mon compilateur et mon chien : tous deux ignorent mes commandes jusqu'au build.\",\n flags: jokeFlags({ sexist: true }),\n safe: false,\n type: \"single\",\n },\n {\n categories: [\"Misc\"],\n createdAt: \"2020-01-09 00:00:00\",\n id: 9,\n lang: \"en\",\n joke: \"A SQL query walks into a bar, walks up to two tables and asks: Can I join you?\",\n flags: jokeFlags({ explicit: true }),\n safe: false,\n type: \"single\",\n },\n {\n categories: [\"Pun\"],\n createdAt: \"2020-01-10 00:00:00\",\n id: 10,\n lang: \"fr\",\n joke: \"J'ai cherch\u00E9 \u00AB chien \u00BB dans la doc : le mock API a r\u00E9pondu.\",\n flags: jokeFlags(),\n safe: true,\n type: \"single\",\n },\n {\n categories: [\"Misc\"],\n createdAt: \"2020-01-11 00:00:00\",\n id: 11,\n lang: \"en\",\n joke: \"There are only two hard things in Computer Science: cache invalidation and naming things.\",\n flags: jokeFlags({ political: true, explicit: true }),\n safe: false,\n type: \"single\",\n },\n {\n categories: [\"Animals\"],\n createdAt: \"2020-01-12 00:00:00\",\n id: 12,\n lang: \"fr\",\n joke: \"Le golden retriever a r\u00E9cup\u00E9r\u00E9 la donn\u00E9e du publisher.\",\n flags: jokeFlags({ racist: true, sexist: true }),\n safe: false,\n type: \"two-liner\",\n setup: \"Pourquoi le chien aime Concorde ?\",\n delivery: \"Parce qu'il sait fetch.\",\n },\n {\n categories: [\"Misc\"],\n createdAt: \"2020-01-13 00:00:00\",\n id: 13,\n lang: \"en\",\n joke: \"My code and my conscience: both have too many warnings.\",\n flags: jokeFlags(),\n safe: true,\n type: \"single\",\n },\n {\n categories: [\"Programming\"],\n createdAt: \"2020-01-14 00:00:00\",\n id: 14,\n lang: \"en\",\n joke: \"Git blame is just a mirror.\",\n flags: jokeFlags({ religious: true, political: true }),\n safe: false,\n type: \"single\",\n },\n];\n\n/** `blacklistFlags` from queue filter form (comma-separated or repeated param). */\nexport function parseBlacklistFlags(url: URL): DocsJokeFlagKey[] {\n const keys = new Set<DocsJokeFlagKey>();\n const valid: DocsJokeFlagKey[] = [\n \"nsfw\",\n \"religious\",\n \"political\",\n \"racist\",\n \"sexist\",\n \"explicit\",\n ];\n const add = (raw: string) => {\n raw\n .split(\",\")\n .map((s) => s.trim().toLowerCase())\n .forEach((part) => {\n if (valid.includes(part as DocsJokeFlagKey)) keys.add(part as DocsJokeFlagKey);\n });\n };\n url.searchParams.getAll(\"blacklistFlags\").forEach(add);\n const single = url.searchParams.get(\"blacklistFlags\");\n if (single) add(single);\n return [...keys];\n}\n\n/** Exclude jokes that match any blacklisted flag (JokeAPI-style). */\nexport function applyJokeBlacklist(\n jokes: DocsJokeItem[],\n blacklist: DocsJokeFlagKey[],\n): DocsJokeItem[] {\n if (!blacklist.length) return jokes;\n return jokes.filter(\n (j) => !blacklist.some((flag) => j.flags[flag] === true),\n );\n}\n\n/** Filtre doc joke queue (`contains`, `lang`, `blacklistFlags` via `dataFilterProvider`). */\nexport function filterDocsJokes(\n jokes: DocsJokeItem[],\n url: URL,\n): DocsJokeItem[] {\n let pool = jokes;\n const contains = url.searchParams.get(\"contains\")?.trim().toLowerCase();\n if (contains) {\n pool = pool.filter((j) => {\n const haystack = [j.joke, j.setup, j.delivery, ...j.categories]\n .filter(Boolean)\n .join(\" \")\n .toLowerCase();\n return haystack.includes(contains);\n });\n }\n const lang = url.searchParams.get(\"lang\")?.trim().toLowerCase();\n if (lang === \"fr\" || lang === \"en\") {\n pool = pool.filter((j) => j.lang === lang);\n }\n pool = applyJokeBlacklist(pool, parseBlacklistFlags(url));\n return pool;\n}\n\n/**\n * R\u00E9ponse type JokeAPI v2.\n * - `amount` : taille du lot si pas de `offset`\n * - `offset` + `limit` / `per_page` : pagination (sonic-queue lazy)\n * - `contains`, `lang`, `blacklistFlags` : filtres formulaire (queue \u00AB Remove following jokes \u00BB)\n */\nexport function buildJokeApiResponse(url: URL): DocsJokeApiResponse {\n const pool = filterDocsJokes(SAMPLE_JOKES, url);\n const amount = Math.max(\n 1,\n parseInt(url.searchParams.get(\"amount\") || \"10\", 10),\n );\n const hasOffset = url.searchParams.has(\"offset\");\n const limit = Math.max(\n 1,\n parseInt(\n url.searchParams.get(\"limit\") ||\n url.searchParams.get(\"per_page\") ||\n String(amount),\n 10,\n ),\n );\n const offset = Math.max(0, parseInt(url.searchParams.get(\"offset\") || \"0\", 10));\n const start = hasOffset ? offset : 0;\n const count = hasOffset ? limit : amount;\n const jokes = pool.slice(start, start + count).map((j, i) => ({\n ...j,\n id: start + i + 1,\n }));\n return { error: false, amount: jokes.length, jokes };\n}\n", "import { handleApiConfigMockRequest } from \"./api-config-mock\";\nimport {\n DOCS_GEO_COMMUNES,\n DOCS_REQRES_USERS,\n buildJokeApiResponse,\n filterDocsUsers,\n type GeoCommuneRow,\n type ReqresUserResponse,\n type ReqresUsersListResponse,\n} from \"./fixtures\";\n\nexport const DOCS_MOCK_API_PREFIX = \"/docs-mock-api\";\n\nfunction json(\n body: unknown,\n init: ResponseInit & { delayMs?: number } = {},\n): Promise<Response> {\n const { delayMs = 0, ...responseInit } = init;\n const headers = new Headers(responseInit.headers);\n headers.set(\"Content-Type\", \"application/json; charset=utf-8\");\n headers.set(\"Access-Control-Allow-Origin\", \"*\");\n headers.set(\"X-Docs-Mock-Api\", \"concorde\");\n\n const build = () =>\n new Response(JSON.stringify(body), { ...responseInit, headers });\n\n if (delayMs <= 0) return Promise.resolve(build());\n return new Promise((resolve) =>\n setTimeout(() => resolve(build()), delayMs),\n );\n}\n\n/**\n * Liste utilisateurs type ReqRes.\n * - `offset` + `per_page` / `limit` : pagination index\u00E9e (sonic-queue avec `$offset` / `$limit`)\n * - `page` + `per_page` : num\u00E9ro de page 1-based (sonic-fetch, ex. `?page=2`)\n * - `q` : recherche sur pr\u00E9nom, nom, email (sonic-queue + `dataFilterProvider`, champ `name=\"q\"`)\n */\nexport function paginateUsers(url: URL): ReqresUsersListResponse {\n const perPage = Math.max(\n 1,\n parseInt(\n url.searchParams.get(\"per_page\") ||\n url.searchParams.get(\"limit\") ||\n \"6\",\n 10,\n ),\n );\n const pool = filterDocsUsers(DOCS_REQRES_USERS, url.searchParams.get(\"q\"));\n const total = pool.length;\n\n let start: number;\n let page: number;\n\n if (url.searchParams.has(\"offset\")) {\n start = Math.max(0, parseInt(url.searchParams.get(\"offset\") || \"0\", 10));\n page = Math.floor(start / perPage) + 1;\n } else {\n page = Math.max(1, parseInt(url.searchParams.get(\"page\") || \"1\", 10));\n start = (page - 1) * perPage;\n }\n\n const data = pool.slice(start, start + perPage);\n\n return {\n page,\n per_page: perPage,\n total,\n total_pages: Math.max(1, Math.ceil(total / perPage)),\n data,\n };\n}\n\nasync function parseRegisterFields(\n request: Request,\n): Promise<{ email?: string; password?: string }> {\n const contentType = (request.headers.get(\"content-type\") || \"\").toLowerCase();\n\n if (contentType.includes(\"multipart/form-data\")) {\n const fd = await request.formData();\n return {\n email: String(fd.get(\"email\") ?? \"\"),\n password: String(fd.get(\"password\") ?? \"\"),\n };\n }\n\n if (contentType.includes(\"application/x-www-form-urlencoded\")) {\n const text = await request.text();\n const params = new URLSearchParams(text);\n return {\n email: params.get(\"email\") ?? \"\",\n password: params.get(\"password\") ?? \"\",\n };\n }\n\n const text = await request.text();\n if (!text.trim()) return {};\n\n try {\n const parsed = JSON.parse(text) as { email?: unknown; password?: unknown };\n return {\n email: parsed.email != null ? String(parsed.email) : \"\",\n password: parsed.password != null ? String(parsed.password) : \"\",\n };\n } catch {\n const params = new URLSearchParams(text);\n if (params.has(\"email\") || params.has(\"password\")) {\n return {\n email: params.get(\"email\") ?? \"\",\n password: params.get(\"password\") ?? \"\",\n };\n }\n return {};\n }\n}\n\nfunction registerOkBody(\n fields: { email: string; password: string },\n nested = false,\n) {\n const payload = {\n id: 4,\n token: \"docs-mock-register-token\",\n email: fields.email,\n };\n return nested ? { data: payload } : payload;\n}\n\nfunction geoCommunes(url: URL): GeoCommuneRow[] {\n const limit = Math.max(\n 1,\n parseInt(url.searchParams.get(\"limit\") || \"5\", 10),\n );\n const offset = Math.max(0, parseInt(url.searchParams.get(\"offset\") || \"0\", 10));\n return DOCS_GEO_COMMUNES.slice(offset, offset + limit);\n}\n\n/**\n * Intercepte les requ\u00EAtes vers /docs-mock-api/* (Service Worker ou middleware Vite).\n */\nexport async function handleDocsMockApiRequest(\n request: Request,\n): Promise<Response | null> {\n const url = new URL(request.url);\n if (!url.pathname.startsWith(DOCS_MOCK_API_PREFIX)) return null;\n\n const subPath = url.pathname.slice(DOCS_MOCK_API_PREFIX.length) || \"/\";\n const method = request.method.toUpperCase();\n\n if (subPath === \"/health\") {\n return json({ ok: true, service: \"concorde-docs-mock-api\" });\n }\n\n const apiConfigResponse = handleApiConfigMockRequest(request, subPath, method);\n if (apiConfigResponse) return apiConfigResponse;\n\n if (\n (subPath === \"/api/register\" || subPath === \"/api/register/nested\") &&\n method === \"POST\"\n ) {\n const body = await parseRegisterFields(request);\n if (!body.email?.trim() || !body.password?.trim()) {\n return json(\n {\n messages: [\n {\n content: \"email and password are required\",\n status: \"error\",\n },\n ],\n },\n { status: 400 },\n );\n }\n return json(\n registerOkBody(\n { email: body.email!.trim(), password: body.password!.trim() },\n subPath === \"/api/register/nested\",\n ),\n );\n }\n\n if (subPath === \"/api/register/echo\" && method === \"GET\") {\n const email = url.searchParams.get(\"email\") ?? \"\";\n const password = url.searchParams.get(\"password\") ?? \"\";\n if (!email.trim() || !password.trim()) {\n return json(\n {\n messages: [\n {\n content: \"email and password query params are required\",\n status: \"error\",\n },\n ],\n },\n { status: 400 },\n );\n }\n return json({\n method: \"GET\",\n query: { email, password },\n email,\n note: \"sonic-submit method=\\\"get\\\" appends form fields to the endpoint URL\",\n });\n }\n\n if (method !== \"GET\") {\n return json({ error: \"Method not allowed\" }, { status: 405 });\n }\n\n const userMatch = subPath.match(/^\\/api\\/users\\/(\\d+)$/);\n if (userMatch) {\n const id = parseInt(userMatch[1], 10);\n const user =\n DOCS_REQRES_USERS.find((u) => u.id === id) ?? DOCS_REQRES_USERS[0];\n const body: ReqresUserResponse = { data: { ...user, id } };\n return json(body);\n }\n\n if (subPath === \"/api/users\") {\n return json(paginateUsers(url));\n }\n\n if (subPath === \"/geo/communes\" || subPath.startsWith(\"/geo/communes\")) {\n return json(geoCommunes(url));\n }\n\n const jokeMatch = subPath.match(/^\\/jokes\\/joke\\/([^/]+)$/);\n if (jokeMatch) {\n return json(buildJokeApiResponse(url));\n }\n\n return json({ error: \"Not found\", path: subPath }, { status: 404 });\n}\n", "/// <reference lib=\"webworker\" />\nimport { handleDocsMockApiRequest} from \"./router\";\n\ndeclare const self: ServiceWorkerGlobalScope;\n\nself.addEventListener(\"install\", (event) => {\n event.waitUntil(self.skipWaiting());\n});\n\nself.addEventListener(\"activate\", (event) => {\n event.waitUntil(self.clients.claim());\n});\n\nself.addEventListener(\"fetch\", (event) => {\n const {request} = event;\n event.respondWith(\n (async () => {\n const mocked = await handleDocsMockApiRequest(request);\n if (mocked) return mocked;\n return fetch(request);\n })(),\n );\n});\n"],
5
+ "mappings": "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",
6
+ "names": ["json", "text"]
7
+ }
package/docs/index.html CHANGED
@@ -10,8 +10,8 @@
10
10
  <!-- <script src="https://cdn.jsdelivr.net/npm/marked/marked.min.js"></script>
11
11
  -->
12
12
  <script src="https://cdn.jsdelivr.net/npm/marked@13.0.3"></script>
13
- <script type="module" crossorigin src="./assets/index-CaysOMFz.js"></script>
14
- <link rel="stylesheet" crossorigin href="./assets/index-D8mGoXzF.css">
13
+ <script type="module" crossorigin src="./assets/index-D9pxaQYK.js"></script>
14
+ <link rel="stylesheet" crossorigin href="./assets/index-t0-i22oI.css">
15
15
  </head>
16
16
  <body class="px-6 lg:px-20">
17
17
  <sonic-theme
@@ -1,6 +1,8 @@
1
1
  # Fetch
2
- The **sonic-fetch** component is used to request and store data from an API.
3
- Fetch extends the mixins Fetcher and [Subscriber](#docs/_core-concept/subscriber.md/subscriber)
2
+
3
+ > **New apps:** prefer [@get](#docs/_decorators/get.md/get) for a typed GET on a component, or [List](#core/components/functional/list/list.md/list) / [Queue](#core/components/functional/queue/queue.md/queue) with `fetch` for collections. Use [Local API demos](#docs/_misc/docs-mock-api.md/docs-mock-api) (`serviceURL="/docs-mock-api"`) to try examples offline.
4
+
5
+ The **sonic-fetch** component requests and stores API data. It extends the Fetcher and [Subscriber](#docs/_core-concept/subscriber.md/subscriber) mixins.
4
6
 
5
7
 
6
8
 
@@ -9,7 +11,7 @@ Fetch extends the mixins Fetcher and [Subscriber](#docs/_core-concept/subscriber
9
11
  ## Basic usage
10
12
  In order to work properly the <b>sonic-fetch</b> component needs at least the following attributes.
11
13
  - **serviceURL** : A base service url. This attribute can be inherited from an ancestor.
12
- *ex : https://reqres.in*
14
+ *ex : /docs-mock-api*
13
15
  - **endPoint** : the specific location where requests for information are sent (see the api docs).
14
16
  *ex : api/users | api/users?page=2 | api/users/2*
15
17
  - **dataProvider *(Required)*** : An ID that is used as a reference to the object storing the data returned by the API.
@@ -19,7 +21,7 @@ This attribute can be inherited from an ancestor.
19
21
 
20
22
  <sonic-code>
21
23
  <template>
22
- <sonic-fetch serviceURL="https://reqres.in" endPoint="api/users?page=2" dataProvider="myDataObj"></sonic-fetch>
24
+ <sonic-fetch serviceURL="/docs-mock-api" endPoint="api/users?page=2" dataProvider="myDataObj"></sonic-fetch>
23
25
  <sonic-button dataProvider="myDataObj" debug>Hover to see the data</sonic-button>
24
26
  </template>
25
27
  </sonic-code>
@@ -29,7 +31,7 @@ If no **endPoint** is specified it will be filled by the **dataProvider ID** ins
29
31
 
30
32
  <sonic-code>
31
33
  <template>
32
- <sonic-fetch serviceURL="https://reqres.in" dataProvider="api/users?page=2" ></sonic-fetch>
34
+ <sonic-fetch serviceURL="/docs-mock-api" dataProvider="api/users?page=2" ></sonic-fetch>
33
35
  <sonic-button dataProvider="api/users?page=2" debug>Hover to see the data</sonic-button>
34
36
  </template>
35
37
  </sonic-code>
@@ -46,19 +48,19 @@ For example if the data is `{my:{data:{a:1,b:2}}}` and the key is `key="my.data"
46
48
 
47
49
  <sonic-code>
48
50
  <template>
49
- <sonic-fetch serviceURL="https://reqres.in" dataProvider="api/users/2" ></sonic-fetch>
51
+ <sonic-fetch serviceURL="/docs-mock-api" dataProvider="api/users/2" ></sonic-fetch>
50
52
  <sonic-button dataProvider="api/users/2" debug>dataProvider object</sonic-button>
51
53
  <!-- Get the user ID -->
52
- <sonic-fetch serviceURL="https://reqres.in" dataProvider="id" endPoint="api/users/2" key="data.id"></sonic-fetch>
54
+ <sonic-fetch serviceURL="/docs-mock-api" dataProvider="id" endPoint="api/users/2" key="data.id"></sonic-fetch>
53
55
  <sonic-button dataProvider="id" debug>data.id</sonic-button>
54
56
  <!-- Get the user First name -->
55
- <sonic-fetch serviceURL="https://reqres.in" dataProvider="first_name" endPoint="api/users/2" key="data.first_name"></sonic-fetch>
57
+ <sonic-fetch serviceURL="/docs-mock-api" dataProvider="first_name" endPoint="api/users/2" key="data.first_name"></sonic-fetch>
56
58
  <sonic-button dataProvider="first_name" debug>data.first_name</sonic-button>
57
59
  <!-- Get the user Last name -->
58
- <sonic-fetch serviceURL="https://reqres.in" dataProvider="last_name" endPoint="api/users/2" key="data.last_name"></sonic-fetch>
60
+ <sonic-fetch serviceURL="/docs-mock-api" dataProvider="last_name" endPoint="api/users/2" key="data.last_name"></sonic-fetch>
59
61
  <sonic-button dataProvider="last_name" debug>data.last_name</sonic-button>
60
62
  <!-- Get the user email -->
61
- <sonic-fetch serviceURL="https://reqres.in" dataProvider="email" endPoint="api/users/2" key="data.email"></sonic-fetch>
63
+ <sonic-fetch serviceURL="/docs-mock-api" dataProvider="email" endPoint="api/users/2" key="data.email"></sonic-fetch>
62
64
  <sonic-button dataProvider="email" debug>data.email</sonic-button>
63
65
  </template>
64
66
  </sonic-code>
@@ -80,7 +82,7 @@ The noLoader attribute disables display of the default loader
80
82
 
81
83
  <sonic-code>
82
84
  <template>
83
- <sonic-fetch noLoader serviceURL="https://reqres.in" endPoint="api/users?page=2" dataProvider="myDataObj"></sonic-fetch>
85
+ <sonic-fetch noLoader serviceURL="/docs-mock-api" endPoint="api/users?page=2" dataProvider="myDataObj"></sonic-fetch>
84
86
  <sonic-button dataProvider="myDataObj" debug>Basic fetch with noLoader attribute</sonic-button>
85
87
  </template>
86
88
  </sonic-code>
@@ -1,16 +1,9 @@
1
1
  # if
2
2
 
3
- The **sonic-if** component shows its content based on the value of its reactive property names **condition**
3
+ The **sonic-if** component shows its slot when **`.condition`** is true. In Lit, bind `.condition` from store-driven state (`@subscribe` on a `formDataProvider` field) live preview and **TypeScript source are the same file** (`docs-lit-demo`):
4
4
 
5
- <sonic-code>
6
- <template>
7
- <div dataProvider="visibilitySettings" formDataProvider="visibilitySettings">
8
- <sonic-checkbox label="Show evacuation plan" name="togglePlan" unique checked value="true"></sonic-checkbox>
9
- <sonic-if data-bind ::condition="$togglePlan" class="mt-4 block">
10
- <sonic-image src="https://www.thebaron.info/assets/mail/concorde-evacuation.jpg" rounded="md" ratio="654/463"></sonic-image>
11
- </sonic-if>
12
- </div>
13
- </template>
14
- </sonic-code>
5
+ <docs-lit-demo for="docs-toggle-if-demo"></docs-lit-demo>
6
+
7
+ Plain HTML without Lit: [HTML integration](#docs/_misc/html-integration.md/html-integration).
15
8
 
16
9
 
@@ -1,199 +1,65 @@
1
1
  # List
2
2
 
3
- The **sonic-list** component creates list items.
4
-
5
-
6
- List extends the mixins [Subscriber](#docs/_core-concept/subscriber.md/subscriber) and Fetcher :
7
- * As a Subscriber it as a **props** reactive property (basically an array of data) and its data is associated to a publisher via its dataProvider attribute.
8
- * List doesn't Fetch by default (see below for activation).
9
-
10
-
11
- ## Template
12
-
13
- The list component loops over its template children to render each items of its props data.
14
- Consider the following example using **2 templates** and **9 items** :
15
- - First template will render the first item.
16
- - Second template will render the second item.
17
- - Then back to the first template and so on and so forth.
18
-
19
- Note that for each line there is a dataProvider found at *[list dataProvider name]/liste-item/[line index or key]*
20
- You can hover the list items in the examples to see it.
21
-
22
- <sonic-code>
23
- <template>
24
- <sonic-list dataProvider="listTemplateExample" class="grid grid-cols-3 gap-4" props='[{"id": "1"}, {"id": "2"}, {"id": "3"}, {"id": "4"}, {"id": "5"}, {"id": "6"}, {"id": "7"}, {"id": "8"}, {"id": "9"}]' debug>
25
- <template>
26
- <div class="bg-neutral-100 text-center p-3">
27
- <sonic-value key="id"></sonic-value>
28
- <div class="text-xs">1st template</div>
29
- </div>
30
- </template>
31
- <template>
32
- <div class="bg-neutral-100 text-info text-center p-3">
33
- <sonic-value key="id"></sonic-value>
34
- <div class="text-xs">2nd template</div>
35
- </div>
36
- </template>
37
- </sonic-list>
38
- </template>
39
- </sonic-code>
40
-
41
-
42
-
43
- ## TemplateKey / data-value
44
-
45
- The **templateKey** attribute allows you to bind a template to a props item.
46
- Consider the following example :
47
- - The list **templateKey** attribute is set to the value **tpl** (any name would do)
48
- - Items 4,5 and 6 are each one **referencing** a templates with a matching **data-value** attribute
49
-
50
- <sonic-code>
51
- <template>
52
- <sonic-list
53
- templateKey="tpl"
54
- class="grid gap-3"
55
- dataProvider="TemplateKeyExample"
56
- props='[
57
- {"id":"1", "default": "The first template with no data-value is used"},
58
- {"id":"2", "default": "The second template with no data-value is used"},
59
- {"id":"3", "default": "Back to the first template with no data-value"},
60
- {"id":"4", "tpl":"custom-tpl-danger"},
61
- {"id":"5", "tpl":"custom-tpl-info"},
62
- {"id":"6", "tpl":"custom-tpl-success"},
63
- {"id":"7", "tpl": "Non-valid tpl name, Back to the first template with no data-value"}
64
- ]'
65
- debug
66
- >
67
- <template>
68
- <div class="p-2 border rounded text-neutral-900">
69
- <sonic-value key="id"> : </sonic-value>
70
- <b>First</b> template with no <b>data-value</b> attribute
71
- </div>
72
- </template>
73
- <template data-value="custom-tpl-danger">
74
- <div class="p-2 border rounded text-danger">
75
- <sonic-value key="id"> : </sonic-value>
76
- data-value : <b>custom-tpl-danger</b>
77
- </div>
78
- </template>
79
- <template data-value="custom-tpl-info">
80
- <div class="p-2 border rounded text-info">
81
- <sonic-value key="id"> : </sonic-value>
82
- data-value : <b>custom-tpl-info</b>
83
- </div>
84
- </template>
85
- <template data-value="custom-tpl-success">
86
- <div class="p-2 border rounded text-success">
87
- <sonic-value key="id"> : </sonic-value>
88
- data-value : <b>custom-tpl-success</b>
89
- </div>
90
- </template>
91
- <template>
92
- <div class="p-2 border rounded text-neutral-400">
93
- <sonic-value key="id"> : </sonic-value>
94
- <b>Second</b> template with no <b>data-value</b> attribute <br>
95
- This one is used if it <b>follows another data-item</b> with <b>no tpl</b> specified in the props
96
- </div>
97
- </template>
98
- </sonic-list>
99
- </template>
100
- </sonic-code>
101
-
102
- ## Special templates : list item separator / empty list view
103
-
104
- A special template with attribute **data-value="separator"** will act as a separator between each list item
105
-
106
- <sonic-code>
107
- <template>
108
- <sonic-list props='[{"id":"1"},{"id":"2"},{"id":"3"}]' dataProvider="ListSeparatorDemo">
109
- <template><sonic-value key="id"></sonic-value></template>
110
- <template data-value="separator"> 🤜 </template>
111
- </sonic-list>
112
- </template>
113
- </sonic-code>
114
-
115
- The same principle can be used to handle empty lists using a template with attribute **data-value="no-item"**
116
-
117
-
118
- ## Fetch
119
-
120
- Enables the list to get data from an external API in order to fill its **props** attribute with an **array of items**
121
- See the [Fetch] web component(#core/components/functional/fetch/fetch.md/fetch)
122
- <sonic-code>
123
- <template>
124
- <sonic-list
125
- props='["a", "b", "c"]' fetch
126
- serviceURL="https://reqres.in"
127
- dataProvider="api/users"
128
- key="data"
129
- class="grid grid-cols-1" debug>
130
- <template>
131
- <docs-user></docs-user>
132
- </template>
133
- </sonic-list>
134
- </template>
135
- </sonic-code>
136
-
137
- ## Extract Values
138
-
139
- Example of using the `extractValues` attribute with a service.
140
- As in the previous example, the `fetch` attribute indicates that a service call should be made.
141
- Note that we use:
142
- * the property `_metadata_` added by the list component to display the key of the extracted value
143
- * the special property `_self_` that allows targeting the item itself. This is useful here because there is no sub-property; we are directly dealing with a string.
144
- <sonic-code>
145
- <template>
146
- <sonic-list
147
- debug
148
- fetch
149
- serviceURL="https://reqres.in"
150
- dataProvider="list-extract-values-test"
151
- endPoint="api/users/2"
152
- key="data"
153
- extractValues
154
- >
155
- <template>
156
- <div class="flex items-center">
157
- <span data-bind ::inner-html="$_metadata_.key : " class="bold w-24"></span>
158
- <span data-bind ::inner-html="$_self_"></span>
159
- <sonic-if data-bind ::condition="|'$_metadata_.key' == 'avatar'">
160
- <sonic-image data-bind ::src="$_self_" rounded="full" ratio="1/1" class="w-20 block"></sonic-image>
161
- </sonic-if>
162
- </div>
163
- </template>
164
- </sonic-list>
165
- </template>
166
- </sonic-code>
3
+ > **Try offline:** `serviceURL="/docs-mock-api"` and `dataProvider="api/users"` with `key="data"` — see [Local API demos](#docs/_misc/docs-mock-api.md/docs-mock-api). Recommended patterns: [Data flow](#docs/_core-concept/dataFlow.md/dataFlow).
4
+
5
+ The **sonic-list** component renders one row per entry in **`props`** (array from fetch or set on the element).
6
+
7
+ List extends [Subscriber](#docs/_core-concept/subscriber.md/subscriber) and **Fetcher**:
8
+ * **Subscriber** `props` + `dataProvider`
9
+ * **Fetcher** — optional `fetch` + `serviceURL` / `key` (see [Fetch](#core/components/functional/fetch/fetch.md/fetch))
10
+
11
+ ## Row renderer (`items`) — recommended
12
+
13
+ From a **Lit** parent, pass a function on the **`items`** property (`ListItems`). Each row is wrapped in a `sonic-subscriber` with `dataProvider="…/list-item/n"` (hover rows with `debug`).
14
+
15
+ ```typescript
16
+ private items = ({ first_name, last_name, email, avatar }) => html`
17
+ <sonic-image src=${avatar} …></sonic-image>
18
+ <div>${first_name} <b>${last_name}</b></div>
19
+ <div>${email}</div>
20
+ `;
21
+
22
+ html`<sonic-list
23
+ fetch
24
+ dataProvider="api/users"
25
+ key="data"
26
+ serviceURL="/docs-mock-api"
27
+ .items=${this.items}
28
+ ></sonic-list>`;
29
+ ```
30
+
31
+ Use **`.items=${fn}`** (property binding): Lit passes functions only as properties, not as HTML attributes — `@property({ type: Function })` does not change that. Same for **`.noItems`**, **`.separator`**, **`.skeleton`**. The callback receives each row object (replacing `data-bind` / `<sonic-value>` in a `<template>`).
32
+
33
+ Live demo + TypeScript source (one file, no Markdown copy):
34
+
35
+ <docs-lit-demo for="docs-users-list"></docs-lit-demo>
36
+
37
+ Implementation: `src/docs/example/docs-users-list.ts` — row markup in the **`items`** callback (`item.first_name`, …), same idea as replacing `data-bind` / `<sonic-value>` in a `<template>`.
38
+
39
+ ## Alternating row layouts
40
+
41
+ Use **`metadata`** (`even`, `odd`, `firstChild`, …) or fields on each item (e.g. `tpl` with `templateKey`):
42
+
43
+ <docs-lit-demo for="docs-list-alternate-demo"></docs-lit-demo>
44
+
45
+ <docs-lit-demo for="docs-list-template-key-demo"></docs-lit-demo>
46
+
47
+ ## Separator and empty list
48
+
49
+ <docs-lit-demo for="docs-list-separator-demo"></docs-lit-demo>
50
+
51
+ ## Fetch + `extractValues`
52
+
53
+ <docs-lit-demo for="docs-list-extract-values-demo"></docs-lit-demo>
54
+
55
+ ## HTML `<template>` children (integration without Lit)
56
+
57
+ For **plain HTML** hosts, you can still declare **`<template>`** children (and `data-value` for `templateKey`, `separator`, `no-item`). That path is for [HTML integration](#docs/_misc/html-integration.md/html-integration) — not used in Concorde doc live demos.
58
+
59
+ Each list row still gets `dataProvider="[list]/list-item/[index]"`.
167
60
 
168
61
  ## Additionnal tips
169
62
 
170
- * If the result of the request is an object, it is nested within an array to ensure functionality.
171
- * The invalidate() method can be called on its publisher to trigger data reloading.
172
- * Each list item publisher has a "_parent_" property pointing to the list publisher
173
-
174
-
175
-
176
- <!--
177
- ## FormDataProvider
178
-
179
- <sonic-alert status="error" background>À déplacer dans la bonne doc</sonic-alert>
180
-
181
- <sonic-code>
182
- <template>
183
- <div formDataProvider="profileInfos">
184
- <form>
185
- <sonic-fieldset label="Edit profile">
186
- <sonic-form-layout>
187
- <sonic-input label="First name" type="text" name="first_name" value="Sponge" size="sm"></sonic-input>
188
- <sonic-input label="Last name" type="text" name="last_name" value="Bob" size="sm"></sonic-input>
189
- <sonic-input label="email" type="email" name="email" value="bob@krustykrab.com" size="sm"></sonic-input>
190
- <sonic-input label="Image url" type="text" name="avatar" value="http://www.bobleponge.fr/goodies/avatars/avatar-bob-eponge_Bob-Eponge-coiffure.jpg" size="sm"></sonic-input>
191
- </sonic-form-layout>
192
- </sonic-fieldset>
193
- </form>
194
- </div>
195
- <sonic-card dataProvider="profileInfos">
196
- <docs-user ></docs-user>
197
- </sonic-card>
198
- </template>
199
- </sonic-code> -->
63
+ * If the request returns an object, it is wrapped in an array (unless `extractValues` is set).
64
+ * Call **`invalidate()`** on the list publisher to reload fetch data.
65
+ * Each row publisher exposes **`_parent_`** pointing at the list publisher.