Files
dreport/frontend/src/components/panels/ToolboxPanel.vue
2026-04-03 01:26:54 +03:00

307 lines
7.4 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<script setup lang="ts">
import { useEditorStore } from '../../stores/editor'
import { useSchemaStore } from '../../stores/schema'
import type { TemplateElement, RepeatingTableElement, TableColumn, ImageElement, PageNumberElement, BarcodeElement, PageBreakElement, CurrentDateElement, ShapeElement, CheckboxElement, CalculatedTextElement, RichTextElement } from '../../core/types'
import { sz } from '../../core/types'
import { schemaFormatToFormatType, defaultAlignForSchema } from '../../core/schema-parser'
const editorStore = useEditorStore()
const schemaStore = useSchemaStore()
let idCounter = Date.now()
function nextId(prefix: string) {
return `${prefix}_${(++idCounter).toString(36)}`
}
interface ToolItem {
label: string
icon: string
create: () => TemplateElement
}
const tools: ToolItem[] = [
{
label: 'Metin',
icon: 'T',
create: () => ({
id: nextId('txt'),
type: 'static_text',
position: { type: 'flow' },
size: { width: sz.auto(), height: sz.auto() },
style: { fontSize: 11, color: '#000000' },
content: 'Yeni metin',
}),
},
{
label: 'Zengin Metin',
icon: 'R',
create: (): RichTextElement => ({
id: nextId('rt'),
type: 'rich_text',
position: { type: 'flow' },
size: { width: sz.auto(), height: sz.auto() },
style: { fontSize: 11, color: '#000000' },
content: [
{ text: 'Zengin ', style: {} },
{ text: 'metin', style: { fontWeight: 'bold' } },
],
}),
},
{
label: 'Container',
icon: '▢',
create: () => ({
id: nextId('cnt'),
type: 'container',
position: { type: 'flow' },
size: { width: sz.fr(1), height: sz.auto() },
direction: 'column' as const,
gap: 3,
padding: { top: 5, right: 5, bottom: 5, left: 5 },
align: 'stretch' as const,
justify: 'start' as const,
style: {},
children: [],
}),
},
{
label: 'Cizgi',
icon: '—',
create: () => ({
id: nextId('ln'),
type: 'line',
position: { type: 'flow' },
size: { width: sz.fr(1), height: sz.auto() },
style: { strokeColor: '#000000', strokeWidth: 0.5 },
}),
},
{
label: 'Tablo',
icon: '▤',
create: (): RepeatingTableElement => {
// Schema'daki ilk array alanını bul ve sütunları otomatik doldur
const arrays = schemaStore.arrayFields
const firstArray = arrays[0]
let dataPath = ''
let columns: TableColumn[] = []
if (firstArray) {
dataPath = firstArray.path
const itemFields = schemaStore.getArrayItemFields(firstArray.path)
columns = itemFields.map(field => ({
id: nextId('col'),
field: field.key,
title: field.title,
width: sz.auto(),
align: defaultAlignForSchema(field),
format: schemaFormatToFormatType(field.format),
}))
}
return {
id: nextId('tbl'),
type: 'repeating_table',
position: { type: 'flow' },
size: { width: sz.fr(1), height: sz.auto() },
dataSource: { type: 'array', path: dataPath },
columns,
style: {
headerBg: '#f0f0f0',
headerColor: '#000000',
fontSize: 10,
headerFontSize: 10,
},
repeatHeader: true,
}
},
},
{
label: 'Gorsel',
icon: '🖼',
create: (): ImageElement => ({
id: nextId('img'),
type: 'image',
position: { type: 'flow' },
size: { width: sz.fixed(40), height: sz.fixed(30) },
style: { objectFit: 'contain' },
}),
},
{
label: 'Sayfa No',
icon: '#',
create: (): PageNumberElement => ({
id: nextId('pgn'),
type: 'page_number',
position: { type: 'flow' },
size: { width: sz.auto(), height: sz.auto() },
style: { fontSize: 10, color: '#666666', align: 'center' },
format: '{current} / {total}',
}),
},
{
label: 'Barkod',
icon: '⣿',
create: (): BarcodeElement => ({
id: nextId('bc'),
type: 'barcode',
position: { type: 'flow' },
size: { width: sz.fixed(30), height: sz.auto() },
format: 'qr',
value: 'https://example.com',
style: {},
}),
},
{
label: 'Onay Kutusu',
icon: '☑',
create: (): CheckboxElement => ({
id: nextId('cb'),
type: 'checkbox',
position: { type: 'flow' },
size: { width: sz.auto(), height: sz.auto() },
checked: false,
style: { size: 4, checkColor: '#000000', borderColor: '#333333', borderWidth: 0.3 },
}),
},
{
label: 'Sekil',
icon: '⬜',
create: (): ShapeElement => ({
id: nextId('shp'),
type: 'shape',
position: { type: 'flow' },
size: { width: sz.fr(1), height: sz.fixed(20) },
shapeType: 'rectangle',
style: { backgroundColor: '#f0f0f0', borderColor: '#333333', borderWidth: 0.5 },
}),
},
{
label: 'Hesaplanan',
icon: 'ƒ',
create: (): CalculatedTextElement => ({
id: nextId('calc'),
type: 'calculated_text',
position: { type: 'flow' },
size: { width: sz.auto(), height: sz.auto() },
expression: '0',
style: { fontSize: 11, color: '#000000' },
}),
},
{
label: 'Tarih',
icon: '📅',
create: (): CurrentDateElement => ({
id: nextId('dt'),
type: 'current_date',
position: { type: 'flow' },
size: { width: sz.auto(), height: sz.auto() },
style: { fontSize: 10, color: '#666666' },
format: 'DD.MM.YYYY',
}),
},
{
label: 'Sayfa Sonu',
icon: '⏎',
create: (): PageBreakElement => ({
id: nextId('pb'),
type: 'page_break',
}),
},
]
function onDragStart(e: DragEvent, tool: ToolItem) {
const el = tool.create()
editorStore.startDragNewElement(el)
// Drag data (fallback)
e.dataTransfer?.setData('text/plain', el.id)
if (e.dataTransfer) {
e.dataTransfer.effectAllowed = 'copy'
}
}
function onDragEnd() {
editorStore.endDragNewElement()
}
</script>
<template>
<div class="toolbox-panel">
<div class="toolbox-panel__title">Arac Kutusu</div>
<div class="toolbox-panel__grid">
<div
v-for="tool in tools"
:key="tool.label"
class="toolbox-item"
draggable="true"
@dragstart="(e) => onDragStart(e, tool)"
@dragend="onDragEnd"
>
<span class="toolbox-item__icon">{{ tool.icon }}</span>
<span class="toolbox-item__label">{{ tool.label }}</span>
</div>
</div>
</div>
</template>
<style scoped>
.toolbox-panel {
padding: 12px;
}
.toolbox-panel__title {
font-size: 11px;
font-weight: 600;
color: #64748b;
text-transform: uppercase;
letter-spacing: 0.5px;
margin-bottom: 10px;
}
.toolbox-panel__grid {
display: flex;
flex-direction: column;
gap: 4px;
}
.toolbox-item {
display: flex;
align-items: center;
gap: 8px;
padding: 8px 10px;
background: white;
border: 1px solid #e2e8f0;
border-radius: 6px;
cursor: grab;
font-size: 13px;
color: #334155;
transition: all 0.15s;
user-select: none;
}
.toolbox-item:hover {
background: #eff6ff;
border-color: #bfdbfe;
}
.toolbox-item:active {
cursor: grabbing;
}
.toolbox-item__icon {
width: 24px;
height: 24px;
display: flex;
align-items: center;
justify-content: center;
background: #f1f5f9;
border-radius: 4px;
font-size: 14px;
font-weight: 600;
color: #475569;
}
.toolbox-item__label {
font-size: 13px;
}
</style>