geoproject-app/src/components/editor/TextSettings.vue

170 lines
5.1 KiB
Vue
Raw Normal View History

<template>
2025-12-10 11:10:14 +01:00
<section class="settings-section" id="settings-section_elem" data-color-type="elem">
2026-03-05 11:42:18 +01:00
<h2>Réglage du texte par défaut</h2>
2025-12-09 17:08:40 +01:00
<div class="container">
<p class="infos">
Ces réglages s'appliquent à l'ensemble des éléments du document. Vous
pouvez modifier ensuite les éléments indépendamment.
</p>
<!-- Police -->
<div class="settings-subsection">
2025-12-09 17:08:40 +01:00
<div class="field field-font">
2026-03-05 16:29:42 +01:00
<label for="text-font" class="label-with-tooltip" data-css="font-family">Police</label>
2026-03-05 11:08:44 +01:00
<div class="field-font__options">
2026-03-05 16:29:42 +01:00
<select id="text-font" v-model="font">
<option value="sans-serif">Police système (sans-serif)</option>
<option v-for="f in projectFonts" :key="f.name" :value="f.name" :style="{ fontFamily: `'${f.name}', ${f.category}` }">{{ f.name }}</option>
</select>
</div>
</div>
</div>
<!-- Taille du texte -->
<div class="settings-subsection">
2025-12-11 13:38:27 +01:00
<div class="field field-text-size">
<label for="text-size-range" class="label-with-tooltip" data-css="font-size">Taille du texte</label>
<InputWithUnit
v-model="fontSize"
:units="['px']"
:min="8"
:max="72"
showRange
/>
</div>
</div>
<!-- Interlignage -->
<div class="settings-subsection">
<div class="field field-text-size">
<label for="text-lineheight-range" class="label-with-tooltip" data-css="line-height">Interlignage</label>
<InputWithUnit
v-model="lineHeight"
:units="['px']"
:min="0"
:max="72"
showRange
/>
</div>
</div>
<!-- Couleurs -->
<div class="settings-subsection">
2025-12-09 17:08:40 +01:00
<div class="field field-simple">
<label for="text-color" class="label-with-tooltip" data-css="color">Couleur</label>
<div class="input-with-color">
<input
ref="colorInput"
id="text-color"
type="text"
v-model="color"
class="color-input"
data-coloris
/>
</div>
</div>
2025-12-09 17:08:40 +01:00
</div>
</div>
</section>
</template>
<script setup>
import { ref, watch, onMounted } from 'vue';
import { initColoris } from '../../composables/useColoris';
import InputWithUnit from '../ui/InputWithUnit.vue';
import { useCssUpdater } from '../../composables/useCssUpdater';
import { useCssSync } from '../../composables/useCssSync';
import { useDebounce } from '../../composables/useDebounce';
2026-03-05 14:49:58 +01:00
import { useTextDefaults } from '../../composables/useTextDefaults';
2026-03-05 16:29:42 +01:00
import { useProjectFonts } from '../../composables/useProjectFonts';
2026-03-05 11:42:18 +01:00
const { updateStyle } = useCssUpdater();
const { extractValue, extractNumericValue } = useCssSync();
const { debouncedUpdate } = useDebounce(500);
2026-03-05 14:49:58 +01:00
const textDefaults = useTextDefaults();
2026-03-05 16:29:42 +01:00
const { fonts: projectFonts, loadFont, loadAllFontPreviews } = useProjectFonts();
// State
2026-03-05 16:29:42 +01:00
const font = ref('sans-serif');
const fontSize = ref({ value: 16, unit: 'px' });
const lineHeight = ref({ value: 20, unit: 'px' });
const color = ref('rgb(0, 0, 0)');
const colorInput = ref(null);
let isUpdatingFromStore = false;
// Watchers for body styles
2026-03-05 16:29:42 +01:00
watch(font, async (val) => {
textDefaults.fontFamily = val;
2026-03-05 16:29:42 +01:00
if (val !== 'sans-serif') await loadFont(val);
if (isUpdatingFromStore) return;
2026-03-05 16:29:42 +01:00
const cssValue = val === 'sans-serif' ? 'sans-serif' : `"${val}"`;
updateStyle('body', 'font-family', cssValue);
}, { immediate: true });
watch(color, (val) => {
textDefaults.color = val;
if (isUpdatingFromStore) return;
updateStyle('body', 'color', val);
}, { immediate: true });
watch(fontSize, (val) => {
2026-03-05 14:49:58 +01:00
textDefaults.fontSize = { value: val.value, unit: val.unit };
if (isUpdatingFromStore) return;
debouncedUpdate(() => {
updateStyle('p', 'font-size', `${val.value}${val.unit}`);
});
2026-03-05 14:49:58 +01:00
}, { deep: true, immediate: true });
watch(lineHeight, (val) => {
textDefaults.lineHeight = { value: val.value, unit: val.unit };
if (isUpdatingFromStore) return;
debouncedUpdate(() => {
updateStyle('p', 'line-height', `${val.value}${val.unit}`);
});
}, { deep: true, immediate: true });
// Sync from store
const syncFromStore = () => {
isUpdatingFromStore = true;
2026-03-05 16:29:42 +01:00
const fontVal = extractValue('body', 'font-family');
if (fontVal) {
const cleaned = fontVal.replace(/['"]/g, '').trim();
font.value = cleaned || 'sans-serif';
}
const colorVal = extractValue('body', 'color');
if (colorVal) color.value = colorVal;
const fontSizeVal = extractNumericValue('p', 'font-size', ['px']); // ['px', 'em', 'rem']
if (fontSizeVal) fontSize.value = fontSizeVal;
const lineHeightVal = extractNumericValue('p', 'line-height', ['px']);
if (lineHeightVal) lineHeight.value = lineHeightVal;
isUpdatingFromStore = false;
};
const updateColorisButtons = () => {
2026-03-05 11:42:18 +01:00
if (colorInput.value) {
colorInput.value.dispatchEvent(new Event('input', { bubbles: true }));
}
};
onMounted(() => {
initColoris({
format: 'auto',
swatches: ['#000000', '#FFFFFF', '#FF0000', '#00FF00', '#0000FF', 'transparent'],
});
syncFromStore();
setTimeout(updateColorisButtons, 100);
2026-03-05 16:29:42 +01:00
loadAllFontPreviews();
});
</script>