@churchapps/apphelper 0.0.9 → 0.0.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (193) hide show
  1. package/.eslintignore +3 -3
  2. package/.eslintrc.json +22 -22
  3. package/LICENSE +21 -21
  4. package/README.md +16 -16
  5. package/dist/components/ExportLink.d.ts.map +1 -1
  6. package/dist/components/ExportLink.js +3 -2
  7. package/dist/components/ExportLink.js.map +1 -1
  8. package/dist/components/markdownEditor/editor.css +787 -787
  9. package/dist/components/markdownEditor/images/icons/arrow-clockwise.svg +3 -3
  10. package/dist/components/markdownEditor/images/icons/arrow-counterclockwise.svg +3 -3
  11. package/dist/components/markdownEditor/images/icons/chat-square-quote.svg +3 -3
  12. package/dist/components/markdownEditor/images/icons/chevron-down.svg +2 -2
  13. package/dist/components/markdownEditor/images/icons/code.svg +2 -2
  14. package/dist/components/markdownEditor/images/icons/journal-code.svg +4 -4
  15. package/dist/components/markdownEditor/images/icons/journal-text.svg +4 -4
  16. package/dist/components/markdownEditor/images/icons/justify.svg +2 -2
  17. package/dist/components/markdownEditor/images/icons/link.svg +3 -3
  18. package/dist/components/markdownEditor/images/icons/list-ol.svg +3 -3
  19. package/dist/components/markdownEditor/images/icons/list-ul.svg +2 -2
  20. package/dist/components/markdownEditor/images/icons/pencil-fill.svg +2 -2
  21. package/dist/components/markdownEditor/images/icons/text-center.svg +2 -2
  22. package/dist/components/markdownEditor/images/icons/text-left.svg +2 -2
  23. package/dist/components/markdownEditor/images/icons/text-paragraph.svg +2 -2
  24. package/dist/components/markdownEditor/images/icons/text-right.svg +2 -2
  25. package/dist/components/markdownEditor/images/icons/type-bold.svg +2 -2
  26. package/dist/components/markdownEditor/images/icons/type-h1.svg +2 -2
  27. package/dist/components/markdownEditor/images/icons/type-h2.svg +2 -2
  28. package/dist/components/markdownEditor/images/icons/type-h3.svg +2 -2
  29. package/dist/components/markdownEditor/images/icons/type-h4.svg +12 -12
  30. package/dist/components/markdownEditor/images/icons/type-italic.svg +2 -2
  31. package/dist/components/markdownEditor/images/icons/type-strikethrough.svg +2 -2
  32. package/dist/components/markdownEditor/images/icons/type-underline.svg +2 -2
  33. package/dist/components/material/SiteWrapper.js +2 -2
  34. package/dist/components/material/SiteWrapper.js.map +1 -1
  35. package/dist/helpers/SocketHelper.d.ts +3 -3
  36. package/dist/helpers/SocketHelper.d.ts.map +1 -1
  37. package/dist/helpers/SocketHelper.js +6 -6
  38. package/dist/helpers/SocketHelper.js.map +1 -1
  39. package/dist/interfaces/Messaging.d.ts +1 -2
  40. package/dist/interfaces/Messaging.d.ts.map +1 -1
  41. package/package.json +84 -84
  42. package/src/components/CreatePerson.tsx +80 -80
  43. package/src/components/DisplayBox.tsx +68 -68
  44. package/src/components/ErrorMessages.tsx +26 -26
  45. package/src/components/ExportLink.tsx +67 -66
  46. package/src/components/FloatingSupport.tsx +16 -16
  47. package/src/components/FormSubmissionEdit.tsx +120 -120
  48. package/src/components/HelpIcon.tsx +10 -10
  49. package/src/components/ImageEditor.tsx +126 -126
  50. package/src/components/InputBox.tsx +73 -73
  51. package/src/components/Loading.tsx +29 -29
  52. package/src/components/PersonAdd.tsx +75 -75
  53. package/src/components/QuestionEdit.tsx +62 -62
  54. package/src/components/SmallButton.tsx +39 -39
  55. package/src/components/SupportModal.tsx +26 -26
  56. package/src/components/TabPanel.tsx +34 -34
  57. package/src/components/gallery/GalleryModal.tsx +102 -102
  58. package/src/components/gallery/StockPhotos.tsx +74 -74
  59. package/src/components/gallery/index.ts +1 -1
  60. package/src/components/index.tsx +23 -23
  61. package/src/components/markdownEditor/Editor.tsx +132 -132
  62. package/src/components/markdownEditor/MarkdownEditor.tsx +16 -16
  63. package/src/components/markdownEditor/MarkdownModal.tsx +46 -46
  64. package/src/components/markdownEditor/MarkdownPreview.tsx +14 -14
  65. package/src/components/markdownEditor/editor.css +787 -787
  66. package/src/components/markdownEditor/images/icons/arrow-clockwise.svg +3 -3
  67. package/src/components/markdownEditor/images/icons/arrow-counterclockwise.svg +3 -3
  68. package/src/components/markdownEditor/images/icons/chat-square-quote.svg +3 -3
  69. package/src/components/markdownEditor/images/icons/chevron-down.svg +2 -2
  70. package/src/components/markdownEditor/images/icons/code.svg +2 -2
  71. package/src/components/markdownEditor/images/icons/journal-code.svg +4 -4
  72. package/src/components/markdownEditor/images/icons/journal-text.svg +4 -4
  73. package/src/components/markdownEditor/images/icons/justify.svg +2 -2
  74. package/src/components/markdownEditor/images/icons/link.svg +3 -3
  75. package/src/components/markdownEditor/images/icons/list-ol.svg +3 -3
  76. package/src/components/markdownEditor/images/icons/list-ul.svg +2 -2
  77. package/src/components/markdownEditor/images/icons/pencil-fill.svg +2 -2
  78. package/src/components/markdownEditor/images/icons/text-center.svg +2 -2
  79. package/src/components/markdownEditor/images/icons/text-left.svg +2 -2
  80. package/src/components/markdownEditor/images/icons/text-paragraph.svg +2 -2
  81. package/src/components/markdownEditor/images/icons/text-right.svg +2 -2
  82. package/src/components/markdownEditor/images/icons/type-bold.svg +2 -2
  83. package/src/components/markdownEditor/images/icons/type-h1.svg +2 -2
  84. package/src/components/markdownEditor/images/icons/type-h2.svg +2 -2
  85. package/src/components/markdownEditor/images/icons/type-h3.svg +2 -2
  86. package/src/components/markdownEditor/images/icons/type-h4.svg +12 -12
  87. package/src/components/markdownEditor/images/icons/type-italic.svg +2 -2
  88. package/src/components/markdownEditor/images/icons/type-strikethrough.svg +2 -2
  89. package/src/components/markdownEditor/images/icons/type-underline.svg +2 -2
  90. package/src/components/markdownEditor/index.ts +2 -2
  91. package/src/components/markdownEditor/plugins/AutoLinkPlugin.tsx +35 -35
  92. package/src/components/markdownEditor/plugins/ControlledEditorPlugin.tsx +24 -24
  93. package/src/components/markdownEditor/plugins/ListMaxIndentLevelPlugin.tsx +68 -68
  94. package/src/components/markdownEditor/plugins/MarkdownTransformers.ts +106 -106
  95. package/src/components/markdownEditor/plugins/ReadOnlyPlugin.tsx +15 -15
  96. package/src/components/markdownEditor/plugins/ToolbarPlugin.tsx +401 -401
  97. package/src/components/markdownEditor/plugins/customLink/CustomLinkNode.tsx +224 -224
  98. package/src/components/markdownEditor/plugins/customLink/CustomLinkNodePlugin.tsx +32 -32
  99. package/src/components/markdownEditor/plugins/customLink/CustomLinkNodeTransformer.tsx +102 -102
  100. package/src/components/markdownEditor/plugins/customLink/FloatingLinkEditor.tsx +243 -243
  101. package/src/components/markdownEditor/plugins/customLink/FloatingLinkEditor.types.ts +11 -11
  102. package/src/components/markdownEditor/plugins/emoji/EmojiNode.tsx +95 -95
  103. package/src/components/markdownEditor/plugins/emoji/EmojiNodeTransform.ts +41 -41
  104. package/src/components/markdownEditor/plugins/emoji/EmojiPickerPlugin.tsx +152 -152
  105. package/src/components/markdownEditor/plugins/emoji/EmojisPlugin.tsx +65 -65
  106. package/src/components/markdownEditor/plugins/index.ts +6 -6
  107. package/src/components/markdownEditor/theme.ts +65 -65
  108. package/src/components/material/AppList.tsx +20 -20
  109. package/src/components/material/ChurchList.tsx +22 -22
  110. package/src/components/material/NavItem.tsx +41 -41
  111. package/src/components/material/NewPrivateMessage.tsx +103 -103
  112. package/src/components/material/PrivateMessageDetails.tsx +23 -23
  113. package/src/components/material/PrivateMessages.tsx +87 -87
  114. package/src/components/material/SiteWrapper.tsx +140 -140
  115. package/src/components/material/UserMenu.tsx +141 -141
  116. package/src/components/material/iconPicker/IconNamesList.ts +2240 -2240
  117. package/src/components/material/iconPicker/IconPicker.tsx +153 -153
  118. package/src/components/material/index.tsx +6 -6
  119. package/src/components/notes/AddNote.tsx +90 -90
  120. package/src/components/notes/Conversation.tsx +82 -82
  121. package/src/components/notes/Conversations.tsx +58 -58
  122. package/src/components/notes/NewConversation.tsx +78 -78
  123. package/src/components/notes/Note.tsx +44 -44
  124. package/src/components/notes/Notes.tsx +52 -52
  125. package/src/components/notes/index.ts +5 -5
  126. package/src/components/reporting/ChartReport.tsx +98 -98
  127. package/src/components/reporting/ReportFilter.tsx +54 -54
  128. package/src/components/reporting/ReportFilterField.tsx +160 -160
  129. package/src/components/reporting/ReportOutput.tsx +79 -79
  130. package/src/components/reporting/ReportWithFilter.tsx +70 -70
  131. package/src/components/reporting/TableReport.tsx +57 -57
  132. package/src/components/reporting/TreeReport.tsx +111 -111
  133. package/src/components/reporting/index.ts +4 -4
  134. package/src/donationComponents/DonationPage.tsx +136 -136
  135. package/src/donationComponents/components/BankForm.tsx +159 -159
  136. package/src/donationComponents/components/CardForm.tsx +104 -104
  137. package/src/donationComponents/components/DonationForm.tsx +235 -235
  138. package/src/donationComponents/components/FundDonation.tsx +49 -49
  139. package/src/donationComponents/components/FundDonations.tsx +39 -39
  140. package/src/donationComponents/components/NonAuthDonation.tsx +31 -31
  141. package/src/donationComponents/components/NonAuthDonationInner.tsx +259 -259
  142. package/src/donationComponents/components/PaymentMethods.tsx +135 -135
  143. package/src/donationComponents/components/RecurringDonations.tsx +121 -121
  144. package/src/donationComponents/components/RecurringDonationsEdit.tsx +93 -93
  145. package/src/donationComponents/components/index.tsx +9 -9
  146. package/src/donationComponents/index.ts +3 -3
  147. package/src/donationComponents/modals/DonationPreviewModal.tsx +66 -66
  148. package/src/helpers/AnalyticsHelper.ts +33 -33
  149. package/src/helpers/ApiHelper.ts +125 -125
  150. package/src/helpers/AppearanceHelper.ts +69 -69
  151. package/src/helpers/ArrayHelper.ts +81 -81
  152. package/src/helpers/CommonEnvironmentHelper.ts +80 -80
  153. package/src/helpers/CurrencyHelper.ts +10 -10
  154. package/src/helpers/DateHelper.ts +108 -108
  155. package/src/helpers/DonationHelper.ts +26 -26
  156. package/src/helpers/ErrorHelper.ts +36 -36
  157. package/src/helpers/EventHelper.ts +52 -52
  158. package/src/helpers/FileHelper.ts +31 -31
  159. package/src/helpers/PersonHelper.ts +60 -60
  160. package/src/helpers/SocketHelper.ts +76 -76
  161. package/src/helpers/Themes.ts +14 -14
  162. package/src/helpers/UniqueIdHelper.ts +36 -36
  163. package/src/helpers/UserHelper.ts +59 -59
  164. package/src/helpers/createEmotionCache.ts +17 -17
  165. package/src/helpers/index.ts +18 -18
  166. package/src/hooks/index.ts +1 -1
  167. package/src/hooks/useMountedState.ts +16 -16
  168. package/src/index.ts +6 -6
  169. package/src/interfaces/Access.ts +24 -24
  170. package/src/interfaces/Attendance.ts +8 -8
  171. package/src/interfaces/Content.ts +10 -10
  172. package/src/interfaces/Doing.ts +24 -24
  173. package/src/interfaces/Donation.ts +45 -45
  174. package/src/interfaces/Error.ts +17 -17
  175. package/src/interfaces/Membership.ts +51 -51
  176. package/src/interfaces/Messaging.ts +20 -20
  177. package/src/interfaces/Permissions.ts +68 -68
  178. package/src/interfaces/Reporting.ts +7 -7
  179. package/src/interfaces/UserContextInterface.ts +13 -13
  180. package/src/interfaces/index.ts +13 -13
  181. package/src/pageComponents/LoginPage.tsx +244 -244
  182. package/src/pageComponents/LogoutPage.tsx +28 -28
  183. package/src/pageComponents/components/Forgot.tsx +79 -79
  184. package/src/pageComponents/components/Login.tsx +54 -54
  185. package/src/pageComponents/components/LoginSetPassword.tsx +63 -63
  186. package/src/pageComponents/components/Register.tsx +107 -107
  187. package/src/pageComponents/components/SelectChurchModal.tsx +41 -41
  188. package/src/pageComponents/components/SelectChurchRegister.tsx +88 -88
  189. package/src/pageComponents/components/SelectChurchSearch.tsx +69 -69
  190. package/src/pageComponents/components/SelectableChurch.tsx +38 -38
  191. package/src/pageComponents/index.ts +3 -3
  192. package/tsconfig.json +34 -34
  193. package/tslint.json +14 -14
@@ -1,73 +1,73 @@
1
- import React from "react";
2
- import { Paper, Box, Typography, Stack, styled, Button, Icon, PaperProps } from "@mui/material";
3
- import { HelpIcon } from "./HelpIcon";
4
-
5
- interface Props {
6
- id?: string;
7
- children?: React.ReactNode;
8
- headerIcon?: string;
9
- headerText?: string;
10
- help?: string;
11
- saveText?: string;
12
- headerActionContent?: React.ReactNode;
13
- cancelFunction?: () => void;
14
- deleteFunction?: () => void;
15
- saveFunction: () => void;
16
- "data-cy"?: string;
17
- className?: string;
18
- isSubmitting?: boolean;
19
- ariaLabelDelete?: string;
20
- ariaLabelSave?: string;
21
- saveButtonType?: "submit" | "button";
22
- mainContainerCssProps?: PaperProps;
23
- }
24
-
25
- const CustomContextBox = styled(Box)({
26
- marginTop: 10,
27
- overflowX: "hidden",
28
- "& p": { color: "#666" },
29
- "& label": { color: "#999" },
30
- "& ul": { paddingLeft: 15 },
31
- "& li": {
32
- marginBottom: 10,
33
- "& i": { marginRight: 5 }
34
- },
35
- "& td": {
36
- "& i": { marginRight: 5 }
37
- }
38
- })
39
-
40
- export function InputBox({ mainContainerCssProps = {}, ...props }: Props) {
41
- let buttons = [];
42
- if (props.cancelFunction) buttons.push(<Button key="cancel" onClick={props.cancelFunction} color="warning" sx={{ "&:focus": { outline: "none" } }}>Cancel</Button>);
43
- if (props.deleteFunction) buttons.push(<Button key="delete" id="delete" variant="outlined" aria-label={props.ariaLabelDelete} onClick={props.deleteFunction} color="error" sx={{ "&:focus": { outline: "none" } }}>Delete</Button>);
44
- if (props.saveFunction) buttons.push(<Button key="save" type={props.saveButtonType || "button"} variant="contained" disableElevation aria-label={props.ariaLabelSave} onClick={props.saveFunction} disabled={props.isSubmitting} sx={{ "&:focus": { outline: "none" } }}>{props.saveText || "save"}</Button>);
45
-
46
- let classNames = ["inputBox"];
47
- if (props.className) {
48
- classNames.push(props.className);
49
- }
50
-
51
- return (
52
- <Paper id={props.id} sx={{ padding: 2, marginBottom: 4 }} data-cy={props["data-cy"]} {...mainContainerCssProps}>
53
- {props.help && <HelpIcon article={props.help} />}
54
- <Box sx={{ display: "flex", justifyContent: "space-between", alignItems: "center", position: "relative" }} data-cy="header">
55
- <Box sx={{ display: "flex", alignItems: "center" }}>
56
- {props.headerIcon && <Icon sx={{ color: "#1976d2" }}>{props.headerIcon}</Icon>}
57
- {props.headerText && (
58
- <Typography component="h2" sx={{ display: "inline-block", marginLeft: props.headerIcon ? 1 : 0 }} variant="h6" color="primary">
59
- {props.headerText}
60
- </Typography>
61
- )}
62
- </Box>
63
- <Box>
64
- {props.headerActionContent}
65
- </Box>
66
- </Box>
67
- <CustomContextBox>{props.children}</CustomContextBox>
68
- <Stack direction="row" sx={{ marginTop: 1 }} spacing={1} justifyContent="end">
69
- {buttons}
70
- </Stack>
71
- </Paper>
72
- );
73
- }
1
+ import React from "react";
2
+ import { Paper, Box, Typography, Stack, styled, Button, Icon, PaperProps } from "@mui/material";
3
+ import { HelpIcon } from "./HelpIcon";
4
+
5
+ interface Props {
6
+ id?: string;
7
+ children?: React.ReactNode;
8
+ headerIcon?: string;
9
+ headerText?: string;
10
+ help?: string;
11
+ saveText?: string;
12
+ headerActionContent?: React.ReactNode;
13
+ cancelFunction?: () => void;
14
+ deleteFunction?: () => void;
15
+ saveFunction: () => void;
16
+ "data-cy"?: string;
17
+ className?: string;
18
+ isSubmitting?: boolean;
19
+ ariaLabelDelete?: string;
20
+ ariaLabelSave?: string;
21
+ saveButtonType?: "submit" | "button";
22
+ mainContainerCssProps?: PaperProps;
23
+ }
24
+
25
+ const CustomContextBox = styled(Box)({
26
+ marginTop: 10,
27
+ overflowX: "hidden",
28
+ "& p": { color: "#666" },
29
+ "& label": { color: "#999" },
30
+ "& ul": { paddingLeft: 15 },
31
+ "& li": {
32
+ marginBottom: 10,
33
+ "& i": { marginRight: 5 }
34
+ },
35
+ "& td": {
36
+ "& i": { marginRight: 5 }
37
+ }
38
+ })
39
+
40
+ export function InputBox({ mainContainerCssProps = {}, ...props }: Props) {
41
+ let buttons = [];
42
+ if (props.cancelFunction) buttons.push(<Button key="cancel" onClick={props.cancelFunction} color="warning" sx={{ "&:focus": { outline: "none" } }}>Cancel</Button>);
43
+ if (props.deleteFunction) buttons.push(<Button key="delete" id="delete" variant="outlined" aria-label={props.ariaLabelDelete} onClick={props.deleteFunction} color="error" sx={{ "&:focus": { outline: "none" } }}>Delete</Button>);
44
+ if (props.saveFunction) buttons.push(<Button key="save" type={props.saveButtonType || "button"} variant="contained" disableElevation aria-label={props.ariaLabelSave} onClick={props.saveFunction} disabled={props.isSubmitting} sx={{ "&:focus": { outline: "none" } }}>{props.saveText || "save"}</Button>);
45
+
46
+ let classNames = ["inputBox"];
47
+ if (props.className) {
48
+ classNames.push(props.className);
49
+ }
50
+
51
+ return (
52
+ <Paper id={props.id} sx={{ padding: 2, marginBottom: 4 }} data-cy={props["data-cy"]} {...mainContainerCssProps}>
53
+ {props.help && <HelpIcon article={props.help} />}
54
+ <Box sx={{ display: "flex", justifyContent: "space-between", alignItems: "center", position: "relative" }} data-cy="header">
55
+ <Box sx={{ display: "flex", alignItems: "center" }}>
56
+ {props.headerIcon && <Icon sx={{ color: "#1976d2" }}>{props.headerIcon}</Icon>}
57
+ {props.headerText && (
58
+ <Typography component="h2" sx={{ display: "inline-block", marginLeft: props.headerIcon ? 1 : 0 }} variant="h6" color="primary">
59
+ {props.headerText}
60
+ </Typography>
61
+ )}
62
+ </Box>
63
+ <Box>
64
+ {props.headerActionContent}
65
+ </Box>
66
+ </Box>
67
+ <CustomContextBox>{props.children}</CustomContextBox>
68
+ <Stack direction="row" sx={{ marginTop: 1 }} spacing={1} justifyContent="end">
69
+ {buttons}
70
+ </Stack>
71
+ </Paper>
72
+ );
73
+ }
@@ -1,29 +1,29 @@
1
- import React from "react";
2
- import { Dots } from "react-activity";
3
- import "react-activity/dist/library.css";
4
-
5
- interface Props { size?: "sm" | "md" | "lg", loadingText?: string, color?: string }
6
-
7
- export const Loading: React.FC<Props> = (props) => {
8
-
9
- const getContents = () => {
10
- const text = (props.loadingText) ? props.loadingText : "Loading"
11
- const color = (props.color) ? props.color : "#222"
12
- let result = <><Dots speed={0.8} animating={true} size={32} color={color} /><p style={{ color: color }}>{text}</p></>
13
- switch (props.size) {
14
- case "sm":
15
- result = <><Dots speed={0.8} animating={true} size={20} color={color} /><p style={{ fontSize: 12, color: color }}>{text}</p></>
16
- break;
17
- case "lg":
18
- result = <><Dots speed={0.8} animating={true} size={48} color={color} /><p style={{ fontSize: 30, color: color }}>{text}</p></>
19
- break;
20
- }
21
- return result;
22
- }
23
-
24
- return (
25
- <div style={{ textAlign: "center", fontFamily: "Roboto" }}>
26
- {getContents()}
27
- </div>
28
- )
29
- }
1
+ import React from "react";
2
+ import { Dots } from "react-activity";
3
+ import "react-activity/dist/library.css";
4
+
5
+ interface Props { size?: "sm" | "md" | "lg", loadingText?: string, color?: string }
6
+
7
+ export const Loading: React.FC<Props> = (props) => {
8
+
9
+ const getContents = () => {
10
+ const text = (props.loadingText) ? props.loadingText : "Loading"
11
+ const color = (props.color) ? props.color : "#222"
12
+ let result = <><Dots speed={0.8} animating={true} size={32} color={color} /><p style={{ color: color }}>{text}</p></>
13
+ switch (props.size) {
14
+ case "sm":
15
+ result = <><Dots speed={0.8} animating={true} size={20} color={color} /><p style={{ fontSize: 12, color: color }}>{text}</p></>
16
+ break;
17
+ case "lg":
18
+ result = <><Dots speed={0.8} animating={true} size={48} color={color} /><p style={{ fontSize: 30, color: color }}>{text}</p></>
19
+ break;
20
+ }
21
+ return result;
22
+ }
23
+
24
+ return (
25
+ <div style={{ textAlign: "center", fontFamily: "Roboto" }}>
26
+ {getContents()}
27
+ </div>
28
+ )
29
+ }
@@ -1,75 +1,75 @@
1
- import React, { useState } from "react";
2
- import { ApiHelper } from "../helpers";
3
- import { PersonInterface } from "../interfaces"
4
- import { TextField, Button, Table, TableBody, TableRow, TableCell } from "@mui/material";
5
- import { SmallButton } from "./SmallButton";
6
- import { CreatePerson } from "./CreatePerson";
7
-
8
- interface Props {
9
- addFunction: (person: PersonInterface) => void;
10
- person?: PersonInterface;
11
- getPhotoUrl: (person: PersonInterface) => string;
12
- searchClicked?: () => void;
13
- filterList?: string[];
14
- includeEmail?: boolean;
15
- actionLabel?: string;
16
- showCreatePersonOnNotFound?: boolean;
17
- }
18
-
19
- export const PersonAdd: React.FC<Props> = ({ addFunction, getPhotoUrl, searchClicked, filterList = [], includeEmail = false, actionLabel, showCreatePersonOnNotFound = false }) => {
20
- const [searchResults, setSearchResults] = useState<PersonInterface[]>([]);
21
- const [searchText, setSearchText] = useState("");
22
- const [hasSearched, setHasSearched] = useState<boolean>(false);
23
-
24
- const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => { e.preventDefault(); setHasSearched(false); setSearchText(e.currentTarget.value); }
25
- const handleKeyDown = (e: React.KeyboardEvent<any>) => { if (e.key === "Enter") { e.preventDefault(); handleSearch(null); } }
26
-
27
- const handleSearch = (e: React.MouseEvent) => {
28
- if (e !== null) e.preventDefault();
29
- let term = searchText.trim();
30
- ApiHelper.post("/people/search", { term: term }, "MembershipApi")
31
- .then((data: PersonInterface[]) => {
32
- setHasSearched(true);
33
- const filteredResult = data.filter(s => !filterList.includes(s.id))
34
- setSearchResults(filteredResult);
35
- if (searchClicked) {
36
- searchClicked();
37
- }
38
- });
39
- }
40
- const handleAdd = (person: PersonInterface) => {
41
- let sr: PersonInterface[] = [...searchResults];
42
- const idx = sr.indexOf(person);
43
- sr.splice(idx, 1);
44
- setSearchResults(sr);
45
- addFunction(person);
46
- }
47
-
48
- //<button className="text-success no-default-style" aria-label="addPerson" data-index={i} onClick={handleAdd}><Icon>person</Icon> Add</button>
49
- let rows = [];
50
- for (let i = 0; i < searchResults.length; i++) {
51
- const sr = searchResults[i];
52
-
53
- rows.push(
54
- <TableRow key={sr.id}>
55
- <TableCell><img src={getPhotoUrl(sr)} alt="avatar" /></TableCell>
56
- <TableCell>{sr.name.display}{includeEmail && (<><br /><i style={{ color: "#999" }}>{sr.contactInfo.email}</i></>)}</TableCell>
57
- <TableCell>
58
- <SmallButton color="success" icon="person" text={actionLabel || "Add"} ariaLabel="addPerson" onClick={() => handleAdd(sr)} />
59
- </TableCell>
60
- </TableRow>
61
- );
62
- }
63
-
64
- return (
65
- <>
66
- <TextField fullWidth name="personAddText" label="Person" value={searchText} onChange={handleChange} onKeyDown={handleKeyDown}
67
- InputProps={{ endAdornment: <Button variant="contained" id="searchButton" data-cy="search-button" onClick={handleSearch}>Search</Button> }}
68
- />
69
- {showCreatePersonOnNotFound && hasSearched && searchText && searchResults.length === 0 && (
70
- <CreatePerson navigateOnCreate={false} onCreate={person => { setSearchText(""); setSearchResults([person]) }} />
71
- )}
72
- <Table size="small" id="householdMemberAddTable"><TableBody>{rows}</TableBody></Table>
73
- </>
74
- );
75
- }
1
+ import React, { useState } from "react";
2
+ import { ApiHelper } from "../helpers";
3
+ import { PersonInterface } from "../interfaces"
4
+ import { TextField, Button, Table, TableBody, TableRow, TableCell } from "@mui/material";
5
+ import { SmallButton } from "./SmallButton";
6
+ import { CreatePerson } from "./CreatePerson";
7
+
8
+ interface Props {
9
+ addFunction: (person: PersonInterface) => void;
10
+ person?: PersonInterface;
11
+ getPhotoUrl: (person: PersonInterface) => string;
12
+ searchClicked?: () => void;
13
+ filterList?: string[];
14
+ includeEmail?: boolean;
15
+ actionLabel?: string;
16
+ showCreatePersonOnNotFound?: boolean;
17
+ }
18
+
19
+ export const PersonAdd: React.FC<Props> = ({ addFunction, getPhotoUrl, searchClicked, filterList = [], includeEmail = false, actionLabel, showCreatePersonOnNotFound = false }) => {
20
+ const [searchResults, setSearchResults] = useState<PersonInterface[]>([]);
21
+ const [searchText, setSearchText] = useState("");
22
+ const [hasSearched, setHasSearched] = useState<boolean>(false);
23
+
24
+ const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => { e.preventDefault(); setHasSearched(false); setSearchText(e.currentTarget.value); }
25
+ const handleKeyDown = (e: React.KeyboardEvent<any>) => { if (e.key === "Enter") { e.preventDefault(); handleSearch(null); } }
26
+
27
+ const handleSearch = (e: React.MouseEvent) => {
28
+ if (e !== null) e.preventDefault();
29
+ let term = searchText.trim();
30
+ ApiHelper.post("/people/search", { term: term }, "MembershipApi")
31
+ .then((data: PersonInterface[]) => {
32
+ setHasSearched(true);
33
+ const filteredResult = data.filter(s => !filterList.includes(s.id))
34
+ setSearchResults(filteredResult);
35
+ if (searchClicked) {
36
+ searchClicked();
37
+ }
38
+ });
39
+ }
40
+ const handleAdd = (person: PersonInterface) => {
41
+ let sr: PersonInterface[] = [...searchResults];
42
+ const idx = sr.indexOf(person);
43
+ sr.splice(idx, 1);
44
+ setSearchResults(sr);
45
+ addFunction(person);
46
+ }
47
+
48
+ //<button className="text-success no-default-style" aria-label="addPerson" data-index={i} onClick={handleAdd}><Icon>person</Icon> Add</button>
49
+ let rows = [];
50
+ for (let i = 0; i < searchResults.length; i++) {
51
+ const sr = searchResults[i];
52
+
53
+ rows.push(
54
+ <TableRow key={sr.id}>
55
+ <TableCell><img src={getPhotoUrl(sr)} alt="avatar" /></TableCell>
56
+ <TableCell>{sr.name.display}{includeEmail && (<><br /><i style={{ color: "#999" }}>{sr.contactInfo.email}</i></>)}</TableCell>
57
+ <TableCell>
58
+ <SmallButton color="success" icon="person" text={actionLabel || "Add"} ariaLabel="addPerson" onClick={() => handleAdd(sr)} />
59
+ </TableCell>
60
+ </TableRow>
61
+ );
62
+ }
63
+
64
+ return (
65
+ <>
66
+ <TextField fullWidth name="personAddText" label="Person" value={searchText} onChange={handleChange} onKeyDown={handleKeyDown}
67
+ InputProps={{ endAdornment: <Button variant="contained" id="searchButton" data-cy="search-button" onClick={handleSearch}>Search</Button> }}
68
+ />
69
+ {showCreatePersonOnNotFound && hasSearched && searchText && searchResults.length === 0 && (
70
+ <CreatePerson navigateOnCreate={false} onCreate={person => { setSearchText(""); setSearchResults([person]) }} />
71
+ )}
72
+ <Table size="small" id="householdMemberAddTable"><TableBody>{rows}</TableBody></Table>
73
+ </>
74
+ );
75
+ }
@@ -1,62 +1,62 @@
1
- import React from "react";
2
- import { AnswerInterface, QuestionInterface } from "../interfaces";
3
- import { Select, MenuItem, SelectChangeEvent, FormControl, InputLabel, TextField } from "@mui/material";
4
-
5
- interface Props {
6
- answer: AnswerInterface
7
- question: QuestionInterface,
8
- noBackground?: boolean,
9
- changeFunction: (questionId: string, value: string) => void
10
- }
11
-
12
- export const QuestionEdit: React.FC<Props> = ({noBackground = false, ...props}) => {
13
- const handleChange = (e: React.ChangeEvent<HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement> | SelectChangeEvent<string>) => {
14
- props.changeFunction(props.question.id, e.target.value);
15
- }
16
- let q = props.question;
17
-
18
- if (q.fieldType === "Heading") return <h5>{q.title}</h5>;
19
- else {
20
- let input = null;
21
- let choiceOptions = [];
22
- if (q.choices !== undefined && q.choices !== null) {
23
- for (let i = 0; i < q.choices.length; i++) choiceOptions.push(<MenuItem key={i} value={q.choices[i].value}>{q.choices[i].text}</MenuItem>);
24
- }
25
-
26
- let answerValue = (props.answer === null) ? "" : props.answer.value
27
- switch (q.fieldType) {
28
- case "Textbox": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />; break;
29
- case "Multiple Choice": {
30
- input = (
31
- <FormControl fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}}>
32
- <InputLabel>{q.title}</InputLabel>
33
- <Select fullWidth label={q.title} value={answerValue} onChange={handleChange}>{choiceOptions}</Select>
34
- </FormControl>);
35
- break;
36
- }
37
- case "Yes/No": {
38
- input = (
39
- <FormControl fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}}>
40
- <InputLabel>{q.title}</InputLabel>
41
- <Select fullWidth label={q.title} value={answerValue} onChange={handleChange}>
42
- <MenuItem value="False">No</MenuItem>
43
- <MenuItem value="True">Yes</MenuItem>
44
- </Select>
45
- </FormControl>);
46
- break;
47
- }
48
- case "Whole Number":
49
- case "Decimal":
50
- input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="number" InputLabelProps={{shrink: true}} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />;
51
- break;
52
- case "Date": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="date" InputLabelProps={{shrink: true}} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />; break;
53
- case "Phone Number": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="number" InputLabelProps={{shrink: true}} label={q.title} placeholder="555-555-5555" value={answerValue} onChange={handleChange} />; break;
54
- case "Email": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="email" label={q.title} placeholder="john@doe.com" value={answerValue} onChange={handleChange} />; break;
55
- case "Text Area": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} multiline rows={4} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />; break;
56
- default: return null;
57
- }
58
- return input;
59
- }
60
-
61
- }
62
-
1
+ import React from "react";
2
+ import { AnswerInterface, QuestionInterface } from "../interfaces";
3
+ import { Select, MenuItem, SelectChangeEvent, FormControl, InputLabel, TextField } from "@mui/material";
4
+
5
+ interface Props {
6
+ answer: AnswerInterface
7
+ question: QuestionInterface,
8
+ noBackground?: boolean,
9
+ changeFunction: (questionId: string, value: string) => void
10
+ }
11
+
12
+ export const QuestionEdit: React.FC<Props> = ({noBackground = false, ...props}) => {
13
+ const handleChange = (e: React.ChangeEvent<HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement> | SelectChangeEvent<string>) => {
14
+ props.changeFunction(props.question.id, e.target.value);
15
+ }
16
+ let q = props.question;
17
+
18
+ if (q.fieldType === "Heading") return <h5>{q.title}</h5>;
19
+ else {
20
+ let input = null;
21
+ let choiceOptions = [];
22
+ if (q.choices !== undefined && q.choices !== null) {
23
+ for (let i = 0; i < q.choices.length; i++) choiceOptions.push(<MenuItem key={i} value={q.choices[i].value}>{q.choices[i].text}</MenuItem>);
24
+ }
25
+
26
+ let answerValue = (props.answer === null) ? "" : props.answer.value
27
+ switch (q.fieldType) {
28
+ case "Textbox": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />; break;
29
+ case "Multiple Choice": {
30
+ input = (
31
+ <FormControl fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}}>
32
+ <InputLabel>{q.title}</InputLabel>
33
+ <Select fullWidth label={q.title} value={answerValue} onChange={handleChange}>{choiceOptions}</Select>
34
+ </FormControl>);
35
+ break;
36
+ }
37
+ case "Yes/No": {
38
+ input = (
39
+ <FormControl fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}}>
40
+ <InputLabel>{q.title}</InputLabel>
41
+ <Select fullWidth label={q.title} value={answerValue} onChange={handleChange}>
42
+ <MenuItem value="False">No</MenuItem>
43
+ <MenuItem value="True">Yes</MenuItem>
44
+ </Select>
45
+ </FormControl>);
46
+ break;
47
+ }
48
+ case "Whole Number":
49
+ case "Decimal":
50
+ input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="number" InputLabelProps={{shrink: true}} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />;
51
+ break;
52
+ case "Date": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="date" InputLabelProps={{shrink: true}} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />; break;
53
+ case "Phone Number": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="number" InputLabelProps={{shrink: true}} label={q.title} placeholder="555-555-5555" value={answerValue} onChange={handleChange} />; break;
54
+ case "Email": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} type="email" label={q.title} placeholder="john@doe.com" value={answerValue} onChange={handleChange} />; break;
55
+ case "Text Area": input = <TextField fullWidth style={noBackground ? {backgroundColor: "white", borderRadius: "4px"} : {}} multiline rows={4} label={q.title} placeholder={q.placeholder} value={answerValue} onChange={handleChange} />; break;
56
+ default: return null;
57
+ }
58
+ return input;
59
+ }
60
+
61
+ }
62
+
@@ -1,39 +1,39 @@
1
- import React from "react";
2
- import { Button, Icon, Tooltip } from "@mui/material";
3
- import { Navigate } from "react-router-dom";
4
-
5
- interface Props {
6
- ariaLabel?: string;
7
- text?: string;
8
- icon: string;
9
- color?: "inherit" | "primary" | "secondary" | "success" | "error" | "info" | "warning";
10
- toolTip?: string;
11
- onClick?: (e: React.MouseEvent) => void;
12
- href?: string;
13
- disabled?: boolean
14
- }
15
-
16
- export const SmallButton = React.forwardRef<HTMLDivElement, Props>((props, ref) => {
17
- const [redirectUrl, setRedirectUrl] = React.useState("");
18
-
19
- const handleClick = (e: React.MouseEvent) => {
20
- e.preventDefault();
21
- if (props.href) setRedirectUrl(props.href);
22
- else props.onClick(e);
23
- }
24
-
25
- const style = (props.text)
26
- ? { backgroundColor: props.color, "& span": { marginRight: 1 } }
27
- : { minWidth: "auto", padding: "4px 4px" }
28
-
29
- if (redirectUrl) return <Navigate to={redirectUrl} />
30
- else return (
31
- <Tooltip title={props.toolTip || ""} arrow placement="top">
32
- <Button sx={style} disabled={props.disabled} variant={props.text ? "outlined" : "text"} color={props.color} aria-label={props.ariaLabel || "editButton"} onClick={handleClick} size="small">
33
- <Icon>{props.icon}</Icon>{(props.text) ? props.text : ""}
34
- </Button>
35
- </Tooltip>
36
- );
37
- });
38
-
39
- SmallButton.displayName = "SmallButton";
1
+ import React from "react";
2
+ import { Button, Icon, Tooltip } from "@mui/material";
3
+ import { Navigate } from "react-router-dom";
4
+
5
+ interface Props {
6
+ ariaLabel?: string;
7
+ text?: string;
8
+ icon: string;
9
+ color?: "inherit" | "primary" | "secondary" | "success" | "error" | "info" | "warning";
10
+ toolTip?: string;
11
+ onClick?: (e: React.MouseEvent) => void;
12
+ href?: string;
13
+ disabled?: boolean
14
+ }
15
+
16
+ export const SmallButton = React.forwardRef<HTMLDivElement, Props>((props, ref) => {
17
+ const [redirectUrl, setRedirectUrl] = React.useState("");
18
+
19
+ const handleClick = (e: React.MouseEvent) => {
20
+ e.preventDefault();
21
+ if (props.href) setRedirectUrl(props.href);
22
+ else props.onClick(e);
23
+ }
24
+
25
+ const style = (props.text)
26
+ ? { backgroundColor: props.color, "& span": { marginRight: 1 } }
27
+ : { minWidth: "auto", padding: "4px 4px" }
28
+
29
+ if (redirectUrl) return <Navigate to={redirectUrl} />
30
+ else return (
31
+ <Tooltip title={props.toolTip || ""} arrow placement="top">
32
+ <Button sx={style} disabled={props.disabled} variant={props.text ? "outlined" : "text"} color={props.color} aria-label={props.ariaLabel || "editButton"} onClick={handleClick} size="small">
33
+ <Icon>{props.icon}</Icon>{(props.text) ? props.text : ""}
34
+ </Button>
35
+ </Tooltip>
36
+ );
37
+ });
38
+
39
+ SmallButton.displayName = "SmallButton";
@@ -1,26 +1,26 @@
1
- import { Button, Dialog, DialogActions, DialogContent, DialogTitle, Icon, Stack } from "@mui/material";
2
- import React from "react";
3
-
4
- interface Props {
5
- appName?: string
6
- onClose: () => void
7
- }
8
-
9
- export const SupportModal: React.FC<Props> = ({ appName = "", onClose }) => {
10
- const subject = appName ? `?subject=${appName} Support` : ""
11
-
12
- return (<>
13
- <Dialog open={true} onClose={onClose}>
14
- <DialogTitle>Get Support</DialogTitle>
15
- <DialogContent>
16
- <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}} mb={2}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>mail</Icon> Email:</Stack></b> <a href={"mailto:support@livecs.org" + subject}>support@livecs.org</a></Stack>
17
- <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}} mb={2}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>phone_iphone</Icon> Phone:</Stack></b> <a href="tel:+19189942638">+1 (918) 994-2638</a></Stack>
18
- <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}} mb={2}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>forum</Icon> Messenger:</Stack></b> <a href="https://m.me/livecsolutions" target="_new">https://m.me/livecsolutions</a></Stack>
19
- <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>info</Icon> Knowledge Base:</Stack></b> <a href="https://support.churchapps.org" target="_new">https://support.churchapps.org</a></Stack>
20
- </DialogContent>
21
- <DialogActions sx={{ paddingX: "16px", paddingBottom: "12px" }}>
22
- <Button variant="outlined" onClick={onClose}>Close</Button>
23
- </DialogActions>
24
- </Dialog>
25
- </>);
26
- };
1
+ import { Button, Dialog, DialogActions, DialogContent, DialogTitle, Icon, Stack } from "@mui/material";
2
+ import React from "react";
3
+
4
+ interface Props {
5
+ appName?: string
6
+ onClose: () => void
7
+ }
8
+
9
+ export const SupportModal: React.FC<Props> = ({ appName = "", onClose }) => {
10
+ const subject = appName ? `?subject=${appName} Support` : ""
11
+
12
+ return (<>
13
+ <Dialog open={true} onClose={onClose}>
14
+ <DialogTitle>Get Support</DialogTitle>
15
+ <DialogContent>
16
+ <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}} mb={2}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>mail</Icon> Email:</Stack></b> <a href={"mailto:support@livecs.org" + subject}>support@livecs.org</a></Stack>
17
+ <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}} mb={2}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>phone_iphone</Icon> Phone:</Stack></b> <a href="tel:+19189942638">+1 (918) 994-2638</a></Stack>
18
+ <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}} mb={2}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>forum</Icon> Messenger:</Stack></b> <a href="https://m.me/livecsolutions" target="_new">https://m.me/livecsolutions</a></Stack>
19
+ <Stack direction="row" alignItems="center" sx={{flexWrap: "wrap"}}><b><Stack direction="row" alignItems="center" mr="5px"><Icon sx={{marginRight: "5px"}}>info</Icon> Knowledge Base:</Stack></b> <a href="https://support.churchapps.org" target="_new">https://support.churchapps.org</a></Stack>
20
+ </DialogContent>
21
+ <DialogActions sx={{ paddingX: "16px", paddingBottom: "12px" }}>
22
+ <Button variant="outlined" onClick={onClose}>Close</Button>
23
+ </DialogActions>
24
+ </Dialog>
25
+ </>);
26
+ };