mirror of
https://github.com/mickael-kerjean/filestash
synced 2026-01-03 22:33:08 +01:00
304 lines
16 KiB
JavaScript
304 lines
16 KiB
JavaScript
import { createElement, createRender } from "../../lib/skeleton/index.js";
|
|
import rxjs, { effect, onClick } from "../../lib/rx.js";
|
|
import { qs } from "../../lib/dom.js";
|
|
import t from "../../locales/index.js";
|
|
import { loadJS, loadCSS } from "../../helpers/loader.js";
|
|
|
|
export default async function(render, { toggle, load$ }) {
|
|
const $page = createElement(`
|
|
<div>
|
|
<div data-bind="header"></div>
|
|
<div data-bind="body">
|
|
<component-icon name="loading"></component-icon>
|
|
</div>
|
|
</div>
|
|
`);
|
|
render($page);
|
|
componentHeader(createRender(qs($page, `[data-bind="header"]`)), { toggle });
|
|
componentBody(createRender(qs($page, `[data-bind="body"]`)), { load$ });
|
|
}
|
|
|
|
function componentHeader(render, { toggle }) {
|
|
const $header = createElement(`
|
|
<div class="header">
|
|
<div>${t("Info")}</div>
|
|
<div style="flex: 1 1 0%;">
|
|
<img class="component_icon" draggable="false" src="" alt="close">
|
|
</div>
|
|
</div>
|
|
`);
|
|
render($header);
|
|
effect(onClick(qs($header, `[alt="close"]`)).pipe(rxjs.tap(toggle)));
|
|
}
|
|
|
|
function componentBody(render, { load$ }) {
|
|
const $page = createElement(`
|
|
<div>
|
|
<div class="content_box">
|
|
<img class="component_icon" draggable="false" src="" alt="schedule">
|
|
<div class="headline ellipsis" data-bind="date">-</div>
|
|
<div class="description ellipsis" data-bind="time">-</div>
|
|
</div>
|
|
<div class="content_box">
|
|
<img class="component_icon" draggable="false" src="" alt="camera">
|
|
<div class="headline ellipsis" data-bind="camera-setting">-</div>
|
|
<div class="description ellipsis" data-bind="camera-name">-</div>
|
|
</div>
|
|
<div data-bind="map"></div>
|
|
<div data-bind="all">
|
|
<component-icon name="loading"></component-icon>
|
|
</div>
|
|
</div>
|
|
`);
|
|
render($page);
|
|
|
|
effect(load$.pipe(rxjs.tap(async($img) => {
|
|
if (!$img) return;
|
|
const metadata = await extractExif($img);
|
|
qs($page, `[data-bind="date"]`).innerText = formatDate(metadata.date) || "-";
|
|
qs($page, `[data-bind="time"]`).innerText = formatTime(metadata.date) || "-";
|
|
qs($page, `[data-bind="camera-setting"]`).innerText = formatCameraSettings(metadata) || "-";
|
|
qs($page, `[data-bind="camera-name"]`).innerText = formatCameraName(metadata) || "-";
|
|
|
|
if (metadata.location) await componentMap(createRender(qs($page, `[data-bind="map"]`)), { metadata });
|
|
componentMore(createRender(qs($page, `[data-bind="all"]`)), { metadata });
|
|
}), rxjs.catchError((err) => {
|
|
qs($page, `[data-bind="all"]`).remove();
|
|
return rxjs.EMPTY;
|
|
})));
|
|
}
|
|
|
|
async function componentMap(render, { metadata }) {
|
|
const DMSToDD = (d) => {
|
|
if (!d || d.length !== 4) return null;
|
|
const [degrees, minutes, seconds, direction] = d;
|
|
const dd = degrees + minutes/60 + seconds/(60*60);
|
|
return direction === "S" || direction === "W" ? -dd : dd;
|
|
};
|
|
const lat = DMSToDD(metadata.location[0]);
|
|
const lng = DMSToDD(metadata.location[1]);
|
|
const $page = createElement(`
|
|
<div class="component_mapshot error">
|
|
<div class="wrapper">
|
|
<div class="mapshot_placeholder error hidden">
|
|
<span><div>Erreur</div></span>
|
|
</div>
|
|
<div class="mapshot_placeholder loading hidden">
|
|
<div class="loader">
|
|
<div class="component_loader">
|
|
<svg width="120px" height="120px" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100" preserveAspectRatio="xMidYMid"><rect x="0" y="0" width="100" height="100" fill="none"></rect><circle cx="50" cy="50" r="40" stroke="rgba(100%,100%,100%,0.679)" fill="none" stroke-width="10" stroke-linecap="round"></circle><circle cx="50" cy="50" r="40" stroke="#6f6f6f" fill="none" stroke-width="6" stroke-linecap="round"><animate attributeName="stroke-dashoffset" dur="2s" repeatCount="indefinite" from="0" to="502"></animate><animate attributeName="stroke-dasharray" dur="2s" repeatCount="indefinite" values="150.6 100.4;1 250;150.6 100.4"></animate></circle></svg>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<a href="https://www.google.com/maps/search/?api=1&query=${lat},${lng}">
|
|
<div class="marker"><img class="component_icon" draggable="false" src="" alt="location"></div>
|
|
<div data-bind="maptile"></div>
|
|
</a>
|
|
</div>
|
|
</div>
|
|
`);
|
|
render($page);
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 500));
|
|
const TILE_SERVER = "https://tile.openstreetmap.org/${z}/${x}/${y}.png";
|
|
const TILE_SIZE = Math.floor($page.clientWidth / 3 * 100) / 100;
|
|
if (TILE_SIZE === 0) return;
|
|
$page.style.height = "${TILE_SIZE*3}px;";
|
|
const mapper = (function map_url(lat, lng, zoom) {
|
|
// https://wiki.openstreetmap.org/wiki/Slippy_map_tilenamse
|
|
const n = Math.pow(2, zoom);
|
|
const tile_numbers = [
|
|
(lng+180)/360*n,
|
|
(1-Math.log(Math.tan(lat*Math.PI/180) + 1/Math.cos(lat*Math.PI/180))/Math.PI)/2*n,
|
|
zoom,
|
|
];
|
|
return {
|
|
tile: function(tile_server, x = 0, y = 0) {
|
|
return tile_server
|
|
.replace("${x}", Math.floor(tile_numbers[0] || 0)+x)
|
|
.replace("${y}", Math.floor(tile_numbers[1] || 0)+y)
|
|
.replace("${z}", Math.floor(zoom));
|
|
},
|
|
position: function() {
|
|
return [
|
|
tile_numbers[0] - Math.floor(tile_numbers[0]),
|
|
tile_numbers[1] - Math.floor(tile_numbers[1]),
|
|
];
|
|
},
|
|
};
|
|
}(lat, lng, 11));
|
|
const $tiles = createElement(`
|
|
<div class="bigpicture">
|
|
<div class="line">
|
|
<img src="${mapper.tile(TILE_SERVER, -1, -1)}" class="btl" style="height: ${TILE_SIZE}px;">
|
|
<img src="${mapper.tile(TILE_SERVER, 0, -1)}" style="height: ${TILE_SIZE}px;">
|
|
<img src="${mapper.tile(TILE_SERVER, 1, -1)}" class="btr" style="height: ${TILE_SIZE}px;">
|
|
</div>
|
|
<div class="line">
|
|
<img src="${mapper.tile(TILE_SERVER, -1, 0)}" style="height: ${TILE_SIZE}px;">
|
|
<img src="${mapper.tile(TILE_SERVER, 0, 0)}" style="height: ${TILE_SIZE}px;">
|
|
<img src="${mapper.tile(TILE_SERVER, 1, 0)}" style="height: ${TILE_SIZE}px;">
|
|
</div>
|
|
<div class="line">
|
|
<img src="${mapper.tile(TILE_SERVER, -1, 1)}" class="bbl" style="height: ${TILE_SIZE}px;">
|
|
<img src="${mapper.tile(TILE_SERVER, 0, 1)}" style="height: ${TILE_SIZE}px;">
|
|
<img src="${mapper.tile(TILE_SERVER, 1, 1)}" class="bbr" style="height: ${TILE_SIZE}px;">
|
|
</div>
|
|
</div>
|
|
`);
|
|
qs($page, `[data-bind="maptile"]`).appendChild($tiles);
|
|
const pos = mapper.position();
|
|
qs($page, ".marker").setAttribute("style", `
|
|
left: ${TILE_SIZE * (1 + pos[0]) - 15}px;
|
|
top: ${TILE_SIZE * (1 + pos[1]) - 30}px;
|
|
`);
|
|
const center = (position, i) => Math.floor(TILE_SIZE * (1 + position[i]) * 1000)/1000;
|
|
$tiles.setAttribute("style", `transform-origin: ${center(mapper.position(), 0)}px ${center(mapper.position(), 1)}px;`);
|
|
}
|
|
|
|
function componentMore(render, { metadata }) {
|
|
const $all = document.createDocumentFragment();
|
|
const formatKey = (str) => str.replace(/([A-Z][a-z])/g, " $1");
|
|
const formatValue = (str) => {
|
|
if (!metadata.all || metadata.all[str] === undefined) return "-";
|
|
if (typeof metadata.all[str] === "number") {
|
|
return Math.floor(metadata.all[str]*100)/100;
|
|
} else if (metadata.all[str].denominator !== undefined &&
|
|
metadata.all[str].numerator !== undefined) {
|
|
if (metadata.all[str].denominator === 1) {
|
|
return metadata.all[str].numerator;
|
|
} else if (metadata.all[str].numerator > metadata.all[str].denominator) {
|
|
return Math.floor(
|
|
metadata.all[str].numerator * 10 / metadata.all[str].denominator,
|
|
) / 10;
|
|
} else {
|
|
return metadata.all[str].numerator+"/"+metadata.all[str].denominator;
|
|
}
|
|
} else if (typeof metadata.all[str] === "string") {
|
|
return metadata.all[str];
|
|
} else if (Array.isArray(metadata.all[str])) {
|
|
let arr = metadata.all[str];
|
|
if (arr.length > 15) {
|
|
arr = arr.slice(0, 3);
|
|
arr.push("...");
|
|
}
|
|
return arr.toString().split(",").join(", ");
|
|
} else {
|
|
return JSON.stringify(metadata.all[str], null, 2);
|
|
}
|
|
};
|
|
Object.keys(metadata.all || {}).sort((a, b) => {
|
|
if (a.toLowerCase().trim() < b.toLowerCase().trim()) return -1;
|
|
else if (a.toLowerCase().trim() > b.toLowerCase().trim()) return +1;
|
|
return 0;
|
|
}).forEach((key) => {
|
|
switch (key) {
|
|
case "undefined":
|
|
case "thumbnail":
|
|
break;
|
|
default: $all.appendChild(createElement(`
|
|
<div class="meta_key">
|
|
<div class="title ellipsis">${formatKey(key)}: </div>
|
|
<div class="value ellipsis" title="${formatValue(key)}">${formatValue(key)}</div>
|
|
</div>
|
|
`));
|
|
}
|
|
});
|
|
render($all);
|
|
}
|
|
|
|
export function init() {
|
|
return Promise.all([
|
|
loadJS(import.meta.url, "../../lib/vendor/exif-js.js"),
|
|
loadCSS(import.meta.url, "./application_image_metadata.css"),
|
|
]);
|
|
}
|
|
|
|
const extractExif = ($img) => new Promise((resolve) => window.EXIF.getData($img, function() {
|
|
const metadata = window.EXIF.getAllTags($img);
|
|
const to_date = (str = "") => {
|
|
if (str === "") return null;
|
|
const digits = str.split(/[ :]/).map((digit) => parseInt(digit));
|
|
return new Date(
|
|
digits[0] || 0,
|
|
digits[1] || 0,
|
|
digits[2] || 0,
|
|
digits[3] || 0,
|
|
digits[4] || 0,
|
|
digits[5] || 0,
|
|
);
|
|
};
|
|
resolve({
|
|
date: to_date(
|
|
metadata["DateTime"] || metadata["DateTimeDigitized"] ||
|
|
metadata["DateTimeOriginal"] || metadata["GPSDateStamp"],
|
|
),
|
|
location: (metadata["GPSLatitude"] && metadata["GPSLongitude"] && [
|
|
[
|
|
metadata["GPSLatitude"][0], metadata["GPSLatitude"][1],
|
|
metadata["GPSLatitude"][2], metadata["GPSLatitudeRef"],
|
|
],
|
|
[
|
|
metadata["GPSLongitude"][0], metadata["GPSLongitude"][1],
|
|
metadata["GPSLongitude"][2], metadata["GPSLongitudeRef"],
|
|
],
|
|
]) || null,
|
|
maker: metadata["Make"] || null,
|
|
model: metadata["Model"] || null,
|
|
focal: metadata["FocalLength"] || null,
|
|
aperture: metadata["FNumber"] || null,
|
|
shutter: metadata["ExposureTime"] || null,
|
|
iso: metadata["ISOSpeedRatings"] || null,
|
|
dimension: (metadata["PixelXDimension"] && metadata["PixelYDimension"] && [
|
|
metadata["PixelXDimension"],
|
|
metadata["PixelYDimension"],
|
|
]) || null,
|
|
all: Object.keys(metadata).length === 0 ? null : metadata,
|
|
});
|
|
}));
|
|
|
|
const formatTime = (t) => t?.toLocaleTimeString(
|
|
"en-us",
|
|
{ weekday: "short", hour: "2-digit", minute: "2-digit" },
|
|
);
|
|
|
|
const formatDate = (t) => t?.toLocaleDateString(
|
|
navigator.language,
|
|
{ year: "numeric", month: "short", day: "numeric" },
|
|
);
|
|
|
|
const formatCameraSettings = (metadata) => {
|
|
const str = format("model", metadata);
|
|
const f = format("focal", metadata);
|
|
if (!f) return str;
|
|
return `${str} (${f})`;
|
|
};
|
|
|
|
const formatCameraName = (metadata) => {
|
|
return [
|
|
format("shutter", metadata),
|
|
format("aperture", metadata),
|
|
format("iso", metadata),
|
|
].join(" ").trim() || "-";
|
|
};
|
|
|
|
const format = (key, metadata) => {
|
|
if (!metadata[key]) return "";
|
|
switch (key) {
|
|
case "focal":
|
|
return `${metadata.focal}mm`;
|
|
case "iso":
|
|
return `ISO${metadata.iso}`;
|
|
case "aperture":
|
|
return `ƒ${Math.floor(metadata.aperture*10)/10}`;
|
|
case "shutter":
|
|
if (metadata.shutter > 60) return metadata.shutter+"m";
|
|
else if (metadata.shutter > 1) return metadata.shutter+"s";
|
|
return `1/${Math.floor(metadata.shutter.denominator / metadata.shutter.numerator)}s`;
|
|
case "dimension":
|
|
if (metadata.dimension.length !== 2 || !metadata.dimension[0] || !metadata.dimension[1]) return "-";
|
|
return metadata.dimension[0]+"x"+metadata.dimension[1];
|
|
default:
|
|
return metadata[key];
|
|
}
|
|
};
|