feat: Vorschaubild für Farbschemata hochladen und anzeigen
This commit is contained in:
12
index.html
12
index.html
@@ -106,6 +106,18 @@
|
|||||||
<input id="schema-name-input" type="text" placeholder="z.B. Website Relaunch">
|
<input id="schema-name-input" type="text" placeholder="z.B. Website Relaunch">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="modal-field">
|
||||||
|
<label>Vorschaubild <span style="color:#aaa;font-size:0.7rem;text-transform:none">(optional)</span></label>
|
||||||
|
<div id="schema-bild-bereich" style="display:flex;align-items:center;gap:0.75rem">
|
||||||
|
<div id="schema-bild-preview" class="schema-bild-preview"></div>
|
||||||
|
<div>
|
||||||
|
<button class="action-btn" id="schema-bild-trigger" type="button">Bild hochladen</button>
|
||||||
|
<input type="file" id="schema-bild-input" accept="image/*" style="display:none">
|
||||||
|
<button class="action-btn" id="schema-bild-entfernen-btn" type="button" style="display:none;margin-left:0.4rem;font-size:0.75rem">Entfernen</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div id="schema-farben-preview" class="schema-farben-preview"></div>
|
<div id="schema-farben-preview" class="schema-farben-preview"></div>
|
||||||
|
|
||||||
<div id="schema-farbe-eingabe">
|
<div id="schema-farbe-eingabe">
|
||||||
|
|||||||
@@ -74,13 +74,14 @@ export function addColorToSchema(hsl) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function saveSchema(name, farben) {
|
export function saveSchema(name, farben, bild) {
|
||||||
const data = load();
|
const data = load();
|
||||||
const existing = data.schemata.find(s => s.name === name);
|
const existing = data.schemata.find(s => s.name === name);
|
||||||
if (existing) {
|
if (existing) {
|
||||||
existing.farben = farben;
|
existing.farben = farben;
|
||||||
|
if (bild !== undefined) existing.bild = bild || null;
|
||||||
} else {
|
} else {
|
||||||
data.schemata.push({ name, farben });
|
data.schemata.push({ name, farben, bild: bild || null });
|
||||||
}
|
}
|
||||||
save(data);
|
save(data);
|
||||||
renderSammlung();
|
renderSammlung();
|
||||||
@@ -227,8 +228,20 @@ export function renderSammlung() {
|
|||||||
const header = document.createElement('div');
|
const header = document.createElement('div');
|
||||||
header.style.cssText = 'display:flex;justify-content:space-between;align-items:center;margin-bottom:0.5rem';
|
header.style.cssText = 'display:flex;justify-content:space-between;align-items:center;margin-bottom:0.5rem';
|
||||||
|
|
||||||
|
// Thumbnail + Name nebeneinander
|
||||||
|
const nameRow = document.createElement('div');
|
||||||
|
nameRow.style.cssText = 'display:flex;align-items:center;gap:0.6rem';
|
||||||
|
|
||||||
|
if (schema.bild) {
|
||||||
|
const thumb = document.createElement('div');
|
||||||
|
thumb.className = 'schema-thumb';
|
||||||
|
thumb.style.backgroundImage = 'url(' + schema.bild + ')';
|
||||||
|
nameRow.appendChild(thumb);
|
||||||
|
}
|
||||||
|
|
||||||
const nameEl = document.createElement('strong');
|
const nameEl = document.createElement('strong');
|
||||||
nameEl.textContent = schema.name;
|
nameEl.textContent = schema.name;
|
||||||
|
nameRow.appendChild(nameEl);
|
||||||
|
|
||||||
const delBtn = document.createElement('button');
|
const delBtn = document.createElement('button');
|
||||||
delBtn.className = 'action-btn';
|
delBtn.className = 'action-btn';
|
||||||
@@ -236,7 +249,7 @@ export function renderSammlung() {
|
|||||||
delBtn.style.fontSize = '0.75rem';
|
delBtn.style.fontSize = '0.75rem';
|
||||||
delBtn.addEventListener('click', () => deleteSchema(schema.name));
|
delBtn.addEventListener('click', () => deleteSchema(schema.name));
|
||||||
|
|
||||||
header.appendChild(nameEl);
|
header.appendChild(nameRow);
|
||||||
header.appendChild(delBtn);
|
header.appendChild(delBtn);
|
||||||
|
|
||||||
const swatchesDiv = document.createElement('div');
|
const swatchesDiv = document.createElement('div');
|
||||||
|
|||||||
@@ -1,14 +1,38 @@
|
|||||||
import { hexToHsl, hslToHex, rgbToHsl, rgbToHex } from './converter.js';
|
import { hexToHsl, hslToHex, rgbToHsl, rgbToHex } from './converter.js';
|
||||||
|
|
||||||
const MAX_FARBEN = 4;
|
const MAX_FARBEN = 4;
|
||||||
|
const THUMB_SIZE = 120; // max px für Vorschaubild
|
||||||
|
|
||||||
// Aktuell gesammelter Zustand des Modals
|
// Aktuell gesammelter Zustand des Modals
|
||||||
let farben = []; // Array von HSL-Objekten
|
let farben = []; // Array von HSL-Objekten
|
||||||
let aktuelleHex = ''; // Hex-Wert des aktuellen Eingabefelds
|
let aktuelleHex = ''; // Hex-Wert des aktuellen Eingabefelds
|
||||||
|
let vorschaubild = null; // Base64-String des Thumbnails oder null
|
||||||
|
|
||||||
|
function compressToThumbnail(file) {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = (e) => {
|
||||||
|
const img = new Image();
|
||||||
|
img.onload = () => {
|
||||||
|
const scale = Math.min(THUMB_SIZE / img.width, THUMB_SIZE / img.height, 1);
|
||||||
|
const w = Math.round(img.width * scale);
|
||||||
|
const h = Math.round(img.height * scale);
|
||||||
|
const c = document.createElement('canvas');
|
||||||
|
c.width = w;
|
||||||
|
c.height = h;
|
||||||
|
c.getContext('2d').drawImage(img, 0, 0, w, h);
|
||||||
|
resolve(c.toDataURL('image/jpeg', 0.8));
|
||||||
|
};
|
||||||
|
img.src = e.target.result;
|
||||||
|
};
|
||||||
|
reader.readAsDataURL(file);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function resetModal() {
|
function resetModal() {
|
||||||
farben = [];
|
farben = [];
|
||||||
aktuelleHex = '';
|
aktuelleHex = '';
|
||||||
|
vorschaubild = null;
|
||||||
document.getElementById('schema-name-input').value = '';
|
document.getElementById('schema-name-input').value = '';
|
||||||
document.getElementById('schema-hex-input').value = '';
|
document.getElementById('schema-hex-input').value = '';
|
||||||
document.getElementById('schema-hex-preview').style.background = '#eee';
|
document.getElementById('schema-hex-preview').style.background = '#eee';
|
||||||
@@ -17,6 +41,12 @@ function resetModal() {
|
|||||||
document.getElementById('schema-canvas').getContext('2d').clearRect(0, 0, 1, 1);
|
document.getElementById('schema-canvas').getContext('2d').clearRect(0, 0, 1, 1);
|
||||||
document.getElementById('schema-farbe-eingabe').style.display = 'block';
|
document.getElementById('schema-farbe-eingabe').style.display = 'block';
|
||||||
document.getElementById('schema-abschliessen-btn').disabled = true;
|
document.getElementById('schema-abschliessen-btn').disabled = true;
|
||||||
|
// Vorschaubild-UI zurücksetzen
|
||||||
|
const preview = document.getElementById('schema-bild-preview');
|
||||||
|
preview.style.backgroundImage = '';
|
||||||
|
preview.classList.remove('hat-bild');
|
||||||
|
document.getElementById('schema-bild-input').value = '';
|
||||||
|
document.getElementById('schema-bild-entfernen-btn').style.display = 'none';
|
||||||
updateFarbeLabel();
|
updateFarbeLabel();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -186,7 +216,31 @@ export function initSchemaModal(onSave) {
|
|||||||
}
|
}
|
||||||
if (farben.length === 0) return;
|
if (farben.length === 0) return;
|
||||||
|
|
||||||
onSave(name, farben);
|
onSave(name, farben, vorschaubild);
|
||||||
document.getElementById('schema-modal-overlay').style.display = 'none';
|
document.getElementById('schema-modal-overlay').style.display = 'none';
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Vorschaubild hochladen
|
||||||
|
document.getElementById('schema-bild-trigger').addEventListener('click', () => {
|
||||||
|
document.getElementById('schema-bild-input').click();
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('schema-bild-input').addEventListener('change', async (e) => {
|
||||||
|
const file = e.target.files[0];
|
||||||
|
if (!file) return;
|
||||||
|
vorschaubild = await compressToThumbnail(file);
|
||||||
|
const preview = document.getElementById('schema-bild-preview');
|
||||||
|
preview.style.backgroundImage = 'url(' + vorschaubild + ')';
|
||||||
|
preview.classList.add('hat-bild');
|
||||||
|
document.getElementById('schema-bild-entfernen-btn').style.display = 'inline-block';
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('schema-bild-entfernen-btn').addEventListener('click', () => {
|
||||||
|
vorschaubild = null;
|
||||||
|
const preview = document.getElementById('schema-bild-preview');
|
||||||
|
preview.style.backgroundImage = '';
|
||||||
|
preview.classList.remove('hat-bild');
|
||||||
|
document.getElementById('schema-bild-input').value = '';
|
||||||
|
document.getElementById('schema-bild-entfernen-btn').style.display = 'none';
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
24
style.css
24
style.css
@@ -225,6 +225,30 @@ button.action-btn:hover { background: #f0f0f0; }
|
|||||||
|
|
||||||
.schema-dropzone p { margin-bottom: 0.5rem; }
|
.schema-dropzone p { margin-bottom: 0.5rem; }
|
||||||
|
|
||||||
|
/* Vorschaubild im Modal */
|
||||||
|
.schema-bild-preview {
|
||||||
|
width: 56px;
|
||||||
|
height: 56px;
|
||||||
|
border-radius: 8px;
|
||||||
|
border: 2px dashed #ddd;
|
||||||
|
background: #f5f5f5;
|
||||||
|
background-size: cover;
|
||||||
|
background-position: center;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
.schema-bild-preview.hat-bild { border-style: solid; border-color: #ccc; }
|
||||||
|
|
||||||
|
/* Thumbnail in der Sammlung */
|
||||||
|
.schema-thumb {
|
||||||
|
width: 40px;
|
||||||
|
height: 40px;
|
||||||
|
border-radius: 6px;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
background-size: cover;
|
||||||
|
background-position: center;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* --- Finale Styles --- */
|
/* --- Finale Styles --- */
|
||||||
h2 { font-size: 1.1rem; margin-bottom: 1rem; }
|
h2 { font-size: 1.1rem; margin-bottom: 1rem; }
|
||||||
h3 { font-size: 0.95rem; margin-bottom: 0.5rem; }
|
h3 { font-size: 0.95rem; margin-bottom: 0.5rem; }
|
||||||
|
|||||||
Reference in New Issue
Block a user