Advertisement
TIMAS_Bro

Musiclo | CC:T music player [QUICK FIX]

Apr 2nd, 2025
516
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
  1. -- Musiclo music player made by timuzkas
  2. -- licensed under Creative Commons CC0
  3. -- Simple, sleek player for YT.
  4. -- //
  5. -- //
  6. -- Backend code from @terreng on github, using MIT license
  7. -- Frontend code by timuzkas, using MIT license
  8. -- Transliterator by timuzkas, using MIT license
  9. -- PrimeUI by JackMacWindows, using CC0 license
  10.  
  11. local expect = require "cc.expect".expect
  12. -- PrimeUI by JackMacWindows
  13. -- Public domain/CC0
  14.  
  15. -- Initialization code
  16. local PrimeUI = {}
  17. do
  18.     local coros = {}
  19.     local restoreCursor
  20.  
  21.     --- Adds a task to run in the main loop.
  22.     ---@param func function The function to run, usually an `os.pullEvent` loop
  23.     function PrimeUI.addTask(func)
  24.         expect(1, func, "function")
  25.         local t = {coro = coroutine.create(func)}
  26.         coros[#coros+1] = t
  27.         _, t.filter = coroutine.resume(t.coro)
  28.     end
  29.  
  30.     --- Sends the provided arguments to the run loop, where they will be returned.
  31.     ---@param ... any The parameters to send
  32.     function PrimeUI.resolve(...)
  33.         coroutine.yield(coros, ...)
  34.     end
  35.  
  36.     --- Clears the screen and resets all components. Do not use any previously
  37.     --- created components after calling this function.
  38.     function PrimeUI.clear()
  39.         -- Reset the screen.
  40.         term.setCursorPos(1, 1)
  41.         term.setCursorBlink(false)
  42.         term.setBackgroundColor(colors.black)
  43.         term.setTextColor(colors.white)
  44.         term.clear()
  45.         -- Reset the task list and cursor restore function.
  46.         coros = {}
  47.         restoreCursor = nil
  48.     end
  49.  
  50.     --- Sets or clears the window that holds where the cursor should be.
  51.     ---@param win window|nil The window to set as the active window
  52.     function PrimeUI.setCursorWindow(win)
  53.         expect(1, win, "table", "nil")
  54.         restoreCursor = win and win.restoreCursor
  55.     end
  56.  
  57.     --- Gets the absolute position of a coordinate relative to a window.
  58.     ---@param win window The window to check
  59.     ---@param x number The relative X position of the point
  60.     ---@param y number The relative Y position of the point
  61.     ---@return number x The absolute X position of the window
  62.     ---@return number y The absolute Y position of the window
  63.     function PrimeUI.getWindowPos(win, x, y)
  64.         if win == term then return x, y end
  65.         while win ~= term.native() and win ~= term.current() do
  66.             if not win.getPosition then return x, y end
  67.             local wx, wy = win.getPosition()
  68.             x, y = x + wx - 1, y + wy - 1
  69.             _, win = debug.getupvalue(select(2, debug.getupvalue(win.isColor, 1)), 1) -- gets the parent window through an upvalue
  70.         end
  71.         return x, y
  72.     end
  73.  
  74.     --- Runs the main loop, returning information on an action.
  75.     ---@return any ... The result of the coroutine that exited
  76.     function PrimeUI.run()
  77.         while true do
  78.             -- Restore the cursor and wait for the next event.
  79.             if restoreCursor then restoreCursor() end
  80.             local ev = table.pack(os.pullEvent())
  81.             -- Run all coroutines.
  82.             for _, v in ipairs(coros) do
  83.                 if v.filter == nil or v.filter == ev[1] then
  84.                     -- Resume the coroutine, passing the current event.
  85.                     local res = table.pack(coroutine.resume(v.coro, table.unpack(ev, 1, ev.n)))
  86.                     -- If the call failed, bail out. Coroutines should never exit.
  87.                     if not res[1] then error(res[2], 2) end
  88.                     -- If the coroutine resolved, return its values.
  89.                     if res[2] == coros then return table.unpack(res, 3, res.n) end
  90.                     -- Set the next event filter.
  91.                     v.filter = res[2]
  92.                 end
  93.             end
  94.         end
  95.     end
  96. end
  97.  
  98. --- Draws a thin border around a screen region.
  99. ---@param win window The window to draw on
  100. ---@param x number The X coordinate of the inside of the box
  101. ---@param y number The Y coordinate of the inside of the box
  102. ---@param width number The width of the inner box
  103. ---@param height number The height of the inner box
  104. ---@param fgColor color|nil The color of the border (defaults to white)
  105. ---@param bgColor color|nil The color of the background (defaults to black)
  106. function PrimeUI.borderBox(win, x, y, width, height, fgColor, bgColor)
  107.     expect(1, win, "table")
  108.     expect(2, x, "number")
  109.     expect(3, y, "number")
  110.     expect(4, width, "number")
  111.     expect(5, height, "number")
  112.     fgColor = expect(6, fgColor, "number", "nil") or colors.white
  113.     bgColor = expect(7, bgColor, "number", "nil") or colors.black
  114.     -- Draw the top-left corner & top border.
  115.     win.setBackgroundColor(bgColor)
  116.     win.setTextColor(fgColor)
  117.     win.setCursorPos(x - 1, y - 1)
  118.     win.write("\x9C" .. ("\x8C"):rep(width))
  119.     -- Draw the top-right corner.
  120.     win.setBackgroundColor(fgColor)
  121.     win.setTextColor(bgColor)
  122.     win.write("\x93")
  123.     -- Draw the right border.
  124.     for i = 1, height do
  125.         win.setCursorPos(win.getCursorPos() - 1, y + i - 1)
  126.         win.write("\x95")
  127.     end
  128.     -- Draw the left border.
  129.     win.setBackgroundColor(bgColor)
  130.     win.setTextColor(fgColor)
  131.     for i = 1, height do
  132.         win.setCursorPos(x - 1, y + i - 1)
  133.         win.write("\x95")
  134.     end
  135.     -- Draw the bottom border and corners.
  136.     win.setCursorPos(x - 1, y + height)
  137.     win.write("\x8D" .. ("\x8C"):rep(width) .. "\x8E")
  138. end
  139.  
  140. --- Creates a clickable button on screen with text.
  141. ---@param win window The window to draw on
  142. ---@param x number The X position of the button
  143. ---@param y number The Y position of the button
  144. ---@param text string The text to draw on the button
  145. ---@param action function|string A function to call when clicked, or a string to send with a `run` event
  146. ---@param fgColor color|nil The color of the button text (defaults to white)
  147. ---@param bgColor color|nil The color of the button (defaults to light gray)
  148. ---@param clickedColor color|nil The color of the button when clicked (defaults to gray)
  149. ---@param periphName string|nil The name of the monitor peripheral, or nil (set if you're using a monitor - events will be filtered to that monitor)
  150. function PrimeUI.button(win, x, y, text, action, fgColor, bgColor, clickedColor, periphName)
  151.     expect(1, win, "table")
  152.     expect(1, win, "table")
  153.     expect(2, x, "number")
  154.     expect(3, y, "number")
  155.     expect(4, text, "string")
  156.     expect(5, action, "function", "string")
  157.     fgColor = expect(6, fgColor, "number", "nil") or colors.white
  158.     bgColor = expect(7, bgColor, "number", "nil") or colors.gray
  159.     clickedColor = expect(8, clickedColor, "number", "nil") or colors.lightGray
  160.     periphName = expect(9, periphName, "string", "nil")
  161.     -- Draw the initial button.
  162.     win.setCursorPos(x, y)
  163.     win.setBackgroundColor(bgColor)
  164.     win.setTextColor(fgColor)
  165.     win.write(" " .. text .. " ")
  166.     -- Get the screen position and add a click handler.
  167.     PrimeUI.addTask(function()
  168.         local buttonDown = false
  169.         while true do
  170.             local event, button, clickX, clickY = os.pullEvent()
  171.             local screenX, screenY = PrimeUI.getWindowPos(win, x, y)
  172.             if event == "mouse_click" and periphName == nil and button == 1 and clickX >= screenX and clickX < screenX + #text + 2 and clickY == screenY then
  173.                 -- Initiate a click action (but don't trigger until mouse up).
  174.                 buttonDown = true
  175.                 -- Redraw the button with the clicked background color.
  176.                 win.setCursorPos(x, y)
  177.                 win.setBackgroundColor(clickedColor)
  178.                 win.setTextColor(fgColor)
  179.                 win.write(" " .. text .. " ")
  180.             elseif (event == "monitor_touch" and periphName == button and clickX >= screenX and clickX < screenX + #text + 2 and clickY == screenY)
  181.                 or (event == "mouse_up" and button == 1 and buttonDown) then
  182.                 -- Finish a click event.
  183.                 if clickX >= screenX and clickX < screenX + #text + 2 and clickY == screenY then
  184.                     -- Trigger the action.
  185.                     if type(action) == "string" then
  186.                         PrimeUI.resolve("button", action)
  187.                     else
  188.                         action()
  189.                     end
  190.                 end
  191.                 -- Redraw the original button state.
  192.                 win.setCursorPos(x, y)
  193.                 win.setBackgroundColor(bgColor)
  194.                 win.setTextColor(fgColor)
  195.                 win.write(" " .. text .. " ")
  196.             end
  197.         end
  198.     end)
  199. end
  200.  
  201. --- Draws a line of text, centering it inside a box horizontally.
  202. ---@param win window The window to draw on
  203. ---@param x number The X position of the left side of the box
  204. ---@param y number The Y position of the box
  205. ---@param width number The width of the box to draw in
  206. ---@param text string The text to draw
  207. ---@param fgColor color|nil The color of the text (defaults to white)
  208. ---@param bgColor color|nil The color of the background (defaults to black)
  209. function PrimeUI.centerLabel(win, x, y, width, text, fgColor, bgColor)
  210.     expect(1, win, "table")
  211.     expect(2, x, "number")
  212.     expect(3, y, "number")
  213.     expect(4, width, "number")
  214.     expect(5, text, "string")
  215.     fgColor = expect(6, fgColor, "number", "nil") or colors.white
  216.     bgColor = expect(7, bgColor, "number", "nil") or colors.black
  217.     assert(#text <= width, "string is too long")
  218.     win.setCursorPos(x + math.floor((width - #text) / 2), y)
  219.     win.setTextColor(fgColor)
  220.     win.setBackgroundColor(bgColor)
  221.     win.write(text)
  222. end
  223.  
  224. --- Creates a list of entries with toggleable check boxes.
  225. ---@param win window The window to draw on
  226. ---@param x number The X coordinate of the inside of the box
  227. ---@param y number The Y coordinate of the inside of the box
  228. ---@param width number The width of the inner box
  229. ---@param height number The height of the inner box
  230. ---@param selections table<string,string|boolean> A list of entries to show, where the value is whether the item is pre-selected (or `"R"` for required/forced selected)
  231. ---@param action function|string|nil A function or `run` event that's called when a selection is made
  232. ---@param fgColor color|nil The color of the text (defaults to white)
  233. ---@param bgColor color|nil The color of the background (defaults to black)
  234. function PrimeUI.checkSelectionBox(win, x, y, width, height, selections, action, fgColor, bgColor)
  235.     expect(1, win, "table")
  236.     expect(2, x, "number")
  237.     expect(3, y, "number")
  238.     expect(4, width, "number")
  239.     expect(5, height, "number")
  240.     expect(6, selections, "table")
  241.     expect(7, action, "function", "string", "nil")
  242.     fgColor = expect(8, fgColor, "number", "nil") or colors.white
  243.     bgColor = expect(9, bgColor, "number", "nil") or colors.black
  244.     -- Calculate how many selections there are.
  245.     local nsel = 0
  246.     for _ in pairs(selections) do nsel = nsel + 1 end
  247.     -- Create the outer display box.
  248.     local outer = window.create(win, x, y, width, height)
  249.     outer.setBackgroundColor(bgColor)
  250.     outer.clear()
  251.     -- Create the inner scroll box.
  252.     local inner = window.create(outer, 1, 1, width - 1, nsel)
  253.     inner.setBackgroundColor(bgColor)
  254.     inner.setTextColor(fgColor)
  255.     inner.clear()
  256.     -- Draw each line in the window.
  257.     local lines = {}
  258.     local nl, selected = 1, 1
  259.     for k, v in pairs(selections) do
  260.         inner.setCursorPos(1, nl)
  261.         inner.write((v and (v == "R" and "[-] " or "[\xD7] ") or "[ ] ") .. k)
  262.         lines[nl] = {k, not not v}
  263.         nl = nl + 1
  264.     end
  265.     -- Draw a scroll arrow if there is scrolling.
  266.     if nsel > height then
  267.         outer.setCursorPos(width, height)
  268.         outer.setBackgroundColor(bgColor)
  269.         outer.setTextColor(fgColor)
  270.         outer.write("\31")
  271.     end
  272.     -- Set cursor blink status.
  273.     inner.setCursorPos(2, selected)
  274.     inner.setCursorBlink(true)
  275.     PrimeUI.setCursorWindow(inner)
  276.     -- Get screen coordinates & add run task.
  277.     local screenX, screenY = PrimeUI.getWindowPos(win, x, y)
  278.     PrimeUI.addTask(function()
  279.         local scrollPos = 1
  280.         while true do
  281.             -- Wait for an event.
  282.             local ev = table.pack(os.pullEvent())
  283.             -- Look for a scroll event or a selection event.
  284.             local dir
  285.             if ev[1] == "key" then
  286.                 if ev[2] == keys.up then dir = -1
  287.                 elseif ev[2] == keys.down then dir = 1
  288.                 elseif ev[2] == keys.space and selections[lines[selected][1]] ~= "R" then
  289.                     -- (Un)select the item.
  290.                     lines[selected][2] = not lines[selected][2]
  291.                     inner.setCursorPos(2, selected)
  292.                     inner.write(lines[selected][2] and "\xD7" or " ")
  293.                     -- Call the action if passed; otherwise, set the original table.
  294.                     if type(action) == "string" then PrimeUI.resolve("checkSelectionBox", action, lines[selected][1], lines[selected][2])
  295.                     elseif action then action(lines[selected][1], lines[selected][2])
  296.                     else selections[lines[selected][1]] = lines[selected][2] end
  297.                     -- Redraw all lines in case of changes.
  298.                     for i, v in ipairs(lines) do
  299.                         local vv = selections[v[1]] == "R" and "R" or v[2]
  300.                         inner.setCursorPos(2, i)
  301.                         inner.write((vv and (vv == "R" and "-" or "\xD7") or " "))
  302.                     end
  303.                     inner.setCursorPos(2, selected)
  304.                 end
  305.             elseif ev[1] == "mouse_scroll" and ev[3] >= screenX and ev[3] < screenX + width and ev[4] >= screenY and ev[4] < screenY + height then
  306.                 dir = ev[2]
  307.             end
  308.             -- Scroll the screen if required.
  309.             if dir and (selected + dir >= 1 and selected + dir <= nsel) then
  310.                 selected = selected + dir
  311.                 if selected - scrollPos < 0 or selected - scrollPos >= height then
  312.                     scrollPos = scrollPos + dir
  313.                     inner.reposition(1, 2 - scrollPos)
  314.                 end
  315.                 inner.setCursorPos(2, selected)
  316.             end
  317.             -- Redraw scroll arrows and reset cursor.
  318.             outer.setCursorPos(width, 1)
  319.             outer.write(scrollPos > 1 and "\30" or " ")
  320.             outer.setCursorPos(width, height)
  321.             outer.write(scrollPos < nsel - height + 1 and "\31" or " ")
  322.             inner.restoreCursor()
  323.         end
  324.     end)
  325. end
  326.  
  327. --- Creates a clickable region on screen without any content.
  328. ---@param win window The window to draw on
  329. ---@param x number The X position of the button
  330. ---@param y number The Y position of the button
  331. ---@param width number The width of the inner box
  332. ---@param height number The height of the inner box
  333. ---@param action function|string A function to call when clicked, or a string to send with a `run` event
  334. ---@param periphName string|nil The name of the monitor peripheral, or nil (set if you're using a monitor - events will be filtered to that monitor)
  335. function PrimeUI.clickRegion(win, x, y, width, height, action, periphName)
  336.     expect(1, win, "table")
  337.     expect(2, x, "number")
  338.     expect(3, y, "number")
  339.     expect(4, width, "number")
  340.     expect(5, height, "number")
  341.     expect(6, action, "function", "string")
  342.     expect(7, periphName, "string", "nil")
  343.     PrimeUI.addTask(function()
  344.         -- Get the screen position and add a click handler.
  345.         local screenX, screenY = PrimeUI.getWindowPos(win, x, y)
  346.         local buttonDown = false
  347.         while true do
  348.             local event, button, clickX, clickY = os.pullEvent()
  349.             if (event == "monitor_touch" and periphName == button)
  350.                 or (event == "mouse_click" and button == 1 and periphName == nil) then
  351.                 -- Finish a click event.
  352.                 if clickX >= screenX and clickX < screenX + width
  353.                     and clickY >= screenY and clickY < screenY + height then
  354.                     -- Trigger the action.
  355.                     if type(action) == "string" then
  356.                         PrimeUI.resolve("clickRegion", action)
  357.                     else
  358.                         action()
  359.                     end
  360.                 end
  361.             end
  362.         end
  363.     end)
  364. end
  365.  
  366. --- Draws a NFT-formatted image to the screen.
  367. ---@param win window The window to draw on
  368. ---@param x number The X position of the top left corner of the image
  369. ---@param y number The Y position of the top left corner of the image
  370. ---@param data string|table The path to the image to load, or the image data itself
  371. function PrimeUI.drawNFT(win, x, y, data)
  372.     expect(1, win, "table")
  373.     expect(2, x, "number")
  374.     expect(3, y, "number")
  375.     expect(4, data, "string", "table")
  376.     -- Load the image file if a string was passed using nft.load.
  377.     if type(data) == "string" then
  378.         data = assert(nft.load("data/example.nft"), "File is not a valid NFT file")
  379.     end
  380.     nft.draw(data, x, y , win)
  381. end
  382.  
  383. --- Draws a block of text inside a window with word wrapping, optionally resizing the window to fit.
  384. ---@param win window The window to draw in
  385. ---@param text string The text to draw
  386. ---@param resizeToFit boolean|nil Whether to resize the window to fit the text (defaults to false). This is useful for scroll boxes.
  387. ---@param fgColor color|nil The color of the text (defaults to white)
  388. ---@param bgColor color|nil The color of the background (defaults to black)
  389. ---@return number lines The total number of lines drawn
  390. function PrimeUI.drawText(win, text, resizeToFit, fgColor, bgColor)
  391.     expect(1, win, "table")
  392.     expect(2, text, "string")
  393.     expect(3, resizeToFit, "boolean", "nil")
  394.     fgColor = expect(4, fgColor, "number", "nil") or colors.white
  395.     bgColor = expect(5, bgColor, "number", "nil") or colors.black
  396.     -- Set colors.
  397.     win.setBackgroundColor(bgColor)
  398.     win.setTextColor(fgColor)
  399.     -- Redirect to the window to use print on it.
  400.     local old = term.redirect(win)
  401.     -- Draw the text using print().
  402.     local lines = print(text)
  403.     -- Redirect back to the original terminal.
  404.     term.redirect(old)
  405.     -- Resize the window if desired.
  406.     if resizeToFit then
  407.         -- Get original parameters.
  408.         local x, y = win.getPosition()
  409.         local w = win.getSize()
  410.         -- Resize the window.
  411.         win.reposition(x, y, w, lines)
  412.     end
  413.     return lines
  414. end
  415.  
  416. --- Draws a horizontal line at a position with the specified width.
  417. ---@param win window The window to draw on
  418. ---@param x number The X position of the left side of the line
  419. ---@param y number The Y position of the line
  420. ---@param width number The width/length of the line
  421. ---@param fgColor color|nil The color of the line (defaults to white)
  422. ---@param bgColor color|nil The color of the background (defaults to black)
  423. function PrimeUI.horizontalLine(win, x, y, width, fgColor, bgColor)
  424.     expect(1, win, "table")
  425.     expect(2, x, "number")
  426.     expect(3, y, "number")
  427.     expect(4, width, "number")
  428.     fgColor = expect(5, fgColor, "number", "nil") or colors.white
  429.     bgColor = expect(6, bgColor, "number", "nil") or colors.black
  430.     -- Use drawing characters to draw a thin line.
  431.     win.setCursorPos(x, y)
  432.     win.setTextColor(fgColor)
  433.     win.setBackgroundColor(bgColor)
  434.     win.write(("\x8C"):rep(width))
  435. end
  436.  
  437. --- Creates a text input box.
  438. ---@param win window The window to draw on
  439. ---@param x number The X position of the left side of the box
  440. ---@param y number The Y position of the box
  441. ---@param width number The width/length of the box
  442. ---@param action function|string A function or `run` event to call when the enter key is pressed
  443. ---@param fgColor color|nil The color of the text (defaults to white)
  444. ---@param bgColor color|nil The color of the background (defaults to black)
  445. ---@param replacement string|nil A character to replace typed characters with
  446. ---@param history string[]|nil A list of previous entries to provide
  447. ---@param completion function|nil A function to call to provide completion
  448. ---@param default string|nil A string to return if the box is empty
  449. function PrimeUI.inputBox(win, x, y, width, action, fgColor, bgColor, replacement, history, completion, default)
  450.     expect(1, win, "table")
  451.     expect(2, x, "number")
  452.     expect(3, y, "number")
  453.     expect(4, width, "number")
  454.     expect(5, action, "function", "string")
  455.     fgColor = expect(6, fgColor, "number", "nil") or colors.white
  456.     bgColor = expect(7, bgColor, "number", "nil") or colors.black
  457.     expect(8, replacement, "string", "nil")
  458.     expect(9, history, "table", "nil")
  459.     expect(10, completion, "function", "nil")
  460.     expect(11, default, "string", "nil")
  461.     -- Create a window to draw the input in.
  462.     local box = window.create(win, x, y, width, 1)
  463.     box.setTextColor(fgColor)
  464.     box.setBackgroundColor(bgColor)
  465.     box.clear()
  466.     -- Call read() in a new coroutine.
  467.     PrimeUI.addTask(function()
  468.         -- We need a child coroutine to be able to redirect back to the window.
  469.         local coro = coroutine.create(read)
  470.         -- Run the function for the first time, redirecting to the window.
  471.         local old = term.redirect(box)
  472.         local ok, res = coroutine.resume(coro, replacement, history, completion, default)
  473.         term.redirect(old)
  474.         -- Run the coroutine until it finishes.
  475.         while coroutine.status(coro) ~= "dead" do
  476.             -- Get the next event.
  477.             local ev = table.pack(os.pullEvent())
  478.             -- Redirect and resume.
  479.             old = term.redirect(box)
  480.             ok, res = coroutine.resume(coro, table.unpack(ev, 1, ev.n))
  481.             term.redirect(old)
  482.             -- Pass any errors along.
  483.             if not ok then error(res) end
  484.         end
  485.         -- Send the result to the receiver.
  486.         if type(action) == "string" then PrimeUI.resolve("inputBox", action, res)
  487.         else action(res) end
  488.         -- Spin forever, because tasks cannot exit.
  489.         while true do os.pullEvent() end
  490.     end)
  491. end
  492.  
  493. --- Runs a function or action repeatedly after a specified time period until canceled.
  494. --- If a function is passed as an action, it may return a number to change the
  495. --- period, or `false` to stop it.
  496. ---@param time number The amount of time to wait for each time, in seconds
  497. ---@param action function|string The function to call when the timer completes, or a `run` event to send
  498. ---@return function cancel A function to cancel the timer
  499. function PrimeUI.interval(time, action)
  500.     expect(1, time, "number")
  501.     expect(2, action, "function", "string")
  502.     -- Start the timer.
  503.     local timer = os.startTimer(time)
  504.     -- Add a task to wait for the timer.
  505.     PrimeUI.addTask(function()
  506.         while true do
  507.             -- Wait for a timer event.
  508.             local _, tm = os.pullEvent("timer")
  509.             if tm == timer then
  510.                 -- Fire the timer action.
  511.                 local res
  512.                 if type(action) == "string" then PrimeUI.resolve("timeout", action)
  513.                 else res = action() end
  514.                 -- Check the return value and adjust time accordingly.
  515.                 if type(res) == "number" then time = res end
  516.                 -- Set a new timer if not canceled.
  517.                 if res ~= false then timer = os.startTimer(time) end
  518.             end
  519.         end
  520.     end)
  521.     -- Return a function to cancel the timer.
  522.     return function() os.cancelTimer(timer) end
  523. end
  524.  
  525. --- Adds an action to trigger when a key is pressed.
  526. ---@param key key The key to trigger on, from `keys.*`
  527. ---@param action function|string A function to call when clicked, or a string to use as a key for a `run` return event
  528. function PrimeUI.keyAction(key, action)
  529.     expect(1, key, "number")
  530.     expect(2, action, "function", "string")
  531.     PrimeUI.addTask(function()
  532.         while true do
  533.             local _, param1 = os.pullEvent("key") -- wait for key
  534.             if param1 == key then
  535.                 if type(action) == "string" then PrimeUI.resolve("keyAction", action)
  536.                 else action() end
  537.             end
  538.         end
  539.     end)
  540. end
  541.  
  542. --- Draws a line of text at a position.
  543. ---@param win window The window to draw on
  544. ---@param x number The X position of the left side of the text
  545. ---@param y number The Y position of the text
  546. ---@param text string The text to draw
  547. ---@param fgColor color|nil The color of the text (defaults to white)
  548. ---@param bgColor color|nil The color of the background (defaults to black)
  549. function PrimeUI.label(win, x, y, text, fgColor, bgColor)
  550.     expect(1, win, "table")
  551.     expect(2, x, "number")
  552.     expect(3, y, "number")
  553.     expect(4, text, "string")
  554.     fgColor = expect(5, fgColor, "number", "nil") or colors.white
  555.     bgColor = expect(6, bgColor, "number", "nil") or colors.black
  556.     win.setCursorPos(x, y)
  557.     win.setTextColor(fgColor)
  558.     win.setBackgroundColor(bgColor)
  559.     win.write(text)
  560. end
  561.  
  562. --- Creates a progress bar, which can be updated by calling the returned function.
  563. ---@param win window The window to draw on
  564. ---@param x number The X position of the left side of the bar
  565. ---@param y number The Y position of the bar
  566. ---@param width number The width of the bar
  567. ---@param fgColor color|nil The color of the activated part of the bar (defaults to white)
  568. ---@param bgColor color|nil The color of the inactive part of the bar (defaults to black)
  569. ---@param useShade boolean|nil Whether to use shaded areas for the inactive part (defaults to false)
  570. ---@return function redraw A function to call to update the progress of the bar, taking a number from 0.0 to 1.0
  571. function PrimeUI.progressBar(win, x, y, width, fgColor, bgColor, useShade)
  572.     expect(1, win, "table")
  573.     expect(2, x, "number")
  574.     expect(3, y, "number")
  575.     expect(4, width, "number")
  576.     fgColor = expect(5, fgColor, "number", "nil") or colors.white
  577.     bgColor = expect(6, bgColor, "number", "nil") or colors.black
  578.     expect(7, useShade, "boolean", "nil")
  579.     local function redraw(progress)
  580.         expect(1, progress, "number")
  581.         if progress < 0 or progress > 1 then error("bad argument #1 (value out of range)", 2) end
  582.         -- Draw the active part of the bar.
  583.         win.setCursorPos(x, y)
  584.         win.setBackgroundColor(bgColor)
  585.         win.setBackgroundColor(fgColor)
  586.         win.write((" "):rep(math.floor(progress * width)))
  587.         -- Draw the inactive part of the bar, using shade if desired.
  588.         win.setBackgroundColor(bgColor)
  589.         win.setTextColor(fgColor)
  590.         win.write((useShade and "\x7F" or " "):rep(width - math.floor(progress * width)))
  591.     end
  592.     redraw(0)
  593.     return redraw
  594. end
  595.  
  596. --- Creates a scrollable window, which allows drawing large content in a small area.
  597. ---@param win window The parent window of the scroll box
  598. ---@param x number The X position of the box
  599. ---@param y number The Y position of the box
  600. ---@param width number The width of the box
  601. ---@param height number The height of the outer box
  602. ---@param innerHeight number The height of the inner scroll area
  603. ---@param allowArrowKeys boolean|nil Whether to allow arrow keys to scroll the box (defaults to true)
  604. ---@param showScrollIndicators boolean|nil Whether to show arrow indicators on the right side when scrolling is available, which reduces the inner width by 1 (defaults to false)
  605. ---@param fgColor number|nil The color of scroll indicators (defaults to white)
  606. ---@param bgColor color|nil The color of the background (defaults to black)
  607. ---@return window inner The inner window to draw inside
  608. ---@return fun(pos:number) scroll A function to manually set the scroll position of the window
  609. function PrimeUI.scrollBox(win, x, y, width, height, innerHeight, allowArrowKeys, showScrollIndicators, fgColor, bgColor)
  610.     expect(1, win, "table")
  611.     expect(2, x, "number")
  612.     expect(3, y, "number")
  613.     expect(4, width, "number")
  614.     expect(5, height, "number")
  615.     expect(6, innerHeight, "number")
  616.     expect(7, allowArrowKeys, "boolean", "nil")
  617.     expect(8, showScrollIndicators, "boolean", "nil")
  618.     fgColor = expect(9, fgColor, "number", "nil") or colors.white
  619.     bgColor = expect(10, bgColor, "number", "nil") or colors.black
  620.     if allowArrowKeys == nil then allowArrowKeys = true end
  621.     -- Create the outer container box.
  622.     local outer = window.create(win == term and term.current() or win, x, y, width, height)
  623.     outer.setBackgroundColor(bgColor)
  624.     outer.clear()
  625.     -- Create the inner scrolling box.
  626.     local inner = window.create(outer, 1, 1, width - (showScrollIndicators and 1 or 0), innerHeight)
  627.     inner.setBackgroundColor(bgColor)
  628.     inner.clear()
  629.     -- Draw scroll indicators if desired.
  630.     if showScrollIndicators then
  631.         outer.setBackgroundColor(bgColor)
  632.         outer.setTextColor(fgColor)
  633.         outer.setCursorPos(width, height)
  634.         outer.write(innerHeight > height and "\31" or " ")
  635.     end
  636.     -- Get the absolute position of the window.
  637.     x, y = PrimeUI.getWindowPos(win, x, y)
  638.     -- Add the scroll handler.
  639.     local scrollPos = 1
  640.    
  641.     -- Store the original event filter function
  642.     local originalEventFilter = PrimeUI.eventFilter
  643.    
  644.     -- Replace the event filter to adjust mouse coordinates for buttons inside the scroll box
  645.     PrimeUI.eventFilter = function(event, ...)
  646.         if event == "mouse_click" or event == "mouse_up" or event == "mouse_drag" then
  647.             local _, mouseX, mouseY = ...
  648.            
  649.             -- Check if the click is within the scroll box boundaries
  650.             if mouseX >= x and mouseX < x + width and mouseY >= y and mouseY < y + height then
  651.                 -- Adjust the Y coordinate based on scroll position
  652.                 local adjustedY = mouseY + scrollPos - 1
  653.                
  654.                 -- Call the original event filter with adjusted coordinates
  655.                 return originalEventFilter(event, _, mouseX, adjustedY, select(4, ...))
  656.             end
  657.         end
  658.        
  659.         -- For other events, use the original filter
  660.         return originalEventFilter(event, ...)
  661.     end
  662.    
  663.     PrimeUI.addTask(function()
  664.         while true do
  665.             -- Wait for next event.
  666.             local ev = table.pack(os.pullEvent())
  667.             -- Update inner height in case it changed.
  668.             innerHeight = select(2, inner.getSize())
  669.             -- Check for scroll events and set direction.
  670.             local dir
  671.             if ev[1] == "key" and allowArrowKeys then
  672.                 if ev[2] == keys.up then dir = -1
  673.                 elseif ev[2] == keys.down then dir = 1 end
  674.             elseif ev[1] == "mouse_scroll" and ev[3] >= x and ev[3] < x + width and ev[4] >= y and ev[4] < y + height then
  675.                 dir = ev[2]
  676.             end
  677.             -- If there's a scroll event, move the window vertically.
  678.             if dir and (scrollPos + dir >= 1 and scrollPos + dir <= innerHeight - height) then
  679.                 scrollPos = scrollPos + dir
  680.                 inner.reposition(1, 2 - scrollPos)
  681.             end
  682.             -- Redraw scroll indicators if desired.
  683.             if showScrollIndicators then
  684.                 outer.setBackgroundColor(bgColor)
  685.                 outer.setTextColor(fgColor)
  686.                 outer.setCursorPos(width, 1)
  687.                 outer.write(scrollPos > 1 and "\30" or " ")
  688.                 outer.setCursorPos(width, height)
  689.                 outer.write(scrollPos < innerHeight - height and "\31" or " ")
  690.             end
  691.         end
  692.     end)
  693.    
  694.     -- Make a function to allow external scrolling.
  695.     local function scroll(pos)
  696.         expect(1, pos, "number")
  697.         pos = math.floor(pos)
  698.         expect.range(pos, 1, innerHeight - height)
  699.         -- Scroll the window.
  700.         scrollPos = pos
  701.         inner.reposition(1, 2 - scrollPos)
  702.         -- Redraw scroll indicators if desired.
  703.         if showScrollIndicators then
  704.             outer.setBackgroundColor(bgColor)
  705.             outer.setTextColor(fgColor)
  706.             outer.setCursorPos(width, 1)
  707.             outer.write(scrollPos > 1 and "\30" or " ")
  708.             outer.setCursorPos(width, height)
  709.             outer.write(scrollPos < innerHeight - height and "\31" or " ")
  710.         end
  711.     end
  712.    
  713.     -- Add a cleanup task to restore the original event filter when the scroll box is destroyed
  714.     PrimeUI.addTask(function()
  715.         while true do
  716.             local event = os.pullEvent("term_resize")
  717.             -- Check if the outer window still exists
  718.             if not outer.isColor then
  719.                 -- Restore the original event filter
  720.                 PrimeUI.eventFilter = originalEventFilter
  721.                 return
  722.             end
  723.         end
  724.     end)
  725.    
  726.     return inner, scroll
  727. end
  728.  
  729. --- Creates a list of entries that can each be selected.
  730. ---@param win window The window to draw on
  731. ---@param x number The X coordinate of the inside of the box
  732. ---@param y number The Y coordinate of the inside of the box
  733. ---@param width number The width of the inner box
  734. ---@param height number The height of the inner box
  735. ---@param entries string[] A list of entries to show, where the value is whether the item is pre-selected (or `"R"` for required/forced selected)
  736. ---@param action function|string A function or `run` event that's called when a selection is made
  737. ---@param selectChangeAction function|string|nil A function or `run` event that's called when the current selection is changed
  738. ---@param fgColor color|nil The color of the text (defaults to white)
  739. ---@param bgColor color|nil The color of the background (defaults to black)
  740. function PrimeUI.selectionBox(win, x, y, width, height, entries, action, selectChangeAction, fgColor, bgColor)
  741.     expect(1, win, "table")
  742.     expect(2, x, "number")
  743.     expect(3, y, "number")
  744.     expect(4, width, "number")
  745.     expect(5, height, "number")
  746.     expect(6, entries, "table")
  747.     expect(7, action, "function", "string")
  748.     expect(8, selectChangeAction, "function", "string", "nil")
  749.     fgColor = expect(9, fgColor, "number", "nil") or colors.white
  750.     bgColor = expect(10, bgColor, "number", "nil") or colors.black
  751.     -- Check that all entries are strings.
  752.     if #entries == 0 then error("bad argument #6 (table must not be empty)", 2) end
  753.     for i, v in ipairs(entries) do
  754.         if type(v) ~= "string" then error("bad item " .. i .. " in entries table (expected string, got " .. type(v), 2) end
  755.     end
  756.     -- Create container window.
  757.     local entrywin = window.create(win, x, y, width, height)
  758.     local selection, scroll = 1, 1
  759.     -- Create a function to redraw the entries on screen.
  760.     local function drawEntries()
  761.         -- Clear and set invisible for performance.
  762.         entrywin.setVisible(false)
  763.         entrywin.setBackgroundColor(bgColor)
  764.         entrywin.clear()
  765.         -- Draw each entry in the scrolled region.
  766.         for i = scroll, scroll + height - 1 do
  767.             -- Get the entry; stop if there's no more.
  768.             local e = entries[i]
  769.             if not e then break end
  770.             -- Set the colors: invert if selected.
  771.             entrywin.setCursorPos(2, i - scroll + 1)
  772.             if i == selection then
  773.                 entrywin.setBackgroundColor(fgColor)
  774.                 entrywin.setTextColor(bgColor)
  775.             else
  776.                 entrywin.setBackgroundColor(bgColor)
  777.                 entrywin.setTextColor(fgColor)
  778.             end
  779.             -- Draw the selection.
  780.             entrywin.clearLine()
  781.             entrywin.write(#e > width - 1 and e:sub(1, width - 4) .. "..." or e)
  782.         end
  783.         -- Draw scroll arrows.
  784.         entrywin.setBackgroundColor(bgColor)
  785.         entrywin.setTextColor(fgColor)
  786.         entrywin.setCursorPos(width, 1)
  787.         entrywin.write("\30")
  788.         entrywin.setCursorPos(width, height)
  789.         entrywin.write("\31")
  790.         -- Send updates to the screen.
  791.         entrywin.setVisible(true)
  792.     end
  793.     -- Draw first screen.
  794.     drawEntries()
  795.     -- Add a task for selection keys.
  796.     PrimeUI.addTask(function()
  797.         while true do
  798.             local event, key, cx, cy = os.pullEvent()
  799.             if event == "key" then
  800.                 if key == keys.down and selection < #entries then
  801.                     -- Move selection down.
  802.                     selection = selection + 1
  803.                     if selection > scroll + height - 1 then scroll = scroll + 1 end
  804.                     -- Send action if necessary.
  805.                     if type(selectChangeAction) == "string" then PrimeUI.resolve("selectionBox", selectChangeAction, selection)
  806.                     elseif selectChangeAction then selectChangeAction(selection) end
  807.                     -- Redraw screen.
  808.                     drawEntries()
  809.                 elseif key == keys.up and selection > 1 then
  810.                     -- Move selection up.
  811.                     selection = selection - 1
  812.                     if selection < scroll then scroll = scroll - 1 end
  813.                     -- Send action if necessary.
  814.                     if type(selectChangeAction) == "string" then PrimeUI.resolve("selectionBox", selectChangeAction, selection)
  815.                     elseif selectChangeAction then selectChangeAction(selection) end
  816.                     -- Redraw screen.
  817.                     drawEntries()
  818.                 elseif key == keys.enter then
  819.                     -- Select the entry: send the action.
  820.                     if type(action) == "string" then PrimeUI.resolve("selectionBox", action, entries[selection])
  821.                     else action(entries[selection]) end
  822.                 end
  823.             elseif event == "mouse_click" and key == 1 then
  824.                 -- Handle clicking the scroll arrows.
  825.                 local wx, wy = PrimeUI.getWindowPos(entrywin, 1, 1)
  826.                 if cx == wx + width - 1 then
  827.                     if cy == wy and selection > 1 then
  828.                         -- Move selection up.
  829.                         selection = selection - 1
  830.                         if selection < scroll then scroll = scroll - 1 end
  831.                         -- Send action if necessary.
  832.                         if type(selectChangeAction) == "string" then PrimeUI.resolve("selectionBox", selectChangeAction, selection)
  833.                         elseif selectChangeAction then selectChangeAction(selection) end
  834.                         -- Redraw screen.
  835.                         drawEntries()
  836.                     elseif cy == wy + height - 1 and selection < #entries then
  837.                         -- Move selection down.
  838.                         selection = selection + 1
  839.                         if selection > scroll + height - 1 then scroll = scroll + 1 end
  840.                         -- Send action if necessary.
  841.                         if type(selectChangeAction) == "string" then PrimeUI.resolve("selectionBox", selectChangeAction, selection)
  842.                         elseif selectChangeAction then selectChangeAction(selection) end
  843.                         -- Redraw screen.
  844.                         drawEntries()
  845.                     end
  846.                 elseif cx >= wx and cx < wx + width - 1 and cy >= wy and cy < wy + height then
  847.                     local sel = scroll + (cy - wy)
  848.                     if sel == selection then
  849.                         -- Select the entry: send the action.
  850.                         if type(action) == "string" then PrimeUI.resolve("selectionBox", action, entries[selection])
  851.                         else action(entries[selection]) end
  852.                     else
  853.                         selection = sel
  854.                         -- Send action if necessary.
  855.                         if type(selectChangeAction) == "string" then PrimeUI.resolve("selectionBox", selectChangeAction, selection)
  856.                         elseif selectChangeAction then selectChangeAction(selection) end
  857.                         -- Redraw screen.
  858.                         drawEntries()
  859.                     end
  860.                 end
  861.             elseif event == "mouse_scroll" then
  862.                 -- Handle mouse scrolling.
  863.                 local wx, wy = PrimeUI.getWindowPos(entrywin, 1, 1)
  864.                 if cx >= wx and cx < wx + width and cy >= wy and cy < wy + height then
  865.                     if key < 0 and selection > 1 then
  866.                         -- Move selection up.
  867.                         selection = selection - 1
  868.                         if selection < scroll then scroll = scroll - 1 end
  869.                         -- Send action if necessary.
  870.                         if type(selectChangeAction) == "string" then PrimeUI.resolve("selectionBox", selectChangeAction, selection)
  871.                         elseif selectChangeAction then selectChangeAction(selection) end
  872.                         -- Redraw screen.
  873.                         drawEntries()
  874.                     elseif key > 0 and selection < #entries then
  875.                         -- Move selection down.
  876.                         selection = selection + 1
  877.                         if selection > scroll + height - 1 then scroll = scroll + 1 end
  878.                         -- Send action if necessary.
  879.                         if type(selectChangeAction) == "string" then PrimeUI.resolve("selectionBox", selectChangeAction, selection)
  880.                         elseif selectChangeAction then selectChangeAction(selection) end
  881.                         -- Redraw screen.
  882.                         drawEntries()
  883.                     end
  884.                 end
  885.             end
  886.         end
  887.     end)
  888. end
  889.  
  890. --- Creates a text box that wraps text and can have its text modified later.
  891. ---@param win window The parent window of the text box
  892. ---@param x number The X position of the box
  893. ---@param y number The Y position of the box
  894. ---@param width number The width of the box
  895. ---@param height number The height of the box
  896. ---@param text string The initial text to draw
  897. ---@param fgColor color|nil The color of the text (defaults to white)
  898. ---@param bgColor color|nil The color of the background (defaults to black)
  899. ---@return function redraw A function to redraw the window with new contents
  900. function PrimeUI.textBox(win, x, y, width, height, text, fgColor, bgColor)
  901.     expect(1, win, "table")
  902.     expect(2, x, "number")
  903.     expect(3, y, "number")
  904.     expect(4, width, "number")
  905.     expect(5, height, "number")
  906.     expect(6, text, "string")
  907.     fgColor = expect(7, fgColor, "number", "nil") or colors.white
  908.     bgColor = expect(8, bgColor, "number", "nil") or colors.black
  909.     -- Create the box window.
  910.     local box = window.create(win, x, y, width, height)
  911.     -- Override box.getSize to make print not scroll.
  912.     function box.getSize()
  913.         return width, math.huge
  914.     end
  915.     -- Define a function to redraw with.
  916.     local function redraw(_text)
  917.         expect(1, _text, "string")
  918.         -- Set window parameters.
  919.         box.setBackgroundColor(bgColor)
  920.         box.setTextColor(fgColor)
  921.         box.clear()
  922.         box.setCursorPos(1, 1)
  923.         -- Redirect and draw with `print`.
  924.         local old = term.redirect(box)
  925.         print(_text)
  926.         term.redirect(old)
  927.     end
  928.     redraw(text)
  929.     return redraw
  930. end
  931.  
  932. --- Creates a clickable, toggleable button on screen with text.
  933. ---@param win window The window to draw on
  934. ---@param x number The X position of the button
  935. ---@param y number The Y position of the button
  936. ---@param textOn string The text to draw on the button when on
  937. ---@param textOff string The text to draw on the button when off (must be the same length as textOn)
  938. ---@param action function|string A function to call when clicked, or a string to send with a `run` event
  939. ---@param fgColor color|nil The color of the button text (defaults to white)
  940. ---@param bgColor color|nil The color of the button (defaults to light gray)
  941. ---@param clickedColor color|nil The color of the button when clicked (defaults to gray)
  942. ---@param periphName string|nil The name of the monitor peripheral, or nil (set if you're using a monitor - events will be filtered to that monitor)
  943. function PrimeUI.toggleButton(win, x, y, textOn, textOff, action, fgColor, bgColor, clickedColor, periphName)
  944.     expect(1, win, "table")
  945.     expect(1, win, "table")
  946.     expect(2, x, "number")
  947.     expect(3, y, "number")
  948.     expect(4, textOn, "string")
  949.     expect(5, textOff, "string")
  950.     if #textOn ~= #textOff then error("On and off text must be the same length", 2) end
  951.     expect(6, action, "function", "string")
  952.     fgColor = expect(7, fgColor, "number", "nil") or colors.white
  953.     bgColor = expect(8, bgColor, "number", "nil") or colors.gray
  954.     clickedColor = expect(9, clickedColor, "number", "nil") or colors.lightGray
  955.     periphName = expect(10, periphName, "string", "nil")
  956.     -- Draw the initial button.
  957.     win.setCursorPos(x, y)
  958.     win.setBackgroundColor(bgColor)
  959.     win.setTextColor(fgColor)
  960.     win.write(" " .. textOff .. " ")
  961.     local state = false
  962.     -- Get the screen position and add a click handler.
  963.     PrimeUI.addTask(function()
  964.         local screenX, screenY = PrimeUI.getWindowPos(win, x, y)
  965.         local buttonDown = false
  966.         while true do
  967.             local event, button, clickX, clickY = os.pullEvent()
  968.             if event == "mouse_click" and periphName == nil and button == 1 and clickX >= screenX and clickX < screenX + #textOn + 2 and clickY == screenY then
  969.                 -- Initiate a click action (but don't trigger until mouse up).
  970.                 buttonDown = true
  971.                 -- Redraw the button with the clicked background color.
  972.                 win.setCursorPos(x, y)
  973.                 win.setBackgroundColor(clickedColor)
  974.                 win.setTextColor(fgColor)
  975.                 win.write(" " .. (state and textOn or textOff) .. " ")
  976.             elseif (event == "monitor_touch" and periphName == button and clickX >= screenX and clickX < screenX + #textOn + 2 and clickY == screenY)
  977.                 or (event == "mouse_up" and button == 1 and buttonDown) then
  978.                 -- Finish a click event.
  979.                 state = not state
  980.                 if clickX >= screenX and clickX < screenX + #textOn + 2 and clickY == screenY then
  981.                     -- Trigger the action.
  982.                     if type(action) == "string" then
  983.                         PrimeUI.resolve("toggleButton", action, state)
  984.                     else
  985.                         action(state)
  986.                     end
  987.                 end
  988.                 -- Redraw the original button state.
  989.                 win.setCursorPos(x, y)
  990.                 win.setBackgroundColor(bgColor)
  991.                 win.setTextColor(fgColor)
  992.                 win.write(" " .. (state and textOn or textOff) .. " ")
  993.             end
  994.         end
  995.     end)
  996. end
  997.  
  998. --- Draws a vertical line at a position with the specified height.
  999. ---@param win window The window to draw on
  1000. ---@param x number The X position of the line
  1001. ---@param y number The Y position of the top of the line
  1002. ---@param height number The height of the line
  1003. ---@param right boolean|nil Whether to align the line to the right instead of the left (defaults to false)
  1004. ---@param fgColor color|nil The color of the line (defaults to white)
  1005. ---@param bgColor color|nil The color of the background (defaults to black)
  1006. function PrimeUI.verticalLine(win, x, y, height, right, fgColor, bgColor)
  1007.     expect(1, win, "table")
  1008.     expect(2, x, "number")
  1009.     expect(3, y, "number")
  1010.     expect(4, height, "number")
  1011.     right = expect(5, right, "boolean", "nil") or false
  1012.     fgColor = expect(6, fgColor, "number", "nil") or colors.white
  1013.     bgColor = expect(7, bgColor, "number", "nil") or colors.black
  1014.     -- Use drawing characters to draw a thin line.
  1015.     win.setTextColor(right and bgColor or fgColor)
  1016.     win.setBackgroundColor(right and fgColor or bgColor)
  1017.     for j = 1, height do
  1018.         win.setCursorPos(x, y + j - 1)
  1019.         win.write("\x95")
  1020.     end
  1021. end
  1022. -- local ui = require "primeui"
  1023. ui = PrimeUI
  1024.  
  1025. -- Transliterator | made by timuzkas
  1026. local Transliteration = {}
  1027. Transliteration.__index = Transliteration
  1028.  
  1029. local cyrillicAlphabet = {
  1030.   {"А", "а", "A", "a"}, {"Б", "б", "B", "b"}, {"В", "в", "V", "v"}, {"Г", "г", "G", "g"}, {"Д", "д", "D", "d"},
  1031.   {"Е", "е", "E", "e"}, {"Ё", "ё", "YO", "yo"}, {"Ж", "ж", "ZH", "zh"}, {"З", "з", "Z", "z"}, {"И", "и", "I", "i"},
  1032.   {"Й", "й", "Y", "y"}, {"К", "к", "K", "k"}, {"Л", "л", "L", "l"}, {"М", "м", "M", "m"}, {"Н", "н", "N", "n"},
  1033.   {"О", "о", "O", "o"}, {"П", "п", "P", "p"}, {"Р", "р", "R", "r"}, {"С", "с", "S", "s"}, {"Т", "т", "T", "t"},
  1034.   {"У", "у", "U", "u"}, {"Ф", "ф", "F", "f"}, {"Х", "х", "KH", "kh"}, {"Ц", "ц", "TS", "ts"}, {"Ч", "ч", "CH", "ch"},
  1035.   {"Ш", "ш", "SH", "sh"}, {"Щ", "щ", "SHCH", "shch"}, {"Ъ", "ъ", "", ""}, {"Ы", "ы", "Y", "y"}, {"Ь", "ь", "", ""},
  1036.   {"Э", "э", "E", "e"}, {"Ю", "ю", "YU", "yu"}, {"Я", "я", "YA", "ya"}
  1037. }
  1038.  
  1039. function Transliteration.new()
  1040.   local self = setmetatable({}, Transliteration)
  1041.   self.cyrillicToLatin = {}
  1042.   self.latinToCyrillic = {}
  1043.   self.isSetup = false
  1044.   return self
  1045. end
  1046.  
  1047. function Transliteration:setup()
  1048.   if self.isSetup then return end
  1049.   for _, pair in ipairs(cyrillicAlphabet) do
  1050.     self.cyrillicToLatin[utf8.codepoint(pair[1])] = pair[3]
  1051.     self.cyrillicToLatin[utf8.codepoint(pair[2])] = pair[4]
  1052.     if pair[3] ~= "" then
  1053.       if not self.latinToCyrillic[pair[3]] then
  1054.         self.latinToCyrillic[pair[3]] = {}
  1055.       end
  1056.       table.insert(self.latinToCyrillic[pair[3]], pair[1])
  1057.       table.insert(self.latinToCyrillic[pair[3]], pair[2])
  1058.     end
  1059.     if pair[4] ~= "" then
  1060.       if not self.latinToCyrillic[pair[4]] then
  1061.         self.latinToCyrillic[pair[4]] = {}
  1062.       end
  1063.       table.insert(self.latinToCyrillic[pair[4]], pair[1])
  1064.       table.insert(self.latinToCyrillic[pair[4]], pair[2])
  1065.     end
  1066.   end
  1067.   self.isSetup = true
  1068. end
  1069.  
  1070. function Transliteration:translate(str)
  1071.   if not self.isSetup then self:setup() end
  1072.   local result = ""
  1073.  
  1074.   local chars = {}
  1075.   for char in str:gmatch(utf8.charpattern) do
  1076.     table.insert(chars, char)
  1077.   end
  1078.  
  1079.   for _, char in ipairs(chars) do
  1080.     local codepoint = utf8.codepoint(char)
  1081.     local latin = self.cyrillicToLatin[codepoint]
  1082.     if latin then
  1083.       result = result .. latin
  1084.     else
  1085.       result = result .. char
  1086.     end
  1087.   end
  1088.  
  1089.   return result
  1090. end
  1091.  
  1092. -- stripped for size reasons
  1093.  
  1094. local function box(terminal, x, y, width, height, color, cornerStyle)
  1095.     cornerStyle = cornerStyle or "square"
  1096.     terminal.setBackgroundColor(color)
  1097.    
  1098.     if cornerStyle == "square" then
  1099.         for i = y, y + height - 1 do
  1100.             terminal.setCursorPos(x, i)
  1101.             terminal.write(string.rep(" ", width))
  1102.         end
  1103.     elseif cornerStyle == "round" then
  1104.         terminal.setCursorPos(x + 1, y)
  1105.         terminal.write(string.rep(" ", width - 2))
  1106.        
  1107.         for i = y + 1, y + height - 2 do
  1108.             terminal.setCursorPos(x, i)
  1109.             terminal.write(string.rep(" ", width))
  1110.         end
  1111.        
  1112.         terminal.setCursorPos(x + 1, y + height - 1)
  1113.         terminal.write(string.rep(" ", width - 2))
  1114.     end
  1115. end
  1116. ui.box = box    
  1117.  
  1118. local api_base_url = "https://ipod-2to6magyna-uc.a.run.app/"
  1119.  
  1120. local width, height = term.getSize()
  1121.  
  1122.  
  1123. local last_search_url = nil
  1124. local search_results = nil
  1125. local playing = false
  1126. local queue = {}
  1127. local now_playing = nil
  1128. local looping = false
  1129.  
  1130. local playing_id = nil
  1131. local last_download_url = nil
  1132. local playing_status = 0
  1133.  
  1134. local player_handle = nil
  1135. local start = nil
  1136. local pcm = nil
  1137. local size = nil
  1138. local decoder = nil
  1139. local needs_next_chunk = 0
  1140. local buffer
  1141.  
  1142. local speakers = { peripheral.find("speaker") }
  1143.  
  1144. if #speakers == 0 then
  1145.     error("No speakers attached. You need to connect a speaker to this computer. If this is an Advanced Noisy Pocket Computer, then this is a bug, and you should try restarting your Minecraft game.", 0)
  1146. end
  1147.  
  1148. local speaker = speakers[1]
  1149.  
  1150. os.startTimer(1)
  1151.  
  1152. -- ui helper functions
  1153. local function playSong(song)
  1154.     now_playing = song
  1155.     playing = true
  1156.     playing_id = nil
  1157. end
  1158.  
  1159. local function stopPlayback()
  1160.     playing = false
  1161.     speaker.stop()
  1162.     playing_id = nil
  1163. end
  1164.  
  1165. local function togglePlayPause()
  1166.     if playing then
  1167.         stopPlayback()
  1168.     else
  1169.         if now_playing or #queue > 0 then
  1170.             playSong(now_playing or queue[1])
  1171.         end
  1172.     end
  1173. end
  1174.  
  1175. local function skipSong()
  1176.     if #queue > 0 then
  1177.         now_playing = queue[1]
  1178.         table.remove(queue, 1)
  1179.         playing_id = nil
  1180.     else
  1181.         now_playing = nil
  1182.         playing = false
  1183.     end
  1184. end
  1185.  
  1186. local function toggleLoop()
  1187.     looping = not looping
  1188. end
  1189.  
  1190. local function addToQueue(song, position)
  1191.     if position then
  1192.         table.insert(queue, position, song)
  1193.     else
  1194.         table.insert(queue, song)
  1195.     end
  1196. end
  1197.  
  1198. local function removeFromQueue(position)
  1199.     if position and position <= #queue then
  1200.         table.remove(queue, position)
  1201.     end
  1202. end
  1203.  
  1204. local function clearQueue()
  1205.     queue = {}
  1206. end
  1207.  
  1208. local function searchMusic(query)
  1209.     last_search = query
  1210.     last_search_url = api_base_url .. "?search=" .. textutils.urlEncode(query)
  1211.     http.request(last_search_url)
  1212.     search_results = nil
  1213.     search_error = false
  1214. end
  1215.  
  1216. -- not used, tho may need later
  1217. local function handleAudioStream(response)
  1218.     player_handle = response
  1219.     start = response.read(4)
  1220.     size = 16 * 1024 - 4
  1221.     playing_status = 1
  1222.     decoder = require "cc.audio.dfpwm".make_decoder()
  1223. end
  1224.  
  1225. -- custom pallete based on spotify one.
  1226. local original_palette = {}
  1227. local function initCustomPallete()
  1228.     for i=1, 16 do
  1229.         original_palette[i] = term.getPaletteColor(i)
  1230.     end
  1231.  
  1232.     term.setPaletteColor(colors.green, 0x1ED760)
  1233.     term.setPaletteColor(colors.lightGray, 0xb3b3b3)
  1234.     term.setPaletteColor(colors.gray, 0x212121)
  1235.     term.setPaletteColor(colors.purple, 0x457e59)
  1236.     term.setPaletteColor(colors.magenta, 0x62d089)
  1237.     term.setPaletteColor(colors.brown, 0x2e2e2e)
  1238. end
  1239.  
  1240. -- truncation and transliteration for text
  1241. local function fixString(str, limit)
  1242.     if not str then return "" end
  1243.     --local transliterator = Transliteration.new()
  1244.     --str = transliterator.translate(transliterator, str)
  1245.    
  1246.     if #str <= limit then
  1247.         return str
  1248.     end
  1249.    
  1250.     return string.sub(str, 1, limit - 3) .. "..."
  1251. end
  1252.  
  1253.  
  1254. -- UI LOOP
  1255.  
  1256. ui.page = 1
  1257.  
  1258. local function redrawScreen()
  1259.     -- init custom palette
  1260.     initCustomPallete()
  1261.  
  1262.  
  1263.     while true do
  1264.         ui.clear()
  1265.         ui.borderBox(term.current(), 3, 2, width-4, 1, colors.gray)
  1266.  
  1267.         local isSmallScreen = width <= 30
  1268.        
  1269.  
  1270.         if now_playing then
  1271.             if playing then
  1272.                 if isSmallScreen then
  1273.                     ui.button(term.current(), 4, 2, "S", "stop", colors.white, colors.red, colors.orange)
  1274.                 else
  1275.                     ui.button(term.current(), 4, 2, "Stop", "stop", colors.white, colors.red, colors.orange)
  1276.                 end
  1277.             else
  1278.                 ui.button(term.current(), 4, 2, "\16", "pause", colors.white, colors.green, colors.lightGray)
  1279.                 ui.button(term.current(), 8, 2, "R", "clear", colors.white, colors.red, colors.orange)
  1280.             end
  1281.             if not isSmallScreen then
  1282.                 ui.label(term.current(), 12, 2, fixString(now_playing.name, 20), colors.white)
  1283.                 ui.label(term.current(), 12+string.len(fixString(now_playing.name, 20))+1, 2, "| "..fixString(now_playing.artist, 14), colors.lightGray)
  1284.             else
  1285.                 ui.label(term.current(), 8, 2, fixString(now_playing.name, 16), colors.white)
  1286.             end
  1287.         else
  1288.             ui.label(term.current(), 4, 2, "Musiclo", colors.green)
  1289.             if not isSmallScreen then
  1290.                 ui.label(term.current(), 4+string.len("Musiclo")+2, 2, "| CC:T music player made easy", colors.lightGray)
  1291.             else
  1292.                 ui.label(term.current(), 4+string.len("Musiclo")+1, 2, "| CC:T player", colors.lightGray)
  1293.             end
  1294.         end
  1295.  
  1296.  
  1297.         local titleTruncateLimit = 41
  1298.         local artistTruncateLimit = 26
  1299.  
  1300.         if isSmallScreen then
  1301.             titleTruncateLimit = 19
  1302.             artistTruncateLimit = 15
  1303.         end
  1304.  
  1305.        
  1306.  
  1307.         if ui.page == 1 then
  1308.             ui.borderBox(term.current(), 3, 5, width-4, height-6, colors.gray)
  1309.  
  1310.             ui.button(term.current(), width-9, 4, "Search", "page.2", colors.white, colors.magenta, colors.purple)
  1311.             ui.keyAction(keys.enter, "page.2")
  1312.            
  1313.             ui.label(term.current(), 4, 4, "Queue", colors.white)
  1314.            
  1315.             ui.keyAction(keys.space, "pause")
  1316.  
  1317.             if looping then
  1318.                 ui.button(term.current(), 4, height-1, "Loop", "loop", colors.white, colors.magenta, colors.purple)
  1319.             else
  1320.                 ui.button(term.current(), 4, height-1, "Loop", "loop", colors.white, colors.gray, colors.lightGray)
  1321.             end
  1322.  
  1323.             if #queue > 0 then
  1324.                 ui.button(term.current(), 11, height-1, "Skip", "skip", colors.white, colors.gray, colors.lightGray)
  1325.                 if isSmallScreen then
  1326.                     ui.button(term.current(), 18, height-1, "Clr", "clear.q", colors.white, colors.red, colors.orange)
  1327.                 else
  1328.                 ui.button(term.current(), 18, height-1, "Clear queue", "clear.q", colors.white, colors.red, colors.orange)
  1329.                 end
  1330.             end
  1331.             ui.label(term.current(), 4, 6, "Now playing", colors.white)
  1332.  
  1333.             local scroller = ui.scrollBox(term.current(), 3, 5, width-4, height-6, 9000, true, true)
  1334.  
  1335.             y = 2
  1336.             if #queue > 0 then
  1337.                 for i, song in ipairs(queue) do
  1338.                     ui.box(scroller, 1, y, width-5, 5, colors.brown)
  1339.                     ui.label(scroller, 2, y+1, fixString(song.name, titleTruncateLimit), colors.white, colors.brown)
  1340.                     ui.label(scroller, 2, y+2, fixString(song.artist, artistTruncateLimit), colors.lightGray, colors.brown)
  1341.                     if isSmallScreen then y = y + 1 end
  1342.                     ui.button(scroller, width-20, y+2, "Play", "play."..i, colors.white, colors.magenta, colors.purple)
  1343.                     local songInQueue = false
  1344.                     for _, queuedSong in ipairs(queue) do
  1345.                         if queuedSong.id == song.id then
  1346.                             songInQueue = true
  1347.                             break
  1348.                         end
  1349.                     end
  1350.                     if songInQueue then
  1351.                         ui.button(scroller, width-13, y+2, "Remove", "rem."..i, colors.white, colors.red, colors.orange)
  1352.                     else
  1353.                         ui.button(scroller, width-13, y+2, "Add", "add."..i, colors.white, colors.gray, colors.lightGray)
  1354.                     end
  1355.                     y = y + 6
  1356.                 end
  1357.             else
  1358.                 ui.centerLabel(scroller, 1, 5,width-4, "No songs in queue",  colors.lightGray)
  1359.                 ui.button(scroller, ((width-4-3)/2-(string.len("Add song")/2))+1, 7 ,"Add song", "page.2", colors.white, colors.gray, colors.lightGray)
  1360.             end
  1361.         elseif ui.page == 2 then
  1362.             ui.borderBox(term.current(), 3, 5, width-4, height-6, colors.gray)
  1363.  
  1364.             ui.button(term.current(), width-10, 4, "Go back","page.1", colors.white, colors.gray, colors.lightGray)
  1365.             ui.label(term.current(), 4, 4, "Search", colors.white)
  1366.  
  1367.             ui.label(term.current(), 4, 6, "Search on Youtube...", colors.lightGray)
  1368.  
  1369.             ui.horizontalLine(term.current(), 3, 8, width-4, colors.gray)
  1370.            
  1371.             local scroller = ui.scrollBox(term.current(), 3, 9, width-4, height-10, 9000, true, true)
  1372.  
  1373.             y = 2
  1374.             if search_results then
  1375.                 for i, song in ipairs(search_results) do
  1376.                     ui.box(scroller, 1, y, width-6, 5, colors.brown)
  1377.                     ui.label(scroller, 2, y+1, fixString(song.name, titleTruncateLimit), colors.white, colors.brown)
  1378.                     ui.label(scroller, 2, y+2, fixString(song.artist, artistTruncateLimit), colors.lightGray, colors.brown)
  1379.                     if isSmallScreen then y = y + 1 end
  1380.                     ui.button(scroller, width-21, y+2, "Play", "play."..i, colors.white, colors.magenta, colors.purple)
  1381.                     local songInQueue = false
  1382.                     for _, queuedSong in ipairs(queue) do
  1383.                         if queuedSong.id == song.id then
  1384.                             songInQueue = true
  1385.                             break
  1386.                         end
  1387.                     end
  1388.                     if songInQueue then
  1389.                         ui.button(scroller, width-14, y+2, "Remove", "rem."..i, colors.white, colors.red, colors.orange)
  1390.                     else
  1391.                         ui.button(scroller, width-14, y+2, "Add", "add."..i, colors.white, colors.gray, colors.lightGray)
  1392.                     end
  1393.                     y = y + 6
  1394.                 end
  1395.             end
  1396.  
  1397.             ui.inputBox(term.current(), 4, 7, width-7, "search", colors.white, colors.gray)
  1398.         end
  1399.  
  1400.        
  1401.         local object, callback, text = ui.run()
  1402.         term.clear()
  1403.         term.setCursorPos(1, 1)
  1404.  
  1405.         -- callbacks
  1406.  
  1407.         if object == "button" then
  1408.             if callback == "page.2" then
  1409.                 ui.page = 2
  1410.             elseif callback == "page.1" then
  1411.                 ui.page = 1
  1412.             elseif callback:sub(1, 4) == "play" then
  1413.                 local index = tonumber(callback:sub(6))
  1414.                 if index and search_results[index] then
  1415.                     playSong(search_results[index])
  1416.                     ui.page = 1
  1417.                 end
  1418.             elseif callback:sub(1, 3) == "add" then
  1419.                 local index = tonumber(callback:sub(5))
  1420.                 if index and search_results[index] then
  1421.                     addToQueue(search_results[index])
  1422.                 end
  1423.             elseif callback:sub(1, 4) == "rem" then
  1424.                 local index = tonumber(callback:sub(6))
  1425.                 if index and search_results[index] then
  1426.                     removeFromQueue(index)
  1427.                 end
  1428.             elseif callback == "stop" then
  1429.                 stopPlayback()
  1430.             elseif callback == "pause" then
  1431.                 togglePlayPause()
  1432.             elseif callback == "loop" then
  1433.                 toggleLoop()
  1434.             elseif callback == "skip" then
  1435.                 skipSong()
  1436.             elseif callback == "clear.q" then
  1437.                 clearQueue()
  1438.             elseif callback == "clear" then
  1439.                 playing = false
  1440.                 now_playing = nil
  1441.                 playing_id = nil
  1442.             end
  1443.         elseif object == "keyAction" then
  1444.             if callback == "page.2" then
  1445.                 ui.page = 2
  1446.             elseif callback == "page.1" then
  1447.                 ui.page = 1
  1448.             end
  1449.         elseif object == "inputBox" and callback == "search" then
  1450.             if text ~= "" then
  1451.                 searchMusic(text)
  1452.                 term.clear()
  1453.                 local sx, sy = term.getSize()
  1454.                 term.setTextColor(colors.lightGray)
  1455.                 term.setCursorPos(sx/2 - #"Fetching..."/2, sy/2)
  1456.                 term.write("Fetching...")
  1457.                 ui.searchDone = false
  1458.                 repeat
  1459.                     sleep(0.1)
  1460.                 until ui.searchDone == true
  1461.                 ui.searchDone = false
  1462.             end
  1463.         elseif object == "rerender" then
  1464.             print("rerender")
  1465.         else
  1466.             term.clear()
  1467.             term.setCursorPos(1, 1)
  1468.             error("["..(object or "No object").."] "..(callback or "No callback").." "..(text or "No text").." not handled! Exiting",0)
  1469.         end
  1470.     end
  1471. end
  1472.  
  1473. local function audioLoop()
  1474.     while true do
  1475.         -- AUDIO
  1476.         sleep(0.1)
  1477.         if playing and now_playing then
  1478.             if playing_id ~= now_playing.id then
  1479.                 playing_id = now_playing.id
  1480.                 last_download_url = api_base_url .. "?v=2&id=" .. textutils.urlEncode(playing_id)
  1481.                 playing_status = 0
  1482.                 needs_next_chunk = 1
  1483.  
  1484.                 http.request({url = last_download_url, binary = true})
  1485.                 is_loading = true
  1486.  
  1487.             end
  1488.             if playing_status == 1 and needs_next_chunk == 3 then
  1489.                 needs_next_chunk = 1
  1490.                 for _, speaker in ipairs(speakers) do
  1491.                     while not speaker.playAudio(buffer) do
  1492.                         needs_next_chunk = 2
  1493.                         break
  1494.                     end
  1495.                 end
  1496.             end
  1497.             if playing_status == 1 and needs_next_chunk == 1 then
  1498.  
  1499.                 while true do
  1500.                     local chunk = player_handle.read(size)
  1501.                     if not chunk then
  1502.                         if looping then
  1503.                             playing_id = nil
  1504.                         else
  1505.                             if #queue > 0 then
  1506.                                 now_playing = queue[1]
  1507.                                 table.remove(queue, 1)
  1508.                                 playing_id = nil
  1509.                             else
  1510.                                 now_playing = nil
  1511.                                 playing = false
  1512.                                 playing_id = nil
  1513.                                 is_loading = false
  1514.                                 is_error = false
  1515.                             end
  1516.                         end
  1517.  
  1518.  
  1519.                         player_handle.close()
  1520.                         needs_next_chunk = 0
  1521.                         break
  1522.                     else
  1523.                         if start then
  1524.                             chunk, start = start .. chunk, nil
  1525.                             size = size + 4
  1526.                         end
  1527.                
  1528.                         buffer = decoder(chunk)
  1529.                         for _, speaker in ipairs(speakers) do
  1530.                             while not speaker.playAudio(buffer) do
  1531.                                 needs_next_chunk = 2
  1532.                                 break
  1533.                             end
  1534.                         end
  1535.                         if needs_next_chunk == 2 then
  1536.                             break
  1537.                         end
  1538.                     end
  1539.                 end
  1540.  
  1541.             end
  1542.         end
  1543.     end
  1544. end
  1545.  
  1546. -- Events
  1547. local function eventLoop()
  1548.     while true do
  1549.         local event, param1, param2 = os.pullEvent()
  1550.  
  1551.         if event == "timer" then
  1552.             os.startTimer(1)
  1553.         end
  1554.  
  1555.         if event == "speaker_audio_empty" then
  1556.             if needs_next_chunk == 2 then
  1557.                 needs_next_chunk = 3
  1558.             end
  1559.         end
  1560.  
  1561.         if event == "http_success" then
  1562.             local url = param1
  1563.             local handle = param2
  1564.             if url == last_search_url then
  1565.                 search_results = textutils.unserialiseJSON(handle.readAll())
  1566.                 table.remove(search_results, 1)
  1567.                 ui.searchDone = true
  1568.             end
  1569.             if url == last_download_url then
  1570.                 player_handle = handle
  1571.                 start = handle.read(4)
  1572.                 size = 16 * 1024 - 4
  1573.                 if start == "RIFF" then
  1574.                     error("WAV not supported!")
  1575.                 end
  1576.                 playing_status = 1
  1577.                 decoder = require "cc.audio.dfpwm".make_decoder()
  1578.             end
  1579.         end
  1580.  
  1581.         if event == "http_failure" then
  1582.             local url = param1
  1583.  
  1584.             if url == last_search_url then
  1585.                 search_error = true
  1586.             end
  1587.             if url == last_download_url then
  1588.                 if #queue > 0 then
  1589.                     now_playing = queue[1]
  1590.                     table.remove(queue, 1)
  1591.                     playing_id = nil
  1592.                 else
  1593.                     now_playing = nil
  1594.                     playing = false
  1595.                     playing_id = nil
  1596.                 end
  1597.             end
  1598.         end
  1599.     end
  1600. end
  1601.  
  1602. parallel.waitForAny(audioLoop, eventLoop, redrawScreen)
  1603.  
  1604. -- cleanup
  1605. for  i=1, 16 do
  1606.     term.setPaletteColor(i, original_palette[i])
  1607. end
  1608. term.setCursorBlink(false)
  1609. term.clear()
  1610. term.setCursorPos(1, 1)
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement