#include #include extern bool gameRunning; extern Menu *currentMenu; static Menu pauseMenu; static Menu optionsMenu; static Menu controlsMenu; void Menu::gotoParent(void) { if (!parent) { currentMenu = nullptr; game::config::update(); } else { currentMenu = parent; } } inline void segFault() { (*((int *)NULL))++; } std::string& deleteWord(std::string& s) { while (s.back() != ' ') s.pop_back(); return s; } std::string sym2str(const SDL_Keycode& c) { std::string s = ""; switch (c) { case SDLK_UP : s = "UP" ; break; case SDLK_DOWN : s = "DOWN" ; break; case SDLK_LEFT : s = "LEFT" ; break; case SDLK_RIGHT : s = "RIGHT" ; break; case SDLK_LSHIFT : s = "LSHIFT" ; break; case SDLK_RSHIFT : s = "RSHIFT" ; break; case SDLK_LALT : s = "LALT" ; break; case SDLK_RALT : s = "RALT" ; break; case SDLK_LCTRL : s = "LCONTROL"; break; case SDLK_RCTRL : s = "RCONTROL"; break; case SDLK_TAB : s = "TAB" ; break; default : s += static_cast(c); break; } return s; } void initControls(Menu *m) { auto cfg = readFileA("config/controls.dat"); unsigned i = 0; SDL_Keycode z; for (const auto &l : cfg) { z = static_cast(std::stoi(l)); setControl(i, z); m->items[i++].button.text += sym2str(z); } } void saveControls(void) { std::ofstream out ("config/controls.dat"); SDL_Keycode q = 1; unsigned int i = 0; while ((q = getControl(i++)) != 0) { auto d = std::to_string(q) + '\n'; out.write(d.data(), d.size()); } out.close(); } void setControlF(unsigned int index, menuItem &m) { SDL_Event e; do SDL_WaitEvent(&e); while (e.type != SDL_KEYDOWN); setControl(index, e.key.keysym.sym); deleteWord(m.button.text); m.button.text += sym2str(e.key.keysym.sym); saveControls(); } namespace ui { namespace menu { menuItem createButton(vec2 l, dim2 d, Color c, const char* t, menuFunc f) { menuItem temp; temp.member = 0; temp.button.loc = l; temp.button.dim = d; temp.button.color = c; temp.button.text = t; temp.button.func = f; temp.child = nullptr; return temp; } menuItem createChildButton(vec2 l, dim2 d, Color c, const char* t, Menu *_child) { menuItem temp; temp.member = -1; temp.button.loc = l; temp.button.dim = d; temp.button.color = c; temp.button.text = t; temp.button.func = NULL; temp.child = _child; return temp; } menuItem createParentButton(vec2 l, dim2 d, Color c, const char* t) { menuItem temp; temp.member = -2; temp.button.loc = l; temp.button.dim = d; temp.button.color = c; temp.button.text = t; temp.button.func = NULL; temp.child = nullptr; return temp; } menuItem createSlider(vec2 l, dim2 d, Color c, float min, float max, const char* t, float* v) { menuItem temp; temp.member = 1; temp.slider.loc = l; temp.slider.dim = d; temp.slider.color = c; temp.slider.minValue = min; temp.slider.maxValue = max; temp.slider.text = t; temp.slider.var = v; temp.slider.sliderLoc = *v; temp.child = nullptr; return temp; } void init(void) { // Create the main pause menu pauseMenu.items.push_back(ui::menu::createParentButton({-128,0},{256,75},{0.0f,0.0f,0.0f}, "Resume")); pauseMenu.items.push_back(ui::menu::createChildButton({-128,-100},{256,75},{0.0f,0.0f,0.0f}, "Options", &optionsMenu)); pauseMenu.items.push_back(ui::menu::createChildButton({-128,-200},{256,75},{0.0f,0.0f,0.0f}, "Controls", &controlsMenu)); pauseMenu.items.push_back(ui::menu::createButton({-128,-300},{256,75},{0.0f,0.0f,0.0f}, "Save and Quit", ui::quitGame)); pauseMenu.items.push_back(ui::menu::createButton({-128,-400},{256,75},{0.0f,0.0f,0.0f}, "Segfault", segFault)); // Create the options (sound) menu optionsMenu.items.push_back(ui::menu::createSlider({0-static_cast(game::SCREEN_WIDTH)/4,0-(512/2)}, {50,512}, {0.0f, 0.0f, 0.0f}, 0, 100, "Master", &game::config::VOLUME_MASTER)); optionsMenu.items.push_back(ui::menu::createSlider({-200,100}, {512,50}, {0.0f, 0.0f, 0.0f}, 0, 100, "Music", &game::config::VOLUME_MUSIC)); optionsMenu.items.push_back(ui::menu::createSlider({-200,000}, {512,50}, {0.0f, 0.0f, 0.0f}, 0, 100, "SFX", &game::config::VOLUME_SFX)); optionsMenu.parent = &pauseMenu; // Create the controls menu controlsMenu.items.push_back(ui::menu::createButton({-450,300}, {400, 75}, {0.0f, 0.0f, 0.0f}, "Up: ", nullptr)); controlsMenu.items.back().button.func = [](){ setControlF(0, controlsMenu.items[0]); }; controlsMenu.items.push_back(ui::menu::createButton({-450,200}, {400, 75}, {0.0f, 0.0f, 0.0f}, "Left: ", nullptr)); controlsMenu.items.back().button.func = [](){ setControlF(1, controlsMenu.items[1]); }; controlsMenu.items.push_back(ui::menu::createButton({-450,100}, {400, 75}, {0.0f, 0.0f, 0.0f}, "Right: ", nullptr)); controlsMenu.items.back().button.func = [](){ setControlF(2, controlsMenu.items[2]); }; controlsMenu.items.push_back(ui::menu::createButton({-450,0}, {400, 75}, {0.0f, 0.0f, 0.0f}, "Sprint: ", nullptr)); controlsMenu.items.back().button.func = [](){ setControlF(3, controlsMenu.items[3]); }; controlsMenu.items.push_back(ui::menu::createButton({-450,-100}, {400, 75}, {0.0f, 0.0f, 0.0f}, "Creep: ", nullptr)); controlsMenu.items.back().button.func = [](){ setControlF(4, controlsMenu.items[4]); }; controlsMenu.items.push_back(ui::menu::createButton({-450,-200}, {400, 75}, {0.0f, 0.0f, 0.0f}, "Inventory: ", nullptr)); controlsMenu.items.back().button.func = [](){ setControlF(5, controlsMenu.items[5]); }; controlsMenu.parent = &pauseMenu; initControls(&controlsMenu); } void toggle(void) { currentMenu = &pauseMenu; } void draw(void) { auto SCREEN_WIDTH = game::SCREEN_WIDTH; auto SCREEN_HEIGHT = game::SCREEN_HEIGHT; SDL_Event e; useShader(&textShader, &textShader_uniform_texture, &textShader_attribute_coord, &textShader_attribute_tex); setFontSize(24); game::config::update(); setFontZ(-9.0); mouse.x = ui::premouse.x+offset.x-(SCREEN_WIDTH/2); mouse.y = (offset.y+SCREEN_HEIGHT/2)-ui::premouse.y; //custom event polling for menu's so all other events are ignored while(SDL_PollEvent(&e)) { switch (e.type) { case SDL_QUIT: gameRunning = false; return; break; case SDL_MOUSEMOTION: premouse.x=e.motion.x; premouse.y=e.motion.y; break; case SDL_KEYUP: if (SDL_KEY == SDLK_ESCAPE) { currentMenu->gotoParent(); return; } break; default:break; } } static GLuint backTex = Texture::genColor(Color(0, 0, 0, 204)); //static GLuint bsTex = Texture::genColor(Color(30, 30, 30)); static GLuint border = Texture::genColor(Color(245, 245, 245)); GLfloat line_tex[] = {0.0, 0.0, 1.0, 0.0, 1.0, 1.0, 0.0, 1.0, 0.0, 0.0}; //draw the dark transparent background glColor4f(0.0f, 0.0f, 0.0f, .8f); glUseProgram(textShader); glBindTexture(GL_TEXTURE_2D, backTex); drawRect(vec2(offset.x - SCREEN_WIDTH / 2 - 1, offset.y - (SCREEN_HEIGHT / 2)), vec2(offset.x + SCREEN_WIDTH / 2, offset.y + (SCREEN_HEIGHT / 2)), -8.5); glUseProgram(0); //loop through all elements of the menu for (auto &m : currentMenu->items) { //if the menu is any type of button if (m.member == 0 || m.member == -1 || m.member == -2) { //draw the button background GLuint bsTex = Texture::genColor(Color(m.button.color.red,m.button.color.green,m.button.color.blue)); glUseProgram(textShader); glBindTexture(GL_TEXTURE_2D, bsTex); drawRect(vec2(offset.x + m.button.loc.x, offset.y + m.button.loc.y), vec2(offset.x + m.button.loc.x + m.button.dim.x, offset.y + m.button.loc.y + m.button.dim.y), -8.6); //draw the button text putStringCentered(offset.x + m.button.loc.x + (m.button.dim.x/2), (offset.y + m.button.loc.y + (m.button.dim.y/2)) - ui::fontSize/2, m.button.text); //tests if the mouse is over the button if (mouse.x >= offset.x+m.button.loc.x && mouse.x <= offset.x+m.button.loc.x + m.button.dim.x) { if (mouse.y >= offset.y+m.button.loc.y && mouse.y <= offset.y+m.button.loc.y + m.button.dim.y) { //if the mouse if over the button, it draws this white outline glBindTexture(GL_TEXTURE_2D, border); GLfloat verts[] = {offset.x+m.button.loc.x, offset.y+m.button.loc.y, -8.7, offset.x+m.button.loc.x+m.button.dim.x, offset.y+m.button.loc.y, -8.7, offset.x+m.button.loc.x+m.button.dim.x, offset.y+m.button.loc.y+m.button.dim.y, -8.7, offset.x+m.button.loc.x, offset.y+m.button.loc.y+m.button.dim.y, -8.7, offset.x+m.button.loc.x, offset.y+m.button.loc.y, -8.7}; glUseProgram(textShader); glEnableVertexAttribArray(textShader_attribute_coord); glEnableVertexAttribArray(textShader_attribute_tex); glVertexAttribPointer(textShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, verts); glVertexAttribPointer(textShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, line_tex); glDrawArrays(GL_LINE_STRIP, 0, 5); glDisableVertexAttribArray(textShader_attribute_coord); glDisableVertexAttribArray(textShader_attribute_tex); //if the mouse is over the button and clicks if (SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_LEFT)) { switch(m.member) { case 0: //normal button m.button.func(); break; case -1: currentMenu = m.child; break; case -2: currentMenu->gotoParent(); default:break; } } } } //if element is a slider }else if (m.member == 1) { //combining slider text with variable amount char outSV[32]; sprintf(outSV, "%s: %.1f",m.slider.text.c_str(), *m.slider.var); float sliderW, sliderH; if (m.slider.dim.y > m.slider.dim.x) { //width of the slider handle sliderW = m.slider.dim.x; sliderH = m.slider.dim.y * .05; //location of the slider handle m.slider.sliderLoc = m.slider.minValue + (*m.slider.var/m.slider.maxValue)*(m.slider.dim.y-sliderW); }else{ //width of the slider handle sliderW = m.slider.dim.x * .05; sliderH = m.slider.dim.y; //location of the slider handle m.slider.sliderLoc = m.slider.minValue + (*m.slider.var/m.slider.maxValue)*(m.slider.dim.x-sliderW); } //draw the background of the slider glUseProgram(textShader); GLuint bsTex = Texture::genColor(Color(m.slider.color.red, m.slider.color.green, m.slider.color.blue, 175)); GLuint hTex = Texture::genColor(Color(m.slider.color.red, m.slider.color.green, m.slider.color.blue, 255)); glBindTexture(GL_TEXTURE_2D, bsTex); drawRect(vec2(offset.x + m.slider.loc.x, offset.y + m.slider.loc.y), vec2(offset.x + m.slider.loc.x + m.slider.dim.x, offset.y + m.slider.loc.y + m.slider.dim.y), -8.6); //draw the slider handle glBindTexture(GL_TEXTURE_2D, hTex); if (m.slider.dim.y > m.slider.dim.x) { drawRect(vec2(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)), vec2(offset.x+m.slider.loc.x + sliderW, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH), -8.7); //draw the now combined slider text putStringCentered(offset.x + m.slider.loc.x + (m.slider.dim.x/2), (offset.y + m.slider.loc.y + (m.slider.dim.y*1.05)) - ui::fontSize/2, outSV); }else{ drawRect(vec2(offset.x+m.slider.loc.x+m.slider.sliderLoc, offset.y+m.slider.loc.y), vec2(offset.x+m.slider.loc.x + m.slider.sliderLoc + sliderW, offset.y+m.slider.loc.y + sliderH), -8.7); //draw the now combined slider text putStringCentered(offset.x + m.slider.loc.x + (m.slider.dim.x/2), (offset.y + m.slider.loc.y + (m.slider.dim.y/2)) - ui::fontSize/2, outSV); } //test if mouse is inside of the slider's borders if (mouse.x >= offset.x+m.slider.loc.x && mouse.x <= offset.x+m.slider.loc.x + m.slider.dim.x) { if (mouse.y >= offset.y+m.slider.loc.y && mouse.y <= offset.y+m.slider.loc.y + m.slider.dim.y) { //if it is we draw a white border around it glBindTexture(GL_TEXTURE_2D, border); glUseProgram(textShader); glEnableVertexAttribArray(textShader_attribute_coord); glEnableVertexAttribArray(textShader_attribute_tex); GLfloat box_border[] = {offset.x+m.slider.loc.x, offset.y+m.slider.loc.y, -8.8, offset.x+m.slider.loc.x+m.slider.dim.x, offset.y+m.slider.loc.y, -8.8, offset.x+m.slider.loc.x+m.slider.dim.x, offset.y+m.slider.loc.y+m.slider.dim.y, -8.8, offset.x+m.slider.loc.x, offset.y+m.slider.loc.y+m.slider.dim.y, -8.8, offset.x+m.slider.loc.x, offset.y+m.slider.loc.y, -8.8}; glVertexAttribPointer(textShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, box_border); glVertexAttribPointer(textShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, line_tex); glDrawArrays(GL_LINE_STRIP, 0, 5); if (m.slider.dim.y > m.slider.dim.x) { //and a border around the slider handle GLfloat handle_border[] = {offset.x+m.slider.loc.x, static_cast(offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)), -8.8, offset.x+m.slider.loc.x + sliderW, static_cast(offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)), -8.8, offset.x+m.slider.loc.x + sliderW, static_cast(offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH), -8.8, offset.x+m.slider.loc.x, static_cast(offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH), -8.8, offset.x+m.slider.loc.x, static_cast(offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)), -8.8}; glVertexAttribPointer(textShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, handle_border); glVertexAttribPointer(textShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, line_tex); glDrawArrays(GL_LINE_STRIP, 0, 5); }else{ //and a border around the slider handle GLfloat handle_border[] = {offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y, -8.8, offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), offset.y+m.slider.loc.y, -8.8, offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), offset.y+m.slider.loc.y+m.slider.dim.y,-8.8, offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y+m.slider.dim.y, -8.8, offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y, -8.8}; glVertexAttribPointer(textShader_attribute_coord, 3, GL_FLOAT, GL_FALSE, 0, handle_border); glVertexAttribPointer(textShader_attribute_tex, 2, GL_FLOAT, GL_FALSE, 0, line_tex); glDrawArrays(GL_LINE_STRIP, 0, 5); } glDisableVertexAttribArray(textShader_attribute_coord); glDisableVertexAttribArray(textShader_attribute_tex); //if we are inside the slider and click it will set the slider to that point if (SDL_GetMouseState(NULL, NULL) & SDL_BUTTON(SDL_BUTTON_LEFT)) { //change handle location if (m.slider.dim.y > m.slider.dim.x) { *m.slider.var = (((mouse.y-offset.y) - m.slider.loc.y)/m.slider.dim.y)*100; //draw a white box over the handle glBindTexture(GL_TEXTURE_2D, border); drawRect(vec2(offset.x+m.slider.loc.x, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05)), vec2(offset.x+m.slider.loc.x + sliderW, offset.y+m.slider.loc.y + (m.slider.sliderLoc * 1.05) + sliderH), -8.9); }else{ *m.slider.var = (((mouse.x-offset.x) - m.slider.loc.x)/m.slider.dim.x)*100; //draw a white box over the handle glBindTexture(GL_TEXTURE_2D, border); drawRect(vec2(offset.x+m.slider.loc.x + m.slider.sliderLoc, offset.y+m.slider.loc.y), vec2(offset.x+m.slider.loc.x + (m.slider.sliderLoc + sliderW), offset.y+m.slider.loc.y + m.slider.dim.y), -8.9); } } //makes sure handle can't go below or above min and max values if (*m.slider.var >= m.slider.maxValue)*m.slider.var = m.slider.maxValue; else if (*m.slider.var <= m.slider.minValue)*m.slider.var = m.slider.minValue; } } } } setFontSize(16); setFontZ(-8.0); } } }