Difference between revisions of "Paint"
From ComputerCraft Wiki
(Adding the code for the paint program) |
|||
Line 4: | Line 4: | ||
Running paint on a monitor requires using an advanced monitor. For instructions, see [[monitor (program)]]. | Running paint on a monitor requires using an advanced monitor. For instructions, see [[monitor (program)]]. | ||
+ | |||
+ | == Code == | ||
+ | The paint program uses the following code: | ||
+ | |||
+ | <code> | ||
+ | -- Paint created by nitrogenfingers (edited by dan200) | ||
+ | -- http://www.youtube.com/user/NitrogenFingers | ||
+ | |||
+ | ------------ | ||
+ | -- Fields -- | ||
+ | ------------ | ||
+ | |||
+ | -- The width and height of the terminal | ||
+ | local w,h = term.getSize() | ||
+ | |||
+ | -- The selected colours on the left and right mouse button, and the colour of the canvas | ||
+ | local leftColour, rightColour = colours.white, nil | ||
+ | local canvasColour = colours.black | ||
+ | |||
+ | -- The values stored in the canvas | ||
+ | local canvas = {} | ||
+ | |||
+ | -- The menu options | ||
+ | local mChoices = { "Save","Exit" } | ||
+ | |||
+ | -- The message displayed in the footer bar | ||
+ | local fMessage = "Press Ctrl to access menu" | ||
+ | |||
+ | ------------------------- | ||
+ | -- Initialisation -- | ||
+ | ------------------------- | ||
+ | |||
+ | -- Determine if we can even run this | ||
+ | if not term.isColour() then | ||
+ | print("Requires an Advanced Computer") | ||
+ | return | ||
+ | end | ||
+ | |||
+ | -- Determines if the file exists, and can be edited on this computer | ||
+ | local tArgs = {...} | ||
+ | if #tArgs == 0 then | ||
+ | print("Usage: paint <path>") | ||
+ | return | ||
+ | end | ||
+ | local sPath = shell.resolve(tArgs[1]) | ||
+ | local bReadOnly = fs.isReadOnly(sPath) | ||
+ | if fs.exists(sPath) and fs.isDir(sPath) then | ||
+ | print("Cannot edit a directory.") | ||
+ | return | ||
+ | end | ||
+ | |||
+ | --------------- | ||
+ | -- Functions -- | ||
+ | --------------- | ||
+ | |||
+ | local function getCanvasPixel( x, y ) | ||
+ | if canvas[y] then | ||
+ | return canvas[y][x] | ||
+ | end | ||
+ | return nil | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Converts a colour value to a text character | ||
+ | params: colour = the number to convert to a hex value | ||
+ | returns: a string representing the chosen colour | ||
+ | ]] | ||
+ | local function getCharOf( colour ) | ||
+ | -- Incorrect values always convert to nil | ||
+ | if type(colour) == "number" then | ||
+ | local value = math.floor( math.log(colour) / math.log(2) ) + 1 | ||
+ | if value >= 1 and value <= 16 then | ||
+ | return string.sub( "0123456789abcdef", value, value ) | ||
+ | end | ||
+ | end | ||
+ | return " " | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Converts a text character to colour value | ||
+ | params: char = the char (from string.byte) to convert to number | ||
+ | returns: the colour number of the hex value | ||
+ | ]] | ||
+ | local tColourLookup = {} | ||
+ | for n=1,16 do | ||
+ | tColourLookup[ string.byte( "0123456789abcdef",n,n ) ] = 2^(n-1) | ||
+ | end | ||
+ | local function getColourOf( char ) | ||
+ | -- Values not in the hex table are transparent (canvas coloured) | ||
+ | return tColourLookup[char] | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Loads the file into the canvas | ||
+ | params: path = the path of the file to open | ||
+ | returns: nil | ||
+ | ]] | ||
+ | local function load(path) | ||
+ | -- Load the file | ||
+ | if fs.exists(path) then | ||
+ | local file = fs.open(sPath, "r") | ||
+ | local sLine = file.readLine() | ||
+ | while sLine do | ||
+ | local line = {} | ||
+ | for x=1,w-2 do | ||
+ | line[x] = getColourOf( string.byte(sLine,x,x) ) | ||
+ | end | ||
+ | table.insert( canvas, line ) | ||
+ | sLine = file.readLine() | ||
+ | end | ||
+ | file.close() | ||
+ | end | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Saves the current canvas to file | ||
+ | params: path = the path of the file to save | ||
+ | returns: true if save was successful, false otherwise | ||
+ | ]] | ||
+ | local function save(path) | ||
+ | -- Open file | ||
+ | local sDir = string.sub(sPath, 1, #sPath - #fs.getName(sPath)) | ||
+ | if not fs.exists(sDir) then | ||
+ | fs.makeDir(sDir) | ||
+ | end | ||
+ | |||
+ | local file = fs.open( path, "w" ) | ||
+ | if not file then | ||
+ | return false | ||
+ | end | ||
+ | |||
+ | -- Encode (and trim) | ||
+ | local tLines = {} | ||
+ | local nLastLine = 0 | ||
+ | for y=1,h-1 do | ||
+ | local sLine = "" | ||
+ | local nLastChar = 0 | ||
+ | for x=1,w-2 do | ||
+ | local c = getCharOf( getCanvasPixel( x, y ) ) | ||
+ | sLine = sLine .. c | ||
+ | if c ~= " " then | ||
+ | nLastChar = x | ||
+ | end | ||
+ | end | ||
+ | sLine = string.sub( sLine, 1, nLastChar ) | ||
+ | tLines[y] = sLine | ||
+ | if string.len( sLine ) > 0 then | ||
+ | nLastLine = y | ||
+ | end | ||
+ | end | ||
+ | |||
+ | -- Save out | ||
+ | for n=1,nLastLine do | ||
+ | file.writeLine( tLines[ n ] ) | ||
+ | end | ||
+ | file.close() | ||
+ | return true | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Draws colour picker sidebar, the pallette and the footer | ||
+ | returns: nil | ||
+ | ]] | ||
+ | local function drawInterface() | ||
+ | -- Footer | ||
+ | term.setCursorPos(1, h) | ||
+ | term.setBackgroundColour(colours.black) | ||
+ | term.setTextColour(colours.yellow) | ||
+ | term.clearLine() | ||
+ | term.write(fMessage) | ||
+ | |||
+ | -- Colour Picker | ||
+ | for i=1,16 do | ||
+ | term.setCursorPos(w-1, i) | ||
+ | term.setBackgroundColour( 2^(i-1) ) | ||
+ | term.write(" ") | ||
+ | end | ||
+ | |||
+ | term.setCursorPos(w-1, 17) | ||
+ | term.setBackgroundColour( canvasColour ) | ||
+ | term.setTextColour( colours.grey ) | ||
+ | term.write("XX") | ||
+ | |||
+ | -- Left and Right Selected Colours | ||
+ | for i=18,18 do | ||
+ | term.setCursorPos(w-1, i) | ||
+ | if leftColour ~= nil then | ||
+ | term.setBackgroundColour( leftColour ) | ||
+ | term.write(" ") | ||
+ | else | ||
+ | term.setBackgroundColour( canvasColour ) | ||
+ | term.setTextColour( colours.grey ) | ||
+ | term.write("X") | ||
+ | end | ||
+ | if rightColour ~= nil then | ||
+ | term.setBackgroundColour( rightColour ) | ||
+ | term.write(" ") | ||
+ | else | ||
+ | term.setBackgroundColour( canvasColour ) | ||
+ | term.setTextColour( colours.grey ) | ||
+ | term.write("X") | ||
+ | end | ||
+ | end | ||
+ | |||
+ | -- Padding | ||
+ | term.setBackgroundColour( canvasColour ) | ||
+ | for i=20,h-1 do | ||
+ | term.setCursorPos(w-1, i) | ||
+ | term.write(" ") | ||
+ | end | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Converts a single pixel of a single line of the canvas and draws it | ||
+ | returns: nil | ||
+ | ]] | ||
+ | local function drawCanvasPixel( x, y ) | ||
+ | local pixel = getCanvasPixel( x, y ) | ||
+ | if pixel then | ||
+ | term.setBackgroundColour( pixel or canvasColour ) | ||
+ | term.setCursorPos(x, y) | ||
+ | term.write(" ") | ||
+ | else | ||
+ | term.setBackgroundColour( canvasColour ) | ||
+ | term.setTextColour( colours.grey ) | ||
+ | term.setCursorPos(x, y) | ||
+ | term.write("-") | ||
+ | end | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Converts each colour in a single line of the canvas and draws it | ||
+ | returns: nil | ||
+ | ]] | ||
+ | local function drawCanvasLine( y ) | ||
+ | for x = 1, w-2 do | ||
+ | drawCanvasPixel( x, y ) | ||
+ | end | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Converts each colour in the canvas and draws it | ||
+ | returns: nil | ||
+ | ]] | ||
+ | local function drawCanvas() | ||
+ | for y = 1, h-1 do | ||
+ | drawCanvasLine( y ) | ||
+ | end | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Draws menu options and handles input from within the menu. | ||
+ | returns: true if the program is to be exited; false otherwise | ||
+ | ]] | ||
+ | local function accessMenu() | ||
+ | -- Selected menu option | ||
+ | local selection = 1 | ||
+ | |||
+ | term.setBackgroundColour(colours.black) | ||
+ | while true do | ||
+ | -- Draw the menu | ||
+ | term.setCursorPos(1,h) | ||
+ | term.clearLine() | ||
+ | term.setTextColour(colours.white) | ||
+ | for k,v in pairs(mChoices) do | ||
+ | if selection==k then | ||
+ | term.setTextColour(colours.yellow) | ||
+ | local ox,_ = term.getCursorPos() | ||
+ | term.write("["..string.rep(" ",#v).."]") | ||
+ | term.setCursorPos(ox+1,h) | ||
+ | term.setTextColour(colours.white) | ||
+ | term.write(v) | ||
+ | term.setCursorPos(term.getCursorPos()+1,h) | ||
+ | else | ||
+ | term.write(" "..v.." ") | ||
+ | end | ||
+ | end | ||
+ | |||
+ | -- Handle input in the menu | ||
+ | local id,key = os.pullEvent("key") | ||
+ | if id == "key" then | ||
+ | -- S and E are shortcuts | ||
+ | if key == keys.s then | ||
+ | selection = 1 | ||
+ | key = keys.enter | ||
+ | elseif key == keys.e then | ||
+ | selection = 2 | ||
+ | key = keys.enter | ||
+ | end | ||
+ | |||
+ | if key == keys.right then | ||
+ | -- Move right | ||
+ | selection = selection + 1 | ||
+ | if selection > #mChoices then | ||
+ | selection = 1 | ||
+ | end | ||
+ | |||
+ | elseif key == keys.left and selection > 1 then | ||
+ | -- Move left | ||
+ | selection = selection - 1 | ||
+ | if selection < 1 then | ||
+ | selection = #mChoices | ||
+ | end | ||
+ | |||
+ | elseif key == keys.enter then | ||
+ | -- Select an option | ||
+ | if mChoices[selection]=="Save" then | ||
+ | if bReadOnly then | ||
+ | fMessage = "Access Denied" | ||
+ | return false | ||
+ | end | ||
+ | local success = save(sPath) | ||
+ | if success then | ||
+ | fMessage = "Saved to "..sPath | ||
+ | else | ||
+ | fMessage = "Error saving to "..sPath | ||
+ | end | ||
+ | return false | ||
+ | elseif mChoices[selection]=="Exit" then | ||
+ | return true | ||
+ | end | ||
+ | elseif key == keys.leftCtrl or keys == keys.rightCtrl then | ||
+ | -- Cancel the menu | ||
+ | return false | ||
+ | end | ||
+ | end | ||
+ | end | ||
+ | end | ||
+ | |||
+ | --[[ | ||
+ | Runs the main thread of execution. Draws the canvas and interface, and handles | ||
+ | mouse and key events. | ||
+ | returns: nil | ||
+ | ]] | ||
+ | local function handleEvents() | ||
+ | local programActive = true | ||
+ | while programActive do | ||
+ | local id,p1,p2,p3 = os.pullEvent() | ||
+ | if id=="mouse_click" or id=="mouse_drag" then | ||
+ | if p2 >= w-1 and p3 >= 1 and p3 <= 17 then | ||
+ | if id ~= "mouse_drag" then | ||
+ | -- Selecting an items in the colour picker | ||
+ | if p3 <= 16 then | ||
+ | if p1==1 then | ||
+ | leftColour = 2^(p3-1) | ||
+ | else | ||
+ | rightColour = 2^(p3-1) | ||
+ | end | ||
+ | else | ||
+ | if p1==1 then | ||
+ | leftColour = nil | ||
+ | else | ||
+ | rightColour = nil | ||
+ | end | ||
+ | end | ||
+ | --drawCanvas() | ||
+ | drawInterface() | ||
+ | end | ||
+ | elseif p2 < w-1 and p3 <= h-1 then | ||
+ | -- Clicking on the canvas | ||
+ | local paintColour = nil | ||
+ | if p1==1 then | ||
+ | paintColour = leftColour | ||
+ | elseif p1==2 then | ||
+ | paintColour = rightColour | ||
+ | end | ||
+ | if not canvas[p3] then | ||
+ | canvas[p3] = {} | ||
+ | end | ||
+ | canvas[p3][p2] = paintColour | ||
+ | |||
+ | drawCanvasPixel( p2, p3 ) | ||
+ | end | ||
+ | elseif id=="key" then | ||
+ | if p1==keys.leftCtrl or p1==keys.rightCtrl then | ||
+ | programActive = not accessMenu() | ||
+ | drawInterface() | ||
+ | end | ||
+ | elseif id=="term_resize" then | ||
+ | w,h = term.getSize() | ||
+ | drawCanvas() | ||
+ | drawInterface() | ||
+ | end | ||
+ | end | ||
+ | end | ||
+ | |||
+ | -- Init | ||
+ | load(sPath) | ||
+ | drawCanvas() | ||
+ | drawInterface() | ||
+ | |||
+ | -- Main loop | ||
+ | handleEvents() | ||
+ | |||
+ | -- Shutdown | ||
+ | term.setBackgroundColour(colours.black) | ||
+ | term.setTextColour(colours.white) | ||
+ | term.clear() | ||
+ | term.setCursorPos(1,1) | ||
+ | </code> | ||
[[Category:Programs]] | [[Category:Programs]] |
Revision as of 10:44, 22 September 2014
Paint is a program, created by NitrogenFingers, added in ComputerCraft 1.45. It requires an Advanced Computer, and is used for making images that may be drawn later with the paintutils library. This program, along with Advanced Computers is NOT available in Tekkit Classic.
Running paint on a monitor requires using an advanced monitor. For instructions, see monitor (program).
Code
The paint program uses the following code:
-- Paint created by nitrogenfingers (edited by dan200)
-- http://www.youtube.com/user/NitrogenFingers
------------
-- Fields --
------------
-- The width and height of the terminal
local w,h = term.getSize()
-- The selected colours on the left and right mouse button, and the colour of the canvas
local leftColour, rightColour = colours.white, nil
local canvasColour = colours.black
-- The values stored in the canvas
local canvas = {}
-- The menu options
local mChoices = { "Save","Exit" }
-- The message displayed in the footer bar
local fMessage = "Press Ctrl to access menu"
-------------------------
-- Initialisation --
-------------------------
-- Determine if we can even run this
if not term.isColour() then
print("Requires an Advanced Computer")
return
end
-- Determines if the file exists, and can be edited on this computer
local tArgs = {...}
if #tArgs == 0 then
print("Usage: paint <path>")
return
end
local sPath = shell.resolve(tArgs[1])
local bReadOnly = fs.isReadOnly(sPath)
if fs.exists(sPath) and fs.isDir(sPath) then
print("Cannot edit a directory.")
return
end
---------------
-- Functions --
---------------
local function getCanvasPixel( x, y )
if canvas[y] then
return canvas[y][x]
end
return nil
end
--[[
Converts a colour value to a text character
params: colour = the number to convert to a hex value
returns: a string representing the chosen colour
]]
local function getCharOf( colour )
-- Incorrect values always convert to nil
if type(colour) == "number" then
local value = math.floor( math.log(colour) / math.log(2) ) + 1
if value >= 1 and value <= 16 then
return string.sub( "0123456789abcdef", value, value )
end
end
return " "
end
--[[
Converts a text character to colour value
params: char = the char (from string.byte) to convert to number
returns: the colour number of the hex value
]]
local tColourLookup = {}
for n=1,16 do
tColourLookup[ string.byte( "0123456789abcdef",n,n ) ] = 2^(n-1)
end
local function getColourOf( char )
-- Values not in the hex table are transparent (canvas coloured)
return tColourLookup[char]
end
--[[
Loads the file into the canvas
params: path = the path of the file to open
returns: nil
]]
local function load(path)
-- Load the file
if fs.exists(path) then
local file = fs.open(sPath, "r")
local sLine = file.readLine()
while sLine do
local line = {}
for x=1,w-2 do
line[x] = getColourOf( string.byte(sLine,x,x) )
end
table.insert( canvas, line )
sLine = file.readLine()
end
file.close()
end
end
--[[
Saves the current canvas to file
params: path = the path of the file to save
returns: true if save was successful, false otherwise
]]
local function save(path)
-- Open file
local sDir = string.sub(sPath, 1, #sPath - #fs.getName(sPath))
if not fs.exists(sDir) then
fs.makeDir(sDir)
end
local file = fs.open( path, "w" )
if not file then
return false
end
-- Encode (and trim)
local tLines = {}
local nLastLine = 0
for y=1,h-1 do
local sLine = ""
local nLastChar = 0
for x=1,w-2 do
local c = getCharOf( getCanvasPixel( x, y ) )
sLine = sLine .. c
if c ~= " " then
nLastChar = x
end
end
sLine = string.sub( sLine, 1, nLastChar )
tLines[y] = sLine
if string.len( sLine ) > 0 then
nLastLine = y
end
end
-- Save out
for n=1,nLastLine do
file.writeLine( tLines[ n ] )
end
file.close()
return true
end
--[[
Draws colour picker sidebar, the pallette and the footer
returns: nil
]]
local function drawInterface()
-- Footer
term.setCursorPos(1, h)
term.setBackgroundColour(colours.black)
term.setTextColour(colours.yellow)
term.clearLine()
term.write(fMessage)
-- Colour Picker
for i=1,16 do
term.setCursorPos(w-1, i)
term.setBackgroundColour( 2^(i-1) )
term.write(" ")
end
term.setCursorPos(w-1, 17)
term.setBackgroundColour( canvasColour )
term.setTextColour( colours.grey )
term.write("XX")
-- Left and Right Selected Colours
for i=18,18 do
term.setCursorPos(w-1, i)
if leftColour ~= nil then
term.setBackgroundColour( leftColour )
term.write(" ")
else
term.setBackgroundColour( canvasColour )
term.setTextColour( colours.grey )
term.write("X")
end
if rightColour ~= nil then
term.setBackgroundColour( rightColour )
term.write(" ")
else
term.setBackgroundColour( canvasColour )
term.setTextColour( colours.grey )
term.write("X")
end
end
-- Padding
term.setBackgroundColour( canvasColour )
for i=20,h-1 do
term.setCursorPos(w-1, i)
term.write(" ")
end
end
--[[
Converts a single pixel of a single line of the canvas and draws it
returns: nil
]]
local function drawCanvasPixel( x, y )
local pixel = getCanvasPixel( x, y )
if pixel then
term.setBackgroundColour( pixel or canvasColour )
term.setCursorPos(x, y)
term.write(" ")
else
term.setBackgroundColour( canvasColour )
term.setTextColour( colours.grey )
term.setCursorPos(x, y)
term.write("-")
end
end
--[[
Converts each colour in a single line of the canvas and draws it
returns: nil
]]
local function drawCanvasLine( y )
for x = 1, w-2 do
drawCanvasPixel( x, y )
end
end
--[[
Converts each colour in the canvas and draws it
returns: nil
]]
local function drawCanvas()
for y = 1, h-1 do
drawCanvasLine( y )
end
end
--[[
Draws menu options and handles input from within the menu.
returns: true if the program is to be exited; false otherwise
]]
local function accessMenu()
-- Selected menu option
local selection = 1
term.setBackgroundColour(colours.black)
while true do
-- Draw the menu
term.setCursorPos(1,h)
term.clearLine()
term.setTextColour(colours.white)
for k,v in pairs(mChoices) do
if selection==k then
term.setTextColour(colours.yellow)
local ox,_ = term.getCursorPos()
term.write("["..string.rep(" ",#v).."]")
term.setCursorPos(ox+1,h)
term.setTextColour(colours.white)
term.write(v)
term.setCursorPos(term.getCursorPos()+1,h)
else
term.write(" "..v.." ")
end
end
-- Handle input in the menu
local id,key = os.pullEvent("key")
if id == "key" then
-- S and E are shortcuts
if key == keys.s then
selection = 1
key = keys.enter
elseif key == keys.e then
selection = 2
key = keys.enter
end
if key == keys.right then
-- Move right
selection = selection + 1
if selection > #mChoices then
selection = 1
end
elseif key == keys.left and selection > 1 then
-- Move left
selection = selection - 1
if selection < 1 then
selection = #mChoices
end
elseif key == keys.enter then
-- Select an option
if mChoices[selection]=="Save" then
if bReadOnly then
fMessage = "Access Denied"
return false
end
local success = save(sPath)
if success then
fMessage = "Saved to "..sPath
else
fMessage = "Error saving to "..sPath
end
return false
elseif mChoices[selection]=="Exit" then
return true
end
elseif key == keys.leftCtrl or keys == keys.rightCtrl then
-- Cancel the menu
return false
end
end
end
end
--[[
Runs the main thread of execution. Draws the canvas and interface, and handles
mouse and key events.
returns: nil
]]
local function handleEvents()
local programActive = true
while programActive do
local id,p1,p2,p3 = os.pullEvent()
if id=="mouse_click" or id=="mouse_drag" then
if p2 >= w-1 and p3 >= 1 and p3 <= 17 then
if id ~= "mouse_drag" then
-- Selecting an items in the colour picker
if p3 <= 16 then
if p1==1 then
leftColour = 2^(p3-1)
else
rightColour = 2^(p3-1)
end
else
if p1==1 then
leftColour = nil
else
rightColour = nil
end
end
--drawCanvas()
drawInterface()
end
elseif p2 < w-1 and p3 <= h-1 then
-- Clicking on the canvas
local paintColour = nil
if p1==1 then
paintColour = leftColour
elseif p1==2 then
paintColour = rightColour
end
if not canvas[p3] then
canvas[p3] = {}
end
canvas[p3][p2] = paintColour
drawCanvasPixel( p2, p3 )
end
elseif id=="key" then
if p1==keys.leftCtrl or p1==keys.rightCtrl then
programActive = not accessMenu()
drawInterface()
end
elseif id=="term_resize" then
w,h = term.getSize()
drawCanvas()
drawInterface()
end
end
end
-- Init
load(sPath)
drawCanvas()
drawInterface()
-- Main loop
handleEvents()
-- Shutdown
term.setBackgroundColour(colours.black)
term.setTextColour(colours.white)
term.clear()
term.setCursorPos(1,1)