@redacto.io/consent-sdk-react 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/dist/index.mjs ADDED
@@ -0,0 +1,1139 @@
1
+ // src/RedactoNoticeConsent/RedactoNoticeConsent.tsx
2
+ import { useEffect as useEffect2, useMemo, useState as useState2 } from "react";
3
+
4
+ // src/RedactoNoticeConsent/api/index.ts
5
+ import { jwtDecode } from "jwt-decode";
6
+ var BASE_URL = "https://api.redacto.tech";
7
+ var BACKEND_URL = "http://localhost:3000";
8
+ var fetchConsentContent = async ({
9
+ noticeId,
10
+ accessToken,
11
+ refreshToken,
12
+ language = "en"
13
+ }) => {
14
+ try {
15
+ const decodedToken = jwtDecode(accessToken);
16
+ const ORGANISATION_UUID = decodedToken == null ? void 0 : decodedToken.organisation_uuid;
17
+ const WORKSPACE_UUID = decodedToken == null ? void 0 : decodedToken.workspace_uuid;
18
+ const response = await fetch(
19
+ `${BASE_URL}/consent/public/organisations/${ORGANISATION_UUID}/workspaces/${WORKSPACE_UUID}/notices/${noticeId}`,
20
+ {
21
+ method: "GET",
22
+ headers: {
23
+ Authorization: `Bearer ${accessToken}`,
24
+ "Accept-Language": language,
25
+ "Content-Type": "application/json"
26
+ }
27
+ }
28
+ );
29
+ if (response.status === 401) {
30
+ try {
31
+ const response2 = await fetch(
32
+ `${BACKEND_URL}/api/consent/refresh-token`,
33
+ {
34
+ method: "POST",
35
+ headers: {
36
+ "Content-Type": "application/json"
37
+ },
38
+ body: JSON.stringify({
39
+ refreshToken
40
+ })
41
+ }
42
+ );
43
+ if (!response2.ok) {
44
+ throw new Error("Failed to get consent tokens");
45
+ }
46
+ const data2 = await response2.json();
47
+ localStorage.setItem("accessToken", data2.token);
48
+ localStorage.setItem("refreshToken", data2.refresh_token);
49
+ localStorage.setItem("tokenExpiresAt", data2.expires_at);
50
+ } catch (error) {
51
+ console.error("Failed to setup consent:", error);
52
+ throw error;
53
+ }
54
+ }
55
+ if (response.status === 409) {
56
+ const error = new Error("User has already provided consent");
57
+ error.status = 409;
58
+ throw error;
59
+ }
60
+ if (!response.ok) {
61
+ throw new Error(
62
+ `Failed to fetch consent content: ${response.statusText}`
63
+ );
64
+ }
65
+ const data = await response.json();
66
+ return data;
67
+ } catch (error) {
68
+ console.error("Error fetching consent content:", error);
69
+ throw error;
70
+ }
71
+ };
72
+ var submitConsentEvent = async ({
73
+ accessToken,
74
+ noticeUuid,
75
+ purposes,
76
+ declined
77
+ }) => {
78
+ try {
79
+ const primaryEmail = localStorage.getItem("userEmail");
80
+ const decodedToken = jwtDecode(accessToken);
81
+ const ORGANISATION_UUID = decodedToken.organisation_uuid;
82
+ const WORKSPACE_UUID = decodedToken.workspace_uuid;
83
+ const payload = {
84
+ primary_email: primaryEmail || void 0,
85
+ notice_uuid: noticeUuid,
86
+ source: "WEB",
87
+ declined,
88
+ consents: {
89
+ purposes: purposes.map((purpose) => ({
90
+ uuid: purpose.uuid,
91
+ name: purpose.name,
92
+ description: purpose.description,
93
+ industries: purpose.industries,
94
+ selected: purpose.selected,
95
+ data_elements: purpose.data_elements.map((element) => ({
96
+ uuid: element.uuid,
97
+ name: element.name,
98
+ enabled: element.enabled,
99
+ required: element.required,
100
+ selected: element.required ? true : element.selected
101
+ }))
102
+ }))
103
+ }
104
+ };
105
+ const response = await fetch(
106
+ `${BASE_URL}/consent/public/consent-events/organisations/${ORGANISATION_UUID}/workspaces/${WORKSPACE_UUID}/by-token`,
107
+ {
108
+ method: "POST",
109
+ headers: {
110
+ Authorization: `Bearer ${accessToken}`,
111
+ "Content-Type": "application/json"
112
+ },
113
+ body: JSON.stringify(payload)
114
+ }
115
+ );
116
+ if (!response.ok) {
117
+ throw new Error(`Failed to submit consent event: ${response.statusText}`);
118
+ }
119
+ } catch (error) {
120
+ console.error("Error submitting consent event:", error);
121
+ throw error;
122
+ }
123
+ };
124
+
125
+ // src/RedactoNoticeConsent/styles.ts
126
+ var styles = {
127
+ overlay: {
128
+ position: "fixed",
129
+ top: 0,
130
+ left: 0,
131
+ width: "100%",
132
+ height: "100%",
133
+ backgroundColor: "rgba(0, 0, 0, 0.5)",
134
+ display: "flex",
135
+ alignItems: "center",
136
+ justifyContent: "center",
137
+ zIndex: 999,
138
+ animation: "overlayFadeIn 0.2s cubic-bezier(0.16, 1, 0.3, 1)",
139
+ pointerEvents: "all"
140
+ },
141
+ modal: {
142
+ position: "fixed",
143
+ top: "50%",
144
+ left: "50%",
145
+ transform: "translate(-50%, -50%)",
146
+ width: "700px",
147
+ borderRadius: "8px",
148
+ backgroundColor: "#ffffff",
149
+ boxShadow: "4px 0px 4px 0px #00000040",
150
+ border: "none",
151
+ display: "flex",
152
+ flexDirection: "column",
153
+ maxHeight: "90vh",
154
+ zIndex: 1e3
155
+ },
156
+ content: {
157
+ margin: "22px",
158
+ display: "flex",
159
+ flexDirection: "column",
160
+ flexGrow: 1,
161
+ overflow: "hidden",
162
+ minHeight: 0
163
+ },
164
+ topSection: {
165
+ display: "flex",
166
+ justifyContent: "space-between",
167
+ alignItems: "center",
168
+ paddingBottom: "15px",
169
+ borderBottom: "1px solid #e5e7eb",
170
+ boxShadow: "0 1px 2px rgba(0, 0, 0, 0.03)",
171
+ backgroundColor: "#ffffff"
172
+ },
173
+ topLeft: {
174
+ display: "flex",
175
+ alignItems: "center",
176
+ gap: "10px"
177
+ },
178
+ logo: {
179
+ height: "32px",
180
+ width: "auto",
181
+ objectFit: "contain"
182
+ },
183
+ title: {
184
+ fontSize: "18px",
185
+ fontWeight: 700,
186
+ lineHeight: "150%",
187
+ letterSpacing: "0.2px",
188
+ verticalAlign: "middle",
189
+ color: "#101828"
190
+ },
191
+ topRight: {
192
+ borderRadius: "8px",
193
+ padding: "3px 9px",
194
+ border: "1px solid #d0d5dd",
195
+ display: "flex",
196
+ alignItems: "center",
197
+ justifyContent: "center",
198
+ fontWeight: 400,
199
+ fontSize: "12px",
200
+ lineHeight: "150%",
201
+ letterSpacing: "0.2px",
202
+ verticalAlign: "middle",
203
+ color: "#344054",
204
+ gap: "5px",
205
+ cursor: "pointer",
206
+ backgroundColor: "#ffffff"
207
+ },
208
+ middleSection: {
209
+ margin: "20px 0px",
210
+ display: "flex",
211
+ flexDirection: "column",
212
+ gap: "12px",
213
+ flexGrow: 1,
214
+ overflowY: "auto",
215
+ minHeight: 0,
216
+ paddingRight: "15px"
217
+ },
218
+ privacyText: {
219
+ fontSize: "16px",
220
+ fontWeight: 400,
221
+ lineHeight: "150%",
222
+ letterSpacing: "0.2px",
223
+ color: "#344054"
224
+ },
225
+ link: {
226
+ color: "#4f87ff",
227
+ textDecoration: "none"
228
+ },
229
+ subTitle: {
230
+ fontWeight: 600,
231
+ fontSize: "16px",
232
+ lineHeight: "150%",
233
+ letterSpacing: "0.2px",
234
+ color: "#101828"
235
+ },
236
+ optionsContainer: {
237
+ display: "flex",
238
+ flexDirection: "column",
239
+ gap: "14px"
240
+ },
241
+ optionItem: {
242
+ display: "flex",
243
+ justifyContent: "space-between",
244
+ alignItems: "center"
245
+ },
246
+ optionLeft: {
247
+ display: "flex",
248
+ gap: "12px",
249
+ alignItems: "center",
250
+ cursor: "pointer"
251
+ },
252
+ optionTextContainer: {
253
+ display: "flex",
254
+ flexDirection: "column"
255
+ },
256
+ optionTitle: {
257
+ fontWeight: 500,
258
+ fontSize: "16px",
259
+ lineHeight: "150%",
260
+ letterSpacing: "0.2px",
261
+ color: "#101828"
262
+ },
263
+ optionDescription: {
264
+ fontWeight: 400,
265
+ fontSize: "12px",
266
+ lineHeight: "150%",
267
+ letterSpacing: "0.2px",
268
+ color: "#475467",
269
+ verticalAlign: "middle"
270
+ },
271
+ checkboxLarge: {
272
+ height: "20px",
273
+ width: "20px",
274
+ padding: "4px",
275
+ borderRadius: "5px",
276
+ border: "2px solid #d0d5dd"
277
+ },
278
+ dataElementsContainer: {
279
+ marginLeft: "27px",
280
+ display: "flex",
281
+ flexDirection: "column"
282
+ },
283
+ dataElementItem: {
284
+ display: "flex",
285
+ alignItems: "center",
286
+ gap: "10px",
287
+ justifyContent: "space-between"
288
+ },
289
+ dataElementText: {
290
+ fontWeight: 400,
291
+ fontSize: "14px",
292
+ lineHeight: "150%",
293
+ letterSpacing: "0.2px",
294
+ color: "#344054"
295
+ },
296
+ checkboxSmall: {
297
+ height: "16px",
298
+ width: "16px",
299
+ padding: "4px",
300
+ borderRadius: "5px",
301
+ border: "2px solid #d0d5dd"
302
+ },
303
+ bottomSection: {
304
+ display: "flex",
305
+ justifyContent: "space-between",
306
+ gap: "29px",
307
+ paddingTop: "15px",
308
+ borderTop: "1px solid #e5e7eb",
309
+ boxShadow: "0 -1px 2px rgba(0, 0, 0, 0.03)",
310
+ backgroundColor: "#ffffff"
311
+ },
312
+ button: {
313
+ width: "100%",
314
+ borderRadius: "8px",
315
+ padding: "9px 45px",
316
+ fontWeight: 400,
317
+ fontSize: "16px",
318
+ lineHeight: "150%",
319
+ letterSpacing: "0.2px",
320
+ cursor: "pointer"
321
+ },
322
+ acceptButton: {
323
+ backgroundColor: "#4f87ff",
324
+ border: "none",
325
+ color: "#ffffff"
326
+ },
327
+ cancelButton: {
328
+ border: "1px solid #d0d5dd",
329
+ backgroundColor: "#ffffff",
330
+ color: "#000000"
331
+ },
332
+ languageSelectorContainer: {
333
+ position: "relative"
334
+ },
335
+ languageDropdown: {
336
+ position: "absolute",
337
+ top: "100%",
338
+ right: 0,
339
+ backgroundColor: "#ffffff",
340
+ border: "1px solid #d0d5dd",
341
+ borderRadius: "4px",
342
+ boxShadow: "0px 2px 4px rgba(0, 0, 0, 0.1)",
343
+ zIndex: 10,
344
+ width: "max-content",
345
+ marginTop: "4px"
346
+ },
347
+ languageItem: {
348
+ padding: "8px 12px",
349
+ cursor: "pointer",
350
+ fontSize: "12px",
351
+ color: "#344054",
352
+ backgroundColor: "#ffffff",
353
+ borderBottom: "1px solid #e5e7eb"
354
+ },
355
+ selectedLanguageItem: {
356
+ backgroundColor: "#f3f4f6",
357
+ fontWeight: 600
358
+ },
359
+ loadingContainer: {
360
+ display: "flex",
361
+ flexDirection: "column",
362
+ alignItems: "center",
363
+ justifyContent: "center",
364
+ padding: "2rem",
365
+ minHeight: "200px"
366
+ },
367
+ loadingSpinner: {
368
+ width: "40px",
369
+ height: "40px",
370
+ border: "4px solid #f3f3f3",
371
+ borderTop: "4px solid #3498db",
372
+ borderRadius: "50%",
373
+ animation: "spin 1s linear infinite",
374
+ marginBottom: "1rem"
375
+ },
376
+ // Media query styles
377
+ "@media (max-width: 768px)": {
378
+ modal: {
379
+ width: "90%"
380
+ },
381
+ content: {
382
+ margin: "20px"
383
+ },
384
+ title: {
385
+ fontSize: "16px"
386
+ },
387
+ subTitle: {
388
+ fontSize: "14px"
389
+ },
390
+ privacyText: {
391
+ fontSize: "14px"
392
+ },
393
+ optionTitle: {
394
+ fontSize: "14px"
395
+ },
396
+ optionDescription: {
397
+ fontSize: "11px"
398
+ },
399
+ dataElementText: {
400
+ fontSize: "12px"
401
+ },
402
+ topRight: {
403
+ fontSize: "11px",
404
+ padding: "3px 6px",
405
+ height: "auto",
406
+ width: "auto"
407
+ },
408
+ languageItem: {
409
+ fontSize: "11px",
410
+ padding: "6px 10px"
411
+ },
412
+ bottomSection: {
413
+ flexDirection: "column",
414
+ gap: "12px"
415
+ },
416
+ button: {
417
+ fontSize: "14px",
418
+ padding: "10px 20px"
419
+ },
420
+ middleSection: {
421
+ paddingRight: "10px"
422
+ }
423
+ }
424
+ };
425
+
426
+ // src/RedactoNoticeConsent/assets/redacto-logo.png
427
+ var redacto_logo_default = "";
428
+
429
+ // src/RedactoNoticeConsent/useMediaQuery.ts
430
+ import { useState, useEffect } from "react";
431
+ var useMediaQuery = (query) => {
432
+ const getMatches = () => typeof window !== "undefined" ? window.matchMedia(query).matches : false;
433
+ const [matches, setMatches] = useState(getMatches);
434
+ useEffect(() => {
435
+ if (typeof window === "undefined") return;
436
+ const media = window.matchMedia(query);
437
+ const listener = (e) => {
438
+ setMatches(e.matches);
439
+ };
440
+ if (media.addEventListener) {
441
+ media.addEventListener("change", listener);
442
+ } else {
443
+ media.addListener(listener);
444
+ }
445
+ return () => {
446
+ if (media.removeEventListener) {
447
+ media.removeEventListener("change", listener);
448
+ } else {
449
+ media.removeListener(listener);
450
+ }
451
+ };
452
+ }, [query]);
453
+ return matches;
454
+ };
455
+
456
+ // src/RedactoNoticeConsent/RedactoNoticeConsent.tsx
457
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
458
+ var RedactoNoticeConsent = ({
459
+ noticeId,
460
+ accessToken,
461
+ refreshToken,
462
+ language = "en",
463
+ blockUI = false,
464
+ onAccept,
465
+ onDecline,
466
+ onError,
467
+ settings
468
+ }) => {
469
+ var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o, _p, _q, _r, _s, _t;
470
+ const [content, setContent] = useState2(null);
471
+ const [isLoading, setIsLoading] = useState2(true);
472
+ const [isSubmitting, setIsSubmitting] = useState2(false);
473
+ const [isLanguageDropdownOpen, setIsLanguageDropdownOpen] = useState2(false);
474
+ const [selectedLanguage, setSelectedLanguage] = useState2(language);
475
+ const [collapsedPurposes, setCollapsedPurposes] = useState2({});
476
+ const [selectedPurposes, setSelectedPurposes] = useState2({});
477
+ const [selectedDataElements, setSelectedDataElements] = useState2({});
478
+ const [hasAlreadyConsented, setHasAlreadyConsented] = useState2(false);
479
+ const isMobile = useMediaQuery("(max-width: 768px)");
480
+ const responsiveStyles = useMemo(
481
+ () => ({
482
+ modal: {
483
+ width: isMobile ? "90%" : "700px"
484
+ },
485
+ content: {
486
+ margin: isMobile ? "20px" : "22px"
487
+ },
488
+ title: {
489
+ fontSize: isMobile ? "16px" : "18px"
490
+ },
491
+ subTitle: {
492
+ fontSize: isMobile ? "14px" : "16px"
493
+ },
494
+ privacyText: {
495
+ fontSize: isMobile ? "14px" : "16px"
496
+ },
497
+ optionTitle: {
498
+ fontSize: isMobile ? "14px" : "16px"
499
+ },
500
+ optionDescription: {
501
+ fontSize: isMobile ? "11px" : "12px"
502
+ },
503
+ dataElementText: {
504
+ fontSize: isMobile ? "12px" : "14px"
505
+ },
506
+ topRight: {
507
+ fontSize: isMobile ? "11px" : "12px",
508
+ padding: isMobile ? "3px 6px" : "3px 9px",
509
+ height: isMobile ? "auto" : void 0,
510
+ width: isMobile ? "auto" : void 0
511
+ },
512
+ languageItem: {
513
+ fontSize: isMobile ? "11px" : "12px",
514
+ padding: isMobile ? "6px 10px" : "8px 12px"
515
+ },
516
+ bottomSection: {
517
+ flexDirection: isMobile ? "column" : "row",
518
+ gap: isMobile ? "12px" : "29px"
519
+ },
520
+ button: {
521
+ fontSize: isMobile ? "14px" : "16px",
522
+ padding: isMobile ? "10px 20px" : "9px 45px"
523
+ },
524
+ middleSection: {
525
+ paddingRight: isMobile ? "10px" : "15px"
526
+ }
527
+ }),
528
+ [isMobile]
529
+ );
530
+ const areAllRequiredElementsChecked = useMemo(() => {
531
+ if (!content) return false;
532
+ return content.purposes.every((purpose) => {
533
+ const requiredElements = purpose.data_elements.filter(
534
+ (element) => element.required
535
+ );
536
+ return requiredElements.every(
537
+ (element) => selectedDataElements[`${purpose.uuid}-${element.uuid}`]
538
+ );
539
+ });
540
+ }, [content, selectedDataElements]);
541
+ const acceptDisabled = isSubmitting || !areAllRequiredElementsChecked;
542
+ useEffect2(() => {
543
+ if (hasAlreadyConsented) {
544
+ onAccept == null ? void 0 : onAccept();
545
+ }
546
+ }, [hasAlreadyConsented, onAccept]);
547
+ const getTranslatedText = (key, defaultText, itemId) => {
548
+ var _a2, _b2;
549
+ if (!content) return defaultText;
550
+ if (selectedLanguage === content.default_language) {
551
+ if (key === "privacy_policy_anchor_text" && content.privacy_policy_anchor_text) {
552
+ return content.privacy_policy_anchor_text;
553
+ }
554
+ return defaultText;
555
+ }
556
+ if (!content.supported_languages_and_translations) {
557
+ return defaultText;
558
+ }
559
+ const translationMap = content.supported_languages_and_translations[selectedLanguage];
560
+ if (!translationMap) {
561
+ return defaultText;
562
+ }
563
+ if (itemId) {
564
+ if (key === "purposes.name" || key === "purposes.description") {
565
+ return ((_a2 = translationMap.purposes) == null ? void 0 : _a2[itemId]) || defaultText;
566
+ }
567
+ if (key.startsWith("data_elements.")) {
568
+ return ((_b2 = translationMap.data_elements) == null ? void 0 : _b2[itemId]) || defaultText;
569
+ }
570
+ }
571
+ const value = translationMap[key];
572
+ if (typeof value === "string") {
573
+ return value;
574
+ }
575
+ return defaultText;
576
+ };
577
+ useEffect2(() => {
578
+ const fetchNotice = async () => {
579
+ setIsLoading(true);
580
+ try {
581
+ const consentContentData = await fetchConsentContent({
582
+ noticeId,
583
+ accessToken,
584
+ refreshToken,
585
+ language
586
+ });
587
+ setContent(consentContentData.detail.active_config);
588
+ if (consentContentData.detail.active_config.default_language) {
589
+ setSelectedLanguage(
590
+ consentContentData.detail.active_config.default_language
591
+ );
592
+ }
593
+ const initialCollapsedState = consentContentData.detail.active_config.purposes.reduce(
594
+ (acc, purpose) => ({
595
+ ...acc,
596
+ [purpose.uuid]: true
597
+ }),
598
+ {}
599
+ );
600
+ setCollapsedPurposes(initialCollapsedState);
601
+ const initialPurposeState = consentContentData.detail.active_config.purposes.reduce(
602
+ (acc, purpose) => {
603
+ return {
604
+ ...acc,
605
+ [purpose.uuid]: false
606
+ // Initialize all purposes as unchecked
607
+ };
608
+ },
609
+ {}
610
+ );
611
+ setSelectedPurposes(initialPurposeState);
612
+ const initialDataElementState = consentContentData.detail.active_config.purposes.reduce(
613
+ (acc, purpose) => {
614
+ purpose.data_elements.forEach((element) => {
615
+ const combinedId = `${purpose.uuid}-${element.uuid}`;
616
+ acc[combinedId] = false;
617
+ });
618
+ return acc;
619
+ },
620
+ {}
621
+ );
622
+ setSelectedDataElements(initialDataElementState);
623
+ } catch (err) {
624
+ console.error(err);
625
+ if (err.status === 409) {
626
+ setHasAlreadyConsented(true);
627
+ } else {
628
+ onError == null ? void 0 : onError(err);
629
+ }
630
+ } finally {
631
+ setIsLoading(false);
632
+ }
633
+ };
634
+ fetchNotice();
635
+ }, [noticeId, accessToken, refreshToken, language, onError]);
636
+ const togglePurposeCollapse = (purposeUuid) => {
637
+ setCollapsedPurposes((prev) => ({
638
+ ...prev,
639
+ [purposeUuid]: !prev[purposeUuid]
640
+ }));
641
+ };
642
+ const toggleLanguageDropdown = () => {
643
+ setIsLanguageDropdownOpen(!isLanguageDropdownOpen);
644
+ };
645
+ const handleLanguageSelect = (lang) => {
646
+ setSelectedLanguage(lang);
647
+ setIsLanguageDropdownOpen(false);
648
+ };
649
+ const handlePurposeCheckboxChange = (purposeUuid) => {
650
+ if (content) {
651
+ const purpose = content.purposes.find((p) => p.uuid === purposeUuid);
652
+ if (purpose) {
653
+ const newPurposeState = !selectedPurposes[purposeUuid];
654
+ setSelectedPurposes((prev) => ({
655
+ ...prev,
656
+ [purposeUuid]: newPurposeState
657
+ }));
658
+ setSelectedDataElements((prev) => {
659
+ const updated = { ...prev };
660
+ purpose.data_elements.forEach((el) => {
661
+ updated[`${purposeUuid}-${el.uuid}`] = newPurposeState;
662
+ });
663
+ return updated;
664
+ });
665
+ }
666
+ }
667
+ };
668
+ const handleDataElementCheckboxChange = (elementUuid, purposeUuid) => {
669
+ const combinedId = `${purposeUuid}-${elementUuid}`;
670
+ setSelectedDataElements((prev) => {
671
+ const newState = {
672
+ ...prev,
673
+ [combinedId]: !prev[combinedId]
674
+ };
675
+ if (content) {
676
+ const purpose = content.purposes.find((p) => p.uuid === purposeUuid);
677
+ if (purpose) {
678
+ const requiredElements = purpose.data_elements.filter(
679
+ (el) => el.required
680
+ );
681
+ const allRequiredElementsChecked = requiredElements.every(
682
+ (el) => newState[`${purposeUuid}-${el.uuid}`]
683
+ );
684
+ setSelectedPurposes((prevPurposes) => ({
685
+ ...prevPurposes,
686
+ [purposeUuid]: allRequiredElementsChecked
687
+ }));
688
+ }
689
+ }
690
+ return newState;
691
+ });
692
+ };
693
+ const handleAccept = async () => {
694
+ try {
695
+ setIsSubmitting(true);
696
+ if (content) {
697
+ await submitConsentEvent({
698
+ accessToken,
699
+ noticeUuid: content.notice_uuid,
700
+ purposes: content.purposes.map((purpose) => ({
701
+ ...purpose,
702
+ selected: selectedPurposes[purpose.uuid] || false,
703
+ data_elements: purpose.data_elements.map((element) => ({
704
+ ...element,
705
+ selected: element.required ? true : selectedDataElements[`${purpose.uuid}-${element.uuid}`] || false
706
+ }))
707
+ })),
708
+ declined: false
709
+ });
710
+ }
711
+ onAccept == null ? void 0 : onAccept();
712
+ } catch (error) {
713
+ console.error("Error submitting consent:", error);
714
+ onError == null ? void 0 : onError(error);
715
+ } finally {
716
+ setIsSubmitting(false);
717
+ }
718
+ };
719
+ const handleDecline = async () => {
720
+ onDecline == null ? void 0 : onDecline();
721
+ };
722
+ const availableLanguages = useMemo(
723
+ () => content ? [
724
+ content.default_language,
725
+ ...Object.keys(content.supported_languages_and_translations || {})
726
+ ].filter((value, index, self) => self.indexOf(value) === index) : [language],
727
+ [content, language]
728
+ );
729
+ const modalStyle = {
730
+ borderRadius: (settings == null ? void 0 : settings.borderRadius) || "8px",
731
+ backgroundColor: (settings == null ? void 0 : settings.backgroundColor) || "#ffffff"
732
+ };
733
+ const buttonStyle = {
734
+ borderRadius: (settings == null ? void 0 : settings.borderRadius) || "8px"
735
+ };
736
+ const acceptButtonStyle = {
737
+ ...buttonStyle,
738
+ backgroundColor: ((_b = (_a = settings == null ? void 0 : settings.button) == null ? void 0 : _a.accept) == null ? void 0 : _b.backgroundColor) || "#4f87ff",
739
+ color: ((_d = (_c = settings == null ? void 0 : settings.button) == null ? void 0 : _c.accept) == null ? void 0 : _d.textColor) || "#ffffff"
740
+ };
741
+ const declineButtonStyle = {
742
+ ...buttonStyle,
743
+ backgroundColor: ((_f = (_e = settings == null ? void 0 : settings.button) == null ? void 0 : _e.decline) == null ? void 0 : _f.backgroundColor) || "#ffffff",
744
+ color: ((_h = (_g = settings == null ? void 0 : settings.button) == null ? void 0 : _g.decline) == null ? void 0 : _h.textColor) || "#000000",
745
+ borderColor: (settings == null ? void 0 : settings.borderColor) || "#d0d5dd"
746
+ };
747
+ const linkStyle = {
748
+ color: ((_i = settings == null ? void 0 : settings.link) == null ? void 0 : _i.color) || "#4f87ff"
749
+ };
750
+ const languageSelectorStyle = {
751
+ borderRadius: (settings == null ? void 0 : settings.borderRadius) || "8px",
752
+ borderColor: (settings == null ? void 0 : settings.borderColor) || "#d0d5dd",
753
+ backgroundColor: ((_k = (_j = settings == null ? void 0 : settings.button) == null ? void 0 : _j.language) == null ? void 0 : _k.backgroundColor) || "#ffffff",
754
+ color: ((_m = (_l = settings == null ? void 0 : settings.button) == null ? void 0 : _l.language) == null ? void 0 : _m.textColor) || "#344054"
755
+ };
756
+ const selectedLanguageItemStyle = {
757
+ ...languageSelectorStyle,
758
+ backgroundColor: ((_o = (_n = settings == null ? void 0 : settings.button) == null ? void 0 : _n.language) == null ? void 0 : _o.selectedBackgroundColor) || "#f3f4f6",
759
+ color: ((_q = (_p = settings == null ? void 0 : settings.button) == null ? void 0 : _p.language) == null ? void 0 : _q.selectedTextColor) || "#344054",
760
+ fontWeight: 600
761
+ };
762
+ const headingStyle = {
763
+ color: (settings == null ? void 0 : settings.headingColor) || "#101828"
764
+ };
765
+ const textStyle = {
766
+ color: (settings == null ? void 0 : settings.textColor) || "#344054"
767
+ };
768
+ const sectionStyle = {
769
+ backgroundColor: (settings == null ? void 0 : settings.backgroundColor) || "#ffffff"
770
+ };
771
+ return /* @__PURE__ */ jsx(Fragment, { children: hasAlreadyConsented ? null : /* @__PURE__ */ jsxs(Fragment, { children: [
772
+ blockUI && /* @__PURE__ */ jsx("div", { style: styles.overlay, "aria-hidden": "true" }),
773
+ /* @__PURE__ */ jsx(
774
+ "div",
775
+ {
776
+ style: {
777
+ ...styles.modal,
778
+ ...responsiveStyles.modal,
779
+ ...modalStyle
780
+ },
781
+ role: "dialog",
782
+ "aria-modal": "true",
783
+ "aria-labelledby": "privacy-notice-title",
784
+ children: isLoading ? /* @__PURE__ */ jsxs("div", { style: styles.loadingContainer, children: [
785
+ /* @__PURE__ */ jsx("div", { style: styles.loadingSpinner }),
786
+ /* @__PURE__ */ jsx("p", { style: { ...styles.privacyText, ...textStyle }, children: "Loading..." })
787
+ ] }) : /* @__PURE__ */ jsxs("div", { style: { ...styles.content, ...responsiveStyles.content }, children: [
788
+ /* @__PURE__ */ jsxs("div", { style: { ...styles.topSection, ...sectionStyle }, children: [
789
+ /* @__PURE__ */ jsxs("div", { style: styles.topLeft, children: [
790
+ /* @__PURE__ */ jsx(
791
+ "img",
792
+ {
793
+ style: styles.logo,
794
+ src: (content == null ? void 0 : content.logo_url) || redacto_logo_default,
795
+ alt: "logo"
796
+ }
797
+ ),
798
+ /* @__PURE__ */ jsx(
799
+ "h2",
800
+ {
801
+ id: "privacy-notice-title",
802
+ style: {
803
+ ...styles.title,
804
+ ...responsiveStyles.title,
805
+ ...headingStyle
806
+ },
807
+ children: "Your Privacy Matters"
808
+ }
809
+ )
810
+ ] }),
811
+ /* @__PURE__ */ jsxs("div", { style: styles.languageSelectorContainer, children: [
812
+ /* @__PURE__ */ jsxs(
813
+ "button",
814
+ {
815
+ style: {
816
+ ...styles.topRight,
817
+ ...responsiveStyles.topRight,
818
+ ...languageSelectorStyle
819
+ },
820
+ onClick: toggleLanguageDropdown,
821
+ "aria-expanded": isLanguageDropdownOpen,
822
+ "aria-haspopup": "listbox",
823
+ children: [
824
+ selectedLanguage,
825
+ /* @__PURE__ */ jsx(
826
+ "svg",
827
+ {
828
+ xmlns: "http://www.w3.org/2000/svg",
829
+ width: "16",
830
+ height: "16",
831
+ viewBox: "0 0 16 16",
832
+ fill: "none",
833
+ stroke: ((_s = (_r = settings == null ? void 0 : settings.button) == null ? void 0 : _r.language) == null ? void 0 : _s.textColor) || "currentColor",
834
+ strokeWidth: "2",
835
+ strokeLinecap: "round",
836
+ strokeLinejoin: "round",
837
+ style: { flexShrink: 0, marginLeft: "4px" },
838
+ children: /* @__PURE__ */ jsx("path", { d: "M4 6l4 4 4-4" })
839
+ }
840
+ )
841
+ ]
842
+ }
843
+ ),
844
+ isLanguageDropdownOpen && /* @__PURE__ */ jsx(
845
+ "ul",
846
+ {
847
+ style: {
848
+ ...styles.languageDropdown,
849
+ ...languageSelectorStyle
850
+ },
851
+ role: "listbox",
852
+ "aria-label": "Select language",
853
+ children: availableLanguages.map((lang) => /* @__PURE__ */ jsx(
854
+ "li",
855
+ {
856
+ style: {
857
+ ...styles.languageItem,
858
+ ...responsiveStyles.languageItem,
859
+ ...selectedLanguage === lang ? selectedLanguageItemStyle : languageSelectorStyle,
860
+ listStyle: "none"
861
+ },
862
+ onClick: () => handleLanguageSelect(lang),
863
+ role: "option",
864
+ "aria-selected": selectedLanguage === lang,
865
+ tabIndex: 0,
866
+ children: lang
867
+ },
868
+ lang
869
+ ))
870
+ }
871
+ )
872
+ ] })
873
+ ] }),
874
+ /* @__PURE__ */ jsxs(
875
+ "div",
876
+ {
877
+ style: {
878
+ ...styles.middleSection,
879
+ ...responsiveStyles.middleSection,
880
+ ...sectionStyle
881
+ },
882
+ children: [
883
+ /* @__PURE__ */ jsxs(
884
+ "p",
885
+ {
886
+ style: {
887
+ ...styles.privacyText,
888
+ ...responsiveStyles.privacyText,
889
+ ...textStyle
890
+ },
891
+ children: [
892
+ content ? getTranslatedText("notice_text", content.notice_text) : "",
893
+ /* @__PURE__ */ jsx("br", {}),
894
+ "Learn more in our",
895
+ " ",
896
+ /* @__PURE__ */ jsxs(
897
+ "a",
898
+ {
899
+ style: { ...styles.link, ...linkStyle },
900
+ href: (content == null ? void 0 : content.privacy_policy_url) || "#",
901
+ target: "_blank",
902
+ rel: "noopener noreferrer",
903
+ children: [
904
+ "[",
905
+ getTranslatedText(
906
+ "privacy_policy_anchor_text",
907
+ (content == null ? void 0 : content.privacy_policy_anchor_text) || "Privacy Policy"
908
+ ),
909
+ "]"
910
+ ]
911
+ }
912
+ ),
913
+ " ",
914
+ "and",
915
+ " ",
916
+ /* @__PURE__ */ jsxs(
917
+ "a",
918
+ {
919
+ style: { ...styles.link, ...linkStyle },
920
+ href: (content == null ? void 0 : content.sub_processors_url) || "#",
921
+ target: "",
922
+ children: [
923
+ "[",
924
+ getTranslatedText(
925
+ "vendors_list_anchor_text",
926
+ (content == null ? void 0 : content.vendors_list_anchor_text) || "Vendors List"
927
+ ),
928
+ "]"
929
+ ]
930
+ }
931
+ ),
932
+ "."
933
+ ]
934
+ }
935
+ ),
936
+ /* @__PURE__ */ jsx(
937
+ "h2",
938
+ {
939
+ style: {
940
+ ...styles.subTitle,
941
+ ...responsiveStyles.subTitle,
942
+ ...headingStyle
943
+ },
944
+ children: getTranslatedText(
945
+ "purpose_section_heading",
946
+ "Manage What You Share"
947
+ )
948
+ }
949
+ ),
950
+ /* @__PURE__ */ jsx("div", { style: styles.optionsContainer, children: (_t = content == null ? void 0 : content.purposes) == null ? void 0 : _t.map((purpose) => /* @__PURE__ */ jsxs("div", { children: [
951
+ /* @__PURE__ */ jsxs("div", { style: styles.optionItem, children: [
952
+ /* @__PURE__ */ jsxs(
953
+ "div",
954
+ {
955
+ style: styles.optionLeft,
956
+ onClick: () => togglePurposeCollapse(purpose.uuid),
957
+ children: [
958
+ /* @__PURE__ */ jsx(
959
+ "div",
960
+ {
961
+ style: {
962
+ display: "flex",
963
+ alignItems: "center",
964
+ justifyContent: "center",
965
+ marginLeft: "5px"
966
+ },
967
+ children: /* @__PURE__ */ jsx(
968
+ "svg",
969
+ {
970
+ width: "7",
971
+ height: "12",
972
+ viewBox: "0 0 7 12",
973
+ fill: "none",
974
+ xmlns: "http://www.w3.org/2000/svg",
975
+ stroke: (settings == null ? void 0 : settings.headingColor) || "#323B4B",
976
+ strokeWidth: "2",
977
+ strokeLinecap: "round",
978
+ strokeLinejoin: "round",
979
+ style: {
980
+ transform: !collapsedPurposes[purpose.uuid] ? "rotate(90deg)" : "rotate(0deg)",
981
+ transition: "transform 0.3s ease"
982
+ },
983
+ children: /* @__PURE__ */ jsx("path", { d: "M1 1L6 6L1 11" })
984
+ }
985
+ )
986
+ }
987
+ ),
988
+ /* @__PURE__ */ jsxs("div", { style: styles.optionTextContainer, children: [
989
+ /* @__PURE__ */ jsx(
990
+ "h2",
991
+ {
992
+ style: {
993
+ ...styles.optionTitle,
994
+ ...responsiveStyles.optionTitle,
995
+ ...headingStyle
996
+ },
997
+ children: getTranslatedText(
998
+ `purposes.name`,
999
+ purpose.name,
1000
+ purpose.uuid
1001
+ )
1002
+ }
1003
+ ),
1004
+ /* @__PURE__ */ jsx(
1005
+ "h3",
1006
+ {
1007
+ style: {
1008
+ ...styles.optionDescription,
1009
+ ...responsiveStyles.optionDescription,
1010
+ ...textStyle
1011
+ },
1012
+ children: getTranslatedText(
1013
+ `purposes.description`,
1014
+ purpose.description,
1015
+ purpose.uuid
1016
+ )
1017
+ }
1018
+ )
1019
+ ] })
1020
+ ]
1021
+ }
1022
+ ),
1023
+ /* @__PURE__ */ jsx(
1024
+ "input",
1025
+ {
1026
+ style: styles.checkboxLarge,
1027
+ type: "checkbox",
1028
+ checked: selectedPurposes[purpose.uuid] || false,
1029
+ onChange: () => handlePurposeCheckboxChange(purpose.uuid)
1030
+ }
1031
+ )
1032
+ ] }),
1033
+ !collapsedPurposes[purpose.uuid] && purpose.data_elements && purpose.data_elements.length > 0 && /* @__PURE__ */ jsx("div", { style: styles.dataElementsContainer, children: purpose.data_elements.map((dataElement) => /* @__PURE__ */ jsxs(
1034
+ "div",
1035
+ {
1036
+ style: styles.dataElementItem,
1037
+ children: [
1038
+ /* @__PURE__ */ jsxs(
1039
+ "span",
1040
+ {
1041
+ style: {
1042
+ ...styles.dataElementText,
1043
+ ...responsiveStyles.dataElementText,
1044
+ ...textStyle
1045
+ },
1046
+ children: [
1047
+ getTranslatedText(
1048
+ `data_elements.name`,
1049
+ dataElement.name,
1050
+ dataElement.uuid
1051
+ ),
1052
+ dataElement.required && /* @__PURE__ */ jsx(
1053
+ "span",
1054
+ {
1055
+ style: {
1056
+ color: "red",
1057
+ marginLeft: "4px"
1058
+ },
1059
+ children: "*"
1060
+ }
1061
+ )
1062
+ ]
1063
+ }
1064
+ ),
1065
+ /* @__PURE__ */ jsx(
1066
+ "input",
1067
+ {
1068
+ style: styles.checkboxSmall,
1069
+ type: "checkbox",
1070
+ checked: selectedDataElements[`${purpose.uuid}-${dataElement.uuid}`] || false,
1071
+ onChange: () => handleDataElementCheckboxChange(
1072
+ dataElement.uuid,
1073
+ purpose.uuid
1074
+ )
1075
+ }
1076
+ )
1077
+ ]
1078
+ },
1079
+ dataElement.uuid
1080
+ )) })
1081
+ ] }, purpose.uuid)) })
1082
+ ]
1083
+ }
1084
+ ),
1085
+ /* @__PURE__ */ jsxs(
1086
+ "div",
1087
+ {
1088
+ style: {
1089
+ ...styles.bottomSection,
1090
+ ...responsiveStyles.bottomSection,
1091
+ ...sectionStyle
1092
+ },
1093
+ children: [
1094
+ /* @__PURE__ */ jsx(
1095
+ "button",
1096
+ {
1097
+ style: {
1098
+ ...styles.button,
1099
+ ...responsiveStyles.button,
1100
+ ...styles.acceptButton,
1101
+ ...acceptButtonStyle,
1102
+ opacity: acceptDisabled ? 0.5 : 1
1103
+ },
1104
+ onClick: handleAccept,
1105
+ disabled: acceptDisabled,
1106
+ children: isSubmitting ? "Processing..." : getTranslatedText(
1107
+ "confirm_button_text",
1108
+ (content == null ? void 0 : content.confirm_button_text) || "Accept"
1109
+ )
1110
+ }
1111
+ ),
1112
+ /* @__PURE__ */ jsx(
1113
+ "button",
1114
+ {
1115
+ style: {
1116
+ ...styles.button,
1117
+ ...responsiveStyles.button,
1118
+ ...styles.cancelButton,
1119
+ ...declineButtonStyle
1120
+ },
1121
+ onClick: handleDecline,
1122
+ disabled: isSubmitting,
1123
+ children: isSubmitting ? "Processing..." : getTranslatedText(
1124
+ "decline_button_text",
1125
+ (content == null ? void 0 : content.decline_button_text) || "Decline"
1126
+ )
1127
+ }
1128
+ )
1129
+ ]
1130
+ }
1131
+ )
1132
+ ] })
1133
+ }
1134
+ )
1135
+ ] }) });
1136
+ };
1137
+ export {
1138
+ RedactoNoticeConsent
1139
+ };