uiMediaFullpage.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. $(() => {
  2. let fullPageMediaDisplayed = false;
  3. let fullPageMediaList = false;
  4. function serializeFileSize(size) {
  5. let units = [ 'o', 'Ko', 'Mo', 'Go', 'To' ];
  6. let idx = 0;
  7. while (size >= 800 && idx < units.length) {
  8. ++idx;
  9. size /= 1024;
  10. }
  11. size = Math.floor(size * 100) / 100;
  12. return `${size} ${units[idx]}`;
  13. }
  14. function displayMeta(key, value, isRo) {
  15. let li = document.createElement("li");
  16. let type = value?.type || null;
  17. let val = (value?.value !== undefined ? value.value : value);
  18. if (!val && val !== '')
  19. return null;
  20. if (type === 'date')
  21. val = val.toLocaleString();
  22. if (["dimension", "height", "width"].indexOf(key) >= 0)
  23. val += ' px';
  24. if (["fileSize"].indexOf(key) >= 0)
  25. val = serializeFileSize(val);
  26. if (key == 'fNumber')
  27. val = `f/ ${val}`;
  28. let keyTranslate = {
  29. fileSize: "File Size",
  30. photochamberImport: "Photochamber Imported",
  31. lensModel: "Lens Model",
  32. exposureTimeStr: "Exposure Time"
  33. };
  34. let keySpan = document.createElement('span');
  35. let valSpan = document.createElement('div');
  36. li.classList.add("row");
  37. keySpan.className = "metaKey col-xl-12 col-4";
  38. valSpan.className = "metaVal col-xl-12 col-8";
  39. let inputGroup = document.createElement('form');
  40. valSpan.appendChild(inputGroup);
  41. inputGroup.classList.add('input-group');
  42. keySpan.innerText = (keyTranslate[key] || (key[0].toUpperCase()+key.substr(1))) + ':';
  43. let valInput = document.createElement("input");
  44. valInput.classList.add("form-control");
  45. valInput.value = val;
  46. valInput.disabled = isRo;
  47. valInput.addEventListener('keyup', evt => evt.stopPropagation());
  48. valInput.addEventListener('keydown', evt => evt.stopPropagation());
  49. inputGroup.appendChild(valInput);
  50. if (!isRo) {
  51. let bt = document.createElement('button');
  52. bt.className = 'btn btn-outline-secondary';
  53. bt.type = 'button';
  54. bt.innerHTML = '<i class="bi bi-pen"></i>';
  55. inputGroup.appendChild(bt);
  56. bt.addEventListener('click', () => MediaStorage.Instance.setMetaValue(fullPageMediaDisplayed?.fixedSum || fullPageMediaList, key, valInput.value));
  57. inputGroup.addEventListener('submit', evt => {
  58. evt.preventDefault();
  59. MediaStorage.Instance.setMetaValue(fullPageMediaDisplayed?.fixedSum || fullPageMediaList, key, valInput.value);
  60. });
  61. }
  62. li.appendChild(keySpan);
  63. li.appendChild(valSpan);
  64. return li;
  65. }
  66. function displayMap(geo) {
  67. let jsonGeo = geo;
  68. try {
  69. geo = JSON.parse(geo);
  70. }
  71. catch(err) { return null; }
  72. let outerHTML = document.createElement("li");
  73. outerHTML.className = "row";
  74. let container = document.createElement("div");
  75. container.className = "leaflet-container container";
  76. outerHTML.appendChild(container);
  77. let innerHTML = document.createElement("div");
  78. container.appendChild(innerHTML);
  79. let map = L.map(innerHTML, { scrollWheelZoom: false }).setView(geo, 13);
  80. L.tileLayer('https://tile.openstreetmap.org/{z}/{x}/{y}.png', {
  81. attribution: '&copy; <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors'
  82. }).addTo(map);
  83. L.marker(geo).addTo(map);
  84. setTimeout(function () {
  85. map.invalidateSize();
  86. }, 0);
  87. let a = document.createElement("a");
  88. a.href = `https://www.openstreetmap.org/?mlat=${geo[0]}&mlon=${geo[1]}`;
  89. a.target = "_blank";
  90. a.innerHTML = jsonGeo;
  91. container.appendChild(a);
  92. return outerHTML;
  93. }
  94. function displayMetas(metaData, isRo) {
  95. let metaList = document.createElement("ul");
  96. metaData.libraryPath = null;
  97. metaData.tags = metaData.fixedTags = null;
  98. if (metaData.exposureTime && metaData.exposureTimeStr)
  99. metaData.exposureTime = null;
  100. if (metaData.date && metaData.dateTime)
  101. metaData.dateTime = null;
  102. if (metaData.height && metaData.width) {
  103. metaData.dimension = { type: "string", value: `${metaData.width.value} x ${metaData.height.value}` }
  104. metaData.height = metaData.width = null;
  105. }
  106. for (let i of [ "date", "dimension", "height", "width", "fileSize" ]) {
  107. let metaItem = displayMeta(i, metaData[i], true);
  108. metaItem && metaList.appendChild(metaItem);
  109. metaData[i] = null;
  110. }
  111. for (let i of Object.keys(metaData).sort((a, b) => a.toLowerCase().localeCompare(b.toLowerCase()))) {
  112. if (i === 'gpsLocation') {
  113. let dom = displayMap(metaData[i].value);
  114. dom && metaList.appendChild(dom);
  115. continue;
  116. }
  117. let metaItem = displayMeta(i, metaData[i], isRo || !MediaStorage.Instance.allMetaTypes[i]?.canWrite);
  118. metaItem && metaList.appendChild(metaItem);
  119. }
  120. if (!isRo)
  121. for (let i of ['geoCountry', 'geoCity', 'geoAdmin']) {
  122. if (!metaData[i]) {
  123. let metaItem = displayMeta(i, "", false);
  124. metaItem && metaList.appendChild(metaItem);
  125. }
  126. }
  127. return metaList;
  128. }
  129. function reloadCurrentMedia() {
  130. fullPageMediaDisplayed && window.displayMediaFullPage(fullPageMediaDisplayed);
  131. }
  132. function displayTags(fixedTagList, tagList, writeAccess) {
  133. let tagListUi = document.createElement("ul");
  134. tagListUi.className = "taglist";
  135. let createTagUiItem = (i, roTag) => {
  136. let uiItem = document.createElement("li");
  137. uiItem.className = "badge text-bg-light";
  138. let textItem = document.createElement("span");
  139. textItem.textContent = i;
  140. uiItem.appendChild(textItem);
  141. if (!roTag) {
  142. let btItem = document.createElement("a");
  143. btItem.className = "border-start bi bi-x removeBt";
  144. btItem.href = '#';
  145. btItem.addEventListener('click', async e => {
  146. e.preventDefault();
  147. await MediaStorage.Instance.removeTag(fullPageMediaDisplayed?.fixedSum || fullPageMediaList, i);
  148. reloadCurrentMedia();
  149. });
  150. uiItem.appendChild(btItem);
  151. }
  152. tagListUi.appendChild(uiItem);
  153. };
  154. for (let i of fixedTagList)
  155. createTagUiItem(i, true);
  156. for (let i of tagList)
  157. createTagUiItem(i, !writeAccess);
  158. if (writeAccess) {
  159. let inputGroup = document.createElement('form');
  160. tagListUi.appendChild(inputGroup);
  161. inputGroup.classList.add('input-group');
  162. let valInput = document.createElement("input");
  163. valInput.classList.add("form-control");
  164. valInput.addEventListener('keyup', evt => evt.stopPropagation());
  165. valInput.addEventListener('keydown', evt => evt.stopPropagation());
  166. inputGroup.appendChild(valInput);
  167. let bt = document.createElement('button');
  168. bt.className = 'btn btn-outline-secondary';
  169. bt.type = 'button';
  170. bt.innerHTML = '<i class="bi bi-tags"></i>';
  171. inputGroup.appendChild(bt);
  172. bt.addEventListener('click', async () => { await MediaStorage.Instance.addTag(fullPageMediaDisplayed?.fixedSum || fullPageMediaList, valInput.value); reloadCurrentMedia(); });
  173. inputGroup.addEventListener('submit', async evt => {
  174. evt.preventDefault();
  175. await MediaStorage.Instance.addTag(fullPageMediaDisplayed?.fixedSum || fullPageMediaList, valInput.value);
  176. reloadCurrentMedia();
  177. });
  178. }
  179. return tagListUi;
  180. }
  181. function displayDownloadBt(downloadLink) {
  182. let bt = document.createElement("button");
  183. bt.type = "button";
  184. bt.className = "btn btn-primary";
  185. bt.textContent = "Download";
  186. bt.addEventListener("click", (evt) => {
  187. let link = document.createElement('a');
  188. link.target='_blank';
  189. link.setAttribute("download", "");
  190. link.href = downloadLink;
  191. link.click();
  192. });
  193. return bt;
  194. }
  195. function _displayMediaFullPage(fileName, imgUrl, metaData, downloadLink, writeAccess) {
  196. return new Promise(ok => {
  197. document.getElementById("pch-fullPagePreviewContainer").classList.add("loading");
  198. document.getElementById("pch-fullPageMedia-title").innerText = fileName;
  199. document.getElementById("pch-fullPagePreview").onceLoaded = ok;
  200. document.getElementById("pch-fullPagePreview").src = imgUrl ?? "";
  201. document.getElementById("pch-fullPageDetail").innerText = "";
  202. if (downloadLink)
  203. document.getElementById("pch-fullPageDetail").appendChild(displayDownloadBt(downloadLink));
  204. document.getElementById("pch-fullPageDetail").appendChild(displayTags(metaData?.fixedTags || [], metaData?.tags || [], writeAccess));
  205. document.getElementById("pch-fullPageDetail").appendChild(displayMetas(Object.assign({}, metaData || {}), !writeAccess));
  206. });
  207. }
  208. function LoadPreviousMedia() {
  209. let i = fullPageMediaDisplayed;
  210. if (!i) return;
  211. while (i = MediaStorage.Instance.previousMedia(i)) {
  212. if (window.FilterManager.match(i)) {
  213. window.displayMediaFullPage(i);
  214. break;
  215. }
  216. }
  217. }
  218. function LoadNextMedia() {
  219. let i = fullPageMediaDisplayed;
  220. if (!i) return;
  221. while (i = MediaStorage.Instance.nextMedia(i)) {
  222. if (window.FilterManager.match(i)) {
  223. window.displayMediaFullPage(i);
  224. break;
  225. }
  226. }
  227. }
  228. function CloseFullpageMedia() {
  229. if (fullPageMediaDisplayed !== false || fullPageMediaList)
  230. document.body.classList.remove("overlay-visible");
  231. document.getElementById("pch-fullPageMedia").classList.add("hidden");
  232. fullPageMediaDisplayed = false;
  233. history.pushState({}, '', '#');
  234. document.Title.pop();
  235. }
  236. window.displayMediaFullPage = function(mediaItem) {
  237. document.getElementById("pch-fullPageMedia").classList.remove("hidden");
  238. document.getElementById("pch-fullPageMedia").classList.remove("multiple");
  239. if (fullPageMediaDisplayed)
  240. document.Title.replaceTitle(mediaItem.fileName);
  241. else
  242. document.Title.pushTitle(mediaItem.fileName);
  243. fullPageMediaDisplayed = mediaItem ?? null;
  244. document.body.classList.add("overlay-visible");
  245. if (!mediaItem)
  246. return _displayMediaFullPage("Error", null, {}, null, false);
  247. let containerSize = document.getElementById("pch-fullPageMedia").getBoundingClientRect();
  248. let requestSize = mediaItem.resize(containerSize.width, containerSize.height);
  249. document.getElementById("pch-fullPagePreview").parentNode.style.maxWidth = "100%";
  250. document.getElementById("pch-fullPagePreview").parentNode.style.maxHeight = "100%";
  251. let meta = {
  252. ...mediaItem.meta,
  253. date: { type: 'date', value: mediaItem.getDate() } || undefined,
  254. filename: mediaItem.filename ? { type: 'string', value: mediaItem.fileName } : undefined,
  255. fixedTags: mediaItem.fixedTags,
  256. tags: mediaItem.tags
  257. };
  258. if (document.location.hash != `#${mediaItem.fixedSum}`)
  259. history.pushState({}, '', `#${mediaItem.fixedSum}`);
  260. const requestSizeQuery = requestSize ? `&w=${requestSize.width}&h=${requestSize.height}` : "";
  261. return _displayMediaFullPage(mediaItem.fileName, `${mediaItem.thumbnail}?q=6${requestSizeQuery}`, meta, `${mediaItem.original}?trim`, mediaItem.writeAccess);
  262. }
  263. function aggregateMetas(medias) {
  264. let meta = medias.reduce((acc, x) => {
  265. for (let key in x.meta) {
  266. acc[key] = acc[key] || x.meta[key];
  267. acc[key] = acc[key].value != x.meta[key].value ? "(multiple)" : x.meta[key];
  268. }
  269. return acc;
  270. }, {});
  271. delete meta.dateTime;
  272. if (!Number.isInteger(meta.height) || !Number.isInteger(meta.width)) {
  273. delete meta.height;
  274. delete meta.width;
  275. }
  276. return meta;
  277. }
  278. window.displayMultipleMediaFullPage = function(medias) {
  279. const title = "Multiple edit"; // FIXME lang ?
  280. fullPageMediaList = Array.from(new Set(medias.map(x => x.fixedSum)));
  281. document.getElementById("pch-fullPageMedia").classList.remove("hidden");
  282. document.getElementById("pch-fullPageMedia").classList.add("multiple");
  283. document.Title.pushTitle(title);
  284. document.body.classList.add("overlay-visible");
  285. document.getElementById("pch-fullPagePreview").parentNode.style.maxWidth = "100%";
  286. document.getElementById("pch-fullPagePreview").parentNode.style.maxHeight = "100%";
  287. let meta = {
  288. ...aggregateMetas(medias),
  289. fixedTags: medias.reduce((acc, x) => { x.fixedTags.forEach(tag => acc.add(tag)) ; return acc; }, new Set()),
  290. tags: medias.reduce((acc, x) => { x.tags.forEach(tag => acc.add(tag)) ; return acc; }, new Set()),
  291. };
  292. return _displayMediaFullPage(title, "", meta, null, true);
  293. }
  294. document.getElementById("pch-fullPageMedia-closeBt")
  295. .addEventListener("click", () => CloseFullpageMedia());
  296. document.getElementById("pch-fullPagePreview").addEventListener("load", () => {
  297. document.getElementById("pch-fullPagePreviewContainer").classList.remove("loading");
  298. let domItem = document.getElementById("pch-fullPagePreview");
  299. domItem.onceLoaded && domItem.onceLoaded();
  300. domItem.onceLoaded = null;
  301. });
  302. document.addEventListener("keydown", evt => {
  303. if (!fullPageMediaDisplayed)
  304. return;
  305. if (evt.keyCode === 37 || evt.keyCode === 38)
  306. LoadPreviousMedia();
  307. else if (evt.keyCode === 39 || evt.keyCode === 40)
  308. LoadNextMedia();
  309. });
  310. document.onClosePopinRequested(() => { CloseFullpageMedia(); });
  311. });