local GUI = require("GUI") local internet = require("Internet") local system = require("System") local filesystem = require("Filesystem") local image = require("Image") local color = require("Color") local screen = require("Screen") local paths = require("Paths") local text = require("Text") local args, options = system.parseArguments(...) -------------------------------------------------------------------- local recentColorsLimit = 52 local recentFilesLimit = 10 local config = { recentColors = {}, recentFiles = {}, transparencyBackground = 0xFFFFFF, transparencyForeground = 0xD2D2D2, } local currentScriptDirectory = filesystem.path(system.getCurrentScript()) local toolsPath = currentScriptDirectory .. "Tools/" local configPath = paths.user.applicationData .. "Picture Edit/Config2.cfg" local savePath local saveItem local tool local workspace, window, menu = system.addWindow(GUI.filledWindow(1, 1, 125, 37, 0x1E1E1E)) -------------------------------------------------------------------- window.newInput = function(...) return GUI.input(1, 1, 1, 1, 0x1E1E1E, 0xC3C3C3, 0x5A5A5A, 0x1E1E1E, 0xD2D2D2, ...) end window.newSlider = function(...) local slider = GUI.slider(1, 1, 1, 0x66DB80, 0x1E1E1E, 0xE1E1E1, 0x878787, ...) slider.roundValues = true return slider end window.newSwitch = function(...) return GUI.switchAndLabel(1, 1, 1, 6, 0x66DB80, 0x1E1E1E, 0xE1E1E1, 0x878787, ...) end window.newButton1 = function(...) return GUI.roundedButton(1, 1, 36, 1, 0xE1E1E1, 0x4B4B4B, 0x4B4B4B, 0xE1E1E1, ...) end window.newButton2 = function(...) local button = GUI.roundedButton(1, 1, 36, 1, 0x4B4B4B, 0xE1E1E1, 0x2D2D2D, 0xE1E1E1, ...) button.colors.disabled.background, button.colors.disabled.text = 0x4B4B4B, 0x787878 return button end -------------------------------------------------------------------- local function saveConfig() filesystem.writeTable(configPath, config) end local function loadConfig() if filesystem.exists(configPath) then config = filesystem.readTable(configPath) else local perLine = 13 local value, step = 0, 360 / (recentColorsLimit - perLine) for i = 1, recentColorsLimit - perLine do table.insert(config.recentColors, color.HSBToInteger(value, 1, 1)) value = value + step end value, step = 0, 255 / perLine for i = 1, perLine do table.insert(config.recentColors, color.RGBToInteger(math.floor(value), math.floor(value), math.floor(value))) value = value + step end saveConfig() end end local function addRecentFile(path) for i = 1, #config.recentFiles do if config.recentFiles[i] == path then return end end table.insert(config.recentFiles, 1, path) if #config.recentFiles > recentFilesLimit then table.remove(config.recentFiles, #config.recentFiles) end saveConfig() end loadConfig() local function addTitle(container, text) local titleContainer = container:addChild(GUI.container(1, 1, container.width, 1)) titleContainer:addChild(GUI.panel(1, 1, titleContainer.width, 1, 0x1E1E1E)) titleContainer:addChild(GUI.text(2, 1, 0xD2D2D2, text)) return titleContainer end window.sidebarPanel = window:addChild(GUI.panel(1, 1, 28, 1, 0x2D2D2D)) window.sidebarLayout = window:addChild(GUI.layout(1, 1, window.sidebarPanel.width, 1, 1, 1)) window.sidebarLayout:setAlignment(1, 1, GUI.ALIGNMENT_HORIZONTAL_CENTER, GUI.ALIGNMENT_VERTICAL_TOP) window.sidebarLayout.eventHandler = function(workspace, object, e1, e2, e3, e4, e5) if e1 == "scroll" then local h, v = window.sidebarLayout:getMargin(1, 1) local from = 0 local to = -window.sidebarLayout.cells[1][1].childrenHeight + 2 v = v + (e5 > 0 and 2 or -2) if v > from then v = from elseif v < to then v = to end window.sidebarLayout:setMargin(1, 1, h, v) workspace:draw() end end addTitle(window.sidebarLayout, "Recent colors") local recentColorsContainer = window.sidebarLayout:addChild(GUI.container(1, 1, window.sidebarLayout.width - 2, 4)) local x, y = 1, 1 for i = 1, #config.recentColors do local button = recentColorsContainer:addChild(GUI.button(x, y, 2, 1, 0x0, 0x0, 0x0, 0x0, " ")) button.onTouch = function() window.primaryColorSelector.color = config.recentColors[i] workspace:draw() end x = x + 2 if x > recentColorsContainer.width - 1 then x, y = 1, y + 1 end end local currentToolTitle = addTitle(window.sidebarLayout, "Tool properties") window.currentToolLayout = window.sidebarLayout:addChild(GUI.layout(1, 1, window.sidebarLayout.width, 1, 1, 1)) window.currentToolLayout:setAlignment(1, 1, GUI.ALIGNMENT_HORIZONTAL_CENTER, GUI.ALIGNMENT_VERTICAL_TOP) window.currentToolLayout:setFitting(1, 1, true, false, 2, 0) local aboutToolTitle = addTitle(window.sidebarLayout, "About tool") local aboutToolTextBox = window.sidebarLayout:addChild(GUI.textBox(1, 1, window.sidebarLayout.width - 2, 1, nil, 0x787878, {}, 1, 0, 0)) window.toolsList = window:addChild(GUI.list(1, 1, 7, 1, 3, 0, 0x2D2D2D, 0x787878, 0x2D2D2D, 0x787878, 0x3C3C3C, 0xE1E1E1)) window.toolsList:setMargin(0, 3) window.image = window:addChild(GUI.object(1, 1, 1, 1)) window.image.data = {} local function onToolTouch(index) tool = window.toolsList:getItem(index).tool window.toolsList.selectedItem = index window.currentToolOverlay:removeChildren() window.currentToolLayout:removeChildren() currentToolTitle.hidden = not tool.onSelection window.currentToolLayout.hidden = currentToolTitle.hidden window.sidebarLayout:setMargin(1, 1, 0, 0) if tool.onSelection then local result, reason = pcall(tool.onSelection) if result then window.currentToolLayout:update() local lastChild = window.currentToolLayout.children[#window.currentToolLayout.children] if lastChild then window.currentToolLayout.height = lastChild.localY + lastChild.height - 1 end else GUI.alert(reason) end end aboutToolTitle.hidden = not tool.about aboutToolTextBox.hidden = aboutToolTitle.hidden if tool.about then aboutToolTextBox.lines = text.wrap({tool.about}, aboutToolTextBox.width) aboutToolTextBox.height = #aboutToolTextBox.lines end workspace:draw() end local tools = filesystem.list(toolsPath) for i = 1, #tools do if filesystem.extension(tools[i]) == ".lua" then local result, reason = loadfile(toolsPath .. tools[i]) if result then result, reason = pcall(result, workspace, window, menu) if result then local item = window.toolsList:addItem(reason.shortcut) item.tool = reason item.onTouch = function() onToolTouch(i) end else error("Failed to perform pcall() on tool " .. tools[i] .. ": " .. reason) end else error("Failed to perform loadfile() on tool " .. tools[i] .. ": " .. reason) end end end window.image.draw = function(object) GUI.drawShadow(object.x, object.y, object.width, object.height, nil, true) local y, text = object.y + object.height + 1, "Size: " .. object.width .. "x" .. object.height screen.drawText(math.floor(object.x + object.width / 2 - unicode.len(text) / 2), y, 0x5A5A5A, text) if savePath then text = "Path: " .. savePath screen.drawText(math.floor(object.x + object.width / 2 - unicode.len(text) / 2), y + 1, 0x5A5A5A, text) end local x, y, step, notStep, background, foreground, symbol = object.x, object.y, false, window.image.width % 2 for i = 3, #window.image.data, 4 do if window.image.data[i + 2] == 0 then background = window.image.data[i] foreground = window.image.data[i + 1] symbol = window.image.data[i + 3] elseif window.image.data[i + 2] < 1 then background = color.blend(config.transparencyBackground, window.image.data[i], window.image.data[i + 2]) foreground = window.image.data[i + 1] symbol = window.image.data[i + 3] else if window.image.data[i + 3] == " " then background = config.transparencyBackground foreground = config.transparencyForeground symbol = step and "▒" or "░" else background = config.transparencyBackground foreground = window.image.data[i + 1] symbol = window.image.data[i + 3] end end screen.set(x, y, background, foreground, symbol) x, step = x + 1, not step if x > object.x + object.width - 1 then x, y = object.x, y + 1 if notStep == 0 then step = not step end end end end local function updateRecentColorsButtons() for i = 1, #config.recentColors do recentColorsContainer.children[i].colors.default.background = config.recentColors[i] recentColorsContainer.children[i].colors.pressed.background = 0xFFFFFF - config.recentColors[i] end end local function swapColors() window.primaryColorSelector.color, window.secondaryColorSelector.color = window.secondaryColorSelector.color, window.primaryColorSelector.color workspace:draw() end local function colorSelectorDraw(object) screen.drawRectangle(object.x + 1, object.y, object.width - 2, object.height, object.color, 0x0, " ") for y = object.y, object.y + object.height - 1 do screen.drawText(object.x, y, object.color, "⢸") screen.drawText(object.x + object.width - 1, y, object.color, "⡇") end end window.secondaryColorSelector = window:addChild(GUI.colorSelector(3, 1, 5, 2, 0xFFFFFF, " ")) window.primaryColorSelector = window:addChild(GUI.colorSelector(2, 1, 5, 2, 0x000000, " ")) window.secondaryColorSelector.draw, window.primaryColorSelector.draw = colorSelectorDraw, colorSelectorDraw window.swapColorsButton = window:addChild(GUI.button(1, 1, window.toolsList.width, 1, nil, 0x696969, nil, 0xA5A5A5, ">")) window.swapColorsButton.onTouch = swapColors window.image.eventHandler = function(workspace, object, e1, e2, e3, e4, ...) if e1 == "key_down" then -- D if e4 == 32 then window.primaryColorSelector.color, window.secondaryColorSelector.color = 0x0, 0xFFFFFF workspace:draw() -- X elseif e4 == 45 then swapColors() else for i = 1, window.toolsList:count() do if e4 == window.toolsList:getItem(i).tool.keyCode then onToolTouch(i) return end end end end local result, reason = pcall(tool.eventHandler, workspace, object, e1, e2, e3, e4, ...) if not result then GUI.alert("Tool eventHandler() failed: " .. reason) end end window.image.setPosition = function(x, y) window.image.localX = x window.image.localY = y window.currentToolOverlay.localX = x window.currentToolOverlay.localY = y end window.image.reposition = function() window.image.width = window.image.data[1] window.image.height = window.image.data[2] window.currentToolOverlay.width = window.image.width window.currentToolOverlay.height = window.image.height if window.image.width < window.backgroundPanel.width then window.image.setPosition( math.floor(window.backgroundPanel.localX + window.backgroundPanel.width / 2 - window.image.width / 2), math.floor(window.backgroundPanel.localY + window.backgroundPanel.height / 2 - window.image.height / 2) ) else window.image.setPosition( window.backgroundPanel.localX, window.backgroundPanel.localY ) end end local function newNoGUI(width, height) savePath, saveItem.disabled = nil, true window.image.data = {width, height} for i = 1, width * height do table.insert(window.image.data, 0x0) table.insert(window.image.data, 0x0) table.insert(window.image.data, 1) table.insert(window.image.data, " ") end end local function new() local container = GUI.addBackgroundContainer(workspace, true, true, "New picture") local widthInput = container.layout:addChild(GUI.input(1, 1, 36, 3, 0xE1E1E1, 0x696969, 0x696969, 0xE1E1E1, 0x2D2D2D, "51", "Width")) local heightInput = container.layout:addChild(GUI.input(1, 1, 36, 3, 0xE1E1E1, 0x696969, 0x696969, 0xE1E1E1, 0x2D2D2D, "19", "Height")) widthInput.validator = function(text) return tonumber(text) end heightInput.validator = widthInput.validator container.panel.eventHandler = function(workspace, object, e1) if e1 == "touch" then newNoGUI(tonumber(widthInput.text), tonumber(heightInput.text)) window.image.reposition() container:remove() workspace:draw() end end workspace:draw() end local function loadImage(path) local result, reason if filesystem.extension(path) == ".rawpic" then result = image.fromString(filesystem.read(path)) else result, reason = image.load(path) end if result then savePath, saveItem.disabled = path, false addRecentFile(path) window.image.data = result else GUI.alert(reason) end end local function saveImage(path) if filesystem.extension(path) == ".pic" then local result, reason = image.save(path, window.image.data, 6) if result then savePath, saveItem.disabled = path, false addRecentFile(path) else GUI.alert(reason) end else savePath, saveItem.disabled = path, false filesystem.write(path, image.toString(window.image.data)) end computer.pushSignal("system", "updateFileList") end local fileItem = menu:addContextMenuItem("File") fileItem:addItem("New").onTouch = new fileItem:addSeparator() fileItem:addItem("Open").onTouch = function() local filesystemDialog = GUI.addFilesystemDialog(workspace, true, 50, math.floor(window.height * 0.8), "Open", "Cancel", "File name", "/") filesystemDialog:setMode(GUI.IO_MODE_OPEN, GUI.IO_MODE_FILE) filesystemDialog:addExtensionFilter(".pic") filesystemDialog:addExtensionFilter(".rawpic") filesystemDialog:expandPath(paths.user.desktop) filesystemDialog:show() filesystemDialog.onSubmit = function(path) loadImage(path) window.image.reposition() workspace:draw() end end local fileItemSubMenu = fileItem:addSubMenuItem("Open recent", #config.recentFiles == 0) for i = 1, #config.recentFiles do fileItemSubMenu:addItem(text.limit(config.recentFiles[i], 32, "left")).onTouch = function() loadImage(config.recentFiles[i]) window.image.reposition() workspace:draw() end end fileItem:addItem("Open from URL").onTouch = function() local container = GUI.addBackgroundContainer(workspace, true, true, "Open from URL") local input = container.layout:addChild(GUI.input(1, 1, 36, 3, 0xE1E1E1, 0x696969, 0x969696, 0xE1E1E1, 0x2D2D2D, "", "http://example.com/test.pic")) input.onInputFinished = function() if #input.text > 0 then input:remove() container.layout:addChild(GUI.label(1, 1, container.width, 1, 0x969696, "Downloading file..."):setAlignment(GUI.ALIGNMENT_HORIZONTAL_CENTER, GUI.ALIGNMENT_VERTICAL_TOP)) workspace:draw() local temporaryPath = system.getTemporaryPath() .. ".pic" local result, reason = internet.download(input.text, temporaryPath) container:remove() if result then loadImage(temporaryPath) window.image.reposition() filesystem.remove(temporaryPath) savePath, saveItem.disabled = nil, true else GUI.alert(reason) end workspace:draw() end end workspace:draw() end fileItem:addSeparator() saveItem = fileItem:addItem("Save") saveItem.onTouch = function() saveImage(savePath) end fileItem:addItem("Save as").onTouch = function() local filesystemDialog = GUI.addFilesystemDialog(workspace, true, 50, math.floor(window.height * 0.8), "Save", "Cancel", "File name", "/") filesystemDialog:setMode(GUI.IO_MODE_SAVE, GUI.IO_MODE_FILE) filesystemDialog:addExtensionFilter(".pic") filesystemDialog:addExtensionFilter(".ocifstring") filesystemDialog:expandPath(paths.user.desktop) filesystemDialog.filesystemTree.selectedItem = paths.user.desktop filesystemDialog:show() filesystemDialog.onSubmit = function(path) saveImage(path) end end fileItem:addSeparator() fileItem:addItem("Exit").onTouch = function() window:remove() end menu:addItem("View").onTouch = function() local container = GUI.addBackgroundContainer(workspace, true, true, "View") local colorSelector1 = container.layout:addChild(GUI.colorSelector(1, 1, 36, 3, config.transparencyBackground, "Transparency background")) local colorSelector2 = container.layout:addChild(GUI.colorSelector(1, 1, 36, 3, config.transparencyForeground, "Transparency foreground")) container.panel.eventHandler = function(workspace, object, e1) if e1 == "touch" then config.transparencyBackground, config.transparencyForeground = colorSelector1.color, colorSelector2.color container:remove() workspace:draw() saveConfig() end end workspace:draw() end menu:addItem("Hotkeys").onTouch = function() local container = GUI.addBackgroundContainer(workspace, true, true, "Hotkeys") local lines = { "There are some hotkeys that works exactly like in real Photoshop:", " ", "M - selection tool", "V - move tool", "C - resizer tool", "Alt - picker tool", "B - brush tool", "E - eraser tool", "T - text tool", "G - fill tool", "F - braille tool", " ", "X - switch colors", "D - make colors B/W", } container.layout:addChild(GUI.textBox(1, 1, 36, 1, nil, 0x969696, lines, 1, 0, 0, true, true)).eventHandler = nil workspace:draw() end window.currentToolOverlay = window:addChild(GUI.container(1, 1, 1, 1)) window.onResize = function(width, height) window.backgroundPanel.localX = window.toolsList.width + 1 window.backgroundPanel.width = width - window.sidebarLayout.width - window.toolsList.width window.backgroundPanel.height = height window.sidebarPanel.localX = window.width - window.sidebarPanel.width + 1 window.sidebarPanel.height = height window.sidebarLayout.localX = window.sidebarPanel.localX window.sidebarLayout.height = height window.toolsList.height = height window.secondaryColorSelector.localY = height - 4 window.primaryColorSelector.localY = height - 5 window.swapColorsButton.localY = height - 1 window.image.reposition() end ---------------------------------------------------------------- window.actionButtons:moveToFront() updateRecentColorsButtons() if (options.o or options.open) and args[1] and filesystem.exists(args[1]) then loadImage(args[1]) else newNoGUI(51, 19) end window:resize(window.width, window.height) onToolTouch(5)