@selfcommunity/react-ui 0.7.9-alpha.4 → 0.7.9-alpha.40

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 (112) hide show
  1. package/lib/cjs/components/ChangeGroupCover/ChangeGroupCover.js +24 -1
  2. package/lib/cjs/components/ChangeGroupPicture/ChangeGroupPicture.js +32 -11
  3. package/lib/cjs/components/Composer/Attributes/Attributes.js +3 -3
  4. package/lib/cjs/components/Composer/Composer.js +3 -5
  5. package/lib/cjs/components/Composer/Content/ContentPost/ContentPost.js +4 -3
  6. package/lib/cjs/components/Composer/Layer/AudienceLayer/AudienceLayer.d.ts +1 -2
  7. package/lib/cjs/components/Composer/Layer/AudienceLayer/AudienceLayer.js +18 -6
  8. package/lib/cjs/components/EditGroupButton/EditGroupButton.js +4 -0
  9. package/lib/cjs/components/FeedObject/FeedObject.d.ts +1 -0
  10. package/lib/cjs/components/FeedObject/FeedObject.js +23 -6
  11. package/lib/cjs/components/FeedUpdatesWidget/FeedUpdatesWidget.js +1 -1
  12. package/lib/cjs/components/Group/Group.js +9 -3
  13. package/lib/cjs/components/GroupForm/GroupForm.js +31 -9
  14. package/lib/cjs/components/GroupHeader/GroupHeader.d.ts +6 -5
  15. package/lib/cjs/components/GroupHeader/GroupHeader.js +52 -11
  16. package/lib/cjs/components/GroupHeader/Skeleton.d.ts +2 -4
  17. package/lib/cjs/components/GroupHeader/Skeleton.js +10 -10
  18. package/lib/cjs/components/GroupInfoWidget/GroupInfoWidget.js +40 -7
  19. package/lib/cjs/components/GroupInviteButton/GroupInviteButton.js +8 -7
  20. package/lib/cjs/components/GroupMembersButton/GroupMembersButton.d.ts +5 -0
  21. package/lib/cjs/components/GroupMembersButton/GroupMembersButton.js +7 -2
  22. package/lib/cjs/components/GroupMembersWidget/GroupMembersWidget.js +29 -4
  23. package/lib/cjs/components/GroupRequestsWidget/GroupRequestsWidget.d.ts +11 -5
  24. package/lib/cjs/components/GroupRequestsWidget/GroupRequestsWidget.js +18 -7
  25. package/lib/cjs/components/GroupSettingsIconButton/GroupSettingsIconButton.d.ts +48 -0
  26. package/lib/cjs/components/GroupSettingsIconButton/GroupSettingsIconButton.js +145 -0
  27. package/lib/cjs/components/GroupSettingsIconButton/index.d.ts +3 -0
  28. package/lib/cjs/components/GroupSettingsIconButton/index.js +5 -0
  29. package/lib/cjs/components/GroupSubscribeButton/GroupSubscribeButton.d.ts +8 -3
  30. package/lib/cjs/components/GroupSubscribeButton/GroupSubscribeButton.js +31 -10
  31. package/lib/cjs/components/Groups/Groups.d.ts +28 -13
  32. package/lib/cjs/components/Groups/Groups.js +71 -89
  33. package/lib/cjs/components/NavigationMenuIconButton/NavigationMenuIconButton.js +1 -1
  34. package/lib/cjs/components/NavigationToolbarMobile/NavigationToolbarMobile.d.ts +4 -0
  35. package/lib/cjs/components/NavigationToolbarMobile/NavigationToolbarMobile.js +2 -3
  36. package/lib/cjs/components/Notification/Group/Group.d.ts +15 -0
  37. package/lib/cjs/components/Notification/Group/Group.js +78 -0
  38. package/lib/cjs/components/Notification/Group/index.d.ts +3 -0
  39. package/lib/cjs/components/Notification/Group/index.js +5 -0
  40. package/lib/cjs/components/Notification/Notification.js +31 -1
  41. package/lib/cjs/components/Notification/PrivateMessage/PrivateMessage.js +16 -5
  42. package/lib/cjs/components/PrivateMessageComponent/PrivateMessageComponent.d.ts +1 -1
  43. package/lib/cjs/components/PrivateMessageComponent/PrivateMessageComponent.js +10 -7
  44. package/lib/cjs/components/PrivateMessageSnippetItem/PrivateMessageSnippetItem.js +11 -6
  45. package/lib/cjs/components/PrivateMessageSnippets/PrivateMessageSnippets.d.ts +3 -3
  46. package/lib/cjs/components/PrivateMessageSnippets/PrivateMessageSnippets.js +24 -6
  47. package/lib/cjs/components/PrivateMessageThread/PrivateMessageThread.d.ts +6 -1
  48. package/lib/cjs/components/PrivateMessageThread/PrivateMessageThread.js +45 -20
  49. package/lib/cjs/components/SearchAutocomplete/SearchAutocomplete.js +22 -5
  50. package/lib/cjs/components/SnippetNotifications/SnippetNotifications.js +7 -0
  51. package/lib/cjs/components/ToastNotifications/ToastNotifications.js +7 -0
  52. package/lib/cjs/constants/PubSub.d.ts +27 -0
  53. package/lib/cjs/constants/PubSub.js +21 -0
  54. package/lib/cjs/index.d.ts +4 -2
  55. package/lib/cjs/index.js +9 -4
  56. package/lib/esm/components/ChangeGroupCover/ChangeGroupCover.js +24 -1
  57. package/lib/esm/components/ChangeGroupPicture/ChangeGroupPicture.js +32 -11
  58. package/lib/esm/components/Composer/Attributes/Attributes.js +3 -3
  59. package/lib/esm/components/Composer/Composer.js +3 -5
  60. package/lib/esm/components/Composer/Content/ContentPost/ContentPost.js +4 -3
  61. package/lib/esm/components/Composer/Layer/AudienceLayer/AudienceLayer.d.ts +1 -2
  62. package/lib/esm/components/Composer/Layer/AudienceLayer/AudienceLayer.js +18 -6
  63. package/lib/esm/components/EditGroupButton/EditGroupButton.js +5 -1
  64. package/lib/esm/components/FeedObject/FeedObject.d.ts +1 -0
  65. package/lib/esm/components/FeedObject/FeedObject.js +24 -7
  66. package/lib/esm/components/FeedUpdatesWidget/FeedUpdatesWidget.js +1 -1
  67. package/lib/esm/components/Group/Group.js +11 -5
  68. package/lib/esm/components/GroupForm/GroupForm.js +31 -9
  69. package/lib/esm/components/GroupHeader/GroupHeader.d.ts +6 -5
  70. package/lib/esm/components/GroupHeader/GroupHeader.js +54 -13
  71. package/lib/esm/components/GroupHeader/Skeleton.d.ts +2 -4
  72. package/lib/esm/components/GroupHeader/Skeleton.js +10 -10
  73. package/lib/esm/components/GroupInfoWidget/GroupInfoWidget.js +40 -7
  74. package/lib/esm/components/GroupInviteButton/GroupInviteButton.js +8 -7
  75. package/lib/esm/components/GroupMembersButton/GroupMembersButton.d.ts +5 -0
  76. package/lib/esm/components/GroupMembersButton/GroupMembersButton.js +8 -3
  77. package/lib/esm/components/GroupMembersWidget/GroupMembersWidget.js +31 -6
  78. package/lib/esm/components/GroupRequestsWidget/GroupRequestsWidget.d.ts +11 -5
  79. package/lib/esm/components/GroupRequestsWidget/GroupRequestsWidget.js +18 -7
  80. package/lib/esm/components/GroupSettingsIconButton/GroupSettingsIconButton.d.ts +48 -0
  81. package/lib/esm/components/GroupSettingsIconButton/GroupSettingsIconButton.js +142 -0
  82. package/lib/esm/components/GroupSettingsIconButton/index.d.ts +3 -0
  83. package/lib/esm/components/GroupSettingsIconButton/index.js +2 -0
  84. package/lib/esm/components/GroupSubscribeButton/GroupSubscribeButton.d.ts +8 -3
  85. package/lib/esm/components/GroupSubscribeButton/GroupSubscribeButton.js +32 -11
  86. package/lib/esm/components/Groups/Groups.d.ts +28 -13
  87. package/lib/esm/components/Groups/Groups.js +76 -94
  88. package/lib/esm/components/NavigationMenuIconButton/NavigationMenuIconButton.js +1 -1
  89. package/lib/esm/components/NavigationToolbarMobile/NavigationToolbarMobile.d.ts +4 -0
  90. package/lib/esm/components/NavigationToolbarMobile/NavigationToolbarMobile.js +2 -3
  91. package/lib/esm/components/Notification/Group/Group.d.ts +15 -0
  92. package/lib/esm/components/Notification/Group/Group.js +75 -0
  93. package/lib/esm/components/Notification/Group/index.d.ts +3 -0
  94. package/lib/esm/components/Notification/Group/index.js +2 -0
  95. package/lib/esm/components/Notification/Notification.js +31 -1
  96. package/lib/esm/components/Notification/PrivateMessage/PrivateMessage.js +16 -5
  97. package/lib/esm/components/PrivateMessageComponent/PrivateMessageComponent.d.ts +1 -1
  98. package/lib/esm/components/PrivateMessageComponent/PrivateMessageComponent.js +11 -8
  99. package/lib/esm/components/PrivateMessageSnippetItem/PrivateMessageSnippetItem.js +11 -6
  100. package/lib/esm/components/PrivateMessageSnippets/PrivateMessageSnippets.d.ts +3 -3
  101. package/lib/esm/components/PrivateMessageSnippets/PrivateMessageSnippets.js +26 -8
  102. package/lib/esm/components/PrivateMessageThread/PrivateMessageThread.d.ts +6 -1
  103. package/lib/esm/components/PrivateMessageThread/PrivateMessageThread.js +47 -22
  104. package/lib/esm/components/SearchAutocomplete/SearchAutocomplete.js +22 -5
  105. package/lib/esm/components/SnippetNotifications/SnippetNotifications.js +7 -0
  106. package/lib/esm/components/ToastNotifications/ToastNotifications.js +7 -0
  107. package/lib/esm/constants/PubSub.d.ts +27 -0
  108. package/lib/esm/constants/PubSub.js +18 -0
  109. package/lib/esm/index.d.ts +4 -2
  110. package/lib/esm/index.js +6 -4
  111. package/lib/umd/react-ui.js +1 -1
  112. package/package.json +6 -6
@@ -16,6 +16,10 @@ const Bullet_1 = tslib_1.__importDefault(require("../../shared/Bullet"));
16
16
  const ChangeGroupPicture_1 = tslib_1.__importDefault(require("../ChangeGroupPicture"));
17
17
  const GroupMembersButton_1 = tslib_1.__importDefault(require("../GroupMembersButton"));
18
18
  const EditGroupButton_1 = tslib_1.__importDefault(require("../EditGroupButton"));
19
+ const GroupSubscribeButton_1 = tslib_1.__importDefault(require("../GroupSubscribeButton"));
20
+ const GroupInviteButton_1 = tslib_1.__importDefault(require("../GroupInviteButton"));
21
+ const PubSub_1 = require("../../constants/PubSub");
22
+ const pubsub_js_1 = tslib_1.__importDefault(require("pubsub-js"));
19
23
  const classes = {
20
24
  root: `${constants_1.PREFIX}-root`,
21
25
  cover: `${constants_1.PREFIX}-cover`,
@@ -72,22 +76,24 @@ function GroupHeader(inProps) {
72
76
  props: inProps,
73
77
  name: constants_1.PREFIX
74
78
  });
75
- const { id = null, className = null, group, groupId = null, ChangePictureProps = {}, ChangeCoverProps = {}, GroupMembersButtonProps = {}, actions } = props, rest = tslib_1.__rest(props, ["id", "className", "group", "groupId", "ChangePictureProps", "ChangeCoverProps", "GroupMembersButtonProps", "actions"]);
79
+ const { id = null, className = null, group, groupId = null, ChangePictureProps = {}, ChangeCoverProps = {}, GroupSubscribeButtonProps = {}, GroupMembersButtonProps = {} } = props, rest = tslib_1.__rest(props, ["id", "className", "group", "groupId", "ChangePictureProps", "ChangeCoverProps", "GroupSubscribeButtonProps", "GroupMembersButtonProps"]);
76
80
  // PREFERENCES
77
81
  const scPreferences = (0, react_core_1.useSCPreferences)();
78
82
  // CONTEXT
79
83
  const scUserContext = (0, react_core_1.useSCUser)();
80
84
  // HOOKS
81
85
  const { scGroup, setSCGroup } = (0, react_core_1.useSCFetchGroup)({ id: groupId, group });
86
+ // REFS
87
+ const updatesSubscription = (0, react_1.useRef)(null);
82
88
  // CONST
83
- const canEdit = (0, react_1.useMemo)(() => { var _a; return scUserContext.user && ((_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id) === scUserContext.user.id; }, [scUserContext.user, (_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id]);
89
+ const isGroupAdmin = (0, react_1.useMemo)(() => { var _a; return scUserContext.user && ((_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id) === scUserContext.user.id; }, [scUserContext.user, (_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id]);
84
90
  /**
85
91
  * Handles Change Avatar
86
92
  * @param avatar
87
93
  */
88
94
  function handleChangeAvatar(avatar) {
89
- if (canEdit) {
90
- setSCGroup(Object.assign({}, scGroup, { image_bigger: avatar }));
95
+ if (isGroupAdmin) {
96
+ setSCGroup(Object.assign({}, scGroup, { image_big: avatar }));
91
97
  }
92
98
  }
93
99
  /**
@@ -95,10 +101,43 @@ function GroupHeader(inProps) {
95
101
  * @param cover
96
102
  */
97
103
  function handleChangeCover(cover) {
98
- if (canEdit) {
104
+ if (isGroupAdmin) {
99
105
  setSCGroup(Object.assign({}, scGroup, { emotional_image: cover }));
100
106
  }
101
107
  }
108
+ /**
109
+ * Handles callback subscribe/unsubscribe group
110
+ */
111
+ const handleSubscribe = (group, status) => {
112
+ setSCGroup(Object.assign(Object.assign({}, group), { subscribers_counter: group.subscribers_counter + (status ? 1 : -1) }));
113
+ };
114
+ /**
115
+ * Subscriber for pubsub callback
116
+ */
117
+ const onChangeGroupMembersHandler = (0, react_1.useCallback)((msg, data) => {
118
+ if (data && data.group.id === scGroup.id) {
119
+ let _group = Object.assign({}, scGroup);
120
+ if (msg === `${PubSub_1.SCTopicType.GROUP}.${PubSub_1.SCEventType.ADD_MEMBER}`) {
121
+ _group.subscribers_counter = _group.subscribers_counter + 1;
122
+ }
123
+ else if (msg === `${PubSub_1.SCTopicType.GROUP}.${PubSub_1.SCEventType.REMOVE_MEMBER}`) {
124
+ _group.subscribers_counter = Math.max(_group.subscribers_counter - 1, 0);
125
+ }
126
+ console.log(_group);
127
+ setSCGroup(_group);
128
+ }
129
+ }, [scGroup, setSCGroup]);
130
+ /**
131
+ * On mount, subscribe to receive groups updates (only edit)
132
+ */
133
+ (0, react_1.useEffect)(() => {
134
+ if (scGroup) {
135
+ updatesSubscription.current = pubsub_js_1.default.subscribe(`${PubSub_1.SCTopicType.GROUP}.${PubSub_1.SCEventType.MEMBERS}`, onChangeGroupMembersHandler);
136
+ }
137
+ return () => {
138
+ updatesSubscription.current && pubsub_js_1.default.unsubscribe(updatesSubscription.current);
139
+ };
140
+ }, [scGroup]);
102
141
  // RENDER
103
142
  if (!scGroup) {
104
143
  return react_1.default.createElement(Skeleton_1.default, null);
@@ -110,13 +149,13 @@ function GroupHeader(inProps) {
110
149
  react_1.default.createElement(material_1.Paper, { style: _backgroundCover, classes: { root: classes.cover } },
111
150
  react_1.default.createElement(material_1.Box, { className: classes.avatar },
112
151
  react_1.default.createElement(material_1.Avatar, null,
113
- react_1.default.createElement("img", { alt: "group", src: scGroup.image_bigger ? scGroup.image_bigger : '' }))),
114
- canEdit && (react_1.default.createElement(react_1.default.Fragment, null,
152
+ react_1.default.createElement("img", { alt: "group", src: scGroup.image_big ? scGroup.image_big : '' }))),
153
+ isGroupAdmin && (react_1.default.createElement(react_1.default.Fragment, null,
115
154
  react_1.default.createElement(ChangeGroupPicture_1.default, Object.assign({ groupId: scGroup.id, onChange: handleChangeAvatar, className: classes.changePicture }, ChangePictureProps)),
116
155
  react_1.default.createElement("div", { className: classes.changeCover },
117
156
  react_1.default.createElement(ChangeGroupCover_1.default, Object.assign({ groupId: scGroup.id, onChange: handleChangeCover }, ChangeCoverProps)))))),
118
157
  react_1.default.createElement(material_1.Box, { className: classes.info },
119
- canEdit && react_1.default.createElement(EditGroupButton_1.default, { group: scGroup, groupId: scGroup.id, onEditSuccess: (data) => setSCGroup(data) }),
158
+ isGroupAdmin && react_1.default.createElement(EditGroupButton_1.default, { group: scGroup, groupId: scGroup.id, onEditSuccess: (data) => setSCGroup(data) }),
120
159
  react_1.default.createElement(material_1.Typography, { variant: "h5", className: classes.name }, scGroup.name),
121
160
  react_1.default.createElement(material_1.Box, { className: classes.visibility },
122
161
  scGroup.privacy === types_1.SCGroupPrivacyType.PUBLIC ? (react_1.default.createElement(material_1.Typography, { className: classes.visibilityItem },
@@ -130,10 +169,12 @@ function GroupHeader(inProps) {
130
169
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupHeader.visibility.visible", defaultMessage: "ui.groupHeader.visibility.visible" }))) : (react_1.default.createElement(material_1.Typography, { className: classes.visibilityItem },
131
170
  react_1.default.createElement(material_1.Icon, null, "visibility_off"),
132
171
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupHeader.visibility.hidden", defaultMessage: "ui.groupHeader.visibility.hidden" })))),
133
- react_1.default.createElement(material_1.Box, { className: classes.members },
172
+ react_1.default.createElement(react_1.default.Fragment, null, ((scGroup && scGroup.privacy === types_1.SCGroupPrivacyType.PUBLIC) ||
173
+ scGroup.subscription_status === types_1.SCGroupSubscriptionStatusType.SUBSCRIBED ||
174
+ isGroupAdmin) && (react_1.default.createElement(material_1.Box, { className: classes.members },
134
175
  react_1.default.createElement(material_1.Typography, { className: classes.membersCounter, component: "div" },
135
176
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupHeader.members", defaultMessage: "ui.groupHeader.members", values: { total: scGroup.subscribers_counter } })),
136
- react_1.default.createElement(GroupMembersButton_1.default, Object.assign({ groupId: scGroup === null || scGroup === void 0 ? void 0 : scGroup.id, group: scGroup }, GroupMembersButtonProps))),
137
- actions && actions)));
177
+ react_1.default.createElement(GroupMembersButton_1.default, Object.assign({ key: scGroup.subscribers_counter, groupId: scGroup === null || scGroup === void 0 ? void 0 : scGroup.id, group: scGroup, autoHide: !isGroupAdmin }, GroupMembersButtonProps))))),
178
+ isGroupAdmin ? (react_1.default.createElement(GroupInviteButton_1.default, { group: scGroup, groupId: scGroup.id })) : (react_1.default.createElement(GroupSubscribeButton_1.default, Object.assign({ group: scGroup, onSubscribe: handleSubscribe }, GroupSubscribeButtonProps))))));
138
179
  }
139
180
  exports.default = GroupHeader;
@@ -1,5 +1,5 @@
1
1
  /**
2
- * > API documentation for the Community-JS Group Headerr Skeleton component. Learn about the available props and the CSS API.
2
+ * > API documentation for the Community-JS Group Header Skeleton component. Learn about the available props and the CSS API.
3
3
 
4
4
  #### Import
5
5
 
@@ -18,9 +18,7 @@
18
18
  |root|.SCGroupHeader-skeleton-root|Styles applied to the root element.|
19
19
  |avatar|.SCGroupHeader-avatar|Styles applied to the avatar element.|
20
20
  |cover|.SCGroupHeader-cover|Styles applied to the cover element.|
21
- |actions|.SCGroupHeader-actions|Styles applied to the actions section.|
22
- |section|.SCGroupHeader-section|Styles applied to the info section.|
23
- |username|.SCGroupHeader-username|Styles applied to the username element.|
21
+ |info|.SCGroupHeader-info|Styles applied to the info info.|
24
22
  *
25
23
  */
26
24
  declare function GroupHeaderSkeleton(): JSX.Element;
@@ -10,16 +10,14 @@ const classes = {
10
10
  root: `${constants_1.PREFIX}-skeleton-root`,
11
11
  cover: `${constants_1.PREFIX}-cover`,
12
12
  avatar: `${constants_1.PREFIX}-avatar`,
13
- actions: `${constants_1.PREFIX}-actions`,
14
- section: `${constants_1.PREFIX}-section`,
15
- username: `${constants_1.PREFIX}-username`
13
+ info: `${constants_1.PREFIX}-info`
16
14
  };
17
15
  const Root = (0, styles_1.styled)(material_1.Box, {
18
16
  name: constants_1.PREFIX,
19
17
  slot: 'SkeletonRoot'
20
18
  })(() => ({}));
21
19
  /**
22
- * > API documentation for the Community-JS Group Headerr Skeleton component. Learn about the available props and the CSS API.
20
+ * > API documentation for the Community-JS Group Header Skeleton component. Learn about the available props and the CSS API.
23
21
 
24
22
  #### Import
25
23
 
@@ -38,9 +36,7 @@ const Root = (0, styles_1.styled)(material_1.Box, {
38
36
  |root|.SCGroupHeader-skeleton-root|Styles applied to the root element.|
39
37
  |avatar|.SCGroupHeader-avatar|Styles applied to the avatar element.|
40
38
  |cover|.SCGroupHeader-cover|Styles applied to the cover element.|
41
- |actions|.SCGroupHeader-actions|Styles applied to the actions section.|
42
- |section|.SCGroupHeader-section|Styles applied to the info section.|
43
- |username|.SCGroupHeader-username|Styles applied to the username element.|
39
+ |info|.SCGroupHeader-info|Styles applied to the info info.|
44
40
  *
45
41
  */
46
42
  function GroupHeaderSkeleton() {
@@ -49,8 +45,12 @@ function GroupHeaderSkeleton() {
49
45
  react_1.default.createElement(Skeleton_1.default, { className: classes.cover, animation: "wave", variant: "rectangular" }),
50
46
  react_1.default.createElement(material_1.Box, { className: classes.avatar },
51
47
  react_1.default.createElement(Skeleton_1.default, { animation: "wave", variant: "circular", width: theme.selfcommunity.group.avatar.sizeLarge, height: theme.selfcommunity.group.avatar.sizeLarge })),
52
- react_1.default.createElement(material_1.Box, { className: classes.section },
53
- react_1.default.createElement(material_1.Typography, { variant: "h5", className: classes.username },
54
- react_1.default.createElement(Skeleton_1.default, { animation: "wave", sx: { height: 30, width: 100 } })))));
48
+ react_1.default.createElement(material_1.Box, { className: classes.info },
49
+ react_1.default.createElement(material_1.Typography, { variant: "h5" },
50
+ react_1.default.createElement(Skeleton_1.default, { animation: "wave", sx: { height: 30, width: 200 } })),
51
+ react_1.default.createElement(material_1.Typography, null,
52
+ react_1.default.createElement(Skeleton_1.default, { animation: "wave", sx: { height: 20, width: 150 } })),
53
+ react_1.default.createElement(material_1.Typography, null,
54
+ react_1.default.createElement(Skeleton_1.default, { animation: "wave", sx: { height: 20, width: 100 } })))));
55
55
  }
56
56
  exports.default = GroupHeaderSkeleton;
@@ -1,7 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  const tslib_1 = require("tslib");
4
- const react_1 = tslib_1.__importDefault(require("react"));
4
+ const react_1 = tslib_1.__importStar(require("react"));
5
5
  const styles_1 = require("@mui/material/styles");
6
6
  const material_1 = require("@mui/material");
7
7
  const classnames_1 = tslib_1.__importDefault(require("classnames"));
@@ -10,7 +10,10 @@ const system_1 = require("@mui/system");
10
10
  const constants_1 = require("./constants");
11
11
  const react_intl_1 = require("react-intl");
12
12
  const types_1 = require("@selfcommunity/types");
13
+ const pubsub_js_1 = tslib_1.__importDefault(require("pubsub-js"));
13
14
  const react_core_1 = require("@selfcommunity/react-core");
15
+ const Skeleton_1 = tslib_1.__importDefault(require("./Skeleton"));
16
+ const PubSub_1 = require("../../constants/PubSub");
14
17
  const classes = {
15
18
  root: `${constants_1.PREFIX}-root`,
16
19
  title: `${constants_1.PREFIX}-title`,
@@ -19,6 +22,7 @@ const classes = {
19
22
  privacyTitle: `${constants_1.PREFIX}-privacy-title`,
20
23
  visibility: `${constants_1.PREFIX}-visibility`,
21
24
  visibilityTitle: `${constants_1.PREFIX}-visibility-title`,
25
+ admin: `${constants_1.PREFIX}-admin`,
22
26
  date: `${constants_1.PREFIX}-date`
23
27
  };
24
28
  const Root = (0, styles_1.styled)(Widget_1.default, {
@@ -61,9 +65,36 @@ function GroupInfoWidget(inProps) {
61
65
  });
62
66
  const { className, group, groupId, onHeightChange, onStateChange } = props, rest = tslib_1.__rest(props, ["className", "group", "groupId", "onHeightChange", "onStateChange"]);
63
67
  // HOOKS
64
- const { scGroup } = (0, react_core_1.useSCFetchGroup)({ id: groupId, group });
68
+ const { scGroup, setSCGroup } = (0, react_core_1.useSCFetchGroup)({ id: groupId, group });
65
69
  // INTL
66
70
  const intl = (0, react_intl_1.useIntl)();
71
+ // REFS
72
+ const updatesSubscription = (0, react_1.useRef)(null);
73
+ /**
74
+ * Subscriber for pubsub callback
75
+ */
76
+ const onChangeGroupHandler = (0, react_1.useCallback)((_msg, data) => {
77
+ if (data && scGroup.id === data.id) {
78
+ setSCGroup(data);
79
+ }
80
+ }, [scGroup, setSCGroup]);
81
+ /**
82
+ * On mount, subscribe to receive groups updates (only edit)
83
+ */
84
+ (0, react_1.useEffect)(() => {
85
+ if (scGroup) {
86
+ updatesSubscription.current = pubsub_js_1.default.subscribe(`${PubSub_1.SCTopicType.GROUP}.${PubSub_1.SCEventType.EDIT}`, onChangeGroupHandler);
87
+ }
88
+ return () => {
89
+ updatesSubscription.current && pubsub_js_1.default.unsubscribe(updatesSubscription.current);
90
+ };
91
+ }, [scGroup]);
92
+ /**
93
+ * Loading group
94
+ */
95
+ if (!scGroup) {
96
+ return react_1.default.createElement(Skeleton_1.default, null);
97
+ }
67
98
  /**
68
99
  * Renders root object
69
100
  */
@@ -75,14 +106,14 @@ function GroupInfoWidget(inProps) {
75
106
  react_1.default.createElement(material_1.Typography, { component: "div", className: classes.privacy }, scGroup.privacy === types_1.SCGroupPrivacyType.PUBLIC ? (react_1.default.createElement(react_1.default.Fragment, null,
76
107
  react_1.default.createElement(material_1.Typography, { className: classes.privacyTitle },
77
108
  react_1.default.createElement(material_1.Icon, null, "public"),
78
- react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupForm.privacy.public", defaultMessage: "ui.groupForm.privacy.public" })),
109
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInfoWidget.privacy.public", defaultMessage: "ui.groupInfoWidget.privacy.public" })),
79
110
  react_1.default.createElement(material_1.Typography, { variant: "body2" },
80
- react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupForm.privacy.public.info", defaultMessage: "ui.groupForm.privacy.public.info", values: { b: (chunks) => react_1.default.createElement("strong", null, chunks) } })))) : (react_1.default.createElement(react_1.default.Fragment, null,
111
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInfoWidget.privacy.public.info", defaultMessage: "ui.groupInfoWidget.privacy.public.info", values: { b: (chunks) => react_1.default.createElement("strong", null, chunks) } })))) : (react_1.default.createElement(react_1.default.Fragment, null,
81
112
  react_1.default.createElement(material_1.Typography, { className: classes.privacyTitle },
82
113
  react_1.default.createElement(material_1.Icon, null, "private"),
83
- react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupForm.privacy.private", defaultMessage: "ui.groupForm.privacy.private" })),
114
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInfoWidget.privacy.private", defaultMessage: "ui.groupInfoWidget.privacy.private" })),
84
115
  react_1.default.createElement(material_1.Typography, { variant: "body2" },
85
- react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupForm.privacy.private.info", defaultMessage: "ui.groupForm.private.public.info", values: { b: (chunks) => react_1.default.createElement("strong", null, chunks) } }))))),
116
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInfoWidget.privacy.private.info", defaultMessage: "ui.groupInfoWidget.private.public.info", values: { b: (chunks) => react_1.default.createElement("strong", null, chunks) } }))))),
86
117
  scGroup.privacy === types_1.SCGroupPrivacyType.PRIVATE && (react_1.default.createElement(material_1.Typography, { component: "div", className: classes.visibility }, scGroup.visible ? (react_1.default.createElement(react_1.default.Fragment, null,
87
118
  react_1.default.createElement(material_1.Typography, { className: classes.visibilityTitle },
88
119
  react_1.default.createElement(material_1.Icon, null, "visibility"),
@@ -95,6 +126,8 @@ function GroupInfoWidget(inProps) {
95
126
  react_1.default.createElement(material_1.Typography, { variant: "body2" },
96
127
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupForm.visibility.hidden.info", defaultMessage: "ui.groupForm.visibility.hidden.info", values: { b: (chunks) => react_1.default.createElement("strong", null, chunks) } })))))),
97
128
  react_1.default.createElement(material_1.Typography, { variant: "body2", className: classes.date },
98
- react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInfoWidget.date", defaultMessage: "ui.groupInfoWidget.date", values: { date: intl.formatDate(scGroup.created_at, { day: 'numeric', year: 'numeric', month: 'long' }) } })))));
129
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInfoWidget.date", defaultMessage: "ui.groupInfoWidget.date", values: { date: intl.formatDate(scGroup.created_at, { day: 'numeric', year: 'numeric', month: 'long' }) } })),
130
+ react_1.default.createElement(material_1.Typography, { variant: "body2", className: classes.admin },
131
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInfoWidget.admin", defaultMessage: "ui.groupInfoWidget.admin", values: { b: (chunks) => react_1.default.createElement("strong", null, chunks), admin: scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by.username } })))));
99
132
  }
100
133
  exports.default = GroupInfoWidget;
@@ -91,11 +91,11 @@ function GroupInviteButton(inProps) {
91
91
  const [loading, setLoading] = (0, react_1.useState)(false);
92
92
  const [invited, setInvited] = (0, react_1.useState)([]);
93
93
  function convertToInvitedUsersObject(data) {
94
- const invited_users = {};
94
+ const invite_users = {};
95
95
  data.forEach((user, index) => {
96
- invited_users[`invited_users[${index}]`] = user.id;
96
+ invite_users[`invite_users[${index}]`] = user.id;
97
97
  });
98
- return invited_users;
98
+ return invite_users;
99
99
  }
100
100
  /**
101
101
  * Memoized users invited ids
@@ -110,7 +110,7 @@ function GroupInviteButton(inProps) {
110
110
  }, [invited]);
111
111
  // HOOKS
112
112
  const { scGroup } = (0, react_core_1.useSCFetchGroup)({ id: groupId, group });
113
- const canEdit = (0, react_1.useMemo)(() => { var _a; return scUserContext.user && ((_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id) === scUserContext.user.id; }, [scUserContext.user, (_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id]);
113
+ const isGroupAdmin = (0, react_1.useMemo)(() => { var _a; return scUserContext.user && ((_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id) === scUserContext.user.id; }, [scUserContext.user, (_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id]);
114
114
  // INTL
115
115
  const intl = (0, react_intl_1.useIntl)();
116
116
  function fetchResults() {
@@ -181,6 +181,7 @@ function GroupInviteButton(inProps) {
181
181
  .then(() => {
182
182
  setIsSending(false);
183
183
  setOpen(false);
184
+ setInvited([]);
184
185
  })
185
186
  .catch((error) => {
186
187
  setOpen(false);
@@ -225,9 +226,9 @@ function GroupInviteButton(inProps) {
225
226
  setList((prev) => [...prev, option]);
226
227
  };
227
228
  /**
228
- * If there's no authUserId, component is hidden.
229
+ * If in group edit mode and logged-in user is not also the group manager, the component is hidden.
229
230
  // */
230
- if (!canEdit) {
231
+ if (group && !isGroupAdmin) {
231
232
  return null;
232
233
  }
233
234
  /**
@@ -241,7 +242,7 @@ function GroupInviteButton(inProps) {
241
242
  react_1.default.createElement(material_1.Icon, { fontSize: "medium" }, "arrow_back")),
242
243
  react_1.default.createElement(material_1.Typography, { className: classes.dialogTitle },
243
244
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInviteButton.dialog.title", defaultMessage: "ui.groupInviteButton.dialog.title" })),
244
- react_1.default.createElement(lab_1.LoadingButton, { size: "small", color: "secondary", variant: "contained", onClick: handleSendInvitations, loading: isSending },
245
+ react_1.default.createElement(lab_1.LoadingButton, { size: "small", color: "secondary", variant: "contained", onClick: handleSendInvitations, loading: isSending, disabled: !invited.length },
245
246
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupInviteButton.dialog.button.end", defaultMessage: "ui.groupInviteButton.dialog.button.end" }))) },
246
247
  react_1.default.createElement(material_1.Box, { className: classes.dialogContent },
247
248
  react_1.default.createElement(Autocomplete_1.default, { className: classes.autocomplete, loading: loading, size: "small", multiple: true, freeSolo: true, disableClearable: true, options: suggested, onChange: handleChange, onInputChange: handleInputChange, inputValue: value, value: invited, getOptionLabel: (option) => (option ? option.username : '...'), isOptionEqualToValue: (option, value) => (option ? value.id === option.id : false), renderTags: () => null, renderOption: (props, option) => (react_1.default.createElement(material_1.Box, Object.assign({ component: "li" }, props),
@@ -17,6 +17,11 @@ export interface GroupMembersButtonProps extends Pick<ButtonProps, Exclude<keyof
17
17
  * @default {}
18
18
  */
19
19
  DialogProps?: BaseDialogProps;
20
+ /**
21
+ * Hides this component
22
+ * @default false
23
+ */
24
+ autoHide?: boolean;
20
25
  /**
21
26
  * Any other properties
22
27
  */
@@ -10,6 +10,7 @@ const InfiniteScroll_1 = tslib_1.__importDefault(require("../../shared/InfiniteS
10
10
  const User_1 = tslib_1.__importStar(require("../User"));
11
11
  const api_services_1 = require("@selfcommunity/api-services");
12
12
  const react_core_1 = require("@selfcommunity/react-core");
13
+ const types_1 = require("@selfcommunity/types");
13
14
  const AvatarGroupSkeleton_1 = tslib_1.__importDefault(require("../Skeleton/AvatarGroupSkeleton"));
14
15
  const classnames_1 = tslib_1.__importDefault(require("classnames"));
15
16
  const system_1 = require("@mui/system");
@@ -60,7 +61,7 @@ function GroupMembersButton(inProps) {
60
61
  props: inProps,
61
62
  name: PREFIX
62
63
  });
63
- const { className, groupId, group, DialogProps = {} } = props, rest = tslib_1.__rest(props, ["className", "groupId", "group", "DialogProps"]);
64
+ const { className, groupId, group, DialogProps = {}, autoHide = false } = props, rest = tslib_1.__rest(props, ["className", "groupId", "group", "DialogProps", "autoHide"]);
64
65
  // STATE
65
66
  const [loading, setLoading] = (0, react_1.useState)(true);
66
67
  const [next, setNext] = (0, react_1.useState)(null);
@@ -71,7 +72,11 @@ function GroupMembersButton(inProps) {
71
72
  const { scGroup } = (0, react_core_1.useSCFetchGroup)({ id: groupId, group });
72
73
  // FETCH FIRST FOLLOWERS
73
74
  (0, use_deep_compare_effect_1.useDeepCompareEffectNoCheck)(() => {
74
- if (!scGroup) {
75
+ if (!scGroup ||
76
+ (scGroup &&
77
+ scGroup.privacy !== types_1.SCGroupPrivacyType.PUBLIC &&
78
+ scGroup.subscription_status !== types_1.SCGroupSubscriptionStatusType.SUBSCRIBED &&
79
+ autoHide)) {
75
80
  return;
76
81
  }
77
82
  if (members.length === 0) {
@@ -21,10 +21,13 @@ const HiddenPlaceholder_1 = tslib_1.__importDefault(require("../../shared/Hidden
21
21
  const constants_1 = require("./constants");
22
22
  const User_1 = tslib_1.__importStar(require("../User"));
23
23
  const GroupInviteButton_1 = tslib_1.__importDefault(require("../GroupInviteButton"));
24
+ const GroupSettingsIconButton_1 = tslib_1.__importDefault(require("../GroupSettingsIconButton"));
25
+ const Icon_1 = tslib_1.__importDefault(require("@mui/material/Icon"));
24
26
  const classes = {
25
27
  root: `${constants_1.PREFIX}-root`,
26
28
  title: `${constants_1.PREFIX}-title`,
27
29
  actions: `${constants_1.PREFIX}-actions`,
30
+ badge: `${constants_1.PREFIX}-badge`,
28
31
  noResults: `${constants_1.PREFIX}-no-results`,
29
32
  showMore: `${constants_1.PREFIX}-show-more`,
30
33
  dialogRoot: `${constants_1.PREFIX}-dialog-root`,
@@ -70,6 +73,7 @@ const DialogRoot = (0, styles_1.styled)(BaseDialog_1.default, {
70
73
  * @param inProps
71
74
  */
72
75
  function GroupMembersWidget(inProps) {
76
+ var _a;
73
77
  // PROPS
74
78
  const props = (0, system_1.useThemeProps)({
75
79
  props: inProps,
@@ -87,8 +91,11 @@ function GroupMembersWidget(inProps) {
87
91
  const [openDialog, setOpenDialog] = (0, react_1.useState)(false);
88
92
  // CONTEXT
89
93
  const scUserContext = (0, react_core_1.useSCUser)();
94
+ const scRoutingContext = (0, react_core_1.useSCRouting)();
90
95
  const scPreferencesContext = (0, react_core_1.useSCPreferences)();
91
96
  const { scGroup } = (0, react_core_1.useSCFetchGroup)({ id: groupId, group });
97
+ // CONST
98
+ const isGroupAdmin = (0, react_1.useMemo)(() => { var _a; return scUserContext.user && ((_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id) === scUserContext.user.id; }, [scUserContext.user, (_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id]);
92
99
  // MEMO
93
100
  const contentAvailability = (0, react_1.useMemo)(() => react_core_1.SCPreferences.CONFIGURATIONS_CONTENT_AVAILABILITY in scPreferencesContext.preferences &&
94
101
  scPreferencesContext.preferences[react_core_1.SCPreferences.CONFIGURATIONS_CONTENT_AVAILABILITY].value, [scPreferencesContext.preferences]);
@@ -170,6 +177,14 @@ function GroupMembersWidget(inProps) {
170
177
  const handleToggleDialogOpen = () => {
171
178
  setOpenDialog((prev) => !prev);
172
179
  };
180
+ const handleRefresh = (0, react_1.useMemo)(() => (userId) => {
181
+ const newMembers = [...state.results];
182
+ const _updated = newMembers.filter((u) => u.id !== userId);
183
+ dispatch({
184
+ type: widget_1.actionWidgetTypes.SET_RESULTS,
185
+ payload: { results: newMembers.length > 1 ? _updated : [] }
186
+ });
187
+ }, [dispatch, state.count, state.results]);
173
188
  // RENDER
174
189
  if ((autoHide && !state.count && state.initialized) || (!contentAvailability && !scUserContext.user) || !scGroup) {
175
190
  return react_1.default.createElement(HiddenPlaceholder_1.default, null);
@@ -183,15 +198,25 @@ function GroupMembersWidget(inProps) {
183
198
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupMembersWidget.title", defaultMessage: "ui.groupMembersWidget.title" })),
184
199
  !state.count ? (react_1.default.createElement(material_1.Typography, { className: classes.noResults, variant: "body2" },
185
200
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupMembersWidget.subtitle.noResults", defaultMessage: "" }))) : (react_1.default.createElement(react_1.default.Fragment, null,
186
- react_1.default.createElement(List_1.default, null, state.results.slice(0, state.visibleItems).map((user) => (react_1.default.createElement(material_1.ListItem, { key: user.id },
187
- react_1.default.createElement(User_1.default, { elevation: 0, actions: react_1.default.createElement(react_1.default.Fragment, null), user: user, userId: user.id, buttonProps: { onClick: () => console.log(user) } }))))),
201
+ react_1.default.createElement(List_1.default, null, state.results.slice(0, state.visibleItems).map((user) => {
202
+ var _a, _b;
203
+ return (react_1.default.createElement(material_1.ListItem, { key: user.id },
204
+ react_1.default.createElement(User_1.default, { elevation: 0, actions: isGroupAdmin ? (react_1.default.createElement(GroupSettingsIconButton_1.default, { group: scGroup, user: user, onRemoveSuccess: () => handleRefresh(user.id) })) : ((_a = scUserContext === null || scUserContext === void 0 ? void 0 : scUserContext.user) === null || _a === void 0 ? void 0 : _a.id) !== user.id ? (react_1.default.createElement(material_1.Button, { size: "small", variant: "outlined", component: react_core_1.Link, to: scRoutingContext.url(react_core_1.SCRoutes.USER_PRIVATE_MESSAGES_ROUTE_NAME, user) },
205
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupSettingsIconButton.item.message", defaultMessage: "ui.groupSettingsIconButton.item.message" }))) : null, badgeContent: ((_b = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _b === void 0 ? void 0 : _b.id) === user.id ? (react_1.default.createElement(material_1.Avatar, { className: classes.badge },
206
+ react_1.default.createElement(Icon_1.default, null, "face"))) : null, user: user, userId: user.id })));
207
+ })),
188
208
  state.count > state.visibleItems && (react_1.default.createElement(material_1.Button, { className: classes.showMore, onClick: handleToggleDialogOpen },
189
209
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupMembersWidget.button.showMore", defaultMessage: "ui.groupMembersWidget.button.showMore" }))))),
190
210
  openDialog && (react_1.default.createElement(DialogRoot, Object.assign({ className: classes.dialogRoot, title: react_1.default.createElement(react_intl_1.FormattedMessage, { defaultMessage: "ui.groupMembersWidget.dialogTitle", id: "ui.groupMembersWidget.dialogTitle", values: { total: scGroup.subscribers_counter } }), onClose: handleToggleDialogOpen, open: openDialog }, DialogProps),
191
211
  react_1.default.createElement(InfiniteScroll_1.default, { dataLength: state.results.length, next: handleNext, hasMoreNext: Boolean(state.next), loaderNext: react_1.default.createElement(User_1.UserSkeleton, Object.assign({ elevation: 0 }, UserProps)), height: isMobile ? '100%' : 400, endMessage: react_1.default.createElement(material_1.Typography, { className: classes.endMessage },
192
212
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupMembersWidget.noMoreResults", defaultMessage: "ui.groupMembersWidget.noMoreResults" })) },
193
- react_1.default.createElement(List_1.default, null, state.results.map((user) => (react_1.default.createElement(material_1.ListItem, { key: user.id },
194
- react_1.default.createElement(User_1.default, { elevation: 0, actions: react_1.default.createElement(react_1.default.Fragment, null), user: user, userId: user.id, buttonProps: { onClick: () => console.log(user) } }))))))))),
213
+ react_1.default.createElement(List_1.default, null, state.results.map((user) => {
214
+ var _a, _b;
215
+ return (react_1.default.createElement(material_1.ListItem, { key: user.id },
216
+ react_1.default.createElement(User_1.default, { elevation: 0, actions: isGroupAdmin ? (react_1.default.createElement(GroupSettingsIconButton_1.default, { group: scGroup, user: user, onRemoveSuccess: () => handleRefresh(user.id) })) : ((_a = scUserContext === null || scUserContext === void 0 ? void 0 : scUserContext.user) === null || _a === void 0 ? void 0 : _a.id) !== user.id ? (react_1.default.createElement(material_1.Button, { size: "small", variant: "outlined", component: react_core_1.Link, to: scRoutingContext.url(react_core_1.SCRoutes.USER_PRIVATE_MESSAGES_ROUTE_NAME, user) },
217
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupSettingsIconButton.item.message", defaultMessage: "ui.groupSettingsIconButton.item.message" }))) : null, badgeContent: ((_b = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _b === void 0 ? void 0 : _b.id) === user.id ? (react_1.default.createElement(material_1.Avatar, { className: classes.badge },
218
+ react_1.default.createElement(Icon_1.default, null, "face"))) : null, user: user, userId: user.id })));
219
+ })))))),
195
220
  react_1.default.createElement(material_1.CardActions, { className: classes.actions },
196
221
  react_1.default.createElement(GroupInviteButton_1.default, { groupId: scGroup === null || scGroup === void 0 ? void 0 : scGroup.id, group: scGroup }))));
197
222
  return (react_1.default.createElement(Root, Object.assign({ className: (0, classnames_1.default)(classes.root, className) }, rest), content));
@@ -4,6 +4,7 @@ import { CacheStrategies } from '@selfcommunity/utils';
4
4
  import { BaseDialogProps } from '../../shared/BaseDialog';
5
5
  import { VirtualScrollerItemProps } from '../../types/virtualScroller';
6
6
  import { UserProps } from '../User';
7
+ import { GroupSubscribeButtonProps } from '../GroupSubscribeButton';
7
8
  export interface GroupRequestsWidgetProps extends VirtualScrollerItemProps, WidgetProps {
8
9
  /**
9
10
  * Group Object
@@ -15,11 +16,6 @@ export interface GroupRequestsWidgetProps extends VirtualScrollerItemProps, Widg
15
16
  * @default null
16
17
  */
17
18
  groupId?: number | string;
18
- /**
19
- * Hides this component
20
- * @default false
21
- */
22
- autoHide?: boolean;
23
19
  /**
24
20
  * Limit the number of users to show
25
21
  * @default false
@@ -40,6 +36,16 @@ export interface GroupRequestsWidgetProps extends VirtualScrollerItemProps, Widg
40
36
  * @default {}
41
37
  */
42
38
  DialogProps?: BaseDialogProps;
39
+ /**
40
+ * Props to spread to group subscribe button component
41
+ * @default {}
42
+ */
43
+ GroupSubscribeButtonProps?: GroupSubscribeButtonProps;
44
+ /**
45
+ * onSubscribeAction callback
46
+ * @param user
47
+ */
48
+ onSubscribeActon?: (userId: number | string) => any;
43
49
  /**
44
50
  * Other props
45
51
  */
@@ -20,6 +20,7 @@ const system_1 = require("@mui/system");
20
20
  const HiddenPlaceholder_1 = tslib_1.__importDefault(require("../../shared/HiddenPlaceholder"));
21
21
  const constants_1 = require("./constants");
22
22
  const User_1 = tslib_1.__importStar(require("../User"));
23
+ const GroupSubscribeButton_1 = tslib_1.__importDefault(require("../GroupSubscribeButton"));
23
24
  const classes = {
24
25
  root: `${constants_1.PREFIX}-root`,
25
26
  title: `${constants_1.PREFIX}-title`,
@@ -68,12 +69,13 @@ const DialogRoot = (0, styles_1.styled)(BaseDialog_1.default, {
68
69
  * @param inProps
69
70
  */
70
71
  function GroupRequestsWidget(inProps) {
72
+ var _a;
71
73
  // PROPS
72
74
  const props = (0, system_1.useThemeProps)({
73
75
  props: inProps,
74
76
  name: constants_1.PREFIX
75
77
  });
76
- const { groupId, group, autoHide = false, limit = 5, className, cacheStrategy = utils_1.CacheStrategies.NETWORK_ONLY, onHeightChange, onStateChange, UserProps = {}, DialogProps = {} } = props, rest = tslib_1.__rest(props, ["groupId", "group", "autoHide", "limit", "className", "cacheStrategy", "onHeightChange", "onStateChange", "UserProps", "DialogProps"]);
78
+ const { groupId, group, limit = 5, className, cacheStrategy = utils_1.CacheStrategies.NETWORK_ONLY, onHeightChange, onStateChange, UserProps = {}, DialogProps = {}, GroupSubscribeButtonProps = {}, onSubscribeActon } = props, rest = tslib_1.__rest(props, ["groupId", "group", "limit", "className", "cacheStrategy", "onHeightChange", "onStateChange", "UserProps", "DialogProps", "GroupSubscribeButtonProps", "onSubscribeActon"]);
77
79
  // STATE
78
80
  const [state, dispatch] = (0, react_1.useReducer)(widget_1.dataWidgetReducer, {
79
81
  isLoadingNext: false,
@@ -93,6 +95,7 @@ function GroupRequestsWidget(inProps) {
93
95
  // HOOKS
94
96
  const theme = (0, material_1.useTheme)();
95
97
  const isMobile = (0, material_1.useMediaQuery)(theme.breakpoints.down('md'));
98
+ const isGroupAdmin = (0, react_1.useMemo)(() => { var _a; return scUserContext.user && ((_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id) === scUserContext.user.id; }, [scUserContext.user, (_a = scGroup === null || scGroup === void 0 ? void 0 : scGroup.managed_by) === null || _a === void 0 ? void 0 : _a.id]);
96
99
  /**
97
100
  * Initialize component
98
101
  * Fetch data only if the component is not initialized and it is not loading data
@@ -168,8 +171,17 @@ function GroupRequestsWidget(inProps) {
168
171
  const handleToggleDialogOpen = () => {
169
172
  setOpenDialog((prev) => !prev);
170
173
  };
174
+ const handleSubscribeAction = (0, react_1.useMemo)(() => (user) => {
175
+ const newRequests = [...state.results];
176
+ const _updated = newRequests.filter((u) => u.id !== user.id);
177
+ dispatch({
178
+ type: widget_1.actionWidgetTypes.SET_RESULTS,
179
+ payload: { results: newRequests.length > 1 ? _updated : [] }
180
+ });
181
+ onSubscribeActon && onSubscribeActon(user);
182
+ }, [dispatch, state.count, state.results, onSubscribeActon]);
171
183
  // RENDER
172
- if ((autoHide && !state.count && state.initialized) || (!contentAvailability && !scUserContext.user) || !scGroup) {
184
+ if ((!state.count && state.initialized) || !contentAvailability || !scGroup || !state.count || !state.results.length || !isGroupAdmin) {
173
185
  return react_1.default.createElement(HiddenPlaceholder_1.default, null);
174
186
  }
175
187
  if (!state.initialized) {
@@ -178,17 +190,16 @@ function GroupRequestsWidget(inProps) {
178
190
  const content = (react_1.default.createElement(material_1.CardContent, null,
179
191
  react_1.default.createElement(material_1.Typography, { className: classes.title, variant: "h5" },
180
192
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupRequestsWidget.title", defaultMessage: "ui.groupRequestsWidget.title" })),
181
- !state.count ? (react_1.default.createElement(material_1.Typography, { className: classes.noResults, variant: "body2" },
182
- react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupRequestsWidget.subtitle.noResults", defaultMessage: "" }))) : (react_1.default.createElement(react_1.default.Fragment, null,
193
+ react_1.default.createElement(react_1.default.Fragment, null,
183
194
  react_1.default.createElement(List_1.default, null, state.results.slice(0, state.visibleItems).map((user) => (react_1.default.createElement(material_1.ListItem, { key: user.id },
184
- react_1.default.createElement(User_1.default, { elevation: 0, actions: react_1.default.createElement(react_1.default.Fragment, null), user: user, userId: user.id, buttonProps: { onClick: () => console.log(user) } }))))),
195
+ react_1.default.createElement(User_1.default, { elevation: 0, actions: react_1.default.createElement(GroupSubscribeButton_1.default, Object.assign({ group: scGroup, groupId: scGroup === null || scGroup === void 0 ? void 0 : scGroup.id, user: user, onSubscribe: () => handleSubscribeAction(user) }, GroupSubscribeButtonProps)), user: user, userId: user.id }))))),
185
196
  state.count > state.visibleItems && (react_1.default.createElement(material_1.Button, { className: classes.showMore, onClick: handleToggleDialogOpen },
186
- react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupRequestsWidget.button.showMore", defaultMessage: "ui.groupRequestsWidget.button.showMore" }))))),
197
+ react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupRequestsWidget.button.showMore", defaultMessage: "ui.groupRequestsWidget.button.showMore" })))),
187
198
  openDialog && (react_1.default.createElement(DialogRoot, Object.assign({ className: classes.dialogRoot, title: react_1.default.createElement(react_intl_1.FormattedMessage, { defaultMessage: "ui.groupRequestsWidget.dialogTitle", id: "ui.groupRequestsWidget.dialogTitle", values: { total: scGroup.subscribers_counter } }), onClose: handleToggleDialogOpen, open: openDialog }, DialogProps),
188
199
  react_1.default.createElement(InfiniteScroll_1.default, { dataLength: state.results.length, next: handleNext, hasMoreNext: Boolean(state.next), loaderNext: react_1.default.createElement(User_1.UserSkeleton, Object.assign({ elevation: 0 }, UserProps)), height: isMobile ? '100%' : 400, endMessage: react_1.default.createElement(material_1.Typography, { className: classes.endMessage },
189
200
  react_1.default.createElement(react_intl_1.FormattedMessage, { id: "ui.groupRequestsWidget.noMoreResults", defaultMessage: "ui.groupRequestsWidget.noMoreResults" })) },
190
201
  react_1.default.createElement(List_1.default, null, state.results.map((user) => (react_1.default.createElement(material_1.ListItem, { key: user.id },
191
- react_1.default.createElement(User_1.default, { elevation: 0, actions: react_1.default.createElement(react_1.default.Fragment, null), user: user, userId: user.id, buttonProps: { onClick: () => console.log(user) } }))))))))));
202
+ react_1.default.createElement(User_1.default, { elevation: 0, actions: react_1.default.createElement(GroupSubscribeButton_1.default, Object.assign({ group: scGroup, groupId: scGroup === null || scGroup === void 0 ? void 0 : scGroup.id, user: user, onSubscribe: () => handleSubscribeAction(user) }, GroupSubscribeButtonProps)), user: user, userId: user.id }))))))))));
192
203
  return (react_1.default.createElement(Root, Object.assign({ className: (0, classnames_1.default)(classes.root, className) }, rest), content));
193
204
  }
194
205
  exports.default = GroupRequestsWidget;
@@ -0,0 +1,48 @@
1
+ import { IconButtonProps } from '@mui/material';
2
+ import { SCGroupType, SCUserType } from '@selfcommunity/types';
3
+ export interface GroupSettingsIconButtonProps extends IconButtonProps {
4
+ /**
5
+ * Overrides or extends the styles applied to the component.
6
+ * @default null
7
+ */
8
+ className?: string;
9
+ /**
10
+ * Handles callback on delete success
11
+ */
12
+ onRemoveSuccess?: () => void;
13
+ /**
14
+ * The user
15
+ */
16
+ user: SCUserType;
17
+ /**
18
+ * The group obj
19
+ */
20
+ group: SCGroupType;
21
+ /**
22
+ * Any other properties
23
+ */
24
+ [p: string]: any;
25
+ }
26
+ /**
27
+ * > API documentation for the Community-JS PrivateMessageSettingsIconButton component. Learn about the available props and the CSS API.
28
+
29
+ #### Import
30
+
31
+ ```jsx
32
+ import {PrivateMessageSettingsIconButton} from '@selfcommunity/react-ui';
33
+ ```
34
+
35
+ #### Component Name
36
+
37
+ The name `SCGroupSettingsIconButton` can be used when providing style overrides in the theme.
38
+
39
+
40
+ #### CSS
41
+
42
+ |Rule Name|Global class|Description|
43
+ |---|---|---|
44
+ |root|.SCGroupSettingsIconButton-root|Styles applied to the root element.|
45
+
46
+ * @param inProps
47
+ */
48
+ export default function GroupSettingsIconButton(inProps: GroupSettingsIconButtonProps): JSX.Element;