2018-09-19 06:51:25 +02:00
--[[
2019-06-07 23:43:41 +02:00
Hello there !
2018-09-19 06:51:25 +02:00
This will add a player list with " ranks " to your server .
Oh .. and you can also " poke " a player .
pokemessages = 80 % by redlabel
To install , add : require " player_list "
to your scenario control.lua .
---MewMew---
2019-07-06 16:38:03 +02:00
Minor changes by ~~~ Gerkiz ~~~
2018-09-19 06:51:25 +02:00
--]]
2019-01-02 23:27:24 +01:00
local event = require ' utils.event '
2019-07-06 16:38:03 +02:00
local play_time = require ' utils.session_data '
2018-09-19 06:51:25 +02:00
local symbol_asc = " ▲ "
local symbol_desc = " ▼ "
local pokemessages = { " a stick " , " a leaf " , " a moldy carrot " , " a crispy slice of bacon " , " a french fry " , " a realistic toygun " , " a broomstick " , " a thirteen inch iron stick " , " a mechanical keyboard " , " a fly fishing cane " , " a selfie stick " , " an oversized fidget spinner " , " a thumb extender " , " a dirty straw " , " a green bean " , " a banana " , " an umbrella " , " grandpa's walking stick " , " live firework " , " a toilet brush " , " a fake hand " , " an undercooked hotdog " , " a slice of yesterday's microwaved pizza " , " bubblegum " , " a biter leg " , " grandma's toothbrush " , " charred octopus " , " a dollhouse bathtub " , " a length of copper wire " , " a decommissioned nuke " , " a smelly trout " , " an unopened can of deodorant " , " a stone brick " , " a half full barrel of lube " , " a half empty barrel of lube " , " an unexploded cannon shell " , " a blasting programmable speaker " , " a not so straight rail " , " a mismatched pipe to ground " , " a surplus box of landmines " , " decommissioned yellow rounds " , " an oily pumpjack shaft " , " a melted plastic bar in the shape of the virgin mary " , " a bottle of watermelon vitamin water " , " a slice of watermelon " , " a stegosaurus tibia " , " a basking musician's clarinet " , " a twig " , " an undisclosed pokey item " , " a childhood trophy everyone else got " , " a dead starfish " , " a titanium toothpick " , " a nail file " , " a stamp collection " , " a bucket of lego " , " a rolled up carpet " , " a rolled up WELCOME doormat " , " Bobby's favorite bone " , " an empty bottle of cheap vodka " , " a tattooing needle " , " a peeled cucumber " , " a stack of cotton candy " , " a signed baseball bat " , " that 5 dollar bill grandma sent for christmas " , " a stack of overdue phone bills " , " the 'relax' section of the white pages " , " a bag of gym clothes which never made it to the washing machine " , " a handful of peanut butter " , " a pheasant's feather " , " a rusty pickaxe " , " a diamond sword " , " the bill of rights of a banana republic " , " one of those giant airport Toblerone's " , " a long handed inserter " , " a wiimote " , " an easter chocolate rabbit " , " a ball of yarn the cat threw up " , " a slightly expired but perfectly edible cheese sandwich " , " conclusive proof of lizard people existence " , " a pen drive full of high res wallpapers " , " a pet hamster " , " an oversized goldfish " , " a one foot extension cord " , " a CD from Walmart's 1 dollar bucket " , " a magic wand " , " a list of disappointed people who believed in you " , " murder exhibit no. 3 " , " a paperback copy of 'Great Expectations' " , " a baby biter " , " a little biter fang " , " the latest diet fad " , " a belt that no longer fits you " , " an abandoned pet rock " , " a lava lamp " , " some spirit herbs " , " a box of fish sticks found at the back of the freezer " , " a bowl of tofu rice " , " a bowl of ramen noodles " , " a live lobster! " , " a miniature golf cart " , " dunce cap " , " a fully furnished x-mas tree " , " an orphaned power pole " , " an horphaned power pole " , " an box of overpriced girl scout cookies " , " the cheapest item from the yard sale " , " a Sharpie " , " a glowstick " , " a thick unibrow hair " , " a very detailed map of Kazakhstan " , " the official Factorio installation DVD " , " a Liberal Arts degree " , " a pitcher of Kool-Aid " , " a 1/4 pound vegan burrito " , " a bottle of expensive wine " , " a hamster sized gravestone " , " a counterfeit Cuban cigar " , " an old Nokia phone " , " a huge inferiority complex " , " a dead real state agent " , " a deck of tarot cards " , " unreleased Wikileaks documents " , " a mean-looking garden dwarf " , " the actual mythological OBESE cat " , " a telescope used to spy on the MILF next door " , " a fancy candelabra " , " the comic version of the Kama Sutra " , " an inflatable 'Netflix & chill' doll " , " whatever it is redlabel gets high on " , " Obama's birth certificate " , " a deck of Cards Against Humanity " , " a copy of META MEME HUMOR for Dummies " , " an abandoned, not-so-young-anymore puppy " , " one of those useless items advertised on TV " , " a genetic blueprint of a Japanese teen idol " }
local function get_formatted_playtime ( x )
2018-10-10 11:16:24 +02:00
local time_one = 216000
local time_two = 3600
if x < 5184000 then
local y = x / 216000
y = tostring ( y )
local h = " "
2019-06-07 23:43:41 +02:00
for i = 1 , 10 , 1 do
local z = string.sub ( y , i , i )
2018-10-10 11:16:24 +02:00
if z == " . " then
break
2019-06-07 23:43:41 +02:00
else
h = h .. z
end
2018-10-10 11:16:24 +02:00
end
2019-06-07 23:43:41 +02:00
2018-10-10 11:16:24 +02:00
local m = x % 216000
m = m / 3600
m = math.floor ( m )
m = tostring ( m )
2019-06-07 23:43:41 +02:00
2018-10-10 11:16:24 +02:00
if h == " 0 " then
local str = m .. " minutes "
return str
else
local str = h .. " hours "
str = str .. m
str = str .. " minutes "
return str
end
2018-09-19 06:51:25 +02:00
else
2018-10-10 11:16:24 +02:00
local y = x / 5184000
y = tostring ( y )
local h = " "
2019-06-07 23:43:41 +02:00
for i = 1 , 10 , 1 do
local z = string.sub ( y , i , i )
2018-10-10 11:16:24 +02:00
if z == " . " then
break
2019-06-07 23:43:41 +02:00
else
h = h .. z
end
2018-10-10 11:16:24 +02:00
end
2019-06-07 23:43:41 +02:00
2018-10-10 11:16:24 +02:00
local m = x % 5184000
m = m / 216000
m = math.floor ( m )
m = tostring ( m )
2019-06-07 23:43:41 +02:00
2018-10-10 11:16:24 +02:00
if h == " 0 " then
local str = m .. " days "
return str
else
local str = h .. " days "
str = str .. m
str = str .. " hours "
return str
end
2018-09-19 06:51:25 +02:00
end
end
local function get_rank ( player )
2019-07-06 16:38:03 +02:00
local play_table = play_time.get_session_table ( )
2019-07-06 23:44:27 +02:00
local t = play_table [ player.name ]
2019-07-06 16:38:03 +02:00
2018-09-23 03:17:31 +02:00
local m = ( player.online_time + t ) / 3600
2019-06-07 23:43:41 +02:00
2018-09-19 06:51:25 +02:00
local ranks = {
2019-06-07 23:43:41 +02:00
" item/burner-mining-drill " , " item/burner-inserter " , " item/stone-furnace " , " item/light-armor " , " item/steam-engine " ,
2019-03-02 13:06:29 +01:00
" item/inserter " , " item/transport-belt " , " item/underground-belt " , " item/splitter " , " item/assembling-machine-1 " , " item/long-handed-inserter " , " item/electronic-circuit " , " item/electric-mining-drill " , " item/dummy-steel-axe " ,
2019-02-28 18:53:33 +01:00
" item/heavy-armor " , " item/steel-furnace " , " item/gun-turret " , " item/fast-transport-belt " , " item/fast-underground-belt " , " item/fast-splitter " , " item/assembling-machine-2 " , " item/fast-inserter " , " item/radar " , " item/filter-inserter " ,
2019-06-07 23:43:41 +02:00
" item/defender-capsule " , " item/pumpjack " , " item/chemical-plant " , " item/solar-panel " , " item/advanced-circuit " , " item/modular-armor " , " item/accumulator " , " item/construction-robot " ,
2018-09-19 06:51:25 +02:00
" item/distractor-capsule " , " item/stack-inserter " , " item/electric-furnace " , " item/express-transport-belt " , " item/express-underground-belt " , " item/express-splitter " , " item/assembling-machine-3 " , " item/processing-unit " , " item/power-armor " , " item/logistic-robot " , " item/laser-turret " ,
" item/stack-filter-inserter " , " item/destroyer-capsule " , " item/power-armor-mk2 " , " item/flamethrower-turret " , " item/beacon " ,
" item/steam-turbine " , " item/centrifuge " , " item/nuclear-reactor "
}
2019-06-07 23:43:41 +02:00
2018-09-19 06:51:25 +02:00
--52 ranks
2019-06-07 23:43:41 +02:00
2019-04-12 02:57:25 +02:00
local time_needed = 240 -- in minutes between rank upgrades
2018-09-19 06:51:25 +02:00
m = m / time_needed
m = math.floor ( m )
m = m + 1
2019-06-07 23:43:41 +02:00
2018-09-19 06:51:25 +02:00
if m > # ranks then m = # ranks end
return ranks [ m ]
end
2019-06-07 23:43:41 +02:00
local comparators = {
[ " pokes_asc " ] = function ( a , b ) return a.pokes > b.pokes end ,
[ " pokes_desc " ] = function ( a , b ) return a.pokes < b.pokes end ,
[ " total_time_played_asc " ] = function ( a , b ) return a.total_played_ticks < b.total_played_ticks end ,
[ " total_time_played_desc " ] = function ( a , b ) return a.total_played_ticks > b.total_played_ticks end ,
[ " time_played_asc " ] = function ( a , b ) return a.played_ticks < b.played_ticks end ,
[ " time_played_desc " ] = function ( a , b ) return a.played_ticks > b.played_ticks end ,
[ " name_asc " ] = function ( a , b ) return a.name : lower ( ) < b.name : lower ( ) end ,
[ " name_desc " ] = function ( a , b ) return a.name : lower ( ) > b.name : lower ( ) end
}
local function get_comparator ( sort_by )
return comparators [ sort_by ]
end
2018-09-19 06:51:25 +02:00
local function get_sorted_list ( sort_by )
2019-07-06 16:38:03 +02:00
local play_table = play_time.get_session_table ( )
2018-09-19 06:51:25 +02:00
local player_list = { }
for i , player in pairs ( game.connected_players ) do
player_list [ i ] = { }
player_list [ i ] . rank = get_rank ( player )
player_list [ i ] . name = player.name
2019-06-07 23:43:41 +02:00
2019-07-06 16:38:03 +02:00
2019-07-06 23:44:27 +02:00
t = play_table [ player.name ]
2019-07-06 16:38:03 +02:00
2019-06-07 23:43:41 +02:00
player_list [ i ] . total_played_time = get_formatted_playtime ( t + player.online_time )
player_list [ i ] . total_played_ticks = t + player.online_time
2018-09-19 06:51:25 +02:00
player_list [ i ] . played_time = get_formatted_playtime ( player.online_time )
player_list [ i ] . played_ticks = player.online_time
if not global.player_list_pokes_counter [ player.index ] then global.player_list_pokes_counter [ player.index ] = 0 end
player_list [ i ] . pokes = global.player_list_pokes_counter [ player.index ]
player_list [ i ] . player_index = player.index
2019-06-07 23:43:41 +02:00
end
local comparator = get_comparator ( sort_by )
table.sort ( player_list , comparator )
2018-09-19 06:51:25 +02:00
return player_list
end
local function player_list_show ( player , sort_by )
2019-06-07 23:43:41 +02:00
-- Frame management
2018-09-19 06:51:25 +02:00
local frame = player.gui . left [ " player-list-panel " ]
if frame then frame.destroy ( ) end
2019-06-07 23:43:41 +02:00
--player.gui.left.direction = "horizontal"
2018-09-19 06:51:25 +02:00
local frame = player.gui . left.add { type = " frame " , name = " player-list-panel " , direction = " vertical " }
2019-06-07 23:43:41 +02:00
2018-09-19 06:51:25 +02:00
frame.style . minimal_width = 408
frame.style . top_padding = 8
frame.style . left_padding = 8
frame.style . right_padding = 8
frame.style . bottom_padding = 8
2019-06-07 23:43:41 +02:00
-- Header management
local t = frame.add { type = " table " , name = " player_list_panel_header_table " , column_count = 5 }
for _ , w in ipairs ( { 36 , 200 , 160 , 130 , 35 } ) do
local label = t.add { type = " label " , caption = " " }
label.style . minimal_width = w
label.style . maximal_width = w
end
local headers = {
[ 1 ] = " [color=0.1,0.7,0.1] " -- green
.. tostring ( # game.connected_players )
.. " [/color] " ,
[ 2 ] = " Online "
.. " / "
.. " [color=0.7,0.1,0.1] " -- red
.. tostring ( # game.players - # game.connected_players )
.. " [/color] "
.. " Offline " ,
[ 3 ] = " Total Time " ,
[ 4 ] = " Current Time " ,
[ 5 ] = " Poke "
}
local header_modifier = {
[ " name_asc " ] = function ( h ) h [ 2 ] = symbol_asc .. h [ 2 ] end ,
[ " name_desc " ] = function ( h ) h [ 2 ] = symbol_desc .. h [ 2 ] end ,
[ " total_time_played_asc " ] = function ( h ) h [ 3 ] = symbol_asc .. h [ 3 ] end ,
[ " total_time_played_desc " ] = function ( h ) h [ 3 ] = symbol_desc .. h [ 3 ] end ,
[ " time_played_asc " ] = function ( h ) h [ 4 ] = symbol_asc .. h [ 4 ] end ,
[ " time_played_desc " ] = function ( h ) h [ 4 ] = symbol_desc .. h [ 4 ] end ,
[ " pokes_asc " ] = function ( h ) h [ 5 ] = symbol_asc .. h [ 5 ] end ,
[ " pokes_desc " ] = function ( h ) h [ 5 ] = symbol_desc .. h [ 5 ] end
}
header_modifier [ sort_by ] ( headers )
for k , v in ipairs ( headers ) do
local label = t.add {
type = " label " ,
name = " player_list_panel_header_ " .. k ,
caption = v
}
label.style . font = " default-bold "
label.style . font_color = { r = 0.98 , g = 0.66 , b = 0.22 }
end
-- special style on first header
local label = t [ " player_list_panel_header_1 " ]
2018-10-05 10:53:01 +02:00
label.style . minimal_width = 36
label.style . maximal_width = 36
2019-02-26 20:12:29 +01:00
label.style . horizontal_align = " right "
2019-06-07 22:37:45 +02:00
2019-06-07 23:43:41 +02:00
-- List management
2018-09-19 06:51:25 +02:00
local player_list_panel_table = frame.add { type = " scroll-pane " , name = " scroll_pane " , direction = " vertical " , horizontal_scroll_policy = " never " , vertical_scroll_policy = " auto " }
player_list_panel_table.style . maximal_height = 530
2019-06-07 23:43:41 +02:00
2018-09-19 06:51:25 +02:00
player_list_panel_table = player_list_panel_table.add { type = " table " , name = " player_list_panel_table " , column_count = 5 }
2019-06-07 23:43:41 +02:00
local player_list = get_sorted_list ( sort_by )
for i = 1 , # player_list , 1 do
-- Icon
2018-09-19 06:51:25 +02:00
local sprite = player_list_panel_table.add { type = " sprite " , name = " player_rank_sprite_ " .. i , sprite = player_list [ i ] . rank }
2018-10-05 10:53:01 +02:00
sprite.style . minimal_width = 36
2019-06-07 23:43:41 +02:00
-- Name
local label = player_list_panel_table.add { type = " label " , name = " player_list_panel_player_names_ " .. i , caption = player_list [ i ] . name }
label.style . font = " default "
2018-09-19 06:51:25 +02:00
label.style . font_color = {
r = .4 + game.players [ player_list [ i ] . player_index ] . color.r * 0.6 ,
g = .4 + game.players [ player_list [ i ] . player_index ] . color.g * 0.6 ,
b = .4 + game.players [ player_list [ i ] . player_index ] . color.b * 0.6 ,
}
2018-10-05 10:53:01 +02:00
label.style . minimal_width = 200
label.style . maximal_width = 200
2019-06-07 23:43:41 +02:00
-- Total time
local label = player_list_panel_table.add { type = " label " , name = " player_list_panel_player_total_time_played_ " .. i , caption = player_list [ i ] . total_played_time }
2018-09-19 06:51:25 +02:00
label.style . minimal_width = 160
label.style . maximal_width = 160
2019-06-07 23:43:41 +02:00
-- Current time
local label = player_list_panel_table.add { type = " label " , name = " player_list_panel_player_time_played_ " .. i , caption = player_list [ i ] . played_time }
2018-09-19 06:51:25 +02:00
label.style . minimal_width = 130
label.style . maximal_width = 130
2019-06-07 23:43:41 +02:00
-- Poke
2018-09-19 06:51:25 +02:00
local flow = player_list_panel_table.add { type = " flow " , name = " button_flow_ " .. i , direction = " horizontal " }
flow.add { type = " label " , name = " button_spacer_ " .. i , caption = " " }
2019-06-07 23:43:41 +02:00
local button = flow.add { type = " button " , name = " poke_player_ " .. player_list [ i ] . name , caption = player_list [ i ] . pokes }
2018-09-19 06:51:25 +02:00
button.style . font = " default "
label.style . font_color = { r = 0.83 , g = 0.83 , b = 0.83 }
button.style . minimal_height = 30
button.style . minimal_width = 30
button.style . maximal_height = 30
button.style . maximal_width = 30
button.style . top_padding = 0
button.style . left_padding = 0
button.style . right_padding = 0
2019-06-07 23:43:41 +02:00
button.style . bottom_padding = 0
end
2018-09-19 06:51:25 +02:00
end
local function on_gui_click ( event )
if not event then return end
2019-06-07 23:43:41 +02:00
if not event.element then return end
2018-09-19 06:51:25 +02:00
if not event.element . valid then return end
if not event.element . name then return end
2019-06-07 23:43:41 +02:00
2018-10-10 11:16:24 +02:00
local player = game.players [ event.element . player_index ]
local name = event.element . name
2019-06-07 23:43:41 +02:00
local actions = {
[ " player_list_button " ] = function ( )
if player.gui . left [ " player-list-panel " ] then
player.gui . left [ " player-list-panel " ] . destroy ( )
else
player_list_show ( player , " total_time_played_desc " )
end
end ,
[ " player_list_panel_header_2 " ] = function ( )
if string.find ( event.element . caption , symbol_desc ) then
player_list_show ( player , " name_asc " )
else
player_list_show ( player , " name_desc " )
end
end ,
[ " player_list_panel_header_3 " ] = function ( )
if string.find ( event.element . caption , symbol_desc ) then
player_list_show ( player , " total_time_played_asc " )
else
player_list_show ( player , " total_time_played_desc " )
end
end ,
[ " player_list_panel_header_4 " ] = function ( )
if string.find ( event.element . caption , symbol_desc ) then
player_list_show ( player , " time_played_asc " )
else
player_list_show ( player , " time_played_desc " )
end
end ,
[ " player_list_panel_header_5 " ] = function ( )
if string.find ( event.element . caption , symbol_desc ) then
player_list_show ( player , " pokes_asc " )
else
player_list_show ( player , " pokes_desc " )
end
2018-09-19 06:51:25 +02:00
end
2019-06-07 23:43:41 +02:00
}
if actions [ name ] then
actions [ name ] ( )
return
2018-10-10 11:16:24 +02:00
end
2019-06-07 23:43:41 +02:00
2018-11-29 20:30:26 +01:00
if not event.element . valid then return end
2018-10-31 01:01:49 +01:00
--Poke other players
2018-10-10 11:16:24 +02:00
if string.sub ( event.element . name , 1 , 11 ) == " poke_player " then
local poked_player = string.sub ( event.element . name , 13 , string.len ( event.element . name ) )
if player.name == poked_player then return end
2019-06-07 23:43:41 +02:00
if global.poke_spam_protection [ event.element . player_index ] + 420 < game.tick then
2018-10-10 11:16:24 +02:00
local str = " >> "
2019-06-07 23:43:41 +02:00
str = str .. player.name
2018-10-10 11:16:24 +02:00
str = str .. " has poked "
2019-06-07 23:43:41 +02:00
str = str .. poked_player
2018-10-10 11:16:24 +02:00
str = str .. " with "
local z = math.random ( 1 , # pokemessages )
str = str .. pokemessages [ z ]
str = str .. " << "
2019-06-07 23:43:41 +02:00
game.print ( str )
2018-10-10 11:16:24 +02:00
global.poke_spam_protection [ event.element . player_index ] = game.tick
local p = game.players [ poked_player ]
global.player_list_pokes_counter [ p.index ] = global.player_list_pokes_counter [ p.index ] + 1
2019-06-07 23:43:41 +02:00
end
end
2018-09-19 06:51:25 +02:00
end
2018-12-23 17:36:21 +01:00
2019-06-07 23:43:41 +02:00
local function on_player_joined_game ( event )
2018-12-23 17:36:21 +01:00
local player = game.players [ event.player_index ]
if not global.poke_spam_protection then global.poke_spam_protection = { } end
global.poke_spam_protection [ event.player_index ] = game.tick
2019-06-07 23:43:41 +02:00
if not global.player_list_pokes_counter then global.player_list_pokes_counter = { } end
2018-12-23 17:36:21 +01:00
if player.gui . top.player_list_button == nil then
2019-05-03 09:47:16 +02:00
local button = player.gui . top.add ( { type = " sprite-button " , name = " player_list_button " , sprite = " item/heavy-armor " , tooltip = " Player List " } )
2018-12-23 17:36:21 +01:00
button.style . minimal_height = 38
button.style . minimal_width = 38
button.style . top_padding = 2
button.style . left_padding = 4
button.style . right_padding = 4
button.style . bottom_padding = 2
end
2019-06-07 23:43:41 +02:00
2018-12-23 17:36:21 +01:00
if player.gui . left [ " player-list-panel " ] then
player.gui . left [ " player-list-panel " ] . destroy ( )
player_list_show ( player , " total_time_played_desc " )
end
end
2019-01-02 23:27:24 +01:00
event.add ( defines.events . on_player_joined_game , on_player_joined_game )
2019-02-26 20:12:29 +01:00
event.add ( defines.events . on_gui_click , on_gui_click )