Merge pull request #4994 from h3poteto/iss-4917/context
refs #4917 Implement context-menu on status
This commit is contained in:
commit
74802e09d1
|
@ -31,6 +31,10 @@
|
||||||
"bookmark": "Bookmark",
|
"bookmark": "Bookmark",
|
||||||
"emoji_reaction": "Emoji reaction",
|
"emoji_reaction": "Emoji reaction",
|
||||||
"detail": "Details"
|
"detail": "Details"
|
||||||
|
},
|
||||||
|
"context_menu": {
|
||||||
|
"copy_link": "Copy link",
|
||||||
|
"open_detail": "Open original page"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"mark_as_read": "Mark as read"
|
"mark_as_read": "Mark as read"
|
||||||
|
|
|
@ -6,9 +6,9 @@ import Poll from '../status/Poll'
|
||||||
import Card from '../status/Card'
|
import Card from '../status/Card'
|
||||||
import Media from '../status/Media'
|
import Media from '../status/Media'
|
||||||
import { FaBarsProgress, FaHouse, FaPenToSquare, FaRetweet, FaStar } from 'react-icons/fa6'
|
import { FaBarsProgress, FaHouse, FaPenToSquare, FaRetweet, FaStar } from 'react-icons/fa6'
|
||||||
import { useIntl } from 'react-intl'
|
import { FormattedMessage, useIntl } from 'react-intl'
|
||||||
import { useState } from 'react'
|
import { MouseEventHandler, useState } from 'react'
|
||||||
import { Avatar } from '@material-tailwind/react'
|
import { Avatar, List, ListItem, Card as MaterialCard } from '@material-tailwind/react'
|
||||||
import { useRouter } from 'next/router'
|
import { useRouter } from 'next/router'
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
|
@ -37,6 +37,33 @@ export default function Reaction(props: Props) {
|
||||||
router.push({ query: { id: router.query.id, timeline: router.query.timeline, user_id: id, detail: true } })
|
router.push({ query: { id: router.query.id, timeline: router.query.timeline, user_id: id, detail: true } })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const onContextMenu: MouseEventHandler<HTMLDivElement> = e => {
|
||||||
|
e.preventDefault()
|
||||||
|
hideOthers()
|
||||||
|
const context = document.getElementById(`context-${status.id}`)
|
||||||
|
if (context) {
|
||||||
|
context.style.left = `${e.clientX}px`
|
||||||
|
context.style.top = `${e.clientY}px`
|
||||||
|
context.style.display = 'block'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClick: MouseEventHandler<HTMLDivElement> = e => {
|
||||||
|
e.preventDefault()
|
||||||
|
hideOthers()
|
||||||
|
}
|
||||||
|
|
||||||
|
const hideOthers = () => {
|
||||||
|
const menu = document.getElementsByClassName('context-menu')
|
||||||
|
for (let i = 0; i < menu.length; i++) {
|
||||||
|
;(menu[i] as HTMLElement).style.display = 'none'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const copyLink = () => {
|
||||||
|
navigator.clipboard.writeText(status.url)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="border-b border-gray-200 dark:border-gray-800 mr-2 py-1">
|
<div className="border-b border-gray-200 dark:border-gray-800 mr-2 py-1">
|
||||||
<div className="flex items-center">
|
<div className="flex items-center">
|
||||||
|
@ -71,7 +98,12 @@ export default function Reaction(props: Props) {
|
||||||
alt={formatMessage({ id: 'timeline.status.avatar' }, { user: status.account.username })}
|
alt={formatMessage({ id: 'timeline.status.avatar' }, { user: status.account.username })}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className="text-gray-600 dark:text-gray-500 break-all overflow-hidden" style={{ width: 'calc(100% - 56px)' }}>
|
<div
|
||||||
|
className="text-gray-600 dark:text-gray-500 break-all overflow-hidden"
|
||||||
|
style={{ width: 'calc(100% - 56px)' }}
|
||||||
|
onContextMenu={onContextMenu}
|
||||||
|
onClick={onClick}
|
||||||
|
>
|
||||||
<div className="flex justify-between">
|
<div className="flex justify-between">
|
||||||
<div className="flex cursor-pointer" onClick={() => openUser(status.account.id)}>
|
<div className="flex cursor-pointer" onClick={() => openUser(status.account.id)}>
|
||||||
<span
|
<span
|
||||||
|
@ -92,6 +124,18 @@ export default function Reaction(props: Props) {
|
||||||
<Media media={status.media_attachments} sensitive={status.sensitive} openMedia={props.openMedia} />
|
<Media media={status.media_attachments} sensitive={status.sensitive} openMedia={props.openMedia} />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
<div className="fixed hidden context-menu z-50" id={`context-${status.id}`}>
|
||||||
|
<MaterialCard className="rounded-md bg-white dark:bg-gray-800">
|
||||||
|
<List className="my-2 p-0">
|
||||||
|
<ListItem className="ground rounded-none py-1.5 px-3 text-sm" onClick={copyLink}>
|
||||||
|
<FormattedMessage id="timeline.status.context_menu.copy_link" />
|
||||||
|
</ListItem>
|
||||||
|
<ListItem className="ground rounded-none py-1.5 px-3 text-sm" onClick={openStatus}>
|
||||||
|
<FormattedMessage id="timeline.status.context_menu.open_detail" />
|
||||||
|
</ListItem>
|
||||||
|
</List>
|
||||||
|
</MaterialCard>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -11,7 +11,7 @@ import { useRouter } from 'next/router'
|
||||||
import { MouseEventHandler, useState } from 'react'
|
import { MouseEventHandler, useState } from 'react'
|
||||||
import { findAccount, findLink, ParsedAccount, accountMatch, findTag } from '@/utils/statusParser'
|
import { findAccount, findLink, ParsedAccount, accountMatch, findTag } from '@/utils/statusParser'
|
||||||
import { Account } from '@/db'
|
import { Account } from '@/db'
|
||||||
import { Avatar } from '@material-tailwind/react'
|
import { Avatar, List, ListItem, Card as MaterialCard } from '@material-tailwind/react'
|
||||||
import { invoke } from '@/utils/invoke'
|
import { invoke } from '@/utils/invoke'
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
|
@ -88,6 +88,33 @@ export default function Status(props: Props) {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const onContextMenu: MouseEventHandler<HTMLDivElement> = e => {
|
||||||
|
e.preventDefault()
|
||||||
|
hideOthers()
|
||||||
|
const context = document.getElementById(`context-${props.status.id}`)
|
||||||
|
if (context) {
|
||||||
|
context.style.left = `${e.clientX}px`
|
||||||
|
context.style.top = `${e.clientY}px`
|
||||||
|
context.style.display = 'block'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onClick: MouseEventHandler<HTMLDivElement> = e => {
|
||||||
|
e.preventDefault()
|
||||||
|
hideOthers()
|
||||||
|
}
|
||||||
|
|
||||||
|
const hideOthers = () => {
|
||||||
|
const menu = document.getElementsByClassName('context-menu')
|
||||||
|
for (let i = 0; i < menu.length; i++) {
|
||||||
|
;(menu[i] as HTMLElement).style.display = 'none'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const copyLink = () => {
|
||||||
|
navigator.clipboard.writeText(status.url)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="border-b border-gray-200 dark:border-gray-800 mr-2 py-1">
|
<div className="border-b border-gray-200 dark:border-gray-800 mr-2 py-1">
|
||||||
{rebloggedHeader(
|
{rebloggedHeader(
|
||||||
|
@ -114,7 +141,12 @@ export default function Status(props: Props) {
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div className="text-gray-950 dark:text-gray-300 break-all overflow-hidden" style={{ width: 'calc(100% - 56px)' }}>
|
<div
|
||||||
|
className="text-gray-950 dark:text-gray-300 break-all overflow-hidden"
|
||||||
|
style={{ width: 'calc(100% - 56px)' }}
|
||||||
|
onContextMenu={onContextMenu}
|
||||||
|
onClick={onClick}
|
||||||
|
>
|
||||||
<div className="flex justify-between">
|
<div className="flex justify-between">
|
||||||
<div className="flex cursor-pointer" onClick={() => openUser(status.account.id)}>
|
<div className="flex cursor-pointer" onClick={() => openUser(status.account.id)}>
|
||||||
<span
|
<span
|
||||||
|
@ -145,6 +177,18 @@ export default function Status(props: Props) {
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
<div className="fixed hidden context-menu z-50" id={`context-${status.id}`}>
|
||||||
|
<MaterialCard className="rounded-md bg-white dark:bg-gray-800">
|
||||||
|
<List className="my-2 p-0">
|
||||||
|
<ListItem className="ground rounded-none py-1.5 px-3 text-sm" onClick={copyLink}>
|
||||||
|
<FormattedMessage id="timeline.status.context_menu.copy_link" />
|
||||||
|
</ListItem>
|
||||||
|
<ListItem className="ground rounded-none py-1.5 px-3 text-sm" onClick={openStatus}>
|
||||||
|
<FormattedMessage id="timeline.status.context_menu.open_detail" />
|
||||||
|
</ListItem>
|
||||||
|
</List>
|
||||||
|
</MaterialCard>
|
||||||
|
</div>
|
||||||
|
|
||||||
<Actions status={status} client={props.client} account={props.account} onRefresh={onRefresh} />
|
<Actions status={status} client={props.client} account={props.account} onRefresh={onRefresh} />
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Reference in New Issue