0

Add follow+notify to actions menu and refactor how those modals are displayed. Closes #2247

This commit is contained in:
Gabe Kangas 2022-10-23 21:59:25 -07:00
parent dd5d24d3d2
commit 77369a3cbe
No known key found for this signature in database
GPG Key ID: 9A56337728BC81EA
4 changed files with 112 additions and 45 deletions

View File

@ -41,3 +41,22 @@ export const Example = Template.bind({});
Example.args = {
actions,
};
export const ShowFollowExample = Template.bind({});
ShowFollowExample.args = {
actions,
showFollowItem: true,
};
export const ShowNotifyExample = Template.bind({});
ShowNotifyExample.args = {
actions,
showNotifyItem: true,
};
export const ShowNotifyAndFollowExample = Template.bind({});
ShowNotifyAndFollowExample.args = {
actions,
showNotifyItem: true,
showFollowItem: true,
};

View File

@ -1,19 +1,38 @@
import { FC } from 'react';
import { Button, Dropdown, Menu } from 'antd';
import { EllipsisOutlined } from '@ant-design/icons';
import { EllipsisOutlined, HeartOutlined, BellOutlined } from '@ant-design/icons';
import styles from './ActionButtonMenu.module.scss';
import { ExternalAction } from '../../../interfaces/external-action';
const NOTIFY_KEY = 'notify';
const FOLLOW_KEY = 'follow';
export type ActionButtonMenuProps = {
actions: ExternalAction[];
showFollowItem?: boolean;
showNotifyItem?: boolean;
externalActionSelected: (action: ExternalAction) => void;
notifyItemSelected: () => void;
followItemSelected: () => void;
};
export const ActionButtonMenu: FC<ActionButtonMenuProps> = ({
actions,
externalActionSelected,
notifyItemSelected,
followItemSelected,
showFollowItem,
showNotifyItem,
}) => {
const onMenuClick = a => {
if (a.key === NOTIFY_KEY) {
notifyItemSelected();
return;
}
if (a.key === FOLLOW_KEY) {
followItemSelected();
return;
}
const action = actions.find(x => x.url === a.key);
externalActionSelected(action);
};
@ -28,6 +47,29 @@ export const ActionButtonMenu: FC<ActionButtonMenuProps> = ({
),
}));
if (showFollowItem) {
items.unshift({
key: FOLLOW_KEY,
label: (
<span className={styles.item}>
<HeartOutlined className={styles.icon} /> Follow this stream
</span>
),
});
}
if (showNotifyItem) {
items.unshift({
key: NOTIFY_KEY,
label: (
<span className={styles.item}>
<BellOutlined className={styles.icon} />
Notify when live
</span>
),
});
}
const menu = <Menu items={items} onClick={onMenuClick} />;
return (

View File

@ -1,40 +1,22 @@
import { Button, ButtonProps } from 'antd';
import { HeartFilled } from '@ant-design/icons';
import { FC, useState } from 'react';
import { useRecoilValue } from 'recoil';
import { Modal } from '../ui/Modal/Modal';
import { FollowModal } from '../modals/FollowModal/FollowModal';
import { FC } from 'react';
import styles from './ActionButton/ActionButton.module.scss';
import { clientConfigStateAtom } from '../stores/ClientConfigStore';
import { ClientConfig } from '../../interfaces/client-config.model';
export type FollowButtonProps = ButtonProps;
export const FollowButton: FC<FollowButtonProps> = props => {
const [showModal, setShowModal] = useState(false);
const clientConfig = useRecoilValue<ClientConfig>(clientConfigStateAtom);
const { name, federation } = clientConfig;
const { account } = federation;
return (
<>
<Button
{...props}
type="primary"
className={styles.button}
icon={<HeartFilled />}
onClick={() => setShowModal(true)}
>
Follow
</Button>
<Modal
title={`Follow ${name}`}
open={showModal}
handleCancel={() => setShowModal(false)}
width="550px"
>
<FollowModal account={account} name={name} handleClose={() => setShowModal(false)} />
</Modal>
</>
);
export type FollowButtonProps = ButtonProps & {
onClick?: () => void;
props?: ButtonProps;
};
export const FollowButton: FC<FollowButtonProps> = ({ onClick, props }) => (
<Button
{...props}
type="primary"
className={styles.button}
icon={<HeartFilled />}
onClick={onClick}
>
Follow
</Button>
);

View File

@ -37,6 +37,7 @@ import { FollowerCollection } from '../followers/FollowerCollection/FollowerColl
import { ExternalAction } from '../../../interfaces/external-action';
import { Modal } from '../Modal/Modal';
import { ActionButtonMenu } from '../../action-buttons/ActionButtonMenu/ActionButtonMenu';
import { FollowModal } from '../../modals/FollowModal/FollowModal';
const { Content: AntContent } = Layout;
@ -116,6 +117,8 @@ const MobileContent = ({
showChat,
actions,
setExternalActionToDisplay,
setShowNotifyPopup,
setShowFollowModal,
}) => {
if (!currentUser) {
return null;
@ -160,7 +163,14 @@ const MobileContent = ({
const replacementTabBar = (props, DefaultTabBar) => (
<div style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'start' }}>
<DefaultTabBar {...props} style={{ width: '85%' }} />
<ActionButtonMenu actions={actions} externalActionSelected={setExternalActionToDisplay} />
<ActionButtonMenu
showFollowItem
showNotifyItem
actions={actions}
externalActionSelected={setExternalActionToDisplay}
notifyItemSelected={() => setShowNotifyPopup(true)}
followItemSelected={() => setShowFollowModal(true)}
/>
</div>
);
@ -211,7 +221,8 @@ export const Content: FC = () => {
notifications,
} = clientConfig;
const [showNotifyReminder, setShowNotifyReminder] = useState(false);
const [showNotifyPopup, setShowNotifyPopup] = useState(false);
const [showNotifyModal, setShowNotifyModal] = useState(false);
const [showFollowModal, setShowFollowModal] = useState(false);
const { account: fediverseAccount } = federation;
const { browser: browserNotifications } = notifications;
const { enabled: browserNotificationsEnabled } = browserNotifications;
@ -248,7 +259,7 @@ export const Content: FC = () => {
};
const disableNotifyReminderPopup = () => {
setShowNotifyPopup(false);
setShowNotifyModal(false);
setShowNotifyReminder(false);
setLocalStorage(LOCAL_STORAGE_KEYS.hasDisplayedNotificationModal, true);
};
@ -289,7 +300,7 @@ export const Content: FC = () => {
notificationsEnabled={browserNotificationsEnabled}
fediverseAccount={fediverseAccount}
lastLive={lastDisconnectTime}
onNotifyClick={() => setShowNotifyPopup(true)}
onNotifyClick={() => setShowNotifyModal(true)}
/>
)}
{online && (
@ -306,20 +317,20 @@ export const Content: FC = () => {
{!isMobile && (
<ActionButtonRow>
{externalActionButtons}
<FollowButton size="small" />
<FollowButton size="small" onClick={() => setShowFollowModal(true)} />
<NotifyReminderPopup
open={showNotifyReminder}
notificationClicked={() => setShowNotifyPopup(true)}
notificationClicked={() => setShowNotifyModal(true)}
notificationClosed={() => disableNotifyReminderPopup()}
>
<NotifyButton onClick={() => setShowNotifyPopup(true)} />
<NotifyButton onClick={() => setShowNotifyModal(true)} />
</NotifyReminderPopup>
</ActionButtonRow>
)}
<Modal
title="Notify"
open={showNotifyPopup}
open={showNotifyModal}
afterClose={() => disableNotifyReminderPopup()}
handleCancel={() => disableNotifyReminderPopup()}
>
@ -340,6 +351,8 @@ export const Content: FC = () => {
showChat={showChat}
actions={externalActions}
setExternalActionToDisplay={externalActionSelected}
setShowNotifyPopup={setShowNotifyModal}
setShowFollowModal={setShowFollowModal}
/>
) : (
<DesktopContent
@ -357,13 +370,24 @@ export const Content: FC = () => {
</Spin>
{!isMobile && false && <Footer version={version} />}
</div>
{externalActionToDisplay && (
<ExternalModal
externalActionToDisplay={externalActionToDisplay}
setExternalActionToDisplay={setExternalActionToDisplay}
/>
)}
<Modal
title={`Follow ${name}`}
open={showFollowModal}
handleCancel={() => setShowFollowModal(false)}
width="550px"
>
<FollowModal
account={fediverseAccount}
name={name}
handleClose={() => setShowFollowModal(false)}
/>
</Modal>
</>
);
};