--- title: Build a print UI extension description: >- Learn how to build a POS print UI extension that generates, previews, and prints custom documents. source_url: html: >- https://shopify.dev/docs/apps/build/pos/build-print-extension?extension=polaris md: >- https://shopify.dev/docs/apps/build/pos/build-print-extension.md?extension=polaris --- # Build a print UI extension This tutorial shows you how to create a print extension that lets merchants generate and preview documents before printing. ## What you'll learn: * Create a backend service that serves print-ready documents. * Build a POS UI extension with preview and print capabilities. * Implement error handling for a reliable printing experience. * Test your extension in a development environment.  ## Requirements [Create a Partner account](https://www.shopify.com/partners) [Create a development store](https://shopify.dev/docs/apps/tools/development-stores#create-a-development-store-to-test-your-app) [Scaffold an app](https://shopify.dev/docs/apps/build/scaffold-app) Scaffold an app that uses the [React Router template](https://github.com/Shopify/shopify-app-template-react-router). [Generate a POS UI extension](https://shopify.dev/docs/apps/build/pos/getting-started) Generate a POS UI extension using Shopify CLI. Access scopes If you plan to include order, customer, or any other Shopify admin data in your prints, you'll need additional access scopes. This tutorial doesn't include any admin data as the core focus is to build the extension itself, so no additional access scopes are required. [Learn more about access scopes](https://shopify.dev/docs/api/usage/access-scopes) ## Project Polaris ### Create a route to serve printable documents First, create a new route file at `app/routes/print.ts` that serves your printable documents. This example uses [React Router](https://reactrouter.com/start/framework/routing#route-modules), but you can adapt the concepts to your preferred framework. > Need help setting up a React Router server? Check out the [Shopify App React Router documentation](https://shopify.dev/docs/api/shopify-app-react-router). Let's walk through each part of the implementation: Set up the route and handle authentication with Shopify. ## /packages/ui-extensions/docs/surfaces/point-of-sale/mdxExamples/print-example/print.ts ```typescript // This example shows how to create a print endpoint for your Shopify app // This works with both Remix and React Router v7 (which uses Remix's API patterns) // Place this file in your app's routes folder to create a route at '/print' import {authenticate} from '../shopify.server'; export async function loader({request}) { const {cors} = await authenticate.admin(request); const url = new URL(request.url); const printTypes = url.searchParams.get('printTypes')?.split(',') || []; const pages = printTypes.map((type) => createPage(type)); const print = printHTML(pages); return cors( new Response(print, { status: 200, headers: { 'Content-type': 'text/html', }, }), ); } // Helper function to create document pages based on type function createPage(type) { const email = 'customerhelp@example.com'; // Get document content based on type (invoice, packing slip, etc.) const getDocumentInfo = () => { switch (type) { case 'invoice': return { label: 'Receipt / Invoice', content: ` ``` Process URL parameters to determine which documents to generate. ## /packages/ui-extensions/docs/surfaces/point-of-sale/mdxExamples/print-example/print.ts ```typescript // This example shows how to create a print endpoint for your Shopify app // This works with both Remix and React Router v7 (which uses Remix's API patterns) // Place this file in your app's routes folder to create a route at '/print' import {authenticate} from '../shopify.server'; export async function loader({request}) { const {cors} = await authenticate.admin(request); const url = new URL(request.url); const printTypes = url.searchParams.get('printTypes')?.split(',') || []; const pages = printTypes.map((type) => createPage(type)); const print = printHTML(pages); return cors( new Response(print, { status: 200, headers: { 'Content-type': 'text/html', }, }), ); } // Helper function to create document pages based on type function createPage(type) { const email = 'customerhelp@example.com'; // Get document content based on type (invoice, packing slip, etc.) const getDocumentInfo = () => { switch (type) { case 'invoice': return { label: 'Receipt / Invoice', content: ` ``` Generate HTML content with proper styling for each document type. ## /packages/ui-extensions/docs/surfaces/point-of-sale/mdxExamples/print-example/print.ts ```typescript // This example shows how to create a print endpoint for your Shopify app // This works with both Remix and React Router v7 (which uses Remix's API patterns) // Place this file in your app's routes folder to create a route at '/print' import {authenticate} from '../shopify.server'; export async function loader({request}) { const {cors} = await authenticate.admin(request); const url = new URL(request.url); const printTypes = url.searchParams.get('printTypes')?.split(',') || []; const pages = printTypes.map((type) => createPage(type)); const print = printHTML(pages); return cors( new Response(print, { status: 200, headers: { 'Content-type': 'text/html', }, }), ); } // Helper function to create document pages based on type function createPage(type) { const email = 'customerhelp@example.com'; // Get document content based on type (invoice, packing slip, etc.) const getDocumentInfo = () => { switch (type) { case 'invoice': return { label: 'Receipt / Invoice', content: ` ``` Return a properly formatted response with CORS headers. ## /packages/ui-extensions/docs/surfaces/point-of-sale/mdxExamples/print-example/print.ts ```typescript // This example shows how to create a print endpoint for your Shopify app // This works with both Remix and React Router v7 (which uses Remix's API patterns) // Place this file in your app's routes folder to create a route at '/print' import {authenticate} from '../shopify.server'; export async function loader({request}) { const {cors} = await authenticate.admin(request); const url = new URL(request.url); const printTypes = url.searchParams.get('printTypes')?.split(',') || []; const pages = printTypes.map((type) => createPage(type)); const print = printHTML(pages); return cors( new Response(print, { status: 200, headers: { 'Content-type': 'text/html', }, }), ); } // Helper function to create document pages based on type function createPage(type) { const email = 'customerhelp@example.com'; // Get document content based on type (invoice, packing slip, etc.) const getDocumentInfo = () => { switch (type) { case 'invoice': return { label: 'Receipt / Invoice', content: ` ``` Print document requirements * Use only static HTML and CSS - JavaScript won't execute in print documents. * Include all styles in the `
` section or inline. * Use `@media print` CSS rules for print-specific styling. * Ensure proper CORS headers are set for POS access. Page breaks When returning multiple documents, use CSS page breaks to ensure proper printing: ```css @media print { .page-break { page-break-after: always; } } ``` Email obfuscation When using Cloudflare tunnels for development, wrap email addresses in HTML comments to prevent obfuscation: `example@email.com` [Learn more about email obfuscation](https://developers.cloudflare.com/waf/tools/scrape-shield/email-address-obfuscation/#prevent-cloudflare-from-obfuscating-email) ### Build the extension tile Next, you'll build a modal where users interact with your app. You'll add the functionality that implements the printing workflow. Build a **Tile** on the POS smart grid that launches your extension. ### Build the extension modal Next, you'll build a modal where users interact with your app. You'll add the functionality that implements the printing workflow. Initialize the loading and document list state. ## /packages/ui-extensions/docs/surfaces/point-of-sale/mdxExamples/print-example/print-modal.jsx ```jsx import {render} from 'preact'; import {useState, useEffect} from 'preact/hooks'; export default function extension() { render(