<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>Forem: Manoj Vishwakarma</title>
    <description>The latest articles on Forem by Manoj Vishwakarma (@vjmanoj).</description>
    <link>https://forem.com/vjmanoj</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3804742%2Fc79a2a67-1f20-42bc-a619-2a4341c77a82.jpg</url>
      <title>Forem: Manoj Vishwakarma</title>
      <link>https://forem.com/vjmanoj</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/vjmanoj"/>
    <language>en</language>
    <item>
      <title>The CMYK Problem Nobody Warns You About When Building a PDF Editor in the Browser</title>
      <dc:creator>Manoj Vishwakarma</dc:creator>
      <pubDate>Tue, 03 Mar 2026 22:55:11 +0000</pubDate>
      <link>https://forem.com/vjmanoj/the-cmyk-problem-nobody-warns-you-about-when-building-a-pdf-editor-in-the-browser-3a02</link>
      <guid>https://forem.com/vjmanoj/the-cmyk-problem-nobody-warns-you-about-when-building-a-pdf-editor-in-the-browser-3a02</guid>
      <description>&lt;p&gt;I spent three weeks building a browser-based PDF editor before I realized something was quietly wrong with every color in my application.&lt;/p&gt;

&lt;p&gt;I loaded a PDF with a specific shade of teal, defined in CMYK for print. It looked fine on the canvas. I picked that color with the eyedropper, applied it to some text, exported the PDF, and compared the output.&lt;/p&gt;

&lt;p&gt;The colors didn't match.&lt;/p&gt;

&lt;p&gt;Not dramatically off. Just enough to notice when you compared them side by side. That was the start of a rabbit hole I didn't expect.&lt;/p&gt;

&lt;h2&gt;
  
  
  What Actually Happens When You Render a CMYK PDF in the Browser
&lt;/h2&gt;

&lt;p&gt;Most of us reach for pdf.js when we need to display PDFs in the browser. It works great. But here is something that is easy to miss: browsers only understand sRGB. They have no concept of CMYK color at all.&lt;/p&gt;

&lt;p&gt;So when pdf.js encounters a CMYK color value in a PDF, it converts it to RGB for rendering. That conversion is lossy. The RGB value it produces is an approximation, good enough for a screen preview, but it's not the original color anymore.&lt;/p&gt;

&lt;p&gt;Here is where it gets tricky. If your user picks that color from the canvas (using the EyeDropper API or a color input), they get the converted RGB value. If you then use that RGB value when exporting back to PDF, you have lost the original CMYK data entirely.&lt;/p&gt;

&lt;p&gt;The color round-trips through two lossy conversions: CMYK to RGB (by pdf.js for display) and then RGB back to CMYK (by your export code). Each step introduces drift.&lt;/p&gt;

&lt;h2&gt;
  
  
  Why You Can't Just Convert Back
&lt;/h2&gt;

&lt;p&gt;My first instinct was simple. Just convert the sampled RGB back to CMYK and call it a day.&lt;/p&gt;

&lt;p&gt;I tried that. The problem is that RGB to CMYK conversion is not the inverse of CMYK to RGB. The color spaces don't map one-to-one. Different CMYK values can produce the same RGB color on screen, and going back from that RGB value gives you a different CMYK than the one you started with.&lt;/p&gt;

&lt;p&gt;On top of that, professional print work relies on specific CMYK values for brand colors. Pantone 320 C is Pantone 320 C. You can't just give the printer "something close" and hope it works.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Dual Representation Approach
&lt;/h2&gt;

&lt;p&gt;The solution I landed on after some research was to maintain two color values for every color in the editor, side by side.&lt;/p&gt;

&lt;p&gt;One is the &lt;strong&gt;source color&lt;/strong&gt;: the original CMYK (or RGB) value from the PDF, preserved exactly as it was. This is what you use when exporting.&lt;/p&gt;

&lt;p&gt;The other is the &lt;strong&gt;preview color&lt;/strong&gt;: the sRGB approximation that the browser can actually display. This is what you show on the canvas and in color pickers.&lt;/p&gt;

&lt;p&gt;The two stay linked together throughout the editing pipeline. When the user picks a color, you don't just store the hex value. You look up which source color that preview corresponds to, and you keep both.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="c1"&gt;// Instead of storing just this:&lt;/span&gt;
&lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;color&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;#1dc4e2&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="c1"&gt;// You store both the preview and the original source:&lt;/span&gt;
&lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;color&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;#1dc4e2&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;colorSource&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;type&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;cmyk&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;c&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.11&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;m&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.24&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;y&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;k&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.13&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;When it's time to export, you check if there's a source color. If it's CMYK, you write those exact values into the PDF. No conversion, no drift.&lt;/p&gt;

&lt;h2&gt;
  
  
  Palette Snapping
&lt;/h2&gt;

&lt;p&gt;There is still a practical problem though. When a user samples a color from the canvas, you get an RGB hex value. How do you figure out which original CMYK color it came from?&lt;/p&gt;

&lt;p&gt;The approach I use is palette snapping. You take all the original colors from the PDF template and build a palette with both their source values and their RGB preview equivalents. When the user picks a color, you compare it against the palette and snap to the nearest match if it's close enough.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;palette&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;normalizePalette&lt;/span&gt;&lt;span class="p"&gt;([&lt;/span&gt;
  &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;source&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;type&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;cmyk&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;c&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.11&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;m&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.24&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;y&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;k&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.13&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
  &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;source&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="na"&gt;type&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;cmyk&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;c&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;m&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;y&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;0.0&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="na"&gt;k&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="mf"&gt;1.0&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;},&lt;/span&gt;
&lt;span class="p"&gt;]);&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;nearest&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;findNearestPaletteEntry&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;palette&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;pickedHex&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;nearest&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;entry&lt;/span&gt; &lt;span class="o"&gt;&amp;amp;&amp;amp;&lt;/span&gt; &lt;span class="nf"&gt;shouldSnapToPalette&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;nearest&lt;/span&gt;&lt;span class="p"&gt;))&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// Snap: use the original CMYK source&lt;/span&gt;
  &lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;colorSource&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;nearest&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;entry&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;source&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
  &lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;color&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;nearest&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;entry&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;previewHex&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;else&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// Custom color: no CMYK source available&lt;/span&gt;
  &lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;color&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;pickedHex&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
  &lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;colorSource&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="kc"&gt;null&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;"Close enough" is determined by a distance metric. Simple RGB Euclidean distance works for most cases, but if you want perceptual accuracy, CIE76 Delta-E is better because it accounts for how humans actually perceive color differences.&lt;/p&gt;

&lt;h2&gt;
  
  
  Why Not Just Use a Full ICC Pipeline?
&lt;/h2&gt;

&lt;p&gt;The proper, correct solution for color management is ICC profiles. Tools like MuPDF and PDFium do this right. They load the ICC profile embedded in the PDF and perform a proper color space transformation.&lt;/p&gt;

&lt;p&gt;But those tools are 5 to 15 megabytes of WASM. They need ICC profile files bundled with your app. They require a custom rendering pipeline instead of using the browser's native canvas. For many browser-based PDF editors, that's too heavy.&lt;/p&gt;

&lt;p&gt;The dual representation approach is a pragmatic middle ground. It's a few kilobytes, runs in pure JavaScript, and preserves CMYK fidelity for the common case where users are working with a fixed set of template colors. It won't handle arbitrary ICC profiles or spot colors, but for most web-to-print workflows, it does the job.&lt;/p&gt;

&lt;h2&gt;
  
  
  What I Built
&lt;/h2&gt;

&lt;p&gt;I packaged this approach into a small TypeScript library called &lt;a href="https://www.npmjs.com/package/cmyk-preview-toolkit" rel="noopener noreferrer"&gt;cmyk-preview-toolkit&lt;/a&gt;. It handles:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;CMYK to RGB conversion using the same polynomial that pdf.js uses internally&lt;/li&gt;
&lt;li&gt;RGB to CMYK reverse conversion with GCR (Grey Component Replacement)&lt;/li&gt;
&lt;li&gt;HSL and CIE Lab color space conversions&lt;/li&gt;
&lt;li&gt;Palette building and nearest-match snapping&lt;/li&gt;
&lt;li&gt;Immutable state helpers for managing dual color representations&lt;/li&gt;
&lt;li&gt;A React hook if you're building with React&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;It has zero runtime dependencies and ships as both ESM and CommonJS. The whole thing tree-shakes down to a few kilobytes.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;npm &lt;span class="nb"&gt;install &lt;/span&gt;cmyk-preview-toolkit
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Here is what the export step looks like with pdf-lib:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight typescript"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;cmyk&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;rgb&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;from&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;pdf-lib&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;source&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;colorSource&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;if &lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;source&lt;/span&gt;&lt;span class="p"&gt;?.&lt;/span&gt;&lt;span class="kd"&gt;type&lt;/span&gt; &lt;span class="o"&gt;===&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;cmyk&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// Use the preserved CMYK values directly&lt;/span&gt;
  &lt;span class="nx"&gt;page&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;drawText&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;Hello&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="na"&gt;color&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nf"&gt;cmyk&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;source&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;c&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;source&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;m&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;source&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;y&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;source&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;k&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;});&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="k"&gt;else&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="c1"&gt;// Fall back to RGB&lt;/span&gt;
  &lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="nx"&gt;r&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;g&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;b&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;hexToRgb&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;element&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nx"&gt;color&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
  &lt;span class="nx"&gt;page&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;drawText&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;Hello&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="na"&gt;color&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="nf"&gt;rgb&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nx"&gt;r&lt;/span&gt; &lt;span class="o"&gt;/&lt;/span&gt; &lt;span class="mi"&gt;255&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;g&lt;/span&gt; &lt;span class="o"&gt;/&lt;/span&gt; &lt;span class="mi"&gt;255&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nx"&gt;b&lt;/span&gt; &lt;span class="o"&gt;/&lt;/span&gt; &lt;span class="mi"&gt;255&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="p"&gt;});&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Lessons Learned
&lt;/h2&gt;

&lt;p&gt;A few things I picked up along the way that might save someone else some time:&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Test with print, not just screens.&lt;/strong&gt; A color that looks identical on two monitors can print very differently. If your users are sending PDFs to commercial printers, you need to verify with actual printed output.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;The EyeDropper API returns sRGB.&lt;/strong&gt; There is no way around this. Browser APIs operate in sRGB. Any solution for preserving CMYK has to work around this constraint, not fight it.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Perceptual distance matters.&lt;/strong&gt; Two colors can be mathematically close in RGB but look obviously different to a human eye, and the reverse is also true. CIE76 Delta-E is not perfect, but it's a big improvement over raw RGB distance for palette snapping decisions.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Keep it simple until you can't.&lt;/strong&gt; Full ICC color management is the correct answer in theory. But if your users are working with a known set of template colors (which is the case for most web-to-print apps), the dual representation approach gives you 95% of the benefit at 5% of the complexity.&lt;/p&gt;

&lt;p&gt;If you are building anything that touches CMYK PDFs in the browser, I hope this saves you the same three weeks of confusion it cost me.&lt;/p&gt;

&lt;p&gt;The source code is on &lt;a href="https://github.com/vjmanoj/cmyk-preview-toolkit" rel="noopener noreferrer"&gt;GitHub&lt;/a&gt; if you want to look under the hood or contribute.&lt;/p&gt;

</description>
      <category>javascript</category>
      <category>pdf</category>
      <category>cmyk</category>
      <category>webdev</category>
    </item>
  </channel>
</rss>
