From 6175bd646272335c8db93264760760d8f2a611d5 Mon Sep 17 00:00:00 2001 From: Steven Arcangeli <506791+stevearc@users.noreply.github.com> Date: Sun, 5 Nov 2023 12:40:58 -0800 Subject: [PATCH] feat: trash support for linux and mac (#165) * wip: skeleton code for trash adapter * refactor: split trash implementation for mac and linux * fix: ensure we create the .Trash/$uid dir * feat: code complete linux trash implementation * doc: write up trash features * feat: code complete mac trash implementation * cleanup: remove previous, terrible, undocumented trash feature * fix: always disabled trash * feat: show original path of trashed files * doc: add a note about calling actions directly * fix: bugs in trash implementation * fix: schedule_wrap in mac trash * doc: fix typo and line wrapping * fix: parsing of arguments to :Oil command * doc: small documentation tweaks * doc: fix awkward wording in the toggle_trash action * fix: warning on Windows when delete_to_trash = true * feat: :Oil --trash can open specific trash directories * fix: show all trash files in device root * fix: trash mtime should be sortable * fix: shorten_path handles optional trailing slash * refactor: overhaul the UI * fix: keep trash original path vtext from stacking * refactor: replace disable_changes with an error filter * fix: shorten path names in home directory relative to root * doc: small README format changes * cleanup: remove unnecessary preserve_undo logic * test: add a functional test for the freedesktop trash adapter * test: more functional tests for trash * fix: schedule a callback to avoid main loop error * refactor: clean up mutator logic * doc: some comments and type annotations --- README.md | 23 +- doc/oil.txt | 54 ++- lua/oil/actions.lua | 32 ++ lua/oil/adapters/files.lua | 32 +- lua/oil/adapters/ssh.lua | 2 +- lua/oil/adapters/trash.lua | 9 + lua/oil/adapters/trash/freedesktop.lua | 630 +++++++++++++++++++++++++ lua/oil/adapters/trash/mac.lua | 233 +++++++++ lua/oil/adapters/trash/windows.lua | 20 + lua/oil/cache.lua | 21 +- lua/oil/columns.lua | 3 + lua/oil/config.lua | 55 +-- lua/oil/fs.lua | 33 +- lua/oil/init.lua | 96 ++-- lua/oil/mutator/init.lua | 77 ++- lua/oil/mutator/parser.lua | 20 +- lua/oil/util.lua | 13 +- lua/oil/view.lua | 114 ++++- scripts/generate.py | 68 ++- syntax/oil_preview.vim | 8 +- tests/files_spec.lua | 2 - tests/minimal_init.lua | 2 +- tests/parser_spec.lua | 1 + tests/test_util.lua | 56 +++ tests/tmpdir.lua | 35 +- tests/trash_spec.lua | 164 +++++++ tests/url_spec.lua | 2 +- 27 files changed, 1578 insertions(+), 227 deletions(-) create mode 100644 lua/oil/adapters/trash.lua create mode 100644 lua/oil/adapters/trash/freedesktop.lua create mode 100644 lua/oil/adapters/trash/mac.lua create mode 100644 lua/oil/adapters/trash/windows.lua create mode 100644 tests/trash_spec.lua diff --git a/README.md b/README.md index 009ddb46..c198c8b2 100644 --- a/README.md +++ b/README.md @@ -43,11 +43,13 @@ oil.nvim supports all the usual plugin managers Packer ```lua -require('packer').startup(function() - use { - 'stevearc/oil.nvim', - config = function() require('oil').setup() end - } +require("packer").startup(function() + use({ + "stevearc/oil.nvim", + config = function() + require("oil").setup() + end, + }) end) ``` @@ -57,9 +59,9 @@ end) Paq ```lua -require "paq" { - {'stevearc/oil.nvim'}; -} +require("paq")({ + { "stevearc/oil.nvim" }, +}) ``` @@ -154,8 +156,6 @@ require("oil").setup({ delete_to_trash = false, -- Skip the confirmation popup for simple operations skip_confirm_for_simple_edits = false, - -- Change this to customize the command used when deleting to trash - trash_command = "trash-put", -- Selecting a new/moved/renamed file or directory will prompt you to save changes first prompt_save_on_select_new_entry = true, -- Oil will automatically delete hidden buffers after this delay @@ -184,6 +184,7 @@ require("oil").setup({ ["gs"] = "actions.change_sort", ["gx"] = "actions.open_external", ["g."] = "actions.toggle_hidden", + ["g\\"] = "actions.toggle_trash", }, -- Set to false to disable all of the above keymaps use_default_keymaps = true, @@ -277,7 +278,7 @@ nvim oil-ssh://[username@]hostname[:port]/[path] This may look familiar. In fact, this is the same url format that netrw uses. -Note that at the moment the ssh adapter does not support Windows machines, and it requires the server to have a `/bin/sh` binary as well as standard unix commands (`rm`, `mv`, `mkdir`, `chmod`, `cp`, `touch`, `ln`, `echo`). +Note that at the moment the ssh adapter does not support Windows machines, and it requires the server to have a `/bin/sh` binary as well as standard unix commands (`ls`, `rm`, `mv`, `mkdir`, `chmod`, `cp`, `touch`, `ln`, `echo`). ## API diff --git a/doc/oil.txt b/doc/oil.txt index 0bd9fe64..2da5bb2c 100644 --- a/doc/oil.txt +++ b/doc/oil.txt @@ -8,6 +8,7 @@ CONTENTS *oil-content 3. Columns |oil-columns| 4. Actions |oil-actions| 5. Highlights |oil-highlights| + 6. Trash |oil-trash| -------------------------------------------------------------------------------- OPTIONS *oil-options* @@ -45,8 +46,6 @@ OPTIONS *oil-option delete_to_trash = false, -- Skip the confirmation popup for simple operations skip_confirm_for_simple_edits = false, - -- Change this to customize the command used when deleting to trash - trash_command = "trash-put", -- Selecting a new/moved/renamed file or directory will prompt you to save changes first prompt_save_on_select_new_entry = true, -- Oil will automatically delete hidden buffers after this delay @@ -75,6 +74,7 @@ OPTIONS *oil-option ["gs"] = "actions.change_sort", ["gx"] = "actions.open_external", ["g."] = "actions.toggle_hidden", + ["g\\"] = "actions.toggle_trash", }, -- Set to false to disable all of the above keymaps use_default_keymaps = true, @@ -343,6 +343,8 @@ birthtime *column-birthtim ACTIONS *oil-actions* These are actions that can be used in the `keymaps` section of config options. +You can also call them directly with +`require("oil.actions").action_name.callback()` cd *actions.cd* :cd to the current oil directory @@ -408,11 +410,14 @@ tcd *actions.tc toggle_hidden *actions.toggle_hidden* Toggle hidden files and directories +toggle_trash *actions.toggle_trash* + Jump to and from the trash for the current directory + -------------------------------------------------------------------------------- HIGHLIGHTS *oil-highlights* OilDir *hl-OilDir* - Directories in an oil buffer + Directory names in an oil buffer OilDirIcon *hl-OilDirIcon* Icon for directories @@ -423,6 +428,9 @@ OilSocket *hl-OilSocke OilLink *hl-OilLink* Soft links in an oil buffer +OilLinkTarget *hl-OilLinkTarget* + The target of a soft link + OilFile *hl-OilFile* Normal files in an oil buffer @@ -441,5 +449,45 @@ OilCopy *hl-OilCop OilChange *hl-OilChange* Change action in the oil preview window +OilRestore *hl-OilRestore* + Restore (from the trash) action in the oil preview window + +OilPurge *hl-OilPurge* + Purge (Permanently delete a file from trash) action in the oil preview + window + +OilTrash *hl-OilTrash* + Trash (delete a file to trash) action in the oil preview window + +OilTrashSourcePath *hl-OilTrashSourcePath* + Virtual text that shows the original path of file in the trash + +-------------------------------------------------------------------------------- +TRASH *oil-trash* + + +Oil has built-in support for using the system trash. When +`delete_to_trash = true`, any deleted files will be sent to the trash instead +of being permanently deleted. You can browse the trash for a directory using +the `toggle_trash` action (bound to `g\` by default). You can view all files +in the trash with `:Oil --trash /`. + +To restore files, simply delete them from the trash and put them in the desired +destination, the same as any other file operation. If you delete files from the +trash they will be permanently deleted (purged). + +Linux: + Oil supports the FreeDesktop trash specification. + https://specifications.freedesktop.org/trash-spec/trashspec-1.0.html + All features should work. + +Mac: + Oil has limited support for MacOS due to the proprietary nature of the + implementation. The trash bin can only be viewed as a single dir + (instead of being able to see files that were trashed from a directory). + +Windows: + Oil does not yet support the Windows trash. PRs are welcome! + ================================================================================ vim:tw=80:ts=2:ft=help:norl:syntax=help: diff --git a/lua/oil/actions.lua b/lua/oil/actions.lua index d446a0b5..2402750e 100644 --- a/lua/oil/actions.lua +++ b/lua/oil/actions.lua @@ -1,6 +1,9 @@ local oil = require("oil") local util = require("oil.util") +-- TODO remove after https://github.com/folke/neodev.nvim/pull/163 lands +---@diagnostic disable: inject-field + local M = {} M.show_help = { @@ -302,6 +305,35 @@ M.change_sort = { end, } +M.toggle_trash = { + desc = "Jump to and from the trash for the current directory", + callback = function() + local fs = require("oil.fs") + local bufname = vim.api.nvim_buf_get_name(0) + local scheme, path = util.parse_url(bufname) + local bufnr = vim.api.nvim_get_current_buf() + local url + if scheme == "oil://" then + url = "oil-trash://" .. path + elseif scheme == "oil-trash://" then + url = "oil://" .. path + -- The non-linux trash implementations don't support per-directory trash, + -- so jump back to the stored source buffer. + if not fs.is_linux then + local src_bufnr = vim.b.oil_trash_toggle_src + if src_bufnr and vim.api.nvim_buf_is_valid(src_bufnr) then + url = vim.api.nvim_buf_get_name(src_bufnr) + end + end + else + vim.notify("No trash found for buffer", vim.log.levels.WARN) + return + end + vim.cmd.edit({ args = { url } }) + vim.b.oil_trash_toggle_src = bufnr + end, +} + ---List actions for documentation generation ---@private M._get_actions = function() diff --git a/lua/oil/adapters/files.lua b/lua/oil/adapters/files.lua index fcd3bcf2..72131539 100644 --- a/lua/oil/adapters/files.lua +++ b/lua/oil/adapters/files.lua @@ -7,6 +7,7 @@ local permissions = require("oil.adapters.files.permissions") local trash = require("oil.adapters.files.trash") local util = require("oil.util") local uv = vim.uv or vim.loop + local M = {} local FIELD_NAME = constants.FIELD_NAME @@ -147,7 +148,11 @@ if not fs.is_windows then } end -local current_year = vim.fn.strftime("%Y") +local current_year +-- Make sure we run this import-time effect in the main loop (mostly for tests) +vim.schedule(function() + current_year = vim.fn.strftime("%Y") +end) for _, time_key in ipairs({ "ctime", "mtime", "atime", "birthtime" }) do file_columns[time_key] = { @@ -436,7 +441,12 @@ M.render_action = function(action) elseif action.type == "delete" then local _, path = util.parse_url(action.url) assert(path) - return string.format("DELETE %s", M.to_short_os_path(path, action.entry_type)) + local short_path = M.to_short_os_path(path, action.entry_type) + if config.delete_to_trash then + return string.format(" TRASH %s", short_path) + else + return string.format("DELETE %s", short_path) + end elseif action.type == "move" or action.type == "copy" then local dest_adapter = config.get_adapter_by_scheme(action.dest_url) if dest_adapter == M then @@ -451,7 +461,7 @@ M.render_action = function(action) M.to_short_os_path(dest_path, action.entry_type) ) else - -- We should never hit this because we don't implement supported_adapters_for_copy + -- We should never hit this because we don't implement supported_cross_adapter_actions error("files adapter doesn't support cross-adapter move/copy") end else @@ -494,7 +504,15 @@ M.perform_action = function(action, cb) assert(path) path = fs.posix_to_os_path(path) if config.delete_to_trash then - trash.recursive_delete(path, cb) + if config.trash_command then + vim.notify_once( + "Oil now has native support for trash. Remove the `trash_command` from your config to try it out!", + vim.log.levels.WARN + ) + trash.recursive_delete(path, cb) + else + require("oil.adapters.trash").delete_to_trash(path, cb) + end else fs.recursive_delete(action.entry_type, path, cb) end @@ -507,9 +525,9 @@ M.perform_action = function(action, cb) assert(dest_path) src_path = fs.posix_to_os_path(src_path) dest_path = fs.posix_to_os_path(dest_path) - fs.recursive_move(action.entry_type, src_path, dest_path, vim.schedule_wrap(cb)) + fs.recursive_move(action.entry_type, src_path, dest_path, cb) else - -- We should never hit this because we don't implement supported_adapters_for_copy + -- We should never hit this because we don't implement supported_cross_adapter_actions cb("files adapter doesn't support cross-adapter move") end elseif action.type == "copy" then @@ -523,7 +541,7 @@ M.perform_action = function(action, cb) dest_path = fs.posix_to_os_path(dest_path) fs.recursive_copy(action.entry_type, src_path, dest_path, cb) else - -- We should never hit this because we don't implement supported_adapters_for_copy + -- We should never hit this because we don't implement supported_cross_adapter_actions cb("files adapter doesn't support cross-adapter copy") end else diff --git a/lua/oil/adapters/ssh.lua b/lua/oil/adapters/ssh.lua index 1fa0057f..7707b954 100644 --- a/lua/oil/adapters/ssh.lua +++ b/lua/oil/adapters/ssh.lua @@ -348,7 +348,7 @@ M.perform_action = function(action, cb) end end -M.supported_adapters_for_copy = { files = true } +M.supported_cross_adapter_actions = { files = "copy" } ---@param bufnr integer M.read_file = function(bufnr) diff --git a/lua/oil/adapters/trash.lua b/lua/oil/adapters/trash.lua new file mode 100644 index 00000000..3c7ffeb9 --- /dev/null +++ b/lua/oil/adapters/trash.lua @@ -0,0 +1,9 @@ +local fs = require("oil.fs") + +if fs.is_mac then + return require("oil.adapters.trash.mac") +elseif fs.is_windows then + error("Trash is not implemented yet on Windows") +else + return require("oil.adapters.trash.freedesktop") +end diff --git a/lua/oil/adapters/trash/freedesktop.lua b/lua/oil/adapters/trash/freedesktop.lua new file mode 100644 index 00000000..3ba211ff --- /dev/null +++ b/lua/oil/adapters/trash/freedesktop.lua @@ -0,0 +1,630 @@ +-- Based on the FreeDesktop.org trash specification +-- https://specifications.freedesktop.org/trash-spec/trashspec-1.0.html +local cache = require("oil.cache") +local config = require("oil.config") +local constants = require("oil.constants") +local files = require("oil.adapters.files") +local fs = require("oil.fs") +local util = require("oil.util") + +local uv = vim.uv or vim.loop +local FIELD_META = constants.FIELD_META + +local M = {} + +local function touch_dir(path) + uv.fs_mkdir(path, 448) -- 0700 +end + +local function ensure_trash_dir(path) + touch_dir(path) + touch_dir(fs.join(path, "info")) + touch_dir(fs.join(path, "files")) +end + +---Gets the location of the home trash dir, creating it if necessary +---@return string +local function get_home_trash_dir() + local xdg_home = vim.env.XDG_DATA_HOME + if not xdg_home then + xdg_home = fs.join(assert(uv.os_homedir()), ".local", "share") + end + local trash_dir = fs.join(xdg_home, "Trash") + ensure_trash_dir(trash_dir) + return trash_dir +end + +---@param mode integer +---@return boolean +local function is_sticky(mode) + local extra = bit.rshift(mode, 9) + return bit.band(extra, 4) ~= 0 +end + +---Get the topdir .Trash/$uid directory if present and valid +---@param path string +---@return string[] +local function get_top_trash_dirs(path) + local dirs = {} + local dev = (uv.fs_stat(path) or {}).dev + local top_trash_dirs = vim.fs.find(".Trash", { upward = true, path = path, limit = math.huge }) + for _, top_trash_dir in ipairs(top_trash_dirs) do + local stat = uv.fs_stat(top_trash_dir) + if stat and not dev then + dev = stat.dev + end + if stat and stat.dev == dev and stat.type == "directory" and is_sticky(stat.mode) then + local trash_dir = fs.join(top_trash_dir, tostring(uv.getuid())) + ensure_trash_dir(trash_dir) + table.insert(dirs, trash_dir) + end + end + + -- Also search for the .Trash-$uid + top_trash_dirs = vim.fs.find( + string.format(".Trash-%d", uv.getuid()), + { upward = true, path = path, limit = math.huge } + ) + for _, top_trash_dir in ipairs(top_trash_dirs) do + local stat = uv.fs_stat(top_trash_dir) + if stat and stat.dev == dev then + ensure_trash_dir(top_trash_dir) + table.insert(dirs, top_trash_dir) + end + end + + return dirs +end + +---@param path string +---@return string +local function get_write_trash_dir(path) + local dev = uv.fs_stat(path).dev + local home_trash = get_home_trash_dir() + if uv.fs_stat(home_trash).dev == dev then + return home_trash + end + + local top_trash_dirs = get_top_trash_dirs(path) + if not vim.tbl_isempty(top_trash_dirs) then + return top_trash_dirs[1] + end + + local parent = vim.fn.fnamemodify(path, ":h") + local next_parent = vim.fn.fnamemodify(parent, ":h") + while parent ~= next_parent and uv.fs_stat(next_parent).dev == dev do + parent = next_parent + next_parent = vim.fn.fnamemodify(parent, ":h") + end + + local top_trash = fs.join(parent, string.format(".Trash-%d", uv.getuid())) + ensure_trash_dir(top_trash) + return top_trash +end + +---@param path string +---@return string[] +local function get_read_trash_dirs(path) + local dirs = { get_home_trash_dir() } + vim.list_extend(dirs, get_top_trash_dirs(path)) + return dirs +end + +---@param url string +---@param callback fun(url: string) +M.normalize_url = function(url, callback) + local scheme, path = util.parse_url(url) + assert(path) + local os_path = vim.fn.fnamemodify(fs.posix_to_os_path(path), ":p") + uv.fs_realpath( + os_path, + vim.schedule_wrap(function(err, new_os_path) + local realpath = new_os_path or os_path + callback(scheme .. util.addslash(fs.os_to_posix_path(realpath))) + end) + ) +end + +---@param url string +---@param entry oil.Entry +---@param cb fun(path: string) +M.get_entry_path = function(url, entry, cb) + local internal_entry = assert(cache.get_entry_by_id(entry.id)) + local meta = internal_entry[FIELD_META] + ---@type oil.TrashInfo + local trash_info = meta.trash_info + if not trash_info then + -- This is a subpath in the trash + M.normalize_url(url, cb) + return + end + local path = fs.os_to_posix_path(trash_info.trash_file) + if meta.stat.type == "directory" then + path = util.addslash(path) + end + cb("oil://" .. path) +end + +---@class oil.TrashInfo +---@field trash_file string +---@field info_file string +---@field original_path string +---@field deletion_date number +---@field stat uv_fs_t + +---@param info_file string +---@param cb fun(err?: string, info?: oil.TrashInfo) +local function read_trash_info(info_file, cb) + if not vim.endswith(info_file, ".trashinfo") then + return cb("File is not .trashinfo") + end + uv.fs_open(info_file, "r", 448, function(err, fd) + if err then + return cb(err) + end + assert(fd) + uv.fs_fstat(fd, function(stat_err, stat) + if stat_err then + uv.fs_close(fd) + return cb(stat_err) + end + uv.fs_read( + fd, + assert(stat).size, + nil, + vim.schedule_wrap(function(read_err, content) + uv.fs_close(fd) + if read_err then + return cb(read_err) + end + assert(content) + local trash_info = { + info_file = info_file, + } + local lines = vim.split(content, "\r?\n") + if lines[1] ~= "[Trash Info]" then + return cb("File missing [Trash Info] header") + end + local trash_base = vim.fn.fnamemodify(info_file, ":h:h") + for _, line in ipairs(lines) do + local key, value = unpack(vim.split(line, "=", { plain = true, trimempty = true })) + if key == "Path" and not trash_info.original_path then + if not vim.startswith(value, "/") then + value = fs.join(trash_base, value) + end + trash_info.original_path = value + elseif key == "DeletionDate" and not trash_info.deletion_date then + trash_info.deletion_date = vim.fn.strptime("%Y-%m-%dT%H:%M:%S", value) + end + end + + if not trash_info.original_path or not trash_info.deletion_date then + return cb("File missing required fields") + end + + local basename = vim.fn.fnamemodify(info_file, ":t:r") + trash_info.trash_file = fs.join(trash_base, "files", basename) + uv.fs_stat(trash_info.trash_file, function(trash_stat_err, trash_stat) + if trash_stat_err then + cb(".trashinfo file points to non-existant file") + else + trash_info.stat = trash_stat + cb(nil, trash_info) + end + end) + end) + ) + end) + end) +end + +---@param url string +---@param column_defs string[] +---@param cb fun(err?: string, entries?: oil.InternalEntry[], fetch_more?: fun()) +M.list = function(url, column_defs, cb) + cb = vim.schedule_wrap(cb) + local _, path = util.parse_url(url) + assert(path) + local trash_dirs = get_read_trash_dirs(path) + local trash_idx = 0 + + local read_next_trash_dir + read_next_trash_dir = function() + trash_idx = trash_idx + 1 + local trash_dir = trash_dirs[trash_idx] + if not trash_dir then + return cb() + end + + -- Show all files from the trash directory if we are in the root of the device, which we can + -- tell if the trash dir is a subpath of our current path + local show_all_files = fs.is_subpath(path, trash_dir) + -- The first trash dir is a special case; it is in the home directory and we should only show + -- all entries if we are in the top root path "/" + if trash_idx == 1 then + show_all_files = path == "/" + end + + local info_dir = fs.join(trash_dir, "info") + ---@diagnostic disable-next-line: param-type-mismatch + uv.fs_opendir(info_dir, function(open_err, fd) + if open_err then + if open_err:match("^ENOENT: no such file or directory") then + -- If the directory doesn't exist, treat the list as a success. We will be able to traverse + -- and edit a not-yet-existing directory. + return read_next_trash_dir() + else + return cb(open_err) + end + end + local read_next + read_next = function() + uv.fs_readdir(fd, function(err, entries) + if err then + uv.fs_closedir(fd, function() + cb(err) + end) + return + elseif entries then + local internal_entries = {} + local poll = util.cb_collect(#entries, function(inner_err) + if inner_err then + cb(inner_err) + else + cb(nil, internal_entries, read_next) + end + end) + + for _, entry in ipairs(entries) do + read_trash_info( + fs.join(info_dir, entry.name), + vim.schedule_wrap(function(read_err, info) + if read_err then + -- Discard the error. We don't care if there's something wrong with one of these + -- files. + poll() + else + local parent = util.addslash(vim.fn.fnamemodify(info.original_path, ":h")) + if path == parent or show_all_files then + local name = vim.fn.fnamemodify(info.trash_file, ":t") + ---@diagnostic disable-next-line: undefined-field + local cache_entry = cache.create_entry(url, name, info.stat.type) + local display_name = vim.fn.fnamemodify(info.original_path, ":t") + cache_entry[FIELD_META] = { + stat = info.stat, + trash_info = info, + display_name = display_name, + } + table.insert(internal_entries, cache_entry) + end + if path ~= parent and (show_all_files or fs.is_subpath(path, parent)) then + local name = parent:sub(path:len() + 1) + local next_par = vim.fs.dirname(name) + while next_par ~= "." do + name = next_par + next_par = vim.fs.dirname(name) + end + ---@diagnostic disable-next-line: undefined-field + local cache_entry = cache.create_entry(url, name, "directory") + + cache_entry[FIELD_META] = { + stat = info.stat, + } + table.insert(internal_entries, cache_entry) + end + poll() + end + end) + ) + end + else + uv.fs_closedir(fd, function(close_err) + if close_err then + cb(close_err) + else + vim.schedule(read_next_trash_dir) + end + end) + end + end) + end + read_next() + ---@diagnostic disable-next-line: param-type-mismatch + end, 10000) + end + read_next_trash_dir() +end + +---@param bufnr integer +---@return boolean +M.is_modifiable = function(bufnr) + return true +end + +local file_columns = {} + +local current_year +-- Make sure we run this import-time effect in the main loop (mostly for tests) +vim.schedule(function() + current_year = vim.fn.strftime("%Y") +end) + +file_columns.mtime = { + render = function(entry, conf) + local meta = entry[FIELD_META] + ---@type oil.TrashInfo + local trash_info = meta.trash_info + local time = trash_info and trash_info.deletion_date or meta.stat and meta.stat.mtime.sec + if not time then + return nil + end + local fmt = conf and conf.format + local ret + if fmt then + ret = vim.fn.strftime(fmt, time) + else + local year = vim.fn.strftime("%Y", time) + if year ~= current_year then + ret = vim.fn.strftime("%b %d %Y", time) + else + ret = vim.fn.strftime("%b %d %H:%M", time) + end + end + return ret + end, + + get_sort_value = function(entry) + local meta = entry[FIELD_META] + ---@type nil|oil.TrashInfo + local trash_info = meta.trash_info + if trash_info then + return trash_info.deletion_date + else + return 0 + end + end, + + parse = function(line, conf) + local fmt = conf and conf.format + local pattern + if fmt then + pattern = fmt:gsub("%%.", "%%S+") + else + pattern = "%S+%s+%d+%s+%d%d:?%d%d" + end + return line:match("^(" .. pattern .. ")%s+(.+)$") + end, +} + +---@param name string +---@return nil|oil.ColumnDefinition +M.get_column = function(name) + return file_columns[name] +end + +M.supported_cross_adapter_actions = { files = "move" } + +---@param action oil.Action +---@return boolean +M.filter_action = function(action) + if action.type == "create" then + return false + elseif action.type == "delete" then + local entry = assert(cache.get_entry_by_url(action.url)) + local meta = entry[FIELD_META] + return meta.trash_info ~= nil + elseif action.type == "move" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + return src_adapter.name == "files" or dest_adapter.name == "files" + elseif action.type == "copy" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + return src_adapter.name == "files" or dest_adapter.name == "files" + else + error(string.format("Bad action type '%s'", action.type)) + end +end + +---@param err oil.ParseError +---@return boolean +M.filter_error = function(err) + if err.message == "Duplicate filename" then + return false + end + return true +end + +---@param action oil.Action +---@return string +M.render_action = function(action) + if action.type == "delete" then + local entry = assert(cache.get_entry_by_url(action.url)) + local meta = entry[FIELD_META] + ---@type oil.TrashInfo + local trash_info = meta.trash_info + local short_path = fs.shorten_path(trash_info.original_path) + return string.format(" PURGE %s", short_path) + elseif action.type == "move" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + if src_adapter.name == "files" then + local _, path = util.parse_url(action.src_url) + assert(path) + local short_path = files.to_short_os_path(path, action.entry_type) + return string.format(" TRASH %s", short_path) + elseif dest_adapter.name == "files" then + local _, path = util.parse_url(action.dest_url) + assert(path) + local short_path = files.to_short_os_path(path, action.entry_type) + return string.format("RESTORE %s", short_path) + else + error("Must be moving files into or out of trash") + end + elseif action.type == "copy" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + if src_adapter.name == "files" then + local _, path = util.parse_url(action.src_url) + assert(path) + local short_path = files.to_short_os_path(path, action.entry_type) + return string.format(" COPY %s -> TRASH", short_path) + elseif dest_adapter.name == "files" then + local _, path = util.parse_url(action.dest_url) + assert(path) + local short_path = files.to_short_os_path(path, action.entry_type) + return string.format("RESTORE %s", short_path) + else + error("Must be copying files into or out of trash") + end + else + error(string.format("Bad action type '%s'", action.type)) + end +end + +---@param trash_info oil.TrashInfo +---@param cb fun(err?: string) +local function purge(trash_info, cb) + fs.recursive_delete("file", trash_info.info_file, function(err) + if err then + return cb(err) + end + ---@diagnostic disable-next-line: undefined-field + fs.recursive_delete(trash_info.stat.type, trash_info.trash_file, cb) + end) +end + +---@param path string +---@param info_path string +---@param cb fun(err?: string) +local function write_info_file(path, info_path, cb) + uv.fs_open( + info_path, + "w", + 448, + vim.schedule_wrap(function(err, fd) + if err then + return cb(err) + end + assert(fd) + local deletion_date = vim.fn.strftime("%Y-%m-%dT%H:%M:%S") + local contents = string.format("[Trash Info]\nPath=%s\nDeletionDate=%s", path, deletion_date) + uv.fs_write(fd, contents, function(write_err) + uv.fs_close(fd, function(close_err) + cb(write_err or close_err) + end) + end) + end) + ) +end + +---@param path string +---@param cb fun(err?: string, trash_info?: oil.TrashInfo) +local function create_trash_info(path, cb) + local trash_dir = get_write_trash_dir(path) + local basename = vim.fs.basename(path) + local now = os.time() + local name = string.format("%s-%d.%d", basename, now, math.random(100000, 999999)) + local dest_path = fs.join(trash_dir, "files", name) + local dest_info = fs.join(trash_dir, "info", name .. ".trashinfo") + uv.fs_stat(path, function(err, stat) + if err then + return cb(err) + end + assert(stat) + write_info_file(path, dest_info, function(info_err) + if info_err then + return cb(info_err) + end + ---@type oil.TrashInfo + local trash_info = { + original_path = path, + trash_file = dest_path, + info_file = dest_info, + deletion_date = now, + stat = stat, + } + cb(nil, trash_info) + end) + end) +end + +---@param action oil.Action +---@param cb fun(err: nil|string) +M.perform_action = function(action, cb) + if action.type == "delete" then + local entry = assert(cache.get_entry_by_url(action.url)) + local meta = entry[FIELD_META] + ---@type oil.TrashInfo + local trash_info = meta.trash_info + purge(trash_info, cb) + elseif action.type == "move" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + if src_adapter.name == "files" then + local _, path = util.parse_url(action.src_url) + M.delete_to_trash(assert(path), cb) + elseif dest_adapter.name == "files" then + -- Restore + local _, dest_path = util.parse_url(action.dest_url) + assert(dest_path) + local entry = assert(cache.get_entry_by_url(action.src_url)) + local meta = entry[FIELD_META] + ---@type oil.TrashInfo + local trash_info = meta.trash_info + fs.recursive_move(action.entry_type, trash_info.trash_file, dest_path, function(err) + if err then + return cb(err) + end + uv.fs_unlink(trash_info.info_file, cb) + end) + else + error("Must be moving files into or out of trash") + end + elseif action.type == "copy" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + if src_adapter.name == "files" then + local _, path = util.parse_url(action.src_url) + assert(path) + create_trash_info(path, function(err, trash_info) + if err then + cb(err) + else + ---@diagnostic disable-next-line: undefined-field + local stat_type = trash_info.stat.type + fs.recursive_copy(stat_type, path, trash_info.trash_file, vim.schedule_wrap(cb)) + end + end) + elseif dest_adapter.name == "files" then + -- Restore + local _, dest_path = util.parse_url(action.dest_url) + assert(dest_path) + local entry = assert(cache.get_entry_by_url(action.src_url)) + local meta = entry[FIELD_META] + ---@type oil.TrashInfo + local trash_info = meta.trash_info + fs.recursive_copy(action.entry_type, trash_info.trash_file, dest_path, cb) + else + error("Must be moving files into or out of trash") + end + else + cb(string.format("Bad action type: %s", action.type)) + end +end + +---@param path string +---@param cb fun(err?: string) +M.delete_to_trash = function(path, cb) + create_trash_info(path, function(err, trash_info) + if err then + cb(err) + else + ---@diagnostic disable-next-line: undefined-field + local stat_type = trash_info.stat.type + fs.recursive_move(stat_type, path, trash_info.trash_file, vim.schedule_wrap(cb)) + end + end) +end + +return M diff --git a/lua/oil/adapters/trash/mac.lua b/lua/oil/adapters/trash/mac.lua new file mode 100644 index 00000000..46d4b4ae --- /dev/null +++ b/lua/oil/adapters/trash/mac.lua @@ -0,0 +1,233 @@ +local cache = require("oil.cache") +local config = require("oil.config") +local files = require("oil.adapters.files") +local fs = require("oil.fs") +local util = require("oil.util") + +local uv = vim.uv or vim.loop + +local M = {} + +local function touch_dir(path) + uv.fs_mkdir(path, 448) -- 0700 +end + +---Gets the location of the home trash dir, creating it if necessary +---@return string +local function get_trash_dir() + local trash_dir = fs.join(assert(uv.os_homedir()), ".Trash") + touch_dir(trash_dir) + return trash_dir +end + +---@param url string +---@param callback fun(url: string) +M.normalize_url = function(url, callback) + local scheme, path = util.parse_url(url) + assert(path) + callback(scheme .. "/") +end + +---@param url string +---@param entry oil.Entry +---@param cb fun(path: string) +M.get_entry_path = function(url, entry, cb) + local trash_dir = get_trash_dir() + local path = fs.join(trash_dir, entry.name) + if entry.type == "directory" then + path = "oil://" .. path + end + cb(path) +end + +---@param url string +---@param column_defs string[] +---@param cb fun(err?: string, entries?: oil.InternalEntry[], fetch_more?: fun()) +M.list = function(url, column_defs, cb) + cb = vim.schedule_wrap(cb) + local _, path = util.parse_url(url) + assert(path) + local trash_dir = get_trash_dir() + ---@diagnostic disable-next-line: param-type-mismatch + uv.fs_opendir(trash_dir, function(open_err, fd) + if open_err then + if open_err:match("^ENOENT: no such file or directory") then + -- If the directory doesn't exist, treat the list as a success. We will be able to traverse + -- and edit a not-yet-existing directory. + return cb() + else + return cb(open_err) + end + end + local read_next + read_next = function() + uv.fs_readdir(fd, function(err, entries) + if err then + uv.fs_closedir(fd, function() + cb(err) + end) + return + elseif entries then + local internal_entries = {} + local poll = util.cb_collect(#entries, function(inner_err) + if inner_err then + cb(inner_err) + else + cb(nil, internal_entries, read_next) + end + end) + + for _, entry in ipairs(entries) do + -- TODO: read .DS_Store and filter by original dir + local cache_entry = cache.create_entry(url, entry.name, entry.type) + table.insert(internal_entries, cache_entry) + poll() + end + else + uv.fs_closedir(fd, function(close_err) + if close_err then + cb(close_err) + else + cb() + end + end) + end + end) + end + read_next() + ---@diagnostic disable-next-line: param-type-mismatch + end, 10000) +end + +---@param bufnr integer +---@return boolean +M.is_modifiable = function(bufnr) + return true +end + +---@param name string +---@return nil|oil.ColumnDefinition +M.get_column = function(name) + return nil +end + +M.supported_cross_adapter_actions = { files = "move" } + +---@param action oil.Action +---@return string +M.render_action = function(action) + if action.type == "create" then + return string.format("CREATE %s", action.url) + elseif action.type == "delete" then + return string.format(" PURGE %s", action.url) + elseif action.type == "move" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + if src_adapter.name == "files" then + local _, path = util.parse_url(action.src_url) + assert(path) + local short_path = files.to_short_os_path(path, action.entry_type) + return string.format(" TRASH %s", short_path) + elseif dest_adapter.name == "files" then + local _, path = util.parse_url(action.dest_url) + assert(path) + local short_path = files.to_short_os_path(path, action.entry_type) + return string.format("RESTORE %s", short_path) + else + return string.format(" %s %s -> %s", action.type:upper(), action.src_url, action.dest_url) + end + elseif action.type == "copy" then + return string.format(" %s %s -> %s", action.type:upper(), action.src_url, action.dest_url) + else + error("Bad action type") + end +end + +---@param action oil.Action +---@param cb fun(err: nil|string) +M.perform_action = function(action, cb) + local trash_dir = get_trash_dir() + if action.type == "create" then + local _, path = util.parse_url(action.url) + assert(path) + path = trash_dir .. path + if action.entry_type == "directory" then + uv.fs_mkdir(path, 493, function(err) + -- Ignore if the directory already exists + if not err or err:match("^EEXIST:") then + cb() + else + cb(err) + end + end) -- 0755 + elseif action.entry_type == "link" and action.link then + local flags = nil + local target = fs.posix_to_os_path(action.link) + ---@diagnostic disable-next-line: param-type-mismatch + uv.fs_symlink(target, path, flags, cb) + else + fs.touch(path, cb) + end + elseif action.type == "delete" then + local _, path = util.parse_url(action.url) + assert(path) + local fullpath = trash_dir .. path + fs.recursive_delete(action.entry_type, fullpath, cb) + elseif action.type == "move" or action.type == "copy" then + local src_adapter = assert(config.get_adapter_by_scheme(action.src_url)) + local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) + local _, src_path = util.parse_url(action.src_url) + local _, dest_path = util.parse_url(action.dest_url) + assert(src_path and dest_path) + if src_adapter.name == "files" then + dest_path = trash_dir .. dest_path + elseif dest_adapter.name == "files" then + src_path = trash_dir .. src_path + else + dest_path = trash_dir .. dest_path + src_path = trash_dir .. src_path + end + + if action.type == "move" then + fs.recursive_move(action.entry_type, src_path, dest_path, cb) + else + fs.recursive_copy(action.entry_type, src_path, dest_path, cb) + end + else + cb(string.format("Bad action type: %s", action.type)) + end +end + +---@param path string +---@param cb fun(err?: string) +M.delete_to_trash = function(path, cb) + local basename = vim.fs.basename(path) + local trash_dir = get_trash_dir() + local dest = fs.join(trash_dir, basename) + uv.fs_stat( + path, + vim.schedule_wrap(function(stat_err, src_stat) + if stat_err then + return cb(stat_err) + end + assert(src_stat) + if uv.fs_stat(dest) then + local date_str = vim.fn.strftime(" %Y-%m-%dT%H:%M:%S") + local name_pieces = vim.split(basename, ".", { plain = true }) + if #name_pieces > 1 then + table.insert(name_pieces, #name_pieces - 1, date_str) + basename = table.concat(name_pieces) + else + basename = basename .. date_str + end + dest = fs.join(trash_dir, basename) + end + + local stat_type = src_stat.type + ---@cast stat_type oil.EntryType + fs.recursive_move(stat_type, path, dest, vim.schedule_wrap(cb)) + end) + ) +end + +return M diff --git a/lua/oil/adapters/trash/windows.lua b/lua/oil/adapters/trash/windows.lua new file mode 100644 index 00000000..f2c37c91 --- /dev/null +++ b/lua/oil/adapters/trash/windows.lua @@ -0,0 +1,20 @@ +-- Work in progress +local M = {} + +-- ---@return string +-- local function get_trash_dir() +-- -- TODO permission issues when using the recycle bin. The folder gets created without +-- -- read/write perms, so all operations fail +-- local cwd = vim.fn.getcwd() +-- local trash_dir = cwd:sub(1, 3) .. "$Recycle.Bin" +-- if vim.fn.isdirectory(trash_dir) == 1 then +-- return trash_dir +-- end +-- trash_dir = "C:\\$Recycle.Bin" +-- if vim.fn.isdirectory(trash_dir) == 1 then +-- return trash_dir +-- end +-- error("No trash found") +-- end + +return M diff --git a/lua/oil/cache.lua b/lua/oil/cache.lua index 0f6a27e6..09ed44fc 100644 --- a/lua/oil/cache.lua +++ b/lua/oil/cache.lua @@ -4,10 +4,12 @@ local M = {} local FIELD_ID = constants.FIELD_ID local FIELD_NAME = constants.FIELD_NAME +local FIELD_META = constants.FIELD_META local next_id = 1 -- Map> +---@type table> local url_directory = {} ---@type table @@ -118,6 +120,15 @@ M.get_entry_by_id = function(id) return entries_by_id[id] end +---@param url string +---@return nil|oil.InternalEntry +M.get_entry_by_url = function(url) + local scheme, path = util.parse_url(url) + local parent_url = scheme .. vim.fn.fnamemodify(path, ":h") + local basename = vim.fn.fnamemodify(path, ":t") + return M.list_url(parent_url)[basename] +end + ---@param id integer ---@return string M.get_parent_url = function(id) @@ -129,18 +140,12 @@ M.get_parent_url = function(id) end ---@param url string ----@return oil.InternalEntry[] +---@return table M.list_url = function(url) url = util.addslash(url) return url_directory[url] or {} end -M.get_entry_by_url = function(url) - local parent, name = url:match("^(.+)/([^/]+)$") - local cache = url_directory[parent] - return cache and cache[name] -end - ---@param action oil.Action M.perform_action = function(action) if action.type == "create" then @@ -172,6 +177,8 @@ M.perform_action = function(action) dest_parent = {} url_directory[dest_parent_url] = dest_parent end + -- We have to clear the metadata because it can be inaccurate after the move + entry[FIELD_META] = nil dest_parent[dest_name] = entry parent_url_by_id[entry[FIELD_ID]] = dest_parent_url entry[FIELD_NAME] = dest_name diff --git a/lua/oil/columns.lua b/lua/oil/columns.lua index 7140f2a2..7e217afa 100644 --- a/lua/oil/columns.lua +++ b/lua/oil/columns.lua @@ -221,6 +221,9 @@ if has_devicons then icon = conf and conf.directory or "" hl = "OilDirIcon" else + if meta and meta.display_name then + name = meta.display_name + end icon, hl = devicons.get_icon(name) icon = icon or (conf and conf.default_file or "") end diff --git a/lua/oil/config.lua b/lua/oil/config.lua index 8bc18027..0e419d26 100644 --- a/lua/oil/config.lua +++ b/lua/oil/config.lua @@ -1,3 +1,5 @@ +local uv = vim.uv or vim.loop + local default_config = { -- Oil will take over directory buffers (e.g. `vim .` or `:e src/`) -- Set to false if you still want to use netrw. @@ -30,8 +32,6 @@ local default_config = { delete_to_trash = false, -- Skip the confirmation popup for simple operations skip_confirm_for_simple_edits = false, - -- Change this to customize the command used when deleting to trash - trash_command = "trash-put", -- Selecting a new/moved/renamed file or directory will prompt you to save changes first prompt_save_on_select_new_entry = true, -- Oil will automatically delete hidden buffers after this delay @@ -60,6 +60,7 @@ local default_config = { ["gs"] = "actions.change_sort", ["gx"] = "actions.open_external", ["g."] = "actions.toggle_hidden", + ["g\\"] = "actions.toggle_trash", }, -- Set to false to disable all of the above keymaps use_default_keymaps = true, @@ -142,6 +143,7 @@ local default_config = { default_config.adapters = { ["oil://"] = "files", ["oil-ssh://"] = "ssh", + ["oil-trash://"] = "trash", } default_config.adapter_aliases = {} @@ -154,13 +156,10 @@ M.setup = function(opts) end if new_conf.delete_to_trash then - local trash_bin = vim.split(new_conf.trash_command, " ")[1] - if vim.fn.executable(trash_bin) == 0 then + local is_windows = uv.os_uname().version:match("Windows") + if is_windows then vim.notify( - string.format( - "oil.nvim: delete_to_trash is true, but '%s' executable not found.\nDeleted files will be permanently removed.", - new_conf.trash_command - ), + "oil.nvim: delete_to_trash is true, but trash is not yet supported on Windows.\nDeleted files will be permanently removed", vim.log.levels.WARN ) new_conf.delete_to_trash = false @@ -176,46 +175,6 @@ M.setup = function(opts) M.adapter_to_scheme[v] = k end M._adapter_by_scheme = {} - if type(M.trash) == "string" then - M.trash = vim.fn.fnamemodify(vim.fn.expand(M.trash), ":p") - end -end - ----@return nil|string -M.get_trash_url = function() - if not M.trash then - return nil - end - local fs = require("oil.fs") - if M.trash == true then - local data_home = os.getenv("XDG_DATA_HOME") or vim.fn.expand("~/.local/share") - local preferred = fs.join(data_home, "trash") - local candidates = { - preferred, - } - if fs.is_windows then - -- TODO permission issues when using the recycle bin. The folder gets created without - -- read/write perms, so all operations fail - -- local cwd = vim.fn.getcwd() - -- table.insert(candidates, 1, cwd:sub(1, 3) .. "$Recycle.Bin") - -- table.insert(candidates, 1, "C:\\$Recycle.Bin") - else - table.insert(candidates, fs.join(data_home, "Trash", "files")) - table.insert(candidates, fs.join(os.getenv("HOME"), ".Trash")) - end - local trash_dir = preferred - for _, candidate in ipairs(candidates) do - if vim.fn.isdirectory(candidate) == 1 then - trash_dir = candidate - break - end - end - - local oil_trash_dir = vim.fn.fnamemodify(fs.join(trash_dir, "nvim", "oil"), ":p") - fs.mkdirp(oil_trash_dir) - M.trash = oil_trash_dir - end - return M.adapter_to_scheme.files .. fs.os_to_posix_path(M.trash) end ---@param scheme nil|string diff --git a/lua/oil/fs.lua b/lua/oil/fs.lua index 4f612a50..3de1acd6 100644 --- a/lua/oil/fs.lua +++ b/lua/oil/fs.lua @@ -7,6 +7,8 @@ M.is_windows = uv.os_uname().version:match("Windows") M.is_mac = uv.os_uname().sysname == "Darwin" +M.is_linux = not M.is_windows and not M.is_mac + ---@type string M.sep = M.is_windows and "\\" or "/" @@ -114,20 +116,31 @@ end local home_dir = assert(uv.os_homedir()) ---@param path string +---@param relative_to? string Shorten relative to this path (default cwd) ---@return string -M.shorten_path = function(path) - local cwd = vim.fn.getcwd() - if M.is_subpath(cwd, path) then - local relative = path:sub(cwd:len() + 2) - if relative == "" then - relative = "." +M.shorten_path = function(path, relative_to) + if not relative_to then + relative_to = vim.fn.getcwd() + end + local relpath + if M.is_subpath(relative_to, path) then + local idx = relative_to:len() + 1 + -- Trim the dividing slash if it's not included in relative_to + if not vim.endswith(relative_to, "/") and not vim.endswith(relative_to, "\\") then + idx = idx + 1 + end + relpath = path:sub(idx) + if relpath == "" then + relpath = "." end - return relative end if M.is_subpath(home_dir, path) then - return "~" .. path:sub(home_dir:len() + 1) + local homepath = "~" .. path:sub(home_dir:len() + 1) + if not relpath or homepath:len() < relpath:len() then + return homepath + end end - return path + return relpath or path end M.mkdirp = function(dir) @@ -177,7 +190,7 @@ M.listdir = function(dir, cb) end read_next() ---@diagnostic disable-next-line: param-type-mismatch - end, 100) -- TODO do some testing for this + end, 10000) end ---@param entry_type oil.EntryType diff --git a/lua/oil/init.lua b/lua/oil/init.lua index 8cdbd301..94424cd2 100644 --- a/lua/oil/init.lua +++ b/lua/oil/init.lua @@ -8,6 +8,7 @@ local M = {} ---@alias oil.EntryType "file"|"directory"|"socket"|"link"|"fifo" ---@alias oil.TextChunk string|string[] +---@alias oil.CrossAdapterAction "copy"|"move" ---@class (exact) oil.Adapter ---@field name string The unique name of the adapter (this will be set automatically) @@ -20,7 +21,9 @@ local M = {} ---@field perform_action? fun(action: oil.Action, cb: fun(err: nil|string)) Perform a mutation action. Only needed if adapter is modifiable. ---@field read_file? fun(bufnr: integer) Used for adapters that deal with remote/virtual files. Read the contents of the file into a buffer. ---@field write_file? fun(bufnr: integer) Used for adapters that deal with remote/virtual files. Write the contents of a buffer to the destination. ----@field supported_adapters_for_copy? table Mapping of adapter name to true for all other adapters that can be used as a src or dest for move/copy actions. +---@field supported_cross_adapter_actions? table Mapping of adapter name to enum for all other adapters that can be used as a src or dest for move/copy actions. +---@field filter_action? fun(action: oil.Action): boolean When present, filter out actions as they are created +---@field filter_error? fun(action: oil.ParseError): boolean When present, filter out errors from parsing a buffer -- TODO remove after https://github.com/folke/neodev.nvim/pull/163 lands ---@diagnostic disable: undefined-field @@ -110,34 +113,6 @@ M.discard_all_changes = function() end end ----Delete all files in the trash directory ----@private ----@note ---- Trash functionality is incomplete and experimental. -M.empty_trash = function() - local config = require("oil.config") - local fs = require("oil.fs") - local util = require("oil.util") - local trash_url = config.get_trash_url() - if not trash_url then - vim.notify("No trash directory configured", vim.log.levels.WARN) - return - end - local _, path = util.parse_url(trash_url) - assert(path) - local dir = fs.posix_to_os_path(path) - if vim.fn.isdirectory(dir) == 1 then - fs.recursive_delete("directory", dir, function(err) - if err then - vim.notify(string.format("Error emptying trash: %s", err), vim.log.levels.ERROR) - else - vim.notify("Trash emptied") - fs.mkdirp(dir) - end - end) - end -end - ---Change the display columns for oil ---@param cols oil.ColumnSpec[] M.set_columns = function(cols) @@ -177,9 +152,13 @@ end ---Get the oil url for a given directory ---@private ---@param dir nil|string When nil, use the cwd ----@return nil|string The parent url +---@param use_oil_parent nil|boolean If in an oil buffer, return the parent (default true) +---@return string The parent url ---@return nil|string The basename (if present) of the file/dir we were just in -M.get_url_for_path = function(dir) +M.get_url_for_path = function(dir, use_oil_parent) + if use_oil_parent == nil then + use_oil_parent = true + end local config = require("oil.config") local fs = require("oil.fs") local util = require("oil.util") @@ -196,15 +175,16 @@ M.get_url_for_path = function(dir) return config.adapter_to_scheme.files .. path else local bufname = vim.api.nvim_buf_get_name(0) - return M.get_buffer_parent_url(bufname) + return M.get_buffer_parent_url(bufname, use_oil_parent) end end ---@private ---@param bufname string +---@param use_oil_parent boolean If in an oil buffer, return the parent ---@return string ---@return nil|string -M.get_buffer_parent_url = function(bufname) +M.get_buffer_parent_url = function(bufname, use_oil_parent) local config = require("oil.config") local fs = require("oil.fs") local pathutil = require("oil.pathutil") @@ -223,13 +203,15 @@ M.get_buffer_parent_url = function(bufname) return parent_url, basename else assert(path) - -- TODO maybe we should remove this special case and turn it into a config if scheme == "term://" then ---@type string path = vim.fn.expand(path:match("^(.*)//")) ---@diagnostic disable-line: assign-type-mismatch return config.adapter_to_scheme.files .. util.addslash(path) end + if not use_oil_parent then + return bufname + end local adapter = config.get_adapter_by_scheme(scheme) local parent_url if adapter and adapter.get_parent then @@ -672,7 +654,7 @@ M._get_highlights = function() { name = "OilDir", link = "Directory", - desc = "Directories in an oil buffer", + desc = "Directory names in an oil buffer", }, { name = "OilDirIcon", @@ -689,6 +671,11 @@ M._get_highlights = function() link = nil, desc = "Soft links in an oil buffer", }, + { + name = "OilLinkTarget", + link = "Comment", + desc = "The target of a soft link", + }, { name = "OilFile", link = nil, @@ -719,6 +706,26 @@ M._get_highlights = function() link = "Special", desc = "Change action in the oil preview window", }, + { + name = "OilRestore", + link = "OilCreate", + desc = "Restore (from the trash) action in the oil preview window", + }, + { + name = "OilPurge", + link = "OilDelete", + desc = "Purge (Permanently delete a file from trash) action in the oil preview window", + }, + { + name = "OilTrash", + link = "OilDelete", + desc = "Trash (delete a file to trash) action in the oil preview window", + }, + { + name = "OilTrashSourcePath", + link = "Comment", + desc = "Virtual text that shows the original path of file in the trash", + }, } end @@ -855,14 +862,23 @@ M.setup = function(opts) config.setup(opts) set_colors() vim.api.nvim_create_user_command("Oil", function(args) + local util = require("oil.util") if args.smods.tab == 1 then vim.cmd.tabnew() end local float = false - for i, v in ipairs(args.fargs) do + local trash = false + local i = 1 + while i <= #args.fargs do + local v = args.fargs[i] if v == "--float" then float = true table.remove(args.fargs, i) + elseif v == "--trash" then + trash = true + table.remove(args.fargs, i) + else + i = i + 1 end end @@ -875,7 +891,13 @@ M.setup = function(opts) end local method = float and "open_float" or "open" - M[method](unpack(args.fargs)) + local path = args.fargs[1] + if trash then + local url = M.get_url_for_path(path, false) + local _, new_path = util.parse_url(url) + path = "oil-trash://" .. new_path + end + M[method](path) end, { desc = "Open oil file browser on a directory", nargs = "*", complete = "dir" }) local aug = vim.api.nvim_create_augroup("Oil", {}) diff --git a/lua/oil/mutator/init.lua b/lua/oil/mutator/init.lua index c3bfc5ea..29d04a6c 100644 --- a/lua/oil/mutator/init.lua +++ b/lua/oil/mutator/init.lua @@ -7,7 +7,6 @@ local constants = require("oil.constants") local lsp_helpers = require("oil.lsp_helpers") local oil = require("oil") local parser = require("oil.mutator.parser") -local pathutil = require("oil.pathutil") local preview = require("oil.mutator.preview") local util = require("oil.util") local view = require("oil.view") @@ -54,6 +53,7 @@ M.create_actions_from_diffs = function(all_diffs) ---@type oil.Action[] local actions = {} + ---@type table local diff_by_id = setmetatable({}, { __index = function(t, key) local list = {} @@ -61,6 +61,15 @@ M.create_actions_from_diffs = function(all_diffs) return list end, }) + ---@param action oil.Action + local function add_action(action) + local adapter = assert(config.get_adapter_by_scheme(action.dest_url or action.url)) + if not adapter.filter_action or adapter.filter_action(action) then + table.insert(actions, action) + end + end + ---@type table + local dest_by_id = {} for bufnr, diffs in pairs(all_diffs) do local adapter = util.get_adapter(bufnr) if not adapter then @@ -71,9 +80,7 @@ M.create_actions_from_diffs = function(all_diffs) if diff.type == "new" then if diff.id then local by_id = diff_by_id[diff.id] - -- FIXME this is kind of a hack. We shouldn't be setting undocumented fields on the diff - ---@diagnostic disable-next-line: inject-field - diff.dest = parent_url .. diff.name + dest_by_id[diff.id] = parent_url .. diff.name table.insert(by_id, diff) else -- Parse nested files like foo/bar/baz @@ -87,7 +94,7 @@ M.create_actions_from_diffs = function(all_diffs) -- Parse alternations like foo.{js,test.js} for _, alt in ipairs(vim.split(alternation, ",")) do local alt_url = url .. "/" .. v:gsub("{[^}]+}", alt) - table.insert(actions, { + add_action({ type = "create", url = alt_url, entry_type = entry_type, @@ -96,7 +103,7 @@ M.create_actions_from_diffs = function(all_diffs) end else url = url .. "/" .. v - table.insert(actions, { + add_action({ type = "create", url = url, entry_type = entry_type, @@ -106,7 +113,7 @@ M.create_actions_from_diffs = function(all_diffs) end end elseif diff.type == "change" then - table.insert(actions, { + add_action({ type = "change", url = parent_url .. diff.name, entry_type = diff.entry_type, @@ -115,8 +122,9 @@ M.create_actions_from_diffs = function(all_diffs) }) else local by_id = diff_by_id[diff.id] + -- HACK: set has_delete field on a list-like table of diffs by_id.has_delete = true - -- Don't insert the delete. We already know that there is a delete because of the presense + -- Don't insert the delete. We already know that there is a delete because of the presence -- in the diff_by_id map. The list will only include the 'new' diffs. end end @@ -127,21 +135,23 @@ M.create_actions_from_diffs = function(all_diffs) if not entry then error(string.format("Could not find entry %d", id)) end + ---HACK: access the has_delete field on the list-like table of diffs + ---@diagnostic disable-next-line: undefined-field if diffs.has_delete then local has_create = #diffs > 0 if has_create then -- MOVE (+ optional copies) when has both creates and delete for i, diff in ipairs(diffs) do - table.insert(actions, { + add_action({ type = i == #diffs and "move" or "copy", entry_type = entry[FIELD_TYPE], - dest_url = diff.dest, + dest_url = dest_by_id[diff.id], src_url = cache.get_parent_url(id) .. entry[FIELD_NAME], }) end else -- DELETE when no create - table.insert(actions, { + add_action({ type = "delete", entry_type = entry[FIELD_TYPE], url = cache.get_parent_url(id) .. entry[FIELD_NAME], @@ -150,11 +160,11 @@ M.create_actions_from_diffs = function(all_diffs) else -- COPY when create but no delete for _, diff in ipairs(diffs) do - table.insert(actions, { + add_action({ type = "copy", entry_type = entry[FIELD_TYPE], src_url = cache.get_parent_url(id) .. entry[FIELD_NAME], - dest_url = diff.dest, + dest_url = dest_by_id[diff.id], }) end end @@ -353,30 +363,6 @@ end ---@param actions oil.Action[] ---@param cb fun(err: nil|string) M.process_actions = function(actions, cb) - -- convert delete actions to move-to-trash - local trash_url = config.get_trash_url() - if trash_url then - for i, v in ipairs(actions) do - if v.type == "delete" then - local scheme, path = util.parse_url(v.url) - if config.adapters[scheme] == "files" then - assert(path) - ---@type oil.MoveAction - local move_action = { - type = "move", - src_url = v.url, - entry_type = v.entry_type, - dest_url = trash_url .. "/" .. pathutil.basename(path) .. string.format( - "_%06d", - math.random(999999) - ), - } - actions[i] = move_action - end - end - end - end - -- send all renames to LSP servers local moves = {} for _, action in ipairs(actions) do @@ -390,12 +376,12 @@ M.process_actions = function(actions, cb) end lsp_helpers.will_rename_files(moves) - -- Convert cross-adapter moves to a copy + delete + -- Convert some cross-adapter moves to a copy + delete for _, action in ipairs(actions) do if action.type == "move" then - local src_scheme = util.parse_url(action.src_url) - local dest_scheme = util.parse_url(action.dest_url) - if src_scheme ~= dest_scheme then + local _, cross_action = util.get_adapter_for_action(action) + -- Only do the conversion if the cross-adapter support is "copy" + if cross_action == "copy" then action.type = "copy" table.insert(actions, { type = "delete", @@ -488,6 +474,10 @@ M.try_write_changes = function(confirm) if vim.bo[bufnr].modified then local diffs, errors = parser.parse(bufnr) all_diffs[bufnr] = diffs + local adapter = assert(util.get_adapter(bufnr)) + if adapter.filter_error then + errors = vim.tbl_filter(adapter.filter_error, errors) + end if not vim.tbl_isempty(errors) then all_errors[bufnr] = errors end @@ -539,7 +529,7 @@ M.try_write_changes = function(confirm) view.unlock_buffers() if err then vim.notify(string.format("[oil] Error applying actions: %s", err), vim.log.levels.ERROR) - view.rerender_all_oil_buffers({ preserve_undo = false }) + view.rerender_all_oil_buffers() else local current_entry = oil.get_cursor_entry() if current_entry then @@ -549,7 +539,8 @@ M.try_write_changes = function(confirm) vim.split(current_entry.parsed_name or current_entry.name, "/")[1] ) end - view.rerender_all_oil_buffers({ preserve_undo = M.trash }) + view.rerender_all_oil_buffers() + vim.api.nvim_exec_autocmds("User", { pattern = "OilMutationComplete", modeline = false }) end mutation_in_progress = false end) diff --git a/lua/oil/mutator/parser.lua b/lua/oil/mutator/parser.lua index 1c0e32d2..e3823a15 100644 --- a/lua/oil/mutator/parser.lua +++ b/lua/oil/mutator/parser.lua @@ -142,11 +142,18 @@ M.parse_line = function(adapter, line, column_defs) return { data = ret, entry = entry, ranges = ranges } end +---@class (exact) oil.ParseError +---@field lnum integer +---@field col integer +---@field message string + ---@param bufnr integer ----@return oil.Diff[] ----@return table[] Parsing errors +---@return oil.Diff[] diffs +---@return oil.ParseError[] errors Parsing errors M.parse = function(bufnr) + ---@type oil.Diff[] local diffs = {} + ---@type oil.ParseError[] local errors = {} local bufname = vim.api.nvim_buf_get_name(bufnr) local adapter = util.get_adapter(bufnr) @@ -158,11 +165,14 @@ M.parse = function(bufnr) }) return diffs, errors end + + local lines = vim.api.nvim_buf_get_lines(bufnr, 0, -1, true) local scheme, path = util.parse_url(bufname) - local parent_url = scheme .. path local column_defs = columns.get_supported_columns(adapter) + local parent_url = scheme .. path local children = cache.list_url(parent_url) - local lines = vim.api.nvim_buf_get_lines(bufnr, 0, -1, true) + -- map from name to entry ID for all entries previously in the buffer + ---@type table local original_entries = {} for _, child in pairs(children) do local name = child[FIELD_NAME] @@ -184,6 +194,7 @@ M.parse = function(bufnr) end for i, line in ipairs(lines) do if line:match("^/%d+") then + -- Parse the line for an existing entry local result, err = M.parse_line(adapter, line, column_defs) if not result or err then table.insert(errors, { @@ -256,6 +267,7 @@ M.parse = function(bufnr) end end else + -- Parse a new entry local name, isdir = parsedir(vim.trim(line)) if vim.startswith(name, "/") then table.insert(errors, { diff --git a/lua/oil/util.lua b/lua/oil/util.lua index 4020d70d..027b5db8 100644 --- a/lua/oil/util.lua +++ b/lua/oil/util.lua @@ -453,6 +453,7 @@ end ---@param action oil.Action ---@return oil.Adapter +---@return nil|oil.CrossAdapterAction M.get_adapter_for_action = function(action) local adapter = config.get_adapter_by_scheme(action.url or action.src_url) if not adapter then @@ -462,15 +463,15 @@ M.get_adapter_for_action = function(action) local dest_adapter = assert(config.get_adapter_by_scheme(action.dest_url)) if adapter ~= dest_adapter then if - adapter.supported_adapters_for_copy - and adapter.supported_adapters_for_copy[dest_adapter.name] + adapter.supported_cross_adapter_actions + and adapter.supported_cross_adapter_actions[dest_adapter.name] then - return adapter + return adapter, adapter.supported_cross_adapter_actions[dest_adapter.name] elseif - dest_adapter.supported_adapters_for_copy - and dest_adapter.supported_adapters_for_copy[adapter.name] + dest_adapter.supported_cross_adapter_actions + and dest_adapter.supported_cross_adapter_actions[adapter.name] then - return dest_adapter + return dest_adapter, dest_adapter.supported_cross_adapter_actions[adapter.name] else error( string.format( diff --git a/lua/oil/view.lua b/lua/oil/view.lua index d9023392..4a2a7d03 100644 --- a/lua/oil/view.lua +++ b/lua/oil/view.lua @@ -1,7 +1,9 @@ +local uv = vim.uv or vim.loop local cache = require("oil.cache") local columns = require("oil.columns") local config = require("oil.config") local constants = require("oil.constants") +local fs = require("oil.fs") local keymap_util = require("oil.keymap_util") local loading = require("oil.loading") local util = require("oil.util") @@ -142,10 +144,11 @@ M.unlock_buffers = function() end end ----@param opts table +---@param opts? table ---@note --- This DISCARDS ALL MODIFICATIONS a user has made to oil buffers M.rerender_all_oil_buffers = function(opts) + opts = opts or {} local buffers = M.get_all_buffers() local hidden_buffers = {} for _, bufnr in ipairs(buffers) do @@ -177,8 +180,8 @@ end ---Get a list of visible oil buffers and a list of hidden oil buffers ---@note --- If any buffers are modified, return values are nil ----@return nil|integer[] ----@return nil|integer[] +---@return nil|integer[] visible +---@return nil|integer[] hidden local function get_visible_hidden_buffers() local buffers = M.get_all_buffers() local hidden_buffers = {} @@ -227,6 +230,43 @@ local function get_first_mutable_column_col(adapter, ranges) return min_col end +---Redraw original path virtual text for trash buffer +---@param bufnr integer +local function redraw_trash_virtual_text(bufnr) + if not vim.api.nvim_buf_is_valid(bufnr) or not vim.api.nvim_buf_is_loaded(bufnr) then + return + end + local parser = require("oil.mutator.parser") + local adapter = util.get_adapter(bufnr) + if not adapter or adapter.name ~= "trash" then + return + end + local _, buf_path = util.parse_url(vim.api.nvim_buf_get_name(bufnr)) + local os_path = fs.posix_to_os_path(assert(buf_path)) + local ns = vim.api.nvim_create_namespace("OilVtext") + vim.api.nvim_buf_clear_namespace(bufnr, ns, 0, -1) + local column_defs = columns.get_supported_columns(adapter) + for lnum, line in ipairs(vim.api.nvim_buf_get_lines(bufnr, 0, -1, true)) do + local result = parser.parse_line(adapter, line, column_defs) + local entry = result and result.entry + if entry then + local meta = entry[FIELD_META] + ---@type nil|oil.TrashInfo + local trash_info = meta and meta.trash_info + if trash_info then + vim.api.nvim_buf_set_extmark(bufnr, ns, lnum - 1, 0, { + virt_text = { + { + "➜ " .. fs.shorten_path(trash_info.original_path, os_path), + "OilTrashSourcePath", + }, + }, + }) + end + end + end +end + ---@param bufnr integer M.initialize = function(bufnr) if bufnr == 0 then @@ -255,7 +295,7 @@ M.initialize = function(bufnr) nested = true, buffer = bufnr, callback = function() - -- First wait a short time (10ms) for the buffer change to settle + -- First wait a short time (100ms) for the buffer change to settle vim.defer_fn(function() local visible_buffers = get_visible_hidden_buffers() -- Only delete oil buffers if none of them are visible @@ -271,7 +311,7 @@ M.initialize = function(bufnr) end end end - end, 10) + end, 100) end, }) vim.api.nvim_create_autocmd("BufDelete", { @@ -351,6 +391,36 @@ M.initialize = function(bufnr) end) end, }) + + -- Watch for TextChanged and update the trash original path extmarks + local adapter = util.get_adapter(bufnr) + if adapter and adapter.name == "trash" then + local debounce_timer = assert(uv.new_timer()) + local pending = false + vim.api.nvim_create_autocmd("TextChanged", { + desc = "Update oil virtual text of original path", + buffer = bufnr, + callback = function() + -- Respond immediately to prevent flickering, the set the timer for a "cooldown period" + -- If this is called again during the cooldown window, we will rerender after cooldown. + if debounce_timer:is_active() then + pending = true + else + redraw_trash_virtual_text(bufnr) + end + debounce_timer:start( + 50, + 0, + vim.schedule_wrap(function() + if pending then + pending = false + redraw_trash_virtual_text(bufnr) + end + end) + ) + end, + }) + end M.render_buffer_async(bufnr, {}, function(err) if err then vim.notify( @@ -358,6 +428,7 @@ M.initialize = function(bufnr) vim.log.levels.ERROR ) else + vim.b[bufnr].oil_ready = true vim.api.nvim_exec_autocmds( "User", { pattern = "OilEnter", modeline = false, data = { buf = bufnr } } @@ -478,6 +549,7 @@ local function render_buffer(bufnr, opts) vim.bo[bufnr].modifiable = false vim.bo[bufnr].modified = false util.set_highlights(bufnr, highlights) + if opts.jump then -- TODO why is the schedule necessary? vim.schedule(function() @@ -511,6 +583,10 @@ end ---@return oil.TextChunk[] M.format_entry_cols = function(entry, column_defs, col_width, adapter) local name = entry[FIELD_NAME] + local meta = entry[FIELD_META] + if meta and meta.display_name then + name = meta.display_name + end -- First put the unique ID local cols = {} local id_key = cache.format_id(entry[FIELD_ID]) @@ -531,7 +607,6 @@ M.format_entry_cols = function(entry, column_defs, col_width, adapter) elseif entry_type == "socket" then table.insert(cols, { name, "OilSocket" }) elseif entry_type == "link" then - local meta = entry[FIELD_META] local link_text if meta then if meta.link_stat and meta.link_stat.type == "directory" then @@ -548,7 +623,7 @@ M.format_entry_cols = function(entry, column_defs, col_width, adapter) table.insert(cols, { name, "OilLink" }) if link_text then - table.insert(cols, { link_text, "Comment" }) + table.insert(cols, { link_text, "OilLinkTarget" }) end else table.insert(cols, { name, "OilFile" }) @@ -573,29 +648,22 @@ end ---@param bufnr integer ---@param opts nil|table ---- preserve_undo nil|boolean --- refetch nil|boolean Defaults to true ---@param callback nil|fun(err: nil|string) M.render_buffer_async = function(bufnr, opts, callback) opts = vim.tbl_deep_extend("keep", opts or {}, { - preserve_undo = false, refetch = true, }) if bufnr == 0 then bufnr = vim.api.nvim_get_current_buf() end local bufname = vim.api.nvim_buf_get_name(bufnr) - local scheme, dir = util.parse_url(bufname) - local preserve_undo = opts.preserve_undo and config.adapters[scheme] == "files" - if not preserve_undo then - -- Undo should not return to a blank buffer - -- Method taken from :h clear-undo - vim.bo[bufnr].undolevels = -1 - end + local _, dir = util.parse_url(bufname) + -- Undo should not return to a blank buffer + -- Method taken from :h clear-undo + vim.bo[bufnr].undolevels = -1 local handle_error = vim.schedule_wrap(function(message) - if not preserve_undo then - vim.bo[bufnr].undolevels = vim.api.nvim_get_option_value("undolevels", { scope = "global" }) - end + vim.bo[bufnr].undolevels = vim.api.nvim_get_option_value("undolevels", { scope = "global" }) util.render_text(bufnr, { "Error: " .. message }) if callback then callback(message) @@ -612,7 +680,7 @@ M.render_buffer_async = function(bufnr, opts, callback) handle_error(string.format("[oil] no adapter for buffer '%s'", bufname)) return end - local start_ms = vim.loop.hrtime() / 1e6 + local start_ms = uv.hrtime() / 1e6 local seek_after_render_found = false local first = true vim.bo[bufnr].modifiable = false @@ -624,9 +692,7 @@ M.render_buffer_async = function(bufnr, opts, callback) end loading.set_loading(bufnr, false) render_buffer(bufnr, { jump = true }) - if not preserve_undo then - vim.bo[bufnr].undolevels = vim.api.nvim_get_option_value("undolevels", { scope = "global" }) - end + vim.bo[bufnr].undolevels = vim.api.nvim_get_option_value("undolevels", { scope = "global" }) vim.bo[bufnr].modifiable = not buffers_locked and adapter.is_modifiable(bufnr) if callback then callback() @@ -651,7 +717,7 @@ M.render_buffer_async = function(bufnr, opts, callback) end end if fetch_more then - local now = vim.loop.hrtime() / 1e6 + local now = uv.hrtime() / 1e6 local delta = now - start_ms -- If we've been chugging for more than 40ms, go ahead and render what we have if delta > 40 then diff --git a/scripts/generate.py b/scripts/generate.py index 2f8e263f..114af0b5 100755 --- a/scripts/generate.py +++ b/scripts/generate.py @@ -121,22 +121,44 @@ class ColumnDef: "An icon for the entry's type (requires nvim-web-devicons)", HL + [ - LuaParam("default_file", "string", "Fallback icon for files when nvim-web-devicons returns nil"), + LuaParam( + "default_file", + "string", + "Fallback icon for files when nvim-web-devicons returns nil", + ), LuaParam("directory", "string", "Icon for directories"), - LuaParam("add_padding", "boolean", "Set to false to remove the extra whitespace after the icon"), + LuaParam( + "add_padding", + "boolean", + "Set to false to remove the extra whitespace after the icon", + ), ], ), ColumnDef("size", "files, ssh", False, True, "The size of the file", HL + []), ColumnDef( - "permissions", "files, ssh", True, False, "Access permissions of the file", HL + [] + "permissions", + "files, ssh", + True, + False, + "Access permissions of the file", + HL + [], + ), + ColumnDef( + "ctime", "files", False, True, "Change timestamp of the file", HL + TIME + [] ), - ColumnDef("ctime", "files", False, True, "Change timestamp of the file", HL + TIME + []), ColumnDef( "mtime", "files", False, True, "Last modified time of the file", HL + TIME + [] ), - ColumnDef("atime", "files", False, True, "Last access time of the file", HL + TIME + []), ColumnDef( - "birthtime", "files", False, True, "The time the file was created", HL + TIME + [] + "atime", "files", False, True, "Last access time of the file", HL + TIME + [] + ), + ColumnDef( + "birthtime", + "files", + False, + True, + "The time the file was created", + HL + TIME + [], ), ] @@ -170,7 +192,7 @@ def get_actions_vimdoc() -> "VimdocSection": section = VimdocSection("Actions", "oil-actions", ["\n"]) section.body.extend( wrap( - "These are actions that can be used in the `keymaps` section of config options." + """These are actions that can be used in the `keymaps` section of config options. You can also call them directly with `require("oil.actions").action_name.callback()`""" ) ) section.body.append("\n") @@ -210,6 +232,37 @@ def get_columns_vimdoc() -> "VimdocSection": return section +def get_trash_vimdoc() -> "VimdocSection": + section = VimdocSection("Trash", "oil-trash", []) + section.body.append( + """ +Oil has built-in support for using the system trash. When +`delete_to_trash = true`, any deleted files will be sent to the trash instead +of being permanently deleted. You can browse the trash for a directory using +the `toggle_trash` action (bound to `g\\` by default). You can view all files +in the trash with `:Oil --trash /`. + +To restore files, simply delete them from the trash and put them in the desired +destination, the same as any other file operation. If you delete files from the +trash they will be permanently deleted (purged). + +Linux: + Oil supports the FreeDesktop trash specification. + https://specifications.freedesktop.org/trash-spec/trashspec-1.0.html + All features should work. + +Mac: + Oil has limited support for MacOS due to the proprietary nature of the + implementation. The trash bin can only be viewed as a single dir + (instead of being able to see files that were trashed from a directory). + +Windows: + Oil does not yet support the Windows trash. PRs are welcome! +""" + ) + return section + + def generate_vimdoc(): doc = Vimdoc("oil.txt", "oil") funcs = parse_functions(os.path.join(ROOT, "lua", "oil", "init.lua")) @@ -220,6 +273,7 @@ def generate_vimdoc(): get_columns_vimdoc(), get_actions_vimdoc(), get_highlights_vimdoc(), + get_trash_vimdoc(), ] ) diff --git a/syntax/oil_preview.vim b/syntax/oil_preview.vim index 41656ee3..b6c2fab7 100644 --- a/syntax/oil_preview.vim +++ b/syntax/oil_preview.vim @@ -3,9 +3,13 @@ if exists("b:current_syntax") endif syn match oilCreate /^CREATE / -syn match oilMove /^ MOVE / +syn match oilMove /^ MOVE / syn match oilDelete /^DELETE / -syn match oilCopy /^ COPY / +syn match oilCopy /^ COPY / syn match oilChange /^CHANGE / +" Trash operations +syn match oilRestore /^RESTORE / +syn match oilPurge /^ PURGE / +syn match oilTrash /^ TRASH / let b:current_syntax = "oil_preview" diff --git a/tests/files_spec.lua b/tests/files_spec.lua index 3e5e4668..b268b4c3 100644 --- a/tests/files_spec.lua +++ b/tests/files_spec.lua @@ -11,8 +11,6 @@ a.describe("files adapter", function() a.after_each(function() if tmpdir then tmpdir:dispose() - a.util.scheduler() - tmpdir = nil end test_util.reset_editor() end) diff --git a/tests/minimal_init.lua b/tests/minimal_init.lua index 262d9ece..70a66b18 100644 --- a/tests/minimal_init.lua +++ b/tests/minimal_init.lua @@ -1,4 +1,4 @@ -vim.cmd([[set runtimepath+=.]]) +vim.opt.runtimepath:append(".") vim.o.swapfile = false vim.bo.swapfile = false diff --git a/tests/parser_spec.lua b/tests/parser_spec.lua index f3a4262b..74872638 100644 --- a/tests/parser_spec.lua +++ b/tests/parser_spec.lua @@ -18,6 +18,7 @@ describe("parser", function() after_each(function() test_util.reset_editor() end) + it("detects new files", function() vim.cmd.edit({ args = { "oil-test:///foo/" } }) local bufnr = vim.api.nvim_get_current_buf() diff --git a/tests/test_util.lua b/tests/test_util.lua index e2ab2aa6..53d4bfed 100644 --- a/tests/test_util.lua +++ b/tests/test_util.lua @@ -25,6 +25,18 @@ M.reset_editor = function() test_adapter.test_clear() end +local function throwiferr(err, ...) + if err then + error(err) + else + return ... + end +end + +M.await = function(fn, nargs, ...) + return throwiferr(a.wrap(fn, nargs)(...)) +end + M.wait_for_autocmd = a.wrap(function(autocmd, cb) local opts = { pattern = "*", @@ -58,4 +70,48 @@ M.feedkeys = function(actions, timestep) a.util.sleep(timestep) end +M.actions = { + ---Open oil and wait for it to finish rendering + ---@param args string[] + open = function(args) + vim.schedule(function() + vim.cmd.Oil({ args = args }) + -- If this buffer was already open, manually dispatch the autocmd to finish the wait + if vim.b.oil_ready then + vim.api.nvim_exec_autocmds("User", { + pattern = "OilEnter", + modeline = false, + data = { buf = vim.api.nvim_get_current_buf() }, + }) + end + end) + M.wait_for_autocmd({ "User", pattern = "OilEnter" }) + end, + + ---Save all changes and wait for operation to complete + save = function() + vim.schedule_wrap(require("oil").save)({ confirm = false }) + M.wait_for_autocmd({ "User", pattern = "OilMutationComplete" }) + end, + + ---@param bufnr? integer + reload = function(bufnr) + M.await(require("oil.view").render_buffer_async, 3, bufnr or 0) + end, + + ---Move cursor to a file or directory in an oil buffer + ---@param filename string + focus = function(filename) + local lines = vim.api.nvim_buf_get_lines(0, 0, -1, true) + local search = " " .. filename .. "$" + for i, line in ipairs(lines) do + if line:match(search) then + vim.api.nvim_win_set_cursor(0, { i, 0 }) + return + end + end + error("Could not find file " .. filename) + end, +} + return M diff --git a/tests/tmpdir.lua b/tests/tmpdir.lua index 62018f45..bb8a9c0a 100644 --- a/tests/tmpdir.lua +++ b/tests/tmpdir.lua @@ -1,16 +1,7 @@ local fs = require("oil.fs") +local test_util = require("tests.test_util") -local function throwiferr(err, ...) - if err then - error(err) - else - return ... - end -end - -local function await(fn, nargs, ...) - return throwiferr(a.wrap(fn, nargs)(...)) -end +local await = test_util.await ---@param path string ---@param cb fun(err: nil|string) @@ -41,6 +32,7 @@ local TmpDir = {} TmpDir.new = function() local path = await(vim.loop.fs_mkdtemp, 2, "oil_test_XXXXXXXXX") + a.util.scheduler() return setmetatable({ path = path }, { __index = TmpDir, }) @@ -60,6 +52,7 @@ function TmpDir:create(paths) end end end + a.util.scheduler() end ---@param filepath string @@ -72,6 +65,7 @@ local read_file = function(filepath) local stat = vim.loop.fs_fstat(fd) local content = vim.loop.fs_read(fd, stat.size) vim.loop.fs_close(fd) + a.util.scheduler() return content end @@ -99,9 +93,9 @@ local assert_fs = function(root, paths) local pieces = vim.split(k, "/") local partial_path = "" for i, piece in ipairs(pieces) do - partial_path = fs.join(partial_path, piece) .. "/" + partial_path = partial_path .. piece .. "/" if i ~= #pieces then - unlisted_dirs[partial_path:sub(2)] = true + unlisted_dirs[partial_path] = true end end end @@ -152,8 +146,23 @@ function TmpDir:assert_fs(paths) assert_fs(self.path, paths) end +function TmpDir:assert_exists(path) + a.util.scheduler() + path = fs.join(self.path, path) + local stat = vim.loop.fs_stat(path) + assert.truthy(stat, string.format("Expected path '%s' to exist", path)) +end + +function TmpDir:assert_not_exists(path) + a.util.scheduler() + path = fs.join(self.path, path) + local stat = vim.loop.fs_stat(path) + assert.falsy(stat, string.format("Expected path '%s' to not exist", path)) +end + function TmpDir:dispose() await(fs.recursive_delete, 3, "directory", self.path) + a.util.scheduler() end return TmpDir diff --git a/tests/trash_spec.lua b/tests/trash_spec.lua new file mode 100644 index 00000000..e74ae77b --- /dev/null +++ b/tests/trash_spec.lua @@ -0,0 +1,164 @@ +local uv = vim.uv or vim.loop +require("plenary.async").tests.add_to_env() +local TmpDir = require("tests.tmpdir") +local fs = require("oil.fs") +local test_util = require("tests.test_util") + +---Get the raw list of filenames from an unmodified oil buffer +---@param bufnr? integer +---@return string[] +local function parse_entries(bufnr) + bufnr = bufnr or 0 + if vim.bo[bufnr].modified then + error("parse_entries doesn't work on a modified oil buffer") + end + local lines = vim.api.nvim_buf_get_lines(bufnr, 0, -1, true) + return vim.tbl_map(function(line) + return line:match("^/%d+ +(.+)$") + end, lines) +end + +a.describe("freedesktop", function() + local tmpdir + a.before_each(function() + require("oil.config").delete_to_trash = true + tmpdir = TmpDir.new() + package.loaded["oil.adapters.trash"] = require("oil.adapters.trash.freedesktop") + local trash_dir = string.format(".Trash-%d", uv.getuid()) + tmpdir:create({ fs.join(trash_dir, "__dummy__") }) + end) + a.after_each(function() + if tmpdir then + tmpdir:dispose() + end + test_util.reset_editor() + package.loaded["oil.adapters.trash"] = nil + end) + + a.it("files can be moved to the trash", function() + tmpdir:create({ "a.txt", "foo/b.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.open({ "--trash", tmpdir.path }) + vim.api.nvim_feedkeys("p", "x", true) + test_util.actions.save() + tmpdir:assert_not_exists("a.txt") + tmpdir:assert_exists("foo/b.txt") + test_util.actions.reload() + assert.are.same({ "a.txt" }, parse_entries(0)) + end) + + a.it("deleting a file moves it to trash", function() + tmpdir:create({ "a.txt", "foo/b.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + tmpdir:assert_not_exists("a.txt") + tmpdir:assert_exists("foo/b.txt") + test_util.actions.open({ "--trash", tmpdir.path }) + assert.are.same({ "a.txt" }, parse_entries(0)) + end) + + a.it("deleting a directory moves it to trash", function() + tmpdir:create({ "a.txt", "foo/b.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("foo/") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + tmpdir:assert_not_exists("foo") + tmpdir:assert_exists("a.txt") + test_util.actions.open({ "--trash", tmpdir.path }) + assert.are.same({ "foo/" }, parse_entries(0)) + end) + + a.it("deleting a file from trash deletes it permanently", function() + tmpdir:create({ "a.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + test_util.actions.open({ "--trash", tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + test_util.actions.reload() + tmpdir:assert_not_exists("a.txt") + assert.are.same({}, parse_entries(0)) + end) + + a.it("cannot create files in the trash", function() + tmpdir:create({ "a.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + test_util.actions.open({ "--trash", tmpdir.path }) + vim.api.nvim_feedkeys("onew_file.txt", "x", true) + test_util.actions.save() + test_util.actions.reload() + assert.are.same({ "a.txt" }, parse_entries(0)) + end) + + a.it("cannot rename files in the trash", function() + tmpdir:create({ "a.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + test_util.actions.open({ "--trash", tmpdir.path }) + vim.api.nvim_feedkeys("0facwnew_name", "x", true) + test_util.actions.save() + test_util.actions.reload() + assert.are.same({ "a.txt" }, parse_entries(0)) + end) + + a.it("cannot copy files in the trash", function() + tmpdir:create({ "a.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + test_util.actions.open({ "--trash", tmpdir.path }) + vim.api.nvim_feedkeys("yypp", "x", true) + test_util.actions.save() + test_util.actions.reload() + assert.are.same({ "a.txt" }, parse_entries(0)) + end) + + a.it("can restore files from trash", function() + tmpdir:create({ "a.txt" }) + test_util.actions.open({ tmpdir.path }) + test_util.actions.focus("a.txt") + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + test_util.actions.open({ "--trash", tmpdir.path }) + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.open({ tmpdir.path }) + vim.api.nvim_feedkeys("p", "x", true) + test_util.actions.save() + test_util.actions.reload() + assert.are.same({ "a.txt" }, parse_entries(0)) + local uid = uv.getuid() + tmpdir:assert_fs({ + ["a.txt"] = "a.txt", + [".Trash-" .. uid .. "/__dummy__"] = ".Trash-" .. uid .. "/__dummy__", + [".Trash-" .. uid .. "/files/"] = true, + [".Trash-" .. uid .. "/info/"] = true, + }) + end) + + a.it("can have multiple files with the same name in trash", function() + tmpdir:create({ "a.txt" }) + test_util.actions.open({ tmpdir.path }) + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + tmpdir:create({ "a.txt" }) + test_util.actions.reload() + vim.api.nvim_feedkeys("dd", "x", true) + test_util.actions.save() + test_util.actions.open({ "--trash", tmpdir.path }) + assert.are.same({ "a.txt", "a.txt" }, parse_entries(0)) + end) +end) diff --git a/tests/url_spec.lua b/tests/url_spec.lua index d9ecb309..32d801eb 100644 --- a/tests/url_spec.lua +++ b/tests/url_spec.lua @@ -13,7 +13,7 @@ describe("url", function() } for _, case in ipairs(cases) do local input, expected, expected_basename = unpack(case) - local output, basename = oil.get_buffer_parent_url(input) + local output, basename = oil.get_buffer_parent_url(input, true) assert.equals(expected, output, string.format('Parent url for path "%s" failed', input)) assert.equals( expected_basename,