105 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			105 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from textual.widgets import DirectoryTree, Rule
 | |
| from textual.widgets.tree import TreeNode
 | |
| from textual.events import MouseDown
 | |
| 
 | |
| from prompt import Prompt
 | |
| from context_menu import ContextMenu, NoSelectStatic
 | |
| 
 | |
| import os, shutil
 | |
| 
 | |
| 
 | |
| class CustomDirectoryTree(DirectoryTree):
 | |
|     def __init__(self, path, *, name = None, id = None, classes = None, disabled = False):
 | |
|         super().__init__(path, name=name, id=id, classes=classes, disabled=disabled)
 | |
|         self.right_clicked_node: TreeNode | None = None
 | |
| 
 | |
|     
 | |
| 
 | |
|     def context_menu_chosen(self, result):
 | |
|         if result == "Delete":
 | |
|             def delete_confirm(will_delete: bool | None):
 | |
|                 if will_delete == True:
 | |
|                     if os.path.isfile(self.right_clicked_node.data.path):
 | |
|                         os.remove(self.right_clicked_node.data.path)
 | |
|                     else:
 | |
|                         shutil.rmtree(self.right_clicked_node.data.path)
 | |
|                     self.reload()
 | |
| 
 | |
|                     self.notify(f"Deleted \"{self.right_clicked_node.data.path}\".")
 | |
|                 self.right_clicked_node = None
 | |
| 
 | |
|             self.app.push_screen(Prompt(f"Are you sure you want to delete \"{self.right_clicked_node.label}\"?", "confirm", "Confirm deletion"), delete_confirm)
 | |
|         elif result == "Rename":
 | |
|             def rename_confirm(new_name: str | None):
 | |
|                 if new_name == None: return
 | |
|                 if new_name.strip() == "":
 | |
|                     self.notify("Filename can't be empty.", title="Failed to rename", severity="error")
 | |
|                     return
 | |
| 
 | |
|                 os.rename(self.right_clicked_node.data.path, os.path.join(os.path.dirname(self.right_clicked_node.data.path), new_name))
 | |
|                 self.reload()
 | |
| 
 | |
|                 self.notify("Renamed successfully.")
 | |
| 
 | |
|                 self.right_clicked_node = None
 | |
| 
 | |
|             self.app.push_screen(Prompt(f"Enter the new name for \"{self.right_clicked_node.label}\".", "string", "Rename"), rename_confirm)
 | |
|         elif result == "New Folder":
 | |
|             def new_folder(folder_name: str | None):
 | |
|                 if folder_name == None: return
 | |
|                 if folder_name.strip() == "":
 | |
|                     self.notify("Folder name can't be empty.", title="Failed to create folder", severity="error")
 | |
|                     return                
 | |
| 
 | |
|                 new_folder_path = os.path.join(self.right_clicked_node.data.path, folder_name)
 | |
|                 try:
 | |
|                     os.mkdir(new_folder_path)
 | |
|                     self.reload()
 | |
|                 except Exception as e:
 | |
|                     self.notify(str(e), title="Failed to create folder", severity="error")
 | |
|                     return
 | |
| 
 | |
|             self.app.push_screen(Prompt("Enter the name of the new folder.", "string", "Create New Folder"), new_folder) 
 | |
|         elif result == "New File":
 | |
|             def new_file(file_name: str | None):
 | |
|                 if file_name == None: return
 | |
|                 if file_name.strip() == "":
 | |
|                     self.notify("File name can't be empty.", title="Failed to create file", severity="error")
 | |
|                     return
 | |
|                 
 | |
|                 new_file_path = os.path.join(self.right_clicked_node.data.path, file_name)
 | |
| 
 | |
|                 if os.path.isfile(new_file_path):
 | |
|                     self.notify("That file already exists.", title="Failed to create file", severity="error")
 | |
|                     return
 | |
| 
 | |
|                 try:
 | |
|                     with open(new_file_path, "w") as f:
 | |
|                         pass
 | |
|                     self.reload()
 | |
|                 except Exception as e:
 | |
|                     self.notify(str(e), title="Failed to create file", severity="error")
 | |
|                     return
 | |
| 
 | |
|             self.app.push_screen(Prompt("Enter the name of the new file", "string", "Create New File"), new_file)
 | |
| 
 | |
|     def on_mouse_down(self, event: MouseDown):
 | |
|         if event.button != 3 or not "line" in event.style.meta:
 | |
|             return
 | |
|         selected_node = self.get_node_at_line(event.style.meta["line"])
 | |
|         self.right_clicked_node = selected_node
 | |
| 
 | |
|         options = None
 | |
|         if self._safe_is_dir(self.right_clicked_node.data.path):
 | |
|             options = ["New Folder", "New File", NoSelectStatic(f'[d]{"-" * 17}[/]'), "Delete", "Rename"]
 | |
|         else:
 | |
|             options = ["Delete", "Rename"]
 | |
| 
 | |
|         file_name = str(self.right_clicked_node.label) if len(self.right_clicked_node.label) <= 17 else self.right_clicked_node.label[:14] + "..."
 | |
|         
 | |
|         self.app.push_screen(ContextMenu(
 | |
|             [NoSelectStatic(f"[b]{file_name}[/]"), NoSelectStatic(f'[d]{"-" * 17}[/]')] + options,
 | |
|             event.screen_offset
 | |
|         ), self.context_menu_chosen)
 | |
| 
 |