2017-03-17 02:10:49 +01:00
|
|
|
/* Copyright (C) 2017 Wildfire Games.
|
2014-06-05 16:33:44 +02:00
|
|
|
* This file is part of 0 A.D.
|
|
|
|
*
|
|
|
|
* 0 A.D. is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* 0 A.D. is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with 0 A.D. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2015-08-21 19:08:41 +02:00
|
|
|
|
2013-11-07 21:07:24 +01:00
|
|
|
#include "precompiled.h"
|
2015-08-21 19:08:41 +02:00
|
|
|
|
2013-11-07 21:07:24 +01:00
|
|
|
#include "COList.h"
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
#include "i18n/L10n.h"
|
2013-11-07 21:07:24 +01:00
|
|
|
#include "ps/CLogger.h"
|
2015-06-17 11:10:50 +02:00
|
|
|
#include "soundmanager/ISoundManager.h"
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
COList::COList() : CList()
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
|
|
|
AddSetting(GUIST_CGUISpriteInstance, "sprite_heading");
|
2016-08-31 21:29:48 +02:00
|
|
|
AddSetting(GUIST_float, "heading_height");
|
2015-06-17 11:10:50 +02:00
|
|
|
AddSetting(GUIST_bool, "sortable"); // The actual sorting is done in JS for more versatility
|
|
|
|
AddSetting(GUIST_CStr, "selected_column");
|
|
|
|
AddSetting(GUIST_int, "selected_column_order");
|
|
|
|
AddSetting(GUIST_CGUISpriteInstance, "sprite_asc"); // Show the order of sorting
|
|
|
|
AddSetting(GUIST_CGUISpriteInstance, "sprite_desc");
|
|
|
|
AddSetting(GUIST_CGUISpriteInstance, "sprite_not_sorted");
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void COList::SetupText()
|
|
|
|
{
|
|
|
|
if (!GetGUI())
|
|
|
|
return;
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
CGUIList* pList;
|
2016-08-31 21:29:48 +02:00
|
|
|
GUI<CGUIList>::GetSettingPointer(this, "list", pList);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
m_ItemsYPositions.resize(pList->m_Items.size() + 1);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
|
|
|
// Delete all generated texts. Some could probably be saved,
|
|
|
|
// but this is easier, and this function will never be called
|
|
|
|
// continuously, or even often, so it'll probably be okay.
|
2015-08-21 19:08:41 +02:00
|
|
|
for (SGUIText* const& t : m_GeneratedTexts)
|
|
|
|
delete t;
|
2013-11-07 21:07:24 +01:00
|
|
|
m_GeneratedTexts.clear();
|
|
|
|
|
|
|
|
CStrW font;
|
|
|
|
if (GUI<CStrW>::GetSetting(this, "font", font) != PSRETURN_OK || font.empty())
|
|
|
|
// Use the default if none is specified
|
|
|
|
// TODO Gee: (2004-08-14) Don't define standard like this. Do it with the default style.
|
|
|
|
font = L"default";
|
|
|
|
|
|
|
|
bool scrollbar;
|
|
|
|
GUI<bool>::GetSetting(this, "scrollbar", scrollbar);
|
|
|
|
|
|
|
|
float width = GetListRect().GetWidth();
|
|
|
|
// remove scrollbar if applicable
|
|
|
|
if (scrollbar && GetScrollBar(0).GetStyle())
|
|
|
|
width -= GetScrollBar(0).GetStyle()->m_Width;
|
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
m_TotalAvailableColumnWidth = width;
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
float buffer_zone = 0.f;
|
2013-11-07 21:07:24 +01:00
|
|
|
GUI<float>::GetSetting(this, "buffer_zone", buffer_zone);
|
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
for (COListColumn column : m_Columns)
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
2015-08-21 19:08:41 +02:00
|
|
|
SGUIText* text = new SGUIText();
|
2013-11-07 21:07:24 +01:00
|
|
|
CGUIString gui_string;
|
2016-08-31 21:29:48 +02:00
|
|
|
gui_string.SetValue(column.m_Heading);
|
2013-11-07 21:07:24 +01:00
|
|
|
*text = GetGUI()->GenerateText(gui_string, font, width, buffer_zone, this);
|
|
|
|
AddText(text);
|
2016-01-18 20:23:14 +01:00
|
|
|
}
|
|
|
|
|
2013-11-07 21:07:24 +01:00
|
|
|
// Generate texts
|
|
|
|
float buffered_y = 0.f;
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
for (size_t i = 0; i < pList->m_Items.size(); ++i)
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
|
|
|
m_ItemsYPositions[i] = buffered_y;
|
2017-03-17 02:10:49 +01:00
|
|
|
float shift = 0.0f;
|
2016-08-31 21:29:48 +02:00
|
|
|
for (size_t c = 0; c < m_Columns.size(); ++c)
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
2015-08-21 19:08:41 +02:00
|
|
|
CGUIList* pList_c;
|
2016-08-31 21:29:48 +02:00
|
|
|
GUI<CGUIList>::GetSettingPointer(this, "list_" + m_Columns[c].m_Id, pList_c);
|
2015-08-21 19:08:41 +02:00
|
|
|
SGUIText* text = new SGUIText();
|
2017-03-17 02:10:49 +01:00
|
|
|
if (!pList_c->m_Items[i].GetOriginalString().empty())
|
|
|
|
*text = GetGUI()->GenerateText(pList_c->m_Items[i], font, width, buffer_zone, this);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Minimum height of a space character of the current font size
|
|
|
|
CGUIString align_string;
|
|
|
|
align_string.SetValue(L" ");
|
|
|
|
*text = GetGUI()->GenerateText(align_string, font, width, buffer_zone, this);
|
|
|
|
}
|
|
|
|
shift = std::max(shift, text->m_Size.cy);
|
2013-11-07 21:07:24 +01:00
|
|
|
AddText(text);
|
|
|
|
}
|
2017-03-17 02:10:49 +01:00
|
|
|
buffered_y += shift;
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
m_ItemsYPositions[pList->m_Items.size()] = buffered_y;
|
|
|
|
|
|
|
|
if (scrollbar)
|
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
CRect rect = GetListRect();
|
2015-08-21 19:08:41 +02:00
|
|
|
GetScrollBar(0).SetScrollRange(m_ItemsYPositions.back());
|
2016-08-31 21:29:48 +02:00
|
|
|
GetScrollBar(0).SetScrollSpace(rect.GetHeight());
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
GetScrollBar(0).SetX(rect.right);
|
|
|
|
GetScrollBar(0).SetY(rect.top);
|
|
|
|
GetScrollBar(0).SetZ(GetBufferedZ());
|
|
|
|
GetScrollBar(0).SetLength(rect.bottom - rect.top);
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CRect COList::GetListRect() const
|
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
float headingHeight;
|
|
|
|
GUI<float>::GetSetting(this, "heading_height", headingHeight);
|
|
|
|
return m_CachedActualSize + CRect(0, headingHeight, 0, 0);
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
void COList::HandleMessage(SGUIMessage& Message)
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
|
|
|
CList::HandleMessage(Message);
|
2015-06-17 11:10:50 +02:00
|
|
|
|
|
|
|
switch (Message.type)
|
|
|
|
{
|
|
|
|
// If somebody clicks on the column heading
|
|
|
|
case GUIM_MOUSE_PRESS_LEFT:
|
|
|
|
{
|
|
|
|
bool sortable;
|
|
|
|
GUI<bool>::GetSetting(this, "sortable", sortable);
|
|
|
|
if (!sortable)
|
|
|
|
return;
|
|
|
|
|
|
|
|
CPos mouse = GetMousePos();
|
|
|
|
if (!m_CachedActualSize.PointInside(mouse))
|
|
|
|
return;
|
2015-08-21 19:08:41 +02:00
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
CStr selectedColumn;
|
|
|
|
GUI<CStr>::GetSetting(this, "selected_column", selectedColumn);
|
|
|
|
int selectedColumnOrder;
|
|
|
|
GUI<int>::GetSetting(this, "selected_column_order", selectedColumnOrder);
|
|
|
|
float headingHeight;
|
|
|
|
GUI<float>::GetSetting(this, "heading_height", headingHeight);
|
|
|
|
|
2015-06-17 11:10:50 +02:00
|
|
|
float xpos = 0;
|
2016-08-31 21:29:48 +02:00
|
|
|
for (COListColumn column : m_Columns)
|
2015-06-17 11:10:50 +02:00
|
|
|
{
|
2017-03-28 02:25:17 +02:00
|
|
|
bool hidden = false;
|
|
|
|
GUI<bool>::GetSetting(this, "hidden_" + column.m_Id, hidden);
|
|
|
|
if (hidden)
|
|
|
|
continue;
|
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
float width = column.m_Width;
|
2015-06-17 11:10:50 +02:00
|
|
|
// Check if it's a decimal value, and if so, assume relative positioning.
|
2016-08-31 21:29:48 +02:00
|
|
|
if (column.m_Width < 1 && column.m_Width > 0)
|
|
|
|
width *= m_TotalAvailableColumnWidth;
|
2015-09-02 19:55:02 +02:00
|
|
|
CPos leftTopCorner = m_CachedActualSize.TopLeft() + CPos(xpos, 0);
|
2015-06-17 11:10:50 +02:00
|
|
|
if (mouse.x >= leftTopCorner.x &&
|
|
|
|
mouse.x < leftTopCorner.x + width &&
|
2016-08-31 21:29:48 +02:00
|
|
|
mouse.y < leftTopCorner.y + headingHeight)
|
2015-06-17 11:10:50 +02:00
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
if (column.m_Id != selectedColumn)
|
2015-06-17 11:10:50 +02:00
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
selectedColumnOrder = 1;
|
|
|
|
selectedColumn = column.m_Id;
|
2015-06-17 11:10:50 +02:00
|
|
|
}
|
|
|
|
else
|
2016-08-31 21:29:48 +02:00
|
|
|
selectedColumnOrder = -selectedColumnOrder;
|
|
|
|
GUI<CStr>::SetSetting(this, "selected_column", column.m_Id);
|
|
|
|
GUI<int>::SetSetting(this, "selected_column_order", selectedColumnOrder);
|
2015-06-17 11:10:50 +02:00
|
|
|
ScriptEvent("selectioncolumnchange");
|
|
|
|
|
|
|
|
CStrW soundPath;
|
|
|
|
if (g_SoundManager && GUI<CStrW>::GetSetting(this, "sound_selected", soundPath) == PSRETURN_OK && !soundPath.empty())
|
|
|
|
g_SoundManager->PlayAsUI(soundPath.c_str(), false);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
xpos += width;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool COList::HandleAdditionalChildren(const XMBElement& child, CXeromyces* pFile)
|
|
|
|
{
|
2014-04-20 22:03:57 +02:00
|
|
|
#define ELMT(x) int elmt_##x = pFile->GetElementID(#x)
|
|
|
|
#define ATTR(x) int attr_##x = pFile->GetAttributeID(#x)
|
|
|
|
ELMT(item);
|
2016-08-31 21:29:48 +02:00
|
|
|
ELMT(column);
|
2014-04-20 22:03:57 +02:00
|
|
|
ELMT(translatableAttribute);
|
|
|
|
ATTR(id);
|
2014-05-01 00:33:08 +02:00
|
|
|
ATTR(context);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
|
|
|
if (child.GetNodeName() == elmt_item)
|
|
|
|
{
|
|
|
|
AddItem(child.GetText().FromUTF8(), child.GetText().FromUTF8());
|
|
|
|
return true;
|
|
|
|
}
|
2016-08-31 21:29:48 +02:00
|
|
|
else if (child.GetNodeName() == elmt_column)
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
COListColumn column;
|
2017-03-28 02:25:17 +02:00
|
|
|
bool hidden = false;
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-06-01 02:29:35 +02:00
|
|
|
for (XMBAttribute attr : child.GetAttributes())
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
2015-08-21 19:08:41 +02:00
|
|
|
CStr attr_name(pFile->GetAttributeString(attr.Name));
|
|
|
|
CStr attr_value(attr.Value);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
|
|
|
if (attr_name == "color")
|
|
|
|
{
|
|
|
|
CColor color;
|
|
|
|
if (!GUI<CColor>::ParseString(attr_value.FromUTF8(), color))
|
2015-01-22 21:36:24 +01:00
|
|
|
LOGERROR("GUI: Error parsing '%s' (\"%s\")", attr_name.c_str(), attr_value.c_str());
|
2016-08-31 21:29:48 +02:00
|
|
|
else
|
|
|
|
column.m_TextColor = color;
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
else if (attr_name == "id")
|
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
column.m_Id = attr_value;
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
2017-03-28 02:25:17 +02:00
|
|
|
else if (attr_name == "hidden")
|
|
|
|
{
|
|
|
|
if (!GUI<bool>::ParseString(attr_value.FromUTF8(), hidden))
|
|
|
|
LOGERROR("GUI: Error parsing '%s' (\"%s\")", attr_name.c_str(), attr_value.c_str());
|
|
|
|
}
|
2013-11-07 21:07:24 +01:00
|
|
|
else if (attr_name == "width")
|
|
|
|
{
|
|
|
|
float width;
|
|
|
|
if (!GUI<float>::ParseString(attr_value.FromUTF8(), width))
|
2015-01-22 21:36:24 +01:00
|
|
|
LOGERROR("GUI: Error parsing '%s' (\"%s\")", attr_name.c_str(), attr_value.c_str());
|
2013-11-07 21:07:24 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// Check if it's a relative value, and save as decimal if so.
|
|
|
|
if (attr_value.find("%") != std::string::npos)
|
|
|
|
width = width / 100.f;
|
2016-08-31 21:29:48 +02:00
|
|
|
column.m_Width = width;
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (attr_name == "heading")
|
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
column.m_Heading = attr_value.FromUTF8();
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-01 02:29:35 +02:00
|
|
|
for (XMBElement grandchild : child.GetChildNodes())
|
2014-04-20 22:03:57 +02:00
|
|
|
{
|
2015-08-21 19:08:41 +02:00
|
|
|
if (grandchild.GetNodeName() != elmt_translatableAttribute)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CStr attributeName(grandchild.GetAttributes().GetNamedItem(attr_id));
|
2016-08-31 21:29:48 +02:00
|
|
|
// only the heading is translatable for list column
|
2015-08-21 19:08:41 +02:00
|
|
|
if (attributeName.empty() || attributeName != "heading")
|
2014-04-20 22:03:57 +02:00
|
|
|
{
|
2016-08-31 21:29:48 +02:00
|
|
|
LOGERROR("GUI: translatable attribute in olist column that isn't a heading. (object: %s)", this->GetPresentableName().c_str());
|
2015-08-21 19:08:41 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
CStr value(grandchild.GetText());
|
|
|
|
if (value.empty())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
CStr context(grandchild.GetAttributes().GetNamedItem(attr_context)); // Read the context if any.
|
|
|
|
if (!context.empty())
|
|
|
|
{
|
|
|
|
CStr translatedValue(g_L10n.TranslateWithContext(context, value));
|
2016-08-31 21:29:48 +02:00
|
|
|
column.m_Heading = translatedValue.FromUTF8();
|
2015-08-21 19:08:41 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CStr translatedValue(g_L10n.Translate(value));
|
2016-08-31 21:29:48 +02:00
|
|
|
column.m_Heading = translatedValue.FromUTF8();
|
2014-04-20 22:03:57 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
m_Columns.push_back(column);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
AddSetting(GUIST_CGUIList, "list_" + column.m_Id);
|
2017-03-28 02:25:17 +02:00
|
|
|
AddSetting(GUIST_bool, "hidden_" + column.m_Id);
|
|
|
|
GUI<bool>::SetSetting(this, "hidden_" + column.m_Id, hidden);
|
|
|
|
|
2013-11-07 21:07:24 +01:00
|
|
|
SetupText();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
void COList::DrawList(const int& selected, const CStr& _sprite, const CStr& _sprite_selected, const CStr& _textcolor)
|
2013-11-07 21:07:24 +01:00
|
|
|
{
|
|
|
|
float bz = GetBufferedZ();
|
|
|
|
|
|
|
|
bool scrollbar;
|
|
|
|
GUI<bool>::GetSetting(this, "scrollbar", scrollbar);
|
|
|
|
|
|
|
|
if (scrollbar)
|
|
|
|
IGUIScrollBarOwner::Draw();
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
if (!GetGUI())
|
|
|
|
return;
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
CRect rect = GetListRect();
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
CGUISpriteInstance* sprite = NULL;
|
|
|
|
CGUISpriteInstance* sprite_selectarea = NULL;
|
|
|
|
int cell_id;
|
|
|
|
GUI<CGUISpriteInstance>::GetSettingPointer(this, _sprite, sprite);
|
|
|
|
GUI<CGUISpriteInstance>::GetSettingPointer(this, _sprite_selected, sprite_selectarea);
|
|
|
|
GUI<int>::GetSetting(this, "cell_id", cell_id);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
CGUIList* pList;
|
2016-08-31 21:29:48 +02:00
|
|
|
GUI<CGUIList>::GetSettingPointer(this, "list", pList);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
GetGUI()->DrawSprite(*sprite, cell_id, bz, rect);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
float scroll = 0.f;
|
|
|
|
if (scrollbar)
|
|
|
|
scroll = GetScrollBar(0).GetPos();
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-09-02 19:55:02 +02:00
|
|
|
// Draw item selection
|
2015-08-21 19:08:41 +02:00
|
|
|
if (selected != -1)
|
|
|
|
{
|
|
|
|
ENSURE(selected >= 0 && selected+1 < (int)m_ItemsYPositions.size());
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
// Get rectangle of selection:
|
|
|
|
CRect rect_sel(rect.left, rect.top + m_ItemsYPositions[selected] - scroll,
|
|
|
|
rect.right, rect.top + m_ItemsYPositions[selected+1] - scroll);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
if (rect_sel.top <= rect.bottom &&
|
|
|
|
rect_sel.bottom >= rect.top)
|
|
|
|
{
|
|
|
|
if (rect_sel.bottom > rect.bottom)
|
|
|
|
rect_sel.bottom = rect.bottom;
|
|
|
|
if (rect_sel.top < rect.top)
|
|
|
|
rect_sel.top = rect.top;
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
if (scrollbar)
|
|
|
|
{
|
|
|
|
// Remove any overlapping area of the scrollbar.
|
|
|
|
if (rect_sel.right > GetScrollBar(0).GetOuterRect().left &&
|
|
|
|
rect_sel.right <= GetScrollBar(0).GetOuterRect().right)
|
|
|
|
rect_sel.right = GetScrollBar(0).GetOuterRect().left;
|
|
|
|
|
|
|
|
if (rect_sel.left >= GetScrollBar(0).GetOuterRect().left &&
|
|
|
|
rect_sel.left < GetScrollBar(0).GetOuterRect().right)
|
|
|
|
rect_sel.left = GetScrollBar(0).GetOuterRect().right;
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
2015-08-21 19:08:41 +02:00
|
|
|
|
2015-09-02 19:55:02 +02:00
|
|
|
// Draw item selection
|
2015-08-21 19:08:41 +02:00
|
|
|
GetGUI()->DrawSprite(*sprite_selectarea, cell_id, bz+0.05f, rect_sel);
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
2015-08-21 19:08:41 +02:00
|
|
|
}
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
float headingHeight;
|
|
|
|
GUI<float>::GetSetting(this, "heading_height", headingHeight);
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-09-02 19:55:02 +02:00
|
|
|
// Draw line above column header
|
2015-08-21 19:08:41 +02:00
|
|
|
CGUISpriteInstance* sprite_heading = NULL;
|
|
|
|
GUI<CGUISpriteInstance>::GetSettingPointer(this, "sprite_heading", sprite_heading);
|
|
|
|
CRect rect_head(m_CachedActualSize.left, m_CachedActualSize.top, m_CachedActualSize.right,
|
2016-08-31 21:29:48 +02:00
|
|
|
m_CachedActualSize.top + headingHeight);
|
2015-08-21 19:08:41 +02:00
|
|
|
GetGUI()->DrawSprite(*sprite_heading, cell_id, bz, rect_head);
|
2015-06-17 11:10:50 +02:00
|
|
|
|
2016-09-04 00:22:11 +02:00
|
|
|
// Draw column headers
|
|
|
|
bool sortable;
|
|
|
|
GUI<bool>::GetSetting(this, "sortable", sortable);
|
2015-06-17 11:10:50 +02:00
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
CStr selectedColumn;
|
|
|
|
GUI<CStr>::GetSetting(this, "selected_column", selectedColumn);
|
|
|
|
|
2016-09-04 00:22:11 +02:00
|
|
|
int selectedColumnOrder;
|
|
|
|
GUI<int>::GetSetting(this, "selected_column_order", selectedColumnOrder);
|
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
CColor color;
|
|
|
|
GUI<CColor>::GetSetting(this, _textcolor, color);
|
2016-09-04 00:22:11 +02:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
float xpos = 0;
|
2016-08-31 21:29:48 +02:00
|
|
|
for (size_t col = 0; col < m_Columns.size(); ++col)
|
2015-08-21 19:08:41 +02:00
|
|
|
{
|
2017-03-28 02:25:17 +02:00
|
|
|
bool hidden = false;
|
|
|
|
GUI<bool>::GetSetting(this, "hidden_" + m_Columns[col].m_Id, hidden);
|
|
|
|
if (hidden)
|
|
|
|
continue;
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
// Check if it's a decimal value, and if so, assume relative positioning.
|
2016-08-31 21:29:48 +02:00
|
|
|
float width = m_Columns[col].m_Width;
|
|
|
|
if (m_Columns[col].m_Width < 1 && m_Columns[col].m_Width > 0)
|
|
|
|
width *= m_TotalAvailableColumnWidth;
|
2015-08-21 19:08:41 +02:00
|
|
|
|
|
|
|
CPos leftTopCorner = m_CachedActualSize.TopLeft() + CPos(xpos, 0);
|
2015-09-02 19:55:02 +02:00
|
|
|
|
|
|
|
// Draw sort arrows in colum header
|
2016-09-04 00:22:11 +02:00
|
|
|
if (sortable)
|
|
|
|
{
|
|
|
|
CGUISpriteInstance* sprite;
|
|
|
|
if (selectedColumn == m_Columns[col].m_Id)
|
|
|
|
{
|
|
|
|
if (selectedColumnOrder == 0)
|
|
|
|
LOGERROR("selected_column_order must not be 0");
|
|
|
|
|
|
|
|
if (selectedColumnOrder != -1)
|
|
|
|
GUI<CGUISpriteInstance>::GetSettingPointer(this, "sprite_asc", sprite);
|
|
|
|
else
|
|
|
|
GUI<CGUISpriteInstance>::GetSettingPointer(this, "sprite_desc", sprite);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
GUI<CGUISpriteInstance>::GetSettingPointer(this, "sprite_not_sorted", sprite);
|
|
|
|
|
|
|
|
GetGUI()->DrawSprite(*sprite, cell_id, bz + 0.1f, CRect(leftTopCorner + CPos(width - 16, 0), leftTopCorner + CPos(width, 16)));
|
|
|
|
}
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-09-02 19:55:02 +02:00
|
|
|
// Draw column header text
|
2016-08-31 21:29:48 +02:00
|
|
|
DrawText(col, color, leftTopCorner + CPos(0, 4), bz + 0.1f, rect_head);
|
2015-08-21 19:08:41 +02:00
|
|
|
xpos += width;
|
|
|
|
}
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-09-02 19:55:02 +02:00
|
|
|
// Draw list items for each column
|
2016-08-31 21:29:48 +02:00
|
|
|
const size_t objectsCount = m_Columns.size();
|
2015-08-21 19:08:41 +02:00
|
|
|
for (size_t i = 0; i < pList->m_Items.size(); ++i)
|
|
|
|
{
|
|
|
|
if (m_ItemsYPositions[i+1] - scroll < 0 ||
|
|
|
|
m_ItemsYPositions[i] - scroll > rect.GetHeight())
|
|
|
|
continue;
|
2015-07-30 03:33:48 +02:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
const float rowHeight = m_ItemsYPositions[i+1] - m_ItemsYPositions[i];
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
// Clipping area (we'll have to substract the scrollbar)
|
|
|
|
CRect cliparea = GetListRect();
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
if (scrollbar)
|
|
|
|
{
|
|
|
|
if (cliparea.right > GetScrollBar(0).GetOuterRect().left &&
|
|
|
|
cliparea.right <= GetScrollBar(0).GetOuterRect().right)
|
|
|
|
cliparea.right = GetScrollBar(0).GetOuterRect().left;
|
2013-11-07 21:07:24 +01:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
if (cliparea.left >= GetScrollBar(0).GetOuterRect().left &&
|
|
|
|
cliparea.left < GetScrollBar(0).GetOuterRect().right)
|
|
|
|
cliparea.left = GetScrollBar(0).GetOuterRect().right;
|
|
|
|
}
|
2015-07-30 03:33:48 +02:00
|
|
|
|
2015-09-02 19:55:02 +02:00
|
|
|
// Draw all items for that column
|
2015-08-21 19:08:41 +02:00
|
|
|
xpos = 0;
|
2016-08-31 21:29:48 +02:00
|
|
|
for (size_t col = 0; col < objectsCount; ++col)
|
2015-08-21 19:08:41 +02:00
|
|
|
{
|
2017-03-28 02:25:17 +02:00
|
|
|
bool hidden = false;
|
|
|
|
GUI<bool>::GetSetting(this, "hidden_" + m_Columns[col].m_Id, hidden);
|
|
|
|
if (hidden)
|
|
|
|
continue;
|
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
// Determine text position and width
|
|
|
|
const CPos textPos = rect.TopLeft() + CPos(xpos, -scroll + m_ItemsYPositions[i]);
|
2015-07-30 03:33:48 +02:00
|
|
|
|
2016-08-31 21:29:48 +02:00
|
|
|
float width = m_Columns[col].m_Width;
|
2015-08-21 19:08:41 +02:00
|
|
|
// Check if it's a decimal value, and if so, assume relative positioning.
|
2016-08-31 21:29:48 +02:00
|
|
|
if (m_Columns[col].m_Width < 1 && m_Columns[col].m_Width > 0)
|
|
|
|
width *= m_TotalAvailableColumnWidth;
|
2015-07-30 03:33:48 +02:00
|
|
|
|
2015-08-21 19:08:41 +02:00
|
|
|
// Clip text to the column (to prevent drawing text into the neighboring column)
|
|
|
|
CRect cliparea2 = cliparea;
|
|
|
|
cliparea2.right = std::min(cliparea2.right, textPos.x + width);
|
|
|
|
cliparea2.bottom = std::min(cliparea2.bottom, textPos.y + rowHeight);
|
|
|
|
|
2015-09-02 19:55:02 +02:00
|
|
|
// Draw list item
|
2016-08-31 21:29:48 +02:00
|
|
|
DrawText(objectsCount * (i +/*Heading*/1) + col, m_Columns[col].m_TextColor, textPos, bz + 0.1f, cliparea2);
|
2015-08-21 19:08:41 +02:00
|
|
|
xpos += width;
|
2013-11-07 21:07:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|