balibabu commited on
Commit
7f9c7e1
·
1 Parent(s): b371a08

Feat: Add MultiSelect #3221 (#4090)

Browse files

### What problem does this PR solve?

Feat: Add MultiSelect #3221

### Type of change


- [x] New Feature (non-breaking change which adds functionality)

web/package-lock.json CHANGED
@@ -16,6 +16,7 @@
16
  "@radix-ui/react-aspect-ratio": "^1.1.0",
17
  "@radix-ui/react-avatar": "^1.1.1",
18
  "@radix-ui/react-checkbox": "^1.1.2",
 
19
  "@radix-ui/react-dropdown-menu": "^2.1.2",
20
  "@radix-ui/react-icons": "^1.3.1",
21
  "@radix-ui/react-label": "^2.1.0",
@@ -40,6 +41,7 @@
40
  "class-variance-authority": "^0.7.0",
41
  "classnames": "^2.5.1",
42
  "clsx": "^2.1.1",
 
43
  "dayjs": "^1.11.10",
44
  "dompurify": "^3.1.6",
45
  "eventsource-parser": "^1.1.2",
@@ -4240,6 +4242,219 @@
4240
  }
4241
  }
4242
  },
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
4243
  "node_modules/@radix-ui/react-direction": {
4244
  "version": "1.1.0",
4245
  "resolved": "https://registry.npmmirror.com/@radix-ui/react-direction/-/react-direction-1.1.0.tgz",
@@ -10672,6 +10887,29 @@
10672
  "node": ">=6"
10673
  }
10674
  },
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
10675
  "node_modules/co": {
10676
  "version": "4.6.0",
10677
  "resolved": "https://registry.npmmirror.com/co/-/co-4.6.0.tgz",
@@ -25069,19 +25307,19 @@
25069
  }
25070
  },
25071
  "node_modules/react-remove-scroll-bar": {
25072
- "version": "2.3.6",
25073
- "resolved": "https://registry.npmmirror.com/react-remove-scroll-bar/-/react-remove-scroll-bar-2.3.6.tgz",
25074
- "integrity": "sha512-DtSYaao4mBmX+HDo5YWYdBWQwYIQQshUV/dVxFxK+KM26Wjwp1gZ6rv6OC3oujI6Bfu6Xyg3TwK533AQutsn/g==",
25075
  "dependencies": {
25076
- "react-style-singleton": "^2.2.1",
25077
  "tslib": "^2.0.0"
25078
  },
25079
  "engines": {
25080
  "node": ">=10"
25081
  },
25082
  "peerDependencies": {
25083
- "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0",
25084
- "react": "^16.8.0 || ^17.0.0 || ^18.0.0"
25085
  },
25086
  "peerDependenciesMeta": {
25087
  "@types/react": {
@@ -25229,20 +25467,19 @@
25229
  }
25230
  },
25231
  "node_modules/react-style-singleton": {
25232
- "version": "2.2.1",
25233
- "resolved": "https://registry.npmmirror.com/react-style-singleton/-/react-style-singleton-2.2.1.tgz",
25234
- "integrity": "sha512-ZWj0fHEMyWkHzKYUr2Bs/4zU6XLmq9HsgBURm7g5pAVfyn49DgUiNgY2d4lXRlYSiCif9YBGpQleewkcqddc7g==",
25235
  "dependencies": {
25236
  "get-nonce": "^1.0.0",
25237
- "invariant": "^2.2.4",
25238
  "tslib": "^2.0.0"
25239
  },
25240
  "engines": {
25241
  "node": ">=10"
25242
  },
25243
  "peerDependencies": {
25244
- "@types/react": "^16.8.0 || ^17.0.0 || ^18.0.0",
25245
- "react": "^16.8.0 || ^17.0.0 || ^18.0.0"
25246
  },
25247
  "peerDependenciesMeta": {
25248
  "@types/react": {
 
16
  "@radix-ui/react-aspect-ratio": "^1.1.0",
17
  "@radix-ui/react-avatar": "^1.1.1",
18
  "@radix-ui/react-checkbox": "^1.1.2",
19
+ "@radix-ui/react-dialog": "^1.1.4",
20
  "@radix-ui/react-dropdown-menu": "^2.1.2",
21
  "@radix-ui/react-icons": "^1.3.1",
22
  "@radix-ui/react-label": "^2.1.0",
 
41
  "class-variance-authority": "^0.7.0",
42
  "classnames": "^2.5.1",
43
  "clsx": "^2.1.1",
44
+ "cmdk": "^1.0.4",
45
  "dayjs": "^1.11.10",
46
  "dompurify": "^3.1.6",
47
  "eventsource-parser": "^1.1.2",
 
4242
  }
4243
  }
4244
  },
4245
+ "node_modules/@radix-ui/react-dialog": {
4246
+ "version": "1.1.4",
4247
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-dialog/-/react-dialog-1.1.4.tgz",
4248
+ "integrity": "sha512-Ur7EV1IwQGCyaAuyDRiOLA5JIUZxELJljF+MbM/2NC0BYwfuRrbpS30BiQBJrVruscgUkieKkqXYDOoByaxIoA==",
4249
+ "dependencies": {
4250
+ "@radix-ui/primitive": "1.1.1",
4251
+ "@radix-ui/react-compose-refs": "1.1.1",
4252
+ "@radix-ui/react-context": "1.1.1",
4253
+ "@radix-ui/react-dismissable-layer": "1.1.3",
4254
+ "@radix-ui/react-focus-guards": "1.1.1",
4255
+ "@radix-ui/react-focus-scope": "1.1.1",
4256
+ "@radix-ui/react-id": "1.1.0",
4257
+ "@radix-ui/react-portal": "1.1.3",
4258
+ "@radix-ui/react-presence": "1.1.2",
4259
+ "@radix-ui/react-primitive": "2.0.1",
4260
+ "@radix-ui/react-slot": "1.1.1",
4261
+ "@radix-ui/react-use-controllable-state": "1.1.0",
4262
+ "aria-hidden": "^1.1.1",
4263
+ "react-remove-scroll": "^2.6.1"
4264
+ },
4265
+ "peerDependencies": {
4266
+ "@types/react": "*",
4267
+ "@types/react-dom": "*",
4268
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
4269
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4270
+ },
4271
+ "peerDependenciesMeta": {
4272
+ "@types/react": {
4273
+ "optional": true
4274
+ },
4275
+ "@types/react-dom": {
4276
+ "optional": true
4277
+ }
4278
+ }
4279
+ },
4280
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/primitive": {
4281
+ "version": "1.1.1",
4282
+ "resolved": "https://registry.npmmirror.com/@radix-ui/primitive/-/primitive-1.1.1.tgz",
4283
+ "integrity": "sha512-SJ31y+Q/zAyShtXJc8x83i9TYdbAfHZ++tUZnvjJJqFjzsdUnKsxPL6IEtBlxKkU7yzer//GQtZSV4GbldL3YA=="
4284
+ },
4285
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/react-compose-refs": {
4286
+ "version": "1.1.1",
4287
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-compose-refs/-/react-compose-refs-1.1.1.tgz",
4288
+ "integrity": "sha512-Y9VzoRDSJtgFMUCoiZBDVo084VQ5hfpXxVE+NgkdNsjiDBByiImMZKKhxMwCbdHvhlENG6a833CbFkOQvTricw==",
4289
+ "peerDependencies": {
4290
+ "@types/react": "*",
4291
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4292
+ },
4293
+ "peerDependenciesMeta": {
4294
+ "@types/react": {
4295
+ "optional": true
4296
+ }
4297
+ }
4298
+ },
4299
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/react-dismissable-layer": {
4300
+ "version": "1.1.3",
4301
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-dismissable-layer/-/react-dismissable-layer-1.1.3.tgz",
4302
+ "integrity": "sha512-onrWn/72lQoEucDmJnr8uczSNTujT0vJnA/X5+3AkChVPowr8n1yvIKIabhWyMQeMvvmdpsvcyDqx3X1LEXCPg==",
4303
+ "dependencies": {
4304
+ "@radix-ui/primitive": "1.1.1",
4305
+ "@radix-ui/react-compose-refs": "1.1.1",
4306
+ "@radix-ui/react-primitive": "2.0.1",
4307
+ "@radix-ui/react-use-callback-ref": "1.1.0",
4308
+ "@radix-ui/react-use-escape-keydown": "1.1.0"
4309
+ },
4310
+ "peerDependencies": {
4311
+ "@types/react": "*",
4312
+ "@types/react-dom": "*",
4313
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
4314
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4315
+ },
4316
+ "peerDependenciesMeta": {
4317
+ "@types/react": {
4318
+ "optional": true
4319
+ },
4320
+ "@types/react-dom": {
4321
+ "optional": true
4322
+ }
4323
+ }
4324
+ },
4325
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/react-focus-scope": {
4326
+ "version": "1.1.1",
4327
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-focus-scope/-/react-focus-scope-1.1.1.tgz",
4328
+ "integrity": "sha512-01omzJAYRxXdG2/he/+xy+c8a8gCydoQ1yOxnWNcRhrrBW5W+RQJ22EK1SaO8tb3WoUsuEw7mJjBozPzihDFjA==",
4329
+ "dependencies": {
4330
+ "@radix-ui/react-compose-refs": "1.1.1",
4331
+ "@radix-ui/react-primitive": "2.0.1",
4332
+ "@radix-ui/react-use-callback-ref": "1.1.0"
4333
+ },
4334
+ "peerDependencies": {
4335
+ "@types/react": "*",
4336
+ "@types/react-dom": "*",
4337
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
4338
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4339
+ },
4340
+ "peerDependenciesMeta": {
4341
+ "@types/react": {
4342
+ "optional": true
4343
+ },
4344
+ "@types/react-dom": {
4345
+ "optional": true
4346
+ }
4347
+ }
4348
+ },
4349
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/react-portal": {
4350
+ "version": "1.1.3",
4351
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-portal/-/react-portal-1.1.3.tgz",
4352
+ "integrity": "sha512-NciRqhXnGojhT93RPyDaMPfLH3ZSl4jjIFbZQ1b/vxvZEdHsBZ49wP9w8L3HzUQwep01LcWtkUvm0OVB5JAHTw==",
4353
+ "dependencies": {
4354
+ "@radix-ui/react-primitive": "2.0.1",
4355
+ "@radix-ui/react-use-layout-effect": "1.1.0"
4356
+ },
4357
+ "peerDependencies": {
4358
+ "@types/react": "*",
4359
+ "@types/react-dom": "*",
4360
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
4361
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4362
+ },
4363
+ "peerDependenciesMeta": {
4364
+ "@types/react": {
4365
+ "optional": true
4366
+ },
4367
+ "@types/react-dom": {
4368
+ "optional": true
4369
+ }
4370
+ }
4371
+ },
4372
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/react-presence": {
4373
+ "version": "1.1.2",
4374
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-presence/-/react-presence-1.1.2.tgz",
4375
+ "integrity": "sha512-18TFr80t5EVgL9x1SwF/YGtfG+l0BS0PRAlCWBDoBEiDQjeKgnNZRVJp/oVBl24sr3Gbfwc/Qpj4OcWTQMsAEg==",
4376
+ "dependencies": {
4377
+ "@radix-ui/react-compose-refs": "1.1.1",
4378
+ "@radix-ui/react-use-layout-effect": "1.1.0"
4379
+ },
4380
+ "peerDependencies": {
4381
+ "@types/react": "*",
4382
+ "@types/react-dom": "*",
4383
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
4384
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4385
+ },
4386
+ "peerDependenciesMeta": {
4387
+ "@types/react": {
4388
+ "optional": true
4389
+ },
4390
+ "@types/react-dom": {
4391
+ "optional": true
4392
+ }
4393
+ }
4394
+ },
4395
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/react-primitive": {
4396
+ "version": "2.0.1",
4397
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-primitive/-/react-primitive-2.0.1.tgz",
4398
+ "integrity": "sha512-sHCWTtxwNn3L3fH8qAfnF3WbUZycW93SM1j3NFDzXBiz8D6F5UTTy8G1+WFEaiCdvCVRJWj6N2R4Xq6HdiHmDg==",
4399
+ "dependencies": {
4400
+ "@radix-ui/react-slot": "1.1.1"
4401
+ },
4402
+ "peerDependencies": {
4403
+ "@types/react": "*",
4404
+ "@types/react-dom": "*",
4405
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc",
4406
+ "react-dom": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4407
+ },
4408
+ "peerDependenciesMeta": {
4409
+ "@types/react": {
4410
+ "optional": true
4411
+ },
4412
+ "@types/react-dom": {
4413
+ "optional": true
4414
+ }
4415
+ }
4416
+ },
4417
+ "node_modules/@radix-ui/react-dialog/node_modules/@radix-ui/react-slot": {
4418
+ "version": "1.1.1",
4419
+ "resolved": "https://registry.npmmirror.com/@radix-ui/react-slot/-/react-slot-1.1.1.tgz",
4420
+ "integrity": "sha512-RApLLOcINYJA+dMVbOju7MYv1Mb2EBp2nH4HdDzXTSyaR5optlm6Otrz1euW3HbdOR8UmmFK06TD+A9frYWv+g==",
4421
+ "dependencies": {
4422
+ "@radix-ui/react-compose-refs": "1.1.1"
4423
+ },
4424
+ "peerDependencies": {
4425
+ "@types/react": "*",
4426
+ "react": "^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc"
4427
+ },
4428
+ "peerDependenciesMeta": {
4429
+ "@types/react": {
4430
+ "optional": true
4431
+ }
4432
+ }
4433
+ },
4434
+ "node_modules/@radix-ui/react-dialog/node_modules/react-remove-scroll": {
4435
+ "version": "2.6.1",
4436
+ "resolved": "https://registry.npmmirror.com/react-remove-scroll/-/react-remove-scroll-2.6.1.tgz",
4437
+ "integrity": "sha512-jWEvWQidZ/C/FnFlUIB1mDLpY3r7uEb22WZ3uVeKj520caKDiaBsNDEB9J1gHJgpiLo+eTdPl2MVi0JitFTiFg==",
4438
+ "dependencies": {
4439
+ "react-remove-scroll-bar": "^2.3.7",
4440
+ "react-style-singleton": "^2.2.1",
4441
+ "tslib": "^2.1.0",
4442
+ "use-callback-ref": "^1.3.0",
4443
+ "use-sidecar": "^1.1.2"
4444
+ },
4445
+ "engines": {
4446
+ "node": ">=10"
4447
+ },
4448
+ "peerDependencies": {
4449
+ "@types/react": "*",
4450
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc"
4451
+ },
4452
+ "peerDependenciesMeta": {
4453
+ "@types/react": {
4454
+ "optional": true
4455
+ }
4456
+ }
4457
+ },
4458
  "node_modules/@radix-ui/react-direction": {
4459
  "version": "1.1.0",
4460
  "resolved": "https://registry.npmmirror.com/@radix-ui/react-direction/-/react-direction-1.1.0.tgz",
 
10887
  "node": ">=6"
10888
  }
10889
  },
10890
+ "node_modules/cmdk": {
10891
+ "version": "1.0.4",
10892
+ "resolved": "https://registry.npmmirror.com/cmdk/-/cmdk-1.0.4.tgz",
10893
+ "integrity": "sha512-AnsjfHyHpQ/EFeAnG216WY7A5LiYCoZzCSygiLvfXC3H3LFGCprErteUcszaVluGOhuOTbJS3jWHrSDYPBBygg==",
10894
+ "dependencies": {
10895
+ "@radix-ui/react-dialog": "^1.1.2",
10896
+ "@radix-ui/react-id": "^1.1.0",
10897
+ "@radix-ui/react-primitive": "^2.0.0",
10898
+ "use-sync-external-store": "^1.2.2"
10899
+ },
10900
+ "peerDependencies": {
10901
+ "react": "^18 || ^19 || ^19.0.0-rc",
10902
+ "react-dom": "^18 || ^19 || ^19.0.0-rc"
10903
+ }
10904
+ },
10905
+ "node_modules/cmdk/node_modules/use-sync-external-store": {
10906
+ "version": "1.4.0",
10907
+ "resolved": "https://registry.npmmirror.com/use-sync-external-store/-/use-sync-external-store-1.4.0.tgz",
10908
+ "integrity": "sha512-9WXSPC5fMv61vaupRkCKCxsPxBocVnwakBEkMIHHpkTTg6icbJtg6jzgtLDm4bl3cSHAca52rYWih0k4K3PfHw==",
10909
+ "peerDependencies": {
10910
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
10911
+ }
10912
+ },
10913
  "node_modules/co": {
10914
  "version": "4.6.0",
10915
  "resolved": "https://registry.npmmirror.com/co/-/co-4.6.0.tgz",
 
25307
  }
25308
  },
25309
  "node_modules/react-remove-scroll-bar": {
25310
+ "version": "2.3.8",
25311
+ "resolved": "https://registry.npmmirror.com/react-remove-scroll-bar/-/react-remove-scroll-bar-2.3.8.tgz",
25312
+ "integrity": "sha512-9r+yi9+mgU33AKcj6IbT9oRCO78WriSj6t/cF8DWBZJ9aOGPOTEDvdUDz1FwKim7QXWwmHqtdHnRJfhAxEG46Q==",
25313
  "dependencies": {
25314
+ "react-style-singleton": "^2.2.2",
25315
  "tslib": "^2.0.0"
25316
  },
25317
  "engines": {
25318
  "node": ">=10"
25319
  },
25320
  "peerDependencies": {
25321
+ "@types/react": "*",
25322
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
25323
  },
25324
  "peerDependenciesMeta": {
25325
  "@types/react": {
 
25467
  }
25468
  },
25469
  "node_modules/react-style-singleton": {
25470
+ "version": "2.2.3",
25471
+ "resolved": "https://registry.npmmirror.com/react-style-singleton/-/react-style-singleton-2.2.3.tgz",
25472
+ "integrity": "sha512-b6jSvxvVnyptAiLjbkWLE/lOnR4lfTtDAl+eUC7RZy+QQWc6wRzIV2CE6xBuMmDxc2qIihtDCZD5NPOFl7fRBQ==",
25473
  "dependencies": {
25474
  "get-nonce": "^1.0.0",
 
25475
  "tslib": "^2.0.0"
25476
  },
25477
  "engines": {
25478
  "node": ">=10"
25479
  },
25480
  "peerDependencies": {
25481
+ "@types/react": "*",
25482
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc"
25483
  },
25484
  "peerDependenciesMeta": {
25485
  "@types/react": {
web/package.json CHANGED
@@ -27,6 +27,7 @@
27
  "@radix-ui/react-aspect-ratio": "^1.1.0",
28
  "@radix-ui/react-avatar": "^1.1.1",
29
  "@radix-ui/react-checkbox": "^1.1.2",
 
30
  "@radix-ui/react-dropdown-menu": "^2.1.2",
31
  "@radix-ui/react-icons": "^1.3.1",
32
  "@radix-ui/react-label": "^2.1.0",
@@ -51,6 +52,7 @@
51
  "class-variance-authority": "^0.7.0",
52
  "classnames": "^2.5.1",
53
  "clsx": "^2.1.1",
 
54
  "dayjs": "^1.11.10",
55
  "dompurify": "^3.1.6",
56
  "eventsource-parser": "^1.1.2",
 
27
  "@radix-ui/react-aspect-ratio": "^1.1.0",
28
  "@radix-ui/react-avatar": "^1.1.1",
29
  "@radix-ui/react-checkbox": "^1.1.2",
30
+ "@radix-ui/react-dialog": "^1.1.4",
31
  "@radix-ui/react-dropdown-menu": "^2.1.2",
32
  "@radix-ui/react-icons": "^1.3.1",
33
  "@radix-ui/react-label": "^2.1.0",
 
52
  "class-variance-authority": "^0.7.0",
53
  "classnames": "^2.5.1",
54
  "clsx": "^2.1.1",
55
+ "cmdk": "^1.0.4",
56
  "dayjs": "^1.11.10",
57
  "dompurify": "^3.1.6",
58
  "eventsource-parser": "^1.1.2",
web/src/components/ui/command.tsx ADDED
@@ -0,0 +1,153 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ 'use client';
2
+
3
+ import { type DialogProps } from '@radix-ui/react-dialog';
4
+ import { Command as CommandPrimitive } from 'cmdk';
5
+ import { Search } from 'lucide-react';
6
+ import * as React from 'react';
7
+
8
+ import { Dialog, DialogContent } from '@/components/ui/dialog';
9
+ import { cn } from '@/lib/utils';
10
+
11
+ const Command = React.forwardRef<
12
+ React.ElementRef<typeof CommandPrimitive>,
13
+ React.ComponentPropsWithoutRef<typeof CommandPrimitive>
14
+ >(({ className, ...props }, ref) => (
15
+ <CommandPrimitive
16
+ ref={ref}
17
+ className={cn(
18
+ 'flex h-full w-full flex-col overflow-hidden rounded-md bg-popover text-popover-foreground',
19
+ className,
20
+ )}
21
+ {...props}
22
+ />
23
+ ));
24
+ Command.displayName = CommandPrimitive.displayName;
25
+
26
+ const CommandDialog = ({ children, ...props }: DialogProps) => {
27
+ return (
28
+ <Dialog {...props}>
29
+ <DialogContent className="overflow-hidden p-0 shadow-lg">
30
+ <Command className="[&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group-heading]]:text-muted-foreground [&_[cmdk-group]:not([hidden])_~[cmdk-group]]:pt-0 [&_[cmdk-group]]:px-2 [&_[cmdk-input-wrapper]_svg]:h-5 [&_[cmdk-input-wrapper]_svg]:w-5 [&_[cmdk-input]]:h-12 [&_[cmdk-item]]:px-2 [&_[cmdk-item]]:py-3 [&_[cmdk-item]_svg]:h-5 [&_[cmdk-item]_svg]:w-5">
31
+ {children}
32
+ </Command>
33
+ </DialogContent>
34
+ </Dialog>
35
+ );
36
+ };
37
+
38
+ const CommandInput = React.forwardRef<
39
+ React.ElementRef<typeof CommandPrimitive.Input>,
40
+ React.ComponentPropsWithoutRef<typeof CommandPrimitive.Input>
41
+ >(({ className, ...props }, ref) => (
42
+ <div className="flex items-center border-b px-3" cmdk-input-wrapper="">
43
+ <Search className="mr-2 h-4 w-4 shrink-0 opacity-50" />
44
+ <CommandPrimitive.Input
45
+ ref={ref}
46
+ className={cn(
47
+ 'flex h-11 w-full rounded-md bg-transparent py-3 text-sm outline-none placeholder:text-muted-foreground disabled:cursor-not-allowed disabled:opacity-50',
48
+ className,
49
+ )}
50
+ {...props}
51
+ />
52
+ </div>
53
+ ));
54
+
55
+ CommandInput.displayName = CommandPrimitive.Input.displayName;
56
+
57
+ const CommandList = React.forwardRef<
58
+ React.ElementRef<typeof CommandPrimitive.List>,
59
+ React.ComponentPropsWithoutRef<typeof CommandPrimitive.List>
60
+ >(({ className, ...props }, ref) => (
61
+ <CommandPrimitive.List
62
+ ref={ref}
63
+ className={cn('max-h-[300px] overflow-y-auto overflow-x-hidden', className)}
64
+ {...props}
65
+ />
66
+ ));
67
+
68
+ CommandList.displayName = CommandPrimitive.List.displayName;
69
+
70
+ const CommandEmpty = React.forwardRef<
71
+ React.ElementRef<typeof CommandPrimitive.Empty>,
72
+ React.ComponentPropsWithoutRef<typeof CommandPrimitive.Empty>
73
+ >((props, ref) => (
74
+ <CommandPrimitive.Empty
75
+ ref={ref}
76
+ className="py-6 text-center text-sm"
77
+ {...props}
78
+ />
79
+ ));
80
+
81
+ CommandEmpty.displayName = CommandPrimitive.Empty.displayName;
82
+
83
+ const CommandGroup = React.forwardRef<
84
+ React.ElementRef<typeof CommandPrimitive.Group>,
85
+ React.ComponentPropsWithoutRef<typeof CommandPrimitive.Group>
86
+ >(({ className, ...props }, ref) => (
87
+ <CommandPrimitive.Group
88
+ ref={ref}
89
+ className={cn(
90
+ 'overflow-hidden p-1 text-foreground [&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:py-1.5 [&_[cmdk-group-heading]]:text-xs [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group-heading]]:text-muted-foreground',
91
+ className,
92
+ )}
93
+ {...props}
94
+ />
95
+ ));
96
+
97
+ CommandGroup.displayName = CommandPrimitive.Group.displayName;
98
+
99
+ const CommandSeparator = React.forwardRef<
100
+ React.ElementRef<typeof CommandPrimitive.Separator>,
101
+ React.ComponentPropsWithoutRef<typeof CommandPrimitive.Separator>
102
+ >(({ className, ...props }, ref) => (
103
+ <CommandPrimitive.Separator
104
+ ref={ref}
105
+ className={cn('-mx-1 h-px bg-border', className)}
106
+ {...props}
107
+ />
108
+ ));
109
+ CommandSeparator.displayName = CommandPrimitive.Separator.displayName;
110
+
111
+ const CommandItem = React.forwardRef<
112
+ React.ElementRef<typeof CommandPrimitive.Item>,
113
+ React.ComponentPropsWithoutRef<typeof CommandPrimitive.Item>
114
+ >(({ className, ...props }, ref) => (
115
+ <CommandPrimitive.Item
116
+ ref={ref}
117
+ className={cn(
118
+ "relative flex cursor-default gap-2 select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none data-[disabled=true]:pointer-events-none data-[selected='true']:bg-accent data-[selected=true]:text-accent-foreground data-[disabled=true]:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0",
119
+ className,
120
+ )}
121
+ {...props}
122
+ />
123
+ ));
124
+
125
+ CommandItem.displayName = CommandPrimitive.Item.displayName;
126
+
127
+ const CommandShortcut = ({
128
+ className,
129
+ ...props
130
+ }: React.HTMLAttributes<HTMLSpanElement>) => {
131
+ return (
132
+ <span
133
+ className={cn(
134
+ 'ml-auto text-xs tracking-widest text-muted-foreground',
135
+ className,
136
+ )}
137
+ {...props}
138
+ />
139
+ );
140
+ };
141
+ CommandShortcut.displayName = 'CommandShortcut';
142
+
143
+ export {
144
+ Command,
145
+ CommandDialog,
146
+ CommandEmpty,
147
+ CommandGroup,
148
+ CommandInput,
149
+ CommandItem,
150
+ CommandList,
151
+ CommandSeparator,
152
+ CommandShortcut,
153
+ };
web/src/components/ui/dialog.tsx ADDED
@@ -0,0 +1,122 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ 'use client';
2
+
3
+ import * as DialogPrimitive from '@radix-ui/react-dialog';
4
+ import { X } from 'lucide-react';
5
+ import * as React from 'react';
6
+
7
+ import { cn } from '@/lib/utils';
8
+
9
+ const Dialog = DialogPrimitive.Root;
10
+
11
+ const DialogTrigger = DialogPrimitive.Trigger;
12
+
13
+ const DialogPortal = DialogPrimitive.Portal;
14
+
15
+ const DialogClose = DialogPrimitive.Close;
16
+
17
+ const DialogOverlay = React.forwardRef<
18
+ React.ElementRef<typeof DialogPrimitive.Overlay>,
19
+ React.ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay>
20
+ >(({ className, ...props }, ref) => (
21
+ <DialogPrimitive.Overlay
22
+ ref={ref}
23
+ className={cn(
24
+ 'fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0',
25
+ className,
26
+ )}
27
+ {...props}
28
+ />
29
+ ));
30
+ DialogOverlay.displayName = DialogPrimitive.Overlay.displayName;
31
+
32
+ const DialogContent = React.forwardRef<
33
+ React.ElementRef<typeof DialogPrimitive.Content>,
34
+ React.ComponentPropsWithoutRef<typeof DialogPrimitive.Content>
35
+ >(({ className, children, ...props }, ref) => (
36
+ <DialogPortal>
37
+ <DialogOverlay />
38
+ <DialogPrimitive.Content
39
+ ref={ref}
40
+ className={cn(
41
+ 'fixed left-[50%] top-[50%] z-50 grid w-full max-w-lg translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg',
42
+ className,
43
+ )}
44
+ {...props}
45
+ >
46
+ {children}
47
+ <DialogPrimitive.Close className="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground">
48
+ <X className="h-4 w-4" />
49
+ <span className="sr-only">Close</span>
50
+ </DialogPrimitive.Close>
51
+ </DialogPrimitive.Content>
52
+ </DialogPortal>
53
+ ));
54
+ DialogContent.displayName = DialogPrimitive.Content.displayName;
55
+
56
+ const DialogHeader = ({
57
+ className,
58
+ ...props
59
+ }: React.HTMLAttributes<HTMLDivElement>) => (
60
+ <div
61
+ className={cn(
62
+ 'flex flex-col space-y-1.5 text-center sm:text-left',
63
+ className,
64
+ )}
65
+ {...props}
66
+ />
67
+ );
68
+ DialogHeader.displayName = 'DialogHeader';
69
+
70
+ const DialogFooter = ({
71
+ className,
72
+ ...props
73
+ }: React.HTMLAttributes<HTMLDivElement>) => (
74
+ <div
75
+ className={cn(
76
+ 'flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2',
77
+ className,
78
+ )}
79
+ {...props}
80
+ />
81
+ );
82
+ DialogFooter.displayName = 'DialogFooter';
83
+
84
+ const DialogTitle = React.forwardRef<
85
+ React.ElementRef<typeof DialogPrimitive.Title>,
86
+ React.ComponentPropsWithoutRef<typeof DialogPrimitive.Title>
87
+ >(({ className, ...props }, ref) => (
88
+ <DialogPrimitive.Title
89
+ ref={ref}
90
+ className={cn(
91
+ 'text-lg font-semibold leading-none tracking-tight',
92
+ className,
93
+ )}
94
+ {...props}
95
+ />
96
+ ));
97
+ DialogTitle.displayName = DialogPrimitive.Title.displayName;
98
+
99
+ const DialogDescription = React.forwardRef<
100
+ React.ElementRef<typeof DialogPrimitive.Description>,
101
+ React.ComponentPropsWithoutRef<typeof DialogPrimitive.Description>
102
+ >(({ className, ...props }, ref) => (
103
+ <DialogPrimitive.Description
104
+ ref={ref}
105
+ className={cn('text-sm text-muted-foreground', className)}
106
+ {...props}
107
+ />
108
+ ));
109
+ DialogDescription.displayName = DialogPrimitive.Description.displayName;
110
+
111
+ export {
112
+ Dialog,
113
+ DialogClose,
114
+ DialogContent,
115
+ DialogDescription,
116
+ DialogFooter,
117
+ DialogHeader,
118
+ DialogOverlay,
119
+ DialogPortal,
120
+ DialogTitle,
121
+ DialogTrigger,
122
+ };
web/src/components/ui/multi-select.tsx ADDED
@@ -0,0 +1,381 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ // src/components/multi-select.tsx
2
+
3
+ import { cva, type VariantProps } from 'class-variance-authority';
4
+ import {
5
+ CheckIcon,
6
+ ChevronDown,
7
+ WandSparkles,
8
+ XCircle,
9
+ XIcon,
10
+ } from 'lucide-react';
11
+ import * as React from 'react';
12
+
13
+ import { Badge } from '@/components/ui/badge';
14
+ import { Button } from '@/components/ui/button';
15
+ import {
16
+ Command,
17
+ CommandEmpty,
18
+ CommandGroup,
19
+ CommandInput,
20
+ CommandItem,
21
+ CommandList,
22
+ CommandSeparator,
23
+ } from '@/components/ui/command';
24
+ import {
25
+ Popover,
26
+ PopoverContent,
27
+ PopoverTrigger,
28
+ } from '@/components/ui/popover';
29
+ import { Separator } from '@/components/ui/separator';
30
+ import { cn } from '@/lib/utils';
31
+
32
+ /**
33
+ * Variants for the multi-select component to handle different styles.
34
+ * Uses class-variance-authority (cva) to define different styles based on "variant" prop.
35
+ */
36
+ const multiSelectVariants = cva(
37
+ 'm-1 transition ease-in-out delay-150 hover:-translate-y-1 hover:scale-110 duration-300',
38
+ {
39
+ variants: {
40
+ variant: {
41
+ default:
42
+ 'border-foreground/10 text-foreground bg-card hover:bg-card/80',
43
+ secondary:
44
+ 'border-foreground/10 bg-secondary text-secondary-foreground hover:bg-secondary/80',
45
+ destructive:
46
+ 'border-transparent bg-destructive text-destructive-foreground hover:bg-destructive/80',
47
+ inverted: 'inverted',
48
+ },
49
+ },
50
+ defaultVariants: {
51
+ variant: 'default',
52
+ },
53
+ },
54
+ );
55
+
56
+ /**
57
+ * Props for MultiSelect component
58
+ */
59
+ interface MultiSelectProps
60
+ extends React.ButtonHTMLAttributes<HTMLButtonElement>,
61
+ VariantProps<typeof multiSelectVariants> {
62
+ /**
63
+ * An array of option objects to be displayed in the multi-select component.
64
+ * Each option object has a label, value, and an optional icon.
65
+ */
66
+ options: {
67
+ /** The text to display for the option. */
68
+ label: string;
69
+ /** The unique value associated with the option. */
70
+ value: string;
71
+ /** Optional icon component to display alongside the option. */
72
+ icon?: React.ComponentType<{ className?: string }>;
73
+ }[];
74
+
75
+ /**
76
+ * Callback function triggered when the selected values change.
77
+ * Receives an array of the new selected values.
78
+ */
79
+ onValueChange: (value: string[]) => void;
80
+
81
+ /** The default selected values when the component mounts. */
82
+ defaultValue?: string[];
83
+
84
+ /**
85
+ * Placeholder text to be displayed when no values are selected.
86
+ * Optional, defaults to "Select options".
87
+ */
88
+ placeholder?: string;
89
+
90
+ /**
91
+ * Animation duration in seconds for the visual effects (e.g., bouncing badges).
92
+ * Optional, defaults to 0 (no animation).
93
+ */
94
+ animation?: number;
95
+
96
+ /**
97
+ * Maximum number of items to display. Extra selected items will be summarized.
98
+ * Optional, defaults to 3.
99
+ */
100
+ maxCount?: number;
101
+
102
+ /**
103
+ * The modality of the popover. When set to true, interaction with outside elements
104
+ * will be disabled and only popover content will be visible to screen readers.
105
+ * Optional, defaults to false.
106
+ */
107
+ modalPopover?: boolean;
108
+
109
+ /**
110
+ * If true, renders the multi-select component as a child of another component.
111
+ * Optional, defaults to false.
112
+ */
113
+ asChild?: boolean;
114
+
115
+ /**
116
+ * Additional class names to apply custom styles to the multi-select component.
117
+ * Optional, can be used to add custom styles.
118
+ */
119
+ className?: string;
120
+ }
121
+
122
+ export const MultiSelect = React.forwardRef<
123
+ HTMLButtonElement,
124
+ MultiSelectProps
125
+ >(
126
+ (
127
+ {
128
+ options,
129
+ onValueChange,
130
+ variant,
131
+ defaultValue = [],
132
+ placeholder = 'Select options',
133
+ animation = 0,
134
+ maxCount = 3,
135
+ modalPopover = false,
136
+ asChild = false,
137
+ className,
138
+ ...props
139
+ },
140
+ ref,
141
+ ) => {
142
+ const [selectedValues, setSelectedValues] =
143
+ React.useState<string[]>(defaultValue);
144
+ const [isPopoverOpen, setIsPopoverOpen] = React.useState(false);
145
+ const [isAnimating, setIsAnimating] = React.useState(false);
146
+
147
+ const handleInputKeyDown = (
148
+ event: React.KeyboardEvent<HTMLInputElement>,
149
+ ) => {
150
+ if (event.key === 'Enter') {
151
+ setIsPopoverOpen(true);
152
+ } else if (event.key === 'Backspace' && !event.currentTarget.value) {
153
+ const newSelectedValues = [...selectedValues];
154
+ newSelectedValues.pop();
155
+ setSelectedValues(newSelectedValues);
156
+ onValueChange(newSelectedValues);
157
+ }
158
+ };
159
+
160
+ const toggleOption = (option: string) => {
161
+ const newSelectedValues = selectedValues.includes(option)
162
+ ? selectedValues.filter((value) => value !== option)
163
+ : [...selectedValues, option];
164
+ setSelectedValues(newSelectedValues);
165
+ onValueChange(newSelectedValues);
166
+ };
167
+
168
+ const handleClear = () => {
169
+ setSelectedValues([]);
170
+ onValueChange([]);
171
+ };
172
+
173
+ const handleTogglePopover = () => {
174
+ setIsPopoverOpen((prev) => !prev);
175
+ };
176
+
177
+ const clearExtraOptions = () => {
178
+ const newSelectedValues = selectedValues.slice(0, maxCount);
179
+ setSelectedValues(newSelectedValues);
180
+ onValueChange(newSelectedValues);
181
+ };
182
+
183
+ const toggleAll = () => {
184
+ if (selectedValues.length === options.length) {
185
+ handleClear();
186
+ } else {
187
+ const allValues = options.map((option) => option.value);
188
+ setSelectedValues(allValues);
189
+ onValueChange(allValues);
190
+ }
191
+ };
192
+
193
+ return (
194
+ <Popover
195
+ open={isPopoverOpen}
196
+ onOpenChange={setIsPopoverOpen}
197
+ modal={modalPopover}
198
+ >
199
+ <PopoverTrigger asChild>
200
+ <Button
201
+ ref={ref}
202
+ {...props}
203
+ onClick={handleTogglePopover}
204
+ className={cn(
205
+ 'flex w-full p-1 rounded-md border min-h-10 h-auto items-center justify-between bg-inherit hover:bg-inherit [&_svg]:pointer-events-auto',
206
+ className,
207
+ )}
208
+ >
209
+ {selectedValues.length > 0 ? (
210
+ <div className="flex justify-between items-center w-full">
211
+ <div className="flex flex-wrap items-center">
212
+ {selectedValues.slice(0, maxCount).map((value) => {
213
+ const option = options.find((o) => o.value === value);
214
+ const IconComponent = option?.icon;
215
+ return (
216
+ <Badge
217
+ key={value}
218
+ className={cn(
219
+ isAnimating ? 'animate-bounce' : '',
220
+ multiSelectVariants({ variant }),
221
+ )}
222
+ style={{ animationDuration: `${animation}s` }}
223
+ >
224
+ {IconComponent && (
225
+ <IconComponent className="h-4 w-4 mr-2" />
226
+ )}
227
+ {option?.label}
228
+ <XCircle
229
+ className="ml-2 h-4 w-4 cursor-pointer"
230
+ onClick={(event) => {
231
+ event.stopPropagation();
232
+ toggleOption(value);
233
+ }}
234
+ />
235
+ </Badge>
236
+ );
237
+ })}
238
+ {selectedValues.length > maxCount && (
239
+ <Badge
240
+ className={cn(
241
+ 'bg-transparent text-foreground border-foreground/1 hover:bg-transparent',
242
+ isAnimating ? 'animate-bounce' : '',
243
+ multiSelectVariants({ variant }),
244
+ )}
245
+ style={{ animationDuration: `${animation}s` }}
246
+ >
247
+ {`+ ${selectedValues.length - maxCount} more`}
248
+ <XCircle
249
+ className="ml-2 h-4 w-4 cursor-pointer"
250
+ onClick={(event) => {
251
+ event.stopPropagation();
252
+ clearExtraOptions();
253
+ }}
254
+ />
255
+ </Badge>
256
+ )}
257
+ </div>
258
+ <div className="flex items-center justify-between">
259
+ <XIcon
260
+ className="h-4 mx-2 cursor-pointer text-muted-foreground"
261
+ onClick={(event) => {
262
+ event.stopPropagation();
263
+ handleClear();
264
+ }}
265
+ />
266
+ <Separator
267
+ orientation="vertical"
268
+ className="flex min-h-6 h-full"
269
+ />
270
+ <ChevronDown className="h-4 mx-2 cursor-pointer text-muted-foreground" />
271
+ </div>
272
+ </div>
273
+ ) : (
274
+ <div className="flex items-center justify-between w-full mx-auto">
275
+ <span className="text-sm text-muted-foreground mx-3">
276
+ {placeholder}
277
+ </span>
278
+ <ChevronDown className="h-4 cursor-pointer text-muted-foreground mx-2" />
279
+ </div>
280
+ )}
281
+ </Button>
282
+ </PopoverTrigger>
283
+ <PopoverContent
284
+ className="w-auto p-0"
285
+ align="start"
286
+ onEscapeKeyDown={() => setIsPopoverOpen(false)}
287
+ >
288
+ <Command>
289
+ <CommandInput
290
+ placeholder="Search..."
291
+ onKeyDown={handleInputKeyDown}
292
+ />
293
+ <CommandList>
294
+ <CommandEmpty>No results found.</CommandEmpty>
295
+ <CommandGroup>
296
+ <CommandItem
297
+ key="all"
298
+ onSelect={toggleAll}
299
+ className="cursor-pointer"
300
+ >
301
+ <div
302
+ className={cn(
303
+ 'mr-2 flex h-4 w-4 items-center justify-center rounded-sm border border-primary',
304
+ selectedValues.length === options.length
305
+ ? 'bg-primary text-primary-foreground'
306
+ : 'opacity-50 [&_svg]:invisible',
307
+ )}
308
+ >
309
+ <CheckIcon className="h-4 w-4" />
310
+ </div>
311
+ <span>(Select All)</span>
312
+ </CommandItem>
313
+ {options.map((option) => {
314
+ const isSelected = selectedValues.includes(option.value);
315
+ return (
316
+ <CommandItem
317
+ key={option.value}
318
+ onSelect={() => toggleOption(option.value)}
319
+ className="cursor-pointer"
320
+ >
321
+ <div
322
+ className={cn(
323
+ 'mr-2 flex h-4 w-4 items-center justify-center rounded-sm border border-primary',
324
+ isSelected
325
+ ? 'bg-primary text-primary-foreground'
326
+ : 'opacity-50 [&_svg]:invisible',
327
+ )}
328
+ >
329
+ <CheckIcon className="h-4 w-4" />
330
+ </div>
331
+ {option.icon && (
332
+ <option.icon className="mr-2 h-4 w-4 text-muted-foreground" />
333
+ )}
334
+ <span>{option.label}</span>
335
+ </CommandItem>
336
+ );
337
+ })}
338
+ </CommandGroup>
339
+ <CommandSeparator />
340
+ <CommandGroup>
341
+ <div className="flex items-center justify-between">
342
+ {selectedValues.length > 0 && (
343
+ <>
344
+ <CommandItem
345
+ onSelect={handleClear}
346
+ className="flex-1 justify-center cursor-pointer"
347
+ >
348
+ Clear
349
+ </CommandItem>
350
+ <Separator
351
+ orientation="vertical"
352
+ className="flex min-h-6 h-full"
353
+ />
354
+ </>
355
+ )}
356
+ <CommandItem
357
+ onSelect={() => setIsPopoverOpen(false)}
358
+ className="flex-1 justify-center cursor-pointer max-w-full"
359
+ >
360
+ Close
361
+ </CommandItem>
362
+ </div>
363
+ </CommandGroup>
364
+ </CommandList>
365
+ </Command>
366
+ </PopoverContent>
367
+ {animation > 0 && selectedValues.length > 0 && (
368
+ <WandSparkles
369
+ className={cn(
370
+ 'cursor-pointer my-2 text-foreground bg-background w-3 h-3',
371
+ isAnimating ? '' : 'text-muted-foreground',
372
+ )}
373
+ onClick={() => setIsAnimating(!isAnimating)}
374
+ />
375
+ )}
376
+ </Popover>
377
+ );
378
+ },
379
+ );
380
+
381
+ MultiSelect.displayName = 'MultiSelect';
web/src/pages/dataset/setting/advanced-setting-form.tsx CHANGED
@@ -23,9 +23,10 @@ import {
23
  } from '@/components/ui/select';
24
  import { FormSlider } from '@/components/ui/slider';
25
  import { Textarea } from '@/components/ui/textarea';
 
26
 
27
  const formSchema = z.object({
28
- username: z.number().min(2, {
29
  message: 'Username must be at least 2 characters.',
30
  }),
31
  a: z.number().min(2, {
@@ -46,7 +47,7 @@ export default function AdvancedSettingForm() {
46
  const form = useForm<z.infer<typeof formSchema>>({
47
  resolver: zodResolver(formSchema),
48
  defaultValues: {
49
- username: 0,
50
  },
51
  });
52
 
@@ -59,9 +60,9 @@ export default function AdvancedSettingForm() {
59
  <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-8">
60
  <FormField
61
  control={form.control}
62
- name="username"
63
  render={({ field }) => (
64
- <FormItem>
65
  <FormLabel>Username</FormLabel>
66
  <FormControl>
67
  <FormSlider {...field}></FormSlider>
@@ -73,11 +74,12 @@ export default function AdvancedSettingForm() {
73
  </FormItem>
74
  )}
75
  />
 
76
  <FormField
77
  control={form.control}
78
  name="a"
79
  render={({ field }) => (
80
- <FormItem>
81
  <FormLabel>Username</FormLabel>
82
  <FormControl>
83
  <FormSlider {...field}></FormSlider>
@@ -93,7 +95,7 @@ export default function AdvancedSettingForm() {
93
  control={form.control}
94
  name="b"
95
  render={({ field }) => (
96
- <FormItem>
97
  <FormLabel>Username</FormLabel>
98
  <Select onValueChange={field.onChange} defaultValue={field.value}>
99
  <FormControl>
@@ -118,7 +120,7 @@ export default function AdvancedSettingForm() {
118
  control={form.control}
119
  name="c"
120
  render={({ field }) => (
121
- <FormItem>
122
  <FormLabel>Username</FormLabel>
123
  <FormControl>
124
  <FormSlider {...field}></FormSlider>
@@ -134,7 +136,7 @@ export default function AdvancedSettingForm() {
134
  control={form.control}
135
  name="d"
136
  render={({ field }) => (
137
- <FormItem>
138
  <FormLabel>Username</FormLabel>
139
  <FormControl>
140
  <Textarea
@@ -153,7 +155,7 @@ export default function AdvancedSettingForm() {
153
  variant={'tertiary'}
154
  size={'sm'}
155
  type="submit"
156
- className="w-full"
157
  >
158
  Test
159
  </Button>
 
23
  } from '@/components/ui/select';
24
  import { FormSlider } from '@/components/ui/slider';
25
  import { Textarea } from '@/components/ui/textarea';
26
+ import ChunkMethodCard from './chunk-method-card';
27
 
28
  const formSchema = z.object({
29
+ parser_id: z.string().min(1, {
30
  message: 'Username must be at least 2 characters.',
31
  }),
32
  a: z.number().min(2, {
 
47
  const form = useForm<z.infer<typeof formSchema>>({
48
  resolver: zodResolver(formSchema),
49
  defaultValues: {
50
+ parser_id: '',
51
  },
52
  });
53
 
 
60
  <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-8">
61
  <FormField
62
  control={form.control}
63
+ name="a"
64
  render={({ field }) => (
65
+ <FormItem className="w-2/5">
66
  <FormLabel>Username</FormLabel>
67
  <FormControl>
68
  <FormSlider {...field}></FormSlider>
 
74
  </FormItem>
75
  )}
76
  />
77
+ <ChunkMethodCard></ChunkMethodCard>
78
  <FormField
79
  control={form.control}
80
  name="a"
81
  render={({ field }) => (
82
+ <FormItem className="w-2/5">
83
  <FormLabel>Username</FormLabel>
84
  <FormControl>
85
  <FormSlider {...field}></FormSlider>
 
95
  control={form.control}
96
  name="b"
97
  render={({ field }) => (
98
+ <FormItem className="w-2/5">
99
  <FormLabel>Username</FormLabel>
100
  <Select onValueChange={field.onChange} defaultValue={field.value}>
101
  <FormControl>
 
120
  control={form.control}
121
  name="c"
122
  render={({ field }) => (
123
+ <FormItem className="w-2/5">
124
  <FormLabel>Username</FormLabel>
125
  <FormControl>
126
  <FormSlider {...field}></FormSlider>
 
136
  control={form.control}
137
  name="d"
138
  render={({ field }) => (
139
+ <FormItem className="w-2/5">
140
  <FormLabel>Username</FormLabel>
141
  <FormControl>
142
  <Textarea
 
155
  variant={'tertiary'}
156
  size={'sm'}
157
  type="submit"
158
+ className="w-2/5"
159
  >
160
  Test
161
  </Button>
web/src/pages/dataset/setting/basic-setting-form.tsx CHANGED
@@ -4,16 +4,16 @@ import { zodResolver } from '@hookform/resolvers/zod';
4
  import { useForm } from 'react-hook-form';
5
  import { z } from 'zod';
6
 
7
- import { Button } from '@/components/ui/button';
8
  import {
9
  Form,
10
  FormControl,
11
- FormDescription,
12
  FormField,
13
  FormItem,
14
  FormLabel,
15
  FormMessage,
16
  } from '@/components/ui/form';
 
 
17
  import {
18
  Select,
19
  SelectContent,
@@ -21,34 +21,48 @@ import {
21
  SelectTrigger,
22
  SelectValue,
23
  } from '@/components/ui/select';
24
- import { FormSlider } from '@/components/ui/slider';
25
- import { Textarea } from '@/components/ui/textarea';
 
26
 
27
- const formSchema = z.object({
28
- username: z.number().min(2, {
29
- message: 'Username must be at least 2 characters.',
30
- }),
31
- a: z.number().min(2, {
32
- message: 'Username must be at least 2 characters.',
33
- }),
34
- b: z.string().min(2, {
35
- message: 'Username must be at least 2 characters.',
36
- }),
37
- c: z.number().min(2, {
38
- message: 'Username must be at least 2 characters.',
39
- }),
40
- d: z.string().min(2, {
41
- message: 'Username must be at least 2 characters.',
42
- }),
43
- });
44
 
45
  export default function BasicSettingForm() {
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
46
  const form = useForm<z.infer<typeof formSchema>>({
47
  resolver: zodResolver(formSchema),
48
  defaultValues: {
49
- username: 0,
 
50
  },
51
  });
 
 
 
 
52
 
53
  function onSubmit(values: z.infer<typeof formSchema>) {
54
  console.log(values);
@@ -59,42 +73,42 @@ export default function BasicSettingForm() {
59
  <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-8">
60
  <FormField
61
  control={form.control}
62
- name="username"
63
  render={({ field }) => (
64
  <FormItem>
65
- <FormLabel>Username</FormLabel>
66
  <FormControl>
67
- <FormSlider {...field}></FormSlider>
 
 
 
68
  </FormControl>
69
- <FormDescription>
70
- This is your public display name.
71
- </FormDescription>
72
  <FormMessage />
73
  </FormItem>
74
  )}
75
  />
76
  <FormField
77
  control={form.control}
78
- name="a"
79
  render={({ field }) => (
80
  <FormItem>
81
  <FormLabel>Username</FormLabel>
82
  <FormControl>
83
- <FormSlider {...field}></FormSlider>
 
 
 
84
  </FormControl>
85
- <FormDescription>
86
- This is your public display name.
87
- </FormDescription>
88
  <FormMessage />
89
  </FormItem>
90
  )}
91
  />
92
  <FormField
93
  control={form.control}
94
- name="b"
95
  render={({ field }) => (
96
  <FormItem>
97
- <FormLabel>Username</FormLabel>
98
  <Select onValueChange={field.onChange} defaultValue={field.value}>
99
  <FormControl>
100
  <SelectTrigger className="bg-colors-background-inverse-weak">
@@ -107,9 +121,6 @@ export default function BasicSettingForm() {
107
  <SelectItem value="[email protected]">[email protected]</SelectItem>
108
  </SelectContent>
109
  </Select>
110
- <FormDescription>
111
- This is your public display name.
112
- </FormDescription>
113
  <FormMessage />
114
  </FormItem>
115
  )}
@@ -121,42 +132,20 @@ export default function BasicSettingForm() {
121
  <FormItem>
122
  <FormLabel>Username</FormLabel>
123
  <FormControl>
124
- <FormSlider {...field}></FormSlider>
125
- </FormControl>
126
- <FormDescription>
127
- This is your public display name.
128
- </FormDescription>
129
- <FormMessage />
130
- </FormItem>
131
- )}
132
- />
133
- <FormField
134
- control={form.control}
135
- name="d"
136
- render={({ field }) => (
137
- <FormItem>
138
- <FormLabel>Username</FormLabel>
139
- <FormControl>
140
- <Textarea
141
  {...field}
142
- className="bg-colors-background-inverse-weak"
143
- ></Textarea>
144
  </FormControl>
145
- <FormDescription>
146
- This is your public display name.
147
- </FormDescription>
148
  <FormMessage />
149
  </FormItem>
150
  )}
151
  />
152
- <Button
153
- variant={'tertiary'}
154
- size={'sm'}
155
- type="submit"
156
- className="w-full"
157
- >
158
- Test
159
- </Button>
160
  </form>
161
  </Form>
162
  );
 
4
  import { useForm } from 'react-hook-form';
5
  import { z } from 'zod';
6
 
 
7
  import {
8
  Form,
9
  FormControl,
 
10
  FormField,
11
  FormItem,
12
  FormLabel,
13
  FormMessage,
14
  } from '@/components/ui/form';
15
+ import { Input } from '@/components/ui/input';
16
+ import { MultiSelect } from '@/components/ui/multi-select';
17
  import {
18
  Select,
19
  SelectContent,
 
21
  SelectTrigger,
22
  SelectValue,
23
  } from '@/components/ui/select';
24
+ import { useTranslate } from '@/hooks/common-hooks';
25
+ import { Cat, Dog, Fish, Rabbit, Turtle } from 'lucide-react';
26
+ import { useState } from 'react';
27
 
28
+ const frameworksList = [
29
+ { value: 'react', label: 'React', icon: Turtle },
30
+ { value: 'angular', label: 'Angular', icon: Cat },
31
+ { value: 'vue', label: 'Vue', icon: Dog },
32
+ { value: 'svelte', label: 'Svelte', icon: Rabbit },
33
+ { value: 'ember', label: 'Ember', icon: Fish },
34
+ ];
 
 
 
 
 
 
 
 
 
 
35
 
36
  export default function BasicSettingForm() {
37
+ const { t } = useTranslate('knowledgeConfiguration');
38
+
39
+ const formSchema = z.object({
40
+ name: z.string().min(1),
41
+ a: z.number().min(2, {
42
+ message: 'Username must be at least 2 characters.',
43
+ }),
44
+ language: z.string().min(1, {
45
+ message: 'Username must be at least 2 characters.',
46
+ }),
47
+ c: z.number().min(2, {
48
+ message: 'Username must be at least 2 characters.',
49
+ }),
50
+ d: z.string().min(2, {
51
+ message: 'Username must be at least 2 characters.',
52
+ }),
53
+ });
54
+
55
  const form = useForm<z.infer<typeof formSchema>>({
56
  resolver: zodResolver(formSchema),
57
  defaultValues: {
58
+ name: '',
59
+ language: 'English',
60
  },
61
  });
62
+ const [selectedFrameworks, setSelectedFrameworks] = useState<string[]>([
63
+ 'react',
64
+ 'angular',
65
+ ]);
66
 
67
  function onSubmit(values: z.infer<typeof formSchema>) {
68
  console.log(values);
 
73
  <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-8">
74
  <FormField
75
  control={form.control}
76
+ name="name"
77
  render={({ field }) => (
78
  <FormItem>
79
+ <FormLabel>{t('name')}</FormLabel>
80
  <FormControl>
81
+ <Input
82
+ {...field}
83
+ className="bg-colors-background-inverse-weak"
84
+ ></Input>
85
  </FormControl>
 
 
 
86
  <FormMessage />
87
  </FormItem>
88
  )}
89
  />
90
  <FormField
91
  control={form.control}
92
+ name="d"
93
  render={({ field }) => (
94
  <FormItem>
95
  <FormLabel>Username</FormLabel>
96
  <FormControl>
97
+ <Input
98
+ {...field}
99
+ className="bg-colors-background-inverse-weak"
100
+ ></Input>
101
  </FormControl>
 
 
 
102
  <FormMessage />
103
  </FormItem>
104
  )}
105
  />
106
  <FormField
107
  control={form.control}
108
+ name="language"
109
  render={({ field }) => (
110
  <FormItem>
111
+ <FormLabel>{t('language')}</FormLabel>
112
  <Select onValueChange={field.onChange} defaultValue={field.value}>
113
  <FormControl>
114
  <SelectTrigger className="bg-colors-background-inverse-weak">
 
121
  <SelectItem value="[email protected]">[email protected]</SelectItem>
122
  </SelectContent>
123
  </Select>
 
 
 
124
  <FormMessage />
125
  </FormItem>
126
  )}
 
132
  <FormItem>
133
  <FormLabel>Username</FormLabel>
134
  <FormControl>
135
+ <MultiSelect
136
+ options={frameworksList}
137
+ onValueChange={setSelectedFrameworks}
138
+ defaultValue={selectedFrameworks}
139
+ placeholder="Select frameworks"
140
+ variant="inverted"
141
+ maxCount={100}
 
 
 
 
 
 
 
 
 
 
142
  {...field}
143
+ />
 
144
  </FormControl>
 
 
 
145
  <FormMessage />
146
  </FormItem>
147
  )}
148
  />
 
 
 
 
 
 
 
 
149
  </form>
150
  </Form>
151
  );
web/src/pages/dataset/setting/chunk-method-card.tsx ADDED
@@ -0,0 +1,124 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import SvgIcon from '@/components/svg-icon';
2
+ import { Card } from '@/components/ui/card';
3
+ import {
4
+ FormControl,
5
+ FormField,
6
+ FormItem,
7
+ FormLabel,
8
+ FormMessage,
9
+ } from '@/components/ui/form';
10
+ import {
11
+ Select,
12
+ SelectContent,
13
+ SelectItem,
14
+ SelectTrigger,
15
+ SelectValue,
16
+ } from '@/components/ui/select';
17
+ import { useTranslate } from '@/hooks/common-hooks';
18
+ import { useSelectParserList } from '@/hooks/user-setting-hooks';
19
+ import { Col, Divider, Empty, Row, Typography } from 'antd';
20
+ import DOMPurify from 'dompurify';
21
+ import camelCase from 'lodash/camelCase';
22
+ import { useMemo } from 'react';
23
+ import { useFormContext } from 'react-hook-form';
24
+ import styles from './index.less';
25
+ import { ImageMap } from './utils';
26
+
27
+ const { Title, Text } = Typography;
28
+
29
+ const CategoryPanel = ({ chunkMethod }: { chunkMethod: string }) => {
30
+ const parserList = useSelectParserList();
31
+ const { t } = useTranslate('knowledgeConfiguration');
32
+
33
+ const item = useMemo(() => {
34
+ const item = parserList.find((x) => x.value === chunkMethod);
35
+ if (item) {
36
+ return {
37
+ title: item.label,
38
+ description: t(camelCase(item.value)),
39
+ };
40
+ }
41
+ return { title: '', description: '' };
42
+ }, [parserList, chunkMethod, t]);
43
+
44
+ const imageList = useMemo(() => {
45
+ if (chunkMethod in ImageMap) {
46
+ return ImageMap[chunkMethod as keyof typeof ImageMap];
47
+ }
48
+ return [];
49
+ }, [chunkMethod]);
50
+
51
+ return (
52
+ <section className={styles.categoryPanelWrapper}>
53
+ {imageList.length > 0 ? (
54
+ <>
55
+ <Title level={5} className={styles.topTitle}>
56
+ {`"${item.title}" ${t('methodTitle')}`}
57
+ </Title>
58
+ <p
59
+ dangerouslySetInnerHTML={{
60
+ __html: DOMPurify.sanitize(item.description),
61
+ }}
62
+ ></p>
63
+ <Title level={5}>{`"${item.title}" ${t('methodExamples')}`}</Title>
64
+ <Text>{t('methodExamplesDescription')}</Text>
65
+ <Row gutter={[10, 10]} className={styles.imageRow}>
66
+ {imageList.map((x) => (
67
+ <Col span={12} key={x}>
68
+ <SvgIcon
69
+ name={x}
70
+ width={'100%'}
71
+ className={styles.image}
72
+ ></SvgIcon>
73
+ </Col>
74
+ ))}
75
+ </Row>
76
+ <Title level={5}>
77
+ {item.title} {t('dialogueExamplesTitle')}
78
+ </Title>
79
+ <Divider></Divider>
80
+ </>
81
+ ) : (
82
+ <Empty description={''} image={null}>
83
+ <p>{t('methodEmpty')}</p>
84
+ <SvgIcon name={'chunk-method/chunk-empty'} width={'100%'}></SvgIcon>
85
+ </Empty>
86
+ )}
87
+ </section>
88
+ );
89
+ };
90
+
91
+ export default function ChunkMethodCard() {
92
+ const { t } = useTranslate('knowledgeConfiguration');
93
+ const form = useFormContext();
94
+
95
+ return (
96
+ <Card className="border-0 p-6 mb-8 bg-colors-background-inverse-weak flex">
97
+ <div className="w-2/5">
98
+ <FormField
99
+ control={form.control}
100
+ name="parser_id"
101
+ render={({ field }) => (
102
+ <FormItem>
103
+ <FormLabel>{t('chunkMethod')}</FormLabel>
104
+ <Select onValueChange={field.onChange} defaultValue={field.value}>
105
+ <FormControl>
106
+ <SelectTrigger className="bg-colors-background-inverse-weak">
107
+ <SelectValue placeholder="Select a verified email to display" />
108
+ </SelectTrigger>
109
+ </FormControl>
110
+ <SelectContent>
111
+ <SelectItem value="[email protected]">[email protected]</SelectItem>
112
+ <SelectItem value="[email protected]">[email protected]</SelectItem>
113
+ <SelectItem value="[email protected]">[email protected]</SelectItem>
114
+ </SelectContent>
115
+ </Select>
116
+ <FormMessage />
117
+ </FormItem>
118
+ )}
119
+ />
120
+ </div>
121
+ <CategoryPanel chunkMethod=""></CategoryPanel>
122
+ </Card>
123
+ );
124
+ }
web/src/pages/dataset/setting/index.less ADDED
@@ -0,0 +1,45 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ .tags {
2
+ margin-bottom: 24px;
3
+ }
4
+
5
+ .preset {
6
+ display: flex;
7
+ height: 80px;
8
+ background-color: rgba(0, 0, 0, 0.1);
9
+ border-radius: 5px;
10
+ padding: 5px;
11
+ margin-bottom: 24px;
12
+
13
+ .left {
14
+ flex: 1;
15
+ }
16
+
17
+ .right {
18
+ width: 100px;
19
+ border-left: 1px solid rgba(0, 0, 0, 0.4);
20
+ margin: 10px 0px;
21
+ padding: 5px;
22
+ }
23
+ }
24
+
25
+ .configurationWrapper {
26
+ padding: 0 52px;
27
+ .buttonWrapper {
28
+ text-align: right;
29
+ }
30
+ .variableSlider {
31
+ width: 100%;
32
+ }
33
+ }
34
+
35
+ .categoryPanelWrapper {
36
+ .topTitle {
37
+ margin-top: 0;
38
+ }
39
+ .imageRow {
40
+ margin-top: 16px;
41
+ }
42
+ .image {
43
+ width: 100%;
44
+ }
45
+ }
web/src/pages/dataset/setting/index.tsx CHANGED
@@ -14,9 +14,7 @@ export default function DatasetSettings() {
14
 
15
  <div className="text-3xl font-bold mb-6">Advanced settings</div>
16
  <Card className="border-0 p-6 mb-8 bg-colors-background-inverse-weak">
17
- <div className="w-2/5">
18
- <AdvancedSettingForm></AdvancedSettingForm>
19
- </div>
20
  </Card>
21
  </section>
22
  );
 
14
 
15
  <div className="text-3xl font-bold mb-6">Advanced settings</div>
16
  <Card className="border-0 p-6 mb-8 bg-colors-background-inverse-weak">
17
+ <AdvancedSettingForm></AdvancedSettingForm>
 
 
18
  </Card>
19
  </section>
20
  );
web/src/pages/dataset/setting/utils.ts ADDED
@@ -0,0 +1,19 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ const getImageName = (prefix: string, length: number) =>
2
+ new Array(length)
3
+ .fill(0)
4
+ .map((x, idx) => `chunk-method/${prefix}-0${idx + 1}`);
5
+
6
+ export const ImageMap = {
7
+ book: getImageName('book', 4),
8
+ laws: getImageName('law', 2),
9
+ manual: getImageName('manual', 4),
10
+ picture: getImageName('media', 2),
11
+ naive: getImageName('naive', 2),
12
+ paper: getImageName('paper', 2),
13
+ presentation: getImageName('presentation', 2),
14
+ qa: getImageName('qa', 2),
15
+ resume: getImageName('resume', 2),
16
+ table: getImageName('table', 2),
17
+ one: getImageName('one', 2),
18
+ knowledge_graph: getImageName('knowledge-graph', 2),
19
+ };
web/src/pages/profile-setting/sidebar/index.tsx CHANGED
@@ -1,4 +1,4 @@
1
- import { useTheme } from '@/components/theme-provider';
2
  import { Button } from '@/components/ui/button';
3
  import { Label } from '@/components/ui/label';
4
  import { Switch } from '@/components/ui/switch';
@@ -52,6 +52,7 @@ export function SideBar() {
52
  const pathName = useSecondPathName();
53
  const { handleMenuClick } = useHandleMenuClick();
54
  const { setTheme } = useTheme();
 
55
 
56
  const handleThemeChange = useCallback(
57
  (checked: boolean) => {
@@ -89,7 +90,11 @@ export function SideBar() {
89
 
90
  <div className="p-6 mt-auto border-t">
91
  <div className="flex items-center gap-2 mb-6">
92
- <Switch id="dark-mode" onCheckedChange={handleThemeChange} />
 
 
 
 
93
  <Label htmlFor="dark-mode" className="text-sm">
94
  Dark
95
  </Label>
 
1
+ import { useIsDarkTheme, useTheme } from '@/components/theme-provider';
2
  import { Button } from '@/components/ui/button';
3
  import { Label } from '@/components/ui/label';
4
  import { Switch } from '@/components/ui/switch';
 
52
  const pathName = useSecondPathName();
53
  const { handleMenuClick } = useHandleMenuClick();
54
  const { setTheme } = useTheme();
55
+ const isDarkTheme = useIsDarkTheme();
56
 
57
  const handleThemeChange = useCallback(
58
  (checked: boolean) => {
 
90
 
91
  <div className="p-6 mt-auto border-t">
92
  <div className="flex items-center gap-2 mb-6">
93
+ <Switch
94
+ id="dark-mode"
95
+ onCheckedChange={handleThemeChange}
96
+ checked={isDarkTheme}
97
+ />
98
  <Label htmlFor="dark-mode" className="text-sm">
99
  Dark
100
  </Label>