mirror of
https://github.com/documenso/documenso.git
synced 2025-11-12 07:43:16 +10:00
## Description Show a dialog when the document has signers with no signature fields placed. ## Changes Made Created a new dialog that'll be triggered when the document owner tries to send a document to the signers without placing signature fields. The document owners can't proceed to the next step unless they add signature fields. ## Checklist - [x] I have tested these changes locally and they work as expected. - [ ] I have added/updated tests that prove the effectiveness of these changes. - [ ] I have updated the documentation to reflect these changes, if applicable. - [x] I have followed the project's coding style guidelines. - [ ] I have addressed the code review feedback from the previous submission, if applicable. https://github.com/documenso/documenso/assets/25515812/f1b5c34e-2ce0-40e3-804c-f05d23045710 <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit - **New Features** - Introduced "Direct Links" for async signing, allowing users to create documents from templates using public links. - Added `MissingSignatureFieldDialog` component to ensure users don't miss adding signature fields. - **Enhancements** - Updated blog content to provide guidance on contract management and announce new pricing plans. - **Bug Fixes** - Improved async signing process for better efficiency and control. - **Refactor** - Improved internal code structure and import order for stripe-related functionality. - **Tests** - Enhanced e2e tests to verify signature presence before document creation and updated test flows for document approval. <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Co-authored-by: David Nguyen <davidngu28@gmail.com>
60 lines
2.0 KiB
TypeScript
60 lines
2.0 KiB
TypeScript
import type Stripe from 'stripe';
|
|
|
|
import type { STRIPE_PLAN_TYPE } from '@documenso/lib/constants/billing';
|
|
import { stripe } from '@documenso/lib/server-only/stripe';
|
|
|
|
// Utility type to handle usage of the `expand` option.
|
|
type PriceWithProduct = Stripe.Price & { product: Stripe.Product };
|
|
|
|
export type PriceIntervals = Record<Stripe.Price.Recurring.Interval, PriceWithProduct[]>;
|
|
|
|
export type GetPricesByIntervalOptions = {
|
|
/**
|
|
* Filter products by their meta 'plan' attribute.
|
|
*/
|
|
plan?: STRIPE_PLAN_TYPE.COMMUNITY | STRIPE_PLAN_TYPE.REGULAR;
|
|
};
|
|
|
|
export const getPricesByInterval = async ({ plan }: GetPricesByIntervalOptions = {}) => {
|
|
let { data: prices } = await stripe.prices.search({
|
|
query: `active:'true' type:'recurring'`,
|
|
expand: ['data.product'],
|
|
limit: 100,
|
|
});
|
|
|
|
prices = prices.filter((price) => {
|
|
// We use `expand` to get the product, but it's not typed as part of the Price type.
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
const product = price.product as Stripe.Product;
|
|
|
|
const filter = !plan || product.metadata?.plan === plan;
|
|
|
|
// Filter out prices for products that are not active.
|
|
return product.active && filter;
|
|
});
|
|
|
|
const intervals: PriceIntervals = {
|
|
day: [],
|
|
week: [],
|
|
month: [],
|
|
year: [],
|
|
};
|
|
|
|
// Add each price to the correct interval.
|
|
for (const price of prices) {
|
|
if (price.recurring?.interval) {
|
|
// We use `expand` to get the product, but it's not typed as part of the Price type.
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
intervals[price.recurring.interval].push(price as PriceWithProduct);
|
|
}
|
|
}
|
|
|
|
// Order all prices by unit_amount.
|
|
intervals.day.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
intervals.week.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
intervals.month.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
intervals.year.sort((a, b) => Number(a.unit_amount) - Number(b.unit_amount));
|
|
|
|
return intervals;
|
|
};
|