[5d6f37a] | 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
|
---|
[057453c] | 7 | import { Invoice } from 'src/schemas';
|
---|
[5d6f37a] | 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,
|
---|
[87c9f1e] | 131 | issueDate,
|
---|
[5d6f37a] | 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>
|
---|
[87c9f1e] | 231 | <Text style={styles.body2}>{fDate(issueDate)}</Text>
|
---|
[5d6f37a] | 232 | </View>
|
---|
| 233 | <View style={styles.col6}>
|
---|
| 234 | <Text style={[styles.subtitle1, styles.mb4]}>Due date</Text>
|
---|
[057453c] | 235 | <Text style={styles.body2}>{fDate(dueDate)}</Text>
|
---|
[5d6f37a] | 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 | }
|
---|