UNPKG

@pdfme/schemas

Version:

TypeScript base PDF generator and React base UI. Open source, developed by the community, and completely free to use under the MIT license!

170 lines 10.5 kB
import { px2mm } from '@pdfme/common'; import { Image } from 'lucide'; import { convertForPdfLayoutProps, addAlphaToHex, isEditable, readFile, createSvgStr, } from '../utils.js'; import { DEFAULT_OPACITY } from '../constants.js'; import { getImageDimension } from './imagehelper.js'; const getCacheKey = (schema, input) => `${schema.type}${input}`; const fullSize = { width: '100%', height: '100%' }; const defaultValue = ''; const imageSchema = { pdf: async (arg) => { const { value, schema, pdfDoc, page, _cache } = arg; if (!value) return; const inputImageCacheKey = getCacheKey(schema, value); let image = _cache.get(inputImageCacheKey); if (!image) { const isPng = value.startsWith('data:image/png;'); image = await (isPng ? pdfDoc.embedPng(value) : pdfDoc.embedJpg(value)); _cache.set(inputImageCacheKey, image); } const _schema = { ...schema, position: { ...schema.position } }; const dimension = getImageDimension(value); const imageWidth = px2mm(dimension.width); const imageHeight = px2mm(dimension.height); const boxWidth = _schema.width; const boxHeight = _schema.height; const imageRatio = imageWidth / imageHeight; const boxRatio = boxWidth / boxHeight; if (imageRatio > boxRatio) { _schema.width = boxWidth; _schema.height = boxWidth / imageRatio; _schema.position.y += (boxHeight - _schema.height) / 2; } else { _schema.width = boxHeight * imageRatio; _schema.height = boxHeight; _schema.position.x += (boxWidth - _schema.width) / 2; } const pageHeight = page.getHeight(); const lProps = convertForPdfLayoutProps({ schema: _schema, pageHeight }); const { width, height, rotate, position, opacity } = lProps; const { x, y } = position; const drawOptions = { x, y, rotate, width, height, opacity }; page.drawImage(image, drawOptions); }, ui: (arg) => { const { value, rootElement, mode, onChange, stopEditing, tabIndex, placeholder, theme, schema, } = arg; const editable = isEditable(mode, schema); const isDefault = value === defaultValue; const container = document.createElement('div'); const backgroundStyle = placeholder ? `url(${placeholder})` : 'none'; const containerStyle = { ...fullSize, backgroundImage: value ? 'none' : backgroundStyle, backgroundSize: `contain`, backgroundRepeat: 'no-repeat', backgroundPosition: 'center', }; Object.assign(container.style, containerStyle); container.addEventListener('click', (e) => { if (editable) { e.stopPropagation(); } }); rootElement.appendChild(container); // image tag if (value) { const img = document.createElement('img'); const imgStyle = { height: '100%', width: '100%', borderRadius: 0, objectFit: 'contain', }; Object.assign(img.style, imgStyle); img.src = value; container.appendChild(img); } // remove button if (value && !isDefault && editable) { const button = document.createElement('button'); button.textContent = 'x'; const buttonStyle = { position: 'absolute', top: 0, left: 0, zIndex: 1, display: 'flex', justifyContent: 'center', alignItems: 'center', color: '#333', background: '#f2f2f2', borderRadius: '2px', border: '1px solid #767676', cursor: 'pointer', height: '24px', width: '24px', }; Object.assign(button.style, buttonStyle); button.addEventListener('click', () => { if (onChange) onChange({ key: 'content', value: '' }); }); container.appendChild(button); } // file input if ((!value || isDefault) && editable) { const label = document.createElement('label'); const labelStyle = { ...fullSize, display: editable ? 'flex' : 'none', position: 'absolute', top: 0, backgroundColor: editable || value ? addAlphaToHex(theme.colorPrimaryBg, 30) : 'none', cursor: 'pointer', }; Object.assign(label.style, labelStyle); container.appendChild(label); const input = document.createElement('input'); const inputStyle = { ...fullSize, position: 'absolute', top: '50%', left: '50%', width: '180px', height: '30px', marginLeft: '-90px', marginTop: '-15px', }; Object.assign(input.style, inputStyle); input.tabIndex = tabIndex || 0; input.type = 'file'; input.accept = 'image/jpeg, image/png'; input.addEventListener('change', (event) => { const changeEvent = event; readFile(changeEvent.target.files) .then((result) => { if (onChange) onChange({ key: 'content', value: result }); }) .catch((error) => { console.error('Error reading file:', error); }); }); input.addEventListener('blur', () => { if (stopEditing) stopEditing(); }); label.appendChild(input); } }, propPanel: { schema: {}, defaultSchema: { name: '', type: 'image', content: defaultValue, position: { x: 0, y: 0 }, width: 40, height: 40, // If the value of "rotate" is set to undefined or not set at all, rotation will be disabled in the UI. // Check this document: https://pdfme.com//docs/custom-schemas#learning-how-to-create-from-pdfmeschemas-code rotate: 0, opacity: DEFAULT_OPACITY, }, }, icon: createSvgStr(Image), }; export default imageSchema; //# sourceMappingURL=image.js.map