41
41
import java .awt .Dimension ;
42
42
import java .io .*;
43
43
import java .nio .channels .FileChannel ;
44
- import java .nio .file .Files ;
45
- import java .nio .file .Path ;
46
- import java .nio .file .Paths ;
47
- import java .nio .file .StandardOpenOption ;
44
+ import java .nio .file .*;
48
45
import java .util .*;
49
46
import java .util .concurrent .atomic .AtomicBoolean ;
50
47
import java .util .concurrent .atomic .AtomicInteger ;
48
+ import java .util .zip .ZipEntry ;
49
+ import java .util .zip .ZipOutputStream ;
50
+ import java .util .zip .ZipInputStream ;
51
51
import javax .swing .*;
52
52
import javax .swing .border .Border ;
53
53
import javax .swing .filechooser .FileFilter ;
54
+ import javax .swing .filechooser .FileNameExtensionFilter ;
54
55
import org .netbeans .api .progress .ProgressHandle ;
55
56
import org .netbeans .api .progress .ProgressHandleFactory ;
56
57
import org .openide .ErrorManager ;
66
67
import org .openide .windows .Mode ;
67
68
import org .openide .windows .TopComponent ;
68
69
import org .openide .windows .WindowManager ;
70
+ import java .nio .channels .Channels ;
71
+ import java .nio .channels .ReadableByteChannel ;
72
+
69
73
70
74
/**
71
75
*
@@ -76,17 +80,7 @@ public final class OutlineTopComponent extends TopComponent implements ExplorerM
76
80
public static final String PREFERRED_ID = "OutlineTopComponent" ;
77
81
private static final GraphDocument document = new GraphDocument ();
78
82
private static final int WORK_UNITS = 10000 ;
79
- private static final FileFilter xmlFileFilter = new FileFilter () {
80
- @ Override
81
- public boolean accept (File f ) {
82
- return f .getName ().toLowerCase ().endsWith (".xml" ) || f .isDirectory ();
83
- }
84
-
85
- @ Override
86
- public String getDescription () {
87
- return "Graph files (*.xml)" ;
88
- }
89
- };
83
+ private static final FileFilter graphFileFilter = new FileNameExtensionFilter ("Graph files (*.xml, *.igv)" , "xml" , "igv" );
90
84
private static final Server server = new Server (document , OutlineTopComponent ::loadContext );
91
85
public static OutlineTopComponent instance ;
92
86
private final Set <FolderNode > selectedFolders = new HashSet <>();
@@ -161,8 +155,26 @@ private static void saveGraphDocument(GraphDocument doc, String path) throws IOE
161
155
}
162
156
}
163
157
164
- try (Writer writer = new OutputStreamWriter (new FileOutputStream (path ))) {
165
- Printer .exportGraphDocument (writer , doc , saveContexts );
158
+ if (path .endsWith (".igv" )) {
159
+ File zipFile = new File (path );
160
+ String fileName = zipFile .getName ();
161
+ try (FileOutputStream fos = new FileOutputStream (zipFile );
162
+ ZipOutputStream zos = new ZipOutputStream (fos );
163
+ Writer writer = new OutputStreamWriter (zos )) {
164
+
165
+ // Replace the '.igv' extension with '.xml's
166
+ String zipEntryName = fileName .substring (0 , fileName .length () - 4 ) + ".xml" ;
167
+ ZipEntry zipEntry = new ZipEntry (zipEntryName );
168
+ zos .putNextEntry (zipEntry );
169
+
170
+ Printer .exportGraphDocument (writer , doc , saveContexts );
171
+
172
+ zos .closeEntry ();
173
+ }
174
+ } else {
175
+ try (Writer writer = new OutputStreamWriter (new FileOutputStream (path ))) {
176
+ Printer .exportGraphDocument (writer , doc , saveContexts );
177
+ }
166
178
}
167
179
}
168
180
@@ -358,7 +370,7 @@ public void clearWorkspace() {
358
370
**/
359
371
public void openFile () {
360
372
JFileChooser fc = new JFileChooser (Settings .get ().get (Settings .DIRECTORY , Settings .DIRECTORY_DEFAULT ));
361
- fc .setFileFilter (xmlFileFilter );
373
+ fc .setFileFilter (graphFileFilter );
362
374
if (fc .showOpenDialog (null ) == JFileChooser .APPROVE_OPTION ) {
363
375
clearWorkspace ();
364
376
String path = fc .getSelectedFile ().getAbsolutePath ();
@@ -403,9 +415,22 @@ public void save() {
403
415
}
404
416
405
417
public void saveAs () {
406
- JFileChooser fc = new JFileChooser ();
418
+ JFileChooser fc = new JFileChooser () {
419
+ @ Override
420
+ public void approveSelection () {
421
+ File selectedFile = getSelectedFile ();
422
+ if (selectedFile != null ) {
423
+ String fileName = selectedFile .getName ().toLowerCase ();
424
+ if (!fileName .endsWith (".xml" ) && !fileName .endsWith (".igv" )) {
425
+ JOptionPane .showMessageDialog (this , "Please select a graph file with .xml or .igv extension." , "Invalid File" , JOptionPane .ERROR_MESSAGE );
426
+ return ;
427
+ }
428
+ }
429
+ super .approveSelection ();
430
+ }
431
+ };
407
432
fc .setDialogTitle ("Save As..." );
408
- fc .setFileFilter (xmlFileFilter );
433
+ fc .setFileFilter (graphFileFilter );
409
434
fc .setCurrentDirectory (new File (Settings .get ().get (Settings .DIRECTORY , Settings .DIRECTORY_DEFAULT )));
410
435
if (fc .showSaveDialog (null ) == JFileChooser .APPROVE_OPTION ) {
411
436
String path = fc .getSelectedFile ().getAbsolutePath ();
@@ -432,7 +457,7 @@ public void saveAs() {
432
457
**/
433
458
public void importFromXML () {
434
459
JFileChooser fc = new JFileChooser ();
435
- fc .setFileFilter (xmlFileFilter );
460
+ fc .setFileFilter (graphFileFilter );
436
461
fc .setCurrentDirectory (new File (Settings .get ().get (Settings .DIRECTORY , Settings .DIRECTORY_DEFAULT )));
437
462
fc .setMultiSelectionEnabled (true );
438
463
if (fc .showOpenDialog (null ) == JFileChooser .APPROVE_OPTION ) {
@@ -482,60 +507,83 @@ private static void loadContext(GraphContext context) {
482
507
}
483
508
484
509
/**
485
- * Loads a graph document from the given file path, updating progress via a ProgressHandle .
486
- * Parse the XML file, add the parsed document to the workspace, and load associated contexts if specified .
510
+ * Loads a graph document from the specified path, either as an XML file or from a ZIP archive .
511
+ * If loading the context is requested, it loads the context along with the document .
487
512
*/
488
513
private void loadGraphDocument (String path , boolean loadContext ) throws IOException {
489
514
if (Files .notExists (Path .of (path ))) {
490
515
return ;
491
516
}
492
517
File file = new File (path );
493
- final FileChannel channel ;
494
- final long start ;
495
- try {
496
- channel = FileChannel .open (file .toPath (), StandardOpenOption .READ );
497
- start = channel .size ();
498
- } catch (Exception ex ) {
499
- Exceptions .printStackTrace (ex );
500
- return ;
518
+ if (file .getName ().endsWith (".xml" )) {
519
+ try (FileChannel channel = FileChannel .open (file .toPath (), StandardOpenOption .READ )) {
520
+ loadFile (channel , file , loadContext );
521
+ }
522
+ } else if (file .getName ().endsWith (".igv" )) {
523
+ try (ZipInputStream zis = new ZipInputStream (new FileInputStream (file ))) {
524
+ ZipEntry entry = zis .getNextEntry ();
525
+ if (entry != null && entry .getName ().endsWith (".xml" )) {
526
+ loadFile (Channels .newChannel (zis ), file , loadContext );
527
+ }
528
+ }
501
529
}
530
+ }
502
531
532
+ /**
533
+ * Loads an XML or ZIP document from the provided channel, while monitoring the progress of the operation.
534
+ */
535
+ private void loadFile (ReadableByteChannel channel , File file , boolean loadContext ) throws IOException {
503
536
final ProgressHandle handle = ProgressHandleFactory .createHandle ("Opening file " + file .getName ());
504
537
handle .start (WORK_UNITS );
505
538
506
- ParseMonitor monitor = new ParseMonitor () {
507
- @ Override
508
- public void updateProgress () {
509
- try {
510
- int prog = (int ) (WORK_UNITS * (double ) channel .position () / (double ) start );
511
- handle .progress (prog );
512
- } catch (IOException ignored ) {
539
+ ParseMonitor monitor ;
540
+ if (channel instanceof FileChannel fileChannel ) {
541
+ final long start = fileChannel .size ();
542
+ monitor = new ParseMonitor () {
543
+ @ Override
544
+ public void updateProgress () {
545
+ try {
546
+ int prog = (int ) (WORK_UNITS * (double ) fileChannel .position () / (double ) start );
547
+ handle .progress (prog );
548
+ } catch (IOException ignored ) {}
513
549
}
514
- }
515
550
516
- @ Override
517
- public void setState (String state ) {
518
- updateProgress ();
519
- handle .progress (state );
520
- }
521
- };
551
+ @ Override
552
+ public void setState (String state ) {
553
+ updateProgress ();
554
+ handle .progress (state );
555
+ }
556
+ };
557
+ } else {
558
+ monitor = new ParseMonitor () {
559
+ @ Override
560
+ public void updateProgress () {
561
+ handle .progress ("Processing..." );
562
+ }
563
+
564
+ @ Override
565
+ public void setState (String state ) {
566
+ updateProgress ();
567
+ handle .progress (state );
568
+ }
569
+ };
570
+ }
571
+
522
572
try {
523
- if (file .getName ().endsWith (".xml" )) {
524
- ArrayList <GraphContext > contexts = new ArrayList <>();
525
- final Parser parser = new Parser (channel , monitor , document , loadContext ? contexts ::add : null );
526
- parser .parse ();
527
- SwingUtilities .invokeLater (() -> {
528
- for (Node child : manager .getRootContext ().getChildren ().getNodes (true )) {
529
- // Nodes are lazily created. By expanding and collapsing they are all initialized
530
- ((BeanTreeView ) this .treeView ).expandNode (child );
531
- ((BeanTreeView ) this .treeView ).collapseNode (child );
532
- }
533
- requestActive ();
534
- });
573
+ ArrayList <GraphContext > contexts = new ArrayList <>();
574
+ final Parser parser = new Parser (channel , monitor , document , loadContext ? contexts ::add : null );
575
+ parser .parse ();
576
+ SwingUtilities .invokeLater (() -> {
577
+ for (Node child : manager .getRootContext ().getChildren ().getNodes (true )) {
578
+ // Nodes are lazily created. By expanding and collapsing they are all initialized
579
+ ((BeanTreeView ) this .treeView ).expandNode (child );
580
+ ((BeanTreeView ) this .treeView ).collapseNode (child );
581
+ }
582
+ requestActive ();
535
583
for (GraphContext ctx : contexts ) {
536
584
loadContext (ctx );
537
585
}
538
- }
586
+ });
539
587
} catch (IOException ex ) {
540
588
Exceptions .printStackTrace (ex );
541
589
}
0 commit comments