관련 지식
javascript, promise, canvas, html2canvas, polyfill, es6-promise

웹 페이지의 스크린샷을 만들기 위해 더 이상 화면캡쳐 프로그램이 필요가 없을수도 있습니다. html2canvas 는 screenshot 을 만들수 있는 매우 가벼운 라이브러리 입니다.

경로 : https://html2canvas.hertzen.com/
문서 : https://html2canvas.hertzen.com/documentation
설치 : npm, yarn, link …
버전 : v1.0.0-alpha.12

사용법은 문서에도 있지만 매우 간단합니다. 캡쳐 하고 싶은 DOM을 html2canvas() 함수의 파라미터로 전달해서 호출하면 Promise 객체를 리턴받을수 있고 그것을 통해 특정 영역을 포함한 canvas 객체를 받을수가 있습니다.

  1. <div id="capture" style="padding: 10px; background: #f5da55">
  2. <h4 style="color: #000; ">Hello world!</h4>
  3. </div>
  1. html2canvas(document.querySelector("#capture")).then(canvas => {
  2. document.body.appendChild(canvas)
  3. });

공식 예제인 위 샘플은 너무 단조롭죠. 캡쳐된 화면도 단지 화면에 보여줄 뿐입니다. 이것을 좀 응용하는 예제를 살펴 보겠습니다. 앞으로 사용할 HTML 예제는 아래와 같습니다.

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <title>CSS Template</title>
  5. <meta charset="utf-8">
  6. <meta name="viewport" content="width=device-width, initial-scale=1">
  7. <script src="https://code.jquery.com/jquery-3.3.1.min.js"></script>
  8. <script src="https://html2canvas.hertzen.com/dist/html2canvas.min.js"></script>
  9. <style>
  10. * {
  11. box-sizing: border-box;
  12. }
  13. body {
  14. font-family: Arial, Helvetica, sans-serif;
  15. }
  16. /* Style the header */
  17. header {
  18. background-color: #666;
  19. padding: 30px;
  20. text-align: center;
  21. font-size: 35px;
  22. color: white;
  23. }
  24. /* Create two columns/boxes that floats next to each other */
  25. nav {
  26. float: left;
  27. width: 30%;
  28. height: 300px; /* only for demonstration, should be removed */
  29. background: #ccc;
  30. padding: 20px;
  31. }
  32. /* Style the list inside the menu */
  33. nav ul {
  34. list-style-type: none;
  35. padding: 0;
  36. }
  37. article {
  38. float: left;
  39. padding: 20px;
  40. width: 70%;
  41. background-color: #f1f1f1;
  42. height: 300px; /* only for demonstration, should be removed */
  43. }
  44. /* Clear floats after the columns */
  45. section:after {
  46. content: "";
  47. display: table;
  48. clear: both;
  49. }
  50. /* Style the footer */
  51. footer {
  52. background-color: #777;
  53. padding: 10px;
  54. text-align: center;
  55. color: white;
  56. }
  57. /* Responsive layout - makes the two columns/boxes stack on top of each other instead of next to each other, on small screens */
  58. @media (max-width: 600px) {
  59. nav, article {
  60. width: 100%;
  61. height: auto;
  62. }
  63. }
  64. </style>
  65. </head>
  66. <body>
  67. <input type="button" value="캡쳐" />
  68. <h2>CSS Layout Float</h2>
  69. <p>In this example, we have created a header, two columns/boxes and a footer. On smaller screens, the columns will stack on top of each other.</p>
  70. <p>Resize the browser window to see the responsive effect (you will learn more about this in our next chapter - HTML Responsive.)</p>
  71. <header>
  72. <h2>Cities</h2>
  73. </header>
  74. <section>
  75. <nav>
  76. <input type="button" value="캡쳐" />
  77. <ul>
  78. <li><a href="#">London</a></li>
  79. <li><a href="#">Paris</a></li>
  80. <li><a href="#">Tokyo</a></li>
  81. </ul>
  82. </nav>
  83. <article>
  84. <h1>London</h1>
  85. <p>London is the capital city of England. It is the most populous city in the United Kingdom, with a metropolitan area of over 13 million inhabitants.</p>
  86. <p>Standing on the River Thames, London has been a major settlement for two millennia, its history going back to its founding by the Romans, who named it Londinium.</p>
  87. <input type="button" value="캡쳐" />
  88. </article>
  89. </section>
  90. <footer>
  91. <p>Footer</p>
  92. </footer>
  93. </body>
  94. </html>

버튼이 위치한 영역을 캡쳐하기

위 html에는 3개의 버튼이 있습니다. 캡쳐할 요소를 일일이 지정하는 것도 괜찮겠지만 버튼이 존재하는 영역의 DOM을 캡쳐하는 것으로 만들어보겠습니다. 화살표 함수가 익숙치 않은 분들을 위해 보통 함수 형태로 변경했습니다.

  1. $(":button").on('click', async function(e) {
  2. html2canvas(e.target.parentElement).then(function(canvas) {
  3. document.body.appendChild(canvas)
  4. });
  5. });

canvas -> image 변환

콜백함수로 전달된 변수 canvas 에는 실제로 canvas 요소가 들어있습니다. 화면에 그려진 캔버스 요소는 이미지로 저장도 되지만 일반적인 <img> 태그가 더 익숙하실 것입니다. 그렇다면 canvas에서 제공하는 api를 이용하여 이미지 태그에 사용할 수 있는 Data URL 문자열로 변환하면 됩니다.

  1. canvas.toDataURL("image/jpeg")

위 함수를 호출하면 아래와 같은 Data URL 문자열을 리턴합니다.(“image/jpeg” 외 다른 포맷도 가능)

저 문자열을 이미지 태그로 만들어보겠습니다.

  1. html2canvas(e.target.parentElement).then(function(canvas) {
  2. $('body').append('<img src="' + canvas.toDataURL("image/jpeg") + '"/>');
  3. });

이미지 파일로 다운로드 하기

미리보기 기능이 필요한 경우가 아니라면 캡쳐한 이미지를 화면에 보이고 싶은 경우 보다는 ‘파일 업로드’ 또는 ‘파일 다운로드’ 형태가 더 자주 쓰일 것입니다. 이미지 파일로 바로 다운로드 하는 방법은 약간의 꼼수만 추가하면 됩니다.

먼저 HTML 태그 하나를 추가합니다. 화면에 보일 요소가 아니므로 display 속성을 none 으로 합니다.

  1. <a id="target" style="display: none"></a>

이미 짐작하신 분도 있겠지만 위에 추가한 앵커 태그의 속성에 Data URL을 넣어주고 클릭 이벤트만 발생 시키면 됩니다.

  1. html2canvas(e.target.parentElement).then(function(canvas) {
  2. var el = document.getElementById("target");
  3. el.href = canvas.toDataURL("image/jpeg");
  4. el.download = '파일명.jpg';
  5. el.click();
  6. });

폴리필 추가하기

html2canvas는 promise 를 사용하기 때문에 프로미스를 지원하지 않는 인터넷 익스플로러에서는 사용할 수가 없습니다. 그러나 promise를 polyfill 처리를 한다면 IE9 이상에서 동작 가능하다고 나와있습니다. 한번 폴리필까지 적용해보겠습니다.

다양한 폴리필 들이 있지만 (아마도) 가장 많이 사용되는 es6-promise 를 사용하겠습니다.

경로 : https://www.npmjs.com/package/es6-promise
설치 : npm, yarn, CDN, download
버전 : 4.2.6

샘플에서는 CDN 으로 적용할 것입니다. 아래 링크를 추가합니다.

  1. <script src="https://cdn.jsdelivr.net/npm/es6-promise@4/dist/es6-promise.min.js"></script>
  2. <script src="https://cdn.jsdelivr.net/npm/es6-promise@4/dist/es6-promise.auto.min.js"></script>

그리고 익스플로러로 테스트를 해보면 이상하게도 아무런 반응이 없습니다. 개발자도구를 열어보면 콘솔 에러를 볼 수 있습니다.

링크에 Data URL을 넣고 다운로드 받는 방법은 크롬에선 이상 없지만 익스플로러에선 정상적으로 동작하지 않습니다. 따라서 다른 방법을 사용해야 합니다.

  1. if (navigator.msSaveBlob) {
  2. var blob = canvas.msToBlob();
  3. return navigator.msSaveBlob(blob, '파일명.jpg');
  4. }

정리

이상으로 웹 화면을 캡쳐해서 이미지로 저장하는 방법을 알아 보았습니다. 우리가 흔히 사용하던 캡쳐 프로그램과 달리 특정 영역을 겹쳐서 캡쳐할수 있는것은 아니지만 외부 플러그인 사용없이 웹 화면에 보이는 그대로 이미지로 내려받을 수 있었습니다.

그러나 이것도 100% 동일한 이미지로 캡쳐되진 않습니다. 캡쳐 방법이 HTML의 DOM과 CSS 속성을 캔버스에 옮겨 이미지화 시키는 것인데 일부 CSS는 지원하지 않기 때문입니다. 아래 요소로 인해 캡쳐에 문제가 있다면 다른 css를 사용하는 것을 고려하셔야 할것 같습니다.

최종 샘플

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <title>CSS Template</title>
  5. <meta charset="utf-8">
  6. <meta name="viewport" content="width=device-width, initial-scale=1">
  7. <script src="https://code.jquery.com/jquery-3.3.1.min.js"></script>
  8. <script src="https://html2canvas.hertzen.com/dist/html2canvas.min.js"></script>
  9. <script src="https://cdn.jsdelivr.net/npm/es6-promise@4/dist/es6-promise.min.js"></script>
  10. <script src="https://cdn.jsdelivr.net/npm/es6-promise@4/dist/es6-promise.auto.min.js"></script>
  11. <style>
  12. * {
  13. box-sizing: border-box;
  14. }
  15. body {
  16. font-family: Arial, Helvetica, sans-serif;
  17. }
  18. /* Style the header */
  19. header {
  20. background-color: #666;
  21. padding: 30px;
  22. text-align: center;
  23. font-size: 35px;
  24. color: white;
  25. }
  26. /* Create two columns/boxes that floats next to each other */
  27. nav {
  28. float: left;
  29. width: 30%;
  30. height: 300px; /* only for demonstration, should be removed */
  31. background: #ccc;
  32. padding: 20px;
  33. }
  34. /* Style the list inside the menu */
  35. nav ul {
  36. list-style-type: none;
  37. padding: 0;
  38. }
  39. article {
  40. float: left;
  41. padding: 20px;
  42. width: 70%;
  43. background-color: #f1f1f1;
  44. height: 300px; /* only for demonstration, should be removed */
  45. }
  46. /* Clear floats after the columns */
  47. section:after {
  48. content: "";
  49. display: table;
  50. clear: both;
  51. }
  52. /* Style the footer */
  53. footer {
  54. background-color: #777;
  55. padding: 10px;
  56. text-align: center;
  57. color: white;
  58. }
  59. /* Responsive layout - makes the two columns/boxes stack on top of each other instead of next to each other, on small screens */
  60. @media (max-width: 600px) {
  61. nav, article {
  62. width: 100%;
  63. height: auto;
  64. }
  65. }
  66. </style>
  67. </head>
  68. <body>
  69. <input type="button" value="캡쳐" />
  70. <h2>CSS Layout Float</h2>
  71. <p>In this example, we have created a header, two columns/boxes and a footer. On smaller screens, the columns will stack on top of each other.</p>
  72. <p>Resize the browser window to see the responsive effect (you will learn more about this in our next chapter - HTML Responsive.)</p>
  73. <header>
  74. <h2>Cities</h2>
  75. </header>
  76. <section>
  77. <nav>
  78. <input type="button" value="캡쳐" />
  79. <ul>
  80. <li><a href="#">London</a></li>
  81. <li><a href="#">Paris</a></li>
  82. <li><a href="#">Tokyo</a></li>
  83. </ul>
  84. </nav>
  85. <article>
  86. <h1>London</h1>
  87. <p>London is the capital city of England. It is the most populous city in the United Kingdom, with a metropolitan area of over 13 million inhabitants.</p>
  88. <p>Standing on the River Thames, London has been a major settlement for two millennia, its history going back to its founding by the Romans, who named it Londinium.</p>
  89. <input type="button" value="캡쳐" />
  90. </article>
  91. </section>
  92. <footer>
  93. <p>Footer</p>
  94. </footer>
  95. <a id="target" style="display: none"></a>
  96. <script>
  97. $(":button").on('click', function(e) {
  98. // html2canvas(e.target.parentElement).then(function(canvas) {
  99. // document.body.appendChild(canvas)
  100. // });
  101. // html2canvas(e.target.parentElement).then(function(canvas) {
  102. // $('body').append('<img src="' + canvas.toDataURL("image/jpeg") + '"/>');
  103. // });
  104. html2canvas(e.target.parentElement).then(function(canvas) {
  105. if (navigator.msSaveBlob) {
  106. var blob = canvas.msToBlob();
  107. return navigator.msSaveBlob(blob, '파일명.jpg');
  108. } else {
  109. var el = document.getElementById("target");
  110. el.href = canvas.toDataURL("image/jpeg");
  111. el.download = '파일명.jpg';
  112. el.click();
  113. }
  114. });
  115. });
  116. </script>
  117. </body>
  118. </html>