sforkowany z mirror/soapbox
Use entity store for Group Search
rodzic
a916056367
commit
af9439f1d3
|
@ -40,6 +40,8 @@ const GroupActionButton = ({ group }: IGroupActionButton) => {
|
||||||
|
|
||||||
const onJoinGroup = () => joinGroup.mutate({}, {
|
const onJoinGroup = () => joinGroup.mutate({}, {
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
|
joinGroup.invalidate();
|
||||||
|
|
||||||
toast.success(
|
toast.success(
|
||||||
group.locked
|
group.locked
|
||||||
? intl.formatMessage(messages.joinRequestSuccess)
|
? intl.formatMessage(messages.joinRequestSuccess)
|
||||||
|
@ -53,8 +55,9 @@ const GroupActionButton = ({ group }: IGroupActionButton) => {
|
||||||
heading: intl.formatMessage(messages.confirmationHeading),
|
heading: intl.formatMessage(messages.confirmationHeading),
|
||||||
message: intl.formatMessage(messages.confirmationMessage),
|
message: intl.formatMessage(messages.confirmationMessage),
|
||||||
confirm: intl.formatMessage(messages.confirmationConfirm),
|
confirm: intl.formatMessage(messages.confirmationConfirm),
|
||||||
onConfirm: () => leaveGroup.mutate({}, {
|
onConfirm: () => leaveGroup.mutate(group.relationship?.id as string, {
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
|
leaveGroup.invalidate();
|
||||||
toast.success(intl.formatMessage(messages.leaveSuccess));
|
toast.success(intl.formatMessage(messages.leaveSuccess));
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { FormattedMessage } from 'react-intl';
|
||||||
import { Components, Virtuoso, VirtuosoGrid } from 'react-virtuoso';
|
import { Components, Virtuoso, VirtuosoGrid } from 'react-virtuoso';
|
||||||
|
|
||||||
import { HStack, Icon, Stack, Text } from 'soapbox/components/ui';
|
import { HStack, Icon, Stack, Text } from 'soapbox/components/ui';
|
||||||
import { useGroupSearch } from 'soapbox/queries/groups/search';
|
import { useGroupSearch } from 'soapbox/hooks/api';
|
||||||
import { Group } from 'soapbox/types/entities';
|
import { Group } from 'soapbox/types/entities';
|
||||||
|
|
||||||
import GroupGridItem from '../group-grid-item';
|
import GroupGridItem from '../group-grid-item';
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { FormattedMessage } from 'react-intl';
|
||||||
import { Stack } from 'soapbox/components/ui';
|
import { Stack } from 'soapbox/components/ui';
|
||||||
import PlaceholderGroupSearch from 'soapbox/features/placeholder/components/placeholder-group-search';
|
import PlaceholderGroupSearch from 'soapbox/features/placeholder/components/placeholder-group-search';
|
||||||
import { useDebounce, useOwnAccount } from 'soapbox/hooks';
|
import { useDebounce, useOwnAccount } from 'soapbox/hooks';
|
||||||
import { useGroupSearch } from 'soapbox/queries/groups/search';
|
import { useGroupSearch } from 'soapbox/hooks/api';
|
||||||
import { saveGroupSearch } from 'soapbox/utils/groups';
|
import { saveGroupSearch } from 'soapbox/utils/groups';
|
||||||
|
|
||||||
import Blankslate from './blankslate';
|
import Blankslate from './blankslate';
|
||||||
|
|
|
@ -0,0 +1,39 @@
|
||||||
|
import { Entities } from 'soapbox/entity-store/entities';
|
||||||
|
import { useEntities } from 'soapbox/entity-store/hooks';
|
||||||
|
import { groupSchema } from 'soapbox/schemas';
|
||||||
|
|
||||||
|
import { useApi } from '../../useApi';
|
||||||
|
import { useFeatures } from '../../useFeatures';
|
||||||
|
|
||||||
|
import { useGroupRelationships } from './useGroups';
|
||||||
|
|
||||||
|
import type { Group } from 'soapbox/schemas';
|
||||||
|
|
||||||
|
function useGroupSearch(search: string) {
|
||||||
|
const api = useApi();
|
||||||
|
const features = useFeatures();
|
||||||
|
|
||||||
|
const { entities, ...result } = useEntities<Group>(
|
||||||
|
[Entities.GROUPS, 'discover', 'search', search],
|
||||||
|
() => api.get('/api/v1/groups/search', {
|
||||||
|
params: {
|
||||||
|
q: search,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
{ enabled: features.groupsDiscovery && !!search, schema: groupSchema },
|
||||||
|
);
|
||||||
|
|
||||||
|
const { relationships } = useGroupRelationships(entities.map(entity => entity.id));
|
||||||
|
|
||||||
|
const groups = entities.map((group) => ({
|
||||||
|
...group,
|
||||||
|
relationship: relationships[group.id] || null,
|
||||||
|
}));
|
||||||
|
|
||||||
|
return {
|
||||||
|
...result,
|
||||||
|
groups,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export { useGroupSearch };
|
|
@ -2,9 +2,13 @@ import { Entities } from 'soapbox/entity-store/entities';
|
||||||
import { useEntityActions } from 'soapbox/entity-store/hooks';
|
import { useEntityActions } from 'soapbox/entity-store/hooks';
|
||||||
import { groupRelationshipSchema } from 'soapbox/schemas';
|
import { groupRelationshipSchema } from 'soapbox/schemas';
|
||||||
|
|
||||||
|
import { useGroups } from './useGroups';
|
||||||
|
|
||||||
import type { Group, GroupRelationship } from 'soapbox/schemas';
|
import type { Group, GroupRelationship } from 'soapbox/schemas';
|
||||||
|
|
||||||
function useJoinGroup(group: Group) {
|
function useJoinGroup(group: Group) {
|
||||||
|
const { invalidate } = useGroups();
|
||||||
|
|
||||||
const { createEntity, isLoading } = useEntityActions<GroupRelationship>(
|
const { createEntity, isLoading } = useEntityActions<GroupRelationship>(
|
||||||
[Entities.GROUP_RELATIONSHIPS, group.id],
|
[Entities.GROUP_RELATIONSHIPS, group.id],
|
||||||
{ post: `/api/v1/groups/${group.id}/join` },
|
{ post: `/api/v1/groups/${group.id}/join` },
|
||||||
|
@ -14,6 +18,7 @@ function useJoinGroup(group: Group) {
|
||||||
return {
|
return {
|
||||||
mutate: createEntity,
|
mutate: createEntity,
|
||||||
isLoading,
|
isLoading,
|
||||||
|
invalidate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,14 @@
|
||||||
import { Entities } from 'soapbox/entity-store/entities';
|
import { Entities } from 'soapbox/entity-store/entities';
|
||||||
import { useEntityActions } from 'soapbox/entity-store/hooks';
|
import { useEntityActions } from 'soapbox/entity-store/hooks';
|
||||||
import { Group, GroupRelationship, groupRelationshipSchema } from 'soapbox/schemas';
|
import { groupRelationshipSchema } from 'soapbox/schemas';
|
||||||
|
|
||||||
|
import { useGroups } from './useGroups';
|
||||||
|
|
||||||
|
import type { Group, GroupRelationship } from 'soapbox/schemas';
|
||||||
|
|
||||||
function useLeaveGroup(group: Group) {
|
function useLeaveGroup(group: Group) {
|
||||||
|
const { invalidate } = useGroups();
|
||||||
|
|
||||||
const { createEntity, isLoading } = useEntityActions<GroupRelationship>(
|
const { createEntity, isLoading } = useEntityActions<GroupRelationship>(
|
||||||
[Entities.GROUP_RELATIONSHIPS, group.id],
|
[Entities.GROUP_RELATIONSHIPS, group.id],
|
||||||
{ post: `/api/v1/groups/${group.id}/leave` },
|
{ post: `/api/v1/groups/${group.id}/leave` },
|
||||||
|
@ -12,6 +18,7 @@ function useLeaveGroup(group: Group) {
|
||||||
return {
|
return {
|
||||||
mutate: createEntity,
|
mutate: createEntity,
|
||||||
isLoading,
|
isLoading,
|
||||||
|
invalidate,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,6 +6,7 @@ export { useCancelMembershipRequest } from './groups/useCancelMembershipRequest'
|
||||||
export { useDeleteGroup } from './groups/useDeleteGroup';
|
export { useDeleteGroup } from './groups/useDeleteGroup';
|
||||||
export { useDemoteGroupMember } from './groups/useDemoteGroupMember';
|
export { useDemoteGroupMember } from './groups/useDemoteGroupMember';
|
||||||
export { useGroup, useGroups } from './groups/useGroups';
|
export { useGroup, useGroups } from './groups/useGroups';
|
||||||
|
export { useGroupSearch } from './groups/useGroupSearch';
|
||||||
export { useJoinGroup } from './groups/useJoinGroup';
|
export { useJoinGroup } from './groups/useJoinGroup';
|
||||||
export { useLeaveGroup } from './groups/useLeaveGroup';
|
export { useLeaveGroup } from './groups/useLeaveGroup';
|
||||||
export { usePromoteGroupMember } from './groups/usePromoteGroupMember';
|
export { usePromoteGroupMember } from './groups/usePromoteGroupMember';
|
||||||
|
|
|
@ -8,13 +8,15 @@ import {
|
||||||
fromJS,
|
fromJS,
|
||||||
} from 'immutable';
|
} from 'immutable';
|
||||||
|
|
||||||
|
import { GroupRoles } from 'soapbox/schemas/group-member';
|
||||||
|
|
||||||
export const GroupRelationshipRecord = ImmutableRecord({
|
export const GroupRelationshipRecord = ImmutableRecord({
|
||||||
id: '',
|
id: '',
|
||||||
blocked_by: false,
|
blocked_by: false,
|
||||||
member: false,
|
member: false,
|
||||||
notifying: null,
|
notifying: null,
|
||||||
requested: false,
|
requested: false,
|
||||||
role: null as 'admin' | 'moderator' | 'user' | null,
|
role: 'user' as GroupRoles,
|
||||||
});
|
});
|
||||||
|
|
||||||
export const normalizeGroupRelationship = (relationship: Record<string, any>) => {
|
export const normalizeGroupRelationship = (relationship: Record<string, any>) => {
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
import z from 'zod';
|
import z from 'zod';
|
||||||
|
|
||||||
|
import { GroupRoles } from './group-member';
|
||||||
|
|
||||||
const groupRelationshipSchema = z.object({
|
const groupRelationshipSchema = z.object({
|
||||||
id: z.string(),
|
id: z.string(),
|
||||||
member: z.boolean().catch(false),
|
member: z.boolean().catch(false),
|
||||||
requested: z.boolean().catch(false),
|
requested: z.boolean().catch(false),
|
||||||
role: z.string().nullish().catch(null),
|
role: z.nativeEnum(GroupRoles).catch(GroupRoles.USER),
|
||||||
blocked_by: z.boolean().catch(false),
|
blocked_by: z.boolean().catch(false),
|
||||||
notifying: z.boolean().nullable().catch(null),
|
notifying: z.boolean().nullable().catch(null),
|
||||||
});
|
});
|
||||||
|
|
Ładowanie…
Reference in New Issue