local SS13 = require("SS13") -- Change this to your ckey so that this works. Don't run it with my ckey please :) local admin = "homek" -- Admins that can also take administrative action, though they can't freely upload videos that bypass the time limit. local trustedAdmins = { [admin] = true, } -- The auth token. You'll need to update this every time you run the script because the python script generates a new one each time it runs for security purposes. authToken = "ag5TwYCgbVGLvEJBiUgTg" -- Whether users can submit requests or not. local acceptingRequests = true -- Whether it's one request per user until their video is played local onePerUser = false -- The size of the TV. Current available options are 1, 2, 4, 8, 16 local scale = 8 -- Channel to play on. Don't modify if you don't know what you're doing local channel = 1023 -- Whether to auto accept requests or not local autoAccept = true -- Number of people required to vote skip local voteSkipRequired = 5 -- Whether the 'admin' should be able to bypass video length limit local bypassVidLength = true -- Whether the TV range is infinite or not. Keep it off if you don't want people nowhere near the TV to lag when videos load. -- Useful if you plan on curating or limiting the videos that will be played so that no matter local infiniteRange = false -- Set to a value if you'd like to force the FPS of the video. Useful if the video itself is not important local forcedFps = nil local voteSkipData = { voteSkip = 0, voteSkipVoters = {} } local AUDIO_DIRECTIONAL = "Directional" local AUDIO_MONO = "Mono" local blockPlayerRequest = {} local me = dm.global_vars:get_var("GLOB"):get_var("directory"):get(admin) local spawnLocation = me:get_var("mob"):get_var("loc") -- hehe trustedAdmins["waltermeldron"] = true local function wget(url, body, headers, outfile) local request = SS13.new("/datum/http_request") request:call_proc("prepare", "get", url, body or "", headers, outfile) request:call_proc("begin_async") while request:call_proc("is_complete") == 0 do sleep() end SS13.stop_tracking(request) local response = request:call_proc("into_response") if response:get_var("errored") == 1 then error("HTTP request for "..url.." failed to parse the returned json object") end local status = response:get_var("status_code") if status ~= 200 then error("HTTP request for "..url.." returned response code "..status) end local body = response:get_var("body") return body end if not json then json = assert(loadstring(wget("https://raw.githubusercontent.com/rxi/json.lua/master/json.lua")))() end local function fetchVideo(url, ckey) return end channelCache = {} channels = {} local currentChannel = nil local behindSign = SS13.new("/obj/structure/sign") behindSign:set_var("icon_state", "standby") behindSign:set_var("vis_flags", 16) do local request = SS13.new("/datum/http_request") local file_name = "tmp/custom_map_icon.dmi" request:call_proc("prepare", "get", "http://raw.githubusercontent.com/tgstation/auxlua-cookbook/main/waltermeldron/assets/tv/inprogress.dmi", "", "", file_name) request:call_proc("begin_async") while request:call_proc("is_complete") == 0 do sleep() end SS13.stop_tracking(request) behindSign:set_var("icon", SS13.new_untracked("/icon", file_name)) end local tv = SS13.new("/obj/structure/showcase/machinery/tv", spawnLocation) tv:get_var("vis_contents"):add(behindSign) local sign = SS13.new("/image", nil, tv) sign:set_var("icon_state", "off") local scales = { [1] = { pixel_x = 8, pixel_y = 10, behind_pixel_x = 3, behind_pixel_y = 5, behindSignScale = 0.5, volume = 1, sampling = "nearest", fps = 30, maxVidLength = 240, }, [2] = { pixel_x = 0, pixel_y = 4, behind_pixel_x = 0, behind_pixel_y = 4, behindSignScale = 1, volume = 1, sampling = "nearest", fps = 30, maxVidLength = 60, }, [4] = { pixel_x = -16, pixel_y = -8, behind_pixel_x = -5, behind_pixel_y = 2, behindSignScale = 2, volume = 1, extraWaitTime = 3, sampling = "bicubic", realScaleX = 3, realScaleY = 3, realPosX = -1, realPosY = -1, fps = 30, maxVidLength = 60, }, [8] = { pixel_x = -48, pixel_y = -32, behind_pixel_x = -15, behind_pixel_y = -2, behindSignScale = 4, volume = 1, extraWaitTime = 5, sampling = "bicubic", realScaleX = 5, realScaleY = 6, realPosX = -2, realPosY = -2, fps = 20, maxVidLength = 60, }, [16] = { pixel_x = -112, pixel_y = -80, behind_pixel_x = -35, behind_pixel_y = -10, behindSignScale = 8, volume = 1, extraWaitTime = 5, sampling = "bicubic", realScaleX = 11, realScaleY = 11, realPosX = -5, realPosY = -4, fps = 10, maxVidLength = 30, }, } local scaleConfig = scales[scale] sign:set_var("pixel_x", scaleConfig.pixel_x) sign:set_var("pixel_y", scaleConfig.pixel_y) behindSign:set_var("pixel_x", scaleConfig.behind_pixel_x) behindSign:set_var("pixel_y", scaleConfig.behind_pixel_y) behindSign:set_var("transform", dm.global_proc("_matrix", scaleConfig.behindSignScale, 0, 0, 0, scaleConfig.behindSignScale, 0)) tv:set_var("transform", dm.global_proc("_matrix", scale, 0, 0, 0, scale, 0)) tv:call_proc("set_light_power", 100) tv:call_proc("set_light_range", math.floor(scale / 2)) tv:call_proc("set_light_on", true) tv:call_proc("update_light") if scaleConfig.realScaleX then tv:set_var("bound_width", 32 * scaleConfig.realScaleX) tv:set_var("bound_height", 32 * scaleConfig.realScaleY) tv:set_var("bound_x", 32 * scaleConfig.realPosX) tv:set_var("bound_y", 32 * scaleConfig.realPosY) end sign:set_var("appearance_flags", 520) behindSign:set_var("mouse_opacity", 0) behindSign:set_var("appearance_flags", 520) globalPlayerSettings = globalPlayerSettings or nil local function getPlayerSettings(ckey) globalPlayerSettings = globalPlayerSettings or {} local playerSettings = globalPlayerSettings[ckey] if not playerSettings then playerSettings = { disableTv = false, audioMode = AUDIO_DIRECTIONAL, volume = 100 } globalPlayerSettings[ckey] = playerSettings end return playerSettings end local function escape(str) str = string.gsub(str, "([^0-9a-zA-Z !'()*._~-])", -- locale independent function (c) return string.format ("%%%02X", string.byte(c)) end) str = string.gsub(str, " ", "+") return str end local playClip = function() end local startTimeOfDay = dm.world:get_var("timeofday") local playingChannel local animationEnd = 0 local listeners = {} local function startTvLoop(players) local playerClientImageMap = {} while animationEnd > dm.world:get_var("timeofday") and not tv:is_null() and tv:get_var("gc_destroyed") == nil do if dm.world:get_var("timeofday") < startTimeOfDay then startTimeOfDay = dm.world:get_var("timeofday") animationEnd = 0 break end if playingChannel == nil then break end if tv:is_null() or tv:get_var("gc_destroyed") ~= nil then break end local location = tv:get_var("loc") for _, playerData in players do local playerClient = playerData.client local playerConfig = playerData.data if animationEnd <= dm.world:get_var("timeofday") then break end if playingChannel == nil then break end if playerClient:is_null() then continue end local player = playerClient:get_var("mob") if playerClientImageMap[playerClient] == nil then playerClientImageMap[playerClient] = player end local playerPos = player:call_proc("drop_location") local dist = dm.global_proc("_get_dist", playerPos, location) local volume = playerConfig.volume * scaleConfig.volume local playLocation = location if playerConfig.audioMode == AUDIO_MONO then playLocation = playerPos end if not SS13.istype(location, "/turf") or dist > 12 or location:get_var("z") ~= playerPos:get_var("z") then playLocation = nil volume = 0 end if playerClientImageMap[playerClient] ~= player then dm.global_proc("_list_add", playerClient:get_var("images"), sign) playerClientImageMap[playerClient] = currentPlayerMob end player:call_proc("playsound_local", playLocation, playingChannel.sound_file, volume, false, nil, 6, channel, true, playingChannel.sound_file, 17, 1, 1, true) end sleep() end playingChannel = nil for _, playerClient in dm.global_vars:get_var("GLOB"):get_var("clients") do if not playerClient then continue end local player = playerClient:get_var("mob") player:call_proc("stop_sound_channel", channel) dm.global_proc("_list_remove", playerClient:get_var("images"), sign) end animationEnd = 0 sleep() sign:set_var("icon_state", "off") if #channels > 0 then sleep() table.remove(channels, 1) if #channels > 0 then behindSign:set_var("icon_state", "loading") currentChannel = channels[1].channel blockPlayerRequest[channels[1].submitter] = nil sign:set_var("icon", currentChannel.icon_file) currentChannel.sound_file:set_var("status", 0) local tvZLoc = tv:call_proc("drop_location"):get_var("z") for _, player in dm.global_vars:get_var("GLOB"):get_var("player_list") do if over_exec_usage(0.7) then sleep() end local dist = dm.global_proc("_get_dist", player, tv) if (dist > 12 and not infiniteRange) then continue end local playerLocation = player:call_proc("drop_location") local playerClient = player:get_var("client") if player:is_null() or not playerClient or not playerLocation or playerLocation:is_null() then continue end local playerSettings = getPlayerSettings(player:get_var("ckey")) if playerSettings.disableTv then continue end if (player:call_proc("drop_location"):get_var("z") == tvZLoc or infiniteRange) then player:call_proc("playsound_local", nil, currentChannel.sound_file, 0, false, nil, 6, channel, true, currentChannel.sound_file, 17, 1, 1, true) local client = player:get_var("client") dm.global_proc("_list_add", client:get_var("images"), sign) end end SS13.set_timeout(3 + (scaleConfig.extraWaitTime or 0), function() playClip() end) else if #queuedRequests > 0 then behindSign:set_var("icon_state", "loading") else behindSign:set_var("icon_state", "standby") end end end end playClip = function() if not currentChannel then return end sign:set_var("icon_state", "on") SS13.set_timeout(0, function() if animationEnd > dm.world:get_var("timeofday") then return end playingChannel = currentChannel voteSkipData = { voteSkip = 0, voteSkipVoters = {} } queuedUrls[playingChannel.url] = false animationEnd = dm.world:get_var("timeofday") + playingChannel.duration playingChannel.sound_file:set_var("status", 0) local playerList = {} local tvZLoc = tv:call_proc("drop_location"):get_var("z") behindSign:set_var("icon_state", "playing") for _, player in dm.global_vars:get_var("GLOB"):get_var("player_list") do if over_exec_usage(0.7)then sleep() end local dist = dm.global_proc("_get_dist", player, tv) if (dist > 12 and not infiniteRange) then continue end if (player:call_proc("drop_location"):get_var("z") == tvZLoc or infiniteRange) then local playerSettings = getPlayerSettings(player:get_var("ckey")) if playerSettings.disableTv then continue end player:call_proc("playsound_local", nil, playingChannel.sound_file, 0, false, nil, 6, channel, true, playingChannel.sound_file, 17, 1, 1, true) local client = player:get_var("client") dm.global_proc("_list_add", client:get_var("images"), sign) table.insert(playerList, { client = client, data = playerSettings }) end end dm.global_proc("_flick", "on", sign) playingChannel.sound_file:set_var("status", 16) startTvLoop(playerList) end) end local createHref = function(args, content, brackets) brackets = brackets == nil and true or false local data = ""..content.."" if brackets then return "("..data..")" else return data end end queuedRequests = {} queuedUrls = {} local saveData = function() end local saveRequired = false if authToken ~= "" then local address = me:get_var("address") or "localhost" saveData = function() local request = SS13.new("/datum/http_request") request:call_proc("prepare", "get", "http://"..address..":30020/set-settings-data", json.encode(globalPlayerSettings), { Authorization = authToken }) request:call_proc("begin_async") while request:call_proc("is_complete") == 0 do sleep() end local response = request:call_proc("into_response") if response:get_var("errored") == 1 then print("Failed to save settings. Please check API server") end SS13.stop_tracking(request) end if not globalPlayerSettings then local request = SS13.new("/datum/http_request") request:call_proc("prepare", "get", "http://"..address..":30020/get-settings-data", "", { Authorization = authToken }) request:call_proc("begin_async") while request:call_proc("is_complete") == 0 do sleep() end local response = request:call_proc("into_response") if response:get_var("errored") == 1 then print("Failed to fetch settings. Please check API server") else local jsonData = response:get_var("body") if jsonData and jsonData ~= "" then globalPlayerSettings = json.decode(jsonData) else globalPlayerSettings = {} end end SS13.stop_tracking(request) end local function saveDataLoop() if not SS13.is_valid(tv) then return end if saveRequired then saveData() saveRequired = false end SS13.set_timeout(30, saveDataLoop) end saveDataLoop() queryInProgress = false fetchVideo = function() if #queuedRequests == 0 then return end local requestData = queuedRequests[1] local url = requestData.url local ckey = requestData.ckey local startPos = requestData.startPos local duration = requestData.duration if url == nil then table.remove(queuedRequests, 1) return end queuedUrls[url] = true if queryInProgress then return end if channelCache[url] then table.remove(queuedRequests, 1) local channel = channelCache[url] table.insert(channels, { channel = channel, submitter = ckey, startPos = startPos }) if #channels == 1 then currentChannel = channels[1].channel blockPlayerRequest[channels[1].submitter] = nil sign:set_var("icon", currentChannel.icon_file) playClip() end return end if behindSign:get_var("icon_state") == "standby" then behindSign:set_var("icon_state", "loading") end local frames = forcedFps or scaleConfig.fps or 30 local channel = {} channel.url = url queryInProgress = true local vidLength = scaleConfig.maxVidLength if ckey == admin and bypassVidLength then -- Load in however long you want it to be, but this is just here so that you don't crash people's clients with 1 hr long videos vidLength = 1200 end if startPos < 0 then startPos = 0 end if duration <= 0 then duration = vidLength end duration = math.min(duration, vidLength) local performFetch = SS13.new("/datum/http_request") performFetch:call_proc("prepare", "get", "http://"..address..":30020/perform-fetch?youtube-url="..escape(url).."&size="..scale.."&sampling="..scaleConfig.sampling.."&frames="..frames.."&max-video-length="..vidLength.."&start-time="..startPos.."&duration="..duration, "", { Authorization = authToken }) performFetch:call_proc("begin_async") while performFetch:call_proc("is_complete") == 0 do sleep() end PERFORM_FETCH_RESULT = performFetch:call_proc("into_response") SS13.stop_tracking(performFetch) local fetchCompleted = false local errored = false local errors = "" local responseData = "" while fetchCompleted == false do local checkFetch = SS13.new("/datum/http_request") checkFetch:call_proc("prepare", "get", "http://"..address..":30020/check-fetch", "", { Authorization = authToken }) checkFetch:call_proc("begin_async") while checkFetch:call_proc("is_complete") == 0 do sleep() end SS13.stop_tracking(checkFetch) local response = checkFetch:call_proc("into_response") if response:get_var("errored") == 1 then fetchCompleted = true errored = true errors = response:get_var("error") break end responseData = response:get_var("body") if responseData ~= "Not ready" then fetchCompleted = true break end SS13.wait(3) end if responseData ~= "1" then local playerClient = dm.global_vars:get_var("GLOB"):get_var("directory"):get(ckey) if behindSign:get_var("icon_state") == "loading" and #channels == 0 then behindSign:set_var("icon_state", "standby") end table.remove(queuedRequests, 1) if playerClient == nil then return end local player = playerClient:get_var("mob") player:call_proc("playsound_local", nil, "sound/effects/adminhelp.ogg", 75) dm.global_proc("to_chat", player, "Your video request was rejected. This is because an error occured with the request. Please input appropriate video details to avoid this from happening again.") dm.global_proc("message_admins", "TV: "..dm.global_proc("key_name_admin", player).." has been warned about their request due to bad input data.") queryInProgress = false queuedUrls[url] = false return end if errored then dm.global_proc("message_admins", "TV: Unable to fetch video for TV due to API errors!") queryInProgress = false table.remove(queuedRequests, 1) return end local request = SS13.new("/datum/http_request") local file_name = "tmp/custom_map_icon.dmi" request:call_proc("prepare", "get", "http://"..address..":30020/get-dmi?size="..scale, "", { Authorization = authToken }, file_name) request:call_proc("begin_async") while request:call_proc("is_complete") == 0 do sleep() end local response = request:call_proc("into_response") channel.icon_file = SS13.await(SS13.global_proc, "_new", "/icon", { file_name }) channel.title = response:get_var("headers"):get("video-title") local references = SS13.state.vars.references references:add(channel.icon_file) SS13.stop_tracking(request) sleep() local request = SS13.new("/datum/http_request") local file_name = "tmp/custom_map_sound.ogg" request:call_proc("prepare", "get", "http://"..address..":30020/get-audio", "", { Authorization = authToken }, file_name) request:call_proc("begin_async") while request:call_proc("is_complete") == 0 do sleep() end local response = request:call_proc("into_response") if response:get_var("errored") == 1 or tonumber(response:get_var("headers"):get("audio-length")) == nil then queryInProgress = false if behindSign:get_var("icon_state") == "loading" and #channels == 0 then behindSign:set_var("icon_state", "standby") end queuedUrls[url] = false table.remove(queuedRequests, 1) return end channel.sound_file = SS13.new("/sound", file_name) channel.duration = tonumber(response:get_var("headers"):get("audio-length")) * 10 SS13.stop_tracking(request) table.insert(channels, { channel = channel, submitter = ckey, startPos = startPos }) table.remove(queuedRequests, 1) channelCache[url] = channel request = nil queryInProgress = false dm.global_proc("message_admins", "TV: Loaded youtube video "..dm.global_proc("sanitize", channel.title).." - "..dm.global_proc("sanitize", url).." for use in the TV. "..createHref("skip="..escape(url), "SKIP")) sleep() if tv:is_null() then return end if #channels == 1 then currentChannel = channels[1].channel blockPlayerRequest[channels[1].submitter] = nil sign:set_var("icon", currentChannel.icon_file) currentChannel.sound_file:set_var("status", 0) local tvZLoc = tv:call_proc("drop_location"):get_var("z") sleep() for _, player in dm.global_vars:get_var("GLOB"):get_var("player_list") do if over_exec_usage(0.7) then sleep() end local dist = dm.global_proc("_get_dist", player, tv) if dist > 12 and not infiniteRange then continue end if (player:call_proc("drop_location"):get_var("z") == tvZLoc or infiniteRange) then local playerSettings = getPlayerSettings(player:get_var("ckey")) if playerSettings.disableTv then continue end player:call_proc("playsound_local", nil, currentChannel.sound_file, 0, false, nil, 6, channel, true, currentChannel.sound_file, 17, 1, 1, true) local client = player:get_var("client") dm.global_proc("_list_add", client:get_var("images"), sign) end end SS13.set_timeout(3 + (scaleConfig.extraWaitTime or 0), function() playClip() end) end SS13.set_timeout(0, function() fetchVideo() end) end end SS13.register_signal(tv, "parent_qdeleting", function() dm.global_proc("qdel", behindSign) for _, player in dm.global_vars:get_var("GLOB"):get_var("player_list") do player:call_proc("stop_sound_channel", channel) end SS13.stop_tracking(sign) saveData() end) local playerOpen = {} local blocked = {} local requestCounter = 0 local canMakeRequest = function(user, isAdmin) if onePerUser and not isAdmin then if blockPlayerRequest[user:get_var("ckey")] then user:call_proc("balloon_alert", user, "only 1 request at a time!") return false end end return true end local makeRequest = function(user, isAdmin) local ckey = user:get_var("ckey") if blocked[ckey] then user:call_proc("balloon_alert", user, "blocked from making requests!") return end if not acceptingRequests and not isAdmin then return end if not canMakeRequest(user, isAdmin) then return end SS13.set_timeout(0, function() if not ckey then return end local vidLength = scaleConfig.maxVidLength if isAdmin and bypassVidLength then -- Load in however long you want it to be, but this is just here so that you don't crash people's clients with 1 hr long videos vidLength = 1200 end playerOpen[ckey] = true local input = SS13.await(SS13.global_proc, "tgui_input_text", user, "Input Youtube URL. Optionally include timestamp to start at a specific video location. Videos longer than "..vidLength.." seconds will be cut down in length.", "Request Youtube Video") playerOpen[ckey] = false if not canMakeRequest(user, isAdmin) then return end if input == nil or input == "" then return end if queuedUrls[input] then user:call_proc("balloon_alert", user, "that request is already queued!") return end local duration = -1 local startTime = tonumber(string.match(input, "t=(%d+)")) if startTime then playerOpen[ckey] = true duration = SS13.await(SS13.global_proc, "tgui_input_number", user, "Detected starting location, please specify video length to showcase in seconds", "Specify video length", vidLength, vidLength, 1) playerOpen[ckey] = false else startTime = -1 end -- This does not make it safe, but there are further protections in the api script anyways local scrubbed = dm.global_proc("shell_url_scrub", input) if not isAdmin then blockPlayerRequest[ckey] = true if autoAccept then dm.global_proc("message_admins", "TV: "..dm.global_proc("key_name_admin", user).." queued the youtube video "..scrubbed.." to be played on the TV. "..createHref("skip="..escape(scrubbed)..";", "SKIP").." "..createHref("block=1;ckey="..ckey, "BLOCK")) table.insert(queuedRequests, { url = scrubbed, ckey = ckey, startPos = startTime, duration = duration }) fetchVideo() user:call_proc("playsound_local", nil, "sound/misc/asay_ping.ogg", 15) dm.global_proc("to_chat", user, "Your video request was queued.") else dm.global_proc("message_admins", "TV: "..dm.global_proc("key_name_admin", user).." requested the youtube video "..scrubbed.." to be played on the TV. "..createHref("link="..escape(scrubbed)..";ckey="..ckey..";startTime="..startTime..";duration="..duration..";play_id="..requestCounter, "PLAY").." "..createHref("reject=1;reject_id="..requestCounter..";ckey="..ckey, "REJECT").." "..createHref("block=1;ckey="..ckey, "BLOCK")) end requestCounter += 1 else table.insert(queuedRequests, { url = scrubbed, ckey = ckey, startPos = startTime, duration = duration }) fetchVideo() end end) end local function openClientSettings(user) local userCkey = user:get_var("ckey") local browser = SS13.new_untracked("/datum/browser", user, "Client TV Settings", "Client TV Settings", 300, 200) local data = "" local playerSettings = getPlayerSettings(userCkey) local tvDisableToggle = createHref("client_disable=1", "YES", false) if playerSettings.disableTv then tvDisableToggle = createHref("client_disable=0", "NO", false) end data = data.."