aboutsummaryrefslogtreecommitdiff
path: root/src
ModeNameSize
-rw-r--r--CMakeLists.txt23550logplain
-rw-r--r--activeobject.h2501logplain
-rw-r--r--ban.cpp3577logplain
-rw-r--r--ban.h1478logplain
-rw-r--r--camera.cpp21437logplain
-rw-r--r--camera.h5950logplain
-rw-r--r--chat.cpp18001logplain
-rw-r--r--chat.h7908logplain
-rw-r--r--chat_interface.h1880logplain
-rw-r--r--chatmessage.h1416logplain
-rw-r--r--client.cpp49126logplain
-rw-r--r--client.h17701logplain
d---------client588logplain
-rw-r--r--clientenvironment.cpp13848logplain
-rw-r--r--clientenvironment.h4295logplain
-rw-r--r--clientiface.cpp23000logplain
-rw-r--r--clientiface.h19356logplain
-rw-r--r--clientmap.cpp19029logplain
-rw-r--r--clientmap.h3267logplain
-rw-r--r--clientmedia.cpp19175logplain
-rw-r--r--clientmedia.h4477logplain
-rw-r--r--clientobject.cpp1694logplain
-rw-r--r--clientobject.h3430logplain
-rw-r--r--clientsimpleobject.h1052logplain
-rw-r--r--cloudparams.h941logplain
-rw-r--r--clouds.cpp11392logplain
-rw-r--r--clouds.h3135logplain
-rw-r--r--cmake_config.h.in1149logplain
-rw-r--r--cmake_config_githash.h.in169logplain
-rw-r--r--collision.cpp16274logplain
-rw-r--r--collision.h2271logplain
-rw-r--r--config.h953logplain
-rw-r--r--constants.h3605logplain
-rw-r--r--content_cao.cpp45662logplain
-rw-r--r--content_cao.h5123logplain
-rw-r--r--content_cso.cpp2578logplain
-rw-r--r--content_cso.h982logplain
-rw-r--r--content_mapblock.cpp41403logplain
-rw-r--r--content_mapblock.h4365logplain
-rw-r--r--content_mapnode.cpp5081logplain
-rw-r--r--content_mapnode.h1171logplain
-rw-r--r--content_nodemeta.cpp5335logplain
-rw-r--r--content_nodemeta.h1072logplain
-rw-r--r--content_sao.cpp38817logplain
-rw-r--r--content_sao.h11331logplain
-rw-r--r--convert_json.cpp2370logplain
-rw-r--r--convert_json.h972logplain
-rw-r--r--craftdef.cpp29921logplain
-rw-r--r--craftdef.h13450logplain
d---------database678logplain
-rw-r--r--daynightratio.h1786logplain
-rw-r--r--debug.cpp5529logplain
-rw-r--r--debug.h2829logplain
-rw-r--r--defaultsettings.cpp20410logplain
-rw-r--r--defaultsettings.h1179logplain
-rw-r--r--emerge.cpp15583logplain
-rw-r--r--emerge.h4403logplain
-rw-r--r--environment.cpp8977logplain
-rw-r--r--environment.h4038logplain
-rw-r--r--event.h1898logplain
-rw-r--r--event_manager.h2793logplain
-rw-r--r--exceptions.h3276logplain
-rw-r--r--face_position_cache.cpp3452logplain
-rw-r--r--face_position_cache.h1272logplain
-rw-r--r--filecache.cpp2268logplain
-rw-r--r--filecache.h1316logplain
-rw-r--r--filesys.cpp18757logplain
-rw-r--r--filesys.h4475logplain
-rw-r--r--fontengine.cpp14691logplain
-rw-r--r--fontengine.h3570logplain
-rw-r--r--game.cpp139646logplain
-rw-r--r--game.h1588logplain
-rw-r--r--gamedef.h2935logplain
-rw-r--r--gameparams.h970logplain
-rw-r--r--genericobject.cpp5518logplain
-rw-r--r--genericobject.h2674logplain
-rw-r--r--gettext.cpp7306logplain
-rw-r--r--gettext.h1978logplain
-rw-r--r--gettime.h1263logplain
d---------gui1173logplain
-rw-r--r--guiscalingfilter.cpp5706logplain
-rw-r--r--guiscalingfilter.h2126logplain
-rw-r--r--httpfetch.cpp20955logplain
-rw-r--r--httpfetch.h3631logplain
-rw-r--r--hud.cpp25430logplain
-rw-r--r--hud.h5103logplain
-rw-r--r--imagefilters.cpp5527logplain
-rw-r--r--imagefilters.h1939logplain
-rw-r--r--inventory.cpp19674logplain
-rw-r--r--inventory.h8498logplain
-rw-r--r--inventorymanager.cpp24710logplain
-rw-r--r--inventorymanager.h5345logplain
-rw-r--r--irr_aabb3d.h891logplain
-rw-r--r--irr_v2d.h1028logplain
-rw-r--r--irr_v3d.h1025logplain
d---------irrlicht_changes248logplain
-rw-r--r--irrlichttypes.h1893logplain
-rw-r--r--irrlichttypes_bloated.h919logplain
-rw-r--r--irrlichttypes_extrabloated.h1107logplain
-rw-r--r--itemdef.cpp15927logplain
-rw-r--r--itemdef.h5436logplain
-rw-r--r--itemgroup.h1103logplain
-rw-r--r--itemstackmetadata.cpp2058logplain
-rw-r--r--itemstackmetadata.h1536logplain
-rw-r--r--keycode.cpp10183logplain
-rw-r--r--keycode.h1776logplain
-rw-r--r--light.cpp2136logplain
-rw-r--r--light.h3183logplain
-rw-r--r--localplayer.cpp31336logplain
-rw-r--r--localplayer.h5779logplain
-rw-r--r--log.cpp8261logplain
-rw-r--r--log.h5979logplain
-rw-r--r--main.cpp29846logplain
-rw-r--r--map.cpp59712logplain
-rw-r--r--map.h12913logplain
-rw-r--r--map_settings_manager.cpp5116logplain
-rw-r--r--map_settings_manager.h2641logplain
-rw-r--r--mapblock.cpp26709logplain
-rw-r--r--mapblock.h15007logplain
-rw-r--r--mapblock_mesh.cpp43929logplain
-rw-r--r--mapblock_mesh.h7751logplain
d---------mapgen1372logplain
-rw-r--r--mapnode.cpp21144logplain
-rw-r--r--mapnode.h8277logplain
-rw-r--r--mapsector.cpp2774logplain
-rw-r--r--mapsector.h1936logplain
-rw-r--r--mesh.cpp29448logplain
-rw-r--r--mesh.h3889logplain
-rw-r--r--mesh_generator_thread.cpp8219logplain
-rw-r--r--mesh_generator_thread.h3379logplain
-rw-r--r--metadata.cpp2424logplain
-rw-r--r--metadata.h1625logplain
-rw-r--r--minimap.cpp17807logplain
-rw-r--r--minimap.h4058logplain
-rw-r--r--modchannels.cpp3920logplain
-rw-r--r--modchannels.h2753logplain
-rw-r--r--modifiedstate.h1117logplain
-rw-r--r--mods.cpp12402logplain
-rw-r--r--mods.h4766logplain
-rw-r--r--nameidmapping.cpp1486logplain
-rw-r--r--nameidmapping.h2173logplain
d---------network869logplain
-rw-r--r--nodedef.cpp49670logplain
-rw-r--r--nodedef.h15289logplain
-rw-r--r--nodemetadata.cpp5297logplain
-rw-r--r--nodemetadata.h2411logplain
-rw-r--r--nodetimer.cpp3831logplain
-rw-r--r--nodetimer.h3753logplain
-rw-r--r--noise.cpp19547logplain
-rw-r--r--noise.h6607logplain
-rw-r--r--objdef.cpp4171logplain
-rw-r--r--objdef.h2843logplain
-rw-r--r--object_properties.cpp5592logplain
-rw-r--r--object_properties.h2165logplain
-rw-r--r--particles.cpp16675logplain
-rw-r--r--particles.h4729logplain
-rw-r--r--pathfinder.cpp36386logplain
-rw-r--r--pathfinder.h2460logplain
-rw-r--r--player.cpp3302logplain
-rw-r--r--player.h3874logplain
-rw-r--r--porting.cpp14781logplain
-rw-r--r--porting.h7813logplain
-rw-r--r--porting_android.cpp9051logplain
-rw-r--r--porting_android.h2113logplain
-rw-r--r--profiler.cpp1501logplain
-rw-r--r--profiler.h4420logplain
-rw-r--r--quicktune.cpp2598logplain
-rw-r--r--quicktune.h2778logplain
-rw-r--r--quicktune_shortcutter.h2155logplain
-rw-r--r--raycast.cpp3567logplain
-rw-r--r--raycast.h2623logplain
-rw-r--r--reflowscan.cpp7045logplain
-rw-r--r--reflowscan.h1460logplain
-rw-r--r--remoteplayer.cpp7279logplain
-rw-r--r--remoteplayer.h4528logplain
-rw-r--r--rollback.cpp27394logplain
-rw-r--r--rollback.h3519logplain
-rw-r--r--rollback_interface.cpp7242logplain
-rw-r--r--rollback_interface.h4009logplain
d---------script429logplain
-rw-r--r--serialization.cpp6176logplain
-rw-r--r--serialization.h3480logplain
-rw-r--r--server.cpp97416logplain
-rw-r--r--server.h22011logplain
-rw-r--r--serverenvironment.cpp64390logplain
-rw-r--r--serverenvironment.h13606logplain
-rw-r--r--serverlist.cpp7651logplain
-rw-r--r--serverlist.h1853logplain
-rw-r--r--serverobject.cpp2475logplain
-rw-r--r--serverobject.h8119logplain
-rw-r--r--settings.cpp21810logplain
-rw-r--r--settings.h8069logplain
-rw-r--r--settings_translation_file.cpp59815logplain
-rw-r--r--shader.cpp25205logplain
-rw-r--r--shader.h4393logplain
-rw-r--r--sky.cpp26023logplain
-rw-r--r--sky.h4817logplain
-rw-r--r--sound.cpp882logplain
-rw-r--r--sound.h3860logplain
-rw-r--r--sound_openal.cpp17485logplain
-rw-r--r--sound_openal.h891logplain
-rw-r--r--staticobject.cpp2160logplain
-rw-r--r--staticobject.h2244logplain
-rw-r--r--subgame.cpp9623logplain
-rw-r--r--subgame.h2776logplain
-rw-r--r--terminal_chat_console.cpp11894logplain
-rw-r--r--terminal_chat_console.h2961logplain
d---------threading313logplain
-rw-r--r--tileanimation.cpp3705logplain
-rw-r--r--tileanimation.h1717logplain
-rw-r--r--tool.cpp7514logplain
-rw-r--r--tool.h3153logplain
-rw-r--r--translation.cpp3751logplain
-rw-r--r--translation.h1212logplain
d---------unittest1362logplain
d---------util1318logplain
-rw-r--r--version.cpp1201logplain
-rw-r--r--version.h904logplain
-rw-r--r--voxel.cpp12431logplain
-rw-r--r--voxel.h11208logplain
-rw-r--r--voxelalgorithms.cpp46355logplain
-rw-r--r--voxelalgorithms.h5309logplain
-rw-r--r--wieldmesh.cpp22357logplain
-rw-r--r--wieldmesh.h4067logplain
n> i; m_visible_rows.push_back(i); } } if (m_has_tree_column) { // Treeview: convert tree to indent cells on leaf rows for (s32 i = 0; i < rowcount; ++i) { if (i == rowcount-1 || m_rows[i].indent >= m_rows[i+1].indent) for (s32 j = 0; j < m_rows[i].cellcount; ++j) if (m_rows[i].cells[j].content_type == COLUMN_TYPE_TREE) m_rows[i].cells[j].content_type = COLUMN_TYPE_INDENT; } // Treeview: close rows according to opendepth option std::set<s32> opened_trees; for (s32 i = 0; i < rowcount; ++i) if (m_rows[i].indent < opendepth) opened_trees.insert(i); setOpenedTrees(opened_trees); } // Delete temporary information used only during setTable() delete[] rows; allocationComplete(); // Clamp scroll bar position updateScrollBar(); } void GUITable::clear() { // Clean up cells and rows for (size_t i = 0; i < m_rows.size(); ++i) delete[] m_rows[i].cells; m_rows.clear(); m_visible_rows.clear(); // Get colors from skin gui::IGUISkin *skin = Environment->getSkin(); m_color = skin->getColor(gui::EGDC_BUTTON_TEXT); m_background = skin->getColor(gui::EGDC_3D_HIGH_LIGHT); m_highlight = skin->getColor(gui::EGDC_HIGH_LIGHT); m_highlight_text = skin->getColor(gui::EGDC_HIGH_LIGHT_TEXT); // Reset members m_is_textlist = false; m_has_tree_column = false; m_selected = -1; m_sel_column = 0; m_sel_doubleclick = false; m_keynav_time = 0; m_keynav_buffer = L""; m_border = true; m_strings.clear(); m_images.clear(); m_alloc_strings.clear(); m_alloc_images.clear(); } std::string GUITable::checkEvent() { s32 sel = getSelected(); assert(sel >= 0); if (sel == 0) { return "INV"; } std::ostringstream os(std::ios::binary); if (m_sel_doubleclick) { os<<"DCL:"; m_sel_doubleclick = false; } else { os<<"CHG:"; } os<<sel; if (!m_is_textlist) { os<<":"<<m_sel_column; } return os.str(); } s32 GUITable::getSelected() const { if (m_selected < 0) return 0; assert(m_selected >= 0 && m_selected < (s32) m_visible_rows.size()); return m_visible_rows[m_selected] + 1; } void GUITable::setSelected(s32 index) { s32 old_selected = m_selected; m_selected = -1; m_sel_column = 0; m_sel_doubleclick = false; --index; // Switch from 1-based indexing to 0-based indexing s32 rowcount = m_rows.size(); if (rowcount == 0) { return; } else if (index < 0) { index = 0; } else if (index >= rowcount) { index = rowcount - 1; } // If the selected row is not visible, open its ancestors to make it visible bool selection_invisible = m_rows[index].visible_index < 0; if (selection_invisible) { std::set<s32> opened_trees; getOpenedTrees(opened_trees); s32 indent = m_rows[index].indent; for (s32 j = index - 1; j >= 0; --j) { if (m_rows[j].indent < indent) { opened_trees.insert(j); indent = m_rows[j].indent; } } setOpenedTrees(opened_trees); } if (index >= 0) { m_selected = m_rows[index].visible_index; assert(m_selected >= 0 && m_selected < (s32) m_visible_rows.size()); } if (m_selected != old_selected || selection_invisible) { autoScroll(); } } GUITable::DynamicData GUITable::getDynamicData() const { DynamicData dyndata; dyndata.selected = getSelected(); dyndata.scrollpos = m_scrollbar->getPos(); dyndata.keynav_time = m_keynav_time; dyndata.keynav_buffer = m_keynav_buffer; if (m_has_tree_column) getOpenedTrees(dyndata.opened_trees); return dyndata; } void GUITable::setDynamicData(const DynamicData &dyndata) { if (m_has_tree_column) setOpenedTrees(dyndata.opened_trees); m_keynav_time = dyndata.keynav_time; m_keynav_buffer = dyndata.keynav_buffer; setSelected(dyndata.selected); m_sel_column = 0; m_sel_doubleclick = false; m_scrollbar->setPos(dyndata.scrollpos); } const c8* GUITable::getTypeName() const { return "GUITable"; } void GUITable::updateAbsolutePosition() { IGUIElement::updateAbsolutePosition(); updateScrollBar(); } void GUITable::draw() { if (!IsVisible) return; gui::IGUISkin *skin = Environment->getSkin(); // draw background bool draw_background = m_background.getAlpha() > 0; if (m_border) skin->draw3DSunkenPane(this, m_background, true, draw_background, AbsoluteRect, &AbsoluteClippingRect); else if (draw_background) skin->draw2DRectangle(this, m_background, AbsoluteRect, &AbsoluteClippingRect); // get clipping rect core::rect<s32> client_clip(AbsoluteRect); client_clip.UpperLeftCorner.Y += 1; client_clip.UpperLeftCorner.X += 1; client_clip.LowerRightCorner.Y -= 1; client_clip.LowerRightCorner.X -= 1; if (m_scrollbar->isVisible()) { client_clip.LowerRightCorner.X = m_scrollbar->getAbsolutePosition().UpperLeftCorner.X; } client_clip.clipAgainst(AbsoluteClippingRect); // draw visible rows s32 scrollpos = m_scrollbar->getPos(); s32 row_min = scrollpos / m_rowheight; s32 row_max = (scrollpos + AbsoluteRect.getHeight() - 1) / m_rowheight + 1; row_max = MYMIN(row_max, (s32) m_visible_rows.size()); core::rect<s32> row_rect(AbsoluteRect); if (m_scrollbar->isVisible()) row_rect.LowerRightCorner.X -= skin->getSize(gui::EGDS_SCROLLBAR_SIZE); row_rect.UpperLeftCorner.Y += row_min * m_rowheight - scrollpos; row_rect.LowerRightCorner.Y = row_rect.UpperLeftCorner.Y + m_rowheight; for (s32 i = row_min; i < row_max; ++i) { Row *row = &m_rows[m_visible_rows[i]]; bool is_sel = i == m_selected; video::SColor color = m_color; if (is_sel) { skin->draw2DRectangle(this, m_highlight, row_rect, &client_clip); color = m_highlight_text; } for (s32 j = 0; j < row->cellcount; ++j) drawCell(&row->cells[j], color, row_rect, client_clip); row_rect.UpperLeftCorner.Y += m_rowheight; row_rect.LowerRightCorner.Y += m_rowheight; } // Draw children IGUIElement::draw(); } void GUITable::drawCell(const Cell *cell, video::SColor color, const core::rect<s32> &row_rect, const core::rect<s32> &client_clip) { if ((cell->content_type == COLUMN_TYPE_TEXT) || (cell->content_type == COLUMN_TYPE_TREE)) { core::rect<s32> text_rect = row_rect; text_rect.UpperLeftCorner.X = row_rect.UpperLeftCorner.X + cell->xpos; text_rect.LowerRightCorner.X = row_rect.UpperLeftCorner.X + cell->xmax; if (cell->color_defined) color = cell->color; if (m_font) { if (cell->content_type == COLUMN_TYPE_TEXT) m_font->draw(m_strings[cell->content_index], text_rect, color, false, true, &client_clip); else // tree m_font->draw(cell->content_index ? L"+" : L"-", text_rect, color, false, true, &client_clip); } } else if (cell->content_type == COLUMN_TYPE_IMAGE) { if (cell->content_index < 0) return; video::IVideoDriver *driver = Environment->getVideoDriver(); video::ITexture *image = m_images[cell->content_index]; if (image) { core::position2d<s32> dest_pos = row_rect.UpperLeftCorner; dest_pos.X += cell->xpos; core::rect<s32> source_rect( core::position2d<s32>(0, 0), image->getOriginalSize()); s32 imgh = source_rect.LowerRightCorner.Y; s32 rowh = row_rect.getHeight(); if (imgh < rowh) dest_pos.Y += (rowh - imgh) / 2; else source_rect.LowerRightCorner.Y = rowh; video::SColor color(255, 255, 255, 255); driver->draw2DImage(image, dest_pos, source_rect, &client_clip, color, true); } } } bool GUITable::OnEvent(const SEvent &event) { if (!isEnabled()) return IGUIElement::OnEvent(event); if (event.EventType == EET_KEY_INPUT_EVENT) { if (event.KeyInput.PressedDown && ( event.KeyInput.Key == KEY_DOWN || event.KeyInput.Key == KEY_UP || event.KeyInput.Key == KEY_HOME || event.KeyInput.Key == KEY_END || event.KeyInput.Key == KEY_NEXT || event.KeyInput.Key == KEY_PRIOR)) { s32 offset = 0; switch (event.KeyInput.Key) { case KEY_DOWN: offset = 1; break; case KEY_UP: offset = -1; break; case KEY_HOME: offset = - (s32) m_visible_rows.size(); break; case KEY_END: offset = m_visible_rows.size(); break; case KEY_NEXT: offset = AbsoluteRect.getHeight() / m_rowheight; break; case KEY_PRIOR: offset = - (s32) (AbsoluteRect.getHeight() / m_rowheight); break; default: break; } s32 old_selected = m_selected; s32 rowcount = m_visible_rows.size(); if (rowcount != 0) { m_selected = rangelim(m_selected + offset, 0, rowcount-1); autoScroll(); } if (m_selected != old_selected) sendTableEvent(0, false); return true; } else if (event.KeyInput.PressedDown && ( event.KeyInput.Key == KEY_LEFT || event.KeyInput.Key == KEY_RIGHT)) { // Open/close subtree via keyboard if (m_selected >= 0) { int dir = event.KeyInput.Key == KEY_LEFT ? -1 : 1; toggleVisibleTree(m_selected, dir, true); } return true; } else if (!event.KeyInput.PressedDown && ( event.KeyInput.Key == KEY_RETURN || event.KeyInput.Key == KEY_SPACE)) { sendTableEvent(0, true); return true; } else if (event.KeyInput.Key == KEY_ESCAPE || event.KeyInput.Key == KEY_SPACE) { // pass to parent } else if (event.KeyInput.PressedDown && event.KeyInput.Char) { // change selection based on text as it is typed s32 now = getTimeMs(); if (now - m_keynav_time >= 500) m_keynav_buffer = L""; m_keynav_time = now; // add to key buffer if not a key repeat if (!(m_keynav_buffer.size() == 1 && m_keynav_buffer[0] == event.KeyInput.Char)) { m_keynav_buffer.append(event.KeyInput.Char); } // find the selected item, starting at the current selection // don't change selection if the key buffer matches the current item s32 old_selected = m_selected; s32 start = MYMAX(m_selected, 0); s32 rowcount = m_visible_rows.size(); for (s32 k = 1; k < rowcount; ++k) { s32 current = start + k; if (current >= rowcount) current -= rowcount; if (doesRowStartWith(getRow(current), m_keynav_buffer)) { m_selected = current; break; } } autoScroll(); if (m_selected != old_selected) sendTableEvent(0, false); return true; } } if (event.EventType == EET_MOUSE_INPUT_EVENT) { core::position2d<s32> p(event.MouseInput.X, event.MouseInput.Y); if (event.MouseInput.Event == EMIE_MOUSE_WHEEL) { m_scrollbar->setPos(m_scrollbar->getPos() + (event.MouseInput.Wheel < 0 ? -3 : 3) * - (s32) m_rowheight / 2); return true; } // Find hovered row and cell bool really_hovering = false; s32 row_i = getRowAt(p.Y, really_hovering); const Cell *cell = NULL; if (really_hovering) { s32 cell_j = getCellAt(p.X, row_i); if (cell_j >= 0) cell = &(getRow(row_i)->cells[cell_j]); } // Update tooltip setToolTipText(cell ? m_strings[cell->tooltip_index].c_str() : L""); // Fix for #1567/#1806: // IGUIScrollBar passes double click events to its parent, // which we don't want. Detect this case and discard the event if (event.MouseInput.Event != EMIE_MOUSE_MOVED && m_scrollbar->isVisible() && m_scrollbar->isPointInside(p)) return true; if (event.MouseInput.isLeftPressed() && (isPointInside(p) || event.MouseInput.Event == EMIE_MOUSE_MOVED)) { s32 sel_column = 0; bool sel_doubleclick = (event.MouseInput.Event == EMIE_LMOUSE_DOUBLE_CLICK); bool plusminus_clicked = false; // For certain events (left click), report column // Also open/close subtrees when the +/- is clicked if (cell && ( event.MouseInput.Event == EMIE_LMOUSE_PRESSED_DOWN || event.MouseInput.Event == EMIE_LMOUSE_DOUBLE_CLICK || event.MouseInput.Event == EMIE_LMOUSE_TRIPLE_CLICK)) { sel_column = cell->reported_column; if (cell->content_type == COLUMN_TYPE_TREE) plusminus_clicked = true; } if (plusminus_clicked) { if (event.MouseInput.Event == EMIE_LMOUSE_PRESSED_DOWN) { toggleVisibleTree(row_i, 0, false); } } else { // Normal selection s32 old_selected = m_selected; m_selected = row_i; autoScroll(); if (m_selected != old_selected || sel_column >= 1 || sel_doubleclick) { sendTableEvent(sel_column, sel_doubleclick); } // Treeview: double click opens/closes trees if (m_has_tree_column && sel_doubleclick) { toggleVisibleTree(m_selected, 0, false); } } } return true; } if (event.EventType == EET_GUI_EVENT && event.GUIEvent.EventType == gui::EGET_SCROLL_BAR_CHANGED && event.GUIEvent.Caller == m_scrollbar) { // Don't pass events from our scrollbar to the parent return true; } return IGUIElement::OnEvent(event); } /******************************************************************************/ /* GUITable helper functions */ /******************************************************************************/ s32 GUITable::allocString(const std::string &text) { std::map<std::string, s32>::iterator it = m_alloc_strings.find(text); if (it == m_alloc_strings.end()) { s32 id = m_strings.size(); std::wstring wtext = utf8_to_wide(text); m_strings.push_back(core::stringw(wtext.c_str())); m_alloc_strings.insert(std::make_pair(text, id)); return id; } else { return it->second; } } s32 GUITable::allocImage(const std::string &imagename) { std::map<std::string, s32>::iterator it = m_alloc_images.find(imagename); if (it == m_alloc_images.end()) { s32 id = m_images.size(); m_images.push_back(m_tsrc->getTexture(imagename)); m_alloc_images.insert(std::make_pair(imagename, id)); return id; } else { return it->second; } } void GUITable::allocationComplete() { // Called when done with creating rows and cells from table data, // i.e. when allocString and allocImage won't be called anymore m_alloc_strings.clear(); m_alloc_images.clear(); } const GUITable::Row* GUITable::getRow(s32 i) const { if (i >= 0 && i < (s32) m_visible_rows.size()) return &m_rows[m_visible_rows[i]]; else return NULL; } bool GUITable::doesRowStartWith(const Row *row, const core::stringw &str) const { if (row == NULL) return false; for (s32 j = 0; j < row->cellcount; ++j) { Cell *cell = &row->cells[j]; if (cell->content_type == COLUMN_TYPE_TEXT) { const core::stringw &cellstr = m_strings[cell->content_index]; if (cellstr.size() >= str.size() && str.equals_ignore_case(cellstr.subString(0, str.size()))) return true; } } return false; } s32 GUITable::getRowAt(s32 y, bool &really_hovering) const { really_hovering = false; s32 rowcount = m_visible_rows.size(); if (rowcount == 0) return -1; // Use arithmetic to find row s32 rel_y = y - AbsoluteRect.UpperLeftCorner.Y - 1; s32 i = (rel_y + m_scrollbar->getPos()) / m_rowheight; if (i >= 0 && i < rowcount) { really_hovering = true; return i; } else if (i < 0) return 0; else return rowcount - 1; } s32 GUITable::getCellAt(s32 x, s32 row_i) const { const Row *row = getRow(row_i); if (row == NULL) return -1; // Use binary search to find cell in row s32 rel_x = x - AbsoluteRect.UpperLeftCorner.X - 1; s32 jmin = 0; s32 jmax = row->cellcount - 1; while (jmin < jmax) { s32 pivot = jmin + (jmax - jmin) / 2; assert(pivot >= 0 && pivot < row->cellcount); const Cell *cell = &row->cells[pivot]; if (rel_x >= cell->xmin && rel_x <= cell->xmax) return pivot; else if (rel_x < cell->xmin) jmax = pivot - 1; else jmin = pivot + 1; } if (jmin >= 0 && jmin < row->cellcount && rel_x >= row->cells[jmin].xmin && rel_x <= row->cells[jmin].xmax) return jmin; else return -1; } void GUITable::autoScroll() { if (m_selected >= 0) { s32 pos = m_scrollbar->getPos(); s32 maxpos = m_selected * m_rowheight; s32 minpos = maxpos - (AbsoluteRect.getHeight() - m_rowheight); if (pos > maxpos) m_scrollbar->setPos(maxpos); else if (pos < minpos) m_scrollbar->setPos(minpos); } } void GUITable::updateScrollBar() { s32 totalheight = m_rowheight * m_visible_rows.size(); s32 scrollmax = MYMAX(0, totalheight - AbsoluteRect.getHeight()); m_scrollbar->setVisible(scrollmax > 0); m_scrollbar->setMax(scrollmax); m_scrollbar->setSmallStep(m_rowheight); m_scrollbar->setLargeStep(2 * m_rowheight); } void GUITable::sendTableEvent(s32 column, bool doubleclick) { m_sel_column = column; m_sel_doubleclick = doubleclick; if (Parent) { SEvent e; memset(&e, 0, sizeof e); e.EventType = EET_GUI_EVENT; e.GUIEvent.Caller = this; e.GUIEvent.Element = 0; e.GUIEvent.EventType = gui::EGET_TABLE_CHANGED; Parent->OnEvent(e); } } void GUITable::getOpenedTrees(std::set<s32> &opened_trees) const { opened_trees.clear(); s32 rowcount = m_rows.size(); for (s32 i = 0; i < rowcount - 1; ++i) { if (m_rows[i].indent < m_rows[i+1].indent && m_rows[i+1].visible_index != -2) opened_trees.insert(i); } } void GUITable::setOpenedTrees(const std::set<s32> &opened_trees) { s32 old_selected = -1; if (m_selected >= 0) old_selected = m_visible_rows[m_selected]; std::vector<s32> parents; std::vector<s32> closed_parents; m_visible_rows.clear(); for (size_t i = 0; i < m_rows.size(); ++i) { Row *row = &m_rows[i]; // Update list of ancestors while (!parents.empty() && m_rows[parents.back()].indent >= row->indent) parents.pop_back(); while (!closed_parents.empty() && m_rows[closed_parents.back()].indent >= row->indent) closed_parents.pop_back(); assert(closed_parents.size() <= parents.size()); if (closed_parents.empty()) { // Visible row row->visible_index = m_visible_rows.size(); m_visible_rows.push_back(i); } else if (parents.back() == closed_parents.back()) { // Invisible row, direct parent is closed row->visible_index = -2; } else { // Invisible row, direct parent is open, some ancestor is closed row->visible_index = -1; } // If not a leaf, add to parents list if (i < m_rows.size()-1 && row->indent < m_rows[i+1].indent) { parents.push_back(i); s32 content_index = 0; // "-", open if (opened_trees.count(i) == 0) { closed_parents.push_back(i); content_index = 1; // "+", closed } // Update all cells of type "tree" for (s32 j = 0; j < row->cellcount; ++j) if (row->cells[j].content_type == COLUMN_TYPE_TREE) row->cells[j].content_index = content_index; } } updateScrollBar(); // m_selected must be updated since it is a visible row index if (old_selected >= 0) m_selected = m_rows[old_selected].visible_index; } void GUITable::openTree(s32 to_open) { std::set<s32> opened_trees; getOpenedTrees(opened_trees); opened_trees.insert(to_open); setOpenedTrees(opened_trees); } void GUITable::closeTree(s32 to_close) { std::set<s32> opened_trees; getOpenedTrees(opened_trees); opened_trees.erase(to_close); setOpenedTrees(opened_trees); } // The following function takes a visible row index (hidden rows skipped) // dir: -1 = left (close), 0 = auto (toggle), 1 = right (open) void GUITable::toggleVisibleTree(s32 row_i, int dir, bool move_selection) { // Check if the chosen tree is currently open const Row *row = getRow(row_i); if (row == NULL) return; bool was_open = false; for (s32 j = 0; j < row->cellcount; ++j) { if (row->cells[j].content_type == COLUMN_TYPE_TREE) { was_open = row->cells[j].content_index == 0; break; } } // Check if the chosen tree should be opened bool do_open = !was_open; if (dir < 0) do_open = false; else if (dir > 0) do_open = true; // Close or open the tree; the heavy lifting is done by setOpenedTrees if (was_open && !do_open) closeTree(m_visible_rows[row_i]); else if (!was_open && do_open) openTree(m_visible_rows[row_i]); // Change selected row if requested by caller, // this is useful for keyboard navigation if (move_selection) { s32 sel = row_i; if (was_open && do_open) { // Move selection to first child const Row *maybe_child = getRow(sel + 1); if (maybe_child && maybe_child->indent > row->indent) sel++; } else if (!was_open && !do_open) { // Move selection to parent assert(getRow(sel) != NULL); while (sel > 0 && getRow(sel - 1)->indent >= row->indent) sel--; sel--; if (sel < 0) // was root already selected? sel = row_i; } if (sel != m_selected) { m_selected = sel; autoScroll(); sendTableEvent(0, false); } } } void GUITable::alignContent(Cell *cell, s32 xmax, s32 content_width, s32 align) { // requires that cell.xmin, cell.xmax are properly set // align = 0: left aligned, 1: centered, 2: right aligned, 3: inline if (align == 0) { cell->xpos = cell->xmin; cell->xmax = xmax; } else if (align == 1) { cell->xpos = (cell->xmin + xmax - content_width) / 2; cell->xmax = xmax; } else if (align == 2) { cell->xpos = xmax - content_width; cell->xmax = xmax; } else { // inline alignment: the cells of the column don't have an aligned // right border, the right border of each cell depends on the content cell->xpos = cell->xmin; cell->xmax = cell->xmin + content_width; } }