mirror of
https://github.com/spacebarchat/client.git
synced 2024-11-25 11:42:30 +01:00
wip context menus
This commit is contained in:
parent
71e47c96cd
commit
7a907d38ec
@ -1,5 +1,7 @@
|
|||||||
import { PresenceUpdateStatus } from "@spacebarchat/spacebar-api-types/v9";
|
import { PresenceUpdateStatus } from "@spacebarchat/spacebar-api-types/v9";
|
||||||
|
import { useContext } from "react";
|
||||||
import styled from "styled-components";
|
import styled from "styled-components";
|
||||||
|
import { ContextMenuContext } from "../../contexts/ContextMenuContext";
|
||||||
import { useAppStore } from "../../stores/AppStore";
|
import { useAppStore } from "../../stores/AppStore";
|
||||||
import GuildMember from "../../stores/objects/GuildMember";
|
import GuildMember from "../../stores/objects/GuildMember";
|
||||||
import Avatar from "../Avatar";
|
import Avatar from "../Avatar";
|
||||||
@ -62,6 +64,7 @@ interface Props {
|
|||||||
function MemberListItem({ item }: Props) {
|
function MemberListItem({ item }: Props) {
|
||||||
const app = useAppStore();
|
const app = useAppStore();
|
||||||
const presence = app.presences.get(item.user!.id);
|
const presence = app.presences.get(item.user!.id);
|
||||||
|
const contextMenu = useContext(ContextMenuContext);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Floating
|
<Floating
|
||||||
@ -73,7 +76,11 @@ function MemberListItem({ item }: Props) {
|
|||||||
member: item,
|
member: item,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<ListItem key={item.user?.id}>
|
<ListItem
|
||||||
|
key={item.user?.id}
|
||||||
|
ref={contextMenu.setReferenceElement}
|
||||||
|
onContextMenu={(e) => contextMenu.onContextMenu(e, { user: item.user! })}
|
||||||
|
>
|
||||||
<Container>
|
<Container>
|
||||||
<Wrapper offline={presence?.status === PresenceUpdateStatus.Offline}>
|
<Wrapper offline={presence?.status === PresenceUpdateStatus.Offline}>
|
||||||
<AvatarWrapper>
|
<AvatarWrapper>
|
||||||
|
66
src/components/contextMenus/ContextMenu.tsx
Normal file
66
src/components/contextMenus/ContextMenu.tsx
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
// modified from https://github.com/revoltchat/frontend/blob/master/components/app/menus/ContextMenu.tsx
|
||||||
|
// changed some styling
|
||||||
|
|
||||||
|
import { ComponentProps } from "react";
|
||||||
|
import styled from "styled-components";
|
||||||
|
import Icon, { IconProps } from "../Icon";
|
||||||
|
|
||||||
|
export const ContextMenu = styled.div`
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding: 6px 8px;
|
||||||
|
|
||||||
|
overflow: hidden;
|
||||||
|
border-radius: 4px;
|
||||||
|
background: var(--background-tertiary);
|
||||||
|
color: var(--text);
|
||||||
|
|
||||||
|
box-shadow: 0 0 10px 0 rgba(0, 0, 0, 0.5);
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const ContextMenuDivider = styled.div`
|
||||||
|
height: 1px;
|
||||||
|
margin: 4px;
|
||||||
|
background: var(--text);
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const ContextMenuItem = styled("a")`
|
||||||
|
display: block;
|
||||||
|
padding: 6px 8px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 14px;
|
||||||
|
margin: 2px 0;
|
||||||
|
cursor: pointer;
|
||||||
|
`;
|
||||||
|
|
||||||
|
const ButtonBase = styled(ContextMenuItem)<{ destructive?: boolean }>`
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 8px;
|
||||||
|
|
||||||
|
> span {
|
||||||
|
margin-top: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background: ${(props) => (props.destructive ? "var(--danger)" : "var(--primary)")};
|
||||||
|
${(props) => (props.destructive ? `color: var(--text)` : "")}
|
||||||
|
}
|
||||||
|
|
||||||
|
${(props) => (props.destructive ? `fill: var(--danger); color: var(--danger)` : "")}
|
||||||
|
`;
|
||||||
|
|
||||||
|
type ButtonProps = ComponentProps<typeof ContextMenuItem> & {
|
||||||
|
icon?: IconProps["icon"];
|
||||||
|
destructive?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function ContextMenuButton(props: ButtonProps) {
|
||||||
|
return (
|
||||||
|
<ButtonBase {...props}>
|
||||||
|
<span>{props.children}</span>
|
||||||
|
{props.icon && <Icon icon={props.icon} size="18px" />}
|
||||||
|
</ButtonBase>
|
||||||
|
);
|
||||||
|
}
|
32
src/components/contextMenus/UserContextMenu.tsx
Normal file
32
src/components/contextMenus/UserContextMenu.tsx
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
// loosely based on https://github.com/revoltchat/frontend/blob/master/components/app/menus/UserContextMenu.tsx
|
||||||
|
|
||||||
|
import User from "../../stores/objects/User";
|
||||||
|
import { ContextMenu, ContextMenuButton, ContextMenuDivider } from "./ContextMenu";
|
||||||
|
|
||||||
|
interface MenuProps {
|
||||||
|
user: User;
|
||||||
|
}
|
||||||
|
|
||||||
|
function UserContextMenu({ user }: MenuProps) {
|
||||||
|
/**
|
||||||
|
* Copy user id to clipboard
|
||||||
|
*/
|
||||||
|
function copyId() {
|
||||||
|
navigator.clipboard.writeText(user.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ContextMenu>
|
||||||
|
<ContextMenuButton icon="mdiAt" destructive>
|
||||||
|
Mention
|
||||||
|
</ContextMenuButton>
|
||||||
|
<ContextMenuDivider />
|
||||||
|
|
||||||
|
<ContextMenuButton icon="mdiIdentifier" onClick={copyId}>
|
||||||
|
Copy user ID
|
||||||
|
</ContextMenuButton>
|
||||||
|
</ContextMenu>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default UserContextMenu;
|
56
src/contexts/ContextMenuContext.tsx
Normal file
56
src/contexts/ContextMenuContext.tsx
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
|
import { FloatingPortal, useFloating } from "@floating-ui/react";
|
||||||
|
import React from "react";
|
||||||
|
import UserContextMenu from "../components/contextMenus/UserContextMenu";
|
||||||
|
import useContextMenu from "../hooks/useContextMenu";
|
||||||
|
import User from "../stores/objects/User";
|
||||||
|
|
||||||
|
interface MenuProps {
|
||||||
|
user: User;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ContextMenuContextType = {
|
||||||
|
setReferenceElement: ReturnType<typeof useFloating>["refs"]["setReference"];
|
||||||
|
onContextMenu: (e: React.MouseEvent, props: MenuProps) => void;
|
||||||
|
close: () => void;
|
||||||
|
open: (user: User) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
// @ts-expect-error not specifying a default value here
|
||||||
|
export const ContextMenuContext = React.createContext<ContextMenuContextType>();
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
export const ContextMenuContextProvider: React.FC<any> = ({ children }) => {
|
||||||
|
const contextMenu = useContextMenu("user");
|
||||||
|
|
||||||
|
const open = (user: User) => {
|
||||||
|
contextMenu.open({
|
||||||
|
user,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ContextMenuContext.Provider
|
||||||
|
value={{
|
||||||
|
close: contextMenu.close,
|
||||||
|
open,
|
||||||
|
setReferenceElement: contextMenu.refs.setReference,
|
||||||
|
onContextMenu: contextMenu.onContextMenu,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<FloatingPortal>
|
||||||
|
{contextMenu.isOpen && (
|
||||||
|
<div
|
||||||
|
className="ContextMenu"
|
||||||
|
ref={contextMenu.refs.setFloating}
|
||||||
|
style={contextMenu.floatingStyles}
|
||||||
|
{...contextMenu.getFloatingProps()}
|
||||||
|
>
|
||||||
|
<UserContextMenu {...(contextMenu.props as any)} />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</FloatingPortal>
|
||||||
|
</ContextMenuContext.Provider>
|
||||||
|
);
|
||||||
|
};
|
92
src/hooks/useContextMenu.tsx
Normal file
92
src/hooks/useContextMenu.tsx
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
import { autoUpdate, flip, offset, shift, useDismiss, useFloating, useInteractions, useRole } from "@floating-ui/react";
|
||||||
|
import { useMemo, useState } from "react";
|
||||||
|
import User from "../stores/objects/User";
|
||||||
|
|
||||||
|
interface MenuProps {
|
||||||
|
user: User;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function (type: "user") {
|
||||||
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
|
const [props, setProps] = useState<MenuProps | null>(null);
|
||||||
|
|
||||||
|
const data = useFloating({
|
||||||
|
placement: "right-start",
|
||||||
|
strategy: "fixed",
|
||||||
|
open: isOpen && props !== null,
|
||||||
|
onOpenChange: setIsOpen,
|
||||||
|
whileElementsMounted: autoUpdate,
|
||||||
|
middleware: [
|
||||||
|
offset(5),
|
||||||
|
flip({
|
||||||
|
fallbackPlacements: ["left-start"],
|
||||||
|
}),
|
||||||
|
shift({
|
||||||
|
padding: 8,
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
const context = data.context;
|
||||||
|
|
||||||
|
const role = useRole(context, { role: "menu" });
|
||||||
|
const dismiss = useDismiss(context);
|
||||||
|
// const listNavigation = useListNavigation(context, {
|
||||||
|
// listRef: listItemsRef,
|
||||||
|
// onNavigate: setActiveIndex,
|
||||||
|
// activeIndex
|
||||||
|
// });
|
||||||
|
// const typeahead = useTypeahead(context, {
|
||||||
|
// enabled: isOpen,
|
||||||
|
// listRef: listContentRef,
|
||||||
|
// onMatch: setActiveIndex,
|
||||||
|
// activeIndex
|
||||||
|
// });
|
||||||
|
|
||||||
|
const interactions = useInteractions([dismiss, role]);
|
||||||
|
|
||||||
|
const open = (props: MenuProps) => {
|
||||||
|
setProps(props);
|
||||||
|
setIsOpen(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const close = () => {
|
||||||
|
setIsOpen(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
function onContextMenu(e: React.MouseEvent, props: MenuProps) {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
data.refs.setPositionReference({
|
||||||
|
getBoundingClientRect() {
|
||||||
|
return {
|
||||||
|
width: 0,
|
||||||
|
height: 0,
|
||||||
|
x: e.clientX,
|
||||||
|
y: e.clientY,
|
||||||
|
top: e.clientY,
|
||||||
|
right: e.clientX,
|
||||||
|
bottom: e.clientY,
|
||||||
|
left: e.clientX,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
setProps(props);
|
||||||
|
setIsOpen(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
return useMemo(
|
||||||
|
() => ({
|
||||||
|
isOpen,
|
||||||
|
props,
|
||||||
|
setProps,
|
||||||
|
open,
|
||||||
|
close,
|
||||||
|
onContextMenu,
|
||||||
|
...interactions,
|
||||||
|
...data,
|
||||||
|
}),
|
||||||
|
[isOpen, props, setProps, open, close, onContextMenu, interactions, data],
|
||||||
|
);
|
||||||
|
}
|
@ -19,6 +19,7 @@ import { BrowserRouter } from "react-router-dom";
|
|||||||
import { ErrorBoundaryContext } from "react-use-error-boundary";
|
import { ErrorBoundaryContext } from "react-use-error-boundary";
|
||||||
import App from "./App";
|
import App from "./App";
|
||||||
import { BannerContextProvider } from "./contexts/BannerContext";
|
import { BannerContextProvider } from "./contexts/BannerContext";
|
||||||
|
import { ContextMenuContextProvider } from "./contexts/ContextMenuContext";
|
||||||
import Theme from "./contexts/Theme";
|
import Theme from "./contexts/Theme";
|
||||||
import ModalRenderer from "./controllers/modals/ModalRenderer";
|
import ModalRenderer from "./controllers/modals/ModalRenderer";
|
||||||
import "./index.css";
|
import "./index.css";
|
||||||
@ -31,8 +32,10 @@ ReactDOM.createRoot(document.getElementById("root") as HTMLElement).render(
|
|||||||
<ErrorBoundaryContext>
|
<ErrorBoundaryContext>
|
||||||
<BrowserRouter>
|
<BrowserRouter>
|
||||||
<BannerContextProvider>
|
<BannerContextProvider>
|
||||||
<App />
|
<ContextMenuContextProvider>
|
||||||
<ModalRenderer />
|
<App />
|
||||||
|
<ModalRenderer />
|
||||||
|
</ContextMenuContextProvider>
|
||||||
</BannerContextProvider>
|
</BannerContextProvider>
|
||||||
<Theme />
|
<Theme />
|
||||||
</BrowserRouter>
|
</BrowserRouter>
|
||||||
|
Loading…
Reference in New Issue
Block a user