ComputerCraft Archive

7.multishell

computer operating-system kepler155c github

Description

ComputerCraft OS

Installation

Copy one of these commands into your ComputerCraft terminal:

wget:wget https://raw.githubusercontent.com/kepler155c/opus/develop-1.8/sys/init/7.multishell.lua 7.multishell
Archive:wget https://cc.shobie.xyz/cc/get/gh-kepler155c-opus-sys-init-7-multishell 7.multishell
Quick Install: wget https://cc.shobie.xyz/cc/get/gh-kepler155c-opus-sys-init-7-multishell 7.multishell

Usage

Run: 7.multishell

Tags

none

Source

View Original Source

Code Preview

local Blit     = require('opus.ui.blit')
local Config   = require('opus.config')
local Util     = require('opus.util')

local colors     = _G.colors
local fs         = _G.fs
local kernel     = _G.kernel
local keys       = _G.keys
local os         = _G.os
local printError = _G.printError
local window     = _G.window

local parentTerm = _G.device.terminal
local w,h = parentTerm.getSize()
local overviewId
local tabsDirty = false
local closeInd = Util.getVersion() >= 1.76 and '\215' or '*'
local multishell = { }

_ENV.multishell = multishell

kernel.window.reposition(1, 2, w, h - 1)

local config = {
	standard = {
		textColor  = colors.lightGray,
		tabBarTextColor = colors.lightGray,
		focusTextColor = colors.white,
		backgroundColor = colors.gray,
		tabBarBackgroundColor = colors.gray,
		focusBackgroundColor = colors.gray,
		errorColor = colors.black,
	},
	color = {
		textColor  = colors.lightGray,
		tabBarTextColor = colors.lightGray,
		focusTextColor = colors.white,
		backgroundColor = colors.gray,
		tabBarBackgroundColor = colors.gray,
		focusBackgroundColor = colors.gray,
		errorColor = colors.red,
	},
}
Config.load('multishell', config)

local _colors = parentTerm.isColor() and config.color or config.standard
local palette = parentTerm.isColor() and Blit.colorPalette or Blit.grayscalePalette

local function redrawMenu()
	if not tabsDirty then
		os.queueEvent('multishell_redraw')
		tabsDirty = true
	end
end

function multishell.getFocus()
	local currentTab = kernel.getFocused()
	return currentTab.uid
end

function multishell.setFocus(tabId)
	return kernel.raise(tabId)
end

function multishell.getTitle(tabId)
	local tab = kernel.find(tabId)
	return tab and tab.title
end

function multishell.setTitle(tabId, title)
	local tab = kernel.find(tabId)
	if tab then
		tab.title = title
		redrawMenu()
	end
end

function multishell.getCurrent()
	local runningTab = kernel.getCurrent()
	return runningTab and runningTab.uid
end

function multishell.getTab(tabId)
	return kernel.find(tabId)
end

function multishell.terminate(tabId)
	os.queueEvent('multishell_terminate', tabId)
end

function multishell.getTabs()
	return kernel.routines
end

function multishell.launch(env, path, ...)
	-- backwards compatibility
	return multishell.openTab(env, {
		path = path,
		args = { ... },
	})
end

local function chain(orig, fn)
	if not orig then
		return fn
	end

	if type(orig) == 'table' then
		table.insert(orig, fn)
		return orig
	end

	return setmetatable({ orig, fn }, {
		__call = function(self, ...)
			for _,v in pairs(self) do
				v(...)
			end
		end
	})
end

function multishell.openTab(env, tab)
	if not tab.title and tab.path then
		tab.title = fs.getName(tab.path):match('([^%.]+)')
	end
	tab.title = tab.title or 'untitled'
	tab.window = tab.window or window.create(parentTerm, 1, 2, w, h - 1, false)
		-- require('opus.terminal').window(parentTerm, 1, 2, w, h - 1, false)
	tab.onExit = chain(tab.onExit, function(self, result, err, stack)
		if not result and err and err ~= 'Terminated' then
			self.terminal.setTextColor(colors.white)
			self.terminal.setCursorBlink(false)
			print('\nThe program terminated with an error.\n')
			if tonumber(err) then
				printError('Process exited with error code: ' .. err)
			elseif err then
				printError(tostring(err))
			end
			if type(stack) == 'table' and #stack > 0 then
				local _, cy = self.terminal.getCursorPos()
				local _, th = self.terminal.getSize()
				self.terminal.setTextColor(colors.white)
				if cy < th - 4 then
					print('\nstack traceback:')
					for _, v in ipairs(stack or { }) do
						_, cy = self.terminal.getCursorPos()
						if cy > th - 3 then
							print(' ...')
							break
						end
						print(v)
					end
				end
			end
			self.terminal.setTextColor(parentTerm.isColor() and colors.yellow or colors.white)
			_G.write('\nPress enter to close')
			self.isDead = true
			self.hidden = false
			redrawMenu()
			while true do
				local e, code = os.pullEventRaw('key')
				if e == 'terminate' or e == 'key' and code == keys.enter then
					break
				end
			end
		end
	end)

	local routine, message = kernel.run(env, tab)

	if routine then
		if tab.focused then
			multishell.setFocus(routine.uid)
		else
			redrawMenu()
		end
	end

	return routine and routine.uid, message
end

function multishell.hideTab(tabId)
	local tab = kernel.find(tabId)
	if tab then
		tab.hidden = true
		kernel.lower(tab.uid)
		redrawMenu()
	end
end

function multishell.unhideTab(tabId)
	local tab = kernel.find(tabId)
	if tab then
		tab.hidden = false
		redrawMenu()
	end
end

function multishell.getCount()
	return #kernel.routines
end

kernel.hook('kernel_focus', function()
	redrawMenu()
end)

kernel.hook('multishell_terminate', function(_, eventData)
	local tab = kernel.find(eventData[1])
	if tab and not tab.isOverview then
		if coroutine.status(tab.co) ~= 'dead' then
			tab:resume("terminate")
		end
	end
	return true
end)

kernel.hook('terminate', function()
	return kernel.getFocused().isOverview
end)

kernel.hook('multishell_redraw', function()
	tabsDirty = false

	local blit = Blit(w, {
		bg = _colors.tabBarBackgroundColor,
		fg = _colors.textColor,
		palette = palette,
	})

	local currentTab = kernel.getFocused()

	for _,tab in pairs(kernel.routines) do
		if tab.hidden and tab ~= currentTab then
			tab.width = 0
		else
			tab.width = #tab.title + 1
		end
	end

	local function width()
		local tw = 0
		Util.each(kernel.routines, function(t) tw = tw + t.width end)
		return tw
	end

	while width() > w - 3 do
		local tab = select(2,
			Util.spairs(kernel.routines, function(a, b) return a.width > b.width end)())
		tab.width = tab.width - 1
	end

	local function compareTab(a, b)
		if a.hidden then return false end
		return b.hidden or a.uid < b.uid
	end

	local tabX = 0
	for _,tab in Util.spairs(kernel.routines, compareTab) do
		if tab.width > 0 then
			tab.sx = tabX + 1
			tab.ex = tabX + tab.width
			tabX = tabX + tab.width
			if tab ~= currentTab then
				local textColor = tab.isDead and _colors.errorColor or _colors.textColor
				blit:write(tab.sx, tab.title:sub(1, tab.width - 1),
					_colors.backgroundColor, textColor)
			end
		end
	end

	if currentTab then
		if currentTab.sx then
			local textColor = currentTab.isDead and _colors.errorColor or _colors.focusTextColor
			blit:write(currentTab.sx - 1,
				' ' .. currentTab.title:sub(1, currentTab.width - 1) .. ' ',
				_colors.focusBackgroundColor, textColor)
		end
		if not currentTab.noTerminate then
			blit:write(w, closeInd, nil, _colors.focusTextColor)
		end
	end

	parentTerm.setCursorPos(1, 1)
	parentTerm.blit(blit.text, blit.fg, blit.bg)

	if currentTab and currentTab.window then
		currentTab.window.restoreCursor()
	end

	return true
end)

kernel.hook('term_resize', function(_, eventData)
	if not eventData[1] then                            --- TEST
		w,h = parentTerm.getSize()

		local windowHeight = h-1

		for _,key in pairs(Util.keys(kernel.routines)) do
			local tab = kernel.routines[key]
			local x,y = tab.window.getCursorPos()
			if y > windowHeight then
				tab.window.scroll(y - windowHeight)
				tab.window.setCursorPos(x, windowHeight)
			end
			tab.window.reposition(1, 2, w, windowHeight)
		end

		redrawMenu()
	end
end)

kernel.hook('mouse_click', function(_, eventData)
	if not eventData[4] then
		local x, y = eventData[2], eventData[3]

		if y == 1 then
			if x == 1 then
				multishell.setFocus(overviewId)
			elseif x == w then
				local currentTab = kernel.getFocused()
				if currentTab then
					multishell.terminate(currentTab.uid)
				end
			else
				for _,tab in pairs(kernel.routines) do
					if not tab.hidden and tab.sx then
						if x >= tab.sx and x <= tab.ex then
							multishell.setFocus(tab.uid)
							break
						end
					end
				end
			end
			return true
		end
		eventData[3] = eventData[3] - 1
	end
end)

kernel.hook({ 'mouse_up', 'mouse_drag' }, function(_, eventData)
	if not eventData[4] then
		eventData[3] = eventData[3] - 1
	end
end)

kernel.hook('mouse_scroll', function(_, eventData)
	if not eventData[4] then
		if eventData[3] == 1 then
			return true
		end
		eventData[3] = eventData[3] - 1
	end
end)

kernel.hook('kernel_ready', function()
	overviewId = multishell.openTab(_ENV, {
		path = 'sys/apps/shell.lua',
		args = { config.launcher or 'sys/apps/Overview.lua' },
		isOverview = true,
		noTerminate = true,
		focused = true,
		title = '+',
		onExit = function(_, s, m)
			if not s then
				kernel.halt(s, m)
			end
		end,
	})
	multishell.setTitle(overviewId, '+')

	multishell.openTab(_ENV, {
		path = 'sys/apps/shell.lua',
		args = { 'sys/apps/autorun.lua' },
		title = 'Autorun',
	})
end)