1 | import { useMemo } from 'react';
|
---|
2 | import { Page, View, Text, Image, Document, Font, StyleSheet } from '@react-pdf/renderer';
|
---|
3 | // utils
|
---|
4 | import { fDate } from 'src/utils/format-time';
|
---|
5 | import { fCurrency } from 'src/utils/format-number';
|
---|
6 | // types
|
---|
7 | import { Invoice } from 'src/schemas';
|
---|
8 | import { createFullAddress } from 'src/utils/create-full-address';
|
---|
9 | import { getQuantityType } from 'src/utils/get-invoice-quantity-type';
|
---|
10 | // import signatureImage from 'src/assets/png/signature.png';
|
---|
11 |
|
---|
12 | // ----------------------------------------------------------------------
|
---|
13 |
|
---|
14 | Font.register({
|
---|
15 | family: 'Roboto',
|
---|
16 | fonts: [{ src: '/fonts/Roboto-Regular.ttf' }, { src: '/fonts/Roboto-Bold.ttf' }],
|
---|
17 | });
|
---|
18 |
|
---|
19 | const useStyles = () =>
|
---|
20 | useMemo(
|
---|
21 | () =>
|
---|
22 | StyleSheet.create({
|
---|
23 | col4: { width: '25%' },
|
---|
24 | col8: { width: '75%' },
|
---|
25 | col6: { width: '50%' },
|
---|
26 | mb4: { marginBottom: 4 },
|
---|
27 | mb8: { marginBottom: 8 },
|
---|
28 | mb40: { marginBottom: 40 },
|
---|
29 | h3: { fontSize: 16, fontWeight: 700 },
|
---|
30 | h4: { fontSize: 13, fontWeight: 700 },
|
---|
31 | body1: { fontSize: 11 },
|
---|
32 | body2: { fontSize: 10 },
|
---|
33 | subtitle1: { fontSize: 11, fontWeight: 700 },
|
---|
34 | subtitle2: { fontSize: 10, fontWeight: 700 },
|
---|
35 | alignRight: { textAlign: 'right' },
|
---|
36 | page: {
|
---|
37 | fontSize: 9,
|
---|
38 | lineHeight: 1.6,
|
---|
39 | fontFamily: 'Roboto',
|
---|
40 | backgroundColor: '#FFFFFF',
|
---|
41 | textTransform: 'capitalize',
|
---|
42 | padding: '40px 24px 120px 24px',
|
---|
43 | },
|
---|
44 | footer: {
|
---|
45 | left: 0,
|
---|
46 | right: 0,
|
---|
47 | bottom: 0,
|
---|
48 | padding: 24,
|
---|
49 | margin: 'auto',
|
---|
50 | borderTopWidth: 1,
|
---|
51 | borderStyle: 'solid',
|
---|
52 | position: 'absolute',
|
---|
53 | borderColor: '#DFE3E8',
|
---|
54 | },
|
---|
55 | bankInfo: {
|
---|
56 | alignItems: 'stretch', // Keeps original alignment from your structure
|
---|
57 | flexDirection: 'column',
|
---|
58 | },
|
---|
59 | bankRow: {
|
---|
60 | flexDirection: 'row',
|
---|
61 | alignItems: 'center',
|
---|
62 | justifyContent: 'space-between', // Ensures the key is left and value is right aligned
|
---|
63 | gap: 24,
|
---|
64 | },
|
---|
65 | bankKey: {
|
---|
66 | color: '#808080', // Light gray for the key
|
---|
67 | },
|
---|
68 | bankValue: {
|
---|
69 | textAlign: 'right', // Aligns the value to the right
|
---|
70 | },
|
---|
71 | signatures: {
|
---|
72 | left: 0,
|
---|
73 | right: 0,
|
---|
74 | bottom: 0,
|
---|
75 | padding: 24,
|
---|
76 | margin: 'auto',
|
---|
77 | borderStyle: 'solid',
|
---|
78 | position: 'absolute',
|
---|
79 | },
|
---|
80 | gridContainer: {
|
---|
81 | flexDirection: 'row',
|
---|
82 | justifyContent: 'space-between',
|
---|
83 | },
|
---|
84 | table: {
|
---|
85 | display: 'flex',
|
---|
86 | width: 'auto',
|
---|
87 | },
|
---|
88 | tableHead: {
|
---|
89 | padding: '5px 5px',
|
---|
90 | flexDirection: 'row',
|
---|
91 | },
|
---|
92 | tableRow: {
|
---|
93 | padding: '5px 5px',
|
---|
94 | flexDirection: 'row',
|
---|
95 | borderBottomWidth: 1,
|
---|
96 | borderStyle: 'solid',
|
---|
97 | borderColor: '#DFE3E8',
|
---|
98 | },
|
---|
99 | noBorder: {
|
---|
100 | paddingTop: 8,
|
---|
101 | paddingBottom: 0,
|
---|
102 | borderBottomWidth: 0,
|
---|
103 | },
|
---|
104 | tableCell_1: {
|
---|
105 | width: '5%',
|
---|
106 | },
|
---|
107 | tableCell_2: {
|
---|
108 | width: '50%',
|
---|
109 | paddingRight: 16,
|
---|
110 | },
|
---|
111 | tableCell_3: {
|
---|
112 | width: '15%',
|
---|
113 | },
|
---|
114 | }),
|
---|
115 | []
|
---|
116 | );
|
---|
117 |
|
---|
118 | // ----------------------------------------------------------------------
|
---|
119 |
|
---|
120 | type Props = {
|
---|
121 | invoice: Invoice;
|
---|
122 | currentStatus: string;
|
---|
123 | };
|
---|
124 |
|
---|
125 | export default function InvoiceEEPDF({ invoice, currentStatus }: Props) {
|
---|
126 | const {
|
---|
127 | items,
|
---|
128 | dueDate,
|
---|
129 | discount,
|
---|
130 | invoiceTo,
|
---|
131 | issueDate,
|
---|
132 | totalAmount,
|
---|
133 | invoiceFrom,
|
---|
134 | invoiceNumber,
|
---|
135 | subTotal,
|
---|
136 | currency,
|
---|
137 | quantityType,
|
---|
138 | } = invoice;
|
---|
139 |
|
---|
140 | const styles = useStyles();
|
---|
141 |
|
---|
142 | return (
|
---|
143 | <Document>
|
---|
144 | <Page size="A4" style={styles.page}>
|
---|
145 | <View style={[styles.gridContainer, styles.mb40]}>
|
---|
146 | <Image source="/logo/logo_single.png" style={{ width: 48, height: 48 }} />
|
---|
147 |
|
---|
148 | <View style={{ alignItems: 'stretch', flexDirection: 'column' }}>
|
---|
149 | <Text style={styles.h3}>Invoice No. {invoiceNumber}</Text>
|
---|
150 | <Text style={styles.body2}>{invoiceFrom.name}</Text>
|
---|
151 | {currency === 'EUR' ? (
|
---|
152 | <>
|
---|
153 | <View style={styles.bankRow}>
|
---|
154 | <Text style={styles.bankKey}>WISE EUROPE SA:</Text>
|
---|
155 | <Text style={styles.bankValue}>
|
---|
156 | {invoiceFrom.bankAccounts?.eur?.accountNumber}
|
---|
157 | </Text>
|
---|
158 | </View>
|
---|
159 | <View style={styles.bankRow}>
|
---|
160 | <Text style={styles.bankKey}>IBAN:</Text>
|
---|
161 | <Text style={styles.bankValue}>{invoiceFrom.bankAccounts?.eur?.iban}</Text>
|
---|
162 | </View>
|
---|
163 | <View style={styles.bankRow}>
|
---|
164 | <Text style={styles.bankKey}>SWIFT/BIC:</Text>
|
---|
165 | <Text style={styles.bankValue}>{invoiceFrom.bankAccounts?.eur?.bicSwift}</Text>
|
---|
166 | </View>
|
---|
167 | </>
|
---|
168 | ) : (
|
---|
169 | <>
|
---|
170 | <View style={styles.bankRow}>
|
---|
171 | <Text style={styles.bankKey}>WISE EUROPE SA:</Text>
|
---|
172 | <Text style={styles.bankValue}>
|
---|
173 | {invoiceFrom.bankAccounts?.usd?.accountNumber}
|
---|
174 | </Text>
|
---|
175 | </View>
|
---|
176 | <View style={styles.bankRow}>
|
---|
177 | <Text style={styles.bankKey}>ROUTING NUMBER:</Text>
|
---|
178 | <Text style={styles.bankValue}>
|
---|
179 | {invoiceFrom.bankAccounts?.usd?.routingNumber}
|
---|
180 | </Text>
|
---|
181 | </View>
|
---|
182 | <View style={styles.bankRow}>
|
---|
183 | <Text style={styles.bankKey}>SWIFT/BIC:</Text>
|
---|
184 | <Text style={styles.bankValue}>{invoiceFrom.bankAccounts?.usd?.bicSwift}</Text>
|
---|
185 | </View>
|
---|
186 | </>
|
---|
187 | )}
|
---|
188 | </View>
|
---|
189 | </View>
|
---|
190 |
|
---|
191 | <View style={[styles.gridContainer, styles.mb40, { gap: 16 }]}>
|
---|
192 | <View
|
---|
193 | style={[
|
---|
194 | styles.col6,
|
---|
195 | { border: 1, borderColor: '#DFE3E8', borderRadius: 5, padding: 4 },
|
---|
196 | ]}
|
---|
197 | >
|
---|
198 | <Text style={[styles.subtitle1, styles.mb4]}>Invoice from</Text>
|
---|
199 | <Text style={[styles.body2, { fontWeight: 700 }]}>{invoiceFrom.name}</Text>
|
---|
200 | <Text style={styles.body2}>{createFullAddress(invoiceFrom.address)}</Text>
|
---|
201 | <Text style={styles.body2}>{invoiceFrom.phoneNumber}</Text>
|
---|
202 | <View style={{ marginTop: 5 }} />
|
---|
203 | <Text style={{ ...styles.body2, textTransform: 'lowercase' }}>{invoiceFrom.email}</Text>
|
---|
204 | </View>
|
---|
205 |
|
---|
206 | <View
|
---|
207 | style={[
|
---|
208 | styles.col6,
|
---|
209 | { border: 1, borderColor: '#DFE3E8', borderRadius: 5, padding: 4 },
|
---|
210 | ]}
|
---|
211 | >
|
---|
212 | <Text style={[styles.subtitle1, styles.mb4]}>Invoice to</Text>
|
---|
213 | <Text style={[styles.body2, { fontWeight: 700 }]}>{invoiceTo.name}</Text>
|
---|
214 | {invoiceTo.companyId && (
|
---|
215 | <Text style={[styles.body2]}>Company ID: {invoiceTo.companyId}</Text>
|
---|
216 | )}
|
---|
217 | <Text style={styles.body2}>{createFullAddress(invoiceTo.address)}</Text>
|
---|
218 | <Text style={styles.body2}>{invoiceTo.phoneNumber}</Text>
|
---|
219 | <View style={{ marginTop: 5 }} />
|
---|
220 | <Text style={{ ...styles.body2, textTransform: 'lowercase' }}>{invoiceTo.email}</Text>
|
---|
221 | </View>
|
---|
222 | </View>
|
---|
223 |
|
---|
224 | <View style={[styles.gridContainer, styles.mb40]}>
|
---|
225 | <View style={styles.col6}>
|
---|
226 | <Text style={[styles.subtitle1, styles.mb4]}>Invoice Number</Text>
|
---|
227 | <Text style={styles.body2}>{invoiceNumber}</Text>
|
---|
228 | </View>
|
---|
229 | <View style={styles.col6}>
|
---|
230 | <Text style={[styles.subtitle1, styles.mb4]}>Date Issued</Text>
|
---|
231 | <Text style={styles.body2}>{fDate(issueDate)}</Text>
|
---|
232 | </View>
|
---|
233 | <View style={styles.col6}>
|
---|
234 | <Text style={[styles.subtitle1, styles.mb4]}>Due date</Text>
|
---|
235 | <Text style={styles.body2}>{fDate(dueDate)}</Text>
|
---|
236 | </View>
|
---|
237 | </View>
|
---|
238 |
|
---|
239 | <Text style={[styles.subtitle1, styles.mb8]}>Invoice Details</Text>
|
---|
240 |
|
---|
241 | <View style={styles.table}>
|
---|
242 | <View>
|
---|
243 | <View style={[styles.tableHead, { backgroundColor: 'black', color: 'white' }]}>
|
---|
244 | <View style={styles.tableCell_1}>
|
---|
245 | <Text style={styles.subtitle1}>#</Text>
|
---|
246 | </View>
|
---|
247 |
|
---|
248 | <View style={styles.tableCell_2}>
|
---|
249 | <Text style={styles.subtitle1}>Description</Text>
|
---|
250 | </View>
|
---|
251 |
|
---|
252 | <View style={styles.tableCell_3}>
|
---|
253 | <Text style={styles.subtitle1}>{getQuantityType(invoice)}</Text>
|
---|
254 | </View>
|
---|
255 |
|
---|
256 | <View style={styles.tableCell_3}>
|
---|
257 | <Text style={styles.subtitle1}>Unit price</Text>
|
---|
258 | </View>
|
---|
259 |
|
---|
260 | <View style={styles.tableCell_3}>
|
---|
261 | <Text style={styles.subtitle1}>VAT</Text>
|
---|
262 | </View>
|
---|
263 |
|
---|
264 | <View style={[styles.tableCell_3, styles.alignRight]}>
|
---|
265 | <Text style={styles.subtitle1}>Total</Text>
|
---|
266 | </View>
|
---|
267 | </View>
|
---|
268 | </View>
|
---|
269 |
|
---|
270 | <View>
|
---|
271 | {items.map((item, index) => (
|
---|
272 | <View style={styles.tableRow} key={item.title}>
|
---|
273 | <View style={styles.tableCell_1}>
|
---|
274 | <Text>{index + 1}</Text>
|
---|
275 | </View>
|
---|
276 |
|
---|
277 | <View style={styles.tableCell_2}>
|
---|
278 | <Text style={styles.subtitle2}>{item.title}</Text>
|
---|
279 | <Text>{item.description}</Text>
|
---|
280 | </View>
|
---|
281 |
|
---|
282 | <View style={styles.tableCell_3}>
|
---|
283 | <Text>{item.quantity}</Text>
|
---|
284 | </View>
|
---|
285 |
|
---|
286 | <View style={styles.tableCell_3}>
|
---|
287 | <Text>{item.price}</Text>
|
---|
288 | </View>
|
---|
289 |
|
---|
290 | <View style={styles.tableCell_3}>
|
---|
291 | <Text>0% [1]</Text>
|
---|
292 | </View>
|
---|
293 |
|
---|
294 | <View style={[styles.tableCell_3, styles.alignRight]}>
|
---|
295 | <Text>{fCurrency(item.price * item.quantity, currency)}</Text>
|
---|
296 | </View>
|
---|
297 | </View>
|
---|
298 | ))}
|
---|
299 |
|
---|
300 | <View style={[styles.tableRow, styles.noBorder]}>
|
---|
301 | <View style={styles.tableCell_1} />
|
---|
302 | <View style={styles.tableCell_2} />
|
---|
303 | <View style={styles.tableCell_3} />
|
---|
304 | <View style={styles.tableCell_3}>
|
---|
305 | <Text>Subtotal</Text>
|
---|
306 | </View>
|
---|
307 | <View style={[styles.tableCell_3, styles.alignRight]}>
|
---|
308 | <Text>{fCurrency(subTotal, currency)}</Text>
|
---|
309 | </View>
|
---|
310 | </View>
|
---|
311 |
|
---|
312 | <View style={[styles.tableRow, styles.noBorder]}>
|
---|
313 | <View style={styles.tableCell_1} />
|
---|
314 | <View style={styles.tableCell_2} />
|
---|
315 | <View style={styles.tableCell_3} />
|
---|
316 | <View style={styles.tableCell_3}>
|
---|
317 | <Text>VAT [1]</Text>
|
---|
318 | </View>
|
---|
319 | <View style={[styles.tableCell_3, styles.alignRight]}>
|
---|
320 | <Text>0,00</Text>
|
---|
321 | </View>
|
---|
322 | </View>
|
---|
323 |
|
---|
324 | {!!discount && (
|
---|
325 | <View style={[styles.tableRow, styles.noBorder]}>
|
---|
326 | <View style={styles.tableCell_1} />
|
---|
327 | <View style={styles.tableCell_2} />
|
---|
328 | <View style={styles.tableCell_3} />
|
---|
329 | <View style={styles.tableCell_3}>
|
---|
330 | <Text>Discount</Text>
|
---|
331 | </View>
|
---|
332 | <View style={[styles.tableCell_3, styles.alignRight]}>
|
---|
333 | <Text>{fCurrency(-discount, currency)}</Text>
|
---|
334 | </View>
|
---|
335 | </View>
|
---|
336 | )}
|
---|
337 |
|
---|
338 | {/* <View style={[styles.tableRow, styles.noBorder]}>
|
---|
339 | <View style={styles.tableCell_1} />
|
---|
340 | <View style={styles.tableCell_2} />
|
---|
341 | <View style={styles.tableCell_3} />
|
---|
342 | <View style={styles.tableCell_3}>
|
---|
343 | <Text>Taxes</Text>
|
---|
344 | </View>
|
---|
345 | <View style={[styles.tableCell_3, styles.alignRight]}>
|
---|
346 | <Text>{fCurrency(taxes)}</Text>
|
---|
347 | </View>
|
---|
348 | </View> */}
|
---|
349 |
|
---|
350 | <View style={[styles.tableRow, styles.noBorder]}>
|
---|
351 | <View style={styles.tableCell_1} />
|
---|
352 | <View style={styles.tableCell_2} />
|
---|
353 | <View style={styles.tableCell_3} />
|
---|
354 | <View style={styles.tableCell_3}>
|
---|
355 | <Text style={styles.h4}>Total</Text>
|
---|
356 | </View>
|
---|
357 | <View style={[styles.tableCell_3, styles.alignRight]}>
|
---|
358 | <Text style={styles.h4}>{fCurrency(totalAmount, currency)}</Text>
|
---|
359 | </View>
|
---|
360 | </View>
|
---|
361 | </View>
|
---|
362 | </View>
|
---|
363 |
|
---|
364 | <View style={[styles.gridContainer, styles.footer]} fixed>
|
---|
365 | <View style={styles.col8}>
|
---|
366 | <Text style={styles.subtitle2}>VAT references</Text>
|
---|
367 | <Text>[1] - Reverse charge 0%</Text>
|
---|
368 | </View>
|
---|
369 | <View style={[styles.col4, styles.alignRight]}>
|
---|
370 | <Text style={[styles.subtitle2]}>Have a question?</Text>
|
---|
371 | <Text style={{ textTransform: 'lowercase' }}>{invoiceFrom.email}</Text>
|
---|
372 | </View>
|
---|
373 | </View>
|
---|
374 | </Page>
|
---|
375 | </Document>
|
---|
376 | );
|
---|
377 | }
|
---|