mirror of
https://github.com/overcuriousity/autopsy-flatpak.git
synced 2025-07-18 18:47:43 +00:00
Merge pull request #3450 from millmanorama/930-use_getRelationshipsCount_to_increase_performance
930 use get relationships count to increase performance
This commit is contained in:
commit
14a55af19c
@ -24,19 +24,22 @@ VisualizationPanel.jButton6.text=Hierarchy
|
|||||||
VisualizationPanel.jButton7.text=Circle
|
VisualizationPanel.jButton7.text=Circle
|
||||||
VisualizationPanel.jButton8.text=Organic
|
VisualizationPanel.jButton8.text=Organic
|
||||||
VisualizationPanel.fitGraphButton.text=
|
VisualizationPanel.fitGraphButton.text=
|
||||||
VisualizationPanel.jTextArea1.text=Right-click an account in the Browse Accounts table, and select 'Visualize' to begin.
|
|
||||||
VisualizationPanel.jLabel1.text=Layouts:
|
|
||||||
VisualizationPanel.fitZoomButton.text=
|
|
||||||
VisualizationPanel.zoomActualButton.text=
|
|
||||||
VisualizationPanel.zoomInButton.text=
|
|
||||||
VisualizationPanel.zoomOutButton.text=
|
VisualizationPanel.zoomOutButton.text=
|
||||||
|
# To change this license header, choose License Headers in Project Properties.
|
||||||
|
# To change this template file, choose Tools | Templates
|
||||||
|
# and open the template in the editor.
|
||||||
VisualizationPanel.circleLayoutButton.text=Circle
|
VisualizationPanel.circleLayoutButton.text=Circle
|
||||||
|
VisualizationPanel.organicLayoutButton.text=Organic
|
||||||
VisualizationPanel.fastOrganicLayoutButton.text=Fast Organic
|
VisualizationPanel.fastOrganicLayoutButton.text=Fast Organic
|
||||||
VisualizationPanel.hierarchyLayoutButton.text=Hierarchy
|
VisualizationPanel.hierarchyLayoutButton.text=Hierarchy
|
||||||
VisualizationPanel.jLabel2.text=Zoom:
|
|
||||||
VisualizationPanel.zoomLabel.text=100%
|
VisualizationPanel.zoomLabel.text=100%
|
||||||
VisualizationPanel.zoomOutButton.toolTipText=Zoom out
|
VisualizationPanel.jLabel1.text=Layouts:
|
||||||
VisualizationPanel.zoomInButton.toolTipText=Zoom in
|
VisualizationPanel.jLabel2.text=Zoom:
|
||||||
VisualizationPanel.zoomActualButton.toolTipText=reset zoom
|
|
||||||
VisualizationPanel.fitZoomButton.toolTipText=fit visualization
|
VisualizationPanel.fitZoomButton.toolTipText=fit visualization
|
||||||
VisualizationPanel.organicLayoutButton.text=Organic
|
VisualizationPanel.fitZoomButton.text=
|
||||||
|
VisualizationPanel.jTextArea1.text=Right-click an account in the Browse Accounts table, and select 'Visualize' to begin.
|
||||||
|
VisualizationPanel.zoomActualButton.toolTipText=reset zoom
|
||||||
|
VisualizationPanel.zoomActualButton.text=
|
||||||
|
VisualizationPanel.zoomInButton.toolTipText=Zoom in
|
||||||
|
VisualizationPanel.zoomInButton.text=
|
||||||
|
VisualizationPanel.zoomOutButton.toolTipText=Zoom out
|
||||||
|
@ -20,9 +20,9 @@ package org.sleuthkit.autopsy.communications;
|
|||||||
|
|
||||||
import com.github.mustachejava.DefaultMustacheFactory;
|
import com.github.mustachejava.DefaultMustacheFactory;
|
||||||
import com.github.mustachejava.Mustache;
|
import com.github.mustachejava.Mustache;
|
||||||
import com.google.common.collect.ImmutableSet;
|
|
||||||
import com.google.common.collect.Multimap;
|
import com.google.common.collect.Multimap;
|
||||||
import com.google.common.collect.MultimapBuilder;
|
import com.google.common.collect.MultimapBuilder;
|
||||||
|
import com.google.common.eventbus.Subscribe;
|
||||||
import com.mxgraph.model.mxCell;
|
import com.mxgraph.model.mxCell;
|
||||||
import com.mxgraph.model.mxICell;
|
import com.mxgraph.model.mxICell;
|
||||||
import com.mxgraph.util.mxConstants;
|
import com.mxgraph.util.mxConstants;
|
||||||
@ -33,10 +33,7 @@ import java.io.InputStream;
|
|||||||
import java.io.InputStreamReader;
|
import java.io.InputStreamReader;
|
||||||
import java.io.StringWriter;
|
import java.io.StringWriter;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
@ -50,8 +47,13 @@ import org.sleuthkit.datamodel.AccountDeviceInstance;
|
|||||||
import org.sleuthkit.datamodel.CommunicationsFilter;
|
import org.sleuthkit.datamodel.CommunicationsFilter;
|
||||||
import org.sleuthkit.datamodel.CommunicationsManager;
|
import org.sleuthkit.datamodel.CommunicationsManager;
|
||||||
import org.sleuthkit.datamodel.Content;
|
import org.sleuthkit.datamodel.Content;
|
||||||
|
import org.sleuthkit.datamodel.Relationship;
|
||||||
import org.sleuthkit.datamodel.TskCoreException;
|
import org.sleuthkit.datamodel.TskCoreException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Implementation of mxGraph customized for our use in the CVT visualize mode.
|
||||||
|
* Acts as the primary entry point into the JGraphX API.
|
||||||
|
*/
|
||||||
final class CommunicationsGraph extends mxGraph {
|
final class CommunicationsGraph extends mxGraph {
|
||||||
|
|
||||||
private static final Logger logger = Logger.getLogger(CommunicationsGraph.class.getName());
|
private static final Logger logger = Logger.getLogger(CommunicationsGraph.class.getName());
|
||||||
@ -68,12 +70,11 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
labelMustache = new DefaultMustacheFactory().compile(new InputStreamReader(templateStream), "Vertex_Label");
|
labelMustache = new DefaultMustacheFactory().compile(new InputStreamReader(templateStream), "Vertex_Label");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Style sheet for default vertex and edge styles. These are initialized in
|
||||||
|
* the static block below.
|
||||||
|
*/
|
||||||
static final private mxStylesheet mxStylesheet = new mxStylesheet();
|
static final private mxStylesheet mxStylesheet = new mxStylesheet();
|
||||||
private final Set<AccountDeviceInstanceKey> pinnedAccountDevices = new HashSet<>();
|
|
||||||
private final Set<mxCell> lockedVertices = new HashSet<>();
|
|
||||||
|
|
||||||
private final Map<String, mxCell> nodeMap = new HashMap<>();
|
|
||||||
private final Multimap<Content, mxCell> edgeMap = MultimapBuilder.hashKeys().hashSetValues().build();
|
|
||||||
|
|
||||||
static {
|
static {
|
||||||
//initialize defaul vertex properties
|
//initialize defaul vertex properties
|
||||||
@ -88,8 +89,22 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
mxStylesheet.getDefaultEdgeStyle().put(mxConstants.STYLE_STARTARROW, mxConstants.NONE);
|
mxStylesheet.getDefaultEdgeStyle().put(mxConstants.STYLE_STARTARROW, mxConstants.NONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map from type specific account identifier to mxCell(vertex).
|
||||||
|
*/
|
||||||
|
private final Map<String, mxCell> nodeMap = new HashMap<>();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map from relationship source (Content) to mxCell (edge).
|
||||||
|
*/
|
||||||
|
private final Multimap<Content, mxCell> edgeMap = MultimapBuilder.hashKeys().hashSetValues().build();
|
||||||
|
private final LockedVertexModel lockedVertexModel;
|
||||||
|
|
||||||
|
private final PinnedAccountModel pinnedAccountModel;
|
||||||
|
|
||||||
CommunicationsGraph() {
|
CommunicationsGraph() {
|
||||||
super(mxStylesheet);
|
super(mxStylesheet);
|
||||||
|
//set fixed properties of graph.
|
||||||
setAutoSizeCells(true);
|
setAutoSizeCells(true);
|
||||||
setCellsCloneable(false);
|
setCellsCloneable(false);
|
||||||
setDropEnabled(false);
|
setDropEnabled(false);
|
||||||
@ -107,6 +122,33 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
setKeepEdgesInBackground(true);
|
setKeepEdgesInBackground(true);
|
||||||
setResetEdgesOnMove(true);
|
setResetEdgesOnMove(true);
|
||||||
setHtmlLabels(true);
|
setHtmlLabels(true);
|
||||||
|
|
||||||
|
lockedVertexModel = new LockedVertexModel();
|
||||||
|
lockedVertexModel.registerhandler(new EventHandler<LockedVertexModel.VertexLockEvent>() {
|
||||||
|
@Override
|
||||||
|
@Subscribe
|
||||||
|
public void handle(LockedVertexModel.VertexLockEvent event) {
|
||||||
|
if (event.isVertexLocked()) {
|
||||||
|
getView().clear(event.getVertex(), true, true);
|
||||||
|
getView().validate();
|
||||||
|
} else {
|
||||||
|
final mxCellState state = getView().getState(event.getVertex(), true);
|
||||||
|
getView().updateLabel(state);
|
||||||
|
getView().updateLabelBounds(state);
|
||||||
|
getView().updateBoundingBox(state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
pinnedAccountModel = new PinnedAccountModel(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
LockedVertexModel getLockedVertexModel() {
|
||||||
|
return lockedVertexModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
PinnedAccountModel getPinnedAccountModel() {
|
||||||
|
return pinnedAccountModel;
|
||||||
}
|
}
|
||||||
|
|
||||||
void clear() {
|
void clear() {
|
||||||
@ -115,10 +157,6 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
removeCells(getChildVertices(getDefaultParent()));
|
removeCells(getChildVertices(getDefaultParent()));
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean isAccountPinned(AccountDeviceInstanceKey account) {
|
|
||||||
return pinnedAccountDevices.contains(account);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String convertValueToString(Object cell) {
|
public String convertValueToString(Object cell) {
|
||||||
final StringWriter stringWriter = new StringWriter();
|
final StringWriter stringWriter = new StringWriter();
|
||||||
@ -132,9 +170,9 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
scopes.put("size", Math.round(Math.log(adiKey.getMessageCount()) + 5));
|
scopes.put("size", Math.round(Math.log(adiKey.getMessageCount()) + 5));
|
||||||
scopes.put("iconFileName", CommunicationsGraph.class.getResource("/org/sleuthkit/autopsy/communications/images/"
|
scopes.put("iconFileName", CommunicationsGraph.class.getResource("/org/sleuthkit/autopsy/communications/images/"
|
||||||
+ Utils.getIconFileName(adiKey.getAccountDeviceInstance().getAccount().getAccountType())));
|
+ Utils.getIconFileName(adiKey.getAccountDeviceInstance().getAccount().getAccountType())));
|
||||||
scopes.put("pinned", pinnedAccountDevices.contains(adiKey));
|
scopes.put("pinned", pinnedAccountModel.isAccountPinned(adiKey));
|
||||||
scopes.put("MARKER_PIN_URL", MARKER_PIN_URL);
|
scopes.put("MARKER_PIN_URL", MARKER_PIN_URL);
|
||||||
scopes.put("locked", lockedVertices.contains((mxCell) cell));
|
scopes.put("locked", lockedVertexModel.isVertexLocked((mxCell) cell));
|
||||||
scopes.put("LOCK_URL", LOCK_URL);
|
scopes.put("LOCK_URL", LOCK_URL);
|
||||||
|
|
||||||
labelMustache.execute(stringWriter, scopes);
|
labelMustache.execute(stringWriter, scopes);
|
||||||
@ -158,9 +196,9 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
scopes.put("size", 12);// Math.round(Math.log(adiKey.getMessageCount()) + 5));
|
scopes.put("size", 12);// Math.round(Math.log(adiKey.getMessageCount()) + 5));
|
||||||
scopes.put("iconFileName", CommunicationsGraph.class.getResource("/org/sleuthkit/autopsy/communications/images/"
|
scopes.put("iconFileName", CommunicationsGraph.class.getResource("/org/sleuthkit/autopsy/communications/images/"
|
||||||
+ Utils.getIconFileName(adiKey.getAccountDeviceInstance().getAccount().getAccountType())));
|
+ Utils.getIconFileName(adiKey.getAccountDeviceInstance().getAccount().getAccountType())));
|
||||||
scopes.put("pinned", pinnedAccountDevices.contains(adiKey));
|
scopes.put("pinned", pinnedAccountModel.isAccountPinned(adiKey));
|
||||||
scopes.put("MARKER_PIN_URL", MARKER_PIN_URL);
|
scopes.put("MARKER_PIN_URL", MARKER_PIN_URL);
|
||||||
scopes.put("locked", lockedVertices.contains((mxCell) cell));
|
scopes.put("locked", lockedVertexModel.isVertexLocked((mxCell) cell));
|
||||||
scopes.put("LOCK_URL", LOCK_URL);
|
scopes.put("LOCK_URL", LOCK_URL);
|
||||||
|
|
||||||
labelMustache.execute(stringWriter, scopes);
|
labelMustache.execute(stringWriter, scopes);
|
||||||
@ -171,54 +209,6 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Unpin the given accounts from the graph. Pinned accounts will always be
|
|
||||||
* shown regardless of the filter state. Furthermore, accounts with
|
|
||||||
* relationships that pass the filters will also be shown.
|
|
||||||
*
|
|
||||||
* @param accountDeviceInstances The accounts to unpin.
|
|
||||||
*/
|
|
||||||
void unpinAccount(ImmutableSet<AccountDeviceInstanceKey> accountDeviceInstances) {
|
|
||||||
pinnedAccountDevices.removeAll(accountDeviceInstances);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Pin the given accounts to the graph. Pinned accounts will always be shown
|
|
||||||
* regardless of the filter state. Furthermore, accounts with relationships
|
|
||||||
* that pass the filters will also be shown.
|
|
||||||
*
|
|
||||||
* @param accountDeviceInstances The accounts to pin.
|
|
||||||
*/
|
|
||||||
void pinAccount(ImmutableSet<AccountDeviceInstanceKey> accountDeviceInstances) {
|
|
||||||
pinnedAccountDevices.addAll(accountDeviceInstances);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Lock the given vertex so that applying a layout algorithm doesn't move
|
|
||||||
* it. The user can still manually position the vertex.
|
|
||||||
*
|
|
||||||
* @param vertex The vertex to lock.
|
|
||||||
*/
|
|
||||||
void lockVertex(mxCell vertex) {
|
|
||||||
lockedVertices.add(vertex);
|
|
||||||
getView().clear(vertex, true, true);
|
|
||||||
getView().validate();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Lock the given vertex so that applying a layout algorithm can move it.
|
|
||||||
*
|
|
||||||
* @param vertex The vertex to unlock.
|
|
||||||
*/
|
|
||||||
void unlockVertex(mxCell vertex) {
|
|
||||||
lockedVertices.remove(vertex);
|
|
||||||
|
|
||||||
final mxCellState state = getView().getState(vertex, true);
|
|
||||||
getView().updateLabel(state);
|
|
||||||
getView().updateLabelBounds(state);
|
|
||||||
getView().updateBoundingBox(state);
|
|
||||||
}
|
|
||||||
|
|
||||||
SwingWorker<?, ?> rebuild(ProgressIndicator progress, CommunicationsManager commsManager, CommunicationsFilter currentFilter) {
|
SwingWorker<?, ?> rebuild(ProgressIndicator progress, CommunicationsManager commsManager, CommunicationsFilter currentFilter) {
|
||||||
return new RebuildWorker(progress, commsManager, currentFilter);
|
return new RebuildWorker(progress, commsManager, currentFilter);
|
||||||
}
|
}
|
||||||
@ -226,8 +216,8 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
void resetGraph() {
|
void resetGraph() {
|
||||||
clear();
|
clear();
|
||||||
getView().setScale(1);
|
getView().setScale(1);
|
||||||
pinnedAccountDevices.clear();
|
pinnedAccountModel.clear();
|
||||||
lockedVertices.clear();
|
lockedVertexModel.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
private mxCell getOrCreateVertex(AccountDeviceInstanceKey accountDeviceInstanceKey) {
|
private mxCell getOrCreateVertex(AccountDeviceInstanceKey accountDeviceInstanceKey) {
|
||||||
@ -247,49 +237,33 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
size);
|
size);
|
||||||
return newVertex;
|
return newVertex;
|
||||||
});
|
});
|
||||||
final mxCellState state = getView().getState(vertex, true);
|
// final mxCellState state = getView().getState(vertex, true);
|
||||||
|
//
|
||||||
getView().updateLabel(state);
|
// getView().updateLabel(state);
|
||||||
getView().updateLabelBounds(state);
|
// getView().updateLabelBounds(state);
|
||||||
getView().updateBoundingBox(state);
|
// getView().updateBoundingBox(state);
|
||||||
|
|
||||||
return vertex;
|
return vertex;
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
private mxCell addEdge(Collection<Content> relSources, AccountDeviceInstanceKey account1, AccountDeviceInstanceKey account2) {
|
private mxCell addOrUpdateEdge(long relSources, AccountDeviceInstanceKey account1, AccountDeviceInstanceKey account2) {
|
||||||
mxCell vertex1 = getOrCreateVertex(account1);
|
mxCell vertex1 = getOrCreateVertex(account1);
|
||||||
mxCell vertex2 = getOrCreateVertex(account2);
|
mxCell vertex2 = getOrCreateVertex(account2);
|
||||||
Object[] edgesBetween = getEdgesBetween(vertex1, vertex2);
|
Object[] edgesBetween = getEdgesBetween(vertex1, vertex2);
|
||||||
mxCell edge;
|
mxCell edge;
|
||||||
if (edgesBetween.length == 0) {
|
if (edgesBetween.length == 0) {
|
||||||
final String edgeName = vertex1.getId() + " <-> " + vertex2.getId();
|
final String edgeName = vertex1.getId() + " <-> " + vertex2.getId();
|
||||||
final HashSet<Content> hashSet = new HashSet<>(relSources);
|
edge = (mxCell) insertEdge(getDefaultParent(), edgeName, relSources, vertex1, vertex2,
|
||||||
edge = (mxCell) insertEdge(getDefaultParent(), edgeName, hashSet, vertex1, vertex2,
|
"strokeWidth=" + (Math.log(relSources) + 1));
|
||||||
"strokeWidth=" + (Math.log(hashSet.size()) + 1));
|
|
||||||
} else {
|
} else {
|
||||||
edge = (mxCell) edgesBetween[0];
|
edge = (mxCell) edgesBetween[0];
|
||||||
((Collection<Content>) edge.getValue()).addAll(relSources);
|
// ((Collection<Content>) edge.getValue()).addAll(relSources);
|
||||||
edge.setStyle("strokeWidth=" + (Math.log(((Collection) edge.getValue()).size()) + 1));
|
edge.setStyle("strokeWidth=" + (Math.log(relSources) + 1));
|
||||||
}
|
}
|
||||||
return edge;
|
return edge;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Are there any accounts in this graph? If there are no pinned accounts the
|
|
||||||
* graph will be empty.
|
|
||||||
*
|
|
||||||
* @return True if this graph is empty.
|
|
||||||
*/
|
|
||||||
boolean isEmpty() {
|
|
||||||
return pinnedAccountDevices.isEmpty();
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean isVertexLocked(mxCell vertex) {
|
|
||||||
return lockedVertices.contains(vertex);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* SwingWorker that loads the accounts and edges for this graph according to
|
* SwingWorker that loads the accounts and edges for this graph according to
|
||||||
* the pinned accounts and the current filters.
|
* the pinned accounts and the current filters.
|
||||||
@ -310,52 +284,68 @@ final class CommunicationsGraph extends mxGraph {
|
|||||||
@Override
|
@Override
|
||||||
protected Void doInBackground() throws Exception {
|
protected Void doInBackground() throws Exception {
|
||||||
progress.start("Loading accounts");
|
progress.start("Loading accounts");
|
||||||
// progress.switchToDeterminate("Loading accounts", 0,pinnedAccountDevices.size());
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
try {
|
try {
|
||||||
/**
|
/**
|
||||||
* set to keep track of accounts related to pinned accounts
|
* set to keep track of accounts related to pinned accounts
|
||||||
*/
|
*/
|
||||||
Set<AccountDeviceInstanceKey> relatedAccounts = new HashSet<>();
|
Map<Long, AccountDeviceInstanceKey> relatedAccounts = new HashMap<>();
|
||||||
for (AccountDeviceInstanceKey adiKey : pinnedAccountDevices) {
|
for (AccountDeviceInstanceKey adiKey : pinnedAccountModel.getPinnedAccounts()) {
|
||||||
if (isCancelled()) {
|
if (isCancelled()) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
List<AccountDeviceInstance> relatedAccountDeviceInstances =
|
List<AccountDeviceInstance> relatedAccountDeviceInstances =
|
||||||
commsManager.getRelatedAccountDeviceInstances(adiKey.getAccountDeviceInstance(), currentFilter);
|
commsManager.getRelatedAccountDeviceInstances(adiKey.getAccountDeviceInstance(), currentFilter);
|
||||||
relatedAccounts.add(adiKey);
|
relatedAccounts.put(adiKey.getAccountDeviceInstance().getAccount().getAccountID(), adiKey);
|
||||||
|
getOrCreateVertex(adiKey);
|
||||||
|
|
||||||
//get accounts related to pinned account
|
//get accounts related to pinned account
|
||||||
for (AccountDeviceInstance relatedADI : relatedAccountDeviceInstances) {
|
for (AccountDeviceInstance relatedADI : relatedAccountDeviceInstances) {
|
||||||
// handle.progress(1);
|
|
||||||
long adiRelationshipsCount = commsManager.getRelationshipSourcesCount(relatedADI, currentFilter);
|
long adiRelationshipsCount = commsManager.getRelationshipSourcesCount(relatedADI, currentFilter);
|
||||||
final AccountDeviceInstanceKey relatedADIKey = new AccountDeviceInstanceKey(relatedADI, currentFilter, adiRelationshipsCount);
|
final AccountDeviceInstanceKey relatedADIKey = new AccountDeviceInstanceKey(relatedADI, currentFilter, adiRelationshipsCount);
|
||||||
relatedAccounts.add(relatedADIKey); //store related accounts
|
relatedAccounts.put(relatedADI.getAccount().getAccountID(), relatedADIKey); //store related accounts
|
||||||
}
|
}
|
||||||
progress.progress(++i);
|
progress.progress(++i);
|
||||||
}
|
}
|
||||||
|
|
||||||
//for each pair of related accounts add edges if they are related o each other.
|
Set<Long> accountIDs = relatedAccounts.keySet();
|
||||||
// this is O(n^2) in the number of related accounts!!!
|
|
||||||
List<AccountDeviceInstanceKey> relatedAccountsList = new ArrayList<>(relatedAccounts);
|
Map<Relationship.RelationshipKey, Long> relationshipCounts = commsManager.getRelationshipCounts(accountIDs, currentFilter);
|
||||||
progress.switchToDeterminate("", 0, relatedAccountsList.size());
|
|
||||||
for (i = 0; i < relatedAccountsList.size(); i++) {
|
int total = relationshipCounts.size();
|
||||||
AccountDeviceInstanceKey adiKey1 = relatedAccountsList.get(i);
|
int k = 0;
|
||||||
for (int j = i; j < relatedAccountsList.size(); j++) {
|
progress.switchToDeterminate("", 0,total);
|
||||||
if (isCancelled()) {
|
for (Map.Entry<Relationship.RelationshipKey, Long> entry : relationshipCounts.entrySet()) {
|
||||||
break;
|
Long count = entry.getValue();
|
||||||
}
|
Relationship.RelationshipKey relationshipKey = entry.getKey();
|
||||||
AccountDeviceInstanceKey adiKey2 = relatedAccountsList.get(j);
|
AccountDeviceInstanceKey account1 = relatedAccounts.get(relationshipKey.getAccount1ID());
|
||||||
List<Content> relationships = commsManager.getRelationshipSources(
|
AccountDeviceInstanceKey account2 = relatedAccounts.get(relationshipKey.getAccount2ID());
|
||||||
adiKey1.getAccountDeviceInstance(),
|
mxCell addEdge = addOrUpdateEdge(count, account1, account2);
|
||||||
adiKey2.getAccountDeviceInstance(),
|
progress.progress(addEdge.getId(),k++);
|
||||||
currentFilter);
|
|
||||||
if (relationships.size() > 0) {
|
|
||||||
mxCell addEdge = addEdge(relationships, adiKey1, adiKey2);
|
|
||||||
progress.progress(addEdge.getId());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
progress.progress(i);
|
|
||||||
}
|
}
|
||||||
|
// //for each pair of related accounts add edges if they are related o each other.
|
||||||
|
// // this is O(n^2) in the number of related accounts!!!
|
||||||
|
// List<AccountDeviceInstanceKey> relatedAccountsList = new ArrayList<>(relatedAccounts);
|
||||||
|
// progress.switchToDeterminate("", 0, relatedAccountsList.size());
|
||||||
|
//
|
||||||
|
// for (i = 0; i < relatedAccountsList.size(); i++) {
|
||||||
|
// AccountDeviceInstanceKey adiKey1 = relatedAccountsList.get(i);
|
||||||
|
// for (int j = i; j < relatedAccountsList.size(); j++) {
|
||||||
|
// if (isCancelled()) {
|
||||||
|
// break;
|
||||||
|
// }
|
||||||
|
// AccountDeviceInstanceKey adiKey2 = relatedAccountsList.get(j);
|
||||||
|
// List<Content> relationships = commsManager.getRelationshipSources(
|
||||||
|
// adiKey1.getAccountDeviceInstance(),
|
||||||
|
// adiKey2.getAccountDeviceInstance(),
|
||||||
|
// currentFilter);
|
||||||
|
// if (relationships.size() > 0) {
|
||||||
|
// mxCell addEdge = addEdge(relationships, adiKey1, adiKey2);
|
||||||
|
// progress.progress(addEdge.getId());
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// progress.progress(i);
|
||||||
|
// }
|
||||||
} catch (TskCoreException tskCoreException) {
|
} catch (TskCoreException tskCoreException) {
|
||||||
logger.log(Level.SEVERE, "Error", tskCoreException);
|
logger.log(Level.SEVERE, "Error", tskCoreException);
|
||||||
} finally {
|
} finally {
|
||||||
|
@ -0,0 +1,28 @@
|
|||||||
|
/*
|
||||||
|
* Autopsy Forensic Browser
|
||||||
|
*
|
||||||
|
* Copyright 2018 Basis Technology Corp.
|
||||||
|
* Contact: carrier <at> sleuthkit <dot> org
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.sleuthkit.autopsy.communications;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public interface EventHandler<T> {
|
||||||
|
|
||||||
|
void handle(T event);
|
||||||
|
}
|
@ -0,0 +1,98 @@
|
|||||||
|
/*
|
||||||
|
* Autopsy Forensic Browser
|
||||||
|
*
|
||||||
|
* Copyright 2018 Basis Technology Corp.
|
||||||
|
* Contact: carrier <at> sleuthkit <dot> org
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.sleuthkit.autopsy.communications;
|
||||||
|
|
||||||
|
import com.google.common.eventbus.EventBus;
|
||||||
|
import com.mxgraph.model.mxCell;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
class LockedVertexModel {
|
||||||
|
|
||||||
|
void registerhandler(EventHandler<VertexLockEvent> handler) {
|
||||||
|
eventBus.register(handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
void unregisterhandler(EventHandler<VertexLockEvent> handler) {
|
||||||
|
eventBus.unregister(handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
private final EventBus eventBus = new EventBus();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set of mxCells (vertices) that are 'locked'. Locked vertices are not
|
||||||
|
* affected by layout algorithms, but may be repositioned manually by the
|
||||||
|
* user.
|
||||||
|
*/
|
||||||
|
private final Set<mxCell> lockedVertices = new HashSet<>();
|
||||||
|
|
||||||
|
LockedVertexModel() {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lock the given vertex so that applying a layout algorithm doesn't move
|
||||||
|
* it. The user can still manually position the vertex.
|
||||||
|
*
|
||||||
|
* @param vertex The vertex to lock.
|
||||||
|
*/
|
||||||
|
void lockVertex(mxCell vertex) {
|
||||||
|
lockedVertices.add(vertex);
|
||||||
|
eventBus.post(new VertexLockEvent(vertex, true));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lock the given vertex so that applying a layout algorithm can move it.
|
||||||
|
*
|
||||||
|
* @param vertex The vertex to unlock.
|
||||||
|
*/
|
||||||
|
void unlockVertex(mxCell vertex) {
|
||||||
|
lockedVertices.remove(vertex);
|
||||||
|
eventBus.post(new VertexLockEvent(vertex, false));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean isVertexLocked(mxCell vertex) {
|
||||||
|
return lockedVertices.contains(vertex);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void clear() {
|
||||||
|
lockedVertices.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
static class VertexLockEvent {
|
||||||
|
|
||||||
|
private final mxCell vertex;
|
||||||
|
|
||||||
|
public mxCell getVertex() {
|
||||||
|
return vertex;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isVertexLocked() {
|
||||||
|
return locked;
|
||||||
|
}
|
||||||
|
private final boolean locked;
|
||||||
|
|
||||||
|
VertexLockEvent(mxCell vertex, boolean locked) {
|
||||||
|
this.vertex = vertex;
|
||||||
|
this.locked = locked;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,84 @@
|
|||||||
|
/*
|
||||||
|
* Autopsy Forensic Browser
|
||||||
|
*
|
||||||
|
* Copyright 2018 Basis Technology Corp.
|
||||||
|
* Contact: carrier <at> sleuthkit <dot> org
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.sleuthkit.autopsy.communications;
|
||||||
|
|
||||||
|
import com.google.common.collect.ImmutableSet;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
class PinnedAccountModel {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set of AccountDeviceInstanceKeys that are 'Pinned' to this graph. Pinned
|
||||||
|
* accounts are shown regardless of filters, and accounts that are related
|
||||||
|
* to pinned accounts and pass the filters are show. Pinning accounts is the
|
||||||
|
* primary way to populate the graph.
|
||||||
|
*/
|
||||||
|
private final Set<AccountDeviceInstanceKey> pinnedAccountDevices = new HashSet<>();
|
||||||
|
private final CommunicationsGraph graph;
|
||||||
|
|
||||||
|
PinnedAccountModel(CommunicationsGraph graph) {
|
||||||
|
this.graph = graph;
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean isAccountPinned(AccountDeviceInstanceKey account) {
|
||||||
|
return pinnedAccountDevices.contains(account);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unpin the given accounts from the graph. Pinned accounts will always be
|
||||||
|
* shown regardless of the filter state. Furthermore, accounts with
|
||||||
|
* relationships that pass the filters will also be shown.
|
||||||
|
*
|
||||||
|
* @param accountDeviceInstances The accounts to unpin.
|
||||||
|
*/
|
||||||
|
void unpinAccount(ImmutableSet<AccountDeviceInstanceKey> accountDeviceInstances) {
|
||||||
|
pinnedAccountDevices.removeAll(accountDeviceInstances);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pin the given accounts to the graph. Pinned accounts will always be shown
|
||||||
|
* regardless of the filter state. Furthermore, accounts with relationships
|
||||||
|
* that pass the filters will also be shown.
|
||||||
|
*
|
||||||
|
* @param accountDeviceInstances The accounts to pin.
|
||||||
|
*/
|
||||||
|
void pinAccount(ImmutableSet<AccountDeviceInstanceKey> accountDeviceInstances) {
|
||||||
|
pinnedAccountDevices.addAll(accountDeviceInstances);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Are there any accounts in this graph? If there are no pinned accounts the
|
||||||
|
* graph will be empty.
|
||||||
|
*
|
||||||
|
* @return True if this graph is empty.
|
||||||
|
*/
|
||||||
|
boolean isEmpty() {
|
||||||
|
return pinnedAccountDevices.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
void clear() {
|
||||||
|
pinnedAccountDevices.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
Iterable<AccountDeviceInstanceKey> getPinnedAccounts() {
|
||||||
|
return ImmutableSet.copyOf(pinnedAccountDevices);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -53,7 +53,7 @@ import java.util.Arrays;
|
|||||||
import static java.util.Collections.singleton;
|
import static java.util.Collections.singleton;
|
||||||
import java.util.EnumSet;
|
import java.util.EnumSet;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.Set;
|
import java.util.List;
|
||||||
import java.util.concurrent.Future;
|
import java.util.concurrent.Future;
|
||||||
import java.util.logging.Level;
|
import java.util.logging.Level;
|
||||||
import javax.swing.AbstractAction;
|
import javax.swing.AbstractAction;
|
||||||
@ -91,8 +91,9 @@ import org.sleuthkit.datamodel.TskCoreException;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* A panel that goes in the Visualize tab of the Communications Visualization
|
* A panel that goes in the Visualize tab of the Communications Visualization
|
||||||
* Tool. Hosts an JGraphX mxGraphComponent that host the communications network
|
* Tool. Hosts an JGraphX mxGraphComponent that implements the communications
|
||||||
* visualization and a MessageBrowser for viewing details of communications.
|
* network visualization and a MessageBrowser for viewing details of
|
||||||
|
* communications.
|
||||||
*
|
*
|
||||||
* The Lookup provided by getLookup will be proxied by the lookup of the
|
* The Lookup provided by getLookup will be proxied by the lookup of the
|
||||||
* CVTTopComponent when this tab is active allowing for context sensitive
|
* CVTTopComponent when this tab is active allowing for context sensitive
|
||||||
@ -140,10 +141,14 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
|
|
||||||
@ThreadConfined(type = ThreadConfined.ThreadType.AWT)
|
@ThreadConfined(type = ThreadConfined.ThreadType.AWT)
|
||||||
private SwingWorker<?, ?> worker;
|
private SwingWorker<?, ?> worker;
|
||||||
|
private final PinnedAccountModel pinnedAccountModel;
|
||||||
|
private final LockedVertexModel lockedVertexModel;
|
||||||
|
|
||||||
public VisualizationPanel() {
|
public VisualizationPanel() {
|
||||||
initComponents();
|
initComponents();
|
||||||
graph = new CommunicationsGraph();
|
graph = new CommunicationsGraph();
|
||||||
|
pinnedAccountModel = graph.getPinnedAccountModel();
|
||||||
|
lockedVertexModel = graph.getLockedVertexModel();
|
||||||
|
|
||||||
fastOrganicLayout = new mxFastOrganicLayoutImpl(graph);
|
fastOrganicLayout = new mxFastOrganicLayoutImpl(graph);
|
||||||
circleLayout = new mxCircleLayoutImpl(graph);
|
circleLayout = new mxCircleLayoutImpl(graph);
|
||||||
@ -193,22 +198,22 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
final JPopupMenu jPopupMenu = new JPopupMenu();
|
final JPopupMenu jPopupMenu = new JPopupMenu();
|
||||||
final AccountDeviceInstanceKey adiKey = (AccountDeviceInstanceKey) cellAt.getValue();
|
final AccountDeviceInstanceKey adiKey = (AccountDeviceInstanceKey) cellAt.getValue();
|
||||||
|
|
||||||
if (graph.isVertexLocked(cellAt)) {
|
if (lockedVertexModel.isVertexLocked(cellAt)) {
|
||||||
jPopupMenu.add(new JMenuItem(new AbstractAction("UnLock " + cellAt.getId(), unlockIcon) {
|
jPopupMenu.add(new JMenuItem(new AbstractAction("UnLock " + cellAt.getId(), unlockIcon) {
|
||||||
@Override
|
@Override
|
||||||
public void actionPerformed(ActionEvent e) {
|
public void actionPerformed(ActionEvent e) {
|
||||||
graph.unlockVertex(cellAt);
|
lockedVertexModel.unlockVertex(cellAt);
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
} else {
|
} else {
|
||||||
jPopupMenu.add(new JMenuItem(new AbstractAction("Lock " + cellAt.getId(), lockIcon) {
|
jPopupMenu.add(new JMenuItem(new AbstractAction("Lock " + cellAt.getId(), lockIcon) {
|
||||||
@Override
|
@Override
|
||||||
public void actionPerformed(ActionEvent e) {
|
public void actionPerformed(ActionEvent e) {
|
||||||
graph.lockVertex(cellAt);
|
lockedVertexModel.lockVertex(cellAt);
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
if (graph.isAccountPinned(adiKey)) {
|
if (pinnedAccountModel.isAccountPinned(adiKey)) {
|
||||||
jPopupMenu.add(new JMenuItem(new AbstractAction("Unpin " + cellAt.getId(), unpinIcon) {
|
jPopupMenu.add(new JMenuItem(new AbstractAction("Unpin " + cellAt.getId(), unpinIcon) {
|
||||||
@Override
|
@Override
|
||||||
public void actionPerformed(ActionEvent e) {
|
public void actionPerformed(ActionEvent e) {
|
||||||
@ -247,6 +252,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
||||||
public Lookup getLookup() {
|
public Lookup getLookup() {
|
||||||
return proxyLookup;
|
return proxyLookup;
|
||||||
}
|
}
|
||||||
@ -254,7 +260,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
@Subscribe
|
@Subscribe
|
||||||
void handleUnPinEvent(CVTEvents.UnpinAccountsEvent pinEvent) {
|
void handleUnPinEvent(CVTEvents.UnpinAccountsEvent pinEvent) {
|
||||||
graph.getModel().beginUpdate();
|
graph.getModel().beginUpdate();
|
||||||
graph.unpinAccount(pinEvent.getAccountDeviceInstances());
|
pinnedAccountModel.unpinAccount(pinEvent.getAccountDeviceInstances());
|
||||||
graph.clear();
|
graph.clear();
|
||||||
rebuildGraph();
|
rebuildGraph();
|
||||||
// Updates the display
|
// Updates the display
|
||||||
@ -268,7 +274,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
if (pinEvent.isReplace()) {
|
if (pinEvent.isReplace()) {
|
||||||
graph.resetGraph();
|
graph.resetGraph();
|
||||||
}
|
}
|
||||||
graph.pinAccount(pinEvent.getAccountDeviceInstances());
|
pinnedAccountModel.pinAccount(pinEvent.getAccountDeviceInstances());
|
||||||
rebuildGraph();
|
rebuildGraph();
|
||||||
// Updates the display
|
// Updates the display
|
||||||
graph.getModel().endUpdate();
|
graph.getModel().endUpdate();
|
||||||
@ -289,7 +295,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
|
|
||||||
@ThreadConfined(type = ThreadConfined.ThreadType.AWT)
|
@ThreadConfined(type = ThreadConfined.ThreadType.AWT)
|
||||||
private void rebuildGraph() {
|
private void rebuildGraph() {
|
||||||
if (graph.isEmpty()) {
|
if (pinnedAccountModel.isEmpty()) {
|
||||||
borderLayoutPanel.remove(graphComponent);
|
borderLayoutPanel.remove(graphComponent);
|
||||||
borderLayoutPanel.add(placeHolderPanel, BorderLayout.CENTER);
|
borderLayoutPanel.add(placeHolderPanel, BorderLayout.CENTER);
|
||||||
repaint();
|
repaint();
|
||||||
@ -717,11 +723,23 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
HashSet<AccountDeviceInstance> adis = new HashSet<>();
|
HashSet<AccountDeviceInstance> adis = new HashSet<>();
|
||||||
for (mxICell cell : selectedCells) {
|
for (mxICell cell : selectedCells) {
|
||||||
if (cell.isEdge()) {
|
if (cell.isEdge()) {
|
||||||
relationshipSources.addAll((Set<Content>) cell.getValue());
|
mxICell source = (mxICell) graph.getModel().getTerminal(cell, true);
|
||||||
|
AccountDeviceInstanceKey account1 = (AccountDeviceInstanceKey) source.getValue();
|
||||||
|
mxICell target = (mxICell) graph.getModel().getTerminal(cell, false);
|
||||||
|
AccountDeviceInstanceKey account2 = (AccountDeviceInstanceKey) target.getValue();
|
||||||
|
try {
|
||||||
|
final List<Content> relationshipSources1 = commsManager.getRelationshipSources(account1.getAccountDeviceInstance(),
|
||||||
|
account2.getAccountDeviceInstance(),
|
||||||
|
currentFilter);
|
||||||
|
relationshipSources.addAll(relationshipSources1);
|
||||||
|
} catch (TskCoreException tskCoreException) {
|
||||||
|
logger.log(Level.SEVERE, " Error getting relationsips....", tskCoreException);
|
||||||
|
}
|
||||||
} else if (cell.isVertex()) {
|
} else if (cell.isVertex()) {
|
||||||
adis.add(((AccountDeviceInstanceKey) cell.getValue()).getAccountDeviceInstance());
|
adis.add(((AccountDeviceInstanceKey) cell.getValue()).getAccountDeviceInstance());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
rootNode = SelectionNode.createFromAccountsAndRelationships(relationshipSources, adis, currentFilter, commsManager);
|
rootNode = SelectionNode.createFromAccountsAndRelationships(relationshipSources, adis, currentFilter, commsManager);
|
||||||
selectedNodes = new Node[]{rootNode};
|
selectedNodes = new Node[]{rootNode};
|
||||||
}
|
}
|
||||||
@ -743,7 +761,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
@Override
|
@Override
|
||||||
public boolean isVertexIgnored(Object vertex) {
|
public boolean isVertexIgnored(Object vertex) {
|
||||||
return super.isVertexIgnored(vertex)
|
return super.isVertexIgnored(vertex)
|
||||||
|| VisualizationPanel.this.graph.isVertexLocked((mxCell) vertex);
|
|| lockedVertexModel.isVertexLocked((mxCell) vertex);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -766,7 +784,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
@Override
|
@Override
|
||||||
public boolean isVertexIgnored(Object vertex) {
|
public boolean isVertexIgnored(Object vertex) {
|
||||||
return super.isVertexIgnored(vertex)
|
return super.isVertexIgnored(vertex)
|
||||||
|| VisualizationPanel.this.graph.isVertexLocked((mxCell) vertex);
|
|| lockedVertexModel.isVertexLocked((mxCell) vertex);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -789,7 +807,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
@Override
|
@Override
|
||||||
public boolean isVertexIgnored(Object vertex) {
|
public boolean isVertexIgnored(Object vertex) {
|
||||||
return super.isVertexIgnored(vertex)
|
return super.isVertexIgnored(vertex)
|
||||||
|| VisualizationPanel.this.graph.isVertexLocked((mxCell) vertex);
|
|| lockedVertexModel.isVertexLocked((mxCell) vertex);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -811,7 +829,7 @@ final public class VisualizationPanel extends JPanel implements Lookup.Provider
|
|||||||
@Override
|
@Override
|
||||||
public boolean isVertexIgnored(Object vertex) {
|
public boolean isVertexIgnored(Object vertex) {
|
||||||
return super.isVertexIgnored(vertex)
|
return super.isVertexIgnored(vertex)
|
||||||
|| VisualizationPanel.this.graph.isVertexLocked((mxCell) vertex);
|
|| lockedVertexModel.isVertexLocked((mxCell) vertex);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
Loading…
x
Reference in New Issue
Block a user