2025-08-08 15:56:20 +01:00
|
|
|
/**
|
|
|
|
* 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';
|
SEO init (#4197)
# 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.
2025-08-14 15:52:38 +01:00
|
|
|
import { useToolUrlRouting } from '../hooks/useToolUrlRouting';
|
|
|
|
import { Tool } from '../types/tool';
|
2025-08-08 15:56:20 +01:00
|
|
|
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;
|
SEO init (#4197)
# 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.
2025-08-14 15:52:38 +01:00
|
|
|
selectedTool: Tool | null;
|
2025-08-08 15:56:20 +01:00
|
|
|
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]);
|
|
|
|
|
SEO init (#4197)
# 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.
2025-08-14 15:52:38 +01:00
|
|
|
// 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: ''
|
|
|
|
});
|
|
|
|
|
2025-08-08 15:56:20 +01:00
|
|
|
// 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;
|