Skip to content

Commit 1a8dc5f

Browse files
authored
Merge pull request #8 from willothy/pipe-from-term
feat: Allow piping from term into new buffer
2 parents 3ff516e + 73a0a02 commit 1a8dc5f

File tree

2 files changed

+110
-36
lines changed

2 files changed

+110
-36
lines changed

lua/flatten/core.lua

+70-26
Original file line numberDiff line numberDiff line change
@@ -41,14 +41,25 @@ local function notify_when_done(pipe, bufnr, callback, ft)
4141
})
4242
end
4343

44-
M.edit_files = function(args, response_pipe, guest_cwd)
44+
M.edit_files = function(args, response_pipe, guest_cwd, stdin)
4545
local config = require("flatten").config
4646
local callbacks = config.callbacks
4747
local focus_first = config.window.focus == "first"
4848
local open = config.window.open
4949

50+
local nargs = #args
51+
local stdin_lines = #stdin
52+
53+
if nargs == 0 and stdin_lines == 0 then
54+
-- If there are no new bufs, don't open anything
55+
-- and tell the guest not to block
56+
return false
57+
end
58+
5059
callbacks.pre_open()
51-
if #args > 0 then
60+
61+
-- Open files
62+
if nargs > 0 then
5263
local argstr = ""
5364
for _, arg in ipairs(args) do
5465
local p = vim.loop.fs_realpath(arg) or guest_cwd .. '/' .. arg
@@ -60,37 +71,67 @@ M.edit_files = function(args, response_pipe, guest_cwd)
6071
end
6172

6273
vim.cmd("0argadd " .. argstr)
74+
end
6375

64-
if type(open) == "function" then
65-
-- Pass list of new buffer IDs
66-
local bufs = vim.api.nvim_list_bufs()
67-
local start = #bufs - #args
68-
local newbufs = {}
69-
for i, buf in ipairs(bufs) do
70-
if i > start then
71-
table.insert(newbufs, buf)
72-
end
76+
-- Create buffer for stdin pipe input
77+
local stdin_buf = nil
78+
if stdin_lines > 0 then
79+
-- Create buffer for stdin
80+
stdin_buf = vim.api.nvim_create_buf(true, false)
81+
-- Add text to buffer
82+
vim.api.nvim_buf_set_lines(stdin_buf, 0, 0, true, stdin)
83+
-- Set buffer name based on the first line of stdin
84+
local name = stdin[1]:sub(1, 12):gsub("[^%w%.]", "")
85+
-- Ensure the name isn't empty or a duplicate
86+
if vim.fn.bufname(name) ~= "" or name == "" then
87+
local i = 1
88+
local newname = name .. i
89+
while vim.fn.bufname(newname) ~= "" do
90+
i = i + 1
91+
newname = name .. i
7392
end
74-
open(newbufs)
75-
elseif type(open) == "string" then
76-
local focus = vim.fn.argv(focus_first and 0 or (#args - 1))
77-
if open == "current" then
78-
vim.cmd("edit " .. focus)
79-
elseif open == "split" then
80-
vim.cmd("split " .. focus)
81-
elseif open == "vsplit" then
82-
vim.cmd("vsplit " .. focus)
83-
else
84-
vim.cmd("tabedit " .. focus)
93+
name = newname
94+
end
95+
vim.api.nvim_buf_set_name(stdin_buf, name)
96+
end
97+
98+
-- Open window
99+
if type(open) == "function" then
100+
-- Pass list of new buffer IDs
101+
local bufs = vim.api.nvim_list_bufs()
102+
local start = #bufs - #args
103+
-- Add buffer for stdin
104+
local newbufs = {}
105+
-- If there's an stdin buf, push it to the table
106+
if stdin_buf then
107+
start = start - 1
108+
table.insert(newbufs, stdin_buf)
109+
end
110+
for i, buf in ipairs(bufs) do
111+
if i > start then
112+
table.insert(newbufs, buf)
85113
end
114+
end
115+
open(newbufs)
116+
elseif type(open) == "string" then
117+
local focus = vim.fn.argv(focus_first and 0 or (#args - 1))
118+
-- If there's an stdin buf, focus that
119+
if stdin_buf then
120+
focus = vim.api.nvim_buf_get_name(stdin_buf)
121+
end
122+
if open == "current" then
123+
vim.cmd("edit " .. focus)
124+
elseif open == "split" then
125+
vim.cmd("split " .. focus)
126+
elseif open == "vsplit" then
127+
vim.cmd("vsplit " .. focus)
86128
else
87-
vim.api.nvim_err_writeln("Flatten: 'config.open.focus' expects a function or string, got " .. type(open))
129+
vim.cmd("tabedit " .. focus)
88130
end
89131
else
90-
-- If there weren't any args, don't open anything
91-
-- and tell the guest not to block
92-
return false
132+
vim.api.nvim_err_writeln("Flatten: 'config.open.focus' expects a function or string, got " .. type(open))
93133
end
134+
94135
local ft = vim.bo.filetype
95136

96137
local winnr = vim.api.nvim_get_current_win()
@@ -100,6 +141,9 @@ M.edit_files = function(args, response_pipe, guest_cwd)
100141
local block = config.block_for[ft] == true
101142
if block then
102143
notify_when_done(response_pipe, bufnr, callbacks.block_end, ft)
144+
-- else
145+
-- local response_sock = vim.fn.sockconnect("pipe", response_pipe, { rpc = true })
146+
-- vim.fn.rpcnotify(response_sock, "nvim_exec_lua", "vim.cmd('qa!')")
103147
end
104148
return block
105149
end

lua/flatten/guest.lua

+40-10
Original file line numberDiff line numberDiff line change
@@ -1,31 +1,61 @@
11
local M = {}
22

3-
M.init = function(host_pipe)
4-
local args = vim.fn.argv()
5-
6-
local host = vim.fn.sockconnect("pipe", host_pipe, { rpc = true })
3+
local function send_files(host, files, stdin)
4+
if #files < 1 and #stdin < 1 then return end
75

86
local call = string.format([[
97
return require('flatten.core').edit_files(
108
%s, -- `args` passed into nested instance.
119
'%s', -- guest default socket.
12-
'%s' -- guest global cwd.
10+
'%s', -- guest global cwd.
11+
%s -- stdin lines or {}.
1312
)]],
14-
vim.inspect(args),
13+
vim.inspect(files),
1514
vim.v.servername,
16-
vim.fn.getcwd()
15+
vim.fn.getcwd(),
16+
vim.inspect(stdin)
1717
)
1818

19-
if #args < 1 then return end
20-
2119
local block = vim.fn.rpcrequest(host, "nvim_exec_lua", call, {})
2220
if not block then
23-
vim.cmd("qa!")
21+
vim.cmd('qa!')
2422
end
2523
vim.fn.chanclose(host)
2624
while true do
2725
vim.cmd("sleep 1")
2826
end
2927
end
3028

29+
M.init = function(host_pipe)
30+
-- Connect to host process
31+
local host = vim.fn.sockconnect("pipe", host_pipe, { rpc = true })
32+
-- Exit on connection error
33+
if host == 0 then vim.cmd("qa!") end
34+
35+
-- Get new files
36+
local files = vim.fn.argv()
37+
local nfiles = #files
38+
39+
vim.api.nvim_create_autocmd("StdinReadPost", {
40+
pattern = '*',
41+
callback = function()
42+
local readlines = vim.api.nvim_buf_get_lines(0, 0, -1, true)
43+
send_files(host, files, readlines)
44+
end
45+
})
46+
47+
-- No arguments, user is probably opening a nested session intentionally
48+
-- Or only piping input from stdin
49+
vim.api.nvim_create_autocmd("BufEnter", {
50+
pattern = '*',
51+
callback = function()
52+
if nfiles < 1 then
53+
vim.cmd('qa!')
54+
end
55+
56+
send_files(host, files, {})
57+
end
58+
})
59+
end
60+
3161
return M

0 commit comments

Comments
 (0)