Compare commits
137 Commits
1bf1c37296
...
feat/servi
| Author | SHA1 | Date | |
|---|---|---|---|
| cb7d154f64 | |||
| a3f404b9ae | |||
| 1f7eb5d4ea | |||
| e1ae3ffccb | |||
| 3e3bfaa904 | |||
| f8422ed94c | |||
| c64c63925e | |||
| 2c8ae23203 | |||
| b4be718994 | |||
| 33bf5b1a51 | |||
| 6a9792bb57 | |||
| 6954a86358 | |||
| d6c5b63c49 | |||
| 8ca360d47e | |||
| 2dadefecc5 | |||
| 162f066b08 | |||
| 6592642945 | |||
| 855f22cf87 | |||
| 726673e277 | |||
| 65d615f43e | |||
| f0f92c5400 | |||
| 7e34f01f6a | |||
| f449767bf9 | |||
| f8ec6065c2 | |||
| fabfe80666 | |||
| bdb3863c07 | |||
| e5295b8302 | |||
| 00a9cf06da | |||
| 1b387238e8 | |||
| bda4f3dbef | |||
| 2458d0c07a | |||
| a72cbab195 | |||
| bcf4e63f7c | |||
| 5aabeb4510 | |||
| 32edc6474d | |||
| 71bfe3b323 | |||
| 089d2bd1cb | |||
| 65c72bcf77 | |||
| 2d8375f26a | |||
| a7cfb099fa | |||
| 7a0d4e2fa7 | |||
| 3955ee9f39 | |||
| 9bdf24d7d5 | |||
| 993ca82596 | |||
| ce9958a8b1 | |||
| 6c54473d5a | |||
| 9ad4352e02 | |||
| 90031db6a5 | |||
| bc07af9622 | |||
| 4caf8cd192 | |||
| 5423d5317b | |||
| 6c8c8f22a3 | |||
| 5100504f16 | |||
| 15f8baf54c | |||
| 4e847fa737 | |||
| 53ff48dc3c | |||
| ab67b3fba2 | |||
| 3855a8fa1e | |||
| 052da02bd0 | |||
| 53a9a3e3eb | |||
| c2ab18e184 | |||
| 8f845e8a9a | |||
| 7af934ea19 | |||
| b65367fb6d | |||
| 25c31108cb | |||
| 4083833b19 | |||
| f53bc65acf | |||
| 070d9be00c | |||
| 6f7b04a26e | |||
| 473358e2b9 | |||
| 8f3aa69cbf | |||
| eededeb1f1 | |||
| 737774f077 | |||
| 69499273cc | |||
| ead33ab2e7 | |||
| d1e2840617 | |||
| a39ef6a0e2 | |||
| 7d620988d8 | |||
| cc4a7d8c20 | |||
| 0b8068f926 | |||
| 9fd7a68798 | |||
| d6e2690bb8 | |||
| f04385a03c | |||
| c88e7b2b9e | |||
| 0ee7ec82d6 | |||
| a6d6dc8c4d | |||
| ab038df8b9 | |||
| 302249451b | |||
| a4082fca45 | |||
| f3ae8a9c49 | |||
| d6058c41d0 | |||
| 79d093c492 | |||
| 56c65b6fbb | |||
| 2af18842c4 | |||
| 41e0d4691b | |||
| 64e0a4f749 | |||
| 4cf54d36b9 | |||
| 020cecabf3 | |||
| 40c899e47e | |||
| bae62c7c08 | |||
| 41dde6f016 | |||
| c5b167f0d4 | |||
| a70faebde6 | |||
| 3e01052579 | |||
| f49b907574 | |||
| 86ed96ae76 | |||
| 9dbfdaa15e | |||
| 9d66f1d19e | |||
| 14cef05d02 | |||
| bdfb973d55 | |||
| 15a18b1bfd | |||
| 6fac1cd96a | |||
| 79d2636c14 | |||
| 2172ce8203 | |||
| 51ab9fb094 | |||
| d8a002cd22 | |||
| e36684820e | |||
| 2a5d953623 | |||
| 0aba9e91e2 | |||
| 1a0973075b | |||
| b8fd26611c | |||
| 0ca372bf45 | |||
| 5230076d5d | |||
| b28084bf19 | |||
| cbc983e070 | |||
| 3b06f3fdac | |||
| a12d74871d | |||
| 45b715ed74 | |||
| 9d5e2e6ad2 | |||
| c5d2d79563 | |||
| 15d4573321 | |||
| eb46590c1d | |||
| c93c32307a | |||
| 41cca7fa64 | |||
| b01c17ab18 | |||
| c8834dc3be | |||
| e385fdda95 |
@@ -1,182 +0,0 @@
|
|||||||
# EnvelopeGenerator — Copilot Context Notes (English)
|
|
||||||
|
|
||||||
## Purpose
|
|
||||||
A digital document signing system. Senders upload PDFs and place signature annotation fields via PSPDFKit (EnvelopeGenerator.Web). Receivers open the document in a Blazor WASM viewer, confirm each signature field via a checkbox overlay, draw/type/upload their signature, and export the stamped PDF.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Solution Structure
|
|
||||||
|
|
||||||
| Project | Target | Description |
|
|
||||||
|---|---|---|
|
|
||||||
| `EnvelopeGenerator.API` | net8.0 | ASP.NET Core Web API. Receiver auth (cookie), annotation reading, PDF serving. |
|
|
||||||
| `EnvelopeGenerator.ReceiverUI` | net8.0 WASM | Blazor WebAssembly. Receiver UI. YARP proxies API calls. |
|
|
||||||
| `EnvelopeGenerator.Web` | net7/8/9 | Razor Pages. Sender UI + PSPDFKit annotation placement. |
|
|
||||||
| `EnvelopeGenerator.Application` | multi | MediatR CQRS handlers. |
|
|
||||||
| `EnvelopeGenerator.Domain` | multi | Domain models, constants, interfaces. |
|
|
||||||
| `EnvelopeGenerator.Infrastructure` | multi | EF Core repos, DB context. |
|
|
||||||
| `EnvelopeGenerator.PdfEditor` | multi | iText7 utilities. NOT used in ReceiverUI flow. |
|
|
||||||
| `EnvelopeGenerator.DependencyInjection` | multi | DI registration helpers. |
|
|
||||||
| VB.NET projects (Service/Form/BBTests) | net462 | Legacy. Do NOT touch. |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Key Files
|
|
||||||
|
|
||||||
| File | Purpose |
|
|
||||||
|---|---|
|
|
||||||
| `ReceiverUI/Pages/ReportViewer.razor` | Main receiver page. All signing logic. |
|
|
||||||
| `ReceiverUI/wwwroot/js/receiver-signature.js` | JS: checkbox overlay, signature pad (draw/type/image). |
|
|
||||||
| `ReceiverUI/wwwroot/fake-data/annotations.json` | Dev-mode fake annotations (YARP proxy target). |
|
|
||||||
| `ReceiverUI/Models/AnnotationDto.cs` | Annotation position model. All properties non-nullable. |
|
|
||||||
| `ReceiverUI/Services/AnnotationService.cs` | Fetches `List<AnnotationDto>` from API or fake-data. |
|
|
||||||
| `ReceiverUI/Services/DocumentService.cs` | Fetches PDF bytes from API. |
|
|
||||||
| `ReceiverUI/Services/AuthService.cs` | Manages receiver session cookie. |
|
|
||||||
| `API/Controllers/AnnotationController.cs` | GET `api/Annotation/{key}` ? annotation list. |
|
|
||||||
| `API/Controllers/DocumentController.cs` | GET `api/Document/{key}` ? PDF bytes. |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## AnnotationDto — Coordinate System
|
|
||||||
|
|
||||||
```
|
|
||||||
Unit : 1/100 inch (DX units) — DevExpress XtraReports native
|
|
||||||
Origin : Top-left corner of page
|
|
||||||
X : increases rightward
|
|
||||||
Y : increases downward
|
|
||||||
|
|
||||||
A4 in DX units: Width = 827, Height = 1169
|
|
||||||
|
|
||||||
Conversions:
|
|
||||||
PSPDFKit (pt, top-left): xDX = xPsPdf * (100/72)
|
|
||||||
GDPicture (pt, bottom-left): yDX = (pageHeightPt - yGD - elemHeightPt) * (100/72)
|
|
||||||
DX ? PDF points: pt = dx * (72/100)
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## ReceiverUI Signing Flow (ReportViewer.razor)
|
|
||||||
|
|
||||||
### On Load (`OnInitializedAsync`)
|
|
||||||
1. `AuthService.CheckEnvelopeAccessAsync` ? redirect to login if unauthorized
|
|
||||||
2. `AnnotationService.GetAnnotationsAsync` ? fills `_annotations`
|
|
||||||
3. `DocumentService.GetDocumentAsync` ? fills `_basePdfBytes` (real mode)
|
|
||||||
4. `BuildFreshBaseReport()` ? `XtraReport` for `DxReportViewer`
|
|
||||||
|
|
||||||
### Signature Popup
|
|
||||||
- Tabs: Draw / Text / Image
|
|
||||||
- Fields: full name (required), position (optional), place (required)
|
|
||||||
- Saved to `_capturedSignature` record
|
|
||||||
- If annotations exist ? popup closes ? JS checkbox overlays installed
|
|
||||||
|
|
||||||
### JS Checkbox Overlay (`receiver-signature.js`)
|
|
||||||
- `receiverSignature.installAnnotationCheckboxes(annotations, checkedIds, dotNetRef)`
|
|
||||||
- One `.annot-sig-cb-wrapper` div per annotation, absolutely positioned over viewer scroll container
|
|
||||||
- Position: `left = pageRect.left + ann.x * scaleX`, `top = pageRect.top + ann.y * scaleY`
|
|
||||||
- `scaleX = pagePixelWidth / 827`, `scaleY = pagePixelHeight / 1169`
|
|
||||||
- Click ? `dotNetRef.invokeMethodAsync('OnAnnotationToggled', id, checked)`
|
|
||||||
|
|
||||||
### Apply Signatures ("Unterschriften anwenden" — `SubmitSignaturesAsync`)
|
|
||||||
|
|
||||||
**Real PDF mode (`_basePdfBytes` is set):**
|
|
||||||
- Calls `StampSignaturesOnPdf` using **iText7** directly on PDF bytes
|
|
||||||
- Coordinate conversion: `xPt = ann.X * (72/100)`, `yPt = pageHeight - ann.Y * (72/100) - sigHeight` (Y flip: DX top-down ? PDF bottom-up)
|
|
||||||
- Returns stamped bytes ? loaded into new `XtraReport` with `XRPdfContent`
|
|
||||||
- Viewer refreshed with `ViewerKey++`
|
|
||||||
|
|
||||||
**Dev/fake mode (`_basePdfBytes` is null):**
|
|
||||||
- Falls back to `AddSignatureAtAnnotation` (XtraReports DetailBand + BeforePrint counter)
|
|
||||||
- This is intentionally left as a dev fallback only
|
|
||||||
|
|
||||||
### Export
|
|
||||||
- `reportViewer.ExportToAsync(ExportFormat.Pdf)`
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## StampSignaturesOnPdf — iText7 Implementation
|
|
||||||
|
|
||||||
```csharp
|
|
||||||
// Located in ReportViewer.razor @code section
|
|
||||||
// Called by SubmitSignaturesAsync when _basePdfBytes is available
|
|
||||||
|
|
||||||
static byte[] StampSignaturesOnPdf(
|
|
||||||
byte[] sourcePdfBytes, byte[] signatureImageBytes,
|
|
||||||
string signerFullName, string signerPosition, string signaturePlace,
|
|
||||||
IReadOnlyList<AnnotationDto> annotations)
|
|
||||||
{
|
|
||||||
// Opens PDF with PdfReader/PdfWriter
|
|
||||||
// For each annotation:
|
|
||||||
// pageNum = ann.Page (clamped to totalPages)
|
|
||||||
// xPt = ann.X * (72f/100f)
|
|
||||||
// imgBottomY = pageHeight - ann.Y * (72f/100f) - sigHeightPt ? Y-axis flip
|
|
||||||
// PdfCanvas.AddImageFittedIntoRectangle(imageData, rect, false)
|
|
||||||
// Separator line at bottom of image
|
|
||||||
// Canvas text block below separator (font: Helvetica 7pt, color: RGB(73,80,87))
|
|
||||||
// Returns stamped byte[]
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## BuildFreshBaseReport()
|
|
||||||
|
|
||||||
```csharp
|
|
||||||
// Real PDF mode:
|
|
||||||
var report = new XtraReport();
|
|
||||||
var detail = new DetailBand();
|
|
||||||
report.Bands.Add(detail);
|
|
||||||
detail.Controls.Add(new XRPdfContent { Source = _basePdfBytes, GenerateOwnPages = true });
|
|
||||||
return report;
|
|
||||||
|
|
||||||
// Dev/fake mode: returns pre-built report from ReportStorage
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## NuGet Packages in ReceiverUI
|
|
||||||
|
|
||||||
| Package | Version | Purpose |
|
|
||||||
|---|---|---|
|
|
||||||
| `DevExpress.Blazor.Reporting.Viewer` | 25.2.3 | DxReportViewer component |
|
|
||||||
| `DevExpress.Blazor.PdfViewer` | 25.2.3 | PDF viewer |
|
|
||||||
| `DevExpress.Drawing.Skia` | 25.2.3 | Drawing backend |
|
|
||||||
| `itext` | 8.0.5 | PDF stamping (iText7) |
|
|
||||||
| `SkiaSharp.*` | 3.119.1 | WASM native rendering |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Mistakes History — Do NOT Repeat
|
|
||||||
|
|
||||||
| Mistake | Why Wrong |
|
|
||||||
|---|---|
|
|
||||||
| `BottomMarginBand` for per-page signatures | Repeats on every page; Y offset wrong |
|
|
||||||
| `imageY = (page-1) * 1169 + ann.Y` | Inflates DetailBand; 35 pages ? 140 pages |
|
|
||||||
| `e.Graph?.PrintingSystem` in BeforePrint | `Graph` not on `CancelEventArgs` |
|
|
||||||
| `ctrl.Report?.PrintingSystem` | `PrintingSystem` not on `XtraReportBase` in WASM |
|
|
||||||
| Adding stamp endpoint to `DocumentController` | Not needed; stamping is done client-side in ReceiverUI |
|
|
||||||
| iText7 via API (server-side) | Unnecessary; iText7 runs fine in WASM directly |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## DevExpress Article (2023-08-28) — Why It Does NOT Apply
|
|
||||||
|
|
||||||
The article describes **X.509 cryptographic digital signatures** via `PdfDocumentSigner` + `Pkcs7Signer`.
|
|
||||||
Our use case is **visual/image stamping** at specific page coordinates — different problem, different API.
|
|
||||||
`XRPdfSignature` in the article requires pre-placed fields in the report designer, not runtime coordinates.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Change Log
|
|
||||||
|
|
||||||
| Session | Date | Change |
|
|
||||||
|---|---|---|
|
|
||||||
| 1–3 | — | Core infrastructure: services, YARP proxy, JS overlay, signature pad |
|
|
||||||
| 4 | — | `AddSignatureAtAnnotation` with BottomMarginBand — ? repeated on all pages |
|
|
||||||
| 5 | — | `BeforePrint` + `e.Graph?.PrintingSystem` — ? compile error |
|
|
||||||
| 6 | — | BeforePrint counter — ? correct pattern, wrong band |
|
|
||||||
| 7 | — | Switched to DetailBand — ? correct band |
|
|
||||||
| 8 | — | `(page-1)*1169+Y` offset — ? 35?140 page inflation |
|
|
||||||
| 9 | — | Fixed: `BoundsF.Y = ann.Y` + counter; created COPILOT_CONTEXT.md |
|
|
||||||
| 10 | — | Investigated DevExpress article — not applicable to our case |
|
|
||||||
| 10 | — | Added iText7 to ReceiverUI; implemented `StampSignaturesOnPdf` — ? deterministic coordinates, no page count side effects |
|
|
||||||
| 10 | — | Split COPILOT_CONTEXT.md into COPILOT_CONTEXT_EN.md and COPILOT_CONTEXT_TR.md |
|
|
||||||
@@ -1,199 +0,0 @@
|
|||||||
# EnvelopeGenerator — Copilot Ba?lam Notlar? (Türkçe)
|
|
||||||
|
|
||||||
## Projenin Amac?
|
|
||||||
Dijital belge imzalama sistemi. Göndericiler PDF yükleyip PSPDFKit üzerinden imza alan? (annotation) yerle?tirir. Al?c?lar Blazor WASM viewer'da belgeyi görür, annotation konumlar?nda checkbox overlay ile imza alanlar?n? onaylar, imzalar?n? olu?turur ve imzal? PDF'i export eder.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Çözüm Yap?s?
|
|
||||||
|
|
||||||
| Proje | Hedef | Aç?klama |
|
|
||||||
|---|---|---|
|
|
||||||
| `EnvelopeGenerator.API` | net8.0 | Web API. Receiver auth (cookie), annotation okuma, PDF sunma. |
|
|
||||||
| `EnvelopeGenerator.ReceiverUI` | net8.0 WASM | Blazor WebAssembly. Al?c? arayüzü. YARP proxy ile API'ye ba?lan?r. |
|
|
||||||
| `EnvelopeGenerator.Web` | net7/8/9 | Razor Pages. Gönderen UI + PSPDFKit ile annotation yerle?tirme. |
|
|
||||||
| `EnvelopeGenerator.Application` | multi | MediatR CQRS handler'lar?. |
|
|
||||||
| `EnvelopeGenerator.Domain` | multi | Domain modelleri, sabitler, arayüzler. |
|
|
||||||
| `EnvelopeGenerator.Infrastructure` | multi | EF Core repo'lar?, DB context. |
|
|
||||||
| `EnvelopeGenerator.PdfEditor` | multi | iText7 PDF yard?mc?lar?. ReceiverUI ak???nda KULLANILMIYOR. |
|
|
||||||
| `EnvelopeGenerator.DependencyInjection` | multi | DI kay?t yard?mc?lar?. |
|
|
||||||
| VB.NET projeleri (Service/Form/BBTests) | net462 | Eski legacy. DOKUNMA. |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Önemli Dosyalar
|
|
||||||
|
|
||||||
| Dosya | Amaç |
|
|
||||||
|---|---|
|
|
||||||
| `ReceiverUI/Pages/ReportViewer.razor` | Ana al?c? sayfas?. Tüm imzalama mant??? burada. |
|
|
||||||
| `ReceiverUI/wwwroot/js/receiver-signature.js` | JS: checkbox overlay, imza pad (çizim/yaz?/resim). |
|
|
||||||
| `ReceiverUI/wwwroot/fake-data/annotations.json` | Dev modda sahte annotation konumlar?. YARP proxy bu dosyaya yönlendirir. |
|
|
||||||
| `ReceiverUI/Models/AnnotationDto.cs` | Annotation pozisyon modeli. Tüm property'ler non-nullable. |
|
|
||||||
| `ReceiverUI/Services/AnnotationService.cs` | `List<AnnotationDto>` döner; gerçek modda API'den, dev modda fake-data'dan. |
|
|
||||||
| `ReceiverUI/Services/DocumentService.cs` | PDF byte'lar?n? API'den al?r. |
|
|
||||||
| `ReceiverUI/Services/AuthService.cs` | Al?c? session cookie'sini yönetir. |
|
|
||||||
| `ReceiverUI/wwwroot/appsettings.json` | `ForceToUseFakeDocument: true` ? gerçek PDF yüklenmez, ?ablon rapor kullan?l?r. |
|
|
||||||
| `API/Controllers/AnnotationController.cs` | GET `api/Annotation/{key}` ? annotation listesi. |
|
|
||||||
| `API/Controllers/DocumentController.cs` | GET `api/Document/{key}` ? PDF byte'lar?. |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## AnnotationDto Koordinat Sistemi
|
|
||||||
|
|
||||||
```
|
|
||||||
Birim : 1/100 inch (DX units) — DevExpress XtraReports'un yerel koordinat sistemi
|
|
||||||
Köken : Sol-üst kö?e
|
|
||||||
X artar : sa?a do?ru
|
|
||||||
Y artar : a?a??ya do?ru
|
|
||||||
|
|
||||||
A4 boyutlar? DX units cinsinden: Geni?lik = 827, Yükseklik = 1169
|
|
||||||
|
|
||||||
Dönü?ümler:
|
|
||||||
PSPDFKit (pt, sol-üst): xDX = xPsPdf * (100/72)
|
|
||||||
GDPicture (pt, sol-alt): yDX = (pageHeightPt - yGD - elemHeightPt) * (100/72)
|
|
||||||
DX ? PDF points: pt = dx * (72/100)
|
|
||||||
PDF Y ekseni çevirme: imgBottomY = sayfaYüksekli?iPt - ann.Y*(72/100) - elemanYüksekli?iPt
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## ReceiverUI ?mzalama Ak??? (ReportViewer.razor)
|
|
||||||
|
|
||||||
### Sayfa Yüklenince (`OnInitializedAsync`)
|
|
||||||
1. `AuthService.CheckEnvelopeAccessAsync` ? yetkisizse login sayfas?na yönlendir
|
|
||||||
2. `AnnotationService.GetAnnotationsAsync` ? `_annotations` listesi dolar
|
|
||||||
3. `DocumentService.GetDocumentAsync` ? `_basePdfBytes` dolar (gerçek mod)
|
|
||||||
4. `BuildFreshBaseReport()` ? `XtraReport` olu?turulur, `DxReportViewer`'a verilir
|
|
||||||
|
|
||||||
### `BuildFreshBaseReport()` Mant???
|
|
||||||
```
|
|
||||||
_basePdfBytes dolu ? XtraReport + DetailBand + XRPdfContent { GenerateOwnPages = true }
|
|
||||||
_basePdfBytes bo? (ForceToUseFakeDocument=true) ? ReportStorage'dan LargeDatasetReport ?ablonu (XtraReport, designer'dan geldi?i gibi)
|
|
||||||
```
|
|
||||||
|
|
||||||
> NOT: `_basePdfBytes` dal? korunur (gerçek PDF modu). Dev ve test sunucusunda `PredefinedReport` (LargeDatasetReport) `XtraReport` olarak do?rudan kullan?l?r — PDF'e export ED?LMEZ.
|
|
||||||
|
|
||||||
### ?mza Popup'? ("Unterschrift erstellen")
|
|
||||||
- Sekmeler: Çizim / Yaz? / Resim
|
|
||||||
- Alanlar: Ad soyad (zorunlu), pozisyon (opsiyonel), yer (zorunlu)
|
|
||||||
- `_capturedSignature` record'una kaydedilir
|
|
||||||
- Annotation varsa popup kapan?r ? JS checkbox overlay kurulur
|
|
||||||
|
|
||||||
### JS Checkbox Overlay (`receiver-signature.js`)
|
|
||||||
- `receiverSignature.installAnnotationCheckboxes(annotations, checkedIds, dotNetRef)` C#'tan ça?r?l?r
|
|
||||||
- Her annotation için `.annot-sig-cb-wrapper` div'i, viewer scroll container'?na absolute olarak yerle?tirilir
|
|
||||||
- **Koordinat hesab?:** `left = pageRect.left + ann.x * scaleX`, `top = pageRect.top + ann.y * scaleY`
|
|
||||||
- `scaleX = sayfaPixelGeni?li?i / 827`, `scaleY = sayfaPixelYüksekli?i / 1169`
|
|
||||||
- Bu koordinatlar sayfa-relatif ve do?ru çal???yor
|
|
||||||
- T?klan?nca `dotNetRef.invokeMethodAsync('OnAnnotationToggled', id, checked)` ça?r?l?r
|
|
||||||
|
|
||||||
### ?mza Uygulama ("Unterschriften anwenden" — `SubmitSignaturesAsync`)
|
|
||||||
|
|
||||||
**Tek ortak yol (her iki mod):**
|
|
||||||
- `SubmitSignaturesAsync` ? `BuildFreshBaseReport()` + `WireAnnotationSignatures(report, _capturedSignature)`
|
|
||||||
- `WireAnnotationSignatures` ? `report.AfterPrint` olay?na abone olur
|
|
||||||
- Belge olu?unca `report.PrintingSystem.Pages` dolu olur; her annotation için `Pages[ann.Page-1]` sayfas?na:
|
|
||||||
- `ImageBrick { Image = imza görseli, Rect = (ann.X, ann.Y, 230, 70), SizeMode = ZoomImage }`
|
|
||||||
- `TextBrick { Text = bilgi metni, Rect = (ann.X, ann.Y+75, 230, 65) }`
|
|
||||||
- `Page.AddBrick(brick)` ile bas?l?r
|
|
||||||
- Brick `Rect`'leri annotation `X`/`Y` (1/100 inch) ? checkbox overlay ile birebir ayn? koordinat, do?ru sayfa+konum
|
|
||||||
- `ViewerKey++` ile viewer yenilenir
|
|
||||||
|
|
||||||
**Gerçek PDF modu (`_basePdfBytes` dolu):** Yukar?daki ile ayn?; rapor `XRPdfContent`'ten olu?ur, brick'ler yine `AfterPrint` ile `PrintingSystem.Pages` üzerine bas?l?r.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## ReceiverUI'daki NuGet Paketleri
|
|
||||||
|
|
||||||
| Paket | Versiyon | Amaç |
|
|
||||||
|---|---|---|
|
|
||||||
| `DevExpress.Blazor.Reporting.Viewer` | 25.2.3 | DxReportViewer bile?eni |
|
|
||||||
| `DevExpress.Blazor.PdfViewer` | 25.2.3 | PDF görüntüleyici |
|
|
||||||
| `DevExpress.Drawing.Skia` | 25.2.3 | Çizim backend'i |
|
|
||||||
| `SkiaSharp.*` | 3.119.1 | WASM native render |
|
|
||||||
|
|
||||||
> Not: iText7, ReceiverUI imzalama ak???nda KULLANILMIYOR. ?mza yerle?tirme tamamen DevExpress XtraReports brick mekanizmas?yla (`PrintingSystem.Pages[i].AddBrick`) yap?l?r.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## GÖREV 1: ?mza Konum Hatas? (BUG) — ÇÖZÜLDÜ
|
|
||||||
|
|
||||||
### Kullan?c?n?n ?ste?i
|
|
||||||
Annotation'lardan okunan sayfa ve X/Y koordinatlar?na göre, t?pk? checkbox overlay'ler gibi, imzalar do?ru sayfa ve konumda görünsün. `_basePdfBytes` dal? korunsun; dev/test'te designer ile olu?turulan `PredefinedReport` `XtraReport` olarak do?rudan kullan?lmaya devam etsin (PDF'e export yok).
|
|
||||||
|
|
||||||
### ÇÖZÜM (Oturum 12) ?
|
|
||||||
`WireAnnotationSignatures` metodu, `report.AfterPrint` olay?nda `report.PrintingSystem.Pages[ann.Page-1].AddBrick(...)` ça??rarak imza görselini (`ImageBrick`) ve bilgi metnini (`TextBrick`) do?rudan hedef sayfaya, annotation `X`/`Y` (1/100 inch) konumunda basar.
|
|
||||||
|
|
||||||
**Neden çal???r:**
|
|
||||||
- `AfterPrint`, belge tamamen olu?tuktan sonra tetiklenir; `PrintingSystem.Pages` art?k gerçek/nihai sayfalar? içerir.
|
|
||||||
- Sayfa indeksleme (`Pages[ann.Page-1]`) band veya veri-sat?r? tekrar?ndan **ba??ms?zd?r** ? `LargeDatasetReport`'un veri-ba?l? `detailBand1` sorununu tamamen atlar.
|
|
||||||
- Brick `Rect` koordinatlar? raporun yerel 1/100 inch sistemindedir ? checkbox overlay ile birebir ayn?, do?ru konum.
|
|
||||||
- Yeni sayfa eklenmedi?i için sayfa say?s? katlanmaz (35 sayfa ? 35 sayfa).
|
|
||||||
|
|
||||||
**Derleme s?ras?nda ö?renilen API gerçekleri:**
|
|
||||||
- `PrintOnPage` olay? `e.Page` VERMEZ (yaln?zca `PageIndex`/`PageCount`) ? brick eklenemez. Do?ru olay `AfterPrint` + `PrintingSystem.Pages`.
|
|
||||||
- `Page` tipinde `InsertBrick` YOK; do?ru metot `Page.AddBrick(brick)` (brick'in `Rect`'i konumu belirler).
|
|
||||||
- `ImageBrick.BorderStyle` tipi `BrickBorderStyle`'dir (`BorderDashStyle` de?il). Border için `Sides` + `BorderColor` kullan?ld?.
|
|
||||||
|
|
||||||
### Denenen Eski Çözümler (ba?ar?s?z — referans)
|
|
||||||
|
|
||||||
| Deneme | Yakla??m | Sonuç | Ba?ar?s?zl?k Sebebi |
|
|
||||||
|---|---|---|---|
|
|
||||||
| 1 | `BottomMarginBand` + `XRPictureBox`/`XRLabel` | Her sayfan?n alt?na ç?kt? | Band her sayfada tekrarlan?r, sayfa filtresi yok |
|
|
||||||
| 2 | `BeforePrint` + `e.Graph?.PrintingSystem` | Derleme hatas? | `CancelEventArgs`'ta `Graph` yok |
|
|
||||||
| 3–6 | `DetailBand` + `BeforePrint` counter | Yanl?? sayfa/konum | ?ablonun `detailBand1`'i veri sat?r? ba??na tetiklenir, sayfa ba??na de?il |
|
|
||||||
| 7 | iText7 export/reload döngüsü | 35 ? 70 sayfa | Margin uyu?mazl???, `GenerateOwnPages` sayfalar? böldü |
|
|
||||||
| 8 | Fake modda `BottomMarginBand` fallback | Her sayfan?n alt?nda | Koordinat yanl?? |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## YAPILMAMASI GEREKENLER
|
|
||||||
|
|
||||||
| Hata | Neden Yanl?? |
|
|
||||||
|---|---|
|
|
||||||
| `BottomMarginBand`/`DetailBand` ile sayfa-spesifik imza | Band veri-sat?r?/sayfa ba??na tekrarlan?r, koordinat kayar |
|
|
||||||
| `BeforePrint` counter ile sayfa filtresi | Veri-ba?l? raporda sat?r ba??na tetiklenir, güvenilmez |
|
|
||||||
| `PrintOnPage` ile brick ekleme | `e.Page` yok; brick eklenemez |
|
|
||||||
| `Page.InsertBrick(...)` | Yok; do?ru metot `Page.AddBrick(...)` |
|
|
||||||
| iText7 export+reload döngüsü | Margin uyu?mazl???ndan sayfa say?s? katlan?r |
|
|
||||||
| ?ablonu PDF'e export edip `XRPdfContent`'e yükleme | ?stenmiyor; designer raporu do?rudan kullan?lmal? |
|
|
||||||
| Stamplama için API endpoint ekleme | Gereksiz; brick'ler client'ta bas?l?r |
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## BEKLEYEN D??ER GÖREVLER (Sonraki Chat'te Yap?lacak)
|
|
||||||
|
|
||||||
### 2. ?mza Arka Plan? Özelli?i
|
|
||||||
?mza görselinin ve bilgilerinin kaplad??? alan kadar, yar? saydam hafif gri opak dikdörtgen arka plan ekle. Böylece imza ve bilgiler arka plandaki metinlerden etkilenmez ve okunur kal?r. (Art?k brick tabanl?: `ImageBrick`/`TextBrick`'in arkas?na bir arka plan `Brick` dikdörtgeni eklenebilir.)
|
|
||||||
|
|
||||||
### 3. Checkbox Renk ve Stil ?yile?tirmesi
|
|
||||||
Mevcut checkbox'lar?n rengi ve kenarl?klar? çok dikkat çekici. Koyu füme tonlar?nda, desenli, sade ve profesyonel görünümlü bir stil olsun. (`receiver-signature.js` ve ilgili CSS.)
|
|
||||||
|
|
||||||
### 4. Sayfa Aç?l???nda Otomatik ?mza Popup'?
|
|
||||||
Sayfa aç?l?r aç?lmaz imza popup'? ç?ks?n. "Kay?tl? hiç bir imzan?z yok, tan?mlay?n?z" mesaj? gösterilsin. Kullan?c? imzas?n? tan?mlamadan ilerleyemesin. Mevcut "Unterschrift erstellen" butonu "?mzay? de?i?tir" olarak güncellensin.
|
|
||||||
|
|
||||||
### 5. Otomatik ?mza Uygulama
|
|
||||||
Kullan?c? tüm checkbox'lar? onaylad??? anda imzalar otomatik olarak uygulanmaya ba?las?n (butona t?klamaya gerek kalmas?n). Sayfan?n üstünde imza say?s? ve imzalanmas? gereken sayfalar hakk?nda bilgi gösterilsin.
|
|
||||||
|
|
||||||
### 6. Checkbox - DevExpress Toolbar Pozisyon Uyumsuzlu?u (BUG)
|
|
||||||
- Checkbox'lar browser'?n boyut/konumuna neredeyse anl?k tepki veriyor
|
|
||||||
- DevExpress toolbar de?i?ikliklerine geç tepki gösteriyor
|
|
||||||
- Zoom de?i?ince bazen 2-3 PDF yan yana gelebiliyor; checkbox o konumda do?ru görünüyor ama iki PDF'in yan yana gelmesi kald?r?lmal?
|
|
||||||
- `DocumentViewer.razor`'daki `DxDocumentViewer` + `DxDocumentViewerTabPanelSettings` bile?enleri daha uygun olabilir mi? De?erlendirmesi yap?lacak.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## De?i?iklik Günlü?ü
|
|
||||||
|
|
||||||
| Oturum | De?i?iklik |
|
|
||||||
|---|---|
|
|
||||||
| 1–3 | Temel altyap?: servisler, YARP proxy, JS overlay, imza pad |
|
|
||||||
| 4 | `AddSignatureAtAnnotation` + BottomMarginBand ? ? her sayfada tekrar |
|
|
||||||
| 5 | `BeforePrint` + `e.Graph?.PrintingSystem` ? ? derleme hatas? |
|
|
||||||
| 6 | BeforePrint counter ? ? do?ru yakla??m, yanl?? band |
|
|
||||||
| 7 | DetailBand'e geçi? ? ? do?ru band, koordinat hâlâ yanl?? |
|
|
||||||
| 8 | `(page-1)*1169+Y` ? ? sayfa ?i?mesi (35?140) |
|
|
||||||
| 9 | `BoundsF.Y = ann.Y` + counter ? (?ablon raporda çal??m?yor) |
|
|
||||||
| 10 | iText7 `StampSignaturesOnPdf` gerçek modda ?, fake modda export+reload ? (35?70), BottomMarginBand fallback ? |
|
|
||||||
| 11 | COPILOT_CONTEXT_TR.md ve COPILOT_CONTEXT_EN.md ayr? dosyalar olarak yeniden olu?turuldu |
|
|
||||||
| **12** | **GÖREV 1 ÇÖZÜLDÜ ?** — `WireAnnotationSignatures`: `report.AfterPrint` + `PrintingSystem.Pages[ann.Page-1].AddBrick(ImageBrick/TextBrick)`. Sayfa hedefleme band'dan ba??ms?z, sayfa say?s? katlanm?yor. iText7 ReceiverUI ak???ndan ç?kar?ld?. `AddSignatureAtAnnotation`/`RemoveExistingSignatureById` kald?r?ld?. Derleme ba?ar?l?. |
|
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
using DigitalData.Auth.Claims;
|
|
||||||
using EnvelopeGenerator.API.Controllers.Interfaces;
|
using EnvelopeGenerator.API.Controllers.Interfaces;
|
||||||
using EnvelopeGenerator.API.Models;
|
using EnvelopeGenerator.API.Models;
|
||||||
using EnvelopeGenerator.Domain.Constants;
|
using EnvelopeGenerator.Domain.Constants;
|
||||||
@@ -74,44 +73,4 @@ public partial class AuthController(IOptions<AuthTokenKeys> authTokenKeyOptions,
|
|||||||
=> role is not null && !User.IsInRole(role)
|
=> role is not null && !User.IsInRole(role)
|
||||||
? Unauthorized()
|
? Unauthorized()
|
||||||
: Ok();
|
: Ok();
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Checks whether the caller holds a valid per-envelope receiver token for the given envelope key.
|
|
||||||
/// The request must carry a cookie named <c>AuthTokenSignFLOWReceiver.{envelopeKey}</c>.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="envelopeKey">The unique envelope key extracted from the route.</param>
|
|
||||||
/// <response code="200">Valid per-envelope token found.</response>
|
|
||||||
/// <response code="401">Token is missing, expired or invalid.</response>
|
|
||||||
[ProducesResponseType(typeof(void), StatusCodes.Status200OK)]
|
|
||||||
[ProducesResponseType(typeof(void), StatusCodes.Status401Unauthorized)]
|
|
||||||
[Authorize(Policy = AuthPolicy.Receiver)]
|
|
||||||
[HttpGet("check/envelope/{envelopeKey}")]
|
|
||||||
public IActionResult CheckEnvelopeReceiver([FromRoute] string envelopeKey) => Ok();
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Removes the per-envelope receiver cookie for the given envelope key.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="envelopeKey">The unique envelope key whose cookie should be deleted.</param>
|
|
||||||
/// <response code="200">Cookie successfully deleted.</response>
|
|
||||||
[ProducesResponseType(typeof(void), StatusCodes.Status200OK)]
|
|
||||||
[HttpPost("logout/envelope/{envelopeKey}")]
|
|
||||||
public IActionResult LogoutEnvelopeReceiver([FromRoute] string envelopeKey)
|
|
||||||
{
|
|
||||||
var cookieName = CookieNames.GetEnvelopeReceiverCookieName(authTokenKeys.Cookie, envelopeKey);
|
|
||||||
Response.Cookies.Delete(cookieName);
|
|
||||||
return Ok();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Removes all per-envelope receiver cookies from the current request.
|
|
||||||
/// </summary>
|
|
||||||
/// <response code="200">All envelope receiver cookies successfully deleted.</response>
|
|
||||||
[ProducesResponseType(typeof(void), StatusCodes.Status200OK)]
|
|
||||||
[HttpPost("logout/envelope")]
|
|
||||||
public IActionResult LogoutAllEnvelopeReceivers()
|
|
||||||
{
|
|
||||||
foreach (var cookieName in Request.Cookies.Keys.Where(k => CookieNames.IsEnvelopeReceiverCookie(k, authTokenKeys.Cookie)))
|
|
||||||
Response.Cookies.Delete(cookieName);
|
|
||||||
return Ok();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
using DigitalData.Auth.Claims;
|
|
||||||
using EnvelopeGenerator.API.Controllers.Interfaces;
|
using EnvelopeGenerator.API.Controllers.Interfaces;
|
||||||
using EnvelopeGenerator.API.Extensions;
|
using EnvelopeGenerator.API.Extensions;
|
||||||
using EnvelopeGenerator.Application.Documents.Queries;
|
using EnvelopeGenerator.Application.Documents.Queries;
|
||||||
@@ -15,9 +14,10 @@ namespace EnvelopeGenerator.API.Controllers;
|
|||||||
/// <remarks>
|
/// <remarks>
|
||||||
/// Initializes a new instance of the <see cref="DocumentController"/> class.
|
/// Initializes a new instance of the <see cref="DocumentController"/> class.
|
||||||
/// </remarks>
|
/// </remarks>
|
||||||
|
[Authorize]
|
||||||
[ApiController]
|
[ApiController]
|
||||||
[Route("api/[controller]")]
|
[Route("api/[controller]")]
|
||||||
public class DocumentController(IMediator mediator, IAuthorizationService authService, ILogger<DocumentController> logger) : ControllerBase, IAuthController
|
public class DocumentController(IMediator mediator, IAuthorizationService authService) : ControllerBase, IAuthController
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
@@ -60,25 +60,4 @@ public class DocumentController(IMediator mediator, IAuthorizationService authSe
|
|||||||
|
|
||||||
return Unauthorized();
|
return Unauthorized();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Gets the document for the specified envelope key.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="envelopeKey"></param>
|
|
||||||
/// <param name="cancel"></param>
|
|
||||||
/// <returns></returns>
|
|
||||||
[Authorize(Policy = AuthPolicy.Receiver)]
|
|
||||||
[HttpGet("{envelopeKey}")]
|
|
||||||
public async Task<IActionResult> GetDocumentOfReceiver(string envelopeKey, CancellationToken cancel)
|
|
||||||
{
|
|
||||||
int envelopeId = User.GetEnvelopeIdOfReceiver();
|
|
||||||
|
|
||||||
var senderDoc = await mediator.Send(new ReadDocumentQuery() { EnvelopeId = envelopeId }, cancel);
|
|
||||||
|
|
||||||
if (senderDoc.ByteData is not byte[] senderDocByte)
|
|
||||||
return NotFound("Document is empty.");
|
|
||||||
|
|
||||||
Response.Headers.ContentDisposition = $"inline; filename=\"{envelopeKey}.pdf\"";
|
|
||||||
return File(senderDocByte, "application/pdf");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -71,6 +71,7 @@ public class EnvelopeController : ControllerBase
|
|||||||
[HttpGet("doc-result")]
|
[HttpGet("doc-result")]
|
||||||
public async Task<IActionResult> GetDocResultAsync([FromQuery] ReadEnvelopeQuery query, [FromQuery] bool view = false)
|
public async Task<IActionResult> GetDocResultAsync([FromQuery] ReadEnvelopeQuery query, [FromQuery] bool view = false)
|
||||||
{
|
{
|
||||||
|
query.IncludeDocResult = true;
|
||||||
var envelopes = await _mediator.Send(query.Authorize(User.GetId()));
|
var envelopes = await _mediator.Send(query.Authorize(User.GetId()));
|
||||||
var envelope = envelopes.FirstOrDefault();
|
var envelope = envelopes.FirstOrDefault();
|
||||||
|
|
||||||
@@ -98,7 +99,7 @@ public class EnvelopeController : ControllerBase
|
|||||||
[HttpPost]
|
[HttpPost]
|
||||||
public async Task<IActionResult> CreateAsync([FromBody] CreateEnvelopeCommand command)
|
public async Task<IActionResult> CreateAsync([FromBody] CreateEnvelopeCommand command)
|
||||||
{
|
{
|
||||||
var res = await _mediator.Send(command.WithAuth(User.GetId()));
|
var res = await _mediator.Send(command.Authorize(User.GetId()));
|
||||||
|
|
||||||
if (res is null)
|
if (res is null)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -14,7 +14,6 @@ using EnvelopeGenerator.Application.Common.SQL;
|
|||||||
using EnvelopeGenerator.Application.Common.Dto.Receiver;
|
using EnvelopeGenerator.Application.Common.Dto.Receiver;
|
||||||
using EnvelopeGenerator.Application.Common.Interfaces.SQLExecutor;
|
using EnvelopeGenerator.Application.Common.Interfaces.SQLExecutor;
|
||||||
using EnvelopeGenerator.API.Extensions;
|
using EnvelopeGenerator.API.Extensions;
|
||||||
using EnvelopeGenerator.Domain.Constants;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.API.Controllers;
|
namespace EnvelopeGenerator.API.Controllers;
|
||||||
|
|
||||||
@@ -74,24 +73,6 @@ public class EnvelopeReceiverController : ControllerBase
|
|||||||
return Ok(result);
|
return Ok(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
///
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="envelopeKey"></param>
|
|
||||||
/// <param name="cancel"></param>
|
|
||||||
/// <returns></returns>
|
|
||||||
[Authorize(Policy = AuthPolicy.Receiver)]
|
|
||||||
[HttpGet("{envelopeKey}")]
|
|
||||||
public async Task<IActionResult> GetEnvelopeReceiverOfReceiver([FromRoute] string envelopeKey, CancellationToken cancel)
|
|
||||||
{
|
|
||||||
var er = await _mediator.Send(new ReadEnvelopeReceiverQuery()
|
|
||||||
{
|
|
||||||
Key = envelopeKey
|
|
||||||
}, cancel);
|
|
||||||
|
|
||||||
return Ok(er.SingleOrDefault());
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Ruft den Namen des zuletzt verwendeten Empfängers basierend auf der angegebenen E-Mail-Adresse ab.
|
/// Ruft den Namen des zuletzt verwendeten Empfängers basierend auf der angegebenen E-Mail-Adresse ab.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -16,12 +16,6 @@ public sealed class AuthProxyDocumentFilter : IDocumentFilter
|
|||||||
/// <param name="swaggerDoc"></param>
|
/// <param name="swaggerDoc"></param>
|
||||||
/// <param name="context"></param>
|
/// <param name="context"></param>
|
||||||
public void Apply(OpenApiDocument swaggerDoc, DocumentFilterContext context)
|
public void Apply(OpenApiDocument swaggerDoc, DocumentFilterContext context)
|
||||||
{
|
|
||||||
AddLoginOperation(swaggerDoc, context);
|
|
||||||
AddEnvelopeReceiverLoginOperation(swaggerDoc, context);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void AddLoginOperation(OpenApiDocument swaggerDoc, DocumentFilterContext context)
|
|
||||||
{
|
{
|
||||||
const string path = "/api/auth";
|
const string path = "/api/auth";
|
||||||
|
|
||||||
@@ -73,51 +67,4 @@ public sealed class AuthProxyDocumentFilter : IDocumentFilter
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void AddEnvelopeReceiverLoginOperation(OpenApiDocument swaggerDoc, DocumentFilterContext context)
|
|
||||||
{
|
|
||||||
const string path = "/api/Auth/envelope-receiver/{key}";
|
|
||||||
|
|
||||||
var bodySchema = context.SchemaGenerator.GenerateSchema(typeof(EnvelopeReceiverLogin), context.SchemaRepository);
|
|
||||||
|
|
||||||
var operation = new OpenApiOperation
|
|
||||||
{
|
|
||||||
Summary = "Envelope receiver login (auth-hub proxy)",
|
|
||||||
Description = "Proxies the envelope receiver login to the auth service. " +
|
|
||||||
"The `cookie` query parameter is always forwarded as `true` so the auth service sets the per-envelope cookie automatically.",
|
|
||||||
Tags = [new() { Name = "Auth" }],
|
|
||||||
Parameters =
|
|
||||||
{
|
|
||||||
new OpenApiParameter
|
|
||||||
{
|
|
||||||
Name = "key",
|
|
||||||
In = ParameterLocation.Path,
|
|
||||||
Required = true,
|
|
||||||
Schema = new OpenApiSchema { Type = "string" },
|
|
||||||
Description = "The unique envelope receiver key."
|
|
||||||
}
|
|
||||||
},
|
|
||||||
RequestBody = new OpenApiRequestBody
|
|
||||||
{
|
|
||||||
Required = false,
|
|
||||||
Content =
|
|
||||||
{
|
|
||||||
["multipart/form-data"] = new OpenApiMediaType { Schema = bodySchema }
|
|
||||||
}
|
|
||||||
},
|
|
||||||
Responses =
|
|
||||||
{
|
|
||||||
["200"] = new OpenApiResponse { Description = "OK – per-envelope cookie set by auth service." },
|
|
||||||
["401"] = new OpenApiResponse { Description = "Unauthorized – invalid or missing access code." }
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
swaggerDoc.Paths[path] = new OpenApiPathItem
|
|
||||||
{
|
|
||||||
Operations =
|
|
||||||
{
|
|
||||||
[OperationType.Post] = operation
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk.Web">
|
<Project Sdk="Microsoft.NET.Sdk.Web">
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFrameworks>net8.0</TargetFrameworks>
|
<TargetFrameworks>net9.0</TargetFrameworks>
|
||||||
<Nullable>enable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
<GenerateDocumentationFile>true</GenerateDocumentationFile>
|
||||||
@@ -10,9 +10,9 @@
|
|||||||
<Authors>Digital Data GmbH</Authors>
|
<Authors>Digital Data GmbH</Authors>
|
||||||
<Company>Digital Data GmbH</Company>
|
<Company>Digital Data GmbH</Company>
|
||||||
<Product>EnvelopeGenerator.GeneratorAPI</Product>
|
<Product>EnvelopeGenerator.GeneratorAPI</Product>
|
||||||
<Version>1.3.1</Version>
|
<Version>1.2.3</Version>
|
||||||
<FileVersion>1.3.1</FileVersion>
|
<FileVersion>1.2.3</FileVersion>
|
||||||
<AssemblyVersion>1.3.1</AssemblyVersion>
|
<AssemblyVersion>1.2.3</AssemblyVersion>
|
||||||
<PackageOutputPath>Copyright © 2025 Digital Data GmbH. All rights reserved.</PackageOutputPath>
|
<PackageOutputPath>Copyright © 2025 Digital Data GmbH. All rights reserved.</PackageOutputPath>
|
||||||
<DocumentationFile>bin\$(Configuration)\$(TargetFramework)\$(AssemblyName).xml</DocumentationFile>
|
<DocumentationFile>bin\$(Configuration)\$(TargetFramework)\$(AssemblyName).xml</DocumentationFile>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
@@ -30,16 +30,11 @@
|
|||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="AspNetCore.Scalar" Version="1.1.8" />
|
<PackageReference Include="AspNetCore.Scalar" Version="1.1.8" />
|
||||||
<PackageReference Include="DigitalData.Auth.Claims" Version="1.0.3" />
|
|
||||||
<PackageReference Include="DigitalData.Auth.Client" Version="1.3.7" />
|
<PackageReference Include="DigitalData.Auth.Client" Version="1.3.7" />
|
||||||
<PackageReference Include="DigitalData.Core.API" Version="2.2.1" />
|
<PackageReference Include="DigitalData.Core.API" Version="2.2.1" />
|
||||||
<PackageReference Include="HtmlSanitizer" Version="9.0.892" />
|
<PackageReference Include="HtmlSanitizer" Version="9.0.892" />
|
||||||
<PackageReference Include="itext" Version="8.0.5" />
|
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="9.0.4" />
|
||||||
<PackageReference Include="itext.bouncy-castle-adapter" Version="8.0.5" />
|
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="9.0.6" />
|
||||||
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="9.0.4" Condition="'$(TargetFramework)' == 'net9.0'" />
|
|
||||||
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="8.0.11" Condition="'$(TargetFramework)' == 'net8.0'" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="9.0.6" Condition="'$(TargetFramework)' == 'net9.0'" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="8.0.17" Condition="'$(TargetFramework)' == 'net8.0'" />
|
|
||||||
<PackageReference Include="Microsoft.Identity.Client" Version="4.82.1" />
|
<PackageReference Include="Microsoft.Identity.Client" Version="4.82.1" />
|
||||||
<PackageReference Include="NLog" Version="5.2.5" />
|
<PackageReference Include="NLog" Version="5.2.5" />
|
||||||
<PackageReference Include="NLog.Web.AspNetCore" Version="5.3.0" />
|
<PackageReference Include="NLog.Web.AspNetCore" Version="5.3.0" />
|
||||||
@@ -76,7 +71,6 @@
|
|||||||
<ProjectReference Include="..\EnvelopeGenerator.Application\EnvelopeGenerator.Application.csproj" />
|
<ProjectReference Include="..\EnvelopeGenerator.Application\EnvelopeGenerator.Application.csproj" />
|
||||||
<ProjectReference Include="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj" />
|
<ProjectReference Include="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj" />
|
||||||
<ProjectReference Include="..\EnvelopeGenerator.Infrastructure\EnvelopeGenerator.Infrastructure.csproj" />
|
<ProjectReference Include="..\EnvelopeGenerator.Infrastructure\EnvelopeGenerator.Infrastructure.csproj" />
|
||||||
<ProjectReference Include="..\EnvelopeGenerator.PdfEditor\EnvelopeGenerator.PdfEditor.csproj" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -11,10 +11,6 @@ namespace EnvelopeGenerator.API.Extensions;
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public static class ReceiverClaimExtensions
|
public static class ReceiverClaimExtensions
|
||||||
{
|
{
|
||||||
private static readonly string[] EnvelopeIdClaimTypes = [EnvelopeClaimTypes.Id, "envelope_id", "EnvelopeId"];
|
|
||||||
private static readonly string[] EnvelopeUuidClaimTypes = [ClaimTypes.NameIdentifier, "envelope_uuid", "EnvelopeUuid"];
|
|
||||||
private static readonly string[] ReceiverSignatureClaimTypes = [ClaimTypes.Hash, "receiver_sig", "ReceiverSignature"];
|
|
||||||
|
|
||||||
private static string GetRequiredClaimOfReceiver(this ClaimsPrincipal user, string claimType)
|
private static string GetRequiredClaimOfReceiver(this ClaimsPrincipal user, string claimType)
|
||||||
{
|
{
|
||||||
var value = user.FindFirstValue(claimType);
|
var value = user.FindFirstValue(claimType);
|
||||||
@@ -31,32 +27,15 @@ public static class ReceiverClaimExtensions
|
|||||||
throw new InvalidOperationException(message);
|
throw new InvalidOperationException(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static string GetRequiredClaimOfReceiver(this ClaimsPrincipal user, params string[] claimTypes)
|
|
||||||
{
|
|
||||||
foreach (var claimType in claimTypes.Where(t => !string.IsNullOrWhiteSpace(t)).Distinct())
|
|
||||||
{
|
|
||||||
var value = user.FindFirstValue(claimType);
|
|
||||||
if (!string.IsNullOrWhiteSpace(value))
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
var identity = user.Identity;
|
|
||||||
var principalName = identity?.Name ?? "(anonymous)";
|
|
||||||
var authType = identity?.AuthenticationType ?? "(none)";
|
|
||||||
var availableClaims = string.Join(", ", user.Claims.Select(c => $"{c.Type}={c.Value}"));
|
|
||||||
var message = $"Required claim(s) '{string.Join("', '", claimTypes)}' are missing for user '{principalName}' (auth: {authType}). Available claims: [{availableClaims}].";
|
|
||||||
throw new InvalidOperationException(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the authenticated envelope UUID from the claims.
|
/// Gets the authenticated envelope UUID from the claims.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string GetEnvelopeUuidOfReceiver(this ClaimsPrincipal user) => user.GetRequiredClaimOfReceiver(EnvelopeUuidClaimTypes);
|
public static string GetEnvelopeUuidOfReceiver(this ClaimsPrincipal user) => user.GetRequiredClaimOfReceiver(ClaimTypes.NameIdentifier);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the authenticated receiver signature from the claims.
|
/// Gets the authenticated receiver signature from the claims.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string GetReceiverSignatureOfReceiver(this ClaimsPrincipal user) => user.GetRequiredClaimOfReceiver(ReceiverSignatureClaimTypes);
|
public static string GetReceiverSignatureOfReceiver(this ClaimsPrincipal user) => user.GetRequiredClaimOfReceiver(ClaimTypes.Hash);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the authenticated receiver display name from the claims.
|
/// Gets the authenticated receiver display name from the claims.
|
||||||
@@ -78,7 +57,7 @@ public static class ReceiverClaimExtensions
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public static int GetEnvelopeIdOfReceiver(this ClaimsPrincipal user)
|
public static int GetEnvelopeIdOfReceiver(this ClaimsPrincipal user)
|
||||||
{
|
{
|
||||||
var envIdStr = user.GetRequiredClaimOfReceiver(EnvelopeIdClaimTypes);
|
var envIdStr = user.GetRequiredClaimOfReceiver(EnvelopeClaimTypes.Id);
|
||||||
if (!int.TryParse(envIdStr, out var envId))
|
if (!int.TryParse(envIdStr, out var envId))
|
||||||
{
|
{
|
||||||
throw new InvalidOperationException($"Claim '{EnvelopeClaimTypes.Id}' is not a valid integer.");
|
throw new InvalidOperationException($"Claim '{EnvelopeClaimTypes.Id}' is not a valid integer.");
|
||||||
|
|||||||
@@ -1,7 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.API.Models;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Request body for the envelope-receiver login endpoint.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="AccessCode">The access code sent to the receiver.</param>
|
|
||||||
public record EnvelopeReceiverLogin(string? AccessCode = null);
|
|
||||||
@@ -19,7 +19,6 @@ using DigitalData.Core.Abstractions.Security.Extensions;
|
|||||||
using EnvelopeGenerator.API.Middleware;
|
using EnvelopeGenerator.API.Middleware;
|
||||||
using NLog.Web;
|
using NLog.Web;
|
||||||
using NLog;
|
using NLog;
|
||||||
using DigitalData.Auth.Claims;
|
|
||||||
|
|
||||||
var logger = LogManager.Setup().LoadConfigurationFromAppSettings().GetCurrentClassLogger();
|
var logger = LogManager.Setup().LoadConfigurationFromAppSettings().GetCurrentClassLogger();
|
||||||
logger.Info("Logging initialized!");
|
logger.Info("Logging initialized!");
|
||||||
@@ -111,9 +110,7 @@ try
|
|||||||
|
|
||||||
options.DocumentFilter<EnvelopeGenerator.API.Documentation.AuthProxyDocumentFilter>();
|
options.DocumentFilter<EnvelopeGenerator.API.Documentation.AuthProxyDocumentFilter>();
|
||||||
});
|
});
|
||||||
#if NET9_0_OR_GREATER
|
|
||||||
builder.Services.AddOpenApi();
|
builder.Services.AddOpenApi();
|
||||||
#endif
|
|
||||||
|
|
||||||
//AddEF Core dbcontext
|
//AddEF Core dbcontext
|
||||||
var useDbMigration = Environment.GetEnvironmentVariable("MIGRATION_TEST_MODE") == true.ToString() || config.GetValue<bool>("UseDbMigration");
|
var useDbMigration = Environment.GetEnvironmentVariable("MIGRATION_TEST_MODE") == true.ToString() || config.GetValue<bool>("UseDbMigration");
|
||||||
@@ -129,9 +126,6 @@ try
|
|||||||
|
|
||||||
var authTokenKeys = config.GetOrDefault<AuthTokenKeys>();
|
var authTokenKeys = config.GetOrDefault<AuthTokenKeys>();
|
||||||
|
|
||||||
// Scheme name used for per-envelope receiver JWT authentication.
|
|
||||||
const string EnvelopeReceiverScheme = "EnvelopeReceiverJwt";
|
|
||||||
|
|
||||||
builder.Services.AddAuthentication(options =>
|
builder.Services.AddAuthentication(options =>
|
||||||
{
|
{
|
||||||
options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
|
options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
|
||||||
@@ -169,61 +163,6 @@ try
|
|||||||
return Task.CompletedTask;
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
})
|
|
||||||
// Per-envelope receiver scheme: reads the JWT from the cookie named
|
|
||||||
// AuthTokenSignFLOWReceiver.{envelope_key} where envelope_key is the
|
|
||||||
// last path segment of the request URL.
|
|
||||||
// This enables simultaneous authentication for multiple envelopes
|
|
||||||
// within the same browser session.
|
|
||||||
.AddJwtBearer(EnvelopeReceiverScheme, opt =>
|
|
||||||
{
|
|
||||||
opt.TokenValidationParameters = new TokenValidationParameters
|
|
||||||
{
|
|
||||||
ValidateIssuerSigningKey = true,
|
|
||||||
IssuerSigningKeyResolver = (token, securityToken, identifier, parameters) =>
|
|
||||||
{
|
|
||||||
var clientParams = deferredProvider.GetOptions<ClientParams>();
|
|
||||||
var publicKey = clientParams!.PublicKeys.Get(authTokenKeys.Issuer, authTokenKeys.Audience);
|
|
||||||
return [publicKey.SecurityKey];
|
|
||||||
},
|
|
||||||
ValidateIssuer = true,
|
|
||||||
ValidIssuer = authTokenKeys.Issuer,
|
|
||||||
ValidateAudience = true,
|
|
||||||
ValidAudience = authTokenKeys.Audience,
|
|
||||||
};
|
|
||||||
|
|
||||||
opt.Events = new JwtBearerEvents
|
|
||||||
{
|
|
||||||
OnMessageReceived = context =>
|
|
||||||
{
|
|
||||||
var paths = context.Request.Path.Value?.Split('/', StringSplitOptions.RemoveEmptyEntries);
|
|
||||||
|
|
||||||
// Derive the envelope key from the last route segment: /{envelope_key}
|
|
||||||
var envelopeKey = paths?.LastOrDefault();
|
|
||||||
|
|
||||||
if (envelopeKey is not null)
|
|
||||||
{
|
|
||||||
var cookieName = CookieNames.GetEnvelopeReceiverCookieName(authTokenKeys.Cookie, envelopeKey);
|
|
||||||
if (context.Request.Cookies.TryGetValue(cookieName, out var cookieToken) && cookieToken is not null)
|
|
||||||
context.Token = cookieToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Task.CompletedTask;
|
|
||||||
},
|
|
||||||
OnTokenValidated = context =>
|
|
||||||
{
|
|
||||||
var paths = context.Request.Path.Value?.Split('/', StringSplitOptions.RemoveEmptyEntries);
|
|
||||||
var envelopeKey = paths?.LastOrDefault();
|
|
||||||
|
|
||||||
var sub = context.Principal?.FindFirst(System.Security.Claims.ClaimTypes.NameIdentifier)?.Value
|
|
||||||
?? context.Principal?.FindFirst("sub")?.Value;
|
|
||||||
|
|
||||||
if (envelopeKey is null || sub != envelopeKey)
|
|
||||||
context.Fail("Envelope key in the path does not match the token subject.");
|
|
||||||
|
|
||||||
return Task.CompletedTask;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Authentication
|
// Authentication
|
||||||
@@ -243,13 +182,8 @@ try
|
|||||||
policy.RequireRole(Role.Sender, Role.Receiver.Full))
|
policy.RequireRole(Role.Sender, Role.Receiver.Full))
|
||||||
.AddPolicy(AuthPolicy.Sender, policy =>
|
.AddPolicy(AuthPolicy.Sender, policy =>
|
||||||
policy.RequireRole(Role.Sender))
|
policy.RequireRole(Role.Sender))
|
||||||
// Per-envelope policy: uses the dedicated EnvelopeReceiverJwt scheme so it
|
|
||||||
// never conflicts with the default JwtBearer scheme.
|
|
||||||
.AddPolicy(AuthPolicy.Receiver, policy =>
|
.AddPolicy(AuthPolicy.Receiver, policy =>
|
||||||
policy
|
policy.RequireRole(Role.Receiver.Full))
|
||||||
.AddAuthenticationSchemes(EnvelopeReceiverScheme)
|
|
||||||
.RequireAuthenticatedUser()
|
|
||||||
.RequireRole(Role.Receiver.Full, "receiver"))
|
|
||||||
.AddPolicy(AuthPolicy.ReceiverTFA, policy =>
|
.AddPolicy(AuthPolicy.ReceiverTFA, policy =>
|
||||||
policy.RequireRole(Role.Receiver.TFA));
|
policy.RequireRole(Role.Receiver.TFA));
|
||||||
|
|
||||||
@@ -290,9 +224,7 @@ try
|
|||||||
|
|
||||||
app.UseMiddleware<ExceptionHandlingMiddleware>();
|
app.UseMiddleware<ExceptionHandlingMiddleware>();
|
||||||
|
|
||||||
#if NET9_0_OR_GREATER
|
|
||||||
app.MapOpenApi();
|
app.MapOpenApi();
|
||||||
#endif
|
|
||||||
|
|
||||||
// Configure the HTTP request pipeline.
|
// Configure the HTTP request pipeline.
|
||||||
if (app.Environment.IsDevelopment() || (app.IsDevOrDiP() && config.GetValue<bool>("UseSwagger")))
|
if (app.Environment.IsDevelopment() || (app.IsDevOrDiP() && config.GetValue<bool>("UseSwagger")))
|
||||||
@@ -326,10 +258,8 @@ try
|
|||||||
app.UseAuthentication();
|
app.UseAuthentication();
|
||||||
app.UseAuthorization();
|
app.UseAuthorization();
|
||||||
|
|
||||||
app.MapControllers();
|
|
||||||
|
|
||||||
// Catch-all YARP proxy — only forward requests that are not swagger/scalar/openapi paths.
|
|
||||||
app.MapReverseProxy();
|
app.MapReverseProxy();
|
||||||
|
app.MapControllers();
|
||||||
|
|
||||||
app.Run();
|
app.Run();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<!--
|
|
||||||
https://go.microsoft.com/fwlink/?LinkID=208121.
|
|
||||||
-->
|
|
||||||
<Project>
|
|
||||||
<PropertyGroup>
|
|
||||||
<WebPublishMethod>Package</WebPublishMethod>
|
|
||||||
<LastUsedBuildConfiguration>Release</LastUsedBuildConfiguration>
|
|
||||||
<LastUsedPlatform>Any CPU</LastUsedPlatform>
|
|
||||||
<SiteUrlToLaunchAfterPublish />
|
|
||||||
<LaunchSiteAfterPublish>true</LaunchSiteAfterPublish>
|
|
||||||
<ExcludeApp_Data>false</ExcludeApp_Data>
|
|
||||||
<ProjectGuid>5e0e17c0-ff5a-4246-bf87-1add85376a27</ProjectGuid>
|
|
||||||
<DesktopBuildPackageLocation>M:\App&Service\0 DD - Smart UP\signFLOW\API\net8\$(Version)\EnvelopeGenerator.API.zip</DesktopBuildPackageLocation>
|
|
||||||
<PackageAsSingleFile>true</PackageAsSingleFile>
|
|
||||||
<DeployIisAppPath>EnvelopeGenerator</DeployIisAppPath>
|
|
||||||
<_TargetId>IISWebDeployPackage</_TargetId>
|
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
|
||||||
</PropertyGroup>
|
|
||||||
</Project>
|
|
||||||
@@ -4,17 +4,6 @@
|
|||||||
"Default": "Information",
|
"Default": "Information",
|
||||||
"Microsoft.AspNetCore": "Warning"
|
"Microsoft.AspNetCore": "Warning"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"ReverseProxy": {
|
|
||||||
"Clusters": {
|
|
||||||
"receiver-ui": {
|
|
||||||
"Destinations": {
|
|
||||||
"primary": {
|
|
||||||
"Address": "https://localhost:52936"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"AuthClientParams": {
|
"AuthClientParams": {
|
||||||
"Url": "http://172.24.12.39:9090/auth-hub",
|
"Url": "http://172.24.12.39:9090/auth-hub",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"UseSwagger": true,
|
"UseSwagger": true,
|
||||||
"UseDbMigration": false,
|
"UseDbMigration": true,
|
||||||
"DiPMode": true,
|
"DiPMode": true,
|
||||||
"Logging": {
|
"Logging": {
|
||||||
"LogLevel": {
|
"LogLevel": {
|
||||||
|
|||||||
@@ -1,65 +1,6 @@
|
|||||||
{
|
{
|
||||||
"ReverseProxy": {
|
"ReverseProxy": {
|
||||||
"Routes": {
|
"Routes": {
|
||||||
"receiver-ui-receiver": {
|
|
||||||
"ClusterId": "receiver-ui",
|
|
||||||
"Order": 100,
|
|
||||||
"Match": {
|
|
||||||
"Path": "/receiver/{**catch-all}",
|
|
||||||
"Methods": [ "GET", "HEAD" ]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"receiver-ui-login": {
|
|
||||||
"ClusterId": "receiver-ui",
|
|
||||||
"Order": 100,
|
|
||||||
"Match": {
|
|
||||||
"Path": "/login/{**catch-all}",
|
|
||||||
"Methods": [ "GET", "HEAD" ]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"receiver-ui-sender": {
|
|
||||||
"ClusterId": "receiver-ui",
|
|
||||||
"Order": 100,
|
|
||||||
"Match": {
|
|
||||||
"Path": "/sender/{**catch-all}",
|
|
||||||
"Methods": [ "GET", "HEAD" ]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"receiver-ui-envelope": {
|
|
||||||
"ClusterId": "receiver-ui",
|
|
||||||
"Order": 100,
|
|
||||||
"Match": {
|
|
||||||
"Path": "/envelope/{EnvelopeKey}",
|
|
||||||
"Methods": [ "GET", "HEAD" ]
|
|
||||||
},
|
|
||||||
"Transforms": [
|
|
||||||
{ "PathSet": "/index.html" }
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"receiver-ui-static-assets": {
|
|
||||||
"ClusterId": "receiver-ui",
|
|
||||||
"Order": 999,
|
|
||||||
"Match": {
|
|
||||||
"Path": "{**catch-all}",
|
|
||||||
"Methods": [ "GET", "HEAD" ]
|
|
||||||
},
|
|
||||||
"Transforms": [
|
|
||||||
{ "ResponseHeader": "Cache-Control", "Set": "no-cache, no-store, must-revalidate", "When": "Always" },
|
|
||||||
{ "ResponseHeader": "Pragma", "Set": "no-cache", "When": "Always" },
|
|
||||||
{ "ResponseHeader": "Expires", "Set": "0", "When": "Always" }
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"receiver-ui-annotation-fake": {
|
|
||||||
"ClusterId": "receiver-ui",
|
|
||||||
"Order": 10,
|
|
||||||
"Match": {
|
|
||||||
"Path": "/api/Annotation/{envelopeKey}",
|
|
||||||
"Methods": [ "GET", "HEAD" ]
|
|
||||||
},
|
|
||||||
"Transforms": [
|
|
||||||
{ "PathSet": "/fake-data/annotations.json" }
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"auth-login": {
|
"auth-login": {
|
||||||
"ClusterId": "auth-hub",
|
"ClusterId": "auth-hub",
|
||||||
"Match": {
|
"Match": {
|
||||||
@@ -69,27 +10,9 @@
|
|||||||
"Transforms": [
|
"Transforms": [
|
||||||
{ "PathSet": "/api/auth/sign-flow" }
|
{ "PathSet": "/api/auth/sign-flow" }
|
||||||
]
|
]
|
||||||
},
|
|
||||||
"auth-envelope-receiver-login": {
|
|
||||||
"ClusterId": "auth-hub",
|
|
||||||
"Match": {
|
|
||||||
"Path": "/api/Auth/envelope-receiver/{key}",
|
|
||||||
"Methods": [ "POST" ]
|
|
||||||
},
|
|
||||||
"Transforms": [
|
|
||||||
{ "PathPattern": "/api/auth/envelope-receiver/{key}" },
|
|
||||||
{ "QueryValueParameter": "cookie", "Set": "true" }
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"Clusters": {
|
"Clusters": {
|
||||||
"receiver-ui": {
|
|
||||||
"Destinations": {
|
|
||||||
"primary": {
|
|
||||||
"Address": "https://localhost:52936"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"auth-hub": {
|
"auth-hub": {
|
||||||
"Destinations": {
|
"Destinations": {
|
||||||
"primary": {
|
"primary": {
|
||||||
@@ -100,4 +23,3 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
13
EnvelopeGenerator.Application/Common/CacheKey.cs
Normal file
13
EnvelopeGenerator.Application/Common/CacheKey.cs
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
namespace EnvelopeGenerator.Application.Common;
|
||||||
|
|
||||||
|
// TODO: merge other cache keys here as well, e.g. for templates, etc.
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public static class CacheKey
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public static readonly Guid DefaultConfig = Guid.NewGuid();
|
||||||
|
}
|
||||||
@@ -8,74 +8,42 @@ public record AnnotationCreateDto
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public long Id { get; set; }
|
public int ElementId { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Obsolete("Not required for DevExpress")]
|
public string Name { get; init; } = null!;
|
||||||
public int ElementId { get; init; } = -1;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Obsolete("Not required for DevExpress")]
|
public string Value { get; init; } = null!;
|
||||||
public string Name { get; init; } = string.Empty;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Obsolete("Not required for DevExpress")]
|
public string Type { get; init; } = null!;
|
||||||
public string Value { get; init; } = string.Empty;
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Obsolete("Not required for DevExpress")]
|
|
||||||
public string Type { get; init; } = string.Empty;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Horizontal position of the signature field on the page.
|
|
||||||
/// <br/><br/>
|
|
||||||
/// <b>DevExpress unit:</b> Hundredths of an inch (1/100 inch ≈ 2.83 PDF points), origin at the <b>top-left</b> corner of the page, X increases to the right.
|
|
||||||
/// <br/>
|
|
||||||
/// <b>Difference from PSPDFKit:</b> PSPDFKit also uses top-left origin but measures in PDF points (1/72 inch).
|
|
||||||
/// To convert: <c>xDevExpress = xPsPdfKit * (100.0 / 72.0)</c>
|
|
||||||
/// <br/>
|
|
||||||
/// <b>Difference from GDPicture:</b> GDPicture uses PDF points with <b>bottom-left</b> origin (standard PDF coordinate system).
|
|
||||||
/// The X axis is the same direction, only unit conversion is needed: <c>xDevExpress = xGdPicture * (100.0 / 72.0)</c>
|
|
||||||
/// </summary>
|
|
||||||
public double? X { get; init; }
|
public double? X { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Vertical position of the signature field on the page.
|
///
|
||||||
/// <br/><br/>
|
|
||||||
/// <b>DevExpress unit:</b> Hundredths of an inch (1/100 inch ≈ 2.83 PDF points), origin at the <b>top-left</b> corner of the page, Y increases downward.
|
|
||||||
/// <br/>
|
|
||||||
/// <b>Difference from PSPDFKit:</b> PSPDFKit also uses top-left origin and Y increases downward, but measures in PDF points (1/72 inch).
|
|
||||||
/// To convert: <c>yDevExpress = yPsPdfKit * (100.0 / 72.0)</c>
|
|
||||||
/// <br/>
|
|
||||||
/// <b>Difference from GDPicture:</b> GDPicture uses PDF points with <b>bottom-left</b> origin, so Y increases <b>upward</b> (PDF standard).
|
|
||||||
/// To convert: <c>yDevExpress = (pageHeightInPt - yGdPicture - elementHeightInPt) * (100.0 / 72.0)</c>
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public double? Y { get; init; }
|
public double? Y { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Obsolete("Not required for DevExpress")]
|
|
||||||
public double? Width { get; init; }
|
public double? Width { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Obsolete("Not required for DevExpress")]
|
|
||||||
public double? Height { get; init; }
|
public double? Height { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Added to eliminate the need for SignatureDto in DevExpress
|
|
||||||
/// </summary>
|
|
||||||
public int? Page { get; init; }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -8,6 +8,11 @@ namespace EnvelopeGenerator.Application.Common.Dto;
|
|||||||
[ApiExplorerSettings(IgnoreApi = true)]
|
[ApiExplorerSettings(IgnoreApi = true)]
|
||||||
public class ConfigDto
|
public class ConfigDto
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the default document path.
|
||||||
|
/// </summary>
|
||||||
|
public string? DocumentPath { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the sending profile identifier.
|
/// Gets or sets the sending profile identifier.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -26,5 +31,30 @@ public class ConfigDto
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets or sets the path where exports will be saved.
|
/// Gets or sets the path where exports will be saved.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string? ExportPath { get; set; }
|
public string ExportPath { get; set; } = null!;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the creation timestamp.
|
||||||
|
/// </summary>
|
||||||
|
public DateTime AddedWhen { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the last update timestamp.
|
||||||
|
/// </summary>
|
||||||
|
public DateTime? ChangedWhen { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets the legacy tinyint GUID field.
|
||||||
|
/// </summary>
|
||||||
|
public byte Guid { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets whether default TFA is enabled.
|
||||||
|
/// </summary>
|
||||||
|
public bool DefTfaEnabled { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets or sets whether default TFA uses phone.
|
||||||
|
/// </summary>
|
||||||
|
public bool DefTfaWithPhone { get; set; }
|
||||||
}
|
}
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
using DigitalData.EmailProfilerDispatcher.Abstraction.Attributes;
|
using DigitalData.EmailProfilerDispatcher.Abstraction.Attributes;
|
||||||
using DigitalData.UserManager.Application.DTOs.User;
|
using DigitalData.UserManager.Application.DTOs.User;
|
||||||
|
using EnvelopeGenerator.Application.Common.Dto.EnvelopeReceiver;
|
||||||
using EnvelopeGenerator.Domain.Constants;
|
using EnvelopeGenerator.Domain.Constants;
|
||||||
using EnvelopeGenerator.Domain.Entities;
|
using EnvelopeGenerator.Domain.Entities;
|
||||||
using EnvelopeGenerator.Domain.Interfaces;
|
using EnvelopeGenerator.Domain.Interfaces;
|
||||||
@@ -44,6 +45,16 @@ public record EnvelopeDto : IEnvelope
|
|||||||
[TemplatePlaceholder("[MESSAGE]")]
|
[TemplatePlaceholder("[MESSAGE]")]
|
||||||
public string Message { get; set; } = string.Empty;
|
public string Message { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public DateTime? ExpiresWhen { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public DateTime? ExpiresWarningWhen { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -60,6 +71,11 @@ public record EnvelopeDto : IEnvelope
|
|||||||
[TemplatePlaceholder("[DOCUMENT_TITLE]")]
|
[TemplatePlaceholder("[DOCUMENT_TITLE]")]
|
||||||
public string Title { get; set; } = string.Empty;
|
public string Title { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Default value is string.Empty
|
||||||
|
/// </summary>
|
||||||
|
public string? Comment { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -73,14 +89,27 @@ public record EnvelopeDto : IEnvelope
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public int? EnvelopeTypeId { get; set; }
|
public bool SendReminderEmails { get; set; }
|
||||||
|
|
||||||
// TODO: use ReadAndConfirm property name
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[Obsolete("Use EnvelopeExtensions.IsReadAndConfirm extension metot instead.")]
|
public int? FirstReminderDays { get; set; }
|
||||||
public bool ReadOnly => this.IsReadAndConfirm();
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public int? ReminderIntervalDays { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public int? EnvelopeTypeId { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public bool ReadOnly => EnvelopeTypeId == 2;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
@@ -92,6 +121,26 @@ public record EnvelopeDto : IEnvelope
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public bool UseAccessCode { get; set; } = true;
|
public bool UseAccessCode { get; set; } = true;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public int? FinalEmailToCreator { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public int? FinalEmailToReceivers { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public int? ExpiresWhenDays { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public int? ExpiresWarningWhenDays { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -126,4 +175,9 @@ public record EnvelopeDto : IEnvelope
|
|||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public IEnumerable<DocumentDto>? Documents { get; set; }
|
public IEnumerable<DocumentDto>? Documents { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public IEnumerable<EnvelopeReceiverDto>? EnvelopeReceivers { get; set; }
|
||||||
}
|
}
|
||||||
@@ -13,12 +13,12 @@ public static class AutoMapperAuditingExtensions
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public static IMappingExpression<TSource, TDestination> MapAddedWhen<TSource, TDestination>(this IMappingExpression<TSource, TDestination> expression)
|
public static IMappingExpression<TSource, TDestination> MapAddedWhen<TSource, TDestination>(this IMappingExpression<TSource, TDestination> expression)
|
||||||
where TDestination : IHasAddedWhen
|
where TDestination : IHasAddedWhen
|
||||||
=> expression.ForMember(dest => dest.AddedWhen, opt => opt.MapFrom(_ => DateTime.Now));
|
=> expression.ForMember(dest => dest.AddedWhen, opt => opt.MapFrom(_ => DateTime.UtcNow));
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Maps <see cref="IHasChangedWhen.ChangedWhen"/> to the current UTC time.
|
/// Maps <see cref="IHasChangedWhen.ChangedWhen"/> to the current UTC time.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static IMappingExpression<TSource, TDestination> MapChangedWhen<TSource, TDestination>(this IMappingExpression<TSource, TDestination> expression)
|
public static IMappingExpression<TSource, TDestination> MapChangedWhen<TSource, TDestination>(this IMappingExpression<TSource, TDestination> expression)
|
||||||
where TDestination : IHasChangedWhen
|
where TDestination : IHasChangedWhen
|
||||||
=> expression.ForMember(dest => dest.ChangedWhen, opt => opt.MapFrom(_ => DateTime.Now));
|
=> expression.ForMember(dest => dest.ChangedWhen, opt => opt.MapFrom(_ => DateTime.UtcNow));
|
||||||
}
|
}
|
||||||
@@ -113,7 +113,7 @@ public abstract class SendMailHandler<TNotification> : INotificationHandler<TNot
|
|||||||
EmailAddress = notification.EmailAddress,
|
EmailAddress = notification.EmailAddress,
|
||||||
EmailBody = temp.Body,
|
EmailBody = temp.Body,
|
||||||
EmailSubj = temp.Subject,
|
EmailSubj = temp.Subject,
|
||||||
AddedWhen = DateTime.Now,
|
AddedWhen = DateTime.UtcNow,
|
||||||
AddedWho = DispatcherParams.AddedWho,
|
AddedWho = DispatcherParams.AddedWho,
|
||||||
SendingProfile = DispatcherParams.SendingProfile,
|
SendingProfile = DispatcherParams.SendingProfile,
|
||||||
ReminderTypeId = DispatcherParams.ReminderTypeId,
|
ReminderTypeId = DispatcherParams.ReminderTypeId,
|
||||||
|
|||||||
@@ -14,4 +14,10 @@ public record EnvelopeQueryBase
|
|||||||
/// Die universell eindeutige Kennung des Umschlags.
|
/// Die universell eindeutige Kennung des Umschlags.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public virtual string? Uuid { get; set; }
|
public virtual string? Uuid { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Wenn gesetzt, wird das DocResult-Feld in der Abfrage einbezogen.
|
||||||
|
/// Standardmäßig wird DocResult nicht geladen, um große Binärdaten zu vermeiden.
|
||||||
|
/// </summary>
|
||||||
|
public bool IncludeDocResult { get; set; } = false;
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,69 @@
|
|||||||
|
using AutoMapper;
|
||||||
|
using DigitalData.Core.Abstraction.Application.Repository;
|
||||||
|
using DigitalData.Core.Exceptions;
|
||||||
|
using EnvelopeGenerator.Application.Common;
|
||||||
|
using EnvelopeGenerator.Application.Common.Dto;
|
||||||
|
using EnvelopeGenerator.Domain.Entities;
|
||||||
|
using MediatR;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using Microsoft.Extensions.Caching.Distributed;
|
||||||
|
using Microsoft.Extensions.Caching.Memory;
|
||||||
|
|
||||||
|
namespace EnvelopeGenerator.Application.Configuration.Queries;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public record ReadDefaultConfigQuery : IRequest<ConfigDto>
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public bool EnforceSingleResult { get; init; }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public class ReadDefaultConfigQueryHandler : IRequestHandler<ReadDefaultConfigQuery, ConfigDto>
|
||||||
|
{
|
||||||
|
private readonly IRepository<Config> _repo;
|
||||||
|
|
||||||
|
private readonly IMapper _mapper;
|
||||||
|
|
||||||
|
private readonly IMemoryCache _cache;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="repo"></param>
|
||||||
|
/// <param name="mapper"></param>
|
||||||
|
/// <param name="cache"></param>
|
||||||
|
public ReadDefaultConfigQueryHandler(IRepository<Config> repo, IMapper mapper, IMemoryCache cache)
|
||||||
|
{
|
||||||
|
_repo = repo;
|
||||||
|
_mapper = mapper;
|
||||||
|
_cache = cache;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request"></param>
|
||||||
|
/// <param name="cancel"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
/// <exception cref="InvalidOperationException"></exception>
|
||||||
|
public async Task<ConfigDto> Handle(ReadDefaultConfigQuery request, CancellationToken cancel)
|
||||||
|
{
|
||||||
|
var config = await _cache.GetOrCreateAsync(CacheKey.DefaultConfig, entry =>
|
||||||
|
{
|
||||||
|
entry.AbsoluteExpirationRelativeToNow = TimeSpan.FromMinutes(30);
|
||||||
|
return request.EnforceSingleResult
|
||||||
|
? _repo.Query.SingleOrDefaultAsync(cancel)
|
||||||
|
: _repo.Query.FirstOrDefaultAsync(cancel)
|
||||||
|
?? throw new NotFoundException("Default configuration could not be found. Ensure at least one configuration record exists in the database.");
|
||||||
|
});
|
||||||
|
|
||||||
|
return _mapper.Map<ConfigDto>(config);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -27,7 +27,7 @@ public class MappingProfile : Profile
|
|||||||
CreateMap<UpdateDocStatusCommand, DocumentStatus>()
|
CreateMap<UpdateDocStatusCommand, DocumentStatus>()
|
||||||
.ForMember(dest => dest.Envelope, opt => opt.Ignore())
|
.ForMember(dest => dest.Envelope, opt => opt.Ignore())
|
||||||
.ForMember(dest => dest.Receiver, opt => opt.Ignore())
|
.ForMember(dest => dest.Receiver, opt => opt.Ignore())
|
||||||
.ForMember(dest => dest.StatusChangedWhen, opt => opt.MapFrom(src => DateTime.Now))
|
.ForMember(dest => dest.StatusChangedWhen, opt => opt.MapFrom(src => DateTime.UtcNow))
|
||||||
.MapChangedWhen();
|
.MapChangedWhen();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -41,7 +41,7 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup Condition="'$(TargetFramework)' == 'net9.0'">
|
<ItemGroup Condition="'$(TargetFramework)' == 'net9.0'">
|
||||||
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="9.0.5" />
|
<PackageReference Include="Microsoft.Extensions.DependencyInjection" Version="9.0.6" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
@@ -81,6 +81,7 @@
|
|||||||
<ItemGroup Condition="'$(TargetFramework)' == 'net7.0'">
|
<ItemGroup Condition="'$(TargetFramework)' == 'net7.0'">
|
||||||
<PackageReference Include="AutoMapper" Version="13.0.1" />
|
<PackageReference Include="AutoMapper" Version="13.0.1" />
|
||||||
<PackageReference Include="Microsoft.Data.SqlClient" Version="5.2.2" />
|
<PackageReference Include="Microsoft.Data.SqlClient" Version="5.2.2" />
|
||||||
|
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="7.0.20" />
|
||||||
<PackageReference Include="CommandDotNet">
|
<PackageReference Include="CommandDotNet">
|
||||||
<Version>7.0.5</Version>
|
<Version>7.0.5</Version>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
@@ -88,6 +89,7 @@
|
|||||||
|
|
||||||
<ItemGroup Condition="'$(TargetFramework)' == 'net8.0'">
|
<ItemGroup Condition="'$(TargetFramework)' == 'net8.0'">
|
||||||
<PackageReference Include="AutoMapper" Version="14.0.0" />
|
<PackageReference Include="AutoMapper" Version="14.0.0" />
|
||||||
|
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="8.0.17" />
|
||||||
<PackageReference Include="CommandDotNet">
|
<PackageReference Include="CommandDotNet">
|
||||||
<Version>8.1.1</Version>
|
<Version>8.1.1</Version>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
@@ -95,6 +97,7 @@
|
|||||||
|
|
||||||
<ItemGroup Condition="'$(TargetFramework)' == 'net9.0'">
|
<ItemGroup Condition="'$(TargetFramework)' == 'net9.0'">
|
||||||
<PackageReference Include="AutoMapper" Version="14.0.0" />
|
<PackageReference Include="AutoMapper" Version="14.0.0" />
|
||||||
|
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="9.0.6" />
|
||||||
<PackageReference Include="CommandDotNet">
|
<PackageReference Include="CommandDotNet">
|
||||||
<Version>8.1.1</Version>
|
<Version>8.1.1</Version>
|
||||||
</PackageReference>
|
</PackageReference>
|
||||||
|
|||||||
@@ -1,127 +0,0 @@
|
|||||||
using AutoMapper;
|
|
||||||
using DigitalData.Core.Abstraction.Application.Repository;
|
|
||||||
using EnvelopeGenerator.Application.Envelopes.Queries;
|
|
||||||
using EnvelopeGenerator.Application.Receivers.Queries;
|
|
||||||
using MediatR;
|
|
||||||
using EnvelopeGenerator.Domain.Entities;
|
|
||||||
using Microsoft.EntityFrameworkCore;
|
|
||||||
using EnvelopeGenerator.Application.Common.Dto.EnvelopeReceiver;
|
|
||||||
using EnvelopeGenerator.Application.Common.Query;
|
|
||||||
using EnvelopeGenerator.Application.Common.Extensions;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.Application.EnvelopeReceivers.Queries;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Represents a query for reading an envelope receiver including sensitive fields
|
|
||||||
/// (access code, phone number) that are excluded from the standard <see cref="ReadEnvelopeReceiverQuery"/>.
|
|
||||||
/// </summary>
|
|
||||||
/// <remarks>
|
|
||||||
/// Returns a single <see cref="EnvelopeReceiverSecretDto"/> matched by UUID and receiver signature.
|
|
||||||
/// Equivalent to the legacy <c>ReadWithSecretByUuidSignatureAsync</c> service method.
|
|
||||||
/// </remarks>
|
|
||||||
public record ReadEnvelopeReceiverSecretQuery
|
|
||||||
: EnvelopeReceiverQueryBase<ReadEnvelopeQuery, ReadReceiverQuery>,
|
|
||||||
IRequest<EnvelopeReceiverSecretDto?>;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Extension methods for dispatching <see cref="ReadEnvelopeReceiverSecretQuery"/> via <see cref="IMediator"/>.
|
|
||||||
/// </summary>
|
|
||||||
public static class ReadEnvelopeReceiverSecretQueryExtensions
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Sends a <see cref="ReadEnvelopeReceiverSecretQuery"/> using the composite key (uuid::signature).
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="mediator">The mediator instance.</param>
|
|
||||||
/// <param name="key">Composite key in the format <c>uuid::signature</c>.</param>
|
|
||||||
/// <param name="cancel">Cancellation token.</param>
|
|
||||||
/// <returns>The matching <see cref="EnvelopeReceiverSecretDto"/>, or <c>null</c> if not found.</returns>
|
|
||||||
public static Task<EnvelopeReceiverSecretDto?> ReadEnvelopeReceiverSecretAsync(
|
|
||||||
this IMediator mediator,
|
|
||||||
string key,
|
|
||||||
CancellationToken cancel = default)
|
|
||||||
=> mediator.Send(new ReadEnvelopeReceiverSecretQuery { Key = key }, cancel);
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Sends a <see cref="ReadEnvelopeReceiverSecretQuery"/> using UUID and receiver signature.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="mediator">The mediator instance.</param>
|
|
||||||
/// <param name="uuid">Envelope UUID.</param>
|
|
||||||
/// <param name="signature">Receiver signature.</param>
|
|
||||||
/// <param name="cancel">Cancellation token.</param>
|
|
||||||
/// <returns>The matching <see cref="EnvelopeReceiverSecretDto"/>, or <c>null</c> if not found.</returns>
|
|
||||||
public static Task<EnvelopeReceiverSecretDto?> ReadEnvelopeReceiverSecretAsync(
|
|
||||||
this IMediator mediator,
|
|
||||||
string uuid,
|
|
||||||
string signature,
|
|
||||||
CancellationToken cancel = default)
|
|
||||||
{
|
|
||||||
var q = new ReadEnvelopeReceiverSecretQuery();
|
|
||||||
q.Envelope.Uuid = uuid;
|
|
||||||
q.Receiver.Signature = signature;
|
|
||||||
return mediator.Send(q, cancel);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Handles <see cref="ReadEnvelopeReceiverSecretQuery"/> and returns a
|
|
||||||
/// <see cref="EnvelopeReceiverSecretDto"/> containing sensitive fields.
|
|
||||||
/// </summary>
|
|
||||||
public class ReadEnvelopeReceiverSecretQueryHandler
|
|
||||||
: IRequestHandler<ReadEnvelopeReceiverSecretQuery, EnvelopeReceiverSecretDto?>
|
|
||||||
{
|
|
||||||
private readonly IRepository<EnvelopeReceiver> _repo;
|
|
||||||
private readonly IRepository<Receiver> _rcvRepo;
|
|
||||||
private readonly IMapper _mapper;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Initializes a new instance of <see cref="ReadEnvelopeReceiverSecretQueryHandler"/>.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="envelopeReceiver">Repository for <see cref="EnvelopeReceiver"/>.</param>
|
|
||||||
/// <param name="rcvRepo">Repository for <see cref="Receiver"/>.</param>
|
|
||||||
/// <param name="mapper">AutoMapper instance.</param>
|
|
||||||
public ReadEnvelopeReceiverSecretQueryHandler(
|
|
||||||
IRepository<EnvelopeReceiver> envelopeReceiver,
|
|
||||||
IRepository<Receiver> rcvRepo,
|
|
||||||
IMapper mapper)
|
|
||||||
{
|
|
||||||
_repo = envelopeReceiver;
|
|
||||||
_rcvRepo = rcvRepo;
|
|
||||||
_mapper = mapper;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Handles the query and returns the matching <see cref="EnvelopeReceiverSecretDto"/>.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="request">The query containing filter criteria.</param>
|
|
||||||
/// <param name="cancel">Cancellation token.</param>
|
|
||||||
/// <returns>
|
|
||||||
/// The matched <see cref="EnvelopeReceiverSecretDto"/>, or <c>null</c> if no record is found.
|
|
||||||
/// </returns>
|
|
||||||
public async Task<EnvelopeReceiverSecretDto?> Handle(
|
|
||||||
ReadEnvelopeReceiverSecretQuery request,
|
|
||||||
CancellationToken cancel)
|
|
||||||
{
|
|
||||||
var q = _repo.Query.Where(request, notnull: false);
|
|
||||||
|
|
||||||
var envRcvs = await q
|
|
||||||
.Include(er => er.Envelope).ThenInclude(e => e!.Documents!).ThenInclude(d => d.Elements)
|
|
||||||
.Include(er => er.Envelope).ThenInclude(e => e!.Histories)
|
|
||||||
.Include(er => er.Envelope).ThenInclude(e => e!.User)
|
|
||||||
.Include(er => er.Receiver)
|
|
||||||
.ToListAsync(cancel);
|
|
||||||
|
|
||||||
if (request.Receiver.HasAnyCriteria && envRcvs.Count != 0)
|
|
||||||
{
|
|
||||||
var receiver = await _rcvRepo.Query.Where(request.Receiver).FirstAsync(cancel);
|
|
||||||
|
|
||||||
foreach (var item in envRcvs)
|
|
||||||
item.Envelope?.Documents?.FirstOrDefault()?.Elements?.RemoveAll(s => s.ReceiverId != receiver.Id);
|
|
||||||
}
|
|
||||||
|
|
||||||
var envRcv = envRcvs.FirstOrDefault();
|
|
||||||
if (envRcv is null)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
return _mapper.Map<EnvelopeReceiverSecretDto>(envRcv);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -40,10 +40,10 @@ public record CreateEnvelopeCommand : IRequest<EnvelopeDto?>
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="userId"></param>
|
/// <param name="userId"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public CreateEnvelopeCommand WithAuth(int userId)
|
public bool Authorize(int userId)
|
||||||
{
|
{
|
||||||
UserId = userId;
|
UserId = userId;
|
||||||
return this;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -24,6 +24,12 @@ public record ReadEnvelopeQuery : EnvelopeQueryBase, IRequest<IEnumerable<Envelo
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public int? UserId { get; init; }
|
public int? UserId { get; init; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Optionaler Zeitfilter in Minuten; wenn gesetzt, werden nur Umschläge zurückgegeben,
|
||||||
|
/// deren letzte Änderung mindestens diese Anzahl an Minuten zurückliegt.
|
||||||
|
/// </summary>
|
||||||
|
public int? MinMinutesSinceLastChange { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Setzt den Benutzerkontext für die Abfrage.
|
/// Setzt den Benutzerkontext für die Abfrage.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -132,6 +138,12 @@ public class ReadEnvelopeQueryHandler : IRequestHandler<ReadEnvelopeQuery, IEnum
|
|||||||
query = query.Where(e => !status.Ignore.Contains(e.Status));
|
query = query.Where(e => !status.Ignore.Contains(e.Status));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (request.MinMinutesSinceLastChange is int minMinutesSinceLastChange)
|
||||||
|
{
|
||||||
|
query = query.Where(e => e.ChangedWhen.HasValue
|
||||||
|
&& EF.Functions.DateDiffMinute(e.ChangedWhen.Value, DateTime.Now) >= minMinutesSinceLastChange);
|
||||||
|
}
|
||||||
|
|
||||||
var envelopes = await query
|
var envelopes = await query
|
||||||
.Include(e => e.Documents)
|
.Include(e => e.Documents)
|
||||||
.ToListAsync(cancel);
|
.ToListAsync(cancel);
|
||||||
|
|||||||
@@ -0,0 +1,48 @@
|
|||||||
|
using MediatR;
|
||||||
|
using EnvelopeGenerator.Application.Common.Dto;
|
||||||
|
using DigitalData.Core.Exceptions;
|
||||||
|
|
||||||
|
namespace EnvelopeGenerator.Application.Envelopes.Queries;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Repräsentiert eine Abfrage für Umschläge.
|
||||||
|
/// </summary>
|
||||||
|
public record ReadSingleEnvelopeDocResultQuery() : IRequest<byte[]>
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
public ReadSingleEnvelopeQuery Envelope { get; set; } = null!;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Verarbeitet <see cref="ReadEnvelopeQuery"/> und liefert passende <see cref="EnvelopeDto"/>-Ergebnisse.
|
||||||
|
/// </summary>
|
||||||
|
public class ReadSingleEnvelopeDocResultQueryHandler : IRequestHandler<ReadSingleEnvelopeDocResultQuery, byte[]>
|
||||||
|
{
|
||||||
|
private readonly IMediator _mediator;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="mediator"></param>
|
||||||
|
public ReadSingleEnvelopeDocResultQueryHandler(IMediator mediator)
|
||||||
|
{
|
||||||
|
_mediator = mediator;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request"></param>
|
||||||
|
/// <param name="cancellationToken"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
public async Task<byte[]> Handle(ReadSingleEnvelopeDocResultQuery request, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
request.Envelope.IncludeDocResult = true;
|
||||||
|
var result = await _mediator.Send(request.Envelope, cancellationToken);
|
||||||
|
return result.DocResult is byte[] docResult && docResult.Length > 0
|
||||||
|
? docResult
|
||||||
|
: throw new NotFoundException($"Document for Envelope with ID {request.Envelope.Id} not found");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,78 @@
|
|||||||
|
using MediatR;
|
||||||
|
using EnvelopeGenerator.Application.Common.Query;
|
||||||
|
using EnvelopeGenerator.Application.Common.Dto;
|
||||||
|
using AutoMapper;
|
||||||
|
using DigitalData.Core.Abstraction.Application.Repository;
|
||||||
|
using EnvelopeGenerator.Domain.Entities;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
using DigitalData.Core.Exceptions;
|
||||||
|
|
||||||
|
namespace EnvelopeGenerator.Application.Envelopes.Queries;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Repräsentiert eine Abfrage für einen einzelnen Umschlag.
|
||||||
|
/// </summary>
|
||||||
|
public record ReadSingleEnvelopeQuery : EnvelopeQueryBase, IRequest<EnvelopeDto>
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Optionaler Benutzerfilter; wenn gesetzt, werden nur Umschläge des Benutzers geladen.
|
||||||
|
/// </summary>
|
||||||
|
public int? UserId { get; init; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Setzt den Benutzerkontext für die Abfrage.
|
||||||
|
/// </summary>
|
||||||
|
public ReadSingleEnvelopeQuery Authorize(int userId) => this with { UserId = userId };
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Verarbeitet <see cref="ReadSingleEnvelopeQuery"/> und liefert ein einzelnes <see cref="EnvelopeDto"/>-Ergebnis.
|
||||||
|
/// </summary>
|
||||||
|
public class ReadSingleEnvelopeQueryHandler : IRequestHandler<ReadSingleEnvelopeQuery, EnvelopeDto>
|
||||||
|
{
|
||||||
|
private readonly IRepository<Envelope> _repository;
|
||||||
|
private readonly IMapper _mapper;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="repository"></param>
|
||||||
|
/// <param name="mapper"></param>
|
||||||
|
public ReadSingleEnvelopeQueryHandler(IRepository<Envelope> repository, IMapper mapper)
|
||||||
|
{
|
||||||
|
_repository = repository;
|
||||||
|
_mapper = mapper;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request"></param>
|
||||||
|
/// <param name="cancel"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
public async Task<EnvelopeDto> Handle(ReadSingleEnvelopeQuery request, CancellationToken cancel)
|
||||||
|
{
|
||||||
|
var query = _repository.Query;
|
||||||
|
|
||||||
|
if (request.UserId is int userId)
|
||||||
|
query = query.Where(e => e.UserId == userId);
|
||||||
|
|
||||||
|
if (request.Id is int id)
|
||||||
|
query = query.Where(e => e.Id == id);
|
||||||
|
|
||||||
|
if (request.Uuid is string uuid)
|
||||||
|
query = query.Where(e => e.Uuid == uuid);
|
||||||
|
|
||||||
|
var envelopes = await query
|
||||||
|
.Include(e => e.Documents)
|
||||||
|
.Take(2)
|
||||||
|
.ToListAsync(cancel);
|
||||||
|
|
||||||
|
if (envelopes.Count > 1)
|
||||||
|
throw new BadRequestException($"Multiple envelopes found for the given criteria: Id={request.Id}, Uuid={request.Uuid}, UserId={request.UserId}");
|
||||||
|
|
||||||
|
return envelopes.SingleOrDefault() is Envelope envelope
|
||||||
|
? _mapper.Map<EnvelopeDto>(envelope)
|
||||||
|
: throw new NotFoundException($"Envelope with Id={request.Id}, Uuid={request.Uuid} not found");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -34,7 +34,7 @@ public record CreateHistoryCommand : EnvelopeReceiverQueryBase, IRequest<History
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public DateTime AddedWhen { get; } = DateTime.Now;
|
public DateTime AddedWhen { get; } = DateTime.UtcNow;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
|
|||||||
@@ -448,7 +448,7 @@
|
|||||||
<value>Document has been reset.</value>
|
<value>Document has been reset.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentSuccessfullyConfirmed" xml:space="preserve">
|
<data name="DocumentSuccessfullyConfirmed" xml:space="preserve">
|
||||||
<value>Document successfully read and confirmed!</value>
|
<value>Document successfully red and confirmed!</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentConfirmedConfirmationMessage" xml:space="preserve">
|
<data name="DocumentConfirmedConfirmationMessage" xml:space="preserve">
|
||||||
<value>You have read and confirmed the document. You will receive a written confirmation afterwards.</value>
|
<value>You have read and confirmed the document. You will receive a written confirmation afterwards.</value>
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ using Microsoft.Extensions.Logging;
|
|||||||
using EnvelopeGenerator.Application.Common.Dto;
|
using EnvelopeGenerator.Application.Common.Dto;
|
||||||
using EnvelopeGenerator.Application.Common.Interfaces.Repositories;
|
using EnvelopeGenerator.Application.Common.Interfaces.Repositories;
|
||||||
using EnvelopeGenerator.Application.Common.Interfaces.Services;
|
using EnvelopeGenerator.Application.Common.Interfaces.Services;
|
||||||
|
using EnvelopeGenerator.Application.Common;
|
||||||
|
|
||||||
namespace EnvelopeGenerator.Application.Services;
|
namespace EnvelopeGenerator.Application.Services;
|
||||||
|
|
||||||
@@ -16,8 +17,6 @@ namespace EnvelopeGenerator.Application.Services;
|
|||||||
[Obsolete("Use MediatR")]
|
[Obsolete("Use MediatR")]
|
||||||
public class ConfigService : ReadService<IConfigRepository, ConfigDto, Config, int>, IConfigService
|
public class ConfigService : ReadService<IConfigRepository, ConfigDto, Config, int>, IConfigService
|
||||||
{
|
{
|
||||||
private static readonly Guid DefaultConfigCacheId = Guid.NewGuid();
|
|
||||||
|
|
||||||
private readonly IMemoryCache _cache;
|
private readonly IMemoryCache _cache;
|
||||||
|
|
||||||
private readonly ILogger<ConfigService> _logger;
|
private readonly ILogger<ConfigService> _logger;
|
||||||
@@ -62,7 +61,7 @@ public class ConfigService : ReadService<IConfigRepository, ConfigDto, Config, i
|
|||||||
/// </exception>
|
/// </exception>
|
||||||
public async Task<ConfigDto> ReadDefaultAsync()
|
public async Task<ConfigDto> ReadDefaultAsync()
|
||||||
{
|
{
|
||||||
var config = await _cache.GetOrCreateAsync(DefaultConfigCacheId, _ => ReadFirstAsync().ThenAsync(
|
var config = await _cache.GetOrCreateAsync(CacheKey.DefaultConfig, _ => ReadFirstAsync().ThenAsync(
|
||||||
Success: config => config,
|
Success: config => config,
|
||||||
Fail: (mssg, ntc) =>
|
Fail: (mssg, ntc) =>
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -0,0 +1,92 @@
|
|||||||
|
using DigitalData.Core.Abstraction.Application.Repository;
|
||||||
|
using DigitalData.Core.Exceptions;
|
||||||
|
using EnvelopeGenerator.Domain.Entities;
|
||||||
|
using MediatR;
|
||||||
|
using Microsoft.EntityFrameworkCore;
|
||||||
|
|
||||||
|
namespace EnvelopeGenerator.Application.ThirdPartyModules.Queries;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Query to read the license text of a third-party module.
|
||||||
|
/// Either <see cref="Id"/> or <see cref="Name"/> must be provided, but not both.
|
||||||
|
/// </summary>
|
||||||
|
public record ReadThirdPartyModuleLicenseQuery : IRequest<string>
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// The unique identifier of the third-party module (optional).
|
||||||
|
/// </summary>
|
||||||
|
public int? Id { get; init; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The name of the third-party module (optional).
|
||||||
|
/// </summary>
|
||||||
|
public string? Name { get; init; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Whether to filter only active modules. Defaults to <c>true</c>.
|
||||||
|
/// </summary>
|
||||||
|
public bool Active { get; init; } = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Handles <see cref="ReadThirdPartyModuleLicenseQuery"/> and returns the license text of a third-party module.
|
||||||
|
/// </summary>
|
||||||
|
public class ReadThirdPartyModuleLicenseQueryHandler : IRequestHandler<ReadThirdPartyModuleLicenseQuery, string>
|
||||||
|
{
|
||||||
|
private readonly IRepository<ThirdPartyModule> _repository;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Initializes a new instance of the <see cref="ReadThirdPartyModuleLicenseQueryHandler"/> class.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="repository">The repository for accessing third-party modules.</param>
|
||||||
|
public ReadThirdPartyModuleLicenseQueryHandler(IRepository<ThirdPartyModule> repository)
|
||||||
|
{
|
||||||
|
_repository = repository;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Handles the query by filtering on Id or Name and returning the license text.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The query parameters.</param>
|
||||||
|
/// <param name="cancel">A cancellation token.</param>
|
||||||
|
/// <returns>The license text of the matching third-party module.</returns>
|
||||||
|
/// <exception cref="BadRequestException">
|
||||||
|
/// Thrown when neither Id nor Name is provided, or when both are provided.
|
||||||
|
/// </exception>
|
||||||
|
/// <exception cref="InvalidOperationException">
|
||||||
|
/// Thrown when multiple modules match the given criteria, indicating a data integrity issue.
|
||||||
|
/// </exception>
|
||||||
|
/// <exception cref="NotFoundException">
|
||||||
|
/// Thrown when no module matches the given criteria.
|
||||||
|
/// </exception>
|
||||||
|
public async Task<string> Handle(ReadThirdPartyModuleLicenseQuery request, CancellationToken cancel)
|
||||||
|
{
|
||||||
|
if (request.Id is null && request.Name is null)
|
||||||
|
throw new BadRequestException("Either Id or Name must be provided.");
|
||||||
|
|
||||||
|
if (request.Id is not null && request.Name is not null)
|
||||||
|
throw new BadRequestException("Only one of Id or Name must be provided, not both.");
|
||||||
|
|
||||||
|
var query = _repository.Query
|
||||||
|
.Where(m => m.Active == request.Active);
|
||||||
|
|
||||||
|
if (request.Id is int id)
|
||||||
|
query = query.Where(m => m.Id == id);
|
||||||
|
|
||||||
|
if (request.Name is string name)
|
||||||
|
query = query.Where(m => m.Name == name);
|
||||||
|
|
||||||
|
var modules = await query
|
||||||
|
.Take(2)
|
||||||
|
.ToListAsync(cancel);
|
||||||
|
|
||||||
|
if (modules.Count > 1)
|
||||||
|
throw new InvalidOperationException(
|
||||||
|
$"Data integrity violation: multiple third-party modules found for the given criteria (Id={request.Id}, Name={request.Name}, Active={request.Active}).");
|
||||||
|
|
||||||
|
return modules.SingleOrDefault() is ThirdPartyModule module
|
||||||
|
? module.License
|
||||||
|
: throw new NotFoundException(
|
||||||
|
$"Third-party module not found for the given criteria (Id={request.Id}, Name={request.Name}, Active={request.Active}).");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -487,6 +487,10 @@
|
|||||||
<Project>{6EA0C51F-C2B1-4462-8198-3DE0B32B74F8}</Project>
|
<Project>{6EA0C51F-C2B1-4462-8198-3DE0B32B74F8}</Project>
|
||||||
<Name>EnvelopeGenerator.CommonServices</Name>
|
<Name>EnvelopeGenerator.CommonServices</Name>
|
||||||
</ProjectReference>
|
</ProjectReference>
|
||||||
|
<ProjectReference Include="..\EnvelopeGenerator.CommonServices\EnvelopeGenerator.CommonServices.vbproj">
|
||||||
|
<Project>{6ea0c51f-c2b1-4462-8198-3de0b32b74f8}</Project>
|
||||||
|
<Name>EnvelopeGenerator.CommonServices</Name>
|
||||||
|
</ProjectReference>
|
||||||
<ProjectReference Include="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj">
|
<ProjectReference Include="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj">
|
||||||
<Project>{4f32a98d-e6f0-4a09-bd97-1cf26107e837}</Project>
|
<Project>{4f32a98d-e6f0-4a09-bd97-1cf26107e837}</Project>
|
||||||
<Name>EnvelopeGenerator.Domain</Name>
|
<Name>EnvelopeGenerator.Domain</Name>
|
||||||
|
|||||||
12
EnvelopeGenerator.BBTests/frmFinalizePDF.Designer.vb
generated
12
EnvelopeGenerator.BBTests/frmFinalizePDF.Designer.vb
generated
@@ -27,7 +27,6 @@ Partial Class frmFinalizePDF
|
|||||||
Me.Label2 = New System.Windows.Forms.Label()
|
Me.Label2 = New System.Windows.Forms.Label()
|
||||||
Me.Button1 = New System.Windows.Forms.Button()
|
Me.Button1 = New System.Windows.Forms.Button()
|
||||||
Me.Button2 = New System.Windows.Forms.Button()
|
Me.Button2 = New System.Windows.Forms.Button()
|
||||||
Me.Button3 = New System.Windows.Forms.Button()
|
|
||||||
Me.txtResult = New System.Windows.Forms.TextBox()
|
Me.txtResult = New System.Windows.Forms.TextBox()
|
||||||
Me.txtEnvelope = New System.Windows.Forms.TextBox()
|
Me.txtEnvelope = New System.Windows.Forms.TextBox()
|
||||||
Me.SuspendLayout()
|
Me.SuspendLayout()
|
||||||
@@ -76,15 +75,6 @@ Partial Class frmFinalizePDF
|
|||||||
Me.Button2.Text = "Merge Json"
|
Me.Button2.Text = "Merge Json"
|
||||||
Me.Button2.UseVisualStyleBackColor = True
|
Me.Button2.UseVisualStyleBackColor = True
|
||||||
'
|
'
|
||||||
'Button3
|
|
||||||
'
|
|
||||||
Me.Button3.Location = New System.Drawing.Point(15, 160)
|
|
||||||
Me.Button3.Name = "Button3"
|
|
||||||
Me.Button3.Size = New System.Drawing.Size(166, 23)
|
|
||||||
Me.Button3.TabIndex = 5
|
|
||||||
Me.Button3.Text = "Full Finalize Test"
|
|
||||||
Me.Button3.UseVisualStyleBackColor = True
|
|
||||||
'
|
|
||||||
'txtResult
|
'txtResult
|
||||||
'
|
'
|
||||||
Me.txtResult.Location = New System.Drawing.Point(333, 12)
|
Me.txtResult.Location = New System.Drawing.Point(333, 12)
|
||||||
@@ -107,7 +97,6 @@ Partial Class frmFinalizePDF
|
|||||||
Me.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font
|
Me.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font
|
||||||
Me.ClientSize = New System.Drawing.Size(800, 450)
|
Me.ClientSize = New System.Drawing.Size(800, 450)
|
||||||
Me.Controls.Add(Me.txtResult)
|
Me.Controls.Add(Me.txtResult)
|
||||||
Me.Controls.Add(Me.Button3)
|
|
||||||
Me.Controls.Add(Me.Button2)
|
Me.Controls.Add(Me.Button2)
|
||||||
Me.Controls.Add(Me.Button1)
|
Me.Controls.Add(Me.Button1)
|
||||||
Me.Controls.Add(Me.Label2)
|
Me.Controls.Add(Me.Label2)
|
||||||
@@ -127,6 +116,5 @@ Partial Class frmFinalizePDF
|
|||||||
Friend WithEvents Label2 As Label
|
Friend WithEvents Label2 As Label
|
||||||
Friend WithEvents Button1 As Button
|
Friend WithEvents Button1 As Button
|
||||||
Friend WithEvents Button2 As Button
|
Friend WithEvents Button2 As Button
|
||||||
Friend WithEvents Button3 As Button
|
|
||||||
Friend WithEvents txtResult As TextBox
|
Friend WithEvents txtResult As TextBox
|
||||||
End Class
|
End Class
|
||||||
|
|||||||
@@ -8,8 +8,6 @@ Imports Newtonsoft.Json.Linq
|
|||||||
Imports EnvelopeGenerator.Infrastructure
|
Imports EnvelopeGenerator.Infrastructure
|
||||||
Imports Microsoft.EntityFrameworkCore
|
Imports Microsoft.EntityFrameworkCore
|
||||||
Imports DigitalData.Core.Abstractions
|
Imports DigitalData.Core.Abstractions
|
||||||
Imports DigitalData.Core.Abstraction.Application.Repository
|
|
||||||
Imports EnvelopeGenerator.Domain.Entities
|
|
||||||
|
|
||||||
Public Class frmFinalizePDF
|
Public Class frmFinalizePDF
|
||||||
Private Const CONNECTIONSTRING = "Server=sDD-VMP04-SQL17\DD_DEVELOP01;Database=DD_ECM;User Id=sa;Password=+bk8oAbbQP1AzoHtvZUbd+Mbok2f8Fl4miEx1qssJ5yEaEWoQJ9prg4L14fURpPnqi1WMNs9fE4=;" + "Encrypt=True;TrustServerCertificate=True;"
|
Private Const CONNECTIONSTRING = "Server=sDD-VMP04-SQL17\DD_DEVELOP01;Database=DD_ECM;User Id=sa;Password=+bk8oAbbQP1AzoHtvZUbd+Mbok2f8Fl4miEx1qssJ5yEaEWoQJ9prg4L14fURpPnqi1WMNs9fE4=;" + "Encrypt=True;TrustServerCertificate=True;"
|
||||||
@@ -126,86 +124,4 @@ Public Class frmFinalizePDF
|
|||||||
|
|
||||||
txtResult.Text = oJObject1.ToString()
|
txtResult.Text = oJObject1.ToString()
|
||||||
End Sub
|
End Sub
|
||||||
|
|
||||||
Private Sub Button3_Click(sender As Object, e As EventArgs) Handles Button3.Click
|
|
||||||
Try
|
|
||||||
Dim envelopeId As Integer = CInt(txtEnvelope.Text)
|
|
||||||
Dim log As New System.Text.StringBuilder()
|
|
||||||
|
|
||||||
' 1. Load annotation JSON data (same as Service)
|
|
||||||
Dim oTable = LoadAnnotationDataForEnvelope()
|
|
||||||
Dim oJsonList = oTable.Rows.
|
|
||||||
Cast(Of DataRow).
|
|
||||||
Select(Function(r As DataRow) r.Item("VALUE").ToString()).
|
|
||||||
ToList()
|
|
||||||
log.AppendLine($"Annotation JSON count: {oJsonList.Count}")
|
|
||||||
|
|
||||||
' 2. Load document bytes (same as Service)
|
|
||||||
Dim oBuffer As Byte() = ReadEnvelope(envelopeId)
|
|
||||||
log.AppendLine($"Document bytes: {oBuffer.Length}")
|
|
||||||
|
|
||||||
' 3. Check what BurnAnnotsToPDF will do internally
|
|
||||||
Using scope = Factory.Shared.ScopeFactory.CreateScope()
|
|
||||||
Dim envRepo = scope.ServiceProvider.Repository(Of Envelope)()
|
|
||||||
Dim envelope = envRepo.Where(Function(env) env.Id = envelopeId).FirstOrDefault()
|
|
||||||
|
|
||||||
If envelope Is Nothing Then
|
|
||||||
log.AppendLine("ERROR: Envelope not found in EF Core!")
|
|
||||||
txtResult.Text = log.ToString()
|
|
||||||
Return
|
|
||||||
End If
|
|
||||||
|
|
||||||
log.AppendLine($"Envelope found: Id={envelope.Id}, EnvelopeTypeId={envelope.EnvelopeTypeId}")
|
|
||||||
log.AppendLine($"ReadOnly (IsReadAndConfirm): {envelope.ReadOnly}")
|
|
||||||
|
|
||||||
If envelope.ReadOnly Then
|
|
||||||
log.AppendLine(">>> EARLY RETURN: ReadOnly=True, original PDF returned without burning")
|
|
||||||
txtResult.Text = log.ToString()
|
|
||||||
Return
|
|
||||||
End If
|
|
||||||
|
|
||||||
Dim sigRepo = scope.ServiceProvider.Repository(Of Signature)()
|
|
||||||
Dim elements = sigRepo _
|
|
||||||
.Where(Function(sig) sig.Document.EnvelopeId = envelopeId) _
|
|
||||||
.Include(Function(sig) sig.Annotations) _
|
|
||||||
.ToList()
|
|
||||||
|
|
||||||
log.AppendLine($"Elements (Signature) count: {elements.Count}")
|
|
||||||
|
|
||||||
If elements.Any() Then
|
|
||||||
log.AppendLine(">>> PATH: BurnElementAnnotsToPDF (new element-based path)")
|
|
||||||
For Each elem In elements
|
|
||||||
Dim annotCount = If(elem.Annotations IsNot Nothing, elem.Annotations.Count(), 0)
|
|
||||||
log.AppendLine($" Element Id={elem.Id}, Page={elem.Page}, X={elem.X}, Y={elem.Y}, W={elem.Width}, H={elem.Height}, Annotations={annotCount}")
|
|
||||||
If elem.Annotations IsNot Nothing Then
|
|
||||||
For Each annot In elem.Annotations
|
|
||||||
log.AppendLine($" Annot: Name={annot.Name}, Type={annot.Type}, X={annot.X}, Y={annot.Y}, W={annot.Width}, H={annot.Height}")
|
|
||||||
Next
|
|
||||||
End If
|
|
||||||
Next
|
|
||||||
Else
|
|
||||||
log.AppendLine(">>> PATH: BurnInstantJSONAnnotsToPDF (old JSON-based path)")
|
|
||||||
End If
|
|
||||||
End Using
|
|
||||||
|
|
||||||
' 4. Actually call BurnAnnotsToPDF (same as Service)
|
|
||||||
log.AppendLine("")
|
|
||||||
log.AppendLine("Calling BurnAnnotsToPDF...")
|
|
||||||
Dim oNewBuffer = PDFBurner.BurnAnnotsToPDF(oBuffer, oJsonList, envelopeId)
|
|
||||||
log.AppendLine($"Result bytes: {oNewBuffer.Length}")
|
|
||||||
log.AppendLine($"Same as input: {oBuffer.Length = oNewBuffer.Length AndAlso oBuffer.SequenceEqual(oNewBuffer)}")
|
|
||||||
|
|
||||||
' 5. Write output
|
|
||||||
Dim desktopPath As String = Environment.GetFolderPath(Environment.SpecialFolder.Desktop)
|
|
||||||
Dim oNewPath = Path.Combine(desktopPath, $"E{txtEnvelope.Text}_FullTest.burned.pdf")
|
|
||||||
File.WriteAllBytes(oNewPath, oNewBuffer)
|
|
||||||
log.AppendLine($"Output: {oNewPath}")
|
|
||||||
|
|
||||||
txtResult.Text = log.ToString()
|
|
||||||
Process.Start(oNewPath)
|
|
||||||
|
|
||||||
Catch ex As Exception
|
|
||||||
txtResult.Text = $"ERROR: {ex.Message}{vbCrLf}{vbCrLf}{ex.ToString()}"
|
|
||||||
End Try
|
|
||||||
End Sub
|
|
||||||
End Class
|
End Class
|
||||||
@@ -15,13 +15,13 @@
|
|||||||
<package id="DigitalData.Modules.Messaging" version="1.9.8" targetFramework="net462" />
|
<package id="DigitalData.Modules.Messaging" version="1.9.8" targetFramework="net462" />
|
||||||
<package id="DocumentFormat.OpenXml" version="3.2.0" targetFramework="net462" />
|
<package id="DocumentFormat.OpenXml" version="3.2.0" targetFramework="net462" />
|
||||||
<package id="DocumentFormat.OpenXml.Framework" version="3.2.0" targetFramework="net462" />
|
<package id="DocumentFormat.OpenXml.Framework" version="3.2.0" targetFramework="net462" />
|
||||||
<package id="EntityFramework" version="6.4.4" targetFramework="net462" />
|
<package id="EntityFramework" version="6.5.1" targetFramework="net462" />
|
||||||
<package id="EntityFramework.Firebird" version="6.4.0" targetFramework="net462" />
|
<package id="EntityFramework.Firebird" version="6.4.0" targetFramework="net462" />
|
||||||
<package id="FirebirdSql.Data.FirebirdClient" version="7.5.0" targetFramework="net462" />
|
<package id="FirebirdSql.Data.FirebirdClient" version="7.5.0" targetFramework="net462" />
|
||||||
<package id="GdPicture" version="14.3.3" targetFramework="net462" />
|
<package id="GdPicture" version="14.3.3" targetFramework="net462" />
|
||||||
<package id="GdPicture.runtimes.windows" version="14.3.3" targetFramework="net462" />
|
<package id="GdPicture.runtimes.windows" version="14.3.3" targetFramework="net462" />
|
||||||
<package id="Microsoft.AspNet.WebApi.Client" version="6.0.0" targetFramework="net462" />
|
<package id="Microsoft.AspNet.WebApi.Client" version="6.0.0" targetFramework="net462" />
|
||||||
<package id="Microsoft.Bcl.AsyncInterfaces" version="8.0.0" targetFramework="net462" />
|
<package id="Microsoft.Bcl.AsyncInterfaces" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="Microsoft.Bcl.Cryptography" version="9.0.0" targetFramework="net462" />
|
<package id="Microsoft.Bcl.Cryptography" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="Microsoft.Bcl.HashCode" version="1.1.1" targetFramework="net462" />
|
<package id="Microsoft.Bcl.HashCode" version="1.1.1" targetFramework="net462" />
|
||||||
<package id="Microsoft.CSharp" version="4.7.0" targetFramework="net462" />
|
<package id="Microsoft.CSharp" version="4.7.0" targetFramework="net462" />
|
||||||
@@ -62,10 +62,10 @@
|
|||||||
<package id="protobuf-net.Core" version="3.2.46" targetFramework="net462" />
|
<package id="protobuf-net.Core" version="3.2.46" targetFramework="net462" />
|
||||||
<package id="RtfPipe" version="2.0.7677.4303" targetFramework="net462" />
|
<package id="RtfPipe" version="2.0.7677.4303" targetFramework="net462" />
|
||||||
<package id="S22.Imap" version="3.6.0.0" targetFramework="net462" />
|
<package id="S22.Imap" version="3.6.0.0" targetFramework="net462" />
|
||||||
<package id="System.Buffers" version="4.6.0" targetFramework="net462" />
|
<package id="System.Buffers" version="4.6.1" targetFramework="net462" />
|
||||||
<package id="System.ClientModel" version="1.8.0" targetFramework="net462" />
|
<package id="System.ClientModel" version="1.8.0" targetFramework="net462" />
|
||||||
<package id="System.CodeDom" version="8.0.0" targetFramework="net462" />
|
<package id="System.CodeDom" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.Collections.Immutable" version="8.0.0" targetFramework="net462" />
|
<package id="System.Collections.Immutable" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.ComponentModel.Annotations" version="4.7.0" targetFramework="net462" />
|
<package id="System.ComponentModel.Annotations" version="4.7.0" targetFramework="net462" />
|
||||||
<package id="System.Data.Common" version="4.3.0" targetFramework="net462" />
|
<package id="System.Data.Common" version="4.3.0" targetFramework="net462" />
|
||||||
<package id="System.Data.Odbc" version="6.0.1" targetFramework="net462" />
|
<package id="System.Data.Odbc" version="6.0.1" targetFramework="net462" />
|
||||||
@@ -74,23 +74,23 @@
|
|||||||
<package id="System.Formats.Asn1" version="10.0.3" targetFramework="net462" />
|
<package id="System.Formats.Asn1" version="10.0.3" targetFramework="net462" />
|
||||||
<package id="System.IdentityModel.Tokens.Jwt" version="7.7.1" targetFramework="net462" />
|
<package id="System.IdentityModel.Tokens.Jwt" version="7.7.1" targetFramework="net462" />
|
||||||
<package id="System.IO.FileSystem.AccessControl" version="5.0.0" targetFramework="net462" />
|
<package id="System.IO.FileSystem.AccessControl" version="5.0.0" targetFramework="net462" />
|
||||||
<package id="System.IO.Packaging" version="8.0.1" targetFramework="net462" />
|
<package id="System.IO.Packaging" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.IO.Pipelines" version="9.0.0" targetFramework="net462" />
|
<package id="System.IO.Pipelines" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.Management" version="8.0.0" targetFramework="net462" />
|
<package id="System.Management" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.Memory" version="4.6.0" targetFramework="net462" />
|
<package id="System.Memory" version="4.6.3" targetFramework="net462" />
|
||||||
<package id="System.Memory.Data" version="8.0.1" targetFramework="net462" />
|
<package id="System.Memory.Data" version="8.0.1" targetFramework="net462" />
|
||||||
<package id="System.Numerics.Vectors" version="4.6.0" targetFramework="net462" />
|
<package id="System.Numerics.Vectors" version="4.6.1" targetFramework="net462" />
|
||||||
<package id="System.Runtime.CompilerServices.Unsafe" version="6.1.0" targetFramework="net462" />
|
<package id="System.Runtime.CompilerServices.Unsafe" version="6.1.2" targetFramework="net462" />
|
||||||
<package id="System.Security.AccessControl" version="6.0.1" targetFramework="net462" />
|
<package id="System.Security.AccessControl" version="6.0.1" targetFramework="net462" />
|
||||||
<package id="System.Security.Cryptography.Algorithms" version="4.3.1" targetFramework="net462" />
|
<package id="System.Security.Cryptography.Algorithms" version="4.3.1" targetFramework="net462" />
|
||||||
<package id="System.Security.Cryptography.Cng" version="5.0.0" targetFramework="net462" />
|
<package id="System.Security.Cryptography.Cng" version="5.0.0" targetFramework="net462" />
|
||||||
<package id="System.Security.Cryptography.Pkcs" version="8.0.1" targetFramework="net462" />
|
<package id="System.Security.Cryptography.Pkcs" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.Security.Cryptography.Primitives" version="4.3.0" targetFramework="net462" />
|
<package id="System.Security.Cryptography.Primitives" version="4.3.0" targetFramework="net462" />
|
||||||
<package id="System.Security.Cryptography.ProtectedData" version="4.5.0" targetFramework="net462" />
|
<package id="System.Security.Cryptography.ProtectedData" version="4.5.0" targetFramework="net462" />
|
||||||
<package id="System.Security.Principal.Windows" version="5.0.0" targetFramework="net462" />
|
<package id="System.Security.Principal.Windows" version="5.0.0" targetFramework="net462" />
|
||||||
<package id="System.Text.Encodings.Web" version="8.0.0" targetFramework="net462" />
|
<package id="System.Text.Encodings.Web" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.Text.Json" version="8.0.6" targetFramework="net462" />
|
<package id="System.Text.Json" version="9.0.0" targetFramework="net462" />
|
||||||
<package id="System.Text.RegularExpressions" version="4.3.1" targetFramework="net462" />
|
<package id="System.Text.RegularExpressions" version="4.3.1" targetFramework="net462" />
|
||||||
<package id="System.Threading.Tasks.Extensions" version="4.5.4" targetFramework="net462" />
|
<package id="System.Threading.Tasks.Extensions" version="4.6.0" targetFramework="net462" />
|
||||||
<package id="System.ValueTuple" version="4.5.0" targetFramework="net462" />
|
<package id="System.ValueTuple" version="4.6.1" targetFramework="net462" />
|
||||||
</packages>
|
</packages>
|
||||||
@@ -15,7 +15,6 @@ Imports DigitalData.Core.Abstraction.Application
|
|||||||
Imports EnvelopeGenerator.Infrastructure
|
Imports EnvelopeGenerator.Infrastructure
|
||||||
Imports Microsoft.EntityFrameworkCore
|
Imports Microsoft.EntityFrameworkCore
|
||||||
Imports DigitalData.Core.Abstractions
|
Imports DigitalData.Core.Abstractions
|
||||||
Imports EnvelopeGenerator.Domain.Interfaces
|
|
||||||
|
|
||||||
Namespace Jobs
|
Namespace Jobs
|
||||||
Public Class FinalizeDocumentJob
|
Public Class FinalizeDocumentJob
|
||||||
@@ -231,29 +230,6 @@ Namespace Jobs
|
|||||||
|
|
||||||
Return Task.FromResult(True)
|
Return Task.FromResult(True)
|
||||||
End Function
|
End Function
|
||||||
|
|
||||||
#Region "From BBTests"
|
|
||||||
Private Function ReadEnvelope(pEnvID As Integer) As Byte()
|
|
||||||
Dim strSql As String = "Select [BYTE_DATA] from [TBSIG_ENVELOPE_DOCUMENT] WHERE ENVELOPE_ID = " & pEnvID
|
|
||||||
Dim obyteDB = Database.GetScalarValue(strSql)
|
|
||||||
If Not IsDBNull(obyteDB) Then
|
|
||||||
Dim fileData As Byte() = DirectCast(Database.GetScalarValue(strSql), Byte())
|
|
||||||
If fileData IsNot Nothing Then
|
|
||||||
Return fileData
|
|
||||||
End If
|
|
||||||
End If
|
|
||||||
|
|
||||||
Throw New InvalidOperationException($"Byte data is null. Envelope ID: {pEnvID}")
|
|
||||||
|
|
||||||
End Function
|
|
||||||
|
|
||||||
Private Function LoadAnnotationDataForEnvelope(pEnvID As Integer) As DataTable
|
|
||||||
Dim oSql = $"SELECT VALUE FROM [TBSIG_DOCUMENT_STATUS] WHERE ENVELOPE_ID = {pEnvID}"
|
|
||||||
Return Database.GetDatatable(oSql)
|
|
||||||
|
|
||||||
End Function
|
|
||||||
#End Region
|
|
||||||
|
|
||||||
Private Sub Update_File_DB(pFilePath As String, pEnvelopeID As Long)
|
Private Sub Update_File_DB(pFilePath As String, pEnvelopeID As Long)
|
||||||
Dim SqlCom As SqlCommand
|
Dim SqlCom As SqlCommand
|
||||||
Dim imageData As Byte()
|
Dim imageData As Byte()
|
||||||
@@ -351,7 +327,7 @@ Namespace Jobs
|
|||||||
Logger.Warn($"No SendFinalEmailToCreator - oMailToCreator [{oMailToCreator}] <> [{FinalEmailType.No}] ")
|
Logger.Warn($"No SendFinalEmailToCreator - oMailToCreator [{oMailToCreator}] <> [{FinalEmailType.No}] ")
|
||||||
End If
|
End If
|
||||||
|
|
||||||
If oMailToReceivers <> FinalEmailType.No And pEnvelope.IsReadAndSign() Then
|
If oMailToReceivers <> FinalEmailType.No Then
|
||||||
Logger.Debug("Sending emails to receivers..")
|
Logger.Debug("Sending emails to receivers..")
|
||||||
SendFinalEmailToReceivers(pEnvelope) ', pAttachment
|
SendFinalEmailToReceivers(pEnvelope) ', pAttachment
|
||||||
Else
|
Else
|
||||||
@@ -442,18 +418,7 @@ Namespace Jobs
|
|||||||
End Try
|
End Try
|
||||||
End If
|
End If
|
||||||
|
|
||||||
#Region "From BBTests"
|
Return PDFBurner.BurnAnnotsToPDF(oInputDocumentBuffer, oAnnotations, pEnvelopeData.EnvelopeId)
|
||||||
Dim oTable = LoadAnnotationDataForEnvelope(pEnvelopeId)
|
|
||||||
Dim oJsonList = oTable.Rows.
|
|
||||||
Cast(Of DataRow).
|
|
||||||
Select(Function(r As DataRow) r.Item("VALUE").ToString()).
|
|
||||||
ToList()
|
|
||||||
|
|
||||||
Dim oBuffer As Byte() = ReadEnvelope(pEnvelopeId)
|
|
||||||
|
|
||||||
#End Region
|
|
||||||
|
|
||||||
Return PDFBurner.BurnAnnotsToPDF(oBuffer, oJsonList, pEnvelopeId)
|
|
||||||
End Function
|
End Function
|
||||||
|
|
||||||
Private Function GetEnvelopeData(pEnvelopeId As Integer) As EnvelopeData
|
Private Function GetEnvelopeData(pEnvelopeId As Integer) As EnvelopeData
|
||||||
|
|||||||
@@ -85,37 +85,16 @@ Namespace Jobs.FinalizeDocument
|
|||||||
|
|
||||||
'Add annotations
|
'Add annotations
|
||||||
For Each element In elements
|
For Each element In elements
|
||||||
If element Is Nothing Then
|
|
||||||
Continue For
|
|
||||||
End If
|
|
||||||
|
|
||||||
Dim elementAnnotations = If(element.Annotations, Enumerable.Empty(Of ElementAnnotation)())
|
|
||||||
If Not elementAnnotations.Any() Then
|
|
||||||
Continue For
|
|
||||||
End If
|
|
||||||
|
|
||||||
Dim frameX = (element.Left - 0.7 - margin)
|
Dim frameX = (element.Left - 0.7 - margin)
|
||||||
|
|
||||||
Dim frame = elementAnnotations.FirstOrDefault(Function(a) a.Name = "frame")
|
Dim frame = element.Annotations.FirstOrDefault(Function(a) a.Name = "frame")
|
||||||
Dim frameY = element.Top - 0.5 - margin
|
Dim frameY = element.Top - 0.5 - margin
|
||||||
Dim frameYShift As Double = 0
|
Dim frameYShift = frame.Y - frameY * inchFactor
|
||||||
Dim frameXShift As Double = 0
|
Dim frameXShift = frame.X - frameX * inchFactor
|
||||||
|
|
||||||
If frame IsNot Nothing Then
|
|
||||||
frameYShift = frame.Y - frameY * inchFactor
|
|
||||||
frameXShift = frame.X - frameX * inchFactor
|
|
||||||
End If
|
|
||||||
|
|
||||||
For Each annot In elementAnnotations
|
|
||||||
If annot Is Nothing Then
|
|
||||||
Continue For
|
|
||||||
End If
|
|
||||||
|
|
||||||
Dim yOffsetofFF As Double = 0
|
|
||||||
If Not String.IsNullOrEmpty(annot.Name) Then
|
|
||||||
yOffsetsOfFF.TryGetValue(annot.Name, yOffsetofFF)
|
|
||||||
End If
|
|
||||||
|
|
||||||
|
For Each annot In element.Annotations
|
||||||
|
Dim yOffsetofFF As Double = If(yOffsetsOfFF.TryGetValue(annot.Name, yOffsetofFF), yOffsetofFF, 0)
|
||||||
Dim y = frameY + yOffsetofFF
|
Dim y = frameY + yOffsetofFF
|
||||||
|
|
||||||
If annot.Type = AnnotationType.FormField Then
|
If annot.Type = AnnotationType.FormField Then
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ Public Class PDFMerger
|
|||||||
|
|
||||||
' Convert to PDF/A
|
' Convert to PDF/A
|
||||||
oMergedPDF.ConvertToPDFA(oFinalStream, PDFAConformanceLevel, ALLOW_VECTORIZATION, ALLOW_RASTERIZATION)
|
oMergedPDF.ConvertToPDFA(oFinalStream, PDFAConformanceLevel, ALLOW_VECTORIZATION, ALLOW_RASTERIZATION)
|
||||||
oStatus = oMergedPDF.GetStat()
|
oStatus = oDocumentPDF.GetStat()
|
||||||
If oStatus <> GdPictureStatus.OK Then
|
If oStatus <> GdPictureStatus.OK Then
|
||||||
Throw New MergeDocumentException($"Document could not be converted to PDF/A: {oStatus}")
|
Throw New MergeDocumentException($"Document could not be converted to PDF/A: {oStatus}")
|
||||||
End If
|
End If
|
||||||
|
|||||||
@@ -270,26 +270,12 @@ Public Class ReceiverModel
|
|||||||
Private Function GetSignedDate(pEmailAddress As String, pEnvelopeId As Integer) As Date
|
Private Function GetSignedDate(pEmailAddress As String, pEnvelopeId As Integer) As Date
|
||||||
Try
|
Try
|
||||||
Dim oStatusInt As Integer = EnvelopeStatus.DocumentSigned
|
Dim oStatusInt As Integer = EnvelopeStatus.DocumentSigned
|
||||||
Dim value = Database.GetScalarValue($"SELECT ACTION_DATE FROM [DD_ECM].[dbo].[TBSIG_ENVELOPE_HISTORY] WHERE ENVELOPE_ID = {pEnvelopeId}
|
Return Database.GetScalarValue($"SELECT ACTION_DATE FROM [DD_ECM].[dbo].[TBSIG_ENVELOPE_HISTORY] WHERE ENVELOPE_ID = {pEnvelopeId}
|
||||||
And USER_REFERENCE = '{pEmailAddress}' AND [STATUS] = {oStatusInt}")
|
And USER_REFERENCE = '{pEmailAddress}' AND [STATUS] = {oStatusInt}")
|
||||||
|
|
||||||
If value Is Nothing OrElse value Is DBNull.Value Then
|
|
||||||
Return DateTime.MinValue
|
|
||||||
End If
|
|
||||||
|
|
||||||
If TypeOf value Is DateTime Then
|
|
||||||
Return DirectCast(value, DateTime)
|
|
||||||
End If
|
|
||||||
|
|
||||||
Dim parsedDate As DateTime
|
|
||||||
If DateTime.TryParse(value.ToString(), parsedDate) Then
|
|
||||||
Return parsedDate
|
|
||||||
End If
|
|
||||||
|
|
||||||
Return DateTime.MinValue
|
|
||||||
Catch ex As Exception
|
Catch ex As Exception
|
||||||
Logger.Error(ex)
|
Logger.Error(ex)
|
||||||
Return DateTime.MinValue
|
Return Nothing
|
||||||
End Try
|
End Try
|
||||||
End Function
|
End Function
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
'------------------------------------------------------------------------------
|
'------------------------------------------------------------------------------
|
||||||
' <auto-generated>
|
' <auto-generated>
|
||||||
' This code was generated by a tool.
|
' Dieser Code wurde von einem Tool generiert.
|
||||||
' Runtime Version:4.0.30319.42000
|
' Laufzeitversion:4.0.30319.42000
|
||||||
'
|
'
|
||||||
' Changes to this file may cause incorrect behavior and will be lost if
|
' Änderungen an dieser Datei können falsches Verhalten verursachen und gehen verloren, wenn
|
||||||
' the code is regenerated.
|
' der Code erneut generiert wird.
|
||||||
' </auto-generated>
|
' </auto-generated>
|
||||||
'------------------------------------------------------------------------------
|
'------------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -15,12 +15,12 @@ Imports System
|
|||||||
|
|
||||||
Namespace My.Resources
|
Namespace My.Resources
|
||||||
|
|
||||||
'This class was auto-generated by the StronglyTypedResourceBuilder
|
'Diese Klasse wurde von der StronglyTypedResourceBuilder automatisch generiert
|
||||||
'class via a tool like ResGen or Visual Studio.
|
'-Klasse über ein Tool wie ResGen oder Visual Studio automatisch generiert.
|
||||||
'To add or remove a member, edit your .ResX file then rerun ResGen
|
'Um einen Member hinzuzufügen oder zu entfernen, bearbeiten Sie die .ResX-Datei und führen dann ResGen
|
||||||
'with the /str option, or rebuild your VS project.
|
'mit der /str-Option erneut aus, oder Sie erstellen Ihr VS-Projekt neu.
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' A strongly-typed resource class, for looking up localized strings, etc.
|
''' Eine stark typisierte Ressourcenklasse zum Suchen von lokalisierten Zeichenfolgen usw.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
<Global.System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0"), _
|
<Global.System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0"), _
|
||||||
Global.System.Diagnostics.DebuggerNonUserCodeAttribute(), _
|
Global.System.Diagnostics.DebuggerNonUserCodeAttribute(), _
|
||||||
@@ -37,7 +37,7 @@ Namespace My.Resources
|
|||||||
End Sub
|
End Sub
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Returns the cached ResourceManager instance used by this class.
|
''' Gibt die zwischengespeicherte ResourceManager-Instanz zurück, die von dieser Klasse verwendet wird.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
<Global.System.ComponentModel.EditorBrowsableAttribute(Global.System.ComponentModel.EditorBrowsableState.Advanced)> _
|
<Global.System.ComponentModel.EditorBrowsableAttribute(Global.System.ComponentModel.EditorBrowsableState.Advanced)> _
|
||||||
Public Shared ReadOnly Property ResourceManager() As Global.System.Resources.ResourceManager
|
Public Shared ReadOnly Property ResourceManager() As Global.System.Resources.ResourceManager
|
||||||
@@ -51,8 +51,8 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Overrides the current thread's CurrentUICulture property for all
|
''' Überschreibt die CurrentUICulture-Eigenschaft des aktuellen Threads für alle
|
||||||
''' resource lookups using this strongly typed resource class.
|
''' Ressourcenzuordnungen, die diese stark typisierte Ressourcenklasse verwenden.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
<Global.System.ComponentModel.EditorBrowsableAttribute(Global.System.ComponentModel.EditorBrowsableState.Advanced)> _
|
<Global.System.ComponentModel.EditorBrowsableAttribute(Global.System.ComponentModel.EditorBrowsableState.Advanced)> _
|
||||||
Public Shared Property Culture() As Global.System.Globalization.CultureInfo
|
Public Shared Property Culture() As Global.System.Globalization.CultureInfo
|
||||||
@@ -65,7 +65,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Zugriffscode korrekt eingegeben.
|
''' Sucht eine lokalisierte Zeichenfolge, die Zugriffscode korrekt eingegeben ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property AccessCodeCorrect() As String
|
Public Shared ReadOnly Property AccessCodeCorrect() As String
|
||||||
Get
|
Get
|
||||||
@@ -74,7 +74,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Zugriffscode falsch eingegeben.
|
''' Sucht eine lokalisierte Zeichenfolge, die Zugriffscode falsch eingegeben ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property AccessCodeIncorrect() As String
|
Public Shared ReadOnly Property AccessCodeIncorrect() As String
|
||||||
Get
|
Get
|
||||||
@@ -83,7 +83,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Zugriffscode angefordert.
|
''' Sucht eine lokalisierte Zeichenfolge, die Zugriffscode angefordert ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property AccessCodeRequested() As String
|
Public Shared ReadOnly Property AccessCodeRequested() As String
|
||||||
Get
|
Get
|
||||||
@@ -92,7 +92,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Fortgeschrittene Elektronische Signatur.
|
''' Sucht eine lokalisierte Zeichenfolge, die Fortgeschrittene Elektronische Signatur ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property AdvancedElectronicSignature() As String
|
Public Shared ReadOnly Property AdvancedElectronicSignature() As String
|
||||||
Get
|
Get
|
||||||
@@ -101,7 +101,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Abgeschlossen.
|
''' Sucht eine lokalisierte Zeichenfolge, die Abgeschlossen ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Completed() As String
|
Public Shared ReadOnly Property Completed() As String
|
||||||
Get
|
Get
|
||||||
@@ -110,16 +110,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Vollständig bestätigt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Vollständig Signiert ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property CompletelyConfirmed() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("CompletelyConfirmed", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Vollständig signiert.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property CompletelySigned() As String
|
Public Shared ReadOnly Property CompletelySigned() As String
|
||||||
Get
|
Get
|
||||||
@@ -128,25 +119,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Lesebestätigung.
|
''' Sucht eine lokalisierte Zeichenfolge, die Vertrag ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property Confirmation() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("Confirmation", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Gelesen und Bestätigt.
|
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property Confirmed() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("Confirmed", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Vertrag.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Contract() As String
|
Public Shared ReadOnly Property Contract() As String
|
||||||
Get
|
Get
|
||||||
@@ -155,7 +128,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Erstellt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Erstellt ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Created() As String
|
Public Shared ReadOnly Property Created() As String
|
||||||
Get
|
Get
|
||||||
@@ -164,16 +137,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Dokument gelesen und bestätigt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Dokument Rotation geändert ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property DocumentConfirmed() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("DocumentConfirmed", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Dokument Rotation geändert.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property DocumentMod_Rotation() As String
|
Public Shared ReadOnly Property DocumentMod_Rotation() As String
|
||||||
Get
|
Get
|
||||||
@@ -182,7 +146,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Dokument geöffnet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Dokument geöffnet ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property DocumentOpened() As String
|
Public Shared ReadOnly Property DocumentOpened() As String
|
||||||
Get
|
Get
|
||||||
@@ -191,7 +155,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Unterzeichnung abgelehnt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Unterzeichnung abgelehnt ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property DocumentRejected() As String
|
Public Shared ReadOnly Property DocumentRejected() As String
|
||||||
Get
|
Get
|
||||||
@@ -200,16 +164,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Lesebestätigung abgelehnt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Dokument unterzeichnet ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property DocumentRejectedRaC() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("DocumentRejectedRaC", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Dokument unterzeichnet.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property DocumentSigned() As String
|
Public Shared ReadOnly Property DocumentSigned() As String
|
||||||
Get
|
Get
|
||||||
@@ -218,7 +173,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Entwurf.
|
''' Sucht eine lokalisierte Zeichenfolge, die Entwurf ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Draft() As String
|
Public Shared ReadOnly Property Draft() As String
|
||||||
Get
|
Get
|
||||||
@@ -227,7 +182,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Archiviert.
|
''' Sucht eine lokalisierte Zeichenfolge, die Archiviert ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeArchived() As String
|
Public Shared ReadOnly Property EnvelopeArchived() As String
|
||||||
Get
|
Get
|
||||||
@@ -236,16 +191,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Vollständig bestätigt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Vollständig signiert ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property EnvelopeCompletelyConfirmed() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("EnvelopeCompletelyConfirmed", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Vollständig signiert.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeCompletelySigned() As String
|
Public Shared ReadOnly Property EnvelopeCompletelySigned() As String
|
||||||
Get
|
Get
|
||||||
@@ -254,7 +200,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Umschlag Erstellt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Umschlag Erstellt ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeCreated() As String
|
Public Shared ReadOnly Property EnvelopeCreated() As String
|
||||||
Get
|
Get
|
||||||
@@ -263,7 +209,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Umschlag Gelöscht.
|
''' Sucht eine lokalisierte Zeichenfolge, die Umschlag Gelöscht ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeDeleted() As String
|
Public Shared ReadOnly Property EnvelopeDeleted() As String
|
||||||
Get
|
Get
|
||||||
@@ -272,16 +218,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Teil-Bestätigt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Teil-Signiert ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property EnvelopePartlyConfirmed() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("EnvelopePartlyConfirmed", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Teil-Signiert.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopePartlySigned() As String
|
Public Shared ReadOnly Property EnvelopePartlySigned() As String
|
||||||
Get
|
Get
|
||||||
@@ -290,7 +227,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Umschlag in Queue.
|
''' Sucht eine lokalisierte Zeichenfolge, die Umschlag in Queue ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeQueued() As String
|
Public Shared ReadOnly Property EnvelopeQueued() As String
|
||||||
Get
|
Get
|
||||||
@@ -299,7 +236,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Umschlag abgelehnt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Umschlag abgelehnt ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeRejected() As String
|
Public Shared ReadOnly Property EnvelopeRejected() As String
|
||||||
Get
|
Get
|
||||||
@@ -308,7 +245,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Signierungszertifikat erstellt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Signierungszertifikat erstellt ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeReportCreated() As String
|
Public Shared ReadOnly Property EnvelopeReportCreated() As String
|
||||||
Get
|
Get
|
||||||
@@ -317,16 +254,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Lesebestätigungszertifikat erstellt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Gespeichert ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property EnvelopeReportCreatedRaC() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("EnvelopeReportCreatedRaC", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Gespeichert.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeSaved() As String
|
Public Shared ReadOnly Property EnvelopeSaved() As String
|
||||||
Get
|
Get
|
||||||
@@ -335,7 +263,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Gesendet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Gesendet ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeSent() As String
|
Public Shared ReadOnly Property EnvelopeSent() As String
|
||||||
Get
|
Get
|
||||||
@@ -344,7 +272,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Umschlag zurückgezogen.
|
''' Sucht eine lokalisierte Zeichenfolge, die Umschlag zurückgezogen ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property EnvelopeWithdrawn() As String
|
Public Shared ReadOnly Property EnvelopeWithdrawn() As String
|
||||||
Get
|
Get
|
||||||
@@ -353,7 +281,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Zugriffscode versendet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Zugriffscode versendet ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property MessageAccessCodeSent() As String
|
Public Shared ReadOnly Property MessageAccessCodeSent() As String
|
||||||
Get
|
Get
|
||||||
@@ -362,7 +290,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Abschlussemail versendet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Abschlussemail versendet ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property MessageCompletionSent() As String
|
Public Shared ReadOnly Property MessageCompletionSent() As String
|
||||||
Get
|
Get
|
||||||
@@ -371,7 +299,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Signaturbestätigung versendet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Signaturbestätigung versendet ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property MessageConfirmationSent() As String
|
Public Shared ReadOnly Property MessageConfirmationSent() As String
|
||||||
Get
|
Get
|
||||||
@@ -380,16 +308,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Lesebestätigung versendet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Löschinformation versendet ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property MessageConfirmationSentRaC() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("MessageConfirmationSentRaC", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Löschinformation versendet.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property MessageDeletionSent() As String
|
Public Shared ReadOnly Property MessageDeletionSent() As String
|
||||||
Get
|
Get
|
||||||
@@ -398,7 +317,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Dokumentenlink versendet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Dokumentenlink versendet ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property MessageInvitationSent() As String
|
Public Shared ReadOnly Property MessageInvitationSent() As String
|
||||||
Get
|
Get
|
||||||
@@ -407,7 +326,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Nein.
|
''' Sucht eine lokalisierte Zeichenfolge, die Nein ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property No() As String
|
Public Shared ReadOnly Property No() As String
|
||||||
Get
|
Get
|
||||||
@@ -416,16 +335,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Teil-Bestätigt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Teil-Signiert ähnelt.
|
||||||
'''</summary>
|
|
||||||
Public Shared ReadOnly Property PartlyConfirmed() As String
|
|
||||||
Get
|
|
||||||
Return ResourceManager.GetString("PartlyConfirmed", resourceCulture)
|
|
||||||
End Get
|
|
||||||
End Property
|
|
||||||
|
|
||||||
'''<summary>
|
|
||||||
''' Looks up a localized string similar to Teil-Signiert.
|
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property PartlySigned() As String
|
Public Shared ReadOnly Property PartlySigned() As String
|
||||||
Get
|
Get
|
||||||
@@ -434,7 +344,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Qualifizierte Signatur.
|
''' Sucht eine lokalisierte Zeichenfolge, die Qualifizierte Signatur ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property QualifiedSignature() As String
|
Public Shared ReadOnly Property QualifiedSignature() As String
|
||||||
Get
|
Get
|
||||||
@@ -443,7 +353,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Lesebestätigung.
|
''' Sucht eine lokalisierte Zeichenfolge, die Arbeitsanweisung ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property ReadAndSign() As String
|
Public Shared ReadOnly Property ReadAndSign() As String
|
||||||
Get
|
Get
|
||||||
@@ -452,7 +362,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren!.
|
''' Sucht eine lokalisierte Zeichenfolge, die Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren! ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property ResetTOTPUser() As String
|
Public Shared ReadOnly Property ResetTOTPUser() As String
|
||||||
Get
|
Get
|
||||||
@@ -461,7 +371,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Gespeichert.
|
''' Sucht eine lokalisierte Zeichenfolge, die Gespeichert ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Saved() As String
|
Public Shared ReadOnly Property Saved() As String
|
||||||
Get
|
Get
|
||||||
@@ -470,7 +380,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Gesendet.
|
''' Sucht eine lokalisierte Zeichenfolge, die Gesendet ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Sent() As String
|
Public Shared ReadOnly Property Sent() As String
|
||||||
Get
|
Get
|
||||||
@@ -479,7 +389,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Signatur.
|
''' Sucht eine lokalisierte Zeichenfolge, die Signatur ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Signature() As String
|
Public Shared ReadOnly Property Signature() As String
|
||||||
Get
|
Get
|
||||||
@@ -488,7 +398,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Abschluss bestätigt.
|
''' Sucht eine lokalisierte Zeichenfolge, die Signatur bestätigt ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property SignatureConfirmed() As String
|
Public Shared ReadOnly Property SignatureConfirmed() As String
|
||||||
Get
|
Get
|
||||||
@@ -497,7 +407,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Signiert.
|
''' Sucht eine lokalisierte Zeichenfolge, die Signiert ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Signed() As String
|
Public Shared ReadOnly Property Signed() As String
|
||||||
Get
|
Get
|
||||||
@@ -506,7 +416,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Erfolgreich! Dialog wird geschlossen..
|
''' Sucht eine lokalisierte Zeichenfolge, die Erfolgreich! Dialog wird geschlossen. ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Success_FormClose() As String
|
Public Shared ReadOnly Property Success_FormClose() As String
|
||||||
Get
|
Get
|
||||||
@@ -515,7 +425,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Unsigniert.
|
''' Sucht eine lokalisierte Zeichenfolge, die Unsigniert ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Unsigned() As String
|
Public Shared ReadOnly Property Unsigned() As String
|
||||||
Get
|
Get
|
||||||
@@ -524,7 +434,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Ja.
|
''' Sucht eine lokalisierte Zeichenfolge, die Ja ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property Yes() As String
|
Public Shared ReadOnly Property Yes() As String
|
||||||
Get
|
Get
|
||||||
@@ -533,7 +443,7 @@ Namespace My.Resources
|
|||||||
End Property
|
End Property
|
||||||
|
|
||||||
'''<summary>
|
'''<summary>
|
||||||
''' Looks up a localized string similar to Ja, mit Anhang.
|
''' Sucht eine lokalisierte Zeichenfolge, die Ja, mit Anhang ähnelt.
|
||||||
'''</summary>
|
'''</summary>
|
||||||
Public Shared ReadOnly Property YesWithAttachment() As String
|
Public Shared ReadOnly Property YesWithAttachment() As String
|
||||||
Get
|
Get
|
||||||
|
|||||||
@@ -132,27 +132,12 @@
|
|||||||
<data name="Completed" xml:space="preserve">
|
<data name="Completed" xml:space="preserve">
|
||||||
<value>Completed</value>
|
<value>Completed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Completely confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="CompletelySigned" xml:space="preserve">
|
|
||||||
<value>Completely signed</value>
|
|
||||||
</data>
|
|
||||||
<data name="Confirmation" xml:space="preserve">
|
|
||||||
<value>Read Confirmation</value>
|
|
||||||
</data>
|
|
||||||
<data name="Confirmed" xml:space="preserve">
|
|
||||||
<value>Read and signed</value>
|
|
||||||
</data>
|
|
||||||
<data name="Contract" xml:space="preserve">
|
<data name="Contract" xml:space="preserve">
|
||||||
<value>Contract</value>
|
<value>Contract</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Created" xml:space="preserve">
|
<data name="Created" xml:space="preserve">
|
||||||
<value>Created</value>
|
<value>Created</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentConfirmed" xml:space="preserve">
|
|
||||||
<value>Document read and confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentMod_Rotation" xml:space="preserve">
|
<data name="DocumentMod_Rotation" xml:space="preserve">
|
||||||
<value>Document rotation adapted</value>
|
<value>Document rotation adapted</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -162,9 +147,6 @@
|
|||||||
<data name="DocumentRejected" xml:space="preserve">
|
<data name="DocumentRejected" xml:space="preserve">
|
||||||
<value>Signing rejected</value>
|
<value>Signing rejected</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentRejectedRaC" xml:space="preserve">
|
|
||||||
<value>Read confirmation rejected</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentSigned" xml:space="preserve">
|
<data name="DocumentSigned" xml:space="preserve">
|
||||||
<value>Document signed</value>
|
<value>Document signed</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -174,9 +156,6 @@
|
|||||||
<data name="EnvelopeArchived" xml:space="preserve">
|
<data name="EnvelopeArchived" xml:space="preserve">
|
||||||
<value>Archived</value>
|
<value>Archived</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeCompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Completely confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
||||||
<value>Completely signed</value>
|
<value>Completely signed</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -186,11 +165,8 @@
|
|||||||
<data name="EnvelopeDeleted" xml:space="preserve">
|
<data name="EnvelopeDeleted" xml:space="preserve">
|
||||||
<value>Envelope Deleted</value>
|
<value>Envelope Deleted</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopePartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Partially confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopePartlySigned" xml:space="preserve">
|
<data name="EnvelopePartlySigned" xml:space="preserve">
|
||||||
<value>Partially signed</value>
|
<value>Partly signed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeQueued" xml:space="preserve">
|
<data name="EnvelopeQueued" xml:space="preserve">
|
||||||
<value>Envelope Queued</value>
|
<value>Envelope Queued</value>
|
||||||
@@ -201,9 +177,6 @@
|
|||||||
<data name="EnvelopeReportCreated" xml:space="preserve">
|
<data name="EnvelopeReportCreated" xml:space="preserve">
|
||||||
<value>Signature certificate created</value>
|
<value>Signature certificate created</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeReportCreatedRaC" xml:space="preserve">
|
|
||||||
<value>Read Confirmation Certificate Created</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeSaved" xml:space="preserve">
|
<data name="EnvelopeSaved" xml:space="preserve">
|
||||||
<value>Saved</value>
|
<value>Saved</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -222,9 +195,6 @@
|
|||||||
<data name="MessageConfirmationSent" xml:space="preserve">
|
<data name="MessageConfirmationSent" xml:space="preserve">
|
||||||
<value>Confirmation Sent</value>
|
<value>Confirmation Sent</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MessageConfirmationSentRaC" xml:space="preserve">
|
|
||||||
<value>Read Confirmation Sent</value>
|
|
||||||
</data>
|
|
||||||
<data name="MessageDeletionSent" xml:space="preserve">
|
<data name="MessageDeletionSent" xml:space="preserve">
|
||||||
<value>Deletion Notice Sent</value>
|
<value>Deletion Notice Sent</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -234,12 +204,6 @@
|
|||||||
<data name="No" xml:space="preserve">
|
<data name="No" xml:space="preserve">
|
||||||
<value>No</value>
|
<value>No</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Partially confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="PartlySigned" xml:space="preserve">
|
|
||||||
<value>Partially signed</value>
|
|
||||||
</data>
|
|
||||||
<data name="QualifiedSignature" xml:space="preserve">
|
<data name="QualifiedSignature" xml:space="preserve">
|
||||||
<value>Qualified Signature</value>
|
<value>Qualified Signature</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -259,7 +223,7 @@
|
|||||||
<value>Signature</value>
|
<value>Signature</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SignatureConfirmed" xml:space="preserve">
|
<data name="SignatureConfirmed" xml:space="preserve">
|
||||||
<value>Finalization confirmed</value>
|
<value>Signature confirmed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Signed" xml:space="preserve">
|
<data name="Signed" xml:space="preserve">
|
||||||
<value>Signed</value>
|
<value>Signed</value>
|
||||||
|
|||||||
@@ -132,17 +132,8 @@
|
|||||||
<data name="Completed" xml:space="preserve">
|
<data name="Completed" xml:space="preserve">
|
||||||
<value>Abgeschlossen</value>
|
<value>Abgeschlossen</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Vollständig bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="CompletelySigned" xml:space="preserve">
|
<data name="CompletelySigned" xml:space="preserve">
|
||||||
<value>Vollständig signiert</value>
|
<value>Vollständig Signiert</value>
|
||||||
</data>
|
|
||||||
<data name="Confirmation" xml:space="preserve">
|
|
||||||
<value>Lesebestätigung</value>
|
|
||||||
</data>
|
|
||||||
<data name="Confirmed" xml:space="preserve">
|
|
||||||
<value>Gelesen und Bestätigt</value>
|
|
||||||
</data>
|
</data>
|
||||||
<data name="Contract" xml:space="preserve">
|
<data name="Contract" xml:space="preserve">
|
||||||
<value>Vertrag</value>
|
<value>Vertrag</value>
|
||||||
@@ -150,9 +141,6 @@
|
|||||||
<data name="Created" xml:space="preserve">
|
<data name="Created" xml:space="preserve">
|
||||||
<value>Erstellt</value>
|
<value>Erstellt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentConfirmed" xml:space="preserve">
|
|
||||||
<value>Dokument gelesen und bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentMod_Rotation" xml:space="preserve">
|
<data name="DocumentMod_Rotation" xml:space="preserve">
|
||||||
<value>Dokument Rotation geändert</value>
|
<value>Dokument Rotation geändert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -162,9 +150,6 @@
|
|||||||
<data name="DocumentRejected" xml:space="preserve">
|
<data name="DocumentRejected" xml:space="preserve">
|
||||||
<value>Unterzeichnung abgelehnt</value>
|
<value>Unterzeichnung abgelehnt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentRejectedRaC" xml:space="preserve">
|
|
||||||
<value>Lesebestätigung abgelehnt</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentSigned" xml:space="preserve">
|
<data name="DocumentSigned" xml:space="preserve">
|
||||||
<value>Dokument unterzeichnet</value>
|
<value>Dokument unterzeichnet</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -174,9 +159,6 @@
|
|||||||
<data name="EnvelopeArchived" xml:space="preserve">
|
<data name="EnvelopeArchived" xml:space="preserve">
|
||||||
<value>Archiviert</value>
|
<value>Archiviert</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeCompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Vollständig bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
||||||
<value>Vollständig signiert</value>
|
<value>Vollständig signiert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -186,9 +168,6 @@
|
|||||||
<data name="EnvelopeDeleted" xml:space="preserve">
|
<data name="EnvelopeDeleted" xml:space="preserve">
|
||||||
<value>Umschlag Gelöscht</value>
|
<value>Umschlag Gelöscht</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopePartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Teil-Bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopePartlySigned" xml:space="preserve">
|
<data name="EnvelopePartlySigned" xml:space="preserve">
|
||||||
<value>Teil-Signiert</value>
|
<value>Teil-Signiert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -201,9 +180,6 @@
|
|||||||
<data name="EnvelopeReportCreated" xml:space="preserve">
|
<data name="EnvelopeReportCreated" xml:space="preserve">
|
||||||
<value>Signierungszertifikat erstellt</value>
|
<value>Signierungszertifikat erstellt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeReportCreatedRaC" xml:space="preserve">
|
|
||||||
<value>Lesebestätigungszertifikat erstellt</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeSaved" xml:space="preserve">
|
<data name="EnvelopeSaved" xml:space="preserve">
|
||||||
<value>Gespeichert</value>
|
<value>Gespeichert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -222,9 +198,6 @@
|
|||||||
<data name="MessageConfirmationSent" xml:space="preserve">
|
<data name="MessageConfirmationSent" xml:space="preserve">
|
||||||
<value>Signaturbestätigung versendet</value>
|
<value>Signaturbestätigung versendet</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MessageConfirmationSentRaC" xml:space="preserve">
|
|
||||||
<value>Lesebestätigung versendet</value>
|
|
||||||
</data>
|
|
||||||
<data name="MessageDeletionSent" xml:space="preserve">
|
<data name="MessageDeletionSent" xml:space="preserve">
|
||||||
<value>Löschinformation versendet</value>
|
<value>Löschinformation versendet</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -234,9 +207,6 @@
|
|||||||
<data name="No" xml:space="preserve">
|
<data name="No" xml:space="preserve">
|
||||||
<value>Nein</value>
|
<value>Nein</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Teil-Bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="PartlySigned" xml:space="preserve">
|
<data name="PartlySigned" xml:space="preserve">
|
||||||
<value>Teil-Signiert</value>
|
<value>Teil-Signiert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -244,7 +214,7 @@
|
|||||||
<value>Qualifizierte Signatur</value>
|
<value>Qualifizierte Signatur</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ReadAndSign" xml:space="preserve">
|
<data name="ReadAndSign" xml:space="preserve">
|
||||||
<value>Lesebestätigung</value>
|
<value>Arbeitsanweisung</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ResetTOTPUser" xml:space="preserve">
|
<data name="ResetTOTPUser" xml:space="preserve">
|
||||||
<value>Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren!</value>
|
<value>Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren!</value>
|
||||||
@@ -259,7 +229,7 @@
|
|||||||
<value>Signatur</value>
|
<value>Signatur</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SignatureConfirmed" xml:space="preserve">
|
<data name="SignatureConfirmed" xml:space="preserve">
|
||||||
<value>Abschluss bestätigt</value>
|
<value>Signatur bestätigt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Signed" xml:space="preserve">
|
<data name="Signed" xml:space="preserve">
|
||||||
<value>Signiert</value>
|
<value>Signiert</value>
|
||||||
|
|||||||
@@ -1,145 +0,0 @@
|
|||||||
using Microsoft.Extensions.Configuration;
|
|
||||||
using Microsoft.Extensions.DependencyInjection;
|
|
||||||
using EnvelopeGenerator.Application;
|
|
||||||
using EnvelopeGenerator.Application.Common.Interfaces.Services;
|
|
||||||
using EnvelopeGenerator.Application.Services;
|
|
||||||
using EnvelopeGenerator.Infrastructure;
|
|
||||||
using DigitalData.EmailProfilerDispatcher;
|
|
||||||
using DigitalData.UserManager.DependencyInjection;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.DependencyInjection;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Controls which optional services are registered by <see cref="DependencyInjection.AddEnvelopeGenerator"/>.
|
|
||||||
/// All flags default to <c>true</c>. Set a flag to <c>false</c> if the consuming project
|
|
||||||
/// already registers that service itself or simply does not need it.
|
|
||||||
/// </summary>
|
|
||||||
public sealed class EnvelopeGeneratorOptions
|
|
||||||
{
|
|
||||||
/// <summary>Calls <c>AddHttpContextAccessor()</c>. Default: <c>true</c>.</summary>
|
|
||||||
public bool AddHttpContextAccessor { get; set; } = true;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Calls <c>AddDistributedSqlServerCache()</c> with the supplied <see cref="SqlCacheOptions"/>.
|
|
||||||
/// Requires <see cref="SqlCacheOptions"/> to be configured. Default: <c>true</c>.
|
|
||||||
/// </summary>
|
|
||||||
public bool AddDistributedSqlServerCache { get; set; } = true;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Options for the distributed SQL Server cache.
|
|
||||||
/// Required when <see cref="AddDistributedSqlServerCache"/> is <c>true</c>.
|
|
||||||
/// </summary>
|
|
||||||
public SqlCacheOptions? SqlCacheOptions { get; set; }
|
|
||||||
|
|
||||||
/// <summary>Calls <c>AddDispatcher<TDbContext>()</c>. Default: <c>true</c>.</summary>
|
|
||||||
public bool AddDispatcher { get; set; } = true;
|
|
||||||
|
|
||||||
/// <summary>Calls <c>AddMemoryCache()</c>. Default: <c>true</c>.</summary>
|
|
||||||
public bool AddMemoryCache { get; set; } = true;
|
|
||||||
|
|
||||||
/// <summary>Calls <c>AddUserManager<TDbContext>()</c>. Default: <c>true</c>.</summary>
|
|
||||||
public bool AddUserManager { get; set; } = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>Options for <c>AddDistributedSqlServerCache</c>.</summary>
|
|
||||||
public sealed class SqlCacheOptions
|
|
||||||
{
|
|
||||||
/// <summary>SQL Server connection string.</summary>
|
|
||||||
public string ConnectionString { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
/// <summary>Schema name. Default: <c>dbo</c>.</summary>
|
|
||||||
public string SchemaName { get; set; } = "dbo";
|
|
||||||
|
|
||||||
/// <summary>Table name. Default: <c>TBDD_CACHE</c>.</summary>
|
|
||||||
public string TableName { get; set; } = "TBDD_CACHE";
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Extension methods for registering EnvelopeGenerator services into an <see cref="IServiceCollection"/>.
|
|
||||||
/// Use <see cref="AddEnvelopeGenerator{TDbContext}"/> as the single entry-point for projects that need both the
|
|
||||||
/// application layer (MediatR, AutoMapper, CRUD services, configuration sections) and the infrastructure
|
|
||||||
/// layer (repositories, DbContext, SQL executors).
|
|
||||||
/// For projects that do not need a database (e.g. lightweight API gateways or unit-test hosts), use
|
|
||||||
/// <see cref="AddEnvelopeGeneratorCore"/> to register only the application layer.
|
|
||||||
/// </summary>
|
|
||||||
public static class DependencyInjection
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Registers the full EnvelopeGenerator stack using <see cref="EGDbContext"/> as the DbContext type.
|
|
||||||
/// </summary>
|
|
||||||
public static IServiceCollection AddEnvelopeGenerator(
|
|
||||||
this IServiceCollection services,
|
|
||||||
IConfiguration configuration,
|
|
||||||
Action<EnvelopeGenerator.Infrastructure.DependencyInjection.Config>? infrastructureOptions = null,
|
|
||||||
Action<EnvelopeGeneratorOptions>? options = null)
|
|
||||||
{
|
|
||||||
var opt = new EnvelopeGeneratorOptions();
|
|
||||||
options?.Invoke(opt);
|
|
||||||
|
|
||||||
#pragma warning disable CS0618
|
|
||||||
// Application layer: CRUD services, MediatR, AutoMapper, configuration sections.
|
|
||||||
services.AddEnvelopeGeneratorServices(configuration);
|
|
||||||
|
|
||||||
// Infrastructure layer: repositories, DbContext, Dapper type maps, SQL executors.
|
|
||||||
services.AddEnvelopeGeneratorInfrastructureServices(cfg =>
|
|
||||||
{
|
|
||||||
infrastructureOptions?.Invoke(cfg);
|
|
||||||
});
|
|
||||||
#pragma warning restore CS0618
|
|
||||||
|
|
||||||
if (opt.AddHttpContextAccessor)
|
|
||||||
services.AddHttpContextAccessor();
|
|
||||||
|
|
||||||
if (opt.AddDistributedSqlServerCache && opt.SqlCacheOptions is { } cacheOpts)
|
|
||||||
services.AddDistributedSqlServerCache(o =>
|
|
||||||
{
|
|
||||||
o.ConnectionString = cacheOpts.ConnectionString;
|
|
||||||
o.SchemaName = cacheOpts.SchemaName;
|
|
||||||
o.TableName = cacheOpts.TableName;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (opt.AddDispatcher)
|
|
||||||
services.AddDispatcher<EGDbContext>();
|
|
||||||
|
|
||||||
if (opt.AddMemoryCache)
|
|
||||||
services.AddMemoryCache();
|
|
||||||
|
|
||||||
#pragma warning disable CS0618
|
|
||||||
if (opt.AddUserManager)
|
|
||||||
services.AddUserManager<EGDbContext>();
|
|
||||||
#pragma warning restore CS0618
|
|
||||||
|
|
||||||
return services;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Registers only the <em>application</em> layer services (MediatR handlers, AutoMapper profiles,
|
|
||||||
/// CRUD services, configuration sections) without any infrastructure / database dependencies.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="services">Service collection to register services into.</param>
|
|
||||||
/// <param name="configuration">Application configuration used to bind application-level option sections.</param>
|
|
||||||
/// <returns>The updated <see cref="IServiceCollection"/>.</returns>
|
|
||||||
#pragma warning disable CS0618
|
|
||||||
public static IServiceCollection AddEnvelopeGeneratorCore(
|
|
||||||
this IServiceCollection services,
|
|
||||||
IConfiguration configuration)
|
|
||||||
{
|
|
||||||
services.AddEnvelopeGeneratorServices(configuration);
|
|
||||||
return services;
|
|
||||||
}
|
|
||||||
#pragma warning restore CS0618
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Registers <see cref="EnvelopeMailService"/> as the <see cref="IEnvelopeMailService"/> scoped
|
|
||||||
/// implementation.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="services">Service collection to register services into.</param>
|
|
||||||
/// <returns>The updated <see cref="IServiceCollection"/>.</returns>
|
|
||||||
#pragma warning disable CS0618
|
|
||||||
public static IServiceCollection AddEnvelopeMailService(this IServiceCollection services)
|
|
||||||
{
|
|
||||||
services.AddScoped<IEnvelopeMailService, EnvelopeMailService>();
|
|
||||||
return services;
|
|
||||||
}
|
|
||||||
#pragma warning restore CS0618
|
|
||||||
}
|
|
||||||
@@ -1,176 +0,0 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
|
||||||
|
|
||||||
<PropertyGroup>
|
|
||||||
<TargetFrameworks>net7.0;net8.0;net9.0</TargetFrameworks>
|
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
|
||||||
<Nullable>enable</Nullable>
|
|
||||||
|
|
||||||
<!-- NuGet package metadata -->
|
|
||||||
<PackageId>EnvelopeGenerator</PackageId>
|
|
||||||
<Authors>Digital Data GmbH</Authors>
|
|
||||||
<Company>Digital Data GmbH</Company>
|
|
||||||
<Product>EnvelopeGenerator</Product>
|
|
||||||
<Description>
|
|
||||||
Envelope Generator ist eine Bibliothek zur Verwaltung und Verarbeitung digitaler Umschläge (Envelopes).
|
|
||||||
Dieses Paket enthält die Dependency-Injection-Erweiterungsmethoden und bündelt die Application-
|
|
||||||
sowie Infrastructure-Schicht in einer einzigen NuGet-Referenz.
|
|
||||||
</Description>
|
|
||||||
<Copyright>Copyright 2024 Digital Data GmbH</Copyright>
|
|
||||||
<RepositoryUrl>http://git.dd:3000/AppStd/EnvelopeGenerator.git</RepositoryUrl>
|
|
||||||
<PackageTags>digital data envelope generator di dependency injection</PackageTags>
|
|
||||||
<PackageRequireLicenseAcceptance>false</PackageRequireLicenseAcceptance>
|
|
||||||
<Version>1.2.0.3</Version>
|
|
||||||
<AssemblyVersion>1.2.0.3</AssemblyVersion>
|
|
||||||
<FileVersion>1.2.0.3</FileVersion>
|
|
||||||
<GeneratePackageOnBuild>false</GeneratePackageOnBuild>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<!-- ASP.NET Core shared framework (AddHttpContextAccessor, AddMemoryCache, etc.) -->
|
|
||||||
<ItemGroup>
|
|
||||||
<FrameworkReference Include="Microsoft.AspNetCore.App" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<!--
|
|
||||||
All dependencies are declared here.
|
|
||||||
Because Application and Infrastructure DLLs are bundled via PrivateAssets=all,
|
|
||||||
their PackageReferences have been moved to this project so that the correct
|
|
||||||
dependencies are written to the nuspec and a consuming project works with
|
|
||||||
a single package install.
|
|
||||||
-->
|
|
||||||
<ItemGroup>
|
|
||||||
<!-- DigitalData BaGet packages -->
|
|
||||||
<PackageReference Include="DigitalData.Core.Abstraction.Application" Version="1.6.0" />
|
|
||||||
<PackageReference Include="DigitalData.Core.Application" Version="3.4.0" />
|
|
||||||
<PackageReference Include="DigitalData.Core.Client" Version="2.1.0" />
|
|
||||||
<PackageReference Include="DigitalData.Core.Exceptions" Version="1.1.0" />
|
|
||||||
<PackageReference Include="DigitalData.Core.Infrastructure" Version="2.6.1" />
|
|
||||||
<PackageReference Include="DigitalData.EmailProfilerDispatcher" Version="3.1.1" />
|
|
||||||
<PackageReference Include="UserManager" Version="1.1.3" />
|
|
||||||
|
|
||||||
<!-- ORM / Database -->
|
|
||||||
<PackageReference Include="Dapper" Version="2.1.66" />
|
|
||||||
<PackageReference Include="Microsoft.Data.SqlClient" Version="5.2.2" />
|
|
||||||
|
|
||||||
<!-- Application services -->
|
|
||||||
<PackageReference Include="MediatR" Version="12.5.0" />
|
|
||||||
<PackageReference Include="QRCoder" Version="1.6.0" />
|
|
||||||
<PackageReference Include="QRCoder-ImageSharp" Version="0.10.0" />
|
|
||||||
<PackageReference Include="QuestPDF" Version="2025.7.1" />
|
|
||||||
<PackageReference Include="Otp.NET" Version="1.4.0" />
|
|
||||||
|
|
||||||
<!-- Security / Identity -->
|
|
||||||
<PackageReference Include="Microsoft.Identity.Client" Version="4.82.1" />
|
|
||||||
|
|
||||||
<!-- Utilities -->
|
|
||||||
<PackageReference Include="HtmlSanitizer" Version="9.0.892" />
|
|
||||||
<PackageReference Include="SixLabors.ImageSharp" Version="3.1.12" />
|
|
||||||
<PackageReference Include="System.Formats.Asn1" Version="10.0.3" />
|
|
||||||
<PackageReference Include="System.Security.AccessControl" Version="6.0.1" />
|
|
||||||
|
|
||||||
<!-- DI abstractions -->
|
|
||||||
<PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" Version="9.0.6" />
|
|
||||||
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="9.0.6" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<!-- TFM-specific packages -->
|
|
||||||
<ItemGroup Condition="'$(TargetFramework)' == 'net7.0'">
|
|
||||||
<PackageReference Include="AutoMapper" Version="13.0.1" />
|
|
||||||
<PackageReference Include="CommandDotNet" Version="7.0.5" />
|
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="7.0.20" />
|
|
||||||
<PackageReference Include="Microsoft.Extensions.Caching.SqlServer" Version="7.0.20" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="7.0.20" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Relational" Version="7.0.20" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="7.0.20" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup Condition="'$(TargetFramework)' == 'net8.0'">
|
|
||||||
<PackageReference Include="AutoMapper" Version="14.0.0" />
|
|
||||||
<PackageReference Include="CommandDotNet" Version="8.1.1" />
|
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="8.0.17" />
|
|
||||||
<PackageReference Include="Microsoft.Extensions.Caching.SqlServer" Version="8.0.17" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="8.0.17" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Relational" Version="8.0.17" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="8.0.17" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup Condition="'$(TargetFramework)' == 'net9.0'">
|
|
||||||
<PackageReference Include="AutoMapper" Version="14.0.0" />
|
|
||||||
<PackageReference Include="CommandDotNet" Version="8.1.1" />
|
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="9.0.6" />
|
|
||||||
<PackageReference Include="Microsoft.Extensions.Caching.SqlServer" Version="9.0.6" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="9.0.6" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.Relational" Version="9.0.6" />
|
|
||||||
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="9.0.6" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<ProjectReference Include="..\EnvelopeGenerator.Application\EnvelopeGenerator.Application.csproj">
|
|
||||||
<PrivateAssets>all</PrivateAssets>
|
|
||||||
</ProjectReference>
|
|
||||||
<ProjectReference Include="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj">
|
|
||||||
<PrivateAssets>all</PrivateAssets>
|
|
||||||
</ProjectReference>
|
|
||||||
<ProjectReference Include="..\EnvelopeGenerator.Infrastructure\EnvelopeGenerator.Infrastructure.csproj">
|
|
||||||
<PrivateAssets>all</PrivateAssets>
|
|
||||||
</ProjectReference>
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<!-- Bundle dependency DLLs into the package lib folder -->
|
|
||||||
<Target Name="IncludeDependencyDlls" BeforeTargets="_GetPackageFiles">
|
|
||||||
<ItemGroup>
|
|
||||||
<_DepDlls Include="
 ..\EnvelopeGenerator.Application\bin\$(Configuration)\%(ProjectReference.TargetFramework)\EnvelopeGenerator.Application.dll;
 ..\EnvelopeGenerator.Domain\bin\$(Configuration)\%(ProjectReference.TargetFramework)\EnvelopeGenerator.Domain.dll;
 ..\EnvelopeGenerator.Infrastructure\bin\$(Configuration)\%(ProjectReference.TargetFramework)\EnvelopeGenerator.Infrastructure.dll" />
|
|
||||||
</ItemGroup>
|
|
||||||
</Target>
|
|
||||||
|
|
||||||
<!--
|
|
||||||
Rebuild all dependency projects for every TFM before packing so that
|
|
||||||
the DLLs bundled into the package are always up-to-date.
|
|
||||||
Run with: dotnet pack -c Release
|
|
||||||
-->
|
|
||||||
<Target Name="RebuildDependenciesBeforePack" BeforeTargets="GenerateNuspec">
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Application\EnvelopeGenerator.Application.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net7.0" />
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Application\EnvelopeGenerator.Application.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net8.0" />
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Application\EnvelopeGenerator.Application.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net9.0" />
|
|
||||||
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net7.0" />
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net8.0" />
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Domain\EnvelopeGenerator.Domain.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net9.0" />
|
|
||||||
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Infrastructure\EnvelopeGenerator.Infrastructure.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net7.0" />
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Infrastructure\EnvelopeGenerator.Infrastructure.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net8.0" />
|
|
||||||
<MSBuild Projects="..\EnvelopeGenerator.Infrastructure\EnvelopeGenerator.Infrastructure.csproj"
|
|
||||||
Targets="Build"
|
|
||||||
Properties="Configuration=$(Configuration);TargetFramework=net9.0" />
|
|
||||||
</Target>
|
|
||||||
|
|
||||||
<Target Name="BundleReferencedDlls" AfterTargets="Build">
|
|
||||||
<ItemGroup>
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Application\bin\$(Configuration)\net7.0\EnvelopeGenerator.Application.dll" TargetFramework="net7.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Application\bin\$(Configuration)\net8.0\EnvelopeGenerator.Application.dll" TargetFramework="net8.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Application\bin\$(Configuration)\net9.0\EnvelopeGenerator.Application.dll" TargetFramework="net9.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Domain\bin\$(Configuration)\net7.0\EnvelopeGenerator.Domain.dll" TargetFramework="net7.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Domain\bin\$(Configuration)\net8.0\EnvelopeGenerator.Domain.dll" TargetFramework="net8.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Domain\bin\$(Configuration)\net9.0\EnvelopeGenerator.Domain.dll" TargetFramework="net9.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Infrastructure\bin\$(Configuration)\net7.0\EnvelopeGenerator.Infrastructure.dll" TargetFramework="net7.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Infrastructure\bin\$(Configuration)\net8.0\EnvelopeGenerator.Infrastructure.dll" TargetFramework="net8.0" />
|
|
||||||
<BuildOutputInPackage Include="..\EnvelopeGenerator.Infrastructure\bin\$(Configuration)\net9.0\EnvelopeGenerator.Infrastructure.dll" TargetFramework="net9.0" />
|
|
||||||
</ItemGroup>
|
|
||||||
</Target>
|
|
||||||
|
|
||||||
</Project>
|
|
||||||
@@ -1,11 +1,17 @@
|
|||||||
using System.ComponentModel.DataAnnotations;
|
using System.ComponentModel.DataAnnotations;
|
||||||
using System.ComponentModel.DataAnnotations.Schema;
|
using System.ComponentModel.DataAnnotations.Schema;
|
||||||
|
#if NETFRAMEWORK
|
||||||
|
using System;
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace EnvelopeGenerator.Domain.Entities
|
namespace EnvelopeGenerator.Domain.Entities
|
||||||
{
|
{
|
||||||
[Table("TBSIG_CONFIG", Schema = "dbo")]
|
[Table("TBSIG_CONFIG", Schema = "dbo")]
|
||||||
public class Config
|
public class Config
|
||||||
{
|
{
|
||||||
|
[Column("DOCUMENT_PATH", TypeName = "nvarchar(256)")]
|
||||||
|
public string DocumentPath { get; set; }
|
||||||
|
|
||||||
[Column("SENDING_PROFILE", TypeName = "int")]
|
[Column("SENDING_PROFILE", TypeName = "int")]
|
||||||
[Required]
|
[Required]
|
||||||
public int SendingProfile { get; set; }
|
public int SendingProfile { get; set; }
|
||||||
@@ -19,5 +25,24 @@ namespace EnvelopeGenerator.Domain.Entities
|
|||||||
|
|
||||||
[Column("EXPORT_PATH", TypeName = "nvarchar(256)")]
|
[Column("EXPORT_PATH", TypeName = "nvarchar(256)")]
|
||||||
public string ExportPath { get; set; }
|
public string ExportPath { get; set; }
|
||||||
|
|
||||||
|
[Column("ADDED_WHEN", TypeName = "datetime")]
|
||||||
|
[Required]
|
||||||
|
public DateTime AddedWhen { get; set; }
|
||||||
|
|
||||||
|
[Column("CHANGED_WHEN", TypeName = "datetime")]
|
||||||
|
public DateTime? ChangedWhen { get; set; }
|
||||||
|
|
||||||
|
[Column("GUID", TypeName = "tinyint")]
|
||||||
|
[Required]
|
||||||
|
public byte Guid { get; set; }
|
||||||
|
|
||||||
|
[Column("DEF_TFA_ENABLED", TypeName = "bit")]
|
||||||
|
[Required]
|
||||||
|
public bool DefTfaEnabled { get; set; }
|
||||||
|
|
||||||
|
[Column("DEF_TFA_WITH_PHONE", TypeName = "bit")]
|
||||||
|
[Required]
|
||||||
|
public bool DefTfaWithPhone { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -35,6 +35,7 @@ namespace EnvelopeGenerator.Domain.Entities
|
|||||||
[Column("STATUS")]
|
[Column("STATUS")]
|
||||||
public Constants.DocumentStatus Status { get; set; }
|
public Constants.DocumentStatus Status { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
[Column("STATUS_CHANGED_WHEN", TypeName = "datetime")]
|
[Column("STATUS_CHANGED_WHEN", TypeName = "datetime")]
|
||||||
public DateTime? StatusChangedWhen { get; set; }
|
public DateTime? StatusChangedWhen { get; set; }
|
||||||
|
|
||||||
|
|||||||
@@ -76,11 +76,14 @@ namespace EnvelopeGenerator.Domain.Entities
|
|||||||
#if nullable
|
#if nullable
|
||||||
?
|
?
|
||||||
#endif
|
#endif
|
||||||
Title
|
Title { get; set; }
|
||||||
{ get; set; }
|
|
||||||
|
|
||||||
[Column("COMMENT", TypeName = "nvarchar(128)")]
|
[Column("COMMENT", TypeName = "nvarchar(128)")]
|
||||||
public string Comment { get; set; }
|
public string
|
||||||
|
#if nullable
|
||||||
|
?
|
||||||
|
#endif
|
||||||
|
Comment { get; set; }
|
||||||
|
|
||||||
[Column("CONTRACT_TYPE")]
|
[Column("CONTRACT_TYPE")]
|
||||||
public int? ContractType { get; set; }
|
public int? ContractType { get; set; }
|
||||||
@@ -137,7 +140,9 @@ namespace EnvelopeGenerator.Domain.Entities
|
|||||||
= false;
|
= false;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if NETFRAMEWORK
|
||||||
[NotMapped]
|
[NotMapped]
|
||||||
|
#endif
|
||||||
[Column("DOC_RESULT")]
|
[Column("DOC_RESULT")]
|
||||||
public byte[]
|
public byte[]
|
||||||
#if nullable
|
#if nullable
|
||||||
@@ -151,8 +156,7 @@ namespace EnvelopeGenerator.Domain.Entities
|
|||||||
#if nullable
|
#if nullable
|
||||||
?
|
?
|
||||||
#endif
|
#endif
|
||||||
Type
|
Type { get; set; }
|
||||||
{ get; set; }
|
|
||||||
|
|
||||||
#if NETFRAMEWORK
|
#if NETFRAMEWORK
|
||||||
[NotMapped]
|
[NotMapped]
|
||||||
@@ -166,22 +170,26 @@ namespace EnvelopeGenerator.Domain.Entities
|
|||||||
#if nullable
|
#if nullable
|
||||||
?
|
?
|
||||||
#endif
|
#endif
|
||||||
Documents
|
Documents { get; set; }
|
||||||
{ get; set; }
|
|
||||||
|
[NotMapped]
|
||||||
|
public Document
|
||||||
|
#if nullable
|
||||||
|
?
|
||||||
|
#endif
|
||||||
|
DefaultDocument => Documents?.FirstOrDefault();
|
||||||
|
|
||||||
public List<History>
|
public List<History>
|
||||||
#if nullable
|
#if nullable
|
||||||
?
|
?
|
||||||
#endif
|
#endif
|
||||||
Histories
|
Histories { get; set; }
|
||||||
{ get; set; }
|
|
||||||
|
|
||||||
public List<EnvelopeReceiver>
|
public List<EnvelopeReceiver>
|
||||||
#if nullable
|
#if nullable
|
||||||
?
|
?
|
||||||
#endif
|
#endif
|
||||||
EnvelopeReceivers
|
EnvelopeReceivers { get; set; }
|
||||||
{ get; set; }
|
|
||||||
|
|
||||||
//#if NETFRAMEWORK
|
//#if NETFRAMEWORK
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
36
EnvelopeGenerator.Domain/Entities/EnvelopeReport.cs
Normal file
36
EnvelopeGenerator.Domain/Entities/EnvelopeReport.cs
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
using System;
|
||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using System.ComponentModel.DataAnnotations.Schema;
|
||||||
|
|
||||||
|
namespace EnvelopeGenerator.Domain.Entities
|
||||||
|
{
|
||||||
|
[Table("VWSIG_ENVELOPE_REPORT", Schema = "dbo")]
|
||||||
|
public class EnvelopeReport
|
||||||
|
{
|
||||||
|
[Column("ENVELOPE_ID", TypeName = "int")]
|
||||||
|
[Required]
|
||||||
|
public int EnvelopeId { get; set; }
|
||||||
|
|
||||||
|
[Column("HEAD_UUID", TypeName = "nvarchar(36)")]
|
||||||
|
[Required]
|
||||||
|
public string HeadUuid { get; set; }
|
||||||
|
|
||||||
|
[Column("HEAD_TITLE", TypeName = "nvarchar(128)")]
|
||||||
|
public string HeadTitle { get; set; }
|
||||||
|
|
||||||
|
[Column("HEAD_MESSAGE", TypeName = "nvarchar(max)")]
|
||||||
|
[Required]
|
||||||
|
public string HeadMessage { get; set; }
|
||||||
|
|
||||||
|
[Column("POS_STATUS", TypeName = "int")]
|
||||||
|
[Required]
|
||||||
|
public int PosStatus { get; set; }
|
||||||
|
|
||||||
|
[Column("POS_WHEN", TypeName = "datetime")]
|
||||||
|
public DateTime? PosWhen { get; set; }
|
||||||
|
|
||||||
|
[Column("POS_WHO", TypeName = "nvarchar(128)")]
|
||||||
|
[Required]
|
||||||
|
public string PosWho { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,6 +4,9 @@ using System.ComponentModel;
|
|||||||
using System.ComponentModel.DataAnnotations;
|
using System.ComponentModel.DataAnnotations;
|
||||||
using System.ComponentModel.DataAnnotations.Schema;
|
using System.ComponentModel.DataAnnotations.Schema;
|
||||||
using EnvelopeGenerator.Domain.Interfaces.Auditing;
|
using EnvelopeGenerator.Domain.Interfaces.Auditing;
|
||||||
|
#if NET
|
||||||
|
using System.Text.Json.Serialization;
|
||||||
|
#endif
|
||||||
#if NETFRAMEWORK
|
#if NETFRAMEWORK
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
#endif
|
#endif
|
||||||
@@ -109,22 +112,25 @@ namespace EnvelopeGenerator.Domain.Entities
|
|||||||
#if nullable
|
#if nullable
|
||||||
?
|
?
|
||||||
#endif
|
#endif
|
||||||
Receiver
|
Receiver { get; set; }
|
||||||
{ get; set; }
|
|
||||||
|
|
||||||
public virtual IEnumerable<ElementAnnotation>
|
public virtual IEnumerable<ElementAnnotation>
|
||||||
#if nullable
|
#if nullable
|
||||||
?
|
?
|
||||||
#endif
|
#endif
|
||||||
Annotations
|
Annotations { get; set; }
|
||||||
{ get; set; }
|
|
||||||
|
|
||||||
#if NETFRAMEWORK
|
#if NET
|
||||||
|
[JsonIgnore]
|
||||||
|
#endif
|
||||||
[NotMapped]
|
[NotMapped]
|
||||||
public double Top => Math.Round(Y, 5);
|
public double Top => Math.Round(Y, 5);
|
||||||
|
|
||||||
|
|
||||||
|
#if NET
|
||||||
|
[JsonIgnore]
|
||||||
|
#endif
|
||||||
[NotMapped]
|
[NotMapped]
|
||||||
public double Left => Math.Round(X, 5);
|
public double Left => Math.Round(X, 5);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
61
EnvelopeGenerator.Domain/Entities/ThirdPartyModule.cs
Normal file
61
EnvelopeGenerator.Domain/Entities/ThirdPartyModule.cs
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using System.ComponentModel.DataAnnotations.Schema;
|
||||||
|
using EnvelopeGenerator.Domain.Interfaces.Auditing;
|
||||||
|
#if NETFRAMEWORK
|
||||||
|
using System;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace EnvelopeGenerator.Domain.Entities
|
||||||
|
{
|
||||||
|
[Table("TBDD_3RD_PARTY_MODULES", Schema = "dbo")]
|
||||||
|
public class ThirdPartyModule : IHasChangedWhen, IHasChangedWho
|
||||||
|
{
|
||||||
|
[Key]
|
||||||
|
[DatabaseGenerated(DatabaseGeneratedOption.Identity)]
|
||||||
|
[Column("GUID")]
|
||||||
|
public int Id { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[Column("ACTIVE", TypeName = "bit")]
|
||||||
|
public bool Active { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[Column("NAME", TypeName = "varchar(50)")]
|
||||||
|
public string Name { get; set; }
|
||||||
|
|
||||||
|
[Column("DESCRIPTION", TypeName = "varchar(500)")]
|
||||||
|
public string
|
||||||
|
#if nullable
|
||||||
|
?
|
||||||
|
#endif
|
||||||
|
Description { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[Column("LICENSE", TypeName = "varchar(max)")]
|
||||||
|
public string License { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[Column("VERSION", TypeName = "varchar(20)")]
|
||||||
|
public string Version { get; set; }
|
||||||
|
|
||||||
|
[Column("ADDED_WHO", TypeName = "varchar(50)")]
|
||||||
|
public string
|
||||||
|
#if nullable
|
||||||
|
?
|
||||||
|
#endif
|
||||||
|
AddedWho { get; set; }
|
||||||
|
|
||||||
|
[Column("ADDED_WHEN", TypeName = "datetime")]
|
||||||
|
public DateTime? AddedWhen { get; set; }
|
||||||
|
|
||||||
|
[Column("CHANGED_WHO", TypeName = "varchar(50)")]
|
||||||
|
public string
|
||||||
|
#if nullable
|
||||||
|
?
|
||||||
|
#endif
|
||||||
|
ChangedWho { get; set; }
|
||||||
|
|
||||||
|
[Column("CHANGED_WHEN", TypeName = "datetime")]
|
||||||
|
public DateTime? ChangedWhen { get; set; }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -11,10 +11,5 @@
|
|||||||
{
|
{
|
||||||
return envelope.EnvelopeTypeId == 2;
|
return envelope.EnvelopeTypeId == 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool IsReadAndSign(this IEnvelope envelope)
|
|
||||||
{
|
|
||||||
return envelope.EnvelopeTypeId != 2;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
200
EnvelopeGenerator.Domain/Resources/Model.Designer.cs
generated
200
EnvelopeGenerator.Domain/Resources/Model.Designer.cs
generated
@@ -1,10 +1,10 @@
|
|||||||
//------------------------------------------------------------------------------
|
//------------------------------------------------------------------------------
|
||||||
// <auto-generated>
|
// <auto-generated>
|
||||||
// This code was generated by a tool.
|
// Dieser Code wurde von einem Tool generiert.
|
||||||
// Runtime Version:4.0.30319.42000
|
// Laufzeitversion:4.0.30319.42000
|
||||||
//
|
//
|
||||||
// Changes to this file may cause incorrect behavior and will be lost if
|
// Änderungen an dieser Datei können falsches Verhalten verursachen und gehen verloren, wenn
|
||||||
// the code is regenerated.
|
// der Code erneut generiert wird.
|
||||||
// </auto-generated>
|
// </auto-generated>
|
||||||
//------------------------------------------------------------------------------
|
//------------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -13,12 +13,12 @@ namespace My.Resources {
|
|||||||
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// A strongly-typed resource class, for looking up localized strings, etc.
|
/// Eine stark typisierte Ressourcenklasse zum Suchen von lokalisierten Zeichenfolgen usw.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
// This class was auto-generated by the StronglyTypedResourceBuilder
|
// Diese Klasse wurde von der StronglyTypedResourceBuilder automatisch generiert
|
||||||
// class via a tool like ResGen or Visual Studio.
|
// -Klasse über ein Tool wie ResGen oder Visual Studio automatisch generiert.
|
||||||
// To add or remove a member, edit your .ResX file then rerun ResGen
|
// Um einen Member hinzuzufügen oder zu entfernen, bearbeiten Sie die .ResX-Datei und führen dann ResGen
|
||||||
// with the /str option, or rebuild your VS project.
|
// mit der /str-Option erneut aus, oder Sie erstellen Ihr VS-Projekt neu.
|
||||||
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0")]
|
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0")]
|
||||||
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
|
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
|
||||||
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
|
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
|
||||||
@@ -33,7 +33,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Returns the cached ResourceManager instance used by this class.
|
/// Gibt die zwischengespeicherte ResourceManager-Instanz zurück, die von dieser Klasse verwendet wird.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
|
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
|
||||||
public static global::System.Resources.ResourceManager ResourceManager {
|
public static global::System.Resources.ResourceManager ResourceManager {
|
||||||
@@ -47,8 +47,8 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Overrides the current thread's CurrentUICulture property for all
|
/// Überschreibt die CurrentUICulture-Eigenschaft des aktuellen Threads für alle
|
||||||
/// resource lookups using this strongly typed resource class.
|
/// Ressourcenzuordnungen, die diese stark typisierte Ressourcenklasse verwenden.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
|
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
|
||||||
public static global::System.Globalization.CultureInfo Culture {
|
public static global::System.Globalization.CultureInfo Culture {
|
||||||
@@ -61,7 +61,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Zugriffscode korrekt eingegeben.
|
/// Sucht eine lokalisierte Zeichenfolge, die Zugriffscode korrekt eingegeben ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string AccessCodeCorrect {
|
public static string AccessCodeCorrect {
|
||||||
get {
|
get {
|
||||||
@@ -70,7 +70,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Zugriffscode falsch eingegeben.
|
/// Sucht eine lokalisierte Zeichenfolge, die Zugriffscode falsch eingegeben ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string AccessCodeIncorrect {
|
public static string AccessCodeIncorrect {
|
||||||
get {
|
get {
|
||||||
@@ -79,7 +79,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Zugriffscode angefordert.
|
/// Sucht eine lokalisierte Zeichenfolge, die Zugriffscode angefordert ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string AccessCodeRequested {
|
public static string AccessCodeRequested {
|
||||||
get {
|
get {
|
||||||
@@ -88,7 +88,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Fortgeschrittene Elektronische Signatur.
|
/// Sucht eine lokalisierte Zeichenfolge, die Fortgeschrittene Elektronische Signatur ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string AdvancedElectronicSignature {
|
public static string AdvancedElectronicSignature {
|
||||||
get {
|
get {
|
||||||
@@ -97,7 +97,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Abgeschlossen.
|
/// Sucht eine lokalisierte Zeichenfolge, die Abgeschlossen ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Completed {
|
public static string Completed {
|
||||||
get {
|
get {
|
||||||
@@ -106,16 +106,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Vollständig bestätigt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Vollständig Signiert ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string CompletelyConfirmed {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("CompletelyConfirmed", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Vollständig signiert.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string CompletelySigned {
|
public static string CompletelySigned {
|
||||||
get {
|
get {
|
||||||
@@ -124,25 +115,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Lesebestätigung.
|
/// Sucht eine lokalisierte Zeichenfolge, die Vertrag ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string Confirmation {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("Confirmation", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Gelesen und bestätigt.
|
|
||||||
/// </summary>
|
|
||||||
public static string Confirmed {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("Confirmed", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Vertrag.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Contract {
|
public static string Contract {
|
||||||
get {
|
get {
|
||||||
@@ -151,7 +124,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Erstellt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Erstellt ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Created {
|
public static string Created {
|
||||||
get {
|
get {
|
||||||
@@ -160,16 +133,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Dokument gelesen und bestätigt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Dokument Rotation geändert ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string DocumentConfirmed {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("DocumentConfirmed", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Dokument Rotation geändert.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string DocumentMod_Rotation {
|
public static string DocumentMod_Rotation {
|
||||||
get {
|
get {
|
||||||
@@ -178,7 +142,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Dokument geöffnet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Dokument geöffnet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string DocumentOpened {
|
public static string DocumentOpened {
|
||||||
get {
|
get {
|
||||||
@@ -187,7 +151,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Unterzeichnung abgelehnt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Unterzeichnung abgelehnt ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string DocumentRejected {
|
public static string DocumentRejected {
|
||||||
get {
|
get {
|
||||||
@@ -196,16 +160,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Lesebestätigung abgelehnt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Dokument unterzeichnet ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string DocumentRejectedRaC {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("DocumentRejectedRaC", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Dokument unterzeichnet.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string DocumentSigned {
|
public static string DocumentSigned {
|
||||||
get {
|
get {
|
||||||
@@ -214,7 +169,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Entwurf.
|
/// Sucht eine lokalisierte Zeichenfolge, die Entwurf ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Draft {
|
public static string Draft {
|
||||||
get {
|
get {
|
||||||
@@ -223,7 +178,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Archiviert.
|
/// Sucht eine lokalisierte Zeichenfolge, die Archiviert ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeArchived {
|
public static string EnvelopeArchived {
|
||||||
get {
|
get {
|
||||||
@@ -232,16 +187,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Vollständig gelesen und bestätigt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Vollständig signiert ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string EnvelopeCompletelyConfirmed {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("EnvelopeCompletelyConfirmed", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Vollständig signiert.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeCompletelySigned {
|
public static string EnvelopeCompletelySigned {
|
||||||
get {
|
get {
|
||||||
@@ -250,7 +196,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Umschlag Erstellt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Umschlag Erstellt ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeCreated {
|
public static string EnvelopeCreated {
|
||||||
get {
|
get {
|
||||||
@@ -259,7 +205,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Umschlag Gelöscht.
|
/// Sucht eine lokalisierte Zeichenfolge, die Umschlag Gelöscht ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeDeleted {
|
public static string EnvelopeDeleted {
|
||||||
get {
|
get {
|
||||||
@@ -268,16 +214,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Teil-Bestätigt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Teil-Signiert ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string EnvelopePartlyConfirmed {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("EnvelopePartlyConfirmed", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Teil-Signiert.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopePartlySigned {
|
public static string EnvelopePartlySigned {
|
||||||
get {
|
get {
|
||||||
@@ -286,7 +223,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Umschlag in Queue.
|
/// Sucht eine lokalisierte Zeichenfolge, die Umschlag in Queue ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeQueued {
|
public static string EnvelopeQueued {
|
||||||
get {
|
get {
|
||||||
@@ -295,7 +232,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Umschlag abgelehnt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Umschlag abgelehnt ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeRejected {
|
public static string EnvelopeRejected {
|
||||||
get {
|
get {
|
||||||
@@ -304,7 +241,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Signierungszertifikat erstellt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Signierungszertifikat erstellt ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeReportCreated {
|
public static string EnvelopeReportCreated {
|
||||||
get {
|
get {
|
||||||
@@ -313,16 +250,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Lesebestätigungszertifikat erstellt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Gespeichert ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string EnvelopeReportCreatedRaC {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("EnvelopeReportCreatedRaC", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Gespeichert.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeSaved {
|
public static string EnvelopeSaved {
|
||||||
get {
|
get {
|
||||||
@@ -331,7 +259,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Gesendet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Gesendet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeSent {
|
public static string EnvelopeSent {
|
||||||
get {
|
get {
|
||||||
@@ -340,7 +268,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Umschlag zurückgezogen.
|
/// Sucht eine lokalisierte Zeichenfolge, die Umschlag zurückgezogen ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string EnvelopeWithdrawn {
|
public static string EnvelopeWithdrawn {
|
||||||
get {
|
get {
|
||||||
@@ -349,7 +277,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Zugriffscode versendet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Zugriffscode versendet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string MessageAccessCodeSent {
|
public static string MessageAccessCodeSent {
|
||||||
get {
|
get {
|
||||||
@@ -358,7 +286,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Abschlussemail versendet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Abschlussemail versendet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string MessageCompletionSent {
|
public static string MessageCompletionSent {
|
||||||
get {
|
get {
|
||||||
@@ -367,7 +295,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Abschlussbestätigung versendet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Signaturbestätigung versendet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string MessageConfirmationSent {
|
public static string MessageConfirmationSent {
|
||||||
get {
|
get {
|
||||||
@@ -376,7 +304,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Löschinformation versendet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Löschinformation versendet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string MessageDeletionSent {
|
public static string MessageDeletionSent {
|
||||||
get {
|
get {
|
||||||
@@ -385,7 +313,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Dokumentenlink versendet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Dokumentenlink versendet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string MessageInvitationSent {
|
public static string MessageInvitationSent {
|
||||||
get {
|
get {
|
||||||
@@ -394,7 +322,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Nein.
|
/// Sucht eine lokalisierte Zeichenfolge, die Nein ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string No {
|
public static string No {
|
||||||
get {
|
get {
|
||||||
@@ -403,16 +331,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Teil-Bestätigt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Teil-Signiert ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string PartlyConfirmed {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("PartlyConfirmed", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Teil-Signiert.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string PartlySigned {
|
public static string PartlySigned {
|
||||||
get {
|
get {
|
||||||
@@ -421,7 +340,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Qualifizierte Signatur.
|
/// Sucht eine lokalisierte Zeichenfolge, die Qualifizierte Signatur ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string QualifiedSignature {
|
public static string QualifiedSignature {
|
||||||
get {
|
get {
|
||||||
@@ -430,7 +349,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Lesebestätigung.
|
/// Sucht eine lokalisierte Zeichenfolge, die Arbeitsanweisung ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string ReadAndSign {
|
public static string ReadAndSign {
|
||||||
get {
|
get {
|
||||||
@@ -439,7 +358,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren!.
|
/// Sucht eine lokalisierte Zeichenfolge, die Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren! ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string ResetTOTPUser {
|
public static string ResetTOTPUser {
|
||||||
get {
|
get {
|
||||||
@@ -448,7 +367,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Gespeichert.
|
/// Sucht eine lokalisierte Zeichenfolge, die Gespeichert ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Saved {
|
public static string Saved {
|
||||||
get {
|
get {
|
||||||
@@ -457,7 +376,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Gesendet.
|
/// Sucht eine lokalisierte Zeichenfolge, die Gesendet ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Sent {
|
public static string Sent {
|
||||||
get {
|
get {
|
||||||
@@ -466,7 +385,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Signatur.
|
/// Sucht eine lokalisierte Zeichenfolge, die Signatur ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Signature {
|
public static string Signature {
|
||||||
get {
|
get {
|
||||||
@@ -475,7 +394,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Abschluss bestätigt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Signatur bestätigt ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string SignatureConfirmed {
|
public static string SignatureConfirmed {
|
||||||
get {
|
get {
|
||||||
@@ -484,7 +403,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Signiert.
|
/// Sucht eine lokalisierte Zeichenfolge, die Signiert ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Signed {
|
public static string Signed {
|
||||||
get {
|
get {
|
||||||
@@ -493,7 +412,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Erfolgreich! Dialog wird geschlossen..
|
/// Sucht eine lokalisierte Zeichenfolge, die Erfolgreich! Dialog wird geschlossen. ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Success_FormClose {
|
public static string Success_FormClose {
|
||||||
get {
|
get {
|
||||||
@@ -502,16 +421,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Unbestätigt.
|
/// Sucht eine lokalisierte Zeichenfolge, die Unsigniert ähnelt.
|
||||||
/// </summary>
|
|
||||||
public static string Unconfirmed {
|
|
||||||
get {
|
|
||||||
return ResourceManager.GetString("Unconfirmed", resourceCulture);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Looks up a localized string similar to Unsigniert.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Unsigned {
|
public static string Unsigned {
|
||||||
get {
|
get {
|
||||||
@@ -520,7 +430,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Ja.
|
/// Sucht eine lokalisierte Zeichenfolge, die Ja ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string Yes {
|
public static string Yes {
|
||||||
get {
|
get {
|
||||||
@@ -529,7 +439,7 @@ namespace My.Resources {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Looks up a localized string similar to Ja, mit Anhang.
|
/// Sucht eine lokalisierte Zeichenfolge, die Ja, mit Anhang ähnelt.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string YesWithAttachment {
|
public static string YesWithAttachment {
|
||||||
get {
|
get {
|
||||||
|
|||||||
@@ -132,27 +132,15 @@
|
|||||||
<data name="Completed" xml:space="preserve">
|
<data name="Completed" xml:space="preserve">
|
||||||
<value>Completed</value>
|
<value>Completed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Completely confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="CompletelySigned" xml:space="preserve">
|
<data name="CompletelySigned" xml:space="preserve">
|
||||||
<value>Completely signed</value>
|
<value>Completely signed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Confirmation" xml:space="preserve">
|
|
||||||
<value>Read Confirmation</value>
|
|
||||||
</data>
|
|
||||||
<data name="Confirmed" xml:space="preserve">
|
|
||||||
<value>Read and signed</value>
|
|
||||||
</data>
|
|
||||||
<data name="Contract" xml:space="preserve">
|
<data name="Contract" xml:space="preserve">
|
||||||
<value>Contract</value>
|
<value>Contract</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Created" xml:space="preserve">
|
<data name="Created" xml:space="preserve">
|
||||||
<value>Created</value>
|
<value>Created</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentConfirmed" xml:space="preserve">
|
|
||||||
<value>Document read and signed</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentMod_Rotation" xml:space="preserve">
|
<data name="DocumentMod_Rotation" xml:space="preserve">
|
||||||
<value>Document rotation adapted</value>
|
<value>Document rotation adapted</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -162,9 +150,6 @@
|
|||||||
<data name="DocumentRejected" xml:space="preserve">
|
<data name="DocumentRejected" xml:space="preserve">
|
||||||
<value>Signing rejected</value>
|
<value>Signing rejected</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentRejectedRaC" xml:space="preserve">
|
|
||||||
<value>Read confirmation rejected</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentSigned" xml:space="preserve">
|
<data name="DocumentSigned" xml:space="preserve">
|
||||||
<value>Document signed</value>
|
<value>Document signed</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -174,9 +159,6 @@
|
|||||||
<data name="EnvelopeArchived" xml:space="preserve">
|
<data name="EnvelopeArchived" xml:space="preserve">
|
||||||
<value>Archived</value>
|
<value>Archived</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeCompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Completely confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
||||||
<value>Completely signed</value>
|
<value>Completely signed</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -186,11 +168,8 @@
|
|||||||
<data name="EnvelopeDeleted" xml:space="preserve">
|
<data name="EnvelopeDeleted" xml:space="preserve">
|
||||||
<value>Envelope Deleted</value>
|
<value>Envelope Deleted</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopePartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Partially confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopePartlySigned" xml:space="preserve">
|
<data name="EnvelopePartlySigned" xml:space="preserve">
|
||||||
<value>Partially signed</value>
|
<value>Partly signed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeQueued" xml:space="preserve">
|
<data name="EnvelopeQueued" xml:space="preserve">
|
||||||
<value>Envelope Queued</value>
|
<value>Envelope Queued</value>
|
||||||
@@ -201,9 +180,6 @@
|
|||||||
<data name="EnvelopeReportCreated" xml:space="preserve">
|
<data name="EnvelopeReportCreated" xml:space="preserve">
|
||||||
<value>Signature certificate created</value>
|
<value>Signature certificate created</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeReportCreatedRaC" xml:space="preserve">
|
|
||||||
<value>Read confirmartion certificate created</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeSaved" xml:space="preserve">
|
<data name="EnvelopeSaved" xml:space="preserve">
|
||||||
<value>Saved</value>
|
<value>Saved</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -220,7 +196,7 @@
|
|||||||
<value>Final email sent</value>
|
<value>Final email sent</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MessageConfirmationSent" xml:space="preserve">
|
<data name="MessageConfirmationSent" xml:space="preserve">
|
||||||
<value>Finalization Confirmation Sent</value>
|
<value>Confirmation Sent</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MessageDeletionSent" xml:space="preserve">
|
<data name="MessageDeletionSent" xml:space="preserve">
|
||||||
<value>Deletion Notice Sent</value>
|
<value>Deletion Notice Sent</value>
|
||||||
@@ -231,12 +207,6 @@
|
|||||||
<data name="No" xml:space="preserve">
|
<data name="No" xml:space="preserve">
|
||||||
<value>No</value>
|
<value>No</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Partially confirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="PartlySigned" xml:space="preserve">
|
|
||||||
<value>Partially signed</value>
|
|
||||||
</data>
|
|
||||||
<data name="QualifiedSignature" xml:space="preserve">
|
<data name="QualifiedSignature" xml:space="preserve">
|
||||||
<value>Qualified Signature</value>
|
<value>Qualified Signature</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -256,7 +226,7 @@
|
|||||||
<value>Signature</value>
|
<value>Signature</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SignatureConfirmed" xml:space="preserve">
|
<data name="SignatureConfirmed" xml:space="preserve">
|
||||||
<value>Finalization confirmed</value>
|
<value>Signature confirmed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Signed" xml:space="preserve">
|
<data name="Signed" xml:space="preserve">
|
||||||
<value>Signed</value>
|
<value>Signed</value>
|
||||||
@@ -264,9 +234,6 @@
|
|||||||
<data name="Success_FormClose" xml:space="preserve">
|
<data name="Success_FormClose" xml:space="preserve">
|
||||||
<value>Successful! Dialog is closed.successful! Dialog is closed.</value>
|
<value>Successful! Dialog is closed.successful! Dialog is closed.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Unconfirmed" xml:space="preserve">
|
|
||||||
<value>Unconfirmed</value>
|
|
||||||
</data>
|
|
||||||
<data name="Unsigned" xml:space="preserve">
|
<data name="Unsigned" xml:space="preserve">
|
||||||
<value>Unsigned</value>
|
<value>Unsigned</value>
|
||||||
</data>
|
</data>
|
||||||
|
|||||||
@@ -132,17 +132,8 @@
|
|||||||
<data name="Completed" xml:space="preserve">
|
<data name="Completed" xml:space="preserve">
|
||||||
<value>Abgeschlossen</value>
|
<value>Abgeschlossen</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Vollständig bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="CompletelySigned" xml:space="preserve">
|
<data name="CompletelySigned" xml:space="preserve">
|
||||||
<value>Vollständig signiert</value>
|
<value>Vollständig Signiert</value>
|
||||||
</data>
|
|
||||||
<data name="Confirmation" xml:space="preserve">
|
|
||||||
<value>Lesebestätigung</value>
|
|
||||||
</data>
|
|
||||||
<data name="Confirmed" xml:space="preserve">
|
|
||||||
<value>Gelesen und bestätigt</value>
|
|
||||||
</data>
|
</data>
|
||||||
<data name="Contract" xml:space="preserve">
|
<data name="Contract" xml:space="preserve">
|
||||||
<value>Vertrag</value>
|
<value>Vertrag</value>
|
||||||
@@ -150,9 +141,6 @@
|
|||||||
<data name="Created" xml:space="preserve">
|
<data name="Created" xml:space="preserve">
|
||||||
<value>Erstellt</value>
|
<value>Erstellt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentConfirmed" xml:space="preserve">
|
|
||||||
<value>Dokument gelesen und bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentMod_Rotation" xml:space="preserve">
|
<data name="DocumentMod_Rotation" xml:space="preserve">
|
||||||
<value>Dokument Rotation geändert</value>
|
<value>Dokument Rotation geändert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -162,9 +150,6 @@
|
|||||||
<data name="DocumentRejected" xml:space="preserve">
|
<data name="DocumentRejected" xml:space="preserve">
|
||||||
<value>Unterzeichnung abgelehnt</value>
|
<value>Unterzeichnung abgelehnt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DocumentRejectedRaC" xml:space="preserve">
|
|
||||||
<value>Lesebestätigung abgelehnt</value>
|
|
||||||
</data>
|
|
||||||
<data name="DocumentSigned" xml:space="preserve">
|
<data name="DocumentSigned" xml:space="preserve">
|
||||||
<value>Dokument unterzeichnet</value>
|
<value>Dokument unterzeichnet</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -174,9 +159,6 @@
|
|||||||
<data name="EnvelopeArchived" xml:space="preserve">
|
<data name="EnvelopeArchived" xml:space="preserve">
|
||||||
<value>Archiviert</value>
|
<value>Archiviert</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeCompletelyConfirmed" xml:space="preserve">
|
|
||||||
<value>Vollständig gelesen und bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
<data name="EnvelopeCompletelySigned" xml:space="preserve">
|
||||||
<value>Vollständig signiert</value>
|
<value>Vollständig signiert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -186,9 +168,6 @@
|
|||||||
<data name="EnvelopeDeleted" xml:space="preserve">
|
<data name="EnvelopeDeleted" xml:space="preserve">
|
||||||
<value>Umschlag Gelöscht</value>
|
<value>Umschlag Gelöscht</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopePartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Teil-Bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopePartlySigned" xml:space="preserve">
|
<data name="EnvelopePartlySigned" xml:space="preserve">
|
||||||
<value>Teil-Signiert</value>
|
<value>Teil-Signiert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -201,9 +180,6 @@
|
|||||||
<data name="EnvelopeReportCreated" xml:space="preserve">
|
<data name="EnvelopeReportCreated" xml:space="preserve">
|
||||||
<value>Signierungszertifikat erstellt</value>
|
<value>Signierungszertifikat erstellt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="EnvelopeReportCreatedRaC" xml:space="preserve">
|
|
||||||
<value>Lesebestätigungszertifikat erstellt</value>
|
|
||||||
</data>
|
|
||||||
<data name="EnvelopeSaved" xml:space="preserve">
|
<data name="EnvelopeSaved" xml:space="preserve">
|
||||||
<value>Gespeichert</value>
|
<value>Gespeichert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -220,7 +196,7 @@
|
|||||||
<value>Abschlussemail versendet</value>
|
<value>Abschlussemail versendet</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MessageConfirmationSent" xml:space="preserve">
|
<data name="MessageConfirmationSent" xml:space="preserve">
|
||||||
<value>Abschlussbestätigung versendet</value>
|
<value>Signaturbestätigung versendet</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MessageDeletionSent" xml:space="preserve">
|
<data name="MessageDeletionSent" xml:space="preserve">
|
||||||
<value>Löschinformation versendet</value>
|
<value>Löschinformation versendet</value>
|
||||||
@@ -231,9 +207,6 @@
|
|||||||
<data name="No" xml:space="preserve">
|
<data name="No" xml:space="preserve">
|
||||||
<value>Nein</value>
|
<value>Nein</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PartlyConfirmed" xml:space="preserve">
|
|
||||||
<value>Teil-Bestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="PartlySigned" xml:space="preserve">
|
<data name="PartlySigned" xml:space="preserve">
|
||||||
<value>Teil-Signiert</value>
|
<value>Teil-Signiert</value>
|
||||||
</data>
|
</data>
|
||||||
@@ -241,7 +214,7 @@
|
|||||||
<value>Qualifizierte Signatur</value>
|
<value>Qualifizierte Signatur</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ReadAndSign" xml:space="preserve">
|
<data name="ReadAndSign" xml:space="preserve">
|
||||||
<value>Lesebestätigung</value>
|
<value>Arbeitsanweisung</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ResetTOTPUser" xml:space="preserve">
|
<data name="ResetTOTPUser" xml:space="preserve">
|
||||||
<value>Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren!</value>
|
<value>Wollen Sie die 2-Faktor Definition für diesen Empfänger zurücksetzen. Der Empfänger muss sich dann neu identifizieren!</value>
|
||||||
@@ -256,7 +229,7 @@
|
|||||||
<value>Signatur</value>
|
<value>Signatur</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SignatureConfirmed" xml:space="preserve">
|
<data name="SignatureConfirmed" xml:space="preserve">
|
||||||
<value>Abschluss bestätigt</value>
|
<value>Signatur bestätigt</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Signed" xml:space="preserve">
|
<data name="Signed" xml:space="preserve">
|
||||||
<value>Signiert</value>
|
<value>Signiert</value>
|
||||||
@@ -264,9 +237,6 @@
|
|||||||
<data name="Success_FormClose" xml:space="preserve">
|
<data name="Success_FormClose" xml:space="preserve">
|
||||||
<value>Erfolgreich! Dialog wird geschlossen.</value>
|
<value>Erfolgreich! Dialog wird geschlossen.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Unconfirmed" xml:space="preserve">
|
|
||||||
<value>Unbestätigt</value>
|
|
||||||
</data>
|
|
||||||
<data name="Unsigned" xml:space="preserve">
|
<data name="Unsigned" xml:space="preserve">
|
||||||
<value>Unsigniert</value>
|
<value>Unsigniert</value>
|
||||||
</data>
|
</data>
|
||||||
|
|||||||
@@ -47,8 +47,6 @@ public abstract class EGDbContextBase : DbContext
|
|||||||
|
|
||||||
public DbSet<Signature> DocumentReceiverElements { get; set; }
|
public DbSet<Signature> DocumentReceiverElements { get; set; }
|
||||||
|
|
||||||
public DbSet<ElementAnnotation> DocumentReceiverElementAnnotations { get; set; }
|
|
||||||
|
|
||||||
public DbSet<DocumentStatus> DocumentStatus { get; set; }
|
public DbSet<DocumentStatus> DocumentStatus { get; set; }
|
||||||
|
|
||||||
public DbSet<EmailTemplate> EmailTemplate { get; set; }
|
public DbSet<EmailTemplate> EmailTemplate { get; set; }
|
||||||
@@ -81,6 +79,10 @@ public abstract class EGDbContextBase : DbContext
|
|||||||
|
|
||||||
public DbSet<ClientUser> ClientUsers { get; set; }
|
public DbSet<ClientUser> ClientUsers { get; set; }
|
||||||
|
|
||||||
|
public DbSet<EnvelopeReport> EnvelopeReports { get; set; }
|
||||||
|
|
||||||
|
public DbSet<ThirdPartyModule> ThirdPartyModules { get; set; }
|
||||||
|
|
||||||
private readonly DbTriggerParams _triggers;
|
private readonly DbTriggerParams _triggers;
|
||||||
|
|
||||||
private readonly ILogger
|
private readonly ILogger
|
||||||
@@ -202,6 +204,10 @@ public abstract class EGDbContextBase : DbContext
|
|||||||
.HasForeignKey(annot => annot.ElementId);
|
.HasForeignKey(annot => annot.ElementId);
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
#region EnvelopeReport
|
||||||
|
modelBuilder.Entity<EnvelopeReport>().HasNoKey();
|
||||||
|
#endregion
|
||||||
|
|
||||||
#region Trigger
|
#region Trigger
|
||||||
// Configure entities to handle database triggers
|
// Configure entities to handle database triggers
|
||||||
void AddTrigger<T>() where T : class => _triggers
|
void AddTrigger<T>() where T : class => _triggers
|
||||||
|
|||||||
@@ -105,13 +105,17 @@ namespace EnvelopeGenerator.PdfEditor
|
|||||||
public Pdf<TInputStream, TOutputStream> Background<TSignature>(IEnumerable<TSignature> signatures, double widthPx = 1.9500000000000002, double heightPx = 2.52)
|
public Pdf<TInputStream, TOutputStream> Background<TSignature>(IEnumerable<TSignature> signatures, double widthPx = 1.9500000000000002, double heightPx = 2.52)
|
||||||
where TSignature : ISignature
|
where TSignature : ISignature
|
||||||
{
|
{
|
||||||
|
// once per page
|
||||||
|
Page(page =>
|
||||||
|
{
|
||||||
|
var canvas = new PdfCanvas(page);
|
||||||
|
canvas.ConcatMatrix(1, 0, 0, -1, 0, page.GetPageSize().GetHeight());
|
||||||
|
});
|
||||||
|
|
||||||
foreach (var signature in signatures)
|
foreach (var signature in signatures)
|
||||||
Page(signature.Page, page =>
|
Page(signature.Page, page =>
|
||||||
{
|
{
|
||||||
var canvas = new PdfCanvas(page);
|
var canvas = new PdfCanvas(page);
|
||||||
canvas.SaveState();
|
|
||||||
canvas.ConcatMatrix(1, 0, 0, -1, 0, page.GetPageSize().GetHeight());
|
|
||||||
|
|
||||||
double inchFactor = 72;
|
double inchFactor = 72;
|
||||||
double magin = .2;
|
double magin = .2;
|
||||||
double x = (signature.X - .7 - magin) * inchFactor;
|
double x = (signature.X - .7 - magin) * inchFactor;
|
||||||
@@ -130,8 +134,6 @@ namespace EnvelopeGenerator.PdfEditor
|
|||||||
canvas.SetFillColor(new DeviceRgb(204, 202, 198))
|
canvas.SetFillColor(new DeviceRgb(204, 202, 198))
|
||||||
.Rectangle(x, y + height - bottomLineLength, width, bottomLineLength)
|
.Rectangle(x, y + height - bottomLineLength, width, bottomLineLength)
|
||||||
.Fill();
|
.Fill();
|
||||||
|
|
||||||
canvas.RestoreState();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
|
|||||||
@@ -1,10 +0,0 @@
|
|||||||
<Router AppAssembly="@typeof(Program).Assembly">
|
|
||||||
<Found Context="routeData">
|
|
||||||
<RouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
|
|
||||||
</Found>
|
|
||||||
<NotFound>
|
|
||||||
<LayoutView Layout="@typeof(MainLayout)">
|
|
||||||
<p>Sorry, there's nothing at this address.</p>
|
|
||||||
</LayoutView>
|
|
||||||
</NotFound>
|
|
||||||
</Router>
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.ReceiverUI.Data {
|
|
||||||
public class Adjustment
|
|
||||||
{
|
|
||||||
public static Adjustment CreateBalanceForward(DateTime dt, int random)
|
|
||||||
{
|
|
||||||
var rnd = new DeterministicRandom(random);
|
|
||||||
Adjustment res = new Adjustment();
|
|
||||||
res.currentDateTime = dt;
|
|
||||||
res.currentDescription = "Balance Forward";
|
|
||||||
res.currentAmount = rnd.Random(10, 300) * 10;
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
public static Adjustment CreatePayment(DateTime dt, int random)
|
|
||||||
{
|
|
||||||
var rnd = new DeterministicRandom(random);
|
|
||||||
Adjustment res = new Adjustment();
|
|
||||||
res.currentDateTime = dt;
|
|
||||||
res.currentDescription = "Payment";
|
|
||||||
res.currentAmount = -rnd.Random(1, 40) * 10;
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
public static Adjustment CreateCharge(DateTime dt, int random)
|
|
||||||
{
|
|
||||||
var rnd = new DeterministicRandom(random);
|
|
||||||
Adjustment res = new Adjustment();
|
|
||||||
res.currentDateTime = dt;
|
|
||||||
res.currentDescription = rnd.GetRandomItem(bills);
|
|
||||||
res.currentAmount = rnd.Random(10, 50) * 10;
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
DateTime currentDateTime;
|
|
||||||
string currentDescription = "";
|
|
||||||
double currentAmount = 0;
|
|
||||||
static readonly string[] bills = new string[] { "Bill - Insurance", "Bill - Electricity", "Bill - Rent", "Bill - Phone", "Bill - Office Supplies" };
|
|
||||||
public DateTime Date { get { return currentDateTime; } }
|
|
||||||
public string Description { get { return currentDescription; } }
|
|
||||||
public double Amount { get { return currentAmount; } }
|
|
||||||
|
|
||||||
public Adjustment()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
using DevExpress.DataAccess.Sql;
|
|
||||||
using DevExpress.DataAccess.Sql.DataApi;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Data {
|
|
||||||
public class Customer {
|
|
||||||
static List<Customer> currentCustomers = new List<Customer>();
|
|
||||||
|
|
||||||
public static List<Customer> Customers { get { return currentCustomers; } }
|
|
||||||
static Customer() {
|
|
||||||
try {
|
|
||||||
SqlDataSource ds = new SqlDataSource("NWindConnectionString");
|
|
||||||
SelectQuery query = SelectQueryFluentBuilder
|
|
||||||
.AddTable("Customers")
|
|
||||||
.SelectAllColumns()
|
|
||||||
.Build("Customers");
|
|
||||||
ds.Queries.Add(query);
|
|
||||||
ds.RebuildResultSchema();
|
|
||||||
ds.Fill();
|
|
||||||
ITable src = ds.Result["Customers"];
|
|
||||||
foreach(var row in src) {
|
|
||||||
currentCustomers.Add(new Customer() {
|
|
||||||
CustomerID = row.GetValue<string>("CustomerID"),
|
|
||||||
Address = row.GetValue<string>("Address"),
|
|
||||||
CompanyName = row.GetValue<string>("CompanyName"),
|
|
||||||
ContactName = row.GetValue<string>("ContactName"),
|
|
||||||
ContactTitle = row.GetValue<string>("ContactTitle"),
|
|
||||||
Country = row.GetValue<string>("Country"),
|
|
||||||
City = row.GetValue<string>("City"),
|
|
||||||
Fax = row.GetValue<string>("Fax"),
|
|
||||||
Phone = row.GetValue<string>("Phone"),
|
|
||||||
PostalCode = row.GetValue<string>("PostalCode"),
|
|
||||||
Region = row.GetValue<string>("Region")
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch {
|
|
||||||
currentCustomers.Add(new Customer() {
|
|
||||||
Address = "Obere Str. 57",
|
|
||||||
City = "Berlin",
|
|
||||||
CompanyName = "Alfreds Futterkiste",
|
|
||||||
ContactName = "Maria Anders",
|
|
||||||
ContactTitle = "Sales Representative",
|
|
||||||
Country = "Germany",
|
|
||||||
CustomerID = "ALFKI",
|
|
||||||
Fax = "030-0076545",
|
|
||||||
Phone = "030-0074321",
|
|
||||||
PostalCode = "12209"
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public string CustomerID { get; set; }
|
|
||||||
public string CompanyName { get; set; }
|
|
||||||
public string ContactName { get; set; }
|
|
||||||
public string ContactTitle { get; set; }
|
|
||||||
public string Address { get; set; }
|
|
||||||
public string City { get; set; }
|
|
||||||
public string PostalCode { get; set; }
|
|
||||||
public string Region { get; set; }
|
|
||||||
public string Country { get; set; }
|
|
||||||
public string Phone { get; set; }
|
|
||||||
public string Fax { get; set; }
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.ReceiverUI.Data {
|
|
||||||
public class DataItem {
|
|
||||||
static readonly string[] accountType = new string[] { "Energy", "Manufacturing", "Estate", "Food", "Services" };
|
|
||||||
public string CustomerID { get; set; }
|
|
||||||
public string CompanyName { get; set; }
|
|
||||||
public string ContactName { get; set; }
|
|
||||||
public string ContactTitle { get; set; }
|
|
||||||
public string Address { get; set; }
|
|
||||||
public string City { get; set; }
|
|
||||||
public string PostalCode { get; set; }
|
|
||||||
public string Region { get; set; }
|
|
||||||
public string Country { get; set; }
|
|
||||||
public string Phone { get; set; }
|
|
||||||
public string Fax { get; set; }
|
|
||||||
public string Email { get; set; }
|
|
||||||
public string Invoice { get; set; }
|
|
||||||
public string CustomerAccount { get; set; }
|
|
||||||
public string CustomerIdentifiers { get; set; }
|
|
||||||
public DateTime BillingDate { get; set; }
|
|
||||||
public DateTime BillingPeriodStart { get; set; }
|
|
||||||
public DateTime BillingPeriodEnd { get; set; }
|
|
||||||
public string Terms { get; set; }
|
|
||||||
public string TermsID { get; set; }
|
|
||||||
public Adjustment[] Adjustments { get; set; }
|
|
||||||
|
|
||||||
public DataItem(int i) {
|
|
||||||
var rnd = new DeterministicRandom(i);
|
|
||||||
Customer c = rnd.GetRandomItem(Customer.Customers);
|
|
||||||
CustomerID = c.CustomerID;
|
|
||||||
CompanyName = c.CompanyName;
|
|
||||||
ContactName = c.ContactName;
|
|
||||||
ContactTitle = c.ContactTitle;
|
|
||||||
Address = c.Address;
|
|
||||||
City = c.City;
|
|
||||||
PostalCode = c.PostalCode;
|
|
||||||
Region = c.Region;
|
|
||||||
Country = c.Country;
|
|
||||||
Phone = c.Phone;
|
|
||||||
Fax = c.Fax;
|
|
||||||
Email = ContactName.Split(' ')[0].Replace(' ', '.').ToLower() + "@" + CompanyName.Split(' ')[0].ToLower() + ".com";
|
|
||||||
Invoice = string.Format("{0}{1}-{2}", rnd.RandomChar, rnd.Random(100, 1000), rnd.Random(100, 1000));
|
|
||||||
CustomerAccount = rnd.GetRandomItem(accountType);
|
|
||||||
CustomerIdentifiers = string.Format("{0}-{1}", rnd.Random(1000, 10000), rnd.Random(10, 100));
|
|
||||||
BillingPeriodStart = rnd.RandomTime();
|
|
||||||
BillingPeriodEnd = rnd.RandomTime(BillingPeriodStart, 7 * 24, 30 * 24);
|
|
||||||
BillingDate = rnd.RandomTime(BillingPeriodEnd, 7 * 24, 30 * 24);
|
|
||||||
Term currentTerm = rnd.GetRandomItem(Term.Terms);
|
|
||||||
Terms = currentTerm.Name;
|
|
||||||
|
|
||||||
int adjustmentsCount = rnd.Random(6) + 4;
|
|
||||||
Adjustments = new Adjustment[adjustmentsCount];
|
|
||||||
int h = (int)((BillingPeriodEnd - BillingPeriodStart).TotalHours / adjustmentsCount);
|
|
||||||
|
|
||||||
Adjustments[0] = Adjustment.CreateBalanceForward(rnd.RandomTime(BillingPeriodStart, 0, h), rnd.Random(10000));
|
|
||||||
|
|
||||||
int[] items = rnd.RandomList(adjustmentsCount - 1, 2);
|
|
||||||
|
|
||||||
for(int j = 1; j < Adjustments.Length; j++) {
|
|
||||||
DateTime nextDate = rnd.RandomTime(BillingPeriodStart.AddHours(h * j), 0, h);
|
|
||||||
switch(items[j - 1]) {
|
|
||||||
case 0:
|
|
||||||
Adjustments[j] = Adjustment.CreateCharge(nextDate, rnd.Random(10000));
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
Adjustments[j] = Adjustment.CreatePayment(nextDate, rnd.Random(10000));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,70 +0,0 @@
|
|||||||
using System.Collections;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Data {
|
|
||||||
public class DataItemList : IList<DataItem>, IList {
|
|
||||||
readonly int rowCount;
|
|
||||||
|
|
||||||
public DataItem this[int index] { get { return new DataItem(index); } set { } }
|
|
||||||
public int Count { get { return rowCount; } }
|
|
||||||
public bool IsReadOnly { get { return false; } }
|
|
||||||
public bool IsFixedSize { get { return false; } }
|
|
||||||
public object SyncRoot { get { return true; } }
|
|
||||||
public bool IsSynchronized { get { return true; } }
|
|
||||||
object IList.this[int index] { get { return new DataItem(index); } set { } }
|
|
||||||
|
|
||||||
public DataItemList(int rowCount) {
|
|
||||||
this.rowCount = rowCount;
|
|
||||||
}
|
|
||||||
public IEnumerator<DataItem> GetEnumerator() {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public int Add(object value) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public bool Contains(object value) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void Clear() {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public int IndexOf(object value) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void Insert(int index, object value) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void Remove(object value) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void RemoveAt(int index) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void CopyTo(Array array, int index) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
IEnumerator IEnumerable.GetEnumerator() {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public int IndexOf(DataItem item) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void Insert(int index, DataItem item) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void Add(DataItem item) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public bool Contains(DataItem item) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public void CopyTo(DataItem[] array, int arrayIndex) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
public bool Remove(DataItem item) {
|
|
||||||
throw new NotImplementedException();
|
|
||||||
}
|
|
||||||
void ICollection<DataItem>.CopyTo(DataItem[] array, int arrayIndex) {
|
|
||||||
CopyTo(array, arrayIndex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.ReceiverUI.Data {
|
|
||||||
class DeterministicRandom {
|
|
||||||
const int randomCount = 10000;
|
|
||||||
static readonly int[] deterministicRandomNumbers;
|
|
||||||
static readonly DateTime time;
|
|
||||||
int rnd;
|
|
||||||
int Next {
|
|
||||||
get {
|
|
||||||
rnd = deterministicRandomNumbers[rnd % randomCount];
|
|
||||||
return rnd;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
public char RandomChar {
|
|
||||||
get {
|
|
||||||
return (char)((int)'A' + Random(0, 26));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
public int[] RandomList(int count, int to) {
|
|
||||||
int[] res = new int[count];
|
|
||||||
for(int i = 0; i < Math.Min(count, to); i++)
|
|
||||||
res[i] = i;
|
|
||||||
for(int i = to; i < count; i++)
|
|
||||||
res[i] = Random(to);
|
|
||||||
|
|
||||||
for(int i = 0; i < count; i++) {
|
|
||||||
int ind = Random(count);
|
|
||||||
int temp = res[ind];
|
|
||||||
res[ind] = res[i];
|
|
||||||
res[i] = temp;
|
|
||||||
}
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
public int Random(int to) {
|
|
||||||
return Random(0, to);
|
|
||||||
}
|
|
||||||
public int Random(int from, int to) {
|
|
||||||
return Next % Math.Max(1, to - from) + from;
|
|
||||||
}
|
|
||||||
public T GetRandomItem<T>(IList<T> list) {
|
|
||||||
return list[Next % list.Count];
|
|
||||||
}
|
|
||||||
public DateTime RandomTime() {
|
|
||||||
return RandomTime(time, 0, 30 * 24);
|
|
||||||
}
|
|
||||||
public DateTime RandomTime(DateTime from, int fromHours, int toHours) {
|
|
||||||
return from.AddHours(Next % (toHours - fromHours) + fromHours);
|
|
||||||
}
|
|
||||||
|
|
||||||
static DeterministicRandom() {
|
|
||||||
time = DateTime.Now.AddDays(-62);
|
|
||||||
Random currentRandom = new Random(randomCount);
|
|
||||||
deterministicRandomNumbers = new int[randomCount];
|
|
||||||
for(int i = 0; i < randomCount; i++)
|
|
||||||
deterministicRandomNumbers[i] = currentRandom.Next(randomCount);
|
|
||||||
}
|
|
||||||
public DeterministicRandom(int i) {
|
|
||||||
this.rnd = i + (i >> 10) + (i >> 20);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.ReceiverUI.Data {
|
|
||||||
public struct Term {
|
|
||||||
public static readonly Term[] Terms = new Term[] {
|
|
||||||
new Term("Payment seven days after invoice date" ),
|
|
||||||
new Term("Payment ten days after invoice date" ),
|
|
||||||
new Term("End of month" ),
|
|
||||||
new Term("21st of the month following invoice date" ),
|
|
||||||
};
|
|
||||||
readonly string currentName;
|
|
||||||
public string Name { get { return currentName; } }
|
|
||||||
public Term(string currentName) {
|
|
||||||
this.currentName = currentName;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk.BlazorWebAssembly">
|
|
||||||
<PropertyGroup>
|
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
|
||||||
<WasmBuildNative>true</WasmBuildNative>
|
|
||||||
<InvariantGlobalization>false</InvariantGlobalization>
|
|
||||||
<BlazorWebAssemblyLoadAllGlobalizationData>true</BlazorWebAssemblyLoadAllGlobalizationData>
|
|
||||||
|
|
||||||
<Nullable>enable</Nullable>
|
|
||||||
<PackageId>EnvelopeGenerator.ReceiverUI</PackageId>
|
|
||||||
<Authors>Digital Data GmbH</Authors>
|
|
||||||
<Company>Digital Data GmbH</Company>
|
|
||||||
<Product>EnvelopeGenerator.ReceiverUI</Product>
|
|
||||||
<PackageIcon>Assets\icon.ico</PackageIcon>
|
|
||||||
<PackageTags>digital data envelope generator web</PackageTags>
|
|
||||||
<Description>EnvelopeGenerator.ReceiverUI is a Blazor WebAssembly application developed to manage signing processes. It uses Entity Framework Core (EF Core) for database operations. The user interface for signing processes is developed with Razor View Engine (.cshtml files) and JavaScript under wwwroot, integrated with PSPDFKit. This integration allows users to view and sign documents seamlessly.</Description>
|
|
||||||
<Version>1.3.0</Version>
|
|
||||||
<!-- NuGet package version -->
|
|
||||||
<AssemblyVersion>1.3.0.0</AssemblyVersion>
|
|
||||||
<!-- Assembly version for API compatibility -->
|
|
||||||
<FileVersion>1.3.0.0</FileVersion>
|
|
||||||
<!-- Windows file version -->
|
|
||||||
<Copyright>Copyright © 2026 Digital Data GmbH. All rights reserved.</Copyright>
|
|
||||||
|
|
||||||
</PropertyGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="DevExpress.Blazor.PdfViewer" Version="25.2.3" />
|
|
||||||
<PackageReference Include="DevExpress.Blazor.Reporting.JSBasedControls" Version="25.2.3" />
|
|
||||||
<PackageReference Include="DevExpress.Blazor.Reporting.Viewer" Version="25.2.3" />
|
|
||||||
<PackageReference Include="DevExpress.Drawing.Skia" Version="25.2.3" />
|
|
||||||
<PackageReference Include="HarfBuzzSharp.NativeAssets.WebAssembly" Version="8.3.1.2" />
|
|
||||||
<PackageReference Include="itext" Version="8.0.5" />
|
|
||||||
<PackageReference Include="SkiaSharp.NativeAssets.WebAssembly" Version="3.119.1" />
|
|
||||||
<PackageReference Include="SkiaSharp.Views.Blazor" Version="3.119.1" />
|
|
||||||
<NativeFileReference Include="$(HarfBuzzSharpStaticLibraryPath)\2.0.23\*.a" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Components.WebAssembly" Version="8.0.11" />
|
|
||||||
<PackageReference Include="Microsoft.AspNetCore.Components.WebAssembly.DevServer" Version="8.0.11" PrivateAssets="all" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<Folder Include="Properties\PublishProfiles\" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<Content Update="wwwroot\docs\privacy-policy.en-US.html">
|
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
|
||||||
</Content>
|
|
||||||
<Content Update="wwwroot\docs\privacy-policy.fr-FR.html">
|
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
|
||||||
</Content>
|
|
||||||
</ItemGroup>
|
|
||||||
</Project>
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.ReceiverUI.Models;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Represents a pre-assigned signature annotation position on a specific page.
|
|
||||||
/// <br/><br/>
|
|
||||||
/// <b>Coordinate unit (X, Y):</b> Hundredths of an inch (1/100 inch ? 2.83 PDF points),
|
|
||||||
/// origin at the <b>top-left</b> corner of the page, both axes increase downward/rightward.
|
|
||||||
/// This matches the DevExpress XtraReports coordinate system (<see cref="System.Drawing.RectangleF"/>).
|
|
||||||
/// <br/><br/>
|
|
||||||
/// <b>Difference from PSPDFKit:</b> Same origin (top-left) and direction, but PSPDFKit uses PDF points (1/72 inch).
|
|
||||||
/// Convert: <c>xDX = xPsPdf * (100.0 / 72.0)</c>
|
|
||||||
/// <br/>
|
|
||||||
/// <b>Difference from GDPicture:</b> GDPicture uses PDF points with <b>bottom-left</b> origin (PDF standard); Y is flipped.
|
|
||||||
/// Convert: <c>yDX = (pageHeightPt - yGD - elemHeightPt) * (100.0 / 72.0)</c>
|
|
||||||
/// </summary>
|
|
||||||
public record AnnotationDto
|
|
||||||
{
|
|
||||||
/// <summary>Unique identifier of the annotation.</summary>
|
|
||||||
public long Id { get; init; }
|
|
||||||
|
|
||||||
/// <summary>1-based page number within the document.</summary>
|
|
||||||
public int Page { get; init; }
|
|
||||||
|
|
||||||
/// <summary>Horizontal position in hundredths of an inch from the left edge of the page.</summary>
|
|
||||||
public double X { get; init; }
|
|
||||||
|
|
||||||
/// <summary>Vertical position in hundredths of an inch from the top edge of the page.</summary>
|
|
||||||
public double Y { get; init; }
|
|
||||||
}
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.ReceiverUI.Models;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Client-side model for the envelope receiver returned by
|
|
||||||
/// <c>GET api/EnvelopeReceiver/{envelopeKey}</c>.
|
|
||||||
/// </summary>
|
|
||||||
public record EnvelopeReceiverDto
|
|
||||||
{
|
|
||||||
public int EnvelopeId { get; init; }
|
|
||||||
public int ReceiverId { get; init; }
|
|
||||||
public int Sequence { get; init; }
|
|
||||||
|
|
||||||
public string? Name { get; init; }
|
|
||||||
public string? JobTitle { get; init; }
|
|
||||||
public string? CompanyName { get; init; }
|
|
||||||
public string? PrivateMessage { get; init; }
|
|
||||||
|
|
||||||
public DateTime AddedWhen { get; init; }
|
|
||||||
public DateTime? ChangedWhen { get; init; }
|
|
||||||
public bool HasPhoneNumber { get; init; }
|
|
||||||
|
|
||||||
public EnvelopeClientDto? Envelope { get; init; }
|
|
||||||
public ReceiverClientDto? Receiver { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Client-side model for the envelope data embedded in <see cref="EnvelopeReceiverDto"/>.
|
|
||||||
/// </summary>
|
|
||||||
public record EnvelopeClientDto
|
|
||||||
{
|
|
||||||
public int Id { get; init; }
|
|
||||||
public int UserId { get; init; }
|
|
||||||
public int Status { get; init; }
|
|
||||||
public string StatusName { get; init; } = string.Empty;
|
|
||||||
public string Uuid { get; init; } = string.Empty;
|
|
||||||
public string Title { get; init; } = string.Empty;
|
|
||||||
public string Message { get; init; } = string.Empty;
|
|
||||||
public DateTime AddedWhen { get; init; }
|
|
||||||
public DateTime? ChangedWhen { get; init; }
|
|
||||||
public string Language { get; init; } = "de-DE";
|
|
||||||
public int? EnvelopeTypeId { get; init; }
|
|
||||||
public string? EnvelopeTypeTitle { get; init; }
|
|
||||||
public int? ContractType { get; init; }
|
|
||||||
public int? CertificationType { get; init; }
|
|
||||||
public bool UseAccessCode { get; init; }
|
|
||||||
public bool TFAEnabled { get; init; }
|
|
||||||
public IEnumerable<DocumentClientDto>? Documents { get; init; }
|
|
||||||
public EnvelopeSenderDto? User { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Sender (user) information embedded in <see cref="EnvelopeClientDto"/>.
|
|
||||||
/// </summary>
|
|
||||||
public record EnvelopeSenderDto
|
|
||||||
{
|
|
||||||
public int Id { get; init; }
|
|
||||||
public string? Username { get; init; }
|
|
||||||
public string? FullName { get; init; }
|
|
||||||
public string? Email { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Client-side model for a document embedded in <see cref="EnvelopeClientDto"/>.
|
|
||||||
/// </summary>
|
|
||||||
public record DocumentClientDto
|
|
||||||
{
|
|
||||||
public int Id { get; init; }
|
|
||||||
public int EnvelopeId { get; init; }
|
|
||||||
public DateTime AddedWhen { get; init; }
|
|
||||||
public IEnumerable<SignatureClientDto>? Elements { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Client-side model for a signature/annotation element embedded in <see cref="DocumentClientDto"/>.
|
|
||||||
/// </summary>
|
|
||||||
public record SignatureClientDto
|
|
||||||
{
|
|
||||||
public int Id { get; init; }
|
|
||||||
public int DocumentId { get; init; }
|
|
||||||
public int ReceiverId { get; init; }
|
|
||||||
public int ElementType { get; init; }
|
|
||||||
public double X { get; init; }
|
|
||||||
public double Y { get; init; }
|
|
||||||
public double Width { get; init; }
|
|
||||||
public double Height { get; init; }
|
|
||||||
public int Page { get; init; }
|
|
||||||
public bool Required { get; init; }
|
|
||||||
public string? Tooltip { get; init; }
|
|
||||||
public bool ReadOnly { get; init; }
|
|
||||||
public int AnnotationIndex { get; init; }
|
|
||||||
public DateTime AddedWhen { get; init; }
|
|
||||||
public DateTime? ChangedWhen { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Client-side model for the receiver data embedded in <see cref="EnvelopeReceiverDto"/>.
|
|
||||||
/// </summary>
|
|
||||||
public record ReceiverClientDto
|
|
||||||
{
|
|
||||||
public int Id { get; init; }
|
|
||||||
public string? EmailAddress { get; init; }
|
|
||||||
public string? Signature { get; init; }
|
|
||||||
public DateTime AddedWhen { get; init; }
|
|
||||||
public DateTime? TfaRegDeadline { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
The following open source libraries are used and included within thеse sample/demonstration projects:
|
|
||||||
|
|
||||||
Bootstrap (Open Source – MIT License)
|
|
||||||
Copyright (c) 2011-2021 Twitter, Inc. / Copyright (c) 2011-2021 The Bootstrap Authors
|
|
||||||
https://github.com/twbs/bootstrap/blob/main/LICENSE
|
|
||||||
|
|
||||||
open-iconic (Open Source – MIT License and SIL Open Font License)
|
|
||||||
Copyright (c) 2014 Waybury
|
|
||||||
https://github.com/iconic/open-iconic/blob/master/ICON-LICENSE
|
|
||||||
https://github.com/iconic/open-iconic/blob/master/FONT-LICENSE
|
|
||||||
|
|
||||||
The open source libraries included in these sample/demonstration projects are done so pursuant to each individual open source library license and subject to the disclaimers and limitations on liability set forth in each open source library license.
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
namespace EnvelopeGenerator.ReceiverUI.Options;
|
|
||||||
|
|
||||||
public class ApiOptions
|
|
||||||
{
|
|
||||||
public const string SectionName = "Api";
|
|
||||||
|
|
||||||
public string BaseUrl { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
public bool ForceToUseFakeDocument { get; set; } = false;
|
|
||||||
}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
@page "/documentviewer"
|
|
||||||
|
|
||||||
<DxDocumentViewer ReportName="LargeDatasetReport" CssClass="dx-blazor-reporting-container" Height="@null" Width="@null">
|
|
||||||
<DxDocumentViewerTabPanelSettings Width="340" />
|
|
||||||
</DxDocumentViewer>
|
|
||||||
@@ -1,71 +0,0 @@
|
|||||||
@page "/"
|
|
||||||
@inject IJSRuntime JS
|
|
||||||
|
|
||||||
<link href="_content/DevExpress.Blazor.Themes/blazing-berry.bs5.min.css" rel="stylesheet" />
|
|
||||||
|
|
||||||
<div class="home-page-wrapper">
|
|
||||||
|
|
||||||
<div class="home-hero-header">
|
|
||||||
<div class="home-hero-header__inner">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="48" height="48" fill="currentColor" class="home-hero-header__icon" viewBox="0 0 16 16">
|
|
||||||
<path d="M0 4a2 2 0 0 1 2-2h12a2 2 0 0 1 2 2v8a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2V4Zm2-1a1 1 0 0 0-1 1v.217l7 4.2 7-4.2V4a1 1 0 0 0-1-1H2Zm13 2.383-4.708 2.825L15 11.105V5.383Zm-.034 6.876-5.64-3.471L8 9.583l-1.326-.795-5.64 3.47A1 1 0 0 0 2 13h12a1 1 0 0 0 .966-.741ZM1 11.105l4.708-2.897L1 5.383v5.722Z"/>
|
|
||||||
</svg>
|
|
||||||
<div>
|
|
||||||
<h1 class="home-hero-header__title">SignFlow</h1>
|
|
||||||
<p class="home-hero-header__subtitle">Willkommen im eSign-Portal</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="home-content">
|
|
||||||
<div class="home-card card shadow border-0">
|
|
||||||
<div class="card-body p-4 p-md-5">
|
|
||||||
|
|
||||||
<p class="text-muted mb-4" style="font-size: 0.92rem; line-height: 1.7; text-align: justify; text-align-last: left; min-height: calc(0.92rem * 1.7 * 9);">
|
|
||||||
<span id="home-description"></span>
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<div class="mt-4 pt-3 border-top">
|
|
||||||
<div class="d-flex flex-wrap justify-content-center gap-3">
|
|
||||||
<div class="home-feature-badge">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="15" height="15" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M8 1a2 2 0 0 1 2 2v4H6V3a2 2 0 0 1 2-2zm3 6V3a3 3 0 0 0-6 0v4a2 2 0 0 0-2 2v5a2 2 0 0 0 2 2h6a2 2 0 0 0 2-2V9a2 2 0 0 0-2-2z"/>
|
|
||||||
</svg>
|
|
||||||
Sicherer Zugang
|
|
||||||
</div>
|
|
||||||
<div class="home-feature-badge">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="15" height="15" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M13.854 3.646a.5.5 0 0 1 0 .708l-7 7a.5.5 0 0 1-.708 0l-3.5-3.5a.5.5 0 1 1 .708-.708L6.5 10.293l6.646-6.647a.5.5 0 0 1 .708 0z"/>
|
|
||||||
</svg>
|
|
||||||
Digitale Unterschrift
|
|
||||||
</div>
|
|
||||||
<div class="home-feature-badge">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="15" height="15" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/>
|
|
||||||
<path d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/>
|
|
||||||
</svg>
|
|
||||||
PDF-Export
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
private const string HomePageDescription =
|
|
||||||
"Das digitale Unterschriftenportal ist eine Plattform, die entwickelt wurde, um Ihre Dokumente sicher zu unterschreiben und zu verwalten. " +
|
|
||||||
"Mit seiner benutzerfreundlichen Oberfläche können Sie Ihre Dokumente schnell hochladen, die Unterschriftsprozesse verfolgen und Ihre digitalen Unterschriftenanwendungen einfach durchführen. " +
|
|
||||||
"Dieses Portal beschleunigt Ihren Arbeitsablauf mit rechtlich gültigen Unterschriften und erhöht gleichzeitig die Sicherheit Ihrer Dokumente.";
|
|
||||||
|
|
||||||
protected override async Task OnAfterRenderAsync(bool firstRender)
|
|
||||||
{
|
|
||||||
if (firstRender)
|
|
||||||
{
|
|
||||||
await JS.InvokeVoidAsync("receiverSignature.startTyped", "home-description", HomePageDescription, 15);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,158 +0,0 @@
|
|||||||
@page "/login/{EnvelopeKey}"
|
|
||||||
@using EnvelopeGenerator.ReceiverUI.Services
|
|
||||||
@inject AuthService AuthService
|
|
||||||
@inject NavigationManager Navigation
|
|
||||||
|
|
||||||
<link href="_content/DevExpress.Blazor.Themes/blazing-berry.bs5.min.css" rel="stylesheet" />
|
|
||||||
|
|
||||||
<div class="login-page-wrapper d-flex align-items-center justify-content-center min-vh-100">
|
|
||||||
<div class="login-card card shadow border-0" style="max-width: 440px; width: 100%;">
|
|
||||||
|
|
||||||
<div class="card-header text-white text-center py-4 border-0" style="background: linear-gradient(135deg, #2c3e50 0%, #3498db 100%); border-radius: calc(0.375rem - 1px) calc(0.375rem - 1px) 0 0;">
|
|
||||||
<div class="mb-2">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="40" height="40" fill="currentColor" viewBox="0 0 16 16">
|
|
||||||
<path d="M8 1a2 2 0 0 1 2 2v4H6V3a2 2 0 0 1 2-2zm3 6V3a3 3 0 0 0-6 0v4a2 2 0 0 0-2 2v5a2 2 0 0 0 2 2h6a2 2 0 0 0 2-2V9a2 2 0 0 0-2-2z"/>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
<h5 class="mb-0 fw-semibold">Dokument öffnen</h5>
|
|
||||||
<p class="mb-0 mt-1 opacity-75" style="font-size: 0.85rem;">Sicherer Zugang mit Zugangscode</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="card-body p-4">
|
|
||||||
|
|
||||||
<p class="text-muted mb-4" style="font-size: 0.875rem; line-height: 1.5;">
|
|
||||||
Bitte geben Sie den Zugangscode ein, den Sie per E-Mail erhalten haben, um das Dokument sicher zu öffnen.
|
|
||||||
</p>
|
|
||||||
|
|
||||||
@if (LoginResult == EnvelopeLoginResult.NotFound) {
|
|
||||||
<div class="alert alert-warning d-flex align-items-start gap-2 py-2" role="alert">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" fill="currentColor" class="flex-shrink-0 mt-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M8.982 1.566a1.13 1.13 0 0 0-1.96 0L.165 13.233c-.457.778.091 1.767.98 1.767h13.713c.889 0 1.438-.99.98-1.767L8.982 1.566zM8 5c.535 0 .954.462.9.995l-.35 3.507a.552.552 0 0 1-1.1 0L7.1 5.995A.905.905 0 0 1 8 5zm.002 6a1 1 0 1 1 0 2 1 1 0 0 1 0-2z"/>
|
|
||||||
</svg>
|
|
||||||
<div>
|
|
||||||
<strong>Dokument nicht gefunden.</strong><br />
|
|
||||||
<span style="font-size:0.85rem;">Der angegebene Zugangscode konnte keinem Dokument zugeordnet werden. Bitte prüfen Sie den Link in Ihrer E-Mail.</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
} else if (LoginResult == EnvelopeLoginResult.InvalidCode) {
|
|
||||||
<div class="alert alert-danger d-flex align-items-start gap-2 py-2" role="alert">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" fill="currentColor" class="flex-shrink-0 mt-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M16 8A8 8 0 1 1 0 8a8 8 0 0 1 16 0zM5.354 4.646a.5.5 0 1 0-.708.708L7.293 8l-2.647 2.646a.5.5 0 0 0 .708.708L8 8.707l2.646 2.647a.5.5 0 0 0 .708-.708L8.707 8l2.647-2.646a.5.5 0 0 0-.708-.708L8 7.293 5.354 4.646z"/>
|
|
||||||
</svg>
|
|
||||||
<div>
|
|
||||||
<strong>Ungültiger Zugangscode.</strong><br />
|
|
||||||
<span style="font-size:0.85rem;">Der eingegebene Code ist falsch. Bitte versuchen Sie es erneut.</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
} else if (LoginResult == EnvelopeLoginResult.Error) {
|
|
||||||
<div class="alert alert-secondary d-flex align-items-start gap-2 py-2" role="alert">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="18" height="18" fill="currentColor" class="flex-shrink-0 mt-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M8 15A7 7 0 1 1 8 1a7 7 0 0 1 0 14zm0 1A8 8 0 1 0 8 0a8 8 0 0 0 0 16z"/>
|
|
||||||
<path d="M7.002 11a1 1 0 1 1 2 0 1 1 0 0 1-2 0zM7.1 4.995a.905.905 0 1 1 1.8 0l-.35 3.507a.552.552 0 0 1-1.1 0L7.1 4.995z"/>
|
|
||||||
</svg>
|
|
||||||
<div>
|
|
||||||
<strong>Serverfehler.</strong><br />
|
|
||||||
<span style="font-size:0.85rem;">Ein unerwarteter Fehler ist aufgetreten. Bitte versuchen Sie es später erneut.</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
|
|
||||||
<div class="mb-4">
|
|
||||||
<label class="form-label fw-medium" for="login-access-code">
|
|
||||||
Zugangscode
|
|
||||||
<span class="text-danger ms-1">*</span>
|
|
||||||
</label>
|
|
||||||
<div class="input-group">
|
|
||||||
<span class="input-group-text bg-light border-end-0">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="#6c757d" viewBox="0 0 16 16">
|
|
||||||
<path d="M3.5 11.5a3.5 3.5 0 1 1 3.163-5H14L15.5 8 14 9.5l-1-1-1 1-1-1-1 1-1-1-1.837 1.337A3.5 3.5 0 0 1 3.5 11.5zm0-1a2.5 2.5 0 1 0 0-5 2.5 2.5 0 0 0 0 5z"/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
<input id="login-access-code"
|
|
||||||
type="@(ShowCode ? "text" : "password")"
|
|
||||||
class="form-control border-start-0 border-end-0 @(LoginResult == EnvelopeLoginResult.InvalidCode ? "is-invalid" : null)"
|
|
||||||
placeholder="Zugangscode eingeben"
|
|
||||||
@bind="AccessCode"
|
|
||||||
@bind:event="oninput"
|
|
||||||
@onkeydown="OnKeyDownAsync"
|
|
||||||
disabled="@IsLoading"
|
|
||||||
autocomplete="one-time-code" />
|
|
||||||
<button type="button"
|
|
||||||
class="btn btn-outline-secondary border-start-0"
|
|
||||||
style="border-left: none;"
|
|
||||||
tabindex="-1"
|
|
||||||
@onclick="() => ShowCode = !ShowCode">
|
|
||||||
@if (ShowCode) {
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" viewBox="0 0 16 16">
|
|
||||||
<path d="M13.359 11.238C15.06 9.72 16 8 16 8s-3-5.5-8-5.5a7.028 7.028 0 0 0-2.79.588l.77.771A5.944 5.944 0 0 1 8 3.5c2.12 0 3.879 1.168 5.168 2.457A13.134 13.134 0 0 1 14.828 8c-.058.087-.122.183-.195.288-.335.48-.83 1.12-1.465 1.755-.165.165-.337.328-.517.486l.708.709z"/>
|
|
||||||
<path d="M11.297 9.176a3.5 3.5 0 0 0-4.474-4.474l.823.823a2.5 2.5 0 0 1 2.829 2.829l.822.822zm-2.943 1.299.822.822a3.5 3.5 0 0 1-4.474-4.474l.823.823a2.5 2.5 0 0 0 2.829 2.829z"/>
|
|
||||||
<path d="M3.35 5.47c-.18.16-.353.322-.518.487A13.134 13.134 0 0 0 1.172 8l.195.288c.335.48.83 1.12 1.465 1.755C4.121 11.332 5.881 12.5 8 12.5c.716 0 1.39-.133 2.02-.36l.77.772A7.029 7.029 0 0 1 8 13.5C3 13.5 0 8 0 8s.939-1.721 2.641-3.238l.708.709z"/>
|
|
||||||
<path fill-rule="evenodd" d="M13.646 14.354l-12-12 .708-.708 12 12-.708.708z"/>
|
|
||||||
</svg>
|
|
||||||
} else {
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" viewBox="0 0 16 16">
|
|
||||||
<path d="M16 8s-3-5.5-8-5.5S0 8 0 8s3 5.5 8 5.5S16 8 16 8zM1.173 8a13.133 13.133 0 0 1 1.66-2.043C4.12 4.668 5.88 3.5 8 3.5c2.12 0 3.879 1.168 5.168 2.457A13.133 13.133 0 0 1 14.828 8c-.058.087-.122.183-.195.288-.335.48-.83 1.12-1.465 1.755C11.879 11.332 10.119 12.5 8 12.5c-2.12 0-3.879-1.168-5.168-2.457A13.134 13.134 0 0 1 1.172 8z"/>
|
|
||||||
<path d="M8 5.5a2.5 2.5 0 1 0 0 5 2.5 2.5 0 0 0 0-5zM4.5 8a3.5 3.5 0 1 1 7 0 3.5 3.5 0 0 1-7 0z"/>
|
|
||||||
</svg>
|
|
||||||
}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<button class="btn btn-primary w-100 py-2 fw-medium"
|
|
||||||
style="background: linear-gradient(135deg, #2c3e50 0%, #3498db 100%); border: none;"
|
|
||||||
@onclick="SubmitAsync"
|
|
||||||
disabled="@(IsLoading || string.IsNullOrWhiteSpace(AccessCode))">
|
|
||||||
@if (IsLoading) {
|
|
||||||
<span class="spinner-border spinner-border-sm me-2" role="status" aria-hidden="true"></span>
|
|
||||||
<span>Überprüfen …</span>
|
|
||||||
} else {
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" fill="currentColor" class="me-2" viewBox="0 0 16 16">
|
|
||||||
<path d="M8 1a2 2 0 0 1 2 2v4H6V3a2 2 0 0 1 2-2zm3 6V3a3 3 0 0 0-6 0v4a2 2 0 0 0-2 2v5a2 2 0 0 0 2 2h6a2 2 0 0 0 2-2V9a2 2 0 0 0-2-2z"/>
|
|
||||||
</svg>
|
|
||||||
<span>Dokument öffnen</span>
|
|
||||||
}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="card-footer text-center text-muted py-3 border-0 bg-transparent" style="font-size: 0.78rem;">
|
|
||||||
Bei Problemen wenden Sie sich bitte an den Absender des Dokuments.
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
[Parameter] public string EnvelopeKey { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
string AccessCode = string.Empty;
|
|
||||||
bool ShowCode;
|
|
||||||
bool IsLoading;
|
|
||||||
EnvelopeLoginResult? LoginResult;
|
|
||||||
|
|
||||||
async Task OnKeyDownAsync(Microsoft.AspNetCore.Components.Web.KeyboardEventArgs e) {
|
|
||||||
if (e.Key == "Enter")
|
|
||||||
await SubmitAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task SubmitAsync() {
|
|
||||||
if (string.IsNullOrWhiteSpace(AccessCode) || IsLoading) return;
|
|
||||||
|
|
||||||
IsLoading = true;
|
|
||||||
LoginResult = null;
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
|
|
||||||
var result = await AuthService.LoginEnvelopeReceiverAsync(EnvelopeKey, AccessCode.Trim());
|
|
||||||
|
|
||||||
if (result == EnvelopeLoginResult.Success) {
|
|
||||||
Navigation.NavigateTo($"/receiver/{Uri.EscapeDataString(EnvelopeKey)}", forceLoad: true);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
LoginResult = result;
|
|
||||||
IsLoading = false;
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
@page "/sender"
|
|
||||||
@using DevExpress.DataAccess.Json;
|
|
||||||
@using EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
|
|
||||||
<DxReportDesigner ReportName="LargeDatasetReport" CssClass="dx-blazor-reporting-container" Height="@null" Width="@null" AllowMDI="true" DataSources="DataSources">
|
|
||||||
<DxReportDesignerWizardSettings UseFullscreenWizard="true" />
|
|
||||||
</DxReportDesigner>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
Dictionary<string, object> DataSources = new Dictionary<string, object>();
|
|
||||||
protected override async Task OnInitializedAsync() {
|
|
||||||
await base.OnInitializedAsync();
|
|
||||||
var connection = CustomDataSourceWizardJsonDataConnectionStorage.GetDefaultConnection();
|
|
||||||
JsonDataSource jsonDataSource = new JsonDataSource();
|
|
||||||
jsonDataSource.JsonSource = connection.GetJsonSource();
|
|
||||||
await jsonDataSource.FillAsync();
|
|
||||||
DataSources.Add("Northwind", jsonDataSource);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,728 +0,0 @@
|
|||||||
@page "/receiver/{EnvelopeKey}"
|
|
||||||
@using System.Drawing
|
|
||||||
@using DevExpress.Blazor
|
|
||||||
@using DevExpress.Drawing
|
|
||||||
@using DevExpress.Utils
|
|
||||||
@using DevExpress.XtraPrinting
|
|
||||||
@using DevExpress.XtraPrinting.Drawing
|
|
||||||
@using Microsoft.JSInterop
|
|
||||||
@using XtraReport = DevExpress.XtraReports.UI.XtraReport
|
|
||||||
@using BottomMarginBand = DevExpress.XtraReports.UI.BottomMarginBand
|
|
||||||
@using XRLabel = DevExpress.XtraReports.UI.XRLabel
|
|
||||||
@using XRPictureBox = DevExpress.XtraReports.UI.XRPictureBox
|
|
||||||
@using XRControl = DevExpress.XtraReports.UI.XRControl
|
|
||||||
@using ImageSizeMode = DevExpress.XtraPrinting.ImageSizeMode
|
|
||||||
@using EnvelopeGenerator.ReceiverUI.Services
|
|
||||||
@using DevExpress.Blazor.Reporting
|
|
||||||
@using Microsoft.Extensions.Options
|
|
||||||
@using EnvelopeGenerator.ReceiverUI.Options
|
|
||||||
@using EnvelopeGenerator.ReceiverUI.Models
|
|
||||||
@implements IDisposable
|
|
||||||
@inject IJSRuntime JSRuntime
|
|
||||||
@inject AnnotationService AnnotationService
|
|
||||||
@inject IOptions<ApiOptions> AppOptions
|
|
||||||
@inject NavigationManager Navigation
|
|
||||||
@inject InMemoryReportStorageWebExtension ReportStorage
|
|
||||||
@inject EnvelopeGenerator.ReceiverUI.Services.DocumentService DocumentService
|
|
||||||
@inject EnvelopeGenerator.ReceiverUI.Services.AuthService AuthService
|
|
||||||
@inject EnvelopeGenerator.ReceiverUI.Services.EnvelopeReceiverService EnvelopeReceiverService
|
|
||||||
|
|
||||||
<link href="_content/DevExpress.Blazor.Themes/blazing-berry.bs5.min.css" rel="stylesheet" />
|
|
||||||
<link href="_content/DevExpress.Blazor.Reporting.Viewer/css/dx-blazor-reporting-components.bs5.css" rel="stylesheet" />
|
|
||||||
|
|
||||||
<div class="receiver-page-layout">
|
|
||||||
|
|
||||||
<div class="receiver-signature-panel">
|
|
||||||
|
|
||||||
@* ?? Envelope info header ???????????????????????????????????????????????? *@
|
|
||||||
@if (_envelopeReceiver is not null) {
|
|
||||||
<div class="receiver-info-header">
|
|
||||||
<div class="receiver-info-header__gradient">
|
|
||||||
<div class="receiver-info-header__left">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="22" height="22" fill="currentColor" class="receiver-info-header__icon" viewBox="0 0 16 16">
|
|
||||||
<path d="M0 4a2 2 0 0 1 2-2h12a2 2 0 0 1 2 2v8a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2V4Zm2-1a1 1 0 0 0-1 1v.217l7 4.2 7-4.2V4a1 1 0 0 0-1-1H2Zm13 2.383-4.708 2.825L15 11.105V5.383Zm-.034 6.876-5.64-3.471L8 9.583l-1.326-.795-5.64 3.47A1 1 0 0 0 2 13h12a1 1 0 0 0 .966-.741ZM1 11.105l4.708-2.897L1 5.383v5.722Z"/>
|
|
||||||
</svg>
|
|
||||||
<div>
|
|
||||||
<div class="receiver-info-header__title">@(_envelopeReceiver.Envelope?.Title ?? "Dokument")</div>
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.Envelope?.User?.FullName) || !string.IsNullOrWhiteSpace(_envelopeReceiver.Envelope?.User?.Email)) {
|
|
||||||
<div class="receiver-info-header__sender">
|
|
||||||
Von
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.Envelope?.User?.FullName)) {
|
|
||||||
<strong>@_envelopeReceiver.Envelope!.User!.FullName</strong>
|
|
||||||
}
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.Envelope?.User?.Email)) {
|
|
||||||
<span class="opacity-75"><@_envelopeReceiver.Envelope!.User!.Email></span>
|
|
||||||
}
|
|
||||||
· @_envelopeReceiver.Envelope?.AddedWhen.ToString("dd.MM.yyyy")
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="receiver-info-header__badges">
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.Name)) {
|
|
||||||
<span class="receiver-info-badge">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M8 8a3 3 0 1 0 0-6 3 3 0 0 0 0 6Zm2-3a2 2 0 1 1-4 0 2 2 0 0 1 4 0Zm4 8c0 1-1 1-1 1H3s-1 0-1-1 1-4 6-4 6 3 6 4Z"/>
|
|
||||||
</svg>
|
|
||||||
@_envelopeReceiver.Name
|
|
||||||
</span>
|
|
||||||
}
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.CompanyName)) {
|
|
||||||
<span class="receiver-info-badge">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M14.763.075A.5.5 0 0 1 15 .5v15a.5.5 0 0 1-.5.5h-3a.5.5 0 0 1-.5-.5V14h-1v1.5a.5.5 0 0 1-.5.5h-9a.5.5 0 0 1-.5-.5V10a.5.5 0 0 1 .342-.474L6 7.64V4.5a.5.5 0 0 1 .276-.447l8-4a.5.5 0 0 1 .487.022Z"/>
|
|
||||||
</svg>
|
|
||||||
@_envelopeReceiver.CompanyName
|
|
||||||
</span>
|
|
||||||
}
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.JobTitle)) {
|
|
||||||
<span class="receiver-info-badge receiver-info-badge--muted">@_envelopeReceiver.JobTitle</span>
|
|
||||||
}
|
|
||||||
@{
|
|
||||||
var docElements = _envelopeReceiver.Envelope?.Documents?.FirstOrDefault()?.Elements;
|
|
||||||
int sigCount = docElements?.Count() ?? _annotations.Count;
|
|
||||||
}
|
|
||||||
@if (sigCount > 0) {
|
|
||||||
<span class="receiver-info-badge receiver-info-badge--accent">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168l10-10zM11.207 2.5 13.5 4.793 14.793 3.5 12.5 1.207 11.207 2.5zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293l6.5-6.5zm-9.761 5.175-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325z"/>
|
|
||||||
</svg>
|
|
||||||
@sigCount @(sigCount == 1 ? "Unterschriftsfeld" : "Unterschriftsfelder")
|
|
||||||
</span>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.Envelope?.Message)) {
|
|
||||||
<div class="receiver-info-message">@_envelopeReceiver.Envelope!.Message</div>
|
|
||||||
}
|
|
||||||
@if (!string.IsNullOrWhiteSpace(_envelopeReceiver.PrivateMessage)) {
|
|
||||||
<div class="receiver-info-private-message">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" fill="currentColor" class="me-1 flex-shrink-0" viewBox="0 0 16 16">
|
|
||||||
<path d="M8 1a2 2 0 0 1 2 2v4H6V3a2 2 0 0 1 2-2zm3 6V3a3 3 0 0 0-6 0v4a2 2 0 0 0-2 2v5a2 2 0 0 0 2 2h6a2 2 0 0 0 2-2V9a2 2 0 0 0-2-2z"/>
|
|
||||||
</svg>
|
|
||||||
@_envelopeReceiver.PrivateMessage
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
|
|
||||||
@* ?? Signature action bar ???????????????????????????????????????????????? *@
|
|
||||||
<div class="receiver-action-bar">
|
|
||||||
<div class="receiver-action-bar__inner">
|
|
||||||
<button class="btn btn-sm @(_capturedSignature is not null ? "btn-outline-success" : "btn-primary")" @onclick="OpenSignaturePopupAsync">
|
|
||||||
@if (_capturedSignature is not null) {
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M13.854 3.646a.5.5 0 0 1 0 .708l-7 7a.5.5 0 0 1-.708 0l-3.5-3.5a.5.5 0 1 1 .708-.708L6.5 10.293l6.646-6.647a.5.5 0 0 1 .708 0z"/>
|
|
||||||
</svg>
|
|
||||||
<span>Unterschrift gespeichert</span>
|
|
||||||
} else {
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M12.146.146a.5.5 0 0 1 .708 0l3 3a.5.5 0 0 1 0 .708l-10 10a.5.5 0 0 1-.168.11l-5 2a.5.5 0 0 1-.65-.65l2-5a.5.5 0 0 1 .11-.168l10-10zM11.207 2.5 13.5 4.793 14.793 3.5 12.5 1.207 11.207 2.5zm1.586 3L10.5 3.207 4 9.707V10h.5a.5.5 0 0 1 .5.5v.5h.5a.5.5 0 0 1 .5.5v.5h.293l6.5-6.5zm-9.761 5.175-.106.106-1.528 3.821 3.821-1.528.106-.106A.5.5 0 0 1 5 12.5V12h-.5a.5.5 0 0 1-.5-.5V11h-.5a.5.5 0 0 1-.468-.325z"/>
|
|
||||||
</svg>
|
|
||||||
<span>Unterschrift erstellen</span>
|
|
||||||
}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
@if (_annotations.Count > 0 && !SignatureApplied) {
|
|
||||||
<div class="receiver-action-bar__progress">
|
|
||||||
<div class="progress" style="height:5px; min-width:70px; width:90px;">
|
|
||||||
<div class="progress-bar @(_checkedAnnotations.Count == _annotations.Count ? "bg-success" : "bg-primary")"
|
|
||||||
style="width:@(_annotations.Count > 0 ? (_checkedAnnotations.Count * 100 / _annotations.Count) : 0)%"
|
|
||||||
role="progressbar"></div>
|
|
||||||
</div>
|
|
||||||
<span class="text-muted" style="font-size:0.75rem;">
|
|
||||||
@_checkedAnnotations.Count / @_annotations.Count
|
|
||||||
Seite@(AnnotationPages.Count() == 1 ? "" : "n") @string.Join(",", AnnotationPages)
|
|
||||||
</span>
|
|
||||||
@if (_capturedSignature is null) {
|
|
||||||
<span class="text-muted fst-italic" style="font-size:0.72rem;">Zuerst Unterschrift erstellen</span>
|
|
||||||
} else if (_checkedAnnotations.Count == _annotations.Count) {
|
|
||||||
<span class="text-success fw-semibold" style="font-size:0.72rem;">✓ Wird angewendet…</span>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
}
|
|
||||||
|
|
||||||
@if (!string.IsNullOrWhiteSpace(SignatureValidationMessage)) {
|
|
||||||
<span class="text-danger" style="font-size:0.78rem;">@SignatureValidationMessage</span>
|
|
||||||
}
|
|
||||||
@if (SignatureApplied) {
|
|
||||||
<span class="text-success" style="font-size:0.78rem;">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M16 8A8 8 0 1 1 0 8a8 8 0 0 1 16 0zm-3.97-3.03a.75.75 0 0 0-1.08.022L7.477 9.417 5.384 7.323a.75.75 0 0 0-1.06 1.06L6.97 11.03a.75.75 0 0 0 1.079-.02l3.992-4.99a.75.75 0 0 0-.01-1.05z"/>
|
|
||||||
</svg>
|
|
||||||
Unterschrift angewendet
|
|
||||||
</span>
|
|
||||||
}
|
|
||||||
|
|
||||||
<div class="ms-auto d-flex align-items-center gap-2">
|
|
||||||
<button class="btn btn-sm btn-success" disabled="@(!SignatureApplied)" @onclick="ExportSignedPdfAsync">
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="13" height="13" fill="currentColor" class="me-1" viewBox="0 0 16 16">
|
|
||||||
<path d="M.5 9.9a.5.5 0 0 1 .5.5v2.5a1 1 0 0 0 1 1h12a1 1 0 0 0 1-1v-2.5a.5.5 0 0 1 1 0v2.5a2 2 0 0 1-2 2H2a2 2 0 0 1-2-2v-2.5a.5.5 0 0 1 .5-.5z"/>
|
|
||||||
<path d="M7.646 11.854a.5.5 0 0 0 .708 0l3-3a.5.5 0 0 0-.708-.708L8.5 10.293V1.5a.5.5 0 0 0-1 0v8.793L5.354 8.146a.5.5 0 1 0-.708.708l3 3z"/>
|
|
||||||
</svg>
|
|
||||||
PDF exportieren
|
|
||||||
</button>
|
|
||||||
@if (!string.IsNullOrWhiteSpace(EnvelopeKey)) {
|
|
||||||
<button class="btn btn-sm btn-outline-danger" @onclick="LogoutAsync" disabled="@IsLoggingOut" title="Abmelden">
|
|
||||||
@if (IsLoggingOut) {
|
|
||||||
<span class="spinner-border spinner-border-sm" role="status" aria-hidden="true"></span>
|
|
||||||
} else {
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" fill="currentColor" viewBox="0 0 16 16">
|
|
||||||
<path fill-rule="evenodd" d="M10 12.5a.5.5 0 0 1-.5.5h-8a.5.5 0 0 1-.5-.5v-9a.5.5 0 0 1 .5-.5h8a.5.5 0 0 1 .5.5v2a.5.5 0 0 0 1 0v-2A1.5 1.5 0 0 0 9.5 2h-8A1.5 1.5 0 0 0 0 3.5v9A1.5 1.5 0 0 0 1.5 14h8a1.5 1.5 0 0 0 1.5-1.5v-2a.5.5 0 0 0-1 0v2z"/>
|
|
||||||
<path fill-rule="evenodd" d="M15.854 8.354a.5.5 0 0 0 0-.708l-3-3a.5.5 0 0 0-.708.708L14.293 7.5H5.5a.5.5 0 0 0 0 1h8.793l-2.147 2.146a.5.5 0 0 0 .708.708l3-3z"/>
|
|
||||||
</svg>
|
|
||||||
}
|
|
||||||
</button>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<DxPopup @bind-Visible="SignaturePopupVisible"
|
|
||||||
HeaderText="Unterschrift erfassen"
|
|
||||||
Width="620px"
|
|
||||||
ShowFooter="true"
|
|
||||||
CloseOnEscape="false"
|
|
||||||
ShowCloseButton="false"
|
|
||||||
CloseOnOutsideClick="false"
|
|
||||||
Shown="OnPopupShownAsync">
|
|
||||||
<BodyContentTemplate>
|
|
||||||
<ul class="nav nav-tabs mb-3">
|
|
||||||
<li class="nav-item">
|
|
||||||
<button type="button" class="nav-link @(ActiveSignatureTab == SignatureTabDraw ? "active" : null)" @onclick="() => SetSignatureTabAsync(SignatureTabDraw)">Zeichnen</button>
|
|
||||||
</li>
|
|
||||||
<li class="nav-item">
|
|
||||||
<button type="button" class="nav-link @(ActiveSignatureTab == SignatureTabText ? "active" : null)" @onclick="() => SetSignatureTabAsync(SignatureTabText)">Text</button>
|
|
||||||
</li>
|
|
||||||
<li class="nav-item">
|
|
||||||
<button type="button" class="nav-link @(ActiveSignatureTab == SignatureTabImage ? "active" : null)" @onclick="() => SetSignatureTabAsync(SignatureTabImage)">Bild</button>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
@if(ActiveSignatureTab == SignatureTabDraw) {
|
|
||||||
<p class="text-muted mb-2">Bitte unterschreiben Sie im folgenden Feld.</p>
|
|
||||||
<canvas id="receiver-signature-pad" width="520" height="160" class="border rounded bg-white w-100" style="max-width: 520px; touch-action: none;"></canvas>
|
|
||||||
} else if(ActiveSignatureTab == SignatureTabText) {
|
|
||||||
<p class="text-muted mb-2">Geben Sie Ihre Unterschrift als Text ein und waehlen Sie eine Schriftart.</p>
|
|
||||||
<div class="row g-2 mb-2">
|
|
||||||
<div class="col-12 col-md-7">
|
|
||||||
<input class="form-control" placeholder="Ihre Unterschrift" value="@TypedSignatureText" @oninput="OnTypedSignatureChanged" />
|
|
||||||
</div>
|
|
||||||
<div class="col-12 col-md-5">
|
|
||||||
<select class="form-select" value="@TypedSignatureFont" @onchange="OnTypedSignatureFontChanged">
|
|
||||||
@foreach(var font in TypedSignatureFonts) {
|
|
||||||
<option value="@font.Value" style="font-family: @font.Value">@font.Text</option>
|
|
||||||
}
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<canvas id="receiver-typed-signature-pad" width="520" height="160" class="border rounded bg-white w-100" style="max-width: 520px;"></canvas>
|
|
||||||
} else {
|
|
||||||
<p class="text-muted mb-2">Laden Sie ein Bild Ihrer Unterschrift hoch.</p>
|
|
||||||
<input id="receiver-signature-image-input" class="form-control mb-2" type="file" accept="image/png,image/jpeg,image/webp" />
|
|
||||||
<canvas id="receiver-image-signature-pad" width="520" height="160" class="border rounded bg-white w-100" style="max-width: 520px;"></canvas>
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
<div class="border-top mt-3 pt-3">
|
|
||||||
<p class="text-muted mb-2">Bitte geben Sie die folgenden Angaben ein. Das Datum wird automatisch hinzugefuegt.</p>
|
|
||||||
<div class="row g-2">
|
|
||||||
<div class="col-12 col-md-6">
|
|
||||||
<label class="form-label" for="receiver-signer-name">Vor- und Nachname *</label>
|
|
||||||
<input id="receiver-signer-name" class="form-control" value="@SignerFullName" @oninput="args => SignerFullName = args.Value?.ToString() ?? string.Empty" />
|
|
||||||
</div>
|
|
||||||
<div class="col-12 col-md-6">
|
|
||||||
<label class="form-label" for="receiver-signer-position">Position</label>
|
|
||||||
<input id="receiver-signer-position" class="form-control" value="@SignerPosition" @oninput="args => SignerPosition = args.Value?.ToString() ?? string.Empty" />
|
|
||||||
</div>
|
|
||||||
<div class="col-12 col-md-6">
|
|
||||||
<label class="form-label" for="receiver-signature-place">Ort *</label>
|
|
||||||
<input id="receiver-signature-place" class="form-control" value="@SignaturePlace" @oninput="args => SignaturePlace = args.Value?.ToString() ?? string.Empty" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@if(!string.IsNullOrWhiteSpace(PopupValidationMessage)) {
|
|
||||||
<div class="text-danger mt-2">@PopupValidationMessage</div>
|
|
||||||
}
|
|
||||||
</BodyContentTemplate>
|
|
||||||
<FooterContentTemplate>
|
|
||||||
<div class="d-flex gap-2 flex-wrap justify-content-end w-100">
|
|
||||||
<button class="btn btn-outline-secondary" @onclick="RenewSignatureAsync">Unterschrift erneuern</button>
|
|
||||||
<button class="btn btn-primary" @onclick="SaveSignatureAsync">Speichern</button>
|
|
||||||
</div>
|
|
||||||
</FooterContentTemplate>
|
|
||||||
</DxPopup>
|
|
||||||
|
|
||||||
<div class="receiver-viewer-wrapper">
|
|
||||||
@if(Report is not null) {
|
|
||||||
<DxReportViewer @key="ViewerKey" @ref="reportViewer" Report="Report" RootCssClasses="w-100 h-100" Zoom="1.3" />
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
|
|
||||||
const string SignatureTabDraw = "draw";
|
|
||||||
const string SignatureTabText = "text";
|
|
||||||
const string SignatureTabImage = "image";
|
|
||||||
const string DrawCanvasId = "receiver-signature-pad";
|
|
||||||
const string TypedCanvasId = "receiver-typed-signature-pad";
|
|
||||||
const string ImageInputId = "receiver-signature-image-input";
|
|
||||||
const string ImageCanvasId = "receiver-image-signature-pad";
|
|
||||||
|
|
||||||
readonly (string Text, string Value)[] TypedSignatureFonts = {
|
|
||||||
("Brush Script", "'Brush Script MT', cursive"),
|
|
||||||
("Segoe Script", "'Segoe Script', cursive"),
|
|
||||||
("Lucida Handwriting", "'Lucida Handwriting', cursive"),
|
|
||||||
("Comic Sans", "'Comic Sans MS', cursive"),
|
|
||||||
("Cursive", "cursive")
|
|
||||||
};
|
|
||||||
|
|
||||||
[Parameter] public string EnvelopeKey { get; set; } = string.Empty;
|
|
||||||
|
|
||||||
DxReportViewer? reportViewer;
|
|
||||||
XtraReport? Report;
|
|
||||||
bool SignatureApplied;
|
|
||||||
bool SignaturePopupVisible;
|
|
||||||
string? SignatureValidationMessage;
|
|
||||||
string? PopupValidationMessage;
|
|
||||||
string ActiveSignatureTab = SignatureTabDraw;
|
|
||||||
string TypedSignatureText = string.Empty;
|
|
||||||
string TypedSignatureFont = "'Brush Script MT', cursive";
|
|
||||||
string SignerFullName = string.Empty;
|
|
||||||
string SignerPosition = string.Empty;
|
|
||||||
string SignaturePlace = string.Empty;
|
|
||||||
int ViewerKey;
|
|
||||||
bool IsLoggingOut;
|
|
||||||
|
|
||||||
IReadOnlyList<AnnotationDto> _annotations = [];
|
|
||||||
IEnumerable<int> AnnotationPages => _annotations.Select(a => a.Page).Distinct().OrderBy(p => p);
|
|
||||||
EnvelopeReceiverDto? _envelopeReceiver;
|
|
||||||
record SignatureCapture(string DataUrl, string FullName, string Position, string Place);
|
|
||||||
SignatureCapture? _capturedSignature;
|
|
||||||
byte[]? _basePdfBytes;
|
|
||||||
// annotation IDs the user has checked via overlay checkboxes
|
|
||||||
readonly HashSet<long> _checkedAnnotations = [];
|
|
||||||
DotNetObjectReference<ReportViewer>? _dotNetRef;
|
|
||||||
int _lastOverlayViewerKey = -1;
|
|
||||||
|
|
||||||
async Task LogoutAsync() {
|
|
||||||
if (string.IsNullOrWhiteSpace(EnvelopeKey) || IsLoggingOut) return;
|
|
||||||
IsLoggingOut = true;
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
await AuthService.LogoutEnvelopeReceiverAsync(EnvelopeKey);
|
|
||||||
Navigation.NavigateTo($"/login/{Uri.EscapeDataString(EnvelopeKey)}", forceLoad: true);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync() {
|
|
||||||
|
|
||||||
if (!string.IsNullOrWhiteSpace(EnvelopeKey)) {
|
|
||||||
var hasAccess = await AuthService.CheckEnvelopeAccessAsync(EnvelopeKey);
|
|
||||||
if (!hasAccess) {
|
|
||||||
Navigation.NavigateTo($"/login/{Uri.EscapeDataString(EnvelopeKey)}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
ActiveSignatureTab = SignatureTabDraw;
|
|
||||||
SignaturePopupVisible = true;
|
|
||||||
SignatureValidationMessage = null;
|
|
||||||
PopupValidationMessage = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_annotations = await AnnotationService.GetAnnotationsAsync(EnvelopeKey ?? "fake");
|
|
||||||
_envelopeReceiver = await EnvelopeReceiverService.GetAsync(EnvelopeKey ?? "fake");
|
|
||||||
|
|
||||||
if (!AppOptions.Value.ForceToUseFakeDocument && !string.IsNullOrWhiteSpace(EnvelopeKey)) {
|
|
||||||
var (pdfBytes, _) = await DocumentService.GetDocumentAsync(EnvelopeKey);
|
|
||||||
if (pdfBytes is { Length: > 0 })
|
|
||||||
_basePdfBytes = pdfBytes;
|
|
||||||
}
|
|
||||||
|
|
||||||
var initialReport = BuildFreshBaseReport();
|
|
||||||
Report = initialReport;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override async Task OnAfterRenderAsync(bool firstRender) {
|
|
||||||
if (firstRender)
|
|
||||||
_dotNetRef = DotNetObjectReference.Create(this);
|
|
||||||
|
|
||||||
|
|
||||||
if (Report is not null && _annotations.Count > 0
|
|
||||||
&& _capturedSignature is not null && !SignatureApplied
|
|
||||||
&& _lastOverlayViewerKey != ViewerKey) {
|
|
||||||
_lastOverlayViewerKey = ViewerKey;
|
|
||||||
await JSRuntime.InvokeVoidAsync(
|
|
||||||
"receiverSignature.installAnnotationCheckboxes",
|
|
||||||
_annotations, _checkedAnnotations.ToArray(), _dotNetRef);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[JSInvokable]
|
|
||||||
public async Task OnAnnotationToggled(long annotationId, bool isChecked) {
|
|
||||||
if (isChecked)
|
|
||||||
_checkedAnnotations.Add(annotationId);
|
|
||||||
else
|
|
||||||
_checkedAnnotations.Remove(annotationId);
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
|
|
||||||
if (_capturedSignature is not null
|
|
||||||
&& !SignatureApplied
|
|
||||||
&& _annotations.Count > 0
|
|
||||||
&& _checkedAnnotations.Count == _annotations.Count) {
|
|
||||||
// K?sa bekleme: kullan?c? son tick'in görsel feedback'ini görsün
|
|
||||||
await Task.Delay(400);
|
|
||||||
await SubmitSignaturesAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void OpenSignaturePopupAsync() {
|
|
||||||
ActiveSignatureTab = SignatureTabDraw;
|
|
||||||
SignaturePopupVisible = true;
|
|
||||||
SignatureValidationMessage = null;
|
|
||||||
PopupValidationMessage = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task OnPopupShownAsync() {
|
|
||||||
await InitializeActiveSignatureTabAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task SetSignatureTabAsync(string tab) {
|
|
||||||
ActiveSignatureTab = tab;
|
|
||||||
PopupValidationMessage = null;
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
await Task.Delay(50);
|
|
||||||
await InitializeActiveSignatureTabAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task InitializeActiveSignatureTabAsync() {
|
|
||||||
if(ActiveSignatureTab == SignatureTabDraw) {
|
|
||||||
await JSRuntime.InvokeVoidAsync("receiverSignature.initialize", DrawCanvasId);
|
|
||||||
} else if(ActiveSignatureTab == SignatureTabText) {
|
|
||||||
await JSRuntime.InvokeVoidAsync("receiverSignature.initializeTyped", TypedCanvasId);
|
|
||||||
await RenderTypedSignatureAsync();
|
|
||||||
} else {
|
|
||||||
await JSRuntime.InvokeVoidAsync("receiverSignature.initializeImage", ImageInputId, ImageCanvasId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task RenewSignatureAsync() {
|
|
||||||
PopupValidationMessage = null;
|
|
||||||
|
|
||||||
if(ActiveSignatureTab == SignatureTabDraw) {
|
|
||||||
await JSRuntime.InvokeVoidAsync("receiverSignature.clear", DrawCanvasId);
|
|
||||||
} else if(ActiveSignatureTab == SignatureTabText) {
|
|
||||||
TypedSignatureText = string.Empty;
|
|
||||||
await JSRuntime.InvokeVoidAsync("receiverSignature.clearTyped", TypedCanvasId);
|
|
||||||
} else {
|
|
||||||
await JSRuntime.InvokeVoidAsync("receiverSignature.clearImage", ImageInputId, ImageCanvasId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void CloseSignaturePopup() {
|
|
||||||
PopupValidationMessage = null;
|
|
||||||
SignaturePopupVisible = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task OnTypedSignatureChanged(Microsoft.AspNetCore.Components.ChangeEventArgs args) {
|
|
||||||
TypedSignatureText = args.Value?.ToString() ?? string.Empty;
|
|
||||||
await RenderTypedSignatureAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task OnTypedSignatureFontChanged(Microsoft.AspNetCore.Components.ChangeEventArgs args) {
|
|
||||||
TypedSignatureFont = args.Value?.ToString() ?? TypedSignatureFont;
|
|
||||||
await RenderTypedSignatureAsync();
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task RenderTypedSignatureAsync() {
|
|
||||||
await JSRuntime.InvokeVoidAsync("receiverSignature.renderTypedSignature", TypedCanvasId, TypedSignatureText, TypedSignatureFont);
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task SaveSignatureAsync() {
|
|
||||||
if (string.IsNullOrWhiteSpace(SignerFullName)) {
|
|
||||||
PopupValidationMessage = "Bitte geben Sie Vor- und Nachname ein.";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (string.IsNullOrWhiteSpace(SignaturePlace)) {
|
|
||||||
PopupValidationMessage = "Bitte geben Sie den Ort ein.";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
var signatureDataUrl = await GetActiveSignatureDataUrlAsync();
|
|
||||||
if (string.IsNullOrWhiteSpace(signatureDataUrl)) {
|
|
||||||
PopupValidationMessage = "Die Unterschrift ist fuer den PDF-Export erforderlich.";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
PopupValidationMessage = null;
|
|
||||||
SignatureValidationMessage = null;
|
|
||||||
_capturedSignature = new(signatureDataUrl, SignerFullName.Trim(), SignerPosition.Trim(), SignaturePlace.Trim());
|
|
||||||
|
|
||||||
// If no annotations, apply immediately (no checkbox step needed)
|
|
||||||
if (_annotations.Count == 0) {
|
|
||||||
var freshReport = BuildFreshBaseReport();
|
|
||||||
AddSignature(freshReport, _capturedSignature.DataUrl, _capturedSignature.FullName, _capturedSignature.Position, _capturedSignature.Place);
|
|
||||||
Report = freshReport;
|
|
||||||
SignatureApplied = true;
|
|
||||||
SignaturePopupVisible = false;
|
|
||||||
ViewerKey++;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Close popup; checkboxes will appear on the PDF via OnAfterRenderAsync
|
|
||||||
SignaturePopupVisible = false;
|
|
||||||
_lastOverlayViewerKey = -1; // force overlay reinstall
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task SubmitSignaturesAsync() {
|
|
||||||
if (_checkedAnnotations.Count == 0) {
|
|
||||||
SignatureValidationMessage = "Bitte markieren Sie mindestens ein Unterschriftsfeld im Dokument.";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (_checkedAnnotations.Count < _annotations.Count) {
|
|
||||||
SignatureValidationMessage = $"Bitte markieren Sie alle {_annotations.Count} Unterschriftsfelder. Noch {_annotations.Count - _checkedAnnotations.Count} offen.";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
SignatureValidationMessage = null;
|
|
||||||
var freshReport = BuildFreshBaseReport();
|
|
||||||
foreach (var ann in _annotations)
|
|
||||||
AddSignatureAtAnnotation(freshReport, ann, _capturedSignature!.DataUrl, _capturedSignature.FullName, _capturedSignature.Position, _capturedSignature.Place);
|
|
||||||
|
|
||||||
Report = freshReport;
|
|
||||||
SignatureApplied = true;
|
|
||||||
ViewerKey++;
|
|
||||||
await InvokeAsync(StateHasChanged);
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task<string?> GetActiveSignatureDataUrlAsync() {
|
|
||||||
if(ActiveSignatureTab == SignatureTabDraw)
|
|
||||||
return await JSRuntime.InvokeAsync<string?>("receiverSignature.getDataUrl", DrawCanvasId);
|
|
||||||
|
|
||||||
if(ActiveSignatureTab == SignatureTabText) {
|
|
||||||
await RenderTypedSignatureAsync();
|
|
||||||
return await JSRuntime.InvokeAsync<string?>("receiverSignature.getTypedDataUrl", TypedCanvasId);
|
|
||||||
}
|
|
||||||
|
|
||||||
return await JSRuntime.InvokeAsync<string?>("receiverSignature.getImageDataUrl", ImageCanvasId);
|
|
||||||
}
|
|
||||||
|
|
||||||
async Task ExportSignedPdfAsync() {
|
|
||||||
if(!SignatureApplied || Report is null) {
|
|
||||||
SignatureValidationMessage = "Bitte fuegen Sie die Unterschrift zuerst zum Bericht hinzu.";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
SignatureValidationMessage = null;
|
|
||||||
await reportViewer!.ExportToAsync(ExportFormat.Pdf);
|
|
||||||
} catch(Exception) {
|
|
||||||
SignatureValidationMessage = "Das signierte PDF konnte nicht exportiert werden. Bitte laden Sie die Seite neu und versuchen Sie es erneut.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
XtraReport BuildFreshBaseReport() {
|
|
||||||
if (_basePdfBytes is { Length: > 0 }) {
|
|
||||||
var report = new XtraReport();
|
|
||||||
var detail = new DevExpress.XtraReports.UI.DetailBand();
|
|
||||||
report.Bands.Add(detail);
|
|
||||||
detail.Controls.Add(new DevExpress.XtraReports.UI.XRPdfContent { Source = _basePdfBytes, GenerateOwnPages = true });
|
|
||||||
return report;
|
|
||||||
}
|
|
||||||
return CreateReportInstance();
|
|
||||||
}
|
|
||||||
|
|
||||||
static void AddAnnotationPlaceholders(XtraReport report, IReadOnlyList<AnnotationDto> annotations) {
|
|
||||||
var bottomMargin = report.Bands.OfType<BottomMarginBand>().FirstOrDefault();
|
|
||||||
if (bottomMargin is null) {
|
|
||||||
bottomMargin = new BottomMarginBand();
|
|
||||||
report.Bands.Add(bottomMargin);
|
|
||||||
}
|
|
||||||
|
|
||||||
const float sigWidth = 230F;
|
|
||||||
const float sigHeight = 154F;
|
|
||||||
const float bottomPad = 6F;
|
|
||||||
const float defaultTopPad = 8F;
|
|
||||||
const float maxBandHeight = 210F;
|
|
||||||
|
|
||||||
float requiredHeight = defaultTopPad + sigHeight + bottomPad;
|
|
||||||
bottomMargin.HeightF = Math.Min(maxBandHeight, Math.Max(bottomMargin.HeightF, requiredHeight));
|
|
||||||
float topPad = Math.Max(0F, bottomMargin.HeightF - bottomPad - sigHeight);
|
|
||||||
|
|
||||||
foreach (var ann in annotations) {
|
|
||||||
float sigX = (float)(ann.X);
|
|
||||||
var annotId = ann.Id.ToString();
|
|
||||||
|
|
||||||
var placeholder = new XRLabel {
|
|
||||||
Name = $"receiverSignaturePlaceholder_{annotId}",
|
|
||||||
Text = "\u270e Bitte unterschreiben",
|
|
||||||
BoundsF = new RectangleF(sigX, topPad, sigWidth, sigHeight),
|
|
||||||
Borders = BorderSide.All,
|
|
||||||
BorderColor = System.Drawing.Color.FromArgb(230, 81, 0),
|
|
||||||
BackColor = System.Drawing.Color.FromArgb(30, 255, 236, 153),
|
|
||||||
Font = new DXFont("Open Sans", 10F, DXFontStyle.Regular),
|
|
||||||
ForeColor = System.Drawing.Color.FromArgb(94, 38, 0),
|
|
||||||
TextAlignment = TextAlignment.MiddleCenter
|
|
||||||
};
|
|
||||||
|
|
||||||
bottomMargin.Controls.Add(placeholder);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
XtraReport CreateReportInstance() {
|
|
||||||
return ReportStorage.TryGetReport("LargeDatasetReport", out var savedReport)
|
|
||||||
? savedReport
|
|
||||||
: PredefinedReports.ReportsFactory.GetReport("LargeDatasetReport");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void AddSignature(XtraReport report, string signatureDataUrl, string signerFullName, string signerPosition, string signaturePlace) {
|
|
||||||
var imageBytes = Convert.FromBase64String(signatureDataUrl[(signatureDataUrl.IndexOf(',') + 1)..]);
|
|
||||||
using var imageStream = new MemoryStream(imageBytes);
|
|
||||||
var imageSource = new ImageSource(DXImage.FromStream(imageStream));
|
|
||||||
var bottomMargin = report.Bands.OfType<BottomMarginBand>().FirstOrDefault();
|
|
||||||
|
|
||||||
if(bottomMargin is null) {
|
|
||||||
bottomMargin = new BottomMarginBand();
|
|
||||||
report.Bands.Add(bottomMargin);
|
|
||||||
}
|
|
||||||
|
|
||||||
RemoveExistingSignature(bottomMargin);
|
|
||||||
|
|
||||||
// Layout constants
|
|
||||||
const float sigX = 390F;
|
|
||||||
const float sigWidth = 230F;
|
|
||||||
const float sigImgHeight = 70F;
|
|
||||||
const float infoHeight = 65F; // up to 4 lines at 8pt
|
|
||||||
const float innerGap = 5F;
|
|
||||||
const float bottomPad = 6F;
|
|
||||||
const float defaultTopPad = 8F;
|
|
||||||
const float maxBandHeight = 210F;
|
|
||||||
|
|
||||||
float requiredHeight = defaultTopPad + sigImgHeight + innerGap + infoHeight + bottomPad;
|
|
||||||
|
|
||||||
// Grow band if needed, but cap at maxBandHeight to avoid overlapping page content
|
|
||||||
bottomMargin.HeightF = Math.Min(maxBandHeight, Math.Max(bottomMargin.HeightF, requiredHeight));
|
|
||||||
|
|
||||||
// If band is tighter than required, compress top padding so content still fits
|
|
||||||
float topPad = Math.Max(0F, bottomMargin.HeightF - bottomPad - infoHeight - innerGap - sigImgHeight);
|
|
||||||
|
|
||||||
float imageY = topPad;
|
|
||||||
float labelY = imageY + sigImgHeight + innerGap;
|
|
||||||
|
|
||||||
var signatureInformation = string.IsNullOrWhiteSpace(signerPosition)
|
|
||||||
? $"{signerFullName}\n{signaturePlace}, {DateTime.Now:d}"
|
|
||||||
: $"{signerFullName}\n{signerPosition}\n{signaturePlace}, {DateTime.Now:d}";
|
|
||||||
|
|
||||||
var signature = new XRPictureBox {
|
|
||||||
Name = "receiverSignatureImage",
|
|
||||||
ImageSource = imageSource,
|
|
||||||
BoundsF = new RectangleF(sigX, imageY, sigWidth, sigImgHeight),
|
|
||||||
Sizing = ImageSizeMode.ZoomImage,
|
|
||||||
Borders = BorderSide.Bottom,
|
|
||||||
BorderColor = System.Drawing.Color.FromArgb(73, 80, 87)
|
|
||||||
};
|
|
||||||
|
|
||||||
var signatureLabel = new XRLabel {
|
|
||||||
Name = "receiverSignatureLabel",
|
|
||||||
Text = signatureInformation,
|
|
||||||
Multiline = true,
|
|
||||||
BoundsF = new RectangleF(sigX, labelY, sigWidth, infoHeight),
|
|
||||||
Font = new DXFont("Open Sans", 8F, DXFontStyle.Regular),
|
|
||||||
ForeColor = System.Drawing.Color.FromArgb(73, 80, 87),
|
|
||||||
TextAlignment = TextAlignment.TopLeft
|
|
||||||
};
|
|
||||||
|
|
||||||
bottomMargin.Controls.AddRange(new XRControl[] { signature, signatureLabel });
|
|
||||||
}
|
|
||||||
|
|
||||||
static void RemoveExistingSignature(BottomMarginBand bottomMargin) {
|
|
||||||
var controls = bottomMargin.Controls
|
|
||||||
.Cast<XRControl>()
|
|
||||||
.Where(control => control.Name is "receiverSignatureLabel" or "receiverSignatureImage")
|
|
||||||
.ToArray();
|
|
||||||
|
|
||||||
foreach(var control in controls)
|
|
||||||
bottomMargin.Controls.Remove(control);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void AddSignatureAtAnnotation(XtraReport report, AnnotationDto? annotation, string signatureDataUrl, string signerFullName, string signerPosition, string signaturePlace) {
|
|
||||||
var imageBytes = Convert.FromBase64String(signatureDataUrl[(signatureDataUrl.IndexOf(',') + 1)..]);
|
|
||||||
using var imageStream = new MemoryStream(imageBytes);
|
|
||||||
var imageSource = new ImageSource(DXImage.FromStream(imageStream));
|
|
||||||
|
|
||||||
var detail = report.Bands.OfType<DevExpress.XtraReports.UI.DetailBand>().FirstOrDefault();
|
|
||||||
if (detail is null) return;
|
|
||||||
|
|
||||||
var annotId = annotation?.Id.ToString() ?? "0";
|
|
||||||
RemoveExistingSignatureById(detail, annotId);
|
|
||||||
|
|
||||||
const float sigWidth = 230F;
|
|
||||||
const float sigImgHeight = 70F;
|
|
||||||
const float infoHeight = 48.75F;
|
|
||||||
const float innerGap = 5F;
|
|
||||||
|
|
||||||
float sigX = (float)(annotation?.X ?? 390.0);
|
|
||||||
float imageY = (float)(annotation?.Y ?? 900.0);
|
|
||||||
float labelY = imageY + sigImgHeight + innerGap;
|
|
||||||
int targetPage = annotation?.Page ?? 1;
|
|
||||||
|
|
||||||
var signatureInformation = string.IsNullOrWhiteSpace(signerPosition)
|
|
||||||
? $"{signerFullName}\n{signaturePlace}, {DateTime.Now:d}"
|
|
||||||
: $"{signerFullName}\n{signerPosition}\n{signaturePlace}, {DateTime.Now:d}";
|
|
||||||
|
|
||||||
var signature = new XRPictureBox {
|
|
||||||
Name = $"receiverSignatureImage_{annotId}",
|
|
||||||
ImageSource = imageSource,
|
|
||||||
BoundsF = new RectangleF(sigX, imageY, sigWidth, sigImgHeight),
|
|
||||||
Sizing = ImageSizeMode.ZoomImage,
|
|
||||||
Borders = BorderSide.Bottom,
|
|
||||||
BorderColor = System.Drawing.Color.FromArgb(73, 80, 87),
|
|
||||||
BackColor = Color.FromArgb(219, 219, 219)
|
|
||||||
};
|
|
||||||
|
|
||||||
var signatureLabel = new XRLabel {
|
|
||||||
Name = $"receiverSignatureLabel_{annotId}",
|
|
||||||
Text = signatureInformation,
|
|
||||||
Multiline = true,
|
|
||||||
BoundsF = new RectangleF(sigX, labelY - 5, sigWidth, infoHeight),
|
|
||||||
Font = new DXFont("Open Sans", 8F, DXFontStyle.Regular),
|
|
||||||
ForeColor = System.Drawing.Color.FromArgb(73, 80, 87),
|
|
||||||
TextAlignment = TextAlignment.TopCenter,
|
|
||||||
BackColor = Color.FromArgb(219, 219, 219)
|
|
||||||
};
|
|
||||||
|
|
||||||
// Show each control only on the target page using an independent print counter
|
|
||||||
int sigPrintCount = 0;
|
|
||||||
signature.BeforePrint += (_, e) => {
|
|
||||||
sigPrintCount++;
|
|
||||||
e.Cancel = sigPrintCount != targetPage;
|
|
||||||
};
|
|
||||||
|
|
||||||
int lblPrintCount = 0;
|
|
||||||
signatureLabel.BeforePrint += (_, e) => {
|
|
||||||
lblPrintCount++;
|
|
||||||
e.Cancel = lblPrintCount != targetPage;
|
|
||||||
};
|
|
||||||
|
|
||||||
detail.Controls.AddRange(new XRControl[] { signature, signatureLabel });
|
|
||||||
}
|
|
||||||
|
|
||||||
static void RemoveExistingSignatureById(DevExpress.XtraReports.UI.DetailBand detail, string annotId) {
|
|
||||||
var controls = detail.Controls
|
|
||||||
.Cast<XRControl>()
|
|
||||||
.Where(c => c.Name == $"receiverSignatureImage_{annotId}" || c.Name == $"receiverSignatureLabel_{annotId}")
|
|
||||||
.ToArray();
|
|
||||||
foreach (var c in controls)
|
|
||||||
detail.Controls.Remove(c);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Dispose() {
|
|
||||||
_dotNetRef?.Dispose();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,81 +0,0 @@
|
|||||||
@page "/test"
|
|
||||||
@using System.Drawing
|
|
||||||
@using DevExpress.Drawing
|
|
||||||
@using DevExpress.Utils
|
|
||||||
@using DevExpress.XtraPrinting
|
|
||||||
@using DevExpress.XtraPrinting.Drawing
|
|
||||||
@using XtraReport = DevExpress.XtraReports.UI.XtraReport
|
|
||||||
@using BottomMarginBand = DevExpress.XtraReports.UI.BottomMarginBand
|
|
||||||
@using XRLabel = DevExpress.XtraReports.UI.XRLabel
|
|
||||||
@using XRPictureBox = DevExpress.XtraReports.UI.XRPictureBox
|
|
||||||
@using XRControl = DevExpress.XtraReports.UI.XRControl
|
|
||||||
@using ImageSizeMode = DevExpress.XtraPrinting.ImageSizeMode
|
|
||||||
@using EnvelopeGenerator.ReceiverUI.Services
|
|
||||||
@using DevExpress.Blazor.Reporting
|
|
||||||
@inject IJSRuntime JSRuntime
|
|
||||||
@inject InMemoryReportStorageWebExtension ReportStorage
|
|
||||||
@inject EnvelopeGenerator.ReceiverUI.Services.DocumentService DocumentService
|
|
||||||
@inject Microsoft.Extensions.Configuration.IConfiguration Configuration
|
|
||||||
@inject HttpClient Http
|
|
||||||
@using System;
|
|
||||||
|
|
||||||
<link href="_content/DevExpress.Blazor.Themes/blazing-berry.bs5.min.css" rel="stylesheet" />
|
|
||||||
<link href="_content/DevExpress.Blazor.Reporting.Viewer/css/dx-blazor-reporting-components.bs5.css" rel="stylesheet" />
|
|
||||||
|
|
||||||
<div class="receiver-page-layout">
|
|
||||||
|
|
||||||
<div class="receiver-viewer-wrapper">
|
|
||||||
<embed class="w-100 h-50" src="/docs/Document.pdf" type="application/pdf" />
|
|
||||||
@if (PdfBytes is { Length: > 0 }) {
|
|
||||||
<DxPdfViewer DocumentContent="PdfBytes" CssClass="w-100 h-50" />
|
|
||||||
}
|
|
||||||
else{
|
|
||||||
<div>Not found</div>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
|
|
||||||
const string SignatureTabDraw = "draw";
|
|
||||||
const string SignatureTabText = "text";
|
|
||||||
const string SignatureTabImage = "image";
|
|
||||||
const string DrawCanvasId = "receiver-signature-pad";
|
|
||||||
const string TypedCanvasId = "receiver-typed-signature-pad";
|
|
||||||
const string ImageInputId = "receiver-signature-image-input";
|
|
||||||
const string ImageCanvasId = "receiver-image-signature-pad";
|
|
||||||
|
|
||||||
readonly (string Text, string Value)[] TypedSignatureFonts = {
|
|
||||||
("Brush Script", "'Brush Script MT', cursive"),
|
|
||||||
("Segoe Script", "'Segoe Script', cursive"),
|
|
||||||
("Lucida Handwriting", "'Lucida Handwriting', cursive"),
|
|
||||||
("Comic Sans", "'Comic Sans MS', cursive"),
|
|
||||||
("Cursive", "cursive")
|
|
||||||
};
|
|
||||||
|
|
||||||
[Parameter] public string? EnvelopeKey { get; set; }
|
|
||||||
|
|
||||||
DxReportViewer? reportViewer;
|
|
||||||
XtraReport? Report;
|
|
||||||
string PdfViewerUrl = string.Empty;
|
|
||||||
byte[]? PdfBytes;
|
|
||||||
byte[]? SignedPdfBytes;
|
|
||||||
bool SignatureApplied;
|
|
||||||
bool SignaturePopupVisible;
|
|
||||||
string? PopupValidationMessage;
|
|
||||||
string ActiveSignatureTab = SignatureTabDraw;
|
|
||||||
string TypedSignatureText = string.Empty;
|
|
||||||
string TypedSignatureFont = "'Brush Script MT', cursive";
|
|
||||||
string SignerFullName = string.Empty;
|
|
||||||
string SignerPosition = string.Empty;
|
|
||||||
string SignaturePlace = string.Empty;
|
|
||||||
int ViewerKey;
|
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync() {
|
|
||||||
PdfBytes = await Http.GetByteArrayAsync("/docs/Document.pdf");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,123 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<root>
|
|
||||||
<!--
|
|
||||||
Microsoft ResX Schema
|
|
||||||
|
|
||||||
Version 2.0
|
|
||||||
|
|
||||||
The primary goals of this format is to allow a simple XML format
|
|
||||||
that is mostly human readable. The generation and parsing of the
|
|
||||||
various data types are done through the TypeConverter classes
|
|
||||||
associated with the data types.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
... ado.net/XML headers & schema ...
|
|
||||||
<resheader name="resmimetype">text/microsoft-resx</resheader>
|
|
||||||
<resheader name="version">2.0</resheader>
|
|
||||||
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
|
|
||||||
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
|
|
||||||
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
|
|
||||||
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
|
|
||||||
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
|
|
||||||
<value>[base64 mime encoded serialized .NET Framework object]</value>
|
|
||||||
</data>
|
|
||||||
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
|
|
||||||
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
|
|
||||||
<comment>This is a comment</comment>
|
|
||||||
</data>
|
|
||||||
|
|
||||||
There are any number of "resheader" rows that contain simple
|
|
||||||
name/value pairs.
|
|
||||||
|
|
||||||
Each data row contains a name, and value. The row also contains a
|
|
||||||
type or mimetype. Type corresponds to a .NET class that support
|
|
||||||
text/value conversion through the TypeConverter architecture.
|
|
||||||
Classes that don't support this are serialized and stored with the
|
|
||||||
mimetype set.
|
|
||||||
|
|
||||||
The mimetype is used for serialized objects, and tells the
|
|
||||||
ResXResourceReader how to depersist the object. This is currently not
|
|
||||||
extensible. For a given mimetype the value must be set accordingly:
|
|
||||||
|
|
||||||
Note - application/x-microsoft.net.object.binary.base64 is the format
|
|
||||||
that the ResXResourceWriter will generate, however the reader can
|
|
||||||
read any of the formats listed below.
|
|
||||||
|
|
||||||
mimetype: application/x-microsoft.net.object.binary.base64
|
|
||||||
value : The object must be serialized with
|
|
||||||
: System.Runtime.Serialization.Formatters.Binary.BinaryFormatter
|
|
||||||
: and then encoded with base64 encoding.
|
|
||||||
|
|
||||||
mimetype: application/x-microsoft.net.object.soap.base64
|
|
||||||
value : The object must be serialized with
|
|
||||||
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter
|
|
||||||
: and then encoded with base64 encoding.
|
|
||||||
|
|
||||||
mimetype: application/x-microsoft.net.object.bytearray.base64
|
|
||||||
value : The object must be serialized into a byte array
|
|
||||||
: using a System.ComponentModel.TypeConverter
|
|
||||||
: and then encoded with base64 encoding.
|
|
||||||
-->
|
|
||||||
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
|
|
||||||
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" />
|
|
||||||
<xsd:element name="root" msdata:IsDataSet="true">
|
|
||||||
<xsd:complexType>
|
|
||||||
<xsd:choice maxOccurs="unbounded">
|
|
||||||
<xsd:element name="metadata">
|
|
||||||
<xsd:complexType>
|
|
||||||
<xsd:sequence>
|
|
||||||
<xsd:element name="value" type="xsd:string" minOccurs="0" />
|
|
||||||
</xsd:sequence>
|
|
||||||
<xsd:attribute name="name" use="required" type="xsd:string" />
|
|
||||||
<xsd:attribute name="type" type="xsd:string" />
|
|
||||||
<xsd:attribute name="mimetype" type="xsd:string" />
|
|
||||||
<xsd:attribute ref="xml:space" />
|
|
||||||
</xsd:complexType>
|
|
||||||
</xsd:element>
|
|
||||||
<xsd:element name="assembly">
|
|
||||||
<xsd:complexType>
|
|
||||||
<xsd:attribute name="alias" type="xsd:string" />
|
|
||||||
<xsd:attribute name="name" type="xsd:string" />
|
|
||||||
</xsd:complexType>
|
|
||||||
</xsd:element>
|
|
||||||
<xsd:element name="data">
|
|
||||||
<xsd:complexType>
|
|
||||||
<xsd:sequence>
|
|
||||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
|
||||||
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
|
|
||||||
</xsd:sequence>
|
|
||||||
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" />
|
|
||||||
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
|
|
||||||
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
|
|
||||||
<xsd:attribute ref="xml:space" />
|
|
||||||
</xsd:complexType>
|
|
||||||
</xsd:element>
|
|
||||||
<xsd:element name="resheader">
|
|
||||||
<xsd:complexType>
|
|
||||||
<xsd:sequence>
|
|
||||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
|
|
||||||
</xsd:sequence>
|
|
||||||
<xsd:attribute name="name" type="xsd:string" use="required" />
|
|
||||||
</xsd:complexType>
|
|
||||||
</xsd:element>
|
|
||||||
</xsd:choice>
|
|
||||||
</xsd:complexType>
|
|
||||||
</xsd:element>
|
|
||||||
</xsd:schema>
|
|
||||||
<resheader name="resmimetype">
|
|
||||||
<value>text/microsoft-resx</value>
|
|
||||||
</resheader>
|
|
||||||
<resheader name="version">
|
|
||||||
<value>2.0</value>
|
|
||||||
</resheader>
|
|
||||||
<resheader name="reader">
|
|
||||||
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
|
||||||
</resheader>
|
|
||||||
<resheader name="writer">
|
|
||||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
|
|
||||||
</resheader>
|
|
||||||
<metadata name="objectDataSource1.TrayLocation" type="System.Drawing.Point, System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a">
|
|
||||||
<value>17, 17</value>
|
|
||||||
</metadata>
|
|
||||||
</root>
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
using DevExpress.XtraReports.UI;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.PredefinedReports {
|
|
||||||
public static class ReportsFactory
|
|
||||||
{
|
|
||||||
public static readonly Dictionary<string, Func<XtraReport>> Reports = new() {
|
|
||||||
["LargeDatasetReport"] = () => new PredefinedReports.Report()
|
|
||||||
};
|
|
||||||
|
|
||||||
public static XtraReport GetReport(string reportName) {
|
|
||||||
return Reports[reportName]();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
using Microsoft.AspNetCore.Components.WebAssembly.Hosting;
|
|
||||||
using Microsoft.AspNetCore.Components.Web;
|
|
||||||
using EnvelopeGenerator.ReceiverUI;
|
|
||||||
using DevExpress.DataAccess.Web;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Options;
|
|
||||||
using DevExpress.XtraReports.Services;
|
|
||||||
using DevExpress.Blazor.Reporting;
|
|
||||||
using DevExpress.XtraReports.Web.Extensions;
|
|
||||||
|
|
||||||
var builder = WebAssemblyHostBuilder.CreateDefault(args);
|
|
||||||
builder.RootComponents.Add<App>("#app");
|
|
||||||
builder.RootComponents.Add<HeadOutlet>("head::after");
|
|
||||||
builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri(builder.HostEnvironment.BaseAddress) });
|
|
||||||
builder.Services.Configure<ApiOptions>(opts =>
|
|
||||||
builder.Configuration.GetSection(ApiOptions.SectionName).Bind(opts));
|
|
||||||
builder.Services.AddScoped<EnvelopeGenerator.ReceiverUI.Services.DocumentService>();
|
|
||||||
builder.Services.AddScoped<EnvelopeGenerator.ReceiverUI.Services.AuthService>();
|
|
||||||
builder.Services.AddScoped<EnvelopeGenerator.ReceiverUI.Services.AnnotationService>();
|
|
||||||
builder.Services.AddScoped<EnvelopeGenerator.ReceiverUI.Services.EnvelopeReceiverService>();
|
|
||||||
|
|
||||||
builder.Services.AddDevExpressWebAssemblyBlazorReportViewer();
|
|
||||||
builder.Services.AddDevExpressWebAssemblyBlazorPdfViewer();
|
|
||||||
|
|
||||||
builder.Services.AddDevExpressBlazorReportingWebAssembly(configure => {
|
|
||||||
configure.UseDevelopmentMode();
|
|
||||||
});
|
|
||||||
builder.Services.AddScoped<IDataSourceWizardJsonConnectionStorage, CustomDataSourceWizardJsonDataConnectionStorage>();
|
|
||||||
builder.Services.AddScoped<IJsonDataConnectionProviderFactory, CustomJsonDataConnectionProviderFactory>();
|
|
||||||
builder.Services.AddScoped<IObjectDataSourceWizardTypeProvider, ObjectDataSourceWizardCustomTypeProvider>();
|
|
||||||
DevExpress.Utils.DeserializationSettings.RegisterTrustedClass(typeof(EnvelopeGenerator.ReceiverUI.Data.DataItemList));
|
|
||||||
DevExpress.Utils.DeserializationSettings.RegisterTrustedClass(typeof(EnvelopeGenerator.ReceiverUI.PredefinedReports.Report));
|
|
||||||
builder.Services.AddSingleton<InMemoryReportStorageWebExtension>();
|
|
||||||
builder.Services.AddSingleton<ReportStorageWebExtension>(sp => sp.GetRequiredService<InMemoryReportStorageWebExtension>());
|
|
||||||
builder.Services.AddScoped<IReportProviderAsync, CustomReportProvider>();
|
|
||||||
ReportStorageWebExtension.RegisterExtensionGlobal(new InMemoryReportStorageWebExtension());
|
|
||||||
|
|
||||||
var host = builder.Build();
|
|
||||||
await FontLoader.LoadFonts(host.Services.GetRequiredService<HttpClient>(), new List<string> { "opensans.ttf" });
|
|
||||||
await host.RunAsync();
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<!-- https://go.microsoft.com/fwlink/?LinkID=208121. -->
|
|
||||||
<Project>
|
|
||||||
<PropertyGroup>
|
|
||||||
<WebPublishMethod>Package</WebPublishMethod>
|
|
||||||
<LastUsedBuildConfiguration>Release</LastUsedBuildConfiguration>
|
|
||||||
<LastUsedPlatform>Any CPU</LastUsedPlatform>
|
|
||||||
<SiteUrlToLaunchAfterPublish />
|
|
||||||
<LaunchSiteAfterPublish>true</LaunchSiteAfterPublish>
|
|
||||||
<ExcludeApp_Data>false</ExcludeApp_Data>
|
|
||||||
<ProjectGuid>fb2d306b-1042-4a70-31ed-f991a1599371</ProjectGuid>
|
|
||||||
<DesktopBuildPackageLocation>M:\App&Service\0 DD - Smart UP\signFLOW\ReceiverUI\net8\$(Version)\EnvelopeGenerator.ReceiverUI.zip</DesktopBuildPackageLocation>
|
|
||||||
<PackageAsSingleFile>true</PackageAsSingleFile>
|
|
||||||
<DeployIisAppPath>EnvelopeGenerator.ReceiverUI</DeployIisAppPath>
|
|
||||||
<_TargetId>IISWebDeployPackage</_TargetId>
|
|
||||||
</PropertyGroup>
|
|
||||||
</Project>
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
{
|
|
||||||
"profiles": {
|
|
||||||
"EnvelopeGenerator.ReceiverUI": {
|
|
||||||
"commandName": "Project",
|
|
||||||
"launchBrowser": true,
|
|
||||||
"environmentVariables": {
|
|
||||||
"ASPNETCORE_ENVIRONMENT": "Development"
|
|
||||||
},
|
|
||||||
"applicationUrl": "https://localhost:52936;http://localhost:52937"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
using System.Net.Http.Json;
|
|
||||||
using System.Text.Json;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Models;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Options;
|
|
||||||
using Microsoft.Extensions.Options;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Retrieves annotation positions from the API.
|
|
||||||
/// The URL is composed as <c>{BaseUrl}/api/Annotation/{envelopeKey}</c>.
|
|
||||||
/// During development, <c>BaseUrl</c> is empty so the request resolves to the
|
|
||||||
/// YARP-proxied route on the same origin, which currently serves
|
|
||||||
/// <c>fake-data/annotations.json</c>. To switch to real data, update the
|
|
||||||
/// YARP route in <c>yarp.json</c> — no code change required.
|
|
||||||
/// </summary>
|
|
||||||
public class AnnotationService(HttpClient http, IOptions<ApiOptions> apiOptions)
|
|
||||||
{
|
|
||||||
private static readonly JsonSerializerOptions _jsonOptions = new(JsonSerializerDefaults.Web);
|
|
||||||
|
|
||||||
public async Task<IReadOnlyList<AnnotationDto>> GetAnnotationsAsync(string envelopeKey, CancellationToken cancel = default)
|
|
||||||
{
|
|
||||||
var url = $"{apiOptions.Value.BaseUrl}/api/Annotation/{Uri.EscapeDataString(envelopeKey)}";
|
|
||||||
var response = await http.GetAsync(url, cancel);
|
|
||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
|
||||||
return [];
|
|
||||||
|
|
||||||
var result = await response.Content.ReadFromJsonAsync<List<AnnotationDto>>(_jsonOptions, cancel);
|
|
||||||
return result ?? [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,57 +0,0 @@
|
|||||||
using System.Net;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Options;
|
|
||||||
using Microsoft.Extensions.Options;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
|
|
||||||
public enum EnvelopeLoginResult { Success, InvalidCode, NotFound, Error }
|
|
||||||
|
|
||||||
public class AuthService(HttpClient http, IOptions<ApiOptions> apiOptions)
|
|
||||||
{
|
|
||||||
private readonly ApiOptions _api = apiOptions.Value;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Checks whether the current user holds a valid receiver token for the given envelope key.
|
|
||||||
/// Calls GET /api/auth/check/envelope/{envelopeKey}.
|
|
||||||
/// </summary>
|
|
||||||
public async Task<bool> CheckEnvelopeAccessAsync(string envelopeKey, CancellationToken cancel = default)
|
|
||||||
{
|
|
||||||
var response = await http.GetAsync($"{_api.BaseUrl}/api/auth/check/envelope/{Uri.EscapeDataString(envelopeKey)}", cancel);
|
|
||||||
return response.StatusCode == HttpStatusCode.OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Submits the access code for the given envelope key.
|
|
||||||
/// Calls POST /api/Auth/envelope-receiver/{key} with multipart/form-data.
|
|
||||||
/// On success the API sets an authentication cookie automatically.
|
|
||||||
/// </summary>
|
|
||||||
public async Task<EnvelopeLoginResult> LoginEnvelopeReceiverAsync(string envelopeKey, string accessCode, CancellationToken cancel = default)
|
|
||||||
{
|
|
||||||
var form = new MultipartFormDataContent();
|
|
||||||
form.Add(new StringContent(accessCode), "AccessCode");
|
|
||||||
|
|
||||||
var response = await http.PostAsync(
|
|
||||||
$"{_api.BaseUrl}/api/Auth/envelope-receiver/{Uri.EscapeDataString(envelopeKey)}",
|
|
||||||
form, cancel);
|
|
||||||
|
|
||||||
return response.StatusCode switch
|
|
||||||
{
|
|
||||||
HttpStatusCode.OK => EnvelopeLoginResult.Success,
|
|
||||||
HttpStatusCode.Unauthorized => EnvelopeLoginResult.InvalidCode,
|
|
||||||
HttpStatusCode.NotFound => EnvelopeLoginResult.NotFound,
|
|
||||||
_ => EnvelopeLoginResult.Error
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Removes the per-envelope receiver cookie for the given envelope key.
|
|
||||||
/// Calls POST /api/auth/logout/envelope/{envelopeKey}.
|
|
||||||
/// </summary>
|
|
||||||
public async Task<bool> LogoutEnvelopeReceiverAsync(string envelopeKey, CancellationToken cancel = default)
|
|
||||||
{
|
|
||||||
var response = await http.PostAsync(
|
|
||||||
$"{_api.BaseUrl}/api/auth/logout/envelope/{Uri.EscapeDataString(envelopeKey)}",
|
|
||||||
null, cancel);
|
|
||||||
return response.IsSuccessStatusCode;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
using DevExpress.DataAccess.Json;
|
|
||||||
using DevExpress.DataAccess.Web;
|
|
||||||
using DevExpress.DataAccess.Wizard.Services;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services
|
|
||||||
{
|
|
||||||
public class CustomDataSourceWizardJsonDataConnectionStorage : IDataSourceWizardJsonConnectionStorage
|
|
||||||
{
|
|
||||||
public static JsonDataConnection GetDefaultConnection() {
|
|
||||||
var uriJsonSource = new UriJsonSource() {
|
|
||||||
Uri = new Uri(@"https://raw.githubusercontent.com/DevExpress-Examples/DataSources/master/JSON/customers.json"),
|
|
||||||
};
|
|
||||||
return new JsonDataConnection(uriJsonSource) { StoreConnectionNameOnly = true, Name = "NWindProductsJson" };
|
|
||||||
}
|
|
||||||
public static List<JsonDataConnection> GetConnections() {
|
|
||||||
var connections = new List<JsonDataConnection> {
|
|
||||||
GetDefaultConnection()
|
|
||||||
};
|
|
||||||
return connections;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IJsonConnectionStorageService.CanSaveConnection => false;
|
|
||||||
bool IJsonConnectionStorageService.ContainsConnection(string connectionName) {
|
|
||||||
return GetConnections().Any(x => x.Name == connectionName);
|
|
||||||
}
|
|
||||||
|
|
||||||
IEnumerable<JsonDataConnection> IJsonConnectionStorageService.GetConnections() {
|
|
||||||
return GetConnections();
|
|
||||||
}
|
|
||||||
|
|
||||||
JsonDataConnection IJsonDataConnectionProviderService.GetJsonDataConnection(string name) {
|
|
||||||
var connection = GetConnections().FirstOrDefault(x => x.Name == name);
|
|
||||||
if(connection == null)
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
return connection;
|
|
||||||
}
|
|
||||||
|
|
||||||
void IJsonConnectionStorageService.SaveConnection(string connectionName, JsonDataConnection connection, bool saveCredentials) { }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
using DevExpress.DataAccess.Json;
|
|
||||||
using DevExpress.DataAccess.Web;
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services
|
|
||||||
{
|
|
||||||
public class CustomJsonDataConnectionProviderFactory : IJsonDataConnectionProviderFactory {
|
|
||||||
public IJsonDataConnectionProviderService Create() {
|
|
||||||
return new WebDocumentViewerJsonDataConnectionProvider(CustomDataSourceWizardJsonDataConnectionStorage.GetConnections());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class WebDocumentViewerJsonDataConnectionProvider : IJsonDataConnectionProviderService
|
|
||||||
{
|
|
||||||
readonly List<JsonDataConnection> jsonDataConnections;
|
|
||||||
public WebDocumentViewerJsonDataConnectionProvider(List<JsonDataConnection> jsonDataConnections) {
|
|
||||||
this.jsonDataConnections = jsonDataConnections;
|
|
||||||
}
|
|
||||||
public JsonDataConnection GetJsonDataConnection(string name) {
|
|
||||||
var connection = jsonDataConnections.FirstOrDefault(x => x.Name == name);
|
|
||||||
if(connection == null)
|
|
||||||
throw new InvalidOperationException();
|
|
||||||
return connection;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
using DevExpress.XtraReports.UI;
|
|
||||||
using DevExpress.XtraReports.Services;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.PredefinedReports;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services
|
|
||||||
{
|
|
||||||
public class CustomReportProvider : IReportProviderAsync {
|
|
||||||
private readonly InMemoryReportStorageWebExtension reportStorage;
|
|
||||||
|
|
||||||
public CustomReportProvider(InMemoryReportStorageWebExtension reportStorage) {
|
|
||||||
this.reportStorage = reportStorage;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Task<XtraReport> GetReportAsync(string id, ReportProviderContext context) {
|
|
||||||
if(reportStorage.TryGetReport(id, out var savedReport))
|
|
||||||
return Task.FromResult(savedReport);
|
|
||||||
|
|
||||||
return Task.FromResult(ReportsFactory.GetReport(id));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
using System.Net;
|
|
||||||
using System.Net.Http;
|
|
||||||
using Microsoft.Extensions.Options;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Options;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
|
|
||||||
public class DocumentService(HttpClient http, IOptions<ApiOptions> apiOptions)
|
|
||||||
{
|
|
||||||
private readonly ApiOptions _api = apiOptions.Value;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Fetches the PDF bytes for the given envelope key from the API.
|
|
||||||
/// Returns null bytes with the HTTP status code on failure.
|
|
||||||
/// </summary>
|
|
||||||
public async Task<(byte[]? Bytes, HttpStatusCode StatusCode)> GetDocumentAsync(string envelopeKey, CancellationToken cancel = default)
|
|
||||||
{
|
|
||||||
var response = await http.GetAsync($"{_api.BaseUrl}/api/Document/{Uri.EscapeDataString(envelopeKey)}", cancel);
|
|
||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
|
||||||
return (null, response.StatusCode);
|
|
||||||
|
|
||||||
var bytes = await response.Content.ReadAsByteArrayAsync(cancel);
|
|
||||||
return (bytes, response.StatusCode);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
using System.Net.Http.Json;
|
|
||||||
using System.Text.Json;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Models;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.Options;
|
|
||||||
using Microsoft.Extensions.Options;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Retrieves the <see cref="EnvelopeReceiverDto"/> for the authenticated receiver
|
|
||||||
/// from <c>GET api/EnvelopeReceiver/{envelopeKey}</c>.
|
|
||||||
/// </summary>
|
|
||||||
public class EnvelopeReceiverService(HttpClient http, IOptions<ApiOptions> apiOptions)
|
|
||||||
{
|
|
||||||
private static readonly JsonSerializerOptions _jsonOptions = new(JsonSerializerDefaults.Web);
|
|
||||||
|
|
||||||
public async Task<EnvelopeReceiverDto?> GetAsync(string envelopeKey, CancellationToken cancel = default)
|
|
||||||
{
|
|
||||||
var url = $"{apiOptions.Value.BaseUrl}/api/EnvelopeReceiver/{Uri.EscapeDataString(envelopeKey)}";
|
|
||||||
var response = await http.GetAsync(url, cancel);
|
|
||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
return await response.Content.ReadFromJsonAsync<EnvelopeReceiverDto>(_jsonOptions, cancel);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
using DevExpress.Drawing;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services {
|
|
||||||
public static class FontLoader {
|
|
||||||
public async static Task LoadFonts(HttpClient httpClient, List<string> fontNames) {
|
|
||||||
foreach(var fontName in fontNames) {
|
|
||||||
var fontBytes = await httpClient.GetByteArrayAsync($"fonts/{fontName}");
|
|
||||||
DXFontRepository.Instance.AddFont(fontBytes);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,83 +0,0 @@
|
|||||||
using DevExpress.XtraReports.UI;
|
|
||||||
using DevExpress.XtraReports.Web.Extensions;
|
|
||||||
using EnvelopeGenerator.ReceiverUI.PredefinedReports;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
|
|
||||||
public class InMemoryReportStorageWebExtension : ReportStorageWebExtension
|
|
||||||
{
|
|
||||||
private const string DefaultReportName = "LargeDatasetReport";
|
|
||||||
private static readonly Dictionary<string, byte[]> Reports = new(StringComparer.OrdinalIgnoreCase);
|
|
||||||
|
|
||||||
public override bool CanSetData(string url) => IsValidUrl(url);
|
|
||||||
|
|
||||||
public override byte[] GetData(string url)
|
|
||||||
{
|
|
||||||
url = NormalizeUrl(url);
|
|
||||||
|
|
||||||
if (Reports.TryGetValue(url, out var reportLayout))
|
|
||||||
return reportLayout;
|
|
||||||
|
|
||||||
if (ReportsFactory.Reports.TryGetValue(url, out var reportFactory))
|
|
||||||
return SaveReport(reportFactory());
|
|
||||||
|
|
||||||
throw new DevExpress.XtraReports.Web.ClientControls.FaultException($"Report '{url}' was not found.");
|
|
||||||
}
|
|
||||||
|
|
||||||
public override Dictionary<string, string> GetUrls()
|
|
||||||
{
|
|
||||||
var urls = ReportsFactory.Reports.Keys
|
|
||||||
.Concat(Reports.Keys)
|
|
||||||
.Distinct(StringComparer.OrdinalIgnoreCase)
|
|
||||||
.ToDictionary(name => name, name => name, StringComparer.OrdinalIgnoreCase);
|
|
||||||
|
|
||||||
return urls;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool IsValidUrl(string url)
|
|
||||||
{
|
|
||||||
return !string.IsNullOrWhiteSpace(url)
|
|
||||||
&& url.IndexOfAny(Path.GetInvalidFileNameChars()) < 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void SetData(XtraReport report, string url)
|
|
||||||
{
|
|
||||||
url = NormalizeUrl(url);
|
|
||||||
Reports[url] = SaveReport(report);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string SetNewData(XtraReport report, string defaultUrl)
|
|
||||||
{
|
|
||||||
var url = NormalizeUrl(defaultUrl);
|
|
||||||
Reports[url] = SaveReport(report);
|
|
||||||
return url;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryGetReport(string url, out XtraReport report)
|
|
||||||
{
|
|
||||||
url = NormalizeUrl(url);
|
|
||||||
|
|
||||||
if (!Reports.ContainsKey(url))
|
|
||||||
{
|
|
||||||
report = null!;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
using var stream = new MemoryStream(Reports[url]);
|
|
||||||
report = XtraReport.FromXmlStream(stream, true);
|
|
||||||
report.Name = url;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static string NormalizeUrl(string url)
|
|
||||||
{
|
|
||||||
return string.IsNullOrWhiteSpace(url) ? DefaultReportName : url;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static byte[] SaveReport(XtraReport report)
|
|
||||||
{
|
|
||||||
using var stream = new MemoryStream();
|
|
||||||
report.SaveLayoutToXml(stream);
|
|
||||||
return stream.ToArray();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
using DevExpress.DataAccess.Web;
|
|
||||||
|
|
||||||
namespace EnvelopeGenerator.ReceiverUI.Services {
|
|
||||||
public class ObjectDataSourceWizardCustomTypeProvider : IObjectDataSourceWizardTypeProvider {
|
|
||||||
public IEnumerable<Type> GetAvailableTypes(string context) {
|
|
||||||
return new[] { typeof(Data.DataItemList) };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
<nav class="navbar header-navbar p-0">
|
|
||||||
<button class="navbar-toggler bg-primary d-block" @onclick="OnToggleClick">
|
|
||||||
<span class="navbar-toggler-icon"></span>
|
|
||||||
</button>
|
|
||||||
<div class="ms-3 fw-bold title pe-4">EnvelopeGenerator.ReceiverUI</div>
|
|
||||||
</nav>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
[Parameter] public bool ToggleOn { get; set; }
|
|
||||||
[Parameter] public EventCallback<bool> ToggleOnChanged { get; set; }
|
|
||||||
|
|
||||||
async Task OnToggleClick() => await Toggle();
|
|
||||||
|
|
||||||
async Task Toggle(bool? value = null) {
|
|
||||||
var newValue = value ?? !ToggleOn;
|
|
||||||
if(ToggleOn != newValue) {
|
|
||||||
ToggleOn = newValue;
|
|
||||||
await ToggleOnChanged.InvokeAsync(ToggleOn);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
.navbar.header-navbar {
|
|
||||||
flex-grow: 0;
|
|
||||||
flex-wrap: nowrap;
|
|
||||||
border: none;
|
|
||||||
background-color: inherit;
|
|
||||||
border-radius: 0;
|
|
||||||
height: 3.5rem;
|
|
||||||
min-height: 3.5rem;
|
|
||||||
box-shadow: 0px 2px 6px 0px rgba(0, 0, 0, 0.12);
|
|
||||||
justify-content: flex-start;
|
|
||||||
}
|
|
||||||
|
|
||||||
.header-navbar .navbar-toggler {
|
|
||||||
outline: none;
|
|
||||||
border-radius: 0;
|
|
||||||
padding-left: .75rem;
|
|
||||||
padding-right: .75rem;
|
|
||||||
box-shadow: none;
|
|
||||||
align-self: stretch;
|
|
||||||
}
|
|
||||||
|
|
||||||
.header-navbar .navbar-toggler .navbar-toggler-icon {
|
|
||||||
background-image: url("data:image/svg+xml;charset=utf8,%3Csvg viewBox='0 0 32 32' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath stroke='rgba(255,255,255, 1)' stroke-width='2' stroke-linecap='round' stroke-miterlimit='10' d='M4 8h24M4 16h24M4 24h24'/%3E%3C/svg%3E");
|
|
||||||
background-color: transparent !important;
|
|
||||||
height: 2rem;
|
|
||||||
width: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.title {
|
|
||||||
font-size: 1.1rem;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 350px) {
|
|
||||||
.title {
|
|
||||||
font-size: inherit;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
@using EnvelopeGenerator.ReceiverUI.Services;
|
|
||||||
@inherits LayoutComponentBase
|
|
||||||
|
|
||||||
<div class="page">
|
|
||||||
<main>
|
|
||||||
<article class="content">
|
|
||||||
@Body
|
|
||||||
</article>
|
|
||||||
</main>
|
|
||||||
<footer class="receiver-footer">
|
|
||||||
<span>© SignFlow 2023-2024 <a href="https://digitaldata.works" target="_blank" rel="noopener">Digital Data GmbH</a></span>
|
|
||||||
<span class="receiver-footer__sep">|</span>
|
|
||||||
<a href="docs/privacy-policy.de-DE.html" target="_blank" rel="noopener">Datenschutz</a>
|
|
||||||
</footer>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
[Inject] HttpClient Http { get; set; }
|
|
||||||
List<string> RequiredFonts = new() {
|
|
||||||
"opensans.ttf"
|
|
||||||
};
|
|
||||||
|
|
||||||
protected async override Task OnInitializedAsync() {
|
|
||||||
await FontLoader.LoadFonts(Http, RequiredFonts);
|
|
||||||
await base.OnInitializedAsync();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
.page {
|
|
||||||
position: relative;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
overflow: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
main {
|
|
||||||
flex: 1;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
article {
|
|
||||||
padding: 0 !important;
|
|
||||||
margin: 0 !important;
|
|
||||||
}
|
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
<div class="top-row ps-3 navbar navbar-dark">
|
|
||||||
<div class="container-fluid">
|
|
||||||
<a class="navbar-brand" href="">EnvelopeGenerator.ReceiverUI</a>
|
|
||||||
<button title="Navigation menu" class="navbar-toggler" @onclick="ToggleNavMenu">
|
|
||||||
<span class="navbar-toggler-icon"></span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="@NavMenuCssClass" @onclick="ToggleNavMenu">
|
|
||||||
<nav class="flex-column">
|
|
||||||
@*
|
|
||||||
<div class="nav-item px-3">
|
|
||||||
<NavLink class="nav-link" href="" Match="NavLinkMatch.All">
|
|
||||||
<span class="oi oi-home" aria-hidden="true"></span> Home
|
|
||||||
</NavLink>
|
|
||||||
</div>
|
|
||||||
<div class="nav-item px-3">
|
|
||||||
<NavLink class="nav-link" href="documentviewer">
|
|
||||||
<span class="oi oi-plus" aria-hidden="true"></span> Document Viewer (JS-Based)
|
|
||||||
</NavLink>
|
|
||||||
</div>
|
|
||||||
*@
|
|
||||||
<div class="nav-item px-3">
|
|
||||||
<NavLink class="nav-link" href="receiver">
|
|
||||||
<span class="oi oi-plus" aria-hidden="true"></span> Empfänger-UI
|
|
||||||
</NavLink>
|
|
||||||
</div>
|
|
||||||
<div class="nav-item px-3">
|
|
||||||
<NavLink class="nav-link" href="sender">
|
|
||||||
<span class="oi oi-plus" aria-hidden="true"></span> Umschlag-UI
|
|
||||||
</NavLink>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@code {
|
|
||||||
private bool collapseNavMenu = true;
|
|
||||||
|
|
||||||
private string NavMenuCssClass => collapseNavMenu ? "collapse" : null;
|
|
||||||
|
|
||||||
private void ToggleNavMenu()
|
|
||||||
{
|
|
||||||
collapseNavMenu = !collapseNavMenu;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
.navbar-toggler {
|
|
||||||
background-color: rgba(255, 255, 255, 0.1);
|
|
||||||
}
|
|
||||||
|
|
||||||
.top-row {
|
|
||||||
height: 3.5rem;
|
|
||||||
background-color: rgba(0,0,0,0.4);
|
|
||||||
}
|
|
||||||
|
|
||||||
.navbar-brand {
|
|
||||||
font-size: 1.1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.oi {
|
|
||||||
width: 2rem;
|
|
||||||
font-size: 1.1rem;
|
|
||||||
vertical-align: text-top;
|
|
||||||
top: -2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item {
|
|
||||||
font-size: 0.9rem;
|
|
||||||
padding-bottom: 0.5rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item:first-of-type {
|
|
||||||
padding-top: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item:last-of-type {
|
|
||||||
padding-bottom: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep a {
|
|
||||||
color: #d7d7d7;
|
|
||||||
border-radius: 4px;
|
|
||||||
height: 3rem;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
line-height: 3rem;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep a.active {
|
|
||||||
background-color: rgba(255,255,255,0.25);
|
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
|
|
||||||
.nav-item ::deep a:hover {
|
|
||||||
background-color: rgba(255,255,255,0.1);
|
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (min-width: 641px) {
|
|
||||||
.navbar-toggler {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sidebar .collapse {
|
|
||||||
/* Never collapse the sidebar for wide screens */
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
@using System.Net.Http
|
|
||||||
@using Microsoft.AspNetCore.Components.Forms
|
|
||||||
@using Microsoft.AspNetCore.Components.Routing
|
|
||||||
@using Microsoft.AspNetCore.Components.Web
|
|
||||||
@using Microsoft.JSInterop
|
|
||||||
@using EnvelopeGenerator.ReceiverUI
|
|
||||||
@using EnvelopeGenerator.ReceiverUI.Shared
|
|
||||||
@using DevExpress.Blazor.Reporting
|
|
||||||
@using DevExpress.Blazor.PdfViewer
|
|
||||||
@using DevExpress.Blazor
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user