GEOINT-Modus aus Monitor entfernt
Wird als eigenstaendige Anwendung auf separater Subdomain neu aufgebaut. Alle GEOINT-Dateien entfernt, dashboard.html/components.js/main.py auf pre-GEOINT Stand zurueckgesetzt.
Dieser Commit ist enthalten in:
@@ -76,6 +76,7 @@ App.selectNetworkAnalysis = async function(id) {
|
||||
if (analysis.status === 'ready') {
|
||||
this._hideNetworkProgress();
|
||||
var graphData = await API.getNetworkGraph(id);
|
||||
document.getElementById('network-graph-area').innerHTML = '';
|
||||
NetworkGraph.init('network-graph-area', graphData);
|
||||
this._setupNetworkFilters(graphData);
|
||||
|
||||
@@ -394,6 +395,7 @@ App._handleNetworkComplete = async function(msg) {
|
||||
// Graph neu laden
|
||||
try {
|
||||
var graphData = await API.getNetworkGraph(msg.analysis_id);
|
||||
document.getElementById('network-graph-area').innerHTML = '';
|
||||
NetworkGraph.init('network-graph-area', graphData);
|
||||
this._setupNetworkFilters(graphData);
|
||||
|
||||
@@ -452,3 +454,109 @@ function _escHtml(text) {
|
||||
d.textContent = text || '';
|
||||
return d.innerHTML;
|
||||
}
|
||||
|
||||
|
||||
// ==========================================================================
|
||||
// Cluster View Integration
|
||||
|
||||
|
||||
// ==========================================================================
|
||||
// Cluster Graph Integration (replaces flat NetworkGraph view)
|
||||
// ==========================================================================
|
||||
|
||||
App._cachedGraphData = null;
|
||||
|
||||
/**
|
||||
* Hide sidebar filter controls that dont apply to cluster view.
|
||||
*/
|
||||
App._hideNetworkSidebarFilters = function() {
|
||||
var sidebar = document.querySelector('.network-sidebar');
|
||||
if (!sidebar) return;
|
||||
var sections = sidebar.querySelectorAll('.network-sidebar-section');
|
||||
// Hide ALL old filter sections — ClusterGraph uses the detail panel directly
|
||||
for (var i = 0; i < sections.length; i++) {
|
||||
sections[i].style.display = 'none';
|
||||
}
|
||||
};
|
||||
|
||||
// Override selectNetworkAnalysis to use ClusterGraph
|
||||
(function() {
|
||||
App.selectNetworkAnalysis = async function(id) {
|
||||
this.currentNetworkId = id;
|
||||
this.currentIncidentId = null;
|
||||
localStorage.removeItem('selectedIncidentId');
|
||||
localStorage.setItem('selectedNetworkId', id);
|
||||
|
||||
document.getElementById('empty-state').style.display = 'none';
|
||||
document.getElementById('incident-view').style.display = 'none';
|
||||
document.getElementById('network-view').style.display = 'flex';
|
||||
|
||||
this.renderSidebar();
|
||||
this.renderNetworkSidebar();
|
||||
|
||||
try {
|
||||
var analysis = await API.getNetworkAnalysis(id);
|
||||
this._renderNetworkHeader(analysis);
|
||||
|
||||
if (analysis.status === 'ready') {
|
||||
this._hideNetworkProgress();
|
||||
var graphData = await API.getNetworkGraph(id);
|
||||
this._cachedGraphData = graphData;
|
||||
|
||||
var graphArea = document.getElementById('network-graph-area');
|
||||
graphArea.innerHTML = '';
|
||||
|
||||
var breadcrumb = document.getElementById('cluster-breadcrumb');
|
||||
if (breadcrumb) breadcrumb.style.display = 'flex';
|
||||
|
||||
ClusterGraph.init('network-graph-area', graphData.entities, graphData.relations);
|
||||
this._hideNetworkSidebarFilters();
|
||||
|
||||
try {
|
||||
var updateCheck = await API.checkNetworkUpdate(id);
|
||||
var badge = document.getElementById('network-update-badge');
|
||||
if (badge) badge.style.display = updateCheck.has_update ? 'inline-flex' : 'none';
|
||||
} catch (e) { /* ignorieren */ }
|
||||
} else if (analysis.status === 'generating') {
|
||||
this._showNetworkProgress('entity_extraction', 0);
|
||||
} else if (analysis.status === 'error') {
|
||||
this._hideNetworkProgress();
|
||||
var errArea = document.getElementById('network-graph-area');
|
||||
if (errArea) errArea.innerHTML = '<div class="network-empty-state"><div class="network-empty-state-icon">⚠</div><div class="network-empty-state-text">Fehler bei der Generierung. Versuche es erneut.</div></div>';
|
||||
}
|
||||
} catch (err) {
|
||||
UI.showToast('Fehler beim Laden der Netzwerkanalyse: ' + err.message, 'error');
|
||||
}
|
||||
};
|
||||
})();
|
||||
|
||||
// Override _handleNetworkComplete to use ClusterGraph
|
||||
(function() {
|
||||
App._handleNetworkComplete = async function(msg) {
|
||||
this._networkGenerating.delete(msg.analysis_id);
|
||||
|
||||
if (msg.analysis_id === this.currentNetworkId) {
|
||||
this._hideNetworkProgress();
|
||||
try {
|
||||
var graphData = await API.getNetworkGraph(msg.analysis_id);
|
||||
this._cachedGraphData = graphData;
|
||||
var graphArea = document.getElementById('network-graph-area');
|
||||
graphArea.innerHTML = '';
|
||||
|
||||
var breadcrumb = document.getElementById('cluster-breadcrumb');
|
||||
if (breadcrumb) breadcrumb.style.display = 'flex';
|
||||
|
||||
ClusterGraph.init('network-graph-area', graphData.entities, graphData.relations);
|
||||
this._hideNetworkSidebarFilters();
|
||||
|
||||
var analysis = await API.getNetworkAnalysis(msg.analysis_id);
|
||||
this._renderNetworkHeader(analysis);
|
||||
} catch (e) {
|
||||
console.error('Graph nach Generierung laden fehlgeschlagen:', e);
|
||||
}
|
||||
UI.showToast('Netzwerkanalyse fertig: ' + (msg.entity_count || 0) + ' Entitaeten, ' + (msg.relation_count || 0) + ' Beziehungen', 'success');
|
||||
}
|
||||
|
||||
await this.loadNetworkAnalyses();
|
||||
};
|
||||
})();
|
||||
|
||||
721
src/static/js/cluster-data.js
Normale Datei
721
src/static/js/cluster-data.js
Normale Datei
@@ -0,0 +1,721 @@
|
||||
/**
|
||||
* AegisSight OSINT Monitor - Cluster Data Transformation
|
||||
*
|
||||
* Transforms flat entity/relation data into hierarchical country-based clusters.
|
||||
* Used by ClusterGraph for the hierarchical network visualization.
|
||||
*
|
||||
* Usage:
|
||||
* const result = ClusterData.buildClusterData(entities, relations);
|
||||
* // result = { countries: [...], edges: [...], assignments: Map, entityToCountry: Map }
|
||||
*/
|
||||
|
||||
/* exported ClusterData */
|
||||
|
||||
const ClusterData = {
|
||||
|
||||
/**
|
||||
* Canonical country names with all known aliases (lowercase).
|
||||
* Maps alias -> canonical name (German UI labels).
|
||||
*/
|
||||
COUNTRY_ALIASES: {
|
||||
// Hauptakteure Irankonflikt
|
||||
'iran': 'Iran',
|
||||
'islamic republic of iran': 'Iran',
|
||||
'islamische republik iran': 'Iran',
|
||||
'persia': 'Iran',
|
||||
'persien': 'Iran',
|
||||
|
||||
'israel': 'Israel',
|
||||
'state of israel': 'Israel',
|
||||
'staat israel': 'Israel',
|
||||
|
||||
'united states': 'USA',
|
||||
'united states of america': 'USA',
|
||||
'usa': 'USA',
|
||||
'us': 'USA',
|
||||
'u.s.': 'USA',
|
||||
'u.s.a.': 'USA',
|
||||
'amerika': 'USA',
|
||||
'vereinigte staaten': 'USA',
|
||||
|
||||
// Naher Osten
|
||||
'lebanon': 'Libanon',
|
||||
'libanon': 'Libanon',
|
||||
'lebanese republic': 'Libanon',
|
||||
|
||||
'syria': 'Syrien',
|
||||
'syrien': 'Syrien',
|
||||
'syrian arab republic': 'Syrien',
|
||||
|
||||
'iraq': 'Irak',
|
||||
'irak': 'Irak',
|
||||
'republic of iraq': 'Irak',
|
||||
|
||||
'yemen': 'Jemen',
|
||||
'jemen': 'Jemen',
|
||||
'republic of yemen': 'Jemen',
|
||||
|
||||
'saudi arabia': 'Saudi-Arabien',
|
||||
'saudi-arabien': 'Saudi-Arabien',
|
||||
'kingdom of saudi arabia': 'Saudi-Arabien',
|
||||
'ksa': 'Saudi-Arabien',
|
||||
|
||||
'united arab emirates': 'VAE',
|
||||
'uae': 'VAE',
|
||||
'vae': 'VAE',
|
||||
'vereinigte arabische emirate': 'VAE',
|
||||
|
||||
'jordan': 'Jordanien',
|
||||
'jordanien': 'Jordanien',
|
||||
|
||||
'egypt': 'Ägypten',
|
||||
'ägypten': 'Ägypten',
|
||||
'aegypten': 'Ägypten',
|
||||
|
||||
'bahrain': 'Bahrain',
|
||||
'kingdom of bahrain': 'Bahrain',
|
||||
|
||||
'kuwait': 'Kuwait',
|
||||
'state of kuwait': 'Kuwait',
|
||||
|
||||
'qatar': 'Katar',
|
||||
'katar': 'Katar',
|
||||
|
||||
'oman': 'Oman',
|
||||
'sultanate of oman': 'Oman',
|
||||
|
||||
'palestine': 'Palästina',
|
||||
'palästina': 'Palästina',
|
||||
'palestinian territories': 'Palästina',
|
||||
'state of palestine': 'Palästina',
|
||||
'gaza': 'Palästina',
|
||||
'gaza strip': 'Palästina',
|
||||
'west bank': 'Palästina',
|
||||
|
||||
// Großmächte
|
||||
'russia': 'Russland',
|
||||
'russland': 'Russland',
|
||||
'russian federation': 'Russland',
|
||||
'russische föderation': 'Russland',
|
||||
|
||||
'china': 'China',
|
||||
'people\'s republic of china': 'China',
|
||||
'volksrepublik china': 'China',
|
||||
'prc': 'China',
|
||||
|
||||
'united kingdom': 'Großbritannien',
|
||||
'uk': 'Großbritannien',
|
||||
'großbritannien': 'Großbritannien',
|
||||
'grossbritannien': 'Großbritannien',
|
||||
'great britain': 'Großbritannien',
|
||||
'britain': 'Großbritannien',
|
||||
'england': 'Großbritannien',
|
||||
|
||||
'france': 'Frankreich',
|
||||
'frankreich': 'Frankreich',
|
||||
'french republic': 'Frankreich',
|
||||
|
||||
'germany': 'Deutschland',
|
||||
'deutschland': 'Deutschland',
|
||||
'federal republic of germany': 'Deutschland',
|
||||
'bundesrepublik deutschland': 'Deutschland',
|
||||
|
||||
// Weitere relevante Staaten
|
||||
'turkey': 'Türkei',
|
||||
'türkei': 'Türkei',
|
||||
'turkei': 'Türkei',
|
||||
'republic of turkey': 'Türkei',
|
||||
'türkiye': 'Türkei',
|
||||
|
||||
'india': 'Indien',
|
||||
'indien': 'Indien',
|
||||
'republic of india': 'Indien',
|
||||
|
||||
'pakistan': 'Pakistan',
|
||||
'islamic republic of pakistan': 'Pakistan',
|
||||
|
||||
'afghanistan': 'Afghanistan',
|
||||
|
||||
'ukraine': 'Ukraine',
|
||||
|
||||
'north korea': 'Nordkorea',
|
||||
'nordkorea': 'Nordkorea',
|
||||
'dprk': 'Nordkorea',
|
||||
|
||||
'south korea': 'Südkorea',
|
||||
'südkorea': 'Südkorea',
|
||||
'republic of korea': 'Südkorea',
|
||||
|
||||
'japan': 'Japan',
|
||||
|
||||
'italy': 'Italien',
|
||||
'italien': 'Italien',
|
||||
|
||||
'spain': 'Spanien',
|
||||
'spanien': 'Spanien',
|
||||
|
||||
'netherlands': 'Niederlande',
|
||||
'niederlande': 'Niederlande',
|
||||
'holland': 'Niederlande',
|
||||
|
||||
'poland': 'Polen',
|
||||
'polen': 'Polen',
|
||||
|
||||
'canada': 'Kanada',
|
||||
'kanada': 'Kanada',
|
||||
|
||||
'australia': 'Australien',
|
||||
'australien': 'Australien',
|
||||
|
||||
'brazil': 'Brasilien',
|
||||
'brasilien': 'Brasilien',
|
||||
|
||||
'mexico': 'Mexiko',
|
||||
'mexiko': 'Mexiko',
|
||||
|
||||
'south africa': 'Südafrika',
|
||||
'südafrika': 'Südafrika',
|
||||
|
||||
'nigeria': 'Nigeria',
|
||||
|
||||
'ethiopia': 'Äthiopien',
|
||||
'äthiopien': 'Äthiopien',
|
||||
|
||||
'somalia': 'Somalia',
|
||||
|
||||
'sudan': 'Sudan',
|
||||
|
||||
'libya': 'Libyen',
|
||||
'libyen': 'Libyen',
|
||||
|
||||
'tunisia': 'Tunesien',
|
||||
'tunesien': 'Tunesien',
|
||||
|
||||
'morocco': 'Marokko',
|
||||
'marokko': 'Marokko',
|
||||
|
||||
'algeria': 'Algerien',
|
||||
'algerien': 'Algerien',
|
||||
|
||||
'sweden': 'Schweden',
|
||||
'schweden': 'Schweden',
|
||||
|
||||
'norway': 'Norwegen',
|
||||
'norwegen': 'Norwegen',
|
||||
|
||||
'switzerland': 'Schweiz',
|
||||
'schweiz': 'Schweiz',
|
||||
|
||||
'austria': 'Österreich',
|
||||
'österreich': 'Österreich',
|
||||
'oesterreich': 'Österreich',
|
||||
},
|
||||
|
||||
/**
|
||||
* Country keyword patterns for name/description matching.
|
||||
* Each entry: [regex, canonical country name]
|
||||
* Order matters: more specific patterns first.
|
||||
*/
|
||||
COUNTRY_PATTERNS: [
|
||||
[/\biran/i, 'Iran'],
|
||||
[/\bpersi/i, 'Iran'],
|
||||
[/\bisrael/i, 'Israel'],
|
||||
[/\bjewish state/i, 'Israel'],
|
||||
[/\bunited states/i, 'USA'],
|
||||
[/\bamerican?\b/i, 'USA'],
|
||||
[/\bu\.?s\.?\b(?![\w-])/i, 'USA'],
|
||||
[/\bpentagon/i, 'USA'],
|
||||
[/\bwhite house/i, 'USA'],
|
||||
[/\bcongress\b/i, 'USA'],
|
||||
[/\bleban/i, 'Libanon'],
|
||||
[/\bhezbollah/i, 'Libanon'],
|
||||
[/\bhisbollah/i, 'Libanon'],
|
||||
[/\bsyri/i, 'Syrien'],
|
||||
[/\biraq/i, 'Irak'],
|
||||
[/\birak/i, 'Irak'],
|
||||
[/\byemen/i, 'Jemen'],
|
||||
[/\bjemen/i, 'Jemen'],
|
||||
[/\bhouthi/i, 'Jemen'],
|
||||
[/\bsaudi/i, 'Saudi-Arabien'],
|
||||
[/\bemira/i, 'VAE'],
|
||||
[/\bdubai/i, 'VAE'],
|
||||
[/\bjordan/i, 'Jordanien'],
|
||||
[/\begypt/i, 'Ägypten'],
|
||||
[/\bägypt/i, 'Ägypten'],
|
||||
[/\bbahrain/i, 'Bahrain'],
|
||||
[/\bkuwait/i, 'Kuwait'],
|
||||
[/\bqatar/i, 'Katar'],
|
||||
[/\bkatar/i, 'Katar'],
|
||||
[/\bpalesti/i, 'Palästina'],
|
||||
[/\bgaza/i, 'Palästina'],
|
||||
[/\bhamas\b/i, 'Palästina'],
|
||||
[/\brussi/i, 'Russland'],
|
||||
[/\bkreml/i, 'Russland'],
|
||||
[/\bputin/i, 'Russland'],
|
||||
[/\bmoscow/i, 'Russland'],
|
||||
[/\bmoskau/i, 'Russland'],
|
||||
[/\bchines/i, 'China'],
|
||||
[/\bchinai/i, 'China'],
|
||||
[/\bchina/i, 'China'],
|
||||
[/\bbeijing/i, 'China'],
|
||||
[/\bpeking/i, 'China'],
|
||||
[/\bbriti/i, 'Großbritannien'],
|
||||
[/\bengland/i, 'Großbritannien'],
|
||||
[/\blondon\b/i, 'Großbritannien'],
|
||||
[/\bfrench/i, 'Frankreich'],
|
||||
[/\bfranz/i, 'Frankreich'],
|
||||
[/\bfrance/i, 'Frankreich'],
|
||||
[/\bgerman/i, 'Deutschland'],
|
||||
[/\bdeutsch/i, 'Deutschland'],
|
||||
[/\bturk/i, 'Türkei'],
|
||||
[/\btürk/i, 'Türkei'],
|
||||
[/\bankara/i, 'Türkei'],
|
||||
[/\bindia/i, 'Indien'],
|
||||
[/\bindisch/i, 'Indien'],
|
||||
[/\bpakistan/i, 'Pakistan'],
|
||||
[/\bafghan/i, 'Afghanistan'],
|
||||
[/\bukrain/i, 'Ukraine'],
|
||||
[/\bnorth.?korea/i, 'Nordkorea'],
|
||||
[/\bnordkorea/i, 'Nordkorea'],
|
||||
[/\bpjöngjang/i, 'Nordkorea'],
|
||||
[/\bpyongyang/i, 'Nordkorea'],
|
||||
[/\bjapan/i, 'Japan'],
|
||||
[/\boman\b/i, 'Oman'],
|
||||
],
|
||||
|
||||
/**
|
||||
* Main entry: transform flat entity/relation data into clustered structure.
|
||||
*
|
||||
* @param {Array} entities - All entities from getNetworkGraph
|
||||
* @param {Array} relations - All relations from getNetworkGraph
|
||||
* @returns {Object} { countries, edges, assignments, entityToCountry }
|
||||
*/
|
||||
buildClusterData(entities, relations) {
|
||||
// 1. Identify which entities are countries and merge duplicates
|
||||
var countryMap = this._identifyCountries(entities);
|
||||
|
||||
// 2. Build adjacency for fast lookup
|
||||
var adjacency = this._buildAdjacency(relations);
|
||||
|
||||
// 3. Multi-strategy assignment:
|
||||
// a) Relation-based (direct country connections)
|
||||
// b) Name/Description keyword matching
|
||||
// c) Propagation through assigned neighbors (multiple passes)
|
||||
var result = this._assignEntities(entities, relations, countryMap, adjacency);
|
||||
|
||||
// 4. Aggregate cross-country relations
|
||||
var edges = this._aggregateEdges(relations, result.entityToCountry);
|
||||
|
||||
// 5. Build country node objects for rendering
|
||||
var countries = this._buildCountryNodes(countryMap, result.assignments, entities);
|
||||
|
||||
return {
|
||||
countries: countries,
|
||||
edges: edges,
|
||||
assignments: result.assignments,
|
||||
entityToCountry: result.entityToCountry
|
||||
};
|
||||
},
|
||||
|
||||
// ---- Step 1: Identify countries ------------------------------------------
|
||||
|
||||
_identifyCountries(entities) {
|
||||
// Map: canonical country name -> [entity_id, ...]
|
||||
var countryMap = new Map();
|
||||
|
||||
for (var i = 0; i < entities.length; i++) {
|
||||
var entity = entities[i];
|
||||
|
||||
var normalized = (entity.name_normalized || entity.name || '')
|
||||
.toLowerCase().trim();
|
||||
|
||||
// Strip common suffixes/brackets for matching
|
||||
var cleaned = normalized
|
||||
.replace(/\s*\(als organisation\)/i, '')
|
||||
.replace(/\s*\(organisation\)/i, '')
|
||||
.replace(/^the\s+/, '')
|
||||
.replace(/\s+republic$/, '')
|
||||
.replace(/\s+federation$/, '');
|
||||
|
||||
// Try direct alias match first (exact match in COUNTRY_ALIASES)
|
||||
var directMatch = this.COUNTRY_ALIASES[normalized];
|
||||
var cleanedMatch = !directMatch ? this.COUNTRY_ALIASES[cleaned] : null;
|
||||
var canonical = directMatch || cleanedMatch;
|
||||
|
||||
// For non-location entities: only accept direct alias matches
|
||||
// (prevents "Iranian Drones" from being a country, but allows
|
||||
// "Islamic Republic of Iran" which is a direct alias)
|
||||
if (canonical && entity.entity_type !== 'location' && !directMatch) {
|
||||
// Match came from cleaning — apply length check
|
||||
if (cleaned.length > canonical.length + 15) continue;
|
||||
}
|
||||
|
||||
if (canonical) {
|
||||
if (!countryMap.has(canonical)) {
|
||||
countryMap.set(canonical, []);
|
||||
}
|
||||
countryMap.get(canonical).push(entity.id);
|
||||
}
|
||||
}
|
||||
|
||||
return countryMap;
|
||||
},
|
||||
|
||||
// ---- Step 2: Build adjacency ---------------------------------------------
|
||||
|
||||
_buildAdjacency(relations) {
|
||||
var adj = new Map();
|
||||
for (var i = 0; i < relations.length; i++) {
|
||||
var r = relations[i];
|
||||
var src = r.source_entity_id;
|
||||
var tgt = r.target_entity_id;
|
||||
|
||||
if (!adj.has(src)) adj.set(src, []);
|
||||
if (!adj.has(tgt)) adj.set(tgt, []);
|
||||
adj.get(src).push(r);
|
||||
adj.get(tgt).push(r);
|
||||
}
|
||||
return adj;
|
||||
},
|
||||
|
||||
// ---- Step 3: Assign entities to countries (multi-strategy) ----------------
|
||||
|
||||
_assignEntities(entities, relations, countryMap, adjacency) {
|
||||
var self = this;
|
||||
var entityToCountry = new Map();
|
||||
var countryEntityIds = new Set();
|
||||
|
||||
// Build entity lookup
|
||||
var entityMap = new Map();
|
||||
for (var i = 0; i < entities.length; i++) {
|
||||
entityMap.set(entities[i].id, entities[i]);
|
||||
}
|
||||
|
||||
// Mark all country entity IDs
|
||||
countryMap.forEach(function(ids, canonical) {
|
||||
for (var i = 0; i < ids.length; i++) {
|
||||
entityToCountry.set(ids[i], canonical);
|
||||
countryEntityIds.add(ids[i]);
|
||||
}
|
||||
});
|
||||
|
||||
// Ensure all country keys exist in assignments
|
||||
var assignments = new Map();
|
||||
countryMap.forEach(function(_, canonical) {
|
||||
assignments.set(canonical, []);
|
||||
});
|
||||
assignments.set('__unassigned__', []);
|
||||
|
||||
// Collect unassigned entity IDs
|
||||
var unassigned = [];
|
||||
for (var i = 0; i < entities.length; i++) {
|
||||
if (!countryEntityIds.has(entities[i].id)) {
|
||||
unassigned.push(entities[i].id);
|
||||
}
|
||||
}
|
||||
|
||||
// --- Strategy A: Relation-based (direct connection to country entity) ---
|
||||
var stillUnassigned = [];
|
||||
for (var a = 0; a < unassigned.length; a++) {
|
||||
var eid = unassigned[a];
|
||||
var country = this._findByRelation(eid, adjacency, entityToCountry, countryEntityIds);
|
||||
if (country) {
|
||||
entityToCountry.set(eid, country);
|
||||
if (!assignments.has(country)) assignments.set(country, []);
|
||||
assignments.get(country).push(eid);
|
||||
} else {
|
||||
stillUnassigned.push(eid);
|
||||
}
|
||||
}
|
||||
|
||||
// --- Strategy B: Name + Description keyword matching ---
|
||||
var afterKeyword = [];
|
||||
for (var b = 0; b < stillUnassigned.length; b++) {
|
||||
var eid2 = stillUnassigned[b];
|
||||
var entity = entityMap.get(eid2);
|
||||
var country2 = this._findByKeywords(entity);
|
||||
if (country2) {
|
||||
entityToCountry.set(eid2, country2);
|
||||
if (!assignments.has(country2)) assignments.set(country2, []);
|
||||
assignments.get(country2).push(eid2);
|
||||
} else {
|
||||
afterKeyword.push(eid2);
|
||||
}
|
||||
}
|
||||
|
||||
// --- Strategy C: Propagation through assigned neighbors (max 5 passes) ---
|
||||
var remaining = afterKeyword;
|
||||
for (var pass = 0; pass < 5 && remaining.length > 0; pass++) {
|
||||
var nextRemaining = [];
|
||||
for (var c = 0; c < remaining.length; c++) {
|
||||
var eid3 = remaining[c];
|
||||
var country3 = this._findByNeighborPropagation(eid3, adjacency, entityToCountry);
|
||||
if (country3) {
|
||||
entityToCountry.set(eid3, country3);
|
||||
if (!assignments.has(country3)) assignments.set(country3, []);
|
||||
assignments.get(country3).push(eid3);
|
||||
} else {
|
||||
nextRemaining.push(eid3);
|
||||
}
|
||||
}
|
||||
if (nextRemaining.length === remaining.length) break; // No progress
|
||||
remaining = nextRemaining;
|
||||
}
|
||||
|
||||
// Everything still unassigned goes to "Sonstige"
|
||||
for (var u = 0; u < remaining.length; u++) {
|
||||
assignments.get('__unassigned__').push(remaining[u]);
|
||||
}
|
||||
|
||||
return { entityToCountry: entityToCountry, assignments: assignments };
|
||||
},
|
||||
|
||||
/**
|
||||
* Strategy A: Direct relation to a country entity.
|
||||
*/
|
||||
_findByRelation: function(entityId, adjacency, entityToCountry, countryEntityIds) {
|
||||
var rels = adjacency.get(entityId);
|
||||
if (!rels || rels.length === 0) return null;
|
||||
|
||||
var scores = new Map();
|
||||
for (var i = 0; i < rels.length; i++) {
|
||||
var r = rels[i];
|
||||
var otherId = r.source_entity_id === entityId
|
||||
? r.target_entity_id : r.source_entity_id;
|
||||
|
||||
if (countryEntityIds.has(otherId)) {
|
||||
var country = entityToCountry.get(otherId);
|
||||
scores.set(country, (scores.get(country) || 0) + (r.weight || 1));
|
||||
}
|
||||
}
|
||||
|
||||
return this._bestFromScores(scores);
|
||||
},
|
||||
|
||||
/**
|
||||
* Strategy B: Match country keywords in entity name, aliases and description.
|
||||
* For events mentioning multiple countries, uses first-mentioned country in name
|
||||
* with a bonus, so "Iran-Israel-US War" → Iran.
|
||||
*/
|
||||
_findByKeywords: function(entity) {
|
||||
if (!entity) return null;
|
||||
|
||||
var scores = new Map();
|
||||
var patterns = this.COUNTRY_PATTERNS;
|
||||
var name = entity.name || '';
|
||||
var desc = entity.description || '';
|
||||
|
||||
// For name matches: track position to boost first-mentioned country
|
||||
var firstMatchPos = Infinity;
|
||||
var firstMatchCountry = null;
|
||||
|
||||
for (var i = 0; i < patterns.length; i++) {
|
||||
var pattern = patterns[i][0];
|
||||
var country = patterns[i][1];
|
||||
|
||||
// Check name (stronger signal)
|
||||
var nameMatch = pattern.exec(name);
|
||||
if (nameMatch) {
|
||||
scores.set(country, (scores.get(country) || 0) + 3);
|
||||
// Track first-mentioned country by position in name
|
||||
if (nameMatch.index < firstMatchPos) {
|
||||
firstMatchPos = nameMatch.index;
|
||||
firstMatchCountry = country;
|
||||
}
|
||||
}
|
||||
// Reset regex lastIndex (stateless)
|
||||
pattern.lastIndex = 0;
|
||||
|
||||
// Check description (weaker signal)
|
||||
if (desc && pattern.test(desc)) {
|
||||
scores.set(country, (scores.get(country) || 0) + 1);
|
||||
}
|
||||
pattern.lastIndex = 0;
|
||||
}
|
||||
|
||||
// Check aliases
|
||||
if (entity.aliases && entity.aliases.length > 0) {
|
||||
var aliasText = entity.aliases.join(' ');
|
||||
for (var j = 0; j < patterns.length; j++) {
|
||||
if (patterns[j][0].test(aliasText)) {
|
||||
var c = patterns[j][1];
|
||||
scores.set(c, (scores.get(c) || 0) + 1);
|
||||
}
|
||||
patterns[j][0].lastIndex = 0;
|
||||
}
|
||||
}
|
||||
|
||||
// Boost first-mentioned country in name (important for multi-country events)
|
||||
if (firstMatchCountry && scores.size > 1) {
|
||||
scores.set(firstMatchCountry, (scores.get(firstMatchCountry) || 0) + 2);
|
||||
}
|
||||
|
||||
return this._bestFromScores(scores);
|
||||
},
|
||||
|
||||
/**
|
||||
* Strategy C: Propagate from already-assigned neighbors.
|
||||
*/
|
||||
_findByNeighborPropagation: function(entityId, adjacency, entityToCountry) {
|
||||
var rels = adjacency.get(entityId);
|
||||
if (!rels || rels.length === 0) return null;
|
||||
|
||||
var scores = new Map();
|
||||
for (var i = 0; i < rels.length; i++) {
|
||||
var r = rels[i];
|
||||
var otherId = r.source_entity_id === entityId
|
||||
? r.target_entity_id : r.source_entity_id;
|
||||
|
||||
if (entityToCountry.has(otherId)) {
|
||||
var country = entityToCountry.get(otherId);
|
||||
scores.set(country, (scores.get(country) || 0) + (r.weight || 1));
|
||||
}
|
||||
}
|
||||
|
||||
return this._bestFromScores(scores);
|
||||
},
|
||||
|
||||
/**
|
||||
* Helper: return country with highest score, or null.
|
||||
*/
|
||||
_bestFromScores: function(scores) {
|
||||
if (scores.size === 0) return null;
|
||||
var best = null;
|
||||
var bestScore = 0;
|
||||
scores.forEach(function(score, country) {
|
||||
if (score > bestScore) {
|
||||
best = country;
|
||||
bestScore = score;
|
||||
}
|
||||
});
|
||||
return best;
|
||||
},
|
||||
|
||||
// ---- Step 4: Aggregate cross-country edges -------------------------------
|
||||
|
||||
_aggregateEdges(relations, entityToCountry) {
|
||||
var edgeMap = new Map(); // "A|B" -> { source, target, count, categories, totalWeight }
|
||||
|
||||
for (var i = 0; i < relations.length; i++) {
|
||||
var r = relations[i];
|
||||
var c1 = entityToCountry.get(r.source_entity_id);
|
||||
var c2 = entityToCountry.get(r.target_entity_id);
|
||||
|
||||
// Skip if same country, or either entity unassigned
|
||||
if (!c1 || !c2 || c1 === c2) continue;
|
||||
|
||||
var key = c1 < c2 ? c1 + '|' + c2 : c2 + '|' + c1;
|
||||
|
||||
if (!edgeMap.has(key)) {
|
||||
edgeMap.set(key, {
|
||||
source: c1 < c2 ? c1 : c2,
|
||||
target: c1 < c2 ? c2 : c1,
|
||||
count: 0,
|
||||
totalWeight: 0,
|
||||
categories: {}
|
||||
});
|
||||
}
|
||||
|
||||
var edge = edgeMap.get(key);
|
||||
edge.count += 1;
|
||||
edge.totalWeight += (r.weight || 1);
|
||||
var cat = r.category || 'neutral';
|
||||
edge.categories[cat] = (edge.categories[cat] || 0) + 1;
|
||||
}
|
||||
|
||||
// Determine dominant category per edge
|
||||
var edges = [];
|
||||
edgeMap.forEach(function(edge) {
|
||||
var bestCat = 'neutral';
|
||||
var bestCount = 0;
|
||||
for (var cat in edge.categories) {
|
||||
if (edge.categories[cat] > bestCount) {
|
||||
bestCat = cat;
|
||||
bestCount = edge.categories[cat];
|
||||
}
|
||||
}
|
||||
edge.dominantCategory = bestCat;
|
||||
edges.push(edge);
|
||||
});
|
||||
|
||||
// Sort by count descending
|
||||
edges.sort(function(a, b) { return b.count - a.count; });
|
||||
|
||||
return edges;
|
||||
},
|
||||
|
||||
// ---- Step 5: Build country node objects -----------------------------------
|
||||
|
||||
_buildCountryNodes(countryMap, assignments, entities) {
|
||||
var entityMap = new Map();
|
||||
for (var i = 0; i < entities.length; i++) {
|
||||
entityMap.set(entities[i].id, entities[i]);
|
||||
}
|
||||
|
||||
var countries = [];
|
||||
|
||||
assignments.forEach(function(entityIds, countryName) {
|
||||
if (countryName === '__unassigned__') {
|
||||
if (entityIds.length > 0) {
|
||||
countries.push({
|
||||
name: 'Sonstige',
|
||||
canonicalName: '__unassigned__',
|
||||
entityCount: entityIds.length,
|
||||
isUnassigned: true,
|
||||
typeCounts: ClusterData._countTypes(entityIds, entityMap),
|
||||
topEntities: ClusterData._getTopEntities(entityIds, entityMap, 5)
|
||||
});
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// Count includes the country entity IDs themselves? No — only affiliated entities
|
||||
var totalCount = entityIds.length;
|
||||
if (totalCount === 0) return; // Skip countries with no affiliated entities
|
||||
|
||||
countries.push({
|
||||
name: countryName,
|
||||
canonicalName: countryName,
|
||||
entityCount: totalCount,
|
||||
isUnassigned: false,
|
||||
countryEntityIds: countryMap.get(countryName) || [],
|
||||
typeCounts: ClusterData._countTypes(entityIds, entityMap),
|
||||
topEntities: ClusterData._getTopEntities(entityIds, entityMap, 5)
|
||||
});
|
||||
});
|
||||
|
||||
// Sort by entity count descending
|
||||
countries.sort(function(a, b) { return b.entityCount - a.entityCount; });
|
||||
|
||||
return countries;
|
||||
},
|
||||
|
||||
/**
|
||||
* Count entities by type within a set of IDs.
|
||||
*/
|
||||
_countTypes(entityIds, entityMap) {
|
||||
var counts = { person: 0, organisation: 0, location: 0, event: 0, military: 0 };
|
||||
for (var i = 0; i < entityIds.length; i++) {
|
||||
var e = entityMap.get(entityIds[i]);
|
||||
if (e && counts.hasOwnProperty(e.entity_type)) {
|
||||
counts[e.entity_type]++;
|
||||
}
|
||||
}
|
||||
return counts;
|
||||
},
|
||||
|
||||
/**
|
||||
* Get top N entities by mention_count from a set of IDs.
|
||||
*/
|
||||
_getTopEntities(entityIds, entityMap, n) {
|
||||
var ents = [];
|
||||
for (var i = 0; i < entityIds.length; i++) {
|
||||
var e = entityMap.get(entityIds[i]);
|
||||
if (e) ents.push(e);
|
||||
}
|
||||
ents.sort(function(a, b) {
|
||||
return (b.mention_count || 0) - (a.mention_count || 0);
|
||||
});
|
||||
return ents.slice(0, n);
|
||||
}
|
||||
};
|
||||
@@ -716,9 +716,6 @@ const UI = {
|
||||
|
||||
if (emptyEl) emptyEl.style.display = 'none';
|
||||
|
||||
// Locations fuer GEOINT merken
|
||||
this._lastLocations = locations;
|
||||
|
||||
// Statistik
|
||||
const totalArticles = locations.reduce((s, l) => s + l.article_count, 0);
|
||||
if (statsEl) statsEl.textContent = `${locations.length} Orte / ${totalArticles} Artikel`;
|
||||
@@ -745,10 +742,6 @@ const UI = {
|
||||
}).setView([51.1657, 10.4515], 5); // Deutschland-Zentrum
|
||||
|
||||
this._applyMapTiles();
|
||||
|
||||
// GEOINT-Modus wiederherstellen
|
||||
if (typeof GEOINT !== 'undefined') GEOINT.restoreState(this._map);
|
||||
|
||||
this._mapCluster = L.markerClusterGroup({
|
||||
maxClusterRadius: 40,
|
||||
iconCreateFunction: function(cluster) {
|
||||
@@ -860,11 +853,6 @@ const UI = {
|
||||
this._mapLegendControl = legend;
|
||||
}
|
||||
|
||||
// GEOINT: Timeline mit Artikel-Daten initialisieren
|
||||
if (typeof GEOINT !== 'undefined' && typeof App !== 'undefined') {
|
||||
GEOINT.initTimeline(App._currentArticles || []);
|
||||
}
|
||||
|
||||
// Resize-Fix fuer gridstack (mehrere Versuche, da Container-Hoehe erst spaeter steht)
|
||||
const self = this;
|
||||
[100, 300, 800].forEach(delay => {
|
||||
@@ -882,8 +870,6 @@ const UI = {
|
||||
|
||||
_applyMapTiles() {
|
||||
if (!this._map) return;
|
||||
// Im GEOINT-Modus Tiles nicht ueberschreiben
|
||||
if (typeof GEOINT !== 'undefined' && GEOINT._active) return;
|
||||
// Alte Tile-Layer entfernen
|
||||
this._map.eachLayer(layer => {
|
||||
if (layer instanceof L.TileLayer) this._map.removeLayer(layer);
|
||||
|
||||
@@ -1,492 +0,0 @@
|
||||
/**
|
||||
* GEOINT-Modus: Taktische Kartenansicht mit Echtzeit-Datenlayern.
|
||||
*/
|
||||
const GEOINT = {
|
||||
_active: false,
|
||||
_map: null,
|
||||
_sublayers: {},
|
||||
_canvasRenderer: null,
|
||||
// Layer references
|
||||
_flightLayer: null, _quakeLayer: null, _gdeltLayer: null,
|
||||
_heatLayer: null, _shipsLayer: null,
|
||||
// Data caches
|
||||
_flightsData: null, _shipsData: null,
|
||||
// Intervals
|
||||
_flightInterval: null, _quakeInterval: null, _gdeltInterval: null, _shipsInterval: null,
|
||||
_flightFetching: false,
|
||||
// UI controls
|
||||
_moveHandler: null, _subControl: null,
|
||||
_coordControl: null, _coordHandler: null,
|
||||
_distanceLayers: null, _distancePoints: [], _distanceHandler: null,
|
||||
_satTileLayer: null, _satLabelLayer: null,
|
||||
_timelineData: null,
|
||||
|
||||
// === HAUPTSCHALTER =====================================================
|
||||
toggle(enabled, map) {
|
||||
if (!map) map = this._map;
|
||||
if (!map && typeof UI !== 'undefined') map = UI._map;
|
||||
if (!map) return;
|
||||
this._active = enabled;
|
||||
this._map = map;
|
||||
|
||||
var container = document.getElementById('map-container');
|
||||
if (container) container.classList.toggle('geoint-active', enabled);
|
||||
var fsContainer = document.getElementById('map-fullscreen-container');
|
||||
if (fsContainer) fsContainer.classList.toggle('geoint-active', enabled);
|
||||
var card = container ? container.closest('.map-card') : null;
|
||||
if (card) card.classList.toggle('geoint-card-active', enabled);
|
||||
|
||||
document.querySelectorAll('#geoint-mode-cb, #geoint-mode-cb-fs').forEach(function(cb) { cb.checked = enabled; });
|
||||
|
||||
if (enabled) {
|
||||
if (!this._canvasRenderer) this._canvasRenderer = L.canvas({ padding: 0.5 });
|
||||
this._applySatelliteTiles(map);
|
||||
this._createSubControl(map);
|
||||
this._restoreSublayers(map);
|
||||
} else {
|
||||
this.cleanup();
|
||||
this._restoreOsmTiles(map);
|
||||
}
|
||||
this._saveState();
|
||||
},
|
||||
|
||||
// === TILES ==============================================================
|
||||
_applySatelliteTiles(map) {
|
||||
if (!map) return;
|
||||
var toRemove = [];
|
||||
map.eachLayer(function(l) { if (l instanceof L.TileLayer) toRemove.push(l); });
|
||||
toRemove.forEach(function(l) { map.removeLayer(l); });
|
||||
this._satTileLayer = L.tileLayer(
|
||||
'https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}',
|
||||
{ attribution: 'Tiles © Esri', maxZoom: 19, noWrap: true }
|
||||
).addTo(map);
|
||||
this._satLabelLayer = L.tileLayer(
|
||||
'https://server.arcgisonline.com/ArcGIS/rest/services/Reference/World_Boundaries_and_Places/MapServer/tile/{z}/{y}/{x}',
|
||||
{ maxZoom: 19, noWrap: true }
|
||||
).addTo(map);
|
||||
if (this._satTileLayer.bringToBack) this._satTileLayer.bringToBack();
|
||||
},
|
||||
|
||||
_restoreOsmTiles(map) {
|
||||
if (!map) return;
|
||||
var toRemove = [];
|
||||
map.eachLayer(function(l) { if (l instanceof L.TileLayer) toRemove.push(l); });
|
||||
toRemove.forEach(function(l) { map.removeLayer(l); });
|
||||
if (typeof UI !== 'undefined' && UI._applyMapTiles) {
|
||||
UI._applyMapTiles();
|
||||
} else {
|
||||
L.tileLayer('https://tile.openstreetmap.de/{z}/{x}/{y}.png', {
|
||||
attribution: '© OpenStreetMap', maxZoom: 18, noWrap: true
|
||||
}).addTo(map);
|
||||
}
|
||||
},
|
||||
|
||||
// === SUB-LAYER CONTROL ==================================================
|
||||
_createSubControl(map) {
|
||||
if (this._subControl) return;
|
||||
var self = this;
|
||||
var items = [
|
||||
['flights', 'Flugverkehr', 'flights'],
|
||||
['ships', 'Schiffsverkehr', 'ships'],
|
||||
['quakes', 'Erdbeben', 'quakes'],
|
||||
['gdelt', 'Nachrichten', 'gdelt'],
|
||||
['_sep'],
|
||||
['heatmap', 'Heatmap', 'heatmap'],
|
||||
['coords', 'Koordinaten', 'coords'],
|
||||
['distance', 'Distanz', 'distance'],
|
||||
];
|
||||
var SubCtrl = L.Control.extend({
|
||||
options: { position: 'topright' },
|
||||
onAdd: function() {
|
||||
var div = L.DomUtil.create('div', 'geoint-sub-control');
|
||||
L.DomEvent.disableClickPropagation(div);
|
||||
L.DomEvent.disableScrollPropagation(div);
|
||||
var html = '<h4>GEOINT Layer</h4>';
|
||||
items.forEach(function(it) {
|
||||
if (it[0] === '_sep') { html += '<div class="geoint-sub-separator"></div>'; return; }
|
||||
var checked = self._sublayers[it[0]] ? ' checked' : '';
|
||||
html += '<div class="geoint-sub-item">' +
|
||||
'<input type="checkbox" id="geoint-sub-' + it[0] + '"' + checked + '>' +
|
||||
'<label for="geoint-sub-' + it[0] + '"><span class="geoint-dot geoint-dot-' + it[2] + '"></span>' + it[1] + '</label></div>';
|
||||
});
|
||||
div.innerHTML = html;
|
||||
return div;
|
||||
}
|
||||
});
|
||||
this._subControl = new SubCtrl();
|
||||
map.addControl(this._subControl);
|
||||
items.forEach(function(it) {
|
||||
if (it[0] === '_sep') return;
|
||||
var cb = document.getElementById('geoint-sub-' + it[0]);
|
||||
if (cb) cb.addEventListener('change', function() { self._toggleSub(it[0], this.checked, map); });
|
||||
});
|
||||
},
|
||||
|
||||
_removeSubControl() {
|
||||
if (this._subControl && this._map) { this._map.removeControl(this._subControl); this._subControl = null; }
|
||||
},
|
||||
|
||||
_toggleSub(id, on, map) {
|
||||
this._sublayers[id] = on;
|
||||
this._saveState();
|
||||
var m = { flights: ['_startFlights','_stopFlights'], ships: ['_startShips','_stopShips'],
|
||||
quakes: ['_startQuakes','_stopQuakes'], gdelt: ['_startGdelt','_stopGdelt'],
|
||||
heatmap: ['_startHeatmap','_stopHeatmap'], coords: ['_startCoords','_stopCoords'],
|
||||
distance: ['_startDistance','_stopDistance'] };
|
||||
if (m[id]) this[m[id][on ? 0 : 1]](map);
|
||||
},
|
||||
|
||||
_restoreSublayers(map) {
|
||||
var self = this;
|
||||
Object.keys(this._sublayers).forEach(function(id) {
|
||||
if (self._sublayers[id]) self._toggleSub(id, true, map);
|
||||
});
|
||||
},
|
||||
|
||||
// === FLUGVERKEHR ========================================================
|
||||
_startFlights(map) {
|
||||
if (this._flightLayer) return;
|
||||
this._flightLayer = L.layerGroup().addTo(map);
|
||||
var self = this;
|
||||
this._fetchFlights(map);
|
||||
this._flightInterval = setInterval(function() { self._fetchFlights(map); }, 30000);
|
||||
this._moveHandler = function() {
|
||||
clearTimeout(self._moveDebounce);
|
||||
self._moveDebounce = setTimeout(function() {
|
||||
self._renderFlights(map);
|
||||
self._renderShips(map);
|
||||
}, 500);
|
||||
};
|
||||
map.on('moveend', this._moveHandler);
|
||||
},
|
||||
|
||||
_stopFlights() {
|
||||
if (this._flightInterval) { clearInterval(this._flightInterval); this._flightInterval = null; }
|
||||
if (this._moveHandler && this._map) { this._map.off('moveend', this._moveHandler); this._moveHandler = null; }
|
||||
if (this._flightLayer && this._map) { this._map.removeLayer(this._flightLayer); this._flightLayer = null; }
|
||||
},
|
||||
|
||||
_fetchFlights(map) {
|
||||
if (this._flightFetching || !map) return;
|
||||
this._flightFetching = true;
|
||||
var self = this;
|
||||
var token = localStorage.getItem('osint_token') || '';
|
||||
fetch('/api/geoint/flights', { headers: token ? { 'Authorization': 'Bearer ' + token } : {} })
|
||||
.then(function(r) { return r.ok ? r.json() : { ac: [] }; })
|
||||
.then(function(data) {
|
||||
self._flightsData = data.ac || data.aircraft || [];
|
||||
self._renderFlights(map);
|
||||
})
|
||||
.catch(function() {})
|
||||
.finally(function() { self._flightFetching = false; });
|
||||
},
|
||||
|
||||
_renderFlights(map) {
|
||||
if (!map || !this._flightLayer || !this._flightsData) return;
|
||||
var newLayer = L.layerGroup();
|
||||
var bounds = map.getBounds();
|
||||
var zoom = map.getZoom();
|
||||
var max = zoom >= 10 ? 600 : zoom >= 7 ? 400 : zoom >= 5 ? 200 : 80;
|
||||
var r = zoom >= 10 ? 4 : zoom >= 7 ? 3 : 2;
|
||||
var count = 0;
|
||||
for (var i = 0; i < this._flightsData.length && count < max; i++) {
|
||||
var a = this._flightsData[i];
|
||||
if (!a.lat || !a.lon || !bounds.contains([a.lat, a.lon])) continue;
|
||||
count++;
|
||||
var cs = (a.flight || a.callsign || a.hex || '???').trim();
|
||||
var alt = a.alt_baro || a.altitude || '?';
|
||||
var spd = a.gs || a.ground_speed || '?';
|
||||
var popup = '<div class="geoint-popup"><strong>' + cs + '</strong>' +
|
||||
'<br><span class="geoint-popup-key">ALT</span> ' + (typeof alt === 'number' ? alt.toLocaleString() + ' ft' : alt) +
|
||||
'<br><span class="geoint-popup-key">SPD</span> ' + (typeof spd === 'number' ? Math.round(spd) + ' kts' : spd) + '</div>';
|
||||
L.circleMarker([a.lat, a.lon], {
|
||||
radius: r, fillColor: '#00ff88', color: '#004422',
|
||||
fillOpacity: 0.9, weight: 1, renderer: this._canvasRenderer
|
||||
}).bindPopup(popup, { className: 'geoint-leaflet-popup' }).addTo(newLayer);
|
||||
}
|
||||
this._map.removeLayer(this._flightLayer);
|
||||
this._flightLayer = newLayer.addTo(this._map);
|
||||
},
|
||||
|
||||
// === SCHIFFSVERKEHR =====================================================
|
||||
_startShips(map) {
|
||||
if (this._shipsLayer) return;
|
||||
this._shipsLayer = L.layerGroup().addTo(map);
|
||||
var self = this;
|
||||
this._fetchShips(map);
|
||||
this._shipsInterval = setInterval(function() { self._fetchShips(map); }, 60000);
|
||||
},
|
||||
|
||||
_stopShips() {
|
||||
if (this._shipsInterval) { clearInterval(this._shipsInterval); this._shipsInterval = null; }
|
||||
if (this._shipsLayer && this._map) { this._map.removeLayer(this._shipsLayer); this._shipsLayer = null; }
|
||||
},
|
||||
|
||||
_fetchShips(map) {
|
||||
var self = this;
|
||||
var token = localStorage.getItem('osint_token') || '';
|
||||
fetch('/api/geoint/ships', { headers: token ? { 'Authorization': 'Bearer ' + token } : {} })
|
||||
.then(function(r) { return r.ok ? r.json() : { ships: [] }; })
|
||||
.then(function(data) {
|
||||
self._shipsData = data.ships || [];
|
||||
self._renderShips(map);
|
||||
})
|
||||
.catch(function() {});
|
||||
},
|
||||
|
||||
_renderShips(map) {
|
||||
if (!map || !this._shipsLayer || !this._shipsData) return;
|
||||
var newLayer = L.layerGroup();
|
||||
var bounds = map.getBounds();
|
||||
var zoom = map.getZoom();
|
||||
var max = zoom >= 10 ? 800 : zoom >= 7 ? 400 : zoom >= 5 ? 150 : 50;
|
||||
var minSog = zoom >= 8 ? 0 : zoom >= 5 ? 0.3 : 1.0;
|
||||
var r = zoom >= 10 ? 3.5 : zoom >= 7 ? 2.5 : 2;
|
||||
var count = 0;
|
||||
for (var i = 0; i < this._shipsData.length && count < max; i++) {
|
||||
var s = this._shipsData[i];
|
||||
if (!s.lat || !s.lon || !bounds.contains([s.lat, s.lon])) continue;
|
||||
if ((s.sog || 0) < minSog) continue;
|
||||
count++;
|
||||
var color = (s.sog || 0) > 0.5 ? '#4499ff' : '#556688';
|
||||
var navLabels = {0:'Motor',1:'Anker',2:'N.steuerb.',3:'Eingeschr.',5:'Festgemacht',7:'Fischfang',8:'Segel'};
|
||||
var shipName = s.name || ('MMSI ' + (s.mmsi||'?'));
|
||||
var popup = '<div class="geoint-popup"><strong>' + shipName + '</strong>' +
|
||||
(s.name ? '<br><span class="geoint-popup-key">MMSI</span> ' + (s.mmsi||'?') : '') +
|
||||
'<br><span class="geoint-popup-key">SOG</span> ' + (s.sog||0).toFixed(1) + ' kn' +
|
||||
'<br><span class="geoint-popup-key">COG</span> ' + Math.round(s.cog||0) + '\u00b0' + '</div>';
|
||||
L.circleMarker([s.lat, s.lon], {
|
||||
radius: r, fillColor: color, color: '#223355',
|
||||
fillOpacity: 0.85, weight: 0.5, renderer: this._canvasRenderer
|
||||
}).bindPopup(popup, { className: 'geoint-leaflet-popup' }).addTo(newLayer);
|
||||
}
|
||||
this._map.removeLayer(this._shipsLayer);
|
||||
this._shipsLayer = newLayer.addTo(this._map);
|
||||
},
|
||||
|
||||
// === ERDBEBEN ============================================================
|
||||
_startQuakes(map) {
|
||||
if (this._quakeLayer) return;
|
||||
this._quakeLayer = L.layerGroup().addTo(map);
|
||||
this._fetchQuakes(map);
|
||||
var self = this;
|
||||
this._quakeInterval = setInterval(function() { self._fetchQuakes(map); }, 300000);
|
||||
},
|
||||
|
||||
_stopQuakes() {
|
||||
if (this._quakeInterval) { clearInterval(this._quakeInterval); this._quakeInterval = null; }
|
||||
if (this._quakeLayer && this._map) { this._map.removeLayer(this._quakeLayer); this._quakeLayer = null; }
|
||||
},
|
||||
|
||||
_fetchQuakes() {
|
||||
if (!this._quakeLayer) return;
|
||||
var self = this;
|
||||
fetch('https://earthquake.usgs.gov/earthquakes/feed/v1.0/summary/2.5_day.geojson')
|
||||
.then(function(r) { return r.json(); })
|
||||
.then(function(data) {
|
||||
if (!self._quakeLayer) return;
|
||||
self._quakeLayer.clearLayers();
|
||||
var now = Date.now();
|
||||
(data.features || []).forEach(function(f) {
|
||||
var c = f.geometry.coordinates, p = f.properties;
|
||||
var mag = p.mag || 1;
|
||||
var ageH = (now - p.time) / 3600000;
|
||||
var color = ageH < 1 ? '#ff0000' : ageH < 6 ? '#ff6600' : ageH < 12 ? '#ffaa00' : '#ffdd00';
|
||||
var cls = ageH < 2 ? 'geoint-quake-marker' : '';
|
||||
L.circleMarker([c[1], c[0]], {
|
||||
radius: Math.max(mag * 3.5, 5), fillColor: color, color: color,
|
||||
weight: 1.5, fillOpacity: 0.6, className: cls
|
||||
}).bindPopup('<div class="geoint-popup"><strong>M' + mag.toFixed(1) + '</strong> ' + (p.place||'') +
|
||||
'<br><span class="geoint-popup-key">TIEFE</span> ' + (c[2]||'?') + ' km</div>',
|
||||
{ className: 'geoint-leaflet-popup' }
|
||||
).addTo(self._quakeLayer);
|
||||
});
|
||||
})
|
||||
.catch(function() {});
|
||||
},
|
||||
|
||||
// === GDELT NACHRICHTEN ===================================================
|
||||
_startGdelt(map) {
|
||||
if (this._gdeltLayer) return;
|
||||
this._gdeltLayer = L.markerClusterGroup({
|
||||
maxClusterRadius: 30,
|
||||
iconCreateFunction: function(cluster) {
|
||||
var n = cluster.getChildCount();
|
||||
return L.divIcon({ html: '<div class="geoint-gdelt-icon">' + (n > 99 ? '99+' : n) + '</div>', className: '', iconSize: [22, 22] });
|
||||
}
|
||||
}).addTo(map);
|
||||
this._fetchGdelt(map);
|
||||
var self = this;
|
||||
this._gdeltInterval = setInterval(function() { self._fetchGdelt(map); }, 600000);
|
||||
},
|
||||
|
||||
_stopGdelt() {
|
||||
if (this._gdeltInterval) { clearInterval(this._gdeltInterval); this._gdeltInterval = null; }
|
||||
if (this._gdeltLayer && this._map) { this._map.removeLayer(this._gdeltLayer); this._gdeltLayer = null; }
|
||||
},
|
||||
|
||||
_fetchGdelt() {
|
||||
if (!this._gdeltLayer) return;
|
||||
var self = this;
|
||||
var query = 'conflict OR crisis OR disaster';
|
||||
if (typeof App !== 'undefined' && App.currentIncidentId) {
|
||||
var inc = (App.incidents || []).find(function(i) { return i.id === App.currentIncidentId; });
|
||||
if (inc && inc.title) query = encodeURIComponent(inc.title.substring(0, 80));
|
||||
}
|
||||
var token = localStorage.getItem('osint_token') || '';
|
||||
fetch('/api/geoint/gdelt?query=' + query, { headers: token ? { 'Authorization': 'Bearer ' + token } : {} })
|
||||
.then(function(r) { return r.ok ? r.json() : { features: [] }; })
|
||||
.then(function(data) {
|
||||
if (!self._gdeltLayer) return;
|
||||
self._gdeltLayer.clearLayers();
|
||||
(data.features || []).slice(0, 200).forEach(function(f) {
|
||||
var c = f.geometry.coordinates, p = f.properties || {};
|
||||
var icon = L.divIcon({ className: '', html: '<div class="geoint-gdelt-icon">N</div>', iconSize: [18, 18], iconAnchor: [9, 9] });
|
||||
var popup = '<div class="geoint-popup" style="max-width:240px"><strong>' + (p.name || p.title || 'Nachricht').substring(0, 100) + '</strong>' +
|
||||
(p.url ? '<br><a href="' + p.url + '" target="_blank" rel="noopener" style="color:#44aaff;font-size:10px">Quelle</a>' : '') + '</div>';
|
||||
L.marker([c[1], c[0]], { icon: icon }).bindPopup(popup, { className: 'geoint-leaflet-popup' }).addTo(self._gdeltLayer);
|
||||
});
|
||||
})
|
||||
.catch(function() {});
|
||||
},
|
||||
|
||||
// === HEATMAP =============================================================
|
||||
_startHeatmap(map) {
|
||||
if (this._heatLayer || typeof L.heatLayer === 'undefined') return;
|
||||
var locs = (typeof UI !== 'undefined' && UI._lastLocations) ? UI._lastLocations : [];
|
||||
if (!locs.length) return;
|
||||
var maxC = Math.max.apply(null, locs.map(function(l) { return l.article_count || 1; }));
|
||||
var pts = locs.map(function(l) { return [l.lat, l.lon, (l.article_count || 1) / maxC]; });
|
||||
this._heatLayer = L.heatLayer(pts, {
|
||||
radius: 30, blur: 20, maxZoom: 12,
|
||||
gradient: { 0.2: '#004400', 0.4: '#00ff88', 0.6: '#ffaa00', 0.8: '#ff4400', 1.0: '#ff0000' }
|
||||
}).addTo(map);
|
||||
},
|
||||
|
||||
_stopHeatmap() {
|
||||
if (this._heatLayer && this._map) { this._map.removeLayer(this._heatLayer); this._heatLayer = null; }
|
||||
},
|
||||
|
||||
// === KOORDINATENANZEIGE ==================================================
|
||||
_startCoords(map) {
|
||||
if (this._coordControl) return;
|
||||
var Ctrl = L.Control.extend({
|
||||
options: { position: 'bottomleft' },
|
||||
onAdd: function() {
|
||||
var div = L.DomUtil.create('div', 'geoint-coord-display');
|
||||
div.id = 'geoint-coord-text';
|
||||
div.textContent = 'LAT: -- LON: --';
|
||||
return div;
|
||||
}
|
||||
});
|
||||
this._coordControl = new Ctrl();
|
||||
map.addControl(this._coordControl);
|
||||
var el = document.getElementById('geoint-coord-text');
|
||||
this._coordHandler = function(e) {
|
||||
if (el) el.textContent = 'LAT: ' + e.latlng.lat.toFixed(4) + ' LON: ' + e.latlng.lng.toFixed(4);
|
||||
};
|
||||
map.on('mousemove', this._coordHandler);
|
||||
},
|
||||
|
||||
_stopCoords() {
|
||||
if (this._coordHandler && this._map) { this._map.off('mousemove', this._coordHandler); this._coordHandler = null; }
|
||||
if (this._coordControl && this._map) { this._map.removeControl(this._coordControl); this._coordControl = null; }
|
||||
},
|
||||
|
||||
// === DISTANZMESSUNG ======================================================
|
||||
_startDistance(map) {
|
||||
if (this._distanceLayers) return;
|
||||
this._distanceLayers = L.layerGroup().addTo(map);
|
||||
this._distancePoints = [];
|
||||
map.getContainer().style.cursor = 'crosshair';
|
||||
var self = this;
|
||||
this._distanceHandler = function(e) {
|
||||
self._distancePoints.push(e.latlng);
|
||||
L.circleMarker(e.latlng, { radius: 6, fillColor: '#ff2222', color: '#ffffff', fillOpacity: 1, weight: 2 }).addTo(self._distanceLayers);
|
||||
if (self._distancePoints.length >= 2) {
|
||||
var p1 = self._distancePoints[self._distancePoints.length - 2];
|
||||
var p2 = self._distancePoints[self._distancePoints.length - 1];
|
||||
L.polyline([p1, p2], { color: '#000000', weight: 5, opacity: 0.5 }).addTo(self._distanceLayers);
|
||||
L.polyline([p1, p2], { color: '#ff2222', weight: 3, dashArray: '8 5' }).addTo(self._distanceLayers);
|
||||
var dist = p1.distanceTo(p2);
|
||||
var total = 0;
|
||||
for (var i = 1; i < self._distancePoints.length; i++) total += self._distancePoints[i-1].distanceTo(self._distancePoints[i]);
|
||||
var label = dist >= 1000 ? (dist/1000).toFixed(1) + ' km' : Math.round(dist) + ' m';
|
||||
var tLabel = total >= 1000 ? (total/1000).toFixed(1) + ' km' : Math.round(total) + ' m';
|
||||
var text = self._distancePoints.length > 2 ? label + ' (\u03a3 ' + tLabel + ')' : label;
|
||||
L.marker([(p1.lat+p2.lat)/2, (p1.lng+p2.lng)/2], {
|
||||
icon: L.divIcon({ className: '', html: '<div class="geoint-distance-label">' + text + '</div>', iconSize: [0,0], iconAnchor: [0,12] })
|
||||
}).addTo(self._distanceLayers);
|
||||
}
|
||||
};
|
||||
map.on('click', this._distanceHandler);
|
||||
},
|
||||
|
||||
_stopDistance() {
|
||||
this._distancePoints = [];
|
||||
if (this._distanceHandler && this._map) { this._map.off('click', this._distanceHandler); this._distanceHandler = null; this._map.getContainer().style.cursor = ''; }
|
||||
if (this._distanceLayers && this._map) { this._map.removeLayer(this._distanceLayers); this._distanceLayers = null; }
|
||||
},
|
||||
|
||||
// === TIMELINE ============================================================
|
||||
initTimeline(articles) {
|
||||
if (!articles || !articles.length) return;
|
||||
var dates = articles.map(function(a) { return a.collected_at || a.published_at; })
|
||||
.filter(Boolean).map(function(d) { return new Date(d).getTime(); })
|
||||
.filter(function(t) { return !isNaN(t); }).sort(function(a,b) { return a-b; });
|
||||
if (dates.length < 2) return;
|
||||
this._timelineData = { min: dates[0], max: dates[dates.length-1], articles: articles };
|
||||
var slider = document.getElementById('geoint-timeline-slider');
|
||||
if (slider) { slider.min = dates[0]; slider.max = dates[dates.length-1]; slider.value = dates[dates.length-1]; }
|
||||
var label = document.getElementById('geoint-timeline-label');
|
||||
if (label) label.textContent = this._fmtDate(dates[dates.length-1]);
|
||||
},
|
||||
|
||||
_onTimelineChange(val) {
|
||||
var label = document.getElementById('geoint-timeline-label');
|
||||
if (label) label.textContent = this._fmtDate(parseInt(val));
|
||||
if (!this._map || !UI._mapCluster || !this._timelineData) return;
|
||||
var maxT = parseInt(val), arts = this._timelineData.articles;
|
||||
var vis = new Set();
|
||||
arts.forEach(function(a) { if (new Date(a.collected_at || a.published_at || 0).getTime() <= maxT) vis.add(a.id); });
|
||||
UI._mapCluster.eachLayer(function(m) {
|
||||
if (m._articleIds) m.setOpacity(m._articleIds.some(function(id) { return vis.has(id); }) ? 1 : 0.08);
|
||||
});
|
||||
},
|
||||
|
||||
_resetTimeline() {
|
||||
if (this._timelineData) {
|
||||
var slider = document.getElementById('geoint-timeline-slider');
|
||||
if (slider) { slider.value = this._timelineData.max; this._onTimelineChange(this._timelineData.max); }
|
||||
}
|
||||
},
|
||||
|
||||
_fmtDate(ts) {
|
||||
var d = new Date(ts);
|
||||
return d.toLocaleDateString('de-DE', { day: '2-digit', month: '2-digit' }) + ' ' +
|
||||
d.toLocaleTimeString('de-DE', { hour: '2-digit', minute: '2-digit' });
|
||||
},
|
||||
|
||||
// === CLEANUP =============================================================
|
||||
cleanup() {
|
||||
this._stopFlights(); this._stopShips(); this._stopQuakes();
|
||||
this._stopGdelt(); this._stopHeatmap(); this._stopCoords(); this._stopDistance();
|
||||
this._removeSubControl();
|
||||
document.querySelectorAll('.geoint-active').forEach(function(el) { el.classList.remove('geoint-active'); });
|
||||
},
|
||||
|
||||
// === STATE ===============================================================
|
||||
_saveState() {
|
||||
try {
|
||||
localStorage.setItem('geoint_mode', this._active ? 'true' : 'false');
|
||||
localStorage.setItem('geoint_sublayers', JSON.stringify(this._sublayers));
|
||||
} catch(e) {}
|
||||
},
|
||||
|
||||
restoreState(map) {
|
||||
if (!map) return;
|
||||
this._map = map;
|
||||
try { var s = localStorage.getItem('geoint_sublayers'); if (s) this._sublayers = JSON.parse(s); } catch(e) { this._sublayers = {}; }
|
||||
if (localStorage.getItem('geoint_mode') === 'true') this.toggle(true, map);
|
||||
},
|
||||
};
|
||||
993
src/static/js/network-cluster.js
Normale Datei
993
src/static/js/network-cluster.js
Normale Datei
@@ -0,0 +1,993 @@
|
||||
/**
|
||||
* AegisSight OSINT Monitor - Cluster Graph Visualization v2
|
||||
*
|
||||
* Hierarchical country-based network visualization powered by d3.js v7.
|
||||
* Level 1: Country overview with prominent inter-country edges
|
||||
* Level 2: Country drill-down (entities within a country)
|
||||
*
|
||||
* Requires: d3 (global), ClusterData (cluster-data.js)
|
||||
*/
|
||||
|
||||
/* global d3, ClusterData, NetworkGraph */
|
||||
|
||||
var ClusterGraph = {
|
||||
|
||||
_svg: null,
|
||||
_g: null,
|
||||
_zoom: null,
|
||||
_simulation: null,
|
||||
_tooltip: null,
|
||||
_container: null,
|
||||
_allEntities: null,
|
||||
_allRelations: null,
|
||||
_clusterData: null,
|
||||
_entityMap: null,
|
||||
_currentLevel: 'overview',
|
||||
_currentCountry: null,
|
||||
_width: 960,
|
||||
_height: 640,
|
||||
|
||||
_categoryColors: {
|
||||
conflict: '#EF4444',
|
||||
alliance: '#22C55E',
|
||||
diplomacy: '#3B82F6',
|
||||
economic: '#FBBF24',
|
||||
neutral: '#6B7280',
|
||||
legal: '#A855F7'
|
||||
},
|
||||
|
||||
_entityTypeColors: {
|
||||
person: '#60A5FA',
|
||||
organisation: '#C084FC',
|
||||
location: '#34D399',
|
||||
event: '#FBBF24',
|
||||
military: '#F87171'
|
||||
},
|
||||
|
||||
_categoryLabels: {
|
||||
conflict: 'Konflikt', alliance: 'Allianz', diplomacy: 'Diplomatie',
|
||||
economic: 'Ökonomie', neutral: 'Neutral', legal: 'Recht'
|
||||
},
|
||||
|
||||
_typeLabels: {
|
||||
person: 'Personen', organisation: 'Organisationen',
|
||||
location: 'Orte', event: 'Ereignisse', military: 'Militär'
|
||||
},
|
||||
|
||||
// ---- public API -----------------------------------------------------------
|
||||
|
||||
init: function(containerId, entities, relations) {
|
||||
this.destroy();
|
||||
var wrapper = document.getElementById(containerId);
|
||||
if (!wrapper) return;
|
||||
wrapper.innerHTML = '';
|
||||
this._container = wrapper;
|
||||
this._allEntities = entities;
|
||||
this._allRelations = relations;
|
||||
|
||||
this._entityMap = new Map();
|
||||
for (var i = 0; i < entities.length; i++) {
|
||||
this._entityMap.set(entities[i].id, entities[i]);
|
||||
}
|
||||
|
||||
var rect = wrapper.getBoundingClientRect();
|
||||
this._width = rect.width || 960;
|
||||
this._height = rect.height || 640;
|
||||
|
||||
this._svg = d3.select(wrapper)
|
||||
.append('svg')
|
||||
.attr('width', '100%')
|
||||
.attr('height', '100%')
|
||||
.attr('viewBox', '0 0 ' + this._width + ' ' + this._height)
|
||||
.attr('preserveAspectRatio', 'xMidYMid meet')
|
||||
.style('background', 'transparent');
|
||||
|
||||
this._createDefs();
|
||||
this._g = this._svg.append('g').attr('class', 'cg-zoom-layer');
|
||||
|
||||
this._zoom = d3.zoom()
|
||||
.scaleExtent([0.2, 6])
|
||||
.on('zoom', function(event) {
|
||||
ClusterGraph._g.attr('transform', event.transform);
|
||||
});
|
||||
this._svg.call(this._zoom);
|
||||
this._svg.on('dblclick.zoom', null);
|
||||
|
||||
this._tooltip = d3.select(wrapper)
|
||||
.append('div')
|
||||
.attr('class', 'cg-tooltip')
|
||||
.style('position', 'absolute')
|
||||
.style('pointer-events', 'none')
|
||||
.style('background', 'rgba(15,23,42,0.95)')
|
||||
.style('color', '#e2e8f0')
|
||||
.style('border', '1px solid #334155')
|
||||
.style('border-radius', '8px')
|
||||
.style('padding', '10px 14px')
|
||||
.style('font-size', '12px')
|
||||
.style('max-width', '320px')
|
||||
.style('z-index', '1000')
|
||||
.style('display', 'none')
|
||||
.style('line-height', '1.6');
|
||||
|
||||
this._clusterData = ClusterData.buildClusterData(entities, relations);
|
||||
this._currentLevel = 'overview';
|
||||
this._currentCountry = null;
|
||||
this._renderOverview();
|
||||
this._updateBreadcrumb();
|
||||
this._renderCountrySidebar();
|
||||
},
|
||||
|
||||
destroy: function() {
|
||||
if (this._simulation) { this._simulation.stop(); this._simulation = null; }
|
||||
if (this._svg) { this._svg.remove(); this._svg = null; }
|
||||
if (this._tooltip) { this._tooltip.remove(); this._tooltip = null; }
|
||||
this._g = null;
|
||||
this._clusterData = null;
|
||||
this._allEntities = null;
|
||||
this._allRelations = null;
|
||||
this._entityMap = null;
|
||||
this._currentLevel = 'overview';
|
||||
this._currentCountry = null;
|
||||
},
|
||||
|
||||
// ---- LEVEL 1: Country Overview -------------------------------------------
|
||||
|
||||
_renderOverview: function() {
|
||||
var self = this;
|
||||
if (this._simulation) this._simulation.stop();
|
||||
this._g.selectAll('*').remove();
|
||||
|
||||
// Filter: no "Sonstige", no empty, minimum 10 entities
|
||||
var countries = this._clusterData.countries.filter(function(c) {
|
||||
return c.entityCount >= 10 && !c.isUnassigned;
|
||||
});
|
||||
|
||||
var edges = this._clusterData.edges.slice();
|
||||
|
||||
// Radius scale
|
||||
var maxCount = 1;
|
||||
for (var i = 0; i < countries.length; i++) {
|
||||
if (countries[i].entityCount > maxCount) maxCount = countries[i].entityCount;
|
||||
}
|
||||
var rScale = d3.scaleSqrt().domain([0, maxCount]).range([22, 65]);
|
||||
|
||||
for (var ci = 0; ci < countries.length; ci++) {
|
||||
countries[ci]._radius = rScale(countries[ci].entityCount);
|
||||
countries[ci].id = countries[ci].canonicalName;
|
||||
}
|
||||
|
||||
// Visible edges only
|
||||
var countryNames = new Set(countries.map(function(c) { return c.canonicalName; }));
|
||||
var visibleEdges = edges.filter(function(e) {
|
||||
return countryNames.has(e.source) && countryNames.has(e.target) && e.count >= 3;
|
||||
});
|
||||
|
||||
// Edge scale
|
||||
var maxEdgeCount = 1;
|
||||
for (var ei = 0; ei < visibleEdges.length; ei++) {
|
||||
if (visibleEdges[ei].count > maxEdgeCount) maxEdgeCount = visibleEdges[ei].count;
|
||||
}
|
||||
var edgeScale = d3.scaleSqrt().domain([1, maxEdgeCount]).range([2, 18]);
|
||||
|
||||
// ---- EDGES (drawn first = behind nodes) ----
|
||||
var linkGroup = this._g.append('g').attr('class', 'cg-links');
|
||||
var linkSel = linkGroup.selectAll('line')
|
||||
.data(visibleEdges)
|
||||
.join('line')
|
||||
.attr('stroke', function(d) {
|
||||
return self._categoryColors[d.dominantCategory] || '#6B7280';
|
||||
})
|
||||
.attr('stroke-width', function(d) { return edgeScale(d.count); })
|
||||
.attr('stroke-opacity', 0.6)
|
||||
.attr('stroke-linecap', 'round')
|
||||
.style('cursor', 'pointer')
|
||||
.on('mouseover', function(event, d) {
|
||||
d3.select(this).attr('stroke-opacity', 1);
|
||||
var lines = ['<strong>' + self._esc(d.source) + ' \u2194 ' + self._esc(d.target) + '</strong>'];
|
||||
lines.push('<span style="font-size:14px;font-weight:600;">' + d.count + ' Beziehungen</span>');
|
||||
var cats = Object.keys(d.categories).sort(function(a, b) {
|
||||
return d.categories[b] - d.categories[a];
|
||||
});
|
||||
for (var ci = 0; ci < Math.min(cats.length, 4); ci++) {
|
||||
var c = cats[ci];
|
||||
var color = self._categoryColors[c] || '#6B7280';
|
||||
lines.push('<span style="color:' + color + ';">\u25CF</span> ' +
|
||||
(self._categoryLabels[c] || c) + ': ' + d.categories[c]);
|
||||
}
|
||||
self._showTooltip(event, lines.join('<br>'));
|
||||
})
|
||||
.on('mousemove', function(event) { self._moveTooltip(event); })
|
||||
.on('mouseout', function() {
|
||||
d3.select(this).attr('stroke-opacity', 0.6);
|
||||
self._hideTooltip();
|
||||
});
|
||||
|
||||
// Edge labels (count) on top edges
|
||||
var topEdges = visibleEdges.filter(function(e) { return e.count >= 10; });
|
||||
var edgeLabelGroup = this._g.append('g').attr('class', 'cg-edge-labels');
|
||||
var edgeLabelSel = edgeLabelGroup.selectAll('text')
|
||||
.data(topEdges)
|
||||
.join('text')
|
||||
.attr('text-anchor', 'middle')
|
||||
.attr('fill', function(d) {
|
||||
return self._categoryColors[d.dominantCategory] || '#94a3b8';
|
||||
})
|
||||
.attr('font-size', '11px')
|
||||
.attr('font-weight', '700')
|
||||
.attr('pointer-events', 'none')
|
||||
.text(function(d) { return d.count; });
|
||||
|
||||
// ---- NODES ----
|
||||
var nodeGroup = this._g.append('g').attr('class', 'cg-nodes');
|
||||
var nodeSel = nodeGroup.selectAll('g')
|
||||
.data(countries)
|
||||
.join('g')
|
||||
.attr('class', 'cg-country-node')
|
||||
.style('cursor', 'pointer')
|
||||
.call(this._drag());
|
||||
|
||||
// Main circle
|
||||
nodeSel.append('circle')
|
||||
.attr('class', 'cg-country-circle')
|
||||
.attr('r', function(d) { return d._radius; })
|
||||
.attr('fill', function(d) { return self._getCountryFill(d); })
|
||||
.attr('stroke', '#e2e8f0')
|
||||
.attr('stroke-width', 2)
|
||||
.attr('opacity', 0.9);
|
||||
|
||||
// Mini donut
|
||||
nodeSel.each(function(d) {
|
||||
self._renderMiniDonut(d3.select(this), d);
|
||||
});
|
||||
|
||||
// Country name
|
||||
nodeSel.append('text')
|
||||
.attr('class', 'cg-country-label')
|
||||
.text(function(d) { return d.name; })
|
||||
.attr('text-anchor', 'middle')
|
||||
.attr('dy', -6)
|
||||
.attr('fill', '#f1f5f9')
|
||||
.attr('font-size', function(d) {
|
||||
return Math.max(10, Math.min(15, d._radius / 3.5)) + 'px';
|
||||
})
|
||||
.attr('font-weight', '700')
|
||||
.attr('pointer-events', 'none');
|
||||
|
||||
// Entity count
|
||||
nodeSel.append('text')
|
||||
.attr('text-anchor', 'middle')
|
||||
.attr('dy', 8)
|
||||
.attr('fill', '#cbd5e1')
|
||||
.attr('font-size', '10px')
|
||||
.attr('pointer-events', 'none')
|
||||
.text(function(d) { return d.entityCount; });
|
||||
|
||||
// Top actor name below circle
|
||||
nodeSel.append('text')
|
||||
.attr('text-anchor', 'middle')
|
||||
.attr('dy', function(d) { return d._radius + 16; })
|
||||
.attr('fill', '#94a3b8')
|
||||
.attr('font-size', '9px')
|
||||
.attr('font-style', 'italic')
|
||||
.attr('pointer-events', 'none')
|
||||
.text(function(d) {
|
||||
if (!d.topEntities || d.topEntities.length === 0) return '';
|
||||
var top = d.topEntities[0];
|
||||
var name = top.name.length > 22 ? top.name.slice(0, 20) + '\u2026' : top.name;
|
||||
return name;
|
||||
});
|
||||
|
||||
// Click -> drill down
|
||||
nodeSel.on('click', function(event, d) {
|
||||
event.stopPropagation();
|
||||
self._drillDown(d.canonicalName);
|
||||
});
|
||||
|
||||
// Hover
|
||||
nodeSel.on('mouseover', function(event, d) {
|
||||
d3.select(this).select('.cg-country-circle')
|
||||
.transition().duration(150)
|
||||
.attr('stroke-width', 4).attr('opacity', 1);
|
||||
|
||||
// Highlight connected edges
|
||||
linkSel.attr('stroke-opacity', function(e) {
|
||||
return (e.source === d.canonicalName || e.target === d.canonicalName ||
|
||||
(e.source.id && e.source.id === d.canonicalName) ||
|
||||
(e.target.id && e.target.id === d.canonicalName)) ? 0.9 : 0.15;
|
||||
});
|
||||
|
||||
var lines = ['<strong style="font-size:14px;">' + self._esc(d.name) + '</strong>'];
|
||||
lines.push(d.entityCount + ' Entitäten');
|
||||
var tc = d.typeCounts;
|
||||
var parts = [];
|
||||
if (tc.person) parts.push(tc.person + ' Pers.');
|
||||
if (tc.organisation) parts.push(tc.organisation + ' Org.');
|
||||
if (tc.military) parts.push(tc.military + ' Mil.');
|
||||
if (tc.event) parts.push(tc.event + ' Ereig.');
|
||||
if (parts.length) lines.push(parts.join(' \u00B7 '));
|
||||
if (d.topEntities && d.topEntities.length > 0) {
|
||||
lines.push('<hr style="border-color:#334155;margin:4px 0;">');
|
||||
for (var ti = 0; ti < Math.min(d.topEntities.length, 4); ti++) {
|
||||
var te = d.topEntities[ti];
|
||||
var typeColor = self._entityTypeColors[te.entity_type] || '#94a3b8';
|
||||
lines.push('<span style="color:' + typeColor + ';">\u25CF</span> ' +
|
||||
self._esc(te.name));
|
||||
}
|
||||
}
|
||||
self._showTooltip(event, lines.join('<br>'));
|
||||
});
|
||||
nodeSel.on('mousemove', function(event) { self._moveTooltip(event); });
|
||||
nodeSel.on('mouseout', function(event, d) {
|
||||
d3.select(this).select('.cg-country-circle')
|
||||
.transition().duration(150)
|
||||
.attr('stroke-width', 2).attr('opacity', 0.9);
|
||||
linkSel.attr('stroke-opacity', 0.6);
|
||||
self._hideTooltip();
|
||||
});
|
||||
|
||||
// ---- Force simulation ----
|
||||
var simLinks = visibleEdges.map(function(e) {
|
||||
return { source: e.source, target: e.target, count: e.count };
|
||||
});
|
||||
|
||||
this._simulation = d3.forceSimulation(countries)
|
||||
.force('link', d3.forceLink(simLinks)
|
||||
.id(function(d) { return d.id; })
|
||||
.distance(function(d) { return 180; })
|
||||
.strength(0.5))
|
||||
.force('charge', d3.forceManyBody()
|
||||
.strength(function(d) { return -400 - d._radius * 6; }))
|
||||
.force('center', d3.forceCenter(self._width / 2, self._height / 2))
|
||||
.force('collide', d3.forceCollide()
|
||||
.radius(function(d) { return d._radius + 30; })
|
||||
.strength(0.9))
|
||||
.alphaDecay(0.025);
|
||||
|
||||
this._simulation.on('tick', function() {
|
||||
linkSel
|
||||
.attr('x1', function(d) { return d.source.x; })
|
||||
.attr('y1', function(d) { return d.source.y; })
|
||||
.attr('x2', function(d) { return d.target.x; })
|
||||
.attr('y2', function(d) { return d.target.y; });
|
||||
|
||||
edgeLabelSel
|
||||
.attr('x', function(d) { return (d.source.x + d.target.x) / 2; })
|
||||
.attr('y', function(d) { return (d.source.y + d.target.y) / 2 - 4; });
|
||||
|
||||
nodeSel.attr('transform', function(d) {
|
||||
return 'translate(' + d.x + ',' + d.y + ')';
|
||||
});
|
||||
});
|
||||
|
||||
// Background click
|
||||
this._svg.on('click', function() {
|
||||
linkSel.attr('stroke-opacity', 0.6);
|
||||
});
|
||||
|
||||
// Zoom-to-fit after simulation stabilizes
|
||||
var tickCount = 0;
|
||||
this._simulation.on('tick.zoomfit', function() {
|
||||
tickCount++;
|
||||
if (tickCount === 120) {
|
||||
self._zoomToFit(countries, 40);
|
||||
self._simulation.on('tick.zoomfit', null); // Remove this listener
|
||||
}
|
||||
});
|
||||
},
|
||||
|
||||
_zoomToFit: function(nodes, padding) {
|
||||
if (!nodes || nodes.length === 0 || !this._svg || !this._zoom) return;
|
||||
|
||||
var minX = Infinity, minY = Infinity, maxX = -Infinity, maxY = -Infinity;
|
||||
for (var i = 0; i < nodes.length; i++) {
|
||||
var n = nodes[i];
|
||||
if (n.x === undefined) continue;
|
||||
var r = n._radius || 30;
|
||||
if (n.x - r < minX) minX = n.x - r;
|
||||
if (n.y - r < minY) minY = n.y - r;
|
||||
if (n.x + r > maxX) maxX = n.x + r;
|
||||
if (n.y + r > maxY) maxY = n.y + r;
|
||||
}
|
||||
|
||||
var graphWidth = maxX - minX + padding * 2;
|
||||
var graphHeight = maxY - minY + padding * 2;
|
||||
var scale = Math.min(
|
||||
this._width / graphWidth,
|
||||
this._height / graphHeight,
|
||||
1.5 // Max zoom
|
||||
);
|
||||
scale = Math.max(scale, 0.3); // Min zoom
|
||||
|
||||
var cx = (minX + maxX) / 2;
|
||||
var cy = (minY + maxY) / 2;
|
||||
var tx = this._width / 2 - cx * scale;
|
||||
var ty = this._height / 2 - cy * scale;
|
||||
|
||||
this._svg.transition().duration(600).call(
|
||||
this._zoom.transform,
|
||||
d3.zoomIdentity.translate(tx, ty).scale(scale)
|
||||
);
|
||||
},
|
||||
|
||||
// ---- LEVEL 2: Country Drill-down -----------------------------------------
|
||||
|
||||
_drillDown: function(countryName) {
|
||||
var self = this;
|
||||
this._currentLevel = 'country';
|
||||
this._currentCountry = countryName;
|
||||
this._updateBreadcrumb();
|
||||
this._renderCountrySidebar();
|
||||
|
||||
this._g.transition().duration(350).style('opacity', 0)
|
||||
.on('end', function() {
|
||||
if (self._simulation) self._simulation.stop();
|
||||
self._g.selectAll('*').remove();
|
||||
self._renderCountryDetail(countryName);
|
||||
self._g.style('opacity', 0)
|
||||
.transition().duration(350).style('opacity', 1);
|
||||
});
|
||||
|
||||
this._svg.transition().duration(350).call(
|
||||
this._zoom.transform, d3.zoomIdentity
|
||||
);
|
||||
},
|
||||
|
||||
_renderCountryDetail: function(countryName) {
|
||||
var self = this;
|
||||
var entityIds = this._clusterData.assignments.get(countryName) || [];
|
||||
if (entityIds.length === 0) {
|
||||
this._g.append('text')
|
||||
.attr('x', this._width / 2).attr('y', this._height / 2)
|
||||
.attr('text-anchor', 'middle').attr('fill', '#94a3b8')
|
||||
.attr('font-size', '16px')
|
||||
.text('Keine Entitäten für ' + countryName);
|
||||
return;
|
||||
}
|
||||
|
||||
var idSet = new Set(entityIds);
|
||||
var entities = [];
|
||||
for (var i = 0; i < entityIds.length; i++) {
|
||||
var e = this._entityMap.get(entityIds[i]);
|
||||
if (e) entities.push({ ...e });
|
||||
}
|
||||
|
||||
var internalRelations = [];
|
||||
for (var ri = 0; ri < this._allRelations.length; ri++) {
|
||||
var r = this._allRelations[ri];
|
||||
if (idSet.has(r.source_entity_id) && idSet.has(r.target_entity_id)) {
|
||||
internalRelations.push(r);
|
||||
}
|
||||
}
|
||||
|
||||
// Connection counts for sizing
|
||||
var connCounts = {};
|
||||
for (var ii = 0; ii < internalRelations.length; ii++) {
|
||||
var ir = internalRelations[ii];
|
||||
connCounts[ir.source_entity_id] = (connCounts[ir.source_entity_id] || 0) + 1;
|
||||
connCounts[ir.target_entity_id] = (connCounts[ir.target_entity_id] || 0) + 1;
|
||||
}
|
||||
|
||||
var maxConn = 1;
|
||||
for (var k in connCounts) {
|
||||
if (connCounts[k] > maxConn) maxConn = connCounts[k];
|
||||
}
|
||||
var rScale = d3.scaleSqrt().domain([0, maxConn]).range([4, 26]);
|
||||
|
||||
entities.forEach(function(n) {
|
||||
n._connections = connCounts[n.id] || 0;
|
||||
n._radius = rScale(n._connections);
|
||||
});
|
||||
|
||||
// Show labels for top 30 or nodes with radius >= 10
|
||||
var sorted = entities.slice().sort(function(a, b) { return b._connections - a._connections; });
|
||||
var labelThreshold = sorted.length > 30 ? sorted[29]._connections : 0;
|
||||
|
||||
// Links
|
||||
var linkGroup = this._g.append('g');
|
||||
var simLinks = internalRelations.map(function(r) {
|
||||
return { source: r.source_entity_id, target: r.target_entity_id,
|
||||
category: r.category, weight: r.weight || 1 };
|
||||
});
|
||||
|
||||
var linkSel = linkGroup.selectAll('line')
|
||||
.data(simLinks).join('line')
|
||||
.attr('stroke', function(d) { return self._categoryColors[d.category] || '#6B7280'; })
|
||||
.attr('stroke-width', function(d) { return Math.max(0.5, Math.min(3, d.weight * 0.6)); })
|
||||
.attr('stroke-opacity', 0.25);
|
||||
|
||||
// Nodes
|
||||
var nodeGroup = this._g.append('g');
|
||||
var nodeSel = nodeGroup.selectAll('g')
|
||||
.data(entities, function(d) { return d.id; })
|
||||
.join('g').style('cursor', 'pointer').call(this._drag());
|
||||
|
||||
nodeSel.append('circle')
|
||||
.attr('r', function(d) { return d._radius; })
|
||||
.attr('fill', function(d) { return self._entityTypeColors[d.entity_type] || '#94A3B8'; })
|
||||
.attr('stroke', '#0f172a').attr('stroke-width', 1.5).attr('opacity', 0.85);
|
||||
|
||||
nodeSel.filter(function(d) {
|
||||
return d._connections >= labelThreshold || d._radius >= 10;
|
||||
}).append('text')
|
||||
.text(function(d) { return d.name.length > 20 ? d.name.slice(0, 18) + '\u2026' : d.name; })
|
||||
.attr('dy', function(d) { return d._radius + 13; })
|
||||
.attr('text-anchor', 'middle').attr('fill', '#cbd5e1')
|
||||
.attr('font-size', '10px').attr('pointer-events', 'none');
|
||||
|
||||
// Hover
|
||||
nodeSel.on('mouseover', function(event, d) {
|
||||
d3.select(this).select('circle')
|
||||
.transition().duration(100)
|
||||
.attr('stroke', '#FBBF24').attr('stroke-width', 3).attr('opacity', 1);
|
||||
var lines = ['<strong>' + self._esc(d.name) + '</strong>'];
|
||||
lines.push(self._typeLabels[d.entity_type] || d.entity_type);
|
||||
if (d.description) {
|
||||
lines.push('<span style="color:#94a3b8;">' +
|
||||
self._esc(d.description.length > 100 ? d.description.slice(0, 97) + '...' : d.description) +
|
||||
'</span>');
|
||||
}
|
||||
lines.push('Verbindungen: ' + d._connections);
|
||||
self._showTooltip(event, lines.join('<br>'));
|
||||
});
|
||||
nodeSel.on('mousemove', function(event) { self._moveTooltip(event); });
|
||||
nodeSel.on('mouseout', function() {
|
||||
d3.select(this).select('circle')
|
||||
.transition().duration(100)
|
||||
.attr('stroke', '#0f172a').attr('stroke-width', 1.5).attr('opacity', 0.85);
|
||||
self._hideTooltip();
|
||||
});
|
||||
|
||||
// Click: highlight neighborhood
|
||||
nodeSel.on('click', function(event, d) {
|
||||
event.stopPropagation();
|
||||
var connIds = new Set([d.id]);
|
||||
linkSel.each(function(l) {
|
||||
var s = typeof l.source === 'object' ? l.source.id : l.source;
|
||||
var t = typeof l.target === 'object' ? l.target.id : l.target;
|
||||
if (s === d.id || t === d.id) { connIds.add(s); connIds.add(t); }
|
||||
});
|
||||
linkSel.attr('stroke-opacity', function(l) {
|
||||
var s = typeof l.source === 'object' ? l.source.id : l.source;
|
||||
var t = typeof l.target === 'object' ? l.target.id : l.target;
|
||||
return (s === d.id || t === d.id) ? 0.8 : 0.04;
|
||||
});
|
||||
nodeSel.select('circle').attr('opacity', function(n) { return connIds.has(n.id) ? 1 : 0.12; });
|
||||
nodeSel.select('text').attr('opacity', function(n) { return connIds.has(n.id) ? 1 : 0.08; });
|
||||
self._updateDetailPanel(d);
|
||||
});
|
||||
|
||||
this._svg.on('click', function() {
|
||||
nodeSel.select('circle').attr('stroke', '#0f172a').attr('stroke-width', 1.5).attr('opacity', 0.85);
|
||||
linkSel.attr('stroke-opacity', 0.25);
|
||||
self._clearDetailPanel();
|
||||
});
|
||||
|
||||
// Force
|
||||
this._simulation = d3.forceSimulation(entities)
|
||||
.force('link', d3.forceLink(simLinks).id(function(d) { return d.id; })
|
||||
.distance(function(d) { return Math.max(30, 100 - d.weight * 10); }))
|
||||
.force('charge', d3.forceManyBody().strength(function(d) { return -60 - d._radius * 3; }))
|
||||
.force('center', d3.forceCenter(self._width / 2, self._height / 2))
|
||||
.force('collide', d3.forceCollide().radius(function(d) { return d._radius + 3; }))
|
||||
.alphaDecay(0.02);
|
||||
|
||||
this._simulation.on('tick', function() {
|
||||
linkSel.attr('x1', function(d) { return d.source.x; }).attr('y1', function(d) { return d.source.y; })
|
||||
.attr('x2', function(d) { return d.target.x; }).attr('y2', function(d) { return d.target.y; });
|
||||
nodeSel.attr('transform', function(d) { return 'translate(' + d.x + ',' + d.y + ')'; });
|
||||
});
|
||||
|
||||
// Zoom-to-fit for detail view
|
||||
var detailTickCount = 0;
|
||||
this._simulation.on('tick.zoomfit', function() {
|
||||
detailTickCount++;
|
||||
if (detailTickCount === 100) {
|
||||
self._zoomToFit(entities, 30);
|
||||
self._simulation.on('tick.zoomfit', null);
|
||||
}
|
||||
});
|
||||
},
|
||||
|
||||
// ---- Sidebar: Country List -----------------------------------------------
|
||||
|
||||
// ---- Filter state --------------------------------------------------------
|
||||
|
||||
_activeCategories: null, // null = all active
|
||||
_searchTerm: '',
|
||||
|
||||
_initFilters: function() {
|
||||
this._activeCategories = new Set(['conflict', 'alliance', 'diplomacy', 'economic', 'neutral', 'legal']);
|
||||
this._searchTerm = '';
|
||||
},
|
||||
|
||||
_applyEdgeFilter: function() {
|
||||
if (!this._g) return;
|
||||
var active = this._activeCategories;
|
||||
this._g.selectAll('.cg-links line').attr('display', function(d) {
|
||||
return active.has(d.dominantCategory) ? null : 'none';
|
||||
});
|
||||
this._g.selectAll('.cg-edge-labels text').attr('display', function(d) {
|
||||
return active.has(d.dominantCategory) ? null : 'none';
|
||||
});
|
||||
},
|
||||
|
||||
_applySearch: function(term) {
|
||||
this._searchTerm = (term || '').toLowerCase().trim();
|
||||
if (!this._g || !this._clusterData) return;
|
||||
|
||||
if (!this._searchTerm) {
|
||||
// Reset all nodes
|
||||
this._g.selectAll('.cg-country-node').select('.cg-country-circle')
|
||||
.attr('opacity', 0.9).attr('stroke-width', 2).attr('stroke', '#e2e8f0');
|
||||
this._g.selectAll('.cg-links line').attr('stroke-opacity', 0.6);
|
||||
return;
|
||||
}
|
||||
|
||||
// Find which countries contain matching entities
|
||||
var matchingCountries = new Set();
|
||||
var self = this;
|
||||
this._allEntities.forEach(function(e) {
|
||||
var text = (e.name || '') + ' ' + (e.description || '');
|
||||
if (e.aliases) text += ' ' + e.aliases.join(' ');
|
||||
if (text.toLowerCase().indexOf(self._searchTerm) !== -1) {
|
||||
var country = self._clusterData.entityToCountry.get(e.id);
|
||||
if (country) matchingCountries.add(country);
|
||||
}
|
||||
});
|
||||
|
||||
// Highlight matching country nodes
|
||||
this._g.selectAll('.cg-country-node').select('.cg-country-circle')
|
||||
.attr('opacity', function(d) {
|
||||
return matchingCountries.has(d.canonicalName) ? 1 : 0.15;
|
||||
})
|
||||
.attr('stroke', function(d) {
|
||||
return matchingCountries.has(d.canonicalName) ? '#FBBF24' : '#e2e8f0';
|
||||
})
|
||||
.attr('stroke-width', function(d) {
|
||||
return matchingCountries.has(d.canonicalName) ? 4 : 2;
|
||||
});
|
||||
|
||||
this._g.selectAll('.cg-links line').attr('stroke-opacity', 0.15);
|
||||
},
|
||||
|
||||
toggleCategory: function(cat) {
|
||||
if (!this._activeCategories) this._initFilters();
|
||||
if (this._activeCategories.has(cat)) {
|
||||
this._activeCategories.delete(cat);
|
||||
} else {
|
||||
this._activeCategories.add(cat);
|
||||
}
|
||||
this._applyEdgeFilter();
|
||||
// Update button inline styles
|
||||
var btn = document.querySelector('.cg-cat-btn[data-cat="' + cat + '"]');
|
||||
if (btn) {
|
||||
var isActive = this._activeCategories.has(cat);
|
||||
var color = this._categoryColors[cat] || '#6B7280';
|
||||
btn.style.border = '1px solid ' + (isActive ? color : '#334155');
|
||||
btn.style.background = isActive ? color + '22' : 'transparent';
|
||||
btn.style.color = isActive ? color : '#64748b';
|
||||
}
|
||||
},
|
||||
|
||||
// ---- Sidebar: Country List -----------------------------------------------
|
||||
|
||||
_renderCountrySidebar: function() {
|
||||
var panel = document.getElementById('network-detail-panel');
|
||||
if (!panel) return;
|
||||
var self = this;
|
||||
|
||||
if (!this._activeCategories) this._initFilters();
|
||||
|
||||
if (this._currentLevel === 'overview') {
|
||||
var countries = this._clusterData.countries.filter(function(c) {
|
||||
return c.entityCount >= 10 && !c.isUnassigned;
|
||||
});
|
||||
|
||||
var html = '';
|
||||
|
||||
// Search
|
||||
html += '<div style="margin-bottom:10px;">';
|
||||
html += '<input type="text" id="cg-search" placeholder="Entität suchen..." ' +
|
||||
'style="width:100%;padding:7px 10px;background:#1e293b;border:1px solid #334155;' +
|
||||
'border-radius:4px;color:#e2e8f0;font-size:12px;outline:none;box-sizing:border-box;" ' +
|
||||
'oninput="ClusterGraph._applySearch(this.value)">';
|
||||
html += '</div>';
|
||||
|
||||
// Category filter
|
||||
html += '<div style="margin-bottom:12px;">';
|
||||
html += '<div style="color:#94a3b8;font-size:11px;font-weight:600;margin-bottom:6px;text-transform:uppercase;">Beziehungsfilter</div>';
|
||||
html += '<div style="display:flex;flex-wrap:wrap;gap:4px;">';
|
||||
var cats = ['conflict', 'alliance', 'diplomacy', 'economic', 'neutral', 'legal'];
|
||||
for (var fi = 0; fi < cats.length; fi++) {
|
||||
var cat = cats[fi];
|
||||
var color = self._categoryColors[cat];
|
||||
var label = self._categoryLabels[cat];
|
||||
var isActive = self._activeCategories.has(cat);
|
||||
html += '<button class="cg-cat-btn' + (isActive ? ' active' : '') + '" data-cat="' + cat + '" ' +
|
||||
'onclick="ClusterGraph.toggleCategory(\'' + cat + '\')" ' +
|
||||
'style="display:inline-flex;align-items:center;gap:4px;padding:3px 8px;' +
|
||||
'border-radius:4px;border:1px solid ' + (isActive ? color : '#334155') + ';' +
|
||||
'background:' + (isActive ? color + '22' : 'transparent') + ';' +
|
||||
'color:' + (isActive ? color : '#64748b') + ';font-size:11px;cursor:pointer;' +
|
||||
'font-family:inherit;transition:all 0.15s;">' +
|
||||
'<span style="font-size:14px;">\u25CF</span>' + label + '</button>';
|
||||
}
|
||||
html += '</div></div>';
|
||||
|
||||
// Summary
|
||||
html += '<div style="margin-bottom:8px;">';
|
||||
html += '<h3 style="margin:0 0 4px 0;color:#f1f5f9;font-size:14px;">' +
|
||||
countries.length + ' Akteure</h3>';
|
||||
var unassigned = this._clusterData.countries.find(function(c) { return c.isUnassigned; });
|
||||
if (unassigned && unassigned.entityCount > 0) {
|
||||
html += '<div style="color:#64748b;font-size:11px;">' +
|
||||
unassigned.entityCount + ' ohne Zuordnung</div>';
|
||||
}
|
||||
html += '</div>';
|
||||
|
||||
// Top edges
|
||||
var topEdges = this._clusterData.edges.slice(0, 6);
|
||||
if (topEdges.length > 0) {
|
||||
html += '<div style="border-bottom:1px solid #1e293b;padding-bottom:8px;margin-bottom:8px;">';
|
||||
html += '<div style="color:#94a3b8;font-size:11px;font-weight:600;margin-bottom:6px;text-transform:uppercase;">Top-Beziehungen</div>';
|
||||
for (var ei = 0; ei < topEdges.length; ei++) {
|
||||
var edge = topEdges[ei];
|
||||
var eColor = self._categoryColors[edge.dominantCategory] || '#6B7280';
|
||||
html += '<div style="display:flex;align-items:center;gap:5px;padding:2px 0;font-size:11px;">';
|
||||
html += '<span style="color:' + eColor + ';font-size:14px;line-height:1;">\u25CF</span>';
|
||||
html += '<span style="color:#e2e8f0;flex:1;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;">' +
|
||||
self._esc(edge.source) + ' \u2194 ' + self._esc(edge.target) + '</span>';
|
||||
html += '<span style="color:#64748b;">' + edge.count + '</span>';
|
||||
html += '</div>';
|
||||
}
|
||||
html += '</div>';
|
||||
}
|
||||
|
||||
// Country list
|
||||
html += '<div style="color:#94a3b8;font-size:11px;font-weight:600;margin-bottom:6px;text-transform:uppercase;">Akteure</div>';
|
||||
for (var ci = 0; ci < countries.length; ci++) {
|
||||
var c = countries[ci];
|
||||
html += '<div style="display:flex;align-items:center;gap:6px;' +
|
||||
'padding:4px 6px;border-radius:4px;cursor:pointer;margin-bottom:1px;" ' +
|
||||
'onmouseover="this.style.background=\'rgba(51,65,85,0.5)\'" ' +
|
||||
'onmouseout="this.style.background=\'transparent\'" ' +
|
||||
'onclick="ClusterGraph._drillDown(\'' + self._esc(c.canonicalName) + '\')">';
|
||||
html += '<span style="color:#f1f5f9;font-size:12px;flex:1;">' + self._esc(c.name) + '</span>';
|
||||
html += '<span style="color:#64748b;font-size:11px;">' + c.entityCount + '</span>';
|
||||
html += '</div>';
|
||||
}
|
||||
|
||||
panel.innerHTML = html;
|
||||
panel.style.display = 'block';
|
||||
} else if (this._currentLevel === 'country') {
|
||||
// Show type legend for detail view
|
||||
var countryData = null;
|
||||
for (var fi = 0; fi < this._clusterData.countries.length; fi++) {
|
||||
if (this._clusterData.countries[fi].canonicalName === this._currentCountry) {
|
||||
countryData = this._clusterData.countries[fi]; break;
|
||||
}
|
||||
}
|
||||
|
||||
var html2 = '';
|
||||
if (countryData) {
|
||||
html2 += '<h3 style="margin:0 0 8px 0;color:#f1f5f9;font-size:14px;">' +
|
||||
self._esc(countryData.name) + '</h3>';
|
||||
html2 += '<div style="color:#94a3b8;font-size:12px;margin-bottom:12px;">' +
|
||||
countryData.entityCount + ' Entitäten</div>';
|
||||
|
||||
var tc = countryData.typeCounts;
|
||||
var types = ['person', 'organisation', 'military', 'event', 'location'];
|
||||
html2 += '<div style="border-bottom:1px solid #1e293b;padding-bottom:8px;margin-bottom:8px;">';
|
||||
for (var ti = 0; ti < types.length; ti++) {
|
||||
var t = types[ti];
|
||||
var cnt = tc[t] || 0;
|
||||
if (cnt === 0) continue;
|
||||
var tColor = self._entityTypeColors[t];
|
||||
html2 += '<div style="display:flex;align-items:center;gap:6px;padding:2px 0;font-size:12px;">';
|
||||
html2 += '<span style="display:inline-block;width:8px;height:8px;border-radius:50%;background:' + tColor + ';"></span>';
|
||||
html2 += '<span style="color:#cbd5e1;">' + (self._typeLabels[t] || t) + '</span>';
|
||||
html2 += '<span style="color:#64748b;margin-left:auto;">' + cnt + '</span>';
|
||||
html2 += '</div>';
|
||||
}
|
||||
html2 += '</div>';
|
||||
|
||||
// Top entities
|
||||
if (countryData.topEntities && countryData.topEntities.length > 0) {
|
||||
html2 += '<div style="color:#94a3b8;font-size:11px;font-weight:600;margin-bottom:6px;text-transform:uppercase;">Top-Akteure</div>';
|
||||
for (var tei = 0; tei < countryData.topEntities.length; tei++) {
|
||||
var te = countryData.topEntities[tei];
|
||||
var teColor = self._entityTypeColors[te.entity_type] || '#94a3b8';
|
||||
html2 += '<div style="padding:3px 0;font-size:12px;">';
|
||||
html2 += '<span style="color:' + teColor + ';">\u25CF</span> ';
|
||||
html2 += '<span style="color:#e2e8f0;">' + self._esc(te.name) + '</span>';
|
||||
html2 += '</div>';
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
html2 += '<div style="margin-top:16px;padding-top:8px;border-top:1px solid #1e293b;color:#64748b;font-size:12px;">Klicke auf einen Knoten für Details.</div>';
|
||||
panel.innerHTML = html2;
|
||||
panel.style.display = 'block';
|
||||
}
|
||||
},
|
||||
|
||||
// ---- Detail panel for entity click ---------------------------------------
|
||||
|
||||
_updateDetailPanel: function(entity) {
|
||||
if (typeof NetworkGraph !== 'undefined' && NetworkGraph._updateDetailPanel) {
|
||||
var tempData = NetworkGraph._data;
|
||||
NetworkGraph._data = { entities: this._allEntities, relations: this._allRelations };
|
||||
NetworkGraph._updateDetailPanel(entity);
|
||||
NetworkGraph._data = tempData;
|
||||
return;
|
||||
}
|
||||
var panel = document.getElementById('network-detail-panel');
|
||||
if (!panel) return;
|
||||
var typeColor = this._entityTypeColors[entity.entity_type] || '#94A3B8';
|
||||
var html = '<h3 style="margin:0 0 6px 0;color:#f1f5f9;font-size:16px;">' + this._esc(entity.name) + '</h3>';
|
||||
html += '<span style="display:inline-block;background:' + typeColor + ';color:#0f172a;padding:2px 8px;border-radius:4px;font-size:11px;font-weight:600;">' + this._esc(entity.entity_type) + '</span>';
|
||||
if (entity.description) html += '<p style="color:#94a3b8;font-size:13px;margin:10px 0;">' + this._esc(entity.description) + '</p>';
|
||||
html += '<div style="color:#94a3b8;font-size:12px;">Verbindungen: <strong style="color:#f1f5f9;">' + (entity._connections || 0) + '</strong></div>';
|
||||
panel.innerHTML = html;
|
||||
},
|
||||
|
||||
_clearDetailPanel: function() {
|
||||
this._renderCountrySidebar();
|
||||
},
|
||||
|
||||
// ---- Navigation ----------------------------------------------------------
|
||||
|
||||
goBack: function() {
|
||||
var self = this;
|
||||
if (this._currentLevel !== 'country') return;
|
||||
this._currentLevel = 'overview';
|
||||
this._currentCountry = null;
|
||||
this._updateBreadcrumb();
|
||||
|
||||
this._g.transition().duration(300).style('opacity', 0)
|
||||
.on('end', function() {
|
||||
if (self._simulation) self._simulation.stop();
|
||||
self._g.selectAll('*').remove();
|
||||
self._clusterData = ClusterData.buildClusterData(self._allEntities, self._allRelations);
|
||||
self._renderOverview();
|
||||
self._renderCountrySidebar();
|
||||
self._g.style('opacity', 0).transition().duration(300).style('opacity', 1);
|
||||
});
|
||||
this._svg.transition().duration(300).call(this._zoom.transform, d3.zoomIdentity);
|
||||
},
|
||||
|
||||
_updateBreadcrumb: function() {
|
||||
var container = document.getElementById('cluster-breadcrumb');
|
||||
if (!container) return;
|
||||
var self = this;
|
||||
container.innerHTML = '';
|
||||
container.style.display = 'flex';
|
||||
|
||||
if (this._currentLevel === 'country') {
|
||||
var backBtn = document.createElement('button');
|
||||
backBtn.className = 'cluster-back-btn';
|
||||
backBtn.innerHTML = '\u2190 Zurück';
|
||||
backBtn.onclick = function() { self.goBack(); };
|
||||
container.appendChild(backBtn);
|
||||
|
||||
var sep = document.createElement('span');
|
||||
sep.className = 'breadcrumb-separator';
|
||||
sep.textContent = ' / ';
|
||||
container.appendChild(sep);
|
||||
}
|
||||
|
||||
var overviewSpan = document.createElement('span');
|
||||
overviewSpan.textContent = 'Länder-Übersicht';
|
||||
overviewSpan.className = 'breadcrumb-item' + (this._currentLevel === 'overview' ? ' active' : ' clickable');
|
||||
if (this._currentLevel !== 'overview') overviewSpan.onclick = function() { self.goBack(); };
|
||||
container.appendChild(overviewSpan);
|
||||
|
||||
if (this._currentCountry) {
|
||||
var sep2 = document.createElement('span');
|
||||
sep2.className = 'breadcrumb-separator';
|
||||
sep2.textContent = ' \u203A ';
|
||||
container.appendChild(sep2);
|
||||
|
||||
var cd = null;
|
||||
for (var i = 0; i < this._clusterData.countries.length; i++) {
|
||||
if (this._clusterData.countries[i].canonicalName === this._currentCountry) { cd = this._clusterData.countries[i]; break; }
|
||||
}
|
||||
var cs = document.createElement('span');
|
||||
cs.className = 'breadcrumb-item active';
|
||||
cs.textContent = this._currentCountry + (cd ? ' (' + cd.entityCount + ')' : '');
|
||||
container.appendChild(cs);
|
||||
}
|
||||
},
|
||||
|
||||
// ---- Visual helpers -------------------------------------------------------
|
||||
|
||||
_getCountryFill: function(d) {
|
||||
// Subtle gradient based on dominant relationship
|
||||
var edges = this._clusterData.edges;
|
||||
var catCounts = {};
|
||||
for (var i = 0; i < edges.length; i++) {
|
||||
var e = edges[i];
|
||||
if (e.source === d.canonicalName || e.target === d.canonicalName) {
|
||||
for (var cat in e.categories) catCounts[cat] = (catCounts[cat] || 0) + e.categories[cat];
|
||||
}
|
||||
}
|
||||
var bestCat = 'neutral', bestCount = 0;
|
||||
for (var c in catCounts) { if (catCounts[c] > bestCount) { bestCat = c; bestCount = catCounts[c]; } }
|
||||
return this._darken(this._categoryColors[bestCat] || '#6B7280', 0.45);
|
||||
},
|
||||
|
||||
_renderMiniDonut: function(gSel, d) {
|
||||
var types = ['person', 'organisation', 'military', 'event', 'location'];
|
||||
var counts = [], colors = [];
|
||||
for (var i = 0; i < types.length; i++) {
|
||||
var c = d.typeCounts[types[i]] || 0;
|
||||
if (c > 0) { counts.push(c); colors.push(this._entityTypeColors[types[i]]); }
|
||||
}
|
||||
if (counts.length === 0) return;
|
||||
var outerR = d._radius + 5, innerR = d._radius + 1;
|
||||
var arc = d3.arc().innerRadius(innerR).outerRadius(outerR);
|
||||
var pie = d3.pie().sort(null).value(function(v) { return v; });
|
||||
var arcs = pie(counts);
|
||||
for (var ai = 0; ai < arcs.length; ai++) {
|
||||
gSel.append('path').attr('d', arc(arcs[ai])).attr('fill', colors[ai])
|
||||
.attr('opacity', 0.8).attr('pointer-events', 'none');
|
||||
}
|
||||
},
|
||||
|
||||
_createDefs: function() {
|
||||
var defs = this._svg.append('defs');
|
||||
var filter = defs.append('filter')
|
||||
.attr('id', 'cg-glow').attr('x', '-50%').attr('y', '-50%')
|
||||
.attr('width', '200%').attr('height', '200%');
|
||||
filter.append('feGaussianBlur').attr('in', 'SourceGraphic').attr('stdDeviation', 6).attr('result', 'blur');
|
||||
filter.append('feColorMatrix').attr('in', 'blur').attr('type', 'matrix')
|
||||
.attr('values', '0 0 0 0 0.24 0 0 0 0 0.51 0 0 0 0 0.96 0 0 0 0.5 0').attr('result', 'glow');
|
||||
var merge = filter.append('feMerge');
|
||||
merge.append('feMergeNode').attr('in', 'glow');
|
||||
merge.append('feMergeNode').attr('in', 'SourceGraphic');
|
||||
},
|
||||
|
||||
_drag: function() {
|
||||
var self = this;
|
||||
return d3.drag()
|
||||
.on('start', function(event, d) {
|
||||
if (!event.active && self._simulation) self._simulation.alphaTarget(0.3).restart();
|
||||
d.fx = d.x; d.fy = d.y;
|
||||
})
|
||||
.on('drag', function(event, d) { d.fx = event.x; d.fy = event.y; })
|
||||
.on('end', function(event, d) {
|
||||
if (!event.active && self._simulation) self._simulation.alphaTarget(0);
|
||||
d.fx = null; d.fy = null;
|
||||
});
|
||||
},
|
||||
|
||||
_showTooltip: function(event, html) {
|
||||
if (!this._tooltip) return;
|
||||
this._tooltip.style('display', 'block').html(html);
|
||||
this._moveTooltip(event);
|
||||
},
|
||||
_moveTooltip: function(event) {
|
||||
if (!this._tooltip) return;
|
||||
this._tooltip.style('left', (event.offsetX + 16) + 'px').style('top', (event.offsetY - 10) + 'px');
|
||||
},
|
||||
_hideTooltip: function() {
|
||||
if (this._tooltip) this._tooltip.style('display', 'none');
|
||||
},
|
||||
|
||||
_esc: function(str) {
|
||||
if (!str) return '';
|
||||
var div = document.createElement('div');
|
||||
div.appendChild(document.createTextNode(str));
|
||||
return div.innerHTML;
|
||||
},
|
||||
|
||||
_darken: function(hex, amount) {
|
||||
var r = parseInt(hex.slice(1, 3), 16);
|
||||
var g = parseInt(hex.slice(3, 5), 16);
|
||||
var b = parseInt(hex.slice(5, 7), 16);
|
||||
r = Math.round(r * (1 - amount));
|
||||
g = Math.round(g * (1 - amount));
|
||||
b = Math.round(b * (1 - amount));
|
||||
return '#' + ((1 << 24) + (r << 16) + (g << 8) + b).toString(16).slice(1);
|
||||
}
|
||||
};
|
||||
Datei-Diff unterdrückt, da er zu groß ist
Diff laden
In neuem Issue referenzieren
Einen Benutzer sperren