Switched to using nekohtml as html parser, externalized simmetrics (just copied the sourcefiles before), lots of buxfixes and improvements. Also started using an ant.
86
build.xml
Normal file
@ -0,0 +1,86 @@
|
||||
<?xml version="1.0"?>
|
||||
|
||||
<project name="FileBot" basedir="." default="fatjar">
|
||||
|
||||
<property name="title" value="${ant.project.name}" />
|
||||
<property name="version" value="2.1" />
|
||||
<property name="url" value="http://sourceforge.net/projects/filebot/" />
|
||||
|
||||
<property name="source" location="${basedir}/source" />
|
||||
<property name="build" location="${basedir}/build" />
|
||||
<property name="lib" location="${basedir}/lib" />
|
||||
|
||||
<property name="executable" location="${basedir}/${title}.jar" />
|
||||
|
||||
<property name="lib.nekohtml" value="nekohtml.jar" />
|
||||
<property name="lib.xerces" value="xercesImpl.jar" />
|
||||
<property name="lib.simmetrics" value="simmetrics_jar_v1_6_2_d07_02_07.jar" />
|
||||
|
||||
|
||||
<target name="jar" depends="build">
|
||||
<jar destfile="${executable}" duplicate="fail">
|
||||
<fileset dir="${build}" />
|
||||
|
||||
<manifest>
|
||||
<attribute name="Built-By" value="${user.name}" />
|
||||
<attribute name="Main-Class" value="Main" />
|
||||
<attribute name="Title" value="${title}" />
|
||||
<attribute name="Version" value="${version}" />
|
||||
<attribute name="URL" value="${url}" />
|
||||
</manifest>
|
||||
|
||||
<!-- include libs if fatjar is set -->
|
||||
<zipfileset src="${lib}/${lib.nekohtml}">
|
||||
<include if="fatjar" name="**/*.class" />
|
||||
<include if="fatjar" name="**/*.properties" />
|
||||
</zipfileset>
|
||||
<zipfileset src="${lib}/${lib.xerces}">
|
||||
<include if="fatjar" name="**/*.class" />
|
||||
</zipfileset>
|
||||
<zipfileset src="${lib}/${lib.simmetrics}">
|
||||
<include if="fatjar" name="**/*.class" />
|
||||
<exclude name="**/*Test*" />
|
||||
</zipfileset>
|
||||
</jar>
|
||||
</target>
|
||||
|
||||
|
||||
<target name="fatjar">
|
||||
<antcall target="jar">
|
||||
<param name="fatjar" value="true" />
|
||||
</antcall>
|
||||
</target>
|
||||
|
||||
|
||||
<target name="build" depends="clean">
|
||||
<!-- create build dir -->
|
||||
<mkdir dir="${build}" />
|
||||
|
||||
<!-- compile -->
|
||||
<javac srcdir="${source}" destdir="${build}">
|
||||
<classpath>
|
||||
<fileset dir="${lib}" includes="*.jar" />
|
||||
</classpath>
|
||||
</javac>
|
||||
|
||||
<!-- copy resources -->
|
||||
<copy todir="${build}">
|
||||
<fileset dir="${source}">
|
||||
<include name="**/*.png" />
|
||||
<include name="**/*.gif" />
|
||||
</fileset>
|
||||
</copy>
|
||||
</target>
|
||||
|
||||
|
||||
<target name="clean">
|
||||
<delete dir="${build}" />
|
||||
<delete file="${executable}" verbose="true" />
|
||||
</target>
|
||||
|
||||
|
||||
<target name="run" depends="fatjar">
|
||||
<java jar="${executable}" fork="true" />
|
||||
</target>
|
||||
|
||||
</project>
|
BIN
fw/action.clear.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/action.down.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/action.find.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/action.load.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/action.match.file2name.png
Normal file
After Width: | Height: | Size: 25 KiB |
BIN
fw/action.match.name2file.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/action.rename.png
Normal file
After Width: | Height: | Size: 29 KiB |
BIN
fw/action.save.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/action.up.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/decoration.header.png
Normal file
After Width: | Height: | Size: 46 KiB |
BIN
fw/design.png
Normal file
After Width: | Height: | Size: 117 KiB |
BIN
fw/message.info.png
Normal file
After Width: | Height: | Size: 28 KiB |
BIN
fw/message.warning.png
Normal file
After Width: | Height: | Size: 25 KiB |
BIN
fw/panel.analyze.png
Normal file
After Width: | Height: | Size: 31 KiB |
BIN
fw/panel.create.png
Normal file
After Width: | Height: | Size: 28 KiB |
BIN
fw/panel.list.png
Normal file
After Width: | Height: | Size: 30 KiB |
BIN
fw/panel.rename.png
Normal file
After Width: | Height: | Size: 32 KiB |
BIN
fw/panel.search.png
Normal file
After Width: | Height: | Size: 30 KiB |
BIN
fw/panel.sfv.png
Normal file
After Width: | Height: | Size: 38 KiB |
BIN
fw/search.anidb.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/search.tvdotcom.png
Normal file
After Width: | Height: | Size: 23 KiB |
BIN
fw/search.tvrage.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/status.error.png
Normal file
After Width: | Height: | Size: 30 KiB |
BIN
fw/status.ok.png
Normal file
After Width: | Height: | Size: 34 KiB |
BIN
fw/status.unknown.png
Normal file
After Width: | Height: | Size: 31 KiB |
BIN
fw/status.warning.png
Normal file
After Width: | Height: | Size: 29 KiB |
BIN
fw/tab.close.hover.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/tab.close.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/tree.closed.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/tree.collapse.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
fw/tree.expand.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
fw/tree.leaf.png
Normal file
After Width: | Height: | Size: 23 KiB |
BIN
fw/tree.open.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
fw/window.icon.big.png
Normal file
After Width: | Height: | Size: 27 KiB |
BIN
fw/window.icon.small.png
Normal file
After Width: | Height: | Size: 24 KiB |
BIN
lib/nekohtml.jar
Normal file
BIN
lib/simmetrics_jar_v1_6_2_d07_02_07.jar
Normal file
BIN
lib/xercesImpl.jar
Normal file
27
source/Main.java
Normal file
@ -0,0 +1,27 @@
|
||||
import javax.swing.SwingUtilities;
|
||||
import javax.swing.UIManager;
|
||||
|
||||
import net.sourceforge.filebot.ui.FileBotWindow;
|
||||
|
||||
|
||||
public class Main {
|
||||
|
||||
/**
|
||||
* @param args
|
||||
*/
|
||||
public static void main(String[] args) {
|
||||
try {
|
||||
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
SwingUtilities.invokeLater(new Runnable() {
|
||||
|
||||
public void run() {
|
||||
new FileBotWindow().setVisible(true);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,91 @@
|
||||
|
||||
package net.sourceforge.filebot.resources;
|
||||
|
||||
|
||||
import java.awt.Image;
|
||||
import java.io.IOException;
|
||||
import java.net.URL;
|
||||
import java.util.HashMap;
|
||||
|
||||
import javax.imageio.ImageIO;
|
||||
import javax.swing.ImageIcon;
|
||||
|
||||
|
||||
public class ResourceManager {
|
||||
|
||||
private static HashMap<String, URL> resourceMap = new HashMap<String, URL>();
|
||||
|
||||
static {
|
||||
putResource("window.icon.small", "window.icon.small.png");
|
||||
putResource("window.icon.big", "window.icon.big.png");
|
||||
|
||||
putResource("panel.analyze", "panel.analyze.png");
|
||||
putResource("panel.rename", "panel.rename.png");
|
||||
putResource("panel.create", "panel.create.png");
|
||||
putResource("panel.list", "panel.list.png");
|
||||
putResource("panel.search", "panel.search.png");
|
||||
|
||||
putResource("tree.open", "tree.open.png");
|
||||
putResource("tree.closed", "tree.closed.png");
|
||||
putResource("tree.leaf", "tree.leaf.png");
|
||||
|
||||
putResource("action.rename", "action.rename.png");
|
||||
putResource("action.down", "action.down.png");
|
||||
putResource("action.up", "action.up.png");
|
||||
putResource("action.save", "action.save.png");
|
||||
putResource("action.load", "action.load.png");
|
||||
putResource("action.clear", "action.clear.png");
|
||||
putResource("action.find", "action.find.png");
|
||||
|
||||
putResource("action.match.file2name", "action.match.file2name.png");
|
||||
putResource("action.match.name2file", "action.match.name2file.png");
|
||||
|
||||
putResource("message.info", "message.info.png");
|
||||
putResource("message.warning", "message.warning.png");
|
||||
|
||||
putResource("search.anidb", "search.anidb.png");
|
||||
putResource("search.tvdotcom", "search.tvdotcom.png");
|
||||
putResource("search.tvrage", "search.tvrage.png");
|
||||
|
||||
putResource("tab.close", "tab.close.png");
|
||||
putResource("tab.close.hover", "tab.close.hover.png");
|
||||
putResource("tab.loading", "tab.loading.gif");
|
||||
putResource("tab.history", "action.find.png");
|
||||
|
||||
putResource("loading", "loading.gif");
|
||||
|
||||
putResource("tree.expand", "tree.expand.png");
|
||||
putResource("tree.collapse", "tree.collapse.png");
|
||||
|
||||
putResource("panel.sfv", "panel.sfv.png");
|
||||
putResource("status.error", "status.error.png");
|
||||
putResource("status.ok", "status.ok.png");
|
||||
putResource("status.unknown", "status.unknown.png");
|
||||
putResource("status.warning", "status.warning.png");
|
||||
|
||||
putResource("decoration.header", "decoration.header.png");
|
||||
}
|
||||
|
||||
|
||||
private static void putResource(String name, String file) {
|
||||
resourceMap.put(name, ResourceManager.class.getResource(file));
|
||||
}
|
||||
|
||||
|
||||
public static Image getImage(String name) {
|
||||
if (!resourceMap.containsKey(name))
|
||||
return null;
|
||||
|
||||
try {
|
||||
return ImageIO.read(resourceMap.get(name));
|
||||
} catch (IOException e) {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public static ImageIcon getIcon(String name) {
|
||||
return new ImageIcon(resourceMap.get(name));
|
||||
}
|
||||
|
||||
}
|
BIN
source/net/sourceforge/filebot/resources/action.clear.png
Normal file
After Width: | Height: | Size: 806 B |
BIN
source/net/sourceforge/filebot/resources/action.down.png
Normal file
After Width: | Height: | Size: 572 B |
BIN
source/net/sourceforge/filebot/resources/action.find.png
Normal file
After Width: | Height: | Size: 875 B |
BIN
source/net/sourceforge/filebot/resources/action.load.png
Normal file
After Width: | Height: | Size: 629 B |
After Width: | Height: | Size: 1.3 KiB |
After Width: | Height: | Size: 1.3 KiB |
BIN
source/net/sourceforge/filebot/resources/action.rename.png
Normal file
After Width: | Height: | Size: 1.3 KiB |
BIN
source/net/sourceforge/filebot/resources/action.save.png
Normal file
After Width: | Height: | Size: 758 B |
BIN
source/net/sourceforge/filebot/resources/action.up.png
Normal file
After Width: | Height: | Size: 564 B |
BIN
source/net/sourceforge/filebot/resources/decoration.header.png
Normal file
After Width: | Height: | Size: 11 KiB |
BIN
source/net/sourceforge/filebot/resources/loading.gif
Normal file
After Width: | Height: | Size: 2.6 KiB |
BIN
source/net/sourceforge/filebot/resources/message.info.png
Normal file
After Width: | Height: | Size: 2.2 KiB |
BIN
source/net/sourceforge/filebot/resources/message.warning.png
Normal file
After Width: | Height: | Size: 1.2 KiB |
BIN
source/net/sourceforge/filebot/resources/panel.analyze.png
Normal file
After Width: | Height: | Size: 2.9 KiB |
BIN
source/net/sourceforge/filebot/resources/panel.create.png
Normal file
After Width: | Height: | Size: 2.3 KiB |
BIN
source/net/sourceforge/filebot/resources/panel.list.png
Normal file
After Width: | Height: | Size: 2.7 KiB |
BIN
source/net/sourceforge/filebot/resources/panel.rename.png
Normal file
After Width: | Height: | Size: 3.2 KiB |
BIN
source/net/sourceforge/filebot/resources/panel.search.png
Normal file
After Width: | Height: | Size: 3.1 KiB |
BIN
source/net/sourceforge/filebot/resources/panel.sfv.png
Normal file
After Width: | Height: | Size: 2.4 KiB |
BIN
source/net/sourceforge/filebot/resources/search.anidb.png
Normal file
After Width: | Height: | Size: 1002 B |
BIN
source/net/sourceforge/filebot/resources/search.tvdotcom.png
Normal file
After Width: | Height: | Size: 722 B |
BIN
source/net/sourceforge/filebot/resources/search.tvrage.png
Normal file
After Width: | Height: | Size: 1012 B |
BIN
source/net/sourceforge/filebot/resources/status.error.png
Normal file
After Width: | Height: | Size: 735 B |
BIN
source/net/sourceforge/filebot/resources/status.ok.png
Normal file
After Width: | Height: | Size: 665 B |
BIN
source/net/sourceforge/filebot/resources/status.unknown.png
Normal file
After Width: | Height: | Size: 885 B |
BIN
source/net/sourceforge/filebot/resources/status.warning.png
Normal file
After Width: | Height: | Size: 579 B |
BIN
source/net/sourceforge/filebot/resources/tab.close.hover.png
Normal file
After Width: | Height: | Size: 331 B |
BIN
source/net/sourceforge/filebot/resources/tab.close.png
Normal file
After Width: | Height: | Size: 336 B |
BIN
source/net/sourceforge/filebot/resources/tab.loading.gif
Normal file
After Width: | Height: | Size: 1.7 KiB |
BIN
source/net/sourceforge/filebot/resources/tree.closed.png
Normal file
After Width: | Height: | Size: 899 B |
BIN
source/net/sourceforge/filebot/resources/tree.collapse.png
Normal file
After Width: | Height: | Size: 340 B |
BIN
source/net/sourceforge/filebot/resources/tree.expand.png
Normal file
After Width: | Height: | Size: 390 B |
BIN
source/net/sourceforge/filebot/resources/tree.leaf.png
Normal file
After Width: | Height: | Size: 565 B |
BIN
source/net/sourceforge/filebot/resources/tree.open.png
Normal file
After Width: | Height: | Size: 1011 B |
BIN
source/net/sourceforge/filebot/resources/window.icon.big.png
Normal file
After Width: | Height: | Size: 1.9 KiB |
BIN
source/net/sourceforge/filebot/resources/window.icon.small.png
Normal file
After Width: | Height: | Size: 768 B |
294
source/net/sourceforge/filebot/torrent/BDecoder.java
Normal file
@ -0,0 +1,294 @@
|
||||
/*
|
||||
* BeDecoder.java
|
||||
*
|
||||
* Created on May 30, 2003, 2:44 PM
|
||||
* Copyright (C) 2003, 2004, 2005, 2006 Aelitis, All Rights Reserved.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU General Public License
|
||||
* as published by the Free Software Foundation; either version 2
|
||||
* of the License, or (at your option) any later version.
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
||||
*
|
||||
* AELITIS, SAS au capital de 46,603.30 euros
|
||||
* 8 Allee Lenotre, La Grille Royale, 78600 Le Mesnil le Roi, France.
|
||||
*/
|
||||
|
||||
package net.sourceforge.filebot.torrent;
|
||||
|
||||
|
||||
import java.io.BufferedInputStream;
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.CharBuffer;
|
||||
import java.nio.charset.Charset;
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
|
||||
/**
|
||||
* A set of utility methods to decode a bencoded array of byte into a Map. integer are represented as Long, String as byte[], dictionnaries as Map, and list as List.
|
||||
*
|
||||
* @author TdC_VgA
|
||||
*
|
||||
*/
|
||||
public class BDecoder {
|
||||
|
||||
private boolean recovery_mode;
|
||||
|
||||
private static final Charset BINARY_CHARSET = Charset.forName("ISO-8859-1");
|
||||
|
||||
|
||||
public static Map<?, ?> decode(byte[] data)
|
||||
|
||||
throws IOException {
|
||||
return (new BDecoder().decodeByteArray(data));
|
||||
}
|
||||
|
||||
|
||||
public static Map<?, ?> decode(BufferedInputStream is)
|
||||
|
||||
throws IOException {
|
||||
return (new BDecoder().decodeStream(is));
|
||||
}
|
||||
|
||||
|
||||
public BDecoder() {
|
||||
}
|
||||
|
||||
|
||||
public Map<?, ?> decodeByteArray(byte[] data)
|
||||
|
||||
throws IOException {
|
||||
return (decode(new ByteArrayInputStream(data)));
|
||||
}
|
||||
|
||||
|
||||
public Map<?, ?> decodeStream(BufferedInputStream data)
|
||||
|
||||
throws IOException {
|
||||
Object res = decodeInputStream(data, 0);
|
||||
|
||||
if (res == null)
|
||||
throw (new IOException("BDecoder: zero length file"));
|
||||
else if (!(res instanceof Map))
|
||||
throw (new IOException("BDecoder: top level isn't a Map"));
|
||||
|
||||
return ((Map<?, ?>) res);
|
||||
}
|
||||
|
||||
|
||||
private Map<?, ?> decode(ByteArrayInputStream data)
|
||||
|
||||
throws IOException {
|
||||
Object res = decodeInputStream(data, 0);
|
||||
|
||||
if (res == null)
|
||||
throw (new IOException("BDecoder: zero length file"));
|
||||
else if (!(res instanceof Map))
|
||||
throw (new IOException("BDecoder: top level isn't a Map"));
|
||||
|
||||
return ((Map<?, ?>) res);
|
||||
}
|
||||
|
||||
|
||||
private Object decodeInputStream(InputStream bais, int nesting)
|
||||
|
||||
throws IOException {
|
||||
if (!bais.markSupported())
|
||||
throw new IOException("InputStream must support the mark() method");
|
||||
|
||||
// set a mark
|
||||
bais.mark(Integer.MAX_VALUE);
|
||||
|
||||
// read a byte
|
||||
int tempByte = bais.read();
|
||||
|
||||
// decide what to do
|
||||
switch (tempByte) {
|
||||
case 'd':
|
||||
// create a new dictionary object
|
||||
Map<String, Object> tempMap = new HashMap<String, Object>();
|
||||
|
||||
try {
|
||||
// get the key
|
||||
byte[] tempByteArray = null;
|
||||
|
||||
while ((tempByteArray = (byte[]) decodeInputStream(bais, nesting + 1)) != null) {
|
||||
|
||||
// decode some more
|
||||
|
||||
Object value = decodeInputStream(bais, nesting + 1);
|
||||
|
||||
// add the value to the map
|
||||
|
||||
CharBuffer cb = BINARY_CHARSET.decode(ByteBuffer.wrap(tempByteArray));
|
||||
|
||||
String key = new String(cb.array(), 0, cb.limit());
|
||||
|
||||
tempMap.put(key, value);
|
||||
}
|
||||
|
||||
if (bais.available() < nesting)
|
||||
throw (new IOException("BDecoder: invalid input data, 'e' missing from end of dictionary"));
|
||||
} catch (Throwable e) {
|
||||
|
||||
if (!recovery_mode) {
|
||||
|
||||
if (e instanceof IOException)
|
||||
throw ((IOException) e);
|
||||
|
||||
throw (new IOException(e));
|
||||
}
|
||||
}
|
||||
|
||||
// return the map
|
||||
return tempMap;
|
||||
|
||||
case 'l':
|
||||
// create the list
|
||||
List<Object> tempList = new ArrayList<Object>();
|
||||
|
||||
try {
|
||||
// create the key
|
||||
Object tempElement = null;
|
||||
while ((tempElement = decodeInputStream(bais, nesting + 1)) != null)
|
||||
// add the element
|
||||
tempList.add(tempElement);
|
||||
|
||||
if (bais.available() < nesting)
|
||||
throw (new IOException("BDecoder: invalid input data, 'e' missing from end of list"));
|
||||
} catch (Throwable e) {
|
||||
|
||||
if (!recovery_mode) {
|
||||
|
||||
if (e instanceof IOException)
|
||||
throw ((IOException) e);
|
||||
|
||||
throw (new IOException(e));
|
||||
}
|
||||
}
|
||||
// return the list
|
||||
return tempList;
|
||||
|
||||
case 'e':
|
||||
case -1:
|
||||
return null;
|
||||
|
||||
case 'i':
|
||||
return new Long(getNumberFromStream(bais, 'e'));
|
||||
|
||||
case '0':
|
||||
case '1':
|
||||
case '2':
|
||||
case '3':
|
||||
case '4':
|
||||
case '5':
|
||||
case '6':
|
||||
case '7':
|
||||
case '8':
|
||||
case '9':
|
||||
// move back one
|
||||
bais.reset();
|
||||
// get the string
|
||||
return getByteArrayFromStream(bais);
|
||||
|
||||
default: {
|
||||
|
||||
int rem_len = bais.available();
|
||||
|
||||
if (rem_len > 256)
|
||||
rem_len = 256;
|
||||
|
||||
byte[] rem_data = new byte[rem_len];
|
||||
|
||||
bais.read(rem_data);
|
||||
|
||||
throw (new IOException("BDecoder: unknown command '" + tempByte + ", remainder = " + new String(rem_data)));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private long getNumberFromStream(InputStream bais, char parseChar) throws IOException {
|
||||
int length = 0;
|
||||
|
||||
// place a mark
|
||||
bais.mark(Integer.MAX_VALUE);
|
||||
|
||||
int tempByte = bais.read();
|
||||
while ((tempByte != parseChar) && (tempByte >= 0)) {
|
||||
tempByte = bais.read();
|
||||
length++;
|
||||
}
|
||||
|
||||
// are we at the end of the stream?
|
||||
if (tempByte < 0)
|
||||
return -1;
|
||||
|
||||
// reset the mark
|
||||
bais.reset();
|
||||
|
||||
// get the length
|
||||
byte[] tempArray = new byte[length];
|
||||
int count = 0;
|
||||
int len = 0;
|
||||
|
||||
// get the string
|
||||
while ((count != length) && ((len = bais.read(tempArray, count, length - count)) > 0))
|
||||
count += len;
|
||||
|
||||
// jump ahead in the stream to compensate for the :
|
||||
bais.skip(1);
|
||||
|
||||
// return the value
|
||||
|
||||
CharBuffer cb = BINARY_CHARSET.decode(ByteBuffer.wrap(tempArray));
|
||||
|
||||
String str_value = new String(cb.array(), 0, cb.limit());
|
||||
|
||||
return Long.parseLong(str_value);
|
||||
}
|
||||
|
||||
|
||||
private byte[] getByteArrayFromStream(InputStream bais) throws IOException {
|
||||
int length = (int) getNumberFromStream(bais, ':');
|
||||
|
||||
if (length < 0)
|
||||
return null;
|
||||
|
||||
// note that torrent hashes can be big (consider a 55GB file with 2MB
|
||||
// pieces
|
||||
// this generates a pieces hash of 1/2 meg
|
||||
|
||||
if (length > 8 * 1024 * 1024)
|
||||
throw (new IOException("Byte array length too large (" + length + ")"));
|
||||
|
||||
byte[] tempArray = new byte[length];
|
||||
int count = 0;
|
||||
int len = 0;
|
||||
// get the string
|
||||
while ((count != length) && ((len = bais.read(tempArray, count, length - count)) > 0))
|
||||
count += len;
|
||||
|
||||
if (count != tempArray.length)
|
||||
throw (new IOException("BDecoder::getByteArrayFromStream: truncated"));
|
||||
|
||||
return tempArray;
|
||||
}
|
||||
|
||||
|
||||
public void setRecoveryMode(boolean r) {
|
||||
recovery_mode = r;
|
||||
}
|
||||
}
|
186
source/net/sourceforge/filebot/torrent/Torrent.java
Normal file
@ -0,0 +1,186 @@
|
||||
|
||||
package net.sourceforge.filebot.torrent;
|
||||
|
||||
|
||||
import java.io.BufferedInputStream;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.IOException;
|
||||
import java.nio.charset.Charset;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Iterator;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
|
||||
public class Torrent {
|
||||
|
||||
private String name;
|
||||
private String encoding;
|
||||
private String createdBy;
|
||||
private String announce;
|
||||
private String comment;
|
||||
private Long creationDate;
|
||||
|
||||
private List<Entry> files;
|
||||
|
||||
private boolean singleFileTorrent;
|
||||
|
||||
|
||||
public Torrent(File torrent) throws IOException {
|
||||
FileInputStream in = new FileInputStream(torrent);
|
||||
Map<?, ?> torrentMap = null;
|
||||
|
||||
try {
|
||||
torrentMap = BDecoder.decode(new BufferedInputStream(in));
|
||||
} finally {
|
||||
in.close();
|
||||
}
|
||||
|
||||
Charset charset = Charset.forName("UTF-8");
|
||||
|
||||
encoding = decodeString(torrentMap.get("encoding"), charset);
|
||||
|
||||
try {
|
||||
charset = Charset.forName(encoding);
|
||||
} catch (Exception e) {
|
||||
|
||||
}
|
||||
|
||||
createdBy = decodeString(torrentMap.get("created by"), charset);
|
||||
announce = decodeString(torrentMap.get("announce"), charset);
|
||||
comment = decodeString(torrentMap.get("comment"), charset);
|
||||
creationDate = decodeLong(torrentMap.get("creation date"));
|
||||
|
||||
Map<?, ?> infoMap = (Map<?, ?>) torrentMap.get("info");
|
||||
|
||||
name = decodeString(infoMap.get("name"), charset);
|
||||
|
||||
if (infoMap.containsKey("files")) {
|
||||
singleFileTorrent = false;
|
||||
|
||||
files = new ArrayList<Entry>();
|
||||
|
||||
for (Object fileMapObject : (List<?>) infoMap.get("files")) {
|
||||
Map<?, ?> fileMap = (Map<?, ?>) fileMapObject;
|
||||
List<?> pathList = (List<?>) fileMap.get("path");
|
||||
|
||||
StringBuffer pathBuffer = new StringBuffer();
|
||||
String entryName = null;
|
||||
|
||||
Iterator<?> iterator = pathList.iterator();
|
||||
|
||||
while (iterator.hasNext()) {
|
||||
String pathElement = decodeString(iterator.next(), charset);
|
||||
|
||||
if (iterator.hasNext()) {
|
||||
pathBuffer.append(pathElement);
|
||||
pathBuffer.append("/");
|
||||
} else {
|
||||
// the last element in the path list is the filename
|
||||
entryName = pathElement;
|
||||
}
|
||||
}
|
||||
|
||||
Long length = decodeLong(fileMap.get("length"));
|
||||
|
||||
files.add(new Entry(entryName, length, pathBuffer.toString()));
|
||||
}
|
||||
} else {
|
||||
// single file torrent
|
||||
singleFileTorrent = true;
|
||||
files = new ArrayList<Entry>(1);
|
||||
|
||||
Long length = decodeLong(infoMap.get("length"));
|
||||
|
||||
files.add(new Entry(name, length, ""));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private String decodeString(Object bytearray, Charset charset) {
|
||||
if (bytearray == null)
|
||||
return null;
|
||||
|
||||
return new String((byte[]) bytearray, charset);
|
||||
}
|
||||
|
||||
|
||||
private Long decodeLong(Object number) {
|
||||
if (number == null)
|
||||
return null;
|
||||
|
||||
return (Long) number;
|
||||
}
|
||||
|
||||
|
||||
public String getAnnounce() {
|
||||
return announce;
|
||||
}
|
||||
|
||||
|
||||
public String getComment() {
|
||||
return comment;
|
||||
}
|
||||
|
||||
|
||||
public String getCreatedBy() {
|
||||
return createdBy;
|
||||
}
|
||||
|
||||
|
||||
public Long getCreationDate() {
|
||||
return creationDate;
|
||||
}
|
||||
|
||||
|
||||
public String getEncoding() {
|
||||
return encoding;
|
||||
}
|
||||
|
||||
|
||||
public List<Entry> getFiles() {
|
||||
return files;
|
||||
}
|
||||
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
|
||||
public boolean isSingleFileTorrent() {
|
||||
return singleFileTorrent;
|
||||
}
|
||||
|
||||
|
||||
public static class Entry {
|
||||
|
||||
private String name;
|
||||
private long length;
|
||||
private String path;
|
||||
|
||||
|
||||
public Entry(String name, Long length, String path) {
|
||||
this.name = name;
|
||||
this.length = length;
|
||||
this.path = path;
|
||||
}
|
||||
|
||||
|
||||
public Long getLength() {
|
||||
return length;
|
||||
}
|
||||
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
|
||||
public String getPath() {
|
||||
return path;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
174
source/net/sourceforge/filebot/ui/FileBotList.java
Normal file
@ -0,0 +1,174 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.io.File;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.PrintStream;
|
||||
import java.util.List;
|
||||
|
||||
import javax.swing.AbstractAction;
|
||||
import javax.swing.BorderFactory;
|
||||
import javax.swing.DefaultListModel;
|
||||
import javax.swing.JList;
|
||||
import javax.swing.JPanel;
|
||||
import javax.swing.JScrollPane;
|
||||
import javax.swing.KeyStroke;
|
||||
import javax.swing.ListSelectionModel;
|
||||
import javax.swing.border.TitledBorder;
|
||||
|
||||
import net.sourceforge.filebot.ui.sal.FileTransferable;
|
||||
import net.sourceforge.filebot.ui.sal.Saveable;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.NullTransferablePolicy;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.TransferablePolicy;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.TransferablePolicySupport;
|
||||
import net.sourceforge.tuned.ui.FancyListCellRenderer;
|
||||
|
||||
|
||||
public class FileBotList extends JPanel implements Saveable, TransferablePolicySupport {
|
||||
|
||||
private JList list = new JList(new DefaultListModel());
|
||||
|
||||
private TitledBorder titledBorder;
|
||||
|
||||
private String title;
|
||||
|
||||
private TransferablePolicy transferablePolicy = new NullTransferablePolicy();
|
||||
|
||||
|
||||
public FileBotList(boolean enableDrop, boolean enableDrag, boolean initRemoveAction) {
|
||||
this(enableDrop, enableDrag, initRemoveAction, true);
|
||||
}
|
||||
|
||||
|
||||
public FileBotList(boolean enableDrop, boolean enableDrag, boolean initRemoveAction, boolean border) {
|
||||
super(new BorderLayout());
|
||||
|
||||
JScrollPane listScrollPane = new JScrollPane(list);
|
||||
|
||||
if (border) {
|
||||
titledBorder = new TitledBorder("");
|
||||
setBorder(titledBorder);
|
||||
} else {
|
||||
listScrollPane.setBorder(BorderFactory.createEmptyBorder());
|
||||
}
|
||||
|
||||
list.setCellRenderer(new FancyListCellRenderer());
|
||||
list.setSelectionMode(ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
|
||||
add(listScrollPane, BorderLayout.CENTER);
|
||||
|
||||
TransferablePolicySupport handlerTransferablePolicySupport = null;
|
||||
Saveable handlerSaveable = null;
|
||||
|
||||
if (enableDrop) {
|
||||
handlerTransferablePolicySupport = this;
|
||||
}
|
||||
|
||||
if (enableDrag) {
|
||||
handlerSaveable = this;
|
||||
}
|
||||
|
||||
list.setTransferHandler(new FileBotTransferHandler(handlerTransferablePolicySupport, handlerSaveable));
|
||||
|
||||
if (handlerSaveable != null)
|
||||
MouseDragRecognizeListener.createForComponent(this.getListComponent());
|
||||
|
||||
if (initRemoveAction) {
|
||||
// Shortcut DELETE
|
||||
FileBotUtil.registerActionForKeystroke(this, KeyStroke.getKeyStroke("pressed DELETE"), removeAction);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public JList getListComponent() {
|
||||
return list;
|
||||
}
|
||||
|
||||
|
||||
public void setTransferablePolicy(TransferablePolicy transferablePolicy) {
|
||||
this.transferablePolicy = transferablePolicy;
|
||||
}
|
||||
|
||||
|
||||
public TransferablePolicy getTransferablePolicy() {
|
||||
return transferablePolicy;
|
||||
}
|
||||
|
||||
|
||||
public String getTitle() {
|
||||
return title;
|
||||
}
|
||||
|
||||
|
||||
public void setTitle(String title) {
|
||||
this.title = title;
|
||||
|
||||
if (titledBorder != null)
|
||||
titledBorder.setTitle(title);
|
||||
|
||||
if (isVisible()) {
|
||||
revalidate();
|
||||
repaint();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public DefaultListModel getModel() {
|
||||
return (DefaultListModel) list.getModel();
|
||||
}
|
||||
|
||||
|
||||
public void save(File file) {
|
||||
try {
|
||||
PrintStream out = new PrintStream(file);
|
||||
|
||||
DefaultListModel model = getModel();
|
||||
|
||||
for (int i = 0; i < model.getSize(); ++i)
|
||||
out.println(model.get(i).toString());
|
||||
|
||||
out.close();
|
||||
} catch (FileNotFoundException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public String getDefaultFileName() {
|
||||
return title + ".txt";
|
||||
}
|
||||
|
||||
|
||||
public boolean isSaveable() {
|
||||
return !getModel().isEmpty();
|
||||
}
|
||||
|
||||
|
||||
public void load(List<File> files) {
|
||||
FileTransferable tr = new FileTransferable(files);
|
||||
|
||||
if (transferablePolicy.accept(tr))
|
||||
transferablePolicy.handleTransferable(tr, false);
|
||||
}
|
||||
|
||||
private final AbstractAction removeAction = new AbstractAction("Remove") {
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
int index = list.getSelectedIndex();
|
||||
Object values[] = list.getSelectedValues();
|
||||
|
||||
for (Object value : values)
|
||||
getModel().removeElement(value);
|
||||
|
||||
int maxIndex = list.getModel().getSize() - 1;
|
||||
|
||||
if (index > maxIndex)
|
||||
index = maxIndex;
|
||||
|
||||
list.setSelectedIndex(index);
|
||||
}
|
||||
};
|
||||
|
||||
}
|
40
source/net/sourceforge/filebot/ui/FileBotPanel.java
Normal file
@ -0,0 +1,40 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
|
||||
import javax.swing.Icon;
|
||||
import javax.swing.JPanel;
|
||||
|
||||
|
||||
public class FileBotPanel extends JPanel {
|
||||
|
||||
private String text;
|
||||
|
||||
private Icon icon;
|
||||
|
||||
|
||||
public FileBotPanel(String text, Icon icon) {
|
||||
super(new BorderLayout(10, 20));
|
||||
this.text = text;
|
||||
this.icon = icon;
|
||||
}
|
||||
|
||||
|
||||
public Icon getIcon() {
|
||||
return icon;
|
||||
}
|
||||
|
||||
|
||||
public String getText() {
|
||||
return text;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return getText();
|
||||
}
|
||||
|
||||
}
|
124
source/net/sourceforge/filebot/ui/FileBotPanelSelectionList.java
Normal file
@ -0,0 +1,124 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.Color;
|
||||
import java.awt.Component;
|
||||
import java.awt.dnd.DropTarget;
|
||||
import java.awt.dnd.DropTargetAdapter;
|
||||
import java.awt.dnd.DropTargetDragEvent;
|
||||
import java.awt.dnd.DropTargetDropEvent;
|
||||
import java.awt.dnd.DropTargetEvent;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.awt.event.ActionListener;
|
||||
|
||||
import javax.swing.DefaultListModel;
|
||||
import javax.swing.JList;
|
||||
import javax.swing.ListSelectionModel;
|
||||
import javax.swing.SwingConstants;
|
||||
import javax.swing.Timer;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
|
||||
import net.sourceforge.filebot.ui.panel.analyze.AnalyzePanel;
|
||||
import net.sourceforge.filebot.ui.panel.create.CreatePanel;
|
||||
import net.sourceforge.filebot.ui.panel.list.ListPanel;
|
||||
import net.sourceforge.filebot.ui.panel.rename.RenamePanel;
|
||||
import net.sourceforge.filebot.ui.panel.search.SearchPanel;
|
||||
import net.sourceforge.filebot.ui.panel.sfv.SfvPanel;
|
||||
import net.sourceforge.tuned.ui.FancyListCellRenderer;
|
||||
import net.sourceforge.tuned.ui.GradientStyle;
|
||||
|
||||
|
||||
public class FileBotPanelSelectionList extends JList {
|
||||
|
||||
private static final int SELECTDELAY_ON_DRAG_OVER = 300;
|
||||
|
||||
|
||||
public FileBotPanelSelectionList() {
|
||||
DefaultListModel model = new DefaultListModel();
|
||||
setModel(model);
|
||||
|
||||
setCellRenderer(new PanelCellRenderer());
|
||||
setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
|
||||
|
||||
setBorder(new EmptyBorder(4, 5, 4, 5));
|
||||
|
||||
new DropTarget(this, new DragDropListener());
|
||||
|
||||
model.addElement(new ListPanel());
|
||||
model.addElement(new RenamePanel());
|
||||
model.addElement(new AnalyzePanel());
|
||||
model.addElement(new SearchPanel());
|
||||
model.addElement(new CreatePanel());
|
||||
model.addElement(new SfvPanel());
|
||||
}
|
||||
|
||||
|
||||
private class PanelCellRenderer extends FancyListCellRenderer {
|
||||
|
||||
public PanelCellRenderer() {
|
||||
super(10, Color.decode("#163264"), GradientStyle.TOP_TO_BOTTOM);
|
||||
setVerticalTextPosition(SwingConstants.BOTTOM);
|
||||
setHorizontalTextPosition(SwingConstants.CENTER);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public Component getListCellRendererComponent(JList list, Object value, int index, boolean isSelected, boolean cellHasFocus) {
|
||||
super.getListCellRendererComponent(list, value, index, isSelected, cellHasFocus);
|
||||
|
||||
FileBotPanel panel = (FileBotPanel) value;
|
||||
setText(panel.getText());
|
||||
setIcon(panel.getIcon());
|
||||
|
||||
return this;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private class DragDropListener extends DropTargetAdapter implements ActionListener {
|
||||
|
||||
private boolean selectEnabled = false;
|
||||
|
||||
private Timer timer = new Timer(SELECTDELAY_ON_DRAG_OVER, this);
|
||||
|
||||
|
||||
public DragDropListener() {
|
||||
timer.setRepeats(false);
|
||||
}
|
||||
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
selectEnabled = true;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void dragOver(DropTargetDragEvent dtde) {
|
||||
if (selectEnabled) {
|
||||
int index = locationToIndex(dtde.getLocation());
|
||||
setSelectedIndex(index);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void dragEnter(DropTargetDragEvent dtde) {
|
||||
timer.start();
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void dragExit(DropTargetEvent dte) {
|
||||
timer.stop();
|
||||
selectEnabled = false;
|
||||
}
|
||||
|
||||
|
||||
public void drop(DropTargetDropEvent dtde) {
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
181
source/net/sourceforge/filebot/ui/FileBotTransferHandler.java
Normal file
@ -0,0 +1,181 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.datatransfer.Clipboard;
|
||||
import java.awt.datatransfer.DataFlavor;
|
||||
import java.awt.datatransfer.StringSelection;
|
||||
import java.awt.datatransfer.Transferable;
|
||||
import java.awt.dnd.InvalidDnDOperationException;
|
||||
import java.io.File;
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Iterator;
|
||||
import java.util.List;
|
||||
|
||||
import javax.swing.JComponent;
|
||||
import javax.swing.JList;
|
||||
import javax.swing.JTable;
|
||||
import javax.swing.JTree;
|
||||
import javax.swing.TransferHandler;
|
||||
import javax.swing.tree.TreePath;
|
||||
|
||||
import net.sourceforge.filebot.ui.sal.FileTransferable;
|
||||
import net.sourceforge.filebot.ui.sal.Saveable;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.TransferablePolicySupport;
|
||||
|
||||
|
||||
public class FileBotTransferHandler extends TransferHandler {
|
||||
|
||||
private TransferablePolicySupport transferablePolicySupport;
|
||||
|
||||
private Saveable saveable;
|
||||
|
||||
private boolean dragging;
|
||||
|
||||
private String tmpdir = System.getProperty("java.io.tmpdir");
|
||||
|
||||
|
||||
public FileBotTransferHandler(TransferablePolicySupport transferablePolicySupport, Saveable saveable) {
|
||||
this.transferablePolicySupport = transferablePolicySupport;
|
||||
this.saveable = saveable;
|
||||
}
|
||||
|
||||
private boolean canImportCache = false;
|
||||
|
||||
|
||||
@Override
|
||||
public boolean canImport(TransferSupport support) {
|
||||
// show "drop allowed" mousecursor when dragging even though drop is not allowed
|
||||
if (dragging)
|
||||
return true;
|
||||
|
||||
if (transferablePolicySupport == null)
|
||||
return false;
|
||||
|
||||
if (support.isDrop())
|
||||
support.setShowDropLocation(false);
|
||||
|
||||
Transferable t = support.getTransferable();
|
||||
|
||||
try {
|
||||
canImportCache = transferablePolicySupport.getTransferablePolicy().accept(t);
|
||||
} catch (InvalidDnDOperationException e) {
|
||||
// for some reason the last transferable has no drop current
|
||||
}
|
||||
|
||||
return canImportCache;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public boolean importData(TransferSupport support) {
|
||||
if (dragging)
|
||||
return false;
|
||||
|
||||
if (!canImport(support))
|
||||
return false;
|
||||
|
||||
boolean add = false;
|
||||
|
||||
if (support.isDrop())
|
||||
add = support.getDropAction() == COPY;
|
||||
|
||||
Transferable t = support.getTransferable();
|
||||
|
||||
return transferablePolicySupport.getTransferablePolicy().handleTransferable(t, add);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void exportDone(JComponent source, Transferable data, int action) {
|
||||
dragging = false;
|
||||
|
||||
if (data == null)
|
||||
return;
|
||||
|
||||
try {
|
||||
List<?> list = (List<?>) data.getTransferData(DataFlavor.javaFileListFlavor);
|
||||
|
||||
for (Object object : list) {
|
||||
File temporaryFile = (File) object;
|
||||
temporaryFile.deleteOnExit();
|
||||
}
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public int getSourceActions(JComponent c) {
|
||||
if (saveable == null || !saveable.isSaveable())
|
||||
return NONE;
|
||||
|
||||
return MOVE | COPY;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void exportToClipboard(JComponent comp, Clipboard clip, int action) throws IllegalStateException {
|
||||
ArrayList<String> lines = new ArrayList<String>();
|
||||
|
||||
if (comp instanceof JList) {
|
||||
JList list = (JList) comp;
|
||||
for (Object value : list.getSelectedValues()) {
|
||||
lines.add(value.toString());
|
||||
}
|
||||
} else if (comp instanceof JTree) {
|
||||
JTree tree = (JTree) comp;
|
||||
for (TreePath path : tree.getSelectionPaths()) {
|
||||
lines.add(path.getPathComponent(path.getPathCount() - 1).toString());
|
||||
}
|
||||
} else if (comp instanceof JTable) {
|
||||
JTable table = (JTable) comp;
|
||||
|
||||
for (int row : table.getSelectedRows()) {
|
||||
StringBuffer b = new StringBuffer();
|
||||
int maxCol = table.getColumnCount() - 1;
|
||||
for (int col = 0; col <= maxCol; col++) {
|
||||
b.append(table.getModel().getValueAt(row, col));
|
||||
|
||||
if (col != maxCol)
|
||||
b.append("\t");
|
||||
}
|
||||
|
||||
lines.add(b.toString());
|
||||
}
|
||||
}
|
||||
|
||||
StringBuffer b = new StringBuffer();
|
||||
Iterator<String> it = lines.iterator();
|
||||
|
||||
while (it.hasNext()) {
|
||||
b.append(it.next());
|
||||
|
||||
if (it.hasNext())
|
||||
b.append("\n");
|
||||
}
|
||||
|
||||
clip.setContents(new StringSelection(b.toString()), null);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected Transferable createTransferable(JComponent c) {
|
||||
dragging = true;
|
||||
|
||||
try {
|
||||
File temporaryFile = new File(tmpdir, saveable.getDefaultFileName());
|
||||
temporaryFile.createNewFile();
|
||||
|
||||
saveable.save(temporaryFile);
|
||||
return new FileTransferable(temporaryFile);
|
||||
} catch (IOException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
}
|
236
source/net/sourceforge/filebot/ui/FileBotTree.java
Normal file
@ -0,0 +1,236 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.Desktop;
|
||||
import java.awt.Font;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.awt.event.MouseAdapter;
|
||||
import java.awt.event.MouseEvent;
|
||||
import java.io.File;
|
||||
import java.util.Enumeration;
|
||||
import java.util.LinkedList;
|
||||
|
||||
import javax.swing.AbstractAction;
|
||||
import javax.swing.JMenuItem;
|
||||
import javax.swing.JPopupMenu;
|
||||
import javax.swing.JTree;
|
||||
import javax.swing.SwingUtilities;
|
||||
import javax.swing.tree.DefaultMutableTreeNode;
|
||||
import javax.swing.tree.DefaultTreeModel;
|
||||
import javax.swing.tree.TreeModel;
|
||||
import javax.swing.tree.TreeNode;
|
||||
import javax.swing.tree.TreePath;
|
||||
import javax.swing.tree.TreeSelectionModel;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.NullTransferablePolicy;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.TransferablePolicy;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.TransferablePolicySupport;
|
||||
|
||||
|
||||
public class FileBotTree extends JTree implements TransferablePolicySupport {
|
||||
|
||||
private TransferablePolicy transferablePolicy = new NullTransferablePolicy();
|
||||
|
||||
|
||||
public FileBotTree() {
|
||||
super(new DefaultTreeModel(new DefaultMutableTreeNode()));
|
||||
getSelectionModel().setSelectionMode(TreeSelectionModel.DISCONTIGUOUS_TREE_SELECTION);
|
||||
setCellRenderer(new FileBotTreeCellRenderer());
|
||||
setShowsRootHandles(true);
|
||||
setRootVisible(false);
|
||||
setRowHeight(22);
|
||||
|
||||
setTransferHandler(new FileBotTransferHandler(this, null));
|
||||
|
||||
addMouseListener(new ExpandCollapsePopupListener());
|
||||
}
|
||||
|
||||
|
||||
public void clear() {
|
||||
setModel(new DefaultTreeModel(new DefaultMutableTreeNode()));
|
||||
}
|
||||
|
||||
|
||||
public void setTransferablePolicy(TransferablePolicy transferablePolicy) {
|
||||
this.transferablePolicy = transferablePolicy;
|
||||
}
|
||||
|
||||
|
||||
public TransferablePolicy getTransferablePolicy() {
|
||||
return transferablePolicy;
|
||||
}
|
||||
|
||||
|
||||
public LinkedList<File> convertToList() {
|
||||
LinkedList<File> list = new LinkedList<File>();
|
||||
TreeModel m = getModel();
|
||||
walk(m, m.getRoot(), list);
|
||||
return list;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public String convertValueToText(Object value, boolean selected, boolean expanded, boolean leaf, int row, boolean hasFocus) {
|
||||
if (value instanceof DefaultMutableTreeNode) {
|
||||
DefaultMutableTreeNode node = (DefaultMutableTreeNode) value;
|
||||
|
||||
Object userObject = node.getUserObject();
|
||||
|
||||
if (userObject != null && userObject instanceof File) {
|
||||
File file = (File) node.getUserObject();
|
||||
return file.getName();
|
||||
}
|
||||
}
|
||||
|
||||
return value.toString();
|
||||
}
|
||||
|
||||
|
||||
private void walk(TreeModel model, Object o, LinkedList<File> list) {
|
||||
int cc = model.getChildCount(o);
|
||||
|
||||
for (int i = 0; i < cc; i++) {
|
||||
DefaultMutableTreeNode child = (DefaultMutableTreeNode) model.getChild(o, i);
|
||||
if (model.isLeaf(child))
|
||||
list.add((File) child.getUserObject());
|
||||
else
|
||||
walk(model, child, list);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public void expandOrCollapseAll(boolean expand) {
|
||||
TreeNode node = (TreeNode) getModel().getRoot();
|
||||
Enumeration<?> e = node.children();
|
||||
|
||||
while (e.hasMoreElements()) {
|
||||
TreeNode n = (TreeNode) e.nextElement();
|
||||
TreePath path = new TreePath(node).pathByAddingChild(n);
|
||||
expandOrCollapseAllImpl(path, expand);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private void expandOrCollapseAllImpl(TreePath parent, boolean expand) {
|
||||
// Traverse children
|
||||
TreeNode node = (TreeNode) parent.getLastPathComponent();
|
||||
if (node.getChildCount() >= 0)
|
||||
for (Enumeration<?> e = node.children(); e.hasMoreElements();) {
|
||||
TreeNode n = (TreeNode) e.nextElement();
|
||||
TreePath path = parent.pathByAddingChild(n);
|
||||
expandOrCollapseAllImpl(path, expand);
|
||||
}
|
||||
|
||||
// Expansion or collapse must be done bottom-up
|
||||
if (expand)
|
||||
expandPath(parent);
|
||||
else
|
||||
collapsePath(parent);
|
||||
}
|
||||
|
||||
|
||||
private class ExpandCollapsePopup extends JPopupMenu {
|
||||
|
||||
public ExpandCollapsePopup() {
|
||||
if (getSelectionCount() >= 1) {
|
||||
Object pathComponent = getLastSelectedPathComponent();
|
||||
|
||||
if (pathComponent instanceof DefaultMutableTreeNode) {
|
||||
DefaultMutableTreeNode node = (DefaultMutableTreeNode) pathComponent;
|
||||
|
||||
Object userObject = node.getUserObject();
|
||||
|
||||
if (userObject instanceof File) {
|
||||
File file = (File) userObject;
|
||||
|
||||
if (file.isFile()) {
|
||||
JMenuItem openItem = new JMenuItem(new OpenAction("Open", file));
|
||||
openItem.setFont(openItem.getFont().deriveFont(Font.BOLD));
|
||||
|
||||
add(openItem);
|
||||
add(new OpenAction("Open Folder", file.getParentFile()));
|
||||
addSeparator();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
add(new ExpandOrCollapseAction(true));
|
||||
add(new ExpandOrCollapseAction(false));
|
||||
}
|
||||
|
||||
|
||||
private class OpenAction extends AbstractAction {
|
||||
|
||||
private File file;
|
||||
|
||||
|
||||
public OpenAction(String text, File file) {
|
||||
super(text);
|
||||
this.file = file;
|
||||
|
||||
if (file == null)
|
||||
setEnabled(false);
|
||||
}
|
||||
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
try {
|
||||
Desktop.getDesktop().open(file);
|
||||
} catch (Exception ex) {
|
||||
MessageManager.showWarning(ex.getMessage());
|
||||
ex.printStackTrace();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private class ExpandOrCollapseAction extends AbstractAction {
|
||||
|
||||
private boolean expand;
|
||||
|
||||
|
||||
public ExpandOrCollapseAction(boolean expand) {
|
||||
this.expand = expand;
|
||||
|
||||
if (expand) {
|
||||
putValue(SMALL_ICON, ResourceManager.getIcon("tree.expand"));
|
||||
putValue(NAME, "Expand all");
|
||||
} else {
|
||||
putValue(SMALL_ICON, ResourceManager.getIcon("tree.collapse"));
|
||||
putValue(NAME, "Collapse all");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
expandOrCollapseAll(expand);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
private class ExpandCollapsePopupListener extends MouseAdapter {
|
||||
|
||||
@Override
|
||||
public void mouseReleased(MouseEvent e) {
|
||||
if (SwingUtilities.isRightMouseButton(e)) {
|
||||
ExpandCollapsePopup popup = new ExpandCollapsePopup();
|
||||
popup.show(e.getComponent(), e.getX(), e.getY());
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void mousePressed(MouseEvent e) {
|
||||
if (SwingUtilities.isRightMouseButton(e)) {
|
||||
TreePath path = getPathForLocation(e.getX(), e.getY());
|
||||
setSelectionPath(path);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,47 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.Component;
|
||||
import java.io.File;
|
||||
|
||||
import javax.swing.JTree;
|
||||
import javax.swing.tree.DefaultMutableTreeNode;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.tuned.ui.FancyTreeCellRenderer;
|
||||
import net.sourceforge.tuned.ui.GradientStyle;
|
||||
|
||||
|
||||
public class FileBotTreeCellRenderer extends FancyTreeCellRenderer {
|
||||
|
||||
public FileBotTreeCellRenderer() {
|
||||
super(GradientStyle.TOP_TO_BOTTOM);
|
||||
openIcon = ResourceManager.getIcon("tree.open");
|
||||
closedIcon = ResourceManager.getIcon("tree.closed");
|
||||
leafIcon = ResourceManager.getIcon("tree.leaf");
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public Component getTreeCellRendererComponent(JTree tree, Object value, boolean sel, boolean expanded, boolean leaf, int row, boolean hasFocus) {
|
||||
if (leaf && isFolder(value))
|
||||
super.getTreeCellRendererComponent(tree, value, sel, true, false, row, hasFocus);
|
||||
else
|
||||
super.getTreeCellRendererComponent(tree, value, sel, expanded, leaf, row, hasFocus);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
|
||||
private boolean isFolder(Object value) {
|
||||
DefaultMutableTreeNode node = (DefaultMutableTreeNode) value;
|
||||
Object object = node.getUserObject();
|
||||
|
||||
if (object instanceof File)
|
||||
return ((File) object).isDirectory();
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
}
|
23
source/net/sourceforge/filebot/ui/FileBotUtil.java
Normal file
@ -0,0 +1,23 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import javax.swing.Action;
|
||||
import javax.swing.JComponent;
|
||||
import javax.swing.KeyStroke;
|
||||
|
||||
|
||||
public class FileBotUtil {
|
||||
|
||||
private FileBotUtil() {
|
||||
// hide construktor
|
||||
}
|
||||
|
||||
|
||||
public static void registerActionForKeystroke(JComponent component, KeyStroke keystroke, Action action) {
|
||||
Integer key = action.hashCode();
|
||||
component.getInputMap(JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT).put(keystroke, key);
|
||||
component.getActionMap().put(key, action);
|
||||
}
|
||||
|
||||
}
|
139
source/net/sourceforge/filebot/ui/FileBotWindow.java
Normal file
@ -0,0 +1,139 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.CardLayout;
|
||||
import java.awt.Image;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.util.ArrayList;
|
||||
|
||||
import javax.swing.AbstractAction;
|
||||
import javax.swing.JComponent;
|
||||
import javax.swing.JFrame;
|
||||
import javax.swing.JPanel;
|
||||
import javax.swing.JScrollPane;
|
||||
import javax.swing.KeyStroke;
|
||||
import javax.swing.ListModel;
|
||||
import javax.swing.OverlayLayout;
|
||||
import javax.swing.ScrollPaneConstants;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
import javax.swing.event.ListSelectionEvent;
|
||||
import javax.swing.event.ListSelectionListener;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.tuned.ui.ShadowBorder;
|
||||
|
||||
|
||||
public class FileBotWindow extends JFrame implements ListSelectionListener {
|
||||
|
||||
private JPanel pagePanel = new JPanel(new CardLayout());
|
||||
|
||||
private FileBotPanelSelectionList selectionListPanel = new FileBotPanelSelectionList();
|
||||
|
||||
private HeaderPanel headerPanel = new HeaderPanel();
|
||||
|
||||
|
||||
public FileBotWindow() {
|
||||
super("FileBot");
|
||||
setLocationByPlatform(true);
|
||||
setDefaultCloseOperation(EXIT_ON_CLOSE);
|
||||
|
||||
ArrayList<Image> icons = new ArrayList<Image>(2);
|
||||
icons.add(ResourceManager.getImage("window.icon.small"));
|
||||
icons.add(ResourceManager.getImage("window.icon.big"));
|
||||
setIconImages(icons);
|
||||
|
||||
selectionListPanel.addListSelectionListener(this);
|
||||
|
||||
JComponent contentPane = createContentPane();
|
||||
|
||||
setContentPane(contentPane);
|
||||
|
||||
// Shortcut ESC
|
||||
FileBotUtil.registerActionForKeystroke(contentPane, KeyStroke.getKeyStroke("released ESCAPE"), closeAction);
|
||||
|
||||
setSize(760, 615);
|
||||
|
||||
selectionListPanel.setSelectedIndex(3);
|
||||
}
|
||||
|
||||
|
||||
public void valueChanged(ListSelectionEvent e) {
|
||||
FileBotPanel currentPanel = (FileBotPanel) selectionListPanel.getSelectedValue();
|
||||
|
||||
headerPanel.setTitle(currentPanel.getText());
|
||||
CardLayout cardLayout = (CardLayout) pagePanel.getLayout();
|
||||
cardLayout.show(pagePanel, currentPanel.getText());
|
||||
|
||||
JComponent c = (JComponent) getContentPane();
|
||||
c.updateUI();
|
||||
}
|
||||
|
||||
|
||||
private JComponent createSelectionListLayer() {
|
||||
JPanel selectionListLayer = new JPanel(new BorderLayout());
|
||||
selectionListLayer.setOpaque(false);
|
||||
|
||||
JPanel shadowBorderPanel = new JPanel(new BorderLayout());
|
||||
shadowBorderPanel.setOpaque(false);
|
||||
|
||||
JScrollPane selectListScrollPane = new JScrollPane(selectionListPanel);
|
||||
selectListScrollPane.setHorizontalScrollBarPolicy(ScrollPaneConstants.HORIZONTAL_SCROLLBAR_NEVER);
|
||||
selectListScrollPane.setVerticalScrollBarPolicy(ScrollPaneConstants.VERTICAL_SCROLLBAR_NEVER);
|
||||
shadowBorderPanel.add(selectListScrollPane, BorderLayout.CENTER);
|
||||
shadowBorderPanel.setBorder(new ShadowBorder());
|
||||
|
||||
selectionListLayer.setBorder(new EmptyBorder(10, 6, 12, 0));
|
||||
selectionListLayer.add(shadowBorderPanel, BorderLayout.WEST);
|
||||
|
||||
selectionListLayer.setAlignmentX(0.0f);
|
||||
selectionListLayer.setAlignmentY(0.0f);
|
||||
selectionListLayer.setMaximumSize(selectionListLayer.getPreferredSize());
|
||||
|
||||
return selectionListLayer;
|
||||
}
|
||||
|
||||
|
||||
private JComponent createPageLayer() {
|
||||
JPanel pageLayer = new JPanel(new BorderLayout());
|
||||
|
||||
pagePanel.setBorder(new EmptyBorder(10, 110, 10, 10));
|
||||
|
||||
pageLayer.add(headerPanel, BorderLayout.NORTH);
|
||||
pageLayer.add(pagePanel, BorderLayout.CENTER);
|
||||
|
||||
ListModel model = selectionListPanel.getModel();
|
||||
|
||||
for (int i = 0; i < model.getSize(); i++) {
|
||||
FileBotPanel panel = (FileBotPanel) model.getElementAt(i);
|
||||
panel.setVisible(false);
|
||||
pagePanel.add(panel, panel.getText());
|
||||
}
|
||||
|
||||
pageLayer.setAlignmentX(0.0f);
|
||||
pageLayer.setAlignmentY(0.0f);
|
||||
|
||||
return pageLayer;
|
||||
}
|
||||
|
||||
|
||||
private JComponent createContentPane() {
|
||||
JPanel contentPane = new JPanel();
|
||||
contentPane.setLayout(new OverlayLayout(contentPane));
|
||||
|
||||
contentPane.add(createSelectionListLayer());
|
||||
contentPane.add(createPageLayer());
|
||||
|
||||
return contentPane;
|
||||
}
|
||||
|
||||
private final AbstractAction closeAction = new AbstractAction("Close") {
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
setVisible(false);
|
||||
dispose();
|
||||
System.exit(0);
|
||||
}
|
||||
};
|
||||
}
|
95
source/net/sourceforge/filebot/ui/FileFormat.java
Normal file
@ -0,0 +1,95 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.io.File;
|
||||
import java.text.NumberFormat;
|
||||
|
||||
|
||||
public class FileFormat {
|
||||
|
||||
private static NumberFormat nf = NumberFormat.getNumberInstance();
|
||||
|
||||
public final static long KILO = 1024;
|
||||
|
||||
public final static long MEGA = KILO * 1024;
|
||||
|
||||
public final static long GIGA = MEGA * 1024;
|
||||
|
||||
static {
|
||||
nf.setMaximumFractionDigits(0);
|
||||
}
|
||||
|
||||
|
||||
public static String formatSize(long size) {
|
||||
if (size >= MEGA)
|
||||
return nf.format((double) size / MEGA) + " MB";
|
||||
else if (size >= KILO)
|
||||
return nf.format((double) size / KILO) + " KB";
|
||||
else
|
||||
return nf.format((double) size) + " Byte";
|
||||
}
|
||||
|
||||
|
||||
public static String formatName(File f) {
|
||||
String name = f.getName();
|
||||
|
||||
if (f.isDirectory())
|
||||
return name;
|
||||
|
||||
return getNameWithoutSuffix(name);
|
||||
}
|
||||
|
||||
|
||||
public static String formatNumberOfFiles(int n) {
|
||||
if (n == 1)
|
||||
return n + " file";
|
||||
else
|
||||
return n + " files";
|
||||
}
|
||||
|
||||
|
||||
public static String getSuffix(File f) {
|
||||
return getSuffix(f, false);
|
||||
}
|
||||
|
||||
|
||||
public static String getSuffix(File f, boolean dot) {
|
||||
String name = f.getName();
|
||||
int dotIndex = name.lastIndexOf(".");
|
||||
|
||||
if (dotIndex > 1) {
|
||||
String suffix = name.substring(dotIndex + 1, name.length());
|
||||
if (dot)
|
||||
return "." + suffix;
|
||||
else
|
||||
return suffix;
|
||||
} else
|
||||
return "";
|
||||
}
|
||||
|
||||
|
||||
public static String getNameWithoutSuffix(String name) {
|
||||
int dotIndex = name.lastIndexOf(".");
|
||||
|
||||
if (dotIndex < 1)
|
||||
return name;
|
||||
|
||||
return name.substring(0, dotIndex);
|
||||
}
|
||||
|
||||
|
||||
public static String getNameWithoutSuffix(File file) {
|
||||
return getNameWithoutSuffix(file.getName());
|
||||
}
|
||||
|
||||
|
||||
public static String getName(File file) {
|
||||
if (!file.getName().isEmpty()) {
|
||||
return file.getName();
|
||||
} else {
|
||||
return file.toString();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
76
source/net/sourceforge/filebot/ui/HeaderPanel.java
Normal file
@ -0,0 +1,76 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.Color;
|
||||
import java.awt.Font;
|
||||
import java.awt.Graphics;
|
||||
import java.awt.Graphics2D;
|
||||
import java.awt.LinearGradientPaint;
|
||||
|
||||
import javax.swing.JLabel;
|
||||
import javax.swing.JPanel;
|
||||
import javax.swing.SwingConstants;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.tuned.ui.GradientStyle;
|
||||
import net.sourceforge.tuned.ui.notification.SeparatorBorder;
|
||||
import net.sourceforge.tuned.ui.notification.SeparatorBorder.Position;
|
||||
|
||||
|
||||
public class HeaderPanel extends JPanel {
|
||||
|
||||
private JLabel titleLabel = new JLabel();
|
||||
|
||||
private float[] gradientFractions = { 0.0f, 0.5f, 1.0f };
|
||||
private Color[] gradientColors = { new Color(0xF6F6F6), new Color(0xF8F8F8), new Color(0xF3F3F3) };
|
||||
|
||||
|
||||
public HeaderPanel() {
|
||||
super(new BorderLayout());
|
||||
setBackground(Color.WHITE);
|
||||
|
||||
JPanel centerPanel = new JPanel(new BorderLayout());
|
||||
centerPanel.setOpaque(false);
|
||||
|
||||
titleLabel.setHorizontalAlignment(SwingConstants.CENTER);
|
||||
titleLabel.setVerticalAlignment(SwingConstants.CENTER);
|
||||
titleLabel.setOpaque(false);
|
||||
titleLabel.setForeground(new Color(0x101010));
|
||||
titleLabel.setFont(new Font(Font.SANS_SERIF, Font.PLAIN, 24));
|
||||
|
||||
int margin = 9;
|
||||
titleLabel.setBorder(new EmptyBorder(margin - 1, 90, margin + 1, 0));
|
||||
|
||||
JLabel decorationLabel = new JLabel(ResourceManager.getIcon("decoration.header"));
|
||||
decorationLabel.setHorizontalAlignment(SwingConstants.CENTER);
|
||||
decorationLabel.setVerticalAlignment(SwingConstants.CENTER);
|
||||
|
||||
centerPanel.setBorder(new EmptyBorder(0, 78, 0, 0));
|
||||
centerPanel.add(decorationLabel, BorderLayout.EAST);
|
||||
centerPanel.add(titleLabel, BorderLayout.CENTER);
|
||||
|
||||
add(centerPanel, BorderLayout.CENTER);
|
||||
|
||||
setBorder(new SeparatorBorder(1, new Color(0xB4B4B4), new Color(0xACACAC), GradientStyle.LEFT_TO_RIGHT, Position.BOTTOM));
|
||||
}
|
||||
|
||||
|
||||
public void setTitle(String title) {
|
||||
titleLabel.setText(title);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void paintComponent(Graphics g) {
|
||||
Graphics2D g2d = (Graphics2D) g;
|
||||
|
||||
LinearGradientPaint paint = new LinearGradientPaint(0, 0, getWidth(), 0, gradientFractions, gradientColors);
|
||||
|
||||
g2d.setPaint(paint);
|
||||
g2d.fill(getBounds());
|
||||
}
|
||||
|
||||
}
|
34
source/net/sourceforge/filebot/ui/MessageManager.java
Normal file
@ -0,0 +1,34 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import javax.swing.Icon;
|
||||
import javax.swing.SwingConstants;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.tuned.ui.notification.MessageNotification;
|
||||
import net.sourceforge.tuned.ui.notification.NotificationManager;
|
||||
import net.sourceforge.tuned.ui.notification.QueueNotificationLayout;
|
||||
|
||||
|
||||
public class MessageManager {
|
||||
|
||||
private static final int TIMEOUT = 2500;
|
||||
private static final NotificationManager manager = new NotificationManager(new QueueNotificationLayout(SwingConstants.NORTH, SwingConstants.SOUTH));
|
||||
|
||||
|
||||
public static void showInfo(String message) {
|
||||
show(message, ResourceManager.getIcon("message.info"), TIMEOUT);
|
||||
}
|
||||
|
||||
|
||||
public static void showWarning(String message) {
|
||||
show(message, ResourceManager.getIcon("message.warning"), TIMEOUT * 2);
|
||||
}
|
||||
|
||||
|
||||
private static void show(String message, Icon icon, int timeout) {
|
||||
manager.show(new MessageNotification("FileBot", message, icon, timeout));
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,57 @@
|
||||
|
||||
package net.sourceforge.filebot.ui;
|
||||
|
||||
|
||||
import java.awt.event.MouseAdapter;
|
||||
import java.awt.event.MouseEvent;
|
||||
|
||||
import javax.swing.JComponent;
|
||||
import javax.swing.TransferHandler;
|
||||
|
||||
|
||||
public class MouseDragRecognizeListener extends MouseAdapter {
|
||||
|
||||
private MouseEvent firstMouseEvent = null;
|
||||
|
||||
private int dragShift = 5;
|
||||
|
||||
|
||||
@Override
|
||||
public void mousePressed(MouseEvent e) {
|
||||
firstMouseEvent = e;
|
||||
e.consume();
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void mouseDragged(MouseEvent e) {
|
||||
if (firstMouseEvent != null) {
|
||||
e.consume();
|
||||
int dx = Math.abs(e.getX() - firstMouseEvent.getX());
|
||||
int dy = Math.abs(e.getY() - firstMouseEvent.getY());
|
||||
|
||||
if (dx > dragShift || dy > dragShift) {
|
||||
// This is a drag, not a click.
|
||||
JComponent c = (JComponent) e.getSource();
|
||||
TransferHandler handler = c.getTransferHandler();
|
||||
handler.exportAsDrag(c, firstMouseEvent, TransferHandler.COPY);
|
||||
firstMouseEvent = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void mouseReleased(MouseEvent e) {
|
||||
firstMouseEvent = null;
|
||||
}
|
||||
|
||||
|
||||
public static void createForComponent(JComponent component) {
|
||||
MouseDragRecognizeListener l = new MouseDragRecognizeListener();
|
||||
|
||||
component.addMouseListener(l);
|
||||
component.addMouseMotionListener(l);
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,81 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.analyze;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.Dimension;
|
||||
import java.beans.PropertyChangeEvent;
|
||||
import java.beans.PropertyChangeListener;
|
||||
import java.io.File;
|
||||
import java.util.LinkedList;
|
||||
import java.util.List;
|
||||
|
||||
import javax.swing.BorderFactory;
|
||||
import javax.swing.Box;
|
||||
import javax.swing.BoxLayout;
|
||||
import javax.swing.JPanel;
|
||||
import javax.swing.JTabbedPane;
|
||||
import javax.swing.SwingConstants;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.filebot.ui.FileBotPanel;
|
||||
import net.sourceforge.filebot.ui.panel.analyze.tools.SplitPanel;
|
||||
import net.sourceforge.filebot.ui.panel.analyze.tools.ToolPanel;
|
||||
import net.sourceforge.filebot.ui.panel.analyze.tools.TypePanel;
|
||||
|
||||
|
||||
public class AnalyzePanel extends FileBotPanel {
|
||||
|
||||
private FileTreePanel filePanel = new FileTreePanel();
|
||||
private JTabbedPane toolsPanel = new JTabbedPane(SwingConstants.TOP, JTabbedPane.SCROLL_TAB_LAYOUT);
|
||||
|
||||
|
||||
public AnalyzePanel() {
|
||||
super("Analyze", ResourceManager.getIcon("panel.analyze"));
|
||||
|
||||
Box panel = new Box(BoxLayout.X_AXIS);
|
||||
|
||||
panel.add(filePanel);
|
||||
|
||||
panel.add(Box.createHorizontalStrut(50));
|
||||
|
||||
JPanel right = new JPanel();
|
||||
right.setLayout(new BorderLayout());
|
||||
right.setBorder(BorderFactory.createTitledBorder("Tools"));
|
||||
|
||||
right.add(toolsPanel, BorderLayout.CENTER);
|
||||
|
||||
panel.add(right);
|
||||
|
||||
add(panel, BorderLayout.CENTER);
|
||||
|
||||
filePanel.getFileTree().addPropertyChangeListener(FileTree.CONTENT_PROPERTY, fileTreeChangeListener);
|
||||
|
||||
addTool(new TypePanel());
|
||||
addTool(new SplitPanel());
|
||||
|
||||
Dimension min = new Dimension(300, 300);
|
||||
filePanel.setMinimumSize(min);
|
||||
toolsPanel.setMinimumSize(min);
|
||||
}
|
||||
|
||||
private PropertyChangeListener fileTreeChangeListener = new PropertyChangeListener() {
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
public void propertyChange(PropertyChangeEvent evt) {
|
||||
List<File> files = (List<File>) evt.getNewValue();
|
||||
|
||||
for (ToolPanel toolPanel : toolPanels)
|
||||
toolPanel.update(files);
|
||||
}
|
||||
};
|
||||
|
||||
private LinkedList<ToolPanel> toolPanels = new LinkedList<ToolPanel>();
|
||||
|
||||
|
||||
public void addTool(ToolPanel toolPanel) {
|
||||
toolsPanel.addTab(toolPanel.getName(), toolPanel);
|
||||
toolPanels.add(toolPanel);
|
||||
}
|
||||
|
||||
}
|
149
source/net/sourceforge/filebot/ui/panel/analyze/FileTree.java
Normal file
@ -0,0 +1,149 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.analyze;
|
||||
|
||||
|
||||
import java.awt.datatransfer.Transferable;
|
||||
import java.beans.PropertyChangeEvent;
|
||||
import java.beans.PropertyChangeListener;
|
||||
import java.io.File;
|
||||
import java.util.List;
|
||||
|
||||
import javax.swing.tree.DefaultMutableTreeNode;
|
||||
import javax.swing.tree.TreePath;
|
||||
|
||||
import net.sourceforge.filebot.ui.FileBotTree;
|
||||
import net.sourceforge.filebot.ui.sal.FileTransferable;
|
||||
import net.sourceforge.filebot.ui.transferablepolicies.BackgroundFileTransferablePolicy;
|
||||
|
||||
|
||||
public class FileTree extends FileBotTree {
|
||||
|
||||
public static final String LOADING_PROPERTY = "loading";
|
||||
public static final String CONTENT_PROPERTY = "content";
|
||||
|
||||
|
||||
public FileTree() {
|
||||
setTransferablePolicy(new FileTreeTransferPolicy());
|
||||
}
|
||||
|
||||
|
||||
public void removeTreeItems(TreePath paths[]) {
|
||||
for (TreePath element : paths) {
|
||||
DefaultMutableTreeNode node = (DefaultMutableTreeNode) (element.getLastPathComponent());
|
||||
node.removeFromParent();
|
||||
}
|
||||
|
||||
updateUI();
|
||||
|
||||
firePropertyChange(LOADING_PROPERTY, null, true);
|
||||
contentChanged();
|
||||
firePropertyChange(LOADING_PROPERTY, null, false);
|
||||
}
|
||||
|
||||
|
||||
public void load(List<File> files) {
|
||||
FileTransferable tr = new FileTransferable(files);
|
||||
|
||||
if (getTransferablePolicy().accept(tr))
|
||||
getTransferablePolicy().handleTransferable(tr, true);
|
||||
}
|
||||
|
||||
|
||||
private void contentChanged() {
|
||||
List<File> files = convertToList();
|
||||
firePropertyChange(CONTENT_PROPERTY, null, files);
|
||||
}
|
||||
|
||||
|
||||
public void addFiles(DefaultMutableTreeNode parent, File files[]) {
|
||||
// folders first
|
||||
for (File f : files)
|
||||
if (f.isDirectory()) {
|
||||
// run through file tree
|
||||
DefaultMutableTreeNode node = new DefaultMutableTreeNode(f);
|
||||
addFiles(node, f.listFiles());
|
||||
parent.add(node);
|
||||
}
|
||||
|
||||
for (File f : files) {
|
||||
if (!f.isDirectory())
|
||||
parent.add(new DefaultMutableTreeNode(f));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void clear() {
|
||||
super.clear();
|
||||
contentChanged();
|
||||
}
|
||||
|
||||
|
||||
private class FileTreeTransferPolicy extends BackgroundFileTransferablePolicy<Object> implements PropertyChangeListener {
|
||||
|
||||
public FileTreeTransferPolicy() {
|
||||
addPropertyChangeListener(LOADING_PROPERTY, this);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected boolean accept(File file) {
|
||||
return file.isFile() || file.isDirectory();
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public boolean handleTransferable(Transferable tr, boolean add) {
|
||||
return super.handleTransferable(tr, true);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void clear() {
|
||||
FileTree.this.clear();
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected boolean load(List<File> files) {
|
||||
DefaultMutableTreeNode root = (DefaultMutableTreeNode) getModel().getRoot();
|
||||
|
||||
File fileArray[] = new File[files.size()];
|
||||
files.toArray(fileArray);
|
||||
|
||||
addFiles(root, fileArray);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* This method will not be used
|
||||
*/
|
||||
@Override
|
||||
protected boolean load(File file) {
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
public void propertyChange(PropertyChangeEvent evt) {
|
||||
Boolean loading = (Boolean) evt.getNewValue();
|
||||
|
||||
if (loading) {
|
||||
FileTree.this.firePropertyChange(LOADING_PROPERTY, null, true);
|
||||
} else {
|
||||
FileTree.this.firePropertyChange(LOADING_PROPERTY, null, false);
|
||||
updateUI();
|
||||
contentChanged();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public String getDescription() {
|
||||
return "files and folders";
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,98 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.analyze;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.beans.PropertyChangeEvent;
|
||||
import java.beans.PropertyChangeListener;
|
||||
|
||||
import javax.swing.AbstractAction;
|
||||
import javax.swing.BorderFactory;
|
||||
import javax.swing.Box;
|
||||
import javax.swing.JButton;
|
||||
import javax.swing.JPanel;
|
||||
import javax.swing.JScrollPane;
|
||||
import javax.swing.KeyStroke;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.filebot.ui.FileBotUtil;
|
||||
import net.sourceforge.filebot.ui.sal.LoadAction;
|
||||
import net.sourceforge.tuned.ui.LoadingOverlayPanel;
|
||||
|
||||
|
||||
public class FileTreePanel extends JPanel {
|
||||
|
||||
private FileTree fileTree = new FileTree();
|
||||
|
||||
private LoadingOverlayPanel loadingOverlay;
|
||||
|
||||
|
||||
public FileTreePanel() {
|
||||
super(new BorderLayout());
|
||||
fileTree.addPropertyChangeListener(FileTree.LOADING_PROPERTY, loadingOverlayUpdateListener);
|
||||
|
||||
setBorder(BorderFactory.createTitledBorder("File Tree"));
|
||||
|
||||
loadingOverlay = new LoadingOverlayPanel(new JScrollPane(fileTree), ResourceManager.getIcon("loading"));
|
||||
|
||||
Box buttons = Box.createHorizontalBox();
|
||||
buttons.setBorder(new EmptyBorder(5, 5, 5, 5));
|
||||
buttons.add(Box.createGlue());
|
||||
buttons.add(new JButton(loadAction));
|
||||
buttons.add(Box.createHorizontalStrut(5));
|
||||
buttons.add(new JButton(clearAction));
|
||||
buttons.add(Box.createGlue());
|
||||
|
||||
// Shortcut DELETE
|
||||
FileBotUtil.registerActionForKeystroke(fileTree, KeyStroke.getKeyStroke("pressed DELETE"), removeAction);
|
||||
|
||||
add(loadingOverlay, BorderLayout.CENTER);
|
||||
add(buttons, BorderLayout.SOUTH);
|
||||
}
|
||||
|
||||
private PropertyChangeListener loadingOverlayUpdateListener = new PropertyChangeListener() {
|
||||
|
||||
public void propertyChange(PropertyChangeEvent evt) {
|
||||
Boolean loading = (Boolean) evt.getNewValue();
|
||||
|
||||
loadingOverlay.setOverlayVisible(loading);
|
||||
loadingOverlay.updateOverlayUI();
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
public FileTree getFileTree() {
|
||||
return fileTree;
|
||||
}
|
||||
|
||||
private final LoadAction loadAction = new LoadAction(fileTree);
|
||||
|
||||
private final AbstractAction clearAction = new AbstractAction("Clear", ResourceManager.getIcon("action.clear")) {
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
fileTree.clear();
|
||||
}
|
||||
};
|
||||
|
||||
private final AbstractAction removeAction = new AbstractAction("Remove") {
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
if (fileTree.getSelectionCount() < 1)
|
||||
return;
|
||||
|
||||
int row = fileTree.getMinSelectionRow();
|
||||
|
||||
fileTree.removeTreeItems(fileTree.getSelectionPaths());
|
||||
|
||||
int maxRow = fileTree.getRowCount() - 1;
|
||||
|
||||
if (row > maxRow)
|
||||
row = maxRow;
|
||||
|
||||
fileTree.setSelectionRow(row);
|
||||
}
|
||||
};
|
||||
|
||||
}
|
@ -0,0 +1,186 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.analyze.tools;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.Color;
|
||||
import java.awt.Dimension;
|
||||
import java.io.File;
|
||||
import java.util.Collection;
|
||||
|
||||
import javax.swing.BorderFactory;
|
||||
import javax.swing.Box;
|
||||
import javax.swing.JLabel;
|
||||
import javax.swing.JScrollPane;
|
||||
import javax.swing.JSpinner;
|
||||
import javax.swing.SpinnerNumberModel;
|
||||
import javax.swing.SwingWorker;
|
||||
import javax.swing.border.CompoundBorder;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
import javax.swing.event.ChangeEvent;
|
||||
import javax.swing.event.ChangeListener;
|
||||
import javax.swing.tree.DefaultMutableTreeNode;
|
||||
import javax.swing.tree.DefaultTreeModel;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.filebot.ui.FileBotTree;
|
||||
import net.sourceforge.filebot.ui.FileFormat;
|
||||
import net.sourceforge.tuned.ui.GradientStyle;
|
||||
import net.sourceforge.tuned.ui.LoadingOverlayPanel;
|
||||
import net.sourceforge.tuned.ui.notification.SeparatorBorder;
|
||||
|
||||
|
||||
public class SplitPanel extends ToolPanel implements ChangeListener {
|
||||
|
||||
private FileBotTree tree = new FileBotTree();
|
||||
|
||||
private SpinnerNumberModel spinnerModel = new SpinnerNumberModel(4480, 0, Integer.MAX_VALUE, 100);
|
||||
|
||||
|
||||
public SplitPanel() {
|
||||
super("Split");
|
||||
setLayout(new BorderLayout());
|
||||
|
||||
JScrollPane sp = new JScrollPane(tree);
|
||||
sp.setBorder(BorderFactory.createEmptyBorder());
|
||||
LoadingOverlayPanel loadingOverlay = new LoadingOverlayPanel(sp, ResourceManager.getIcon("loading"));
|
||||
JSpinner spinner = new JSpinner(spinnerModel);
|
||||
spinner.setMaximumSize(spinner.getPreferredSize());
|
||||
spinner.setEditor(new JSpinner.NumberEditor(spinner, "#"));
|
||||
|
||||
Box spinnerBox = Box.createHorizontalBox();
|
||||
spinnerBox.add(Box.createGlue());
|
||||
spinnerBox.add(new JLabel("Split every"));
|
||||
spinnerBox.add(Box.createHorizontalStrut(5));
|
||||
spinnerBox.add(spinner);
|
||||
spinnerBox.add(Box.createHorizontalStrut(5));
|
||||
spinnerBox.add(new JLabel("MB."));
|
||||
spinnerBox.add(Box.createGlue());
|
||||
|
||||
add(loadingOverlay, BorderLayout.CENTER);
|
||||
add(spinnerBox, BorderLayout.SOUTH);
|
||||
|
||||
Color beginColor = new Color(0, 0, 0, 90);
|
||||
SeparatorBorder separatorBorder = new SeparatorBorder(2, beginColor, GradientStyle.TOP_TO_BOTTOM, SeparatorBorder.Position.TOP);
|
||||
spinnerBox.setBorder(new CompoundBorder(separatorBorder, new EmptyBorder(6, 5, 7, 5)));
|
||||
|
||||
setLoadingOverlayPane(loadingOverlay);
|
||||
spinnerModel.addChangeListener(this);
|
||||
spinner.setPreferredSize(new Dimension(80, 20));
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* splitsize change callback
|
||||
*/
|
||||
public void stateChanged(ChangeEvent e) {
|
||||
if (files != null)
|
||||
update();
|
||||
}
|
||||
|
||||
|
||||
private long getSplitSize() {
|
||||
return spinnerModel.getNumber().intValue() * FileFormat.MEGA;
|
||||
}
|
||||
|
||||
private UpdateTask latestUpdateTask;
|
||||
|
||||
private Collection<File> files;
|
||||
|
||||
|
||||
@Override
|
||||
public void update(Collection<File> files) {
|
||||
this.files = files;
|
||||
update();
|
||||
}
|
||||
|
||||
|
||||
private void update() {
|
||||
latestUpdateTask = new UpdateTask();
|
||||
firePropertyChange(LOADING_PROPERTY, null, true);
|
||||
latestUpdateTask.execute();
|
||||
}
|
||||
|
||||
|
||||
private class UpdateTask extends SwingWorker<DefaultTreeModel, Object> {
|
||||
|
||||
private boolean isLatest() {
|
||||
if (this == latestUpdateTask)
|
||||
return true;
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
private void setLastChildUserObject(DefaultMutableTreeNode root, int part, long size) {
|
||||
DefaultMutableTreeNode node = ((DefaultMutableTreeNode) root.getLastChild());
|
||||
String uo = "Part " + part + " (" + FileFormat.formatSize(size) + ")";
|
||||
node.setUserObject(uo);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected DefaultTreeModel doInBackground() throws Exception {
|
||||
long currentSize = 0;
|
||||
|
||||
DefaultMutableTreeNode root = new DefaultMutableTreeNode();
|
||||
DefaultMutableTreeNode first = new DefaultMutableTreeNode();
|
||||
root.add(first);
|
||||
DefaultMutableTreeNode remainder = new DefaultMutableTreeNode("Remainder");
|
||||
|
||||
int p = 1;
|
||||
long splitSize = getSplitSize();
|
||||
|
||||
for (File f : files) {
|
||||
long fileSize = f.length();
|
||||
DefaultMutableTreeNode fileNode = new DefaultMutableTreeNode(f);
|
||||
|
||||
if (fileSize > splitSize)
|
||||
remainder.add(fileNode);
|
||||
else if (currentSize + fileSize <= splitSize) {
|
||||
currentSize += fileSize;
|
||||
((DefaultMutableTreeNode) root.getLastChild()).add(fileNode);
|
||||
} else {
|
||||
setLastChildUserObject(root, p, currentSize);
|
||||
|
||||
currentSize = fileSize;
|
||||
p++;
|
||||
DefaultMutableTreeNode node = new DefaultMutableTreeNode();
|
||||
node.add(fileNode);
|
||||
root.add(node);
|
||||
}
|
||||
|
||||
if (!isLatest())
|
||||
return null;
|
||||
}
|
||||
|
||||
setLastChildUserObject(root, p, currentSize);
|
||||
|
||||
if (!remainder.isLeaf())
|
||||
root.add(remainder);
|
||||
|
||||
if (first.isLeaf())
|
||||
first.removeFromParent();
|
||||
|
||||
return new DefaultTreeModel(root);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void done() {
|
||||
try {
|
||||
DefaultTreeModel model = get();
|
||||
|
||||
if (model == null)
|
||||
return;
|
||||
|
||||
tree.setModel(model);
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
SplitPanel.this.firePropertyChange(LOADING_PROPERTY, null, false);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,55 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.analyze.tools;
|
||||
|
||||
|
||||
import java.beans.PropertyChangeEvent;
|
||||
import java.beans.PropertyChangeListener;
|
||||
import java.io.File;
|
||||
import java.util.Collection;
|
||||
|
||||
import javax.swing.JComponent;
|
||||
|
||||
import net.sourceforge.tuned.ui.LoadingOverlayPanel;
|
||||
|
||||
|
||||
public abstract class ToolPanel extends JComponent {
|
||||
|
||||
private String name = null;
|
||||
|
||||
private LoadingOverlayPanel loadingOverlay;
|
||||
|
||||
public static final String LOADING_PROPERTY = "loading";
|
||||
|
||||
|
||||
public ToolPanel(String name) {
|
||||
this.name = name;
|
||||
addPropertyChangeListener(LOADING_PROPERTY, loadingOverlayUpdateListener);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
|
||||
protected void setLoadingOverlayPane(LoadingOverlayPanel c) {
|
||||
loadingOverlay = c;
|
||||
}
|
||||
|
||||
private PropertyChangeListener loadingOverlayUpdateListener = new PropertyChangeListener() {
|
||||
|
||||
public void propertyChange(PropertyChangeEvent evt) {
|
||||
Boolean loading = (Boolean) evt.getNewValue();
|
||||
|
||||
if (loadingOverlay == null)
|
||||
return;
|
||||
|
||||
loadingOverlay.setOverlayVisible(loading);
|
||||
loadingOverlay.updateOverlayUI();
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
public abstract void update(Collection<File> list);
|
||||
}
|
@ -0,0 +1,134 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.analyze.tools;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.io.File;
|
||||
import java.util.Collection;
|
||||
import java.util.Iterator;
|
||||
import java.util.LinkedList;
|
||||
import java.util.TreeMap;
|
||||
|
||||
import javax.swing.BorderFactory;
|
||||
import javax.swing.JScrollPane;
|
||||
import javax.swing.SwingWorker;
|
||||
import javax.swing.tree.DefaultMutableTreeNode;
|
||||
import javax.swing.tree.DefaultTreeModel;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.filebot.ui.FileBotTree;
|
||||
import net.sourceforge.filebot.ui.FileFormat;
|
||||
import net.sourceforge.tuned.ui.LoadingOverlayPanel;
|
||||
|
||||
|
||||
public class TypePanel extends ToolPanel {
|
||||
|
||||
private FileBotTree tree = new FileBotTree();
|
||||
|
||||
|
||||
public TypePanel() {
|
||||
super("Types");
|
||||
setLayout(new BorderLayout());
|
||||
|
||||
JScrollPane sp = new JScrollPane(tree);
|
||||
sp.setBorder(BorderFactory.createEmptyBorder());
|
||||
LoadingOverlayPanel loadingOverlay = new LoadingOverlayPanel(sp, ResourceManager.getIcon("loading"));
|
||||
add(loadingOverlay, BorderLayout.CENTER);
|
||||
|
||||
setLoadingOverlayPane(loadingOverlay);
|
||||
}
|
||||
|
||||
UpdateTask latestUpdateTask;
|
||||
|
||||
|
||||
@Override
|
||||
public void update(Collection<File> files) {
|
||||
latestUpdateTask = new UpdateTask(files);
|
||||
|
||||
firePropertyChange(LOADING_PROPERTY, null, true);
|
||||
latestUpdateTask.execute();
|
||||
}
|
||||
|
||||
|
||||
private class UpdateTask extends SwingWorker<DefaultTreeModel, Object> {
|
||||
|
||||
private Collection<File> files;
|
||||
|
||||
|
||||
public UpdateTask(Collection<File> files) {
|
||||
this.files = files;
|
||||
}
|
||||
|
||||
|
||||
private boolean isLatest() {
|
||||
if (this == latestUpdateTask)
|
||||
return true;
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected DefaultTreeModel doInBackground() throws Exception {
|
||||
TreeMap<String, Collection<File>> map = new TreeMap<String, Collection<File>>();
|
||||
|
||||
for (File f : files) {
|
||||
String suffix = FileFormat.getSuffix(f);
|
||||
|
||||
Collection<File> list = map.get(suffix);
|
||||
|
||||
if (list != null)
|
||||
list.add(f);
|
||||
else {
|
||||
list = new LinkedList<File>();
|
||||
list.add(f);
|
||||
map.put(suffix, list);
|
||||
}
|
||||
|
||||
if (!isLatest())
|
||||
return null;
|
||||
}
|
||||
|
||||
DefaultMutableTreeNode root = new DefaultMutableTreeNode();
|
||||
Iterator<String> i = map.keySet().iterator();
|
||||
|
||||
while (i.hasNext()) {
|
||||
String key = i.next();
|
||||
Collection<File> list = map.get(key);
|
||||
DefaultMutableTreeNode node = new DefaultMutableTreeNode();
|
||||
long size = 0;
|
||||
|
||||
for (File f : list) {
|
||||
node.add(new DefaultMutableTreeNode(f));
|
||||
size += f.length();
|
||||
}
|
||||
|
||||
node.setUserObject(key + " (" + FileFormat.formatNumberOfFiles(list.size()) + ", " + FileFormat.formatSize(size) + ")");
|
||||
root.add(node);
|
||||
|
||||
if (!isLatest())
|
||||
return null;
|
||||
}
|
||||
|
||||
return new DefaultTreeModel(root);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void done() {
|
||||
try {
|
||||
DefaultTreeModel model = get();
|
||||
|
||||
if (model == null)
|
||||
return;
|
||||
|
||||
tree.setModel(model);
|
||||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
TypePanel.this.firePropertyChange(LOADING_PROPERTY, null, false);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,33 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.create;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
|
||||
import javax.swing.Box;
|
||||
import javax.swing.JButton;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
|
||||
import net.sourceforge.filebot.ui.FileBotList;
|
||||
import net.sourceforge.filebot.ui.sal.SaveAction;
|
||||
|
||||
|
||||
public class CreateList extends FileBotList {
|
||||
|
||||
private SaveAction saveAction = new SaveAction(this);
|
||||
|
||||
|
||||
public CreateList() {
|
||||
super(false, true, true);
|
||||
|
||||
setTitle("List");
|
||||
|
||||
Box buttons = Box.createHorizontalBox();
|
||||
buttons.setBorder(new EmptyBorder(5, 5, 5, 5));
|
||||
buttons.add(Box.createGlue());
|
||||
buttons.add(new JButton(saveAction));
|
||||
buttons.add(Box.createGlue());
|
||||
|
||||
add(buttons, BorderLayout.SOUTH);
|
||||
}
|
||||
}
|
108
source/net/sourceforge/filebot/ui/panel/create/CreatePanel.java
Normal file
@ -0,0 +1,108 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.create;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
import java.awt.Dimension;
|
||||
import java.awt.event.ActionEvent;
|
||||
import java.text.NumberFormat;
|
||||
|
||||
import javax.swing.AbstractAction;
|
||||
import javax.swing.Box;
|
||||
import javax.swing.JButton;
|
||||
import javax.swing.JComponent;
|
||||
import javax.swing.JLabel;
|
||||
import javax.swing.JPanel;
|
||||
import javax.swing.JSpinner;
|
||||
import javax.swing.JTextField;
|
||||
import javax.swing.KeyStroke;
|
||||
import javax.swing.SpinnerNumberModel;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
|
||||
import net.sourceforge.filebot.resources.ResourceManager;
|
||||
import net.sourceforge.filebot.ui.FileBotPanel;
|
||||
import net.sourceforge.filebot.ui.FileBotUtil;
|
||||
|
||||
|
||||
public class CreatePanel extends FileBotPanel {
|
||||
|
||||
private SpinnerNumberModel fromSpinnerModel = new SpinnerNumberModel(1, 1, Integer.MAX_VALUE, 1);
|
||||
private SpinnerNumberModel toSpinnerModel = new SpinnerNumberModel(20, 1, Integer.MAX_VALUE, 1);
|
||||
|
||||
private CreateList list = new CreateList();
|
||||
|
||||
private JTextField textField = new JTextField("Name - $i", 25);
|
||||
|
||||
|
||||
public CreatePanel() {
|
||||
super("Create", ResourceManager.getIcon("panel.create"));
|
||||
|
||||
JSpinner fromSpinner = new JSpinner(fromSpinnerModel);
|
||||
JSpinner toSpinner = new JSpinner(toSpinnerModel);
|
||||
|
||||
fromSpinner.setEditor(new JSpinner.NumberEditor(fromSpinner, "#"));
|
||||
toSpinner.setEditor(new JSpinner.NumberEditor(toSpinner, "#"));
|
||||
|
||||
Dimension spinnerDimension = new Dimension(50, textField.getPreferredSize().height);
|
||||
fromSpinner.setPreferredSize(spinnerDimension);
|
||||
toSpinner.setPreferredSize(spinnerDimension);
|
||||
|
||||
Box spinners = Box.createHorizontalBox();
|
||||
spinners.setBorder(new EmptyBorder(5, 5, 5, 5));
|
||||
spinners.add(Box.createHorizontalGlue());
|
||||
|
||||
spinners.add(createLabeledComponent("Pattern:", textField));
|
||||
spinners.add(Box.createHorizontalStrut(15));
|
||||
spinners.add(createLabeledComponent("From:", fromSpinner));
|
||||
spinners.add(Box.createHorizontalStrut(10));
|
||||
spinners.add(createLabeledComponent("To:", toSpinner));
|
||||
spinners.add(Box.createHorizontalStrut(15));
|
||||
spinners.add(new JButton(createAction));
|
||||
spinners.add(Box.createHorizontalGlue());
|
||||
|
||||
JPanel panel = new JPanel(new BorderLayout());
|
||||
|
||||
panel.add(spinners, BorderLayout.NORTH);
|
||||
panel.add(list, BorderLayout.CENTER);
|
||||
|
||||
add(panel, BorderLayout.CENTER);
|
||||
|
||||
FileBotUtil.registerActionForKeystroke(this, KeyStroke.getKeyStroke("ENTER"), createAction);
|
||||
}
|
||||
|
||||
|
||||
private JComponent createLabeledComponent(String label, JComponent component) {
|
||||
Box box = Box.createHorizontalBox();
|
||||
box.setBorder(new EmptyBorder(5, 5, 5, 5));
|
||||
box.add(new JLabel(label));
|
||||
box.add(Box.createHorizontalStrut(6));
|
||||
box.add(component);
|
||||
|
||||
box.setMaximumSize(box.getPreferredSize());
|
||||
|
||||
return box;
|
||||
}
|
||||
|
||||
private AbstractAction createAction = new AbstractAction("Create") {
|
||||
|
||||
public void actionPerformed(ActionEvent e) {
|
||||
list.getModel().clear();
|
||||
|
||||
int from = fromSpinnerModel.getNumber().intValue();
|
||||
int to = toSpinnerModel.getNumber().intValue();
|
||||
|
||||
NumberFormat format = NumberFormat.getInstance();
|
||||
format.setGroupingUsed(false);
|
||||
|
||||
format.setMinimumIntegerDigits(Math.max(Integer.toString(to).length(), 2));
|
||||
|
||||
String pattern = textField.getText();
|
||||
|
||||
for (int i = from; i <= to; i++) {
|
||||
String s = pattern.replaceAll("\\$i", format.format(i));
|
||||
list.getModel().addElement(s);
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
}
|
42
source/net/sourceforge/filebot/ui/panel/list/FileList.java
Normal file
@ -0,0 +1,42 @@
|
||||
|
||||
package net.sourceforge.filebot.ui.panel.list;
|
||||
|
||||
|
||||
import java.awt.BorderLayout;
|
||||
|
||||
import javax.swing.Box;
|
||||
import javax.swing.JButton;
|
||||
import javax.swing.border.EmptyBorder;
|
||||
|
||||
import net.sourceforge.filebot.ui.FileBotList;
|
||||
import net.sourceforge.filebot.ui.sal.LoadAction;
|
||||
import net.sourceforge.filebot.ui.sal.SaveAction;
|
||||
|
||||
|
||||
public class FileList extends FileBotList {
|
||||
|
||||
private SaveAction saveAction = new SaveAction(this);
|
||||
|
||||
private LoadAction loadAction = new LoadAction(this);
|
||||
|
||||
|
||||
public FileList() {
|
||||
super(true, true, true);
|
||||
|
||||
setTransferablePolicy(new FileListTransferablePolicy(this));
|
||||
|
||||
setTitle("Folder");
|
||||
|
||||
Box buttons = Box.createHorizontalBox();
|
||||
buttons.setBorder(new EmptyBorder(5, 5, 5, 5));
|
||||
buttons.add(Box.createHorizontalGlue());
|
||||
|
||||
buttons.add(new JButton(loadAction));
|
||||
buttons.add(Box.createHorizontalStrut(5));
|
||||
buttons.add(new JButton(saveAction));
|
||||
buttons.add(Box.createHorizontalGlue());
|
||||
|
||||
add(buttons, BorderLayout.SOUTH);
|
||||
}
|
||||
|
||||
}
|