Compare commits
31 Commits
feat/save-
...
v1.7.2-rc.
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
faf2bd5384 | ||
|
|
d40ed94b74 | ||
|
|
cd3d9b701b | ||
|
|
e40f47a73c | ||
|
|
64ea4a6f9f | ||
|
|
18115e95d7 | ||
|
|
e736261056 | ||
|
|
2e57da7549 | ||
|
|
574454db0a | ||
|
|
f05b670d93 | ||
|
|
318149fbf3 | ||
|
|
5f19dcf25c | ||
|
|
c99cf4b848 | ||
|
|
18ec40f6af | ||
|
|
ddee8a8272 | ||
|
|
efb2bc94ab | ||
|
|
97ee69e7a0 | ||
|
|
3da344fc5f | ||
|
|
404ca3202f | ||
|
|
c043fa9c06 | ||
|
|
9852e8971f | ||
|
|
5091112e4b | ||
|
|
e76f732990 | ||
|
|
b7c3deb6cd | ||
|
|
08114f7b97 | ||
|
|
6e368cc333 | ||
|
|
4ce4ca3f34 | ||
|
|
7644c0d855 | ||
|
|
fa6453e811 | ||
|
|
f7a20113e5 | ||
|
|
3d644db286 |
48
.cursorrules
Normal file
48
.cursorrules
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
Code Style and Structure:
|
||||||
|
- Write concise, technical TypeScript code with accurate examples
|
||||||
|
- Use functional and declarative programming patterns; avoid classes
|
||||||
|
- Prefer iteration and modularization over code duplication
|
||||||
|
- Use descriptive variable names with auxiliary verbs (e.g., isLoading, hasError)
|
||||||
|
- Structure files: exported component, subcomponents, helpers, static content, types
|
||||||
|
|
||||||
|
Naming Conventions:
|
||||||
|
- Use lowercase with dashes for directories (e.g., components/auth-wizard)
|
||||||
|
- Favor named exports for components
|
||||||
|
|
||||||
|
TypeScript Usage:
|
||||||
|
- Use TypeScript for all code; prefer interfaces over types
|
||||||
|
- Avoid enums; use maps instead
|
||||||
|
- Use functional components with TypeScript interfaces
|
||||||
|
|
||||||
|
Syntax and Formatting:
|
||||||
|
- Use the "function" keyword for pure functions
|
||||||
|
- Avoid unnecessary curly braces in conditionals; use concise syntax for simple statements
|
||||||
|
- Use declarative JSX
|
||||||
|
|
||||||
|
Error Handling and Validation:
|
||||||
|
- Prioritize error handling: handle errors and edge cases early
|
||||||
|
- Use early returns and guard clauses
|
||||||
|
- Implement proper error logging and user-friendly messages
|
||||||
|
- Use Zod for form validation
|
||||||
|
- Model expected errors as return values in Server Actions
|
||||||
|
- Use error boundaries for unexpected errors
|
||||||
|
|
||||||
|
UI and Styling:
|
||||||
|
- Use Shadcn UI, Radix, and Tailwind Aria for components and styling
|
||||||
|
- Implement responsive design with Tailwind CSS; use a mobile-first approach
|
||||||
|
|
||||||
|
Performance Optimization:
|
||||||
|
- Minimize 'use client', 'useEffect', and 'setState'; favor React Server Components (RSC)
|
||||||
|
- Wrap client components in Suspense with fallback
|
||||||
|
- Use dynamic loading for non-critical components
|
||||||
|
- Optimize images: use WebP format, include size data, implement lazy loading
|
||||||
|
|
||||||
|
Key Conventions:
|
||||||
|
- Use 'nuqs' for URL search parameter state management
|
||||||
|
- Optimize Web Vitals (LCP, CLS, FID)
|
||||||
|
- Limit 'use client':
|
||||||
|
- Favor server components and Next.js SSR
|
||||||
|
- Use only for Web API access in small components
|
||||||
|
- Avoid for data fetching or state management
|
||||||
|
|
||||||
|
Follow Next.js docs for Data Fetching, Rendering, and Routing
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
{
|
||||||
|
"index": "Getting Started",
|
||||||
|
"contributing-translations": "Contributing Translations"
|
||||||
|
}
|
||||||
@@ -0,0 +1,69 @@
|
|||||||
|
---
|
||||||
|
title: Contributing Translations
|
||||||
|
description: Learn how to contribute translations to Documenso and become part of our community.
|
||||||
|
---
|
||||||
|
|
||||||
|
import { Callout, Steps } from 'nextra/components';
|
||||||
|
|
||||||
|
# Contributing Translations
|
||||||
|
|
||||||
|
We are always open for help with translations! Currently we utilise AI to generate the initial translations for new languages, which are then improved over time by our awesome community.
|
||||||
|
|
||||||
|
If you are looking for development notes on translations, you can find them [here](/developers/local-development/translations).
|
||||||
|
|
||||||
|
<Callout type="info">
|
||||||
|
Contributions are made through GitHub Pull Requests, so you will need a GitHub account to
|
||||||
|
contribute.
|
||||||
|
</Callout>
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
We store our translations in PO files, which are located in our GitHub repository [here](https://github.com/documenso/documenso/tree/main/packages/lib/translations).
|
||||||
|
|
||||||
|
The translation files are organized into folders represented by their respective language codes (`en` for English, `de` for German, etc). Each language folder contains three PO files:
|
||||||
|
|
||||||
|
1. `web.po`: Translations for the web application
|
||||||
|
2. `marketing.po`: Translations for the marketing application
|
||||||
|
3. `common.po`: Shared translations between web and marketing
|
||||||
|
|
||||||
|
Each PO file contains translations which look like this:
|
||||||
|
|
||||||
|
```po
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/no-longer-available.tsx:61
|
||||||
|
msgid "Want to send slick signing links like this one? <0>Check out Documenso.</0>"
|
||||||
|
msgstr "Möchten Sie auffällige Signatur-Links wie diesen senden? <0>Überprüfen Sie Documenso.</0>"
|
||||||
|
```
|
||||||
|
|
||||||
|
- `msgid`: The original text in English (never edit this manually)
|
||||||
|
- `msgstr`: The translated text in the target language
|
||||||
|
|
||||||
|
<Callout type="warning">
|
||||||
|
Notice the `<0>` tags? These represent HTML elements and must remain in both the `msgid` and `msgstr`. Make sure to translate the content between these tags while keeping the tags intact.
|
||||||
|
</Callout>
|
||||||
|
|
||||||
|
## How to Contribute
|
||||||
|
|
||||||
|
### Updating Existing Translations
|
||||||
|
|
||||||
|
1. Fork the repository.
|
||||||
|
2. Navigate to the appropriate language folder.
|
||||||
|
3. Open the PO file you want to update (web.po, marketing.po, or common.po).
|
||||||
|
4. Make your changes, ensuring you follow the PO file format.
|
||||||
|
5. Commit your changes with a message such as `chore: update German translations`
|
||||||
|
6. Create a Pull Request.
|
||||||
|
|
||||||
|
### Adding a New Language
|
||||||
|
|
||||||
|
If you want to add translations for a language that doesn't exist yet:
|
||||||
|
|
||||||
|
1. Create an issue in our GitHub repository requesting the addition of the new language.
|
||||||
|
2. Wait for our team to review and approve the request.
|
||||||
|
3. Once approved, we will set up the necessary files and kickstart the translations with AI to provide initial coverage.
|
||||||
|
|
||||||
|
## Need Help?
|
||||||
|
|
||||||
|
<Callout type="info">
|
||||||
|
If you have any questions, hop into our [Discord](https://documen.so/discord) and ask us directly!
|
||||||
|
</Callout>
|
||||||
|
|
||||||
|
Thank you for helping make Documenso more accessible to users around the world!
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
---
|
---
|
||||||
title: Contributing Guide
|
title: Getting started
|
||||||
description: Learn how to contribute to Documenso and become part of our community.
|
description: Learn how to contribute to Documenso and become part of our community.
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -10,6 +10,7 @@
|
|||||||
"signing-documents": "Signing Documents",
|
"signing-documents": "Signing Documents",
|
||||||
"templates": "Templates",
|
"templates": "Templates",
|
||||||
"direct-links": "Direct Signing Links",
|
"direct-links": "Direct Signing Links",
|
||||||
|
"document-visibility": "Document Visibility",
|
||||||
"-- Legal Overview": {
|
"-- Legal Overview": {
|
||||||
"type": "separator",
|
"type": "separator",
|
||||||
"title": "Legal Overview"
|
"title": "Legal Overview"
|
||||||
|
|||||||
18
apps/documentation/pages/users/document-visibility.mdx
Normal file
18
apps/documentation/pages/users/document-visibility.mdx
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
---
|
||||||
|
title: Document Visibility
|
||||||
|
description: Learn how to control the visibility of your team documents.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Team's Document Visibility
|
||||||
|
|
||||||
|
By default, all documents created in a team are visible to all team members. However, you can control the visibility of your documents by changing the document's visibility settings.
|
||||||
|
|
||||||
|
To set the visibility of a document, click on the **Document visibility** dropdown in the document's settings panel.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
The document visibility can be set to one of the following options:
|
||||||
|
|
||||||
|
- **Everyone** - The document is visible to all team members.
|
||||||
|
- **Managers and above** - The document is visible to people with the role of Manager or above.
|
||||||
|
- **Admin only** - The document is only visible to the team's admins.
|
||||||
BIN
apps/documentation/public/document-visibility-settings.webp
Normal file
BIN
apps/documentation/public/document-visibility-settings.webp
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 72 KiB |
@@ -0,0 +1,85 @@
|
|||||||
|
---
|
||||||
|
title: 'Customer Story Prisma: 4 Reasons why Prisma chose Documenso for Signatures'
|
||||||
|
description: We are happy to welcome Prisma, another OSS company, as a customer. Read here why they choose us.
|
||||||
|
authorName: 'Timur Ercan'
|
||||||
|
authorImage: '/blog/blog-author-timur.jpeg'
|
||||||
|
authorRole: 'Co-Founder'
|
||||||
|
date: 2024-09-26
|
||||||
|
tags:
|
||||||
|
- Prisma
|
||||||
|
- Customer Story
|
||||||
|
- Open Source
|
||||||
|
---
|
||||||
|
|
||||||
|
<figure>
|
||||||
|
<MdxNextImage
|
||||||
|
src="/blog/prisma.png"
|
||||||
|
width="1200"
|
||||||
|
height="675"
|
||||||
|
alt="Primsa Landing Page We simplify database migration, connection pooling, database queries, and readable data models."
|
||||||
|
/>
|
||||||
|
|
||||||
|
<figcaption className="text-center">
|
||||||
|
Prisma uses Documenso for collaborative team signing.
|
||||||
|
</figcaption>
|
||||||
|
</figure>
|
||||||
|
|
||||||
|
> TLDR; Prisma is now using Documenso, and [we added visibility scopes](https://docs.documenso.com/users/document-visibility)
|
||||||
|
|
||||||
|
# Prisma
|
||||||
|
|
||||||
|
Prisma is an open-source company known for its modern OSS ORM (Object-Relational Mapping) tools that simplify database interactions for developers. Their flagship product, Prisma ORM, provides a type-safe way to query databases like PostgreSQL, MySQL, and many more. With the addition of Prisma Studio, an intuitive database management interface, Prisma makes it easier and more efficient for developers to work with databases. With their new additions, Prisma Pulse and Accelerate, you can react to real-time database changes and optimize your queries. And they are completely [open source](https://github.com/prisma/prisma)!
|
||||||
|
|
||||||
|
# We choose Prisma too!
|
||||||
|
|
||||||
|
I discovered Prisma when planning the tech stack for the [first version of Documenso](https://github.com/documenso/documenso/releases/tag/0.9-developer-preview). Prisma has felt natural to use since day one and has been the base of our database architecture ever since. It's great to see them develop and grow with us.
|
||||||
|
|
||||||
|
# Why they choose us
|
||||||
|
|
||||||
|
## 1. Signature Flows
|
||||||
|
|
||||||
|
Documenso signing flows are highly configurable, designed to adapt to the needs of any document signing process. Whether you're working with different roles, varying settings, or specific delivery methods, Documenso offers the flexibility to suit your requirements. You can choose to send documents via email, share a manual link, generate a link through the API, or even use a static direct link for quick access—all while ensuring a smooth signing experience.
|
||||||
|
|
||||||
|
Additionally, you can create templates to streamline and reuse common workflows, saving valuable time. Direct link templates enable users to drive the flow themselves, providing a straightforward path for signing. For a seamless experience, Documenso also allows you to embed the signing process directly into your website, ensuring an uninterrupted, integrated workflow tailored to your needs.
|
||||||
|
|
||||||
|
## 2. Modern UX
|
||||||
|
|
||||||
|
<figure>
|
||||||
|
<MdxNextImage
|
||||||
|
src="/blog/dsux.png"
|
||||||
|
width="1200"
|
||||||
|
height="675"
|
||||||
|
alt="A completed document in Documenso, ready to download."
|
||||||
|
/>
|
||||||
|
|
||||||
|
<figcaption className="text-center">
|
||||||
|
We call Documenso's design "Happy Minimalism"
|
||||||
|
</figcaption>
|
||||||
|
</figure>
|
||||||
|
|
||||||
|
We’ve crafted Documenso with a sleek, modern interface that makes it incredibly easy to use. Whether you’re signing documents, managing workflows, or fine-tuning settings, its intuitive design allows you to accomplish tasks quickly and effortlessly. More than just powerful, Documenso is a pleasure to navigate—designed to be accessible to everyone, no matter their level of tech experience.
|
||||||
|
|
||||||
|
## 3. Teams
|
||||||
|
|
||||||
|
### Teamwork Makes the Dream Work
|
||||||
|
|
||||||
|
Documenso makes teamwork a breeze with its team management features. You can easily set up and organize teams, making it simple to share and manage documents and workflows together. This is a lifesaver for larger organizations or teams spread across different departments, ensuring everyone stays in sync and on track. Different visibility scopes ensure private documents stay private and others are shared for easy collaboration.
|
||||||
|
|
||||||
|
### Document Visibility
|
||||||
|
|
||||||
|
Collaboration within a team often demands different levels of access to documents. For instance, the Documenso team at Prisma needed a way to set custom visibility on some documents while keeping others accessible to everyone. To address this need, we introduced role-based visibility scopes. This feature allows teams to manage documents more effectively. They can make certain documents visible only to managers or, in special cases, restricted to admins. This ensures sensitive information stays protected while general documents remain accessible to those who need them.
|
||||||
|
|
||||||
|
Learn more about visibility scopes and [how they can benefit your team here](https://docs.documenso.com/users/document-visibility).
|
||||||
|
|
||||||
|
## 4. OSS!
|
||||||
|
|
||||||
|
As you might know, we are open-source! This means you can peek under the hood, tweak things to your liking, and even contribute to making the platform better. We love the community-driven aspect of open-source, and it aligns perfectly with our goal to keep improving and innovating with input from our users.
|
||||||
|
|
||||||
|
So, whether you're looking to streamline your document workflows or just need a solid, reliable platform, Documenso has got your back. And we're thrilled to serve another OSS company and help make the space more open.
|
||||||
|
|
||||||
|
If you have any questions or comments, please reach out on [Twitter / X](https://twitter.com/eltimuro) (DM open) or [Discord](https://documen.so/discord).
|
||||||
|
|
||||||
|
Thinking about switching to a modern signing platform? Reach out anytime: [https://documen.so/sales](https://documen.so/sales)
|
||||||
|
|
||||||
|
Best from Hamburg\
|
||||||
|
Timur
|
||||||
86
apps/marketing/content/blog/go-fork-yourself.mdx
Normal file
86
apps/marketing/content/blog/go-fork-yourself.mdx
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
---
|
||||||
|
title: Go Fork Yourself
|
||||||
|
description: Curious about our take on open-source and code forking? Discover why we see forking not as a threat but as a vital part of the Open Source ecosystem.
|
||||||
|
authorName: 'Timur Ercan'
|
||||||
|
authorImage: '/blog/blog-author-timur.jpeg'
|
||||||
|
authorRole: 'Co-Founder'
|
||||||
|
date: 2024-10-03
|
||||||
|
tags:
|
||||||
|
- Culture
|
||||||
|
- Open Startup
|
||||||
|
- Open Source
|
||||||
|
---
|
||||||
|
|
||||||
|
> TLDR; At Documenso, we see OSS as co-owned by all. Forking—collaborative or not—is part of the open-source spirit.
|
||||||
|
|
||||||
|
## Freedom vs. Ownership
|
||||||
|
|
||||||
|
Recently, there has been a lot of debate on the subject of forks and the usage of OSS IP (Open Source Software Intellectual Property). While I mostly aim to stay out of these controversies (as there is no “winning”), I wanted to take this opportunity to share my views on IP and forking culture here at Documenso. I don’t presume this is the ideal path, but for me, it’s the only path that makes sense.
|
||||||
|
|
||||||
|
What these issues show foremost, in my opinion, is that the concept of Open Source is still evolving. I have heard many say, “Open Source is clearly defined” and that there is no ambiguity anymore. That may be true on the legal side, but there are vast differences in how these rules are interpreted and lived out. Here are a few questions to illustrate the point:
|
||||||
|
|
||||||
|
1. Is it okay to use an open-source project without ever giving back?
|
||||||
|
2. Is it okay to fork (some might say copy) an OSS product and build something on top of it?
|
||||||
|
3. Are we morally obliged to fight those who provide different answers to these questions than we do?
|
||||||
|
|
||||||
|
## Embracing Forks and Collaboration
|
||||||
|
|
||||||
|
Since starting Documenso, I’ve thought a lot about what it actually means to be Open Source for us. So far, it has been about openness in working with everyone, from contributors to customers and sharing our work transparently. For this, we have been richly rewarded with attention and reach. This collaborative give-and-take is what people commonly associate with being Open Source, and it seems ideal.
|
||||||
|
|
||||||
|
Yet, there are the questions mentioned above. And while these may be contentious, my take is straightforward:
|
||||||
|
|
||||||
|
1. Yes.
|
||||||
|
2. Yes.
|
||||||
|
3. No.
|
||||||
|
|
||||||
|
I say this because, to me, the principles of Open Source are rooted in freedom and collaboration. That means allowing others to use, improve, or even compete with what you’ve built without feeling possessive over the code. The beauty of Open Source lies in its openness—its ability to be forked, reused, and adapted by anyone.
|
||||||
|
|
||||||
|
You may answer these questions differently for your own reasons. One thing I’ve found lacking in the discourse is the fact that Open Source is still being treated as socially proprietary. If it’s under an open-source license, you can fork it and try to improve upon the original, and there’s nothing wrong with that. The same is true for closed-source startups. Yet in Open Source, there’s a notion that it’s somehow “dirty,” even though the license explicitly allows it.
|
||||||
|
|
||||||
|
## Forking in Action: Real-World Examples
|
||||||
|
|
||||||
|
When the team behind **Node.js** disagreed with its governance and pace of development, they forked the project to create **io.js**. This wasn’t seen as dirty but as a necessary push for change. In fact, the fork resulted in positive changes—better community governance and faster development—which eventually led to the merge of the two projects under the Node.js Foundation. It shows that forking can be a catalyst for improvement, not just competition.
|
||||||
|
|
||||||
|
## The Misconception of “Exploitative” Usage
|
||||||
|
|
||||||
|
However, sometimes forks don’t merge back but still bring positive change. A good example is **Jenkins**, which was forked from **Hudson** over disagreements in governance after Oracle acquired Sun Microsystems. Jenkins quickly overtook Hudson in terms of community support, development, and innovation. Rather than being seen as a hostile move, the fork enabled Jenkins to become a thriving project, better aligned with the open-source ethos of collaboration and transparency. It emphasizes that forking isn’t inherently exploitative; it can simply be a way to realize a project’s full potential.
|
||||||
|
|
||||||
|
And then there’s **MariaDB**, a fork of **MySQL**. After Oracle acquired MySQL, many in the community feared the project’s open-source nature could be compromised. The fork preserved its spirit, and MariaDB has since grown to become a popular and thriving database. It’s a reminder that sometimes, forking is not just acceptable—it’s necessary to uphold the values and freedoms of open-source software.
|
||||||
|
|
||||||
|
<figure>
|
||||||
|
<MdxNextImage
|
||||||
|
src="/blog/owncode.jpeg"
|
||||||
|
width="1200"
|
||||||
|
height="675"
|
||||||
|
alt="Meme: If everyone owns the code, no one does."
|
||||||
|
/>
|
||||||
|
|
||||||
|
<figcaption className="text-center">
|
||||||
|
Funny Meme to drive the point home.
|
||||||
|
</figcaption>
|
||||||
|
</figure>
|
||||||
|
|
||||||
|
My view is that the code is not “your” code, just as Documenso’s code is not “our” code. It’s been co-owned by the world ever since we published the repo under AGPL V3. That is the whole point. It’s finally not owned by anyone (cue the “everyone/no one” meme). Open Source is for everyone, even competitors. Yet, we are still treating the licenses as extensions of the old, proprietary world and defending perceived injustices based on that model.
|
||||||
|
|
||||||
|
> Side Note: Full compliance with all license and other legal rules is a given here.
|
||||||
|
|
||||||
|
## Documenso’s Approach: Co-Ownership and Community
|
||||||
|
|
||||||
|
So, if you want to fork Documenso and build a business on it, you can. Whether that’s a cool thing to do is another matter. Whether you do a better job than us is also another matter (you won’t). But if you do, I’ll be the first to join. But why not join us from the start since you already have the upside? We exist because we believe this to be the best way forward—not because we force it.
|
||||||
|
|
||||||
|
## The Bigger Picture: Open-Source as Progress
|
||||||
|
|
||||||
|
I’ve also thought a lot about question #3. I understand the impulse to fight anyone who doesn’t appreciate this collaborative approach, but there is no part of this model that backs that up. You are free to “exploit” as long as it’s in a way that adds value. The fallacy is in considering someone else using the OSS part for their business as treason, which it’s not. It’s the whole point.
|
||||||
|
|
||||||
|
While some might say this is theoretical and that reality is different, this is the version of Open Source on which we are building Documenso. The point here is that OSS companies must be resilient to handle forking and competition; without this resilience, an open source driven economy can’t thrive. The focus on freedom and collaboration means being prepared for forks and challenges as part of the growth, not as threats.
|
||||||
|
|
||||||
|
Of course, all of this applies to Documenso, the OSS project, not Documenso Inc., the company, which is very much a privately owned, for-profit entity. However, since the goal is to scale Documenso to the entire world, there is plenty of room to see everyone as co-owners of the Open Source project rather than as competitors. In the end, Open Source is about progress through freedom. If you don’t like how we run things, go fork yourself and hold us accountable. We don’t own this; we just happened to start it.
|
||||||
|
|
||||||
|
> Since this article is open source as well, you are free to fork it and change it here: [https://documen.so/repo](https://documen.so/repo)
|
||||||
|
|
||||||
|
If you have any questions or comments, please reach out on [Twitter / X](https://twitter.com/eltimuro) (DM open) or [Discord](https://documen.so/discord).
|
||||||
|
|
||||||
|
Thinking about switching to a modern signing platform? Reach out anytime: [https://documen.so/sales](https://documen.so/sales)
|
||||||
|
|
||||||
|
Best from Hamburg\
|
||||||
|
Timur
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@documenso/marketing",
|
"name": "@documenso/marketing",
|
||||||
"version": "1.7.1-rc.0",
|
"version": "1.7.2-rc.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -34,7 +34,7 @@
|
|||||||
"micro": "^10.0.1",
|
"micro": "^10.0.1",
|
||||||
"next": "14.2.6",
|
"next": "14.2.6",
|
||||||
"next-auth": "4.24.5",
|
"next-auth": "4.24.5",
|
||||||
"next-axiom": "^1.1.1",
|
"next-axiom": "^1.5.1",
|
||||||
"next-contentlayer": "^0.3.4",
|
"next-contentlayer": "^0.3.4",
|
||||||
"next-plausible": "^3.10.1",
|
"next-plausible": "^3.10.1",
|
||||||
"perfect-freehand": "^1.2.0",
|
"perfect-freehand": "^1.2.0",
|
||||||
|
|||||||
BIN
apps/marketing/public/blog/dsux.png
Normal file
BIN
apps/marketing/public/blog/dsux.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 692 KiB |
BIN
apps/marketing/public/blog/owncode.jpeg
Normal file
BIN
apps/marketing/public/blog/owncode.jpeg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 52 KiB |
BIN
apps/marketing/public/blog/prisma.png
Normal file
BIN
apps/marketing/public/blog/prisma.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 435 KiB |
@@ -19,10 +19,10 @@ export const TEAM_MEMBERS = [
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'Ephraim Atta-Duncan',
|
name: 'Ephraim Atta-Duncan',
|
||||||
role: 'Software Engineer - Intern',
|
role: 'Software Engineer - I',
|
||||||
salary: 15_000,
|
salary: 60_000,
|
||||||
location: 'Ghana',
|
location: 'Ghana',
|
||||||
engagement: msg`Part-Time`,
|
engagement: msg`Full-Time`,
|
||||||
joinDate: 'June 6th, 2023',
|
joinDate: 'June 6th, 2023',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -168,6 +168,7 @@ export const SinglePlayerClient = () => {
|
|||||||
sendStatus: 'NOT_SENT',
|
sendStatus: 'NOT_SENT',
|
||||||
role: 'SIGNER',
|
role: 'SIGNER',
|
||||||
authOptions: null,
|
authOptions: null,
|
||||||
|
signingOrder: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
const onFileDrop = async (file: File) => {
|
const onFileDrop = async (file: File) => {
|
||||||
|
|||||||
@@ -108,14 +108,21 @@ export const Carousel = () => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
setSelectedIndex(emblaApi.selectedScrollSnap());
|
const newIndex = emblaApi.selectedScrollSnap();
|
||||||
emblaThumbsApi.scrollTo(emblaApi.selectedScrollSnap());
|
|
||||||
|
setSelectedIndex(newIndex);
|
||||||
|
emblaThumbsApi.scrollTo(newIndex);
|
||||||
|
|
||||||
resetProgress();
|
resetProgress();
|
||||||
|
|
||||||
|
const currentVideo = videoRefs.current[newIndex];
|
||||||
|
if (currentVideo) {
|
||||||
|
currentVideo.currentTime = 0;
|
||||||
|
}
|
||||||
|
|
||||||
// moduleResolution: bundler breaks this type
|
// moduleResolution: bundler breaks this type
|
||||||
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
||||||
const autoplay = emblaApi.plugins()?.autoplay as unknown as AutoplayType | undefined;
|
const autoplay = emblaApi?.plugins()?.autoplay as unknown as AutoplayType | undefined;
|
||||||
|
|
||||||
if (autoplay) {
|
if (autoplay) {
|
||||||
autoplay.reset();
|
autoplay.reset();
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@documenso/web",
|
"name": "@documenso/web",
|
||||||
"version": "1.7.1-rc.0",
|
"version": "1.7.2-rc.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -37,7 +37,7 @@
|
|||||||
"micro": "^10.0.1",
|
"micro": "^10.0.1",
|
||||||
"next": "14.2.6",
|
"next": "14.2.6",
|
||||||
"next-auth": "4.24.5",
|
"next-auth": "4.24.5",
|
||||||
"next-axiom": "^1.1.1",
|
"next-axiom": "^1.5.1",
|
||||||
"next-plausible": "^3.10.1",
|
"next-plausible": "^3.10.1",
|
||||||
"next-themes": "^0.2.1",
|
"next-themes": "^0.2.1",
|
||||||
"papaparse": "^5.4.1",
|
"papaparse": "^5.4.1",
|
||||||
@@ -53,7 +53,7 @@
|
|||||||
"react-icons": "^4.11.0",
|
"react-icons": "^4.11.0",
|
||||||
"react-rnd": "^10.4.1",
|
"react-rnd": "^10.4.1",
|
||||||
"recharts": "^2.7.2",
|
"recharts": "^2.7.2",
|
||||||
"remeda": "^1.27.1",
|
"remeda": "^2.12.1",
|
||||||
"sharp": "0.32.6",
|
"sharp": "0.32.6",
|
||||||
"ts-pattern": "^5.0.5",
|
"ts-pattern": "^5.0.5",
|
||||||
"ua-parser-js": "^1.0.37",
|
"ua-parser-js": "^1.0.37",
|
||||||
|
|||||||
@@ -12,10 +12,12 @@ import { getDocumentById } from '@documenso/lib/server-only/document/get-documen
|
|||||||
import { getServerComponentFlag } from '@documenso/lib/server-only/feature-flags/get-server-component-feature-flag';
|
import { getServerComponentFlag } from '@documenso/lib/server-only/feature-flags/get-server-component-feature-flag';
|
||||||
import { getFieldsForDocument } from '@documenso/lib/server-only/field/get-fields-for-document';
|
import { getFieldsForDocument } from '@documenso/lib/server-only/field/get-fields-for-document';
|
||||||
import { getRecipientsForDocument } from '@documenso/lib/server-only/recipient/get-recipients-for-document';
|
import { getRecipientsForDocument } from '@documenso/lib/server-only/recipient/get-recipients-for-document';
|
||||||
|
import { DocumentVisibility } from '@documenso/lib/types/document-visibility';
|
||||||
import { symmetricDecrypt } from '@documenso/lib/universal/crypto';
|
import { symmetricDecrypt } from '@documenso/lib/universal/crypto';
|
||||||
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
|
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
|
||||||
import { DocumentStatus } from '@documenso/prisma/client';
|
import { DocumentStatus } from '@documenso/prisma/client';
|
||||||
import type { Team, TeamEmail } from '@documenso/prisma/client';
|
import type { Team, TeamEmail } from '@documenso/prisma/client';
|
||||||
|
import { TeamMemberRole } from '@documenso/prisma/client';
|
||||||
import { Badge } from '@documenso/ui/primitives/badge';
|
import { Badge } from '@documenso/ui/primitives/badge';
|
||||||
import { Button } from '@documenso/ui/primitives/button';
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
import { Card, CardContent } from '@documenso/ui/primitives/card';
|
import { Card, CardContent } from '@documenso/ui/primitives/card';
|
||||||
@@ -39,7 +41,7 @@ export type DocumentPageViewProps = {
|
|||||||
params: {
|
params: {
|
||||||
id: string;
|
id: string;
|
||||||
};
|
};
|
||||||
team?: Team & { teamEmail: TeamEmail | null };
|
team?: Team & { teamEmail: TeamEmail | null } & { currentTeamMember: { role: TeamMemberRole } };
|
||||||
};
|
};
|
||||||
|
|
||||||
export const DocumentPageView = async ({ params, team }: DocumentPageViewProps) => {
|
export const DocumentPageView = async ({ params, team }: DocumentPageViewProps) => {
|
||||||
@@ -62,11 +64,35 @@ export const DocumentPageView = async ({ params, team }: DocumentPageViewProps)
|
|||||||
teamId: team?.id,
|
teamId: team?.id,
|
||||||
}).catch(() => null);
|
}).catch(() => null);
|
||||||
|
|
||||||
|
if (document?.teamId && !team?.url) {
|
||||||
|
redirect(documentRootPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
const documentVisibility = document?.visibility;
|
||||||
|
const currentTeamMemberRole = team?.currentTeamMember?.role;
|
||||||
|
const isRecipient = document?.Recipient.find((recipient) => recipient.email === user.email);
|
||||||
|
let canAccessDocument = true;
|
||||||
|
|
||||||
|
if (team && !isRecipient) {
|
||||||
|
canAccessDocument = match([documentVisibility, currentTeamMemberRole])
|
||||||
|
.with([DocumentVisibility.EVERYONE, TeamMemberRole.ADMIN], () => true)
|
||||||
|
.with([DocumentVisibility.EVERYONE, TeamMemberRole.MANAGER], () => true)
|
||||||
|
.with([DocumentVisibility.EVERYONE, TeamMemberRole.MEMBER], () => true)
|
||||||
|
.with([DocumentVisibility.MANAGER_AND_ABOVE, TeamMemberRole.ADMIN], () => true)
|
||||||
|
.with([DocumentVisibility.MANAGER_AND_ABOVE, TeamMemberRole.MANAGER], () => true)
|
||||||
|
.with([DocumentVisibility.ADMIN, TeamMemberRole.ADMIN], () => true)
|
||||||
|
.otherwise(() => false);
|
||||||
|
}
|
||||||
|
|
||||||
const isDocumentHistoryEnabled = await getServerComponentFlag(
|
const isDocumentHistoryEnabled = await getServerComponentFlag(
|
||||||
'app_document_page_view_history_sheet',
|
'app_document_page_view_history_sheet',
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!document || !document.documentData) {
|
if (!document || !document.documentData || (team && !canAccessDocument)) {
|
||||||
|
redirect(documentRootPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (team && !canAccessDocument) {
|
||||||
redirect(documentRootPath);
|
redirect(documentRootPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -85,6 +85,20 @@ export const EditDocumentForm = ({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const { mutateAsync: setSigningOrderForDocument } =
|
||||||
|
trpc.document.setSigningOrderForDocument.useMutation({
|
||||||
|
...DO_NOT_INVALIDATE_QUERY_ON_MUTATION,
|
||||||
|
onSuccess: (newData) => {
|
||||||
|
utils.document.getDocumentWithDetailsById.setData(
|
||||||
|
{
|
||||||
|
id: initialDocument.id,
|
||||||
|
teamId: team?.id,
|
||||||
|
},
|
||||||
|
(oldData) => ({ ...(oldData || initialDocument), ...newData, id: Number(newData.id) }),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const { mutateAsync: addFields } = trpc.field.addFields.useMutation({
|
const { mutateAsync: addFields } = trpc.field.addFields.useMutation({
|
||||||
...DO_NOT_INVALIDATE_QUERY_ON_MUTATION,
|
...DO_NOT_INVALIDATE_QUERY_ON_MUTATION,
|
||||||
onSuccess: (newFields) => {
|
onSuccess: (newFields) => {
|
||||||
@@ -177,6 +191,7 @@ export const EditDocumentForm = ({
|
|||||||
data: {
|
data: {
|
||||||
title: data.title,
|
title: data.title,
|
||||||
externalId: data.externalId || null,
|
externalId: data.externalId || null,
|
||||||
|
visibility: data.visibility,
|
||||||
globalAccessAuth: data.globalAccessAuth ?? null,
|
globalAccessAuth: data.globalAccessAuth ?? null,
|
||||||
globalActionAuth: data.globalActionAuth ?? null,
|
globalActionAuth: data.globalActionAuth ?? null,
|
||||||
},
|
},
|
||||||
@@ -204,7 +219,13 @@ export const EditDocumentForm = ({
|
|||||||
|
|
||||||
const onAddSignersFormSubmit = async (data: TAddSignersFormSchema) => {
|
const onAddSignersFormSubmit = async (data: TAddSignersFormSchema) => {
|
||||||
try {
|
try {
|
||||||
await addSigners({
|
await Promise.all([
|
||||||
|
setSigningOrderForDocument({
|
||||||
|
documentId: document.id,
|
||||||
|
signingOrder: data.signingOrder,
|
||||||
|
}),
|
||||||
|
|
||||||
|
addSigners({
|
||||||
documentId: document.id,
|
documentId: document.id,
|
||||||
teamId: team?.id,
|
teamId: team?.id,
|
||||||
signers: data.signers.map((signer) => ({
|
signers: data.signers.map((signer) => ({
|
||||||
@@ -212,7 +233,8 @@ export const EditDocumentForm = ({
|
|||||||
// Explicitly set to null to indicate we want to remove auth if required.
|
// Explicitly set to null to indicate we want to remove auth if required.
|
||||||
actionAuth: signer.actionAuth || null,
|
actionAuth: signer.actionAuth || null,
|
||||||
})),
|
})),
|
||||||
});
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
// Router refresh is here to clear the router cache for when navigating to /documents.
|
// Router refresh is here to clear the router cache for when navigating to /documents.
|
||||||
router.refresh();
|
router.refresh();
|
||||||
@@ -339,6 +361,7 @@ export const EditDocumentForm = ({
|
|||||||
key={recipients.length}
|
key={recipients.length}
|
||||||
documentFlow={documentFlow.settings}
|
documentFlow={documentFlow.settings}
|
||||||
document={document}
|
document={document}
|
||||||
|
currentTeamMemberRole={team?.currentTeamMember?.role}
|
||||||
recipients={recipients}
|
recipients={recipients}
|
||||||
fields={fields}
|
fields={fields}
|
||||||
isDocumentEnterprise={isDocumentEnterprise}
|
isDocumentEnterprise={isDocumentEnterprise}
|
||||||
@@ -350,6 +373,7 @@ export const EditDocumentForm = ({
|
|||||||
key={recipients.length}
|
key={recipients.length}
|
||||||
documentFlow={documentFlow.signers}
|
documentFlow={documentFlow.signers}
|
||||||
recipients={recipients}
|
recipients={recipients}
|
||||||
|
signingOrder={document.documentMeta?.signingOrder}
|
||||||
fields={fields}
|
fields={fields}
|
||||||
isDocumentEnterprise={isDocumentEnterprise}
|
isDocumentEnterprise={isDocumentEnterprise}
|
||||||
onSubmit={onAddSignersFormSubmit}
|
onSubmit={onAddSignersFormSubmit}
|
||||||
|
|||||||
@@ -3,14 +3,17 @@ import { redirect } from 'next/navigation';
|
|||||||
|
|
||||||
import { Plural, Trans } from '@lingui/macro';
|
import { Plural, Trans } from '@lingui/macro';
|
||||||
import { ChevronLeft, Users2 } from 'lucide-react';
|
import { ChevronLeft, Users2 } from 'lucide-react';
|
||||||
|
import { match } from 'ts-pattern';
|
||||||
|
|
||||||
import { isUserEnterprise } from '@documenso/ee/server-only/util/is-document-enterprise';
|
import { isUserEnterprise } from '@documenso/ee/server-only/util/is-document-enterprise';
|
||||||
import { DOCUMENSO_ENCRYPTION_KEY } from '@documenso/lib/constants/crypto';
|
import { DOCUMENSO_ENCRYPTION_KEY } from '@documenso/lib/constants/crypto';
|
||||||
import { getRequiredServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
import { getRequiredServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
||||||
import { getDocumentWithDetailsById } from '@documenso/lib/server-only/document/get-document-with-details-by-id';
|
import { getDocumentWithDetailsById } from '@documenso/lib/server-only/document/get-document-with-details-by-id';
|
||||||
|
import { DocumentVisibility } from '@documenso/lib/types/document-visibility';
|
||||||
import { symmetricDecrypt } from '@documenso/lib/universal/crypto';
|
import { symmetricDecrypt } from '@documenso/lib/universal/crypto';
|
||||||
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
|
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
|
||||||
import type { Team } from '@documenso/prisma/client';
|
import type { Team } from '@documenso/prisma/client';
|
||||||
|
import { TeamMemberRole } from '@documenso/prisma/client';
|
||||||
import { DocumentStatus as InternalDocumentStatus } from '@documenso/prisma/client';
|
import { DocumentStatus as InternalDocumentStatus } from '@documenso/prisma/client';
|
||||||
|
|
||||||
import { EditDocumentForm } from '~/app/(dashboard)/documents/[id]/edit-document';
|
import { EditDocumentForm } from '~/app/(dashboard)/documents/[id]/edit-document';
|
||||||
@@ -21,7 +24,7 @@ export type DocumentEditPageViewProps = {
|
|||||||
params: {
|
params: {
|
||||||
id: string;
|
id: string;
|
||||||
};
|
};
|
||||||
team?: Team;
|
team?: Team & { currentTeamMember: { role: TeamMemberRole } };
|
||||||
};
|
};
|
||||||
|
|
||||||
export const DocumentEditPageView = async ({ params, team }: DocumentEditPageViewProps) => {
|
export const DocumentEditPageView = async ({ params, team }: DocumentEditPageViewProps) => {
|
||||||
@@ -43,10 +46,34 @@ export const DocumentEditPageView = async ({ params, team }: DocumentEditPageVie
|
|||||||
teamId: team?.id,
|
teamId: team?.id,
|
||||||
}).catch(() => null);
|
}).catch(() => null);
|
||||||
|
|
||||||
|
if (document?.teamId && !team?.url) {
|
||||||
|
redirect(documentRootPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
const documentVisibility = document?.visibility;
|
||||||
|
const currentTeamMemberRole = team?.currentTeamMember?.role;
|
||||||
|
const isRecipient = document?.Recipient.find((recipient) => recipient.email === user.email);
|
||||||
|
let canAccessDocument = true;
|
||||||
|
|
||||||
|
if (!isRecipient) {
|
||||||
|
canAccessDocument = match([documentVisibility, currentTeamMemberRole])
|
||||||
|
.with([DocumentVisibility.EVERYONE, TeamMemberRole.ADMIN], () => true)
|
||||||
|
.with([DocumentVisibility.EVERYONE, TeamMemberRole.MANAGER], () => true)
|
||||||
|
.with([DocumentVisibility.EVERYONE, TeamMemberRole.MEMBER], () => true)
|
||||||
|
.with([DocumentVisibility.MANAGER_AND_ABOVE, TeamMemberRole.ADMIN], () => true)
|
||||||
|
.with([DocumentVisibility.MANAGER_AND_ABOVE, TeamMemberRole.MANAGER], () => true)
|
||||||
|
.with([DocumentVisibility.ADMIN, TeamMemberRole.ADMIN], () => true)
|
||||||
|
.otherwise(() => false);
|
||||||
|
}
|
||||||
|
|
||||||
if (!document) {
|
if (!document) {
|
||||||
redirect(documentRootPath);
|
redirect(documentRootPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (team && !canAccessDocument) {
|
||||||
|
redirect(documentRootPath);
|
||||||
|
}
|
||||||
|
|
||||||
if (document.status === InternalDocumentStatus.COMPLETED) {
|
if (document.status === InternalDocumentStatus.COMPLETED) {
|
||||||
redirect(`${documentRootPath}/${documentId}`);
|
redirect(`${documentRootPath}/${documentId}`);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,12 +10,13 @@ import type { GetStatsInput } from '@documenso/lib/server-only/document/get-stat
|
|||||||
import { getStats } from '@documenso/lib/server-only/document/get-stats';
|
import { getStats } from '@documenso/lib/server-only/document/get-stats';
|
||||||
import { parseToIntegerArray } from '@documenso/lib/utils/params';
|
import { parseToIntegerArray } from '@documenso/lib/utils/params';
|
||||||
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
|
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
|
||||||
import type { Team, TeamEmail } from '@documenso/prisma/client';
|
import type { Team, TeamEmail, TeamMemberRole } from '@documenso/prisma/client';
|
||||||
import { isExtendedDocumentStatus } from '@documenso/prisma/guards/is-extended-document-status';
|
import { isExtendedDocumentStatus } from '@documenso/prisma/guards/is-extended-document-status';
|
||||||
import { ExtendedDocumentStatus } from '@documenso/prisma/types/extended-document-status';
|
import { ExtendedDocumentStatus } from '@documenso/prisma/types/extended-document-status';
|
||||||
import { Avatar, AvatarFallback, AvatarImage } from '@documenso/ui/primitives/avatar';
|
import { Avatar, AvatarFallback, AvatarImage } from '@documenso/ui/primitives/avatar';
|
||||||
import { Tabs, TabsList, TabsTrigger } from '@documenso/ui/primitives/tabs';
|
import { Tabs, TabsList, TabsTrigger } from '@documenso/ui/primitives/tabs';
|
||||||
|
|
||||||
|
import { DocumentSearch } from '~/components/(dashboard)/document-search/document-search';
|
||||||
import { PeriodSelector } from '~/components/(dashboard)/period-selector/period-selector';
|
import { PeriodSelector } from '~/components/(dashboard)/period-selector/period-selector';
|
||||||
import { isPeriodSelectorValue } from '~/components/(dashboard)/period-selector/types';
|
import { isPeriodSelectorValue } from '~/components/(dashboard)/period-selector/types';
|
||||||
import { DocumentStatus } from '~/components/formatter/document-status';
|
import { DocumentStatus } from '~/components/formatter/document-status';
|
||||||
@@ -25,16 +26,17 @@ import { DataTableSenderFilter } from './data-table-sender-filter';
|
|||||||
import { EmptyDocumentState } from './empty-state';
|
import { EmptyDocumentState } from './empty-state';
|
||||||
import { UploadDocument } from './upload-document';
|
import { UploadDocument } from './upload-document';
|
||||||
|
|
||||||
export type DocumentsPageViewProps = {
|
export interface DocumentsPageViewProps {
|
||||||
searchParams?: {
|
searchParams?: {
|
||||||
status?: ExtendedDocumentStatus;
|
status?: ExtendedDocumentStatus;
|
||||||
period?: PeriodSelectorValue;
|
period?: PeriodSelectorValue;
|
||||||
page?: string;
|
page?: string;
|
||||||
perPage?: string;
|
perPage?: string;
|
||||||
senderIds?: string;
|
senderIds?: string;
|
||||||
|
search?: string;
|
||||||
};
|
};
|
||||||
team?: Team & { teamEmail?: TeamEmail | null };
|
team?: Team & { teamEmail?: TeamEmail | null } & { currentTeamMember?: { role: TeamMemberRole } };
|
||||||
};
|
}
|
||||||
|
|
||||||
export const DocumentsPageView = async ({ searchParams = {}, team }: DocumentsPageViewProps) => {
|
export const DocumentsPageView = async ({ searchParams = {}, team }: DocumentsPageViewProps) => {
|
||||||
const { user } = await getRequiredServerComponentSession();
|
const { user } = await getRequiredServerComponentSession();
|
||||||
@@ -44,13 +46,16 @@ export const DocumentsPageView = async ({ searchParams = {}, team }: DocumentsPa
|
|||||||
const page = Number(searchParams.page) || 1;
|
const page = Number(searchParams.page) || 1;
|
||||||
const perPage = Number(searchParams.perPage) || 20;
|
const perPage = Number(searchParams.perPage) || 20;
|
||||||
const senderIds = parseToIntegerArray(searchParams.senderIds ?? '');
|
const senderIds = parseToIntegerArray(searchParams.senderIds ?? '');
|
||||||
|
const search = searchParams.search || '';
|
||||||
const currentTeam = team
|
const currentTeam = team
|
||||||
? { id: team.id, url: team.url, teamEmail: team.teamEmail?.email }
|
? { id: team.id, url: team.url, teamEmail: team.teamEmail?.email }
|
||||||
: undefined;
|
: undefined;
|
||||||
|
const currentTeamMemberRole = team?.currentTeamMember?.role;
|
||||||
|
|
||||||
const getStatOptions: GetStatsInput = {
|
const getStatOptions: GetStatsInput = {
|
||||||
user,
|
user,
|
||||||
period,
|
period,
|
||||||
|
search,
|
||||||
};
|
};
|
||||||
|
|
||||||
if (team) {
|
if (team) {
|
||||||
@@ -58,6 +63,9 @@ export const DocumentsPageView = async ({ searchParams = {}, team }: DocumentsPa
|
|||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
teamEmail: team.teamEmail?.email,
|
teamEmail: team.teamEmail?.email,
|
||||||
senderIds,
|
senderIds,
|
||||||
|
currentTeamMemberRole,
|
||||||
|
currentUserEmail: user.email,
|
||||||
|
userId: user.id,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,6 +83,7 @@ export const DocumentsPageView = async ({ searchParams = {}, team }: DocumentsPa
|
|||||||
perPage,
|
perPage,
|
||||||
period,
|
period,
|
||||||
senderIds,
|
senderIds,
|
||||||
|
search,
|
||||||
});
|
});
|
||||||
|
|
||||||
const getTabHref = (value: typeof status) => {
|
const getTabHref = (value: typeof status) => {
|
||||||
@@ -131,10 +140,7 @@ export const DocumentsPageView = async ({ searchParams = {}, team }: DocumentsPa
|
|||||||
<DocumentStatus status={value} />
|
<DocumentStatus status={value} />
|
||||||
|
|
||||||
{value !== ExtendedDocumentStatus.ALL && (
|
{value !== ExtendedDocumentStatus.ALL && (
|
||||||
<span className="ml-1 inline-block opacity-50">
|
<span className="ml-1 inline-block opacity-50">{stats[value]}</span>
|
||||||
{Math.min(stats[value], 99)}
|
|
||||||
{stats[value] > 99 && '+'}
|
|
||||||
</span>
|
|
||||||
)}
|
)}
|
||||||
</Link>
|
</Link>
|
||||||
</TabsTrigger>
|
</TabsTrigger>
|
||||||
@@ -147,6 +153,9 @@ export const DocumentsPageView = async ({ searchParams = {}, team }: DocumentsPa
|
|||||||
<div className="flex w-48 flex-wrap items-center justify-between gap-x-2 gap-y-4">
|
<div className="flex w-48 flex-wrap items-center justify-between gap-x-2 gap-y-4">
|
||||||
<PeriodSelector />
|
<PeriodSelector />
|
||||||
</div>
|
</div>
|
||||||
|
<div className="flex w-48 flex-wrap items-center justify-between gap-x-2 gap-y-4">
|
||||||
|
<DocumentSearch initialValue={search} />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ export const UserPasskeysDataTable = () => {
|
|||||||
cell: ({ row }) =>
|
cell: ({ row }) =>
|
||||||
row.original.lastUsedAt
|
row.original.lastUsedAt
|
||||||
? DateTime.fromJSDate(row.original.lastUsedAt).toRelative()
|
? DateTime.fromJSDate(row.original.lastUsedAt).toRelative()
|
||||||
: msg`Never`,
|
: _(msg`Never`),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'actions',
|
id: 'actions',
|
||||||
|
|||||||
@@ -103,6 +103,19 @@ export const EditTemplateForm = ({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const { mutateAsync: setSigningOrderForTemplate } =
|
||||||
|
trpc.template.setSigningOrderForTemplate.useMutation({
|
||||||
|
...DO_NOT_INVALIDATE_QUERY_ON_MUTATION,
|
||||||
|
onSuccess: (newData) => {
|
||||||
|
utils.template.getTemplateWithDetailsById.setData(
|
||||||
|
{
|
||||||
|
id: initialTemplate.id,
|
||||||
|
},
|
||||||
|
(oldData) => ({ ...(oldData || initialTemplate), ...newData }),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const { mutateAsync: addTemplateFields } = trpc.field.addTemplateFields.useMutation({
|
const { mutateAsync: addTemplateFields } = trpc.field.addTemplateFields.useMutation({
|
||||||
...DO_NOT_INVALIDATE_QUERY_ON_MUTATION,
|
...DO_NOT_INVALIDATE_QUERY_ON_MUTATION,
|
||||||
onSuccess: (newData) => {
|
onSuccess: (newData) => {
|
||||||
@@ -160,11 +173,19 @@ export const EditTemplateForm = ({
|
|||||||
data: TAddTemplatePlacholderRecipientsFormSchema,
|
data: TAddTemplatePlacholderRecipientsFormSchema,
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
await addTemplateSigners({
|
await Promise.all([
|
||||||
|
setSigningOrderForTemplate({
|
||||||
|
templateId: template.id,
|
||||||
|
teamId: team?.id,
|
||||||
|
signingOrder: data.signingOrder,
|
||||||
|
}),
|
||||||
|
|
||||||
|
addTemplateSigners({
|
||||||
templateId: template.id,
|
templateId: template.id,
|
||||||
teamId: team?.id,
|
teamId: team?.id,
|
||||||
signers: data.signers,
|
signers: data.signers,
|
||||||
});
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
// Router refresh is here to clear the router cache for when navigating to /documents.
|
// Router refresh is here to clear the router cache for when navigating to /documents.
|
||||||
router.refresh();
|
router.refresh();
|
||||||
@@ -262,6 +283,7 @@ export const EditTemplateForm = ({
|
|||||||
documentFlow={documentFlow.signers}
|
documentFlow={documentFlow.signers}
|
||||||
recipients={recipients}
|
recipients={recipients}
|
||||||
fields={fields}
|
fields={fields}
|
||||||
|
signingOrder={template.templateMeta?.signingOrder}
|
||||||
templateDirectLink={template.directLink}
|
templateDirectLink={template.directLink}
|
||||||
onSubmit={onAddTemplatePlaceholderFormSubmit}
|
onSubmit={onAddTemplatePlaceholderFormSubmit}
|
||||||
isEnterprise={isEnterprise}
|
isEnterprise={isEnterprise}
|
||||||
|
|||||||
@@ -267,14 +267,14 @@ export const CheckboxField = ({
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<div className="flex flex-col gap-y-2">
|
<div className="flex flex-col gap-y-1">
|
||||||
{values?.map((item: { id: number; value: string; checked: boolean }, index: number) => {
|
{values?.map((item: { id: number; value: string; checked: boolean }, index: number) => {
|
||||||
const itemValue = item.value || `empty-value-${item.id}`;
|
const itemValue = item.value || `empty-value-${item.id}`;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div key={index} className="flex items-center gap-x-1.5">
|
<div key={index} className="flex items-center gap-x-1.5">
|
||||||
<Checkbox
|
<Checkbox
|
||||||
className="h-4 w-4"
|
className="h-3 w-3"
|
||||||
checkClassName="text-white"
|
checkClassName="text-white"
|
||||||
id={`checkbox-${index}`}
|
id={`checkbox-${index}`}
|
||||||
checked={field.customText
|
checked={field.customText
|
||||||
@@ -283,7 +283,7 @@ export const CheckboxField = ({
|
|||||||
disabled={isLoading}
|
disabled={isLoading}
|
||||||
onCheckedChange={() => void handleCheckboxOptionClick(item)}
|
onCheckedChange={() => void handleCheckboxOptionClick(item)}
|
||||||
/>
|
/>
|
||||||
<Label htmlFor={`checkbox-${index}`}>
|
<Label htmlFor={`checkbox-${index}`} className="text-xs">
|
||||||
{item.value.includes('empty-value-') ? '' : item.value}
|
{item.value.includes('empty-value-') ? '' : item.value}
|
||||||
</Label>
|
</Label>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -204,14 +204,17 @@ export default async function CompletedSigningPage({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div className="flex flex-col items-center">
|
||||||
{canSignUp && (
|
{canSignUp && (
|
||||||
<div className={`flex max-w-xl flex-col items-center justify-center p-4 md:p-12`}>
|
<div className="flex max-w-xl flex-col items-center justify-center p-4 md:p-12">
|
||||||
<h2 className="mt-8 text-center text-xl font-semibold md:mt-0">
|
<h2 className="mt-8 text-center text-xl font-semibold md:mt-0">
|
||||||
<Trans>Need to sign documents?</Trans>
|
<Trans>Need to sign documents?</Trans>
|
||||||
</h2>
|
</h2>
|
||||||
|
|
||||||
<p className="text-muted-foreground/60 mt-4 max-w-[55ch] text-center leading-normal">
|
<p className="text-muted-foreground/60 mt-4 max-w-[55ch] text-center leading-normal">
|
||||||
<Trans>Create your account and start using state-of-the-art document signing.</Trans>
|
<Trans>
|
||||||
|
Create your account and start using state-of-the-art document signing.
|
||||||
|
</Trans>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<ClaimAccount defaultName={recipientName} defaultEmail={recipient.email} />
|
<ClaimAccount defaultName={recipientName} defaultEmail={recipient.email} />
|
||||||
@@ -219,11 +222,12 @@ export default async function CompletedSigningPage({
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{isLoggedIn && (
|
{isLoggedIn && (
|
||||||
<Link href="/documents" className="text-documenso-700 hover:text-documenso-600 mt-36">
|
<Link href="/documents" className="text-documenso-700 hover:text-documenso-600">
|
||||||
<Trans>Go Back Home</Trans>
|
<Trans>Go Back Home</Trans>
|
||||||
</Link>
|
</Link>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<PollUntilDocumentCompleted document={document} />
|
<PollUntilDocumentCompleted document={document} />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -150,7 +150,7 @@ export const DateField = ({
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<p className="text-muted-foreground dark:text-background/80 text-sm duration-200">
|
<p className="text-muted-foreground dark:text-background/80 text-[clamp(0.625rem,1cqw,0.825rem)] duration-200">
|
||||||
{localDateString}
|
{localDateString}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
import { useRouter } from 'next/navigation';
|
||||||
|
|
||||||
import { Trans } from '@lingui/macro';
|
import { Trans } from '@lingui/macro';
|
||||||
import { DateTime } from 'luxon';
|
|
||||||
import { signOut } from 'next-auth/react';
|
import { signOut } from 'next-auth/react';
|
||||||
|
|
||||||
import { RecipientRole } from '@documenso/prisma/client';
|
import { RecipientRole } from '@documenso/prisma/client';
|
||||||
import { trpc } from '@documenso/trpc/react';
|
|
||||||
import { Alert, AlertDescription } from '@documenso/ui/primitives/alert';
|
import { Alert, AlertDescription } from '@documenso/ui/primitives/alert';
|
||||||
import { Button } from '@documenso/ui/primitives/button';
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
import { DialogFooter } from '@documenso/ui/primitives/dialog';
|
import { DialogFooter } from '@documenso/ui/primitives/dialog';
|
||||||
@@ -25,22 +25,19 @@ export const DocumentActionAuthAccount = ({
|
|||||||
}: DocumentActionAuthAccountProps) => {
|
}: DocumentActionAuthAccountProps) => {
|
||||||
const { recipient } = useRequiredDocumentAuthContext();
|
const { recipient } = useRequiredDocumentAuthContext();
|
||||||
|
|
||||||
const [isSigningOut, setIsSigningOut] = useState(false);
|
const router = useRouter();
|
||||||
|
|
||||||
const { mutateAsync: encryptSecondaryData } = trpc.crypto.encryptSecondaryData.useMutation();
|
const [isSigningOut, setIsSigningOut] = useState(false);
|
||||||
|
|
||||||
const handleChangeAccount = async (email: string) => {
|
const handleChangeAccount = async (email: string) => {
|
||||||
try {
|
try {
|
||||||
setIsSigningOut(true);
|
setIsSigningOut(true);
|
||||||
|
|
||||||
const encryptedEmail = await encryptSecondaryData({
|
await signOut({
|
||||||
data: email,
|
redirect: false,
|
||||||
expiresAt: DateTime.now().plus({ days: 1 }).toMillis(),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await signOut({
|
router.push(`/signin#email=${email}`);
|
||||||
callbackUrl: `/signin?email=${encodeURIComponent(encryptedEmail)}`,
|
|
||||||
});
|
|
||||||
} catch {
|
} catch {
|
||||||
setIsSigningOut(false);
|
setIsSigningOut(false);
|
||||||
|
|
||||||
|
|||||||
@@ -189,7 +189,10 @@ export const DropdownField = ({
|
|||||||
},
|
},
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
<SelectValue placeholder={`${_(msg`Select`)}`} />
|
<SelectValue
|
||||||
|
className="text-[clamp(0.625rem,1cqw,0.825rem)]"
|
||||||
|
placeholder={`${_(msg`Select`)}`}
|
||||||
|
/>
|
||||||
</SelectTrigger>
|
</SelectTrigger>
|
||||||
<SelectContent className="w-full ring-0 focus:ring-0" position="popper">
|
<SelectContent className="w-full ring-0 focus:ring-0" position="popper">
|
||||||
{parsedFieldMeta?.values?.map((item, index) => (
|
{parsedFieldMeta?.values?.map((item, index) => (
|
||||||
@@ -203,7 +206,7 @@ export const DropdownField = ({
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<p className="text-muted-foreground dark:text-background/80 flex items-center justify-center gap-x-1 duration-200">
|
<p className="text-muted-foreground dark:text-background/80 text-[clamp(0.625rem,1cqw,0.825rem)] duration-200">
|
||||||
{field.customText}
|
{field.customText}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -128,7 +128,7 @@ export const EmailField = ({ field, recipient, onSignField, onUnsignField }: Ema
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<p className="text-muted-foreground dark:text-background/80 truncate duration-200">
|
<p className="text-muted-foreground dark:text-background/80 text-[clamp(0.625rem,1cqw,0.825rem)] duration-200">
|
||||||
{field.customText}
|
{field.customText}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -29,9 +29,16 @@ export type SigningFormProps = {
|
|||||||
recipient: Recipient;
|
recipient: Recipient;
|
||||||
fields: Field[];
|
fields: Field[];
|
||||||
redirectUrl?: string | null;
|
redirectUrl?: string | null;
|
||||||
|
isRecipientsTurn: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const SigningForm = ({ document, recipient, fields, redirectUrl }: SigningFormProps) => {
|
export const SigningForm = ({
|
||||||
|
document,
|
||||||
|
recipient,
|
||||||
|
fields,
|
||||||
|
redirectUrl,
|
||||||
|
isRecipientsTurn,
|
||||||
|
}: SigningFormProps) => {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const analytics = useAnalytics();
|
const analytics = useAnalytics();
|
||||||
const { data: session } = useSession();
|
const { data: session } = useSession();
|
||||||
@@ -150,6 +157,7 @@ export const SigningForm = ({ document, recipient, fields, redirectUrl }: Signin
|
|||||||
fields={fields}
|
fields={fields}
|
||||||
fieldsValidated={fieldsValidated}
|
fieldsValidated={fieldsValidated}
|
||||||
role={recipient.role}
|
role={recipient.role}
|
||||||
|
disabled={!isRecipientsTurn}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -213,6 +221,7 @@ export const SigningForm = ({ document, recipient, fields, redirectUrl }: Signin
|
|||||||
fields={fields}
|
fields={fields}
|
||||||
fieldsValidated={fieldsValidated}
|
fieldsValidated={fieldsValidated}
|
||||||
role={recipient.role}
|
role={recipient.role}
|
||||||
|
disabled={!isRecipientsTurn}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -131,7 +131,7 @@ export const InitialsField = ({
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<p className="text-muted-foreground dark:text-background/80 truncate duration-200">
|
<p className="text-muted-foreground dark:text-background/80 text-[clamp(0.625rem,1cqw,0.825rem)] duration-200">
|
||||||
{field.customText}
|
{field.customText}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -172,7 +172,7 @@ export const NameField = ({ field, recipient, onSignField, onUnsignField }: Name
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<p className="text-muted-foreground dark:text-background/80 truncate duration-200">
|
<p className="text-muted-foreground dark:text-background/80 text-[clamp(0.625rem,1cqw,0.825rem)] duration-200">
|
||||||
{field.customText}
|
{field.customText}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -259,7 +259,7 @@ export const NumberField = ({ field, recipient, onSignField, onUnsignField }: Nu
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<p className="text-muted-foreground dark:text-background/80 flex items-center justify-center gap-x-1 duration-200">
|
<p className="text-muted-foreground dark:text-background/80 text-[clamp(0.625rem,1cqw,0.825rem)] duration-200">
|
||||||
{field.customText}
|
{field.customText}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
@@ -267,7 +267,7 @@ export const NumberField = ({ field, recipient, onSignField, onUnsignField }: Nu
|
|||||||
<Dialog open={showRadioModal} onOpenChange={setShowRadioModal}>
|
<Dialog open={showRadioModal} onOpenChange={setShowRadioModal}>
|
||||||
<DialogContent>
|
<DialogContent>
|
||||||
<DialogTitle>
|
<DialogTitle>
|
||||||
{parsedFieldMeta?.label ? parsedFieldMeta?.label : <Trans>Add number</Trans>}
|
{parsedFieldMeta?.label ? parsedFieldMeta?.label : <Trans>Number</Trans>}
|
||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import { isRecipientAuthorized } from '@documenso/lib/server-only/document/is-re
|
|||||||
import { viewedDocument } from '@documenso/lib/server-only/document/viewed-document';
|
import { viewedDocument } from '@documenso/lib/server-only/document/viewed-document';
|
||||||
import { getCompletedFieldsForToken } from '@documenso/lib/server-only/field/get-completed-fields-for-token';
|
import { getCompletedFieldsForToken } from '@documenso/lib/server-only/field/get-completed-fields-for-token';
|
||||||
import { getFieldsForToken } from '@documenso/lib/server-only/field/get-fields-for-token';
|
import { getFieldsForToken } from '@documenso/lib/server-only/field/get-fields-for-token';
|
||||||
|
import { getIsRecipientsTurnToSign } from '@documenso/lib/server-only/recipient/get-is-recipient-turn';
|
||||||
import { getRecipientByToken } from '@documenso/lib/server-only/recipient/get-recipient-by-token';
|
import { getRecipientByToken } from '@documenso/lib/server-only/recipient/get-recipient-by-token';
|
||||||
import { getRecipientSignatures } from '@documenso/lib/server-only/recipient/get-recipient-signatures';
|
import { getRecipientSignatures } from '@documenso/lib/server-only/recipient/get-recipient-signatures';
|
||||||
import { getUserByEmail } from '@documenso/lib/server-only/user/get-user-by-email';
|
import { getUserByEmail } from '@documenso/lib/server-only/user/get-user-by-email';
|
||||||
@@ -42,6 +43,12 @@ export default async function SigningPage({ params: { token } }: SigningPageProp
|
|||||||
|
|
||||||
const requestMetadata = extractNextHeaderRequestMetadata(requestHeaders);
|
const requestMetadata = extractNextHeaderRequestMetadata(requestHeaders);
|
||||||
|
|
||||||
|
const isRecipientsTurn = await getIsRecipientsTurnToSign({ token });
|
||||||
|
|
||||||
|
if (!isRecipientsTurn) {
|
||||||
|
return redirect(`/sign/${token}/waiting`);
|
||||||
|
}
|
||||||
|
|
||||||
const [document, fields, recipient, completedFields] = await Promise.all([
|
const [document, fields, recipient, completedFields] = await Promise.all([
|
||||||
getDocumentAndSenderByToken({
|
getDocumentAndSenderByToken({
|
||||||
token,
|
token,
|
||||||
@@ -146,6 +153,7 @@ export default async function SigningPage({ params: { token } }: SigningPageProp
|
|||||||
document={document}
|
document={document}
|
||||||
fields={fields}
|
fields={fields}
|
||||||
completedFields={completedFields}
|
completedFields={completedFields}
|
||||||
|
isRecipientsTurn={isRecipientsTurn}
|
||||||
/>
|
/>
|
||||||
</DocumentAuthProvider>
|
</DocumentAuthProvider>
|
||||||
</SigningProvider>
|
</SigningProvider>
|
||||||
|
|||||||
@@ -173,16 +173,16 @@ export const RadioField = ({ field, recipient, onSignField, onUnsignField }: Rad
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{field.inserted && (
|
{field.inserted && (
|
||||||
<RadioGroup>
|
<RadioGroup className="gap-y-1">
|
||||||
{values?.map((item, index) => (
|
{values?.map((item, index) => (
|
||||||
<div key={index} className="flex items-center gap-x-1.5">
|
<div key={index} className="flex items-center gap-x-1.5">
|
||||||
<RadioGroupItem
|
<RadioGroupItem
|
||||||
className=""
|
className="h-3 w-3"
|
||||||
value={item.value}
|
value={item.value}
|
||||||
id={`option-${index}`}
|
id={`option-${index}`}
|
||||||
checked={item.value === field.customText}
|
checked={item.value === field.customText}
|
||||||
/>
|
/>
|
||||||
<Label htmlFor={`option-${index}`}>
|
<Label htmlFor={`option-${index}`} className="text-xs">
|
||||||
{item.value.includes('empty-value-') ? '' : item.value}
|
{item.value.includes('empty-value-') ? '' : item.value}
|
||||||
</Label>
|
</Label>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ export type SignDialogProps = {
|
|||||||
fieldsValidated: () => void | Promise<void>;
|
fieldsValidated: () => void | Promise<void>;
|
||||||
onSignatureComplete: () => void | Promise<void>;
|
onSignatureComplete: () => void | Promise<void>;
|
||||||
role: RecipientRole;
|
role: RecipientRole;
|
||||||
|
disabled?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const SignDialog = ({
|
export const SignDialog = ({
|
||||||
@@ -32,6 +33,7 @@ export const SignDialog = ({
|
|||||||
fieldsValidated,
|
fieldsValidated,
|
||||||
onSignatureComplete,
|
onSignatureComplete,
|
||||||
role,
|
role,
|
||||||
|
disabled = false,
|
||||||
}: SignDialogProps) => {
|
}: SignDialogProps) => {
|
||||||
const [showDialog, setShowDialog] = useState(false);
|
const [showDialog, setShowDialog] = useState(false);
|
||||||
const truncatedTitle = truncateTitle(documentTitle);
|
const truncatedTitle = truncateTitle(documentTitle);
|
||||||
@@ -54,6 +56,7 @@ export const SignDialog = ({
|
|||||||
size="lg"
|
size="lg"
|
||||||
onClick={fieldsValidated}
|
onClick={fieldsValidated}
|
||||||
loading={isSubmitting}
|
loading={isSubmitting}
|
||||||
|
disabled={disabled}
|
||||||
>
|
>
|
||||||
{isComplete ? <Trans>Complete</Trans> : <Trans>Next field</Trans>}
|
{isComplete ? <Trans>Complete</Trans> : <Trans>Next field</Trans>}
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -2,12 +2,12 @@
|
|||||||
|
|
||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
import { useRouter } from 'next/navigation';
|
||||||
|
|
||||||
import { Trans, msg } from '@lingui/macro';
|
import { Trans, msg } from '@lingui/macro';
|
||||||
import { useLingui } from '@lingui/react';
|
import { useLingui } from '@lingui/react';
|
||||||
import { DateTime } from 'luxon';
|
|
||||||
import { signOut } from 'next-auth/react';
|
import { signOut } from 'next-auth/react';
|
||||||
|
|
||||||
import { trpc } from '@documenso/trpc/react';
|
|
||||||
import { Button } from '@documenso/ui/primitives/button';
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
import { useToast } from '@documenso/ui/primitives/use-toast';
|
import { useToast } from '@documenso/ui/primitives/use-toast';
|
||||||
|
|
||||||
@@ -20,24 +20,19 @@ export const SigningAuthPageView = ({ email, emailHasAccount }: SigningAuthPageV
|
|||||||
const { _ } = useLingui();
|
const { _ } = useLingui();
|
||||||
const { toast } = useToast();
|
const { toast } = useToast();
|
||||||
|
|
||||||
const [isSigningOut, setIsSigningOut] = useState(false);
|
const router = useRouter();
|
||||||
|
|
||||||
const { mutateAsync: encryptSecondaryData } = trpc.crypto.encryptSecondaryData.useMutation();
|
const [isSigningOut, setIsSigningOut] = useState(false);
|
||||||
|
|
||||||
const handleChangeAccount = async (email: string) => {
|
const handleChangeAccount = async (email: string) => {
|
||||||
try {
|
try {
|
||||||
setIsSigningOut(true);
|
setIsSigningOut(true);
|
||||||
|
|
||||||
const encryptedEmail = await encryptSecondaryData({
|
await signOut({
|
||||||
data: email,
|
redirect: false,
|
||||||
expiresAt: DateTime.now().plus({ days: 1 }).toMillis(),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await signOut({
|
router.push(emailHasAccount ? `/signin#email=${email}` : `/signup#email=${email}`);
|
||||||
callbackUrl: emailHasAccount
|
|
||||||
? `/signin?email=${encodeURIComponent(encryptedEmail)}`
|
|
||||||
: `/signup?email=${encodeURIComponent(encryptedEmail)}`,
|
|
||||||
});
|
|
||||||
} catch {
|
} catch {
|
||||||
toast({
|
toast({
|
||||||
title: _(msg`Something went wrong`),
|
title: _(msg`Something went wrong`),
|
||||||
|
|||||||
@@ -128,7 +128,7 @@ export const SigningFieldContainer = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={cn(type === 'Checkbox' ? 'group' : '')}>
|
<div className={cn('[container-type:size]', type === 'Checkbox' ? 'group' : '')}>
|
||||||
<FieldRootContainer field={field}>
|
<FieldRootContainer field={field}>
|
||||||
{!field.inserted && !loading && !readOnlyField && (
|
{!field.inserted && !loading && !readOnlyField && (
|
||||||
<button
|
<button
|
||||||
|
|||||||
@@ -39,6 +39,7 @@ export type SigningPageViewProps = {
|
|||||||
recipient: Recipient;
|
recipient: Recipient;
|
||||||
fields: Field[];
|
fields: Field[];
|
||||||
completedFields: CompletedField[];
|
completedFields: CompletedField[];
|
||||||
|
isRecipientsTurn: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const SigningPageView = ({
|
export const SigningPageView = ({
|
||||||
@@ -46,6 +47,7 @@ export const SigningPageView = ({
|
|||||||
recipient,
|
recipient,
|
||||||
fields,
|
fields,
|
||||||
completedFields,
|
completedFields,
|
||||||
|
isRecipientsTurn,
|
||||||
}: SigningPageViewProps) => {
|
}: SigningPageViewProps) => {
|
||||||
const { documentData, documentMeta } = document;
|
const { documentData, documentMeta } = document;
|
||||||
|
|
||||||
@@ -99,6 +101,7 @@ export const SigningPageView = ({
|
|||||||
recipient={recipient}
|
recipient={recipient}
|
||||||
fields={fields}
|
fields={fields}
|
||||||
redirectUrl={documentMeta?.redirectUrl}
|
redirectUrl={documentMeta?.redirectUrl}
|
||||||
|
isRecipientsTurn={isRecipientsTurn}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -253,7 +253,7 @@ export const TextField = ({ field, recipient, onSignField, onUnsignField }: Text
|
|||||||
>
|
>
|
||||||
<span className="flex items-center justify-center gap-x-1">
|
<span className="flex items-center justify-center gap-x-1">
|
||||||
<Type />
|
<Type />
|
||||||
{fieldDisplayName || <Trans>Add text</Trans>}
|
{fieldDisplayName || <Trans>Text</Trans>}
|
||||||
</span>
|
</span>
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
@@ -269,7 +269,7 @@ export const TextField = ({ field, recipient, onSignField, onUnsignField }: Text
|
|||||||
<Dialog open={showCustomTextModal} onOpenChange={setShowCustomTextModal}>
|
<Dialog open={showCustomTextModal} onOpenChange={setShowCustomTextModal}>
|
||||||
<DialogContent>
|
<DialogContent>
|
||||||
<DialogTitle>
|
<DialogTitle>
|
||||||
{parsedFieldMeta?.label ? parsedFieldMeta?.label : <Trans>Add Text</Trans>}
|
{parsedFieldMeta?.label ? parsedFieldMeta?.label : <Trans>Text</Trans>}
|
||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
|
|||||||
100
apps/web/src/app/(signing)/sign/[token]/waiting/page.tsx
Normal file
100
apps/web/src/app/(signing)/sign/[token]/waiting/page.tsx
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
import Link from 'next/link';
|
||||||
|
import { notFound, redirect } from 'next/navigation';
|
||||||
|
|
||||||
|
import { Trans } from '@lingui/macro';
|
||||||
|
|
||||||
|
import { setupI18nSSR } from '@documenso/lib/client-only/providers/i18n.server';
|
||||||
|
import { getServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
||||||
|
import { getDocumentById } from '@documenso/lib/server-only/document/get-document-by-id';
|
||||||
|
import { getDocumentAndSenderByToken } from '@documenso/lib/server-only/document/get-document-by-token';
|
||||||
|
import { getRecipientByToken } from '@documenso/lib/server-only/recipient/get-recipient-by-token';
|
||||||
|
import { getTeamById } from '@documenso/lib/server-only/team/get-team';
|
||||||
|
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
|
||||||
|
import type { Team } from '@documenso/prisma/client';
|
||||||
|
import { DocumentStatus } from '@documenso/prisma/client';
|
||||||
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
|
|
||||||
|
type WaitingForTurnToSignPageProps = {
|
||||||
|
params: { token?: string };
|
||||||
|
};
|
||||||
|
|
||||||
|
export default async function WaitingForTurnToSignPage({
|
||||||
|
params: { token },
|
||||||
|
}: WaitingForTurnToSignPageProps) {
|
||||||
|
setupI18nSSR();
|
||||||
|
|
||||||
|
if (!token) {
|
||||||
|
return notFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
const { user } = await getServerComponentSession();
|
||||||
|
|
||||||
|
const [document, recipient] = await Promise.all([
|
||||||
|
getDocumentAndSenderByToken({ token }).catch(() => null),
|
||||||
|
getRecipientByToken({ token }).catch(() => null),
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (!document || !recipient) {
|
||||||
|
return notFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (document.status === DocumentStatus.COMPLETED) {
|
||||||
|
return redirect(`/sign/${token}/complete`);
|
||||||
|
}
|
||||||
|
|
||||||
|
let isOwnerOrTeamMember = false;
|
||||||
|
|
||||||
|
let team: Team | null = null;
|
||||||
|
|
||||||
|
if (user) {
|
||||||
|
isOwnerOrTeamMember = await getDocumentById({
|
||||||
|
id: document.id,
|
||||||
|
userId: user.id,
|
||||||
|
teamId: document.teamId ?? undefined,
|
||||||
|
})
|
||||||
|
.then((document) => !!document)
|
||||||
|
.catch(() => false);
|
||||||
|
|
||||||
|
if (document.teamId) {
|
||||||
|
team = await getTeamById({
|
||||||
|
userId: user.id,
|
||||||
|
teamId: document.teamId,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="relative flex flex-col items-center justify-center px-4 py-12 sm:px-6 lg:px-8">
|
||||||
|
<div className="w-full max-w-md text-center">
|
||||||
|
<h2 className="tracking-tigh text-3xl font-bold">
|
||||||
|
<Trans>Waiting for Your Turn</Trans>
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<p className="text-muted-foreground mt-2 text-sm">
|
||||||
|
<Trans>
|
||||||
|
It's currently not your turn to sign. You will receive an email with instructions once
|
||||||
|
it's your turn to sign the document.
|
||||||
|
</Trans>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p className="text-muted-foreground mt-4 text-sm">
|
||||||
|
<Trans>Please check your email for updates.</Trans>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div className="mt-4">
|
||||||
|
{isOwnerOrTeamMember ? (
|
||||||
|
<Button variant="link" asChild>
|
||||||
|
<Link href={`${formatDocumentsPath(team?.url)}/${document.id}`}>
|
||||||
|
<Trans>Were you trying to edit this document instead?</Trans>
|
||||||
|
</Link>
|
||||||
|
</Button>
|
||||||
|
) : (
|
||||||
|
<Button variant="link" asChild>
|
||||||
|
<Link href="/documents">Return Home</Link>
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import { redirect } from 'next/navigation';
|
|
||||||
|
|
||||||
import { Trans } from '@lingui/macro';
|
import { Trans } from '@lingui/macro';
|
||||||
import { env } from 'next-runtime-env';
|
import { env } from 'next-runtime-env';
|
||||||
@@ -11,7 +10,6 @@ import {
|
|||||||
IS_OIDC_SSO_ENABLED,
|
IS_OIDC_SSO_ENABLED,
|
||||||
OIDC_PROVIDER_LABEL,
|
OIDC_PROVIDER_LABEL,
|
||||||
} from '@documenso/lib/constants/auth';
|
} from '@documenso/lib/constants/auth';
|
||||||
import { decryptSecondaryData } from '@documenso/lib/server-only/crypto/decrypt';
|
|
||||||
|
|
||||||
import { SignInForm } from '~/components/forms/signin';
|
import { SignInForm } from '~/components/forms/signin';
|
||||||
|
|
||||||
@@ -19,24 +17,11 @@ export const metadata: Metadata = {
|
|||||||
title: 'Sign In',
|
title: 'Sign In',
|
||||||
};
|
};
|
||||||
|
|
||||||
type SignInPageProps = {
|
export default function SignInPage() {
|
||||||
searchParams: {
|
|
||||||
email?: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function SignInPage({ searchParams }: SignInPageProps) {
|
|
||||||
setupI18nSSR();
|
setupI18nSSR();
|
||||||
|
|
||||||
const NEXT_PUBLIC_DISABLE_SIGNUP = env('NEXT_PUBLIC_DISABLE_SIGNUP');
|
const NEXT_PUBLIC_DISABLE_SIGNUP = env('NEXT_PUBLIC_DISABLE_SIGNUP');
|
||||||
|
|
||||||
const rawEmail = typeof searchParams.email === 'string' ? searchParams.email : undefined;
|
|
||||||
const email = rawEmail ? decryptSecondaryData(rawEmail) : null;
|
|
||||||
|
|
||||||
if (!email && rawEmail) {
|
|
||||||
redirect('/signin');
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="w-screen max-w-lg px-4">
|
<div className="w-screen max-w-lg px-4">
|
||||||
<div className="border-border dark:bg-background z-10 rounded-xl border bg-neutral-100 p-6">
|
<div className="border-border dark:bg-background z-10 rounded-xl border bg-neutral-100 p-6">
|
||||||
@@ -50,7 +35,6 @@ export default function SignInPage({ searchParams }: SignInPageProps) {
|
|||||||
<hr className="-mx-6 my-4" />
|
<hr className="-mx-6 my-4" />
|
||||||
|
|
||||||
<SignInForm
|
<SignInForm
|
||||||
initialEmail={email || undefined}
|
|
||||||
isGoogleSSOEnabled={IS_GOOGLE_SSO_ENABLED}
|
isGoogleSSOEnabled={IS_GOOGLE_SSO_ENABLED}
|
||||||
isOIDCSSOEnabled={IS_OIDC_SSO_ENABLED}
|
isOIDCSSOEnabled={IS_OIDC_SSO_ENABLED}
|
||||||
oidcProviderLabel={OIDC_PROVIDER_LABEL}
|
oidcProviderLabel={OIDC_PROVIDER_LABEL}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import { env } from 'next-runtime-env';
|
|||||||
|
|
||||||
import { setupI18nSSR } from '@documenso/lib/client-only/providers/i18n.server';
|
import { setupI18nSSR } from '@documenso/lib/client-only/providers/i18n.server';
|
||||||
import { IS_GOOGLE_SSO_ENABLED, IS_OIDC_SSO_ENABLED } from '@documenso/lib/constants/auth';
|
import { IS_GOOGLE_SSO_ENABLED, IS_OIDC_SSO_ENABLED } from '@documenso/lib/constants/auth';
|
||||||
import { decryptSecondaryData } from '@documenso/lib/server-only/crypto/decrypt';
|
|
||||||
|
|
||||||
import { SignUpFormV2 } from '~/components/forms/v2/signup';
|
import { SignUpFormV2 } from '~/components/forms/v2/signup';
|
||||||
|
|
||||||
@@ -13,13 +12,7 @@ export const metadata: Metadata = {
|
|||||||
title: 'Sign Up',
|
title: 'Sign Up',
|
||||||
};
|
};
|
||||||
|
|
||||||
type SignUpPageProps = {
|
export default function SignUpPage() {
|
||||||
searchParams: {
|
|
||||||
email?: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function SignUpPage({ searchParams }: SignUpPageProps) {
|
|
||||||
setupI18nSSR();
|
setupI18nSSR();
|
||||||
|
|
||||||
const NEXT_PUBLIC_DISABLE_SIGNUP = env('NEXT_PUBLIC_DISABLE_SIGNUP');
|
const NEXT_PUBLIC_DISABLE_SIGNUP = env('NEXT_PUBLIC_DISABLE_SIGNUP');
|
||||||
@@ -28,17 +21,9 @@ export default function SignUpPage({ searchParams }: SignUpPageProps) {
|
|||||||
redirect('/signin');
|
redirect('/signin');
|
||||||
}
|
}
|
||||||
|
|
||||||
const rawEmail = typeof searchParams.email === 'string' ? searchParams.email : undefined;
|
|
||||||
const email = rawEmail ? decryptSecondaryData(rawEmail) : null;
|
|
||||||
|
|
||||||
if (!email && rawEmail) {
|
|
||||||
redirect('/signup');
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SignUpFormV2
|
<SignUpFormV2
|
||||||
className="w-screen max-w-screen-2xl px-4 md:px-16 lg:-my-16"
|
className="w-screen max-w-screen-2xl px-4 md:px-16 lg:-my-16"
|
||||||
initialEmail={email || undefined}
|
|
||||||
isGoogleSSOEnabled={IS_GOOGLE_SSO_ENABLED}
|
isGoogleSSOEnabled={IS_GOOGLE_SSO_ENABLED}
|
||||||
isOIDCSSOEnabled={IS_OIDC_SSO_ENABLED}
|
isOIDCSSOEnabled={IS_OIDC_SSO_ENABLED}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -1,5 +1,8 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
export const ZBaseEmbedDataSchema = z.object({
|
export const ZBaseEmbedDataSchema = z.object({
|
||||||
css: z.string().optional().transform(value => value || undefined),
|
css: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.transform((value) => value || undefined),
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -18,15 +18,18 @@ export const EmbedDocumentCompleted = ({ name, signature }: EmbedDocumentComplet
|
|||||||
|
|
||||||
<div className="mt-8 w-full max-w-md">
|
<div className="mt-8 w-full max-w-md">
|
||||||
<SigningCard3D
|
<SigningCard3D
|
||||||
className='w-full mx-auto'
|
className="mx-auto w-full"
|
||||||
name={name || 'Documenso'}
|
name={name || 'Documenso'}
|
||||||
signature={signature}
|
signature={signature}
|
||||||
signingCelebrationImage={signingCelebration}
|
signingCelebrationImage={signingCelebration}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<p className="mt-8 max-w-[50ch] text-center text-muted-foreground text-sm">
|
<p className="text-muted-foreground mt-8 max-w-[50ch] text-center text-sm">
|
||||||
<Trans>The document is now completed, please follow any instructions provided within the parent application.</Trans>
|
<Trans>
|
||||||
|
The document is now completed, please follow any instructions provided within the parent
|
||||||
|
application.
|
||||||
|
</Trans>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import { useSearchParams } from 'next/navigation';
|
|||||||
|
|
||||||
import { Trans, msg } from '@lingui/macro';
|
import { Trans, msg } from '@lingui/macro';
|
||||||
import { useLingui } from '@lingui/react';
|
import { useLingui } from '@lingui/react';
|
||||||
|
import { LucideChevronDown, LucideChevronUp } from 'lucide-react';
|
||||||
import { DateTime } from 'luxon';
|
import { DateTime } from 'luxon';
|
||||||
|
|
||||||
import { useThrottleFn } from '@documenso/lib/client-only/hooks/use-throttle-fn';
|
import { useThrottleFn } from '@documenso/lib/client-only/hooks/use-throttle-fn';
|
||||||
@@ -14,7 +15,7 @@ import { PDF_VIEWER_PAGE_SELECTOR } from '@documenso/lib/constants/pdf-viewer';
|
|||||||
import { DEFAULT_DOCUMENT_TIME_ZONE } from '@documenso/lib/constants/time-zones';
|
import { DEFAULT_DOCUMENT_TIME_ZONE } from '@documenso/lib/constants/time-zones';
|
||||||
import { validateFieldsInserted } from '@documenso/lib/utils/fields';
|
import { validateFieldsInserted } from '@documenso/lib/utils/fields';
|
||||||
import type { DocumentMeta, Recipient, TemplateMeta } from '@documenso/prisma/client';
|
import type { DocumentMeta, Recipient, TemplateMeta } from '@documenso/prisma/client';
|
||||||
import { FieldType, type DocumentData, type Field } from '@documenso/prisma/client';
|
import { type DocumentData, type Field, FieldType } from '@documenso/prisma/client';
|
||||||
import { trpc } from '@documenso/trpc/react';
|
import { trpc } from '@documenso/trpc/react';
|
||||||
import type {
|
import type {
|
||||||
TRemovedSignedFieldWithTokenMutationSchema,
|
TRemovedSignedFieldWithTokenMutationSchema,
|
||||||
@@ -34,7 +35,6 @@ import type { DirectTemplateLocalField } from '~/app/(recipient)/d/[token]/sign-
|
|||||||
import { useRequiredSigningContext } from '~/app/(signing)/sign/[token]/provider';
|
import { useRequiredSigningContext } from '~/app/(signing)/sign/[token]/provider';
|
||||||
import { Logo } from '~/components/branding/logo';
|
import { Logo } from '~/components/branding/logo';
|
||||||
|
|
||||||
import { LucideChevronDown, LucideChevronUp } from 'lucide-react';
|
|
||||||
import { EmbedClientLoading } from '../../client-loading';
|
import { EmbedClientLoading } from '../../client-loading';
|
||||||
import { EmbedDocumentCompleted } from '../../completed';
|
import { EmbedDocumentCompleted } from '../../completed';
|
||||||
import { EmbedDocumentFields } from '../../document-fields';
|
import { EmbedDocumentFields } from '../../document-fields';
|
||||||
@@ -307,7 +307,7 @@ export const EmbedDirectTemplateClientPage = ({
|
|||||||
<div className="relative mx-auto flex min-h-[100dvh] max-w-screen-lg flex-col items-center justify-center p-6">
|
<div className="relative mx-auto flex min-h-[100dvh] max-w-screen-lg flex-col items-center justify-center p-6">
|
||||||
{(!hasFinishedInit || !hasDocumentLoaded) && <EmbedClientLoading />}
|
{(!hasFinishedInit || !hasDocumentLoaded) && <EmbedClientLoading />}
|
||||||
|
|
||||||
<div className="relative flex flex-col md:flex-row w-full gap-x-6 gap-y-12">
|
<div className="relative flex w-full flex-col gap-x-6 gap-y-12 md:flex-row">
|
||||||
{/* Viewer */}
|
{/* Viewer */}
|
||||||
<div className="flex-1">
|
<div className="flex-1">
|
||||||
<LazyPDFViewer
|
<LazyPDFViewer
|
||||||
@@ -318,26 +318,26 @@ export const EmbedDirectTemplateClientPage = ({
|
|||||||
|
|
||||||
{/* Widget */}
|
{/* Widget */}
|
||||||
<div
|
<div
|
||||||
className="group/document-widget fixed md:sticky md:top-4 left-0 w-full bottom-8 px-6 md:px-0 z-50 md:z-auto md:w-[350px] flex-shrink-0 h-fit"
|
className="group/document-widget fixed bottom-8 left-0 z-50 h-fit w-full flex-shrink-0 px-6 md:sticky md:top-4 md:z-auto md:w-[350px] md:px-0"
|
||||||
data-expanded={isExpanded || undefined}
|
data-expanded={isExpanded || undefined}
|
||||||
>
|
>
|
||||||
<div className="w-full border-border bg-widget flex md:min-h-[min(calc(100dvh-2rem),48rem)] h-fit flex-col rounded-xl border px-4 py-4 md:py-6">
|
<div className="border-border bg-widget flex h-fit w-full flex-col rounded-xl border px-4 py-4 md:min-h-[min(calc(100dvh-2rem),48rem)] md:py-6">
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<div>
|
<div>
|
||||||
<div className="flex items-center justify-between gap-x-2">
|
<div className="flex items-center justify-between gap-x-2">
|
||||||
<h3 className="text-foreground text-xl md:text-2xl font-semibold">
|
<h3 className="text-foreground text-xl font-semibold md:text-2xl">
|
||||||
<Trans>Sign document</Trans>
|
<Trans>Sign document</Trans>
|
||||||
</h3>
|
</h3>
|
||||||
|
|
||||||
<Button variant="outline" className="h-8 w-8 p-0 md:hidden">
|
<Button variant="outline" className="h-8 w-8 p-0 md:hidden">
|
||||||
{isExpanded ? (
|
{isExpanded ? (
|
||||||
<LucideChevronDown
|
<LucideChevronDown
|
||||||
className="h-5 w-5 text-muted-foreground"
|
className="text-muted-foreground h-5 w-5"
|
||||||
onClick={() => setIsExpanded(false)}
|
onClick={() => setIsExpanded(false)}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<LucideChevronUp
|
<LucideChevronUp
|
||||||
className="h-5 w-5 text-muted-foreground"
|
className="text-muted-foreground h-5 w-5"
|
||||||
onClick={() => setIsExpanded(true)}
|
onClick={() => setIsExpanded(true)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
@@ -354,7 +354,7 @@ export const EmbedDirectTemplateClientPage = ({
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Form */}
|
{/* Form */}
|
||||||
<div className="-mx-2 px-2 hidden group-data-[expanded]/document-widget:block md:block">
|
<div className="-mx-2 hidden px-2 group-data-[expanded]/document-widget:block md:block">
|
||||||
<div className="flex flex-1 flex-col gap-y-4">
|
<div className="flex flex-1 flex-col gap-y-4">
|
||||||
<div>
|
<div>
|
||||||
<Label htmlFor="full-name">
|
<Label htmlFor="full-name">
|
||||||
@@ -408,9 +408,9 @@ export const EmbedDirectTemplateClientPage = ({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex-1 hidden group-data-[expanded]/document-widget:block md:block" />
|
<div className="hidden flex-1 group-data-[expanded]/document-widget:block md:block" />
|
||||||
|
|
||||||
<div className="w-full grid-cols-2 items-center mt-4 hidden group-data-[expanded]/document-widget:grid md:grid">
|
<div className="mt-4 hidden w-full grid-cols-2 items-center group-data-[expanded]/document-widget:grid md:grid">
|
||||||
{pendingFields.length > 0 ? (
|
{pendingFields.length > 0 ? (
|
||||||
<Button className="col-start-2" onClick={() => onNextFieldClick()}>
|
<Button className="col-start-2" onClick={() => onNextFieldClick()}>
|
||||||
<Trans>Next</Trans>
|
<Trans>Next</Trans>
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
export default function EmbedDirectTemplateNotFound() {
|
export default function EmbedDirectTemplateNotFound() {
|
||||||
return <div>Not Found</div>
|
return <div>Not Found</div>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,11 +73,7 @@ export default async function EmbedDirectTemplatePage({ params }: EmbedDirectTem
|
|||||||
const fields = template.Field.filter((field) => field.recipientId === directTemplateRecipientId);
|
const fields = template.Field.filter((field) => field.recipientId === directTemplateRecipientId);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SigningProvider
|
<SigningProvider email={user?.email} fullName={user?.name} signature={user?.signature}>
|
||||||
email={user?.email}
|
|
||||||
fullName={user?.name}
|
|
||||||
signature={user?.signature}
|
|
||||||
>
|
|
||||||
<DocumentAuthProvider
|
<DocumentAuthProvider
|
||||||
documentAuthOptions={template.authOptions}
|
documentAuthOptions={template.authOptions}
|
||||||
recipient={recipient}
|
recipient={recipient}
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
ZTextFieldMeta,
|
ZTextFieldMeta,
|
||||||
} from '@documenso/lib/types/field-meta';
|
} from '@documenso/lib/types/field-meta';
|
||||||
import type { DocumentMeta, Recipient, TemplateMeta } from '@documenso/prisma/client';
|
import type { DocumentMeta, Recipient, TemplateMeta } from '@documenso/prisma/client';
|
||||||
import { FieldType, type Field } from '@documenso/prisma/client';
|
import { type Field, FieldType } from '@documenso/prisma/client';
|
||||||
import type { FieldWithSignatureAndFieldMeta } from '@documenso/prisma/types/field-with-signature-and-fieldmeta';
|
import type { FieldWithSignatureAndFieldMeta } from '@documenso/prisma/types/field-with-signature-and-fieldmeta';
|
||||||
import type {
|
import type {
|
||||||
TRemovedSignedFieldWithTokenMutationSchema,
|
TRemovedSignedFieldWithTokenMutationSchema,
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
export const EmbedPaywall = () => {
|
export const EmbedPaywall = () => {
|
||||||
return <div>
|
return (
|
||||||
|
<div>
|
||||||
<h1>Paywall</h1>
|
<h1>Paywall</h1>
|
||||||
</div>
|
</div>
|
||||||
}
|
);
|
||||||
|
};
|
||||||
|
|||||||
@@ -1,15 +1,17 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
import { Trans, msg } from '@lingui/macro';
|
||||||
|
import { useLingui } from '@lingui/react';
|
||||||
|
import { LucideChevronDown, LucideChevronUp } from 'lucide-react';
|
||||||
|
|
||||||
import { useThrottleFn } from '@documenso/lib/client-only/hooks/use-throttle-fn';
|
import { useThrottleFn } from '@documenso/lib/client-only/hooks/use-throttle-fn';
|
||||||
import { PDF_VIEWER_PAGE_SELECTOR } from '@documenso/lib/constants/pdf-viewer';
|
import { PDF_VIEWER_PAGE_SELECTOR } from '@documenso/lib/constants/pdf-viewer';
|
||||||
import { validateFieldsInserted } from '@documenso/lib/utils/fields';
|
import { validateFieldsInserted } from '@documenso/lib/utils/fields';
|
||||||
import type { DocumentMeta, Recipient, TemplateMeta } from '@documenso/prisma/client';
|
import type { DocumentMeta, Recipient, TemplateMeta } from '@documenso/prisma/client';
|
||||||
import { type DocumentData, type Field } from '@documenso/prisma/client';
|
import { type DocumentData, type Field } from '@documenso/prisma/client';
|
||||||
import { trpc } from '@documenso/trpc/react';
|
import { trpc } from '@documenso/trpc/react';
|
||||||
import { Trans, msg } from '@lingui/macro';
|
|
||||||
import { useLingui } from '@lingui/react';
|
|
||||||
import { useEffect, useState } from 'react';
|
|
||||||
|
|
||||||
import { FieldToolTip } from '@documenso/ui/components/field/field-tooltip';
|
import { FieldToolTip } from '@documenso/ui/components/field/field-tooltip';
|
||||||
import { Button } from '@documenso/ui/primitives/button';
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
import { Card, CardContent } from '@documenso/ui/primitives/card';
|
import { Card, CardContent } from '@documenso/ui/primitives/card';
|
||||||
@@ -20,9 +22,9 @@ import { LazyPDFViewer } from '@documenso/ui/primitives/lazy-pdf-viewer';
|
|||||||
import { SignaturePad } from '@documenso/ui/primitives/signature-pad';
|
import { SignaturePad } from '@documenso/ui/primitives/signature-pad';
|
||||||
import { useToast } from '@documenso/ui/primitives/use-toast';
|
import { useToast } from '@documenso/ui/primitives/use-toast';
|
||||||
|
|
||||||
import { LucideChevronDown, LucideChevronUp } from 'lucide-react';
|
|
||||||
import { useRequiredSigningContext } from '~/app/(signing)/sign/[token]/provider';
|
import { useRequiredSigningContext } from '~/app/(signing)/sign/[token]/provider';
|
||||||
import { Logo } from '~/components/branding/logo';
|
import { Logo } from '~/components/branding/logo';
|
||||||
|
|
||||||
import { EmbedClientLoading } from '../../client-loading';
|
import { EmbedClientLoading } from '../../client-loading';
|
||||||
import { EmbedDocumentCompleted } from '../../completed';
|
import { EmbedDocumentCompleted } from '../../completed';
|
||||||
import { EmbedDocumentFields } from '../../document-fields';
|
import { EmbedDocumentFields } from '../../document-fields';
|
||||||
@@ -185,7 +187,7 @@ export const EmbedSignDocumentClientPage = ({
|
|||||||
<div className="relative mx-auto flex min-h-[100dvh] max-w-screen-lg flex-col items-center justify-center p-6">
|
<div className="relative mx-auto flex min-h-[100dvh] max-w-screen-lg flex-col items-center justify-center p-6">
|
||||||
{(!hasFinishedInit || !hasDocumentLoaded) && <EmbedClientLoading />}
|
{(!hasFinishedInit || !hasDocumentLoaded) && <EmbedClientLoading />}
|
||||||
|
|
||||||
<div className="relative flex flex-col md:flex-row w-full gap-x-6 gap-y-12">
|
<div className="relative flex w-full flex-col gap-x-6 gap-y-12 md:flex-row">
|
||||||
{/* Viewer */}
|
{/* Viewer */}
|
||||||
<div className="flex-1">
|
<div className="flex-1">
|
||||||
<LazyPDFViewer
|
<LazyPDFViewer
|
||||||
@@ -196,26 +198,26 @@ export const EmbedSignDocumentClientPage = ({
|
|||||||
|
|
||||||
{/* Widget */}
|
{/* Widget */}
|
||||||
<div
|
<div
|
||||||
className="group/document-widget fixed md:sticky md:top-4 left-0 w-full bottom-8 px-6 md:px-0 z-50 md:z-auto md:w-[350px] flex-shrink-0 h-fit"
|
className="group/document-widget fixed bottom-8 left-0 z-50 h-fit w-full flex-shrink-0 px-6 md:sticky md:top-4 md:z-auto md:w-[350px] md:px-0"
|
||||||
data-expanded={isExpanded || undefined}
|
data-expanded={isExpanded || undefined}
|
||||||
>
|
>
|
||||||
<div className="w-full border-border bg-widget flex flex-col rounded-xl border px-4 py-4 md:py-6">
|
<div className="border-border bg-widget flex w-full flex-col rounded-xl border px-4 py-4 md:py-6">
|
||||||
{/* Header */}
|
{/* Header */}
|
||||||
<div>
|
<div>
|
||||||
<div className="flex items-center justify-between gap-x-2">
|
<div className="flex items-center justify-between gap-x-2">
|
||||||
<h3 className="text-foreground text-xl md:text-2xl font-semibold">
|
<h3 className="text-foreground text-xl font-semibold md:text-2xl">
|
||||||
<Trans>Sign document</Trans>
|
<Trans>Sign document</Trans>
|
||||||
</h3>
|
</h3>
|
||||||
|
|
||||||
<Button variant="outline" className="h-8 w-8 p-0 md:hidden">
|
<Button variant="outline" className="h-8 w-8 p-0 md:hidden">
|
||||||
{isExpanded ? (
|
{isExpanded ? (
|
||||||
<LucideChevronDown
|
<LucideChevronDown
|
||||||
className="h-5 w-5 text-muted-foreground"
|
className="text-muted-foreground h-5 w-5"
|
||||||
onClick={() => setIsExpanded(false)}
|
onClick={() => setIsExpanded(false)}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<LucideChevronUp
|
<LucideChevronUp
|
||||||
className="h-5 w-5 text-muted-foreground"
|
className="text-muted-foreground h-5 w-5"
|
||||||
onClick={() => setIsExpanded(true)}
|
onClick={() => setIsExpanded(true)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
@@ -232,7 +234,7 @@ export const EmbedSignDocumentClientPage = ({
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Form */}
|
{/* Form */}
|
||||||
<div className="-mx-2 px-2 hidden group-data-[expanded]/document-widget:block md:block">
|
<div className="-mx-2 hidden px-2 group-data-[expanded]/document-widget:block md:block">
|
||||||
<div className="flex flex-1 flex-col gap-y-4">
|
<div className="flex flex-1 flex-col gap-y-4">
|
||||||
<div>
|
<div>
|
||||||
<Label htmlFor="full-name">
|
<Label htmlFor="full-name">
|
||||||
@@ -285,9 +287,9 @@ export const EmbedSignDocumentClientPage = ({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex-1 hidden group-data-[expanded]/document-widget:block md:block" />
|
<div className="hidden flex-1 group-data-[expanded]/document-widget:block md:block" />
|
||||||
|
|
||||||
<div className="w-full grid-cols-2 items-center mt-4 hidden group-data-[expanded]/document-widget:grid md:grid">
|
<div className="mt-4 hidden w-full grid-cols-2 items-center group-data-[expanded]/document-widget:grid md:grid">
|
||||||
{pendingFields.length > 0 ? (
|
{pendingFields.length > 0 ? (
|
||||||
<Button className="col-start-2" onClick={() => onNextFieldClick()}>
|
<Button className="col-start-2" onClick={() => onNextFieldClick()}>
|
||||||
<Trans>Next</Trans>
|
<Trans>Next</Trans>
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
export default function EmbedDirectTemplateNotFound() {
|
export default function EmbedDirectTemplateNotFound() {
|
||||||
return <div>Not Found</div>
|
return <div>Not Found</div>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,8 +4,12 @@ import { match } from 'ts-pattern';
|
|||||||
|
|
||||||
import { IS_BILLING_ENABLED } from '@documenso/lib/constants/app';
|
import { IS_BILLING_ENABLED } from '@documenso/lib/constants/app';
|
||||||
import { getServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
import { getServerComponentSession } from '@documenso/lib/next-auth/get-server-component-session';
|
||||||
|
import { getDocumentAndSenderByToken } from '@documenso/lib/server-only/document/get-document-by-token';
|
||||||
|
import { getFieldsForToken } from '@documenso/lib/server-only/field/get-fields-for-token';
|
||||||
|
import { getRecipientByToken } from '@documenso/lib/server-only/recipient/get-recipient-by-token';
|
||||||
import { DocumentAccessAuth } from '@documenso/lib/types/document-auth';
|
import { DocumentAccessAuth } from '@documenso/lib/types/document-auth';
|
||||||
import { extractDocumentAuthMethods } from '@documenso/lib/utils/document-auth';
|
import { extractDocumentAuthMethods } from '@documenso/lib/utils/document-auth';
|
||||||
|
import { DocumentStatus } from '@documenso/prisma/client';
|
||||||
|
|
||||||
import { DocumentAuthProvider } from '~/app/(signing)/sign/[token]/document-auth-provider';
|
import { DocumentAuthProvider } from '~/app/(signing)/sign/[token]/document-auth-provider';
|
||||||
import { SigningProvider } from '~/app/(signing)/sign/[token]/provider';
|
import { SigningProvider } from '~/app/(signing)/sign/[token]/provider';
|
||||||
@@ -13,10 +17,6 @@ import { SigningProvider } from '~/app/(signing)/sign/[token]/provider';
|
|||||||
import { EmbedAuthenticateView } from '../../authenticate';
|
import { EmbedAuthenticateView } from '../../authenticate';
|
||||||
import { EmbedPaywall } from '../../paywall';
|
import { EmbedPaywall } from '../../paywall';
|
||||||
import { EmbedSignDocumentClientPage } from './client';
|
import { EmbedSignDocumentClientPage } from './client';
|
||||||
import { getFieldsForToken } from '@documenso/lib/server-only/field/get-fields-for-token';
|
|
||||||
import { getRecipientByToken } from '@documenso/lib/server-only/recipient/get-recipient-by-token';
|
|
||||||
import { getDocumentAndSenderByToken } from '@documenso/lib/server-only/document/get-document-by-token';
|
|
||||||
import { DocumentStatus } from '@documenso/prisma/client';
|
|
||||||
|
|
||||||
export type EmbedSignDocumentPageProps = {
|
export type EmbedSignDocumentPageProps = {
|
||||||
params: {
|
params: {
|
||||||
@@ -66,7 +66,12 @@ export default async function EmbedSignDocumentPage({ params }: EmbedSignDocumen
|
|||||||
.exhaustive();
|
.exhaustive();
|
||||||
|
|
||||||
if (!isAccessAuthValid) {
|
if (!isAccessAuthValid) {
|
||||||
return <EmbedAuthenticateView email={user?.email || recipient.email} returnTo={`/embed/direct/${token}`} />;
|
return (
|
||||||
|
<EmbedAuthenticateView
|
||||||
|
email={user?.email || recipient.email}
|
||||||
|
returnTo={`/embed/direct/${token}`}
|
||||||
|
/>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -0,0 +1,41 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useCallback, useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
import { useRouter, useSearchParams } from 'next/navigation';
|
||||||
|
|
||||||
|
import { useDebouncedValue } from '@documenso/lib/client-only/hooks/use-debounced-value';
|
||||||
|
import { Input } from '@documenso/ui/primitives/input';
|
||||||
|
|
||||||
|
export const DocumentSearch = ({ initialValue = '' }: { initialValue?: string }) => {
|
||||||
|
const router = useRouter();
|
||||||
|
const searchParams = useSearchParams();
|
||||||
|
const [searchTerm, setSearchTerm] = useState(initialValue);
|
||||||
|
const debouncedSearchTerm = useDebouncedValue(searchTerm, 500);
|
||||||
|
|
||||||
|
const handleSearch = useCallback(
|
||||||
|
(term: string) => {
|
||||||
|
const params = new URLSearchParams(searchParams?.toString() ?? '');
|
||||||
|
if (term) {
|
||||||
|
params.set('search', term);
|
||||||
|
} else {
|
||||||
|
params.delete('search');
|
||||||
|
}
|
||||||
|
router.push(`?${params.toString()}`);
|
||||||
|
},
|
||||||
|
[router, searchParams],
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
handleSearch(searchTerm);
|
||||||
|
}, [debouncedSearchTerm]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Input
|
||||||
|
type="search"
|
||||||
|
placeholder="Search documents..."
|
||||||
|
value={searchTerm}
|
||||||
|
onChange={(e) => setSearchTerm(e.target.value)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -335,6 +335,23 @@ export const DocumentHistorySheet = ({
|
|||||||
]}
|
]}
|
||||||
/>
|
/>
|
||||||
))
|
))
|
||||||
|
.with(
|
||||||
|
{ type: DOCUMENT_AUDIT_LOG_TYPE.DOCUMENT_VISIBILITY_UPDATED },
|
||||||
|
({ data }) => (
|
||||||
|
<DocumentHistorySheetChanges
|
||||||
|
values={[
|
||||||
|
{
|
||||||
|
key: 'Old',
|
||||||
|
value: data.from,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'New',
|
||||||
|
value: data.to,
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
),
|
||||||
|
)
|
||||||
.exhaustive()}
|
.exhaustive()}
|
||||||
|
|
||||||
{isUserDetailsVisible && (
|
{isUserDetailsVisible && (
|
||||||
|
|||||||
@@ -89,7 +89,7 @@ export const DocumentReadOnlyFields = ({ documentMeta, fields }: DocumentReadOnl
|
|||||||
className="h-full w-full object-contain dark:invert"
|
className="h-full w-full object-contain dark:invert"
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<p className="font-signature text-muted-foreground text-lg duration-200 sm:text-xl md:text-2xl lg:text-3xl">
|
<p className="font-signature text-muted-foreground text-lg duration-200 sm:text-xl md:text-2xl">
|
||||||
{field.Signature?.typedSignature}
|
{field.Signature?.typedSignature}
|
||||||
</p>
|
</p>
|
||||||
),
|
),
|
||||||
@@ -122,7 +122,7 @@ export const DocumentReadOnlyFields = ({ documentMeta, fields }: DocumentReadOnl
|
|||||||
{field.Recipient.signingStatus === SigningStatus.NOT_SIGNED && (
|
{field.Recipient.signingStatus === SigningStatus.NOT_SIGNED && (
|
||||||
<p
|
<p
|
||||||
className={cn('text-muted-foreground text-lg duration-200', {
|
className={cn('text-muted-foreground text-lg duration-200', {
|
||||||
'font-signature sm:text-xl md:text-2xl lg:text-3xl':
|
'font-signature sm:text-xl md:text-2xl':
|
||||||
field.type === FieldType.SIGNATURE ||
|
field.type === FieldType.SIGNATURE ||
|
||||||
field.type === FieldType.FREE_SIGNATURE,
|
field.type === FieldType.FREE_SIGNATURE,
|
||||||
})}
|
})}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { useMemo, useState } from 'react';
|
import { useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import { useRouter } from 'next/navigation';
|
import { useRouter } from 'next/navigation';
|
||||||
@@ -307,6 +307,18 @@ export const SignInForm = ({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const hash = window.location.hash.slice(1);
|
||||||
|
|
||||||
|
const params = new URLSearchParams(hash);
|
||||||
|
|
||||||
|
const email = params.get('email');
|
||||||
|
|
||||||
|
if (email) {
|
||||||
|
form.setValue('email', email);
|
||||||
|
}
|
||||||
|
}, [form]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Form {...form}>
|
<Form {...form}>
|
||||||
<form
|
<form
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import Image from 'next/image';
|
import Image from 'next/image';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
@@ -203,6 +203,18 @@ export const SignUpFormV2 = ({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const hash = window.location.hash.slice(1);
|
||||||
|
|
||||||
|
const params = new URLSearchParams(hash);
|
||||||
|
|
||||||
|
const email = params.get('email');
|
||||||
|
|
||||||
|
if (email) {
|
||||||
|
form.setValue('email', email);
|
||||||
|
}
|
||||||
|
}, [form]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={cn('flex justify-center gap-x-12', className)}>
|
<div className={cn('flex justify-center gap-x-12', className)}>
|
||||||
<div className="border-border relative hidden flex-1 overflow-hidden rounded-xl border xl:flex">
|
<div className="border-border relative hidden flex-1 overflow-hidden rounded-xl border xl:flex">
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ async function middleware(req: NextRequest): Promise<NextResponse> {
|
|||||||
// Allow third parties to iframe the document.
|
// Allow third parties to iframe the document.
|
||||||
res.headers.set('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE, OPTIONS');
|
res.headers.set('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE, OPTIONS');
|
||||||
res.headers.set('Access-Control-Allow-Origin', '*');
|
res.headers.set('Access-Control-Allow-Origin', '*');
|
||||||
res.headers.set('Content-Security-Policy', "frame-ancestors *");
|
res.headers.set('Content-Security-Policy', 'frame-ancestors *');
|
||||||
res.headers.set('Referrer-Policy', 'strict-origin-when-cross-origin');
|
res.headers.set('Referrer-Policy', 'strict-origin-when-cross-origin');
|
||||||
res.headers.set('X-Content-Type-Options', 'nosniff');
|
res.headers.set('X-Content-Type-Options', 'nosniff');
|
||||||
res.headers.set('X-Frame-Options', 'ALLOW-ALL');
|
res.headers.set('X-Frame-Options', 'ALLOW-ALL');
|
||||||
|
|||||||
@@ -3,14 +3,14 @@
|
|||||||
import { createContext, useContext } from 'react';
|
import { createContext, useContext } from 'react';
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
import type { Team } from '@documenso/prisma/client';
|
import type { GetTeamResponse } from '@documenso/lib/server-only/team/get-team';
|
||||||
|
|
||||||
interface TeamProviderProps {
|
interface TeamProviderProps {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
team: Team;
|
team: GetTeamResponse;
|
||||||
}
|
}
|
||||||
|
|
||||||
const TeamContext = createContext<Team | null>(null);
|
const TeamContext = createContext<GetTeamResponse | null>(null);
|
||||||
|
|
||||||
export const useCurrentTeam = () => {
|
export const useCurrentTeam = () => {
|
||||||
const context = useContext(TeamContext);
|
const context = useContext(TeamContext);
|
||||||
|
|||||||
230
package-lock.json
generated
230
package-lock.json
generated
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "@documenso/root",
|
"name": "@documenso/root",
|
||||||
"version": "1.7.1-rc.0",
|
"version": "1.7.2-rc.0",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "@documenso/root",
|
"name": "@documenso/root",
|
||||||
"version": "1.7.1-rc.0",
|
"version": "1.7.2-rc.0",
|
||||||
"workspaces": [
|
"workspaces": [
|
||||||
"apps/*",
|
"apps/*",
|
||||||
"packages/*"
|
"packages/*"
|
||||||
@@ -80,7 +80,7 @@
|
|||||||
},
|
},
|
||||||
"apps/marketing": {
|
"apps/marketing": {
|
||||||
"name": "@documenso/marketing",
|
"name": "@documenso/marketing",
|
||||||
"version": "1.7.1-rc.0",
|
"version": "1.7.2-rc.0",
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@documenso/assets": "*",
|
"@documenso/assets": "*",
|
||||||
@@ -103,7 +103,7 @@
|
|||||||
"micro": "^10.0.1",
|
"micro": "^10.0.1",
|
||||||
"next": "14.2.6",
|
"next": "14.2.6",
|
||||||
"next-auth": "4.24.5",
|
"next-auth": "4.24.5",
|
||||||
"next-axiom": "^1.1.1",
|
"next-axiom": "^1.5.1",
|
||||||
"next-contentlayer": "^0.3.4",
|
"next-contentlayer": "^0.3.4",
|
||||||
"next-plausible": "^3.10.1",
|
"next-plausible": "^3.10.1",
|
||||||
"perfect-freehand": "^1.2.0",
|
"perfect-freehand": "^1.2.0",
|
||||||
@@ -410,6 +410,23 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"apps/marketing/node_modules/next-axiom": {
|
||||||
|
"version": "1.5.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/next-axiom/-/next-axiom-1.5.1.tgz",
|
||||||
|
"integrity": "sha512-sWxIzuJOex48ugMDlXWzvGvDGv5YHZ3w8gLZbUQ/Yml7oy5jcCItJNws9D0qmASirp2e5/BnvHxs44+9CO0GAQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"use-deep-compare": "^1.2.1",
|
||||||
|
"whatwg-fetch": "^3.6.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"next": ">=14.0",
|
||||||
|
"react": ">=18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"apps/marketing/node_modules/typescript": {
|
"apps/marketing/node_modules/typescript": {
|
||||||
"version": "5.2.2",
|
"version": "5.2.2",
|
||||||
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
|
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
|
||||||
@@ -424,7 +441,7 @@
|
|||||||
},
|
},
|
||||||
"apps/web": {
|
"apps/web": {
|
||||||
"name": "@documenso/web",
|
"name": "@documenso/web",
|
||||||
"version": "1.7.1-rc.0",
|
"version": "1.7.2-rc.0",
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@documenso/api": "*",
|
"@documenso/api": "*",
|
||||||
@@ -449,7 +466,7 @@
|
|||||||
"micro": "^10.0.1",
|
"micro": "^10.0.1",
|
||||||
"next": "14.2.6",
|
"next": "14.2.6",
|
||||||
"next-auth": "4.24.5",
|
"next-auth": "4.24.5",
|
||||||
"next-axiom": "^1.1.1",
|
"next-axiom": "^1.5.1",
|
||||||
"next-plausible": "^3.10.1",
|
"next-plausible": "^3.10.1",
|
||||||
"next-themes": "^0.2.1",
|
"next-themes": "^0.2.1",
|
||||||
"papaparse": "^5.4.1",
|
"papaparse": "^5.4.1",
|
||||||
@@ -465,7 +482,7 @@
|
|||||||
"react-icons": "^4.11.0",
|
"react-icons": "^4.11.0",
|
||||||
"react-rnd": "^10.4.1",
|
"react-rnd": "^10.4.1",
|
||||||
"recharts": "^2.7.2",
|
"recharts": "^2.7.2",
|
||||||
"remeda": "^1.27.1",
|
"remeda": "^2.12.1",
|
||||||
"sharp": "0.32.6",
|
"sharp": "0.32.6",
|
||||||
"ts-pattern": "^5.0.5",
|
"ts-pattern": "^5.0.5",
|
||||||
"ua-parser-js": "^1.0.37",
|
"ua-parser-js": "^1.0.37",
|
||||||
@@ -493,6 +510,23 @@
|
|||||||
"integrity": "sha512-O+z53uwx64xY7D6roOi4+jApDGFg0qn6WHcxe5QeqjMaTezBO/mxdfFXIVAVVyNWKx84OmPB3L8kbVYOTeN34A==",
|
"integrity": "sha512-O+z53uwx64xY7D6roOi4+jApDGFg0qn6WHcxe5QeqjMaTezBO/mxdfFXIVAVVyNWKx84OmPB3L8kbVYOTeN34A==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"apps/web/node_modules/next-axiom": {
|
||||||
|
"version": "1.5.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/next-axiom/-/next-axiom-1.5.1.tgz",
|
||||||
|
"integrity": "sha512-sWxIzuJOex48ugMDlXWzvGvDGv5YHZ3w8gLZbUQ/Yml7oy5jcCItJNws9D0qmASirp2e5/BnvHxs44+9CO0GAQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"use-deep-compare": "^1.2.1",
|
||||||
|
"whatwg-fetch": "^3.6.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"next": ">=14.0",
|
||||||
|
"react": ">=18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"apps/web/node_modules/typescript": {
|
"apps/web/node_modules/typescript": {
|
||||||
"version": "5.2.2",
|
"version": "5.2.2",
|
||||||
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
|
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
|
||||||
@@ -3433,6 +3467,34 @@
|
|||||||
"react-dom": "^16 || ^17 || ^18"
|
"react-dom": "^16 || ^17 || ^18"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@hello-pangea/dnd": {
|
||||||
|
"version": "16.6.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@hello-pangea/dnd/-/dnd-16.6.0.tgz",
|
||||||
|
"integrity": "sha512-vfZ4GydqbtUPXSLfAvKvXQ6xwRzIjUSjVU0Sx+70VOhc2xx6CdmJXJ8YhH70RpbTUGjxctslQTHul9sIOxCfFQ==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"@babel/runtime": "^7.24.1",
|
||||||
|
"css-box-model": "^1.2.1",
|
||||||
|
"memoize-one": "^6.0.0",
|
||||||
|
"raf-schd": "^4.0.3",
|
||||||
|
"react-redux": "^8.1.3",
|
||||||
|
"redux": "^4.2.1",
|
||||||
|
"use-memo-one": "^1.1.3"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": "^16.8.5 || ^17.0.0 || ^18.0.0",
|
||||||
|
"react-dom": "^16.8.5 || ^17.0.0 || ^18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@hello-pangea/dnd/node_modules/redux": {
|
||||||
|
"version": "4.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/redux/-/redux-4.2.1.tgz",
|
||||||
|
"integrity": "sha512-LAUYz4lc+Do8/g7aeRa8JkyDErK6ekstQaqWQrNRW//MY1TvCEpMtpTWvlQ+FPbWCx+Xixu/6SHt5N0HR+SB4w==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@babel/runtime": "^7.9.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@hexagon/base64": {
|
"node_modules/@hexagon/base64": {
|
||||||
"version": "1.1.28",
|
"version": "1.1.28",
|
||||||
"resolved": "https://registry.npmjs.org/@hexagon/base64/-/base64-1.1.28.tgz",
|
"resolved": "https://registry.npmjs.org/@hexagon/base64/-/base64-1.1.28.tgz",
|
||||||
@@ -11048,6 +11110,16 @@
|
|||||||
"@types/unist": "^2"
|
"@types/unist": "^2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@types/hoist-non-react-statics": {
|
||||||
|
"version": "3.3.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/hoist-non-react-statics/-/hoist-non-react-statics-3.3.5.tgz",
|
||||||
|
"integrity": "sha512-SbcrWzkKBw2cdwRTwQAswfpB9g9LJWfjtUeW/jvNwbhC8cpmmNYVePa+ncbUe0rGTQ7G3Ff6mYUN2VMfLVr+Sg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@types/react": "*",
|
||||||
|
"hoist-non-react-statics": "^3.3.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@types/http-cache-semantics": {
|
"node_modules/@types/http-cache-semantics": {
|
||||||
"version": "4.0.4",
|
"version": "4.0.4",
|
||||||
"resolved": "https://registry.npmjs.org/@types/http-cache-semantics/-/http-cache-semantics-4.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/@types/http-cache-semantics/-/http-cache-semantics-4.0.4.tgz",
|
||||||
@@ -14663,6 +14735,15 @@
|
|||||||
"node": ">=8"
|
"node": ">=8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/css-box-model": {
|
||||||
|
"version": "1.2.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/css-box-model/-/css-box-model-1.2.1.tgz",
|
||||||
|
"integrity": "sha512-a7Vr4Q/kd/aw96bnJG332W9V9LkJO69JRcaCYDUqjp6/z0w6VcZjgAcTbgFxEPfBgdnAwlh3iwu+hLopa+flJw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"tiny-invariant": "^1.0.6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/css.escape": {
|
"node_modules/css.escape": {
|
||||||
"version": "1.5.1",
|
"version": "1.5.1",
|
||||||
"resolved": "https://registry.npmjs.org/css.escape/-/css.escape-1.5.1.tgz",
|
"resolved": "https://registry.npmjs.org/css.escape/-/css.escape-1.5.1.tgz",
|
||||||
@@ -19256,6 +19337,15 @@
|
|||||||
"node": "*"
|
"node": "*"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/hoist-non-react-statics": {
|
||||||
|
"version": "3.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz",
|
||||||
|
"integrity": "sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==",
|
||||||
|
"license": "BSD-3-Clause",
|
||||||
|
"dependencies": {
|
||||||
|
"react-is": "^16.7.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/hosted-git-info": {
|
"node_modules/hosted-git-info": {
|
||||||
"version": "4.1.0",
|
"version": "4.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-4.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-4.1.0.tgz",
|
||||||
@@ -22639,6 +22729,12 @@
|
|||||||
"node": ">= 4.0.0"
|
"node": ">= 4.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/memoize-one": {
|
||||||
|
"version": "6.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/memoize-one/-/memoize-one-6.0.0.tgz",
|
||||||
|
"integrity": "sha512-rkpe71W0N0c0Xz6QD0eJETuWAJGnJ9afsl1srmwPrI+yBCkge5EycXXbYRyvL29zZVUWQCY7InPRCv3GDXuZNw==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/meow": {
|
"node_modules/meow": {
|
||||||
"version": "8.1.2",
|
"version": "8.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/meow/-/meow-8.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/meow/-/meow-8.1.2.tgz",
|
||||||
@@ -24088,22 +24184,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/next-axiom": {
|
|
||||||
"version": "1.1.1",
|
|
||||||
"resolved": "https://registry.npmjs.org/next-axiom/-/next-axiom-1.1.1.tgz",
|
|
||||||
"integrity": "sha512-0r/TJ+/zetD+uDc7B+2E7WpC86hEtQ1U+DuWYrP/JNmUz+ZdPFbrZgzOSqaZ6TwYbXP56VVlPfYwq1YsKHTHYQ==",
|
|
||||||
"dependencies": {
|
|
||||||
"remeda": "^1.29.0",
|
|
||||||
"whatwg-fetch": "^3.6.2"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": ">=18"
|
|
||||||
},
|
|
||||||
"peerDependencies": {
|
|
||||||
"next": ">=13.4",
|
|
||||||
"react": ">=18.0.0"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/next-contentlayer": {
|
"node_modules/next-contentlayer": {
|
||||||
"version": "0.3.4",
|
"version": "0.3.4",
|
||||||
"resolved": "https://registry.npmjs.org/next-contentlayer/-/next-contentlayer-0.3.4.tgz",
|
"resolved": "https://registry.npmjs.org/next-contentlayer/-/next-contentlayer-0.3.4.tgz",
|
||||||
@@ -27686,6 +27766,12 @@
|
|||||||
"node": ">=8"
|
"node": ">=8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/raf-schd": {
|
||||||
|
"version": "4.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/raf-schd/-/raf-schd-4.0.3.tgz",
|
||||||
|
"integrity": "sha512-tQkJl2GRWh83ui2DiPTJz9wEiMN20syf+5oKfB03yYP7ioZcJwsIK8FjrtLwH1m7C7e+Tt2yYBlrOpdT+dyeIQ==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/railroad-diagrams": {
|
"node_modules/railroad-diagrams": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/railroad-diagrams/-/railroad-diagrams-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/railroad-diagrams/-/railroad-diagrams-1.0.0.tgz",
|
||||||
@@ -28523,6 +28609,51 @@
|
|||||||
"resolved": "https://registry.npmjs.org/react-property/-/react-property-2.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/react-property/-/react-property-2.0.0.tgz",
|
||||||
"integrity": "sha512-kzmNjIgU32mO4mmH5+iUyrqlpFQhF8K2k7eZ4fdLSOPFrD1XgEuSBv9LDEgxRXTMBqMd8ppT0x6TIzqE5pdGdw=="
|
"integrity": "sha512-kzmNjIgU32mO4mmH5+iUyrqlpFQhF8K2k7eZ4fdLSOPFrD1XgEuSBv9LDEgxRXTMBqMd8ppT0x6TIzqE5pdGdw=="
|
||||||
},
|
},
|
||||||
|
"node_modules/react-redux": {
|
||||||
|
"version": "8.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/react-redux/-/react-redux-8.1.3.tgz",
|
||||||
|
"integrity": "sha512-n0ZrutD7DaX/j9VscF+uTALI3oUPa/pO4Z3soOBIjuRn/FzVu6aehhysxZCLi6y7duMf52WNZGMl7CtuK5EnRw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"@babel/runtime": "^7.12.1",
|
||||||
|
"@types/hoist-non-react-statics": "^3.3.1",
|
||||||
|
"@types/use-sync-external-store": "^0.0.3",
|
||||||
|
"hoist-non-react-statics": "^3.3.2",
|
||||||
|
"react-is": "^18.0.0",
|
||||||
|
"use-sync-external-store": "^1.0.0"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"@types/react": "^16.8 || ^17.0 || ^18.0",
|
||||||
|
"@types/react-dom": "^16.8 || ^17.0 || ^18.0",
|
||||||
|
"react": "^16.8 || ^17.0 || ^18.0",
|
||||||
|
"react-dom": "^16.8 || ^17.0 || ^18.0",
|
||||||
|
"react-native": ">=0.59",
|
||||||
|
"redux": "^4 || ^5.0.0-beta.0"
|
||||||
|
},
|
||||||
|
"peerDependenciesMeta": {
|
||||||
|
"@types/react": {
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
|
"@types/react-dom": {
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
|
"react-dom": {
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
|
"react-native": {
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
|
"redux": {
|
||||||
|
"optional": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/react-redux/node_modules/react-is": {
|
||||||
|
"version": "18.3.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz",
|
||||||
|
"integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/react-remove-scroll": {
|
"node_modules/react-remove-scroll": {
|
||||||
"version": "2.5.5",
|
"version": "2.5.5",
|
||||||
"resolved": "https://registry.npmjs.org/react-remove-scroll/-/react-remove-scroll-2.5.5.tgz",
|
"resolved": "https://registry.npmjs.org/react-remove-scroll/-/react-remove-scroll-2.5.5.tgz",
|
||||||
@@ -30017,9 +30148,25 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/remeda": {
|
"node_modules/remeda": {
|
||||||
"version": "1.29.0",
|
"version": "2.12.1",
|
||||||
"resolved": "https://registry.npmjs.org/remeda/-/remeda-1.29.0.tgz",
|
"resolved": "https://registry.npmjs.org/remeda/-/remeda-2.12.1.tgz",
|
||||||
"integrity": "sha512-M3LQ14KtMdQ1879lj/kKji3zBk158s7Rwg963mEkTfQFMxnKrIEAMxJfo/+0sp/+uGgN/KMVU2MBA4LNjqf8YQ=="
|
"integrity": "sha512-hKFAbxbQe8PMd4+CYO1DYCrCbcZsUSa7e21g7+4co91GBy7BD+Ub6JdaLy76yPOp7PCPTAXRz/9NXtZ9w15jbg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"type-fest": "^4.26.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/remeda/node_modules/type-fest": {
|
||||||
|
"version": "4.26.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.26.1.tgz",
|
||||||
|
"integrity": "sha512-yOGpmOAL7CkKe/91I5O3gPICmJNLJ1G4zFYVAsRHg7M64biSnPtRj0WNQt++bRkjYOqjWXrhnUw1utzmVErAdg==",
|
||||||
|
"license": "(MIT OR CC0-1.0)",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"node_modules/remote-git-tags": {
|
"node_modules/remote-git-tags": {
|
||||||
"version": "3.0.0",
|
"version": "3.0.0",
|
||||||
@@ -34082,6 +34229,27 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/use-deep-compare": {
|
||||||
|
"version": "1.3.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/use-deep-compare/-/use-deep-compare-1.3.0.tgz",
|
||||||
|
"integrity": "sha512-94iG+dEdEP/Sl3WWde+w9StIunlV8Dgj+vkt5wTwMoFQLaijiEZSXXy8KtcStpmEDtIptRJiNeD4ACTtVvnIKA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"dequal": "2.0.3"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": ">=16.8.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/use-memo-one": {
|
||||||
|
"version": "1.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/use-memo-one/-/use-memo-one-1.1.3.tgz",
|
||||||
|
"integrity": "sha512-g66/K7ZQGYrI6dy8GLpVcMsBp4s17xNkYJVSMvTEevGy3nDxHOfE6z8BVE22+5G5x7t3+bhzrlTDB7ObrEE0cQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": "^16.8.0 || ^17.0.0 || ^18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/use-sidecar": {
|
"node_modules/use-sidecar": {
|
||||||
"version": "1.1.2",
|
"version": "1.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/use-sidecar/-/use-sidecar-1.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/use-sidecar/-/use-sidecar-1.1.2.tgz",
|
||||||
@@ -36629,7 +36797,7 @@
|
|||||||
"pg": "^8.11.3",
|
"pg": "^8.11.3",
|
||||||
"playwright": "1.43.0",
|
"playwright": "1.43.0",
|
||||||
"react": "^18",
|
"react": "^18",
|
||||||
"remeda": "^1.27.1",
|
"remeda": "^2.12.1",
|
||||||
"sharp": "0.32.6",
|
"sharp": "0.32.6",
|
||||||
"stripe": "^12.7.0",
|
"stripe": "^12.7.0",
|
||||||
"ts-pattern": "^5.0.5",
|
"ts-pattern": "^5.0.5",
|
||||||
@@ -36828,6 +36996,7 @@
|
|||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@documenso/lib": "*",
|
"@documenso/lib": "*",
|
||||||
|
"@hello-pangea/dnd": "^16.6.0",
|
||||||
"@hookform/resolvers": "^3.3.0",
|
"@hookform/resolvers": "^3.3.0",
|
||||||
"@lingui/macro": "^4.11.3",
|
"@lingui/macro": "^4.11.3",
|
||||||
"@lingui/react": "^4.11.3",
|
"@lingui/react": "^4.11.3",
|
||||||
@@ -36874,6 +37043,7 @@
|
|||||||
"react-hook-form": "^7.45.4",
|
"react-hook-form": "^7.45.4",
|
||||||
"react-pdf": "7.7.3",
|
"react-pdf": "7.7.3",
|
||||||
"react-rnd": "^10.4.1",
|
"react-rnd": "^10.4.1",
|
||||||
|
"remeda": "^1.27.1",
|
||||||
"tailwind-merge": "^1.12.0",
|
"tailwind-merge": "^1.12.0",
|
||||||
"tailwindcss-animate": "^1.0.5",
|
"tailwindcss-animate": "^1.0.5",
|
||||||
"ts-pattern": "^5.0.5",
|
"ts-pattern": "^5.0.5",
|
||||||
@@ -36926,6 +37096,12 @@
|
|||||||
"node": ">=6"
|
"node": ">=6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"packages/ui/node_modules/remeda": {
|
||||||
|
"version": "1.61.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/remeda/-/remeda-1.61.0.tgz",
|
||||||
|
"integrity": "sha512-caKfSz9rDeSKBQQnlJnVW3mbVdFgxgGWQKq1XlFokqjf+hQD5gxutLGTTY2A/x24UxVyJe9gH5fAkFI63ULw4A==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"packages/ui/node_modules/typescript": {
|
"packages/ui/node_modules/typescript": {
|
||||||
"version": "5.2.2",
|
"version": "5.2.2",
|
||||||
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
|
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "1.7.1-rc.0",
|
"version": "1.7.2-rc.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "turbo run build",
|
"build": "turbo run build",
|
||||||
"build:web": "turbo run build --filter=@documenso/web",
|
"build:web": "turbo run build --filter=@documenso/web",
|
||||||
|
|||||||
@@ -293,6 +293,7 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
timezone,
|
timezone,
|
||||||
dateFormat: dateFormat?.value,
|
dateFormat: dateFormat?.value,
|
||||||
redirectUrl: body.meta.redirectUrl,
|
redirectUrl: body.meta.redirectUrl,
|
||||||
|
signingOrder: body.meta.signingOrder,
|
||||||
requestMetadata: extractNextApiRequestMetadata(args.req),
|
requestMetadata: extractNextApiRequestMetadata(args.req),
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -301,7 +302,9 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
documentId: document.id,
|
documentId: document.id,
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
teamId: team?.id,
|
teamId: team?.id,
|
||||||
data: body.authOptions,
|
data: {
|
||||||
|
...body.authOptions,
|
||||||
|
},
|
||||||
requestMetadata: extractNextApiRequestMetadata(args.req),
|
requestMetadata: extractNextApiRequestMetadata(args.req),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -325,6 +328,7 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
token: recipient.token,
|
token: recipient.token,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
|
|
||||||
signingUrl: `${NEXT_PUBLIC_WEBAPP_URL()}/sign/${recipient.token}`,
|
signingUrl: `${NEXT_PUBLIC_WEBAPP_URL()}/sign/${recipient.token}`,
|
||||||
})),
|
})),
|
||||||
@@ -496,6 +500,7 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
token: recipient.token,
|
token: recipient.token,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
|
|
||||||
signingUrl: `${NEXT_PUBLIC_WEBAPP_URL()}/sign/${recipient.token}`,
|
signingUrl: `${NEXT_PUBLIC_WEBAPP_URL()}/sign/${recipient.token}`,
|
||||||
})),
|
})),
|
||||||
@@ -588,6 +593,7 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
token: recipient.token,
|
token: recipient.token,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
|
|
||||||
signingUrl: `${NEXT_PUBLIC_WEBAPP_URL()}/sign/${recipient.token}`,
|
signingUrl: `${NEXT_PUBLIC_WEBAPP_URL()}/sign/${recipient.token}`,
|
||||||
})),
|
})),
|
||||||
@@ -713,7 +719,7 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
|
|
||||||
createRecipient: authenticatedMiddleware(async (args, user, team) => {
|
createRecipient: authenticatedMiddleware(async (args, user, team) => {
|
||||||
const { id: documentId } = args.params;
|
const { id: documentId } = args.params;
|
||||||
const { name, email, role, authOptions } = args.body;
|
const { name, email, role, authOptions, signingOrder } = args.body;
|
||||||
|
|
||||||
const document = await getDocumentById({
|
const document = await getDocumentById({
|
||||||
id: Number(documentId),
|
id: Number(documentId),
|
||||||
@@ -762,11 +768,16 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
userId: user.id,
|
userId: user.id,
|
||||||
teamId: team?.id,
|
teamId: team?.id,
|
||||||
recipients: [
|
recipients: [
|
||||||
...recipients,
|
...recipients.map(({ email, name }) => ({
|
||||||
|
email,
|
||||||
|
name,
|
||||||
|
role,
|
||||||
|
})),
|
||||||
{
|
{
|
||||||
email,
|
email,
|
||||||
name,
|
name,
|
||||||
role,
|
role,
|
||||||
|
signingOrder,
|
||||||
actionAuth: authOptions?.actionAuth ?? null,
|
actionAuth: authOptions?.actionAuth ?? null,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@@ -799,7 +810,7 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
|
|
||||||
updateRecipient: authenticatedMiddleware(async (args, user, team) => {
|
updateRecipient: authenticatedMiddleware(async (args, user, team) => {
|
||||||
const { id: documentId, recipientId } = args.params;
|
const { id: documentId, recipientId } = args.params;
|
||||||
const { name, email, role, authOptions } = args.body;
|
const { name, email, role, authOptions, signingOrder } = args.body;
|
||||||
|
|
||||||
const document = await getDocumentById({
|
const document = await getDocumentById({
|
||||||
id: Number(documentId),
|
id: Number(documentId),
|
||||||
@@ -833,6 +844,7 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
email,
|
email,
|
||||||
name,
|
name,
|
||||||
role,
|
role,
|
||||||
|
signingOrder,
|
||||||
actionAuth: authOptions?.actionAuth,
|
actionAuth: authOptions?.actionAuth,
|
||||||
requestMetadata: extractNextApiRequestMetadata(args.req),
|
requestMetadata: extractNextApiRequestMetadata(args.req),
|
||||||
}).catch(() => null);
|
}).catch(() => null);
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import {
|
|||||||
import { ZFieldMetaSchema } from '@documenso/lib/types/field-meta';
|
import { ZFieldMetaSchema } from '@documenso/lib/types/field-meta';
|
||||||
import {
|
import {
|
||||||
DocumentDataType,
|
DocumentDataType,
|
||||||
|
DocumentSigningOrder,
|
||||||
FieldType,
|
FieldType,
|
||||||
ReadStatus,
|
ReadStatus,
|
||||||
RecipientRole,
|
RecipientRole,
|
||||||
@@ -103,6 +104,7 @@ export const ZCreateDocumentMutationSchema = z.object({
|
|||||||
name: z.string().min(1),
|
name: z.string().min(1),
|
||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
meta: z
|
meta: z
|
||||||
@@ -123,6 +125,7 @@ export const ZCreateDocumentMutationSchema = z.object({
|
|||||||
enum: DATE_FORMATS.map((format) => format.value),
|
enum: DATE_FORMATS.map((format) => format.value),
|
||||||
}),
|
}),
|
||||||
redirectUrl: z.string(),
|
redirectUrl: z.string(),
|
||||||
|
signingOrder: z.nativeEnum(DocumentSigningOrder).optional(),
|
||||||
})
|
})
|
||||||
.partial(),
|
.partial(),
|
||||||
authOptions: z
|
authOptions: z
|
||||||
@@ -147,6 +150,7 @@ export const ZCreateDocumentMutationResponseSchema = z.object({
|
|||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
token: z.string(),
|
token: z.string(),
|
||||||
role: z.nativeEnum(RecipientRole),
|
role: z.nativeEnum(RecipientRole),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
|
|
||||||
signingUrl: z.string(),
|
signingUrl: z.string(),
|
||||||
}),
|
}),
|
||||||
@@ -165,6 +169,7 @@ export const ZCreateDocumentFromTemplateMutationSchema = z.object({
|
|||||||
name: z.string().min(1),
|
name: z.string().min(1),
|
||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
meta: z
|
meta: z
|
||||||
@@ -174,6 +179,7 @@ export const ZCreateDocumentFromTemplateMutationSchema = z.object({
|
|||||||
timezone: z.string(),
|
timezone: z.string(),
|
||||||
dateFormat: z.string(),
|
dateFormat: z.string(),
|
||||||
redirectUrl: z.string(),
|
redirectUrl: z.string(),
|
||||||
|
signingOrder: z.nativeEnum(DocumentSigningOrder).optional(),
|
||||||
})
|
})
|
||||||
.partial()
|
.partial()
|
||||||
.optional(),
|
.optional(),
|
||||||
@@ -200,6 +206,7 @@ export const ZCreateDocumentFromTemplateMutationResponseSchema = z.object({
|
|||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
token: z.string(),
|
token: z.string(),
|
||||||
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
|
|
||||||
signingUrl: z.string(),
|
signingUrl: z.string(),
|
||||||
}),
|
}),
|
||||||
@@ -219,6 +226,7 @@ export const ZGenerateDocumentFromTemplateMutationSchema = z.object({
|
|||||||
id: z.number(),
|
id: z.number(),
|
||||||
name: z.string().optional(),
|
name: z.string().optional(),
|
||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.refine(
|
.refine(
|
||||||
@@ -237,6 +245,7 @@ export const ZGenerateDocumentFromTemplateMutationSchema = z.object({
|
|||||||
timezone: z.string(),
|
timezone: z.string(),
|
||||||
dateFormat: z.string(),
|
dateFormat: z.string(),
|
||||||
redirectUrl: ZUrlSchema,
|
redirectUrl: ZUrlSchema,
|
||||||
|
signingOrder: z.nativeEnum(DocumentSigningOrder).optional(),
|
||||||
})
|
})
|
||||||
.partial()
|
.partial()
|
||||||
.optional(),
|
.optional(),
|
||||||
@@ -263,6 +272,7 @@ export const ZGenerateDocumentFromTemplateMutationResponseSchema = z.object({
|
|||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
token: z.string(),
|
token: z.string(),
|
||||||
role: z.nativeEnum(RecipientRole),
|
role: z.nativeEnum(RecipientRole),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
|
|
||||||
signingUrl: z.string(),
|
signingUrl: z.string(),
|
||||||
}),
|
}),
|
||||||
@@ -277,6 +287,7 @@ export const ZCreateRecipientMutationSchema = z.object({
|
|||||||
name: z.string().min(1),
|
name: z.string().min(1),
|
||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
role: z.nativeEnum(RecipientRole).optional().default(RecipientRole.SIGNER),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
authOptions: z
|
authOptions: z
|
||||||
.object({
|
.object({
|
||||||
actionAuth: ZRecipientActionAuthTypesSchema.optional(),
|
actionAuth: ZRecipientActionAuthTypesSchema.optional(),
|
||||||
@@ -305,6 +316,7 @@ export const ZSuccessfulRecipientResponseSchema = z.object({
|
|||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
name: z.string(),
|
name: z.string(),
|
||||||
role: z.nativeEnum(RecipientRole),
|
role: z.nativeEnum(RecipientRole),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
token: z.string(),
|
token: z.string(),
|
||||||
// !: Not used for now
|
// !: Not used for now
|
||||||
// expired: z.string(),
|
// expired: z.string(),
|
||||||
@@ -422,6 +434,7 @@ export const ZTemplateMetaSchema = z.object({
|
|||||||
dateFormat: z.string().nullish(),
|
dateFormat: z.string().nullish(),
|
||||||
templateId: z.number(),
|
templateId: z.number(),
|
||||||
redirectUrl: z.string().nullish(),
|
redirectUrl: z.string().nullish(),
|
||||||
|
signingOrder: z.nativeEnum(DocumentSigningOrder).nullish().default(DocumentSigningOrder.PARALLEL),
|
||||||
});
|
});
|
||||||
|
|
||||||
export const ZTemplateSchema = z.object({
|
export const ZTemplateSchema = z.object({
|
||||||
@@ -443,6 +456,7 @@ export const ZRecipientSchema = z.object({
|
|||||||
email: z.string().email().min(1),
|
email: z.string().email().min(1),
|
||||||
name: z.string(),
|
name: z.string(),
|
||||||
token: z.string(),
|
token: z.string(),
|
||||||
|
signingOrder: z.number().nullish(),
|
||||||
documentDeletedAt: z.date().nullish(),
|
documentDeletedAt: z.date().nullish(),
|
||||||
expired: z.date().nullish(),
|
expired: z.date().nullish(),
|
||||||
signedAt: z.date().nullish(),
|
signedAt: z.date().nullish(),
|
||||||
@@ -496,6 +510,7 @@ export const ZTemplateWithDataSchema = ZTemplateSchema.extend({
|
|||||||
id: true,
|
id: true,
|
||||||
email: true,
|
email: true,
|
||||||
name: true,
|
name: true,
|
||||||
|
signingOrder: true,
|
||||||
authOptions: true,
|
authOptions: true,
|
||||||
role: true,
|
role: true,
|
||||||
}).array(),
|
}).array(),
|
||||||
|
|||||||
@@ -41,11 +41,10 @@ test.describe('[EE_ONLY]', () => {
|
|||||||
// Add 2 signers.
|
// Add 2 signers.
|
||||||
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
||||||
await page.getByPlaceholder('Name').fill('Recipient 1');
|
await page.getByPlaceholder('Name').fill('Recipient 1');
|
||||||
|
|
||||||
await page.getByRole('button', { name: 'Add Signer' }).click();
|
await page.getByRole('button', { name: 'Add Signer' }).click();
|
||||||
await page
|
await page.getByLabel('Email').nth(1).fill('recipient2@documenso.com');
|
||||||
.getByRole('textbox', { name: 'Email', exact: true })
|
await page.getByLabel('Name').nth(1).fill('Recipient 2');
|
||||||
.fill('recipient2@documenso.com');
|
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('Recipient 2');
|
|
||||||
|
|
||||||
// Display advanced settings.
|
// Display advanced settings.
|
||||||
await page.getByLabel('Show advanced settings').check();
|
await page.getByLabel('Show advanced settings').check();
|
||||||
@@ -77,9 +76,11 @@ test('[DOCUMENT_FLOW]: add signers', async ({ page }) => {
|
|||||||
// Add 2 signers.
|
// Add 2 signers.
|
||||||
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
||||||
await page.getByPlaceholder('Name').fill('Recipient 1');
|
await page.getByPlaceholder('Name').fill('Recipient 1');
|
||||||
|
|
||||||
await page.getByRole('button', { name: 'Add Signer' }).click();
|
await page.getByRole('button', { name: 'Add Signer' }).click();
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).fill('recipient2@documenso.com');
|
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('Recipient 2');
|
await page.getByLabel('Email').nth(1).fill('recipient2@documenso.com');
|
||||||
|
await page.getByLabel('Name').nth(1).fill('Recipient 2');
|
||||||
|
|
||||||
// Advanced settings should not be visible for non EE users.
|
// Advanced settings should not be visible for non EE users.
|
||||||
await expect(page.getByLabel('Show advanced settings')).toBeHidden();
|
await expect(page.getByLabel('Show advanced settings')).toBeHidden();
|
||||||
|
|||||||
@@ -4,7 +4,13 @@ import path from 'node:path';
|
|||||||
|
|
||||||
import { getRecipientByEmail } from '@documenso/lib/server-only/recipient/get-recipient-by-email';
|
import { getRecipientByEmail } from '@documenso/lib/server-only/recipient/get-recipient-by-email';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import { DocumentStatus, FieldType, RecipientRole } from '@documenso/prisma/client';
|
import {
|
||||||
|
DocumentSigningOrder,
|
||||||
|
DocumentStatus,
|
||||||
|
FieldType,
|
||||||
|
RecipientRole,
|
||||||
|
SigningStatus,
|
||||||
|
} from '@documenso/prisma/client';
|
||||||
import {
|
import {
|
||||||
seedBlankDocument,
|
seedBlankDocument,
|
||||||
seedPendingDocumentWithFullFields,
|
seedPendingDocumentWithFullFields,
|
||||||
@@ -137,8 +143,9 @@ test('[DOCUMENT_FLOW]: should be able to create a document with multiple recipie
|
|||||||
await page.getByPlaceholder('Email').fill('user1@example.com');
|
await page.getByPlaceholder('Email').fill('user1@example.com');
|
||||||
await page.getByPlaceholder('Name').fill('User 1');
|
await page.getByPlaceholder('Name').fill('User 1');
|
||||||
await page.getByRole('button', { name: 'Add Signer' }).click();
|
await page.getByRole('button', { name: 'Add Signer' }).click();
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).fill('user2@example.com');
|
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('User 2');
|
await page.getByLabel('Email').nth(1).fill('user2@example.com');
|
||||||
|
await page.getByLabel('Name').nth(1).fill('User 2');
|
||||||
|
|
||||||
await page.getByRole('button', { name: 'Continue' }).click();
|
await page.getByRole('button', { name: 'Continue' }).click();
|
||||||
|
|
||||||
@@ -217,20 +224,20 @@ test('[DOCUMENT_FLOW]: should be able to create a document with multiple recipie
|
|||||||
await page.getByPlaceholder('Name').fill('User 1');
|
await page.getByPlaceholder('Name').fill('User 1');
|
||||||
await page.getByRole('button', { name: 'Add Signer' }).click();
|
await page.getByRole('button', { name: 'Add Signer' }).click();
|
||||||
|
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).fill('user2@example.com');
|
await page.getByLabel('Email').nth(1).fill('user2@example.com');
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('User 2');
|
await page.getByLabel('Name').nth(1).fill('User 2');
|
||||||
await page.locator('button[role="combobox"]').nth(1).click();
|
await page.locator('button[role="combobox"]').nth(1).click();
|
||||||
await page.getByLabel('Receives copy').click();
|
await page.getByLabel('Receives copy').click();
|
||||||
await page.getByRole('button', { name: 'Add Signer' }).click();
|
await page.getByRole('button', { name: 'Add Signer' }).click();
|
||||||
|
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).nth(1).fill('user3@example.com');
|
await page.getByLabel('Email').nth(2).fill('user3@example.com');
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(2).fill('User 3');
|
await page.getByLabel('Name').nth(2).fill('User 3');
|
||||||
await page.locator('button[role="combobox"]').nth(2).click();
|
await page.locator('button[role="combobox"]').nth(2).click();
|
||||||
await page.getByLabel('Needs to approve').click();
|
await page.getByLabel('Needs to approve').click();
|
||||||
await page.getByRole('button', { name: 'Add Signer' }).click();
|
await page.getByRole('button', { name: 'Add Signer' }).click();
|
||||||
|
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).nth(2).fill('user4@example.com');
|
await page.getByLabel('Email').nth(3).fill('user4@example.com');
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(3).fill('User 4');
|
await page.getByLabel('Name').nth(3).fill('User 4');
|
||||||
await page.locator('button[role="combobox"]').nth(3).click();
|
await page.locator('button[role="combobox"]').nth(3).click();
|
||||||
await page.getByLabel('Needs to view').click();
|
await page.getByLabel('Needs to view').click();
|
||||||
|
|
||||||
@@ -503,3 +510,163 @@ test('[DOCUMENT_FLOW]: should be able to sign a document with custom date', asyn
|
|||||||
expect(completedStatus).toBe(DocumentStatus.COMPLETED);
|
expect(completedStatus).toBe(DocumentStatus.COMPLETED);
|
||||||
}).toPass();
|
}).toPass();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('[DOCUMENT_FLOW]: should be able to create and sign a document with 3 recipients in sequential order', async ({
|
||||||
|
page,
|
||||||
|
}) => {
|
||||||
|
const user = await seedUser();
|
||||||
|
const document = await seedBlankDocument(user);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: user.email,
|
||||||
|
redirectPath: `/documents/${document.id}/edit`,
|
||||||
|
});
|
||||||
|
|
||||||
|
const documentTitle = `Sequential-Signing-${Date.now()}.pdf`;
|
||||||
|
|
||||||
|
await expect(page.getByRole('heading', { name: 'General' })).toBeVisible();
|
||||||
|
await page.getByLabel('Title').fill(documentTitle);
|
||||||
|
await page.getByRole('button', { name: 'Continue' }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole('heading', { name: 'Add Signers' })).toBeVisible();
|
||||||
|
await page.getByLabel('Enable signing order').check();
|
||||||
|
|
||||||
|
for (let i = 1; i <= 3; i++) {
|
||||||
|
if (i > 1) {
|
||||||
|
await page.getByRole('button', { name: 'Add Signer' }).click();
|
||||||
|
}
|
||||||
|
await page
|
||||||
|
.getByPlaceholder('Email')
|
||||||
|
.nth(i - 1)
|
||||||
|
.fill(`user${i}@example.com`);
|
||||||
|
await page
|
||||||
|
.getByPlaceholder('Name')
|
||||||
|
.nth(i - 1)
|
||||||
|
.fill(`User ${i}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Continue' }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole('heading', { name: 'Add Fields' })).toBeVisible();
|
||||||
|
|
||||||
|
for (let i = 1; i <= 3; i++) {
|
||||||
|
if (i > 1) {
|
||||||
|
await page.getByText(`User ${i} (user${i}@example.com)`).click();
|
||||||
|
}
|
||||||
|
await page.getByRole('button', { name: 'Signature' }).click();
|
||||||
|
await page.locator('canvas').click({
|
||||||
|
position: {
|
||||||
|
x: 100,
|
||||||
|
y: 100 * i,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
await page.getByText(`User ${i} (user${i}@example.com)`).click();
|
||||||
|
}
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Continue' }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole('heading', { name: 'Add Subject' })).toBeVisible();
|
||||||
|
await page.getByRole('button', { name: 'Send' }).click();
|
||||||
|
|
||||||
|
await page.waitForURL('/documents');
|
||||||
|
|
||||||
|
await expect(page.getByRole('link', { name: documentTitle })).toBeVisible();
|
||||||
|
|
||||||
|
const createdDocument = await prisma.document.findFirst({
|
||||||
|
where: { title: documentTitle },
|
||||||
|
include: { Recipient: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(createdDocument).not.toBeNull();
|
||||||
|
expect(createdDocument?.Recipient.length).toBe(3);
|
||||||
|
|
||||||
|
for (let i = 0; i < 3; i++) {
|
||||||
|
const recipient = createdDocument?.Recipient.find(
|
||||||
|
(r) => r.email === `user${i + 1}@example.com`,
|
||||||
|
);
|
||||||
|
expect(recipient).not.toBeNull();
|
||||||
|
|
||||||
|
const fields = await prisma.field.findMany({
|
||||||
|
where: { recipientId: recipient?.id, documentId: createdDocument?.id },
|
||||||
|
});
|
||||||
|
const recipientField = fields[0];
|
||||||
|
|
||||||
|
if (i > 0) {
|
||||||
|
const previousRecipient = await prisma.recipient.findFirst({
|
||||||
|
where: { email: `user${i}@example.com`, documentId: createdDocument?.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(previousRecipient?.signingStatus).toBe(SigningStatus.SIGNED);
|
||||||
|
}
|
||||||
|
|
||||||
|
await page.goto(`/sign/${recipient?.token}`);
|
||||||
|
await expect(page.getByRole('heading', { name: 'Sign Document' })).toBeVisible();
|
||||||
|
|
||||||
|
await page.locator(`#field-${recipientField.id}`).getByRole('button').click();
|
||||||
|
|
||||||
|
const canvas = page.locator('canvas#signature');
|
||||||
|
const box = await canvas.boundingBox();
|
||||||
|
if (box) {
|
||||||
|
await page.mouse.move(box.x + box.width / 2, box.y + box.height / 2);
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(box.x + box.width / 4, box.y + box.height / 4);
|
||||||
|
await page.mouse.up();
|
||||||
|
}
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Sign', exact: true }).click();
|
||||||
|
await page.getByRole('button', { name: 'Complete' }).click();
|
||||||
|
await page.getByRole('button', { name: 'Sign' }).click();
|
||||||
|
|
||||||
|
await page.waitForURL(`/sign/${recipient?.token}/complete`);
|
||||||
|
await expect(page.getByText('Document Signed')).toBeVisible();
|
||||||
|
|
||||||
|
const updatedRecipient = await prisma.recipient.findFirst({
|
||||||
|
where: { id: recipient?.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updatedRecipient?.signingStatus).toBe(SigningStatus.SIGNED);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for the document to be signed.
|
||||||
|
await page.waitForTimeout(5000);
|
||||||
|
|
||||||
|
const finalDocument = await prisma.document.findFirst({
|
||||||
|
where: { id: createdDocument?.id },
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(finalDocument?.status).toBe(DocumentStatus.COMPLETED);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[DOCUMENT_FLOW]: should prevent out-of-order signing in sequential mode', async ({
|
||||||
|
page,
|
||||||
|
}) => {
|
||||||
|
const user = await seedUser();
|
||||||
|
|
||||||
|
const { document, recipients } = await seedPendingDocumentWithFullFields({
|
||||||
|
owner: user,
|
||||||
|
recipients: ['user1@example.com', 'user2@example.com', 'user3@example.com'],
|
||||||
|
fields: [FieldType.SIGNATURE],
|
||||||
|
recipientsCreateOptions: [{ signingOrder: 1 }, { signingOrder: 2 }, { signingOrder: 3 }],
|
||||||
|
updateDocumentOptions: {
|
||||||
|
documentMeta: {
|
||||||
|
create: {
|
||||||
|
signingOrder: DocumentSigningOrder.SEQUENTIAL,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const pendingRecipient = recipients.find((r) => r.signingOrder === 2);
|
||||||
|
|
||||||
|
await page.goto(`/sign/${pendingRecipient?.token}`);
|
||||||
|
|
||||||
|
await expect(page).toHaveURL(`/sign/${pendingRecipient?.token}/waiting`);
|
||||||
|
|
||||||
|
const activeRecipient = recipients.find((r) => r.signingOrder === 1);
|
||||||
|
|
||||||
|
await page.goto(`/sign/${activeRecipient?.token}`);
|
||||||
|
|
||||||
|
await expect(page).not.toHaveURL(`/sign/${activeRecipient?.token}/waiting`);
|
||||||
|
await expect(page.getByRole('heading', { name: 'Sign Document' })).toBeVisible();
|
||||||
|
});
|
||||||
|
|||||||
249
packages/app-tests/e2e/teams/search-documents.spec.ts
Normal file
249
packages/app-tests/e2e/teams/search-documents.spec.ts
Normal file
@@ -0,0 +1,249 @@
|
|||||||
|
import { expect, test } from '@playwright/test';
|
||||||
|
|
||||||
|
import { DocumentStatus, TeamMemberRole } from '@documenso/prisma/client';
|
||||||
|
import { seedDocuments, seedTeamDocuments } from '@documenso/prisma/seed/documents';
|
||||||
|
import { seedTeam, seedTeamMember } from '@documenso/prisma/seed/teams';
|
||||||
|
import { seedUser } from '@documenso/prisma/seed/users';
|
||||||
|
|
||||||
|
import { apiSignin, apiSignout } from '../fixtures/authentication';
|
||||||
|
import { checkDocumentTabCount } from '../fixtures/documents';
|
||||||
|
|
||||||
|
test('[TEAMS]: search respects team document visibility', async ({ page }) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
const adminUser = await seedTeamMember({ teamId: team.id, role: TeamMemberRole.ADMIN });
|
||||||
|
const managerUser = await seedTeamMember({ teamId: team.id, role: TeamMemberRole.MANAGER });
|
||||||
|
const memberUser = await seedTeamMember({ teamId: team.id, role: TeamMemberRole.MEMBER });
|
||||||
|
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Searchable Document for Everyone',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'MANAGER_AND_ABOVE',
|
||||||
|
title: 'Searchable Document for Managers',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Searchable Document for Admins',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
const testCases = [
|
||||||
|
{ user: adminUser, visibleDocs: 3 },
|
||||||
|
{ user: managerUser, visibleDocs: 2 },
|
||||||
|
{ user: memberUser, visibleDocs: 1 },
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const { user, visibleDocs } of testCases) {
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: user.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents`,
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.getByPlaceholder('Search documents...').fill('Searchable');
|
||||||
|
await page.waitForURL(/search=Searchable/);
|
||||||
|
|
||||||
|
await checkDocumentTabCount(page, 'All', visibleDocs);
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: search does not reveal documents from other teams', async ({ page }) => {
|
||||||
|
const { team: teamA, teamMember2: teamAMember } = await seedTeamDocuments();
|
||||||
|
const { team: teamB } = await seedTeamDocuments();
|
||||||
|
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: teamA.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: teamA.id,
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Unique Team A Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
sender: teamB.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: teamB.id,
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Unique Team B Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: teamAMember.email,
|
||||||
|
redirectPath: `/t/${teamA.url}/documents`,
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.getByPlaceholder('Search documents...').fill('Unique');
|
||||||
|
await page.waitForURL(/search=Unique/);
|
||||||
|
|
||||||
|
await checkDocumentTabCount(page, 'All', 1);
|
||||||
|
await expect(page.getByRole('link', { name: 'Unique Team A Document' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('link', { name: 'Unique Team B Document' })).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[PERSONAL]: search does not reveal team documents in personal account', async ({ page }) => {
|
||||||
|
const { team, teamMember2 } = await seedTeamDocuments();
|
||||||
|
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: teamMember2,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: null,
|
||||||
|
title: 'Personal Unique Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Team Unique Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: teamMember2.email,
|
||||||
|
redirectPath: '/documents',
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.getByPlaceholder('Search documents...').fill('Unique');
|
||||||
|
await page.waitForURL(/search=Unique/);
|
||||||
|
|
||||||
|
await checkDocumentTabCount(page, 'All', 1);
|
||||||
|
await expect(page.getByRole('link', { name: 'Personal Unique Document' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('link', { name: 'Team Unique Document' })).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: search respects recipient visibility regardless of team visibility', async ({
|
||||||
|
page,
|
||||||
|
}) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
const memberUser = await seedTeamMember({ teamId: team.id, role: TeamMemberRole.MEMBER });
|
||||||
|
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [memberUser],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Admin Document with Member Recipient',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: memberUser.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents`,
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.getByPlaceholder('Search documents...').fill('Admin Document');
|
||||||
|
await page.waitForURL(/search=Admin(%20|\+|\s)Document/);
|
||||||
|
|
||||||
|
await checkDocumentTabCount(page, 'All', 1);
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Admin Document with Member Recipient' }),
|
||||||
|
).toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: search by recipient name respects visibility', async ({ page }) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
const adminUser = await seedTeamMember({ teamId: team.id, role: TeamMemberRole.ADMIN });
|
||||||
|
const memberUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.MEMBER,
|
||||||
|
name: 'Team Member',
|
||||||
|
});
|
||||||
|
|
||||||
|
const uniqueRecipient = await seedUser();
|
||||||
|
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [uniqueRecipient],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Admin Document for Unique Recipient',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Admin should see the document when searching by recipient name
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: adminUser.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents`,
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.getByPlaceholder('Search documents...').fill('Unique Recipient');
|
||||||
|
await page.waitForURL(/search=Unique(%20|\+|\s)Recipient/);
|
||||||
|
|
||||||
|
await checkDocumentTabCount(page, 'All', 1);
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Admin Document for Unique Recipient' }),
|
||||||
|
).toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
|
||||||
|
// Member should not see the document when searching by recipient name
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: memberUser.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents`,
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.getByPlaceholder('Search documents...').fill('Unique Recipient');
|
||||||
|
await page.waitForURL(/search=Unique(%20|\+|\s)Recipient/);
|
||||||
|
|
||||||
|
await checkDocumentTabCount(page, 'All', 0);
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Admin Document for Unique Recipient' }),
|
||||||
|
).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
import { expect, test } from '@playwright/test';
|
import { expect, test } from '@playwright/test';
|
||||||
|
|
||||||
import { DocumentStatus } from '@documenso/prisma/client';
|
import { DocumentStatus, TeamMemberRole } from '@documenso/prisma/client';
|
||||||
import { seedDocuments, seedTeamDocuments } from '@documenso/prisma/seed/documents';
|
import { seedDocuments, seedTeamDocuments } from '@documenso/prisma/seed/documents';
|
||||||
import { seedTeamEmail } from '@documenso/prisma/seed/teams';
|
import { seedTeam, seedTeamEmail, seedTeamMember } from '@documenso/prisma/seed/teams';
|
||||||
import { seedUser } from '@documenso/prisma/seed/users';
|
import { seedUser } from '@documenso/prisma/seed/users';
|
||||||
|
|
||||||
import { apiSignin, apiSignout } from '../fixtures/authentication';
|
import { apiSignin, apiSignout } from '../fixtures/authentication';
|
||||||
@@ -355,3 +355,354 @@ test('[TEAMS]: delete completed team document', async ({ page }) => {
|
|||||||
await apiSignout({ page });
|
await apiSignout({ page });
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: check document visibility based on team member role', async ({ page }) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
|
||||||
|
// Seed users with different roles
|
||||||
|
const adminUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.ADMIN,
|
||||||
|
});
|
||||||
|
|
||||||
|
const managerUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.MANAGER,
|
||||||
|
});
|
||||||
|
|
||||||
|
const memberUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.MEMBER,
|
||||||
|
});
|
||||||
|
|
||||||
|
const outsideUser = await seedUser();
|
||||||
|
|
||||||
|
// Seed documents with different visibility levels
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Document Visible to Everyone',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'MANAGER_AND_ABOVE',
|
||||||
|
title: 'Document Visible to Manager and Above',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Document Visible to Admin',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [outsideUser],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Document Visible to Admin with Recipient',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Test cases for each role
|
||||||
|
const testCases = [
|
||||||
|
{
|
||||||
|
user: adminUser,
|
||||||
|
expectedDocuments: [
|
||||||
|
'Document Visible to Everyone',
|
||||||
|
'Document Visible to Manager and Above',
|
||||||
|
'Document Visible to Admin',
|
||||||
|
'Document Visible to Admin with Recipient',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
user: managerUser,
|
||||||
|
expectedDocuments: ['Document Visible to Everyone', 'Document Visible to Manager and Above'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
user: memberUser,
|
||||||
|
expectedDocuments: ['Document Visible to Everyone'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
user: outsideUser,
|
||||||
|
expectedDocuments: ['Document Visible to Admin with Recipient'],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const testCase of testCases) {
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: testCase.user.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check that the user sees the expected documents
|
||||||
|
for (const documentTitle of testCase.expectedDocuments) {
|
||||||
|
await expect(page.getByRole('link', { name: documentTitle, exact: true })).toBeVisible();
|
||||||
|
}
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: ensure recipient can see document regardless of visibility', async ({ page }) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
|
||||||
|
// Seed a member user
|
||||||
|
const memberUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.MEMBER,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Seed a document with ADMIN visibility but make the member user a recipient
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [memberUser],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Admin Document with Member Recipient',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: memberUser.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check that the member user can see the document
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Admin Document with Member Recipient', exact: true }),
|
||||||
|
).toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: check that members cannot see ADMIN-only documents', async ({ page }) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
|
||||||
|
// Seed a member user
|
||||||
|
const memberUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.MEMBER,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Seed an ADMIN-only document
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Admin Only Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: memberUser.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check that the member user cannot see the ADMIN-only document
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Admin Only Document', exact: true }),
|
||||||
|
).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: check that managers cannot see ADMIN-only documents', async ({ page }) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
|
||||||
|
// Seed a manager user
|
||||||
|
const managerUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.MANAGER,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Seed an ADMIN-only document
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'ADMIN',
|
||||||
|
title: 'Admin Only Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: managerUser.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check that the manager user cannot see the ADMIN-only document
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Admin Only Document', exact: true }),
|
||||||
|
).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: check that admin can see MANAGER_AND_ABOVE documents', async ({ page }) => {
|
||||||
|
const team = await seedTeam();
|
||||||
|
|
||||||
|
// Seed an admin user
|
||||||
|
const adminUser = await seedTeamMember({
|
||||||
|
teamId: team.id,
|
||||||
|
role: TeamMemberRole.ADMIN,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Seed a MANAGER_AND_ABOVE document
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: team.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'MANAGER_AND_ABOVE',
|
||||||
|
title: 'Manager and Above Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: adminUser.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check that the admin user can see the MANAGER_AND_ABOVE document
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Manager and Above Document', exact: true }),
|
||||||
|
).toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: users cannot see documents from other teams', async ({ page }) => {
|
||||||
|
// Seed two teams with documents
|
||||||
|
const { team: teamA, teamMember2: teamAMember } = await seedTeamDocuments();
|
||||||
|
const { team: teamB, teamMember2: teamBMember } = await seedTeamDocuments();
|
||||||
|
|
||||||
|
// Seed a document in team B
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: teamB.owner,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: teamB.id,
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Team B Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Sign in as a member of team A
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: teamAMember.email,
|
||||||
|
redirectPath: `/t/${teamA.url}/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify that the user cannot see the document from team B
|
||||||
|
await expect(page.getByRole('link', { name: 'Team B Document', exact: true })).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[TEAMS]: personal documents are not visible in team context', async ({ page }) => {
|
||||||
|
// Seed a team and a user with personal documents
|
||||||
|
const { team, teamMember2 } = await seedTeamDocuments();
|
||||||
|
const personalUser = await seedUser();
|
||||||
|
|
||||||
|
// Seed a personal document for teamMember2
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: teamMember2,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: null, // Indicates a personal document
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Personal Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Sign in as teamMember2 in the team context
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: teamMember2.email,
|
||||||
|
redirectPath: `/t/${team.url}/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify that the personal document is not visible in the team context
|
||||||
|
await expect(
|
||||||
|
page.getByRole('link', { name: 'Personal Document', exact: true }),
|
||||||
|
).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('[PERSONAL]: team documents are not visible in personal account', async ({ page }) => {
|
||||||
|
// Seed a team and a user with personal documents
|
||||||
|
const { team, teamMember2 } = await seedTeamDocuments();
|
||||||
|
|
||||||
|
// Seed a team document
|
||||||
|
await seedDocuments([
|
||||||
|
{
|
||||||
|
sender: teamMember2,
|
||||||
|
recipients: [],
|
||||||
|
type: DocumentStatus.COMPLETED,
|
||||||
|
documentOptions: {
|
||||||
|
teamId: team.id,
|
||||||
|
visibility: 'EVERYONE',
|
||||||
|
title: 'Team Document',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Sign in as teamMember2 in the personal context
|
||||||
|
await apiSignin({
|
||||||
|
page,
|
||||||
|
email: teamMember2.email,
|
||||||
|
redirectPath: `/documents?status=COMPLETED`,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify that the team document is not visible in the personal context
|
||||||
|
await expect(page.getByRole('link', { name: 'Team Document', exact: true })).not.toBeVisible();
|
||||||
|
|
||||||
|
await apiSignout({ page });
|
||||||
|
});
|
||||||
|
|||||||
@@ -42,10 +42,8 @@ test.describe('[EE_ONLY]', () => {
|
|||||||
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
||||||
await page.getByPlaceholder('Name').fill('Recipient 1');
|
await page.getByPlaceholder('Name').fill('Recipient 1');
|
||||||
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
||||||
await page
|
await page.getByPlaceholder('Email').nth(1).fill('recipient2@documenso.com');
|
||||||
.getByRole('textbox', { name: 'Email', exact: true })
|
await page.getByPlaceholder('Name').nth(1).fill('Recipient 2');
|
||||||
.fill('recipient2@documenso.com');
|
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('Recipient 2');
|
|
||||||
|
|
||||||
// Display advanced settings.
|
// Display advanced settings.
|
||||||
await page.getByLabel('Show advanced settings').check();
|
await page.getByLabel('Show advanced settings').check();
|
||||||
@@ -94,8 +92,8 @@ test('[TEMPLATE_FLOW]: add placeholder', async ({ page }) => {
|
|||||||
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
||||||
await page.getByPlaceholder('Name').fill('Recipient 1');
|
await page.getByPlaceholder('Name').fill('Recipient 1');
|
||||||
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).fill('recipient2@documenso.com');
|
await page.getByPlaceholder('Email').nth(1).fill('recipient2@documenso.com');
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('Recipient 2');
|
await page.getByPlaceholder('Name').nth(1).fill('Recipient 2');
|
||||||
|
|
||||||
// Advanced settings should not be visible for non EE users.
|
// Advanced settings should not be visible for non EE users.
|
||||||
await expect(page.getByLabel('Show advanced settings')).toBeHidden();
|
await expect(page.getByLabel('Show advanced settings')).toBeHidden();
|
||||||
|
|||||||
@@ -76,8 +76,8 @@ test('[TEMPLATE]: should create a document from a template', async ({ page }) =>
|
|||||||
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
||||||
await page.getByPlaceholder('Name').fill('Recipient 1');
|
await page.getByPlaceholder('Name').fill('Recipient 1');
|
||||||
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).fill('recipient2@documenso.com');
|
await page.getByPlaceholder('Email').nth(1).fill('recipient2@documenso.com');
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('Recipient 2');
|
await page.getByPlaceholder('Name').nth(1).fill('Recipient 2');
|
||||||
|
|
||||||
// Apply require passkey for Recipient 1.
|
// Apply require passkey for Recipient 1.
|
||||||
if (isBillingEnabled) {
|
if (isBillingEnabled) {
|
||||||
@@ -211,8 +211,8 @@ test('[TEMPLATE]: should create a team document from a team template', async ({
|
|||||||
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
await page.getByPlaceholder('Email').fill('recipient1@documenso.com');
|
||||||
await page.getByPlaceholder('Name').fill('Recipient 1');
|
await page.getByPlaceholder('Name').fill('Recipient 1');
|
||||||
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
await page.getByRole('button', { name: 'Add Placeholder Recipient' }).click();
|
||||||
await page.getByRole('textbox', { name: 'Email', exact: true }).fill('recipient2@documenso.com');
|
await page.getByPlaceholder('Email').nth(1).fill('recipient2@documenso.com');
|
||||||
await page.getByRole('textbox', { name: 'Name', exact: true }).nth(1).fill('Recipient 2');
|
await page.getByPlaceholder('Name').nth(1).fill('Recipient 2');
|
||||||
|
|
||||||
// Apply require passkey for Recipient 1.
|
// Apply require passkey for Recipient 1.
|
||||||
if (isBillingEnabled) {
|
if (isBillingEnabled) {
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
import { createContext, useCallback, useContext, useEffect, useState } from 'react';
|
import { createContext, useCallback, useContext, useEffect, useState } from 'react';
|
||||||
|
|
||||||
import { equals } from 'remeda';
|
import { isDeepEqual } from 'remeda';
|
||||||
|
|
||||||
import { getLimits } from '../client';
|
import { getLimits } from '../client';
|
||||||
import { FREE_PLAN_LIMITS } from '../constants';
|
import { FREE_PLAN_LIMITS } from '../constants';
|
||||||
@@ -42,7 +42,7 @@ export const LimitsProvider = ({
|
|||||||
const newLimits = await getLimits({ teamId });
|
const newLimits = await getLimits({ teamId });
|
||||||
|
|
||||||
setLimits((oldLimits) => {
|
setLimits((oldLimits) => {
|
||||||
if (equals(oldLimits, newLimits)) {
|
if (isDeepEqual(oldLimits, newLimits)) {
|
||||||
return oldLimits;
|
return oldLimits;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
78
packages/email/templates/recipient-removed-from-document.tsx
Normal file
78
packages/email/templates/recipient-removed-from-document.tsx
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
import config from '@documenso/tailwind-config';
|
||||||
|
|
||||||
|
import {
|
||||||
|
Body,
|
||||||
|
Container,
|
||||||
|
Head,
|
||||||
|
Hr,
|
||||||
|
Html,
|
||||||
|
Img,
|
||||||
|
Preview,
|
||||||
|
Section,
|
||||||
|
Tailwind,
|
||||||
|
Text,
|
||||||
|
} from '../components';
|
||||||
|
import type { TemplateDocumentCancelProps } from '../template-components/template-document-cancel';
|
||||||
|
import TemplateDocumentImage from '../template-components/template-document-image';
|
||||||
|
import { TemplateFooter } from '../template-components/template-footer';
|
||||||
|
|
||||||
|
export type DocumentCancelEmailTemplateProps = Partial<TemplateDocumentCancelProps>;
|
||||||
|
|
||||||
|
export const RecipientRemovedFromDocumentTemplate = ({
|
||||||
|
inviterName = 'Lucas Smith',
|
||||||
|
documentName = 'Open Source Pledge.pdf',
|
||||||
|
assetBaseUrl = 'http://localhost:3002',
|
||||||
|
}: DocumentCancelEmailTemplateProps) => {
|
||||||
|
const previewText = `${inviterName} has removed you from the document ${documentName}.`;
|
||||||
|
|
||||||
|
const getAssetUrl = (path: string) => {
|
||||||
|
return new URL(path, assetBaseUrl).toString();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Html>
|
||||||
|
<Head />
|
||||||
|
<Preview>{previewText}</Preview>
|
||||||
|
<Tailwind
|
||||||
|
config={{
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
colors: config.theme.extend.colors,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Body className="mx-auto my-auto bg-white font-sans">
|
||||||
|
<Section>
|
||||||
|
<Container className="mx-auto mb-2 mt-8 max-w-xl rounded-lg border border-solid border-slate-200 p-4 backdrop-blur-sm">
|
||||||
|
<Section>
|
||||||
|
<Img
|
||||||
|
src={getAssetUrl('/static/logo.png')}
|
||||||
|
alt="Documenso Logo"
|
||||||
|
className="mb-4 h-6"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<TemplateDocumentImage className="mt-6" assetBaseUrl={assetBaseUrl} />
|
||||||
|
|
||||||
|
<Section>
|
||||||
|
<Text className="text-primary mx-auto mb-0 max-w-[80%] text-center text-lg font-semibold">
|
||||||
|
{inviterName} has removed you from the document
|
||||||
|
<br />"{documentName}"
|
||||||
|
</Text>
|
||||||
|
</Section>
|
||||||
|
</Section>
|
||||||
|
</Container>
|
||||||
|
|
||||||
|
<Hr className="mx-auto mt-12 max-w-xl" />
|
||||||
|
|
||||||
|
<Container className="mx-auto max-w-xl">
|
||||||
|
<TemplateFooter />
|
||||||
|
</Container>
|
||||||
|
</Section>
|
||||||
|
</Body>
|
||||||
|
</Tailwind>
|
||||||
|
</Html>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default RecipientRemovedFromDocumentTemplate;
|
||||||
@@ -2,7 +2,7 @@ import { useCallback, useEffect, useState } from 'react';
|
|||||||
|
|
||||||
import { getBoundingClientRect } from '@documenso/lib/client-only/get-bounding-client-rect';
|
import { getBoundingClientRect } from '@documenso/lib/client-only/get-bounding-client-rect';
|
||||||
import { PDF_VIEWER_PAGE_SELECTOR } from '@documenso/lib/constants/pdf-viewer';
|
import { PDF_VIEWER_PAGE_SELECTOR } from '@documenso/lib/constants/pdf-viewer';
|
||||||
import { Field } from '@documenso/prisma/client';
|
import type { Field } from '@documenso/prisma/client';
|
||||||
|
|
||||||
export const useFieldPageCoords = (field: Field) => {
|
export const useFieldPageCoords = (field: Field) => {
|
||||||
const [coords, setCoords] = useState({
|
const [coords, setCoords] = useState({
|
||||||
|
|||||||
@@ -3,12 +3,12 @@ import { useCallback, useRef, useState } from 'react';
|
|||||||
type ThrottleOptions = {
|
type ThrottleOptions = {
|
||||||
leading?: boolean;
|
leading?: boolean;
|
||||||
trailing?: boolean;
|
trailing?: boolean;
|
||||||
}
|
};
|
||||||
|
|
||||||
export function useThrottleFn<T extends (...args: unknown[]) => unknown>(
|
export function useThrottleFn<T extends (...args: unknown[]) => unknown>(
|
||||||
fn: T,
|
fn: T,
|
||||||
ms = 500,
|
ms = 500,
|
||||||
options: ThrottleOptions = {}
|
options: ThrottleOptions = {},
|
||||||
): [(...args: Parameters<T>) => void, boolean, () => void] {
|
): [(...args: Parameters<T>) => void, boolean, () => void] {
|
||||||
const [isThrottling, setIsThrottling] = useState(false);
|
const [isThrottling, setIsThrottling] = useState(false);
|
||||||
const $isThrottling = useRef(false);
|
const $isThrottling = useRef(false);
|
||||||
@@ -44,7 +44,7 @@ export function useThrottleFn<T extends (...args: unknown[]) => unknown>(
|
|||||||
$lastArgs.current = args;
|
$lastArgs.current = args;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
[fn, ms, leading, trailing, $setIsThrottling]
|
[fn, ms, leading, trailing, $setIsThrottling],
|
||||||
);
|
);
|
||||||
|
|
||||||
const cancel = useCallback(() => {
|
const cancel = useCallback(() => {
|
||||||
|
|||||||
@@ -5,7 +5,8 @@ export const APP_DOCUMENT_UPLOAD_SIZE_LIMIT =
|
|||||||
|
|
||||||
export const NEXT_PUBLIC_WEBAPP_URL = () => env('NEXT_PUBLIC_WEBAPP_URL');
|
export const NEXT_PUBLIC_WEBAPP_URL = () => env('NEXT_PUBLIC_WEBAPP_URL');
|
||||||
export const NEXT_PUBLIC_MARKETING_URL = () => env('NEXT_PUBLIC_MARKETING_URL');
|
export const NEXT_PUBLIC_MARKETING_URL = () => env('NEXT_PUBLIC_MARKETING_URL');
|
||||||
export const NEXT_PRIVATE_INTERNAL_WEBAPP_URL = process.env.NEXT_PRIVATE_INTERNAL_WEBAPP_URL ?? NEXT_PUBLIC_WEBAPP_URL();
|
export const NEXT_PRIVATE_INTERNAL_WEBAPP_URL =
|
||||||
|
process.env.NEXT_PRIVATE_INTERNAL_WEBAPP_URL ?? NEXT_PUBLIC_WEBAPP_URL();
|
||||||
|
|
||||||
export const IS_APP_MARKETING = process.env.NEXT_PUBLIC_PROJECT === 'marketing';
|
export const IS_APP_MARKETING = process.env.NEXT_PUBLIC_PROJECT === 'marketing';
|
||||||
export const IS_APP_WEB = process.env.NEXT_PUBLIC_PROJECT === 'web';
|
export const IS_APP_WEB = process.env.NEXT_PUBLIC_PROJECT === 'web';
|
||||||
|
|||||||
23
packages/lib/constants/document-visibility.ts
Normal file
23
packages/lib/constants/document-visibility.ts
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import { DocumentVisibility } from '@documenso/lib/types/document-visibility';
|
||||||
|
|
||||||
|
import type { TDocumentVisibility } from '../types/document-visibility';
|
||||||
|
|
||||||
|
type DocumentVisibilityTypeData = {
|
||||||
|
key: TDocumentVisibility;
|
||||||
|
value: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const DOCUMENT_VISIBILITY: Record<string, DocumentVisibilityTypeData> = {
|
||||||
|
[DocumentVisibility.ADMIN]: {
|
||||||
|
key: DocumentVisibility.ADMIN,
|
||||||
|
value: 'Admins only',
|
||||||
|
},
|
||||||
|
[DocumentVisibility.EVERYONE]: {
|
||||||
|
key: DocumentVisibility.EVERYONE,
|
||||||
|
value: 'Everyone',
|
||||||
|
},
|
||||||
|
[DocumentVisibility.MANAGER_AND_ABOVE]: {
|
||||||
|
key: DocumentVisibility.MANAGER_AND_ABOVE,
|
||||||
|
value: 'Managers and above',
|
||||||
|
},
|
||||||
|
} satisfies Record<TDocumentVisibility, DocumentVisibilityTypeData>;
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
export const SUPPORTED_LANGUAGE_CODES = ['de', 'en'] as const;
|
export const SUPPORTED_LANGUAGE_CODES = ['de', 'en', 'fr'] as const;
|
||||||
|
|
||||||
export const ZSupportedLanguageCodeSchema = z.enum(SUPPORTED_LANGUAGE_CODES).catch('en');
|
export const ZSupportedLanguageCodeSchema = z.enum(SUPPORTED_LANGUAGE_CODES).catch('en');
|
||||||
|
|
||||||
@@ -38,4 +38,8 @@ export const SUPPORTED_LANGUAGES: Record<string, SupportedLanguage> = {
|
|||||||
full: 'English',
|
full: 'English',
|
||||||
short: 'en',
|
short: 'en',
|
||||||
},
|
},
|
||||||
|
fr: {
|
||||||
|
full: 'French',
|
||||||
|
short: 'fr',
|
||||||
|
},
|
||||||
} satisfies Record<SupportedLanguageCodes, SupportedLanguage>;
|
} satisfies Record<SupportedLanguageCodes, SupportedLanguage>;
|
||||||
|
|||||||
@@ -48,7 +48,7 @@
|
|||||||
"pg": "^8.11.3",
|
"pg": "^8.11.3",
|
||||||
"playwright": "1.43.0",
|
"playwright": "1.43.0",
|
||||||
"react": "^18",
|
"react": "^18",
|
||||||
"remeda": "^1.27.1",
|
"remeda": "^2.12.1",
|
||||||
"sharp": "0.32.6",
|
"sharp": "0.32.6",
|
||||||
"stripe": "^12.7.0",
|
"stripe": "^12.7.0",
|
||||||
"ts-pattern": "^5.0.5",
|
"ts-pattern": "^5.0.5",
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import {
|
|||||||
diffDocumentMetaChanges,
|
diffDocumentMetaChanges,
|
||||||
} from '@documenso/lib/utils/document-audit-logs';
|
} from '@documenso/lib/utils/document-audit-logs';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
|
import type { DocumentSigningOrder } from '@documenso/prisma/client';
|
||||||
|
|
||||||
export type CreateDocumentMetaOptions = {
|
export type CreateDocumentMetaOptions = {
|
||||||
documentId: number;
|
documentId: number;
|
||||||
@@ -16,6 +17,7 @@ export type CreateDocumentMetaOptions = {
|
|||||||
password?: string;
|
password?: string;
|
||||||
dateFormat?: string;
|
dateFormat?: string;
|
||||||
redirectUrl?: string;
|
redirectUrl?: string;
|
||||||
|
signingOrder?: DocumentSigningOrder;
|
||||||
userId: number;
|
userId: number;
|
||||||
requestMetadata: RequestMetadata;
|
requestMetadata: RequestMetadata;
|
||||||
};
|
};
|
||||||
@@ -29,6 +31,7 @@ export const upsertDocumentMeta = async ({
|
|||||||
password,
|
password,
|
||||||
userId,
|
userId,
|
||||||
redirectUrl,
|
redirectUrl,
|
||||||
|
signingOrder,
|
||||||
requestMetadata,
|
requestMetadata,
|
||||||
}: CreateDocumentMetaOptions) => {
|
}: CreateDocumentMetaOptions) => {
|
||||||
const user = await prisma.user.findFirstOrThrow({
|
const user = await prisma.user.findFirstOrThrow({
|
||||||
@@ -78,6 +81,7 @@ export const upsertDocumentMeta = async ({
|
|||||||
timezone,
|
timezone,
|
||||||
documentId,
|
documentId,
|
||||||
redirectUrl,
|
redirectUrl,
|
||||||
|
signingOrder,
|
||||||
},
|
},
|
||||||
update: {
|
update: {
|
||||||
subject,
|
subject,
|
||||||
@@ -86,6 +90,7 @@ export const upsertDocumentMeta = async ({
|
|||||||
dateFormat,
|
dateFormat,
|
||||||
timezone,
|
timezone,
|
||||||
redirectUrl,
|
redirectUrl,
|
||||||
|
signingOrder,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -2,11 +2,18 @@ import { DOCUMENT_AUDIT_LOG_TYPE } from '@documenso/lib/types/document-audit-log
|
|||||||
import type { RequestMetadata } from '@documenso/lib/universal/extract-request-metadata';
|
import type { RequestMetadata } from '@documenso/lib/universal/extract-request-metadata';
|
||||||
import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs';
|
import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import { DocumentStatus, SigningStatus } from '@documenso/prisma/client';
|
import {
|
||||||
|
DocumentSigningOrder,
|
||||||
|
DocumentStatus,
|
||||||
|
RecipientRole,
|
||||||
|
SendStatus,
|
||||||
|
SigningStatus,
|
||||||
|
} from '@documenso/prisma/client';
|
||||||
import { WebhookTriggerEvents } from '@documenso/prisma/client';
|
import { WebhookTriggerEvents } from '@documenso/prisma/client';
|
||||||
|
|
||||||
import { jobs } from '../../jobs/client';
|
import { jobs } from '../../jobs/client';
|
||||||
import type { TRecipientActionAuth } from '../../types/document-auth';
|
import type { TRecipientActionAuth } from '../../types/document-auth';
|
||||||
|
import { getIsRecipientsTurnToSign } from '../recipient/get-is-recipient-turn';
|
||||||
import { triggerWebhook } from '../webhooks/trigger/trigger-webhook';
|
import { triggerWebhook } from '../webhooks/trigger/trigger-webhook';
|
||||||
import { sendPendingEmail } from './send-pending-email';
|
import { sendPendingEmail } from './send-pending-email';
|
||||||
|
|
||||||
@@ -29,6 +36,7 @@ const getDocument = async ({ token, documentId }: CompleteDocumentWithTokenOptio
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
include: {
|
include: {
|
||||||
|
documentMeta: true,
|
||||||
Recipient: {
|
Recipient: {
|
||||||
where: {
|
where: {
|
||||||
token,
|
token,
|
||||||
@@ -59,6 +67,16 @@ export const completeDocumentWithToken = async ({
|
|||||||
throw new Error(`Recipient ${recipient.id} has already signed`);
|
throw new Error(`Recipient ${recipient.id} has already signed`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (document.documentMeta?.signingOrder === DocumentSigningOrder.SEQUENTIAL) {
|
||||||
|
const isRecipientsTurn = await getIsRecipientsTurnToSign({ token: recipient.token });
|
||||||
|
|
||||||
|
if (!isRecipientsTurn) {
|
||||||
|
throw new Error(
|
||||||
|
`Recipient ${recipient.id} attempted to complete the document before it was their turn`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const fields = await prisma.field.findMany({
|
const fields = await prisma.field.findMany({
|
||||||
where: {
|
where: {
|
||||||
documentId: document.id,
|
documentId: document.id,
|
||||||
@@ -120,17 +138,48 @@ export const completeDocumentWithToken = async ({
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const pendingRecipients = await prisma.recipient.count({
|
const pendingRecipients = await prisma.recipient.findMany({
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
signingOrder: true,
|
||||||
|
},
|
||||||
where: {
|
where: {
|
||||||
documentId: document.id,
|
documentId: document.id,
|
||||||
signingStatus: {
|
signingStatus: {
|
||||||
not: SigningStatus.SIGNED,
|
not: SigningStatus.SIGNED,
|
||||||
},
|
},
|
||||||
|
role: {
|
||||||
|
not: RecipientRole.CC,
|
||||||
},
|
},
|
||||||
|
},
|
||||||
|
// Composite sort so our next recipient is always the one with the lowest signing order or id
|
||||||
|
// if there is a tie.
|
||||||
|
orderBy: [{ signingOrder: { sort: 'asc', nulls: 'last' } }, { id: 'asc' }],
|
||||||
});
|
});
|
||||||
|
|
||||||
if (pendingRecipients > 0) {
|
if (pendingRecipients.length > 0) {
|
||||||
await sendPendingEmail({ documentId, recipientId: recipient.id });
|
await sendPendingEmail({ documentId, recipientId: recipient.id });
|
||||||
|
|
||||||
|
if (document.documentMeta?.signingOrder === DocumentSigningOrder.SEQUENTIAL) {
|
||||||
|
const [nextRecipient] = pendingRecipients;
|
||||||
|
|
||||||
|
await prisma.$transaction(async (tx) => {
|
||||||
|
await tx.recipient.update({
|
||||||
|
where: { id: nextRecipient.id },
|
||||||
|
data: { sendStatus: SendStatus.SENT },
|
||||||
|
});
|
||||||
|
|
||||||
|
await jobs.triggerJob({
|
||||||
|
name: 'send.signing.requested.email',
|
||||||
|
payload: {
|
||||||
|
userId: document.userId,
|
||||||
|
documentId: document.id,
|
||||||
|
recipientId: nextRecipient.id,
|
||||||
|
requestMetadata,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const haveAllRecipientsSigned = await prisma.document.findFirst({
|
const haveAllRecipientsSigned = await prisma.document.findFirst({
|
||||||
@@ -138,7 +187,7 @@ export const completeDocumentWithToken = async ({
|
|||||||
id: document.id,
|
id: document.id,
|
||||||
Recipient: {
|
Recipient: {
|
||||||
every: {
|
every: {
|
||||||
signingStatus: SigningStatus.SIGNED,
|
OR: [{ signingStatus: SigningStatus.SIGNED }, { role: RecipientRole.CC }],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -2,10 +2,11 @@ import { DateTime } from 'luxon';
|
|||||||
import { P, match } from 'ts-pattern';
|
import { P, match } from 'ts-pattern';
|
||||||
|
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import { RecipientRole, SigningStatus } from '@documenso/prisma/client';
|
import { RecipientRole, SigningStatus, TeamMemberRole } from '@documenso/prisma/client';
|
||||||
import type { Document, Prisma, Team, TeamEmail, User } from '@documenso/prisma/client';
|
import type { Document, Prisma, Team, TeamEmail, User } from '@documenso/prisma/client';
|
||||||
import { ExtendedDocumentStatus } from '@documenso/prisma/types/extended-document-status';
|
import { ExtendedDocumentStatus } from '@documenso/prisma/types/extended-document-status';
|
||||||
|
|
||||||
|
import { DocumentVisibility } from '../../types/document-visibility';
|
||||||
import type { FindResultSet } from '../../types/find-result-set';
|
import type { FindResultSet } from '../../types/find-result-set';
|
||||||
import { maskRecipientTokensForDocument } from '../../utils/mask-recipient-tokens-for-document';
|
import { maskRecipientTokensForDocument } from '../../utils/mask-recipient-tokens-for-document';
|
||||||
|
|
||||||
@@ -24,6 +25,7 @@ export type FindDocumentsOptions = {
|
|||||||
};
|
};
|
||||||
period?: PeriodSelectorValue;
|
period?: PeriodSelectorValue;
|
||||||
senderIds?: number[];
|
senderIds?: number[];
|
||||||
|
search?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const findDocuments = async ({
|
export const findDocuments = async ({
|
||||||
@@ -36,6 +38,7 @@ export const findDocuments = async ({
|
|||||||
orderBy,
|
orderBy,
|
||||||
period,
|
period,
|
||||||
senderIds,
|
senderIds,
|
||||||
|
search,
|
||||||
}: FindDocumentsOptions) => {
|
}: FindDocumentsOptions) => {
|
||||||
const { user, team } = await prisma.$transaction(async (tx) => {
|
const { user, team } = await prisma.$transaction(async (tx) => {
|
||||||
const user = await tx.user.findFirstOrThrow({
|
const user = await tx.user.findFirstOrThrow({
|
||||||
@@ -58,6 +61,14 @@ export const findDocuments = async ({
|
|||||||
},
|
},
|
||||||
include: {
|
include: {
|
||||||
teamEmail: true,
|
teamEmail: true,
|
||||||
|
members: {
|
||||||
|
where: {
|
||||||
|
userId,
|
||||||
|
},
|
||||||
|
select: {
|
||||||
|
role: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -70,6 +81,7 @@ export const findDocuments = async ({
|
|||||||
|
|
||||||
const orderByColumn = orderBy?.column ?? 'createdAt';
|
const orderByColumn = orderBy?.column ?? 'createdAt';
|
||||||
const orderByDirection = orderBy?.direction ?? 'desc';
|
const orderByDirection = orderBy?.direction ?? 'desc';
|
||||||
|
const teamMemberRole = team?.members[0].role ?? null;
|
||||||
|
|
||||||
const termFilters = match(term)
|
const termFilters = match(term)
|
||||||
.with(P.string.minLength(1), () => {
|
.with(P.string.minLength(1), () => {
|
||||||
@@ -82,7 +94,45 @@ export const findDocuments = async ({
|
|||||||
})
|
})
|
||||||
.otherwise(() => undefined);
|
.otherwise(() => undefined);
|
||||||
|
|
||||||
const filters = team ? findTeamDocumentsFilter(status, team) : findDocumentsFilter(status, user);
|
const searchFilter: Prisma.DocumentWhereInput = {
|
||||||
|
OR: [
|
||||||
|
{ title: { contains: search, mode: 'insensitive' } },
|
||||||
|
{ Recipient: { some: { name: { contains: search, mode: 'insensitive' } } } },
|
||||||
|
{ Recipient: { some: { email: { contains: search, mode: 'insensitive' } } } },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
const visibilityFilters = [
|
||||||
|
match(teamMemberRole)
|
||||||
|
.with(TeamMemberRole.ADMIN, () => ({
|
||||||
|
visibility: {
|
||||||
|
in: [
|
||||||
|
DocumentVisibility.EVERYONE,
|
||||||
|
DocumentVisibility.MANAGER_AND_ABOVE,
|
||||||
|
DocumentVisibility.ADMIN,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
.with(TeamMemberRole.MANAGER, () => ({
|
||||||
|
visibility: {
|
||||||
|
in: [DocumentVisibility.EVERYONE, DocumentVisibility.MANAGER_AND_ABOVE],
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
.otherwise(() => ({ visibility: DocumentVisibility.EVERYONE })),
|
||||||
|
{
|
||||||
|
Recipient: {
|
||||||
|
some: {
|
||||||
|
email: user.email,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
let filters: Prisma.DocumentWhereInput | null = findDocumentsFilter(status, user);
|
||||||
|
|
||||||
|
if (team) {
|
||||||
|
filters = findTeamDocumentsFilter(status, team, visibilityFilters);
|
||||||
|
}
|
||||||
|
|
||||||
if (filters === null) {
|
if (filters === null) {
|
||||||
return {
|
return {
|
||||||
@@ -148,9 +198,7 @@ export const findDocuments = async ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const whereClause: Prisma.DocumentWhereInput = {
|
const whereClause: Prisma.DocumentWhereInput = {
|
||||||
...termFilters,
|
AND: [{ ...termFilters }, { ...filters }, { ...deletedFilter }, { ...searchFilter }],
|
||||||
...filters,
|
|
||||||
...deletedFilter,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
if (period) {
|
if (period) {
|
||||||
@@ -333,6 +381,7 @@ const findDocumentsFilter = (status: ExtendedDocumentStatus, user: User) => {
|
|||||||
const findTeamDocumentsFilter = (
|
const findTeamDocumentsFilter = (
|
||||||
status: ExtendedDocumentStatus,
|
status: ExtendedDocumentStatus,
|
||||||
team: Team & { teamEmail: TeamEmail | null },
|
team: Team & { teamEmail: TeamEmail | null },
|
||||||
|
visibilityFilters: Prisma.DocumentWhereInput[],
|
||||||
) => {
|
) => {
|
||||||
const teamEmail = team.teamEmail?.email ?? null;
|
const teamEmail = team.teamEmail?.email ?? null;
|
||||||
|
|
||||||
@@ -343,6 +392,7 @@ const findTeamDocumentsFilter = (
|
|||||||
OR: [
|
OR: [
|
||||||
{
|
{
|
||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
@@ -358,6 +408,7 @@ const findTeamDocumentsFilter = (
|
|||||||
email: teamEmail,
|
email: teamEmail,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Filter to display all documents that have been sent by the team email.
|
// Filter to display all documents that have been sent by the team email.
|
||||||
@@ -365,6 +416,7 @@ const findTeamDocumentsFilter = (
|
|||||||
User: {
|
User: {
|
||||||
email: teamEmail,
|
email: teamEmail,
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -389,6 +441,7 @@ const findTeamDocumentsFilter = (
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
};
|
};
|
||||||
})
|
})
|
||||||
.with(ExtendedDocumentStatus.DRAFT, () => {
|
.with(ExtendedDocumentStatus.DRAFT, () => {
|
||||||
@@ -397,6 +450,7 @@ const findTeamDocumentsFilter = (
|
|||||||
{
|
{
|
||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
status: ExtendedDocumentStatus.DRAFT,
|
status: ExtendedDocumentStatus.DRAFT,
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
@@ -407,6 +461,7 @@ const findTeamDocumentsFilter = (
|
|||||||
User: {
|
User: {
|
||||||
email: teamEmail,
|
email: teamEmail,
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -418,6 +473,7 @@ const findTeamDocumentsFilter = (
|
|||||||
{
|
{
|
||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
status: ExtendedDocumentStatus.PENDING,
|
status: ExtendedDocumentStatus.PENDING,
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
@@ -436,11 +492,13 @@ const findTeamDocumentsFilter = (
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
User: {
|
User: {
|
||||||
email: teamEmail,
|
email: teamEmail,
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
@@ -454,6 +512,7 @@ const findTeamDocumentsFilter = (
|
|||||||
OR: [
|
OR: [
|
||||||
{
|
{
|
||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
@@ -466,11 +525,13 @@ const findTeamDocumentsFilter = (
|
|||||||
email: teamEmail,
|
email: teamEmail,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
User: {
|
User: {
|
||||||
email: teamEmail,
|
email: teamEmail,
|
||||||
},
|
},
|
||||||
|
OR: visibilityFilters,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,10 @@
|
|||||||
|
import { match } from 'ts-pattern';
|
||||||
|
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import type { Prisma } from '@documenso/prisma/client';
|
import type { Prisma } from '@documenso/prisma/client';
|
||||||
|
import { TeamMemberRole } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
import { DocumentVisibility } from '../../types/document-visibility';
|
||||||
import { getTeamById } from '../team/get-team';
|
import { getTeamById } from '../team/get-team';
|
||||||
|
|
||||||
export type GetDocumentByIdOptions = {
|
export type GetDocumentByIdOptions = {
|
||||||
@@ -28,6 +32,11 @@ export const getDocumentById = async ({ id, userId, teamId }: GetDocumentByIdOpt
|
|||||||
email: true,
|
email: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
Recipient: {
|
||||||
|
select: {
|
||||||
|
email: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
team: {
|
team: {
|
||||||
select: {
|
select: {
|
||||||
id: true,
|
id: true,
|
||||||
@@ -115,5 +124,35 @@ export const getDocumentWhereInput = async ({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return documentWhereInput;
|
const user = await prisma.user.findFirstOrThrow({
|
||||||
|
where: {
|
||||||
|
id: userId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const visibilityFilters = [
|
||||||
|
...match(team.currentTeamMember?.role)
|
||||||
|
.with(TeamMemberRole.ADMIN, () => [
|
||||||
|
{ visibility: DocumentVisibility.EVERYONE },
|
||||||
|
{ visibility: DocumentVisibility.MANAGER_AND_ABOVE },
|
||||||
|
{ visibility: DocumentVisibility.ADMIN },
|
||||||
|
])
|
||||||
|
.with(TeamMemberRole.MANAGER, () => [
|
||||||
|
{ visibility: DocumentVisibility.EVERYONE },
|
||||||
|
{ visibility: DocumentVisibility.MANAGER_AND_ABOVE },
|
||||||
|
])
|
||||||
|
.otherwise(() => [{ visibility: DocumentVisibility.EVERYONE }]),
|
||||||
|
{
|
||||||
|
Recipient: {
|
||||||
|
some: {
|
||||||
|
email: user.email,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
return {
|
||||||
|
...documentWhereInput,
|
||||||
|
OR: [...visibilityFilters],
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,19 +1,24 @@
|
|||||||
import { DateTime } from 'luxon';
|
import { DateTime } from 'luxon';
|
||||||
|
import { match } from 'ts-pattern';
|
||||||
|
|
||||||
import type { PeriodSelectorValue } from '@documenso/lib/server-only/document/find-documents';
|
import type { PeriodSelectorValue } from '@documenso/lib/server-only/document/find-documents';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
|
import { TeamMemberRole } from '@documenso/prisma/client';
|
||||||
import type { Prisma, User } from '@documenso/prisma/client';
|
import type { Prisma, User } from '@documenso/prisma/client';
|
||||||
import { SigningStatus } from '@documenso/prisma/client';
|
import { SigningStatus } from '@documenso/prisma/client';
|
||||||
import { isExtendedDocumentStatus } from '@documenso/prisma/guards/is-extended-document-status';
|
import { isExtendedDocumentStatus } from '@documenso/prisma/guards/is-extended-document-status';
|
||||||
import { ExtendedDocumentStatus } from '@documenso/prisma/types/extended-document-status';
|
import { ExtendedDocumentStatus } from '@documenso/prisma/types/extended-document-status';
|
||||||
|
|
||||||
|
import { DocumentVisibility } from '../../types/document-visibility';
|
||||||
|
|
||||||
export type GetStatsInput = {
|
export type GetStatsInput = {
|
||||||
user: User;
|
user: User;
|
||||||
team?: Omit<GetTeamCountsOption, 'createdAt'>;
|
team?: Omit<GetTeamCountsOption, 'createdAt'>;
|
||||||
period?: PeriodSelectorValue;
|
period?: PeriodSelectorValue;
|
||||||
|
search?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getStats = async ({ user, period, ...options }: GetStatsInput) => {
|
export const getStats = async ({ user, period, search, ...options }: GetStatsInput) => {
|
||||||
let createdAt: Prisma.DocumentWhereInput['createdAt'];
|
let createdAt: Prisma.DocumentWhereInput['createdAt'];
|
||||||
|
|
||||||
if (period) {
|
if (period) {
|
||||||
@@ -27,8 +32,14 @@ export const getStats = async ({ user, period, ...options }: GetStatsInput) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const [ownerCounts, notSignedCounts, hasSignedCounts] = await (options.team
|
const [ownerCounts, notSignedCounts, hasSignedCounts] = await (options.team
|
||||||
? getTeamCounts({ ...options.team, createdAt })
|
? getTeamCounts({
|
||||||
: getCounts({ user, createdAt }));
|
...options.team,
|
||||||
|
createdAt,
|
||||||
|
currentUserEmail: user.email,
|
||||||
|
userId: user.id,
|
||||||
|
search,
|
||||||
|
})
|
||||||
|
: getCounts({ user, createdAt, search }));
|
||||||
|
|
||||||
const stats: Record<ExtendedDocumentStatus, number> = {
|
const stats: Record<ExtendedDocumentStatus, number> = {
|
||||||
[ExtendedDocumentStatus.DRAFT]: 0,
|
[ExtendedDocumentStatus.DRAFT]: 0,
|
||||||
@@ -68,9 +79,18 @@ export const getStats = async ({ user, period, ...options }: GetStatsInput) => {
|
|||||||
type GetCountsOption = {
|
type GetCountsOption = {
|
||||||
user: User;
|
user: User;
|
||||||
createdAt: Prisma.DocumentWhereInput['createdAt'];
|
createdAt: Prisma.DocumentWhereInput['createdAt'];
|
||||||
|
search?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getCounts = async ({ user, createdAt, search }: GetCountsOption) => {
|
||||||
|
const searchFilter: Prisma.DocumentWhereInput = {
|
||||||
|
OR: [
|
||||||
|
{ title: { contains: search, mode: 'insensitive' } },
|
||||||
|
{ Recipient: { some: { name: { contains: search, mode: 'insensitive' } } } },
|
||||||
|
{ Recipient: { some: { email: { contains: search, mode: 'insensitive' } } } },
|
||||||
|
],
|
||||||
};
|
};
|
||||||
|
|
||||||
const getCounts = async ({ user, createdAt }: GetCountsOption) => {
|
|
||||||
return Promise.all([
|
return Promise.all([
|
||||||
// Owner counts.
|
// Owner counts.
|
||||||
prisma.document.groupBy({
|
prisma.document.groupBy({
|
||||||
@@ -83,6 +103,7 @@ const getCounts = async ({ user, createdAt }: GetCountsOption) => {
|
|||||||
createdAt,
|
createdAt,
|
||||||
teamId: null,
|
teamId: null,
|
||||||
deletedAt: null,
|
deletedAt: null,
|
||||||
|
AND: [searchFilter],
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
// Not signed counts.
|
// Not signed counts.
|
||||||
@@ -101,6 +122,7 @@ const getCounts = async ({ user, createdAt }: GetCountsOption) => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
createdAt,
|
createdAt,
|
||||||
|
AND: [searchFilter],
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
// Has signed counts.
|
// Has signed counts.
|
||||||
@@ -138,6 +160,7 @@ const getCounts = async ({ user, createdAt }: GetCountsOption) => {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
AND: [searchFilter],
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
]);
|
]);
|
||||||
@@ -147,7 +170,11 @@ type GetTeamCountsOption = {
|
|||||||
teamId: number;
|
teamId: number;
|
||||||
teamEmail?: string;
|
teamEmail?: string;
|
||||||
senderIds?: number[];
|
senderIds?: number[];
|
||||||
|
currentUserEmail: string;
|
||||||
|
userId: number;
|
||||||
createdAt: Prisma.DocumentWhereInput['createdAt'];
|
createdAt: Prisma.DocumentWhereInput['createdAt'];
|
||||||
|
currentTeamMemberRole?: TeamMemberRole;
|
||||||
|
search?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
const getTeamCounts = async (options: GetTeamCountsOption) => {
|
const getTeamCounts = async (options: GetTeamCountsOption) => {
|
||||||
@@ -162,6 +189,14 @@ const getTeamCounts = async (options: GetTeamCountsOption) => {
|
|||||||
}
|
}
|
||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
|
const searchFilter: Prisma.DocumentWhereInput = {
|
||||||
|
OR: [
|
||||||
|
{ title: { contains: options.search, mode: 'insensitive' } },
|
||||||
|
{ Recipient: { some: { name: { contains: options.search, mode: 'insensitive' } } } },
|
||||||
|
{ Recipient: { some: { email: { contains: options.search, mode: 'insensitive' } } } },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
let ownerCountsWhereInput: Prisma.DocumentWhereInput = {
|
let ownerCountsWhereInput: Prisma.DocumentWhereInput = {
|
||||||
userId: userIdWhereClause,
|
userId: userIdWhereClause,
|
||||||
createdAt,
|
createdAt,
|
||||||
@@ -172,6 +207,50 @@ const getTeamCounts = async (options: GetTeamCountsOption) => {
|
|||||||
let notSignedCountsGroupByArgs = null;
|
let notSignedCountsGroupByArgs = null;
|
||||||
let hasSignedCountsGroupByArgs = null;
|
let hasSignedCountsGroupByArgs = null;
|
||||||
|
|
||||||
|
const visibilityFilters = [
|
||||||
|
...match(options.currentTeamMemberRole)
|
||||||
|
.with(TeamMemberRole.ADMIN, () => [
|
||||||
|
{ visibility: DocumentVisibility.EVERYONE },
|
||||||
|
{ visibility: DocumentVisibility.MANAGER_AND_ABOVE },
|
||||||
|
{ visibility: DocumentVisibility.ADMIN },
|
||||||
|
])
|
||||||
|
.with(TeamMemberRole.MANAGER, () => [
|
||||||
|
{ visibility: DocumentVisibility.EVERYONE },
|
||||||
|
{ visibility: DocumentVisibility.MANAGER_AND_ABOVE },
|
||||||
|
])
|
||||||
|
.otherwise(() => [{ visibility: DocumentVisibility.EVERYONE }]),
|
||||||
|
];
|
||||||
|
|
||||||
|
ownerCountsWhereInput = {
|
||||||
|
...ownerCountsWhereInput,
|
||||||
|
OR: [
|
||||||
|
{
|
||||||
|
AND: [
|
||||||
|
{
|
||||||
|
visibility: {
|
||||||
|
in: visibilityFilters.map((filter) => filter.visibility),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Recipient: {
|
||||||
|
none: {
|
||||||
|
email: options.currentUserEmail,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Recipient: {
|
||||||
|
some: {
|
||||||
|
email: options.currentUserEmail,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
...searchFilter,
|
||||||
|
};
|
||||||
|
|
||||||
if (teamEmail) {
|
if (teamEmail) {
|
||||||
ownerCountsWhereInput = {
|
ownerCountsWhereInput = {
|
||||||
userId: userIdWhereClause,
|
userId: userIdWhereClause,
|
||||||
|
|||||||
@@ -3,7 +3,13 @@ import type { RequestMetadata } from '@documenso/lib/universal/extract-request-m
|
|||||||
import { putPdfFile } from '@documenso/lib/universal/upload/put-file';
|
import { putPdfFile } from '@documenso/lib/universal/upload/put-file';
|
||||||
import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs';
|
import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import { DocumentStatus, RecipientRole, SendStatus, SigningStatus } from '@documenso/prisma/client';
|
import {
|
||||||
|
DocumentSigningOrder,
|
||||||
|
DocumentStatus,
|
||||||
|
RecipientRole,
|
||||||
|
SendStatus,
|
||||||
|
SigningStatus,
|
||||||
|
} from '@documenso/prisma/client';
|
||||||
import { WebhookTriggerEvents } from '@documenso/prisma/client';
|
import { WebhookTriggerEvents } from '@documenso/prisma/client';
|
||||||
|
|
||||||
import { jobs } from '../../jobs/client';
|
import { jobs } from '../../jobs/client';
|
||||||
@@ -57,7 +63,9 @@ export const sendDocument = async ({
|
|||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
include: {
|
include: {
|
||||||
Recipient: true,
|
Recipient: {
|
||||||
|
orderBy: [{ signingOrder: { sort: 'asc', nulls: 'last' } }, { id: 'asc' }],
|
||||||
|
},
|
||||||
documentMeta: true,
|
documentMeta: true,
|
||||||
documentData: true,
|
documentData: true,
|
||||||
},
|
},
|
||||||
@@ -75,6 +83,21 @@ export const sendDocument = async ({
|
|||||||
throw new Error('Can not send completed document');
|
throw new Error('Can not send completed document');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const signingOrder = document.documentMeta?.signingOrder || DocumentSigningOrder.PARALLEL;
|
||||||
|
|
||||||
|
let recipientsToNotify = document.Recipient;
|
||||||
|
|
||||||
|
if (signingOrder === DocumentSigningOrder.SEQUENTIAL) {
|
||||||
|
// Get the currently active recipient.
|
||||||
|
recipientsToNotify = document.Recipient.filter(
|
||||||
|
(r) => r.signingStatus === SigningStatus.NOT_SIGNED && r.role !== RecipientRole.CC,
|
||||||
|
).slice(0, 1);
|
||||||
|
|
||||||
|
// Secondary filter so we aren't resending if the current active recipient has already
|
||||||
|
// received the document.
|
||||||
|
recipientsToNotify.filter((r) => r.sendStatus !== SendStatus.SENT);
|
||||||
|
}
|
||||||
|
|
||||||
const { documentData } = document;
|
const { documentData } = document;
|
||||||
|
|
||||||
if (!documentData.data) {
|
if (!documentData.data) {
|
||||||
@@ -135,7 +158,7 @@ export const sendDocument = async ({
|
|||||||
|
|
||||||
if (sendEmail) {
|
if (sendEmail) {
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
document.Recipient.map(async (recipient) => {
|
recipientsToNotify.map(async (recipient) => {
|
||||||
if (recipient.sendStatus === SendStatus.SENT || recipient.role === RecipientRole.CC) {
|
if (recipient.sendStatus === SendStatus.SENT || recipient.role === RecipientRole.CC) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import type { RequestMetadata } from '@documenso/lib/universal/extract-request-m
|
|||||||
import type { CreateDocumentAuditLogDataResponse } from '@documenso/lib/utils/document-audit-logs';
|
import type { CreateDocumentAuditLogDataResponse } from '@documenso/lib/utils/document-audit-logs';
|
||||||
import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs';
|
import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
|
import type { DocumentVisibility } from '@documenso/prisma/client';
|
||||||
import { DocumentStatus } from '@documenso/prisma/client';
|
import { DocumentStatus } from '@documenso/prisma/client';
|
||||||
|
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
import { AppError, AppErrorCode } from '../../errors/app-error';
|
||||||
@@ -19,6 +20,7 @@ export type UpdateDocumentSettingsOptions = {
|
|||||||
data: {
|
data: {
|
||||||
title?: string;
|
title?: string;
|
||||||
externalId?: string | null;
|
externalId?: string | null;
|
||||||
|
visibility?: string | null;
|
||||||
globalAccessAuth?: TDocumentAccessAuthTypes | null;
|
globalAccessAuth?: TDocumentAccessAuthTypes | null;
|
||||||
globalActionAuth?: TDocumentActionAuthTypes | null;
|
globalActionAuth?: TDocumentActionAuthTypes | null;
|
||||||
};
|
};
|
||||||
@@ -91,10 +93,14 @@ export const updateDocumentSettings = async ({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const isTitleSame = data.title === document.title;
|
const isTitleSame = data.title === undefined || data.title === document.title;
|
||||||
const isExternalIdSame = data.externalId === document.externalId;
|
const isExternalIdSame = data.externalId === undefined || data.externalId === document.externalId;
|
||||||
const isGlobalAccessSame = documentGlobalAccessAuth === newGlobalAccessAuth;
|
const isGlobalAccessSame =
|
||||||
const isGlobalActionSame = documentGlobalActionAuth === newGlobalActionAuth;
|
documentGlobalAccessAuth === undefined || documentGlobalAccessAuth === newGlobalAccessAuth;
|
||||||
|
const isGlobalActionSame =
|
||||||
|
documentGlobalActionAuth === undefined || documentGlobalActionAuth === newGlobalActionAuth;
|
||||||
|
const isDocumentVisibilitySame =
|
||||||
|
data.visibility === undefined || data.visibility === document.visibility;
|
||||||
|
|
||||||
const auditLogs: CreateDocumentAuditLogDataResponse[] = [];
|
const auditLogs: CreateDocumentAuditLogDataResponse[] = [];
|
||||||
|
|
||||||
@@ -165,6 +171,21 @@ export const updateDocumentSettings = async ({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!isDocumentVisibilitySame) {
|
||||||
|
auditLogs.push(
|
||||||
|
createDocumentAuditLogData({
|
||||||
|
type: DOCUMENT_AUDIT_LOG_TYPE.DOCUMENT_VISIBILITY_UPDATED,
|
||||||
|
documentId,
|
||||||
|
user,
|
||||||
|
requestMetadata,
|
||||||
|
data: {
|
||||||
|
from: document.visibility,
|
||||||
|
to: data.visibility || '',
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
// Early return if nothing is required.
|
// Early return if nothing is required.
|
||||||
if (auditLogs.length === 0) {
|
if (auditLogs.length === 0) {
|
||||||
return document;
|
return document;
|
||||||
@@ -182,7 +203,8 @@ export const updateDocumentSettings = async ({
|
|||||||
},
|
},
|
||||||
data: {
|
data: {
|
||||||
title: data.title,
|
title: data.title,
|
||||||
externalId: data.externalId || null,
|
externalId: data.externalId,
|
||||||
|
visibility: data.visibility as DocumentVisibility,
|
||||||
authOptions,
|
authOptions,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { isDeepEqual } from 'remeda';
|
||||||
|
|
||||||
import { validateCheckboxField } from '@documenso/lib/advanced-fields-validation/validate-checkbox';
|
import { validateCheckboxField } from '@documenso/lib/advanced-fields-validation/validate-checkbox';
|
||||||
import { validateDropdownField } from '@documenso/lib/advanced-fields-validation/validate-dropdown';
|
import { validateDropdownField } from '@documenso/lib/advanced-fields-validation/validate-dropdown';
|
||||||
import { validateNumberField } from '@documenso/lib/advanced-fields-validation/validate-number';
|
import { validateNumberField } from '@documenso/lib/advanced-fields-validation/validate-number';
|
||||||
@@ -20,22 +22,15 @@ import {
|
|||||||
} from '@documenso/lib/utils/document-audit-logs';
|
} from '@documenso/lib/utils/document-audit-logs';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import type { Field } from '@documenso/prisma/client';
|
import type { Field } from '@documenso/prisma/client';
|
||||||
import { FieldType, SendStatus, SigningStatus } from '@documenso/prisma/client';
|
import { FieldType } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
import { AppError, AppErrorCode } from '../../errors/app-error';
|
||||||
|
import { canRecipientFieldsBeModified } from '../../utils/recipients';
|
||||||
|
|
||||||
export interface SetFieldsForDocumentOptions {
|
export interface SetFieldsForDocumentOptions {
|
||||||
userId: number;
|
userId: number;
|
||||||
documentId: number;
|
documentId: number;
|
||||||
fields: {
|
fields: FieldData[];
|
||||||
id?: number | null;
|
|
||||||
type: FieldType;
|
|
||||||
signerEmail: string;
|
|
||||||
pageNumber: number;
|
|
||||||
pageX: number;
|
|
||||||
pageY: number;
|
|
||||||
pageWidth: number;
|
|
||||||
pageHeight: number;
|
|
||||||
fieldMeta?: FieldMeta;
|
|
||||||
}[];
|
|
||||||
requestMetadata?: RequestMetadata;
|
requestMetadata?: RequestMetadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,6 +58,9 @@ export const setFieldsForDocument = async ({
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
include: {
|
||||||
|
Recipient: true,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const user = await prisma.user.findFirstOrThrow({
|
const user = await prisma.user.findFirstOrThrow({
|
||||||
@@ -97,20 +95,35 @@ export const setFieldsForDocument = async ({
|
|||||||
(existingField) => !fields.find((field) => field.id === existingField.id),
|
(existingField) => !fields.find((field) => field.id === existingField.id),
|
||||||
);
|
);
|
||||||
|
|
||||||
const linkedFields = fields
|
const linkedFields = fields.map((field) => {
|
||||||
.map((field) => {
|
|
||||||
const existing = existingFields.find((existingField) => existingField.id === field.id);
|
const existing = existingFields.find((existingField) => existingField.id === field.id);
|
||||||
|
|
||||||
|
const recipient = document.Recipient.find(
|
||||||
|
(recipient) => recipient.email.toLowerCase() === field.signerEmail.toLowerCase(),
|
||||||
|
);
|
||||||
|
|
||||||
|
// Each field MUST have a recipient associated with it.
|
||||||
|
if (!recipient) {
|
||||||
|
throw new AppError(AppErrorCode.INVALID_REQUEST, `Recipient not found for field ${field.id}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether the existing field can be modified.
|
||||||
|
if (
|
||||||
|
existing &&
|
||||||
|
hasFieldBeenChanged(existing, field) &&
|
||||||
|
!canRecipientFieldsBeModified(recipient, existingFields)
|
||||||
|
) {
|
||||||
|
throw new AppError(
|
||||||
|
AppErrorCode.INVALID_REQUEST,
|
||||||
|
'Cannot modify a field where the recipient has already interacted with the document',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...field,
|
...field,
|
||||||
_persisted: existing,
|
_persisted: existing,
|
||||||
|
_recipient: recipient,
|
||||||
};
|
};
|
||||||
})
|
|
||||||
.filter((field) => {
|
|
||||||
return (
|
|
||||||
field._persisted?.Recipient?.sendStatus !== SendStatus.SENT &&
|
|
||||||
field._persisted?.Recipient?.signingStatus !== SigningStatus.SIGNED
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const persistedFields = await prisma.$transaction(async (tx) => {
|
const persistedFields = await prisma.$transaction(async (tx) => {
|
||||||
@@ -322,3 +335,33 @@ export const setFieldsForDocument = async ({
|
|||||||
|
|
||||||
return [...filteredFields, ...persistedFields];
|
return [...filteredFields, ...persistedFields];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If you change this you MUST update the `hasFieldBeenChanged` function.
|
||||||
|
*/
|
||||||
|
type FieldData = {
|
||||||
|
id?: number | null;
|
||||||
|
type: FieldType;
|
||||||
|
signerEmail: string;
|
||||||
|
pageNumber: number;
|
||||||
|
pageX: number;
|
||||||
|
pageY: number;
|
||||||
|
pageWidth: number;
|
||||||
|
pageHeight: number;
|
||||||
|
fieldMeta?: FieldMeta;
|
||||||
|
};
|
||||||
|
|
||||||
|
const hasFieldBeenChanged = (field: Field, newFieldData: FieldData) => {
|
||||||
|
const currentFieldMeta = field.fieldMeta || null;
|
||||||
|
const newFieldMeta = newFieldData.fieldMeta || null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
field.type !== newFieldData.type ||
|
||||||
|
field.page !== newFieldData.pageNumber ||
|
||||||
|
field.positionX.toNumber() !== newFieldData.pageX ||
|
||||||
|
field.positionY.toNumber() !== newFieldData.pageY ||
|
||||||
|
field.width.toNumber() !== newFieldData.pageWidth ||
|
||||||
|
field.height.toNumber() !== newFieldData.pageHeight ||
|
||||||
|
!isDeepEqual(currentFieldMeta, newFieldMeta)
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|||||||
@@ -4,5 +4,8 @@ import { cookies } from 'next/headers';
|
|||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/require-await
|
// eslint-disable-next-line @typescript-eslint/require-await
|
||||||
export const switchI18NLanguage = async (lang: string) => {
|
export const switchI18NLanguage = async (lang: string) => {
|
||||||
cookies().set('language', lang);
|
// Two year expiry.
|
||||||
|
const maxAge = 60 * 60 * 24 * 365 * 2;
|
||||||
|
|
||||||
|
cookies().set('language', lang, { maxAge });
|
||||||
};
|
};
|
||||||
|
|||||||
46
packages/lib/server-only/recipient/get-is-recipient-turn.ts
Normal file
46
packages/lib/server-only/recipient/get-is-recipient-turn.ts
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
import { DocumentSigningOrder, SigningStatus } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
export type GetIsRecipientTurnOptions = {
|
||||||
|
token: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export async function getIsRecipientsTurnToSign({ token }: GetIsRecipientTurnOptions) {
|
||||||
|
const document = await prisma.document.findFirstOrThrow({
|
||||||
|
where: {
|
||||||
|
Recipient: {
|
||||||
|
some: {
|
||||||
|
token,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
documentMeta: true,
|
||||||
|
Recipient: {
|
||||||
|
orderBy: {
|
||||||
|
signingOrder: 'asc',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (document.documentMeta?.signingOrder !== DocumentSigningOrder.SEQUENTIAL) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
const recipients = document.Recipient;
|
||||||
|
|
||||||
|
const currentRecipientIndex = recipients.findIndex((r) => r.token === token);
|
||||||
|
|
||||||
|
if (currentRecipientIndex === -1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < currentRecipientIndex; i++) {
|
||||||
|
if (recipients[i].signingStatus !== SigningStatus.SIGNED) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
@@ -1,4 +1,9 @@
|
|||||||
|
import { createElement } from 'react';
|
||||||
|
|
||||||
import { isUserEnterprise } from '@documenso/ee/server-only/util/is-document-enterprise';
|
import { isUserEnterprise } from '@documenso/ee/server-only/util/is-document-enterprise';
|
||||||
|
import { mailer } from '@documenso/email/mailer';
|
||||||
|
import { render } from '@documenso/email/render';
|
||||||
|
import RecipientRemovedFromDocumentTemplate from '@documenso/email/templates/recipient-removed-from-document';
|
||||||
import { DOCUMENT_AUDIT_LOG_TYPE } from '@documenso/lib/types/document-audit-logs';
|
import { DOCUMENT_AUDIT_LOG_TYPE } from '@documenso/lib/types/document-audit-logs';
|
||||||
import {
|
import {
|
||||||
type TRecipientActionAuthTypes,
|
type TRecipientActionAuthTypes,
|
||||||
@@ -16,19 +21,16 @@ import type { Recipient } from '@documenso/prisma/client';
|
|||||||
import { RecipientRole } from '@documenso/prisma/client';
|
import { RecipientRole } from '@documenso/prisma/client';
|
||||||
import { SendStatus, SigningStatus } from '@documenso/prisma/client';
|
import { SendStatus, SigningStatus } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
import { NEXT_PUBLIC_WEBAPP_URL } from '../../constants/app';
|
||||||
|
import { FROM_ADDRESS, FROM_NAME } from '../../constants/email';
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
import { AppError, AppErrorCode } from '../../errors/app-error';
|
||||||
|
import { canRecipientBeModified } from '../../utils/recipients';
|
||||||
|
|
||||||
export interface SetRecipientsForDocumentOptions {
|
export interface SetRecipientsForDocumentOptions {
|
||||||
userId: number;
|
userId: number;
|
||||||
teamId?: number;
|
teamId?: number;
|
||||||
documentId: number;
|
documentId: number;
|
||||||
recipients: {
|
recipients: RecipientData[];
|
||||||
id?: number | null;
|
|
||||||
email: string;
|
|
||||||
name: string;
|
|
||||||
role: RecipientRole;
|
|
||||||
actionAuth?: TRecipientActionAuthTypes | null;
|
|
||||||
}[];
|
|
||||||
requestMetadata?: RequestMetadata;
|
requestMetadata?: RequestMetadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -58,6 +60,9 @@ export const setRecipientsForDocument = async ({
|
|||||||
teamId: null,
|
teamId: null,
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
include: {
|
||||||
|
Field: true,
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const user = await prisma.user.findFirstOrThrow({
|
const user = await prisma.user.findFirstOrThrow({
|
||||||
@@ -115,24 +120,27 @@ export const setRecipientsForDocument = async ({
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
const linkedRecipients = normalizedRecipients
|
const linkedRecipients = normalizedRecipients.map((recipient) => {
|
||||||
.map((recipient) => {
|
|
||||||
const existing = existingRecipients.find(
|
const existing = existingRecipients.find(
|
||||||
(existingRecipient) =>
|
(existingRecipient) =>
|
||||||
existingRecipient.id === recipient.id || existingRecipient.email === recipient.email,
|
existingRecipient.id === recipient.id || existingRecipient.email === recipient.email,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (
|
||||||
|
existing &&
|
||||||
|
hasRecipientBeenChanged(existing, recipient) &&
|
||||||
|
!canRecipientBeModified(existing, document.Field)
|
||||||
|
) {
|
||||||
|
throw new AppError(
|
||||||
|
AppErrorCode.INVALID_REQUEST,
|
||||||
|
'Cannot modify a recipient who has already interacted with the document',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...recipient,
|
...recipient,
|
||||||
_persisted: existing,
|
_persisted: existing,
|
||||||
};
|
};
|
||||||
})
|
|
||||||
.filter((recipient) => {
|
|
||||||
return (
|
|
||||||
recipient._persisted?.role === RecipientRole.CC ||
|
|
||||||
(recipient._persisted?.sendStatus !== SendStatus.SENT &&
|
|
||||||
recipient._persisted?.signingStatus !== SigningStatus.SIGNED)
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const persistedRecipients = await prisma.$transaction(async (tx) => {
|
const persistedRecipients = await prisma.$transaction(async (tx) => {
|
||||||
@@ -156,6 +164,7 @@ export const setRecipientsForDocument = async ({
|
|||||||
name: recipient.name,
|
name: recipient.name,
|
||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
documentId,
|
documentId,
|
||||||
sendStatus: recipient.role === RecipientRole.CC ? SendStatus.SENT : SendStatus.NOT_SENT,
|
sendStatus: recipient.role === RecipientRole.CC ? SendStatus.SENT : SendStatus.NOT_SENT,
|
||||||
signingStatus:
|
signingStatus:
|
||||||
@@ -166,6 +175,7 @@ export const setRecipientsForDocument = async ({
|
|||||||
name: recipient.name,
|
name: recipient.name,
|
||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
token: nanoid(),
|
token: nanoid(),
|
||||||
documentId,
|
documentId,
|
||||||
sendStatus: recipient.role === RecipientRole.CC ? SendStatus.SENT : SendStatus.NOT_SENT,
|
sendStatus: recipient.role === RecipientRole.CC ? SendStatus.SENT : SendStatus.NOT_SENT,
|
||||||
@@ -265,6 +275,37 @@ export const setRecipientsForDocument = async ({
|
|||||||
),
|
),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Send emails to deleted recipients.
|
||||||
|
await Promise.all(
|
||||||
|
removedRecipients.map(async (recipient) => {
|
||||||
|
if (recipient.sendStatus !== SendStatus.SENT) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const assetBaseUrl = NEXT_PUBLIC_WEBAPP_URL() || 'http://localhost:3000';
|
||||||
|
|
||||||
|
const template = createElement(RecipientRemovedFromDocumentTemplate, {
|
||||||
|
documentName: document.title,
|
||||||
|
inviterName: user.name || undefined,
|
||||||
|
assetBaseUrl,
|
||||||
|
});
|
||||||
|
|
||||||
|
await mailer.sendMail({
|
||||||
|
to: {
|
||||||
|
address: recipient.email,
|
||||||
|
name: recipient.name,
|
||||||
|
},
|
||||||
|
from: {
|
||||||
|
name: FROM_NAME,
|
||||||
|
address: FROM_ADDRESS,
|
||||||
|
},
|
||||||
|
subject: 'You have been removed from a document',
|
||||||
|
html: render(template),
|
||||||
|
text: render(template, { plainText: true }),
|
||||||
|
});
|
||||||
|
}),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Filter out recipients that have been removed or have been updated.
|
// Filter out recipients that have been removed or have been updated.
|
||||||
@@ -281,3 +322,27 @@ export const setRecipientsForDocument = async ({
|
|||||||
|
|
||||||
return [...filteredRecipients, ...persistedRecipients];
|
return [...filteredRecipients, ...persistedRecipients];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If you change this you MUST update the `hasRecipientBeenChanged` function.
|
||||||
|
*/
|
||||||
|
type RecipientData = {
|
||||||
|
id?: number | null;
|
||||||
|
email: string;
|
||||||
|
name: string;
|
||||||
|
role: RecipientRole;
|
||||||
|
signingOrder?: number | null;
|
||||||
|
actionAuth?: TRecipientActionAuthTypes | null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const hasRecipientBeenChanged = (recipient: Recipient, newRecipientData: RecipientData) => {
|
||||||
|
const authOptions = ZRecipientAuthOptionsSchema.parse(recipient.authOptions);
|
||||||
|
|
||||||
|
return (
|
||||||
|
recipient.email !== newRecipientData.email ||
|
||||||
|
recipient.name !== newRecipientData.name ||
|
||||||
|
recipient.role !== newRecipientData.role ||
|
||||||
|
recipient.signingOrder !== newRecipientData.signingOrder ||
|
||||||
|
authOptions.actionAuth !== newRecipientData.actionAuth
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ export type SetRecipientsForTemplateOptions = {
|
|||||||
email: string;
|
email: string;
|
||||||
name: string;
|
name: string;
|
||||||
role: RecipientRole;
|
role: RecipientRole;
|
||||||
|
signingOrder?: number | null;
|
||||||
actionAuth?: TRecipientActionAuthTypes | null;
|
actionAuth?: TRecipientActionAuthTypes | null;
|
||||||
}[];
|
}[];
|
||||||
};
|
};
|
||||||
@@ -162,6 +163,7 @@ export const setRecipientsForTemplate = async ({
|
|||||||
name: recipient.name,
|
name: recipient.name,
|
||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
templateId,
|
templateId,
|
||||||
authOptions,
|
authOptions,
|
||||||
},
|
},
|
||||||
@@ -169,6 +171,7 @@ export const setRecipientsForTemplate = async ({
|
|||||||
name: recipient.name,
|
name: recipient.name,
|
||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
token: nanoid(),
|
token: nanoid(),
|
||||||
templateId,
|
templateId,
|
||||||
authOptions,
|
authOptions,
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ export type UpdateRecipientOptions = {
|
|||||||
email?: string;
|
email?: string;
|
||||||
name?: string;
|
name?: string;
|
||||||
role?: RecipientRole;
|
role?: RecipientRole;
|
||||||
|
signingOrder?: number | null;
|
||||||
actionAuth?: TRecipientActionAuthTypes | null;
|
actionAuth?: TRecipientActionAuthTypes | null;
|
||||||
userId: number;
|
userId: number;
|
||||||
teamId?: number;
|
teamId?: number;
|
||||||
@@ -30,6 +31,7 @@ export const updateRecipient = async ({
|
|||||||
email,
|
email,
|
||||||
name,
|
name,
|
||||||
role,
|
role,
|
||||||
|
signingOrder,
|
||||||
actionAuth,
|
actionAuth,
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
@@ -112,6 +114,7 @@ export const updateRecipient = async ({
|
|||||||
email: email?.toLowerCase() ?? recipient.email,
|
email: email?.toLowerCase() ?? recipient.email,
|
||||||
name: name ?? recipient.name,
|
name: name ?? recipient.name,
|
||||||
role: role ?? recipient.role,
|
role: role ?? recipient.role,
|
||||||
|
signingOrder,
|
||||||
authOptions: createRecipientAuthOptions({
|
authOptions: createRecipientAuthOptions({
|
||||||
accessAuth: recipientAuthOptions.accessAuth,
|
accessAuth: recipientAuthOptions.accessAuth,
|
||||||
actionAuth: actionAuth ?? null,
|
actionAuth: actionAuth ?? null,
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ export type GetTeamByIdOptions = {
|
|||||||
teamId: number;
|
teamId: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type GetTeamResponse = Awaited<ReturnType<typeof getTeamById>>;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a team given a teamId.
|
* Get a team given a teamId.
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ export type CreateDocumentFromTemplateLegacyOptions = {
|
|||||||
name?: string;
|
name?: string;
|
||||||
email: string;
|
email: string;
|
||||||
role?: RecipientRole;
|
role?: RecipientRole;
|
||||||
|
signingOrder?: number | null;
|
||||||
}[];
|
}[];
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -73,6 +74,7 @@ export const createDocumentFromTemplateLegacy = async ({
|
|||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
name: recipient.name,
|
name: recipient.name,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
token: nanoid(),
|
token: nanoid(),
|
||||||
})),
|
})),
|
||||||
},
|
},
|
||||||
@@ -129,12 +131,14 @@ export const createDocumentFromTemplateLegacy = async ({
|
|||||||
name: recipient.name,
|
name: recipient.name,
|
||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
},
|
},
|
||||||
create: {
|
create: {
|
||||||
documentId: document.id,
|
documentId: document.id,
|
||||||
email: recipient.email,
|
email: recipient.email,
|
||||||
name: recipient.name,
|
name: recipient.name,
|
||||||
role: recipient.role,
|
role: recipient.role,
|
||||||
|
signingOrder: recipient.signingOrder,
|
||||||
token: nanoid(),
|
token: nanoid(),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
import { nanoid } from '@documenso/lib/universal/id';
|
import { nanoid } from '@documenso/lib/universal/id';
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import type { Field } from '@documenso/prisma/client';
|
|
||||||
import {
|
import {
|
||||||
|
DocumentSigningOrder,
|
||||||
DocumentSource,
|
DocumentSource,
|
||||||
|
type Field,
|
||||||
type Recipient,
|
type Recipient,
|
||||||
RecipientRole,
|
RecipientRole,
|
||||||
SendStatus,
|
SendStatus,
|
||||||
@@ -41,6 +42,7 @@ export type CreateDocumentFromTemplateOptions = {
|
|||||||
id: number;
|
id: number;
|
||||||
name?: string;
|
name?: string;
|
||||||
email: string;
|
email: string;
|
||||||
|
signingOrder?: number | null;
|
||||||
}[];
|
}[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -54,6 +56,7 @@ export type CreateDocumentFromTemplateOptions = {
|
|||||||
password?: string;
|
password?: string;
|
||||||
dateFormat?: string;
|
dateFormat?: string;
|
||||||
redirectUrl?: string;
|
redirectUrl?: string;
|
||||||
|
signingOrder?: DocumentSigningOrder;
|
||||||
};
|
};
|
||||||
requestMetadata?: RequestMetadata;
|
requestMetadata?: RequestMetadata;
|
||||||
};
|
};
|
||||||
@@ -134,6 +137,7 @@ export const createDocumentFromTemplate = async ({
|
|||||||
name: foundRecipient ? foundRecipient.name ?? '' : templateRecipient.name,
|
name: foundRecipient ? foundRecipient.name ?? '' : templateRecipient.name,
|
||||||
email: foundRecipient ? foundRecipient.email : templateRecipient.email,
|
email: foundRecipient ? foundRecipient.email : templateRecipient.email,
|
||||||
role: templateRecipient.role,
|
role: templateRecipient.role,
|
||||||
|
signingOrder: foundRecipient?.signingOrder ?? templateRecipient.signingOrder,
|
||||||
authOptions: templateRecipient.authOptions,
|
authOptions: templateRecipient.authOptions,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
@@ -150,7 +154,7 @@ export const createDocumentFromTemplate = async ({
|
|||||||
const document = await tx.document.create({
|
const document = await tx.document.create({
|
||||||
data: {
|
data: {
|
||||||
source: DocumentSource.TEMPLATE,
|
source: DocumentSource.TEMPLATE,
|
||||||
externalId,
|
externalId: externalId || template.externalId,
|
||||||
templateId: template.id,
|
templateId: template.id,
|
||||||
userId,
|
userId,
|
||||||
teamId: template.teamId,
|
teamId: template.teamId,
|
||||||
@@ -168,6 +172,10 @@ export const createDocumentFromTemplate = async ({
|
|||||||
password: override?.password || template.templateMeta?.password,
|
password: override?.password || template.templateMeta?.password,
|
||||||
dateFormat: override?.dateFormat || template.templateMeta?.dateFormat,
|
dateFormat: override?.dateFormat || template.templateMeta?.dateFormat,
|
||||||
redirectUrl: override?.redirectUrl || template.templateMeta?.redirectUrl,
|
redirectUrl: override?.redirectUrl || template.templateMeta?.redirectUrl,
|
||||||
|
signingOrder:
|
||||||
|
override?.signingOrder ||
|
||||||
|
template.templateMeta?.signingOrder ||
|
||||||
|
DocumentSigningOrder.PARALLEL,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
Recipient: {
|
Recipient: {
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ export const updateTemplateSettings = async ({
|
|||||||
meta,
|
meta,
|
||||||
data,
|
data,
|
||||||
}: UpdateTemplateSettingsOptions) => {
|
}: UpdateTemplateSettingsOptions) => {
|
||||||
if (Object.values(data).length === 0) {
|
if (Object.values(data).length === 0 && Object.keys(meta ?? {}).length === 0) {
|
||||||
throw new AppError(AppErrorCode.INVALID_BODY, 'Missing data to update');
|
throw new AppError(AppErrorCode.INVALID_BODY, 'Missing data to update');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,7 +100,7 @@ export const updateTemplateSettings = async ({
|
|||||||
},
|
},
|
||||||
data: {
|
data: {
|
||||||
title: data.title,
|
title: data.title,
|
||||||
externalId: data.externalId || null,
|
externalId: data.externalId,
|
||||||
type: data.type,
|
type: data.type,
|
||||||
publicDescription: data.publicDescription,
|
publicDescription: data.publicDescription,
|
||||||
publicTitle: data.publicTitle,
|
publicTitle: data.publicTitle,
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"POT-Creation-Date: 2024-08-22 05:16+0000\n"
|
"POT-Creation-Date: 2024-07-24 13:01+1000\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=utf-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"X-Generator: @lingui/cli\n"
|
"X-Generator: @lingui/cli\n"
|
||||||
"Language: de\n"
|
"Language: de\n"
|
||||||
"Project-Id-Version: documenso-app\n"
|
"Project-Id-Version: documenso-app\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"PO-Revision-Date: 2024-09-05 06:04\n"
|
"PO-Revision-Date: 2024-09-16 16:03\n"
|
||||||
"Last-Translator: \n"
|
"Last-Translator: \n"
|
||||||
"Language-Team: German\n"
|
"Language-Team: German\n"
|
||||||
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
||||||
@@ -60,12 +60,12 @@ msgstr "<0>Passkey erforderlich</0> - Der Empfänger muss ein Konto haben und de
|
|||||||
msgid "Add a document"
|
msgid "Add a document"
|
||||||
msgstr "Dokument hinzufügen"
|
msgstr "Dokument hinzufügen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:305
|
#: packages/ui/primitives/document-flow/add-settings.tsx:336
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:339
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:339
|
||||||
msgid "Add a URL to redirect the user to once the document is signed"
|
msgid "Add a URL to redirect the user to once the document is signed"
|
||||||
msgstr "Fügen Sie eine URL hinzu, um den Benutzer nach der Unterzeichnung des Dokuments weiterzuleiten"
|
msgstr "Fügen Sie eine URL hinzu, um den Benutzer nach der Unterzeichnung des Dokuments weiterzuleiten"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:217
|
#: packages/ui/primitives/document-flow/add-settings.tsx:248
|
||||||
msgid "Add an external ID to the document. This can be used to identify the document in external systems."
|
msgid "Add an external ID to the document. This can be used to identify the document in external systems."
|
||||||
msgstr "Fügen Sie dem Dokument eine externe ID hinzu. Diese kann verwendet werden, um das Dokument in externen Systemen zu identifizieren."
|
msgstr "Fügen Sie dem Dokument eine externe ID hinzu. Diese kann verwendet werden, um das Dokument in externen Systemen zu identifizieren."
|
||||||
|
|
||||||
@@ -82,19 +82,19 @@ msgstr "Weitere Option hinzufügen"
|
|||||||
msgid "Add another value"
|
msgid "Add another value"
|
||||||
msgstr "Weiteren Wert hinzufügen"
|
msgstr "Weiteren Wert hinzufügen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:359
|
#: packages/ui/primitives/document-flow/add-signers.tsx:662
|
||||||
msgid "Add myself"
|
msgid "Add myself"
|
||||||
msgstr "Mich selbst hinzufügen"
|
msgstr "Mich selbst hinzufügen"
|
||||||
|
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:369
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:637
|
||||||
msgid "Add Myself"
|
msgid "Add Myself"
|
||||||
msgstr "Mich hinzufügen"
|
msgstr "Mich hinzufügen"
|
||||||
|
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:355
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:623
|
||||||
msgid "Add Placeholder Recipient"
|
msgid "Add Placeholder Recipient"
|
||||||
msgstr "Platzhalterempfänger hinzufügen"
|
msgstr "Platzhalterempfänger hinzufügen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:348
|
#: packages/ui/primitives/document-flow/add-signers.tsx:651
|
||||||
msgid "Add Signer"
|
msgid "Add Signer"
|
||||||
msgstr "Unterzeichner hinzufügen"
|
msgstr "Unterzeichner hinzufügen"
|
||||||
|
|
||||||
@@ -110,12 +110,12 @@ msgstr "Text zum Feld hinzufügen"
|
|||||||
msgid "Admin"
|
msgid "Admin"
|
||||||
msgstr "Admin"
|
msgstr "Admin"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:199
|
#: packages/ui/primitives/document-flow/add-settings.tsx:230
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:238
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:238
|
||||||
msgid "Advanced Options"
|
msgid "Advanced Options"
|
||||||
msgstr "Erweiterte Optionen"
|
msgstr "Erweiterte Optionen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:510
|
#: packages/ui/primitives/document-flow/add-fields.tsx:565
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:402
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:402
|
||||||
msgid "Advanced settings"
|
msgid "Advanced settings"
|
||||||
msgstr "Erweiterte Einstellungen"
|
msgstr "Erweiterte Einstellungen"
|
||||||
@@ -142,21 +142,25 @@ msgstr "Genehmigung"
|
|||||||
|
|
||||||
#: packages/ui/primitives/signature-pad/signature-pad.tsx:276
|
#: packages/ui/primitives/signature-pad/signature-pad.tsx:276
|
||||||
msgid "Black"
|
msgid "Black"
|
||||||
msgstr ""
|
msgstr "Schwarz"
|
||||||
|
|
||||||
#: packages/ui/primitives/signature-pad/signature-pad.tsx:290
|
#: packages/ui/primitives/signature-pad/signature-pad.tsx:290
|
||||||
msgid "Blue"
|
msgid "Blue"
|
||||||
msgstr ""
|
msgstr "Blau"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/field-item-advanced-settings.tsx:287
|
#: packages/ui/primitives/document-flow/field-item-advanced-settings.tsx:287
|
||||||
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:58
|
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:58
|
||||||
msgid "Cancel"
|
msgid "Cancel"
|
||||||
msgstr "Abbrechen"
|
msgstr "Abbrechen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:164
|
#: packages/ui/primitives/document-flow/add-signers.tsx:193
|
||||||
msgid "Cannot remove signer"
|
msgid "Cannot remove signer"
|
||||||
msgstr "Unterzeichner kann nicht entfernt werden"
|
msgstr "Unterzeichner kann nicht entfernt werden"
|
||||||
|
|
||||||
|
#: packages/ui/primitives/document-flow/add-signers.tsx:221
|
||||||
|
#~ msgid "Cannot update signer because they have already signed a field"
|
||||||
|
#~ msgstr ""
|
||||||
|
|
||||||
#: packages/lib/constants/recipient-roles.ts:17
|
#: packages/lib/constants/recipient-roles.ts:17
|
||||||
msgid "Cc"
|
msgid "Cc"
|
||||||
msgstr "Cc"
|
msgstr "Cc"
|
||||||
@@ -174,7 +178,7 @@ msgstr "CC'd"
|
|||||||
msgid "Character Limit"
|
msgid "Character Limit"
|
||||||
msgstr "Zeichenbeschränkung"
|
msgstr "Zeichenbeschränkung"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:932
|
#: packages/ui/primitives/document-flow/add-fields.tsx:993
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:788
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:788
|
||||||
msgid "Checkbox"
|
msgid "Checkbox"
|
||||||
msgstr "Checkbox"
|
msgstr "Checkbox"
|
||||||
@@ -203,7 +207,7 @@ msgstr "Schließen"
|
|||||||
msgid "Configure Direct Recipient"
|
msgid "Configure Direct Recipient"
|
||||||
msgstr "Direkten Empfänger konfigurieren"
|
msgstr "Direkten Empfänger konfigurieren"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:511
|
#: packages/ui/primitives/document-flow/add-fields.tsx:566
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:403
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:403
|
||||||
msgid "Configure the {0} field"
|
msgid "Configure the {0} field"
|
||||||
msgstr "Konfigurieren Sie das Feld {0}"
|
msgstr "Konfigurieren Sie das Feld {0}"
|
||||||
@@ -220,22 +224,22 @@ msgstr "In die Zwischenablage kopiert"
|
|||||||
msgid "Custom Text"
|
msgid "Custom Text"
|
||||||
msgstr "Benutzerdefinierter Text"
|
msgstr "Benutzerdefinierter Text"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:828
|
#: packages/ui/primitives/document-flow/add-fields.tsx:889
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:684
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:684
|
||||||
msgid "Date"
|
msgid "Date"
|
||||||
msgstr "Datum"
|
msgstr "Datum"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:240
|
#: packages/ui/primitives/document-flow/add-settings.tsx:271
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:279
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:279
|
||||||
msgid "Date Format"
|
msgid "Date Format"
|
||||||
msgstr "Datumsformat"
|
msgstr "Datumsformat"
|
||||||
|
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:312
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:570
|
||||||
msgid "Direct link receiver"
|
msgid "Direct link receiver"
|
||||||
msgstr "Empfänger des direkten Links"
|
msgstr "Empfänger des direkten Links"
|
||||||
|
|
||||||
#: packages/ui/components/document/document-global-auth-access-select.tsx:62
|
#: packages/ui/components/document/document-global-auth-access-select.tsx:62
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:166
|
#: packages/ui/primitives/document-flow/add-settings.tsx:174
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:151
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:151
|
||||||
msgid "Document access"
|
msgid "Document access"
|
||||||
msgstr "Dokumentenzugriff"
|
msgstr "Dokumentenzugriff"
|
||||||
@@ -252,7 +256,7 @@ msgstr "Herunterladen"
|
|||||||
msgid "Drag & drop your PDF here."
|
msgid "Drag & drop your PDF here."
|
||||||
msgstr "Ziehen Sie Ihr PDF hierher."
|
msgstr "Ziehen Sie Ihr PDF hierher."
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:958
|
#: packages/ui/primitives/document-flow/add-fields.tsx:1019
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:814
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:814
|
||||||
msgid "Dropdown"
|
msgid "Dropdown"
|
||||||
msgstr "Dropdown"
|
msgstr "Dropdown"
|
||||||
@@ -261,13 +265,12 @@ msgstr "Dropdown"
|
|||||||
msgid "Dropdown options"
|
msgid "Dropdown options"
|
||||||
msgstr "Dropdown-Optionen"
|
msgstr "Dropdown-Optionen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:776
|
#: packages/ui/primitives/document-flow/add-fields.tsx:837
|
||||||
#: packages/ui/primitives/document-flow/add-signature.tsx:272
|
#: packages/ui/primitives/document-flow/add-signature.tsx:272
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:232
|
#: packages/ui/primitives/document-flow/add-signers.tsx:500
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:239
|
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:632
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:632
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:210
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:463
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:217
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:470
|
||||||
msgid "Email"
|
msgid "Email"
|
||||||
msgstr "E-Mail"
|
msgstr "E-Mail"
|
||||||
|
|
||||||
@@ -275,10 +278,19 @@ msgstr "E-Mail"
|
|||||||
msgid "Email Options"
|
msgid "Email Options"
|
||||||
msgstr "E-Mail-Optionen"
|
msgstr "E-Mail-Optionen"
|
||||||
|
|
||||||
|
#: packages/ui/primitives/document-flow/add-fields.tsx:1082
|
||||||
|
msgid "Empty field"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
#: packages/lib/constants/template.ts:8
|
#: packages/lib/constants/template.ts:8
|
||||||
msgid "Enable Direct Link Signing"
|
msgid "Enable Direct Link Signing"
|
||||||
msgstr "Direktlink-Signierung aktivieren"
|
msgstr "Direktlink-Signierung aktivieren"
|
||||||
|
|
||||||
|
#: packages/ui/primitives/document-flow/add-signers.tsx:401
|
||||||
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:362
|
||||||
|
msgid "Enable signing order"
|
||||||
|
msgstr "Aktiviere die Signaturreihenfolge"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-password-dialog.tsx:84
|
#: packages/ui/primitives/document-password-dialog.tsx:84
|
||||||
msgid "Enter password"
|
msgid "Enter password"
|
||||||
msgstr "Passwort eingeben"
|
msgstr "Passwort eingeben"
|
||||||
@@ -287,7 +299,7 @@ msgstr "Passwort eingeben"
|
|||||||
msgid "Error"
|
msgid "Error"
|
||||||
msgstr "Fehler"
|
msgstr "Fehler"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:210
|
#: packages/ui/primitives/document-flow/add-settings.tsx:241
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:249
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:249
|
||||||
msgid "External ID"
|
msgid "External ID"
|
||||||
msgstr "Externe ID"
|
msgstr "Externe ID"
|
||||||
@@ -322,7 +334,7 @@ msgstr "Zurück"
|
|||||||
|
|
||||||
#: packages/ui/primitives/signature-pad/signature-pad.tsx:297
|
#: packages/ui/primitives/signature-pad/signature-pad.tsx:297
|
||||||
msgid "Green"
|
msgid "Green"
|
||||||
msgstr ""
|
msgstr "Grün"
|
||||||
|
|
||||||
#: packages/lib/constants/recipient-roles.ts:72
|
#: packages/lib/constants/recipient-roles.ts:72
|
||||||
msgid "I am a signer of this document"
|
msgid "I am a signer of this document"
|
||||||
@@ -367,7 +379,7 @@ msgstr "Max"
|
|||||||
msgid "Member"
|
msgid "Member"
|
||||||
msgstr "Mitglied"
|
msgstr "Mitglied"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-subject.tsx:229
|
#: packages/ui/primitives/document-flow/add-subject.tsx:95
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:215
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:215
|
||||||
msgid "Message <0>(Optional)</0>"
|
msgid "Message <0>(Optional)</0>"
|
||||||
msgstr "Nachricht <0>(Optional)</0>"
|
msgstr "Nachricht <0>(Optional)</0>"
|
||||||
@@ -376,12 +388,13 @@ msgstr "Nachricht <0>(Optional)</0>"
|
|||||||
msgid "Min"
|
msgid "Min"
|
||||||
msgstr "Min"
|
msgstr "Min"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:802
|
#: packages/ui/primitives/document-flow/add-fields.tsx:863
|
||||||
#: packages/ui/primitives/document-flow/add-signature.tsx:298
|
#: packages/ui/primitives/document-flow/add-signature.tsx:298
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:265
|
#: packages/ui/primitives/document-flow/add-signers.tsx:535
|
||||||
|
#: packages/ui/primitives/document-flow/add-signers.tsx:541
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:658
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:658
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:245
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:498
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:251
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:504
|
||||||
msgid "Name"
|
msgid "Name"
|
||||||
msgstr "Name"
|
msgstr "Name"
|
||||||
|
|
||||||
@@ -397,12 +410,12 @@ msgstr "Muss unterzeichnen"
|
|||||||
msgid "Needs to view"
|
msgid "Needs to view"
|
||||||
msgstr "Muss sehen"
|
msgstr "Muss sehen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:613
|
#: packages/ui/primitives/document-flow/add-fields.tsx:674
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:497
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:497
|
||||||
msgid "No recipient matching this description was found."
|
msgid "No recipient matching this description was found."
|
||||||
msgstr "Kein passender Empfänger mit dieser Beschreibung gefunden."
|
msgstr "Kein passender Empfänger mit dieser Beschreibung gefunden."
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:629
|
#: packages/ui/primitives/document-flow/add-fields.tsx:690
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:513
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:513
|
||||||
msgid "No recipients with this role"
|
msgid "No recipients with this role"
|
||||||
msgstr "Keine Empfänger mit dieser Rolle"
|
msgstr "Keine Empfänger mit dieser Rolle"
|
||||||
@@ -427,7 +440,7 @@ msgstr "Kein Unterschriftsfeld gefunden"
|
|||||||
msgid "No value found."
|
msgid "No value found."
|
||||||
msgstr "Kein Wert gefunden."
|
msgstr "Kein Wert gefunden."
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:880
|
#: packages/ui/primitives/document-flow/add-fields.tsx:941
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:736
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:736
|
||||||
msgid "Number"
|
msgid "Number"
|
||||||
msgstr "Nummer"
|
msgstr "Nummer"
|
||||||
@@ -462,7 +475,7 @@ msgstr "Wählen Sie eine Zahl"
|
|||||||
msgid "Placeholder"
|
msgid "Placeholder"
|
||||||
msgstr "Platzhalter"
|
msgstr "Platzhalter"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:906
|
#: packages/ui/primitives/document-flow/add-fields.tsx:967
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:762
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:762
|
||||||
msgid "Radio"
|
msgid "Radio"
|
||||||
msgstr "Radio"
|
msgstr "Radio"
|
||||||
@@ -484,21 +497,21 @@ msgid "Receives copy"
|
|||||||
msgstr "Erhält Kopie"
|
msgstr "Erhält Kopie"
|
||||||
|
|
||||||
#: packages/ui/components/recipient/recipient-action-auth-select.tsx:39
|
#: packages/ui/components/recipient/recipient-action-auth-select.tsx:39
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:184
|
#: packages/ui/primitives/document-flow/add-settings.tsx:215
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:169
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:169
|
||||||
msgid "Recipient action authentication"
|
msgid "Recipient action authentication"
|
||||||
msgstr "Empfängeraktion Authentifizierung"
|
msgstr "Empfängeraktion Authentifizierung"
|
||||||
|
|
||||||
#: packages/ui/primitives/signature-pad/signature-pad.tsx:283
|
#: packages/ui/primitives/signature-pad/signature-pad.tsx:283
|
||||||
msgid "Red"
|
msgid "Red"
|
||||||
msgstr ""
|
msgstr "Rot"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:298
|
#: packages/ui/primitives/document-flow/add-settings.tsx:329
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:332
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:332
|
||||||
msgid "Redirect URL"
|
msgid "Redirect URL"
|
||||||
msgstr "Weiterleitungs-URL"
|
msgstr "Weiterleitungs-URL"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:996
|
#: packages/ui/primitives/document-flow/add-fields.tsx:1069
|
||||||
msgid "Remove"
|
msgid "Remove"
|
||||||
msgstr "Entfernen"
|
msgstr "Entfernen"
|
||||||
|
|
||||||
@@ -542,7 +555,7 @@ msgstr "Wählen Sie mindestens"
|
|||||||
msgid "Select default option"
|
msgid "Select default option"
|
||||||
msgstr "Standardoption auswählen"
|
msgstr "Standardoption auswählen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-subject.tsx:275
|
#: packages/ui/primitives/document-flow/add-subject.tsx:124
|
||||||
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:34
|
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:34
|
||||||
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:64
|
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:64
|
||||||
msgid "Send"
|
msgid "Send"
|
||||||
@@ -560,8 +573,8 @@ msgstr "Unterschriftenkarte teilen"
|
|||||||
msgid "Share the Link"
|
msgid "Share the Link"
|
||||||
msgstr "Link teilen"
|
msgstr "Link teilen"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:377
|
#: packages/ui/primitives/document-flow/add-signers.tsx:680
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:387
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:655
|
||||||
msgid "Show advanced settings"
|
msgid "Show advanced settings"
|
||||||
msgstr "Erweiterte Einstellungen anzeigen"
|
msgstr "Erweiterte Einstellungen anzeigen"
|
||||||
|
|
||||||
@@ -569,7 +582,7 @@ msgstr "Erweiterte Einstellungen anzeigen"
|
|||||||
msgid "Sign"
|
msgid "Sign"
|
||||||
msgstr "Unterschreiben"
|
msgstr "Unterschreiben"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:724
|
#: packages/ui/primitives/document-flow/add-fields.tsx:785
|
||||||
#: packages/ui/primitives/document-flow/add-signature.tsx:323
|
#: packages/ui/primitives/document-flow/add-signature.tsx:323
|
||||||
#: packages/ui/primitives/document-flow/field-icon.tsx:52
|
#: packages/ui/primitives/document-flow/field-icon.tsx:52
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:580
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:580
|
||||||
@@ -604,7 +617,7 @@ msgstr "Etwas ist schief gelaufen."
|
|||||||
msgid "Step <0>{step} of {maxStep}</0>"
|
msgid "Step <0>{step} of {maxStep}</0>"
|
||||||
msgstr "Schritt <0>{step} von {maxStep}</0>"
|
msgstr "Schritt <0>{step} von {maxStep}</0>"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-subject.tsx:194
|
#: packages/ui/primitives/document-flow/add-subject.tsx:78
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:195
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:195
|
||||||
msgid "Subject <0>(Optional)</0>"
|
msgid "Subject <0>(Optional)</0>"
|
||||||
msgstr "Betreff <0>(Optional)</0>"
|
msgstr "Betreff <0>(Optional)</0>"
|
||||||
@@ -617,7 +630,7 @@ msgstr "Einreichen"
|
|||||||
msgid "Template title"
|
msgid "Template title"
|
||||||
msgstr "Vorlagentitel"
|
msgstr "Vorlagentitel"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:854
|
#: packages/ui/primitives/document-flow/add-fields.tsx:915
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:710
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:710
|
||||||
msgid "Text"
|
msgid "Text"
|
||||||
msgstr "Text"
|
msgstr "Text"
|
||||||
@@ -678,7 +691,7 @@ msgstr "Der Name des Unterzeichners"
|
|||||||
msgid "This can be overriden by setting the authentication requirements directly on each recipient in the next step."
|
msgid "This can be overriden by setting the authentication requirements directly on each recipient in the next step."
|
||||||
msgstr "Dies kann überschrieben werden, indem die Authentifizierungsanforderungen im nächsten Schritt direkt für jeden Empfänger festgelegt werden."
|
msgstr "Dies kann überschrieben werden, indem die Authentifizierungsanforderungen im nächsten Schritt direkt für jeden Empfänger festgelegt werden."
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:685
|
#: packages/ui/primitives/document-flow/add-fields.tsx:746
|
||||||
msgid "This document has already been sent to this recipient. You can no longer edit this recipient."
|
msgid "This document has already been sent to this recipient. You can no longer edit this recipient."
|
||||||
msgstr "Dieses Dokument wurde bereits an diesen Empfänger gesendet. Sie können diesen Empfänger nicht mehr bearbeiten."
|
msgstr "Dieses Dokument wurde bereits an diesen Empfänger gesendet. Sie können diesen Empfänger nicht mehr bearbeiten."
|
||||||
|
|
||||||
@@ -686,33 +699,41 @@ msgstr "Dieses Dokument wurde bereits an diesen Empfänger gesendet. Sie können
|
|||||||
msgid "This document is password protected. Please enter the password to view the document."
|
msgid "This document is password protected. Please enter the password to view the document."
|
||||||
msgstr "Dieses Dokument ist durch ein Passwort geschützt. Bitte geben Sie das Passwort ein, um das Dokument anzusehen."
|
msgstr "Dieses Dokument ist durch ein Passwort geschützt. Bitte geben Sie das Passwort ein, um das Dokument anzusehen."
|
||||||
|
|
||||||
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:315
|
#: packages/ui/primitives/template-flow/add-template-placeholder-recipients.tsx:573
|
||||||
msgid "This field cannot be modified or deleted. When you share this template's direct link or add it to your public profile, anyone who accesses it can input their name and email, and fill in the fields assigned to them."
|
msgid "This field cannot be modified or deleted. When you share this template's direct link or add it to your public profile, anyone who accesses it can input their name and email, and fill in the fields assigned to them."
|
||||||
msgstr "Dieses Feld kann nicht geändert oder gelöscht werden. Wenn Sie den direkten Link dieser Vorlage teilen oder zu Ihrem öffentlichen Profil hinzufügen, kann jeder, der darauf zugreift, seinen Namen und seine E-Mail-Adresse eingeben und die ihm zugewiesenen Felder ausfüllen."
|
msgstr "Dieses Feld kann nicht geändert oder gelöscht werden. Wenn Sie den direkten Link dieser Vorlage teilen oder zu Ihrem öffentlichen Profil hinzufügen, kann jeder, der darauf zugreift, seinen Namen und seine E-Mail-Adresse eingeben und die ihm zugewiesenen Felder ausfüllen."
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-signers.tsx:165
|
#: packages/ui/primitives/document-flow/add-fields.tsx:1050
|
||||||
msgid "This signer has already received the document."
|
msgid "This recipient can no longer be modified as they have signed a field, or completed the document."
|
||||||
msgstr "Dieser Unterzeichner hat das Dokument bereits erhalten."
|
msgstr ""
|
||||||
|
|
||||||
|
#: packages/ui/primitives/document-flow/add-signers.tsx:195
|
||||||
|
#~ msgid "This signer has already received the document."
|
||||||
|
#~ msgstr "Dieser Unterzeichner hat das Dokument bereits erhalten."
|
||||||
|
|
||||||
|
#: packages/ui/primitives/document-flow/add-signers.tsx:194
|
||||||
|
msgid "This signer has already signed the document."
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
#: packages/ui/components/recipient/recipient-action-auth-select.tsx:48
|
#: packages/ui/components/recipient/recipient-action-auth-select.tsx:48
|
||||||
msgid "This will override any global settings."
|
msgid "This will override any global settings."
|
||||||
msgstr "Dies überschreibt alle globalen Einstellungen."
|
msgstr "Dies überschreibt alle globalen Einstellungen."
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:274
|
#: packages/ui/primitives/document-flow/add-settings.tsx:305
|
||||||
#: packages/ui/primitives/template-flow/add-template-settings.tsx:309
|
#: packages/ui/primitives/template-flow/add-template-settings.tsx:309
|
||||||
msgid "Time Zone"
|
msgid "Time Zone"
|
||||||
msgstr "Zeitzone"
|
msgstr "Zeitzone"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-settings.tsx:145
|
#: packages/ui/primitives/document-flow/add-settings.tsx:153
|
||||||
msgid "Title"
|
msgid "Title"
|
||||||
msgstr "Titel"
|
msgstr "Titel"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-fields.tsx:971
|
#: packages/ui/primitives/document-flow/add-fields.tsx:1033
|
||||||
#: packages/ui/primitives/template-flow/add-template-fields.tsx:828
|
#: packages/ui/primitives/template-flow/add-template-fields.tsx:828
|
||||||
msgid "To proceed further, please set at least one value for the {0} field."
|
msgid "To proceed further, please set at least one value for the {0} field."
|
||||||
msgstr "Um fortzufahren, legen Sie bitte mindestens einen Wert für das Feld {0} fest."
|
msgstr "Um fortzufahren, legen Sie bitte mindestens einen Wert für das Feld {0} fest."
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/add-subject.tsx:275
|
#: packages/ui/primitives/document-flow/add-subject.tsx:124
|
||||||
msgid "Update"
|
msgid "Update"
|
||||||
msgstr "Aktualisieren"
|
msgstr "Aktualisieren"
|
||||||
|
|
||||||
@@ -756,7 +777,7 @@ msgstr "Viewing"
|
|||||||
|
|
||||||
#: packages/ui/primitives/signature-pad/signature-pad.tsx:280
|
#: packages/ui/primitives/signature-pad/signature-pad.tsx:280
|
||||||
#~ msgid "White"
|
#~ msgid "White"
|
||||||
#~ msgstr ""
|
#~ msgstr "White"
|
||||||
|
|
||||||
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:44
|
#: packages/ui/primitives/document-flow/send-document-action-dialog.tsx:44
|
||||||
msgid "You are about to send this document to the recipients. Are you sure you want to continue?"
|
msgid "You are about to send this document to the recipients. Are you sure you want to continue?"
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
@@ -1,14 +1,14 @@
|
|||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"POT-Creation-Date: 2024-08-22 05:16+0000\n"
|
"POT-Creation-Date: 2024-07-24 13:01+1000\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=utf-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"X-Generator: @lingui/cli\n"
|
"X-Generator: @lingui/cli\n"
|
||||||
"Language: de\n"
|
"Language: de\n"
|
||||||
"Project-Id-Version: documenso-app\n"
|
"Project-Id-Version: documenso-app\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"PO-Revision-Date: 2024-09-05 06:04\n"
|
"PO-Revision-Date: 2024-09-16 14:04\n"
|
||||||
"Last-Translator: \n"
|
"Last-Translator: \n"
|
||||||
"Language-Team: German\n"
|
"Language-Team: German\n"
|
||||||
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
||||||
@@ -20,19 +20,19 @@ msgstr ""
|
|||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/blog/page.tsx:45
|
#: apps/marketing/src/app/(marketing)/blog/page.tsx:45
|
||||||
msgid "{0}"
|
msgid "{0}"
|
||||||
msgstr ""
|
msgstr "{0}"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:100
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:100
|
||||||
msgid "5 standard documents per month"
|
msgid "5 standard documents per month"
|
||||||
msgstr ""
|
msgstr "5 Standarddokumente pro Monat"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:198
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:198
|
||||||
msgid "5 Users Included"
|
msgid "5 Users Included"
|
||||||
msgstr ""
|
msgstr "5 Benutzer inbegriffen"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:34
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:34
|
||||||
msgid "A 10x better signing experience."
|
msgid "A 10x better signing experience."
|
||||||
msgstr ""
|
msgstr "Eine 10x bessere Signaturerfahrung."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:51
|
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:51
|
||||||
msgid "Add document"
|
msgid "Add document"
|
||||||
@@ -40,138 +40,142 @@ msgstr "Dokument hinzufügen"
|
|||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:201
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:201
|
||||||
msgid "Add More Users for {0}"
|
msgid "Add More Users for {0}"
|
||||||
msgstr ""
|
msgstr "Mehr Benutzer hinzufügen für {0}"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:165
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:165
|
||||||
msgid "All our metrics, finances, and learnings are public. We believe in transparency and want to share our journey with you. You can read more about why here: <0>Announcing Open Metrics</0>"
|
msgid "All our metrics, finances, and learnings are public. We believe in transparency and want to share our journey with you. You can read more about why here: <0>Announcing Open Metrics</0>"
|
||||||
msgstr ""
|
msgstr "Alle unsere Kennzahlen, Finanzen und Erkenntnisse sind öffentlich. Wir glauben an Transparenz und möchten unsere Reise mit Ihnen teilen. Mehr erfahren Sie hier: <0>Ankündigung Offene Kennzahlen</0>"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/funding-raised.tsx:58
|
#: apps/marketing/src/app/(marketing)/open/funding-raised.tsx:58
|
||||||
#: apps/marketing/src/app/(marketing)/open/funding-raised.tsx:65
|
#: apps/marketing/src/app/(marketing)/open/funding-raised.tsx:65
|
||||||
msgid "Amount Raised"
|
msgid "Amount Raised"
|
||||||
msgstr ""
|
msgstr "Erhobener Betrag"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:145
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:145
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:189
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:189
|
||||||
msgid "API Access"
|
msgid "API Access"
|
||||||
msgstr ""
|
msgstr "API-Zugriff"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:67
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:67
|
||||||
msgid "Beautiful."
|
msgid "Beautiful."
|
||||||
msgstr ""
|
msgstr "Schön."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:69
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:69
|
||||||
msgid "Because signing should be celebrated. That’s why we care about the smallest detail in our product."
|
msgid "Because signing should be celebrated. That’s why we care about the smallest detail in our product."
|
||||||
msgstr ""
|
msgstr "Weil Unterschriften gefeiert werden sollten. Deshalb kümmern wir uns um jedes kleinste Detail in unserem Produkt."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:35
|
#: apps/marketing/src/components/(marketing)/footer.tsx:35
|
||||||
#: apps/marketing/src/components/(marketing)/header.tsx:57
|
#: apps/marketing/src/components/(marketing)/header.tsx:57
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:36
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:36
|
||||||
msgid "Blog"
|
msgid "Blog"
|
||||||
msgstr ""
|
msgstr "Blog"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:64
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:64
|
||||||
msgid "Build on top."
|
msgid "Build on top."
|
||||||
msgstr ""
|
msgstr "Aufbauen oben drauf."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:163
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:163
|
||||||
msgid "Can I use Documenso commercially?"
|
msgid "Can I use Documenso commercially?"
|
||||||
msgstr ""
|
msgstr "Kann ich Documenso kommerziell nutzen?"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:42
|
#: apps/marketing/src/components/(marketing)/footer.tsx:42
|
||||||
msgid "Careers"
|
msgid "Careers"
|
||||||
msgstr ""
|
msgstr "Karrieren"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:36
|
#: apps/marketing/src/components/(marketing)/footer.tsx:36
|
||||||
msgid "Changelog"
|
msgid "Changelog"
|
||||||
msgstr ""
|
msgstr "Änderungsprotokoll"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:85
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:85
|
||||||
msgid "Choose a template from the community app store. Or submit your own template for others to use."
|
msgid "Choose a template from the community app store. Or submit your own template for others to use."
|
||||||
msgstr ""
|
msgstr "Wählen Sie eine Vorlage aus dem Community-App-Store. Oder reichen Sie Ihre eigene Vorlage ein, damit andere sie benutzen können."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:219
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:219
|
||||||
msgid "Community"
|
msgid "Community"
|
||||||
msgstr ""
|
msgstr "Gemeinschaft"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-completed-documents-chart.tsx:55
|
#: apps/marketing/src/app/(marketing)/open/monthly-completed-documents-chart.tsx:55
|
||||||
msgid "Completed Documents"
|
msgid "Completed Documents"
|
||||||
msgstr ""
|
msgstr "Fertige Dokumente"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-completed-documents-chart.tsx:33
|
#: apps/marketing/src/app/(marketing)/open/monthly-completed-documents-chart.tsx:33
|
||||||
msgid "Completed Documents per Month"
|
msgid "Completed Documents per Month"
|
||||||
msgstr ""
|
msgstr "Fertige Dokumente pro Monat"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:65
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:65
|
||||||
msgid "Connections"
|
msgid "Connections"
|
||||||
msgstr ""
|
msgstr "Verbindungen"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/enterprise.tsx:35
|
#: apps/marketing/src/components/(marketing)/enterprise.tsx:35
|
||||||
msgid "Contact Us"
|
msgid "Contact Us"
|
||||||
msgstr ""
|
msgstr "Kontaktiere uns"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:67
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:67
|
||||||
msgid "Create connections and automations with Zapier and more to integrate with your favorite tools."
|
msgid "Create connections and automations with Zapier and more to integrate with your favorite tools."
|
||||||
msgstr ""
|
msgstr "Erstellen Sie Verbindungen und Automatisierungen mit Zapier und mehr, um sich mit Ihren Lieblingstools zu integrieren."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/call-to-action.tsx:23
|
#: apps/marketing/src/components/(marketing)/call-to-action.tsx:23
|
||||||
msgid "Create your account and start using state-of-the-art document signing. Open and beautiful signing is within your grasp."
|
msgid "Create your account and start using state-of-the-art document signing. Open and beautiful signing is within your grasp."
|
||||||
msgstr ""
|
msgstr "Erstellen Sie Ihr Konto und beginnen Sie mit der Nutzung modernster Dokumentensignaturen. Offene und schöne Signaturen sind zum Greifen nah."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/tooltip.tsx:35
|
#: apps/marketing/src/app/(marketing)/open/tooltip.tsx:35
|
||||||
msgid "Customers with an Active Subscriptions."
|
msgid "Customers with an Active Subscriptions."
|
||||||
msgstr ""
|
msgstr "Kunden mit einer aktiven Abonnements."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:33
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:33
|
||||||
msgid "Customise and expand."
|
msgid "Customise and expand."
|
||||||
msgstr ""
|
msgstr "Anpassen und erweitern."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:38
|
#: apps/marketing/src/components/(marketing)/footer.tsx:38
|
||||||
msgid "Design"
|
msgid "Design"
|
||||||
msgstr ""
|
msgstr "Design"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:44
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:44
|
||||||
msgid "Designed for every stage of your journey."
|
msgid "Designed for every stage of your journey."
|
||||||
msgstr ""
|
msgstr "Entwickelt für jede Phase Ihrer Reise."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/carousel.tsx:40
|
#: apps/marketing/src/components/(marketing)/carousel.tsx:40
|
||||||
msgid "Direct Link"
|
msgid "Direct Link"
|
||||||
msgstr ""
|
msgstr "Direktlink"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:181
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:181
|
||||||
msgid "Documenso is a community effort to create an open and vibrant ecosystem around a tool, everybody is free to use and adapt. By being truly open we want to create trusted infrastructure for the future of the internet."
|
msgid "Documenso is a community effort to create an open and vibrant ecosystem around a tool, everybody is free to use and adapt. By being truly open we want to create trusted infrastructure for the future of the internet."
|
||||||
msgstr ""
|
msgstr "Documenso ist eine Gemeinschaftsanstrengung, um ein offenes und lebendiges Ökosystem um ein Werkzeug zu schaffen, das jeder frei nutzen und anpassen kann. Indem wir wirklich offen sind, wollen wir vertrauenswürdige Infrastruktur für die Zukunft des Internets schaffen."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:28
|
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:28
|
||||||
msgid "Documenso on X"
|
msgid "Documenso on X"
|
||||||
msgstr ""
|
msgstr "Documenso auf X"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/hero.tsx:104
|
#: apps/marketing/src/components/(marketing)/hero.tsx:104
|
||||||
msgid "Document signing,<0/>finally open source."
|
msgid "Document signing,<0/>finally open source."
|
||||||
msgstr ""
|
msgstr "Unterschriften,<0/>endlich Open Source."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:33
|
#: apps/marketing/src/components/(marketing)/footer.tsx:33
|
||||||
#: apps/marketing/src/components/(marketing)/header.tsx:50
|
#: apps/marketing/src/components/(marketing)/header.tsx:50
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:28
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:28
|
||||||
msgid "Documentation"
|
msgid "Documentation"
|
||||||
msgstr ""
|
msgstr "Dokumentation"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:110
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:110
|
||||||
msgid "Easily embed Documenso into your product. Simply copy and paste our react widget into your application."
|
msgid "Easily embed Documenso into your product. Simply copy and paste our react widget into your application."
|
||||||
msgstr ""
|
msgstr "Betten Sie Documenso ganz einfach in Ihr Produkt ein. Kopieren und fügen Sie einfach unser React-Widget in Ihre Anwendung ein."
|
||||||
|
|
||||||
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:42
|
||||||
|
#~ msgid "Easy Sharing (Soon)."
|
||||||
|
#~ msgstr "Easy Sharing (Soon)."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:46
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:46
|
||||||
msgid "Easy Sharing."
|
msgid "Easy Sharing."
|
||||||
msgstr ""
|
msgstr "Einfaches Teilen."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:148
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:148
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:192
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:192
|
||||||
msgid "Email and Discord Support"
|
msgid "Email and Discord Support"
|
||||||
msgstr ""
|
msgstr "E-Mail- und Discord-Support"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:43
|
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:43
|
||||||
msgid "Engagement"
|
msgid "Engagement"
|
||||||
msgstr ""
|
msgstr "Beteiligung"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:64
|
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:64
|
||||||
msgid "Enter your details."
|
msgid "Enter your details."
|
||||||
@@ -179,266 +183,272 @@ msgstr "Geben Sie Ihre Details ein."
|
|||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/enterprise.tsx:16
|
#: apps/marketing/src/components/(marketing)/enterprise.tsx:16
|
||||||
msgid "Enterprise Compliance, License or Technical Needs?"
|
msgid "Enterprise Compliance, License or Technical Needs?"
|
||||||
msgstr ""
|
msgstr "Enterprise-Konformität, Lizenz- oder technische Bedürfnisse?"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:128
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:128
|
||||||
msgid "Everything you need for a great signing experience."
|
msgid "Everything you need for a great signing experience."
|
||||||
msgstr ""
|
msgstr "Alles, was Sie für ein großartiges Signaturerlebnis benötigen."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:45
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:45
|
||||||
msgid "Fast."
|
msgid "Fast."
|
||||||
msgstr ""
|
msgstr "Schnell."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:36
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:36
|
||||||
msgid "Faster, smarter and more beautiful."
|
msgid "Faster, smarter and more beautiful."
|
||||||
msgstr ""
|
msgstr "Schneller, intelligenter und schöner."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:210
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:210
|
||||||
msgid "Finances"
|
msgid "Finances"
|
||||||
msgstr ""
|
msgstr "Finanzen"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:38
|
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:38
|
||||||
msgid "Follow us on X"
|
msgid "Follow us on X"
|
||||||
msgstr ""
|
msgstr "Folgen Sie uns auf X"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:172
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:172
|
||||||
msgid "For companies looking to scale across multiple teams."
|
msgid "For companies looking to scale across multiple teams."
|
||||||
msgstr ""
|
msgstr "Für Unternehmen, die über mehrere Teams skalieren möchten."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:85
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:85
|
||||||
msgid "For small teams and individuals with basic needs."
|
msgid "For small teams and individuals with basic needs."
|
||||||
msgstr ""
|
msgstr "Für kleine Teams und Einzelpersonen mit grundlegenden Bedürfnissen."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:80
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:80
|
||||||
msgid "Free"
|
msgid "Free"
|
||||||
msgstr ""
|
msgstr "Kostenlos"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/blog/page.tsx:26
|
#: apps/marketing/src/app/(marketing)/blog/page.tsx:26
|
||||||
msgid "From the blog"
|
msgid "From the blog"
|
||||||
msgstr ""
|
msgstr "Aus dem Blog"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/data.ts:9
|
#: apps/marketing/src/app/(marketing)/open/data.ts:9
|
||||||
#: apps/marketing/src/app/(marketing)/open/data.ts:17
|
#: apps/marketing/src/app/(marketing)/open/data.ts:17
|
||||||
|
#: apps/marketing/src/app/(marketing)/open/data.ts:25
|
||||||
#: apps/marketing/src/app/(marketing)/open/data.ts:33
|
#: apps/marketing/src/app/(marketing)/open/data.ts:33
|
||||||
#: apps/marketing/src/app/(marketing)/open/data.ts:41
|
#: apps/marketing/src/app/(marketing)/open/data.ts:41
|
||||||
#: apps/marketing/src/app/(marketing)/open/data.ts:49
|
#: apps/marketing/src/app/(marketing)/open/data.ts:49
|
||||||
msgid "Full-Time"
|
msgid "Full-Time"
|
||||||
msgstr ""
|
msgstr "Vollzeit"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:87
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:87
|
||||||
msgid "Get paid (Soon)."
|
msgid "Get paid (Soon)."
|
||||||
msgstr ""
|
msgstr "Lassen Sie sich bezahlen (Bald)."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/call-to-action.tsx:31
|
#: apps/marketing/src/components/(marketing)/call-to-action.tsx:31
|
||||||
msgid "Get started"
|
msgid "Get started"
|
||||||
msgstr ""
|
msgstr "Loslegen"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:75
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:75
|
||||||
msgid "Get Started"
|
msgid "Get Started"
|
||||||
msgstr ""
|
msgstr "Loslegen"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:47
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:47
|
||||||
msgid "Get started today."
|
msgid "Get started today."
|
||||||
msgstr ""
|
msgstr "Fangen Sie heute an."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/blog/page.tsx:30
|
#: apps/marketing/src/app/(marketing)/blog/page.tsx:30
|
||||||
msgid "Get the latest news from Documenso, including product updates, team announcements and more!"
|
msgid "Get the latest news from Documenso, including product updates, team announcements and more!"
|
||||||
msgstr ""
|
msgstr "Erhalten Sie die neuesten Nachrichten von Documenso, einschließlich Produkt-Updates, Team-Ankündigungen und mehr!"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:233
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:233
|
||||||
msgid "GitHub: Total Merged PRs"
|
msgid "GitHub: Total Merged PRs"
|
||||||
msgstr ""
|
msgstr "GitHub: Gesamte PRs zusammengeführt"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:251
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:251
|
||||||
msgid "GitHub: Total Open Issues"
|
msgid "GitHub: Total Open Issues"
|
||||||
msgstr ""
|
msgstr "GitHub: Gesamte offene Issues"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:225
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:225
|
||||||
msgid "GitHub: Total Stars"
|
msgid "GitHub: Total Stars"
|
||||||
msgstr ""
|
msgstr "GitHub: Gesamtanzahl Sterne"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:23
|
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:23
|
||||||
msgid "Global Salary Bands"
|
msgid "Global Salary Bands"
|
||||||
msgstr ""
|
msgstr "Globale Gehaltsbänder"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:261
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:261
|
||||||
msgid "Growth"
|
msgid "Growth"
|
||||||
msgstr ""
|
msgstr "Wachstum"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:134
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:134
|
||||||
msgid "How can I contribute?"
|
msgid "How can I contribute?"
|
||||||
msgstr ""
|
msgstr "Wie kann ich beitragen?"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:105
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:105
|
||||||
msgid "How do you handle my data?"
|
msgid "How do you handle my data?"
|
||||||
msgstr ""
|
msgstr "Wie gehen Sie mit meinen Daten um?"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:118
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:118
|
||||||
msgid "Individual"
|
msgid "Individual"
|
||||||
msgstr ""
|
msgstr "Einzelperson"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:89
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:89
|
||||||
msgid "Integrated payments with Stripe so you don’t have to worry about getting paid."
|
msgid "Integrated payments with Stripe so you don’t have to worry about getting paid."
|
||||||
msgstr ""
|
msgstr "Integrierte Zahlungen mit Stripe, sodass Sie sich keine Sorgen ums Bezahlen machen müssen."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:35
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:35
|
||||||
msgid "Integrates with all your favourite tools."
|
msgid "Integrates with all your favourite tools."
|
||||||
msgstr ""
|
msgstr "Integriert sich mit all Ihren Lieblingstools."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:289
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:289
|
||||||
msgid "Is there more?"
|
msgid "Is there more?"
|
||||||
msgstr ""
|
msgstr "Gibt es mehr?"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:44
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:44
|
||||||
msgid "It’s up to you. Either clone our repository or rely on our easy to use hosting solution."
|
msgid "It’s up to you. Either clone our repository or rely on our easy to use hosting solution."
|
||||||
msgstr ""
|
msgstr "Es liegt an Ihnen. Entweder klonen Sie unser Repository oder nutzen unsere einfach zu bedienende Hosting-Lösung."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:49
|
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:49
|
||||||
msgid "Join Date"
|
msgid "Join Date"
|
||||||
msgstr ""
|
msgstr "Eintrittsdatum"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/call-to-action.tsx:19
|
#: apps/marketing/src/components/(marketing)/call-to-action.tsx:19
|
||||||
msgid "Join the Open Signing Movement"
|
msgid "Join the Open Signing Movement"
|
||||||
msgstr ""
|
msgstr "Treten Sie der Open Signing-Bewegung bei"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:46
|
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:46
|
||||||
msgid "Location"
|
msgid "Location"
|
||||||
msgstr ""
|
msgstr "Standort"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:66
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:66
|
||||||
msgid "Make it your own through advanced customization and adjustability."
|
msgid "Make it your own through advanced customization and adjustability."
|
||||||
msgstr ""
|
msgstr "Machen Sie es zu Ihrem eigenen durch erweiterte Anpassung und Einstellbarkeit."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:199
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:199
|
||||||
msgid "Merged PR's"
|
msgid "Merged PR's"
|
||||||
msgstr ""
|
msgstr "Zusammengeführte PRs"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:234
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:234
|
||||||
msgid "Merged PRs"
|
msgid "Merged PRs"
|
||||||
msgstr ""
|
msgstr "Zusammengeführte PRs"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:40
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:40
|
||||||
msgid "Monthly"
|
msgid "Monthly"
|
||||||
msgstr ""
|
msgstr "Monatlich"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:34
|
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:34
|
||||||
msgid "Name"
|
msgid "Name"
|
||||||
msgstr ""
|
msgstr "Name"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-new-users-chart.tsx:30
|
#: apps/marketing/src/app/(marketing)/open/monthly-new-users-chart.tsx:30
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-new-users-chart.tsx:43
|
#: apps/marketing/src/app/(marketing)/open/monthly-new-users-chart.tsx:43
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-new-users-chart.tsx:52
|
#: apps/marketing/src/app/(marketing)/open/monthly-new-users-chart.tsx:52
|
||||||
msgid "New Users"
|
msgid "New Users"
|
||||||
msgstr ""
|
msgstr "Neue Benutzer"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:106
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:106
|
||||||
msgid "No credit card required"
|
msgid "No credit card required"
|
||||||
msgstr ""
|
msgstr "Keine Kreditkarte erforderlich"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/callout.tsx:29
|
#: apps/marketing/src/components/(marketing)/callout.tsx:29
|
||||||
#: apps/marketing/src/components/(marketing)/hero.tsx:125
|
#: apps/marketing/src/components/(marketing)/hero.tsx:125
|
||||||
msgid "No Credit Card required"
|
msgid "No Credit Card required"
|
||||||
msgstr ""
|
msgstr "Keine Kreditkarte erforderlich"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:61
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:61
|
||||||
msgid "None of these work for you? Try self-hosting!"
|
msgid "None of these work for you? Try self-hosting!"
|
||||||
msgstr ""
|
msgstr "Keines dieser Angebote passt zu Ihnen? Versuchen Sie das Selbst-Hosting!"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:194
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:194
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:252
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:252
|
||||||
msgid "Open Issues"
|
msgid "Open Issues"
|
||||||
msgstr ""
|
msgstr "Offene Issues"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:42
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:42
|
||||||
msgid "Open Source or Hosted."
|
msgid "Open Source or Hosted."
|
||||||
msgstr ""
|
msgstr "Open Source oder Hosted."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:161
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:161
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:37
|
#: apps/marketing/src/components/(marketing)/footer.tsx:37
|
||||||
#: apps/marketing/src/components/(marketing)/header.tsx:64
|
#: apps/marketing/src/components/(marketing)/header.tsx:64
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:40
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:40
|
||||||
msgid "Open Startup"
|
msgid "Open Startup"
|
||||||
msgstr ""
|
msgstr "Offenes Startup"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:41
|
#: apps/marketing/src/components/(marketing)/footer.tsx:41
|
||||||
msgid "OSS Friends"
|
msgid "OSS Friends"
|
||||||
msgstr ""
|
msgstr "OSS-Freunde"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:91
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:91
|
||||||
msgid "Our custom templates come with smart rules that can help you save time and energy."
|
msgid "Our custom templates come with smart rules that can help you save time and energy."
|
||||||
msgstr ""
|
msgstr "Unsere benutzerdefinierten Vorlagen verfügen über intelligente Regeln, die Ihnen Zeit und Energie sparen können."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/enterprise.tsx:20
|
#: apps/marketing/src/components/(marketing)/enterprise.tsx:20
|
||||||
msgid "Our Enterprise License is great for large organizations looking to switch to Documenso for all their signing needs. It's available for our cloud offering as well as self-hosted setups and offers a wide range of compliance and Adminstration Features."
|
msgid "Our Enterprise License is great for large organizations looking to switch to Documenso for all their signing needs. It's available for our cloud offering as well as self-hosted setups and offers a wide range of compliance and Adminstration Features."
|
||||||
msgstr ""
|
msgstr "Unsere Enterprise-Lizenz ist ideal für große Organisationen, die auf Documenso für all ihre Signaturanforderungen umsteigen möchten. Sie ist sowohl für unser Cloud-Angebot als auch für selbstgehostete Setups verfügbar und bietet eine breite Palette an Compliance- und Verwaltungsfunktionen."
|
||||||
|
|
||||||
|
#: apps/marketing/src/components/(marketing)/enterprise.tsx:20
|
||||||
|
#~ msgid "Our Enterprise License is great large organizations looking to switch to Documenso for all their signing needs. It's availible for our cloud offering as well as self-hosted setups and offer a wide range of compliance and Adminstration Features."
|
||||||
|
#~ msgstr "Our Enterprise License is great large organizations looking to switch to Documenso for all their signing needs. It's availible for our cloud offering as well as self-hosted setups and offer a wide range of compliance and Adminstration Features."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:65
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:65
|
||||||
msgid "Our self-hosted option is great for small teams and individuals who need a simple solution. You can use our docker based setup to get started in minutes. Take control with full customizability and data ownership."
|
msgid "Our self-hosted option is great for small teams and individuals who need a simple solution. You can use our docker based setup to get started in minutes. Take control with full customizability and data ownership."
|
||||||
msgstr ""
|
msgstr "Unsere selbstgehostete Option ist ideal für kleine Teams und Einzelpersonen, die eine einfache Lösung benötigen. Sie können unser docker-basiertes Setup verwenden, um in wenigen Minuten loszulegen. Übernehmen Sie die Kontrolle mit vollständiger Anpassbarkeit und Datenhoheit."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/data.ts:25
|
#: apps/marketing/src/app/(marketing)/open/data.ts:25
|
||||||
msgid "Part-Time"
|
#~ msgid "Part-Time"
|
||||||
msgstr ""
|
#~ msgstr "Teilzeit"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:151
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:151
|
||||||
msgid "Premium Profile Name"
|
msgid "Premium Profile Name"
|
||||||
msgstr ""
|
msgstr "Premium Profilname"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:40
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:40
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:31
|
#: apps/marketing/src/components/(marketing)/footer.tsx:31
|
||||||
#: apps/marketing/src/components/(marketing)/header.tsx:42
|
#: apps/marketing/src/components/(marketing)/header.tsx:42
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:24
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:24
|
||||||
msgid "Pricing"
|
msgid "Pricing"
|
||||||
msgstr ""
|
msgstr "Preise"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:43
|
#: apps/marketing/src/components/(marketing)/footer.tsx:43
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:53
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:53
|
||||||
msgid "Privacy"
|
msgid "Privacy"
|
||||||
msgstr ""
|
msgstr "Datenschutz"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/carousel.tsx:58
|
#: apps/marketing/src/components/(marketing)/carousel.tsx:58
|
||||||
msgid "Profile"
|
msgid "Profile"
|
||||||
msgstr ""
|
msgstr "Profil"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:108
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:108
|
||||||
msgid "React Widget (Soon)."
|
msgid "React Widget (Soon)."
|
||||||
msgstr ""
|
msgstr "React Widget (Demnächst)."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:48
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:48
|
||||||
msgid "Receive your personal link to share with everyone you care about."
|
msgid "Receive your personal link to share with everyone you care about."
|
||||||
msgstr ""
|
msgstr "Erhalten Sie Ihren persönlichen Link zum Teilen mit allen, die Ihnen wichtig sind."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:37
|
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:37
|
||||||
msgid "Role"
|
msgid "Role"
|
||||||
msgstr ""
|
msgstr "Rolle"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:37
|
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:37
|
||||||
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:40
|
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:40
|
||||||
msgid "Salary"
|
msgid "Salary"
|
||||||
msgstr ""
|
msgstr "Gehalt"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:62
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:62
|
||||||
msgid "Save $60 or $120"
|
msgid "Save $60 or $120"
|
||||||
msgstr ""
|
msgstr "Sparen Sie $60 oder $120"
|
||||||
|
|
||||||
|
#: apps/marketing/src/components/(marketing)/i18n-switcher.tsx:47
|
||||||
#~ msgid "Search languages..."
|
#~ msgid "Search languages..."
|
||||||
#~ msgstr "Sprachen suchen..."
|
#~ msgstr "Search languages..."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:109
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:109
|
||||||
msgid "Securely. Our data centers are located in Frankfurt (Germany), giving us the best local privacy laws. We are very aware of the sensitive nature of our data and follow best practices to ensure the security and integrity of the data entrusted to us."
|
msgid "Securely. Our data centers are located in Frankfurt (Germany), giving us the best local privacy laws. We are very aware of the sensitive nature of our data and follow best practices to ensure the security and integrity of the data entrusted to us."
|
||||||
msgstr ""
|
msgstr "Sicher. Unsere Rechenzentren befinden sich in Frankfurt (Deutschland) und bieten uns die besten lokalen Datenschutzgesetze. Uns ist die sensible Natur unserer Daten sehr bewusst und wir folgen bewährten Praktiken, um die Sicherheit und Integrität der uns anvertrauten Daten zu gewährleisten."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:37
|
#: apps/marketing/src/components/(marketing)/share-connect-paid-widget-bento.tsx:37
|
||||||
msgid "Send, connect, receive and embed everywhere."
|
msgid "Send, connect, receive and embed everywhere."
|
||||||
msgstr ""
|
msgstr "Überall senden, verbinden, empfangen und einbetten."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:34
|
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:34
|
||||||
msgid "Seniority"
|
msgid "Seniority"
|
||||||
msgstr ""
|
msgstr "Dienstalter"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:39
|
#: apps/marketing/src/components/(marketing)/footer.tsx:39
|
||||||
msgid "Shop"
|
msgid "Shop"
|
||||||
msgstr ""
|
msgstr "Shop"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:63
|
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:63
|
||||||
msgid "Sign"
|
msgid "Sign"
|
||||||
@@ -447,115 +457,115 @@ msgstr "Signieren"
|
|||||||
#: apps/marketing/src/components/(marketing)/header.tsx:72
|
#: apps/marketing/src/components/(marketing)/header.tsx:72
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:61
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:61
|
||||||
msgid "Sign in"
|
msgid "Sign in"
|
||||||
msgstr ""
|
msgstr "Anmelden"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/header.tsx:77
|
#: apps/marketing/src/components/(marketing)/header.tsx:77
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:57
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:57
|
||||||
msgid "Sign up"
|
msgid "Sign up"
|
||||||
msgstr ""
|
msgstr "Registrieren"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/carousel.tsx:22
|
#: apps/marketing/src/components/(marketing)/carousel.tsx:22
|
||||||
msgid "Signing Process"
|
msgid "Signing Process"
|
||||||
msgstr ""
|
msgstr "Signaturprozess"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:94
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:94
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:136
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:136
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:180
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:180
|
||||||
msgid "Signup Now"
|
msgid "Signup Now"
|
||||||
msgstr ""
|
msgstr "Jetzt registrieren"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:89
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:89
|
||||||
msgid "Smart."
|
msgid "Smart."
|
||||||
msgstr ""
|
msgstr "Intelligent."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/hero.tsx:132
|
#: apps/marketing/src/components/(marketing)/hero.tsx:132
|
||||||
msgid "Star on GitHub"
|
msgid "Star on GitHub"
|
||||||
msgstr ""
|
msgstr "Auf GitHub favorisieren"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:226
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:226
|
||||||
msgid "Stars"
|
msgid "Stars"
|
||||||
msgstr ""
|
msgstr "Favoriten"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:40
|
#: apps/marketing/src/components/(marketing)/footer.tsx:40
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:44
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:44
|
||||||
msgid "Status"
|
msgid "Status"
|
||||||
msgstr ""
|
msgstr "Status"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/footer.tsx:34
|
#: apps/marketing/src/components/(marketing)/footer.tsx:34
|
||||||
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:48
|
#: apps/marketing/src/components/(marketing)/mobile-navigation.tsx:48
|
||||||
msgid "Support"
|
msgid "Support"
|
||||||
msgstr ""
|
msgstr "Support"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:26
|
#: apps/marketing/src/app/(marketing)/open/team-members.tsx:26
|
||||||
msgid "Team"
|
msgid "Team"
|
||||||
msgstr ""
|
msgstr "Team"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:195
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:195
|
||||||
msgid "Team Inbox"
|
msgid "Team Inbox"
|
||||||
msgstr ""
|
msgstr "Team-Posteingang"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/carousel.tsx:28
|
#: apps/marketing/src/components/(marketing)/carousel.tsx:28
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:162
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:162
|
||||||
msgid "Teams"
|
msgid "Teams"
|
||||||
msgstr ""
|
msgstr "Teams"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:83
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:83
|
||||||
msgid "Template Store (Soon)."
|
msgid "Template Store (Soon)."
|
||||||
msgstr ""
|
msgstr "Vorlagen-Shop (Demnächst)."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:138
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:138
|
||||||
msgid "That's awesome. You can take a look at the current <0>Issues</0> and join our <1>Discord Community</1> to keep up to date, on what the current priorities are. In any case, we are an open community and welcome all input, technical and non-technical ❤️"
|
msgid "That's awesome. You can take a look at the current <0>Issues</0> and join our <1>Discord Community</1> to keep up to date, on what the current priorities are. In any case, we are an open community and welcome all input, technical and non-technical ❤️"
|
||||||
msgstr ""
|
msgstr "Das ist großartig. Sie können sich die aktuellen <0>Issues</0> ansehen und unserer <1>Discord-Community</1> beitreten, um auf dem neuesten Stand zu bleiben, was die aktuellen Prioritäten sind. In jedem Fall sind wir eine offene Gemeinschaft und begrüßen jegliche Beiträge, technische und nicht-technische ❤️"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:293
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:293
|
||||||
msgid "This page is evolving as we learn what makes a great signing company. We'll update it when we have more to share."
|
msgid "This page is evolving as we learn what makes a great signing company. We'll update it when we have more to share."
|
||||||
msgstr ""
|
msgstr "Diese Seite entwickelt sich weiter, während wir lernen, was ein großartiges Signing-Unternehmen ausmacht. Wir werden sie aktualisieren, wenn wir mehr zu teilen haben."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:31
|
#: apps/marketing/src/app/(marketing)/open/salary-bands.tsx:31
|
||||||
msgid "Title"
|
msgid "Title"
|
||||||
msgstr ""
|
msgstr "Titel"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/total-signed-documents-chart.tsx:30
|
#: apps/marketing/src/app/(marketing)/open/total-signed-documents-chart.tsx:30
|
||||||
#: apps/marketing/src/app/(marketing)/open/total-signed-documents-chart.tsx:55
|
#: apps/marketing/src/app/(marketing)/open/total-signed-documents-chart.tsx:55
|
||||||
msgid "Total Completed Documents"
|
msgid "Total Completed Documents"
|
||||||
msgstr ""
|
msgstr "Insgesamt Abgeschlossene Dokumente"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:267
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:267
|
||||||
#: apps/marketing/src/app/(marketing)/open/page.tsx:268
|
#: apps/marketing/src/app/(marketing)/open/page.tsx:268
|
||||||
msgid "Total Customers"
|
msgid "Total Customers"
|
||||||
msgstr ""
|
msgstr "Insgesamt Kunden"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/funding-raised.tsx:29
|
#: apps/marketing/src/app/(marketing)/open/funding-raised.tsx:29
|
||||||
msgid "Total Funding Raised"
|
msgid "Total Funding Raised"
|
||||||
msgstr ""
|
msgstr "Insgesamt Finanzierungsvolumen"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-total-users-chart.tsx:30
|
#: apps/marketing/src/app/(marketing)/open/monthly-total-users-chart.tsx:30
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-total-users-chart.tsx:43
|
#: apps/marketing/src/app/(marketing)/open/monthly-total-users-chart.tsx:43
|
||||||
#: apps/marketing/src/app/(marketing)/open/monthly-total-users-chart.tsx:52
|
#: apps/marketing/src/app/(marketing)/open/monthly-total-users-chart.tsx:52
|
||||||
msgid "Total Users"
|
msgid "Total Users"
|
||||||
msgstr ""
|
msgstr "Gesamtanzahl der Benutzer"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:31
|
#: apps/marketing/src/components/(marketing)/open-build-template-bento.tsx:31
|
||||||
msgid "Truly your own."
|
msgid "Truly your own."
|
||||||
msgstr ""
|
msgstr "Wirklich Ihr Eigenes."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/callout.tsx:27
|
#: apps/marketing/src/components/(marketing)/callout.tsx:27
|
||||||
#: apps/marketing/src/components/(marketing)/hero.tsx:123
|
#: apps/marketing/src/components/(marketing)/hero.tsx:123
|
||||||
msgid "Try our Free Plan"
|
msgid "Try our Free Plan"
|
||||||
msgstr ""
|
msgstr "Probieren Sie unseren Gratisplan aus"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:20
|
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:20
|
||||||
msgid "Twitter Stats"
|
msgid "Twitter Stats"
|
||||||
msgstr ""
|
msgstr "Twitter-Statistiken"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:142
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:142
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:186
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:186
|
||||||
msgid "Unlimited Documents per Month"
|
msgid "Unlimited Documents per Month"
|
||||||
msgstr ""
|
msgstr "Unbegrenzte Dokumente pro Monat"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:103
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:103
|
||||||
msgid "Up to 10 recipients per document"
|
msgid "Up to 10 recipients per document"
|
||||||
msgstr ""
|
msgstr "Bis zu 10 Empfänger pro Dokument"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:52
|
#: apps/marketing/src/app/(marketing)/singleplayer/client.tsx:52
|
||||||
msgid "Upload a document and add fields."
|
msgid "Upload a document and add fields."
|
||||||
@@ -563,52 +573,48 @@ msgstr "Laden Sie ein Dokument hoch und fügen Sie Felder hinzu."
|
|||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:123
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:123
|
||||||
msgid "Using our hosted version is the easiest way to get started, you can simply subscribe and start signing your documents. We take care of the infrastructure, so you can focus on your business. Additionally, when using our hosted version you benefit from our trusted signing certificates which helps you to build trust with your customers."
|
msgid "Using our hosted version is the easiest way to get started, you can simply subscribe and start signing your documents. We take care of the infrastructure, so you can focus on your business. Additionally, when using our hosted version you benefit from our trusted signing certificates which helps you to build trust with your customers."
|
||||||
msgstr ""
|
msgstr "Die Nutzung unserer gehosteten Version ist der einfachste Weg, um zu starten. Sie können einfach abonnieren und mit der Unterzeichnung Ihrer Dokumente beginnen. Wir kümmern uns um die Infrastruktur, damit Sie sich auf Ihr Geschäft konzentrieren können. Zudem profitieren Sie bei der Nutzung unserer gehosteten Version von unseren vertrauenswürdigen Signaturzertifikaten, die Ihnen helfen, Vertrauen bei Ihren Kunden aufzubauen."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:33
|
#: apps/marketing/src/app/(marketing)/open/typefully.tsx:33
|
||||||
msgid "View all stats"
|
msgid "View all stats"
|
||||||
msgstr ""
|
msgstr "Alle Statistiken anzeigen"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:195
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:195
|
||||||
msgid "We are happy to assist you at <0>support@documenso.com</0> or <1>in our Discord-Support-Channel</1> please message either Lucas or Timur to get added to the channel if you are not already a member."
|
msgid "We are happy to assist you at <0>support@documenso.com</0> or <1>in our Discord-Support-Channel</1> please message either Lucas or Timur to get added to the channel if you are not already a member."
|
||||||
msgstr ""
|
msgstr "Wir helfen Ihnen gerne unter <0>support@documenso.com</0> oder <1>in unserem Discord-Support-Kanal</1>. Bitte senden Sie Lucas oder Timur eine Nachricht, um dem Kanal beizutreten, falls Sie noch kein Mitglied sind."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:89
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:89
|
||||||
msgid "What is the difference between the plans?"
|
msgid "What is the difference between the plans?"
|
||||||
msgstr ""
|
msgstr "Was ist der Unterschied zwischen den Plänen?"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:47
|
#: apps/marketing/src/components/(marketing)/faster-smarter-beautiful-bento.tsx:47
|
||||||
msgid "When it comes to sending or receiving a contract, you can count on lightning-fast speeds."
|
msgid "When it comes to sending or receiving a contract, you can count on lightning-fast speeds."
|
||||||
msgstr ""
|
msgstr "Wenn es um das Senden oder Empfangen eines Vertrags geht, können Sie auf blitzschnelle Geschwindigkeiten zählen."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:191
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:191
|
||||||
msgid "Where can I get support?"
|
msgid "Where can I get support?"
|
||||||
msgstr ""
|
msgstr "Wo kann ich Unterstützung bekommen?"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:177
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:177
|
||||||
msgid "Why should I prefer Documenso over DocuSign or some other signing tool?"
|
msgid "Why should I prefer Documenso over DocuSign or some other signing tool?"
|
||||||
msgstr ""
|
msgstr "Warum sollte ich Documenso gegenüber DocuSign oder einem anderen Signatur-Tool bevorzugen?"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:119
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:119
|
||||||
msgid "Why should I use your hosting service?"
|
msgid "Why should I use your hosting service?"
|
||||||
msgstr ""
|
msgstr "Warum sollte ich Ihren Hosting-Service nutzen?"
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:60
|
#: apps/marketing/src/components/(marketing)/pricing-table.tsx:60
|
||||||
msgid "Yearly"
|
msgid "Yearly"
|
||||||
msgstr ""
|
msgstr "Jährlich"
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:167
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:167
|
||||||
msgid "Yes! Documenso is offered under the GNU AGPL V3 open source license. This means you can use it for free and even modify it to fit your needs, as long as you publish your changes under the same license."
|
msgid "Yes! Documenso is offered under the GNU AGPL V3 open source license. This means you can use it for free and even modify it to fit your needs, as long as you publish your changes under the same license."
|
||||||
msgstr ""
|
msgstr "Ja! Documenso wird unter der GNU AGPL V3 Open-Source-Lizenz angeboten. Das bedeutet, dass Sie es kostenlos nutzen und sogar an Ihre Bedürfnisse anpassen können, solange Sie Ihre Änderungen unter derselben Lizenz veröffentlichen."
|
||||||
|
|
||||||
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:93
|
#: apps/marketing/src/app/(marketing)/pricing/page.tsx:93
|
||||||
msgid "You can self-host Documenso for free or use our ready-to-use hosted version. The hosted version comes with additional support, painless scalability and more. Early adopters will get access to all features we build this year, for no additional cost! Forever! Yes, that includes multiple users per account later. If you want Documenso for your enterprise, we are happy to talk about your needs."
|
msgid "You can self-host Documenso for free or use our ready-to-use hosted version. The hosted version comes with additional support, painless scalability and more. Early adopters will get access to all features we build this year, for no additional cost! Forever! Yes, that includes multiple users per account later. If you want Documenso for your enterprise, we are happy to talk about your needs."
|
||||||
msgstr ""
|
msgstr "Sie können Documenso kostenlos selbst hosten oder unsere sofort einsatzbereite gehostete Version nutzen. Die gehostete Version bietet zusätzlichen Support, schmerzfreie Skalierbarkeit und mehr. Frühzeitige Anwender erhalten in diesem Jahr Zugriff auf alle Funktionen, die wir entwickeln, ohne zusätzliche Kosten! Für immer! Ja, das beinhaltet später mehrere Benutzer pro Konto. Wenn Sie Documenso für Ihr Unternehmen möchten, sprechen wir gerne über Ihre Bedürfnisse."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/carousel.tsx:265
|
#: apps/marketing/src/components/(marketing)/carousel.tsx:272
|
||||||
msgid "Your browser does not support the video tag."
|
msgid "Your browser does not support the video tag."
|
||||||
msgstr ""
|
msgstr "Ihr Browser unterstützt das Video-Tag nicht."
|
||||||
|
|
||||||
#: apps/marketing/src/components/(marketing)/carousel.tsx:265
|
|
||||||
#~ msgid "Your browser does not support the video tag.<<<<<<< HEAD"
|
|
||||||
#~ msgstr "Ihr Browser unterstützt das Video-Tag nicht.>>>>>>> main"
|
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
@@ -1,14 +1,14 @@
|
|||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"POT-Creation-Date: 2024-08-22 05:16+0000\n"
|
"POT-Creation-Date: 2024-07-24 13:01+1000\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=utf-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"X-Generator: @lingui/cli\n"
|
"X-Generator: @lingui/cli\n"
|
||||||
"Language: de\n"
|
"Language: de\n"
|
||||||
"Project-Id-Version: documenso-app\n"
|
"Project-Id-Version: documenso-app\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"PO-Revision-Date: 2024-09-05 06:04\n"
|
"PO-Revision-Date: 2024-09-16 16:03\n"
|
||||||
"Last-Translator: \n"
|
"Last-Translator: \n"
|
||||||
"Language-Team: German\n"
|
"Language-Team: German\n"
|
||||||
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
|
||||||
@@ -26,15 +26,15 @@ msgstr "\"{0}\" wird im Dokument erscheinen, da es eine Zeitzone von \"{timezone
|
|||||||
msgid "\"{documentTitle}\" has been successfully deleted"
|
msgid "\"{documentTitle}\" has been successfully deleted"
|
||||||
msgstr "\"{documentTitle}\" wurde erfolgreich gelöscht"
|
msgstr "\"{documentTitle}\" wurde erfolgreich gelöscht"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-page-view.tsx:76
|
#: apps/web/src/app/(signing)/sign/[token]/signing-page-view.tsx:78
|
||||||
msgid "({0}) has invited you to approve this document"
|
msgid "({0}) has invited you to approve this document"
|
||||||
msgstr "({0}) hat dich eingeladen, dieses Dokument zu genehmigen"
|
msgstr "({0}) hat dich eingeladen, dieses Dokument zu genehmigen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-page-view.tsx:73
|
#: apps/web/src/app/(signing)/sign/[token]/signing-page-view.tsx:75
|
||||||
msgid "({0}) has invited you to sign this document"
|
msgid "({0}) has invited you to sign this document"
|
||||||
msgstr "({0}) hat dich eingeladen, dieses Dokument zu unterzeichnen"
|
msgstr "({0}) hat dich eingeladen, dieses Dokument zu unterzeichnen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-page-view.tsx:70
|
#: apps/web/src/app/(signing)/sign/[token]/signing-page-view.tsx:72
|
||||||
msgid "({0}) has invited you to view this document"
|
msgid "({0}) has invited you to view this document"
|
||||||
msgstr "({0}) hat dich eingeladen, dieses Dokument zu betrachten"
|
msgstr "({0}) hat dich eingeladen, dieses Dokument zu betrachten"
|
||||||
|
|
||||||
@@ -60,11 +60,11 @@ msgstr "{0, plural, one {# Sitz} other {# Sitze}}"
|
|||||||
msgid "{0, plural, one {<0>You have <1>1</1> pending team invitation</0>} other {<2>You have <3>#</3> pending team invitations</2>}}"
|
msgid "{0, plural, one {<0>You have <1>1</1> pending team invitation</0>} other {<2>You have <3>#</3> pending team invitations</2>}}"
|
||||||
msgstr "{0, plural, one {<0>Du hast <1>1</1> ausstehende Team-Einladung</0>} other {<2>Du hast <3>#</3> ausstehende Team-Einladungen</2>}}"
|
msgstr "{0, plural, one {<0>Du hast <1>1</1> ausstehende Team-Einladung</0>} other {<2>Du hast <3>#</3> ausstehende Team-Einladungen</2>}}"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit/document-edit-page-view.tsx:102
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit/document-edit-page-view.tsx:129
|
||||||
msgid "{0, plural, one {1 Recipient} other {# Recipients}}"
|
msgid "{0, plural, one {1 Recipient} other {# Recipients}}"
|
||||||
msgstr "{0, plural, one {1 Empfänger} other {# Empfänger}}"
|
msgstr "{0, plural, one {1 Empfänger} other {# Empfänger}}"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:204
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:230
|
||||||
msgid "{0, plural, one {Waiting on 1 recipient} other {Waiting on # recipients}}"
|
msgid "{0, plural, one {Waiting on 1 recipient} other {Waiting on # recipients}}"
|
||||||
msgstr "{0, plural, one {Warte auf 1 Empfänger} other {Warte auf # Empfänger}}"
|
msgstr "{0, plural, one {Warte auf 1 Empfänger} other {Warte auf # Empfänger}}"
|
||||||
|
|
||||||
@@ -84,7 +84,7 @@ msgstr "{0} Dokument"
|
|||||||
msgid "{0} of {1} documents remaining this month."
|
msgid "{0} of {1} documents remaining this month."
|
||||||
msgstr "{0} von {1} Dokumenten verbleibend in diesem Monat."
|
msgstr "{0} von {1} Dokumenten verbleibend in diesem Monat."
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:139
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:165
|
||||||
msgid "{0} Recipient(s)"
|
msgid "{0} Recipient(s)"
|
||||||
msgstr "{0} Empfänger(in) / -n"
|
msgstr "{0} Empfänger(in) / -n"
|
||||||
|
|
||||||
@@ -220,7 +220,7 @@ msgstr "Aktive Abonnements"
|
|||||||
msgid "Add"
|
msgid "Add"
|
||||||
msgstr "Hinzufügen"
|
msgstr "Hinzufügen"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:143
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:157
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:87
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:87
|
||||||
msgid "Add all relevant fields for each recipient."
|
msgid "Add all relevant fields for each recipient."
|
||||||
msgstr "Fügen Sie alle relevanten Felder für jeden Empfänger hinzu."
|
msgstr "Fügen Sie alle relevanten Felder für jeden Empfänger hinzu."
|
||||||
@@ -241,7 +241,7 @@ msgstr "Fügen Sie einen Authenticator hinzu, um als sekundäre Authentifizierun
|
|||||||
msgid "Add email"
|
msgid "Add email"
|
||||||
msgstr "E-Mail hinzufügen"
|
msgstr "E-Mail hinzufügen"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:142
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:156
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:86
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:86
|
||||||
msgid "Add Fields"
|
msgid "Add Fields"
|
||||||
msgstr "Felder hinzufügen"
|
msgstr "Felder hinzufügen"
|
||||||
@@ -251,8 +251,8 @@ msgid "Add more"
|
|||||||
msgstr "Mehr hinzufügen"
|
msgstr "Mehr hinzufügen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/number-field.tsx:270
|
#: apps/web/src/app/(signing)/sign/[token]/number-field.tsx:270
|
||||||
msgid "Add number"
|
#~ msgid "Add number"
|
||||||
msgstr "Nummer hinzufügen"
|
#~ msgstr "Add number"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/settings/security/passkeys/create-passkey-dialog.tsx:146
|
#: apps/web/src/app/(dashboard)/settings/security/passkeys/create-passkey-dialog.tsx:146
|
||||||
#: apps/web/src/app/(dashboard)/settings/security/passkeys/create-passkey-dialog.tsx:154
|
#: apps/web/src/app/(dashboard)/settings/security/passkeys/create-passkey-dialog.tsx:154
|
||||||
@@ -263,11 +263,11 @@ msgstr "Passkey hinzufügen"
|
|||||||
msgid "Add Placeholders"
|
msgid "Add Placeholders"
|
||||||
msgstr "Platzhalter hinzufügen"
|
msgstr "Platzhalter hinzufügen"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:137
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:151
|
||||||
msgid "Add Signers"
|
msgid "Add Signers"
|
||||||
msgstr "Unterzeichner hinzufügen"
|
msgstr "Unterzeichner hinzufügen"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:147
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:161
|
||||||
msgid "Add Subject"
|
msgid "Add Subject"
|
||||||
msgstr "Betreff hinzufügen"
|
msgstr "Betreff hinzufügen"
|
||||||
|
|
||||||
@@ -276,14 +276,14 @@ msgid "Add team email"
|
|||||||
msgstr "Team-E-Mail hinzufügen"
|
msgstr "Team-E-Mail hinzufügen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:256
|
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:256
|
||||||
msgid "Add text"
|
#~ msgid "Add text"
|
||||||
msgstr "Text hinzufügen"
|
#~ msgstr "Add text"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:272
|
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:272
|
||||||
msgid "Add Text"
|
#~ msgid "Add Text"
|
||||||
msgstr "Text hinzufügen"
|
#~ msgstr "Add Text"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:138
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:152
|
||||||
msgid "Add the people who will sign the document."
|
msgid "Add the people who will sign the document."
|
||||||
msgstr "Fügen Sie die Personen hinzu, die das Dokument unterschreiben werden."
|
msgstr "Fügen Sie die Personen hinzu, die das Dokument unterschreiben werden."
|
||||||
|
|
||||||
@@ -291,7 +291,7 @@ msgstr "Fügen Sie die Personen hinzu, die das Dokument unterschreiben werden."
|
|||||||
msgid "Add the recipients to create the document with"
|
msgid "Add the recipients to create the document with"
|
||||||
msgstr "Fügen Sie die Empfänger hinzu, um das Dokument zu erstellen"
|
msgstr "Fügen Sie die Empfänger hinzu, um das Dokument zu erstellen"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:148
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:162
|
||||||
msgid "Add the subject and message you wish to send to signers."
|
msgid "Add the subject and message you wish to send to signers."
|
||||||
msgstr "Fügen Sie den Betreff und die Nachricht hinzu, die Sie den Unterzeichnern senden möchten."
|
msgstr "Fügen Sie den Betreff und die Nachricht hinzu, die Sie den Unterzeichnern senden möchten."
|
||||||
|
|
||||||
@@ -339,7 +339,7 @@ msgstr "Alle Zeiten"
|
|||||||
msgid "Allows authenticating using biometrics, password managers, hardware keys, etc."
|
msgid "Allows authenticating using biometrics, password managers, hardware keys, etc."
|
||||||
msgstr "Erlaubt die Authentifizierung mit biometrischen Daten, Passwort-Managern, Hardware-Schlüsseln usw."
|
msgstr "Erlaubt die Authentifizierung mit biometrischen Daten, Passwort-Managern, Hardware-Schlüsseln usw."
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:411
|
#: apps/web/src/components/forms/v2/signup.tsx:423
|
||||||
msgid "Already have an account? <0>Sign in instead</0>"
|
msgid "Already have an account? <0>Sign in instead</0>"
|
||||||
msgstr "Hast du bereits ein Konto? <0>Stattdessen anmelden</0>"
|
msgstr "Hast du bereits ein Konto? <0>Stattdessen anmelden</0>"
|
||||||
|
|
||||||
@@ -370,13 +370,13 @@ msgstr "Eine E-Mail, in der die Übertragung dieses Teams angefordert wird, wurd
|
|||||||
msgid "An error occurred"
|
msgid "An error occurred"
|
||||||
msgstr "Ein Fehler ist aufgetreten"
|
msgstr "Ein Fehler ist aufgetreten"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:226
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:248
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:176
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:197
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:210
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:231
|
||||||
msgid "An error occurred while adding signers."
|
msgid "An error occurred while adding signers."
|
||||||
msgstr "Ein Fehler ist aufgetreten, während Unterzeichner hinzugefügt wurden."
|
msgstr "Ein Fehler ist aufgetreten, während Unterzeichner hinzugefügt wurden."
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:256
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:278
|
||||||
msgid "An error occurred while adding the fields."
|
msgid "An error occurred while adding the fields."
|
||||||
msgstr "Ein Fehler ist aufgetreten, während die Felder hinzugefügt wurden."
|
msgstr "Ein Fehler ist aufgetreten, während die Felder hinzugefügt wurden."
|
||||||
|
|
||||||
@@ -434,7 +434,7 @@ msgstr "Ein Fehler ist aufgetreten, während die Unterschrift entfernt wurde."
|
|||||||
msgid "An error occurred while removing the text."
|
msgid "An error occurred while removing the text."
|
||||||
msgstr "Ein Fehler ist aufgetreten, während der Text entfernt wurde."
|
msgstr "Ein Fehler ist aufgetreten, während der Text entfernt wurde."
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:287
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:309
|
||||||
msgid "An error occurred while sending the document."
|
msgid "An error occurred while sending the document."
|
||||||
msgstr "Ein Fehler ist aufgetreten, während das Dokument gesendet wurde."
|
msgstr "Ein Fehler ist aufgetreten, während das Dokument gesendet wurde."
|
||||||
|
|
||||||
@@ -458,8 +458,8 @@ msgstr "Ein Fehler ist aufgetreten, während das Dokument unterzeichnet wurde."
|
|||||||
msgid "An error occurred while trying to create a checkout session."
|
msgid "An error occurred while trying to create a checkout session."
|
||||||
msgstr "Ein Fehler ist aufgetreten, während versucht wurde, eine Checkout-Sitzung zu erstellen."
|
msgstr "Ein Fehler ist aufgetreten, während versucht wurde, eine Checkout-Sitzung zu erstellen."
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:199
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:214
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:153
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:166
|
||||||
msgid "An error occurred while updating the document settings."
|
msgid "An error occurred while updating the document settings."
|
||||||
msgstr "Ein Fehler ist aufgetreten, während die Dokumenteinstellungen aktualisiert wurden."
|
msgstr "Ein Fehler ist aufgetreten, während die Dokumenteinstellungen aktualisiert wurden."
|
||||||
|
|
||||||
@@ -539,7 +539,7 @@ msgstr "App-Version"
|
|||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view-button.tsx:89
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view-button.tsx:89
|
||||||
#: apps/web/src/app/(dashboard)/documents/data-table-action-button.tsx:120
|
#: apps/web/src/app/(dashboard)/documents/data-table-action-button.tsx:120
|
||||||
#: apps/web/src/app/(dashboard)/documents/data-table-action-dropdown.tsx:144
|
#: apps/web/src/app/(dashboard)/documents/data-table-action-dropdown.tsx:144
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:122
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:125
|
||||||
msgid "Approve"
|
msgid "Approve"
|
||||||
msgstr "Genehmigen"
|
msgstr "Genehmigen"
|
||||||
|
|
||||||
@@ -578,7 +578,7 @@ msgid "Audit Log"
|
|||||||
msgstr "Audit-Protokoll"
|
msgstr "Audit-Protokoll"
|
||||||
|
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:41
|
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:41
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:57
|
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:52
|
||||||
msgid "Authentication required"
|
msgid "Authentication required"
|
||||||
msgstr "Authentifizierung erforderlich"
|
msgstr "Authentifizierung erforderlich"
|
||||||
|
|
||||||
@@ -596,7 +596,7 @@ msgstr "Warte auf E-Mail-Bestätigung"
|
|||||||
|
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/sign-direct-template.tsx:369
|
#: apps/web/src/app/(recipient)/d/[token]/sign-direct-template.tsx:369
|
||||||
#: apps/web/src/components/(dashboard)/settings/layout/activity-back.tsx:20
|
#: apps/web/src/components/(dashboard)/settings/layout/activity-back.tsx:20
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:497
|
#: apps/web/src/components/forms/v2/signup.tsx:509
|
||||||
msgid "Back"
|
msgid "Back"
|
||||||
msgstr "Zurück"
|
msgstr "Zurück"
|
||||||
|
|
||||||
@@ -609,7 +609,7 @@ msgid "Background Color"
|
|||||||
msgstr "Hintergrundfarbe"
|
msgstr "Hintergrundfarbe"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/2fa/disable-authenticator-app-dialog.tsx:167
|
#: apps/web/src/components/forms/2fa/disable-authenticator-app-dialog.tsx:167
|
||||||
#: apps/web/src/components/forms/signin.tsx:473
|
#: apps/web/src/components/forms/signin.tsx:485
|
||||||
msgid "Backup Code"
|
msgid "Backup Code"
|
||||||
msgstr "Backup-Code"
|
msgstr "Backup-Code"
|
||||||
|
|
||||||
@@ -621,7 +621,7 @@ msgstr "Backup-Codes"
|
|||||||
msgid "Banner Updated"
|
msgid "Banner Updated"
|
||||||
msgstr "Banner aktualisiert"
|
msgstr "Banner aktualisiert"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:460
|
#: apps/web/src/components/forms/v2/signup.tsx:472
|
||||||
msgid "Basic details"
|
msgid "Basic details"
|
||||||
msgstr "Basisdetails"
|
msgstr "Basisdetails"
|
||||||
|
|
||||||
@@ -662,13 +662,13 @@ msgstr "Durch die Aktivierung von 2FA müssen Sie jedes Mal, wenn Sie sich anmel
|
|||||||
#: apps/web/src/app/(dashboard)/templates/move-template-dialog.tsx:119
|
#: apps/web/src/app/(dashboard)/templates/move-template-dialog.tsx:119
|
||||||
#: apps/web/src/app/(dashboard)/templates/template-direct-link-dialog.tsx:470
|
#: apps/web/src/app/(dashboard)/templates/template-direct-link-dialog.tsx:470
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-2fa.tsx:178
|
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-2fa.tsx:178
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-account.tsx:74
|
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-account.tsx:71
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-passkey.tsx:164
|
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-passkey.tsx:164
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-passkey.tsx:189
|
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-passkey.tsx:189
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/form.tsx:143
|
#: apps/web/src/app/(signing)/sign/[token]/form.tsx:150
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/name-field.tsx:215
|
#: apps/web/src/app/(signing)/sign/[token]/name-field.tsx:215
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/number-field.tsx:327
|
#: apps/web/src/app/(signing)/sign/[token]/number-field.tsx:327
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:110
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:113
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:250
|
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:250
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:333
|
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:333
|
||||||
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/team-transfer-status.tsx:121
|
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/team-transfer-status.tsx:121
|
||||||
@@ -720,7 +720,7 @@ msgstr "Wählen..."
|
|||||||
msgid "Claim account"
|
msgid "Claim account"
|
||||||
msgstr "Konto beanspruchen"
|
msgstr "Konto beanspruchen"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:469
|
#: apps/web/src/components/forms/v2/signup.tsx:481
|
||||||
msgid "Claim username"
|
msgid "Claim username"
|
||||||
msgstr "Benutzername beanspruchen"
|
msgstr "Benutzername beanspruchen"
|
||||||
|
|
||||||
@@ -728,7 +728,7 @@ msgstr "Benutzername beanspruchen"
|
|||||||
msgid "Claim your profile later"
|
msgid "Claim your profile later"
|
||||||
msgstr "Profile später beanspruchen"
|
msgstr "Profile später beanspruchen"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:267
|
#: apps/web/src/components/forms/v2/signup.tsx:279
|
||||||
msgid "Claim your username now"
|
msgid "Claim your username now"
|
||||||
msgstr "Benutzername jetzt beanspruchen"
|
msgstr "Benutzername jetzt beanspruchen"
|
||||||
|
|
||||||
@@ -752,9 +752,9 @@ msgid "Click to copy signing link for sending to recipient"
|
|||||||
msgstr "Klicken Sie, um den Signatur-Link zu kopieren, um ihn an den Empfänger zu senden"
|
msgstr "Klicken Sie, um den Signatur-Link zu kopieren, um ihn an den Empfänger zu senden"
|
||||||
|
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/sign-direct-template.tsx:175
|
#: apps/web/src/app/(recipient)/d/[token]/sign-direct-template.tsx:175
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/form.tsx:107
|
#: apps/web/src/app/(signing)/sign/[token]/form.tsx:114
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:435
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:435
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:312
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:314
|
||||||
msgid "Click to insert field"
|
msgid "Click to insert field"
|
||||||
msgstr "Klicken Sie, um das Feld einzufügen"
|
msgstr "Klicken Sie, um das Feld einzufügen"
|
||||||
|
|
||||||
@@ -770,22 +770,22 @@ msgstr "Klicken Sie, um das Feld einzufügen"
|
|||||||
msgid "Close"
|
msgid "Close"
|
||||||
msgstr "Schließen"
|
msgstr "Schließen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:58
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:61
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:425
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:425
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:302
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:304
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:522
|
#: apps/web/src/components/forms/v2/signup.tsx:534
|
||||||
msgid "Complete"
|
msgid "Complete"
|
||||||
msgstr "Vollständig"
|
msgstr "Vollständig"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:67
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:70
|
||||||
msgid "Complete Approval"
|
msgid "Complete Approval"
|
||||||
msgstr "Genehmigung abschließen"
|
msgstr "Genehmigung abschließen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:66
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:69
|
||||||
msgid "Complete Signing"
|
msgid "Complete Signing"
|
||||||
msgstr "Unterzeichnung abschließen"
|
msgstr "Unterzeichnung abschließen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:65
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:68
|
||||||
msgid "Complete Viewing"
|
msgid "Complete Viewing"
|
||||||
msgstr "Betrachten abschließen"
|
msgstr "Betrachten abschließen"
|
||||||
|
|
||||||
@@ -802,7 +802,7 @@ msgstr "Abgeschlossene Dokumente"
|
|||||||
msgid "Completed Documents"
|
msgid "Completed Documents"
|
||||||
msgstr "Abgeschlossene Dokumente"
|
msgstr "Abgeschlossene Dokumente"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:133
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:147
|
||||||
msgid "Configure general settings for the document."
|
msgid "Configure general settings for the document."
|
||||||
msgstr "Konfigurieren Sie die allgemeinen Einstellungen für das Dokument."
|
msgstr "Konfigurieren Sie die allgemeinen Einstellungen für das Dokument."
|
||||||
|
|
||||||
@@ -887,7 +887,7 @@ msgstr "Token kopieren"
|
|||||||
msgid "Create"
|
msgid "Create"
|
||||||
msgstr "Erstellen"
|
msgstr "Erstellen"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:252
|
#: apps/web/src/components/forms/v2/signup.tsx:264
|
||||||
msgid "Create a new account"
|
msgid "Create a new account"
|
||||||
msgstr "Ein neues Konto erstellen"
|
msgstr "Ein neues Konto erstellen"
|
||||||
|
|
||||||
@@ -952,11 +952,11 @@ msgstr "Webhook erstellen"
|
|||||||
msgid "Create Webhook"
|
msgid "Create Webhook"
|
||||||
msgstr "Webhook erstellen"
|
msgstr "Webhook erstellen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/complete/page.tsx:214
|
#: apps/web/src/app/(signing)/sign/[token]/complete/page.tsx:215
|
||||||
msgid "Create your account and start using state-of-the-art document signing."
|
msgid "Create your account and start using state-of-the-art document signing."
|
||||||
msgstr "Erstellen Sie Ihr Konto und beginnen Sie mit dem modernen Dokumentensignieren."
|
msgstr "Erstellen Sie Ihr Konto und beginnen Sie mit dem modernen Dokumentensignieren."
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:256
|
#: apps/web/src/components/forms/v2/signup.tsx:268
|
||||||
msgid "Create your account and start using state-of-the-art document signing. Open and beautiful signing is within your grasp."
|
msgid "Create your account and start using state-of-the-art document signing. Open and beautiful signing is within your grasp."
|
||||||
msgstr "Erstellen Sie Ihr Konto und beginnen Sie mit dem modernen Dokumentensignieren. Offenes und schönes Signieren liegt in Ihrer Reichweite."
|
msgstr "Erstellen Sie Ihr Konto und beginnen Sie mit dem modernen Dokumentensignieren. Offenes und schönes Signieren liegt in Ihrer Reichweite."
|
||||||
|
|
||||||
@@ -986,10 +986,12 @@ msgstr "Erstellt am"
|
|||||||
#: apps/web/src/app/(dashboard)/settings/webhooks/page.tsx:88
|
#: apps/web/src/app/(dashboard)/settings/webhooks/page.tsx:88
|
||||||
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/webhooks/page.tsx:93
|
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/webhooks/page.tsx:93
|
||||||
msgid "Created on {0}"
|
msgid "Created on {0}"
|
||||||
msgstr ""
|
msgstr "Erstellt am {0}"
|
||||||
|
|
||||||
|
#: apps/web/src/app/(dashboard)/settings/webhooks/page.tsx:89
|
||||||
|
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/webhooks/page.tsx:94
|
||||||
#~ msgid "Created on <0/>"
|
#~ msgid "Created on <0/>"
|
||||||
#~ msgstr ""
|
#~ msgstr "Created on <0/>"
|
||||||
|
|
||||||
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/tokens/page.tsx:100
|
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/tokens/page.tsx:100
|
||||||
msgid "Created on{0}"
|
msgid "Created on{0}"
|
||||||
@@ -1218,7 +1220,7 @@ msgid "Document created"
|
|||||||
msgstr "Dokument erstellt"
|
msgstr "Dokument erstellt"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/admin/documents/[id]/super-delete-document-dialog.tsx:51
|
#: apps/web/src/app/(dashboard)/admin/documents/[id]/super-delete-document-dialog.tsx:51
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:147
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:173
|
||||||
#: apps/web/src/app/(dashboard)/documents/delete-document-dialog.tsx:59
|
#: apps/web/src/app/(dashboard)/documents/delete-document-dialog.tsx:59
|
||||||
msgid "Document deleted"
|
msgid "Document deleted"
|
||||||
msgstr "Dokument gelöscht"
|
msgstr "Dokument gelöscht"
|
||||||
@@ -1231,7 +1233,7 @@ msgstr "Dokument-Entwurf"
|
|||||||
msgid "Document Duplicated"
|
msgid "Document Duplicated"
|
||||||
msgstr "Dokument dupliziert"
|
msgstr "Dokument dupliziert"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:158
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:184
|
||||||
#: apps/web/src/components/document/document-history-sheet.tsx:104
|
#: apps/web/src/components/document/document-history-sheet.tsx:104
|
||||||
msgid "Document history"
|
msgid "Document history"
|
||||||
msgstr "Dokumentverlauf"
|
msgstr "Dokumentverlauf"
|
||||||
@@ -1272,7 +1274,7 @@ msgstr "Dokument erneut gesendet"
|
|||||||
msgid "Document resealed"
|
msgid "Document resealed"
|
||||||
msgstr "Dokument wieder versiegelt"
|
msgstr "Dokument wieder versiegelt"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:276
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:298
|
||||||
msgid "Document sent"
|
msgid "Document sent"
|
||||||
msgstr "Dokument gesendet"
|
msgstr "Dokument gesendet"
|
||||||
|
|
||||||
@@ -1310,11 +1312,11 @@ msgstr "Dokument wird dauerhaft gelöscht"
|
|||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/admin/nav.tsx:65
|
#: apps/web/src/app/(dashboard)/admin/nav.tsx:65
|
||||||
#: apps/web/src/app/(dashboard)/admin/users/data-table-users.tsx:92
|
#: apps/web/src/app/(dashboard)/admin/users/data-table-users.tsx:92
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:113
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:139
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit/document-edit-page-view.tsx:82
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit/document-edit-page-view.tsx:109
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/loading.tsx:16
|
#: apps/web/src/app/(dashboard)/documents/[id]/loading.tsx:16
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/sent/page.tsx:15
|
#: apps/web/src/app/(dashboard)/documents/[id]/sent/page.tsx:15
|
||||||
#: apps/web/src/app/(dashboard)/documents/documents-page-view.tsx:110
|
#: apps/web/src/app/(dashboard)/documents/documents-page-view.tsx:119
|
||||||
#: apps/web/src/app/(profile)/p/[url]/page.tsx:166
|
#: apps/web/src/app/(profile)/p/[url]/page.tsx:166
|
||||||
#: apps/web/src/app/not-found.tsx:21
|
#: apps/web/src/app/not-found.tsx:21
|
||||||
#: apps/web/src/components/(dashboard)/common/command-menu.tsx:205
|
#: apps/web/src/components/(dashboard)/common/command-menu.tsx:205
|
||||||
@@ -1333,7 +1335,7 @@ msgid "Documents Viewed"
|
|||||||
msgstr "Dokumente angesehen"
|
msgstr "Dokumente angesehen"
|
||||||
|
|
||||||
#: apps/web/src/app/(unauthenticated)/reset-password/[token]/page.tsx:40
|
#: apps/web/src/app/(unauthenticated)/reset-password/[token]/page.tsx:40
|
||||||
#: apps/web/src/app/(unauthenticated)/signin/page.tsx:61
|
#: apps/web/src/app/(unauthenticated)/signin/page.tsx:45
|
||||||
msgid "Don't have an account? <0>Sign up</0>"
|
msgid "Don't have an account? <0>Sign up</0>"
|
||||||
msgstr "Haben Sie kein Konto? <0>Registrieren</0>"
|
msgstr "Haben Sie kein Konto? <0>Registrieren</0>"
|
||||||
|
|
||||||
@@ -1405,12 +1407,12 @@ msgstr "Webhook bearbeiten"
|
|||||||
#: apps/web/src/app/(recipient)/d/[token]/configure-direct-template.tsx:118
|
#: apps/web/src/app/(recipient)/d/[token]/configure-direct-template.tsx:118
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/email-field.tsx:126
|
#: apps/web/src/app/(signing)/sign/[token]/email-field.tsx:126
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:376
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:376
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:254
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:256
|
||||||
#: apps/web/src/components/(teams)/dialogs/add-team-email-dialog.tsx:169
|
#: apps/web/src/components/(teams)/dialogs/add-team-email-dialog.tsx:169
|
||||||
#: apps/web/src/components/(teams)/dialogs/update-team-email-dialog.tsx:153
|
#: apps/web/src/components/(teams)/dialogs/update-team-email-dialog.tsx:153
|
||||||
#: apps/web/src/components/forms/forgot-password.tsx:81
|
#: apps/web/src/components/forms/forgot-password.tsx:81
|
||||||
#: apps/web/src/components/forms/profile.tsx:122
|
#: apps/web/src/components/forms/profile.tsx:122
|
||||||
#: apps/web/src/components/forms/signin.tsx:326
|
#: apps/web/src/components/forms/signin.tsx:338
|
||||||
#: apps/web/src/components/forms/signup.tsx:180
|
#: apps/web/src/components/forms/signup.tsx:180
|
||||||
msgid "Email"
|
msgid "Email"
|
||||||
msgstr "E-Mail"
|
msgstr "E-Mail"
|
||||||
@@ -1421,7 +1423,7 @@ msgstr "E-Mail"
|
|||||||
msgid "Email address"
|
msgid "Email address"
|
||||||
msgstr "E-Mail-Adresse"
|
msgstr "E-Mail-Adresse"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:316
|
#: apps/web/src/components/forms/v2/signup.tsx:328
|
||||||
msgid "Email Address"
|
msgid "Email Address"
|
||||||
msgstr "E-Mail-Adresse"
|
msgstr "E-Mail-Adresse"
|
||||||
|
|
||||||
@@ -1489,16 +1491,16 @@ msgstr "Geben Sie hier Ihren Text ein"
|
|||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/admin/documents/[id]/admin-actions.tsx:41
|
#: apps/web/src/app/(dashboard)/admin/documents/[id]/admin-actions.tsx:41
|
||||||
#: apps/web/src/app/(dashboard)/admin/users/[id]/page.tsx:78
|
#: apps/web/src/app/(dashboard)/admin/users/[id]/page.tsx:78
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:198
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:213
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:225
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:247
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:255
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:277
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:286
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:308
|
||||||
#: apps/web/src/app/(dashboard)/documents/move-document-dialog.tsx:57
|
#: apps/web/src/app/(dashboard)/documents/move-document-dialog.tsx:57
|
||||||
#: apps/web/src/app/(dashboard)/documents/upload-document.tsx:106
|
#: apps/web/src/app/(dashboard)/documents/upload-document.tsx:106
|
||||||
#: apps/web/src/app/(dashboard)/documents/upload-document.tsx:112
|
#: apps/web/src/app/(dashboard)/documents/upload-document.tsx:112
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:152
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:165
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:175
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:196
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:209
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:230
|
||||||
#: apps/web/src/app/(dashboard)/templates/duplicate-template-dialog.tsx:51
|
#: apps/web/src/app/(dashboard)/templates/duplicate-template-dialog.tsx:51
|
||||||
#: apps/web/src/app/(dashboard)/templates/move-template-dialog.tsx:56
|
#: apps/web/src/app/(dashboard)/templates/move-template-dialog.tsx:56
|
||||||
#: apps/web/src/app/(dashboard)/templates/use-template-dialog.tsx:152
|
#: apps/web/src/app/(dashboard)/templates/use-template-dialog.tsx:152
|
||||||
@@ -1542,8 +1544,10 @@ msgstr "Zeitüberschreitung überschritten"
|
|||||||
msgid "Expired"
|
msgid "Expired"
|
||||||
msgstr "Abgelaufen"
|
msgstr "Abgelaufen"
|
||||||
|
|
||||||
|
#: apps/web/src/app/(dashboard)/settings/tokens/page.tsx:73
|
||||||
|
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/tokens/page.tsx:106
|
||||||
#~ msgid "Expires on"
|
#~ msgid "Expires on"
|
||||||
#~ msgstr ""
|
#~ msgstr "Expires on"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/settings/tokens/page.tsx:71
|
#: apps/web/src/app/(dashboard)/settings/tokens/page.tsx:71
|
||||||
msgid "Expires on {0}"
|
msgid "Expires on {0}"
|
||||||
@@ -1556,7 +1560,7 @@ msgstr ">>>>>>> 4ca18b99 (fix: refactor dates)"
|
|||||||
|
|
||||||
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/tokens/page.tsx:107
|
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/tokens/page.tsx:107
|
||||||
msgid "Expires on{0}"
|
msgid "Expires on{0}"
|
||||||
msgstr ""
|
msgstr "Läuft ab am{0}"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/admin/documents/[id]/admin-actions.tsx:42
|
#: apps/web/src/app/(dashboard)/admin/documents/[id]/admin-actions.tsx:42
|
||||||
msgid "Failed to reseal document"
|
msgid "Failed to reseal document"
|
||||||
@@ -1580,20 +1584,20 @@ msgid "File cannot be larger than {APP_DOCUMENT_UPLOAD_SIZE_LIMIT}MB"
|
|||||||
msgstr "Die Datei darf nicht größer als {APP_DOCUMENT_UPLOAD_SIZE_LIMIT} MB sein"
|
msgstr "Die Datei darf nicht größer als {APP_DOCUMENT_UPLOAD_SIZE_LIMIT} MB sein"
|
||||||
|
|
||||||
#: apps/web/src/app/(unauthenticated)/forgot-password/page.tsx:21
|
#: apps/web/src/app/(unauthenticated)/forgot-password/page.tsx:21
|
||||||
#: apps/web/src/components/forms/signin.tsx:358
|
#: apps/web/src/components/forms/signin.tsx:370
|
||||||
msgid "Forgot your password?"
|
msgid "Forgot your password?"
|
||||||
msgstr "Haben Sie Ihr Passwort vergessen?"
|
msgstr "Haben Sie Ihr Passwort vergessen?"
|
||||||
|
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/sign-direct-template.tsx:326
|
#: apps/web/src/app/(recipient)/d/[token]/sign-direct-template.tsx:326
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/name-field.tsx:193
|
#: apps/web/src/app/(signing)/sign/[token]/name-field.tsx:193
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:361
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:361
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:239
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:241
|
||||||
#: apps/web/src/components/forms/profile.tsx:110
|
#: apps/web/src/components/forms/profile.tsx:110
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:300
|
#: apps/web/src/components/forms/v2/signup.tsx:312
|
||||||
msgid "Full Name"
|
msgid "Full Name"
|
||||||
msgstr "Vollständiger Name"
|
msgstr "Vollständiger Name"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:132
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:146
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:76
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:76
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/direct-template.tsx:60
|
#: apps/web/src/app/(recipient)/d/[token]/direct-template.tsx:60
|
||||||
#: apps/web/src/components/(teams)/settings/layout/desktop-nav.tsx:43
|
#: apps/web/src/components/(teams)/settings/layout/desktop-nav.tsx:43
|
||||||
@@ -1616,7 +1620,7 @@ msgstr "Zurück"
|
|||||||
msgid "Go back home"
|
msgid "Go back home"
|
||||||
msgstr "Zurück nach Hause"
|
msgstr "Zurück nach Hause"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/complete/page.tsx:223
|
#: apps/web/src/app/(signing)/sign/[token]/complete/page.tsx:226
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/no-longer-available.tsx:57
|
#: apps/web/src/app/(signing)/sign/[token]/no-longer-available.tsx:57
|
||||||
msgid "Go Back Home"
|
msgid "Go Back Home"
|
||||||
msgstr "Zurück nach Hause"
|
msgstr "Zurück nach Hause"
|
||||||
@@ -1774,9 +1778,13 @@ msgstr "Es scheint, dass kein Token bereitgestellt wurde. Wenn Sie versuchen, Ih
|
|||||||
msgid "It seems that there is no token provided. Please check your email and try again."
|
msgid "It seems that there is no token provided. Please check your email and try again."
|
||||||
msgstr "Es scheint, dass kein Token bereitgestellt wurde. Bitte überprüfen Sie Ihre E-Mail und versuchen Sie es erneut."
|
msgstr "Es scheint, dass kein Token bereitgestellt wurde. Bitte überprüfen Sie Ihre E-Mail und versuchen Sie es erneut."
|
||||||
|
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/waiting/page.tsx:74
|
||||||
|
msgid "It's currently not your turn to sign. You will receive an email with instructions once it's your turn to sign the document."
|
||||||
|
msgstr "Es ist derzeit nicht deine Reihe zu unterschreiben. Du erhältst eine E-Mail mit Anweisungen, sobald es deine Reihe ist, das Dokument zu unterschreiben."
|
||||||
|
|
||||||
#: apps/web/src/components/(dashboard)/layout/menu-switcher.tsx:286
|
#: apps/web/src/components/(dashboard)/layout/menu-switcher.tsx:286
|
||||||
msgid "Language"
|
msgid "Language"
|
||||||
msgstr ""
|
msgstr "Sprache"
|
||||||
|
|
||||||
#: apps/web/src/components/(dashboard)/period-selector/period-selector.tsx:61
|
#: apps/web/src/components/(dashboard)/period-selector/period-selector.tsx:61
|
||||||
msgid "Last 14 days"
|
msgid "Last 14 days"
|
||||||
@@ -1856,8 +1864,8 @@ msgid "Loading..."
|
|||||||
msgstr "Wird geladen..."
|
msgstr "Wird geladen..."
|
||||||
|
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:54
|
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:54
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-account.tsx:78
|
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-account.tsx:75
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:72
|
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:67
|
||||||
msgid "Login"
|
msgid "Login"
|
||||||
msgstr "Anmelden"
|
msgstr "Anmelden"
|
||||||
|
|
||||||
@@ -1929,7 +1937,7 @@ msgstr "Verwalten Sie Ihre Passkeys."
|
|||||||
msgid "Manage your site settings here"
|
msgid "Manage your site settings here"
|
||||||
msgstr "Verwalten Sie hier Ihre Seiteneinstellungen"
|
msgstr "Verwalten Sie hier Ihre Seiteneinstellungen"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:120
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:123
|
||||||
msgid "Mark as Viewed"
|
msgid "Mark as Viewed"
|
||||||
msgstr "Als angesehen markieren"
|
msgstr "Als angesehen markieren"
|
||||||
|
|
||||||
@@ -2008,7 +2016,7 @@ msgstr "Meine Vorlagen"
|
|||||||
msgid "Name"
|
msgid "Name"
|
||||||
msgstr "Name"
|
msgstr "Name"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/complete/page.tsx:210
|
#: apps/web/src/app/(signing)/sign/[token]/complete/page.tsx:211
|
||||||
msgid "Need to sign documents?"
|
msgid "Need to sign documents?"
|
||||||
msgstr "Müssen Dokumente signieren?"
|
msgstr "Müssen Dokumente signieren?"
|
||||||
|
|
||||||
@@ -2030,12 +2038,12 @@ msgid "New Template"
|
|||||||
msgstr "Neue Vorlage"
|
msgstr "Neue Vorlage"
|
||||||
|
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:416
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:416
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:293
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:295
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:509
|
#: apps/web/src/components/forms/v2/signup.tsx:521
|
||||||
msgid "Next"
|
msgid "Next"
|
||||||
msgstr "Nächster"
|
msgstr "Nächster"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:58
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:61
|
||||||
msgid "Next field"
|
msgid "Next field"
|
||||||
msgstr "Nächstes Feld"
|
msgstr "Nächstes Feld"
|
||||||
|
|
||||||
@@ -2093,6 +2101,10 @@ msgstr "Nicht unterstützt"
|
|||||||
msgid "Nothing to do"
|
msgid "Nothing to do"
|
||||||
msgstr "Nichts zu tun"
|
msgstr "Nichts zu tun"
|
||||||
|
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/number-field.tsx:270
|
||||||
|
msgid "Number"
|
||||||
|
msgstr "Nummer"
|
||||||
|
|
||||||
#: apps/web/src/components/(dashboard)/settings/webhooks/create-webhook-dialog.tsx:128
|
#: apps/web/src/components/(dashboard)/settings/webhooks/create-webhook-dialog.tsx:128
|
||||||
msgid "On this page, you can create a new webhook."
|
msgid "On this page, you can create a new webhook."
|
||||||
msgstr "Auf dieser Seite können Sie einen neuen Webhook erstellen."
|
msgstr "Auf dieser Seite können Sie einen neuen Webhook erstellen."
|
||||||
@@ -2138,11 +2150,11 @@ msgstr "Geöffnet"
|
|||||||
#: apps/web/src/app/(dashboard)/templates/template-direct-link-dialog.tsx:337
|
#: apps/web/src/app/(dashboard)/templates/template-direct-link-dialog.tsx:337
|
||||||
#: apps/web/src/components/forms/signup.tsx:243
|
#: apps/web/src/components/forms/signup.tsx:243
|
||||||
#: apps/web/src/components/forms/signup.tsx:267
|
#: apps/web/src/components/forms/signup.tsx:267
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:371
|
#: apps/web/src/components/forms/v2/signup.tsx:383
|
||||||
msgid "Or"
|
msgid "Or"
|
||||||
msgstr "Oder"
|
msgstr "Oder"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/signin.tsx:378
|
#: apps/web/src/components/forms/signin.tsx:390
|
||||||
msgid "Or continue with"
|
msgid "Or continue with"
|
||||||
msgstr "Oder fahren Sie fort mit"
|
msgstr "Oder fahren Sie fort mit"
|
||||||
|
|
||||||
@@ -2159,7 +2171,7 @@ msgstr "Besitzer"
|
|||||||
msgid "Paid"
|
msgid "Paid"
|
||||||
msgstr "Bezahlt"
|
msgstr "Bezahlt"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/signin.tsx:423
|
#: apps/web/src/components/forms/signin.tsx:435
|
||||||
msgid "Passkey"
|
msgid "Passkey"
|
||||||
msgstr "Passkey"
|
msgstr "Passkey"
|
||||||
|
|
||||||
@@ -2199,9 +2211,9 @@ msgstr "Passkeys werden von diesem Browser nicht unterstützt"
|
|||||||
#: apps/web/src/components/(dashboard)/common/command-menu.tsx:70
|
#: apps/web/src/components/(dashboard)/common/command-menu.tsx:70
|
||||||
#: apps/web/src/components/forms/password.tsx:123
|
#: apps/web/src/components/forms/password.tsx:123
|
||||||
#: apps/web/src/components/forms/reset-password.tsx:110
|
#: apps/web/src/components/forms/reset-password.tsx:110
|
||||||
#: apps/web/src/components/forms/signin.tsx:344
|
#: apps/web/src/components/forms/signin.tsx:356
|
||||||
#: apps/web/src/components/forms/signup.tsx:196
|
#: apps/web/src/components/forms/signup.tsx:196
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:332
|
#: apps/web/src/components/forms/v2/signup.tsx:344
|
||||||
msgid "Password"
|
msgid "Password"
|
||||||
msgstr "Passwort"
|
msgstr "Passwort"
|
||||||
|
|
||||||
@@ -2266,6 +2278,10 @@ msgstr "Wählen Sie eine der folgenden Vereinbarungen aus und beginnen Sie das S
|
|||||||
msgid "Please check the CSV file and make sure it is according to our format"
|
msgid "Please check the CSV file and make sure it is according to our format"
|
||||||
msgstr "Bitte prüfen Sie die CSV-Datei und stellen Sie sicher, dass sie unserem Format entspricht"
|
msgstr "Bitte prüfen Sie die CSV-Datei und stellen Sie sicher, dass sie unserem Format entspricht"
|
||||||
|
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/waiting/page.tsx:81
|
||||||
|
msgid "Please check your email for updates."
|
||||||
|
msgstr "Bitte überprüfe deine E-Mail auf Updates."
|
||||||
|
|
||||||
#: apps/web/src/app/(unauthenticated)/reset-password/[token]/page.tsx:34
|
#: apps/web/src/app/(unauthenticated)/reset-password/[token]/page.tsx:34
|
||||||
msgid "Please choose your new password"
|
msgid "Please choose your new password"
|
||||||
msgstr "Bitte wählen Sie Ihr neues Passwort"
|
msgstr "Bitte wählen Sie Ihr neues Passwort"
|
||||||
@@ -2278,7 +2294,7 @@ msgstr "Bitte kontaktieren Sie den Support, wenn Sie diese Aktion rückgängig m
|
|||||||
msgid "Please enter a meaningful name for your token. This will help you identify it later."
|
msgid "Please enter a meaningful name for your token. This will help you identify it later."
|
||||||
msgstr "Bitte geben Sie einen aussagekräftigen Namen für Ihr Token ein. Dies wird Ihnen helfen, es später zu identifizieren."
|
msgstr "Bitte geben Sie einen aussagekräftigen Namen für Ihr Token ein. Dies wird Ihnen helfen, es später zu identifizieren."
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/form.tsx:127
|
#: apps/web/src/app/(signing)/sign/[token]/form.tsx:134
|
||||||
msgid "Please mark as viewed to complete"
|
msgid "Please mark as viewed to complete"
|
||||||
msgstr "Bitte als angesehen markieren, um abzuschließen"
|
msgstr "Bitte als angesehen markieren, um abzuschließen"
|
||||||
|
|
||||||
@@ -2388,7 +2404,7 @@ msgstr "Öffentliches Profil"
|
|||||||
msgid "Public profile URL"
|
msgid "Public profile URL"
|
||||||
msgstr "Öffentlicher Profil-URL"
|
msgstr "Öffentlicher Profil-URL"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:438
|
#: apps/web/src/components/forms/v2/signup.tsx:450
|
||||||
msgid "Public profile username"
|
msgid "Public profile username"
|
||||||
msgstr "Öffentlicher Profil-Benutzername"
|
msgstr "Öffentlicher Profil-Benutzername"
|
||||||
|
|
||||||
@@ -2609,7 +2625,7 @@ msgstr "Sicherheit"
|
|||||||
msgid "Security activity"
|
msgid "Security activity"
|
||||||
msgstr "Sicherheitsaktivität"
|
msgstr "Sicherheitsaktivität"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/dropdown-field.tsx:192
|
#: apps/web/src/app/(signing)/sign/[token]/dropdown-field.tsx:194
|
||||||
msgid "Select"
|
msgid "Select"
|
||||||
msgstr "Auswählen"
|
msgstr "Auswählen"
|
||||||
|
|
||||||
@@ -2713,7 +2729,7 @@ msgstr "Vorlagen in Ihrem Team-Öffentliches Profil anzeigen, damit Ihre Zielgru
|
|||||||
#: apps/web/src/app/(profile)/p/[url]/page.tsx:192
|
#: apps/web/src/app/(profile)/p/[url]/page.tsx:192
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-2fa.tsx:182
|
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-2fa.tsx:182
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/name-field.tsx:224
|
#: apps/web/src/app/(signing)/sign/[token]/name-field.tsx:224
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:121
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:124
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:259
|
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:259
|
||||||
#: apps/web/src/components/ui/user-profile-skeleton.tsx:75
|
#: apps/web/src/components/ui/user-profile-skeleton.tsx:75
|
||||||
#: apps/web/src/components/ui/user-profile-timur.tsx:81
|
#: apps/web/src/components/ui/user-profile-timur.tsx:81
|
||||||
@@ -2733,7 +2749,7 @@ msgid "Sign as<0>{0} <1>({1})</1></0>"
|
|||||||
msgstr "Unterzeichnen als<0>{0} <1>({1})</1></0>"
|
msgstr "Unterzeichnen als<0>{0} <1>({1})</1></0>"
|
||||||
|
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:329
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:329
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:207
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:209
|
||||||
msgid "Sign document"
|
msgid "Sign document"
|
||||||
msgstr "Dokument unterschreiben"
|
msgstr "Dokument unterschreiben"
|
||||||
|
|
||||||
@@ -2742,17 +2758,17 @@ msgid "Sign field"
|
|||||||
msgstr "Unterzeichnen-Feld"
|
msgstr "Unterzeichnen-Feld"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/signup.tsx:212
|
#: apps/web/src/components/forms/signup.tsx:212
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:350
|
#: apps/web/src/components/forms/v2/signup.tsx:362
|
||||||
msgid "Sign Here"
|
msgid "Sign Here"
|
||||||
msgstr "Hier unterzeichnen"
|
msgstr "Hier unterzeichnen"
|
||||||
|
|
||||||
#: apps/web/src/app/not-found.tsx:29
|
#: apps/web/src/app/not-found.tsx:29
|
||||||
#: apps/web/src/components/forms/signin.tsx:371
|
#: apps/web/src/components/forms/signin.tsx:383
|
||||||
#: apps/web/src/components/forms/signin.tsx:498
|
#: apps/web/src/components/forms/signin.tsx:510
|
||||||
msgid "Sign In"
|
msgid "Sign In"
|
||||||
msgstr "Einloggen"
|
msgstr "Einloggen"
|
||||||
|
|
||||||
#: apps/web/src/app/(unauthenticated)/signin/page.tsx:44
|
#: apps/web/src/app/(unauthenticated)/signin/page.tsx:29
|
||||||
msgid "Sign in to your account"
|
msgid "Sign in to your account"
|
||||||
msgstr "Melden Sie sich bei Ihrem Konto an"
|
msgstr "Melden Sie sich bei Ihrem Konto an"
|
||||||
|
|
||||||
@@ -2762,11 +2778,11 @@ msgid "Sign Out"
|
|||||||
msgstr "Ausloggen"
|
msgstr "Ausloggen"
|
||||||
|
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:350
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:350
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:228
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:230
|
||||||
msgid "Sign the document to complete the process."
|
msgid "Sign the document to complete the process."
|
||||||
msgstr "Unterschreiben Sie das Dokument, um den Vorgang abzuschließen."
|
msgstr "Unterschreiben Sie das Dokument, um den Vorgang abzuschließen."
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:72
|
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:67
|
||||||
msgid "Sign up"
|
msgid "Sign up"
|
||||||
msgstr "Registrieren"
|
msgstr "Registrieren"
|
||||||
|
|
||||||
@@ -2775,12 +2791,12 @@ msgid "Sign Up"
|
|||||||
msgstr "Registrieren"
|
msgstr "Registrieren"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/signup.tsx:257
|
#: apps/web/src/components/forms/signup.tsx:257
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:389
|
#: apps/web/src/components/forms/v2/signup.tsx:401
|
||||||
msgid "Sign Up with Google"
|
msgid "Sign Up with Google"
|
||||||
msgstr "Registrieren mit Google"
|
msgstr "Registrieren mit Google"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/signup.tsx:281
|
#: apps/web/src/components/forms/signup.tsx:281
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:405
|
#: apps/web/src/components/forms/v2/signup.tsx:417
|
||||||
msgid "Sign Up with OIDC"
|
msgid "Sign Up with OIDC"
|
||||||
msgstr "Registrieren mit OIDC"
|
msgstr "Registrieren mit OIDC"
|
||||||
|
|
||||||
@@ -2789,7 +2805,7 @@ msgstr "Registrieren mit OIDC"
|
|||||||
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:197
|
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:197
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:227
|
#: apps/web/src/app/(signing)/sign/[token]/signature-field.tsx:227
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:391
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:391
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:268
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:270
|
||||||
#: apps/web/src/components/forms/profile.tsx:132
|
#: apps/web/src/components/forms/profile.tsx:132
|
||||||
msgid "Signature"
|
msgid "Signature"
|
||||||
msgstr "Unterschrift"
|
msgstr "Unterschrift"
|
||||||
@@ -2806,8 +2822,8 @@ msgstr "Unterschriften erscheinen, sobald das Dokument abgeschlossen ist"
|
|||||||
msgid "Signed"
|
msgid "Signed"
|
||||||
msgstr "Unterzeichnet"
|
msgstr "Unterzeichnet"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/signin.tsx:371
|
#: apps/web/src/components/forms/signin.tsx:383
|
||||||
#: apps/web/src/components/forms/signin.tsx:498
|
#: apps/web/src/components/forms/signin.tsx:510
|
||||||
msgid "Signing in..."
|
msgid "Signing in..."
|
||||||
msgstr "Anmeldung..."
|
msgstr "Anmeldung..."
|
||||||
|
|
||||||
@@ -2852,12 +2868,12 @@ msgstr "Website Einstellungen"
|
|||||||
#: apps/web/src/app/(dashboard)/templates/template-direct-link-dialog.tsx:151
|
#: apps/web/src/app/(dashboard)/templates/template-direct-link-dialog.tsx:151
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/direct-template.tsx:117
|
#: apps/web/src/app/(recipient)/d/[token]/direct-template.tsx:117
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:27
|
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:27
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:43
|
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:38
|
||||||
#: apps/web/src/app/(teams)/t/[teamUrl]/layout-billing-banner.tsx:53
|
#: apps/web/src/app/(teams)/t/[teamUrl]/layout-billing-banner.tsx:53
|
||||||
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/team-email-dropdown.tsx:39
|
#: apps/web/src/app/(teams)/t/[teamUrl]/settings/team-email-dropdown.tsx:39
|
||||||
#: apps/web/src/app/(unauthenticated)/verify-email/[token]/page.tsx:61
|
#: apps/web/src/app/(unauthenticated)/verify-email/[token]/page.tsx:61
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:243
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:243
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:123
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:125
|
||||||
#: apps/web/src/components/(teams)/dialogs/create-team-checkout-dialog.tsx:50
|
#: apps/web/src/components/(teams)/dialogs/create-team-checkout-dialog.tsx:50
|
||||||
#: apps/web/src/components/(teams)/dialogs/create-team-checkout-dialog.tsx:99
|
#: apps/web/src/components/(teams)/dialogs/create-team-checkout-dialog.tsx:99
|
||||||
#: apps/web/src/components/(teams)/dialogs/invite-team-member-dialog.tsx:210
|
#: apps/web/src/components/(teams)/dialogs/invite-team-member-dialog.tsx:210
|
||||||
@@ -3113,7 +3129,7 @@ msgstr "Template has been updated."
|
|||||||
msgid "Template moved"
|
msgid "Template moved"
|
||||||
msgstr "Template moved"
|
msgstr "Template moved"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:198
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:219
|
||||||
msgid "Template saved"
|
msgid "Template saved"
|
||||||
msgstr "Template saved"
|
msgstr "Template saved"
|
||||||
|
|
||||||
@@ -3129,6 +3145,11 @@ msgstr "Templates"
|
|||||||
msgid "Templates allow you to quickly generate documents with pre-filled recipients and fields."
|
msgid "Templates allow you to quickly generate documents with pre-filled recipients and fields."
|
||||||
msgstr "Templates allow you to quickly generate documents with pre-filled recipients and fields."
|
msgstr "Templates allow you to quickly generate documents with pre-filled recipients and fields."
|
||||||
|
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:256
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/text-field.tsx:272
|
||||||
|
msgid "Text"
|
||||||
|
msgstr "Text"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/admin/site-settings/banner-form.tsx:166
|
#: apps/web/src/app/(dashboard)/admin/site-settings/banner-form.tsx:166
|
||||||
msgid "Text Color"
|
msgid "Text Color"
|
||||||
msgstr "Text Color"
|
msgstr "Text Color"
|
||||||
@@ -3306,11 +3327,11 @@ msgstr "This document has been cancelled by the owner and is no longer available
|
|||||||
msgid "This document has been cancelled by the owner."
|
msgid "This document has been cancelled by the owner."
|
||||||
msgstr "This document has been cancelled by the owner."
|
msgstr "This document has been cancelled by the owner."
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:193
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:219
|
||||||
msgid "This document has been signed by all recipients"
|
msgid "This document has been signed by all recipients"
|
||||||
msgstr "This document has been signed by all recipients"
|
msgstr "This document has been signed by all recipients"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:196
|
#: apps/web/src/app/(dashboard)/documents/[id]/document-page-view.tsx:222
|
||||||
msgid "This document is currently a draft and has not been sent"
|
msgid "This document is currently a draft and has not been sent"
|
||||||
msgstr "This document is currently a draft and has not been sent"
|
msgstr "This document is currently a draft and has not been sent"
|
||||||
|
|
||||||
@@ -3413,7 +3434,7 @@ msgstr "To enable two-factor authentication, scan the following QR code using yo
|
|||||||
msgid "To gain access to your account, please confirm your email address by clicking on the confirmation link from your inbox."
|
msgid "To gain access to your account, please confirm your email address by clicking on the confirmation link from your inbox."
|
||||||
msgstr "To gain access to your account, please confirm your email address by clicking on the confirmation link from your inbox."
|
msgstr "To gain access to your account, please confirm your email address by clicking on the confirmation link from your inbox."
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-account.tsx:57
|
#: apps/web/src/app/(signing)/sign/[token]/document-action-auth-account.tsx:54
|
||||||
msgid "To mark this document as viewed, you need to be logged in as <0>{0}</0>"
|
msgid "To mark this document as viewed, you need to be logged in as <0>{0}</0>"
|
||||||
msgstr "To mark this document as viewed, you need to be logged in as <0>{0}</0>"
|
msgstr "To mark this document as viewed, you need to be logged in as <0>{0}</0>"
|
||||||
|
|
||||||
@@ -3502,7 +3523,7 @@ msgstr "Two factor authentication"
|
|||||||
msgid "Two factor authentication recovery codes are used to access your account in the event that you lose access to your authenticator app."
|
msgid "Two factor authentication recovery codes are used to access your account in the event that you lose access to your authenticator app."
|
||||||
msgstr "Two factor authentication recovery codes are used to access your account in the event that you lose access to your authenticator app."
|
msgstr "Two factor authentication recovery codes are used to access your account in the event that you lose access to your authenticator app."
|
||||||
|
|
||||||
#: apps/web/src/components/forms/signin.tsx:436
|
#: apps/web/src/components/forms/signin.tsx:448
|
||||||
msgid "Two-Factor Authentication"
|
msgid "Two-Factor Authentication"
|
||||||
msgstr "Two-Factor Authentication"
|
msgstr "Two-Factor Authentication"
|
||||||
|
|
||||||
@@ -3537,7 +3558,7 @@ msgstr "Uh oh! Looks like you're missing a token"
|
|||||||
|
|
||||||
#: apps/web/src/components/(dashboard)/common/command-menu.tsx:303
|
#: apps/web/src/components/(dashboard)/common/command-menu.tsx:303
|
||||||
msgid "Unable to change the language at this time. Please try again later."
|
msgid "Unable to change the language at this time. Please try again later."
|
||||||
msgstr ""
|
msgstr "Derzeit kann die Sprache nicht geändert werden. Bitte versuchen Sie es später erneut."
|
||||||
|
|
||||||
#: apps/web/src/components/forms/2fa/recovery-code-list.tsx:31
|
#: apps/web/src/components/forms/2fa/recovery-code-list.tsx:31
|
||||||
msgid "Unable to copy recovery code"
|
msgid "Unable to copy recovery code"
|
||||||
@@ -3712,12 +3733,12 @@ msgid "Uploaded file not an allowed file type"
|
|||||||
msgstr "Die hochgeladene Datei ist kein zulässiger Dateityp"
|
msgstr "Die hochgeladene Datei ist kein zulässiger Dateityp"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/2fa/disable-authenticator-app-dialog.tsx:187
|
#: apps/web/src/components/forms/2fa/disable-authenticator-app-dialog.tsx:187
|
||||||
#: apps/web/src/components/forms/signin.tsx:493
|
#: apps/web/src/components/forms/signin.tsx:505
|
||||||
msgid "Use Authenticator"
|
msgid "Use Authenticator"
|
||||||
msgstr "Authenticator verwenden"
|
msgstr "Authenticator verwenden"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/2fa/disable-authenticator-app-dialog.tsx:185
|
#: apps/web/src/components/forms/2fa/disable-authenticator-app-dialog.tsx:185
|
||||||
#: apps/web/src/components/forms/signin.tsx:491
|
#: apps/web/src/components/forms/signin.tsx:503
|
||||||
msgid "Use Backup Code"
|
msgid "Use Backup Code"
|
||||||
msgstr "Backup-Code verwenden"
|
msgstr "Backup-Code verwenden"
|
||||||
|
|
||||||
@@ -3733,7 +3754,7 @@ msgstr "Benutzer"
|
|||||||
msgid "User ID"
|
msgid "User ID"
|
||||||
msgstr "Benutzer-ID"
|
msgstr "Benutzer-ID"
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:222
|
#: apps/web/src/components/forms/v2/signup.tsx:234
|
||||||
msgid "User profiles are here!"
|
msgid "User profiles are here!"
|
||||||
msgstr "Benutzerprofile sind hier!"
|
msgstr "Benutzerprofile sind hier!"
|
||||||
|
|
||||||
@@ -3835,6 +3856,10 @@ msgstr "Warten"
|
|||||||
msgid "Waiting for others to sign"
|
msgid "Waiting for others to sign"
|
||||||
msgstr "Warten auf andere, um zu unterschreiben"
|
msgstr "Warten auf andere, um zu unterschreiben"
|
||||||
|
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/waiting/page.tsx:70
|
||||||
|
msgid "Waiting for Your Turn"
|
||||||
|
msgstr "Warten auf deine Reihe"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/no-longer-available.tsx:61
|
#: apps/web/src/app/(signing)/sign/[token]/no-longer-available.tsx:61
|
||||||
msgid "Want to send slick signing links like this one? <0>Check out Documenso.</0>"
|
msgid "Want to send slick signing links like this one? <0>Check out Documenso.</0>"
|
||||||
msgstr "Möchten Sie auffällige Signatur-Links wie diesen senden? <0>Überprüfen Sie Documenso.</0>"
|
msgstr "Möchten Sie auffällige Signatur-Links wie diesen senden? <0>Überprüfen Sie Documenso.</0>"
|
||||||
@@ -4008,7 +4033,7 @@ msgid "We were unable to disable two-factor authentication for your account. Ple
|
|||||||
msgstr "Wir konnten die Zwei-Faktor-Authentifizierung für Ihr Konto nicht deaktivieren. Bitte stellen Sie sicher, dass Sie Ihr Passwort und den Backup-Code korrekt eingegeben haben und versuchen Sie es erneut."
|
msgstr "Wir konnten die Zwei-Faktor-Authentifizierung für Ihr Konto nicht deaktivieren. Bitte stellen Sie sicher, dass Sie Ihr Passwort und den Backup-Code korrekt eingegeben haben und versuchen Sie es erneut."
|
||||||
|
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:28
|
#: apps/web/src/app/(recipient)/d/[token]/signing-auth-page.tsx:28
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:44
|
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:39
|
||||||
msgid "We were unable to log you out at this time."
|
msgid "We were unable to log you out at this time."
|
||||||
msgstr "Wir konnten Sie zurzeit nicht abmelden."
|
msgstr "Wir konnten Sie zurzeit nicht abmelden."
|
||||||
|
|
||||||
@@ -4023,7 +4048,7 @@ msgstr "Wir konnten die Zwei-Faktor-Authentifizierung für Ihr Konto nicht einri
|
|||||||
|
|
||||||
#: apps/web/src/app/(recipient)/d/[token]/direct-template.tsx:119
|
#: apps/web/src/app/(recipient)/d/[token]/direct-template.tsx:119
|
||||||
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:245
|
#: apps/web/src/app/embed/direct/[[...url]]/client.tsx:245
|
||||||
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:125
|
#: apps/web/src/app/embed/sign/[[...url]]/client.tsx:127
|
||||||
msgid "We were unable to submit this document at this time. Please try again later."
|
msgid "We were unable to submit this document at this time. Please try again later."
|
||||||
msgstr "Wir konnten dieses Dokument zurzeit nicht einreichen. Bitte versuchen Sie es später erneut."
|
msgstr "Wir konnten dieses Dokument zurzeit nicht einreichen. Bitte versuchen Sie es später erneut."
|
||||||
|
|
||||||
@@ -4075,10 +4100,14 @@ msgstr "Webhooks"
|
|||||||
msgid "Weekly"
|
msgid "Weekly"
|
||||||
msgstr "Wöchentlich"
|
msgstr "Wöchentlich"
|
||||||
|
|
||||||
#: apps/web/src/app/(unauthenticated)/signin/page.tsx:48
|
#: apps/web/src/app/(unauthenticated)/signin/page.tsx:33
|
||||||
msgid "Welcome back, we are lucky to have you."
|
msgid "Welcome back, we are lucky to have you."
|
||||||
msgstr "Willkommen zurück, wir freuen uns, Sie zu haben."
|
msgstr "Willkommen zurück, wir freuen uns, Sie zu haben."
|
||||||
|
|
||||||
|
#: apps/web/src/app/(signing)/sign/[token]/waiting/page.tsx:88
|
||||||
|
msgid "Were you trying to edit this document instead?"
|
||||||
|
msgstr "Hast du stattdessen versucht, dieses Dokument zu bearbeiten?"
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/settings/security/passkeys/create-passkey-dialog.tsx:189
|
#: apps/web/src/app/(dashboard)/settings/security/passkeys/create-passkey-dialog.tsx:189
|
||||||
msgid "When you click continue, you will be prompted to add the first available authenticator on your system."
|
msgid "When you click continue, you will be prompted to add the first available authenticator on your system."
|
||||||
msgstr "Wenn Sie auf Fortfahren klicken, werden Sie aufgefordert, den ersten verfügbaren Authenticator auf Ihrem System hinzuzufügen."
|
msgstr "Wenn Sie auf Fortfahren klicken, werden Sie aufgefordert, den ersten verfügbaren Authenticator auf Ihrem System hinzuzufügen."
|
||||||
@@ -4107,15 +4136,15 @@ msgstr "Jährlich"
|
|||||||
msgid "You"
|
msgid "You"
|
||||||
msgstr "Sie"
|
msgstr "Sie"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:90
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:93
|
||||||
msgid "You are about to complete approving \"{truncatedTitle}\".<0/> Are you sure?"
|
msgid "You are about to complete approving \"{truncatedTitle}\".<0/> Are you sure?"
|
||||||
msgstr "Sie stehen kurz davor, die Genehmigung für \"{truncatedTitle}\" abzuschließen.<0/> Sind Sie sicher?"
|
msgstr "Sie stehen kurz davor, die Genehmigung für \"{truncatedTitle}\" abzuschließen.<0/> Sind Sie sicher?"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:82
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:85
|
||||||
msgid "You are about to complete signing \"{truncatedTitle}\".<0/> Are you sure?"
|
msgid "You are about to complete signing \"{truncatedTitle}\".<0/> Are you sure?"
|
||||||
msgstr "Sie stehen kurz davor, \"{truncatedTitle}\" zu unterzeichnen.<0/> Sind Sie sicher?"
|
msgstr "Sie stehen kurz davor, \"{truncatedTitle}\" zu unterzeichnen.<0/> Sind Sie sicher?"
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:74
|
#: apps/web/src/app/(signing)/sign/[token]/sign-dialog.tsx:77
|
||||||
msgid "You are about to complete viewing \"{truncatedTitle}\".<0/> Are you sure?"
|
msgid "You are about to complete viewing \"{truncatedTitle}\".<0/> Are you sure?"
|
||||||
msgstr "Sie stehen kurz davor, \"{truncatedTitle}\" anzusehen.<0/> Sind Sie sicher?"
|
msgstr "Sie stehen kurz davor, \"{truncatedTitle}\" anzusehen.<0/> Sind Sie sicher?"
|
||||||
|
|
||||||
@@ -4297,7 +4326,7 @@ msgstr "Sie müssen mindestens einen anderen Teamkollegen haben, um die Eigentum
|
|||||||
msgid "You must set a profile URL before enabling your public profile."
|
msgid "You must set a profile URL before enabling your public profile."
|
||||||
msgstr "Sie müssen eine Profil-URL festlegen, bevor Sie Ihr öffentliches Profil aktivieren."
|
msgstr "Sie müssen eine Profil-URL festlegen, bevor Sie Ihr öffentliches Profil aktivieren."
|
||||||
|
|
||||||
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:61
|
#: apps/web/src/app/(signing)/sign/[token]/signing-auth-page.tsx:56
|
||||||
msgid "You need to be logged in as <0>{email}</0> to view this page."
|
msgid "You need to be logged in as <0>{email}</0> to view this page."
|
||||||
msgstr "Sie müssen als <0>{email}</0> angemeldet sein, um diese Seite anzuzeigen."
|
msgstr "Sie müssen als <0>{email}</0> angemeldet sein, um diese Seite anzuzeigen."
|
||||||
|
|
||||||
@@ -4309,7 +4338,7 @@ msgstr "Sie müssen angemeldet sein, um diese Seite anzuzeigen."
|
|||||||
msgid "You need to setup 2FA to mark this document as viewed."
|
msgid "You need to setup 2FA to mark this document as viewed."
|
||||||
msgstr "Sie müssen 2FA einrichten, um dieses Dokument als angesehen zu markieren."
|
msgstr "Sie müssen 2FA einrichten, um dieses Dokument als angesehen zu markieren."
|
||||||
|
|
||||||
#: apps/web/src/components/forms/v2/signup.tsx:271
|
#: apps/web/src/components/forms/v2/signup.tsx:283
|
||||||
msgid "You will get notified & be able to set up your documenso public profile when we launch the feature."
|
msgid "You will get notified & be able to set up your documenso public profile when we launch the feature."
|
||||||
msgstr "Sie werden benachrichtigt und können Ihr Documenso öffentliches Profil einrichten, wenn wir die Funktion starten."
|
msgstr "Sie werden benachrichtigt und können Ihr Documenso öffentliches Profil einrichten, wenn wir die Funktion starten."
|
||||||
|
|
||||||
@@ -4353,7 +4382,7 @@ msgstr "Ihr Dokument wurde erfolgreich aus der Vorlage erstellt."
|
|||||||
msgid "Your document has been re-sent successfully."
|
msgid "Your document has been re-sent successfully."
|
||||||
msgstr "Ihr Dokument wurde erfolgreich erneut gesendet."
|
msgstr "Ihr Dokument wurde erfolgreich erneut gesendet."
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:277
|
#: apps/web/src/app/(dashboard)/documents/[id]/edit-document.tsx:299
|
||||||
msgid "Your document has been sent successfully."
|
msgid "Your document has been sent successfully."
|
||||||
msgstr "Ihr Dokument wurde erfolgreich gesendet."
|
msgstr "Ihr Dokument wurde erfolgreich gesendet."
|
||||||
|
|
||||||
@@ -4444,7 +4473,7 @@ msgstr "Ihre Vorlage wurde erfolgreich gelöscht."
|
|||||||
msgid "Your template will be duplicated."
|
msgid "Your template will be duplicated."
|
||||||
msgstr "Ihre Vorlage wird dupliziert."
|
msgstr "Ihre Vorlage wird dupliziert."
|
||||||
|
|
||||||
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:199
|
#: apps/web/src/app/(dashboard)/templates/[id]/edit-template.tsx:220
|
||||||
msgid "Your templates has been saved successfully."
|
msgid "Your templates has been saved successfully."
|
||||||
msgstr "Ihre Vorlagen wurden erfolgreich gespeichert."
|
msgstr "Ihre Vorlagen wurden erfolgreich gespeichert."
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user