Dieser Commit ist enthalten in:
Claude Project Manager
2025-09-24 23:22:26 +02:00
Ursprung 1176db42e8
Commit 5e698adf09
2 geänderte Dateien mit 60 neuen und 9 gelöschten Zeilen

Datei anzeigen

@ -374,8 +374,12 @@ export default function OrganizationEditor() {
const handleAutoLayout = () => {
const laneWidth = 420
const xMargin = 40
const yStart = 80
const yGap = 110
// Zweistufiges Top-Band: Direktion ganz oben, darunter weitere ROOT-Knoten
const topYDirector = 10
const topYRootRow = topYDirector + yGap
// Spaltenstart mit Abstand unterhalb der Top-Row, um Überlappungen zu vermeiden
const yStart = topYRootRow + yGap
const newNodes: Node[] = nodes.map(n => ({ ...n, position: { ...n.position } }))
const byId: Record<string, Node> = {}
@ -442,16 +446,36 @@ export default function OrganizationEditor() {
return va - vb
}
// Position Direktion and root-like nodes at the top center
// Position Direktion und ROOT-ähnliche Knoten als Top-Zeilen
const rootNodes = lanesMap.get('ROOT') || []
let globalMinX = xMargin
const totalLanes = laneOrder.filter(k => k !== 'ROOT' && k !== 'UNK').length
const rootX = Math.max(xMargin, (totalLanes * laneWidth) / 2 - 150)
let rootY = 10
rootNodes.forEach(n => {
n.position = { x: rootX, y: rootY }
rootY += yGap
})
const lanesForWidth = laneOrder.filter(k => k !== 'ROOT' && k !== 'UNK')
const laneCount = Math.max(1, lanesForWidth.length)
const totalWidth = (laneCount - 1) * laneWidth
const cardHalfWidth = 150 // angenommene halbe Kartenbreite
// 1) Direktion zentriert ganz oben
const directorIndex = rootNodes.findIndex(n => (String(n.data?.type).toLowerCase() === 'direktion') || String(n.data?.code).toUpperCase() === 'DIR')
let placedIndices = new Set<number>()
if (directorIndex >= 0) {
const centerX = xMargin + totalWidth / 2 - cardHalfWidth
const x = Math.max(xMargin, centerX)
rootNodes[directorIndex].position = { x, y: topYDirector }
placedIndices.add(directorIndex)
}
// 2) Übrige ROOT-Knoten in einer zweiten Top-Reihe gleichmäßig verteilen
const others: Node[] = rootNodes.filter((_, idx) => !placedIndices.has(idx))
if (others.length === 1) {
const x = Math.max(xMargin, xMargin + totalWidth / 2 - cardHalfWidth)
others[0].position = { x, y: topYRootRow }
} else if (others.length > 1) {
const step = others.length > 1 ? (totalWidth / (others.length - 1)) : 0
others.forEach((n, i) => {
const xi = xMargin + i * step - cardHalfWidth
n.position = { x: Math.max(xMargin, xi), y: topYRootRow }
})
}
// Arrange lanes
let laneIndex = 0

Datei anzeigen

@ -83,6 +83,7 @@ function parseOrganizationFromText(text: string) {
continue
}
// Abteilung (inkl. Zentralabteilung erkennen)
const abtMatch = line.match(/Abteilung\s+(\d+|Zentralabteilung)/i)
if (abtMatch) {
const abtNum = abtMatch[1] === 'Zentralabteilung' ? 'ZA' : abtMatch[1]
@ -100,6 +101,32 @@ function parseOrganizationFromText(text: string) {
continue
}
// Zentralabteilung alleinstehend (ohne Präfix "Abteilung")
if (/^Zentralabteilung\b/i.test(line) || /^ZA\b/i.test(line)) {
currentAbteilung = ensure({
code: 'Abt ZA',
name: 'Zentralabteilung',
type: 'abteilung',
level: 1,
parentId: 'DIR',
color: colors['ZA'] || '#6b7280',
hasFuehrungsstelle: false
})
currentDezernat = null
// nicht continue; nachfolgende Muster können weitere Details liefern
}
// Dezernat ZA N (z. B. "Dez ZA 1" bis "Dez ZA 5")
const dezZaMatch = line.match(/^(?:Dezernat|Dez)\s+ZA\s*(\d{1,2})/i)
if (dezZaMatch) {
const zaNum = dezZaMatch[1].trim()
const dezName = line.replace(/^(?:Dezernat|Dez)\s+ZA\s*\d{1,2}\s*-?\s*/i, '').trim() || `Dezernat ZA ${zaNum}`
// Ensure Abt ZA exists
ensure({ code: 'Abt ZA', name: 'Zentralabteilung', type: 'abteilung', level: 1, parentId: 'DIR', color: colors['ZA'] || '#6b7280', hasFuehrungsstelle: false })
currentDezernat = ensure({ code: `Dez ZA ${zaNum}`, name: dezName, type: 'dezernat', level: 2, parentId: 'Abt ZA' })
continue
}
const dezMatch = line.match(/^(?:Dezernat|Dez)\s+([\d]+)/i)
if (dezMatch) {
const dezNum = dezMatch[1].trim()