@yungu-fed/class-student-roster 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,2 @@
1
+ # classStudentRoster
2
+
@@ -0,0 +1,9 @@
1
+ "use strict";Object.defineProperties(exports,{__esModule:{value:!0},[Symbol.toStringTag]:{value:"Module"}});var o=require("react");function _(m){return m&&typeof m=="object"&&"default"in m?m:{default:m}}var $=_(o);const ee="_iconfont_zvw5p_7",se="_csrMask_zvw5p_14",te="_csrModal_zvw5p_24",ce="_csrHeader_zvw5p_34",re="_csrTitle_zvw5p_44",ne="_csrClose_zvw5p_50",ae="_csrBody_zvw5p_62",de="_csrPanel_zvw5p_69",ie="_csrPanelHeader_zvw5p_81",Ae="_csrSearch_zvw5p_86",ue="_csrSearchIcon_zvw5p_102",me="_csrTree_zvw5p_113",oe="_csrTreeItem_zvw5p_118",le="_csrTreeItemText_zvw5p_135",he="_csrMeta_zvw5p_138",fe="_csrTreeItemActive_zvw5p_142",we="_csrTreeChildren_zvw5p_151",Ie="_csrList_zvw5p_154",ve="_csrListItem_zvw5p_161",Ce="_csrOps_zvw5p_185",Le="_hasSelected_zvw5p_196",pe="_clearBtn_zvw5p_201",ge="_csrChipWrap_zvw5p_205",ze="_csrChipList_zvw5p_210",Ee="_deleteIcon_zvw5p_218",Ne="_csrChip_zvw5p_205",Be="_csrFooter_zvw5p_238",Fe="_csrBtn_zvw5p_245",Me="_csrBtnPrimary_zvw5p_254",Pe="_csrBtnCancel_zvw5p_259",Ze="_csrEmpty_zvw5p_264",be="_csrCheck_zvw5p_279",We="_csrCheckLabel_zvw5p_289";var t={iconfont:ee,csrMask:se,csrModal:te,csrHeader:ce,csrTitle:re,csrClose:ne,csrBody:ae,csrPanel:de,csrPanelHeader:ie,csrSearch:Ae,csrSearchIcon:ue,csrTree:me,csrTreeItem:oe,csrTreeItemText:le,csrMeta:he,csrTreeItemActive:fe,csrTreeChildren:we,csrList:Ie,csrListItem:ve,csrOps:Ce,hasSelected:Le,clearBtn:pe,csrChipWrap:ge,csrChipList:ze,deleteIcon:Ee,csrChip:Ne,csrFooter:Be,csrBtn:Fe,csrBtnPrimary:Me,csrBtnCancel:Pe,csrEmpty:Ze,csrCheck:be,csrCheckLabel:We},Q="",S={exports:{}},W={};/**
2
+ * @license React
3
+ * react-jsx-runtime.production.min.js
4
+ *
5
+ * Copyright (c) Facebook, Inc. and its affiliates.
6
+ *
7
+ * This source code is licensed under the MIT license found in the
8
+ * LICENSE file in the root directory of this source tree.
9
+ */var ye=$.default,Se=Symbol.for("react.element"),je=Symbol.for("react.fragment"),Oe=Object.prototype.hasOwnProperty,ke=ye.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,Te={key:!0,ref:!0,__self:!0,__source:!0};function H(m,l,E){var h,p={},f=null,B=null;E!==void 0&&(f=""+E),l.key!==void 0&&(f=""+l.key),l.ref!==void 0&&(B=l.ref);for(h in l)Oe.call(l,h)&&!Te.hasOwnProperty(h)&&(p[h]=l[h]);if(m&&m.defaultProps)for(h in l=m.defaultProps,l)p[h]===void 0&&(p[h]=l[h]);return{$$typeof:Se,type:m,key:f,ref:B,props:p,_owner:ke.current}}W.Fragment=je;W.jsx=H;W.jsxs=H;S.exports=W;const i=S.exports.jsx,A=S.exports.jsxs;function xe({visible:m=!1,onClose:l,onSave:E,loadData:h,title:p="\u9009\u62E9\u5B66\u751F"}){const[f,B]=o.useState([]),[F,J]=o.useState([]),[M,I]=o.useState(null),[N,v]=o.useState(null),[C,g]=o.useState(new Set),[z,D]=o.useState(""),[y,j]=o.useState(!1),[P,O]=o.useState(""),G=e=>Array.isArray(e==null?void 0:e.grades)&&e.grades.length?e.grades:Array.isArray(e==null?void 0:e.classes)&&e.classes.length?[{id:"virtual-grade",name:e.defaultGradeName||"\u6240\u6709\u73ED\u7EA7",classes:e.classes}]:[];o.useEffect(()=>{if(!m)return;let e=!0;return(async()=>{var r,s;try{j(!0),O("");const a=h?await h():{grades:[],students:[]};if(!e)return;const d=G(a);B(d),J(a.students||[]);const c=d==null?void 0:d[0];I((c==null?void 0:c.id)||null),v(((s=(r=c==null?void 0:c.classes)==null?void 0:r[0])==null?void 0:s.id)||null)}catch(a){if(!e)return;O((a==null?void 0:a.message)||"\u52A0\u8F7D\u5931\u8D25")}finally{e&&j(!1)}})(),()=>{e=!1}},[m,h]);const Z=o.useMemo(()=>{const e=new Map;return F.forEach(r=>e.set(r.id,r)),e},[F]),L=o.useMemo(()=>{var a,d,c;if(!M)return[];const e=f.find(n=>n.id===M);if(!e)return[];const r=new Set;if(N){const n=(a=e.classes)==null?void 0:a.find(u=>u.id===N);(d=n==null?void 0:n.studentIds)==null||d.forEach(u=>r.add(u))}else(c=e.classes)==null||c.forEach(n=>{var u;return(u=n.studentIds)==null?void 0:u.forEach(w=>r.add(w))});let s=Array.from(r).map(n=>Z.get(n)).filter(Boolean);if(z.trim()){const n=z.trim().toLowerCase();s=s.filter(u=>u.name.toLowerCase().includes(n))}return s},[M,N,f,Z,z]),b=o.useMemo(()=>Array.from(C).map(e=>Z.get(e)).filter(Boolean),[C,Z]),k=e=>{g(r=>{const s=new Set(r);return s.has(e)?s.delete(e):s.add(e),s})},V=e=>{if(e){const r=L.map(s=>s.id);g(s=>new Set([...s,...r]))}else g(r=>{const s=new Set(r);return L.forEach(a=>s.delete(a.id)),s})},Y=(e,r,s)=>{const a=(r==null?void 0:r.studentIds)||[];g(d=>{const c=new Set(d);return s?a.forEach(n=>c.add(n)):a.forEach(n=>c.delete(n)),c}),I(e),v((r==null?void 0:r.id)||null)},K=(e,r)=>{var a;const s=[];(a=e.classes)==null||a.forEach(d=>{var c;return(c=d.studentIds)==null?void 0:c.forEach(n=>s.push(n))}),g(d=>{const c=new Set(d);return r?s.forEach(n=>c.add(n)):s.forEach(n=>c.delete(n)),c}),I(e.id),v(null)},R=(e,r)=>{var u;const s=f.find(w=>w.id===e),a=(u=s==null?void 0:s.classes)==null?void 0:u.find(w=>w.id===r),d=(a==null?void 0:a.studentIds)||[],c=d.length,n=d.filter(w=>C.has(w)).length;return{checked:c>0&&n===c,indeterminate:n>0&&n<c}},T=e=>{var d;const r=[];(d=e.classes)==null||d.forEach(c=>{var n;return(n=c.studentIds)==null?void 0:n.forEach(u=>r.push(u))});const s=r.length,a=r.filter(c=>C.has(c)).length;return{checked:s>0&&a===s,indeterminate:a>0&&a<s}},X=(e,r)=>{var n;const s=f.find(u=>u.id===e),a=(n=s==null?void 0:s.classes)==null?void 0:n.find(u=>u.id===r),d=(a==null?void 0:a.studentIds)||[];return{selectedCount:d.filter(u=>C.has(u)).length,total:d.length}},q=()=>g(new Set),U=()=>{const e=b.map(r=>({id:r.id,name:r.name}));E&&E(e)};return o.useEffect(()=>{const e=z.trim().toLowerCase();if(!e)return;const r=f.find(c=>{var n;return(n=c.name)==null?void 0:n.toLowerCase().includes(e)});if(r){I(r.id),v(null);return}let s=null,a=null;if(f.some(c=>{var u;const n=(u=c.classes)==null?void 0:u.find(w=>{var x;return(x=w.name)==null?void 0:x.toLowerCase().includes(e)});return n?(s=n,a=c.id,!0):!1}),s){I(a),v(s.id);return}const d=F.find(c=>{var n;return(n=c.name)==null?void 0:n.toLowerCase().includes(e)});d&&(I(d.gradeId||null),v(d.classId||null))},[z,f,F]),m?i("div",{className:t.csrMask,children:A("div",{className:t.csrModal,children:[A("div",{className:t.csrHeader,children:[i("h3",{className:t.csrTitle,children:p}),i("i",{className:`${t.iconfont} ${t.csrClose}`,onClick:l,children:"\uE6A9"})]}),A("div",{className:t.csrBody,children:[A("div",{className:t.csrPanel,children:[A("div",{className:t.csrSearch,children:[i("input",{placeholder:"\u8BF7\u8F93\u5165\u5B66\u751F\u59D3\u540D/\u5B66\u53F7\u641C\u7D22",value:z,onChange:e=>D(e.target.value)}),i("i",{className:`${t.iconfont} ${t.csrSearchIcon}`,children:"\uE61D"})]}),A("div",{className:t.csrTree,children:[y&&i("div",{className:t.csrEmpty,children:"\u52A0\u8F7D\u4E2D..."}),P&&i("div",{className:t.csrEmpty,children:P}),!y&&!P&&f.map(e=>{var r;return A("div",{children:[A("div",{className:`${t.csrTreeItem} ${M===e.id&&!N?t.csrTreeItemActive:""}`,children:[i("input",{className:t.csrCheck,type:"checkbox",name:"grade",checked:T(e).checked,ref:s=>{s&&(s.indeterminate=T(e).indeterminate)},onClick:s=>s.stopPropagation(),onChange:s=>K(e,s.target.checked)}),i("span",{className:t.csrTreeItemText,onClick:()=>{I(e.id),v(null)},children:e.name})]}),i("div",{className:t.csrTreeChildren,children:(r=e.classes)==null?void 0:r.map(s=>{const a=X(e.id,s.id),d=R(e.id,s.id);return A("div",{className:`${t.csrTreeItem} ${N===s.id?t.csrTreeItemActive:""}`,children:[i("input",{className:t.csrCheck,type:"checkbox",name:"class",checked:d.checked,ref:c=>{c&&(c.indeterminate=d.indeterminate)},onClick:c=>c.stopPropagation(),onChange:c=>Y(e.id,s,c.target.checked)}),i("span",{className:t.csrTreeItemText,onClick:()=>{I(e.id),v(s.id)},children:s.name}),A("span",{className:t.csrMeta,children:[a.selectedCount,a.total?`/${a.total}`:""]})]},s.id)})})]},e.id)}),!y&&!P&&!f.length&&A("div",{className:t.csrEmpty,children:[i("img",{src:Q,alt:"blank"}),i("span",{children:"\u6682\u65E0\u6570\u636E"})]})]})]}),A("div",{className:t.csrPanel,children:[A("div",{className:t.csrOps,children:[A("label",{className:t.csrCheckLabel,children:[i("input",{className:t.csrCheck,type:"checkbox",checked:L.length>0&&L.every(e=>C.has(e.id)),onChange:e=>V(e.target.checked)}),"\u5168\u9009"]}),A("span",{className:t.csrMeta,children:["\u5171 ",L.length," \u4EBA"]})]}),A("ul",{className:t.csrList,children:[L.map(e=>A("li",{className:t.csrListItem,children:[i("input",{className:t.csrCheck,type:"checkbox",checked:C.has(e.id),onChange:()=>k(e.id)}),i("div",{children:e.name})]},e.id)),!L.length&&A("div",{className:t.csrEmpty,children:[i("img",{src:Q,alt:"blank"}),i("span",{children:"\u6682\u65E0\u5B66\u751F"})]})]})]}),A("div",{className:t.csrPanel,children:[A("div",{className:t.csrOps,children:[A("span",{className:t.hasSelected,children:["\u5DF2\u9009 ",b.length," \u4EBA"]}),i("span",{className:t.clearBtn,onClick:q,children:"\u6E05\u7A7A"})]}),i("div",{className:t.csrChipWrap,children:b.length?i("ul",{className:t.csrChipList,children:b.map(e=>A("li",{className:t.csrChip,children:[i("span",{children:e.name}),i("i",{className:`${t.iconfont} ${t.deleteIcon}`,onClick:()=>k(e.id),children:"\uE6A9"})]},e.id))}):i("div",{className:t.csrEmpty,children:"\u8BF7\u9009\u62E9"})})]})]}),A("div",{className:t.csrFooter,children:[i("button",{className:`${t.csrBtn} ${t.csrBtnCancel}`,onClick:l,children:"\u53D6\u6D88"}),i("button",{className:`${t.csrBtn} ${t.csrBtnPrimary}`,onClick:U,children:"\u4FDD\u5B58"})]})]})}):null}exports.default=xe;
@@ -0,0 +1,524 @@
1
+ import require$$0, { useState, useEffect, useMemo } from "react";
2
+ const iconfont = "_iconfont_zvw5p_7";
3
+ const csrMask = "_csrMask_zvw5p_14";
4
+ const csrModal = "_csrModal_zvw5p_24";
5
+ const csrHeader = "_csrHeader_zvw5p_34";
6
+ const csrTitle = "_csrTitle_zvw5p_44";
7
+ const csrClose = "_csrClose_zvw5p_50";
8
+ const csrBody = "_csrBody_zvw5p_62";
9
+ const csrPanel = "_csrPanel_zvw5p_69";
10
+ const csrPanelHeader = "_csrPanelHeader_zvw5p_81";
11
+ const csrSearch = "_csrSearch_zvw5p_86";
12
+ const csrSearchIcon = "_csrSearchIcon_zvw5p_102";
13
+ const csrTree = "_csrTree_zvw5p_113";
14
+ const csrTreeItem = "_csrTreeItem_zvw5p_118";
15
+ const csrTreeItemText = "_csrTreeItemText_zvw5p_135";
16
+ const csrMeta = "_csrMeta_zvw5p_138";
17
+ const csrTreeItemActive = "_csrTreeItemActive_zvw5p_142";
18
+ const csrTreeChildren = "_csrTreeChildren_zvw5p_151";
19
+ const csrList = "_csrList_zvw5p_154";
20
+ const csrListItem = "_csrListItem_zvw5p_161";
21
+ const csrOps = "_csrOps_zvw5p_185";
22
+ const hasSelected = "_hasSelected_zvw5p_196";
23
+ const clearBtn = "_clearBtn_zvw5p_201";
24
+ const csrChipWrap = "_csrChipWrap_zvw5p_205";
25
+ const csrChipList = "_csrChipList_zvw5p_210";
26
+ const deleteIcon = "_deleteIcon_zvw5p_218";
27
+ const csrChip = "_csrChip_zvw5p_205";
28
+ const csrFooter = "_csrFooter_zvw5p_238";
29
+ const csrBtn = "_csrBtn_zvw5p_245";
30
+ const csrBtnPrimary = "_csrBtnPrimary_zvw5p_254";
31
+ const csrBtnCancel = "_csrBtnCancel_zvw5p_259";
32
+ const csrEmpty = "_csrEmpty_zvw5p_264";
33
+ const csrCheck = "_csrCheck_zvw5p_279";
34
+ const csrCheckLabel = "_csrCheckLabel_zvw5p_289";
35
+ var styles = {
36
+ iconfont,
37
+ csrMask,
38
+ csrModal,
39
+ csrHeader,
40
+ csrTitle,
41
+ csrClose,
42
+ csrBody,
43
+ csrPanel,
44
+ csrPanelHeader,
45
+ csrSearch,
46
+ csrSearchIcon,
47
+ csrTree,
48
+ csrTreeItem,
49
+ csrTreeItemText,
50
+ csrMeta,
51
+ csrTreeItemActive,
52
+ csrTreeChildren,
53
+ csrList,
54
+ csrListItem,
55
+ csrOps,
56
+ hasSelected,
57
+ clearBtn,
58
+ csrChipWrap,
59
+ csrChipList,
60
+ deleteIcon,
61
+ csrChip,
62
+ csrFooter,
63
+ csrBtn,
64
+ csrBtnPrimary,
65
+ csrBtnCancel,
66
+ csrEmpty,
67
+ csrCheck,
68
+ csrCheckLabel
69
+ };
70
+ var blankIcon = "";
71
+ var jsxRuntime = { exports: {} };
72
+ var reactJsxRuntime_production_min = {};
73
+ /**
74
+ * @license React
75
+ * react-jsx-runtime.production.min.js
76
+ *
77
+ * Copyright (c) Facebook, Inc. and its affiliates.
78
+ *
79
+ * This source code is licensed under the MIT license found in the
80
+ * LICENSE file in the root directory of this source tree.
81
+ */
82
+ var f = require$$0, k = Symbol.for("react.element"), l = Symbol.for("react.fragment"), m = Object.prototype.hasOwnProperty, n = f.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner, p = { key: true, ref: true, __self: true, __source: true };
83
+ function q(c, a, g) {
84
+ var b, d = {}, e = null, h = null;
85
+ void 0 !== g && (e = "" + g);
86
+ void 0 !== a.key && (e = "" + a.key);
87
+ void 0 !== a.ref && (h = a.ref);
88
+ for (b in a)
89
+ m.call(a, b) && !p.hasOwnProperty(b) && (d[b] = a[b]);
90
+ if (c && c.defaultProps)
91
+ for (b in a = c.defaultProps, a)
92
+ void 0 === d[b] && (d[b] = a[b]);
93
+ return { $$typeof: k, type: c, key: e, ref: h, props: d, _owner: n.current };
94
+ }
95
+ reactJsxRuntime_production_min.Fragment = l;
96
+ reactJsxRuntime_production_min.jsx = q;
97
+ reactJsxRuntime_production_min.jsxs = q;
98
+ {
99
+ jsxRuntime.exports = reactJsxRuntime_production_min;
100
+ }
101
+ const jsx = jsxRuntime.exports.jsx;
102
+ const jsxs = jsxRuntime.exports.jsxs;
103
+ function ClassStudentRoster({
104
+ visible = false,
105
+ onClose,
106
+ onSave,
107
+ loadData,
108
+ title = "\u9009\u62E9\u5B66\u751F"
109
+ }) {
110
+ const [grades, setGrades] = useState([]);
111
+ const [students, setStudents] = useState([]);
112
+ const [activeGradeId, setActiveGradeId] = useState(null);
113
+ const [activeClassId, setActiveClassId] = useState(null);
114
+ const [selectedIds, setSelectedIds] = useState(/* @__PURE__ */ new Set());
115
+ const [query, setQuery] = useState("");
116
+ const [loading, setLoading] = useState(false);
117
+ const [error, setError] = useState("");
118
+ const normalizeGrades = (res) => {
119
+ if (Array.isArray(res == null ? void 0 : res.grades) && res.grades.length)
120
+ return res.grades;
121
+ if (Array.isArray(res == null ? void 0 : res.classes) && res.classes.length) {
122
+ return [{
123
+ id: "virtual-grade",
124
+ name: res.defaultGradeName || "\u6240\u6709\u73ED\u7EA7",
125
+ classes: res.classes
126
+ }];
127
+ }
128
+ return [];
129
+ };
130
+ useEffect(() => {
131
+ if (!visible)
132
+ return;
133
+ let mounted = true;
134
+ (async () => {
135
+ var _a, _b;
136
+ try {
137
+ setLoading(true);
138
+ setError("");
139
+ const res = loadData ? await loadData() : {
140
+ grades: [],
141
+ students: []
142
+ };
143
+ if (!mounted)
144
+ return;
145
+ const normalizedGrades = normalizeGrades(res);
146
+ setGrades(normalizedGrades);
147
+ setStudents(res.students || []);
148
+ const firstGrade = normalizedGrades == null ? void 0 : normalizedGrades[0];
149
+ setActiveGradeId((firstGrade == null ? void 0 : firstGrade.id) || null);
150
+ setActiveClassId(((_b = (_a = firstGrade == null ? void 0 : firstGrade.classes) == null ? void 0 : _a[0]) == null ? void 0 : _b.id) || null);
151
+ } catch (e) {
152
+ if (!mounted)
153
+ return;
154
+ setError((e == null ? void 0 : e.message) || "\u52A0\u8F7D\u5931\u8D25");
155
+ } finally {
156
+ if (mounted)
157
+ setLoading(false);
158
+ }
159
+ })();
160
+ return () => {
161
+ mounted = false;
162
+ };
163
+ }, [visible, loadData]);
164
+ const studentMap = useMemo(() => {
165
+ const map = /* @__PURE__ */ new Map();
166
+ students.forEach((s) => map.set(s.id, s));
167
+ return map;
168
+ }, [students]);
169
+ const activeStudents = useMemo(() => {
170
+ var _a, _b, _c;
171
+ if (!activeGradeId)
172
+ return [];
173
+ const grade = grades.find((g) => g.id === activeGradeId);
174
+ if (!grade)
175
+ return [];
176
+ const ids = /* @__PURE__ */ new Set();
177
+ if (activeClassId) {
178
+ const cls = (_a = grade.classes) == null ? void 0 : _a.find((c) => c.id === activeClassId);
179
+ (_b = cls == null ? void 0 : cls.studentIds) == null ? void 0 : _b.forEach((id) => ids.add(id));
180
+ } else {
181
+ (_c = grade.classes) == null ? void 0 : _c.forEach((c) => {
182
+ var _a2;
183
+ return (_a2 = c.studentIds) == null ? void 0 : _a2.forEach((id) => ids.add(id));
184
+ });
185
+ }
186
+ let list = Array.from(ids).map((id) => studentMap.get(id)).filter(Boolean);
187
+ if (query.trim()) {
188
+ const q2 = query.trim().toLowerCase();
189
+ list = list.filter((s) => s.name.toLowerCase().includes(q2));
190
+ }
191
+ return list;
192
+ }, [activeGradeId, activeClassId, grades, studentMap, query]);
193
+ const selectedList = useMemo(() => Array.from(selectedIds).map((id) => studentMap.get(id)).filter(Boolean), [selectedIds, studentMap]);
194
+ const toggleOne = (id) => {
195
+ setSelectedIds((prev) => {
196
+ const next = new Set(prev);
197
+ if (next.has(id))
198
+ next.delete(id);
199
+ else
200
+ next.add(id);
201
+ return next;
202
+ });
203
+ };
204
+ const toggleAll = (checked) => {
205
+ if (checked) {
206
+ const ids = activeStudents.map((s) => s.id);
207
+ setSelectedIds((prev) => /* @__PURE__ */ new Set([...prev, ...ids]));
208
+ } else {
209
+ setSelectedIds((prev) => {
210
+ const next = new Set(prev);
211
+ activeStudents.forEach((s) => next.delete(s.id));
212
+ return next;
213
+ });
214
+ }
215
+ };
216
+ const toggleClass = (gradeId, cls, checked) => {
217
+ const ids = (cls == null ? void 0 : cls.studentIds) || [];
218
+ setSelectedIds((prev) => {
219
+ const next = new Set(prev);
220
+ if (checked)
221
+ ids.forEach((id) => next.add(id));
222
+ else
223
+ ids.forEach((id) => next.delete(id));
224
+ return next;
225
+ });
226
+ setActiveGradeId(gradeId);
227
+ setActiveClassId((cls == null ? void 0 : cls.id) || null);
228
+ };
229
+ const toggleGrade = (grade, checked) => {
230
+ var _a;
231
+ const ids = [];
232
+ (_a = grade.classes) == null ? void 0 : _a.forEach((c) => {
233
+ var _a2;
234
+ return (_a2 = c.studentIds) == null ? void 0 : _a2.forEach((id) => ids.push(id));
235
+ });
236
+ setSelectedIds((prev) => {
237
+ const next = new Set(prev);
238
+ if (checked)
239
+ ids.forEach((id) => next.add(id));
240
+ else
241
+ ids.forEach((id) => next.delete(id));
242
+ return next;
243
+ });
244
+ setActiveGradeId(grade.id);
245
+ setActiveClassId(null);
246
+ };
247
+ const getClassStatus = (gradeId, classId) => {
248
+ var _a;
249
+ const grade = grades.find((g) => g.id === gradeId);
250
+ const cls = (_a = grade == null ? void 0 : grade.classes) == null ? void 0 : _a.find((c) => c.id === classId);
251
+ const ids = (cls == null ? void 0 : cls.studentIds) || [];
252
+ const total = ids.length;
253
+ const selectedCount = ids.filter((id) => selectedIds.has(id)).length;
254
+ return {
255
+ checked: total > 0 && selectedCount === total,
256
+ indeterminate: selectedCount > 0 && selectedCount < total
257
+ };
258
+ };
259
+ const getGradeStatus = (grade) => {
260
+ var _a;
261
+ const ids = [];
262
+ (_a = grade.classes) == null ? void 0 : _a.forEach((c) => {
263
+ var _a2;
264
+ return (_a2 = c.studentIds) == null ? void 0 : _a2.forEach((id) => ids.push(id));
265
+ });
266
+ const total = ids.length;
267
+ const selectedCount = ids.filter((id) => selectedIds.has(id)).length;
268
+ return {
269
+ checked: total > 0 && selectedCount === total,
270
+ indeterminate: selectedCount > 0 && selectedCount < total
271
+ };
272
+ };
273
+ const getClassSelectedInfo = (gradeId, classId) => {
274
+ var _a;
275
+ const grade = grades.find((g) => g.id === gradeId);
276
+ const cls = (_a = grade == null ? void 0 : grade.classes) == null ? void 0 : _a.find((c) => c.id === classId);
277
+ const ids = (cls == null ? void 0 : cls.studentIds) || [];
278
+ const selectedCount = ids.filter((id) => selectedIds.has(id)).length;
279
+ return {
280
+ selectedCount,
281
+ total: ids.length
282
+ };
283
+ };
284
+ const clearAll = () => setSelectedIds(/* @__PURE__ */ new Set());
285
+ const handleSave = () => {
286
+ const payload = selectedList.map((s) => ({
287
+ id: s.id,
288
+ name: s.name
289
+ }));
290
+ onSave && onSave(payload);
291
+ };
292
+ useEffect(() => {
293
+ const keyword = query.trim().toLowerCase();
294
+ if (!keyword)
295
+ return;
296
+ const gradeHit = grades.find((g) => {
297
+ var _a;
298
+ return (_a = g.name) == null ? void 0 : _a.toLowerCase().includes(keyword);
299
+ });
300
+ if (gradeHit) {
301
+ setActiveGradeId(gradeHit.id);
302
+ setActiveClassId(null);
303
+ return;
304
+ }
305
+ let classHit = null;
306
+ let classHitGradeId = null;
307
+ grades.some((g) => {
308
+ var _a;
309
+ const found = (_a = g.classes) == null ? void 0 : _a.find((c) => {
310
+ var _a2;
311
+ return (_a2 = c.name) == null ? void 0 : _a2.toLowerCase().includes(keyword);
312
+ });
313
+ if (found) {
314
+ classHit = found;
315
+ classHitGradeId = g.id;
316
+ return true;
317
+ }
318
+ return false;
319
+ });
320
+ if (classHit) {
321
+ setActiveGradeId(classHitGradeId);
322
+ setActiveClassId(classHit.id);
323
+ return;
324
+ }
325
+ const studentHit = students.find((s) => {
326
+ var _a;
327
+ return (_a = s.name) == null ? void 0 : _a.toLowerCase().includes(keyword);
328
+ });
329
+ if (studentHit) {
330
+ setActiveGradeId(studentHit.gradeId || null);
331
+ setActiveClassId(studentHit.classId || null);
332
+ }
333
+ }, [query, grades, students]);
334
+ if (!visible)
335
+ return null;
336
+ return /* @__PURE__ */ jsx("div", {
337
+ className: styles.csrMask,
338
+ children: /* @__PURE__ */ jsxs("div", {
339
+ className: styles.csrModal,
340
+ children: [/* @__PURE__ */ jsxs("div", {
341
+ className: styles.csrHeader,
342
+ children: [/* @__PURE__ */ jsx("h3", {
343
+ className: styles.csrTitle,
344
+ children: title
345
+ }), /* @__PURE__ */ jsx("i", {
346
+ className: `${styles.iconfont} ${styles.csrClose}`,
347
+ onClick: onClose,
348
+ children: "\uE6A9"
349
+ })]
350
+ }), /* @__PURE__ */ jsxs("div", {
351
+ className: styles.csrBody,
352
+ children: [/* @__PURE__ */ jsxs("div", {
353
+ className: styles.csrPanel,
354
+ children: [/* @__PURE__ */ jsxs("div", {
355
+ className: styles.csrSearch,
356
+ children: [/* @__PURE__ */ jsx("input", {
357
+ placeholder: "\u8BF7\u8F93\u5165\u5B66\u751F\u59D3\u540D/\u5B66\u53F7\u641C\u7D22",
358
+ value: query,
359
+ onChange: (e) => setQuery(e.target.value)
360
+ }), /* @__PURE__ */ jsx("i", {
361
+ className: `${styles.iconfont} ${styles.csrSearchIcon}`,
362
+ children: "\uE61D"
363
+ })]
364
+ }), /* @__PURE__ */ jsxs("div", {
365
+ className: styles.csrTree,
366
+ children: [loading && /* @__PURE__ */ jsx("div", {
367
+ className: styles.csrEmpty,
368
+ children: "\u52A0\u8F7D\u4E2D..."
369
+ }), error && /* @__PURE__ */ jsx("div", {
370
+ className: styles.csrEmpty,
371
+ children: error
372
+ }), !loading && !error && grades.map((g) => {
373
+ var _a;
374
+ return /* @__PURE__ */ jsxs("div", {
375
+ children: [/* @__PURE__ */ jsxs("div", {
376
+ className: `${styles.csrTreeItem} ${activeGradeId === g.id && !activeClassId ? styles.csrTreeItemActive : ""}`,
377
+ children: [/* @__PURE__ */ jsx("input", {
378
+ className: styles.csrCheck,
379
+ type: "checkbox",
380
+ name: "grade",
381
+ checked: getGradeStatus(g).checked,
382
+ ref: (el) => {
383
+ if (el)
384
+ el.indeterminate = getGradeStatus(g).indeterminate;
385
+ },
386
+ onClick: (e) => e.stopPropagation(),
387
+ onChange: (e) => toggleGrade(g, e.target.checked)
388
+ }), /* @__PURE__ */ jsx("span", {
389
+ className: styles.csrTreeItemText,
390
+ onClick: () => {
391
+ setActiveGradeId(g.id);
392
+ setActiveClassId(null);
393
+ },
394
+ children: g.name
395
+ })]
396
+ }), /* @__PURE__ */ jsx("div", {
397
+ className: styles.csrTreeChildren,
398
+ children: (_a = g.classes) == null ? void 0 : _a.map((c) => {
399
+ const info = getClassSelectedInfo(g.id, c.id);
400
+ const status = getClassStatus(g.id, c.id);
401
+ return /* @__PURE__ */ jsxs("div", {
402
+ className: `${styles.csrTreeItem} ${activeClassId === c.id ? styles.csrTreeItemActive : ""}`,
403
+ children: [/* @__PURE__ */ jsx("input", {
404
+ className: styles.csrCheck,
405
+ type: "checkbox",
406
+ name: "class",
407
+ checked: status.checked,
408
+ ref: (el) => {
409
+ if (el)
410
+ el.indeterminate = status.indeterminate;
411
+ },
412
+ onClick: (e) => e.stopPropagation(),
413
+ onChange: (e) => toggleClass(g.id, c, e.target.checked)
414
+ }), /* @__PURE__ */ jsx("span", {
415
+ className: styles.csrTreeItemText,
416
+ onClick: () => {
417
+ setActiveGradeId(g.id);
418
+ setActiveClassId(c.id);
419
+ },
420
+ children: c.name
421
+ }), /* @__PURE__ */ jsxs("span", {
422
+ className: styles.csrMeta,
423
+ children: [info.selectedCount, info.total ? `/${info.total}` : ""]
424
+ })]
425
+ }, c.id);
426
+ })
427
+ })]
428
+ }, g.id);
429
+ }), !loading && !error && !grades.length && /* @__PURE__ */ jsxs("div", {
430
+ className: styles.csrEmpty,
431
+ children: [/* @__PURE__ */ jsx("img", {
432
+ src: blankIcon,
433
+ alt: "blank"
434
+ }), /* @__PURE__ */ jsx("span", {
435
+ children: "\u6682\u65E0\u6570\u636E"
436
+ })]
437
+ })]
438
+ })]
439
+ }), /* @__PURE__ */ jsxs("div", {
440
+ className: styles.csrPanel,
441
+ children: [/* @__PURE__ */ jsxs("div", {
442
+ className: styles.csrOps,
443
+ children: [/* @__PURE__ */ jsxs("label", {
444
+ className: styles.csrCheckLabel,
445
+ children: [/* @__PURE__ */ jsx("input", {
446
+ className: styles.csrCheck,
447
+ type: "checkbox",
448
+ checked: activeStudents.length > 0 && activeStudents.every((s) => selectedIds.has(s.id)),
449
+ onChange: (e) => toggleAll(e.target.checked)
450
+ }), "\u5168\u9009"]
451
+ }), /* @__PURE__ */ jsxs("span", {
452
+ className: styles.csrMeta,
453
+ children: ["\u5171 ", activeStudents.length, " \u4EBA"]
454
+ })]
455
+ }), /* @__PURE__ */ jsxs("ul", {
456
+ className: styles.csrList,
457
+ children: [activeStudents.map((s) => /* @__PURE__ */ jsxs("li", {
458
+ className: styles.csrListItem,
459
+ children: [/* @__PURE__ */ jsx("input", {
460
+ className: styles.csrCheck,
461
+ type: "checkbox",
462
+ checked: selectedIds.has(s.id),
463
+ onChange: () => toggleOne(s.id)
464
+ }), /* @__PURE__ */ jsx("div", {
465
+ children: s.name
466
+ })]
467
+ }, s.id)), !activeStudents.length && /* @__PURE__ */ jsxs("div", {
468
+ className: styles.csrEmpty,
469
+ children: [/* @__PURE__ */ jsx("img", {
470
+ src: blankIcon,
471
+ alt: "blank"
472
+ }), /* @__PURE__ */ jsx("span", {
473
+ children: "\u6682\u65E0\u5B66\u751F"
474
+ })]
475
+ })]
476
+ })]
477
+ }), /* @__PURE__ */ jsxs("div", {
478
+ className: styles.csrPanel,
479
+ children: [/* @__PURE__ */ jsxs("div", {
480
+ className: styles.csrOps,
481
+ children: [/* @__PURE__ */ jsxs("span", {
482
+ className: styles.hasSelected,
483
+ children: ["\u5DF2\u9009 ", selectedList.length, " \u4EBA"]
484
+ }), /* @__PURE__ */ jsx("span", {
485
+ className: styles.clearBtn,
486
+ onClick: clearAll,
487
+ children: "\u6E05\u7A7A"
488
+ })]
489
+ }), /* @__PURE__ */ jsx("div", {
490
+ className: styles.csrChipWrap,
491
+ children: selectedList.length ? /* @__PURE__ */ jsx("ul", {
492
+ className: styles.csrChipList,
493
+ children: selectedList.map((s) => /* @__PURE__ */ jsxs("li", {
494
+ className: styles.csrChip,
495
+ children: [/* @__PURE__ */ jsx("span", {
496
+ children: s.name
497
+ }), /* @__PURE__ */ jsx("i", {
498
+ className: `${styles.iconfont} ${styles.deleteIcon}`,
499
+ onClick: () => toggleOne(s.id),
500
+ children: "\uE6A9"
501
+ })]
502
+ }, s.id))
503
+ }) : /* @__PURE__ */ jsx("div", {
504
+ className: styles.csrEmpty,
505
+ children: "\u8BF7\u9009\u62E9"
506
+ })
507
+ })]
508
+ })]
509
+ }), /* @__PURE__ */ jsxs("div", {
510
+ className: styles.csrFooter,
511
+ children: [/* @__PURE__ */ jsx("button", {
512
+ className: `${styles.csrBtn} ${styles.csrBtnCancel}`,
513
+ onClick: onClose,
514
+ children: "\u53D6\u6D88"
515
+ }), /* @__PURE__ */ jsx("button", {
516
+ className: `${styles.csrBtn} ${styles.csrBtnPrimary}`,
517
+ onClick: handleSave,
518
+ children: "\u4FDD\u5B58"
519
+ })]
520
+ })]
521
+ })
522
+ });
523
+ }
524
+ export { ClassStudentRoster as default };
package/dist/style.css ADDED
@@ -0,0 +1 @@
1
+ @font-face{font-family:iconfont;src:url(//at.alicdn.com/t/c/font_789461_6fumqpsw41m.woff2?t=1766568844647) format("woff2"),url(//at.alicdn.com/t/c/font_789461_6fumqpsw41m.woff?t=1766568844647) format("woff"),url(//at.alicdn.com/t/c/font_789461_6fumqpsw41m.ttf?t=1766568844647) format("truetype")}._iconfont_zvw5p_7{font-family:iconfont!important;font-size:16px;font-style:normal;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}._csrMask_zvw5p_14{position:fixed;inset:0;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:1000;font-family:PingFangSC-Regular}._csrModal_zvw5p_24{width:980px;background:#fff;border-radius:11px;box-shadow:0 12px 40px #00000029;display:flex;flex-direction:column;overflow:hidden;font-family:PingFangSC-Regular}._csrHeader_zvw5p_34{height:56px;min-height:56px;padding:0 12px;border-bottom:1px solid #E6E7EC;display:flex;align-items:center;justify-content:center;position:relative}._csrTitle_zvw5p_44{margin:0;font-size:16px;font-weight:600;color:#01113d}._csrClose_zvw5p_50{position:absolute;right:16px;top:50%;border:none;background:transparent;font-size:18px;cursor:pointer;color:#27355a;font-weight:600;margin-top:-9px}._csrBody_zvw5p_62{flex:1;overflow:hidden;display:flex;background:#f5f6f7;padding:12px}._csrPanel_zvw5p_69{background:#fff;border-radius:11px;display:flex;flex-direction:column;min-height:0;flex:1;height:450px}._csrPanel_zvw5p_69:nth-child(2){margin:0 10px}._csrPanelHeader_zvw5p_81{padding:12px 12px 8px;font-weight:600;color:#333}._csrSearch_zvw5p_86{padding:10px 12px;display:flex;align-items:center;position:relative;border-bottom:1px solid #E6E7EC}._csrSearch_zvw5p_86 input{padding:9px 30px 9px 10px;border-radius:7px;border:1px solid #E6E7EC;font-size:14px;width:100%;z-index:1;position:relative}._csrSearch_zvw5p_86 ._csrSearchIcon_zvw5p_102{position:absolute;right:24px;top:50%;transform:translateY(-50%);font-size:14px;color:#5a6481;font-weight:600;z-index:2;cursor:pointer}._csrTree_zvw5p_113{flex:1;overflow:auto;padding:10px 12px}._csrTreeItem_zvw5p_118{display:flex;align-items:center;padding:6px 8px 6px 2px;border-radius:4px;cursor:pointer;color:#01113d;font-size:14px;font-family:PingFangSC-Regular;position:relative;margin-bottom:8px;transition:background .18s ease,color .18s ease,transform .18s ease,box-shadow .18s ease}._csrTreeItem_zvw5p_118:hover{background:rgba(4,69,252,.06);transform:translate(2px)}._csrTreeItem_zvw5p_118 ._csrTreeItemText_zvw5p_135{padding-right:32px}._csrTreeItem_zvw5p_118 ._csrMeta_zvw5p_138{position:absolute;right:11px}._csrTreeItemActive_zvw5p_142{background:rgba(4,69,252,.05);border-radius:2px;transform:translate(4px)}._csrTreeItemActive_zvw5p_142 span{color:#0445fc;font-weight:500}._csrTreeChildren_zvw5p_151{padding-left:22px}._csrList_zvw5p_154{list-style:none;margin:0;padding:10px 12px 12px;flex:1;overflow:auto}._csrList_zvw5p_154 ._csrListItem_zvw5p_161{display:flex;align-items:center;font-size:14px;font-family:PingFangSC-Regular;padding:8px 4px;margin-bottom:4px;border-radius:4px;transition:background .18s ease,transform .18s ease,box-shadow .18s ease}._csrList_zvw5p_154 ._csrListItem_zvw5p_161:hover{background:rgba(4,69,252,.05);transform:translate(2px)}._csrList_zvw5p_154 ._csrListItem_zvw5p_161:last-child{border-bottom:none}._csrMeta_zvw5p_138{display:flex;align-items:center;gap:6px;font-size:12px;color:#666}._csrOps_zvw5p_185{display:flex;align-items:center;justify-content:space-between;padding:0 12px;height:56px;font-size:14px;color:#01113d;font-family:PingFangSC-Regular;border-bottom:1px solid #E6E7EC}._csrOps_zvw5p_185 ._hasSelected_zvw5p_196{font-size:14px;font-weight:500;color:#01113d}._csrOps_zvw5p_185 ._clearBtn_zvw5p_201{color:#0445fc;cursor:pointer}._csrChipWrap_zvw5p_205{flex:1;overflow:auto;padding:10px 12px 12px}._csrChipList_zvw5p_210{display:flex;flex-wrap:wrap;gap:8px;margin:0;padding:0;list-style:none}._csrChipList_zvw5p_210 ._deleteIcon_zvw5p_218{margin-left:8px;cursor:pointer;font-size:14px}._csrChip_zvw5p_205{display:inline-flex;align-items:center;background:#F2F3F5;border-radius:4px;padding:4px 10px;font-size:12px;color:#01113dd9}._csrChip_zvw5p_205 button{border:none;background:transparent;cursor:pointer;color:#666}._csrFooter_zvw5p_238{border-top:1px solid #f0f0f0;padding:12px 16px;display:flex;justify-content:flex-end;gap:12px}._csrBtn_zvw5p_245{min-width:76px;height:32px;border-radius:7px;cursor:pointer;border:1px solid #d9d9d9;background:#fff;font-size:14px}._csrBtnPrimary_zvw5p_254{background:#0445FC;border-color:#0445fc;color:#fff}._csrBtnCancel_zvw5p_259{background:#EDEEF1;border-color:#edeef1;color:#01113d}._csrEmpty_zvw5p_264{color:#01113da6;padding:8px 12px;font-size:13px;display:flex;align-items:center;justify-content:center;flex-direction:column;height:80%}._csrEmpty_zvw5p_264 img{width:120px;height:120px;margin-bottom:12px}._csrCheck_zvw5p_279{margin-right:8px;width:14px;height:14px;accent-color:#0445FC;transition:transform .12s ease}._csrCheck_zvw5p_279:active{transform:scale(.9)}._csrCheckLabel_zvw5p_289{display:flex;align-items:center}
package/package.json ADDED
@@ -0,0 +1,30 @@
1
+ {
2
+ "name": "@yungu-fed/class-student-roster",
3
+ "version": "0.0.1",
4
+ "description": "student list under a class or a grade",
5
+ "main": "./dist/index.cjs.js",
6
+ "module": "./dist/index.es.js",
7
+ "sideEffects": false,
8
+ "files": [
9
+ "dist"
10
+ ],
11
+ "scripts": {
12
+ "dev": "vite",
13
+ "build": "vite build",
14
+ "preview": "vite preview",
15
+ "test": "echo \"Error: no test specified\" && exit 1"
16
+ },
17
+ "author": "",
18
+ "license": "ISC",
19
+ "peerDependencies": {
20
+ "react": "^16.8.0 || ^17 || ^18 || ^19",
21
+ "react-dom": "^16.8.0 || ^17 || ^18 || ^19"
22
+ },
23
+ "devDependencies": {
24
+ "@vitejs/plugin-react": "^1.3.2",
25
+ "less": "^4.2.0",
26
+ "react": "^18.2.0",
27
+ "react-dom": "^18.2.0",
28
+ "vite": "^2.9.16"
29
+ }
30
+ }