mirror of
https://github.com/Maschell/JNUSTool.git
synced 2025-01-23 05:31:17 +01:00
Added simple GUI
-added simple GUI -fixed download of some meta files
This commit is contained in:
parent
d83f486fa3
commit
20813d4430
92
src/Directory.java
Normal file
92
src/Directory.java
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
import java.util.Collection;
|
||||||
|
import java.util.TreeMap;
|
||||||
|
|
||||||
|
import javax.swing.tree.DefaultMutableTreeNode;
|
||||||
|
|
||||||
|
public class Directory {
|
||||||
|
String name = "";
|
||||||
|
TreeMap<String,Directory> folder = new TreeMap<>();
|
||||||
|
TreeMap<String,FEntry> files = new TreeMap<>();
|
||||||
|
|
||||||
|
public Directory get(String s){
|
||||||
|
return folder.get(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Directory(String name){
|
||||||
|
setName(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean containsFolder(String s){
|
||||||
|
return folder.containsKey(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Directory getFolder(String s){
|
||||||
|
return folder.get(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Directory addFolder(Directory s){
|
||||||
|
return folder.put(s.getName(),s);
|
||||||
|
}
|
||||||
|
public boolean containsFile(String s){
|
||||||
|
return files.containsKey(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public FEntry getFile(String s){
|
||||||
|
return files.get(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public FEntry addFile(FEntry s){
|
||||||
|
return files.put(s.getFileName(),s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public Collection<Directory> getFolder() {
|
||||||
|
return folder.values();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public Collection<FEntry> getFiles() {
|
||||||
|
return files.values();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFiles(TreeMap<String, FEntry> files) {
|
||||||
|
this.files = files;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString(){
|
||||||
|
System.out.println(name + ":");
|
||||||
|
for(Directory d : folder.values()){
|
||||||
|
System.out.println(d);
|
||||||
|
}
|
||||||
|
for(String s : files.keySet()){
|
||||||
|
System.out.println(s);
|
||||||
|
}
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
public DefaultMutableTreeNode getNodes(){
|
||||||
|
DefaultMutableTreeNode node = new DefaultMutableTreeNode(getName());
|
||||||
|
|
||||||
|
for(Directory f: getFolder()){
|
||||||
|
node.add(f.getNodes());
|
||||||
|
}
|
||||||
|
|
||||||
|
for(FEntry f: getFiles()){
|
||||||
|
node.add(new DefaultMutableTreeNode(f));
|
||||||
|
}
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
@ -30,7 +30,6 @@ public class Downloader {
|
|||||||
dlFileLength = ((dlFileLength/BLOCKSIZE)*BLOCKSIZE) +BLOCKSIZE;
|
dlFileLength = ((dlFileLength/BLOCKSIZE)*BLOCKSIZE) +BLOCKSIZE;
|
||||||
}
|
}
|
||||||
|
|
||||||
fileOffset = ((fileOffset / BLOCKSIZE) * BLOCKSIZE);
|
|
||||||
connection.setRequestProperty("Range", "bytes=" + fileOffset+"-");
|
connection.setRequestProperty("Range", "bytes=" + fileOffset+"-");
|
||||||
connection.connect();
|
connection.connect();
|
||||||
|
|
||||||
@ -235,17 +234,21 @@ public class Downloader {
|
|||||||
String [] path = toDownload.getFullPath().split("/");
|
String [] path = toDownload.getFullPath().split("/");
|
||||||
|
|
||||||
String folder = String.format("%016X", titleID) +"/";
|
String folder = String.format("%016X", titleID) +"/";
|
||||||
|
File folder_ = null;
|
||||||
for(int i = 0;i<path.length-1;i++){
|
for(int i = 0;i<path.length-1;i++){
|
||||||
if(!path[i].equals("")){
|
if(!path[i].equals("")){
|
||||||
folder += path[i] + "/";
|
folder += path[i] + "/";
|
||||||
|
folder_ = new File(folder);
|
||||||
|
if(!folder_.exists()){
|
||||||
|
folder_.mkdir();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
File folder_ = new File(folder);
|
}
|
||||||
if(!folder_.exists()) folder_.mkdir();
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
//if(toDownload.isExtractWithHash()){
|
//if(toDownload.isExtractWithHash()){
|
||||||
if(path[1].equals("content")){
|
if(!path[1].equals("code") && toDownload.isExtractWithHash()){
|
||||||
downloadAndDecryptHash(URL,toDownload.getFileOffset(),toDownload.getFileLength(),toDownload,ticket);
|
downloadAndDecryptHash(URL,toDownload.getFileOffset(),toDownload.getFileLength(),toDownload,ticket);
|
||||||
}else{
|
}else{
|
||||||
downloadAndDecrypt(URL,toDownload.getFileOffset(),toDownload.getFileLength(),toDownload,ticket);
|
downloadAndDecrypt(URL,toDownload.getFileOffset(),toDownload.getFileLength(),toDownload,ticket);
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import java.util.List;
|
||||||
|
|
||||||
public class FEntry {
|
public class FEntry {
|
||||||
|
|
||||||
@ -16,10 +17,11 @@ public class FEntry {
|
|||||||
private long fileLength = 0;
|
private long fileLength = 0;
|
||||||
private int contentID = 0;
|
private int contentID = 0;
|
||||||
private int NUScontentID = 0;
|
private int NUScontentID = 0;
|
||||||
|
private List<String> pathList;
|
||||||
|
|
||||||
|
|
||||||
public FEntry(String path, String filename, int contentID,int NUScontentID, long fileOffset, long fileLength, boolean dir,
|
public FEntry(String path, String filename, int contentID,int NUScontentID, long fileOffset, long fileLength, boolean dir,
|
||||||
boolean in_nus_title, boolean extract_withHash) {
|
boolean in_nus_title, boolean extract_withHash, List<String> pathList) {
|
||||||
setPath(path);
|
setPath(path);
|
||||||
setFileName(filename);
|
setFileName(filename);
|
||||||
setContentID(contentID);
|
setContentID(contentID);
|
||||||
@ -29,7 +31,7 @@ public class FEntry {
|
|||||||
setInNusTitle(in_nus_title);
|
setInNusTitle(in_nus_title);
|
||||||
setExtractWithHash(extract_withHash);
|
setExtractWithHash(extract_withHash);
|
||||||
setNUScontentID(NUScontentID);
|
setNUScontentID(NUScontentID);
|
||||||
|
setPathList(pathList);
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean isDir() {
|
public boolean isDir() {
|
||||||
@ -118,5 +120,16 @@ public class FEntry {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public List<String> getPathList() {
|
||||||
|
return pathList;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPathList(List<String> pathList) {
|
||||||
|
this.pathList = pathList;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
46
src/FST.java
46
src/FST.java
@ -14,10 +14,36 @@ public class FST {
|
|||||||
int totalEntries = 0;
|
int totalEntries = 0;
|
||||||
int dirEntries = 0;
|
int dirEntries = 0;
|
||||||
|
|
||||||
|
private Directory directory = new Directory("root");
|
||||||
|
|
||||||
public FST(byte[] decrypteddata, TitleMetaData tmd) throws IOException {
|
public FST(byte[] decrypteddata, TitleMetaData tmd) throws IOException {
|
||||||
parse(decrypteddata,tmd);
|
parse(decrypteddata,tmd);
|
||||||
|
buildDirectory();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void buildDirectory() {
|
||||||
|
for(FEntry f : getFileEntries()){
|
||||||
|
Directory current = directory;
|
||||||
|
int i = 0;
|
||||||
|
for(String s :f.getPathList()){
|
||||||
|
|
||||||
|
if(current.containsFolder(s)){
|
||||||
|
current = current.get(s);
|
||||||
|
}else{
|
||||||
|
Directory newDir = new Directory(s);
|
||||||
|
current.addFolder(newDir);
|
||||||
|
current = newDir;
|
||||||
|
}
|
||||||
|
i++;
|
||||||
|
if(i==f.getPathList().size()){
|
||||||
|
current.addFile(f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
private void parse(byte[] decrypteddata, TitleMetaData tmd) throws IOException {
|
private void parse(byte[] decrypteddata, TitleMetaData tmd) throws IOException {
|
||||||
|
|
||||||
@ -99,7 +125,7 @@ public class FST {
|
|||||||
this.totalContentSize += fileLength;
|
this.totalContentSize += fileLength;
|
||||||
if(in_nus_title)this.totalContentSizeInNUS += fileLength;
|
if(in_nus_title)this.totalContentSizeInNUS += fileLength;
|
||||||
|
|
||||||
|
List<String> pathList = new ArrayList<>();
|
||||||
//getting the full path of entry
|
//getting the full path of entry
|
||||||
if(dir)
|
if(dir)
|
||||||
{
|
{
|
||||||
@ -114,20 +140,28 @@ public class FST {
|
|||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
int k = 0;
|
int k = 0;
|
||||||
int nameoffoff,nameoff_entrypath;
|
int nameoffoff,nameoff_entrypath;
|
||||||
|
|
||||||
for( j=0; j<level; ++j )
|
for( j=0; j<level; ++j )
|
||||||
{
|
{
|
||||||
nameoffoff = Util.getIntFromBytes(decrypteddata,base_offset+Entry[j]*0x10);
|
nameoffoff = Util.getIntFromBytes(decrypteddata,base_offset+Entry[j]*0x10);
|
||||||
k=0;
|
k=0;
|
||||||
nameoff_entrypath = nameOff + nameoffoff;
|
nameoff_entrypath = nameOff + nameoffoff;
|
||||||
while(decrypteddata[nameoff_entrypath + k] != 0){k++;}
|
while(decrypteddata[nameoff_entrypath + k] != 0){k++;}
|
||||||
sb.append(new String(Arrays.copyOfRange(decrypteddata,nameoff_entrypath, nameoff_entrypath + k)));
|
String tmpname = new String(Arrays.copyOfRange(decrypteddata,nameoff_entrypath, nameoff_entrypath + k));
|
||||||
|
if(!tmpname.equals("")){
|
||||||
|
pathList.add(tmpname);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
sb.append(tmpname);
|
||||||
sb.append("/");
|
sb.append("/");
|
||||||
}
|
}
|
||||||
path = sb.toString();
|
path = sb.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
//add this to the List!
|
//add this to the List!
|
||||||
fileEntries.add(new FEntry(path,filename,contentID,tmd.contents[contentID].ID,fileOffset,fileLength,dir,in_nus_title,extract_withHash));
|
fileEntries.add(new FEntry(path,filename,contentID,tmd.contents[contentID].ID,fileOffset,fileLength,dir,in_nus_title,extract_withHash,pathList));
|
||||||
//System.out.println(fileEntries.get(i));
|
//System.out.println(fileEntries.get(i));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -218,4 +252,10 @@ public class FST {
|
|||||||
}
|
}
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Directory getDirectory() {
|
||||||
|
return directory;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
274
src/JCheckBoxTree.java
Normal file
274
src/JCheckBoxTree.java
Normal file
@ -0,0 +1,274 @@
|
|||||||
|
/**
|
||||||
|
* Based on
|
||||||
|
* http://stackoverflow.com/questions/21847411/java-swing-need-a-good-quality-developed-jtree-with-checkboxes/21851201#21851201
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.awt.BorderLayout;
|
||||||
|
import java.awt.Component;
|
||||||
|
import java.awt.event.MouseEvent;
|
||||||
|
import java.awt.event.MouseListener;
|
||||||
|
import java.util.EventListener;
|
||||||
|
import java.util.EventObject;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
|
||||||
|
import javax.swing.JCheckBox;
|
||||||
|
import javax.swing.JPanel;
|
||||||
|
import javax.swing.JTree;
|
||||||
|
import javax.swing.event.EventListenerList;
|
||||||
|
import javax.swing.tree.DefaultMutableTreeNode;
|
||||||
|
import javax.swing.tree.DefaultTreeModel;
|
||||||
|
import javax.swing.tree.DefaultTreeSelectionModel;
|
||||||
|
import javax.swing.tree.TreeCellRenderer;
|
||||||
|
import javax.swing.tree.TreeModel;
|
||||||
|
import javax.swing.tree.TreeNode;
|
||||||
|
import javax.swing.tree.TreePath;
|
||||||
|
|
||||||
|
public class JCheckBoxTree extends JTree {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = -4194122328392241790L;
|
||||||
|
|
||||||
|
JCheckBoxTree selfPointer = this;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Defining data structure that will enable to fast check-indicate the state of each node
|
||||||
|
// It totally replaces the "selection" mechanism of the JTree
|
||||||
|
private class CheckedNode {
|
||||||
|
boolean isSelected;
|
||||||
|
boolean hasChildren;
|
||||||
|
boolean allChildrenSelected;
|
||||||
|
|
||||||
|
public CheckedNode(boolean isSelected_, boolean hasChildren_, boolean allChildrenSelected_) {
|
||||||
|
isSelected = isSelected_;
|
||||||
|
hasChildren = hasChildren_;
|
||||||
|
allChildrenSelected = allChildrenSelected_;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
HashMap<TreePath, CheckedNode> nodesCheckingState;
|
||||||
|
HashSet<TreePath> checkedPaths = new HashSet<TreePath>();
|
||||||
|
|
||||||
|
// Defining a new event type for the checking mechanism and preparing event-handling mechanism
|
||||||
|
protected EventListenerList listenerList = new EventListenerList();
|
||||||
|
|
||||||
|
public class CheckChangeEvent extends EventObject {
|
||||||
|
private static final long serialVersionUID = -8100230309044193368L;
|
||||||
|
|
||||||
|
public CheckChangeEvent(Object source) {
|
||||||
|
super(source);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface CheckChangeEventListener extends EventListener {
|
||||||
|
public void checkStateChanged(CheckChangeEvent event);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addCheckChangeEventListener(CheckChangeEventListener listener) {
|
||||||
|
listenerList.add(CheckChangeEventListener.class, listener);
|
||||||
|
}
|
||||||
|
public void removeCheckChangeEventListener(CheckChangeEventListener listener) {
|
||||||
|
listenerList.remove(CheckChangeEventListener.class, listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
void fireCheckChangeEvent(CheckChangeEvent evt) {
|
||||||
|
Object[] listeners = listenerList.getListenerList();
|
||||||
|
for (int i = 0; i < listeners.length; i++) {
|
||||||
|
if (listeners[i] == CheckChangeEventListener.class) {
|
||||||
|
((CheckChangeEventListener) listeners[i + 1]).checkStateChanged(evt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Override
|
||||||
|
public void setModel(TreeModel newModel) {
|
||||||
|
super.setModel(newModel);
|
||||||
|
resetCheckingState();
|
||||||
|
}
|
||||||
|
|
||||||
|
// New method that returns only the checked paths (totally ignores original "selection" mechanism)
|
||||||
|
public TreePath[] getCheckedPaths() {
|
||||||
|
return checkedPaths.toArray(new TreePath[checkedPaths.size()]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns true in case that the node is selected, has children but not all of them are selected
|
||||||
|
public boolean isSelectedPartially(TreePath path) {
|
||||||
|
CheckedNode cn = nodesCheckingState.get(path);
|
||||||
|
return cn.isSelected && cn.hasChildren && !cn.allChildrenSelected;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void resetCheckingState() {
|
||||||
|
nodesCheckingState = new HashMap<TreePath, CheckedNode>();
|
||||||
|
checkedPaths = new HashSet<TreePath>();
|
||||||
|
DefaultMutableTreeNode node = (DefaultMutableTreeNode)getModel().getRoot();
|
||||||
|
if (node == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
addSubtreeToCheckingStateTracking(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Creating data structure of the current model for the checking mechanism
|
||||||
|
private void addSubtreeToCheckingStateTracking(DefaultMutableTreeNode node) {
|
||||||
|
TreeNode[] path = node.getPath();
|
||||||
|
TreePath tp = new TreePath(path);
|
||||||
|
CheckedNode cn = new CheckedNode(false, node.getChildCount() > 0, false);
|
||||||
|
nodesCheckingState.put(tp, cn);
|
||||||
|
for (int i = 0 ; i < node.getChildCount() ; i++) {
|
||||||
|
addSubtreeToCheckingStateTracking((DefaultMutableTreeNode) tp.pathByAddingChild(node.getChildAt(i)).getLastPathComponent());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Overriding cell renderer by a class that ignores the original "selection" mechanism
|
||||||
|
// It decides how to show the nodes due to the checking-mechanism
|
||||||
|
private class CheckBoxCellRenderer extends JPanel implements TreeCellRenderer {
|
||||||
|
private static final long serialVersionUID = -7341833835878991719L;
|
||||||
|
JCheckBox checkBox;
|
||||||
|
public CheckBoxCellRenderer() {
|
||||||
|
super();
|
||||||
|
this.setLayout(new BorderLayout());
|
||||||
|
checkBox = new JCheckBox();
|
||||||
|
add(checkBox, BorderLayout.CENTER);
|
||||||
|
setOpaque(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Component getTreeCellRendererComponent(JTree tree, Object value,
|
||||||
|
boolean selected, boolean expanded, boolean leaf, int row,
|
||||||
|
boolean hasFocus) {
|
||||||
|
DefaultMutableTreeNode node = (DefaultMutableTreeNode)value;
|
||||||
|
Object obj = node.getUserObject();
|
||||||
|
TreePath tp = new TreePath(node.getPath());
|
||||||
|
CheckedNode cn = nodesCheckingState.get(tp);
|
||||||
|
if (cn == null) {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
checkBox.setSelected(cn.isSelected);
|
||||||
|
if(obj instanceof FEntry){
|
||||||
|
FEntry f = (FEntry) obj;
|
||||||
|
checkBox.setText(f.getFileName());
|
||||||
|
}else{
|
||||||
|
checkBox.setText(obj.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
checkBox.setOpaque(cn.isSelected && cn.hasChildren && ! cn.allChildrenSelected);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public JCheckBoxTree(NUSTitle nus) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
parent.add(new DefaultMutableTreeNode("hot dogs"));
|
||||||
|
parent.add(new DefaultMutableTreeNode("pizza"));
|
||||||
|
parent.add(new DefaultMutableTreeNode("ravioli"));
|
||||||
|
parent.add(new DefaultMutableTreeNode("bananas"));*/
|
||||||
|
//return ;
|
||||||
|
|
||||||
|
setModel(new DefaultTreeModel(nus.getFst().getDirectory().getNodes()));
|
||||||
|
|
||||||
|
// Disabling toggling by double-click
|
||||||
|
this.setToggleClickCount(0);
|
||||||
|
// Overriding cell renderer by new one defined above
|
||||||
|
CheckBoxCellRenderer cellRenderer = new CheckBoxCellRenderer();
|
||||||
|
this.setCellRenderer(cellRenderer);
|
||||||
|
|
||||||
|
// Overriding selection model by an empty one
|
||||||
|
DefaultTreeSelectionModel dtsm = new DefaultTreeSelectionModel() {
|
||||||
|
private static final long serialVersionUID = -8190634240451667286L;
|
||||||
|
// Totally disabling the selection mechanism
|
||||||
|
public void setSelectionPath(TreePath path) {
|
||||||
|
}
|
||||||
|
public void addSelectionPath(TreePath path) {
|
||||||
|
}
|
||||||
|
public void removeSelectionPath(TreePath path) {
|
||||||
|
}
|
||||||
|
public void setSelectionPaths(TreePath[] pPaths) {
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Calling checking mechanism on mouse click
|
||||||
|
this.addMouseListener(new MouseListener() {
|
||||||
|
public void mouseClicked(MouseEvent arg0) {
|
||||||
|
TreePath tp = selfPointer.getPathForLocation(arg0.getX(), arg0.getY());
|
||||||
|
if (tp == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
boolean checkMode = ! nodesCheckingState.get(tp).isSelected;
|
||||||
|
checkSubTree(tp, checkMode);
|
||||||
|
updatePredecessorsWithCheckMode(tp, checkMode);
|
||||||
|
// Firing the check change event
|
||||||
|
fireCheckChangeEvent(new CheckChangeEvent(new Object()));
|
||||||
|
// Repainting tree after the data structures were updated
|
||||||
|
selfPointer.repaint();
|
||||||
|
}
|
||||||
|
public void mouseEntered(MouseEvent arg0) {
|
||||||
|
}
|
||||||
|
public void mouseExited(MouseEvent arg0) {
|
||||||
|
}
|
||||||
|
public void mousePressed(MouseEvent arg0) {
|
||||||
|
}
|
||||||
|
public void mouseReleased(MouseEvent arg0) {
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
this.setSelectionModel(dtsm);
|
||||||
|
}
|
||||||
|
|
||||||
|
// When a node is checked/unchecked, updating the states of the predecessors
|
||||||
|
protected void updatePredecessorsWithCheckMode(TreePath tp, boolean check) {
|
||||||
|
TreePath parentPath = tp.getParentPath();
|
||||||
|
// If it is the root, stop the recursive calls and return
|
||||||
|
if (parentPath == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
CheckedNode parentCheckedNode = nodesCheckingState.get(parentPath);
|
||||||
|
DefaultMutableTreeNode parentNode = (DefaultMutableTreeNode) parentPath.getLastPathComponent();
|
||||||
|
parentCheckedNode.allChildrenSelected = true;
|
||||||
|
parentCheckedNode.isSelected = false;
|
||||||
|
for (int i = 0 ; i < parentNode.getChildCount() ; i++) {
|
||||||
|
TreePath childPath = parentPath.pathByAddingChild(parentNode.getChildAt(i));
|
||||||
|
CheckedNode childCheckedNode = nodesCheckingState.get(childPath);
|
||||||
|
// It is enough that even one subtree is not fully selected
|
||||||
|
// to determine that the parent is not fully selected
|
||||||
|
if (! childCheckedNode.allChildrenSelected) {
|
||||||
|
parentCheckedNode.allChildrenSelected = false;
|
||||||
|
}
|
||||||
|
// If at least one child is selected, selecting also the parent
|
||||||
|
if (childCheckedNode.isSelected) {
|
||||||
|
parentCheckedNode.isSelected = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (parentCheckedNode.isSelected) {
|
||||||
|
checkedPaths.add(parentPath);
|
||||||
|
} else {
|
||||||
|
checkedPaths.remove(parentPath);
|
||||||
|
}
|
||||||
|
// Go to upper predecessor
|
||||||
|
updatePredecessorsWithCheckMode(parentPath, check);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Recursively checks/unchecks a subtree
|
||||||
|
protected void checkSubTree(TreePath tp, boolean check) {
|
||||||
|
CheckedNode cn = nodesCheckingState.get(tp);
|
||||||
|
cn.isSelected = check;
|
||||||
|
DefaultMutableTreeNode node = (DefaultMutableTreeNode) tp.getLastPathComponent();
|
||||||
|
for (int i = 0 ; i < node.getChildCount() ; i++) {
|
||||||
|
checkSubTree(tp.pathByAddingChild(node.getChildAt(i)), check);
|
||||||
|
}
|
||||||
|
cn.allChildrenSelected = check;
|
||||||
|
if (check) {
|
||||||
|
checkedPaths.add(tp);
|
||||||
|
} else {
|
||||||
|
checkedPaths.remove(tp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
70
src/NUSGUI.java
Normal file
70
src/NUSGUI.java
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
import java.awt.BorderLayout;
|
||||||
|
import java.awt.event.ActionEvent;
|
||||||
|
import java.awt.event.ActionListener;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.ForkJoinPool;
|
||||||
|
|
||||||
|
import javax.swing.JButton;
|
||||||
|
import javax.swing.JFrame;
|
||||||
|
import javax.swing.JScrollPane;
|
||||||
|
import javax.swing.tree.DefaultMutableTreeNode;
|
||||||
|
import javax.swing.tree.TreePath;
|
||||||
|
|
||||||
|
public class NUSGUI extends JFrame {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = 4648172894076113183L;
|
||||||
|
|
||||||
|
public NUSGUI(NUSTitle nus) {
|
||||||
|
super();
|
||||||
|
setSize(800, 600);
|
||||||
|
getContentPane().setLayout(new BorderLayout(0, 0));
|
||||||
|
|
||||||
|
final JCheckBoxTree cbt = new JCheckBoxTree(nus);
|
||||||
|
JScrollPane qPane = new JScrollPane(cbt,
|
||||||
|
JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED,
|
||||||
|
JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
|
||||||
|
this.getContentPane().add(qPane);
|
||||||
|
|
||||||
|
|
||||||
|
JButton btnNewButton = new JButton("Download");
|
||||||
|
btnNewButton.addActionListener(new ActionListener() {
|
||||||
|
public void actionPerformed(ActionEvent e) {
|
||||||
|
new Thread(new Runnable() { public void run() {
|
||||||
|
ForkJoinPool pool = ForkJoinPool.commonPool();
|
||||||
|
List<TitleDownloader> list = new ArrayList<>();
|
||||||
|
|
||||||
|
|
||||||
|
TreePath[] paths = cbt.getCheckedPaths();
|
||||||
|
for (TreePath tp : paths) {
|
||||||
|
Object obj = tp.getPath()[tp.getPath().length-1];
|
||||||
|
if(((DefaultMutableTreeNode)obj).getUserObject() instanceof FEntry){
|
||||||
|
FEntry f = (FEntry) ((DefaultMutableTreeNode)obj).getUserObject();
|
||||||
|
if(!f.isDir() && f.isInNUSTitle())
|
||||||
|
list.add(new TitleDownloader(f));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pool.invokeAll(list);
|
||||||
|
System.out.println("Done!");
|
||||||
|
}}).start();
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
getContentPane().add(btnNewButton, BorderLayout.SOUTH);
|
||||||
|
|
||||||
|
/*cbt.addCheckChangeEventListener(new JCheckBoxTree.CheckChangeEventListener() {
|
||||||
|
public void checkStateChanged(JCheckBoxTree.CheckChangeEvent event) {
|
||||||
|
System.out.println("event");
|
||||||
|
TreePath[] paths = cbt.getCheckedPaths();
|
||||||
|
for (TreePath tp : paths) {
|
||||||
|
for (Object pathPart : tp.getPath()) {
|
||||||
|
System.out.print(pathPart + ",");
|
||||||
|
}
|
||||||
|
System.out.println();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});*/
|
||||||
|
|
||||||
|
this.setDefaultCloseOperation(EXIT_ON_CLOSE);
|
||||||
|
}
|
||||||
|
}
|
@ -1,14 +1,10 @@
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Date;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.concurrent.ForkJoinPool;
|
|
||||||
|
|
||||||
public class NUSTitle {
|
public class NUSTitle {
|
||||||
private TitleMetaData tmd;
|
private TitleMetaData tmd;
|
||||||
private long titleID;
|
private long titleID;
|
||||||
private TIK ticket;
|
private TIK ticket;
|
||||||
|
private FST fst;
|
||||||
public NUSTitle(long titleId,String key){
|
public NUSTitle(long titleId,String key){
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@ -31,33 +27,13 @@ public class NUSTitle {
|
|||||||
byte[] encryptedFST = Downloader.getInstance().downloadContentToByteArray(tmd.contents[0].ID);
|
byte[] encryptedFST = Downloader.getInstance().downloadContentToByteArray(tmd.contents[0].ID);
|
||||||
byte[] decryptedFST = decryption.decrypt(encryptedFST);
|
byte[] decryptedFST = decryption.decrypt(encryptedFST);
|
||||||
|
|
||||||
FST fst = new FST(decryptedFST,tmd);
|
fst = new FST(decryptedFST,tmd);
|
||||||
|
|
||||||
long start = new Date().getTime();
|
|
||||||
|
|
||||||
System.out.println("Total Size of Content Files: " + ((int)((getTotalContentSize()/1024.0/1024.0)*100))/100.0 +" MB");
|
System.out.println("Total Size of Content Files: " + ((int)((getTotalContentSize()/1024.0/1024.0)*100))/100.0 +" MB");
|
||||||
System.out.println("Total Size of Decrypted Files: " + ((int)((fst.getTotalContentSizeInNUS()/1024.0/1024.0)*100))/100.0 +" MB");
|
System.out.println("Total Size of Decrypted Files: " + ((int)((fst.getTotalContentSizeInNUS()/1024.0/1024.0)*100))/100.0 +" MB");
|
||||||
System.out.println("Entries: " + fst.getTotalEntries());
|
System.out.println("Entries: " + fst.getTotalEntries());
|
||||||
System.out.println("Entries: " + fst.getFileCount());
|
System.out.println("Entries: " + fst.getFileCount());
|
||||||
System.out.println("Files in NUSTitle: " + fst.getFileCountInNUS());
|
System.out.println("Files in NUSTitle: " + fst.getFileCountInNUS());
|
||||||
System.out.println("");
|
|
||||||
System.out.println("Downloading all files.");
|
|
||||||
System.out.println("");
|
|
||||||
|
|
||||||
ForkJoinPool pool = ForkJoinPool.commonPool();
|
|
||||||
List<TitleDownloader> list = new ArrayList<>();
|
|
||||||
for(FEntry f: fst.getFileEntries()){
|
|
||||||
if(!f.isDir() && f.isInNUSTitle())
|
|
||||||
list.add(new TitleDownloader(f));
|
|
||||||
}
|
|
||||||
pool.invokeAll(list);
|
|
||||||
|
|
||||||
long runningTime = new Date().getTime() - start;
|
|
||||||
System.out.println("");
|
|
||||||
System.out.println("Done in: " + runningTime/1000.0 + " seconds");
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// TODO Auto-generated catch block
|
// TODO Auto-generated catch block
|
||||||
@ -65,6 +41,17 @@ public class NUSTitle {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public FST getFst() {
|
||||||
|
return fst;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public void setFst(FST fst) {
|
||||||
|
this.fst = fst;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public long getTitleID() {
|
public long getTitleID() {
|
||||||
return titleID;
|
return titleID;
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,8 @@ import java.io.IOException;
|
|||||||
public class Starter {
|
public class Starter {
|
||||||
|
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
|
System.out.println("JNUSTool 0.0.1 - pre alpha - by Maschell");
|
||||||
|
System.out.println();
|
||||||
try {
|
try {
|
||||||
readConfig();
|
readConfig();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
@ -20,14 +22,15 @@ public class Starter {
|
|||||||
if( args.length > 1 && args[1].length() == 32){
|
if( args.length > 1 && args[1].length() == 32){
|
||||||
key = args[1].substring(0, 32);
|
key = args[1].substring(0, 32);
|
||||||
}
|
}
|
||||||
new NUSTitle(titleID, key);
|
NUSGUI m = new NUSGUI(new NUSTitle(titleID, key));
|
||||||
|
m.setVisible(true);
|
||||||
}else{
|
}else{
|
||||||
System.out.println("Need parameters: TITLEID [KEY]");
|
System.out.println("Need parameters: TITLEID [KEY]");
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void readConfig() throws IOException {
|
public static void readConfig() throws IOException {
|
||||||
BufferedReader in = new BufferedReader(new FileReader(new File("config")));
|
BufferedReader in = new BufferedReader(new FileReader(new File("config")));
|
||||||
Downloader.URL_BASE = in.readLine();
|
Downloader.URL_BASE = in.readLine();
|
||||||
Util.commonKey = Util.hexStringToByteArray(in.readLine());
|
Util.commonKey = Util.hexStringToByteArray(in.readLine());
|
||||||
|
Loading…
x
Reference in New Issue
Block a user