RE: AI 이미지 EXIF 뷰어

AI 이미지 메타데이터 보기

// ==UserScript==
// @name        RE: AI 이미지 EXIF 뷰어
// @namespace   https://github.com/panta5/AI-Image-EXIF-Viewer
// @match       https://www.pixiv.net/*
// @match       https://arca.live/b/aiart*
// @match       https://arca.live/b/hypernetworks*
// @match       https://arca.live/b/aiartreal*
// @match       https://arca.live/b/aireal*
// @match       https://arca.live/b/characterai*
// @version     2.1.1+1.0
// @author      PantaFive
// @homepageURL https://github.com/panta5/AI-Image-EXIF-Viewer
// @require     https://cdnjs.cloudflare.com/ajax/libs/pako/2.1.0/pako.min.js
// @require     https://greasyfork.org/scripts/452821-upng-js/code/UPNGjs.js?version=1103227
// @require     https://cdn.jsdelivr.net/npm/casestry-exif-library@2.0.3/dist/exif-library.min.js
// @require     https://cdn.jsdelivr.net/npm/sweetalert2@11
// @require     https://cdn.jsdelivr.net/npm/clipboard@2.0.10/dist/clipboard.min.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/arrive/2.4.1/arrive.min.js
// @require     https://greasyfork.org/scripts/421384-gm-fetch/code/GM_fetch.js
// @grant       GM_xmlhttpRequest
// @grant       GM_registerMenuCommand
// @grant       GM_setValue
// @grant       GM_getValue
// @grant       GM_addStyle
// @grant       GM_download
// @grant       GM_info

// @description AI 이미지 메타데이터 보기
// @license MIT
// ==/UserScript==

//this URL must be changed manually to be linked properly
// const scriptGreasyforkURL = '#';
//toast timer in ms
const toastTimer = 3000;
const colorOption1 = '#5cc964';
const colorOption2 = '#ff9d0b';
const colorClose = '#b41b29';

const footerString = `<div class="version">v${GM_info.script.version}  -  <a href="${GM_info.script.namespace}" target="_blank">Forked Repository</a>  -  <a href="https://github.com/nyqui/AI-Image-EXIF-Viewer" target="_blank">Original GitHub Repository</a></div>`;

(async function () {
    'use strict';

    const modalCSS = /* css */ `
  font-family: -apple-system, BlinkMacSystemFont, NanumBarunGothic, NanumGothic, system-ui, sans-serif;
  .swal2-popup {
    font-size: 15px;
  }
  .swal2-actions {
    margin: .4em auto 0;
  }
  .swal2-footer{
    margin: 1em 1.6em .3em;
    padding: 1em 0 0;
    overflow: auto;
    font-size: 1.125em;
  }
  #dropzone {
    z-index: 100000000;
    display: none;
    position: fixed;
    width: 100%;
    height: 100%;
    left: 0;
    top: 0;
  }

  .md-grid {
    display: grid;
    grid-template-rows: repeat(3, auto);
    text-align: left;
  }

  .md-grid-item {
    border-bottom: 1px solid #b3b3b3;
    padding: .6em;
  }

  .md-grid-item:last-child {
    border-bottom: 0px;
  }

  .md-nested-grid {
    display: grid;
    grid-template-columns: repeat(3, 1fr);
    grid-template-rows: repeat(4, auto);
    gap: .5em;
  }

  .md-title {
    line-height: 1em;
    font-weight: bold;
    font-size: .9em;
    padding-bottom: .2em;
    display: flex;
    color: #1A1A1A;
  }

  .md-info {
    line-height: 1.5em;
    font-size: .8em;
    word-break: break-word;
    color: #444444;
  }

  .md-hidden {
    overflow: hidden;
    position: relative;
    max-height: 5em;
  }

  .md-hidden:after {
    content: "";
    position: absolute;
    bottom: 0;
    left: 0;
    width: 100%;
    height: 2em;
    background: linear-gradient(to bottom, rgba(255, 255, 255, 0.4) 0%, white 100%);
  }
  .md-info > a{
    text-decoration: none;
  }
  .md-info > a:hover{
    text-decoration: underline !important;
  }
  pre.md-show-and-hide{
    font-family: monospace;
    margin: 0px;
    white-space: pre-line;
  }

  .md-visible {
    height: auto;
    overflow: auto;
  }

  .md-model {
    grid-column-start: 1;
    grid-column-end: 3;
  }

  .md-show-more {
    text-align: center;
    cursor: pointer;
  }

  #md-tags {
    width: 100%;
    height: 20em;
    padding-top: .5em;
    text-align: left;
    font-size: 0.9em;
  }
  span.md-button {
    margin-left: .15em;
    cursor: pointer;
  }

  span.md-copy {
    content: url("data:image/svg+xml,%3Csvg width='16' height='16' viewBox='0 0 16 16' xmlns='http://www.w3.org/2000/svg' xmlns:xlink='http://www.w3.org/1999/xlink'%3E%3Crect width='16' height='16' stroke='none' fill='%23000000' opacity='0'/%3E%3Cg transform='matrix(0.6 0 0 0.6 8 8)' %3E%3Cpath style='stroke: none; stroke-width: 1; stroke-dasharray: none; stroke-linecap: butt; stroke-dashoffset: 0; stroke-linejoin: miter; stroke-miterlimit: 4; fill: rgb(0,0,0); fill-rule: nonzero; opacity: 1;' transform=' translate(-12, -12)' d='M 4 2 C 2.895 2 2 2.895 2 4 L 2 18 L 4 18 L 4 4 L 18 4 L 18 2 L 4 2 z M 8 6 C 6.895 6 6 6.895 6 8 L 6 20 C 6 21.105 6.895 22 8 22 L 20 22 C 21.105 22 22 21.105 22 20 L 22 8 C 22 6.895 21.105 6 20 6 L 8 6 z M 8 8 L 20 8 L 20 20 L 8 20 L 8 8 z' stroke-linecap='round' /%3E%3C/g%3E%3C/svg%3E");
  }

  span.md-civitai {
    content: url("");
  }

  .version {
    margin: 1px;
    text-align: right;
    font-size: .5em;
    font-style: italic;
  }
  `;

    const toastmix = Swal.mixin({
        toast: true,
        position: 'bottom',
        showConfirmButton: false,
        timer: `${toastTimer}`,
        timerProgressBar: true,
    });

    function registerMenu() {
        try {
            if (typeof GM_registerMenuCommand == undefined) {
                return;
            } else {
                GM_registerMenuCommand('(로그인 필수) Pixiv 뷰어 사용 토글', () => {
                    if (GM_getValue('usePixiv', false)) {
                        GM_setValue('usePixiv', false);
                        toastmix.fire({
                            icon: 'error',
                            title: `Pixiv 비활성화
                      창이 닫힌 후 새로고침 됩니다`,
                            didDestroy: () => {
                                location.reload();
                            },
                        });
                    } else {
                        GM_setValue('usePixiv', true);
                        toastmix.fire({
                            icon: 'success',
                            title: `Pixiv 활성화
                      창이 닫힌 후 새로고침 됩니다`,
                            didDestroy: () => {
                                location.reload();
                            },
                        });
                    }
                });
                GM_registerMenuCommand('아카라이브 EXIF 보존 토글', () => {
                    if (GM_getValue('saveExifDefault', true)) {
                        GM_setValue('saveExifDefault', false);
                        toastmix.fire({
                            icon: 'error',
                            title: `아카라이브 EXIF 보존 비활성화
                      다음번 작성시부터 버려집니다`,
                        });
                    } else {
                        GM_setValue('saveExifDefault', true);
                        toastmix.fire({
                            icon: 'success',
                            title: `아카라이브 EXIF 보존 활성화
                      다음번 작성시부터 보존됩니다`,
                        });
                    }
                });
                GM_registerMenuCommand('아카라이브 글쓰기 창 스크립트 토글', () => {
                    if (GM_getValue('useDragdropUpload', true)) {
                        GM_setValue('useDragdropUpload', false);
                        toastmix.fire({
                            icon: 'error',
                            title: `아카 글쓰기 창 스크립트 비활성화
                      다음번 작성시부터 적용됩니다`,
                        });
                    } else {
                        GM_setValue('useDragdropUpload', true);
                        toastmix.fire({
                            icon: 'success',
                            title: `아카 글쓰기 창 스크립트 활성화
                      다음번 작성시부터 적용됩니다`,
                        });
                    }
                });
            }
        } catch (err) {
            console.log(err);
        }
    }

    class DropZone {
        constructor() {
            const dropZone = document.createElement('div');
            dropZone.setAttribute('id', 'dropzone');
            document.body.appendChild(dropZone);
            this.dropZone = document.getElementById('dropzone');
            this.setupEventListeners();
        }

        showDropZone() {
            this.dropZone.style.display = 'block';
        }

        hideDropZone() {
            this.dropZone.style.display = 'none';
        }

        allowDrag(e) {
            e.preventDefault();
        }

        async handleDrop(e) {
            e.preventDefault();
            this.hideDropZone();

            const file = e.dataTransfer.files[0];
            if (!file) return;

            const blob = await fileToBlob(file);
            const type = blob.type;
            if (isArcaEditor) {
                const uploadableType = handleUploadable(type);
                let editor = document.querySelector('.write-body .fr-element');
                let saveEXIF = GM_getValue('saveExifDefault', true);
                if (uploadableType == 'image') {
                    try {
                        saveEXIF = document.getElementById('saveExif').checked;
                    } catch {}
                    uploadArca(blob, uploadableType, saveEXIF).then((url) => {
                        editor.innerHTML =
                            editor.innerHTML + `<p><img src="${url}" class="fr-fic fr-dii"></p><p><br></p>`;
                        Swal.close();
                    });
                } else if (uploadableType == 'video') {
                    uploadArca(blob, uploadableType, false).then((url) => {
                        editor.innerHTML =
                            editor.innerHTML +
                            `<p><span class="fr-video fr-dvi fr-draggable"><video class="fr-draggable" controls="" loop="" muted="" playsinline="" src="${url}">귀하의 브라우저는 html5 video를 지원하지 않습니다.</video></span></p><p><br></p>`;
                        Swal.close();
                    });
                } else {
                    Swal.close();
                    toastmix.fire({
                        icon: 'error',
                        title: `업로드 오류:
                    업로드 할 수 있는 포맷이 아닙니다.`,
                    });
                }
            } else {
                if (!isSupportedImageFormat(blob.type)) {
                    notSupportedFormat();
                    return;
                }
                const metadata = await extractImageMetadata(blob, type);
                metadata ? showMetadataModal(metadata) : showTagExtractionModal(null, blob);
            }
        }

        setupEventListeners() {
            window.addEventListener('dragenter', () => this.showDropZone());
            this.dropZone.addEventListener('dragenter', (e) => this.allowDrag(e));
            this.dropZone.addEventListener('dragover', (e) => this.allowDrag(e));
            this.dropZone.addEventListener('dragleave', () => this.hideDropZone());
            this.dropZone.addEventListener('drop', (e) => this.handleDrop(e));
        }
    }

    function getMetadataPNGChunk(chunk) {
        const isValidPNG = chunk.slice(0, 8).every((byte, index) => [137, 80, 78, 71, 13, 10, 26, 10][index] === byte);
        if (!isValidPNG) {
            console.error('Invalid PNG');
            return null;
        }

        const textDecoder = new TextDecoder('utf-8');
        let metadata = {};

        function checkForChunks() {
            let position = 8;
            while (true) {
                const chunkLength = getUint32(position);

                if (chunk.byteLength < position + chunkLength + 12) {
                    return;
                }
                const name = String.fromCharCode(...chunk.subarray(position + 4, position + 8));
                const data = chunk.subarray(position + 8, position + chunkLength + 8);
                const dataString = textDecoder.decode(data);

                if (name === 'tEXt') {
                    const [key, value] = dataString.split('\0');
                    metadata[key] = value;
                } else if (name === 'iTXt') {
                    const [key, value] = dataString.split('\0\0\0\0\0');
                    metadata[key] = value;
                } else if (name === 'IDAT') {
                    metadata[name] = true;
                    return;
                }
                position += chunkLength + 12;
            }
        }

        function getUint32(offset) {
            return (chunk[offset] << 24) | (chunk[offset + 1] << 16) | (chunk[offset + 2] << 8) | chunk[offset + 3];
        }
        checkForChunks();
        return metadata;
    }

    function getMetadataJPEGChunk(chunk) {
        if (chunk[0] !== 255 || chunk[1] !== 216) {
            // 0xFF 0xD8
            console.error('Invalid JPEG');
            return null;
        }
        const textDecoder = new TextDecoder();
        let offset = 2;
        if (chunk[offset] === 0xff) {
            switch (chunk[offset + 1]) {
                case 0xe0: {
                    offset += ((chunk[offset + 2] << 8) | chunk[offset + 3]) + 2;
                }
                case 0xe1: {
                    const length = (chunk[offset + 2] << 8) | chunk[offset + 3];
                    const data = chunk.subarray(offset + 4, offset + 2 + length);
                    if (
                        data[0] === 69 && //0x45 E
                        data[1] === 120 && //0x78 x
                        data[2] === 105 && //0x69 i
                        data[3] === 102 && //0x66 f
                        data[4] === 0 && // null
                        data[5] === 0 // null
                    ) {
                        const userCommentData = data.subarray(46, offset + 2 + length);
                        const parameters = textDecoder
                            .decode(userCommentData)
                            .replace('UNICODE', '')
                            .replaceAll('\u0000', '');
                        return {
                            parameters,
                        };
                    }
                }
                default:
                    return null;
            }
        }
        return null;
    }

    function getFileName(url) {
        if (url === '/') return;
        const fileName = url.split('?')[0];
        return fileName;
    }

    function parseMetadata(exif) {
        try {
            let metadata = {};
            if (exif.parameters) {
                let parameters = exif.parameters.replaceAll('<', '&lt;').replaceAll('>', '&gt;');
                metadata.rawMetadata = parameters;

                if (!parameters.includes('Negative prompt')) {
                    parameters = parameters.replace('Steps', '\nNegative prompt: 정보 없음\nSteps');
                }

                parameters = parameters.split('Steps: ');
                parameters = `${parameters[0]
                    .replaceAll(': ', ':')
                    .replace('Negative prompt:', 'Negative prompt: ')}Steps: ${parameters[1]}`;

                const metadataStr = parameters.substring(parameters.indexOf('Steps'), parameters.length);
                const keyValuePairs = metadataStr.split(', ');

                for (const pair of keyValuePairs) {
                    const [key, value] = pair.split(': ');
                    metadata[key] = value;
                }

                metadata.prompt =
                    parameters.indexOf('Negative prompt') === 0
                        ? '정보 없음'
                        : parameters.substring(0, parameters.indexOf('Negative prompt:'));
                metadata.negativePrompt = parameters.includes('Negative prompt:')
                    ? parameters
                          .substring(parameters.indexOf('Negative prompt:'), parameters.indexOf('Steps:'))
                          .replace('Negative prompt:', '')
                    : null;

                return metadata;
            } else if (exif.Description) {
                metadata.rawMetadata = `${exif.Description}\n${exif.Comment}`;
                const comment = JSON.parse(exif.Comment);

                metadata.prompt = exif.Description;
                metadata.negativePrompt = comment.uc;
                metadata['Steps'] = comment.steps;
                metadata['Sampler'] = comment.sampler;
                metadata['CFG scale'] = comment.scale;
                metadata['Seed'] = comment.seed;
                metadata['Software'] = 'NovelAI';

                return metadata;
            } else if (exif['sd-metadata']) {
                metadata.rawMetadata = exif['sd-metadata'];
                const parameters = JSON.parse(exif['sd-metadata']);
                const rowPrompt = parameters.image.prompt[0].prompt;
                const PromptRegex = /[^[\]]+(?=\[|$)/g;
                const negativePromptRegex = /\[.*?\]/g;
                const promptArray = rowPrompt.match(PromptRegex);
                const negativePromptArray = rowPrompt.match(negativePromptRegex);
                const prompt = promptArray.map((prompt) => prompt.replace(/^\,|\,$/g, ''));
                const negativePrompt = negativePromptArray.map((prompt) =>
                    prompt.replace(/^\[|\]$/g, '').replace(/^\,|\,$/g, ''),
                );

                metadata.prompt = prompt.join(', ');
                metadata.negativePrompt = negativePrompt.join(', ');
                metadata['Steps'] = parameters?.image.steps;
                metadata['Model'] = parameters?.model;
                metadata['Model hash'] = parameters?.model_hash;
                metadata['Sampler'] = parameters?.image.sampler;
                metadata['CFG scale'] = parameters?.image.cfg_scale;
                metadata['Seed'] = parameters?.image.seed;
                metadata['Size'] = `${parameters?.image.width}x${parameters?.image.height}`;
                metadata['Software'] = 'InvokeAI';

                return metadata;
            }
        } catch (error) {
            console.log(error);
            Swal.fire({
                icon: 'error',
                confirmButtonColor: `${colorClose}`,
                confirmButtonText: '닫기',
                title: '분석 오류',
                html: `
        ${error}<br>
        오류내용과 이미지를 댓글로 알려주세요`,
            });
        }
    }

    function infer(metadata) {
        if (metadata?.Software) return [metadata.Software];
        const inferList = [];
        const denoising = metadata?.['Denoising strength'];
        const hires = metadata?.['Hires upscaler'];

        inferList.push('T2I');
        if (denoising && !hires) {
            inferList[0] = 'I2I';
        } else if (hires) {
            inferList.push('Hires. fix');
        }
        (metadata?.['AddNet Enabled'] ||
            metadata?.prompt?.includes('lora:') ||
            metadata?.negativePrompt?.includes('lora:')) &&
            inferList.push('LoRa');
        (metadata?.prompt?.includes('lyco:') || metadata?.negativePrompt?.includes('lyco:')) &&
            inferList.push('LyCORIS');
        (metadata?.['Hypernet'] ||
            metadata?.prompt?.includes('hypernet:') ||
            metadata?.negativePrompt?.includes('hypernet:')) &&
            inferList.push('Hypernet');

        const controlNetRegex = /(ControlNet)/;
        for (const key in metadata) {
            if (controlNetRegex.test(key)) {
                inferList.push('ControlNet');
                break;
            }
        }
        metadata?.['SD upscale upscaler'] && inferList.push('SD upscale');
        metadata?.['Ultimate SD upscale upscaler'] && inferList.push('Ultimate SD upscale');
        metadata?.['Latent Couple'] && inferList.push('Latent Couple');
        metadata?.['Dynamic thresholding enabled'] && inferList.push('Dynamic thresholding');
        metadata?.['LLuL Enabled'] && inferList.push('LLuL');
        metadata?.['Cutoff enabled'] && inferList.push('Cutoff');
        metadata?.['Tiled Diffusion'] && inferList.push('Tiled Diffusion');
        metadata?.['DDetailer model a'] && inferList.push('DDetailer');
        metadata?.['ADetailer version'] && inferList.push('ADetailer'); // DDetailer/ADetailer를 DINO 하나로 묶는 게 나을까?

        return inferList;
    }

    function showAndHide(elementSelector) {
        const contentEls = document.querySelectorAll(elementSelector);

        contentEls.forEach((contentEl) => {
            const containerEl = contentEl.parentElement;
            const showMoreEl = containerEl.nextElementSibling;

            if (contentEl.offsetHeight > containerEl.offsetHeight) {
                showMoreEl.style.display = 'block';
                containerEl.classList.add('md-hidden');
            } else {
                showMoreEl.style.display = 'none';
                containerEl.classList.remove('md-hidden');
                containerEl.classList.add('md-visible');
            }

            showMoreEl.addEventListener('click', () => {
                const isMore = showMoreEl.textContent === '더 보기';
                showMoreEl.textContent = isMore ? '숨기기' : '더 보기';
                containerEl.classList.toggle('md-hidden', !isMore);
                containerEl.classList.toggle('md-visible', isMore);
            });
        });
    }

    function showMetadataModal(metadata, url) {
        metadata = parseMetadata(metadata);
        const inferList = infer(metadata);
        const showMeta = Swal.mixin({
            title: '메타데이터 요약',
            html: /*html*/ `
    <div class="md-grid">
      <div class="md-grid-item">
        <div class="md-title">Prompt <span class="md-copy md-button" data-clipboard-target="#prompt"></span></div>
        <div class="md-info" id="prompt">
          ${metadata.prompt ?? '정보 없음'}
        </div>
      </div>
      <div class="md-grid-item">
        <div class="md-title">Negative Prompt
          <span class="md-copy md-button" data-clipboard-target="#negative-prompt"></span>
        </div>
        <div class="md-info">
          <div class="md-hidden">
            <div class="md-show-and-hide" id="negative-prompt">
              ${metadata.negativePrompt ?? '정보 없음'}
            </div>
          </div>
          <div class="md-show-more">더 보기</div>
        </div>
      </div>
      <div class="md-grid-item">
        <div class="md-nested-grid">
          <div>
            <div class="md-title">Sampler <span class="md-copy md-button" data-clipboard-target="#sampler"></span></div>
            <div class="md-info" id="sampler">${metadata['Sampler'] ?? '정보 없음'}</div>
          </div>
          <div>
            <div class="md-title">Seed <span class="md-copy md-button" data-clipboard-target="#seed"></span></div>
            <div class="md-info" id="seed">${metadata['Seed'] ?? '정보 없음'}</div>
          </div>
          <div>
            <div class="md-title">Steps <span class="md-copy md-button" data-clipboard-target="#steps"></span></div>
            <div class="md-info" id="steps">${metadata['Steps'] ?? '정보 없음'}</div>
          </div>
          <div>
            <div class="md-title">Size <span class="md-copy md-button" data-clipboard-target="#size"></span></div>
            <div class="md-info" id="size">${metadata['Size'] ?? '정보 없음'}</div>
          </div>
          <div>
            <div class="md-title">CFG scale <span class="md-copy md-button" data-clipboard-target="#cfg-scale"></span></div>
            <div class="md-info" id="cfg-scale">${metadata['CFG scale'] ?? '정보 없음'}</div>
          </div>
          <div>
            <div class="md-title">Denoising strength <span class="md-copy md-button" data-clipboard-target="#denoising-strength"></span></div>
            <div class="md-info" id="denoising-strength">${metadata['Denoising strength'] ?? '정보 없음'}</div>
          </div>
          <div class="md-model">
            <div class="md-title">Model
              <span class="md-copy md-button" data-clipboard-target="#model"></span>
              <a href='https://civitai.com/?query=${
                  metadata['Model hash']
              }' target='_blank'><span class="md-civitai md-button"></span></a>
            </div>
            <div class="md-info" id="model">${
                metadata['Model']
                    ? `${metadata['Model']} [${metadata['Model hash']}]`
                    : metadata['Model hash'] ?? '정보 없음'
            }</div>
          </div>
          <div>
            <div class="md-title">Infer...</div>
            <div class="md-info">${inferList.join(', ')}</div>
          </div>
        </div>
      </div>
      `,
            footer: /*html*/ `
      <div class="md-grid-item">
      <div class="md-title">Raw Metadata <span class="md-copy md-button" data-clipboard-target="#raw-metadata"></span>
      </div>
      <div class="md-info">
        <div class="md-hidden">
          <pre class="md-show-and-hide" id="raw-metadata">
          ${metadata.rawMetadata ?? '정보 없음'}
        </pre>
        </div>
        <div class="md-show-more">더 보기</div>
      </div>
      ${footerString}
      </div>
      `,
            width: '50em',
            showDenyButton: true,
            showCancelButton: true,
            focusCancel: true,
            confirmButtonColor: `${colorOption1}`,
            denyButtonColor: `${colorOption2}`,
            cancelButtonColor: `${colorClose}`,
            confirmButtonText: '이미지 열기',
            denyButtonText: '이미지 저장',
            cancelButtonText: '닫기',
        });

        // if image has URL, options are available to open in new tab or download
        if (url != null) {
            showMeta.fire().then((result) => {
                if (result.isConfirmed) {
                    window.open(url, '_blank');
                } else if (result.isDenied) {
                    GM_download(url, getFileName(url));
                }
            });
        } else {
            // if image has no URL, then it must have been dragged and dropped, hence no open in new tab or download options
            showMeta.fire({
                showDenyButton: false,
                showCancelButton: false,
                focusCancel: false,
                focusConfirm: true,
                confirmButtonColor: `${colorClose}`,
                confirmButtonText: '닫기',
            });
        }
        showAndHide('.md-show-and-hide');
    }

    function showTagExtractionModal(url, blob) {
        let noMeta = Swal.mixin({
            footer: `
      <div style="width: 100%;">
        <div class="md-info" style="text-align: center;">
          <a href="${url}" target="_blank">Open image...</a>
        </div>
        ${footerString}
      </div>
      `,
        });
        if (url == null) {
            noMeta = Swal.mixin({
                footer: `
        <div style="width: 100%;">
          ${footerString}
        </div>
        `,
            });
        }

        function getOptimizedImageURL(url) {
            if (isArca) {
                return url.replace('ac.namu.la', 'ac-o.namu.la').replace('&type=orig', '');
            }
            if (isPixiv) {
                const extension = url.substring(url.lastIndexOf('.') + 1);
                return url
                    .replace('/img-original/', '/c/600x1200_90_webp/img-master/')
                    .replace(`.${extension}`, '_master1200.jpg');
            }
        }
        noMeta
            .fire({
                icon: 'error',
                title: '메타데이터 없음!',
                text: '찾아볼까요?',
                showCancelButton: true,
                showDenyButton: true,
                confirmButtonText: 'Danbooru Autotagger',
                denyButtonText: 'WD 1.4 Tagger',
                cancelButtonText: '아니오',
                showLoaderOnConfirm: true,
                showLoaderOnDeny: true,
                focusCancel: true,
                confirmButtonColor: `${colorOption1}`,
                denyButtonColor: `${colorOption2}`,
                cancelButtonColor: `${colorClose}`,
                backdrop: true,
                preConfirm: async () => {
                    if (url != null) {
                        const res = await GM_fetch(getOptimizedImageURL(url), {
                            headers: {
                                Referer: `${location.protocol}//${location.hostname}`,
                            },
                        });
                        blob = await res.blob();
                    }
                    let formData = new FormData();
                    formData.append('threshold', '0.4');
                    formData.append('format', 'json');
                    formData.append('file', blob);

                    return GM_fetch('https://autotagger.donmai.us/evaluate', {
                        method: 'POST',
                        body: formData,
                    })
                        .then((res) => {
                            if (!res.status === 200) {
                                Swal.showValidationMessage(`https://autotagger.donmai.us 접속되는지 확인!`);
                            }
                            return res.json();
                        })
                        .catch((error) => {
                            console.log(error);
                            Swal.showValidationMessage(`https://autotagger.donmai.us 접속되는지 확인!`);
                        });
                },
                preDeny: async () => {
                    if (url != null) {
                        const res = await GM_fetch(getOptimizedImageURL(url), {
                            headers: {
                                Referer: `${location.protocol}//${location.hostname}`,
                            },
                        });
                        blob = await res.blob();
                    }
                    const optimizedBase64 = await blobToBase64(blob);

                    return fetch('https://smilingwolf-wd-v1-4-tags.hf.space/run/predict', {
                        method: 'POST',
                        headers: {
                            'Content-Type': 'application/json',
                        },
                        body: JSON.stringify({
                            data: [optimizedBase64, 'SwinV2', 0.35, 0.85],
                        }),
                    })
                        .then((res) => res.json())
                        .catch((error) => {
                            Swal.showValidationMessage(error);
                        });
                },
                allowOutsideClick: () => !Swal.isLoading(),
            })
            .then((result) => {
                if (result.isDismissed) return;
                let tags;
                if (result.isConfirmed) {
                    tags = Object.keys(result.value[0].tags).join(', ').replaceAll('_', ' ');
                } else if (result.isDenied) {
                    tags = result.value.data[3]?.label
                        ? `${result.value.data[3]?.label}, ${result.value.data[0]}`
                        : result.value.data[0];
                }

                Swal.fire({
                    confirmButtonColor: `${colorClose}`,
                    confirmButtonText: '닫기',
                    html: /*html*/ `
            <div class="md-title">Output
              <span class="md-copy md-button" data-clipboard-target="#md-tags"></span>
            </div>
            <div class="md-info" id="md-tags">${tags}</div>
            `,
                });
            });
    }

    function fileToBlob(file) {
        return new Promise((resolve) => {
            const reader = new FileReader();
            reader.onload = () =>
                resolve(
                    new Blob([reader.result], {
                        type: file.type,
                    }),
                );
            reader.readAsArrayBuffer(file);
        });
    }

    function blobToBase64(blob) {
        return new Promise((resolve) => {
            const reader = new FileReader();
            reader.onloadend = () => resolve(reader.result);
            reader.readAsDataURL(blob);
        });
    }

    function notSupportedFormat() {
        toastmix.fire({
            position: 'top-end',
            icon: 'error',
            title: '지원하지 않는 파일 형식입니다.',
        });
    }

    function isSupportedImageFormat(url) {
        const supportedExtensions = /\.(png|jpe?g|webp)|image\/(jpeg|webp|png)/;
        return supportedExtensions.test(url);
    }

    function handleUploadable(MIME) {
        const uploadableSubtypes = /(jpe?g|jfif|pjp|png|gif|web[pm]|mov|mp4|m4[ab])/;
        const [type, subtype] = MIME.split('/');
        if (uploadableSubtypes.test(subtype)) {
            return type;
        } else {
            return null;
        }
    }

    async function getStealthExif(src) {
        let canvas = document.createElement('canvas');
        let ctx = canvas.getContext('2d', { willReadFrequently: true, alpha: true });
        let img = new Image();
        img.src = src;
        img.crossOrigin = 'Anonymous';

        await img.decode();

        canvas.width = img.width;
        canvas.height = img.height;
        ctx.drawImage(img, 0, 0);

        let binary = '';
        const signature = 'stealth_pngcomp';
        let index = 0;
        let reading = 'signature';
        let length = 0;

        for (let x = 0; x < img.width; x++) {
            for (let y = 0; y < img.height; y++) {
                let data = ctx.getImageData(x, y, 1, 1).data;
                let a = data[3];
                binary += String(a & 1);
                index++;

                if (reading == 'signature') {
                    if (index == signature.length * 8) {
                        let str = '';
                        for (let i = 0; i < binary.length / 8; i++) {
                            str += String.fromCharCode(parseInt(binary.substring(i * 8, i * 8 + 8), 2));
                        }

                        if (str == signature) {
                            reading = 'length';
                            binary = '';
                            index = 0;
                        } else {
                            return null;
                        }
                    }
                } else if (reading == 'length') {
                    if (index == 32) {
                        length = parseInt(binary, 2);
                        reading = 'data';
                        binary = '';
                        index = 0;
                    }
                } else if (reading == 'data') {
                    if (index == length) {
                        let array = new Uint8Array(length);
                        for (let i = 0; i < binary.length / 8; i++) {
                            array[i] = parseInt(binary.substring(i * 8, i * 8 + 8), 2);
                        }

                        let temp = pako.ungzip(array);
                        let prompt = new TextDecoder('utf-8').decode(temp);
                        return JSON.parse(prompt);
                    }
                }
            }
        }

        return null;
    }

    async function extractImageMetadata(blob, type) {
        try {
            switch (type) {
                case 'image/jpeg':
                case 'image/webp': {
                    const exif = exifLib.load(await blobToBase64(blob));
                    const parameters = exif.Exif[37510].replace('UNICODE', '').replaceAll('\u0000', '');
                    return {
                        parameters,
                    };
                }
                case 'image/png': {
                    const chunks = UPNG.decode(await blob.arrayBuffer());
                    let parameters = chunks.tabs.tEXt?.parameters || chunks.tabs.iTXt?.parameters;
                    const description = chunks.tabs.tEXt?.Description || chunks.tabs.iTXt?.Description;
                    if (parameters) {
                        return {
                            parameters,
                        };
                    } else if (description) {
                        return chunks.tabs?.tEXt || chunks.tabs?.iTXt;
                    } else {
                        return null;
                    }
                }
            }
        } catch (error) {
            console.log(error);
            return null;
        }
    }

    async function fetchAndDecode(url) {
        try {
            let response, contentType, reader;
            const Referer = `${location.protocol}//${location.hostname}`;
            if (isArca) {
                response = await fetch(url.replace('ac.namu.la', 'ac-o.namu.la'));
                contentType = response.headers.get('content-type');
                reader = response.body.getReader();
            } else if (useTampermonkey) {
                response = await new Promise((resolve) => {
                    GM_xmlhttpRequest({
                        url,
                        responseType: 'stream',
                        headers: {
                            Referer,
                        },
                        onreadystatechange: (data) => {
                            resolve(data);
                        },
                    });
                });
                const headers = Object.fromEntries(
                    response.responseHeaders.split('\n').map((line) => {
                        const [key, value] = line.split(':').map((part) => part.trim());
                        return [key, value];
                    }),
                );
                contentType = headers['content-type'];
                reader = response.response.getReader();
            } else {
                response = await GM_fetch(url, {
                    headers: {
                        Referer,
                    },
                });
                contentType = response.headers.get('content-type');
                reader = response.body.getReader();
            }
            if (
                (isPixiv && !url.includes('.jpg') && contentType === 'text/html') ||
                (isPixiv && url.includes('.jpg'))
            ) {
                url = url.replace('.png', '.jpg');
                showTagExtractionModal(url);
                return;
            }

            let metadata;
            let chunks = [];
            while (true) {
                const { done, value } = await reader.read();
                if (done || metadata || metadata === null) {
                    reader.cancel();
                    break;
                }
                switch (contentType) {
                    case 'image/jpeg':
                        metadata = getMetadataJPEGChunk(value);
                        break;
                    case 'image/png':
                        metadata = getMetadataPNGChunk(value);
                        metadata?.IDAT && reader.cancel();
                        break;
                    case 'image/webp':
                        chunks.push(value);
                        break;
                    default:
                        notSupportedFormat();
                        reader.cancel();
                        break;
                }
            }
            if (contentType === 'image/webp') {
                const blob = new Blob(chunks, {
                    type: 'image/webp',
                });
                const base64 = await blobToBase64(blob);
                const exif = exifLib.load(base64);
                const parameters = exif.Exif[37510].replace('UNICODE', '').replaceAll('\u0000', '');
                metadata = {
                    parameters,
                };
            }
            const stealthData = await getStealthExif(url);
            if ((await stealthData) != null) {
                console.log('stealth data:', await stealthData);
                return await stealthData;
            }
            console.log('EXIF data:', metadata);
            return metadata;
        } catch (error) {
            console.log(error);
            return null;
        }
    }

    async function extract(url) {
        if (!isSupportedImageFormat(url)) {
            notSupportedFormat();
            return;
        }

        Swal.fire({
            title: '로드 중!',
            width: '15rem',
            didOpen: () => {
                Swal.showLoading();
            },
        });

        console.time('modal open');
        console.time('fetch');
        const metadata = await fetchAndDecode(url);
        console.timeEnd('fetch');
        console.log(metadata);

        if (metadata?.Description || metadata?.parameters || metadata?.['sd-metadata']) {
            showMetadataModal(metadata, url);
        } else {
            showTagExtractionModal(url);
        }
        console.timeEnd('modal open');
    }

    function getCSRFToken() {
        return new Promise((resolve) => {
            const csrf = document.querySelector('input[name=_csrf]');
            const token = document.querySelector('input[name=token]');
            if (csrf && token) {
                resolve([csrf.value, token.value]);
            }
        });
    }

    function uploadArca(blob, type, saveEXIF = true, token = null) {
        return new Promise(async (resolve, reject) => {
            let swalText = '비디오는 EXIF 보존 설정에 영향을 받지 않습니다.';
            if (type == 'image') {
                swalText = 'EXIF 보존: ' + saveEXIF;
            }
            let xhr = new XMLHttpRequest();
            xhr.upload.addEventListener('progress', null, false);
            let formData = new FormData();
            if (!document.querySelector('#article_write_form > input[name=token]')) {
                await getCSRFToken().then((tokenList) => {
                    token = tokenList[1];
                });
            }

            formData.append('upload', blob);
            formData.append('token', token || document.querySelector('#article_write_form > input[name=token]').value);
            formData.append('saveExif', saveEXIF);
            formData.append('saveFilename', false);

            xhr.onload = function () {
                let response = JSON.parse(xhr.responseText);
                if (response.uploaded === true) {
                    resolve(response.url);
                } else {
                    Swal.close();
                    console.error(xhr.responseText);
                    toastmix.fire({
                        icon: 'error',
                        title: `업로드 오류`,
                    });
                }
            };
            xhr.open('POST', 'https://arca.live/b/upload');
            xhr.send(formData);
            Swal.fire({
                title: '파일 업로드중',
                text: swalText,
                showConfirmButton: false,
                allowOutsideClick: false,
                didOpen: () => {
                    Swal.showLoading();
                },
            });
        });
    }

    const { hostname, href, pathname } = location;
    const isMobile = /iPhone|iPad|iPod|Android/i.test(navigator.userAgent);
    const isPixiv = hostname === 'www.pixiv.net';
    const isArca = hostname === 'arca.live';
    const isArcaViewer = /(arca.live)(\/)(b\/.*)(\/)(\d*)/.test(href);
    const isArcaEditor = /(arca.live\/b\/.*\/)(edit|write)/.test(href);
    const useTampermonkey = GM_xmlhttpRequest?.RESPONSE_TYPE_STREAM && true;
    const isPixivDragUpload = pathname === '/illustration/create' || pathname === '/upload.php';

    if (GM_getValue('usePixiv', false) && isPixiv) {
        function getOriginalUrl(url) {
            const extension = url.substring(url.lastIndexOf('.') + 1);
            const originalUrl = url
                .replace('/c/600x1200_90_webp/img-master/', '/img-original/')
                .replace('/c/100x100/img-master/', '/img-original/')
                .replace('_master1200', '')
                .replace(`.${extension}`, '.png');
            return originalUrl;
        }

        let isAi = false;
        if (!isMobile) {
            document.arrive('footer > ul > li > span > a', function () {
                if (this.href === 'https://www.pixiv.help/hc/articles/11866167926809') isAi = true;
            });
            document.arrive('div[role=presentation]:last-child > div > div', function () {
                isAi && this.click();
            });
        } else {
            document.arrive('a.ai-generated', () => {
                isAi = true;
            });
            document.arrive('button.nav-back', function () {
                isAi && this.click();
            });
        }

        document.arrive('a > img', function () {
            if (this.alt === 'pixiv') return;

            if (isAi) {
                let src;
                if (!isMobile) {
                    src = this.parentNode.href;
                } else {
                    src = getOriginalUrl(this.src);
                }

                this.onclick = function () {
                    extract(src);
                };
            }
        });
    }

    if (isArcaViewer) {
        document.arrive(
            'a[href$="type=orig"] > img',
            {
                existing: true,
            },
            function () {
                if (this.classList.contains('channel-icon')) return;

                this.parentNode.onclick = (event) => {
                    if (event.button === 0) {
                        event.preventDefault();
                    }
                };
                this.onclick = function () {
                    const src = `${this.src}&type=orig`;
                    extract(src);
                };
            },
        );
    }

    let ArcaDragUpload = true;
    if (isArcaEditor) {
        if (GM_getValue('saveExifDefault', true)) {
            document.arrive(
                '.images-multi-upload',
                {
                    onceOnly: true,
                },
                () => {
                    document.getElementById('saveExif').checked = true;
                },
            );
        }
        if (!GM_getValue('useDragdropUpload', true)) ArcaDragUpload = false;
    }

    !isMobile && !isPixivDragUpload && ArcaDragUpload && new DropZone();
    GM_addStyle(modalCSS);
    new ClipboardJS('.md-copy');
    registerMenu();
})();