1
0
mirror of https://github.com/SquidDev-CC/CC-Tweaked synced 2025-01-21 22:46:57 +00:00

Merge branch 'master' into mc-1.14.x

This commit is contained in:
SquidDev 2020-04-23 10:03:45 +01:00
commit a049502d12
15 changed files with 496 additions and 153 deletions

View File

@ -1,4 +1,4 @@
# ![CC: Tweaked](logo.png) [![Download CC: Tweaked on CurseForge](http://cf.way2muchnoise.eu/title/cc-tweaked.svg)](https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked on CurseForge") # ![CC: Tweaked](logo.png) [![Download CC: Tweaked on CurseForge](https://cf.way2muchnoise.eu/title/cc-tweaked.svg)](https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked on CurseForge")
CC: Tweaked is a fork of [ComputerCraft], adding programmable computers, turtles and more to Minecraft. CC: Tweaked is a fork of [ComputerCraft], adding programmable computers, turtles and more to Minecraft.

View File

@ -23,7 +23,7 @@ body {
"Droid Sans", "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol"; "Droid Sans", "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
} }
code, pre, .parameter, .type, .definition-name, .reference { code, pre, .parameter, .type, .definition-name, .reference-code {
font-family: "SFMono-Regular", Consolas, "Liberation Mono", Menlo, Courier, monospace; font-family: "SFMono-Regular", Consolas, "Liberation Mono", Menlo, Courier, monospace;
} }

View File

@ -1,5 +1,5 @@
# Mod properties # Mod properties
mod_version=1.86.2 mod_version=1.87.0
# Minecraft properties (update mods.toml when changing) # Minecraft properties (update mods.toml when changing)
mc_version=1.14.4 mc_version=1.14.4

View File

@ -2,10 +2,11 @@
(sources (sources
/doc/stub/ /doc/stub/
/src/main/resources/data/computercraft/lua/bios.lua /src/main/resources/*/computercraft/lua/bios.lua
/src/main/resources/data/computercraft/lua/rom/ /src/main/resources/*/computercraft/lua/rom/
/src/test/resources/test-rom) /src/test/resources/test-rom)
(doc (doc
(title "CC: Tweaked") (title "CC: Tweaked")
(index doc/index.md) (index doc/index.md)
@ -14,13 +15,13 @@
(library-path (library-path
/doc/stub/ /doc/stub/
/src/main/resources/data/computercraft/lua/rom/apis /src/main/resources/*/computercraft/lua/rom/apis
/src/main/resources/data/computercraft/lua/rom/apis/command /src/main/resources/*/computercraft/lua/rom/apis/command
/src/main/resources/data/computercraft/lua/rom/apis/turtle /src/main/resources/*/computercraft/lua/rom/apis/turtle
/src/main/resources/data/computercraft/lua/rom/modules/main /src/main/resources/*/computercraft/lua/rom/modules/main
/src/main/resources/data/computercraft/lua/rom/modules/command /src/main/resources/*/computercraft/lua/rom/modules/command
/src/main/resources/data/computercraft/lua/rom/modules/turtle)) /src/main/resources/*/computercraft/lua/rom/modules/turtle))
(at / (at /
(linters (linters
@ -34,9 +35,7 @@
;; be good to find a compromise in the future, but this works for now. ;; be good to find a compromise in the future, but this works for now.
-var:unused-arg -var:unused-arg
;; Suppress a couple of documentation comments warnings for now. We'll ;; Some APIS (keys, colour and os mainly) are incomplete right now.
;; hopefully be able to remove them in the future.
-doc:undocumented -doc:undocumented-arg -doc:unresolved-reference
-var:unresolved-member) -var:unresolved-member)
(lint (lint
(bracket-spaces (bracket-spaces
@ -49,8 +48,8 @@
;; We disable the unused global linter in bios.lua and the APIs. In the future ;; We disable the unused global linter in bios.lua and the APIs. In the future
;; hopefully we'll get illuaminate to handle this. ;; hopefully we'll get illuaminate to handle this.
(at (at
(/src/main/resources/data/computercraft/lua/bios.lua (/src/main/resources/*/computercraft/lua/bios.lua
/src/main/resources/data/computercraft/lua/rom/apis/) /src/main/resources/*/computercraft/lua/rom/apis/)
(linters -var:unused-global) (linters -var:unused-global)
(lint (allow-toplevel-global true))) (lint (allow-toplevel-global true)))
@ -59,19 +58,27 @@
(linters -var:unused-global) (linters -var:unused-global)
(lint (allow-toplevel-global true))) (lint (allow-toplevel-global true)))
;; Ensure any fully documented modules stay fully documented. ;; Suppress warnings for currently undocumented modules.
(at (at
(/src/main/resources/data/computercraft/lua/rom/apis/colors.lua (/doc/stub/commands.lua
/src/main/resources/data/computercraft/lua/rom/apis/colours.lua /doc/stub/fs.lua
/src/main/resources/data/computercraft/lua/rom/apis/disk.lua /doc/stub/http.lua
/src/main/resources/data/computercraft/lua/rom/apis/gps.lua /doc/stub/os.lua
/src/main/resources/data/computercraft/lua/rom/apis/help.lua /doc/stub/redstone.lua
/src/main/resources/data/computercraft/lua/rom/apis/keys.lua /doc/stub/term.lua
/src/main/resources/data/computercraft/lua/rom/apis/paintutils.lua /doc/stub/turtle.lua
/src/main/resources/data/computercraft/lua/rom/apis/parallel.lua /src/main/resources/*/computercraft/lua/rom/apis/command/commands.lua
/src/main/resources/data/computercraft/lua/rom/apis/peripheral.lua /src/main/resources/*/computercraft/lua/rom/apis/io.lua
/src/main/resources/data/computercraft/lua/rom/apis/rednet.lua /src/main/resources/*/computercraft/lua/rom/apis/window.lua
/src/main/resources/data/computercraft/lua/rom/apis/settings.lua /src/main/resources/*/computercraft/lua/rom/modules/main/cc/shell/completion.lua)
/src/main/resources/data/computercraft/lua/rom/apis/texutils.lua
/src/main/resources/data/computercraft/lua/rom/apis/vector.lua) (linters -doc:undocumented -doc:undocumented-arg))
(linters doc:undocumented doc:undocumented-arg))
;; These currently rely on unknown references.
(at
(/src/main/resources/*/computercraft/lua/rom/apis/textutils.lua
/src/main/resources/*/computercraft/lua/rom/modules/main/cc/completion.lua
/src/main/resources/*/computercraft/lua/rom/modules/main/cc/shell/completion.lua
/src/main/resources/*/computercraft/lua/rom/programs/advanced/multishell.lua
/src/main/resources/*/computercraft/lua/rom/programs/shell.lua)
(linters -doc:unresolved-reference))

View File

@ -521,14 +521,11 @@ function os.run(_tEnv, _sPath, ...)
expect(1, _tEnv, "table") expect(1, _tEnv, "table")
expect(2, _sPath, "string") expect(2, _sPath, "string")
local tArgs = table.pack(...)
local tEnv = _tEnv local tEnv = _tEnv
setmetatable(tEnv, { __index = _G }) setmetatable(tEnv, { __index = _G })
local fnFile, err = loadfile(_sPath, nil, tEnv) local fnFile, err = loadfile(_sPath, nil, tEnv)
if fnFile then if fnFile then
local ok, err = pcall(function() local ok, err = pcall(fnFile, ...)
fnFile(table.unpack(tArgs, 1, tArgs.n))
end)
if not ok then if not ok then
if err and err ~= "" then if err and err ~= "" then
printError(err) printError(err)
@ -926,7 +923,7 @@ settings.define("list.show_hidden", {
}) })
settings.define("motd.enable", { settings.define("motd.enable", {
default = false, default = pocket == nil,
description = "Display a random message when the computer starts up.", description = "Display a random message when the computer starts up.",
type = "boolean", type = "boolean",
}) })

View File

@ -106,7 +106,7 @@ end
-- This generally returns the same as @{disk.getLabel} for records. -- This generally returns the same as @{disk.getLabel} for records.
-- --
-- @tparam string name The name of the disk drive. -- @tparam string name The name of the disk drive.
-- @treturn string|false|nil The track title, `false` if there is not a music -- @treturn string|false|nil The track title, @{false} if there is not a music
-- record in the drive or `nil` if no drive is present. -- record in the drive or `nil` if no drive is present.
function getAudioTitle(name) function getAudioTitle(name)
if isDrive(name) then if isDrive(name) then

View File

@ -32,7 +32,7 @@ for _, v in ipairs(valid_types) do valid_types[v] = true end
-- you to provide defaults and additional metadata. -- you to provide defaults and additional metadata.
-- --
-- @tparam string name The name of this option -- @tparam string name The name of this option
-- @tparam[opt] { description? = string, default? = value, type? = string } options -- @tparam[opt] { description? = string, default? = any, type? = string } options
-- Options for this setting. This table accepts the following fields: -- Options for this setting. This table accepts the following fields:
-- --
-- - `description`: A description which may be printed when running the `set` program. -- - `description`: A description which may be printed when running the `set` program.
@ -127,7 +127,7 @@ end
--- Get details about a specific setting. --- Get details about a specific setting.
-- --
-- @tparam string name The name of the setting to get. -- @tparam string name The name of the setting to get.
-- @treturn { description? = string, default? = value, type? = string, value? = value } -- @treturn { description? = string, default? = any, type? = string, value? = any }
-- Information about this setting. This includes all information from @{settings.define}, -- Information about this setting. This includes all information from @{settings.define},
-- as well as this setting's value. -- as well as this setting's value.
function getDetails(name) function getDetails(name)

View File

@ -1,3 +1,34 @@
# New features in CC: Tweaked 1.87.0
* Add documentation to many Lua functions. This is published online at https://tweaked.cc/.
* Replace to pretty-printer in the Lua REPL. It now supports displaying functions and recursive tables. This printer is may be used within your own code through the `cc.pretty` module.
* Add `fs.getCapacity`. A complement to `fs.getFreeSpace`, this returns the capacity of the supplied drive.
* Add `fs.getAttributes`. This provides file size and type, as well as creation and modification time.
* Update Cobalt version. This backports several features from Lua 5.2 and 5.3:
- The `__len` metamethod may now be used by tables.
- Add `\z`, hexadecimal (`\x00`) and unicode (`\u0000`) string escape codes.
- Add `utf8` lib.
- Mirror Lua's behaviour of tail calls more closely. Native functions are no longer tail called, and tail calls are displayed in the stack trace.
- `table.unpack` now uses `__len` and `__index` metamethods.
- Parser errors now include the token where the error occured.
* Add `textutils.unserializeJSON`. This can be used to decode standard JSON and stringified-NBT.
* Switch the monitor renderer to use VBOs, providing a _significant_ performance boost in some cases. You can switch back to the display list renderer via the config.
* The `settings` API now allows "defining" settings. This allows settings to specify a default value and description.
* Enable the motd on non-pocket computers.
* Allow using the menu with the mouse in edit and paint (JakobDev).
* Add Danish and Korean translations (ChristianLW, mindy15963)
* Fire `mouse_up` events in the monitor program.
* Allow specifying a timeout to `websocket.receive`.
* Increase the maximimum limit for websocket messages.
* Optimise capacity checking of computer/disk folders.
And several bug fixes:
* Fix turtle texture being incorrectly oriented (magiczocker10).
* Prevent copying folders into themselves.
* Normalise file paths within shell.setDir (JakobDev)
* Fix turtles treating waterlogged blocks as water.
* Register an entity renderer for the turtle's fake player.
# New features in CC: Tweaked 1.86.2 # New features in CC: Tweaked 1.86.2
* Fix peripheral.getMethods returning an empty table * Fix peripheral.getMethods returning an empty table

View File

@ -1,5 +1,32 @@
New features in CC: Tweaked 1.86.2 New features in CC: Tweaked 1.87.0
* Fix peripheral.getMethods returning an empty table * Add documentation to many Lua functions. This is published online at https://tweaked.cc/.
* Replace to pretty-printer in the Lua REPL. It now supports displaying functions and recursive tables. This printer is may be used within your own code through the `cc.pretty` module.
* Add `fs.getCapacity`. A complement to `fs.getFreeSpace`, this returns the capacity of the supplied drive.
* Add `fs.getAttributes`. This provides file size and type, as well as creation and modification time.
* Update Cobalt version. This backports several features from Lua 5.2 and 5.3:
- The `__len` metamethod may now be used by tables.
- Add `\z`, hexadecimal (`\x00`) and unicode (`\u0000`) string escape codes.
- Add `utf8` lib.
- Mirror Lua's behaviour of tail calls more closely. Native functions are no longer tail called, and tail calls are displayed in the stack trace.
- `table.unpack` now uses `__len` and `__index` metamethods.
- Parser errors now include the token where the error occured.
* Add `textutils.unserializeJSON`. This can be used to decode standard JSON and stringified-NBT.
* Switch the monitor renderer to use VBOs, providing a _significant_ performance boost in some cases. You can switch back to the display list renderer via the config.
* The `settings` API now allows "defining" settings. This allows settings to specify a default value and description.
* Enable the motd on non-pocket computers.
* Allow using the menu with the mouse in edit and paint (JakobDev).
* Add Danish and Korean translations (ChristianLW, mindy15963)
* Fire `mouse_up` events in the monitor program.
* Allow specifying a timeout to `websocket.receive`.
* Increase the maximimum limit for websocket messages.
* Optimise capacity checking of computer/disk folders.
And several bug fixes:
* Fix turtle texture being incorrectly oriented (magiczocker10).
* Prevent copying folders into themselves.
* Normalise file paths within shell.setDir (JakobDev)
* Fix turtles treating waterlogged blocks as water.
* Register an entity renderer for the turtle's fake player.
Type "help changelog" to see the full version history. Type "help changelog" to see the full version history.

View File

@ -16,7 +16,7 @@ local completion = require "cc.completion"
--- Complete the name of a file relative to the current working directory. --- Complete the name of a file relative to the current working directory.
-- --
-- @tparam shell shell The shell we're completing in -- @tparam table shell The shell we're completing in
-- @tparam { string... } choices The list of choices to complete from. -- @tparam { string... } choices The list of choices to complete from.
-- @treturn { string... } A list of suffixes of matching files. -- @treturn { string... } A list of suffixes of matching files.
local function file(shell, text) local function file(shell, text)
@ -25,7 +25,7 @@ end
--- Complete the name of a directory relative to the current working directory. --- Complete the name of a directory relative to the current working directory.
-- --
-- @tparam shell shell The shell we're completing in -- @tparam table shell The shell we're completing in
-- @tparam { string... } choices The list of choices to complete from. -- @tparam { string... } choices The list of choices to complete from.
-- @treturn { string... } A list of suffixes of matching directories. -- @treturn { string... } A list of suffixes of matching directories.
local function dir(shell, text) local function dir(shell, text)
@ -35,7 +35,7 @@ end
--- Complete the name of a file or directory relative to the current working --- Complete the name of a file or directory relative to the current working
-- directory. -- directory.
-- --
-- @tparam shell shell The shell we're completing in -- @tparam table shell The shell we're completing in
-- @tparam { string... } choices The list of choices to complete from. -- @tparam { string... } choices The list of choices to complete from.
-- @tparam { string... } previous The shell arguments before this one. -- @tparam { string... } previous The shell arguments before this one.
-- @tparam[opt] boolean add_space Whether to add a space after the completed item. -- @tparam[opt] boolean add_space Whether to add a space after the completed item.
@ -61,9 +61,10 @@ end
--- Complete the name of a program. --- Complete the name of a program.
-- --
-- @tparam shell shell The shell we're completing in -- @tparam table shell The shell we're completing in
-- @tparam { string... } choices The list of choices to complete from. -- @tparam { string... } choices The list of choices to complete from.
-- @treturn { string... } A list of suffixes of matching programs. -- @treturn { string... } A list of suffixes of matching programs.
-- @see shell.completeProgram
local function program(shell, text) local function program(shell, text)
return shell.completeProgram(text) return shell.completeProgram(text)
end end

View File

@ -1,43 +1,23 @@
View the source code at https://github.com/SquidDev-CC/CC-Tweaked Please report bugs at https://github.com/SquidDev-CC/CC-Tweaked. Thanks!
View the documentation at https://wiki.computercraft.cc View the documentation at https://wiki.computercraft.cc
Visit the forum at https://forums.computercraft.cc Show off your programs or ask for help at our forum: https://forums.computercraft.cc
You can disable these messages by running "set motd.enable false". You can disable these messages by running "set motd.enable false".
You can create directories with "mkdir".
Want to see hidden files? Run "set list.show_hidden true".
Run "list" or "ls" to see all files in a directory.
You can delete files and directories with "delete" or "rm".
Use "pastebin put" to upload a program to pastebin. Use "pastebin put" to upload a program to pastebin.
Use "pastebin get" to download a program from pastebin.
Use "pastebin run" to run a program from pastebin.
Use the "edit" program to create and edit your programs. Use the "edit" program to create and edit your programs.
You can copy files with "copy" or "cp".
You can use "wget run <url>" to run a program from the internet.
You can use "wget" to download a file from the internet. You can use "wget" to download a file from the internet.
On an advanced computer you can use "fg" or "bg" to run multiple programs at the same time. On an advanced computer you can use "fg" or "bg" to run multiple programs at the same time.
Use "type" to see if a path is a file or a directory.
Get a list of all programs with "programs".
Use an advanced computer to use colours and the mouse. Use an advanced computer to use colours and the mouse.
With a speaker you can play sounds. With a speaker you can play sounds.
Use "motd" to print the Message of the Day.
You can disable the startup from a computer with "set shell.allow_startup false".
You can disable the startup from a disk with "set shell.allow_disk_startup false".
Programs that are placed in the "startup" folder in the root of a computer are started on boot. Programs that are placed in the "startup" folder in the root of a computer are started on boot.
Use a modem to connect with other computers. Use a modem to connect with other computers.
With the "gps" program you can get the position of a computer. With the "gps" program you can get the position of a computer.
Use "monitor" to run a program on a attached monitor. Use "monitor" to run a program on a attached monitor.
View all attached peripherals with "peripherals". Don't forget to label your computer with "label set".
Use "time" to see the in-game time.
You can set the label of a computer with "label set".
A computer needs a label to keep its files if it is destroyed.
You can disable auto completion in the shell with "set shell.autocomplete false".
You can disable auto completion in edit with "set edit.autocomplete false".
Feeling creative? Use a printer to print a book! Feeling creative? Use a printer to print a book!
Files beginning with a "." character are hidden from "list" by default. Files beginning with a "." are hidden from "list" by default.
Running "set" lists the current values of all settings. Running "set" lists the current values of all settings.
Some programs are only available on advanced computers, turtles, pocket computers, or command computers. Some programs are only available on advanced computers, turtles, pocket computers or command computers.
The "equip" and "unequip" programs let you add or remove supported upgrades from a turtle or pocket computer without crafting. The "equip" programs let you add upgrades to a turtle without crafting.
You can change the color of a disk by crafting it with dye. You can change the color of a disk by crafting or right clicking it with dye.
Right-clicking a turtle with a dye changes its color.
You can print on a printed page again to get multiple colors. You can print on a printed page again to get multiple colors.
Holding the Control and T keys terminates the running program. Holding the Ctrl and T keys terminates the running program.
Holding Control and S or R shuts down or reboots the computer you are using.

View File

@ -1,3 +1,21 @@
--- Multishell allows multiple programs to be run at the same time.
--
-- When multiple programs are running, it displays a tab bar at the top of the
-- screen, which allows you to switch between programs. New programs can be
-- launched using the `fg` or `bg` programs, or using the @{shell.openTab} and
-- @{multishell.launch} functions.
--
-- Each process is identified by its ID, which corresponds to its position in
-- the tab list. As tabs may be opened and closed, this ID is _not_ constant
-- over a program's run. As such, be careful not to use stale IDs.
--
-- As with @{shell}, @{multishell} is not a "true" API. Instead, it is a
-- standard program, which launches a shell and injects its API into the shell's
-- environment. This API is not available in the global environment, and so is
-- not available to @{os.loadAPI|APIs}.
--
-- @module[module] multishell
local expect = dofile("rom/modules/main/cc/expect.lua").expect local expect = dofile("rom/modules/main/cc/expect.lua").expect
-- Setup process switching -- Setup process switching
@ -190,12 +208,26 @@ local function setMenuVisible(bVis)
end end
end end
local multishell = {} local multishell = {} --- @export
--- Get the currently visible process. This will be the one selected on
-- the tab bar.
--
-- Note, this is different to @{getCurrent}, which returns the process which is
-- currently executing.
--
-- @treturn number The currently visible process's index.
-- @see setFocus
function multishell.getFocus() function multishell.getFocus()
return nCurrentProcess return nCurrentProcess
end end
--- Change the currently visible process.
--
-- @tparam number n The process index to switch to.
-- @treturn boolean If the process was changed successfully. This will
-- return @{false} if there is no process with this id.
-- @see getFocus
function multishell.setFocus(n) function multishell.setFocus(n)
expect(1, n, "number") expect(1, n, "number")
if n >= 1 and n <= #tProcesses then if n >= 1 and n <= #tProcesses then
@ -206,6 +238,13 @@ function multishell.setFocus(n)
return false return false
end end
--- Get the title of the given tab.
--
-- This starts as the name of the program, but may be changed using
-- @{multishell.setTitle}.
-- @tparam number n The process index.
-- @treturn string|nil The current process title, or @{nil} if the
-- process doesn't exist.
function multishell.getTitle(n) function multishell.getTitle(n)
expect(1, n, "number") expect(1, n, "number")
if n >= 1 and n <= #tProcesses then if n >= 1 and n <= #tProcesses then
@ -214,19 +253,45 @@ function multishell.getTitle(n)
return nil return nil
end end
function multishell.setTitle(n, sTitle) --- Set the title of the given process.
--
-- @tparam number n The process index.
-- @tparam string title The new process title.
-- @see getTitle
-- @usage Change the title of the current process
--
-- multishell.setTitle(multishell.getCurrent(), "Hello")
function multishell.setTitle(n, title)
expect(1, n, "number") expect(1, n, "number")
expect(2, sTitle, "string") expect(2, title, "string")
if n >= 1 and n <= #tProcesses then if n >= 1 and n <= #tProcesses then
setProcessTitle(n, sTitle) setProcessTitle(n, title)
redrawMenu() redrawMenu()
end end
end end
--- Get the index of the currently running process.
--
-- @treturn number The currently running process.
function multishell.getCurrent() function multishell.getCurrent()
return nRunningProcess return nRunningProcess
end end
--- Start a new process, with the given environment, program and arguments.
--
-- The returned process index is not constant over the program's run. It can be
-- safely used immediately after launching (for instance, to update the title or
-- switch to that tab). However, after your program has yielded, it may no
-- longer be correct.
--
-- @tparam table tProgramEnv The environment to load the path under.
-- @tparam string sProgramPath The path to the program to run.
-- @param ... Additional arguments to pass to the program.
-- @treturn number The index of the created process.
-- @see os.run
-- @usage Run the "hello" program, and set its title to "Hello!"
-- local id = multishell.launch({}, "/rom/programs/fun/hello.lua")
-- multishell.setTitle(id, "Hello!")
function multishell.launch(tProgramEnv, sProgramPath, ...) function multishell.launch(tProgramEnv, sProgramPath, ...)
expect(1, tProgramEnv, "table") expect(1, tProgramEnv, "table")
expect(2, sProgramPath, "string") expect(2, sProgramPath, "string")
@ -238,6 +303,9 @@ function multishell.launch(tProgramEnv, sProgramPath, ...)
return nResult return nResult
end end
--- Get the number of processes within this multishell.
--
-- @treturn number The number of processes.
function multishell.getCount() function multishell.getCount()
return #tProcesses return #tProcesses
end end

View File

@ -15,7 +15,7 @@ end
-- Create .lua files by default -- Create .lua files by default
if not fs.exists(sPath) and not string.find(sPath, "%.") then if not fs.exists(sPath) and not string.find(sPath, "%.") then
local sExtension = settings.get("edit.default_extension", "") local sExtension = settings.get("edit.default_extension")
if sExtension ~= "" and type(sExtension) == "string" then if sExtension ~= "" and type(sExtension) == "string" then
sPath = sPath .. "." .. sExtension sPath = sPath .. "." .. sExtension
end end
@ -61,7 +61,12 @@ if peripheral.find("printer") then
end end
table.insert(tMenuItems, "Exit") table.insert(tMenuItems, "Exit")
local sStatus = "Press Ctrl to access menu" local sStatus
if term.isColour() then
sStatus = "Press Ctrl or click here to access menu"
else
sStatus = "Press Ctrl to access menu"
end
if #sStatus > w - 5 then if #sStatus > w - 5 then
sStatus = "Press Ctrl for menu" sStatus = "Press Ctrl for menu"
end end
@ -725,16 +730,35 @@ while bRunning do
end end
elseif sEvent == "mouse_click" then elseif sEvent == "mouse_click" then
local cx, cy = param2, param3
if not bMenu then if not bMenu then
if param == 1 then if param == 1 then
-- Left click -- Left click
local cx, cy = param2, param3
if cy < h then if cy < h then
local newY = math.min(math.max(scrollY + cy, 1), #tLines) local newY = math.min(math.max(scrollY + cy, 1), #tLines)
local newX = math.min(math.max(scrollX + cx, 1), #tLines[newY] + 1) local newX = math.min(math.max(scrollX + cx, 1), #tLines[newY] + 1)
setCursor(newX, newY) setCursor(newX, newY)
else
bMenu = true
redrawMenu()
end end
end end
else
if cy == h then
local nMenuPosEnd = 1
local nMenuPosStart = 1
for n, sMenuItem in ipairs(tMenuItems) do
nMenuPosEnd = nMenuPosEnd + #sMenuItem + 1
if cx > nMenuPosStart and cx < nMenuPosEnd then
doMenuItem(n)
end
nMenuPosEnd = nMenuPosEnd + 1
nMenuPosStart = nMenuPosEnd
end
else
bMenu = false
redrawMenu()
end
end end
elseif sEvent == "mouse_scroll" then elseif sEvent == "mouse_scroll" then

View File

@ -19,7 +19,7 @@ local canvas = {}
local mChoices = { "Save", "Exit" } local mChoices = { "Save", "Exit" }
-- The message displayed in the footer bar -- The message displayed in the footer bar
local fMessage = "Press Ctrl to access menu" local fMessage = "Press Ctrl or click here to access menu"
------------------------- -------------------------
-- Initialisation -- -- Initialisation --
@ -46,7 +46,7 @@ end
-- Create .nfp files by default -- Create .nfp files by default
if not fs.exists(sPath) and not string.find(sPath, "%.") then if not fs.exists(sPath) and not string.find(sPath, "%.") then
local sExtension = settings.get("paint.default_extension", "") local sExtension = settings.get("paint.default_extension")
if sExtension ~= "" and type(sExtension) == "string" then if sExtension ~= "" and type(sExtension) == "string" then
sPath = sPath .. "." .. sExtension sPath = sPath .. "." .. sExtension
end end
@ -252,6 +252,29 @@ local function drawCanvas()
end end
end end
local menu_choices = {
Save = function()
if bReadOnly then
fMessage = "Access denied"
return false
end
local success, err = save(sPath)
if success then
fMessage = "Saved to " .. sPath
else
if err then
fMessage = "Error saving to " .. err
else
fMessage = "Error saving to " .. sPath
end
end
return false
end,
Exit = function()
return true
end,
}
--[[ --[[
Draws menu options and handles input from within the menu. Draws menu options and handles input from within the menu.
returns: true if the program is to be exited; false otherwise returns: true if the program is to be exited; false otherwise
@ -261,6 +284,7 @@ local function accessMenu()
local selection = 1 local selection = 1
term.setBackgroundColour(colours.black) term.setBackgroundColour(colours.black)
while true do while true do
-- Draw the menu -- Draw the menu
term.setCursorPos(1, h) term.setCursorPos(1, h)
@ -269,27 +293,28 @@ local function accessMenu()
for k, v in pairs(mChoices) do for k, v in pairs(mChoices) do
if selection == k then if selection == k then
term.setTextColour(colours.yellow) term.setTextColour(colours.yellow)
local ox = term.getCursorPos() term.write("[")
term.write("[" .. string.rep(" ", #v) .. "]")
term.setCursorPos(ox + 1, h)
term.setTextColour(colours.white) term.setTextColour(colours.white)
term.write(v) term.write(v)
term.setCursorPos(term.getCursorPos() + 1, h) term.setTextColour(colours.yellow)
term.write("]")
term.setTextColour(colours.white)
else else
term.write(" " .. v .. " ") term.write(" " .. v .. " ")
end end
end end
-- Handle input in the menu -- Handle input in the menu
local id, key = os.pullEvent("key") local id, param1, param2, param3 = os.pullEvent()
if id == "key" then if id == "key" then
-- S and E are shortcuts local key = param1
if key == keys.s then
selection = 1 -- Handle menu shortcuts.
key = keys.enter for _, menu_item in ipairs(mChoices) do
elseif key == keys.e then local k = keys[menu_item:sub(1, 1):lower()]
selection = 2 if k and k == key then
key = keys.enter return menu_choices[menu_item]()
end
end end
if key == keys.right then if key == keys.right then
@ -308,29 +333,25 @@ local function accessMenu()
elseif key == keys.enter then elseif key == keys.enter then
-- Select an option -- Select an option
if mChoices[selection] == "Save" then return menu_choices[mChoices[selection]]()
if bReadOnly then
fMessage = "Access denied"
return false
end
local success, err = save(sPath)
if success then
fMessage = "Saved to " .. sPath
else
if err then
fMessage = "Error saving to " .. err
else
fMessage = "Error saving to " .. sPath
end
end
return false
elseif mChoices[selection] == "Exit" then
return true
end
elseif key == keys.leftCtrl or keys == keys.rightCtrl then elseif key == keys.leftCtrl or keys == keys.rightCtrl then
-- Cancel the menu -- Cancel the menu
return false return false
end end
elseif id == "mouse_click" then
local cx, cy = param2, param3
if cy ~= h then return false end -- Exit the menu
local nMenuPosEnd = 1
local nMenuPosStart = 1
for _, sMenuItem in ipairs(mChoices) do
nMenuPosEnd = nMenuPosEnd + #sMenuItem + 1
if cx > nMenuPosStart and cx < nMenuPosEnd then
return menu_choices[sMenuItem]()
end
nMenuPosEnd = nMenuPosEnd + 1
nMenuPosStart = nMenuPosEnd
end
end end
end end
end end
@ -378,6 +399,10 @@ local function handleEvents()
canvas[p3][p2] = paintColour canvas[p3][p2] = paintColour
drawCanvasPixel(p2, p3) drawCanvasPixel(p2, p3)
elseif p3 == h and id == "mouse_click" then
-- Open menu
programActive = not accessMenu()
drawInterface()
end end
elseif id == "key" then elseif id == "key" then
if p1 == keys.leftCtrl or p1 == keys.rightCtrl then if p1 == keys.leftCtrl or p1 == keys.rightCtrl then

View File

@ -1,3 +1,15 @@
--- The shell API provides access to CraftOS's command line interface.
--
-- It allows you to @{run|start programs}, @{setCompletionFunction|add
-- completion for a program}, and much more.
--
-- @{shell} is not a "true" API. Instead, it is a standard program, which its
-- API into the programs that it launches. This allows for multiple shells to
-- run at the same time, but means that the API is not available in the global
-- environment, and so is unavailable to other @{os.loadAPI|APIs}.
--
-- @module[module] shell
local expect = dofile("rom/modules/main/cc/expect.lua").expect local expect = dofile("rom/modules/main/cc/expect.lua").expect
local multishell = multishell local multishell = multishell
@ -15,7 +27,7 @@ local tAliases = parentShell and parentShell.aliases() or {}
local tCompletionInfo = parentShell and parentShell.getCompletionInfo() or {} local tCompletionInfo = parentShell and parentShell.getCompletionInfo() or {}
local tProgramStack = {} local tProgramStack = {}
local shell = {} local shell = {} --- @export
local function createShellEnv(sDir) local function createShellEnv(sDir)
local tEnv = {} local tEnv = {}
tEnv.shell = shell tEnv.shell = shell
@ -172,6 +184,18 @@ local function tokenise(...)
end end
-- Install shell API -- Install shell API
--- Run a program with the supplied arguments.
--
-- All arguments are concatenated together and then parsed as a command line. As
-- a result, `shell.run("program a b")` is the same as `shell.run("program",
-- "a", "b")`.
--
-- @tparam string ... The program to run and its arguments.
-- @treturn boolean Whether the program exited successfully.
-- @usage Run `paint my-image` from within your program:
--
-- shell.run("paint", "my-image")
function shell.run(...) function shell.run(...)
local tWords = tokenise(...) local tWords = tokenise(...)
local sCommand = tWords[1] local sCommand = tWords[1]
@ -181,38 +205,83 @@ function shell.run(...)
return false return false
end end
--- Exit the current shell.
--
-- This does _not_ terminate your program, it simply makes the shell terminate
-- after your program has finished. If this is the toplevel shell, then the
-- computer will be shutdown.
function shell.exit() function shell.exit()
bExit = true bExit = true
end end
--- Return the current working directory. This is what is displayed before the
-- `> ` of the shell prompt, and is used by @{shell.resolve} to handle relative
-- paths.
--
-- @treturn string The current working directory.
-- @see setDir To change the working directory.
function shell.dir() function shell.dir()
return sDir return sDir
end end
function shell.setDir(_sDir) --- Set the current working directory.
expect(1, _sDir, "string") --
if not fs.isDir(_sDir) then -- @tparam string dir The new working directory.
-- @throws If the path does not exist or is not a directory.
-- @usage Set the working directory to "rom"
--
-- shell.setDir("rom")
function shell.setDir(dir)
expect(1, dir, "string")
if not fs.isDir(dir) then
error("Not a directory", 2) error("Not a directory", 2)
end end
sDir = fs.combine(_sDir, "") sDir = fs.combine(dir, "")
end end
--- Set the path where programs are located.
--
-- The path is composed of a list of directory names in a string, each separated
-- by a colon (`:`). On normal turtles will look in the current directory (`.`),
-- `/rom/programs` and `/rom/programs/turtle` folder, making the path
-- `.:/rom/programs:/rom/programs/turtle`.
--
-- @treturn string The current shell's path.
-- @see setPath To change the current path.
function shell.path() function shell.path()
return sPath return sPath
end end
function shell.setPath(_sPath) --- Set the @{path|current program path}.
expect(1, _sPath, "string") --
sPath = _sPath -- Be careful to prefix directories with a `/`. Otherwise they will be searched
-- for from the @{shell.dir|current directory}, rather than the computer's root.
--
-- @tparam string path The new program path.
function shell.setPath(path)
expect(1, path, "string")
sPath = path
end end
function shell.resolve(_sPath) --- Resolve a relative path to an absolute path.
expect(1, _sPath, "string") --
local sStartChar = string.sub(_sPath, 1, 1) -- The @{fs} and @{io} APIs work using absolute paths, and so we must convert
-- any paths relative to the @{dir|current directory} to absolute ones. This
-- does nothing when the path starts with `/`.
--
-- @tparam string path The path to resolve.
-- @usage Resolve `startup.lua` when in the `rom` folder.
--
-- shell.setDir("rom")
-- print(shell.resolve("startup.lua"))
-- -- => rom/startup.lua
function shell.resolve(path)
expect(1, path, "string")
local sStartChar = string.sub(path, 1, 1)
if sStartChar == "/" or sStartChar == "\\" then if sStartChar == "/" or sStartChar == "\\" then
return fs.combine("", _sPath) return fs.combine("", path)
else else
return fs.combine(sDir, _sPath) return fs.combine(sDir, path)
end end
end end
@ -226,16 +295,25 @@ local function pathWithExtension(_sPath, _sExt)
return _sPath .. "." .. _sExt return _sPath .. "." .. _sExt
end end
function shell.resolveProgram(_sCommand) --- Resolve a program, using the @{path|program path} and list of @{aliases|aliases}.
expect(1, _sCommand, "string") --
-- @tparam string command The name of the program
-- @treturn string|nil The absolute path to the program, or @{nil} if it could
-- not be found.
-- @usage Locate the `hello` program.
--
-- shell.resolveProgram("hello")
-- -- => rom/programs/fun/hello.lua
function shell.resolveProgram(command)
expect(1, command, "string")
-- Substitute aliases firsts -- Substitute aliases firsts
if tAliases[_sCommand] ~= nil then if tAliases[command] ~= nil then
_sCommand = tAliases[_sCommand] command = tAliases[command]
end end
-- If the path is a global path, use it directly -- If the path is a global path, use it directly
if _sCommand:find("/") or _sCommand:find("\\") then if command:find("/") or command:find("\\") then
local sPath = shell.resolve(_sCommand) local sPath = shell.resolve(command)
if fs.exists(sPath) and not fs.isDir(sPath) then if fs.exists(sPath) and not fs.isDir(sPath) then
return sPath return sPath
else else
@ -249,7 +327,7 @@ function shell.resolveProgram(_sCommand)
-- Otherwise, look on the path variable -- Otherwise, look on the path variable
for sPath in string.gmatch(sPath, "[^:]+") do for sPath in string.gmatch(sPath, "[^:]+") do
sPath = fs.combine(shell.resolve(sPath), _sCommand) sPath = fs.combine(shell.resolve(sPath), command)
if fs.exists(sPath) and not fs.isDir(sPath) then if fs.exists(sPath) and not fs.isDir(sPath) then
return sPath return sPath
else else
@ -264,7 +342,15 @@ function shell.resolveProgram(_sCommand)
return nil return nil
end end
function shell.programs(_bIncludeHidden) --- Return a list of all programs on the @{shell.path|path}.
--
-- @tparam[opt] boolean include_hidden Include hidden files. Namely, any which
-- start with `.`.
-- @treturn { string } A list of available programs.
-- @usage textutils.tabulate(shell.programs())
function shell.programs(include_hidden)
expect(1, include_hidden, "boolean", "nil")
local tItems = {} local tItems = {}
-- Add programs from the path -- Add programs from the path
@ -275,7 +361,7 @@ function shell.programs(_bIncludeHidden)
for n = 1, #tList do for n = 1, #tList do
local sFile = tList[n] local sFile = tList[n]
if not fs.isDir(fs.combine(sPath, sFile)) and if not fs.isDir(fs.combine(sPath, sFile)) and
(_bIncludeHidden or string.sub(sFile, 1, 1) ~= ".") then (include_hidden or string.sub(sFile, 1, 1) ~= ".") then
if #sFile > 4 and sFile:sub(-4) == ".lua" then if #sFile > 4 and sFile:sub(-4) == ".lua" then
sFile = sFile:sub(1, -5) sFile = sFile:sub(1, -5)
end end
@ -351,6 +437,21 @@ local function completeProgramArgument(sProgram, nArgument, sPart, tPreviousPart
return nil return nil
end end
--- Complete a shell command line.
--
-- This accepts an incomplete command, and completes the program name or
-- arguments. For instance, `l` will be completed to `ls`, and `ls ro` will be
-- completed to `ls rom/`.
--
-- Completion handlers for your program may be registered with
-- @{shell.setCompletionFunction}.
--
-- @tparam string sLine The input to complete.
-- @treturn { string }|nil The list of possible completions.
-- @see read For more information about completion.
-- @see shell.completeProgram
-- @see shell.setCompletionFunction
-- @see shell.getCompletionInfo
function shell.complete(sLine) function shell.complete(sLine)
expect(1, sLine, "string") expect(1, sLine, "string")
if #sLine > 0 then if #sLine > 0 then
@ -388,23 +489,66 @@ function shell.complete(sLine)
return nil return nil
end end
function shell.completeProgram(sProgram) --- Complete the name of a program.
expect(1, sProgram, "string") --
return completeProgram(sProgram) -- @tparam string program The name of a program to complete.
-- @treturn { string } A list of possible completions.
-- @see cc.shell.completion.program
function shell.completeProgram(program)
expect(1, program, "string")
return completeProgram(program)
end end
function shell.setCompletionFunction(sProgram, fnComplete) --- Set the completion function for a program. When the program is entered on
expect(1, sProgram, "string") -- the command line, this program will be called to provide auto-complete
expect(2, fnComplete, "function") -- information.
tCompletionInfo[sProgram] = { --
fnComplete = fnComplete, -- The completion function accepts four arguments:
--
-- 1. The current shell. As completion functions are inherited, this is not
-- guaranteed to be the shell you registered this function in.
-- 2. The index of the argument currently being completed.
-- 3. The current argument. This may be the empty string.
-- 4. A list of the previous arguments.
--
-- For instance, when completing `pastebin put rom/st` our pastebin completion
-- function will receive the shell API, an index of 2, `rom/st` as the current
-- argument, and a "previous" table of `{ "put" }`. This function may then wish
-- to return a table containing `artup.lua`, indicating the entire command
-- should be completed to `pastebin put rom/startup.lua`.
--
-- You completion entries may also be followed by a space, if you wish to
-- indicate another argument is expected.
--
-- @tparam string program The path to the program. This should be an absolute path
-- _without_ the leading `/`.
-- @tparam function(shell: table, index: number, argument: string, previous: { string }):({ string }|nil) complete
-- The completion function.
-- @see cc.shell.completion Various utilities to help with writing completion functions.
-- @see shell.complete
-- @see read For more information about completion.
function shell.setCompletionFunction(program, complete)
expect(1, program, "string")
expect(2, complete, "function")
tCompletionInfo[program] = {
fnComplete = complete,
} }
end end
--- Get a table containing all completion functions.
--
-- This should only be needed when building custom shells. Use
-- @{setCompletionFunction} to add a completion function.
--
-- @treturn { [string] = { fnComplete = function } } A table mapping the
-- absolute path of programs, to their completion functions.
function shell.getCompletionInfo() function shell.getCompletionInfo()
return tCompletionInfo return tCompletionInfo
end end
--- Returns the path to the currently running program.
--
-- @treturn string The absolute path to the running program.
function shell.getRunningProgram() function shell.getRunningProgram()
if #tProgramStack > 0 then if #tProgramStack > 0 then
return tProgramStack[#tProgramStack] return tProgramStack[#tProgramStack]
@ -412,17 +556,38 @@ function shell.getRunningProgram()
return nil return nil
end end
function shell.setAlias(_sCommand, _sProgram) --- Add an alias for a program.
expect(1, _sCommand, "string") --
expect(2, _sProgram, "string") -- @tparam string command The name of the alias to add.
tAliases[_sCommand] = _sProgram -- @tparam string program The name or path to the program.
-- @usage Alias `vim` to the `edit` program
--
-- shell.setAlias("vim", "edit")
function shell.setAlias(command, program)
expect(1, command, "string")
expect(2, program, "string")
tAliases[command] = program
end end
function shell.clearAlias(_sCommand) --- Remove an alias.
expect(1, _sCommand, "string") --
tAliases[_sCommand] = nil -- @tparam string command The alias name to remove.
function shell.clearAlias(command)
expect(1, command, "string")
tAliases[command] = nil
end end
--- Get the current aliases for this shell.
--
-- Aliases are used to allow multiple commands to refer to a single program. For
-- instance, the `list` program is aliased `dir` or `ls`. Running `ls`, `dir` or
-- `list` in the shell will all run the `list` program.
--
-- @treturn { [string] = string } A table, where the keys are the names of
-- aliases, and the values are the path to the program.
-- @see shell.setAlias
-- @see shell.resolveProgram This uses aliases when resolving a program name to
-- an absolute path.
function shell.aliases() function shell.aliases()
-- Copy aliases -- Copy aliases
local tCopy = {} local tCopy = {}
@ -433,6 +598,20 @@ function shell.aliases()
end end
if multishell then if multishell then
--- Open a new @{multishell} tab running a command.
--
-- This behaves similarly to @{shell.run}, but instead returns the process
-- index.
--
-- This function is only available if the @{multishell} API is.
--
-- @tparam string ... The command line to run.
-- @see shell.run
-- @see multishell.launch
-- @usage Launch the Lua interpreter and switch to it.
--
-- local id = shell.openTab("lua")
-- shell.switchTab(id)
function shell.openTab(...) function shell.openTab(...)
local tWords = tokenise(...) local tWords = tokenise(...)
local sCommand = tWords[1] local sCommand = tWords[1]
@ -448,9 +627,13 @@ if multishell then
end end
end end
function shell.switchTab(nID) --- Switch to the @{multishell} tab with the given index.
expect(1, nID, "number") --
multishell.setFocus(nID) -- @tparam number id The tab to switch to.
-- @see multishell.setFocus
function shell.switchTab(id)
expect(1, id, "number")
multishell.setFocus(id)
end end
end end