1002 lines
30 KiB
Lua
1002 lines
30 KiB
Lua
local json = require("common.json")
|
|
|
|
local sound = require('common.sound');
|
|
|
|
local resX,resY = game.GetResolution()
|
|
|
|
local mposx = 0;
|
|
local mposy = 0;
|
|
local hovered = nil;
|
|
local buttonWidth = resX*(3/4);
|
|
local buttonHeight = 75;
|
|
local buttonBorder = 2;
|
|
local portrait
|
|
|
|
game.LoadSkinSample("click-02")
|
|
game.LoadSkinSample("click-01")
|
|
game.LoadSkinSample("menu_click")
|
|
|
|
local loading = true;
|
|
local rooms = {};
|
|
local lobby_users = {};
|
|
local selected_room = nil;
|
|
local user_id = nil;
|
|
local jacket = 0;
|
|
local all_ready;
|
|
local user_ready;
|
|
local hard_mode = false;
|
|
local rotate_host = false;
|
|
local start_game_soon = false;
|
|
local host = nil;
|
|
local owner = nil;
|
|
local missing_song = false;
|
|
local placeholderJacket = gfx.CreateSkinImage("song_select/loading.png", 0)
|
|
local did_exit = false;
|
|
|
|
local diffColors = {{0,0,255}, {0,255,0}, {255,0,0}, {255, 0, 255}}
|
|
|
|
local grades = {
|
|
{["max"] = 6999999, ["image"] = gfx.CreateSkinImage("common/grades/D.png", 0)},
|
|
{["max"] = 7999999, ["image"] = gfx.CreateSkinImage("common/grades/C.png", 0)},
|
|
{["max"] = 8699999, ["image"] = gfx.CreateSkinImage("common/grades/B.png", 0)},
|
|
{["max"] = 8999999, ["image"] = gfx.CreateSkinImage("common/grades/A.png", 0)},
|
|
{["max"] = 9299999, ["image"] = gfx.CreateSkinImage("common/grades/A+.png", 0)},
|
|
{["max"] = 9499999, ["image"] = gfx.CreateSkinImage("common/grades/AA.png", 0)},
|
|
{["max"] = 9699999, ["image"] = gfx.CreateSkinImage("common/grades/AA+.png", 0)},
|
|
{["max"] = 9799999, ["image"] = gfx.CreateSkinImage("common/grades/AAA.png", 0)},
|
|
{["max"] = 9899999, ["image"] = gfx.CreateSkinImage("common/grades/AAA+.png", 0)},
|
|
{["max"] = 99999999, ["image"] = gfx.CreateSkinImage("common/grades/S.png", 0)}
|
|
}
|
|
|
|
local badges = {
|
|
gfx.CreateSkinImage("badges/played.png", 0),
|
|
gfx.CreateSkinImage("badges/clear.png", 0),
|
|
gfx.CreateSkinImage("badges/hard-clear.png", 0),
|
|
gfx.CreateSkinImage("badges/full-combo.png", 0),
|
|
gfx.CreateSkinImage("badges/perfect.png", 0)
|
|
}
|
|
|
|
local user_name_key = game.GetSkinSetting('multi.user_name_key')
|
|
if user_name_key == nil then
|
|
user_name_key = 'nick'
|
|
end
|
|
local name = game.GetSkinSetting(user_name_key)
|
|
if name == nil or name == '' then
|
|
name = 'Guest'
|
|
end
|
|
|
|
local normal_font = game.GetSkinSetting('multi.normal_font')
|
|
if normal_font == nil then
|
|
normal_font = 'NotoSans-Regular.ttf'
|
|
end
|
|
local mono_font = game.GetSkinSetting('multi.mono_font')
|
|
if mono_font == nil then
|
|
mono_font = 'NovaMono.ttf'
|
|
end
|
|
|
|
local SERVER = game.GetSkinSetting("multi.server")
|
|
|
|
|
|
|
|
|
|
|
|
mouse_clipped = function(x,y,w,h)
|
|
return mposx > x and mposy > y and mposx < x+w and mposy < y+h;
|
|
end;
|
|
|
|
draw_room = function(name, x, y, selected, hoverindex)
|
|
local buttonWidth = resX*(3/4);
|
|
local rx = x - (buttonWidth / 2);
|
|
local ty = y - (buttonHeight / 2);
|
|
local roomButtonBorder = buttonBorder;
|
|
gfx.BeginPath();
|
|
gfx.FillColor(0,128,255);
|
|
if selected then
|
|
gfx.FillColor(0,255,0);
|
|
roomButtonBorder = 4;
|
|
end
|
|
if mouse_clipped(rx,ty, buttonWidth, buttonHeight) then
|
|
hovered = hoverindex;
|
|
gfx.FillColor(255,128,0);
|
|
end
|
|
gfx.Rect(rx - roomButtonBorder,
|
|
ty - roomButtonBorder,
|
|
buttonWidth + (roomButtonBorder * 2),
|
|
buttonHeight + (roomButtonBorder * 2));
|
|
gfx.Fill();
|
|
gfx.BeginPath();
|
|
gfx.FillColor(40,40,40);
|
|
gfx.Rect(rx, ty, buttonWidth, buttonHeight);
|
|
gfx.Fill();
|
|
gfx.BeginPath();
|
|
gfx.FillColor(255,255,255);
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER + gfx.TEXT_ALIGN_MIDDLE);
|
|
gfx.FontSize(40);
|
|
gfx.Text(name, x, y);
|
|
end;
|
|
|
|
draw_button = function(name, x, y, buttonWidth, hoverindex)
|
|
draw_button_color(name, x, y, buttonWidth, hoverindex, 40,40,40, 0,128,255)
|
|
end
|
|
|
|
draw_button_color = function(name, x, y, buttonWidth, hoverindex,r,g,b, olr,olg,olb)
|
|
local rx = x - (buttonWidth / 2);
|
|
local ty = y - (buttonHeight / 2);
|
|
gfx.BeginPath();
|
|
gfx.FillColor(olr, olg, olb);
|
|
if mouse_clipped(rx,ty, buttonWidth, buttonHeight) then
|
|
hovered = hoverindex;
|
|
gfx.FillColor(255,128,0);
|
|
end
|
|
gfx.Rect(rx - buttonBorder,
|
|
ty - buttonBorder,
|
|
buttonWidth + (buttonBorder * 2),
|
|
buttonHeight + (buttonBorder * 2));
|
|
gfx.Fill();
|
|
gfx.BeginPath();
|
|
gfx.FillColor(r,g,b);
|
|
gfx.Rect(rx, ty, buttonWidth, buttonHeight);
|
|
gfx.Fill();
|
|
gfx.BeginPath();
|
|
gfx.FillColor(255,255,255);
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER + gfx.TEXT_ALIGN_MIDDLE);
|
|
gfx.FontSize(40);
|
|
gfx.Text(name, x, y);
|
|
end;
|
|
|
|
draw_checkbox = function(text, x, y, hoverindex, current, can_click)
|
|
local rx = x - (buttonWidth / 2);
|
|
local ty = y - (buttonHeight / 2);
|
|
gfx.BeginPath();
|
|
|
|
if can_click then
|
|
gfx.FillColor(255,255,255);
|
|
else
|
|
gfx.FillColor(150,100,100);
|
|
end
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER + gfx.TEXT_ALIGN_MIDDLE);
|
|
gfx.FontSize(35);
|
|
gfx.Text(text, x, y)
|
|
|
|
local xmin,ymin,xmax,ymax = gfx.TextBounds(x, y, text);
|
|
|
|
local sx = xmin - 40;
|
|
local sy = y - 15;
|
|
|
|
gfx.StrokeColor(0,128,255);
|
|
if can_click and (mouse_clipped(sx, sy, 31, 30) or mouse_clipped(xmin-10, ymin, xmax-xmin, ymax-ymin)) then
|
|
hovered = hoverindex;
|
|
gfx.StrokeColor(255,128,0);
|
|
end
|
|
|
|
gfx.Rect(sx, y - 15, 30, 30)
|
|
gfx.StrokeWidth(2)
|
|
gfx.Stroke()
|
|
|
|
if current then
|
|
-- Draw checkmark
|
|
gfx.BeginPath();
|
|
gfx.MoveTo(sx+5, sy+10);
|
|
gfx.LineTo(sx+15, y+5);
|
|
gfx.LineTo(sx+35, y-15);
|
|
gfx.StrokeWidth(5)
|
|
gfx.StrokeColor(0,255,0);
|
|
gfx.Stroke()
|
|
|
|
end
|
|
end;
|
|
|
|
local userHeight = 100
|
|
|
|
draw_user = function(user, x, y, buttonWidth, rank)
|
|
local buttonHeight = userHeight;
|
|
local rx = x - (buttonWidth / 2);
|
|
local ty = y - (buttonHeight / 2);
|
|
gfx.BeginPath();
|
|
gfx.FillColor(256,128,255);
|
|
|
|
gfx.Rect(rx - buttonBorder,
|
|
ty - buttonBorder,
|
|
buttonWidth + (buttonBorder * 2),
|
|
buttonHeight + (buttonBorder * 2));
|
|
gfx.Fill();
|
|
gfx.BeginPath();
|
|
if host == user.id then
|
|
gfx.FillColor(80,0,0);
|
|
else
|
|
gfx.FillColor(0,0,40);
|
|
end
|
|
gfx.Rect(rx, ty, buttonWidth, buttonHeight);
|
|
gfx.Fill();
|
|
gfx.BeginPath();
|
|
gfx.FillColor(255,255,255);
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_LEFT + gfx.TEXT_ALIGN_MIDDLE);
|
|
gfx.FontSize(40);
|
|
local name = user.name;
|
|
if user.id == user_id then
|
|
name = name
|
|
end
|
|
if user.id == host then
|
|
name = name..' (host)'
|
|
elseif user.missing_map then
|
|
name = name..' (NO CHART)'
|
|
elseif user.ready then
|
|
name = name..' (ready)'
|
|
end
|
|
if user.score ~= nil then
|
|
name = '#'..rank..' '..name
|
|
end
|
|
first_y = y - 28
|
|
second_y = y + 28
|
|
gfx.Text(name, x - buttonWidth/2 + 5, first_y);
|
|
if user.score ~= nil then
|
|
gfx.FillColor(255,255,0)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_RIGHT + gfx.TEXT_ALIGN_MIDDLE);
|
|
local combo_text = ' '..user.combo..'x'
|
|
gfx.Text(combo_text, x+buttonWidth/2 - 5, second_y-5);
|
|
local xmin,ymin,xmax,ymax = gfx.TextBounds(x+buttonWidth/2 - 5, second_y-5, combo_text);
|
|
|
|
|
|
local score_text = ' '..string.format("%08d",user.score);
|
|
gfx.FillColor(255,255,255)
|
|
gfx.Text(score_text, xmin, second_y-5);
|
|
xmin,ymin,xmax,ymax = gfx.TextBounds(xmin, second_y-5, score_text);
|
|
|
|
if user.grade == nil then
|
|
for i,v in ipairs(grades) do
|
|
if v.max > user.score then
|
|
user.grade = v.image
|
|
break
|
|
end
|
|
end
|
|
end
|
|
if user.badge == nil and user.clear > 1 then
|
|
user.badge = badges[user.clear];
|
|
end
|
|
|
|
gfx.BeginPath()
|
|
local iw, ih = gfx.ImageSize(user.grade)
|
|
local iar = iw/ih
|
|
local grade_height = buttonHeight/2 - 10
|
|
gfx.ImageRect(xmin - iar * grade_height, second_y - buttonHeight/4 , iar * grade_height, grade_height, user.grade, 1, 0)
|
|
end
|
|
if user.level ~= 0 then
|
|
gfx.FillColor(255,255,0)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_LEFT + gfx.TEXT_ALIGN_MIDDLE);
|
|
local level_text = 'Lvl '..user.level..' '
|
|
gfx.Text(level_text, x-buttonWidth/2 + 5, second_y-5)
|
|
local xmin,ymin,xmax,ymax = gfx.TextBounds(x-buttonWidth/2 + 5, second_y-5, level_text);
|
|
|
|
|
|
if user.badge then
|
|
gfx.BeginPath()
|
|
local iw, ih = gfx.ImageSize(user.badge)
|
|
local iar = iw/ih;
|
|
local badge_height = buttonHeight/2 - 10
|
|
gfx.ImageRect(xmax+5, second_y - buttonHeight/4 , iar * badge_height, badge_height, user.badge, 1, 0)
|
|
|
|
|
|
end
|
|
end
|
|
end;
|
|
|
|
function render_loading()
|
|
if not loading then return end
|
|
gfx.Save()
|
|
gfx.ResetTransform()
|
|
gfx.BeginPath()
|
|
gfx.MoveTo(resX, resY)
|
|
gfx.LineTo(resX - 350, resY)
|
|
gfx.LineTo(resX - 300, resY - 50)
|
|
gfx.LineTo(resX, resY - 50)
|
|
gfx.ClosePath()
|
|
gfx.FillColor(33,33,33)
|
|
gfx.Fill()
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_RIGHT, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text("LOADING...", resX - 20, resY - 3)
|
|
gfx.Restore()
|
|
end
|
|
|
|
function render_info()
|
|
gfx.Save()
|
|
gfx.ResetTransform()
|
|
gfx.BeginPath()
|
|
gfx.MoveTo(0, resY)
|
|
gfx.LineTo(550, resY)
|
|
gfx.LineTo(500, resY - 65)
|
|
gfx.LineTo(0, resY - 65)
|
|
gfx.ClosePath()
|
|
gfx.FillColor(33,33,33)
|
|
gfx.Fill()
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_LEFT, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text("Multiplayer", 3, resY - 15)
|
|
local xmin,ymin,xmax,ymax = gfx.TextBounds(3, resY - 3, "Multiplayer")
|
|
gfx.FontSize(20)
|
|
gfx.Text(MULTIPLAYER_VERSION, xmax + 13, resY - 15)
|
|
--gfx.Text('Server: '..'', xmax + 13, resY - 15)
|
|
draw_button_color("Settings", 500-60, resY-25, 120, function()
|
|
mpScreen.OpenSettings();
|
|
end, 33,33,33, 33,33,33)
|
|
gfx.Restore()
|
|
|
|
if searchStatus then
|
|
gfx.BeginPath()
|
|
gfx.FillColor(255,255,255)
|
|
gfx.FontSize(20);
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_LEFT + gfx.TEXT_ALIGN_TOP)
|
|
gfx.Text(searchStatus, 3, 3)
|
|
end
|
|
|
|
|
|
|
|
|
|
end
|
|
|
|
draw_diff_icon = function(diff, x, y, w, h, selected)
|
|
local shrinkX = w/4
|
|
local shrinkY = h/4
|
|
if selected then
|
|
gfx.FontSize(h/2)
|
|
shrinkX = w/6
|
|
shrinkY = h/6
|
|
else
|
|
gfx.FontSize(math.floor(h / 3))
|
|
end
|
|
gfx.BeginPath()
|
|
gfx.RoundedRectVarying(x+shrinkX,y+shrinkY,w-shrinkX*2,h-shrinkY*2,0,0,0,0)
|
|
gfx.FillColor(15,15,15)
|
|
gfx.StrokeColor(table.unpack(diffColors[diff.difficulty + 1]))
|
|
gfx.StrokeWidth(2)
|
|
gfx.Fill()
|
|
gfx.Stroke()
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_MIDDLE + gfx.TEXT_ALIGN_CENTER)
|
|
gfx.FastText(tostring(diff.level), x+(w/2),y+(h/2))
|
|
end
|
|
|
|
local doffset = 0;
|
|
local timer = 0;
|
|
|
|
draw_cursor = function(x,y,rotation,width)
|
|
gfx.Save()
|
|
gfx.BeginPath();
|
|
gfx.Translate(x,y)
|
|
gfx.Rotate(rotation)
|
|
gfx.StrokeColor(255,128,0)
|
|
gfx.StrokeWidth(4)
|
|
gfx.Rect(-width/2, -width/2, width, width)
|
|
gfx.Stroke()
|
|
gfx.Restore()
|
|
end
|
|
|
|
draw_diffs = function(diffs, x, y, w, h, selectedDiff)
|
|
local diffWidth = w/2.5
|
|
local diffHeight = w/2.5
|
|
local diffCount = #diffs
|
|
gfx.Scissor(x,y,w,h)
|
|
for i = math.max(selectedDiff - 2, 1), math.max(selectedDiff - 1,1) do
|
|
local diff = diffs[i]
|
|
local xpos = x + ((w/2 - diffWidth/2) + (selectedDiff - i + doffset)*(-0.8*diffWidth))
|
|
if i ~= selectedDiff then
|
|
draw_diff_icon(diff, xpos, y, diffWidth, diffHeight, false)
|
|
end
|
|
end
|
|
|
|
--after selected
|
|
for i = math.min(selectedDiff + 2, diffCount), selectedDiff + 1,-1 do
|
|
local diff = diffs[i]
|
|
local xpos = x + ((w/2 - diffWidth/2) + (selectedDiff - i + doffset)*(-0.8*diffWidth))
|
|
if i ~= selectedDiff then
|
|
draw_diff_icon(diff, xpos, y, diffWidth, diffHeight, false)
|
|
end
|
|
end
|
|
local diff = diffs[selectedDiff]
|
|
local xpos = x + ((w/2 - diffWidth/2) + (doffset)*(-0.8*diffWidth))
|
|
draw_diff_icon(diff, xpos, y, diffWidth, diffHeight, true)
|
|
gfx.BeginPath()
|
|
gfx.FillColor(0,128,255)
|
|
gfx.Rect(x,y+10,2,diffHeight-h/6)
|
|
gfx.Fill()
|
|
gfx.BeginPath()
|
|
gfx.Rect(x+w-2,y+10,2,diffHeight-h/6)
|
|
gfx.Fill()
|
|
gfx.ResetScissor()
|
|
draw_cursor(x + w/2, y +diffHeight/2, timer * math.pi, diffHeight / 1.5)
|
|
end
|
|
|
|
set_diff = function(oldDiff, newDiff)
|
|
game.PlaySample("click-02")
|
|
doffset = doffset + oldDiff - newDiff
|
|
end;
|
|
|
|
local selected_room_index = 1;
|
|
local ioffset = 0;
|
|
|
|
function draw_rooms(y, h)
|
|
if #rooms == 0 then
|
|
return
|
|
end
|
|
local num_rooms_visible = math.floor(h / (buttonHeight + 10))
|
|
|
|
local first_half_rooms = math.floor(num_rooms_visible/2)
|
|
local second_half_rooms = math.ceil(num_rooms_visible/2) - 1
|
|
|
|
local start_offset = math.max(selected_room_index - first_half_rooms, 1);
|
|
local end_offset = math.min(selected_room_index + second_half_rooms + 2, #rooms);
|
|
|
|
local start_index_offset = 1;
|
|
|
|
-- If our selection is near the start or end we have to offset
|
|
if selected_room_index <= first_half_rooms then
|
|
start_index_offset = 0;
|
|
end_offset = math.min(#rooms, num_rooms_visible + 1)
|
|
end
|
|
if selected_room_index >= #rooms - second_half_rooms then
|
|
start_offset = math.max(1, #rooms - num_rooms_visible)
|
|
end_offset = #rooms
|
|
end
|
|
|
|
for i = start_offset, end_offset do
|
|
local room = rooms[i];
|
|
-- if selected room < halfvis then we start at 1
|
|
-- if sel > #rooms - halfvis then we start at -halfvis
|
|
local offset_index = (start_offset + first_half_rooms) - i + start_index_offset
|
|
|
|
local offsetY = (offset_index + ioffset) * (buttonHeight + 10);
|
|
local ypos = y + (h/2) - offsetY;
|
|
local status = room.current..'/'..room.max
|
|
if room.ingame then
|
|
status = status..' (In Game)'
|
|
end
|
|
if room.password then
|
|
status = status..' <P>'
|
|
end
|
|
draw_room(room.name .. ': '.. status, resX / 2, ypos, i == selected_room_index, function()
|
|
join_room(room)
|
|
end)
|
|
end
|
|
end
|
|
|
|
change_selected_room = function(off)
|
|
|
|
local new_index = selected_room_index + off;
|
|
--selected_room_index = 2;
|
|
if new_index < 1 or new_index > #rooms then
|
|
return;
|
|
end
|
|
|
|
local h = resY - 290;
|
|
|
|
local num_rooms_visible = math.floor(h / (buttonHeight + 10))
|
|
|
|
local first_half_rooms = math.floor(num_rooms_visible/2)
|
|
local second_half_rooms = math.ceil(num_rooms_visible/2) - 1
|
|
|
|
if off > 0 and (selected_room_index < first_half_rooms or selected_room_index >= #rooms - second_half_rooms - 1) then
|
|
elseif off < 0 and (selected_room_index <= first_half_rooms or selected_room_index >= #rooms - second_half_rooms) then
|
|
else
|
|
ioffset = ioffset - new_index + selected_room_index;
|
|
end
|
|
|
|
game.PlaySample("menu_click")
|
|
|
|
selected_room_index = new_index;
|
|
end
|
|
|
|
function render_lobby(deltaTime)
|
|
|
|
local jacket_size;
|
|
|
|
-- split is how the screen is split or not
|
|
if portrait then
|
|
split = resX
|
|
jacket_size = math.min(resX/3, resY/3);
|
|
user_y_off = 375+jacket_size + 70
|
|
song_x_off = 0;
|
|
else
|
|
split = resX/2
|
|
jacket_size = math.min(resX/2, resY/2);
|
|
user_y_off = 0
|
|
song_x_off = 1/2*resX;
|
|
end
|
|
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text(selected_room.name, resX/2, 50)
|
|
|
|
-- === Users ===
|
|
|
|
gfx.Text("Users", split/2, user_y_off+100)
|
|
|
|
buttonY = user_y_off + 125 + userHeight/2
|
|
for i, user in ipairs(lobby_users) do
|
|
draw_user(user, split/2, buttonY, split*3/4, i)
|
|
local side_button_off = 0
|
|
if owner == user_id and user.id ~= user_id then
|
|
draw_button("K",split/2 + split*3/8+10+25, buttonY, 50, function()
|
|
kick_user(user);
|
|
end)
|
|
side_button_off = 60;
|
|
end
|
|
if (owner == user_id or host == user_id) and user.id ~= host then
|
|
draw_button("H",split/2 + split*3/8+10+25+side_button_off, buttonY, 50, function()
|
|
change_host(user);
|
|
end)
|
|
end
|
|
buttonY = buttonY + userHeight
|
|
end
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER + gfx.TEXT_ALIGN_MIDDLE);
|
|
gfx.FillColor(255,255,255)
|
|
|
|
|
|
-- === song select ===
|
|
|
|
gfx.FontSize(60)
|
|
gfx.Text("Selected Song:", split/2 + song_x_off, 100)
|
|
gfx.FontSize(40)
|
|
if selected_song == nil then
|
|
if host == user_id then
|
|
gfx.Text("Select song:", split/2 + song_x_off, 175)
|
|
else
|
|
if missing_song then
|
|
gfx.Text("Missing song!!!!", split/2 + song_x_off, 175)
|
|
else
|
|
gfx.Text("Host is selecting song", split/2 + song_x_off, 175)
|
|
end
|
|
end
|
|
if jacket == 0 then
|
|
jacket = placeholderJacket
|
|
end
|
|
else
|
|
gfx.Text(selected_song.title, split/2 + song_x_off, 175)
|
|
draw_diffs(selected_song.all_difficulties, split/2 + song_x_off - 150, 200, 300, 100, selected_song.diff_index+1)
|
|
|
|
if selected_song.jacket == nil or selected_song.jacket == placeholderJacket then
|
|
selected_song.jacket = gfx.LoadImageJob(selected_song.jacketPath, placeholderJacket)
|
|
jacket = selected_song.jacket
|
|
end
|
|
end
|
|
gfx.Save()
|
|
gfx.BeginPath()
|
|
gfx.Translate(split/2 + song_x_off, 325+jacket_size/2)
|
|
gfx.ImageRect(-jacket_size/2,-jacket_size/2,jacket_size,jacket_size,jacket,1,0)
|
|
|
|
if mouse_clipped(split/2 + song_x_off-jacket_size/2, 325, jacket_size,jacket_size) and host == user_id then
|
|
hovered = function()
|
|
missing_song = false
|
|
mpScreen.SelectSong()
|
|
end
|
|
end
|
|
gfx.Restore()
|
|
if start_game_soon then
|
|
draw_button("Game starting...", split/2 + song_x_off, 375+jacket_size, 600, function() end);
|
|
else
|
|
if host == user_id then
|
|
if selected_song == nil or not selected_song.self_picked then
|
|
draw_button_color("Select song", split/2 + song_x_off, 375+jacket_size, 600, function()
|
|
missing_song = false
|
|
mpScreen.SelectSong()
|
|
end, 0, math.min(255, 128 + math.floor(32 * math.cos(timer * math.pi))), 0, 0,128,255);
|
|
elseif user_ready and all_ready then
|
|
draw_button("Start game", split/2 + song_x_off, 375+jacket_size, 600, start_game)
|
|
elseif user_ready and not all_ready then
|
|
draw_button("Waiting for others", split/2 + song_x_off, 375+jacket_size, 600, function()
|
|
missing_song = false
|
|
mpScreen.SelectSong()
|
|
end)
|
|
else
|
|
draw_button("Ready", split/2 + song_x_off, 375+jacket_size, 600, ready_up);
|
|
end
|
|
elseif host == nil then
|
|
draw_button("Waiting for game to end", split/2 + song_x_off, 375+jacket_size, 600, function() end);
|
|
elseif missing_song then
|
|
draw_button("Missing Song!", split/2 + song_x_off, 375+jacket_size, 600, function() end);
|
|
elseif selected_song ~= nil then
|
|
if user_ready then
|
|
draw_button("Cancel", split/2 + song_x_off, 375+jacket_size, 600, ready_up);
|
|
else
|
|
draw_button("Ready", split/2 + song_x_off, 375+jacket_size, 600, ready_up);
|
|
end
|
|
else
|
|
draw_button("Waiting for host", split/2 + song_x_off, 375+jacket_size, 600, function() end);
|
|
end
|
|
end
|
|
|
|
draw_checkbox("Excessive", split/2 + song_x_off - 150, 375+jacket_size + 70, toggle_hard, hard_mode, not start_game_soon)
|
|
draw_checkbox("Mirror", split/2 + song_x_off, 375+jacket_size + 70, toggle_mirror, mirror_mode, not start_game_soon)
|
|
|
|
draw_checkbox("Rotate Host", split/2 + song_x_off + 150 + 20, 375+jacket_size + 70, toggle_rotate, do_rotate,
|
|
(owner == user_id or host == user_id) and not start_game_soon)
|
|
|
|
if selected_song ~= nil then
|
|
gfx.FillColor(255,255,255)
|
|
gfx.FontSize(20);
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER + gfx.TEXT_ALIGN_TOP)
|
|
if selected_song.min_bpm ~= selected_song.max_bpm then
|
|
gfx.Text(string.format("BPM: %.0f-%.0f, Start BPM: %.0f, Hispeed: %.0f x %.1f = %.0f",
|
|
selected_song.min_bpm, selected_song.max_bpm, selected_song.start_bpm,
|
|
selected_song.speed_bpm, selected_song.hispeed, selected_song.speed_bpm * selected_song.hispeed),
|
|
split/2 + song_x_off, 375+jacket_size + 70 + 30)
|
|
else
|
|
gfx.Text(string.format("BPM: %.0f, Hispeed: %.0f x %.1f = %.0f",
|
|
selected_song.min_bpm,
|
|
selected_song.speed_bpm, selected_song.hispeed, selected_song.speed_bpm * selected_song.hispeed),
|
|
split/2 + song_x_off, 375+jacket_size + 70 + 30)
|
|
end
|
|
end
|
|
end
|
|
|
|
function render_room_list(deltaTime)
|
|
draw_rooms(175, resY - 290);
|
|
|
|
-- Draw cover for rooms out of view
|
|
gfx.BeginPath()
|
|
gfx.FillColor(20, 20, 20)
|
|
gfx.Rect(0, 0, resX, 145)
|
|
gfx.Rect(0, resY-170, resX, 170)
|
|
gfx.Fill()
|
|
|
|
gfx.BeginPath()
|
|
gfx.FillColor(60, 60, 60)
|
|
gfx.Rect(0, 145, resX, 2)
|
|
gfx.Rect(0, resY-170-2, resX, 2)
|
|
gfx.Fill()
|
|
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text("Multiplayer Rooms", resX/2, 100)
|
|
|
|
|
|
if not loading then
|
|
draw_button("Create new room", resX/2, resY-40-buttonHeight, resX*(3/4), new_room);
|
|
end
|
|
end
|
|
|
|
|
|
passwordErrorOffset = 0;
|
|
function render_password_screen(deltaTime)
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text("Joining "..selected_room.name.."...", resX/2, resY/4)
|
|
|
|
gfx.FillColor(50,50,50)
|
|
gfx.BeginPath()
|
|
gfx.Rect(0, resY/2-10, resX, 40)
|
|
gfx.Fill();
|
|
|
|
gfx.FillColor(255,255,255)
|
|
gfx.Text("Please enter room password:", resX/2, resY/2-40)
|
|
gfx.Text(string.rep("*",#textInput.text), resX/2, resY/2+40)
|
|
if passwordError then
|
|
|
|
gfx.FillColor(255,50,50)
|
|
gfx.FontSize(60 + math.floor(passwordErrorOffset*20))
|
|
gfx.Text("Invalid password", resX/2, resY/2+80)
|
|
end
|
|
draw_button("Join", resX/2, resY*3/4, resX/2, mpScreen.JoinWithPassword);
|
|
end
|
|
|
|
function render_new_room_password(delta_time)
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text("Create New Room", resX/2, resY/4)
|
|
|
|
gfx.FillColor(50,50,50)
|
|
gfx.BeginPath()
|
|
gfx.Rect(0, resY/2-10, resX, 40)
|
|
gfx.Fill();
|
|
|
|
gfx.FillColor(255,255,255)
|
|
gfx.Text("Enter room password:", resX/2, resY/2-40)
|
|
gfx.Text(string.rep("*",#textInput.text), resX/2, resY/2+40)
|
|
draw_button("Create Room", resX/2, resY*3/4, resX/2, mpScreen.NewRoomStep);
|
|
end
|
|
|
|
function render_new_room_name(deltaTime)
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text("Create New Room", resX/2, resY/4)
|
|
|
|
gfx.FillColor(50,50,50)
|
|
gfx.BeginPath()
|
|
gfx.Rect(0, resY/2-10, resX, 60)
|
|
gfx.Fill();
|
|
|
|
gfx.FillColor(255,255,255)
|
|
gfx.Text("Please enter room name:", resX/2, resY/2-40)
|
|
gfx.Text(textInput.text, resX/2, resY/2+40)
|
|
draw_button("Next", resX/2, resY*3/4, resX/2, mpScreen.NewRoomStep);
|
|
end
|
|
|
|
function render_set_username(deltaTime)
|
|
gfx.FillColor(255,255,255)
|
|
gfx.TextAlign(gfx.TEXT_ALIGN_CENTER, gfx.TEXT_ALIGN_BOTTOM)
|
|
gfx.FontSize(70)
|
|
gfx.Text("First things first...", resX/2, resY/4)
|
|
|
|
gfx.FillColor(50,50,50)
|
|
gfx.BeginPath()
|
|
gfx.Rect(0, resY/2-10, resX, 60)
|
|
gfx.Fill();
|
|
|
|
gfx.FillColor(255,255,255)
|
|
gfx.Text("Enter a display name:", resX/2, resY/2-40)
|
|
gfx.Text(textInput.text, resX/2, resY/2+40)
|
|
draw_button("Join Multiplayer", resX/2, resY*3/4, resX/2, function()
|
|
loading = true;
|
|
mpScreen.SaveUsername()
|
|
end);
|
|
|
|
end
|
|
|
|
render = function(deltaTime)
|
|
resX,resY = game.GetResolution();
|
|
buttonWidth = resX*(3/4);
|
|
mposx,mposy = game.GetMousePos();
|
|
portrait = resY > resX
|
|
|
|
sound.stopMusic();
|
|
|
|
doffset = doffset * 0.9
|
|
ioffset = ioffset * 0.9
|
|
passwordErrorOffset = passwordErrorOffset * 0.9
|
|
timer = (timer + deltaTime)
|
|
timer = timer % 2
|
|
|
|
hovered = nil;
|
|
|
|
gfx.LoadSkinFont(normal_font);
|
|
|
|
do_sounds(deltaTime);
|
|
|
|
-- Room Listing View
|
|
if screenState == "inRoom" then
|
|
render_lobby(deltaTime);
|
|
elseif screenState == "roomList" then
|
|
render_room_list(deltaTime);
|
|
elseif screenState == "passwordScreen" then
|
|
render_password_screen(deltaTime);
|
|
elseif screenState == "newRoomName" then
|
|
render_new_room_name()
|
|
elseif screenState == "newRoomPassword" then
|
|
render_new_room_password()
|
|
elseif screenState == "setUsername" then
|
|
loading = false;
|
|
render_set_username()
|
|
end
|
|
render_loading();
|
|
render_info();
|
|
|
|
end
|
|
|
|
-- Ready up to play
|
|
function ready_up()
|
|
Tcp.SendLine(json.encode({topic="user.ready.toggle"}))
|
|
end
|
|
|
|
-- Toggle hard gauage
|
|
function toggle_hard()
|
|
Tcp.SendLine(json.encode({topic="user.hard.toggle"}))
|
|
end
|
|
|
|
-- Toggle hard gauage
|
|
function toggle_mirror()
|
|
Tcp.SendLine(json.encode({topic="user.mirror.toggle"}))
|
|
end
|
|
|
|
function new_room()
|
|
host = user_id
|
|
owner = user_id
|
|
mpScreen.NewRoomStep()
|
|
end
|
|
|
|
-- Toggle host rotation
|
|
function toggle_rotate()
|
|
Tcp.SendLine(json.encode({topic="room.option.rotation.toggle"}))
|
|
end
|
|
|
|
-- Change lobby host
|
|
function change_host(user)
|
|
Tcp.SendLine(json.encode({topic="room.host.set", host=user.id}))
|
|
end
|
|
|
|
-- Kick user
|
|
function kick_user(user)
|
|
Tcp.SendLine(json.encode({topic="room.kick", id=user.id}))
|
|
end
|
|
|
|
-- Tell the server to start the game
|
|
function start_game()
|
|
selected_song.self_picked = false
|
|
if (selected_song == nil) then
|
|
return
|
|
end
|
|
if (start_game_soon) then
|
|
return
|
|
end
|
|
|
|
Tcp.SendLine(json.encode({topic="room.game.start"}))
|
|
end
|
|
|
|
-- Join a given room
|
|
function join_room(room)
|
|
host = user_id
|
|
selected_room = room;
|
|
if room.password then
|
|
mpScreen.JoinWithPassword(room.id)
|
|
else
|
|
mpScreen.JoinWithoutPassword(room.id)
|
|
end
|
|
end
|
|
|
|
-- Handle button presses to advance the UI
|
|
button_pressed = function(button)
|
|
if button == game.BUTTON_STA then
|
|
if start_game_soon then
|
|
return
|
|
end
|
|
if screenState == "roomList" then
|
|
if #rooms == 0 then
|
|
new_room()
|
|
else
|
|
-- TODO navigate room selection
|
|
join_room(rooms[selected_room_index])
|
|
end
|
|
elseif screenState == "inRoom" then
|
|
if host == user_id then
|
|
if selected_song and selected_song.self_picked then
|
|
if all_ready then
|
|
start_game()
|
|
else
|
|
missing_song = false
|
|
mpScreen.SelectSong()
|
|
end
|
|
else
|
|
missing_song = false
|
|
mpScreen.SelectSong()
|
|
end
|
|
else
|
|
ready_up()
|
|
end
|
|
end
|
|
end
|
|
|
|
if button == game.BUTTON_FXL then
|
|
toggle_hard();
|
|
end
|
|
if button == game.BUTTON_FXR then
|
|
toggle_mirror();
|
|
end
|
|
end
|
|
|
|
-- Handle the escape key around the UI
|
|
function key_pressed(key)
|
|
if key == 27 then --escape pressed
|
|
if screenState == "roomList" then
|
|
did_exit = true;
|
|
mpScreen.Exit();
|
|
return
|
|
end
|
|
|
|
-- Reset room data
|
|
screenState = "roomList" -- have to update here
|
|
selected_room = nil;
|
|
rooms = {};
|
|
selected_song = nil
|
|
selected_song_index = 1;
|
|
jacket = 0;
|
|
end
|
|
|
|
end
|
|
|
|
-- Handle mouse clicks in the UI
|
|
mouse_pressed = function(button)
|
|
if hovered then
|
|
hovered()
|
|
end
|
|
return 0
|
|
end
|
|
|
|
function init_tcp()
|
|
Tcp.SetTopicHandler("server.info", function(data)
|
|
loading = false
|
|
user_id = data.userid
|
|
end)
|
|
-- Update the list of rooms as well as get user_id for the client
|
|
Tcp.SetTopicHandler("server.rooms", function(data)
|
|
|
|
rooms = {}
|
|
for i, room in ipairs(data.rooms) do
|
|
table.insert(rooms, room)
|
|
end
|
|
end)
|
|
|
|
Tcp.SetTopicHandler("server.room.joined", function(data)
|
|
selected_room = data.room
|
|
end)
|
|
|
|
local sound_time = 0;
|
|
local sound_clip = nil;
|
|
local sounds_left = 0;
|
|
local sound_interval = 0;
|
|
|
|
function repeat_sound(clip, times, interval)
|
|
sound_clip = clip;
|
|
sound_time = 0;
|
|
sounds_left = times - 1;
|
|
sound_interval = interval;
|
|
game.PlaySample(clip)
|
|
end
|
|
|
|
function do_sounds(deltaTime)
|
|
if sound_clip == nil then
|
|
return
|
|
end
|
|
|
|
sound_time = sound_time + deltaTime;
|
|
if sound_time > sound_interval then
|
|
sound_time = sound_time - sound_interval;
|
|
game.PlaySample(sound_clip);
|
|
sounds_left = sounds_left - 1
|
|
if sounds_left <= 0 then
|
|
sound_clip = nil
|
|
end
|
|
end
|
|
end
|
|
|
|
local last_song = nil
|
|
|
|
-- Update the current lobby
|
|
Tcp.SetTopicHandler("room.update", function(data)
|
|
-- Update the users in the lobby
|
|
lobby_users = {}
|
|
local prev_all_ready = all_ready;
|
|
all_ready = true
|
|
for i, user in ipairs(data.users) do
|
|
table.insert(lobby_users, user)
|
|
if user.id == user_id then
|
|
user_ready = user.ready
|
|
end
|
|
if not user.ready then
|
|
all_ready = false
|
|
end
|
|
end
|
|
|
|
if user_id == host and #data.users > 1 and all_ready and not prev_all_ready then
|
|
repeat_sound("click-02", 3, .1)
|
|
end
|
|
|
|
if data.host == user_id and host ~= user_id then
|
|
repeat_sound("click-02", 3, .1)
|
|
end
|
|
|
|
if data.song ~=nil and last_song ~=data.song then
|
|
game.PlaySample("menu_click")
|
|
last_song = data.song
|
|
end
|
|
host = data.host
|
|
if data.owner then
|
|
owner = data.owner
|
|
else
|
|
owner = host
|
|
end
|
|
hard_mode = data.hard_mode
|
|
mirror_mode = data.mirror_mode
|
|
do_rotate = data.do_rotate
|
|
if data.start_soon and not start_game_soon then
|
|
repeat_sound("click-01", 5, 1)
|
|
end
|
|
start_game_soon = data.start_soon
|
|
|
|
end)
|
|
end
|