Feature/v2/sign (#4485)

Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
Co-authored-by: James Brunton <james@stirlingpdf.com>
This commit is contained in:
Reece Browne
2025-09-26 19:11:03 +01:00
committed by GitHub
parent abc0988fdf
commit 416d79aed3
33 changed files with 3407 additions and 18 deletions

View File

@@ -0,0 +1,59 @@
import { useTranslation } from 'react-i18next';
import { useToolOperation, ToolOperationHook, ToolType } from '../shared/useToolOperation';
import { SignParameters, DEFAULT_PARAMETERS } from './useSignParameters';
import { createStandardErrorHandler } from '../../../utils/toolErrorHandler';
// Static configuration that can be used by both the hook and automation executor
export const buildSignFormData = (params: SignParameters, file: File): FormData => {
const formData = new FormData();
formData.append('fileInput', file);
// Add signature data if available
if (params.signatureData) {
formData.append('signatureData', params.signatureData);
}
// Add signature position and size
if (params.signaturePosition) {
formData.append('x', params.signaturePosition.x.toString());
formData.append('y', params.signaturePosition.y.toString());
formData.append('width', params.signaturePosition.width.toString());
formData.append('height', params.signaturePosition.height.toString());
formData.append('page', params.signaturePosition.page.toString());
}
// Add signature type
formData.append('signatureType', params.signatureType || 'draw');
// Add other parameters
if (params.reason) {
formData.append('reason', params.reason);
}
if (params.location) {
formData.append('location', params.location);
}
if (params.signerName) {
formData.append('signerName', params.signerName);
}
return formData;
};
// Static configuration object
export const signOperationConfig = {
toolType: ToolType.singleFile,
buildFormData: buildSignFormData,
operationType: 'sign',
endpoint: '/api/v1/security/add-signature',
filePrefix: 'signed_',
defaultParameters: DEFAULT_PARAMETERS,
} as const;
export const useSignOperation = (): ToolOperationHook<SignParameters> => {
const { t } = useTranslation();
return useToolOperation<SignParameters>({
...signOperationConfig,
getErrorMessage: createStandardErrorHandler(t('sign.error.failed', 'An error occurred while signing the PDF.'))
});
};

View File

@@ -0,0 +1,61 @@
import { useBaseParameters } from '../shared/useBaseParameters';
export interface SignaturePosition {
x: number;
y: number;
width: number;
height: number;
page: number;
}
export interface SignParameters {
signatureType: 'image' | 'text' | 'draw' | 'canvas';
signatureData?: string; // Base64 encoded image or text content
signaturePosition?: SignaturePosition;
reason?: string;
location?: string;
signerName?: string;
fontFamily?: string;
fontSize?: number;
}
export const DEFAULT_PARAMETERS: SignParameters = {
signatureType: 'canvas',
reason: 'Document signing',
location: 'Digital',
signerName: '',
fontFamily: 'Helvetica',
fontSize: 16,
};
const validateSignParameters = (parameters: SignParameters): boolean => {
// Basic validation
if (!parameters.signatureType) return false;
// If signature position is set, validate it
if (parameters.signaturePosition) {
const pos = parameters.signaturePosition;
if (pos.x < 0 || pos.y < 0 || pos.width <= 0 || pos.height <= 0 || pos.page < 0) {
return false;
}
}
// For image and canvas signatures, require signature data
if ((parameters.signatureType === 'image' || parameters.signatureType === 'canvas') && !parameters.signatureData) {
return false;
}
// For text signatures, require signer name
if (parameters.signatureType === 'text' && !parameters.signerName) {
return false;
}
return true;
};
export const useSignParameters = () => {
return useBaseParameters<SignParameters>({
defaultParameters: DEFAULT_PARAMETERS,
endpointName: 'add-signature',
validateFn: validateSignParameters,
});
};