GalGameUI / index.html
SenY's picture
Update index.html
3caa8b1
<!doctype html>
<html lang="ja" class="h-100">
<head>
<!-- Google tag (gtag.js) -->
<script async src="https://www.googletagmanager.com/gtag/js?id=G-HWSPDD19EL"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag() { dataLayer.push(arguments); }
gtag('js', new Date());
gtag('config', 'G-HWSPDD19EL');
</script>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="description" content="Gal-Game UI Editor">
<meta name="author" content="SenY">
<meta name="generator" content="vscode">
<title>Gal-Game UI Editor</title>
<link rel="canonical" href="https://huggingface.co/spaces/SenY/GalGameUI">
<link href="https://unpkg.com/bootstrap@5.3.1/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="https://unpkg.com/font-awesome@4.7.0/css/font-awesome.min.css" rel="stylesheet">
<meta name="theme-color" content="#7952b3">
<style>
[aria-expanded=false] .text-expanded {
display: none;
}
[aria-expanded=true] .text-collapsed {
display: none;
}
a {
color: inherit;
}
hr {
margin: 0.5rem;
padding: 0;
}
#left {
background-repeat: no-repeat;
background-position: center bottom;
background-size: contain;
position: relative;
}
#edit {
background-repeat: no-repeat;
background-size: contain;
}
input {
padding: 0 1rem;
background: rgba(0, 0, 0, 0);
border: rgba(0, 0, 0, 0) solid 1px;
width: 100%;
height: 100%;
display: flex;
align-items: center;
font-size: 100%;
z-index: 3;
}
.shadow-text {
padding: 0 1rem;
display: flex;
align-items: center;
z-index: 3;
}
#edit .row2 {
display: flex;
align-items: stretch;
}
:root {
--messageWindowSize: 25%;
--mainMassageHeight: 100%;
--userInterfaceHeight: 20%;
}
#messageWindow {
margin: 0;
padding: 0;
height: var(--messageWindowSize);
}
#mainMassage {
height: var(--mainMassageHeight);
}
#userInterface {
font-family: Georgia, 'Times New Roman', Times, serif;
height: var(--userInterfaceHeight);
width: 100%;
display: flex;
margin: 0;
padding: 0;
justify-content: space-evenly;
align-items: center;
}
#userInterface>* {
width: 100%;
height: 100%;
display: flex;
align-items: center;
justify-content: center;
background-color: rgba(0, 0, 0, 0.25);
border: 1px dashed rgba(255, 255, 255, 0.5);
font-size: calc(clamp(100%, 1vh, 5vh))
}
#left,
#right {
height: 100%;
}
#right {
padding-left: 0;
padding-right: 0;
padding-top: 0;
padding-bottom: 0;
}
#left * {
text-align: center;
font-weight: 900;
}
#left .row2 {
justify-content: center;
}
#result {
background: rgb(0, 0, 0);
text-align: center;
}
#result>img {
transform-origin: top;
max-width: 100vw;
max-height: 100vh;
}
.relative {
position: relative;
}
#controller {
list-style-type: none;
z-index: 32768;
}
#controller .input-group {
justify-content: space-around;
align-items: center;
}
#emptySpace {
height: calc(100% - var(--messageWindowSize));
flex-direction: column;
}
#emptySpace>* {
width: 100%;
display: flex;
justify-content: end;
align-items: center;
height: 5%;
font-size: 100%;
}
#emptySpace>*>* {
padding-right: 0.5rem;
}
.watermark {
-webkit-text-stroke: 1px rgb(0, 0, 0);
paint-order: stroke fill;
color: rgba(255, 255, 255, 0.8);
font-size: 100%;
}
.recording {
color: red;
font-weight: bold;
}
</style>
<link id="googleFont" href="https://fonts.googleapis.com/css?family=Noto+Sans+JP" rel="stylesheet">
<style id="fontFamily">
body {
font-family: "Noto Sans JP";
}
</style>
<style id="text-shadowPreview">
#mainMassage input {
text-shadow: 1px -0px 0 rgb(0, 0, 0), 1px 0px 0 rgb(0, 0, 0), 1px 1px 0 rgb(0, 0, 0), 0px 1px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0), -1px 0px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0);
}
</style>
<style id="text-shadow">
</style>
<link href="https://getbootstrap.jp/docs/5.0/examples/sticky-footer-navbar/sticky-footer-navbar.css"
rel="stylesheet">
</head>
<body class="d-flex flex-column bg-dark">
<main class="flex-shrink-0">
<div class="container-fluid">
<div class="row">
<div id="result" class="d-none col-12 p-0 m-0 vh-100"></div>
</div>
<div class="row">
<div class="col-lg-7 p-0 m-0 relative" id="edit">
<div id="emptySpace" class="p-0 m-0">
<div class="recording d-none">
<div>●REC</div>
</div>
<div class="watermark">
<div>Gal-Game UI Editor</div>
</div>
<div class="watermark">
<div>https://huggingface.co/spaces/SenY/GalGameUI</div>
</div>
</div>
<div id="messageWindow" class="row">
<div class="col-12 p-0 m-0 h-100">
<div id="mainMassage" class="row">
<div id="left" class="col-3 p-0 m-0">
<div class="p-0" style="height:100%" id="leftTexts">
</div>
</div>
<div id="right" class="col-9 m-0">
<div class="h-100 p-0" id="rightTexts">
</div>
</div>
</div>
<div id="userInterface" class="text-light d-none">
<div>
<div>Q-Save</div>
</div>
<div>
<div>Q-Load</div>
</div>
<div>
<div>Save</div>
</div>
<div>
<div>Load</div>
</div>
<div>
<div>Auto</div>
</div>
<div>
<div>Skip</div>
</div>
<div>
<div>Log</div>
</div>
<div>
<div>Config</div>
</div>
</div>
</div>
</div>
</div>
<div class="col-lg-5 text-light bg-dark bg-gradient" id="controller">
<h6><u>All images are processed on your browser. No data are sent to any servers.</u></h6>
<div class="input-group">
<button id="render" type="button" class="btn btn-success">Render</button>
</div>
<hr>
<h6>Image</h6>
<div class="input-group">
<label for="upload">
<button id="uploadButton" type="button" class="btn btn-primary">Set Image</button>
</label>
<input id="upload" type="file" class="d-none">
<input id="faceUpload" type="file" class="d-none">
<div class="vr"></div>
<label for="faceUpload">
<button id="faceUploadButton" type="button" class="btn btn-primary">Face Icon</button>
</label>
<label for="faceRemove">
<button id="faceRemoveButton" type="button" class="btn btn-danger">Remove Icon</button>
</label>
</div>
<hr>
<h6>Window</h6>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">TopColor</span>
<input name="topColor" type="color" class="form-control form-control-color" id="topColor"
value="#000000">
<div class="vr"></div>
<span class="input-group-text bg-secondary text-light">TopAlpha</span>
<input name="topAlpha" type="range" class="form-control form-range" id="topAlpha" value="0.35"
step="0.01" max="1">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">BtmColor</span>
<input name="btmColor" type="color" class="form-control form-control-color" id="btmColor"
value="#9999EE">
<div class="vr"></div>
<span class="input-group-text bg-secondary text-light">BtmAlpha</span>
<input name="btmAlpha" type="range" class="form-control form-range" id="btmAlpha" value="0.45"
step="0.01" max="1">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">GoogleFont</span>
<select name="googleFontInput" class="form-control" id="googleFontInput">
<option>Aoboshi One</option>
<option>BIZ UDGothic</option>
<option>BIZ UDMincho</option>
<option>BIZ UDPGothic</option>
<option>BIZ UDPMincho</option>
<option>Cherry Bomb One</option>
<option>Chokokutai</option>
<option>Darumadrop One</option>
<option>Dela Gothic One</option>
<option>DotGothic16</option>
<option>Hachi Maru Pop</option>
<option>Hina Mincho</option>
<option>IBM Plex Sans JP</option>
<option>Kaisei Decol</option>
<option>Kaisei HarunoUmi</option>
<option>Kaisei Opti</option>
<option>Kaisei Tokumin</option>
<option>Kiwi Maru</option>
<option>Klee One</option>
<option>Kosugi</option>
<option>Kosugi Maru</option>
<option>M PLUS 1</option>
<option>M PLUS 1 Code</option>
<option>M PLUS 1p</option>
<option>M PLUS 2</option>
<option>M PLUS Rounded 1c</option>
<option>Mochiy Pop One</option>
<option>Mochiy Pop P One</option>
<option>Monomaniac One</option>
<option>Murecho</option>
<option>New Tegomin</option>
<option selected="selected">Noto Sans JP</option>
<option>Noto Serif JP</option>
<option>Palette Mosaic</option>
<option>Potta One</option>
<option>Rampart One</option>
<option>Reggae One</option>
<option>Rock 3D</option>
<option>RocknRoll One</option>
<option>Sawarabi Gothic</option>
<option>Sawarabi Mincho</option>
<option>Shippori Antique</option>
<option>Shippori Antique B1</option>
<option>Shippori Mincho</option>
<option>Shippori Mincho B1</option>
<option>Shizuru</option>
<option>Slackside One</option>
<option>Stick</option>
<option>Train One</option>
<option>Tsukimi Rounded</option>
<option>Yomogi</option>
<option>Yuji Boku</option>
<option>Yuji Hentaigana Akari</option>
<option>Yuji Hentaigana Akebono</option>
<option>Yuji Mai</option>
<option>Yuji Syuku</option>
<option>Yusei Magic</option>
<option>Zen Antique</option>
<option>Zen Antique Soft</option>
<option>Zen Kaku Gothic Antique</option>
<option>Zen Kaku Gothic New</option>
<option>Zen Kurenaido</option>
<option>Zen Maru Gothic</option>
<option>Zen Old Mincho</option>
</select>
<span class="input-group-text bg-secondary text-light">LocalFont</span>
<input name="fontFamilyInput" type="text" class="form-control" id="fontFamilyInput" value="">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">font-size</span>
<input name="fontSizeInput" type="range" class="form-control form-range" id="fontSizeInput"
value="0.8" step="0.01" min="0.25" max="1">
</div>
<div class="input-group">
<button id="textShadowToggleA" type="button" class="btn btn-primary">Text-Stroke
(Recommended)</button>
<button id="textShadowToggleB" type="button" class="btn btn-primary opacity-50">Text-Stroke
(Shadow)</button>
<div class="vr"></div>
<button id="userInterfaceToggle" type="button"
class="btn btn-primary opacity-50">Fake-UI</button>
</div>
<hr>
<div class="input-group">
<button id="waterMarkToggle" type="button" class="btn btn-warning">WaterMark</button>
<button id="recordingToggle" type="button" class="btn btn-warning opacity-50">●REC</button>
</div>
<hr>
<h6>
<a data-bs-toggle="collapse" href="#collapseExample" role="button" aria-expanded="false"
aria-controls="collapseExample">
Advanced
<i class="fa fa-angle-left text-expanded"></i>
<i class="fa fa-angle-down text-collapsed"></i>
</a>
</h6>
<div class="collapse" id="collapseExample">
<div class="input-group">
<span class="input-group-text bg-secondary text-light">NameSpaceLength</span>
<input name="nameSpaceLength" type="range" class="form-control form-range"
id="nameSpaceLength" value="3" step="1" min="0" max="12">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">fontColor</span>
<input name="fontColor" type="color" class="form-control form-control-color"
id="fontColor" value="#ffffff">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">StrokeColor</span>
<input name="strokeColor" type="color" class="form-control form-control-color"
id="strokeColor" value="#000000">
<span class="input-group-text bg-secondary text-light">StrokeWeight</span>
<input name="strokeWeight" type="range" class="form-control form-range" id="strokeWeight"
value="0.05" step="0.01" min="0.01" max="0.2">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">WindowSize</span>
<input name="windowSize" type="range" class="form-control form-range" id="windowSize"
value="25" step="1" min="10" max="100">
<span class="input-group-text bg-secondary text-light">Padding</span>
<input name="mainMessagePaddingSize" type="range" class="form-control form-range"
id="mainMessagePaddingSize" value="0" step="0.1" min="0" max="2">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">leftLines</span>
<input name="leftLineLength" type="number" class="form-control form-number"
id="leftLineLength" value="4" step="1" min="1">
<span class="input-group-text bg-secondary text-light">rightLines</span>
<input name="rightLineLength" type="number" class="form-control form-number"
id="rightLineLength" value="4" step="1" min="1">
</div>
<div class="input-group">
<span class="input-group-text bg-secondary text-light">Render Scale</span>
<input name="renderScale" type="range" class="form-control form-range" id="renderScale"
value="2" step="0.5" min="1" max="4">
</div>
<hr>
<div class="input-group">
<label for="upload">
<button id="resetStorage" type="button" class="btn btn-danger">ResetCookies</button>
</label>
</div>
</div>
</div>
<div class="col-lg-12 text-light bg-dark bg-gradient p-4" id="help">
<h1>ヘルプ</h1>
<h2>概要</h2>
<p>ギャルゲーのスクリーンショット風画像の生成を補助するツールです。</p>
<p>画像データは全てブラウザ上で処理されサーバーに送信されたりはしませんのでお好きな画像をお使いください。</p>
<p>AI生成された画像に限らず好きな画像を利用可能です。</p>
<p>生成される画像に関しては当方は一切権利を主張しませんので元画像の権利に問題が無ければ商用・非商用を問わず自由にご利用頂けます。</p>
<p>どのような使い方をされているのか見たいのでここの<a
href="https://huggingface.co/spaces/SenY/GalGameUI">URL</a>付きで画像を掲載して頂いたり<a
href="https://twitter.com/user_vkff5833/status/1699858475193626909"
target="_blank">Twitter</a>のリプライや引用RTなどで教えて頂けると嬉しいです。
<p>このソフトウェア(webアプリ、HTMLファイル、Javascript)そのものに対するライセンスは付与していないので転載・改変はご遠慮ください。</p>
<h2>基本的な使い方</h2>
<ul>
<li>"Set Image"を押して画像を設定する。</li>
<li>名前欄とテキスト欄に好きなメッセージを入れる</li>
<li>Renderを押すと画像が生成されるのでそれを保存する。</li>
</ul>
<h2>最近の更新</h2>
<ul>
<li>
<ul>
<li>Google Fontsからフォントを選択できるようにした。</li>
<li>FontColorの変更に対応した。</li>
</ul>
</li>
<li>
<ul>
<li>名前欄とメッセージ欄の行数を変更できるようにした。</li>
<li>メッセージウィンドウの高さ変動の幅を10%~100%に拡張した。</li>
<li>UIを整理すると共にAdvanced部を折りたたむようにした。</li>
<li>ストロークの色と太さの変更に対応した。</li>
<li>ストロークの選択をデフォルトでRecommendedにすると共に、プレビューは常にプレビュー用のCSSを使うよう仕様変更した。</li>
<li>名前欄の左右の比率を非表示を含めて調整出来るようにした。</li>
</ul>
</li>
</ul>
<h2>設定項目</h2>
<ul>
<li>Face Icon</li>
<li>Remove Icon</li>
<li>
<ul>
<li>顔アイコンを設定・削除する。背景透過済画像を推奨</li>
</ul>
</li>
<li>Window Color</li>
<li>
<ul>
<li>メッセージウィンドウの背景色と透過度を上下でそれぞれ設定</li>
</ul>
</li>
<li>Font</li>
<li>font-family</li>
<li>
<ul>
<li>デフォルト以外のフォントを利用したい場合はフォント名をここに入力。</li>
</ul>
</li>
<li>font-size</li>
<li>
<ul>
<li>フォントの大きさを設定。</li>
</ul>
</li>
<li>Text-Stroke</li>
<li>
<ul>
<li>Recommended</li>
<li>
<ul>
<li>-webkit-text-strokeをフォントサイズの5%で設定したもの。大抵の場合実際のレンダリングではこちらのほうが綺麗に出るがプレビューでは表示がおかしくなる。
</li>
</ul>
</li>
<li>Shadow</li>
<li>
<ul>
<li>text-shadowで8方向に1pxずつ置いてレンダリングしたもの。webkit意外の場合など</li>
</ul>
</li>
</ul>
</li>
<li>Fake-UI</li>
<li>
<ul>
<li>セーブボタン等のUIっぽいものの表示・非表示を切り替える。</li>
<li>それぞれのボタンはクリックすることで消えるがここでON・OFFをすることで全て復活します。デザインの都合でボタンの数などを減らしたい時に。</li>
</ul>
</li>
<li>Advanced</li>
<li>
<ul>
<li>その他のちょっとややこしい設定など</li>
<li>Window Size</li>
<li>
<ul>
<li>メッセージウィンドウの高さそのものを調整。</li>
</ul>
</li>
<li>Padding</li>
<li>
<ul>
<li>メッセージウィンドウの外周に対するテキスト表示エリアの隙間の大きさ。行間の調整などに。</li>
</ul>
</li>
<li>StrokeColor, StrokeWeight</li>
<li>
<ul>
<li>文字の縁取りと太さの設定。Weightはプレビューには反映されません。</li>
</ul>
</li>
<li>Render Scale</li>
<li>
<ul>
<li>画像としてレンダリングする時の解像度設定。初期値のまま動かさないことを推奨。</li>
</ul>
</li>
<li>WaterMark</li>
<li>
<ul>
<li>右上のURL透かしのON・OFFを切り替えます。現時点では透かしをオフにすることに対する制限は特に設けていません。</li>
</ul>
</li>
<li>Reset Cookies</li>
<li>
<ul>
<li>保存されている過去のパラメータを全て削除して初期化。</li>
</ul>
</li>
</ul>
</li>
</ul>
</div>
</div>
</div>
</main>
<img id="face" class="d-none">
<img id="image"
src=""
class="d-none">
<script src="https://unpkg.com/bootstrap@5.3.1/dist/js/bootstrap.bundle.min.js" crossorigin="anonymous"></script>
<script src="https://unpkg.com/html2canvas@1.4.1/dist/html2canvas.min.js" crossorigin="anonymous"></script>
<script>
const hex2rgb = function (hex) {
if (hex.slice(0, 1) == "#") hex = hex.slice(1);
if (hex.length == 3) hex = hex.slice(0, 1) + hex.slice(0, 1) + hex.slice(1, 2) + hex.slice(1, 2) + hex.slice(2, 3) + hex.slice(2, 3);
return [hex.slice(0, 2), hex.slice(2, 4), hex.slice(4, 6)].map(function (str) {
return parseInt(str, 16);
});
}
const createEditor = function () {
let image = document.getElementById("image");
let face = document.getElementById("face");
let naturalAspect = image.naturalWidth / image.naturalHeight;
let height = image.naturalHeight;
let width = height * naturalAspect;
let col = 12;
while (width > window.document.documentElement.clientWidth || height > window.document.documentElement.clientHeight) {
height--;
width = height * naturalAspect;
}
document.getElementById("edit").style.width = width + 'px';
document.getElementById("edit").style.height = height + 'px';
document.getElementById("left").setAttribute("class", "p-0 m-0");
document.getElementById("right").setAttribute("class", "");
let lc = "col-" + document.getElementById("nameSpaceLength").value;
let rc = "col-" + (12 - document.getElementById("nameSpaceLength").value);
if (document.getElementById("nameSpaceLength").value == 0) {
lc = "d-none";
}
if (document.getElementById("nameSpaceLength").value == 12) {
rc = "d-none";
}
document.getElementById("left").classList.add(lc);
document.getElementById("right").classList.add(rc);
document.getElementById("right").style.paddingTop = document.getElementById("mainMessagePaddingSize").value + 'rem';
document.getElementById("right").style.paddingBottom = document.getElementById("mainMessagePaddingSize").value + 'rem';
document.getElementById("rightTexts").textContent = "";
for (let i = 0; i < document.getElementById("rightLineLength").value; i++) {
let row2 = document.createElement("div");
let input = document.createElement("input");
let shadowText = document.createElement("div");
shadowText.setAttribute("class", "shadow-text d-none text-light");
input.name = "rightText" + (i + 1);
row2.classList.add("row2");
row2.appendChild(input);
row2.appendChild(shadowText);
input.classList.add("text-light");
input.value = "";
if (!window.localStorage.length) {
input.value = "Line of " + (i + 1);
}
let v = window.localStorage.getItem(input.name);
if (v) {
input.value = v;
}
input.addEventListener("change", function (e) {
if (input.name && input.value) {
window.localStorage.setItem(input.name, input.value);
}
if (input.name && !input.value) {
window.localStorage.removeItem(input.name);
}
});
document.getElementById("rightTexts").appendChild(row2);
}
document.getElementById("leftTexts").textContent = "";
for (let i = 0; i < document.getElementById("leftLineLength").value; i++) {
let row2 = document.createElement("div");
let input = document.createElement("input");
let shadowText = document.createElement("div");
shadowText.setAttribute("class", "shadow-text d-none text-light");
input.name = "leftText" + (i + 1);
row2.classList.add("row2");
row2.appendChild(input);
row2.appendChild(shadowText);
input.classList.add("text-light");
input.value = "";
if (!window.localStorage.length && i == 2) {
input.value = "Name";
}
let v = window.localStorage.getItem(input.name);
if (v) {
input.value = v;
}
input.addEventListener("change", function (e) {
if (input.name && input.value) {
window.localStorage.setItem(input.name, input.value);
}
if (input.name && !input.value) {
window.localStorage.removeItem(input.name);
}
});
document.getElementById("leftTexts").appendChild(row2);
}
//linear-gradient(rgba(170, 100, 101, 0.7), rgba(145, 152, 229, 0.9))
let bg = "linear-gradient(rgba(" + hex2rgb(document.getElementById("topColor").value) + ", " + document.getElementById("topAlpha").value + "), rgba(" + hex2rgb(document.getElementById("btmColor").value) + ", " + document.getElementById("btmAlpha").value + "))";
document.getElementById("messageWindow").style.backgroundImage = bg;
document.querySelectorAll(".row2").forEach(el => {
el.style.height = (100 / Array.from(el.parentElement.querySelectorAll(".row2")).length) + "%";
});
let sparse = 0;
if (document.querySelectorAll("#left .row2").length > 1) {
sparse = 100 / (document.querySelectorAll("#left .row2").length - 1);
}
document.getElementById("left").style.height = (100 + sparse) + "%";
document.getElementById("left").style.top = (-1 * sparse) + "%";
document.querySelectorAll("#right input, #left input, #emptySpace>*>*").forEach(el => {
el.style.fontSize = (el.parentNode.offsetHeight * document.querySelector("#fontSizeInput").value) + "px";
});
}
document.querySelectorAll("input").forEach(el => {
addEventListener("change", function (e) {
if (el.name && el.value) {
window.localStorage.setItem(el.name, el.value);
}
if (el.name && !el.value) {
window.localStorage.removeItem(el.name);
}
});
})
const saveImage = function (image) {
document.getElementById("controller").classList.add("d-none");
let edit = document.getElementById("edit");
let result = document.getElementById("result");
let naturalAspect = image.naturalWidth / image.naturalHeight;
let width = window.document.documentElement.clientHeight * naturalAspect
let height = window.document.documentElement.clientHeight;
result.classList.remove("d-none");
result.textContent = "";
let loading = document.createElement("div");
result.appendChild(loading);
loading.classList.add("spinner-border");
loading.classList.add("text-primary");
createEditor();
window.scrollTo(0, 0);
setTimeout(function () {
html2canvas(document.getElementById("edit"), {
scale: document.getElementById("renderScale").value,
windowWidth: image.naturalWidth * 2,
windowHeight: image.naturalHeight * 2,
width: image.naturalWidth,
height: image.naturalHeight,
onclone: (x) => {
x.getElementById("edit").style.width = image.naturalWidth + "px";
x.getElementById("edit").style.height = image.naturalHeight + "px";
x.getElementById("edit").classList.add("m-0");
x.getElementById("edit").classList.add("p-0");
x.querySelectorAll(".shadow-text").forEach(el => {
el.classList.remove("d-none");
el.style.fontSize = (el.parentNode.offsetHeight * document.querySelector("#fontSizeInput").value) + "px";
el.textContent = el.parentNode.querySelector("input").value;
el.parentNode.querySelector("input").classList.add("d-none");
});
return x;
}
}).then(canvas => {
result.textContent = "";
let i = document.createElement("img");
i.setAttribute("src", canvas.toDataURL('image/png'));
result.appendChild(i);
});
}, 1);
}
document.getElementById("result").addEventListener("click", function (e) {
document.getElementById("result").classList.add("d-none");
document.getElementById("controller").classList.remove("d-none");
let image = document.getElementById("image");
document.getElementById("edit").style.backgroundImage = 'url("' + image.getAttribute("src") + '")';
createEditor();
});
document.getElementById("render").addEventListener("click", function (e) {
saveImage(image);
});
document.querySelectorAll("#controller input").forEach(el => {
el.addEventListener("change", function () {
createEditor();
});
});
let image = document.getElementById("image");
document.getElementById("edit").style.backgroundImage = 'url("' + image.getAttribute("src") + '")';
createEditor();
document.getElementById("upload").addEventListener("change", function () {
let reader = new FileReader();
let u = this;
reader.onload = function () {
let image = document.getElementById("image");
image.setAttribute("src", reader.result);
document.getElementById("edit").style.backgroundImage = 'url("' + image.getAttribute("src") + '")';
setTimeout(function () {
createEditor();
u.value = null;
}, 500);
}
reader.readAsDataURL(u.files[0]);
});
document.getElementById("faceUpload").addEventListener("change", function () {
let reader = new FileReader();
let u = this;
reader.onload = function () {
let face = document.getElementById("face");
face.setAttribute("src", reader.result);
document.getElementById("left").style.backgroundImage = 'url("' + face.getAttribute("src") + '")';
setTimeout(function () {
createEditor();
u.value = null;
}, 500);
}
reader.readAsDataURL(u.files[0]);
});
document.getElementById("uploadButton").addEventListener("click", function () {
document.getElementById("upload").click();
});
document.getElementById("faceUploadButton").addEventListener("click", function () {
document.getElementById("faceUpload").click();
});
document.getElementById("resetStorage").addEventListener("click", function () {
window.localStorage.clear();
alert("All settings have been initialized. Please reload to activate.");
});
document.getElementById("faceRemoveButton").addEventListener("click", function () {
let face = document.getElementById("face");
face.removeAttribute("src");
document.getElementById("left").style.backgroundImage = "none";
setTimeout(function () {
createEditor();
}, 1);
});
window.onresize = function () {
createEditor();
}
document.getElementById("fontFamilyInput").addEventListener("change", function () {
let f = document.getElementById("fontFamily");
let i = this;
if (!i.value) {
f.textContent = 'body { font-family: "Noto Sans JP"; }';
} else {
f.textContent = 'body { font-family: "' + i.value.trim() + '"; }';
}
});
document.getElementById("googleFontInput").addEventListener("change", function () {
let f = document.getElementById("fontFamily");
let i = this;
document.getElementById("googleFont").setAttribute("href", "https://fonts.googleapis.com/css?family=" + i.value.trim());
f.textContent = 'body { font-family: "' + i.value.trim() + '"; }';
});
document.getElementById("fontColor").addEventListener("change", function () {
document.documentElement.style.setProperty('--bs-light-rgb', hex2rgb(document.getElementById("fontColor").value));
createEditor();
});
document.getElementById("windowSize").addEventListener("change", function () {
document.documentElement.style.setProperty('--messageWindowSize', this.value + "%");
createEditor();
});
const setStroke = function (mode) {
let fontSize = Math.max.apply(null, [document.querySelector("#left input").style.fontSize.replace("px", ""), document.querySelector("#right input").style.fontSize.replace("px", ""), 8]);
let strokeSize = Math.ceil(fontSize * document.getElementById("strokeWeight").value);
let strokeColor = hex2rgb(document.getElementById("strokeColor").value);
let css = '.shadow-text {-webkit-text-stroke: ' + strokeSize + 'px rgb(0, 0, 0) ;paint-order: stroke fill;}'.replaceAll('rgb(0, 0, 0)', 'rgb(' + strokeColor + ')');
let css2 = '#mainMassage input {text-shadow: 1px -0px 0 rgb(0, 0, 0), 1px 0px 0 rgb(0, 0, 0), 1px 1px 0 rgb(0, 0, 0), 0px 1px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0), -1px 0px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0);}'.replaceAll('rgb(0, 0, 0)', 'rgb(' + strokeColor + ')');
document.getElementById("text-shadow").textContent = css;
document.getElementById("text-shadowPreview").textContent = css2;
if (mode == "b") {
document.getElementById("text-shadow").textContent = '.shadow-text {text-shadow: 1px -0px 0 rgb(0, 0, 0), 1px 0px 0 rgb(0, 0, 0), 1px 1px 0 rgb(0, 0, 0), 0px 1px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0), -1px 0px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0);}'.replaceAll('rgb(0, 0, 0)', 'rgb(' + strokeColor + ')');
document.getElementById("text-shadowPreview").textContent = '#mainMassage input {text-shadow: 1px -0px 0 rgb(0, 0, 0), 1px 0px 0 rgb(0, 0, 0), 1px 1px 0 rgb(0, 0, 0), 0px 1px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0), -1px 0px 0 rgb(0, 0, 0), -1px -1px 0 rgb(0, 0, 0);}'.replaceAll('rgb(0, 0, 0)', 'rgb(' + strokeColor + ')');
}
}
setStroke();
document.getElementById("strokeWeight").addEventListener("change", function () {
setStroke();
});
document.getElementById("strokeColor").addEventListener("change", function () {
setStroke();
});
document.getElementById("textShadowToggleA").addEventListener("click", function () {
document.getElementById("textShadowToggleB").classList.add("opacity-50");
this.classList.toggle("opacity-50");
if (this.classList.contains("opacity-50")) {
document.getElementById("text-shadow").textContent = "";
document.getElementById("text-shadowPreview").textContent = "";
} else {
setStroke();
}
});
document.getElementById("textShadowToggleB").addEventListener("click", function () {
document.getElementById("textShadowToggleA").classList.add("opacity-50");
this.classList.toggle("opacity-50");
if (this.classList.contains("opacity-50")) {
document.getElementById("text-shadow").textContent = "";
document.getElementById("text-shadowPreview").textContent = "";
} else {
setStroke("b");
}
});
document.getElementById("userInterfaceToggle").addEventListener("click", function () {
this.classList.toggle("opacity-50");
if (!this.classList.contains("opacity-50")) {
document.documentElement.style.setProperty('--mainMassageHeight', "80%");
document.getElementById("userInterface").classList.remove("d-none");
document.querySelectorAll("#userInterface>*").forEach(el => el.classList.remove("d-none"));
createEditor();
} else {
document.documentElement.style.setProperty('--mainMassageHeight', "100%");
document.getElementById("userInterface").classList.add("d-none");
createEditor();
}
});
document.getElementById("waterMarkToggle").addEventListener("click", function () {
this.classList.toggle("opacity-50");
if (this.classList.contains("opacity-50")) {
document.querySelectorAll(".watermark").forEach(el => el.classList.add("d-none"));
} else {
document.querySelectorAll(".watermark").forEach(el => el.classList.remove("d-none"));
}
createEditor();
});
document.getElementById("recordingToggle").addEventListener("click", function () {
this.classList.toggle("opacity-50");
if (this.classList.contains("opacity-50")) {
document.querySelectorAll(".recording").forEach(el => el.classList.add("d-none"));
} else {
document.querySelectorAll(".recording").forEach(el => el.classList.remove("d-none"));
}
createEditor();
});
document.querySelectorAll("input").forEach(el => {
let v = window.localStorage.getItem(el.name);
if (v) {
el.value = v;
let evt = new Event("change");
el.dispatchEvent(evt)
}
});
document.querySelectorAll("#userInterface > div").forEach(el => {
el.addEventListener("click", function () {
el.classList.add("d-none");
});
});
</script>
</body>
</html>