
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>D3 Radial Tree Layout - Location Hierarchy</title>
    <script src="https://d3js.org/d3.v7.min.js"></script>
    <style>
        body {
            font-family: Arial, sans-serif;
            margin: 0;
            padding: 0;
            display: flex;
            justify-content: center;
            align-items: center;
            min-height: 100vh;
            background-color: #f0f0f0;
        }
        #tree-container {
            background-color: white;
            border-radius: 8px;
            box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
            overflow: hidden;
        }
        .node circle {
            fill: #fff;
            stroke: steelblue;
            stroke-width: 2px;
            transition: all 0.3s ease;
        }
        .node text {
            font: 11px sans-serif;
            transition: all 0.3s ease;
            pointer-events: none;
        }
        .link {
            fill: none;
            stroke: #ccc;
            stroke-width: 1.5px;
        }
        .node:hover circle {
            fill: #f8f8f8;
            stroke: #4CAF50;
            r: 6;
        }
        .node:hover text {
            font-size: 13px;
            font-weight: bold;
        }
        .controls {
            position: absolute;
            top: 10px;
            left: 10px;
            z-index: 1000;
        }
        .control-btn {
            margin: 5px;
            padding: 8px 12px;
            background: #007bff;
            color: white;
            border: none;
            border-radius: 4px;
            cursor: pointer;
        }
        .control-btn:hover {
            background: #0056b3;
        }
    </style>
</head>
<body>
    <div class="controls">
        <button class="control-btn" onclick="resetZoom()">Reset Zoom</button>
        <button class="control-btn" onclick="toggleLabels()">Toggle Labels</button>
    </div>
    <div id="tree-container"></div>
    <script>
        // JSON 형식의 데이터 삽입
        const data = {'respiratory': {'chest': ['chest', 'chest cage'], 'costophrenic_angles': ['costodiaphragmatic angles', 'bilateral costophrenic angles', 'costophrenic angle', 'costophrenic angles'], 'lung': ['bilateral lungs', 'lung', 'bilateral lung', 'either lung', 'lungs', 'lung fields', 'lungs bilaterally', 'both lung fields', 'both lungs', 'pulmonary'], 'pleura': ['pleural surfaces', 'pleural'], 'lobes': ['lobar airspace', 'lobar', 'both lobes of the lung'], 'zone': {'right': ['right lung', 'right-sided lung', 'entire right lung'], 'lateral': ['right lateral lung'], 'middle': ['right mid lung', 'bilateral mid lung', 'mid lung', 'mid lung regions', 'right-mid lung', 'right mid lung field', 'right mid lung zone', 'right middle lung', 'right midlung'], 'upper': ['right upper lung', 'right upper lung field', 'upper lung', 'bilateral upper lung', 'bilateral upper lung zone', 'bilateral upper lungs', 'bilateral upper lungs zone', 'upper lung fields bilaterally', 'upper lungs'], 'left': {'base': ['left lateral basal lung', 'base of the left lung', 'base the left lung', 'basilar left lung', 'left lung base', 'left lung bases'], 'apical': ['left lung apex'], 'lower': ['left lower lung', 'l lower lung field'], 'middle': ['left mid lung', 'left mid lobe', 'left mid lung field', 'left mid lung zone', 'left midlung'], 'upper': ['left upper lung', 'left upper lung field', 'left upper lung zone']}, 'apical': ['bilateral lung apices', 'lung apices', 'lung apices bilaterally', 'right lung apex', 'right medial lung apex'], 'base': ['basilar lung', 'bilateral lung base', 'bilateral lung bases', 'lung base', 'lung bases', 'lung bases bilaterally', 'both lung bases', 'base of the right lung', 'right lung base', 'r lung base', 'right lateral lung bases', 'right lung bases', 'right medial lung base'], 'lingula': ['lingula', 'lingular'], 'lower': ['lower lung', 'lower lung zones', 'lower lungs', 'bilateral lower lung', 'both lower lungs', 'right lower lung field', 'right lower lung zone', 'right lower lungs']}, 'right': ['right chest', 'right chest wall', 'right lower chest wall', 'right costophrenic sinus', 'right costophrenic sulcus', 'right cp angle', 'right diaphragm', 'right-sided diaphragm', 'right fissure', 'right hemidiaphragmatic contour', 'right hemidiaphragm', 'right hilus', 'right hilar region', 'right hilar', 'right hilum', 'right hilar lung', 'right infrahilar', 'right infrahilar region', 'perihilar right-sided', 'right perihilar', 'right perihilar region', 'right pleural', 'right-sided pleural', 'right retrocardiac'], 'hila': ['bilateral hila', 'bilateral hilar', 'hila', 'hilar'], 'alveolar': ['airspace', 'air space', 'alveolar', 'bibasilar airspace'], 'bronchi': ['bronchial'], 'wall': ['bronchial wall'], 'walls': ['chest wall'], 'diaphragm': ['diaphragm', 'diaphragmatic surfaces', 'diaphragmatic'], 'fissures': ['fissural', 'fissure'], 'hemidiaphragm': ['hemidiaphragm'], 'infrahilar': ['infrahilar'], 'interstitial': ['interstitial'], 'thorax': ['intrathoracic', 'thorax'], 'left': ['left chest', 'left chest wall', 'left chest wall laterally', 'left costophrenic sinus', 'left posterior costophrenic', 'left-sided diaphragm', 'left hemidiaphragm', 'left-sided hemidiaphragm', 'left hilus', 'left hilar', 'left hilar area', 'left hilum', 'left infrahilar', 'left infrahilar regions'], 'left, respiratory': {'hila': {'perihilar': {'left': ['left hilar/perihilar']}}}, 'juxtahilar': {'left': ['left juxta-hilar', 'left juxtahilar', 'left juxtahilar region']}, 'middle': ['left mid chest', 'middle lobe', 'right middle lobe'], 'perihilar': ['bilateral perihilar', 'perihilar'], 'other': {'left': ['left pneumectomy cavity'], 'right': ['rightward'], 'specific_location': ['subpulmonic']}, 'retrocardiac': ['retrocardiac'], 'suprahilar': ['suprahilar'], 'lower': ['lower lobes', 'bilateral lower lobe', 'bilateral lower lobes', 'lower lobe', 'lower lobes bilaterally', 'both lower lobes', 'right lower lobe', 'rll'], 'parenchyma': ['lung parenchyma'], 'major': ['major fissure'], 'minor': ['minor fissure'], 'peribronchial': ['peribronchial', 'preibronchial'], 'space': ['pleural cavity', 'pleural space'], 'cardiophrenic_angles': {'right': ['right cardiophrenic angle', 'right cardiophrenic sulcus', 'right cardiophrenic region']}, 'medical device': ['right chest tube'], 'parahilar': {'right': ['right parahilar']}, 'upper': ['right upper lobe', 'bilateral upper lobe', 'bilateral upper lobes', 'upper lobe', 'upper lobes', 'upper lobes bilaterally', 'both upper lobes', 'upper thorax'], 'cavity': ['thoracic cavity'], 'inlet': ['thoracic inlet', 'thoracic inlet bilaterally'], 'trachea': ['trachea', 'tracheal'], 'trachea, respiratory': {'bronchi': {'tree': ['tracheobronchial tree']}}, 'airway': {'upper': ['upper airway', 'upper respiratory airway']}}, 'cardiovascular': {'vessels': ['pulmonary vascularity', 'vascular', 'pulmonary vascular', 'pulmonary vasculature'], 'heart': ['cardiac silhouette', 'cardiomediastinal silhouette', 'cardiothoracic', 'cardiac', 'heart'], 'vascular_spaces_and_junctions': ['aortopulmonary window', 'aortopulmonic window', 'ap window', 'cavoatrial junction', 'junction of brachiocephalic veins', 'pulmonary outflow tract', 'svc/cavoatrial junction', 'svc-ra junction', 'svc/ra junction'], 'heart, mediastinum': ['cardiomediastinal'], 'heart, respiratory': {'lung': ['cardiopulmonary']}, 'vascular_spaces_and_junctions, cardiovascular': {'heart': {'chambers': {'atrium': {'right': ['cavoatrial junction/proximal right atrium']}}}}, 'chambers': {'atrium': {'left': ['left atrial'], 'right': ['right atrium', 'right atrial', 'ra']}, 'ventricle': ['ventricle'], 'right': {'apical': ['right apex ventricle']}}, 'left': ['left cardiac border', 'left heart border'], 'valves': {'mitral': ['mitral annulus', 'mitral annular']}, 'pericardium': ['pericardial'], 'right': ['right heart border', 'right heart'], 'paracardiac': {'right': {'lower': ['right lower paracardiac region']}}, 'arteries': {'pulmonary': ['main pulmonary artery']}}, 'other_anatomical_structures': ['femoral'], 'mediastinum': ['mediastinum', 'mediastinal', 'mediatinal'], 'musculoskeletal': {'bones': ['osseous structures', 'bone'], 'joints': {'acromioclavicular': ['acromioclavicular joints', 'bilateral acromioclavicular joints'], 'glenohumeral': ['bilateral glenohumeral joints'], 'glenoid': ['glenoid'], 'left': ['left ac joint', 'left glenohumeral joint', 'left glenohumeral joints'], 'shoulder': ['bilateral shoulders', 'shoulders bilaterally'], 'right': ['right ac joint', 'right acromioclavicular joint', 'right acromioclavicular', 'right glenohumeral joint'], 'coracoclavicular': ['right coracoclavicular region']}, 'humerus': {'neck': {'proximal': ['proximal humeral neck'], 'right': ['right humeral neck'], 'specific': ['right humeral surgical neck']}, 'right': ['right humeral']}, 'sternum': ['sternal'], 'ribs': ['rib', 'ribs', 'ribcage'], 'right': ['right-sided rib'], 'clavicle': {'right': ['right clavicle', 'mid shaft of the right clavicle'], 'supraclavicular': {'right': ['right supraclavicular region in the neck', 'right supraclavicular']}}, 'scapula': {'right': ['right scapula']}, 'spine': ['spinal column', 'spine', 'spinal', 'fractured vertebral body', 'vertebral body'], 'thoracic': ['thoracic spine', 'upper thoracic spine', 'thoracic vertebral bodies'], 'thoracolumbar': ['thoracolumbar', 'thoracolumbar junction', 'thoracolumbar spine'], 'upper': ['upper thoracic'], 'middle': ['upper mid thoracic vertebral body']}, 'abdominal': ['abdomen', 'abdominal', 'intrabdominal', 'intraabdominal'], 'anatomical_directions': {'right': ['right', 'right side', 'right sided', 'right-sided', 'right of midline'], 'midline': ['midline'], 'adjacent': ['adjacent'], 'anterior': ['anterior'], 'apical': ['apical', 'apical area', 'apices', 'bi-apical', 'bilateral apical', 'biapical', 'apical areas on either side', 'both apical areas', 'left apex', 'left apical', 'right apex'], 'base': ['basal', 'base', 'bases', 'bilateral basal', 'bilateral bases', 'basilar', 'bi-basal', 'bibasal', 'bibasilar', 'bilateral basilar', 'bilateral basilar areas', 'bilateral bibasilar', 'both bases', 'right bibasilar', 'right medial basal', 'right medial base'], 'left': ['left', 'l', 'left side', 'left sided', 'left-sided', 'leftward'], 'bilateral': ['bilateral', 'bilaterally', 'either side'], 'central': ['central', 'centrally', 'right central'], 'distal': ['distal'], 'inferior': ['inferior', 'inferiorly'], 'lateral': ['lateral', 'left lateral', 'right apicolateral', 'right lateral'], 'lower': ['left lower', 'lower', 'right lower'], 'posterior': ['left posteriorly', 'posterior', 'posterior compartments', 'posteriorly', 'right posterior'], 'superior': ['left superior', 'superior', 'superiorly'], 'upper': ['left upper zone', 'rightward upper', 'upper'], 'middle': ['mid', 'right mid']}, 'imaging_descriptions': ['area of bronchiectasis'], 'organs': {'intestines': ['bowel'], 'small': {'duodenum': ['duodenum']}, 'esophagus': ['esophagus'], 'spleen': ['splenic flexure'], 'large': {'colon': {'transverse': ['transverse colon']}}}, 'stomach': ['stomach', 'body of the stomach'], 'epigastric': ['epigastric abdomen', 'epigastric area'], 'junctions': {'gastroesophageal': ['gastroesophageal junction']}, 'medical devices': ['around g-tube', 'sternal wires', 'superior most wire', 'superior most wires', 'superior to the vertebroplasty'], 'hip': ['hip'], 'left': ['left abdomen'], 'arm': {'left': ['left arm']}, 'axilla': {'left': ['left axilla', 'left axillary region'], 'right': ['right axilla']}, 'specific_location': ['left paraspinal region in the abdomen'], 'extremities': {'left': {'upper': ['left upper extremity']}, 'lower': ['lower extremity']}, 'quadrants': {'left': {'upper': ['left upper quadrant']}, 'right': {'upper': ['right upper abdominal quadrant', 'right upper quadrant', 'right upper quadrant of the abdomen', 'ruq']}}, 'neck': ['neck', 'throat'], 'lymphatic': {'nodes': ['lymph node']}, 'middle': ['mid abdomen'], 'paratracheal': ['paratracheal'], 'location description': ['peripheral', 'lung periphery'], 'right': ['right abdomen', 'right mediastinal', 'right mediastinum', 'right neck', 'right paratracheal', 'right paratracheal region', 'right paratracheal stripe'], 'lower': ['right lower neck'], 'paramediastinal region': {'right': ['right paramediastinal', 'right paramediastinal region']}, 'upper': ['right upper mediastinal contour', 'right upper mediastinum', 'upper abdomen', 'upper abdominal', 'upper portion of the abdomen', 'upper mediastinal', 'upper mediastinum', 'upper stomach'], 'subdiaphragm': ['subdiaphragmatic']};

        // 데이터를 D3 계층 구조로 변환하는 함수            
        function transformData(data, name = "Location", currentDepth = 0, maxDepth = 8) {
            if (currentDepth >= maxDepth) {
                return {name: name};
            }

            let children = [];
            if (Array.isArray(data)) {
                // 리프 노드 (동의어 목록) - 최대 10개만 표시
                if (currentDepth < maxDepth - 1) {
                    children = data.slice(0, 10).map(synonym => ({name: synonym}));
                }
            } else if (typeof data === 'object') {
                for (let [key, value] of Object.entries(data)) {
                    children.push(transformData(value, key, currentDepth + 1, maxDepth));
                }
            }
            return {name: name, children: children.length > 0 ? children : null};
        }

        const hierarchyData = transformData(data);

        // SVG 크기 설정 (더 크게)
        const width = 1800;
        const height = 1600;

        // SVG 생성
        const svg = d3.select("#tree-container")
            .append("svg")
            .attr("width", width)
            .attr("height", height)
            .append("g")
            .attr("transform", `translate(${width / 2},${height / 2})`);

        // 데이터를 계층 구조로 변환
        const root = d3.hierarchy(hierarchyData);

        // 트리의 깊이에 따라 반지름 계산 (더 크게)
        const radius = Math.min(width, height) / 2 - 100;  // 150 → 100으로 증가하여 더 넓은 공간 활용

        // 트리 레이아웃 생성 (간격 조정 - 더 넓게)
        const tree = d3.tree()
            .size([2 * Math.PI, radius])
            .separation((a, b) => {
                // 노드 간 간격을 더 크게 설정
                if (a.parent == b.parent) {
                    return 3.5; // 같은 레벨 노드 간 간격 (2.5 → 3.5)
                } else {
                    return 5.5; // 다른 레벨 노드 간 간격 (4.0 → 5.5)
                }
            });

        tree(root);

        // 링크 생성
        const link = svg.selectAll(".link")
            .data(root.links())
            .join("path")
            .attr("class", "link")
            .attr("d", d3.linkRadial()
                .angle(d => d.x)
                .radius(d => d.y));

        // 노드 생성
        const node = svg.selectAll(".node")
            .data(root.descendants())
            .join("g")
            .attr("class", "node")
            .attr("transform", d => `rotate(${d.x * 180 / Math.PI - 90}) translate(${d.y},0)`);

        // 노드에 원 추가 (크기 조정 - 더 작게)
        node.append("circle")
            .attr("r", d => d.children ? 4 : 2.5);  // 5→4, 3→2.5로 줄여서 간격 확보

        // 노드에 텍스트 추가 (간격 개선)
        node.append("text")
            .attr("dy", ".35em")
            .attr("x", d => d.x < Math.PI === !d.children ? 12 : -12)  // 8 → 12로 증가
            .attr("text-anchor", d => d.x < Math.PI === !d.children ? "start" : "end")
            .attr("transform", d => d.x >= Math.PI ? "rotate(180)" : null)
            .text(d => d.data.name.length > 18 ? d.data.name.substring(0, 18) + "..." : d.data.name)  // 20 → 18로 줄여서 겹침 방지
            .clone(true).lower()
            .attr("stroke", "white")
            .attr("stroke-width", 2);

        // 줌 기능 추가
        const zoom = d3.zoom()
            .scaleExtent([0.2, 4])
            .on("zoom", (event) => {
                svg.attr("transform", `translate(${event.transform.x + width / 2},${event.transform.y + height / 2}) scale(${event.transform.k})`);
            });

        // 초기 줌 레벨 설정 (더 작게 시작하여 전체 구조 보기)
        const initialScale = 0.5;  // 0.6 → 0.5로 더 작게
        const initialTranslate = [(width - width * initialScale) / 2, (height - height * initialScale) / 2];

        d3.select("svg")
            .call(zoom)
            .call(zoom.transform, d3.zoomIdentity.translate(...initialTranslate).scale(initialScale));

        // 컨트롤 함수들
        function resetZoom() {
            d3.select("svg").transition().duration(750).call(
                zoom.transform,
                d3.zoomIdentity.translate(...initialTranslate).scale(initialScale)
            );
        }

        function toggleLabels() {
            const texts = d3.selectAll(".node text");
            const isVisible = texts.style("opacity") !== "0";
            texts.style("opacity", isVisible ? "0" : "1");
        }

    </script>
</body>
</html>
