diff --git a/app/css/interceptor.css b/app/css/interceptor.css
index 578f99a5..6d5af223 100644
--- a/app/css/interceptor.css
+++ b/app/css/interceptor.css
@@ -1737,13 +1737,14 @@ header:has(form[role='search']) h1 {
}
.multiline-card {
- --bg-color: #484848;
+ --bg-color: transparent;
--button-color: #77738ccc;
--image-size: 2.25rem;
--min-text-width: 3ch;
--pad-x: 0;
--pad-y: 0;
--gap-x: 0.5rem;
+ --gap-y: 0.0625rem;
--edge-roundness: 3px;
font: inherit;
@@ -1751,7 +1752,7 @@ header:has(form[role='search']) h1 {
grid-template-columns: [left] minmax(0, min-content) [data] minmax(0, max-content) [right];
grid-template-rows: [top] min-content [sub] min-content [bottom];
column-gap: var(--gap-x);
- row-gap: 2px;
+ row-gap: var(--gap-y);
padding-block: var(--pad-y);
padding-inline: var(--pad-x);
background-color: var(--bg-color);
@@ -1770,8 +1771,8 @@ header:has(form[role='search']) h1 {
background: var(--bg-color);
border: 0 none;
padding: 0;
- cursor: pointer;
+ &:not(:disabled) { cursor: pointer }
&:hover, &:focus { background: var(--bg-color) }
}
@@ -1810,7 +1811,13 @@ header:has(form[role='search']) h1 {
}
}
- > data { grid-area: top / left / bottom / right }
+ > data {
+ grid-area: top / left / bottom / right;
+
+ .multiline-card:hover &:not(:has(+:disabled)), .multiline-card:focus-within &:not(:has(+:disabled)) {
+ opacity: 0;
+ }
+ }
> button {
grid-area: top / left / bottom / right;
@@ -1820,8 +1827,11 @@ header:has(form[role='search']) h1 {
background-color: var(--bg-color);
opacity: 0;
outline: none;
+ position: relative;
- &:is(.multiline-card:hover *, .multiline-card:focus-within *) { opacity: 1 }
+ .multiline-card:hover &, .multiline-card:focus-within & {
+ &:not(:disabled) { opacity: 1 }
+ }
&:hover, &:focus {
> span {
diff --git a/app/ts/components/subcomponents/MultilineCard.tsx b/app/ts/components/subcomponents/MultilineCard.tsx
index ca78b88f..5de0b664 100644
--- a/app/ts/components/subcomponents/MultilineCard.tsx
+++ b/app/ts/components/subcomponents/MultilineCard.tsx
@@ -4,88 +4,128 @@ import { Tooltip, TooltipConfig } from './Tooltip.js'
import { clipboardCopy } from './clipboardcopy.js'
import { CopyIcon } from './icons.js'
-export type CardIcon = {
- component: () => JSX.Element
- onClick?: () => void
- tooltipText?: string
-}
-
export type MultilineCardProps = {
- icon: CardIcon
+ icon: ActionableIconProps
label: ActionableTextProps
note: ActionableTextProps
style?: JSX.CSSProperties
}
export const MultilineCard = ({ icon, label, note, style }: MultilineCardProps) => {
+ return (
+
+ )
+}
+
+type ActionableIconAction = {
+ onClick: JSX.MouseEventHandler
+ hintText?: string
+}
+
+export type ActionableIconProps = {
+ icon: () => JSX.Element
+ hintText?: string
+ action: 'clipboard-copy'
+ copyValue: string
+ copySuccessMessage?: string
+} | {
+ icon: () => JSX.Element
+ hintText?: string
+ action?: ActionableIconAction
+}
+
+const ActionableIcon = (props: ActionableIconProps) => {
const tooltipConfig = useSignal(undefined)
const copyTextToClipboard = async (event: JSX.TargetedMouseEvent) => {
event.currentTarget.blur()
await clipboardCopy(event.currentTarget.value)
- tooltipConfig.value = { message: 'Copied!', x: event.clientX, y: event.clientY }
+ const copySuccessMessage = props.action === 'clipboard-copy' && props.copySuccessMessage ? props.copySuccessMessage : 'Copied!'
+ tooltipConfig.value = { message: copySuccessMessage, x: event.clientX, y: event.clientY }
}
- const CardIcon = icon.component
- const defaultAction:TextAction = { label: 'Copy', icon: () => , onClick: copyTextToClipboard }
+ const CardIcon = props.icon
+ const handleClick = props.action ? props.action === 'clipboard-copy' ? copyTextToClipboard : props.action.onClick : undefined
+ const copyValue = props.action === 'clipboard-copy' ? props.copyValue : undefined
+ const hintText = props.action !== 'clipboard-copy' ? props.action?.hintText : undefined
return (
- <>
-
-
- >
+
+
+
)
}
-export type TextAction = {
- label: string
- icon: () => JSX.Element
- onClick?: JSX.MouseEventHandler
+type TextNodeProps = {
+ displayText: string,
+ value: string
}
+const TextNode = ({ displayText, value }: TextNodeProps) => { displayText }
+
export type ActionableTextProps = {
displayText: string
- value?: string
- action: TextAction | 'noaction' | undefined
+ action: 'clipboard-copy'
+ copyValue?: string
+ copySuccessMessage?: string
+} | {
+ displayText: string
+ action?: ActionableTextAction
}
-type TextNodeProps = {
- displayText: string,
- value: string
+type ActionableTextAction = {
+ label: string
+ icon: () => JSX.Element
+ onClick?: JSX.MouseEventHandler
}
-const TextNode = ({ displayText, value }: TextNodeProps) => { displayText }
+const ActionableText = (props: ActionableTextProps) => {
+ const tooltipConfig = useSignal(undefined)
+
+ const copyTextToClipboard = async (event: JSX.TargetedMouseEvent) => {
+ event.currentTarget.blur()
+ await clipboardCopy(event.currentTarget.value)
+ tooltipConfig.value = { message: 'Copied!', x: event.clientX, y: event.clientY }
+ }
+
+ const copyValue = props.action === 'clipboard-copy' && props.copyValue ? props.copyValue : props.displayText
+ const DisplayText = () =>
+
+ const actionIcon = props.action ? props.action === 'clipboard-copy' ? () => : props.action.icon : () => <>>
+ const actionHandler = props.action ? props.action === 'clipboard-copy' ? copyTextToClipboard : props.action.onClick : undefined
+ const actionButtonLabel = props.action === 'clipboard-copy' ? 'Copy' : props.action?.label || ''
-const ActionableText = ({ displayText, value, action }: ActionableTextProps) => {
- const DisplayText = () =>
return (
- { action !== undefined && action !== 'noaction' ? : <>> }
+
+
)
}
-type TextActionProps = {
- icon: () => JSX.Element
+type TextActionProps = ActionableTextAction & {
textNode: () => JSX.Element
- label: string
- onClick?: JSX.MouseEventHandler
+ copyValue: string
}
-const TextAction = ({ textNode: DisplayText, icon: ActionIcon, label, onClick }: TextActionProps) => {
+const TextAction = (props: TextActionProps) => {
+ const DisplayText = props.textNode
+ const ActionIcon = props.icon
+
return (
-