@superdoc-dev/esign 1.0.0-next.3 → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,6 +1,6 @@
1
1
  # @superdoc-dev/esign
2
2
 
3
- React eSignature component for document signing workflows with SuperDoc.
3
+ React component that wraps SuperDoc for document signing workflows with audit trails and compliance tracking.
4
4
 
5
5
  ## Installation
6
6
 
@@ -11,445 +11,130 @@ npm install @superdoc-dev/esign superdoc
11
11
  ## Quick Start
12
12
 
13
13
  ```jsx
14
+ import React from 'react';
14
15
  import SuperDocESign from '@superdoc-dev/esign';
15
16
  import 'superdoc/dist/style.css';
16
17
 
17
18
  function App() {
19
+ const handleSubmit = async (data) => {
20
+ // Send to your backend
21
+ await fetch('/api/sign', {
22
+ method: 'POST',
23
+ headers: { 'Content-Type': 'application/json' },
24
+ body: JSON.stringify(data)
25
+ });
26
+ alert('Document signed!');
27
+ };
28
+
18
29
  return (
19
30
  <SuperDocESign
20
- eventId="unique-session-id"
31
+ eventId={`session-${Date.now()}`}
21
32
 
22
33
  document={{
23
- source: "contract.pdf",
24
- mode: "full",
25
- displayOptions: {
26
- scrollRequired: true
27
- }
34
+ source: "https://storage.googleapis.com/public_static_hosting/public_demo_docs/employment_agreement.docx",
35
+ validation: { scroll: { required: true } }
28
36
  }}
29
37
 
30
38
  fields={{
31
39
  document: [
32
- { alias: 'company', value: 'Acme Corp' }
40
+ { id: 'employee_name', value: 'Jane Smith' },
41
+ { id: 'position', value: 'Senior Engineer' },
42
+ { id: 'salary', value: '$120,000' }
33
43
  ],
34
44
  signer: [
35
45
  {
36
46
  id: 'signature',
37
47
  type: 'signature',
38
48
  validation: { required: true },
39
- label: 'Your Signature'
49
+ label: 'Type your full name'
50
+ },
51
+ {
52
+ id: 'accept_terms',
53
+ type: 'checkbox',
54
+ validation: { required: true },
55
+ label: 'I accept the terms'
40
56
  }
41
57
  ]
42
58
  }}
43
59
 
44
- onSubmit={async (data) => {
45
- await api.saveSignature(data);
46
- }}
60
+ onSubmit={handleSubmit}
47
61
  />
48
62
  );
49
63
  }
50
64
  ```
51
65
 
52
- ## API Reference
53
-
54
- ### Props
55
-
56
- | Prop | Type | Required | Description |
57
- |------|------|----------|-------------|
58
- | `eventId` | string | ✓ | Unique identifier for signing session |
59
- | `document` | object | ✓ | Document configuration |
60
- | `fields` | object | | Field definitions |
61
- | `download` | object | | Download button config |
62
- | `submit` | object | | Submit button config |
63
- | `onSubmit` | function | ✓ | Submit handler |
64
- | `onDownload` | function | | Download handler |
65
- | `onStateChange` | function | | State change handler |
66
- | `onFieldChange` | function | | Field change handler |
67
- | `isDisabled` | boolean | | Disable all interactions |
68
-
69
- ### Document Configuration
70
-
71
- ```jsx
72
- document={{
73
- source: File | Blob | string, // Required
74
- mode: 'full' | 'download_only' | 'preview',
75
- displayOptions: {
76
- scrollRequired: boolean,
77
- watermark: boolean
78
- }
79
- }}
80
- ```
66
+ ## Backend - Create Signed PDF
81
67
 
82
- ### Field Reference System
68
+ Use the SuperDoc API to create the final signed document:
83
69
 
84
- Fields can be referenced using either `id` or `alias`:
85
-
86
- - **`id`**: System identifier (e.g., "field_123")
87
- - **`alias`**: Human-readable name (e.g., "customer_name")
88
-
89
- This allows the same value to appear multiple places in a document:
90
-
91
- ```jsx
92
- // Document template might have:
93
- // "Dear {{customer_name}},"
94
- // "Agreement between {{company}} and {{customer_name}}"
95
- // "Signed by: {{customer_name}}"
96
-
97
- fields={{
98
- document: [
99
- {
100
- alias: 'customer_name', // Used in document {{customer_name}}
101
- value: 'John Doe'
102
- },
103
- {
104
- alias: 'company',
105
- value: 'Acme Corp'
106
- },
107
- {
108
- id: 'date_field',
109
- alias: 'sign_date', // Can have both id and alias
110
- value: '2024-01-15'
111
- }
112
- ],
70
+ ```javascript
71
+ // Node.js/Express
72
+ app.post('/api/sign', async (req, res) => {
73
+ const { eventId, auditTrail, documentFields, signerFields } = req.body;
113
74
 
114
- signer: [
115
- {
116
- id: 'sig_001',
117
- alias: 'signature', // Document shows {{signature}}
118
- type: 'signature',
119
- validation: { required: true }
75
+ // 1. Fill document fields
76
+ const annotated = await fetch('https://api.superdoc.dev/v1/annotate', {
77
+ method: 'POST',
78
+ headers: {
79
+ 'Authorization': `Bearer ${API_KEY}`,
80
+ 'Content-Type': 'application/json'
120
81
  },
121
- {
122
- id: 'consent_terms',
123
- alias: 'terms_accepted', // Can reference as {{terms_accepted}}
124
- type: 'consent',
125
- label: 'I accept the terms'
126
- }
127
- ]
128
- }}
129
- ```
130
-
131
- **How it works:**
132
- 1. Document fields are injected on load using their id/alias
133
- 2. Signer fields are collected and can replace placeholders using their alias
134
- 3. The same alias can appear multiple times in the document
135
- 4. All instances get the same value
136
-
137
- ### Field Configuration
138
-
139
- ```jsx
140
- fields={{
141
- // Values to inject into document
142
- document: [
143
- {
144
- id: 'field_id', // Optional
145
- alias: 'field_name', // Optional (at least one of id or alias required)
146
- value: 'any value'
147
- }
148
- ],
82
+ body: JSON.stringify({
83
+ document: 'template.docx',
84
+ fields: [...documentFields, ...signerFields]
85
+ })
86
+ });
149
87
 
150
- // Fields for signer to complete
151
- signer: [
152
- {
153
- id: 'unique_id', // Required for signer fields
154
- alias: 'field_ref', // Optional alias for document references
155
- type: 'signature' | 'consent' | 'checkbox' | 'text',
156
- validation: {
157
- required: boolean,
158
- minLength: number,
159
- maxLength: number
160
- },
161
- label: 'Field Label',
162
- component: CustomComponent // Optional - consistent with download/submit!
163
- }
164
- ]
165
- }}
166
- ```
167
-
168
- ### UI Customization
169
-
170
- Simple customization:
171
- ```jsx
172
- download={{
173
- fileName: "contract.pdf",
174
- label: "Download Contract",
175
- variant: "secondary"
176
- }}
177
-
178
- submit={{
179
- label: "Sign Document",
180
- loadingLabel: "Processing...",
181
- variant: "primary"
182
- }}
183
- ```
184
-
185
- Full custom components:
186
- ```jsx
187
- download={{
188
- component: ({ onClick, fileName, isDisabled }) => (
189
- <CustomButton onClick={onClick} disabled={isDisabled}>
190
- Download {fileName}
191
- </CustomButton>
192
- )
193
- }}
194
-
195
- submit={{
196
- component: ({ onClick, isValid, isDisabled, isSubmitting }) => (
197
- <CustomButton
198
- onClick={onClick}
199
- disabled={!isValid || isDisabled}
200
- >
201
- {isSubmitting ? 'Processing...' : 'Sign & Submit'}
202
- </CustomButton>
203
- )
204
- }}
205
- ```
206
-
207
- ### Custom Field Components
208
-
209
- ```jsx
210
- // Custom signature pad component
211
- const CustomSignaturePad = ({ value, onChange, isDisabled, label }) => (
212
- <div className="custom-signature">
213
- <label>{label}</label>
214
- <canvas
215
- // ... canvas signature implementation
216
- onMouseUp={(e) => onChange(canvasDataURL)}
217
- style={{ border: '1px solid #000', cursor: 'crosshair' }}
218
- />
219
- </div>
220
- );
221
-
222
- // Use it in fields
223
- fields={{
224
- signer: [
225
- {
226
- id: 'signature',
227
- type: 'signature',
228
- validation: { required: true },
229
- component: CustomSignaturePad // Consistent with download/submit!
230
- }
231
- ]
232
- }}
233
- ```
234
-
235
- ## Examples
236
-
237
- ### Basic Agreement
238
-
239
- ```jsx
240
- <SuperDocESign
241
- eventId="session-123"
242
- document={{ source: "terms.pdf" }}
243
- fields={{
244
- signer: [
245
- {
246
- id: 'accept',
247
- type: 'consent',
248
- validation: { required: true },
249
- label: 'I accept the terms'
250
- }
251
- ]
252
- }}
253
- onSubmit={handleSubmit}
254
- />
255
- ```
256
-
257
- ### Employment Offer with Repeated Fields
258
-
259
- ```jsx
260
- // Example: Employment offer with repeated fields
261
- <SuperDocESign
262
- eventId="offer-123"
263
-
264
- document={{
265
- source: offerLetter, // Contains {{employee_name}}, {{salary}}, etc.
266
- mode: "full",
267
- displayOptions: {
268
- scrollRequired: true
269
- }
270
- }}
271
-
272
- fields={{
273
- document: [
274
- {
275
- alias: 'employee_name', // Appears 5 times in document
276
- value: 'Jane Smith'
277
- },
278
- {
279
- alias: 'position', // Appears 3 times
280
- value: 'Senior Engineer'
281
- },
282
- {
283
- alias: 'salary', // Appears 2 times
284
- value: '$120,000'
285
- },
286
- {
287
- alias: 'start_date',
288
- value: 'February 1, 2024'
289
- }
290
- ],
291
-
292
- signer: [
293
- {
294
- id: 'employee_signature',
295
- alias: 'employee_sig', // {{employee_sig}} in document
296
- type: 'signature',
297
- validation: { required: true },
298
- label: 'Your Signature'
299
- },
300
- {
301
- id: 'accept_offer',
302
- alias: 'offer_accepted',
303
- type: 'consent',
304
- validation: { required: true },
305
- label: 'I accept this offer'
306
- }
307
- ]
308
- }}
309
-
310
- submit={{
311
- label: 'Accept Offer'
312
- }}
313
-
314
- onSubmit={handleAcceptOffer}
315
- />
316
- ```
317
-
318
- ### Full Contract with All Features
319
-
320
- ```jsx
321
- <SuperDocESign
322
- eventId="contract-456"
323
-
324
- document={{
325
- source: serviceAgreement,
326
- mode: "full",
327
- displayOptions: {
328
- scrollRequired: true,
329
- watermark: true
330
- }
331
- }}
332
-
333
- fields={{
334
- document: [
335
- { alias: 'client_name', value: 'John Doe' },
336
- { alias: 'company', value: 'Acme Corp' },
337
- { alias: 'service_type', value: 'Premium Support' },
338
- { alias: 'contract_date', value: new Date().toLocaleDateString() }
339
- ],
340
- signer: [
341
- {
342
- id: 'client_signature',
343
- alias: 'signature',
344
- type: 'signature',
345
- validation: { required: true },
346
- label: 'Your Signature'
347
- },
348
- {
349
- id: 'consent_terms',
350
- type: 'consent',
351
- validation: { required: true },
352
- label: 'I agree to the terms and conditions'
353
- },
354
- {
355
- id: 'consent_privacy',
356
- type: 'consent',
357
- validation: { required: true },
358
- label: 'I acknowledge the privacy policy'
359
- },
360
- {
361
- id: 'email_updates',
362
- type: 'checkbox',
363
- validation: { required: false },
364
- label: 'Send me email updates'
365
- }
366
- ]
367
- }}
368
-
369
- download={{
370
- fileName: 'service_agreement_signed.pdf',
371
- label: 'Download Agreement',
372
- variant: 'secondary'
373
- }}
374
-
375
- submit={{
376
- label: 'Sign Agreement',
377
- loadingLabel: 'Processing...',
378
- invalidLabel: 'Please complete all required fields',
379
- variant: 'success'
380
- }}
381
-
382
- onSubmit={async (data) => {
383
- await api.saveSignedContract(data);
384
- console.log('Signed by:', data.signerFields);
385
- }}
386
-
387
- onDownload={(blob, fileName) => {
388
- console.log('Downloaded:', fileName);
389
- }}
88
+ // 2. Add digital signature
89
+ const signed = await fetch('https://api.superdoc.dev/v1/sign', {
90
+ method: 'POST',
91
+ headers: {
92
+ 'Authorization': `Bearer ${API_KEY}`,
93
+ 'Content-Type': 'application/json'
94
+ },
95
+ body: JSON.stringify({
96
+ document: await annotated.blob(),
97
+ auditTrail: auditTrail
98
+ })
99
+ });
390
100
 
391
- onStateChange={(state) => {
392
- console.log('Valid:', state.isValid);
393
- }}
101
+ // 3. Save PDF
102
+ await saveToStorage(await signed.blob(), `signed_${eventId}.pdf`);
394
103
 
395
- isDisabled={false}
396
- />
104
+ res.json({ success: true });
105
+ });
397
106
  ```
398
107
 
399
- ## Progressive Customization
400
-
401
- The component supports three levels of customization:
108
+ See [Python, Ruby, and more examples](https://docs.superdoc.dev/solutions/esign/backend).
402
109
 
403
- ```jsx
404
- // Level 1: Use all defaults
405
- <SuperDocESign
406
- eventId="session-1"
407
- document={{ source: "doc.pdf" }}
408
- onSubmit={handleSubmit}
409
- />
410
-
411
- // Level 2: Simple customization
412
- <SuperDocESign
413
- eventId="session-2"
414
- document={{ source: "doc.pdf" }}
415
- submit={{ label: "I Agree", variant: "success" }}
416
- download={{ label: "Get Copy" }}
417
- onSubmit={handleSubmit}
418
- />
110
+ ## What You Receive
419
111
 
420
- // Level 3: Full custom components
421
- <SuperDocESign
422
- eventId="session-3"
423
- document={{ source: "doc.pdf" }}
424
- submit={{ component: CustomSubmitButton }}
425
- download={{ component: CustomDownloadButton }}
426
- fields={{
427
- signer: [{
428
- id: 'sig',
429
- type: 'signature',
430
- component: CustomSignaturePad
431
- }]
432
- }}
433
- onSubmit={handleSubmit}
434
- />
112
+ ```javascript
113
+ {
114
+ eventId: "session-123",
115
+ timestamp: "2024-01-15T10:30:00Z",
116
+ duration: 45000,
117
+ documentFields: [
118
+ { id: "employee_name", value: "Jane Smith" }
119
+ ],
120
+ signerFields: [
121
+ { id: "signature", value: "Jane Smith" },
122
+ { id: "accept_terms", value: true }
123
+ ],
124
+ auditTrail: [
125
+ { type: "ready", timestamp: "..." },
126
+ { type: "scroll", timestamp: "..." },
127
+ { type: "field_change", timestamp: "..." },
128
+ { type: "submit", timestamp: "..." }
129
+ ],
130
+ isFullyCompleted: true
131
+ }
435
132
  ```
436
133
 
437
- ## TypeScript
438
-
439
- Full TypeScript support with exported types:
134
+ ## Documentation
440
135
 
441
- ```typescript
442
- import SuperDocESign from '@superdoc-dev/esign';
443
- import type {
444
- SuperDocESignProps,
445
- SubmitData,
446
- SigningState,
447
- DocumentField,
448
- SignerField,
449
- FieldComponentProps
450
- } from '@superdoc-dev/esign';
451
- ```
136
+ Full docs at [docs.superdoc.dev/solutions/esign](https://docs.superdoc.dev/solutions/esign)
452
137
 
453
138
  ## License
454
139
 
455
- MIT
140
+ AGPLv3
@@ -0,0 +1,4 @@
1
+ import { default as React } from 'react';
2
+ import { FieldComponentProps } from '../types';
3
+ export declare const CheckboxInput: React.FC<FieldComponentProps>;
4
+ //# sourceMappingURL=CheckboxInput.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"CheckboxInput.d.ts","sourceRoot":"","sources":["../../src/defaults/CheckboxInput.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;AAEpD,eAAO,MAAM,aAAa,EAAE,KAAK,CAAC,EAAE,CAAC,mBAAmB,CAoBvD,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"SignatureInput.d.ts","sourceRoot":"","sources":["../../src/defaults/SignatureInput.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAmB,MAAM,OAAO,CAAC;AACxC,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;AAEpD,eAAO,MAAM,cAAc,EAAE,KAAK,CAAC,EAAE,CAAC,mBAAmB,CAiDxD,CAAC"}
1
+ {"version":3,"file":"SignatureInput.d.ts","sourceRoot":"","sources":["../../src/defaults/SignatureInput.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;AAEpD,eAAO,MAAM,cAAc,EAAE,KAAK,CAAC,EAAE,CAAC,mBAAmB,CAsBxD,CAAC"}
@@ -1,5 +1,5 @@
1
1
  export { SignatureInput } from './SignatureInput';
2
- export { ConsentCheckbox } from './ConsentCheckbox';
2
+ export { CheckboxInput } from './CheckboxInput';
3
3
  export { createDownloadButton } from './DownloadButton';
4
4
  export { createSubmitButton } from './SubmitButton';
5
5
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/defaults/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAClD,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACpD,OAAO,EAAE,oBAAoB,EAAE,MAAM,kBAAkB,CAAC;AACxD,OAAO,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/defaults/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAClD,OAAO,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAChD,OAAO,EAAE,oBAAoB,EAAE,MAAM,kBAAkB,CAAC;AACxD,OAAO,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAC"}
package/dist/index.d.ts CHANGED
@@ -1,8 +1,7 @@
1
- import { default as React } from 'react';
2
- import { SignatureInput, ConsentCheckbox } from './defaults';
1
+ import { SignatureInput, CheckboxInput } from './defaults';
3
2
  import type * as Types from "./types";
4
3
  export * from './types';
5
- export { SignatureInput, ConsentCheckbox };
6
- declare const SuperDocESign: React.ForwardRefExoticComponent<Types.SuperDocESignProps & React.RefAttributes<any>>;
4
+ export { SignatureInput, CheckboxInput };
5
+ declare const SuperDocESign: import('react').ForwardRefExoticComponent<Types.SuperDocESignProps & import('react').RefAttributes<any>>;
7
6
  export default SuperDocESign;
8
7
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.tsx"],"names":[],"mappings":"AAAA,OAAO,KAON,MAAM,OAAO,CAAC;AAEf,OAAO,KAAK,KAAK,KAAK,MAAM,SAAS,CAAC;AACtC,OAAO,EACL,cAAc,EACd,eAAe,EAGhB,MAAM,YAAY,CAAC;AAEpB,cAAc,SAAS,CAAC;AACxB,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,CAAC;AAI3C,QAAA,MAAM,aAAa,sFA+alB,CAAC;AAIF,eAAe,aAAa,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.tsx"],"names":[],"mappings":"AASA,OAAO,KAAK,KAAK,KAAK,MAAM,SAAS,CAAC;AACtC,OAAO,EACL,cAAc,EACd,aAAa,EAGd,MAAM,YAAY,CAAC;AAEpB,cAAc,SAAS,CAAC;AACxB,OAAO,EAAE,cAAc,EAAE,aAAa,EAAE,CAAC;AAIzC,QAAA,MAAM,aAAa,0GA0alB,CAAC;AAIF,eAAe,aAAa,CAAC"}