html-images-alt-minifier.js 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. /*
  2. * Copyright 2010-2020 Gildas Lormeau
  3. * contact : gildas.lormeau <at> gmail.com
  4. *
  5. * This file is part of SingleFile.
  6. *
  7. * The code in this file is free software: you can redistribute it and/or
  8. * modify it under the terms of the GNU Affero General Public License
  9. * (GNU AGPL) as published by the Free Software Foundation, either version 3
  10. * of the License, or (at your option) any later version.
  11. *
  12. * The code in this file is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
  15. * General Public License for more details.
  16. *
  17. * As additional permission under GNU AGPL version 3 section 7, you may
  18. * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
  19. * AGPL normally required by section 4, provided you include this license
  20. * notice and a URL through which recipients can access the Corresponding
  21. * Source.
  22. */
  23. import * as srcsetParser from "./../vendor/html-srcset-parser.js";
  24. const EMPTY_IMAGE = "";
  25. export {
  26. process
  27. };
  28. function process(doc) {
  29. doc.querySelectorAll("picture").forEach(pictureElement => {
  30. const imgElement = pictureElement.querySelector("img");
  31. if (imgElement) {
  32. let { src, srcset } = getImgSrcData(imgElement);
  33. if (!src) {
  34. const data = getSourceSrcData(Array.from(pictureElement.querySelectorAll("source")).reverse());
  35. src = data.src;
  36. if (!srcset) {
  37. srcset = data.srcset;
  38. }
  39. }
  40. setSrc({ src, srcset }, imgElement, pictureElement);
  41. }
  42. });
  43. doc.querySelectorAll(":not(picture) > img[srcset]").forEach(imgElement => setSrc(getImgSrcData(imgElement), imgElement));
  44. }
  45. function getImgSrcData(imgElement) {
  46. let src = imgElement.getAttribute("src");
  47. if (src == EMPTY_IMAGE) {
  48. src = null;
  49. }
  50. let srcset = getSourceSrc(imgElement.getAttribute("srcset"));
  51. if (srcset == EMPTY_IMAGE) {
  52. srcset = null;
  53. }
  54. return { src, srcset };
  55. }
  56. function getSourceSrcData(sources) {
  57. let source = sources.find(source => source.src);
  58. let src = source && source.src;
  59. let srcset = source && source.srcset;
  60. if (!src) {
  61. source = sources.find(source => getSourceSrc(source.src));
  62. src = source && source.src;
  63. if (src == EMPTY_IMAGE) {
  64. src = null;
  65. }
  66. }
  67. if (!srcset) {
  68. source = sources.find(source => getSourceSrc(source.srcset));
  69. srcset = source && source.srcset;
  70. if (srcset == EMPTY_IMAGE) {
  71. srcset = null;
  72. }
  73. }
  74. return { src, srcset };
  75. }
  76. function setSrc(srcData, imgElement, pictureElement) {
  77. if (srcData.src) {
  78. imgElement.setAttribute("src", srcData.src);
  79. imgElement.setAttribute("srcset", "");
  80. imgElement.setAttribute("sizes", "");
  81. } else {
  82. imgElement.setAttribute("src", EMPTY_IMAGE);
  83. if (srcData.srcset) {
  84. imgElement.setAttribute("srcset", srcData.srcset);
  85. } else {
  86. imgElement.setAttribute("srcset", "");
  87. imgElement.setAttribute("sizes", "");
  88. }
  89. }
  90. if (pictureElement) {
  91. pictureElement.querySelectorAll("source").forEach(sourceElement => sourceElement.remove());
  92. }
  93. }
  94. function getSourceSrc(sourceSrcSet) {
  95. if (sourceSrcSet) {
  96. const srcset = srcsetParser.process(sourceSrcSet);
  97. if (srcset.length) {
  98. return (srcset.find(srcset => srcset.url)).url;
  99. }
  100. }
  101. }