local api = vim.api local luv = vim.loop local open_mode = luv.constants.O_CREAT + luv.constants.O_WRONLY + luv.constants.O_TRUNC local utils = require'nvim-tree.utils' local lib = require'nvim-tree.lib' local events = require'nvim-tree.events' local M = {} local clipboard = { move = {}, copy = {} } local function clear_prompt() vim.api.nvim_command('normal :esc') end local function refresh_tree() vim.api.nvim_command(":NvimTreeRefresh") end local function create_file(file) if luv.fs_access(file, "r") ~= false then local ans = vim.fn.input(file..' already exists, overwrite ? y/n: ') clear_prompt() if ans ~= "y" then return end end luv.fs_open(file, "w", open_mode, vim.schedule_wrap(function(err, fd) if err then api.nvim_err_writeln('Could not create file '..file) else -- FIXME: i don't know why but libuv keeps creating file with executable permissions -- this is why we need to chmod to default file permissions luv.fs_chmod(file, 420) luv.fs_close(fd) events._dispatch_file_created(file) api.nvim_out_write('File '..file..' was properly created\n') refresh_tree() end end)) end local function get_num_entries(iter) local i = 0 for _ in iter do i = i + 1 end return i end function M.create(node) node = lib.get_last_group_node(node) if node.name == '..' then node = { absolute_path = lib.Tree.cwd, entries = lib.Tree.entries, } end local add_into if node.entries ~= nil then add_into = utils.path_add_trailing(node.absolute_path) else add_into = node.absolute_path:sub(0, -(#node.name + 1)) end local ans = vim.fn.input('Create file '..add_into) clear_prompt() if not ans or #ans == 0 then return end if not ans:match(utils.path_separator) then return create_file(add_into..ans) end -- create a foler for each element until / and create a file when element is not ending with / -- if element is ending with / and it's the last element, we need to manually refresh local relpath = '' local idx = 0 local num_entries = get_num_entries(utils.path_split(ans)) for path in utils.path_split(ans) do idx = idx + 1 relpath = relpath..path if relpath:match('.*'..utils.path_separator..'$') then local success = luv.fs_mkdir(add_into..relpath, 493) if not success then api.nvim_err_writeln('Could not create folder '..add_into..relpath) return end if idx == num_entries then events._dispatch_folder_created(add_into..relpath) api.nvim_out_write('Folder '..add_into..relpath..' was properly created\n') refresh_tree() end else create_file(add_into..relpath) end end end local function clear_buffer(absolute_path) for _, buf in pairs(api.nvim_list_bufs()) do if vim.fn.bufloaded(buf) == 1 and api.nvim_buf_get_name(buf) == absolute_path then api.nvim_command(':bd! '..buf) end end end local function remove_dir(cwd) local handle = luv.fs_scandir(cwd) if type(handle) == 'string' then return api.nvim_err_writeln(handle) end while true do local name, t = luv.fs_scandir_next(handle) if not name then break end local new_cwd = utils.path_join({cwd, name}) if t == 'directory' then local success = remove_dir(new_cwd) if not success then return false end else local success = luv.fs_unlink(new_cwd) if not success then return false end clear_buffer(new_cwd) end end return luv.fs_rmdir(cwd) end local function do_copy(source, destination) local source_stats = luv.fs_stat(source) if source_stats and source_stats.type == 'file' then return luv.fs_copyfile(source, destination) end local handle = luv.fs_scandir(source) if type(handle) == 'string' then return false, handle end luv.fs_mkdir(destination, source_stats.mode) while true do local name, _ = luv.fs_scandir_next(handle) if not name then break end local new_name = utils.path_join({source, name}) local new_destination = utils.path_join({destination, name}) local success, msg = do_copy(new_name, new_destination) if not success then return success, msg end end return true end local function do_single_paste(source, dest, action_type, action_fn) local dest_stats = luv.fs_stat(dest) local should_process = true local should_rename = false if dest_stats then local ans = vim.fn.input(dest..' already exists, overwrite ? y/n/r(ename): ') clear_prompt() should_process = ans:match('^y') should_rename = ans:match('^r') end if should_rename then local new_dest = vim.fn.input('New name: ', dest) return do_single_paste(source, new_dest, action_type, action_fn) end if should_process then local success, errmsg = action_fn(source, dest) if not success then api.nvim_err_writeln('Could not '..action_type..' '..source..' - '..errmsg) end end end local function do_paste(node, action_type, action_fn) node = lib.get_last_group_node(node) if node.name == '..' then return end local clip = clipboard[action_type] if #clip == 0 then return end local destination = node.absolute_path local stats = luv.fs_stat(destination) local is_dir = stats and stats.type == 'directory' if not is_dir then destination = vim.fn.fnamemodify(destination, ':p:h') elseif not node.open then destination = vim.fn.fnamemodify(destination, ':p:h:h') end local msg = #clip..' entries' if #clip == 1 then msg = clip[1].absolute_path end local ans = vim.fn.input(action_type..' '..msg..' to '..destination..'? y/n: ') clear_prompt() if not ans:match('^y') then return api.nvim_out_write('Canceled.\n') end for _, entry in ipairs(clip) do local dest = utils.path_join({destination, entry.name }) do_single_paste(entry.absolute_path, dest, action_type, action_fn) end clipboard[action_type] = {} return refresh_tree() end local function add_to_clipboard(node, clip) if node.name == '..' then return end for idx, entry in ipairs(clip) do if entry.absolute_path == node.absolute_path then table.remove(clip, idx) return api.nvim_out_write(node.absolute_path..' removed to clipboard.\n') end end table.insert(clip, node) api.nvim_out_write(node.absolute_path..' added to clipboard.\n') end function M.remove(node) if node.name == '..' then return end local ans = vim.fn.input("Remove " ..node.name.. " ? y/n: ") clear_prompt() if ans:match('^y') then if node.entries ~= nil then local success = remove_dir(node.absolute_path) if not success then return api.nvim_err_writeln('Could not remove '..node.name) end events._dispatch_folder_removed(node.absolute_path) api.nvim_out_write(node.name..' has been removed\n') else local success = luv.fs_unlink(node.absolute_path) if not success then return api.nvim_err_writeln('Could not remove '..node.name) end events._dispatch_file_removed(node.absolute_path) api.nvim_out_write(node.name..' has been removed\n') clear_buffer(node.absolute_path) end refresh_tree() end end function M.rename(with_sub) return function(node) node = lib.get_last_group_node(node) if node.name == '..' then return end local namelen = node.name:len() local abs_path = with_sub and node.absolute_path:sub(0, namelen * (-1) -1) or node.absolute_path local new_name = vim.fn.input("Rename " ..node.name.. " to ", abs_path) clear_prompt() if not new_name or #new_name == 0 then return end local success = luv.fs_rename(node.absolute_path, new_name) if not success then return api.nvim_err_writeln('Could not rename '..node.absolute_path..' to '..new_name) end api.nvim_out_write(node.absolute_path..' ➜ '..new_name..'\n') for _, buf in pairs(api.nvim_list_bufs()) do if api.nvim_buf_is_loaded(buf) then if api.nvim_buf_get_name(buf) == node.absolute_path then api.nvim_buf_set_name(buf, new_name) -- to avoid the 'overwrite existing file' error message on write vim.api.nvim_buf_call(buf, function() vim.cmd("silent! w!") end) end end end events._dispatch_node_renamed(abs_path, new_name) refresh_tree() end end function M.copy(node) add_to_clipboard(node, clipboard.copy) end function M.cut(node) add_to_clipboard(node, clipboard.move) end function M.paste(node) if clipboard.move[1] ~= nil then return do_paste(node, 'move', luv.fs_rename) end return do_paste(node, 'copy', do_copy) end function M.print_clipboard() local content = {} if #clipboard.move > 0 then table.insert(content, 'Cut') for _, item in pairs(clipboard.move) do table.insert(content, ' * '..item.absolute_path) end end if #clipboard.copy > 0 then table.insert(content, 'Copy') for _, item in pairs(clipboard.copy) do table.insert(content, ' * '..item.absolute_path) end end return api.nvim_out_write(table.concat(content, '\n')..'\n') end return M