mirror of
https://github.com/Stirling-Tools/Stirling-PDF.git
synced 2025-08-23 04:39:24 +00:00

# Description of Changes This pull request introduces dynamic document meta tag management and improves URL routing and tool metadata handling in the frontend. The most significant changes are the addition of a custom hook to update document meta tags (including OpenGraph tags) based on the selected tool, enhancements to the tool selection context for better URL synchronization, and enrichment of the `Tool` type and tool registry with more metadata. **Dynamic document meta management:** * Added a new `useDocumentMeta` hook that updates the page's `<title>`, description, and OpenGraph meta tags based on the currently selected tool, and restores the original values on cleanup. (`frontend/src/hooks/useDocumentMeta.ts`) * Integrated `useDocumentMeta` into `HomePageContent` so the document's meta tags dynamically reflect the selected tool's metadata, improving SEO and social sharing. (`frontend/src/pages/HomePage.tsx`) [[1]](diffhunk://#diff-85c26b21681286c20e97a26a4912f0b91812776c9d4d0c54aa541fded2565c7eR2-R8) [[2]](diffhunk://#diff-85c26b21681286c20e97a26a4912f0b91812776c9d4d0c54aa541fded2565c7eR17) [[3]](diffhunk://#diff-85c26b21681286c20e97a26a4912f0b91812776c9d4d0c54aa541fded2565c7eR28-R37) **Tool metadata and context improvements:** * Enhanced the `Tool` type and tool registry to include `title` and `description` fields, which are now translated and used throughout the UI and meta tags. (`frontend/src/types/tool.ts`, `frontend/src/hooks/useToolManagement.tsx`) [[1]](diffhunk://#diff-0b557df7bd27ac90cd2f925ddd8ef8096ea2decfaee9a5c12a94dc7a03c64bfaR46) [[2]](diffhunk://#diff-57f8a6b3e75ecaec10ad445b01afe8fccc376af6f8ad4d693c68cf98e8863273L116-R118) * Updated the `ToolWorkflowContext` to use the new `Tool` type for `selectedTool`, replacing the previous `ToolConfiguration` type. (`frontend/src/contexts/ToolWorkflowContext.tsx`) [[1]](diffhunk://#diff-9b36e2c06dddbcfba6cb66fd0b303b7860f88ca8b562bb2534af1ab50390d385L6-R8) [[2]](diffhunk://#diff-9b36e2c06dddbcfba6cb66fd0b303b7860f88ca8b562bb2534af1ab50390d385L72-R72) **URL routing and synchronization:** * Implemented logic in `ToolWorkflowContext` to synchronize the selected tool with the browser URL, initialize tool selection from the URL on load, and handle browser navigation (back/forward) for tool selection. (`frontend/src/contexts/ToolWorkflowContext.tsx`) --- ## Checklist ### General - [ ] I have read the [Contribution Guidelines](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/CONTRIBUTING.md) - [ ] I have read the [Stirling-PDF Developer Guide](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/DeveloperGuide.md) (if applicable) - [ ] 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) - [ ] I have performed a self-review of my own code - [ ] 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.
236 lines
7.4 KiB
TypeScript
236 lines
7.4 KiB
TypeScript
/**
|
|
* ToolWorkflowContext - Manages tool selection, UI state, and workflow coordination
|
|
* Eliminates prop drilling with a single, simple context
|
|
*/
|
|
|
|
import React, { createContext, useContext, useReducer, useCallback, useMemo } from 'react';
|
|
import { useToolManagement } from '../hooks/useToolManagement';
|
|
import { useToolUrlRouting } from '../hooks/useToolUrlRouting';
|
|
import { Tool } from '../types/tool';
|
|
import { PageEditorFunctions } from '../types/pageEditor';
|
|
|
|
// State interface
|
|
interface ToolWorkflowState {
|
|
// UI State
|
|
sidebarsVisible: boolean;
|
|
leftPanelView: 'toolPicker' | 'toolContent';
|
|
readerMode: boolean;
|
|
|
|
// File/Preview State
|
|
previewFile: File | null;
|
|
pageEditorFunctions: PageEditorFunctions | null;
|
|
|
|
// Search State
|
|
searchQuery: string;
|
|
}
|
|
|
|
// Actions
|
|
type ToolWorkflowAction =
|
|
| { type: 'SET_SIDEBARS_VISIBLE'; payload: boolean }
|
|
| { type: 'SET_LEFT_PANEL_VIEW'; payload: 'toolPicker' | 'toolContent' }
|
|
| { type: 'SET_READER_MODE'; payload: boolean }
|
|
| { type: 'SET_PREVIEW_FILE'; payload: File | null }
|
|
| { type: 'SET_PAGE_EDITOR_FUNCTIONS'; payload: PageEditorFunctions | null }
|
|
| { type: 'SET_SEARCH_QUERY'; payload: string }
|
|
| { type: 'RESET_UI_STATE' };
|
|
|
|
// Initial state
|
|
const initialState: ToolWorkflowState = {
|
|
sidebarsVisible: true,
|
|
leftPanelView: 'toolPicker',
|
|
readerMode: false,
|
|
previewFile: null,
|
|
pageEditorFunctions: null,
|
|
searchQuery: '',
|
|
};
|
|
|
|
// Reducer
|
|
function toolWorkflowReducer(state: ToolWorkflowState, action: ToolWorkflowAction): ToolWorkflowState {
|
|
switch (action.type) {
|
|
case 'SET_SIDEBARS_VISIBLE':
|
|
return { ...state, sidebarsVisible: action.payload };
|
|
case 'SET_LEFT_PANEL_VIEW':
|
|
return { ...state, leftPanelView: action.payload };
|
|
case 'SET_READER_MODE':
|
|
return { ...state, readerMode: action.payload };
|
|
case 'SET_PREVIEW_FILE':
|
|
return { ...state, previewFile: action.payload };
|
|
case 'SET_PAGE_EDITOR_FUNCTIONS':
|
|
return { ...state, pageEditorFunctions: action.payload };
|
|
case 'SET_SEARCH_QUERY':
|
|
return { ...state, searchQuery: action.payload };
|
|
case 'RESET_UI_STATE':
|
|
return { ...initialState, searchQuery: state.searchQuery }; // Preserve search
|
|
default:
|
|
return state;
|
|
}
|
|
}
|
|
|
|
// Context value interface
|
|
interface ToolWorkflowContextValue extends ToolWorkflowState {
|
|
// Tool management (from hook)
|
|
selectedToolKey: string | null;
|
|
selectedTool: Tool | null;
|
|
toolRegistry: any; // From useToolManagement
|
|
|
|
// UI Actions
|
|
setSidebarsVisible: (visible: boolean) => void;
|
|
setLeftPanelView: (view: 'toolPicker' | 'toolContent') => void;
|
|
setReaderMode: (mode: boolean) => void;
|
|
setPreviewFile: (file: File | null) => void;
|
|
setPageEditorFunctions: (functions: PageEditorFunctions | null) => void;
|
|
setSearchQuery: (query: string) => void;
|
|
|
|
// Tool Actions
|
|
selectTool: (toolId: string) => void;
|
|
clearToolSelection: () => void;
|
|
|
|
// Workflow Actions (compound actions)
|
|
handleToolSelect: (toolId: string) => void;
|
|
handleBackToTools: () => void;
|
|
handleReaderToggle: () => void;
|
|
|
|
// Computed values
|
|
filteredTools: [string, any][]; // Filtered by search
|
|
isPanelVisible: boolean;
|
|
}
|
|
|
|
const ToolWorkflowContext = createContext<ToolWorkflowContextValue | undefined>(undefined);
|
|
|
|
// Provider component
|
|
interface ToolWorkflowProviderProps {
|
|
children: React.ReactNode;
|
|
/** Handler for view changes (passed from parent) */
|
|
onViewChange?: (view: string) => void;
|
|
}
|
|
|
|
export function ToolWorkflowProvider({ children, onViewChange }: ToolWorkflowProviderProps) {
|
|
const [state, dispatch] = useReducer(toolWorkflowReducer, initialState);
|
|
|
|
// Tool management hook
|
|
const {
|
|
selectedToolKey,
|
|
selectedTool,
|
|
toolRegistry,
|
|
selectTool,
|
|
clearToolSelection,
|
|
} = useToolManagement();
|
|
|
|
// UI Action creators
|
|
const setSidebarsVisible = useCallback((visible: boolean) => {
|
|
dispatch({ type: 'SET_SIDEBARS_VISIBLE', payload: visible });
|
|
}, []);
|
|
|
|
const setLeftPanelView = useCallback((view: 'toolPicker' | 'toolContent') => {
|
|
dispatch({ type: 'SET_LEFT_PANEL_VIEW', payload: view });
|
|
}, []);
|
|
|
|
const setReaderMode = useCallback((mode: boolean) => {
|
|
dispatch({ type: 'SET_READER_MODE', payload: mode });
|
|
}, []);
|
|
|
|
const setPreviewFile = useCallback((file: File | null) => {
|
|
dispatch({ type: 'SET_PREVIEW_FILE', payload: file });
|
|
}, []);
|
|
|
|
const setPageEditorFunctions = useCallback((functions: PageEditorFunctions | null) => {
|
|
dispatch({ type: 'SET_PAGE_EDITOR_FUNCTIONS', payload: functions });
|
|
}, []);
|
|
|
|
const setSearchQuery = useCallback((query: string) => {
|
|
dispatch({ type: 'SET_SEARCH_QUERY', payload: query });
|
|
}, []);
|
|
|
|
// Workflow actions (compound actions that coordinate multiple state changes)
|
|
const handleToolSelect = useCallback((toolId: string) => {
|
|
selectTool(toolId);
|
|
onViewChange?.('fileEditor');
|
|
setLeftPanelView('toolContent');
|
|
setReaderMode(false);
|
|
}, [selectTool, onViewChange, setLeftPanelView, setReaderMode]);
|
|
|
|
const handleBackToTools = useCallback(() => {
|
|
setLeftPanelView('toolPicker');
|
|
setReaderMode(false);
|
|
clearToolSelection();
|
|
}, [setLeftPanelView, setReaderMode, clearToolSelection]);
|
|
|
|
const handleReaderToggle = useCallback(() => {
|
|
setReaderMode(true);
|
|
}, [setReaderMode]);
|
|
|
|
// URL routing functionality
|
|
const { getToolUrlSlug, getToolKeyFromSlug } = useToolUrlRouting({
|
|
selectedToolKey,
|
|
toolRegistry,
|
|
selectTool,
|
|
clearToolSelection,
|
|
// During initial load, we want the full UI side-effects (like before):
|
|
onInitSelect: handleToolSelect,
|
|
// For back/forward nav, keep it lightweight like before (selection only):
|
|
onPopStateSelect: selectTool,
|
|
// If your app serves under a subpath, provide basePath here (e.g., '/app')
|
|
// basePath: ''
|
|
});
|
|
|
|
// Filter tools based on search query
|
|
const filteredTools = useMemo(() => {
|
|
if (!toolRegistry) return [];
|
|
return Object.entries(toolRegistry).filter(([_, { name }]) =>
|
|
name.toLowerCase().includes(state.searchQuery.toLowerCase())
|
|
);
|
|
}, [toolRegistry, state.searchQuery]);
|
|
|
|
const isPanelVisible = useMemo(() =>
|
|
state.sidebarsVisible && !state.readerMode,
|
|
[state.sidebarsVisible, state.readerMode]
|
|
);
|
|
|
|
// Simple context value with basic memoization
|
|
const contextValue = useMemo((): ToolWorkflowContextValue => ({
|
|
// State
|
|
...state,
|
|
selectedToolKey,
|
|
selectedTool,
|
|
toolRegistry,
|
|
|
|
// Actions
|
|
setSidebarsVisible,
|
|
setLeftPanelView,
|
|
setReaderMode,
|
|
setPreviewFile,
|
|
setPageEditorFunctions,
|
|
setSearchQuery,
|
|
selectTool,
|
|
clearToolSelection,
|
|
|
|
// Workflow Actions
|
|
handleToolSelect,
|
|
handleBackToTools,
|
|
handleReaderToggle,
|
|
|
|
// Computed
|
|
filteredTools,
|
|
isPanelVisible,
|
|
}), [state, selectedToolKey, selectedTool, toolRegistry, filteredTools, isPanelVisible]);
|
|
|
|
return (
|
|
<ToolWorkflowContext.Provider value={contextValue}>
|
|
{children}
|
|
</ToolWorkflowContext.Provider>
|
|
);
|
|
}
|
|
|
|
// Custom hook to use the context
|
|
export function useToolWorkflow(): ToolWorkflowContextValue {
|
|
const context = useContext(ToolWorkflowContext);
|
|
if (!context) {
|
|
throw new Error('useToolWorkflow must be used within a ToolWorkflowProvider');
|
|
}
|
|
return context;
|
|
}
|
|
|
|
// Convenience exports for specific use cases (optional - components can use useToolWorkflow directly)
|
|
export const useToolSelection = useToolWorkflow;
|
|
export const useToolPanelState = useToolWorkflow;
|
|
export const useWorkbenchState = useToolWorkflow; |