committed by
bunnei
2 changed files with 110 additions and 125 deletions
-
125src/android/app/src/main/java/org/yuzu/yuzu_emu/utils/DocumentsTree.java
-
110src/android/app/src/main/java/org/yuzu/yuzu_emu/utils/DocumentsTree.kt
@ -1,125 +0,0 @@ |
|||
package org.yuzu.yuzu_emu.utils; |
|||
|
|||
import android.content.Context; |
|||
import android.net.Uri; |
|||
|
|||
import androidx.annotation.Nullable; |
|||
import androidx.documentfile.provider.DocumentFile; |
|||
|
|||
import org.yuzu.yuzu_emu.YuzuApplication; |
|||
import org.yuzu.yuzu_emu.model.MinimalDocumentFile; |
|||
|
|||
import java.util.HashMap; |
|||
import java.util.Map; |
|||
import java.util.StringTokenizer; |
|||
|
|||
public class DocumentsTree { |
|||
private DocumentsNode root; |
|||
private final Context context; |
|||
public static final String DELIMITER = "/"; |
|||
|
|||
public DocumentsTree() { |
|||
context = YuzuApplication.getAppContext(); |
|||
} |
|||
|
|||
public void setRoot(Uri rootUri) { |
|||
root = null; |
|||
root = new DocumentsNode(); |
|||
root.uri = rootUri; |
|||
root.isDirectory = true; |
|||
} |
|||
|
|||
public int openContentUri(String filepath, String openmode) { |
|||
DocumentsNode node = resolvePath(filepath); |
|||
if (node == null) { |
|||
return -1; |
|||
} |
|||
return FileUtil.openContentUri(context, node.uri.toString(), openmode); |
|||
} |
|||
|
|||
public long getFileSize(String filepath) { |
|||
DocumentsNode node = resolvePath(filepath); |
|||
if (node == null || node.isDirectory) { |
|||
return 0; |
|||
} |
|||
return FileUtil.getFileSize(context, node.uri.toString()); |
|||
} |
|||
|
|||
public boolean Exists(String filepath) { |
|||
return resolvePath(filepath) != null; |
|||
} |
|||
|
|||
@Nullable |
|||
private DocumentsNode resolvePath(String filepath) { |
|||
StringTokenizer tokens = new StringTokenizer(filepath, DELIMITER, false); |
|||
DocumentsNode iterator = root; |
|||
while (tokens.hasMoreTokens()) { |
|||
String token = tokens.nextToken(); |
|||
if (token.isEmpty()) continue; |
|||
iterator = find(iterator, token); |
|||
if (iterator == null) return null; |
|||
} |
|||
return iterator; |
|||
} |
|||
|
|||
@Nullable |
|||
private DocumentsNode find(DocumentsNode parent, String filename) { |
|||
if (parent.isDirectory && !parent.loaded) { |
|||
structTree(parent); |
|||
} |
|||
return parent.children.get(filename); |
|||
} |
|||
|
|||
/** |
|||
* Construct current level directory tree |
|||
* @param parent parent node of this level |
|||
*/ |
|||
private void structTree(DocumentsNode parent) { |
|||
MinimalDocumentFile[] documents = FileUtil.listFiles(context, parent.uri); |
|||
for (MinimalDocumentFile document: documents) { |
|||
DocumentsNode node = new DocumentsNode(document); |
|||
node.parent = parent; |
|||
parent.children.put(node.name, node); |
|||
} |
|||
parent.loaded = true; |
|||
} |
|||
|
|||
public static boolean isNativePath(String path) { |
|||
if (path.length() > 0) { |
|||
return path.charAt(0) == '/'; |
|||
} |
|||
return false; |
|||
} |
|||
|
|||
private static class DocumentsNode { |
|||
private DocumentsNode parent; |
|||
private final Map<String, DocumentsNode> children = new HashMap<>(); |
|||
private String name; |
|||
private Uri uri; |
|||
private boolean loaded = false; |
|||
private boolean isDirectory = false; |
|||
|
|||
private DocumentsNode() {} |
|||
private DocumentsNode(MinimalDocumentFile document) { |
|||
name = document.getFilename(); |
|||
uri = document.getUri(); |
|||
isDirectory = document.isDirectory(); |
|||
loaded = !isDirectory; |
|||
} |
|||
private DocumentsNode(DocumentFile document, boolean isCreateDir) { |
|||
name = document.getName(); |
|||
uri = document.getUri(); |
|||
isDirectory = isCreateDir; |
|||
loaded = true; |
|||
} |
|||
|
|||
private void rename(String name) { |
|||
if (parent == null) { |
|||
return; |
|||
} |
|||
parent.children.remove(this.name); |
|||
this.name = name; |
|||
parent.children.put(name, this); |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,110 @@ |
|||
package org.yuzu.yuzu_emu.utils |
|||
|
|||
import android.net.Uri |
|||
import androidx.documentfile.provider.DocumentFile |
|||
import org.yuzu.yuzu_emu.YuzuApplication |
|||
import org.yuzu.yuzu_emu.model.MinimalDocumentFile |
|||
import java.io.File |
|||
import java.util.* |
|||
|
|||
class DocumentsTree { |
|||
private var root: DocumentsNode? = null |
|||
|
|||
fun setRoot(rootUri: Uri?) { |
|||
root = null |
|||
root = DocumentsNode() |
|||
root!!.uri = rootUri |
|||
root!!.isDirectory = true |
|||
} |
|||
|
|||
fun openContentUri(filepath: String, openMode: String?): Int { |
|||
val node = resolvePath(filepath) ?: return -1 |
|||
return FileUtil.openContentUri(YuzuApplication.appContext, node.uri.toString(), openMode) |
|||
} |
|||
|
|||
fun getFileSize(filepath: String): Long { |
|||
val node = resolvePath(filepath) |
|||
return if (node == null || node.isDirectory) { |
|||
0 |
|||
} else FileUtil.getFileSize(YuzuApplication.appContext, node.uri.toString()) |
|||
} |
|||
|
|||
fun exists(filepath: String): Boolean { |
|||
return resolvePath(filepath) != null |
|||
} |
|||
|
|||
private fun resolvePath(filepath: String): DocumentsNode? { |
|||
val tokens = StringTokenizer(filepath, File.separator, false) |
|||
var iterator = root |
|||
while (tokens.hasMoreTokens()) { |
|||
val token = tokens.nextToken() |
|||
if (token.isEmpty()) continue |
|||
iterator = find(iterator, token) |
|||
if (iterator == null) return null |
|||
} |
|||
return iterator |
|||
} |
|||
|
|||
private fun find(parent: DocumentsNode?, filename: String): DocumentsNode? { |
|||
if (parent!!.isDirectory && !parent.loaded) { |
|||
structTree(parent) |
|||
} |
|||
return parent.children[filename] |
|||
} |
|||
|
|||
/** |
|||
* Construct current level directory tree |
|||
* @param parent parent node of this level |
|||
*/ |
|||
private fun structTree(parent: DocumentsNode) { |
|||
val documents = FileUtil.listFiles(YuzuApplication.appContext, parent.uri!!) |
|||
for (document in documents) { |
|||
val node = DocumentsNode(document) |
|||
node.parent = parent |
|||
parent.children[node.name] = node |
|||
} |
|||
parent.loaded = true |
|||
} |
|||
|
|||
private class DocumentsNode { |
|||
var parent: DocumentsNode? = null |
|||
val children: MutableMap<String?, DocumentsNode> = HashMap() |
|||
var name: String? = null |
|||
var uri: Uri? = null |
|||
var loaded = false |
|||
var isDirectory = false |
|||
|
|||
constructor() |
|||
constructor(document: MinimalDocumentFile) { |
|||
name = document.filename |
|||
uri = document.uri |
|||
isDirectory = document.isDirectory |
|||
loaded = !isDirectory |
|||
} |
|||
|
|||
private constructor(document: DocumentFile, isCreateDir: Boolean) { |
|||
name = document.name |
|||
uri = document.uri |
|||
isDirectory = isCreateDir |
|||
loaded = true |
|||
} |
|||
|
|||
private fun rename(name: String) { |
|||
if (parent == null) { |
|||
return |
|||
} |
|||
parent!!.children.remove(this.name) |
|||
this.name = name |
|||
parent!!.children[name] = this |
|||
} |
|||
} |
|||
|
|||
companion object { |
|||
@JvmStatic |
|||
fun isNativePath(path: String): Boolean { |
|||
return if (path.isNotEmpty()) { |
|||
path[0] == '/' |
|||
} else false |
|||
} |
|||
} |
|||
} |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue