mirror of
https://github.com/Frooodle/Stirling-PDF.git
synced 2026-03-04 02:20:19 +01:00
# Description of Changes <!-- Please provide a summary of the changes, including: - What was changed - Why the change was made - Any challenges encountered Closes #(issue_number) --> --- ## Checklist ### General - [x] I have read the [Contribution Guidelines](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/CONTRIBUTING.md) - [x] I have read the [Stirling-PDF Developer Guide](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/DeveloperGuide.md) (if applicable) - [x] I have read the [How to add new languages to Stirling-PDF](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/HowToAddNewLanguage.md) (if applicable) - [x] I have performed a self-review of my own code - [x] My changes generate no new warnings ### Documentation - [ ] I have updated relevant docs on [Stirling-PDF's doc repo](https://github.com/Stirling-Tools/Stirling-Tools.github.io/blob/main/docs/) (if functionality has heavily changed) - [ ] I have read the section [Add New Translation Tags](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/HowToAddNewLanguage.md#add-new-translation-tags) (for new translation tags only) ### UI Changes (if applicable) - [ ] Screenshots or videos demonstrating the UI changes are attached (e.g., as comments or direct attachments in the PR) ### Testing (if applicable) - [ ] I have tested my changes locally. Refer to the [Testing Guide](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/DeveloperGuide.md#6-testing) for more details.
137 lines
4.3 KiB
TypeScript
137 lines
4.3 KiB
TypeScript
import React from "react";
|
|
import { Button } from "@mantine/core";
|
|
import { Tooltip } from "../../shared/Tooltip";
|
|
import { ToolRegistryEntry } from "../../../data/toolsTaxonomy";
|
|
import { useToolNavigation } from "../../../hooks/useToolNavigation";
|
|
import { handleUnlessSpecialClick } from "../../../utils/clickHandlers";
|
|
import FitText from "../../shared/FitText";
|
|
|
|
interface ToolButtonProps {
|
|
id: string;
|
|
tool: ToolRegistryEntry;
|
|
isSelected: boolean;
|
|
onSelect: (id: string) => void;
|
|
rounded?: boolean;
|
|
disableNavigation?: boolean;
|
|
matchedSynonym?: string;
|
|
}
|
|
|
|
const ToolButton: React.FC<ToolButtonProps> = ({ id, tool, isSelected, onSelect, disableNavigation = false, matchedSynonym }) => {
|
|
const isUnavailable = !tool.component && !tool.link;
|
|
const { getToolNavigation } = useToolNavigation();
|
|
|
|
const handleClick = (id: string) => {
|
|
if (isUnavailable) return;
|
|
if (tool.link) {
|
|
// Open external link in new tab
|
|
window.open(tool.link, '_blank', 'noopener,noreferrer');
|
|
return;
|
|
}
|
|
// Normal tool selection
|
|
onSelect(id);
|
|
};
|
|
|
|
// Get navigation props for URL support (only if navigation is not disabled)
|
|
const navProps = !isUnavailable && !tool.link && !disableNavigation ? getToolNavigation(id, tool) : null;
|
|
|
|
const tooltipContent = isUnavailable
|
|
? (<span><strong>Coming soon:</strong> {tool.description}</span>)
|
|
: tool.description;
|
|
|
|
const buttonContent = (
|
|
<>
|
|
<div className="tool-button-icon" style={{ color: "var(--tools-text-and-icon-color)", marginRight: "0.5rem", transform: "scale(0.8)", transformOrigin: "center", opacity: isUnavailable ? 0.25 : 1 }}>{tool.icon}</div>
|
|
<div style={{ display: 'flex', flexDirection: 'column', alignItems: 'flex-start', flex: 1, overflow: 'visible' }}>
|
|
<FitText
|
|
text={tool.name}
|
|
lines={1}
|
|
minimumFontScale={0.8}
|
|
as="span"
|
|
style={{ display: 'inline-block', maxWidth: '100%', opacity: isUnavailable ? 0.25 : 1 }}
|
|
/>
|
|
{matchedSynonym && (
|
|
<span style={{
|
|
fontSize: '0.75rem',
|
|
color: 'var(--mantine-color-dimmed)',
|
|
opacity: isUnavailable ? 0.25 : 1,
|
|
marginTop: '1px',
|
|
overflow: 'visible',
|
|
whiteSpace: 'nowrap'
|
|
}}>
|
|
{matchedSynonym}
|
|
</span>
|
|
)}
|
|
</div>
|
|
</>
|
|
);
|
|
|
|
const handleExternalClick = (e: React.MouseEvent) => {
|
|
handleUnlessSpecialClick(e, () => handleClick(id));
|
|
};
|
|
|
|
const buttonElement = navProps ? (
|
|
// For internal tools with URLs, render Button as an anchor for proper link behavior
|
|
<Button
|
|
component="a"
|
|
href={navProps.href}
|
|
onClick={navProps.onClick}
|
|
variant={isSelected ? "filled" : "subtle"}
|
|
size="sm"
|
|
radius="md"
|
|
fullWidth
|
|
justify="flex-start"
|
|
className="tool-button"
|
|
styles={{
|
|
root: { borderRadius: 0, color: "var(--tools-text-and-icon-color)", overflow: 'visible' },
|
|
label: { overflow: 'visible' }
|
|
}}
|
|
>
|
|
{buttonContent}
|
|
</Button>
|
|
) : tool.link && !isUnavailable ? (
|
|
// For external links, render Button as an anchor with proper href
|
|
<Button
|
|
component="a"
|
|
href={tool.link}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
onClick={handleExternalClick}
|
|
variant={isSelected ? "filled" : "subtle"}
|
|
size="sm"
|
|
radius="md"
|
|
fullWidth
|
|
justify="flex-start"
|
|
className="tool-button"
|
|
styles={{
|
|
root: { borderRadius: 0, color: "var(--tools-text-and-icon-color)", overflow: 'visible' },
|
|
label: { overflow: 'visible' }
|
|
}}
|
|
>
|
|
{buttonContent}
|
|
</Button>
|
|
) : (
|
|
// For unavailable tools, use regular button
|
|
<Button
|
|
variant={isSelected ? "filled" : "subtle"}
|
|
onClick={() => handleClick(id)}
|
|
size="sm"
|
|
radius="md"
|
|
fullWidth
|
|
justify="flex-start"
|
|
className="tool-button"
|
|
aria-disabled={isUnavailable}
|
|
styles={{ root: { borderRadius: 0, color: "var(--tools-text-and-icon-color)", cursor: isUnavailable ? 'not-allowed' : undefined, overflow: 'visible' }, label: { overflow: 'visible' } }}
|
|
>
|
|
{buttonContent}
|
|
</Button>
|
|
);
|
|
|
|
return (
|
|
<Tooltip content={tooltipContent} position="right" arrow={true} delay={500}>
|
|
{buttonElement}
|
|
</Tooltip>
|
|
);
|
|
};
|
|
|
|
export default ToolButton;
|