@rytass/bpm-core-react 0.3.1 → 0.3.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/CHANGELOG.md +166 -0
  2. package/dist/chunks/app-navigation-BSkMsEhy.js +268 -0
  3. package/dist/chunks/app-navigation-BSkMsEhy.js.map +1 -0
  4. package/dist/chunks/app-navigation-KnlJCUp1.cjs +2 -0
  5. package/dist/chunks/app-navigation-KnlJCUp1.cjs.map +1 -0
  6. package/dist/chunks/approval-instance-list-page-CVXgE2K3.cjs +2 -0
  7. package/dist/chunks/approval-instance-list-page-CVXgE2K3.cjs.map +1 -0
  8. package/dist/chunks/{approval-instance-list-page-nmzMrj0b.js → approval-instance-list-page-CqNdoZqx.js} +73 -72
  9. package/dist/chunks/approval-instance-list-page-CqNdoZqx.js.map +1 -0
  10. package/dist/chunks/builder-CMlJfQHE.cjs +3 -0
  11. package/dist/chunks/builder-CMlJfQHE.cjs.map +1 -0
  12. package/dist/chunks/{builder-DqZskyXC.js → builder-D950gct_.js} +492 -491
  13. package/dist/chunks/builder-D950gct_.js.map +1 -0
  14. package/dist/chunks/categories-5yEM3p3N.cjs +2 -0
  15. package/dist/chunks/categories-5yEM3p3N.cjs.map +1 -0
  16. package/dist/chunks/{categories-DTEl182t.js → categories-BIpOG451.js} +115 -114
  17. package/dist/chunks/categories-BIpOG451.js.map +1 -0
  18. package/dist/chunks/dashboard-page-1K_jQXQk.cjs +2 -0
  19. package/dist/chunks/dashboard-page-1K_jQXQk.cjs.map +1 -0
  20. package/dist/chunks/dashboard-page-R_T2OEiE.js +122 -0
  21. package/dist/chunks/dashboard-page-R_T2OEiE.js.map +1 -0
  22. package/dist/chunks/{delegations-C5PzZ5Kn.js → delegations-B2j-wNEO.js} +193 -192
  23. package/dist/chunks/delegations-B2j-wNEO.js.map +1 -0
  24. package/dist/chunks/delegations-CsB9ozLu.cjs +2 -0
  25. package/dist/chunks/delegations-CsB9ozLu.cjs.map +1 -0
  26. package/dist/chunks/delegations-CvtwTXNP.cjs +2 -0
  27. package/dist/chunks/delegations-CvtwTXNP.cjs.map +1 -0
  28. package/dist/chunks/{delegations-C-ZrwzvU.js → delegations-dKodb0WW.js} +175 -174
  29. package/dist/chunks/delegations-dKodb0WW.js.map +1 -0
  30. package/dist/chunks/{detail-CfFyU5zC.js → detail-BcGAqJ_R.js} +465 -464
  31. package/dist/chunks/detail-BcGAqJ_R.js.map +1 -0
  32. package/dist/chunks/detail-CqjqLd65.cjs +2 -0
  33. package/dist/chunks/detail-CqjqLd65.cjs.map +1 -0
  34. package/dist/chunks/{format-date-time-isOa3e9q.cjs → format-date-time-26_pFvv4.cjs} +2 -2
  35. package/dist/chunks/{format-date-time-isOa3e9q.cjs.map → format-date-time-26_pFvv4.cjs.map} +1 -1
  36. package/dist/chunks/notifications-2swRqDPF.js +198 -0
  37. package/dist/chunks/notifications-2swRqDPF.js.map +1 -0
  38. package/dist/chunks/notifications-BaYDebFt.cjs +2 -0
  39. package/dist/chunks/notifications-BaYDebFt.cjs.map +1 -0
  40. package/dist/chunks/{orgs-xrdhb3hS.js → orgs-CuHxxd_n.js} +608 -607
  41. package/dist/chunks/orgs-CuHxxd_n.js.map +1 -0
  42. package/dist/chunks/orgs-YMiVLNvL.cjs +2 -0
  43. package/dist/chunks/orgs-YMiVLNvL.cjs.map +1 -0
  44. package/dist/chunks/routes-config-2aKbWq2H.cjs +2 -0
  45. package/dist/chunks/routes-config-2aKbWq2H.cjs.map +1 -0
  46. package/dist/chunks/routes-config-dxahImVe.js +43 -0
  47. package/dist/chunks/routes-config-dxahImVe.js.map +1 -0
  48. package/dist/chunks/templates-DTkbSgFY.cjs +2 -0
  49. package/dist/chunks/templates-DTkbSgFY.cjs.map +1 -0
  50. package/dist/chunks/templates-DoDWM68t.js +384 -0
  51. package/dist/chunks/templates-DoDWM68t.js.map +1 -0
  52. package/dist/chunks/users-3ySyUW4u.cjs +2 -0
  53. package/dist/chunks/users-3ySyUW4u.cjs.map +1 -0
  54. package/dist/chunks/{users-CY4-NK3j.js → users-sMfrSjRQ.js} +75 -74
  55. package/dist/chunks/users-sMfrSjRQ.js.map +1 -0
  56. package/dist/components/admin-pickers.d.ts +1 -1
  57. package/dist/components/approval-instance-list-page.d.ts +1 -1
  58. package/dist/components/org-unit-tree-draft-editor.d.ts +1 -1
  59. package/dist/index.cjs +1 -1
  60. package/dist/index.cjs.map +1 -1
  61. package/dist/index.d.ts +1 -0
  62. package/dist/index.js +101 -99
  63. package/dist/index.js.map +1 -1
  64. package/dist/lib/auth-provider.d.ts +1 -1
  65. package/dist/lib/org-tree-draft.d.ts +1 -1
  66. package/dist/lib/routes-config.d.ts +105 -0
  67. package/dist/next/BPMNextProviders.d.ts +48 -3
  68. package/dist/next/index.cjs +1 -1
  69. package/dist/next/index.cjs.map +1 -1
  70. package/dist/next/index.d.ts +1 -0
  71. package/dist/next/index.js +27 -21
  72. package/dist/next/index.js.map +1 -1
  73. package/dist/pages/admin/delegations/index.cjs +1 -1
  74. package/dist/pages/admin/delegations/index.js +1 -1
  75. package/dist/pages/admin/orgs/index.cjs +1 -1
  76. package/dist/pages/admin/orgs/index.js +1 -1
  77. package/dist/pages/admin/users/index.cjs +1 -1
  78. package/dist/pages/admin/users/index.js +1 -1
  79. package/dist/pages/delegations/index.cjs +1 -1
  80. package/dist/pages/delegations/index.js +1 -1
  81. package/dist/pages/forms/builder/index.cjs +1 -1
  82. package/dist/pages/forms/builder/index.js +1 -1
  83. package/dist/pages/instances/detail/index.cjs +1 -1
  84. package/dist/pages/instances/detail/index.js +1 -1
  85. package/dist/pages/settings/notifications/index.cjs +1 -1
  86. package/dist/pages/settings/notifications/index.js +1 -1
  87. package/dist/pages/templates/categories/index.cjs +1 -1
  88. package/dist/pages/templates/categories/index.js +1 -1
  89. package/dist/pages/templates/index.cjs +1 -1
  90. package/dist/pages/templates/index.js +1 -1
  91. package/dist/views/admin/delegations/index.cjs +1 -1
  92. package/dist/views/admin/delegations/index.js +1 -1
  93. package/dist/views/admin/index.cjs +1 -1
  94. package/dist/views/admin/index.js +3 -3
  95. package/dist/views/admin/orgs/index.cjs +1 -1
  96. package/dist/views/admin/orgs/index.js +1 -1
  97. package/dist/views/admin/users/index.cjs +1 -1
  98. package/dist/views/admin/users/index.js +1 -1
  99. package/dist/views/cc/index.cjs +1 -1
  100. package/dist/views/cc/index.js +1 -1
  101. package/dist/views/dashboard/index.cjs +1 -1
  102. package/dist/views/dashboard/index.js +1 -1
  103. package/dist/views/delegations/index.cjs +1 -1
  104. package/dist/views/delegations/index.js +1 -1
  105. package/dist/views/forms/builder/index.cjs +1 -1
  106. package/dist/views/forms/builder/index.js +1 -1
  107. package/dist/views/forms/index.cjs +1 -1
  108. package/dist/views/forms/index.cjs.map +1 -1
  109. package/dist/views/forms/index.js +78 -77
  110. package/dist/views/forms/index.js.map +1 -1
  111. package/dist/views/forms/renderer/FormRendererView.d.ts +2 -2
  112. package/dist/views/inbox/index.cjs +1 -1
  113. package/dist/views/inbox/index.cjs.map +1 -1
  114. package/dist/views/inbox/index.js +106 -105
  115. package/dist/views/inbox/index.js.map +1 -1
  116. package/dist/views/instances/detail/index.cjs +1 -1
  117. package/dist/views/instances/detail/index.js +1 -1
  118. package/dist/views/instances/new/index.cjs +1 -1
  119. package/dist/views/instances/new/index.cjs.map +1 -1
  120. package/dist/views/instances/new/index.js +90 -89
  121. package/dist/views/instances/new/index.js.map +1 -1
  122. package/dist/views/search/index.cjs +1 -1
  123. package/dist/views/search/index.js +1 -1
  124. package/dist/views/sent/index.cjs +1 -1
  125. package/dist/views/sent/index.js +1 -1
  126. package/dist/views/settings/index.cjs +1 -1
  127. package/dist/views/settings/index.js +1 -1
  128. package/dist/views/settings/notifications/index.cjs +1 -1
  129. package/dist/views/settings/notifications/index.js +1 -1
  130. package/dist/views/templates/categories/index.cjs +1 -1
  131. package/dist/views/templates/categories/index.js +1 -1
  132. package/dist/views/templates/designer/index.cjs +6 -6
  133. package/dist/views/templates/designer/index.cjs.map +1 -1
  134. package/dist/views/templates/designer/index.js +589 -588
  135. package/dist/views/templates/designer/index.js.map +1 -1
  136. package/dist/views/templates/index.cjs +1 -1
  137. package/dist/views/templates/index.js +2 -2
  138. package/dist/views/templates/versions/index.cjs +1 -1
  139. package/dist/views/templates/versions/index.cjs.map +1 -1
  140. package/dist/views/templates/versions/index.js +44 -43
  141. package/dist/views/templates/versions/index.js.map +1 -1
  142. package/package.json +3 -3
  143. package/dist/chunks/app-navigation-C_mbz7jx.cjs +0 -2
  144. package/dist/chunks/app-navigation-C_mbz7jx.cjs.map +0 -1
  145. package/dist/chunks/app-navigation-uwbNEw9P.js +0 -262
  146. package/dist/chunks/app-navigation-uwbNEw9P.js.map +0 -1
  147. package/dist/chunks/approval-instance-list-page-Mo6wpDPb.cjs +0 -2
  148. package/dist/chunks/approval-instance-list-page-Mo6wpDPb.cjs.map +0 -1
  149. package/dist/chunks/approval-instance-list-page-nmzMrj0b.js.map +0 -1
  150. package/dist/chunks/builder-DPhAH381.cjs +0 -3
  151. package/dist/chunks/builder-DPhAH381.cjs.map +0 -1
  152. package/dist/chunks/builder-DqZskyXC.js.map +0 -1
  153. package/dist/chunks/categories-DEijUOnw.cjs +0 -2
  154. package/dist/chunks/categories-DEijUOnw.cjs.map +0 -1
  155. package/dist/chunks/categories-DTEl182t.js.map +0 -1
  156. package/dist/chunks/dashboard-page-DCmuB0Rw.cjs +0 -2
  157. package/dist/chunks/dashboard-page-DCmuB0Rw.cjs.map +0 -1
  158. package/dist/chunks/dashboard-page-Dx5PeEeN.js +0 -117
  159. package/dist/chunks/dashboard-page-Dx5PeEeN.js.map +0 -1
  160. package/dist/chunks/delegations-C-ZrwzvU.js.map +0 -1
  161. package/dist/chunks/delegations-C5PzZ5Kn.js.map +0 -1
  162. package/dist/chunks/delegations-DOGDvybX.cjs +0 -2
  163. package/dist/chunks/delegations-DOGDvybX.cjs.map +0 -1
  164. package/dist/chunks/delegations-DkDBWOQ7.cjs +0 -2
  165. package/dist/chunks/delegations-DkDBWOQ7.cjs.map +0 -1
  166. package/dist/chunks/detail-B2gcOPkd.cjs +0 -2
  167. package/dist/chunks/detail-B2gcOPkd.cjs.map +0 -1
  168. package/dist/chunks/detail-CfFyU5zC.js.map +0 -1
  169. package/dist/chunks/notifications-CPQ-nVar.cjs +0 -2
  170. package/dist/chunks/notifications-CPQ-nVar.cjs.map +0 -1
  171. package/dist/chunks/notifications-DweexUVy.js +0 -197
  172. package/dist/chunks/notifications-DweexUVy.js.map +0 -1
  173. package/dist/chunks/orgs-DgZ0DQ3-.cjs +0 -2
  174. package/dist/chunks/orgs-DgZ0DQ3-.cjs.map +0 -1
  175. package/dist/chunks/orgs-xrdhb3hS.js.map +0 -1
  176. package/dist/chunks/templates-PK_VYvcy.js +0 -383
  177. package/dist/chunks/templates-PK_VYvcy.js.map +0 -1
  178. package/dist/chunks/templates-x1OJZmsG.cjs +0 -2
  179. package/dist/chunks/templates-x1OJZmsG.cjs.map +0 -1
  180. package/dist/chunks/users-CY4-NK3j.js.map +0 -1
  181. package/dist/chunks/users-DHnu_056.cjs +0 -2
  182. package/dist/chunks/users-DHnu_056.cjs.map +0 -1
package/CHANGELOG.md ADDED
@@ -0,0 +1,166 @@
1
+ # Changelog
2
+
3
+ All notable changes to `@rytass/bpm-core-react` are documented in this file.
4
+
5
+ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/)
6
+ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
7
+
8
+ Releases are managed by [`nx release`](https://nx.dev/recipes/nx-release) with
9
+ Conventional Commits — see `nx.json` for the release config.
10
+
11
+ ## 0.3.3 — 2026-05-28
12
+
13
+ ### Fixed
14
+
15
+ - **Dist `.d.ts` files leaked workspace-source paths.** The 0.3.0–0.3.2
16
+ release shipped declaration files that contained relative imports back
17
+ into the vendor's monorepo (e.g. `import { ApiMember } from
18
+ '../../libs/bpm-core-client/src/index.ts'`), which broke `tsc` for any
19
+ consumer that didn't have BPMCore's source on disk at that exact
20
+ relative path. The build now passes `compilerOptions.paths: {}` to
21
+ `vite-plugin-dts` so the published types reference packages by their
22
+ npm names (`@rytass/bpm-core-client`) rather than by resolved aliases.
23
+
24
+ ### Added
25
+
26
+ - **`BPMNextProviders` props are now typed and forwarded.** The
27
+ `loginPath`, `publicPaths`, and `locale` props were always accepted by
28
+ the inner `<Providers>` but the `next` subpath wrapper hid them. They
29
+ are now part of the exported `BPMNextProvidersProps` interface and
30
+ forwarded through. Default `loginPath` remains `'/login'`, so hosts
31
+ with a different auth route must pass their override explicitly.
32
+
33
+ ### Documentation
34
+
35
+ - `BPMRoutes.caseDetail` and `BPMRoutes.caseNew(templateId?)` JSDoc
36
+ carries `@example` annotations showing the default factory's literal
37
+ paths so consumers can see what they're overriding without reading
38
+ the source.
39
+
40
+ ### Why a patch
41
+
42
+ The d.ts fix only affected consumers' type-check experience — runtime
43
+ behavior didn't change. The `BPMNextProviders` props forwarding is
44
+ additive (props were silently dropped before, now they work).
45
+
46
+ ## 0.3.2 — 2026-05-28
47
+
48
+ ### Added
49
+
50
+ - **`BPMRoutesContext` + `BPMRoutesProvider` + `useBPMRoutes()`** — opt-in
51
+ path-mapping for hosts that embed BPM views under their own URL prefix.
52
+ All BPM internal cross-navigation (instance detail, instance new,
53
+ template designer, template versions, template categories, form
54
+ builder, etc.) now resolves through `routes.caseDetail(id)` instead of
55
+ inline `\`/instances/${id}\``. When no provider is mounted,
56
+ `createDefaultBPMRoutes()` reproduces the previous string literals so
57
+ upgrades from 0.3.1 require zero changes.
58
+
59
+ Re-exported from the root barrel and from `@rytass/bpm-core-react/next`.
60
+
61
+ Example (Next.js host mounting BPM under `/workspace`):
62
+
63
+ ```tsx
64
+ import { BPMRoutesProvider } from '@rytass/bpm-core-react/next';
65
+
66
+ <BPMRoutesProvider value={{
67
+ caseDetail: (id) => `/workspace/cases/${id}`,
68
+ caseNew: (templateId) =>
69
+ `/workspace/cases/new${templateId ? `?t=${templateId}` : ''}`,
70
+ templates: () => '/workspace/templates',
71
+ templateDesigner: (id) => `/workspace/templates/${id}/designer`,
72
+ // ...rest follows the BPMRoutes interface
73
+ }}>
74
+ {children}
75
+ </BPMRoutesProvider>
76
+ ```
77
+
78
+ ### Why a patch (additive only)
79
+
80
+ The provider is optional and the default factory preserves the exact
81
+ 0.3.1 routes. No public symbol removed, no existing type signature
82
+ changed.
83
+
84
+ ## 0.3.1 — 2026-05-27
85
+
86
+ ### Fixed
87
+
88
+ - **`AppLayout` slot detection — the "left sidebar disappeared" bug.**
89
+ Mezzanine `<Layout>` discovers its sidebar slot by component-identity
90
+ match (`child.type === Navigation`). The previous `<AppNavigation>`
91
+ wrapper rendered `<Navigation>` internally, so `<Layout><AppNavigation
92
+ />...</Layout>` would silently drop the sidebar because `AppNavigation`
93
+ ≠ `Navigation`. This shipped in 0.3.0 and broke every embedded view.
94
+
95
+ ### ⚠️ Breaking
96
+
97
+ - **`AppNavigation` → `AppLayout`.** The standalone navigation component
98
+ is replaced by `AppLayout`, which composes Mezzanine's
99
+ `<Layout> + <Navigation> + <Layout.Main>{children}</Layout.Main>`
100
+ internally. All 12 internal views were migrated.
101
+
102
+ Migration for consumers who imported `AppNavigation` directly (very
103
+ rare — most consume the `pages/*` shims which already use `AppLayout`):
104
+
105
+ ```diff
106
+ - import { AppNavigation } from '@rytass/bpm-core-react';
107
+ + import { AppLayout } from '@rytass/bpm-core-react';
108
+
109
+ - <Layout>
110
+ - <AppNavigation activeHref="/inbox" />
111
+ - <Layout.Main>
112
+ - <PageHeader title="待簽" />
113
+ - <SectionGroup>...</SectionGroup>
114
+ - </Layout.Main>
115
+ - </Layout>
116
+ + <AppLayout activeHref="/inbox">
117
+ + <PageHeader title="待簽" />
118
+ + <SectionGroup>...</SectionGroup>
119
+ + </AppLayout>
120
+ ```
121
+
122
+ The exported types are now `AppLayoutProps` and `AppNavigationGroup`
123
+ (the latter is the typed shape for overriding the 4-group nav tree via
124
+ the `groups` prop).
125
+
126
+ ## 0.3.0 — 2026-05-27
127
+
128
+ ### Added
129
+
130
+ - **Full admin surface.** Up from the 0.2.0 login-only POC to 12 view
131
+ subpaths, 19 Next.js page shims (`pages/<feature>`), and 4 view group
132
+ barrels (`views/workflow`, `views/instances`, `views/settings`,
133
+ `views/admin`).
134
+ - **`@rytass/bpm-core-react/next` subpath** — `<BPMNextProviders>`
135
+ composes RouterAdapter + AuthProvider + NotificationDrawerProvider +
136
+ NotificationUnreadProvider + CalendarConfigProviderMoment in one wrap.
137
+ - **Root barrel foundation** — Providers, hooks (`useAuth`,
138
+ `useRouterAdapter`, `useNotificationUnread`,
139
+ `useNotificationDrawer`), `AppNavigation`, `NotificationDrawer`,
140
+ `ApprovalInstanceListPage`, `DashboardPage`, `BPMFormField`,
141
+ `MemberPicker`, `OrgUnitPicker`, `PositionPicker`.
142
+
143
+ ### Changed
144
+
145
+ - **Server Components everywhere by default.** `pages/<feature>` modules
146
+ are now Server Components that export `default` + `metadata`. The
147
+ `'use client'` boundary lives inside the underlying `<F>View`. The
148
+ prior intermediate `<F>ClientView` shim layer (19 files) is removed.
149
+ - **`InstanceNewView` lifts `searchParams` to a prop.** Pages read
150
+ `searchParams.templateId` server-side and pass it down instead of the
151
+ view reading the browser URL.
152
+ - **Root page `/` performs a server-side redirect to `/dashboard`** for
153
+ authenticated users (previously a client-side bounce).
154
+
155
+ ## 0.2.0 — 2026-05-26
156
+
157
+ ### Added
158
+
159
+ - Initial POC release. Login chain only (`/views/login`, `/pages/login`,
160
+ providers, RouterAdapter). The remaining views planned for 0.3.0.
161
+ - Vite library-mode build pipeline emitting per-format chunks
162
+ (`.js` for ESM, `.cjs` for CJS), SCSS CSS Modules with `bpm_` scope
163
+ prefix, `'use client'` directive preserved via
164
+ `rollup-preserve-directives`, type definitions via `vite-plugin-dts`.
165
+ - Decoupled from `next/navigation` via the `RouterAdapter` context —
166
+ framework-agnostic consumers can plug in their own router.
@@ -0,0 +1,268 @@
1
+ "use client";
2
+ import { a as e, n as t } from "./auth-provider-Bnox5gsx.js";
3
+ import { r as n } from "./routes-config-dxahImVe.js";
4
+ import { createContext as r, useCallback as i, useContext as a, useEffect as o, useMemo as s, useState as c } from "react";
5
+ import { Layout as l, Navigation as u, NavigationFooter as d, NavigationHeader as f, NavigationIconButton as p, NavigationOption as m, NavigationOptionCategory as h, NavigationUserMenu as g } from "@mezzanine-ui/react";
6
+ import { logoutApi as _ } from "@rytass/bpm-core-client";
7
+ import { Fragment as v, jsx as y, jsxs as b } from "react/jsx-runtime";
8
+ import { readUnreadNotificationCount as x } from "@rytass/bpm-core-client/workflow";
9
+ import { FileIcon as S, FolderIcon as C, HomeIcon as w, ListIcon as T, LogoutIcon as E, MailIcon as D, MailUnreadIcon as O, NotificationUnreadIcon as k, SearchIcon as A, ShareIcon as j, SwitchHorizontalIcon as M, SystemIcon as N, UserIcon as P } from "@mezzanine-ui/icons";
10
+ import '../app-navigation.css';//#region src/lib/notification-drawer-provider.tsx
11
+ var F = r(null);
12
+ function I({ children: e }) {
13
+ let [t, n] = c(!1), r = i(() => {
14
+ n(!0);
15
+ }, []), a = i(() => {
16
+ n(!1);
17
+ }, []), o = i(() => {
18
+ n((e) => !e);
19
+ }, []), l = s(() => ({
20
+ close: a,
21
+ isOpen: t,
22
+ open: r,
23
+ toggle: o
24
+ }), [
25
+ a,
26
+ t,
27
+ r,
28
+ o
29
+ ]);
30
+ return /* @__PURE__ */ y(F.Provider, {
31
+ value: l,
32
+ children: e
33
+ });
34
+ }
35
+ function L() {
36
+ return a(F) || {
37
+ close: () => void 0,
38
+ isOpen: !1,
39
+ open: () => void 0,
40
+ toggle: () => void 0
41
+ };
42
+ }
43
+ //#endregion
44
+ //#region src/lib/notification-unread-provider.tsx
45
+ var R = r(null);
46
+ function z({ children: e }) {
47
+ let { member: n } = t(), r = n?.memberId ?? null, [a, l] = c(0), u = i(async () => {
48
+ if (!r) return l(0), 0;
49
+ let e = await x(r);
50
+ return l(e), e;
51
+ }, [r]);
52
+ o(() => {
53
+ let e = !0;
54
+ return (async () => {
55
+ try {
56
+ let t = await u();
57
+ e && l(t);
58
+ } catch {
59
+ e && l(0);
60
+ }
61
+ })(), () => {
62
+ e = !1;
63
+ };
64
+ }, [u]);
65
+ let d = s(() => ({
66
+ refreshUnreadCount: u,
67
+ unreadCount: a
68
+ }), [u, a]);
69
+ return /* @__PURE__ */ y(R.Provider, {
70
+ value: d,
71
+ children: e
72
+ });
73
+ }
74
+ function B() {
75
+ return a(R) || {
76
+ refreshUnreadCount: async () => 0,
77
+ unreadCount: 0
78
+ };
79
+ }
80
+ var V = {
81
+ logo: "bpm_logo_QvBLU",
82
+ notificationBell: "bpm_notificationBell_W-wl7",
83
+ notificationBadge: "bpm_notificationBadge_Gy3Eq"
84
+ };
85
+ //#endregion
86
+ //#region src/components/app-navigation.tsx
87
+ function H(e) {
88
+ return [
89
+ {
90
+ title: "我的工作",
91
+ items: [
92
+ {
93
+ href: e.dashboard(),
94
+ icon: w,
95
+ label: "工作台"
96
+ },
97
+ {
98
+ href: e.inbox(),
99
+ icon: O,
100
+ label: "我的待簽"
101
+ },
102
+ {
103
+ href: e.sent(),
104
+ icon: D,
105
+ label: "我發起的"
106
+ },
107
+ {
108
+ href: e.cc(),
109
+ icon: j,
110
+ label: "抄送給我"
111
+ }
112
+ ]
113
+ },
114
+ {
115
+ title: "查詢與代理",
116
+ items: [{
117
+ href: e.search(),
118
+ icon: A,
119
+ label: "搜尋"
120
+ }, {
121
+ href: e.delegations(),
122
+ icon: M,
123
+ label: "個人代理"
124
+ }]
125
+ },
126
+ {
127
+ title: "簽核設計",
128
+ items: [
129
+ {
130
+ href: e.templates(),
131
+ icon: C,
132
+ label: "簽核模板",
133
+ requiresAdmin: !0
134
+ },
135
+ {
136
+ href: e.templateCategories(),
137
+ icon: T,
138
+ label: "模板分類",
139
+ requiresAdmin: !0
140
+ },
141
+ {
142
+ href: e.forms(),
143
+ icon: S,
144
+ label: "表單設計",
145
+ requiresAdmin: !0
146
+ }
147
+ ]
148
+ },
149
+ {
150
+ title: "系統管理",
151
+ items: [
152
+ {
153
+ href: e.adminOrgs(),
154
+ icon: N,
155
+ label: "組織管理",
156
+ requiresAdmin: !0
157
+ },
158
+ {
159
+ href: e.adminUsers(),
160
+ icon: P,
161
+ label: "會員對照",
162
+ requiresAdmin: !0
163
+ },
164
+ {
165
+ href: e.adminDelegations(),
166
+ icon: j,
167
+ label: "代理設定",
168
+ requiresAdmin: !0
169
+ }
170
+ ]
171
+ }
172
+ ];
173
+ }
174
+ function U({ activeHref: r, logoSrc: i = "/rytass-logo.png", title: a = "BPM Admin", groups: o, children: s }) {
175
+ let c = e(), v = n(), { member: x } = t(), { unreadCount: S } = B(), C = r ?? c.pathname ?? "", w = W(x), T = (o ?? H(v)).map((e) => ({
176
+ title: e.title,
177
+ items: e.items.filter((e) => !e.requiresAdmin || w)
178
+ })).filter((e) => e.items.length > 0), D = async () => {
179
+ await _(), c.replace("/login");
180
+ };
181
+ return /* @__PURE__ */ b(l, { children: [/* @__PURE__ */ y(u, {
182
+ exactActivatedMatch: !0,
183
+ children: [
184
+ /* @__PURE__ */ y(f, {
185
+ title: a,
186
+ children: /* @__PURE__ */ y("img", {
187
+ alt: "",
188
+ className: V.logo,
189
+ height: 24,
190
+ src: i,
191
+ width: 24
192
+ })
193
+ }, "header"),
194
+ ...T.map((e) => /* @__PURE__ */ y(h, {
195
+ title: e.title,
196
+ children: e.items.map((e) => /* @__PURE__ */ y(m, {
197
+ active: e.href === C,
198
+ href: e.href,
199
+ icon: e.icon,
200
+ title: e.label
201
+ }, e.href))
202
+ }, e.title)),
203
+ /* @__PURE__ */ b(d, { children: [
204
+ /* @__PURE__ */ y(g, {
205
+ options: [{
206
+ id: "notification-settings",
207
+ name: "通知設定"
208
+ }, {
209
+ id: "logout",
210
+ name: "登出"
211
+ }],
212
+ onSelect: (e) => {
213
+ if (e.id === "notification-settings") {
214
+ c.push(v.notificationSettings());
215
+ return;
216
+ }
217
+ e.id === "logout" && D();
218
+ },
219
+ children: /* @__PURE__ */ y(K, {})
220
+ }),
221
+ /* @__PURE__ */ y(G, { unreadCount: S }),
222
+ /* @__PURE__ */ y(p, {
223
+ "aria-label": "登出",
224
+ icon: E,
225
+ onClick: () => {
226
+ D();
227
+ },
228
+ title: "登出",
229
+ type: "button"
230
+ })
231
+ ] }, "footer")
232
+ ]
233
+ }), /* @__PURE__ */ y(l.Main, { children: s })] });
234
+ }
235
+ function W(e) {
236
+ return e ? (e.roles ?? []).includes("BPM_ADMIN") || (e.permissions ?? []).some((e) => [
237
+ "bpm:*",
238
+ "bpm:admin",
239
+ "bpm.admin",
240
+ "bpm:admin:*"
241
+ ].includes(e)) : !1;
242
+ }
243
+ function G({ unreadCount: e }) {
244
+ let { open: t } = L();
245
+ return /* @__PURE__ */ b("span", {
246
+ className: V.notificationBell,
247
+ children: [/* @__PURE__ */ y(p, {
248
+ "aria-label": e > 0 ? `通知中心,${e} 則未讀` : "通知中心",
249
+ icon: k,
250
+ onClick: () => {
251
+ t();
252
+ },
253
+ title: "通知中心",
254
+ type: "button"
255
+ }), e > 0 ? /* @__PURE__ */ y("span", {
256
+ className: V.notificationBadge,
257
+ children: e > 99 ? "99+" : e
258
+ }) : null]
259
+ });
260
+ }
261
+ function K() {
262
+ let { member: e } = t();
263
+ return e ? /* @__PURE__ */ y(v, { children: e.name }) : null;
264
+ }
265
+ //#endregion
266
+ export { L as a, I as i, z as n, B as r, U as t };
267
+
268
+ //# sourceMappingURL=app-navigation-BSkMsEhy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"app-navigation-BSkMsEhy.js","names":[],"sources":["../../src/lib/notification-drawer-provider.tsx","../../src/lib/notification-unread-provider.tsx","../../src/components/app-navigation.module.scss","../../src/components/app-navigation.tsx"],"sourcesContent":["'use client';\n\nimport {\n createContext,\n useCallback,\n useContext,\n useMemo,\n useState,\n type ReactElement,\n type ReactNode,\n} from 'react';\n\ninterface NotificationDrawerContextValue {\n readonly close: () => void;\n readonly isOpen: boolean;\n readonly open: () => void;\n readonly toggle: () => void;\n}\n\nconst NotificationDrawerContext =\n createContext<NotificationDrawerContextValue | null>(null);\n\ninterface NotificationDrawerProviderProps {\n readonly children: ReactNode;\n}\n\n/**\n * Controls the open/closed state of the BPM notification drawer. Wraps\n * children with a context that `<NotificationDrawer />` reads to mount /\n * hide itself, and that `<AppLayout />` reads to open the drawer when\n * the bell icon is clicked.\n *\n * When used outside this provider, the returned hook is a safe no-op so\n * components don't crash in test or storybook environments.\n */\nexport function NotificationDrawerProvider({\n children,\n}: NotificationDrawerProviderProps): ReactElement {\n const [isOpen, setIsOpen] = useState(false);\n\n const open = useCallback((): void => {\n setIsOpen(true);\n }, []);\n const close = useCallback((): void => {\n setIsOpen(false);\n }, []);\n const toggle = useCallback((): void => {\n setIsOpen((current) => !current);\n }, []);\n\n const value = useMemo<NotificationDrawerContextValue>(\n () => ({ close, isOpen, open, toggle }),\n [close, isOpen, open, toggle],\n );\n\n return (\n <NotificationDrawerContext.Provider value={value}>\n {children}\n </NotificationDrawerContext.Provider>\n );\n}\n\n/**\n * Read the BPM notification drawer's open state and control helpers.\n * Returns a no-op stub when used outside `<NotificationDrawerProvider>`.\n */\nexport function useNotificationDrawer(): NotificationDrawerContextValue {\n const context = useContext(NotificationDrawerContext);\n if (!context) {\n return {\n close: (): void => undefined,\n isOpen: false,\n open: (): void => undefined,\n toggle: (): void => undefined,\n };\n }\n return context;\n}\n","'use client';\n\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useState,\n type ReactElement,\n type ReactNode,\n} from 'react';\nimport { readUnreadNotificationCount } from '@rytass/bpm-core-client/workflow';\nimport { useAuth } from './auth-provider';\n\ninterface NotificationUnreadContextValue {\n readonly refreshUnreadCount: () => Promise<number>;\n readonly unreadCount: number;\n}\n\nconst NotificationUnreadContext =\n createContext<NotificationUnreadContextValue | null>(null);\n\ninterface NotificationUnreadProviderProps {\n readonly children: ReactNode;\n}\n\n/**\n * Polls BPM for the current member's unread notification count via\n * `readUnreadNotificationCount` and exposes it through context for\n * `<AppLayout />` (bell badge) and `<NotificationDrawer />` (header\n * count). Refresh is triggered on mount and whenever the auth member id\n * changes; consumers can call `refreshUnreadCount()` after acknowledging\n * a notification.\n */\nexport function NotificationUnreadProvider({\n children,\n}: NotificationUnreadProviderProps): ReactElement {\n const { member } = useAuth();\n const currentMemberId = member?.memberId ?? null;\n const [unreadCount, setUnreadCount] = useState(0);\n\n const refreshUnreadCount = useCallback(async (): Promise<number> => {\n if (!currentMemberId) {\n setUnreadCount(0);\n return 0;\n }\n const next = await readUnreadNotificationCount(currentMemberId);\n setUnreadCount(next);\n return next;\n }, [currentMemberId]);\n\n useEffect((): (() => void) => {\n let active = true;\n void (async () => {\n try {\n const next = await refreshUnreadCount();\n if (active) setUnreadCount(next);\n } catch {\n if (active) setUnreadCount(0);\n }\n })();\n return (): void => {\n active = false;\n };\n }, [refreshUnreadCount]);\n\n const value = useMemo<NotificationUnreadContextValue>(\n () => ({ refreshUnreadCount, unreadCount }),\n [refreshUnreadCount, unreadCount],\n );\n\n return (\n <NotificationUnreadContext.Provider value={value}>\n {children}\n </NotificationUnreadContext.Provider>\n );\n}\n\n/**\n * Read the current unread-notification count and a manual refresh helper.\n * Returns a zero/no-op stub when used outside\n * `<NotificationUnreadProvider>`.\n */\nexport function useNotificationUnread(): NotificationUnreadContextValue {\n const context = useContext(NotificationUnreadContext);\n if (!context) {\n return {\n refreshUnreadCount: async (): Promise<number> => 0,\n unreadCount: 0,\n };\n }\n return context;\n}\n",".logo {\n display: block;\n height: 24px;\n margin: auto;\n object-fit: contain;\n width: 24px;\n}\n\n.notificationBell {\n display: inline-flex;\n position: relative;\n}\n\n.notificationBadge {\n align-items: center;\n background: #d92d20;\n border: 1px solid #fff;\n border-radius: 999px;\n color: #fff;\n display: inline-flex;\n font-size: 10px;\n font-weight: 600;\n height: 16px;\n justify-content: center;\n line-height: 1;\n min-width: 16px;\n padding: 0 4px;\n pointer-events: none;\n position: absolute;\n right: -4px;\n top: -4px;\n}\n","'use client';\n\nimport type { ReactElement, ReactNode } from 'react';\nimport {\n Layout,\n Navigation,\n NavigationFooter,\n NavigationHeader,\n NavigationIconButton,\n NavigationOption,\n NavigationOptionCategory,\n NavigationUserMenu,\n} from '@mezzanine-ui/react';\nimport {\n FileIcon,\n FolderIcon,\n HomeIcon,\n ListIcon,\n LogoutIcon,\n MailIcon,\n MailUnreadIcon,\n NotificationUnreadIcon,\n SearchIcon,\n ShareIcon,\n SystemIcon,\n SwitchHorizontalIcon,\n UserIcon,\n type IconDefinition,\n} from '@mezzanine-ui/icons';\nimport { logoutApi } from '@rytass/bpm-core-client';\nimport { useAuth } from '../lib/auth-provider';\nimport { useRouterAdapter } from '../lib/router-adapter';\nimport { useNotificationDrawer } from '../lib/notification-drawer-provider';\nimport { useNotificationUnread } from '../lib/notification-unread-provider';\nimport { useBPMRoutes, type BPMRoutes } from '../lib/routes-config';\nimport styles from './app-navigation.module.scss';\n\ninterface NavigationItem {\n readonly href: string;\n readonly icon: IconDefinition;\n readonly label: string;\n readonly requiresAdmin?: boolean;\n}\n\nexport interface AppNavigationGroup {\n readonly title: string;\n readonly items: readonly NavigationItem[];\n}\n\nfunction createDefaultNavigationGroups(\n routes: BPMRoutes,\n): readonly AppNavigationGroup[] {\n return [\n {\n title: '我的工作',\n items: [\n { href: routes.dashboard(), icon: HomeIcon, label: '工作台' },\n { href: routes.inbox(), icon: MailUnreadIcon, label: '我的待簽' },\n { href: routes.sent(), icon: MailIcon, label: '我發起的' },\n { href: routes.cc(), icon: ShareIcon, label: '抄送給我' },\n ],\n },\n {\n title: '查詢與代理',\n items: [\n { href: routes.search(), icon: SearchIcon, label: '搜尋' },\n { href: routes.delegations(), icon: SwitchHorizontalIcon, label: '個人代理' },\n ],\n },\n {\n title: '簽核設計',\n items: [\n { href: routes.templates(), icon: FolderIcon, label: '簽核模板', requiresAdmin: true },\n { href: routes.templateCategories(), icon: ListIcon, label: '模板分類', requiresAdmin: true },\n { href: routes.forms(), icon: FileIcon, label: '表單設計', requiresAdmin: true },\n ],\n },\n {\n title: '系統管理',\n items: [\n { href: routes.adminOrgs(), icon: SystemIcon, label: '組織管理', requiresAdmin: true },\n { href: routes.adminUsers(), icon: UserIcon, label: '會員對照', requiresAdmin: true },\n { href: routes.adminDelegations(), icon: ShareIcon, label: '代理設定', requiresAdmin: true },\n ],\n },\n ];\n}\n\nexport interface AppLayoutProps {\n /** Override the active href detection (defaults to router's pathname). */\n readonly activeHref?: string;\n /** Logo image URL displayed in the sidebar header. */\n readonly logoSrc?: string;\n /** Sidebar title (defaults to \"BPM Admin\"). */\n readonly title?: string;\n /**\n * Override the entire navigation tree. When omitted, the default 4-group\n * BPM admin nav (`我的工作` / `查詢與代理` / `簽核設計` / `系統管理`) is used.\n */\n readonly groups?: readonly AppNavigationGroup[];\n /** Page content rendered inside the Mezzanine `<Layout.Main>` slot. */\n readonly children?: ReactNode;\n}\n\n/**\n * BPM admin layout shell — composes Mezzanine `<Layout>` + `<Navigation>`\n * with the default 4-group BPM tree, and exposes a `children` prop that\n * fills the `<Layout.Main>` slot.\n *\n * Why a single component instead of a `<Navigation>` wrapper: Mezzanine\n * `<Layout>` discovers its slot children by component-identity match\n * (`child.type === Navigation` / `LayoutMain` / etc.). Any custom wrapper\n * around `<Navigation>` is silently dropped, so the sidebar disappears.\n * Keeping the `<Navigation>` element as a direct child of `<Layout>` here\n * is mandatory for the slot to register.\n */\nexport function AppLayout({\n activeHref,\n logoSrc = '/rytass-logo.png',\n title = 'BPM Admin',\n groups,\n children,\n}: AppLayoutProps): ReactElement {\n const router = useRouterAdapter();\n const routes = useBPMRoutes();\n const { member } = useAuth();\n const { unreadCount } = useNotificationUnread();\n const resolvedActive = activeHref ?? router.pathname ?? '';\n const isAdmin = isAdminMember(member);\n const resolvedGroups = groups ?? createDefaultNavigationGroups(routes);\n const visibleGroups = resolvedGroups\n .map((group) => ({\n title: group.title,\n items: group.items.filter((item) => !item.requiresAdmin || isAdmin),\n }))\n .filter((group) => group.items.length > 0);\n\n const handleLogout = async (): Promise<void> => {\n await logoutApi();\n router.replace('/login');\n };\n\n const navigationChildren = [\n <NavigationHeader key=\"header\" title={title}>\n <img alt=\"\" className={styles.logo} height={24} src={logoSrc} width={24} />\n </NavigationHeader>,\n ...visibleGroups.map((group) => (\n <NavigationOptionCategory key={group.title} title={group.title}>\n {group.items.map((item) => (\n <NavigationOption\n active={item.href === resolvedActive}\n href={item.href}\n icon={item.icon}\n key={item.href}\n title={item.label}\n />\n ))}\n </NavigationOptionCategory>\n )),\n <NavigationFooter key=\"footer\">\n <NavigationUserMenu\n options={[\n { id: 'notification-settings', name: '通知設定' },\n { id: 'logout', name: '登出' },\n ]}\n onSelect={(option): void => {\n if (option.id === 'notification-settings') {\n router.push(routes.notificationSettings());\n return;\n }\n if (option.id === 'logout') {\n void handleLogout();\n }\n }}\n >\n <NavigationMemberName />\n </NavigationUserMenu>\n <NotificationBell unreadCount={unreadCount} />\n <NavigationIconButton\n aria-label=\"登出\"\n icon={LogoutIcon}\n onClick={(): void => {\n void handleLogout();\n }}\n title=\"登出\"\n type=\"button\"\n />\n </NavigationFooter>,\n ];\n\n return (\n <Layout>\n <Navigation exactActivatedMatch>{navigationChildren}</Navigation>\n <Layout.Main>{children}</Layout.Main>\n </Layout>\n );\n}\n\nfunction isAdminMember(member: ReturnType<typeof useAuth>['member']): boolean {\n if (!member) return false;\n return (\n (member.roles ?? []).includes('BPM_ADMIN') ||\n (member.permissions ?? []).some((p) =>\n ['bpm:*', 'bpm:admin', 'bpm.admin', 'bpm:admin:*'].includes(p),\n )\n );\n}\n\nfunction NotificationBell({\n unreadCount,\n}: {\n readonly unreadCount: number;\n}): ReactElement {\n const { open } = useNotificationDrawer();\n return (\n <span className={styles.notificationBell}>\n <NavigationIconButton\n aria-label={unreadCount > 0 ? `通知中心,${unreadCount} 則未讀` : '通知中心'}\n icon={NotificationUnreadIcon}\n onClick={(): void => {\n open();\n }}\n title=\"通知中心\"\n type=\"button\"\n />\n {unreadCount > 0 ? (\n <span className={styles.notificationBadge}>\n {unreadCount > 99 ? '99+' : unreadCount}\n </span>\n ) : null}\n </span>\n );\n}\n\nfunction NavigationMemberName(): ReactElement | null {\n const { member } = useAuth();\n if (!member) return null;\n return <>{member.name}</>;\n}\n"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ "use client";require('../app-navigation.css');var e=Object.create,t=Object.defineProperty,n=Object.getOwnPropertyDescriptor,r=Object.getOwnPropertyNames,i=Object.getPrototypeOf,a=Object.prototype.hasOwnProperty,o=(e,i,o,s)=>{if(i&&typeof i==`object`||typeof i==`function`)for(var c=r(i),l=0,u=c.length,d;l<u;l++)d=c[l],!a.call(e,d)&&d!==o&&t(e,d,{get:(e=>i[e]).bind(null,d),enumerable:!(s=n(i,d))||s.enumerable});return e},s=(n,r,a)=>(a=n==null?{}:e(i(n)),o(r||!n||!n.__esModule?t(a,`default`,{value:n,enumerable:!0}):a,n));const c=require("./auth-provider-BV8Iiwfb.cjs"),l=require("./routes-config-2aKbWq2H.cjs");let u=require("react"),d=require("@mezzanine-ui/react"),f=require("@rytass/bpm-core-client"),p=require("react/jsx-runtime"),m=require("@rytass/bpm-core-client/workflow"),h=require("@mezzanine-ui/icons");var g=(0,u.createContext)(null);function _({children:e}){let[t,n]=(0,u.useState)(!1),r=(0,u.useCallback)(()=>{n(!0)},[]),i=(0,u.useCallback)(()=>{n(!1)},[]),a=(0,u.useCallback)(()=>{n(e=>!e)},[]),o=(0,u.useMemo)(()=>({close:i,isOpen:t,open:r,toggle:a}),[i,t,r,a]);return(0,p.jsx)(g.Provider,{value:o,children:e})}function v(){return(0,u.useContext)(g)||{close:()=>void 0,isOpen:!1,open:()=>void 0,toggle:()=>void 0}}var y=(0,u.createContext)(null);function b({children:e}){let{member:t}=c.n(),n=t?.memberId??null,[r,i]=(0,u.useState)(0),a=(0,u.useCallback)(async()=>{if(!n)return i(0),0;let e=await(0,m.readUnreadNotificationCount)(n);return i(e),e},[n]);(0,u.useEffect)(()=>{let e=!0;return(async()=>{try{let t=await a();e&&i(t)}catch{e&&i(0)}})(),()=>{e=!1}},[a]);let o=(0,u.useMemo)(()=>({refreshUnreadCount:a,unreadCount:r}),[a,r]);return(0,p.jsx)(y.Provider,{value:o,children:e})}function x(){return(0,u.useContext)(y)||{refreshUnreadCount:async()=>0,unreadCount:0}}var S={logo:`bpm_logo_QvBLU`,notificationBell:`bpm_notificationBell_W-wl7`,notificationBadge:`bpm_notificationBadge_Gy3Eq`};function C(e){return[{title:`我的工作`,items:[{href:e.dashboard(),icon:h.HomeIcon,label:`工作台`},{href:e.inbox(),icon:h.MailUnreadIcon,label:`我的待簽`},{href:e.sent(),icon:h.MailIcon,label:`我發起的`},{href:e.cc(),icon:h.ShareIcon,label:`抄送給我`}]},{title:`查詢與代理`,items:[{href:e.search(),icon:h.SearchIcon,label:`搜尋`},{href:e.delegations(),icon:h.SwitchHorizontalIcon,label:`個人代理`}]},{title:`簽核設計`,items:[{href:e.templates(),icon:h.FolderIcon,label:`簽核模板`,requiresAdmin:!0},{href:e.templateCategories(),icon:h.ListIcon,label:`模板分類`,requiresAdmin:!0},{href:e.forms(),icon:h.FileIcon,label:`表單設計`,requiresAdmin:!0}]},{title:`系統管理`,items:[{href:e.adminOrgs(),icon:h.SystemIcon,label:`組織管理`,requiresAdmin:!0},{href:e.adminUsers(),icon:h.UserIcon,label:`會員對照`,requiresAdmin:!0},{href:e.adminDelegations(),icon:h.ShareIcon,label:`代理設定`,requiresAdmin:!0}]}]}function w({activeHref:e,logoSrc:t=`/rytass-logo.png`,title:n=`BPM Admin`,groups:r,children:i}){let a=c.a(),o=l.r(),{member:s}=c.n(),{unreadCount:u}=x(),m=e??a.pathname??``,g=T(s),_=(r??C(o)).map(e=>({title:e.title,items:e.items.filter(e=>!e.requiresAdmin||g)})).filter(e=>e.items.length>0),v=async()=>{await(0,f.logoutApi)(),a.replace(`/login`)};return(0,p.jsxs)(d.Layout,{children:[(0,p.jsx)(d.Navigation,{exactActivatedMatch:!0,children:[(0,p.jsx)(d.NavigationHeader,{title:n,children:(0,p.jsx)(`img`,{alt:``,className:S.logo,height:24,src:t,width:24})},`header`),..._.map(e=>(0,p.jsx)(d.NavigationOptionCategory,{title:e.title,children:e.items.map(e=>(0,p.jsx)(d.NavigationOption,{active:e.href===m,href:e.href,icon:e.icon,title:e.label},e.href))},e.title)),(0,p.jsxs)(d.NavigationFooter,{children:[(0,p.jsx)(d.NavigationUserMenu,{options:[{id:`notification-settings`,name:`通知設定`},{id:`logout`,name:`登出`}],onSelect:e=>{if(e.id===`notification-settings`){a.push(o.notificationSettings());return}e.id===`logout`&&v()},children:(0,p.jsx)(D,{})}),(0,p.jsx)(E,{unreadCount:u}),(0,p.jsx)(d.NavigationIconButton,{"aria-label":`登出`,icon:h.LogoutIcon,onClick:()=>{v()},title:`登出`,type:`button`})]},`footer`)]}),(0,p.jsx)(d.Layout.Main,{children:i})]})}function T(e){return e?(e.roles??[]).includes(`BPM_ADMIN`)||(e.permissions??[]).some(e=>[`bpm:*`,`bpm:admin`,`bpm.admin`,`bpm:admin:*`].includes(e)):!1}function E({unreadCount:e}){let{open:t}=v();return(0,p.jsxs)(`span`,{className:S.notificationBell,children:[(0,p.jsx)(d.NavigationIconButton,{"aria-label":e>0?`通知中心,${e} 則未讀`:`通知中心`,icon:h.NotificationUnreadIcon,onClick:()=>{t()},title:`通知中心`,type:`button`}),e>0?(0,p.jsx)(`span`,{className:S.notificationBadge,children:e>99?`99+`:e}):null]})}function D(){let{member:e}=c.n();return e?(0,p.jsx)(p.Fragment,{children:e.name}):null}Object.defineProperty(exports,"a",{enumerable:!0,get:function(){return v}}),Object.defineProperty(exports,"i",{enumerable:!0,get:function(){return _}}),Object.defineProperty(exports,"n",{enumerable:!0,get:function(){return b}}),Object.defineProperty(exports,"o",{enumerable:!0,get:function(){return s}}),Object.defineProperty(exports,"r",{enumerable:!0,get:function(){return x}}),Object.defineProperty(exports,"t",{enumerable:!0,get:function(){return w}});
2
+ //# sourceMappingURL=app-navigation-KnlJCUp1.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"app-navigation-KnlJCUp1.cjs","names":[],"sources":["../../src/lib/notification-drawer-provider.tsx","../../src/lib/notification-unread-provider.tsx","../../src/components/app-navigation.module.scss","../../src/components/app-navigation.tsx"],"sourcesContent":["'use client';\n\nimport {\n createContext,\n useCallback,\n useContext,\n useMemo,\n useState,\n type ReactElement,\n type ReactNode,\n} from 'react';\n\ninterface NotificationDrawerContextValue {\n readonly close: () => void;\n readonly isOpen: boolean;\n readonly open: () => void;\n readonly toggle: () => void;\n}\n\nconst NotificationDrawerContext =\n createContext<NotificationDrawerContextValue | null>(null);\n\ninterface NotificationDrawerProviderProps {\n readonly children: ReactNode;\n}\n\n/**\n * Controls the open/closed state of the BPM notification drawer. Wraps\n * children with a context that `<NotificationDrawer />` reads to mount /\n * hide itself, and that `<AppLayout />` reads to open the drawer when\n * the bell icon is clicked.\n *\n * When used outside this provider, the returned hook is a safe no-op so\n * components don't crash in test or storybook environments.\n */\nexport function NotificationDrawerProvider({\n children,\n}: NotificationDrawerProviderProps): ReactElement {\n const [isOpen, setIsOpen] = useState(false);\n\n const open = useCallback((): void => {\n setIsOpen(true);\n }, []);\n const close = useCallback((): void => {\n setIsOpen(false);\n }, []);\n const toggle = useCallback((): void => {\n setIsOpen((current) => !current);\n }, []);\n\n const value = useMemo<NotificationDrawerContextValue>(\n () => ({ close, isOpen, open, toggle }),\n [close, isOpen, open, toggle],\n );\n\n return (\n <NotificationDrawerContext.Provider value={value}>\n {children}\n </NotificationDrawerContext.Provider>\n );\n}\n\n/**\n * Read the BPM notification drawer's open state and control helpers.\n * Returns a no-op stub when used outside `<NotificationDrawerProvider>`.\n */\nexport function useNotificationDrawer(): NotificationDrawerContextValue {\n const context = useContext(NotificationDrawerContext);\n if (!context) {\n return {\n close: (): void => undefined,\n isOpen: false,\n open: (): void => undefined,\n toggle: (): void => undefined,\n };\n }\n return context;\n}\n","'use client';\n\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useState,\n type ReactElement,\n type ReactNode,\n} from 'react';\nimport { readUnreadNotificationCount } from '@rytass/bpm-core-client/workflow';\nimport { useAuth } from './auth-provider';\n\ninterface NotificationUnreadContextValue {\n readonly refreshUnreadCount: () => Promise<number>;\n readonly unreadCount: number;\n}\n\nconst NotificationUnreadContext =\n createContext<NotificationUnreadContextValue | null>(null);\n\ninterface NotificationUnreadProviderProps {\n readonly children: ReactNode;\n}\n\n/**\n * Polls BPM for the current member's unread notification count via\n * `readUnreadNotificationCount` and exposes it through context for\n * `<AppLayout />` (bell badge) and `<NotificationDrawer />` (header\n * count). Refresh is triggered on mount and whenever the auth member id\n * changes; consumers can call `refreshUnreadCount()` after acknowledging\n * a notification.\n */\nexport function NotificationUnreadProvider({\n children,\n}: NotificationUnreadProviderProps): ReactElement {\n const { member } = useAuth();\n const currentMemberId = member?.memberId ?? null;\n const [unreadCount, setUnreadCount] = useState(0);\n\n const refreshUnreadCount = useCallback(async (): Promise<number> => {\n if (!currentMemberId) {\n setUnreadCount(0);\n return 0;\n }\n const next = await readUnreadNotificationCount(currentMemberId);\n setUnreadCount(next);\n return next;\n }, [currentMemberId]);\n\n useEffect((): (() => void) => {\n let active = true;\n void (async () => {\n try {\n const next = await refreshUnreadCount();\n if (active) setUnreadCount(next);\n } catch {\n if (active) setUnreadCount(0);\n }\n })();\n return (): void => {\n active = false;\n };\n }, [refreshUnreadCount]);\n\n const value = useMemo<NotificationUnreadContextValue>(\n () => ({ refreshUnreadCount, unreadCount }),\n [refreshUnreadCount, unreadCount],\n );\n\n return (\n <NotificationUnreadContext.Provider value={value}>\n {children}\n </NotificationUnreadContext.Provider>\n );\n}\n\n/**\n * Read the current unread-notification count and a manual refresh helper.\n * Returns a zero/no-op stub when used outside\n * `<NotificationUnreadProvider>`.\n */\nexport function useNotificationUnread(): NotificationUnreadContextValue {\n const context = useContext(NotificationUnreadContext);\n if (!context) {\n return {\n refreshUnreadCount: async (): Promise<number> => 0,\n unreadCount: 0,\n };\n }\n return context;\n}\n",".logo {\n display: block;\n height: 24px;\n margin: auto;\n object-fit: contain;\n width: 24px;\n}\n\n.notificationBell {\n display: inline-flex;\n position: relative;\n}\n\n.notificationBadge {\n align-items: center;\n background: #d92d20;\n border: 1px solid #fff;\n border-radius: 999px;\n color: #fff;\n display: inline-flex;\n font-size: 10px;\n font-weight: 600;\n height: 16px;\n justify-content: center;\n line-height: 1;\n min-width: 16px;\n padding: 0 4px;\n pointer-events: none;\n position: absolute;\n right: -4px;\n top: -4px;\n}\n","'use client';\n\nimport type { ReactElement, ReactNode } from 'react';\nimport {\n Layout,\n Navigation,\n NavigationFooter,\n NavigationHeader,\n NavigationIconButton,\n NavigationOption,\n NavigationOptionCategory,\n NavigationUserMenu,\n} from '@mezzanine-ui/react';\nimport {\n FileIcon,\n FolderIcon,\n HomeIcon,\n ListIcon,\n LogoutIcon,\n MailIcon,\n MailUnreadIcon,\n NotificationUnreadIcon,\n SearchIcon,\n ShareIcon,\n SystemIcon,\n SwitchHorizontalIcon,\n UserIcon,\n type IconDefinition,\n} from '@mezzanine-ui/icons';\nimport { logoutApi } from '@rytass/bpm-core-client';\nimport { useAuth } from '../lib/auth-provider';\nimport { useRouterAdapter } from '../lib/router-adapter';\nimport { useNotificationDrawer } from '../lib/notification-drawer-provider';\nimport { useNotificationUnread } from '../lib/notification-unread-provider';\nimport { useBPMRoutes, type BPMRoutes } from '../lib/routes-config';\nimport styles from './app-navigation.module.scss';\n\ninterface NavigationItem {\n readonly href: string;\n readonly icon: IconDefinition;\n readonly label: string;\n readonly requiresAdmin?: boolean;\n}\n\nexport interface AppNavigationGroup {\n readonly title: string;\n readonly items: readonly NavigationItem[];\n}\n\nfunction createDefaultNavigationGroups(\n routes: BPMRoutes,\n): readonly AppNavigationGroup[] {\n return [\n {\n title: '我的工作',\n items: [\n { href: routes.dashboard(), icon: HomeIcon, label: '工作台' },\n { href: routes.inbox(), icon: MailUnreadIcon, label: '我的待簽' },\n { href: routes.sent(), icon: MailIcon, label: '我發起的' },\n { href: routes.cc(), icon: ShareIcon, label: '抄送給我' },\n ],\n },\n {\n title: '查詢與代理',\n items: [\n { href: routes.search(), icon: SearchIcon, label: '搜尋' },\n { href: routes.delegations(), icon: SwitchHorizontalIcon, label: '個人代理' },\n ],\n },\n {\n title: '簽核設計',\n items: [\n { href: routes.templates(), icon: FolderIcon, label: '簽核模板', requiresAdmin: true },\n { href: routes.templateCategories(), icon: ListIcon, label: '模板分類', requiresAdmin: true },\n { href: routes.forms(), icon: FileIcon, label: '表單設計', requiresAdmin: true },\n ],\n },\n {\n title: '系統管理',\n items: [\n { href: routes.adminOrgs(), icon: SystemIcon, label: '組織管理', requiresAdmin: true },\n { href: routes.adminUsers(), icon: UserIcon, label: '會員對照', requiresAdmin: true },\n { href: routes.adminDelegations(), icon: ShareIcon, label: '代理設定', requiresAdmin: true },\n ],\n },\n ];\n}\n\nexport interface AppLayoutProps {\n /** Override the active href detection (defaults to router's pathname). */\n readonly activeHref?: string;\n /** Logo image URL displayed in the sidebar header. */\n readonly logoSrc?: string;\n /** Sidebar title (defaults to \"BPM Admin\"). */\n readonly title?: string;\n /**\n * Override the entire navigation tree. When omitted, the default 4-group\n * BPM admin nav (`我的工作` / `查詢與代理` / `簽核設計` / `系統管理`) is used.\n */\n readonly groups?: readonly AppNavigationGroup[];\n /** Page content rendered inside the Mezzanine `<Layout.Main>` slot. */\n readonly children?: ReactNode;\n}\n\n/**\n * BPM admin layout shell — composes Mezzanine `<Layout>` + `<Navigation>`\n * with the default 4-group BPM tree, and exposes a `children` prop that\n * fills the `<Layout.Main>` slot.\n *\n * Why a single component instead of a `<Navigation>` wrapper: Mezzanine\n * `<Layout>` discovers its slot children by component-identity match\n * (`child.type === Navigation` / `LayoutMain` / etc.). Any custom wrapper\n * around `<Navigation>` is silently dropped, so the sidebar disappears.\n * Keeping the `<Navigation>` element as a direct child of `<Layout>` here\n * is mandatory for the slot to register.\n */\nexport function AppLayout({\n activeHref,\n logoSrc = '/rytass-logo.png',\n title = 'BPM Admin',\n groups,\n children,\n}: AppLayoutProps): ReactElement {\n const router = useRouterAdapter();\n const routes = useBPMRoutes();\n const { member } = useAuth();\n const { unreadCount } = useNotificationUnread();\n const resolvedActive = activeHref ?? router.pathname ?? '';\n const isAdmin = isAdminMember(member);\n const resolvedGroups = groups ?? createDefaultNavigationGroups(routes);\n const visibleGroups = resolvedGroups\n .map((group) => ({\n title: group.title,\n items: group.items.filter((item) => !item.requiresAdmin || isAdmin),\n }))\n .filter((group) => group.items.length > 0);\n\n const handleLogout = async (): Promise<void> => {\n await logoutApi();\n router.replace('/login');\n };\n\n const navigationChildren = [\n <NavigationHeader key=\"header\" title={title}>\n <img alt=\"\" className={styles.logo} height={24} src={logoSrc} width={24} />\n </NavigationHeader>,\n ...visibleGroups.map((group) => (\n <NavigationOptionCategory key={group.title} title={group.title}>\n {group.items.map((item) => (\n <NavigationOption\n active={item.href === resolvedActive}\n href={item.href}\n icon={item.icon}\n key={item.href}\n title={item.label}\n />\n ))}\n </NavigationOptionCategory>\n )),\n <NavigationFooter key=\"footer\">\n <NavigationUserMenu\n options={[\n { id: 'notification-settings', name: '通知設定' },\n { id: 'logout', name: '登出' },\n ]}\n onSelect={(option): void => {\n if (option.id === 'notification-settings') {\n router.push(routes.notificationSettings());\n return;\n }\n if (option.id === 'logout') {\n void handleLogout();\n }\n }}\n >\n <NavigationMemberName />\n </NavigationUserMenu>\n <NotificationBell unreadCount={unreadCount} />\n <NavigationIconButton\n aria-label=\"登出\"\n icon={LogoutIcon}\n onClick={(): void => {\n void handleLogout();\n }}\n title=\"登出\"\n type=\"button\"\n />\n </NavigationFooter>,\n ];\n\n return (\n <Layout>\n <Navigation exactActivatedMatch>{navigationChildren}</Navigation>\n <Layout.Main>{children}</Layout.Main>\n </Layout>\n );\n}\n\nfunction isAdminMember(member: ReturnType<typeof useAuth>['member']): boolean {\n if (!member) return false;\n return (\n (member.roles ?? []).includes('BPM_ADMIN') ||\n (member.permissions ?? []).some((p) =>\n ['bpm:*', 'bpm:admin', 'bpm.admin', 'bpm:admin:*'].includes(p),\n )\n );\n}\n\nfunction NotificationBell({\n unreadCount,\n}: {\n readonly unreadCount: number;\n}): ReactElement {\n const { open } = useNotificationDrawer();\n return (\n <span className={styles.notificationBell}>\n <NavigationIconButton\n aria-label={unreadCount > 0 ? `通知中心,${unreadCount} 則未讀` : '通知中心'}\n icon={NotificationUnreadIcon}\n onClick={(): void => {\n open();\n }}\n title=\"通知中心\"\n type=\"button\"\n />\n {unreadCount > 0 ? (\n <span className={styles.notificationBadge}>\n {unreadCount > 99 ? '99+' : unreadCount}\n </span>\n ) : null}\n </span>\n );\n}\n\nfunction NavigationMemberName(): ReactElement | null {\n const { member } = useAuth();\n if (!member) return null;\n return <>{member.name}</>;\n}\n"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ "use client";require('../approval-instance-list-page.css');const e=require("./app-navigation-KnlJCUp1.cjs"),t=require("./auth-provider-BV8Iiwfb.cjs"),n=require("./format-date-time-26_pFvv4.cjs"),r=require("./routes-config-2aKbWq2H.cjs");let i=require("react"),a=require("@mezzanine-ui/react"),o=require("@rytass/bpm-core-client"),s=require("react/jsx-runtime"),c=require("@rytass/bpm-core-client/workflow"),l=require("@mezzanine-ui/react/ContentHeader");l=e.o(l,1);let u=require("@mezzanine-ui/core/form");var d={instanceFilterArea:`bpm_instanceFilterArea_qpvJq`},f=[10,20,50],p=[{id:`ALL`,name:`全部狀態`,state:null},{id:`RUNNING`,name:`進行中`,state:`RUNNING`},{id:`APPROVED`,name:`已通過`,state:`APPROVED`},{id:`REJECTED`,name:`已拒絕`,state:`REJECTED`},{id:`RETURNED`,name:`已退回`,state:`RETURNED`},{id:`CANCELLED`,name:`已取消`,state:`CANCELLED`},{id:`EXPIRED`,name:`已逾期`,state:`EXPIRED`},{id:`DRAFT`,name:`草稿`,state:`DRAFT`}];function m({activeHref:m,defaultState:y,description:b,emptyMessage:w,searchPlaceholder:T,title:E,view:D}){let O=t.a(),k=r.r(),[A,j]=(0,i.useState)(null),[M,N]=(0,i.useState)(new Map),[P,F]=(0,i.useState)(1),[I,L]=(0,i.useState)(10),[R,z]=(0,i.useState)(0),[B,V]=(0,i.useState)(!0),[H,U]=(0,i.useState)([]),[W,G]=(0,i.useState)(``),[K,q]=(0,i.useState)(_(y)),J=(0,i.useCallback)(async()=>{V(!0),j(null);try{let e=await(0,c.listApprovalInstancesPage)({page:P,pageSize:I,searchText:W,state:K.state,templateId:null,view:D});U(e.instances.map(h)),z(e.totalCount)}catch(e){j(C(e))}finally{V(!1)}},[P,I,W,K,D]);(0,i.useEffect)(()=>{J()},[J]),(0,i.useEffect)(()=>{let e=Array.from(new Set(H.map(e=>e.initiatorMemberId).filter(Boolean)));if(e.length===0){N(new Map);return}let t=!1;return(async()=>{try{let n=await(0,o.resolveMembers)(e);if(t)return;N(new Map(n.map(e=>[e.memberId,e])))}catch{if(t)return;N(new Map)}})(),()=>{t=!0}},[H]);let Y=(0,i.useMemo)(()=>[{dataIndex:`caseTitle`,key:`caseTitle`,title:`案件`,width:300},{key:`state`,render:e=>(0,s.jsx)(a.Typography,{color:x(e.state),component:`span`,variant:`body`,children:e.stateLabel}),title:`狀態`,width:120},{key:`initiatorMemberId`,render:e=>(0,s.jsx)(a.Typography,{component:`span`,variant:`body`,children:S(e.initiatorMemberId,M)}),title:`發起人`,width:180},{key:`startedAt`,render:e=>(0,s.jsx)(a.Typography,{component:`span`,variant:`body`,children:n.t(e.startedAt)}),title:`發起時間`,width:220},{key:`completedAt`,render:e=>(0,s.jsx)(a.Typography,{component:`span`,variant:`body`,children:n.t(e.completedAt)}),title:`完成時間`,width:220}],[M]),X=(0,i.useMemo)(()=>({render:e=>[{name:`查看`,onClick:()=>O.push(k.caseDetail(e.id))}],variant:`base-secondary`,width:88}),[O]);return(0,s.jsxs)(e.t,{activeHref:m,children:[(0,s.jsx)(a.PageHeader,{children:(0,s.jsx)(l.default,{description:b,title:E})}),(0,s.jsx)(a.SectionGroup,{children:(0,s.jsxs)(a.Section,{filterArea:(0,s.jsx)(a.FilterArea,{className:d.instanceFilterArea,size:`sub`,children:(0,s.jsxs)(a.FilterLine,{children:[(0,s.jsx)(a.Filter,{span:3,children:(0,s.jsx)(a.FormField,{fullWidth:!0,layout:u.FormFieldLayout.VERTICAL,name:`instanceSearchText`,children:(0,s.jsx)(a.Input,{fullWidth:!0,onChange:e=>{G(e.target.value),F(1)},placeholder:T,size:`sub`,value:W,variant:`base`})})}),(0,s.jsx)(a.Filter,{span:2,children:(0,s.jsx)(a.FormField,{fullWidth:!0,layout:u.FormFieldLayout.VERTICAL,name:`instanceState`,children:(0,s.jsx)(a.Select,{clearable:!1,fullWidth:!0,onChange:e=>{q(g(e)),F(1)},options:[...p],placeholder:`狀態`,renderValue:e=>`狀態:${v(e)}`,size:`sub`,value:K})})})]})}),children:[A?(0,s.jsx)(a.Typography,{color:`text-error`,variant:`body`,children:A}):null,!A&&!B&&H.length===0?(0,s.jsx)(a.Typography,{color:`text-neutral`,variant:`body`,children:w}):null,(0,s.jsx)(a.Table,{actions:X,columns:Y,dataSource:[...H],fullWidth:!0,loading:B,pagination:{current:P,onChange:e=>{F(e)},onChangePageSize:e=>{F(1),L(e)},pageSize:I,pageSizeLabel:`每頁筆數`,pageSizeOptions:f,renderResultSummary:(e,t,n)=>`顯示 ${e}-${t} 筆,共 ${n} 筆`,showPageSizeOptions:!0,total:R}})]})})]})}function h(e){return{...e,caseTitle:(0,c.readApprovalInstanceCaseTitle)(e),key:e.id,stateLabel:b(e.state)}}function g(e){return y(e)?_(e.state):p[0]}function _(e){return p.find(t=>t.state===e)??p[0]}function v(e){return y(e)?e.name:p[0].name}function y(e){return typeof e==`object`&&!!e&&`id`in e&&`name`in e&&`state`in e}function b(e){return e===`RUNNING`?`進行中`:e===`APPROVED`?`已通過`:e===`REJECTED`?`已拒絕`:e===`RETURNED`?`已退回`:e===`CANCELLED`?`已取消`:e===`EXPIRED`?`已逾期`:`草稿`}function x(e){return e===`APPROVED`?`text-success`:e===`REJECTED`||e===`CANCELLED`||e===`EXPIRED`?`text-error`:`text-neutral`}function S(e,t){let n=(e??``).trim();return n?t.get(n)?.name??n:`未知發起人`}function C(e){return e instanceof Error?e.message:`讀取簽核案件失敗。`}Object.defineProperty(exports,"t",{enumerable:!0,get:function(){return m}});
2
+ //# sourceMappingURL=approval-instance-list-page-CVXgE2K3.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"approval-instance-list-page-CVXgE2K3.cjs","names":[],"sources":["../../src/components/approval-instance-list-page.module.scss","../../src/components/approval-instance-list-page.tsx"],"sourcesContent":[".instanceFilterArea {\n :global(.mzn-filter-area__actions) {\n display: none;\n }\n\n :global(.mzn-form-field__label-area) {\n display: none;\n }\n\n :global(.mzn-form-field__control-field-slot--main) {\n width: 100%;\n min-width: 0;\n }\n}\n","'use client';\n\nimport type { ChangeEvent, ReactElement } from 'react';\nimport { useCallback, useEffect, useMemo, useState } from 'react';\nimport {\n Filter,\n FilterArea,\n FilterLine,\n FormField,\n Input,\n PageHeader,\n Section,\n SectionGroup,\n Select,\n Table,\n Typography,\n} from '@mezzanine-ui/react';\nimport ContentHeader from '@mezzanine-ui/react/ContentHeader';\nimport { FormFieldLayout } from '@mezzanine-ui/core/form';\nimport type { TableActions, TableColumn } from '@mezzanine-ui/core/table';\nimport { resolveMembers, type MemberProfileRecord } from '@rytass/bpm-core-client';\nimport {\n listApprovalInstancesPage,\n readApprovalInstanceCaseTitle,\n type ApprovalInstanceRecord,\n type ApprovalInstanceState,\n type ApprovalInstanceView,\n} from '@rytass/bpm-core-client/workflow';\nimport { useRouterAdapter } from '../lib/router-adapter';\nimport { useBPMRoutes } from '../lib/routes-config';\nimport { formatDateTime } from '../lib/format-date-time';\nimport { AppLayout } from './app-navigation';\nimport styles from './approval-instance-list-page.module.scss';\n\nexport interface ApprovalInstanceListPageProps {\n readonly activeHref: string;\n readonly defaultState: ApprovalInstanceState | null;\n readonly description: string;\n readonly emptyMessage: string;\n readonly searchPlaceholder: string;\n readonly title: string;\n readonly view: ApprovalInstanceView;\n}\n\ntype StateFilterOption = Readonly<{\n id: 'ALL' | ApprovalInstanceState;\n name: string;\n state: ApprovalInstanceState | null;\n}>;\n\ntype ApprovalInstanceRow = Readonly<\n Record<string, unknown> &\n ApprovalInstanceRecord & {\n caseTitle: string;\n key: string;\n stateLabel: string;\n }\n>;\n\nconst INSTANCE_PAGE_SIZE_OPTIONS = [10, 20, 50];\nconst STATE_FILTER_OPTIONS: readonly StateFilterOption[] = [\n { id: 'ALL', name: '全部狀態', state: null },\n { id: 'RUNNING', name: '進行中', state: 'RUNNING' },\n { id: 'APPROVED', name: '已通過', state: 'APPROVED' },\n { id: 'REJECTED', name: '已拒絕', state: 'REJECTED' },\n { id: 'RETURNED', name: '已退回', state: 'RETURNED' },\n { id: 'CANCELLED', name: '已取消', state: 'CANCELLED' },\n { id: 'EXPIRED', name: '已逾期', state: 'EXPIRED' },\n { id: 'DRAFT', name: '草稿', state: 'DRAFT' },\n];\n\n/**\n * Shared list page for any approval-instance \"view\" (inbox / sent / cc /\n * delegated). Caller picks the view + default state filter; the page renders\n * the standard BPM filter bar + paginated table and navigates to\n * `/instances/:id` on row action.\n */\nexport function ApprovalInstanceListPage({\n activeHref,\n defaultState,\n description,\n emptyMessage,\n searchPlaceholder,\n title,\n view,\n}: ApprovalInstanceListPageProps): ReactElement {\n const router = useRouterAdapter();\n const routes = useBPMRoutes();\n const [error, setError] = useState<string | null>(null);\n const [initiatorProfilesById, setInitiatorProfilesById] = useState<\n ReadonlyMap<string, MemberProfileRecord>\n >(new Map());\n const [instancePage, setInstancePage] = useState(1);\n const [instancePageSize, setInstancePageSize] = useState(10);\n const [instanceTotalCount, setInstanceTotalCount] = useState(0);\n const [loading, setLoading] = useState(true);\n const [rows, setRows] = useState<readonly ApprovalInstanceRow[]>([]);\n const [searchText, setSearchText] = useState('');\n const [stateFilter, setStateFilter] = useState<StateFilterOption>(\n readStateFilterOption(defaultState),\n );\n\n const refreshInstances = useCallback(async (): Promise<void> => {\n setLoading(true);\n setError(null);\n\n try {\n const result = await listApprovalInstancesPage({\n page: instancePage,\n pageSize: instancePageSize,\n searchText,\n state: stateFilter.state,\n templateId: null,\n view,\n });\n\n setRows(result.instances.map(readApprovalInstanceRow));\n setInstanceTotalCount(result.totalCount);\n } catch (requestError: unknown) {\n setError(readErrorMessage(requestError));\n } finally {\n setLoading(false);\n }\n }, [instancePage, instancePageSize, searchText, stateFilter, view]);\n\n useEffect((): void => {\n void refreshInstances();\n }, [refreshInstances]);\n\n useEffect((): (() => void) | void => {\n const initiatorMemberIds = Array.from(\n new Set(rows.map((row) => row.initiatorMemberId).filter(Boolean)),\n );\n\n if (initiatorMemberIds.length === 0) {\n setInitiatorProfilesById(new Map());\n\n return;\n }\n\n let cancelled = false;\n\n void (async (): Promise<void> => {\n try {\n const profiles = await resolveMembers(initiatorMemberIds);\n\n if (cancelled) {\n return;\n }\n\n setInitiatorProfilesById(\n new Map(profiles.map((profile) => [profile.memberId, profile])),\n );\n } catch {\n if (cancelled) {\n return;\n }\n\n setInitiatorProfilesById(new Map());\n }\n })();\n\n return (): void => {\n cancelled = true;\n };\n }, [rows]);\n\n const columns = useMemo(\n (): TableColumn<ApprovalInstanceRow>[] => [\n { dataIndex: 'caseTitle', key: 'caseTitle', title: '案件', width: 300 },\n {\n key: 'state',\n render: (record: ApprovalInstanceRow): ReactElement => (\n <Typography\n color={readInstanceStateColor(record.state)}\n component=\"span\"\n variant=\"body\"\n >\n {record.stateLabel}\n </Typography>\n ),\n title: '狀態',\n width: 120,\n },\n {\n key: 'initiatorMemberId',\n render: (record: ApprovalInstanceRow): ReactElement => (\n <Typography component=\"span\" variant=\"body\">\n {readInitiatorLabel(\n record.initiatorMemberId,\n initiatorProfilesById,\n )}\n </Typography>\n ),\n title: '發起人',\n width: 180,\n },\n {\n key: 'startedAt',\n render: (record: ApprovalInstanceRow): ReactElement => (\n <Typography component=\"span\" variant=\"body\">\n {formatDateTime(record.startedAt)}\n </Typography>\n ),\n title: '發起時間',\n width: 220,\n },\n {\n key: 'completedAt',\n render: (record: ApprovalInstanceRow): ReactElement => (\n <Typography component=\"span\" variant=\"body\">\n {formatDateTime(record.completedAt)}\n </Typography>\n ),\n title: '完成時間',\n width: 220,\n },\n ],\n [initiatorProfilesById],\n );\n const tableActions = useMemo(\n (): TableActions<ApprovalInstanceRow> => ({\n render: (\n record,\n ): ReturnType<TableActions<ApprovalInstanceRow>['render']> => [\n {\n name: '查看',\n onClick: (): void => router.push(routes.caseDetail(record.id)),\n },\n ],\n variant: 'base-secondary',\n width: 88,\n }),\n [router],\n );\n\n return (\n <AppLayout activeHref={activeHref}>\n <PageHeader>\n <ContentHeader description={description} title={title} />\n </PageHeader>\n\n <SectionGroup>\n <Section\n filterArea={\n <FilterArea className={styles.instanceFilterArea} size=\"sub\">\n <FilterLine>\n <Filter span={3}>\n <FormField\n fullWidth\n layout={FormFieldLayout.VERTICAL}\n name=\"instanceSearchText\"\n >\n <Input\n fullWidth\n onChange={(\n event: ChangeEvent<HTMLInputElement>,\n ): void => {\n setSearchText(event.target.value);\n setInstancePage(1);\n }}\n placeholder={searchPlaceholder}\n size=\"sub\"\n value={searchText}\n variant=\"base\"\n />\n </FormField>\n </Filter>\n <Filter span={2}>\n <FormField\n fullWidth\n layout={FormFieldLayout.VERTICAL}\n name=\"instanceState\"\n >\n <Select\n clearable={false}\n fullWidth\n onChange={(option): void => {\n setStateFilter(readSelectedStateFilterOption(option));\n setInstancePage(1);\n }}\n options={[...STATE_FILTER_OPTIONS]}\n placeholder=\"狀態\"\n renderValue={(value): string =>\n `狀態:${readStateFilterLabel(value)}`\n }\n size=\"sub\"\n value={stateFilter}\n />\n </FormField>\n </Filter>\n </FilterLine>\n </FilterArea>\n }\n >\n {error ? (\n <Typography color=\"text-error\" variant=\"body\">\n {error}\n </Typography>\n ) : null}\n {!error && !loading && rows.length === 0 ? (\n <Typography color=\"text-neutral\" variant=\"body\">\n {emptyMessage}\n </Typography>\n ) : null}\n <Table\n actions={tableActions}\n columns={columns}\n dataSource={[...rows]}\n fullWidth\n loading={loading}\n pagination={{\n current: instancePage,\n onChange: (page): void => {\n setInstancePage(page);\n },\n onChangePageSize: (pageSize): void => {\n setInstancePage(1);\n setInstancePageSize(pageSize);\n },\n pageSize: instancePageSize,\n pageSizeLabel: '每頁筆數',\n pageSizeOptions: INSTANCE_PAGE_SIZE_OPTIONS,\n renderResultSummary: (from, to, total): string =>\n `顯示 ${from}-${to} 筆,共 ${total} 筆`,\n showPageSizeOptions: true,\n total: instanceTotalCount,\n }}\n />\n </Section>\n </SectionGroup>\n </AppLayout>\n );\n}\n\nfunction readApprovalInstanceRow(\n instance: ApprovalInstanceRecord,\n): ApprovalInstanceRow {\n return {\n ...instance,\n caseTitle: readApprovalInstanceCaseTitle(instance),\n key: instance.id,\n stateLabel: readInstanceStateLabel(instance.state),\n };\n}\n\nfunction readSelectedStateFilterOption(option: unknown): StateFilterOption {\n if (!isStateFilterOption(option)) {\n return STATE_FILTER_OPTIONS[0];\n }\n return readStateFilterOption(option.state);\n}\n\nfunction readStateFilterOption(\n state: ApprovalInstanceState | null,\n): StateFilterOption {\n return (\n STATE_FILTER_OPTIONS.find((option) => option.state === state) ??\n STATE_FILTER_OPTIONS[0]\n );\n}\n\nfunction readStateFilterLabel(value: unknown): string {\n return isStateFilterOption(value) ? value.name : STATE_FILTER_OPTIONS[0].name;\n}\n\nfunction isStateFilterOption(value: unknown): value is StateFilterOption {\n return (\n typeof value === 'object' &&\n value !== null &&\n 'id' in value &&\n 'name' in value &&\n 'state' in value\n );\n}\n\nfunction readInstanceStateLabel(state: ApprovalInstanceState): string {\n if (state === 'RUNNING') return '進行中';\n if (state === 'APPROVED') return '已通過';\n if (state === 'REJECTED') return '已拒絕';\n if (state === 'RETURNED') return '已退回';\n if (state === 'CANCELLED') return '已取消';\n if (state === 'EXPIRED') return '已逾期';\n return '草稿';\n}\n\nfunction readInstanceStateColor(\n state: ApprovalInstanceState,\n): 'text-error' | 'text-neutral' | 'text-success' {\n if (state === 'APPROVED') return 'text-success';\n if (state === 'REJECTED' || state === 'CANCELLED' || state === 'EXPIRED') {\n return 'text-error';\n }\n return 'text-neutral';\n}\n\nfunction readInitiatorLabel(\n initiatorMemberId: string | null | undefined,\n initiatorProfilesById: ReadonlyMap<string, MemberProfileRecord>,\n): string {\n const trimmedMemberId = (initiatorMemberId ?? '').trim();\n if (!trimmedMemberId) return '未知發起人';\n return initiatorProfilesById.get(trimmedMemberId)?.name ?? trimmedMemberId;\n}\n\nfunction readErrorMessage(error: unknown): string {\n return error instanceof Error ? error.message : '讀取簽核案件失敗。';\n}\n"],"mappings":"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"}