winamp/Src/Wasabi/api/skin/guitree.cpp
2024-09-24 14:54:57 +02:00

379 lines
9.8 KiB
C++

#include <precomp.h>
#include "guitree.h"
#include <api/skin/skinparse.h>
#include <api/syscb/callbacks/wndcb.h>
#ifdef WASABI_COMPILE_WNDMGR
#include <api/wndmgr/autopopup.h>
#endif
#include <api/skin/skin.h>
#include <bfc/bfc_assert.h>
GuiTree *guiTree=NULL;
GuiTreeItem::GuiTreeItem(int type, const wchar_t *name, ifc_xmlreaderparams *p, int scriptid, const wchar_t *path, GUID g, const wchar_t *windowtype, const wchar_t *xuitag)
{
if (p)
{
for (size_t i=0;i!=p->getNbItems();i++)
params.addItem(p->getItemName(i), p->getItemValue(i));
}
object_type = type;
object_name = name;
scriptId = scriptid;
rootpath = path;
guid = g;
idx=-1;
seccount = incrementor++;
wndtype = windowtype;
tag = xuitag;
}
int GuiTreeItem::incrementor=0;
GuiTreeItem::~GuiTreeItem() {
}
int GuiTreeItem::getScriptId() {
return scriptId;
}
ifc_xmlreaderparams *GuiTreeItem::getParams()
{
// TODO: benski> helps replicate old logic.. we really fix the parts of the code that are assuming getValue() always succeeds with this returned object
if (params.getNbItems())
return &params;
else
return 0;
}
const wchar_t *GuiTreeItem::getXmlRootPath() {
return rootpath;
}
int GuiTreeItem::getType() {
return object_type;
}
const wchar_t *GuiTreeItem::getName()
{
return object_name;
}
SkinItem *GuiTreeItem::getAncestor() {
if (object_type == XML_TAG_GROUPDEF)
return guiTree->getGroupDefAncestor(this);
if (object_type == XML_TAG_CONTAINER)
return guiTree->getContainerAncestor(this);
return NULL;
}
GuiTree::GuiTree() {
cached=-1;
cachedtype=-1;
cached_guid_idx=-1;
cached_guid=INVALID_GUID;
lastdefinedgroupdef = NULL;
}
GuiTree::~GuiTree() {
list.deleteAll();
groupdefs.removeAll();
wndtypes.removeAll();
//groupdefsbyguid.removeAll();
}
void GuiTree::addItem(int object_type, const wchar_t *name, ifc_xmlreaderparams *params, int scriptid, const wchar_t *rootpath) {
cached = -1;
cached_guid_idx = -1;
GUID guid=INVALID_GUID;
const wchar_t *wtype=NULL;
int has_ancestor=0;
const wchar_t *xuitag=NULL;
const wchar_t *_id = NULL;
if (params)
{
_id = params->getItemValue(L"id");
xuitag = params->getItemValue(L"xuitag");
#ifdef WASABI_COMPILE_WNDMGR
if (params->getItemValueInt(L"register_autopopup"))
AutoPopup::registerGroupId(scriptid, _id, params->getItemValue(L"name"));
#endif
const wchar_t *strguid = params->getItemValue(L"guid");
if (strguid)
guid = nsGUID::fromCharW(strguid);
wtype = params->getItemValue(L"windowtype");
}
if (object_type == XML_TAG_GROUPDEF && params)
{
has_ancestor = (getGroupDef(_id) != NULL);
}
GuiTreeItem *item = new GuiTreeItem(object_type, name, params, scriptid, rootpath, guid, wtype, xuitag);
item->setIdx(list.getNumItems());
list.addItem(item);
if (object_type == XML_TAG_GROUPDEF && params) {
lastdefinedgroupdef = item;
// groupdefsbyguid.setAutoSort(0);
groupdefs.addItem(item);
// groupdefsbyguid.addItem(item);
if (wtype && *wtype) {
wndtypes.addItem(item);
deferredInvalidateType(wtype);
}
if (xuitag && *xuitag) {
xuigroupdefs.addItem(item);
}
if (has_ancestor)
deferredInvalidateGroup(_id);
} else if (object_type == XML_TAG_CONTAINER && params && (params->getItemValueInt(L"dynamic", 0) == 1)) {
containers_by_id.setAutoSort(0);
containers_by_id.addItem(item);
}
}
int GuiTree::getNumObject() {
return list.getNumItems();
}
int GuiTree::getNumObject(int object_type) {
if (cachedtype == object_type && cached != -1) return cached;
int n=0;
for (int i=0;i<list.getNumItems();i++) {
GuiTreeItem *it = list.enumItem(i);
if (it->getType() == object_type)
n++;
}
cachedtype = object_type;
cached = n;
return n;
}
SkinItem *GuiTree::getObject(int object_type, int nth) {
int n=0;
for (int i=0;i<list.getNumItems();i++) {
GuiTreeItem *it = list.enumItem(i);
if (it && it->getType() == object_type) {
if (n++ == nth) return it;
}
}
return NULL;
}
SkinItem *GuiTree::getGroupDef(const wchar_t *id)
{
return groupdefs.findLastItem(id);
}
SkinItem *GuiTree::getXuiGroupDef(const wchar_t *xuitag)
{
xuigroupdefs.sort(); // doesn't sort if not necessary
return xuigroupdefs.findLastItem(xuitag);
}
SkinItem *GuiTree::getGroupDefAncestor(SkinItem *_item)
{
//groupdefs.sort(); // doesn't sort if not necessary
int pos = -1;
GuiTreeItem *item = static_cast<GuiTreeItem *>(_item);
if (!item) return NULL;
ASSERT(item->getParams() != NULL);
const wchar_t *iid = item->getParams()->getItemValue(L"id");
pos = groupdefs.searchItem(item);
if (pos <= 0) return NULL;
pos--;
GuiTreeItem *ritem = groupdefs.enumItem(pos);
if (!ritem) return NULL;
ASSERT(ritem->getParams() != NULL);
if (WCSICMP(iid, ritem->getParams()->getItemValue(L"id"))) return NULL;
return ritem;
}
SkinItem *GuiTree::getContainerAncestor(SkinItem *_item) {
int pos = -1;
GuiTreeItem *item = static_cast<GuiTreeItem *>(_item);
if (!item) return NULL;
ASSERT(item->getParams() != NULL);
const wchar_t *iid = item->getParams()->getItemValue(L"id");
pos = containers_by_id.searchItem(item);
if (pos <= 0) return NULL;
pos--;
GuiTreeItem *ritem = containers_by_id.enumItem(pos);
if (!ritem) return NULL;
ASSERT(ritem->getParams() != NULL);
if (WCSICMP(iid, ritem->getParams()->getItemValue(L"id"))) return NULL;
return ritem;
}
/*int GuiTree::getGroupDef(GUID guid) {
groupdefsbyguid.sort(); // doesn't sort if not necessary
GuiTreeItem *item = groupdefsbyguid.findItem(reinterpret_cast<const char *>(&guid));
if (item)
return item->getIdx();
return -1;
}*/
SkinItem *GuiTree::enumGroupDefOfType(const wchar_t *type, int n) {
int c = 0;
foreach(wndtypes)
GuiTreeItem *item = wndtypes.getfor();
if (WCSCASEEQLSAFE(type, item->getWindowType())) {
if (c == n) {
/* if (item->getGuid() != INVALID_GUID) { // see if it has a GUID, in which case we need to instantiate the last overriden version
int n = getGroupDef(item->getGuid());
if (n != -1) return n;
}*/
// take its groupid and find its latest overriden version
ifc_xmlreaderparams *p = item->getParams();
if (p)
{
const wchar_t *id = p->getItemValue(L"id");
if (id)
{
SkinItem *m = getGroupDef(id);
if (m != NULL) return m;
}
}
return item;
}
c++;
}
endfor;
return NULL;
}
int GuiTree::getObjectType(SkinItem *item) {
GuiTreeItem *i = static_cast<GuiTreeItem *>(item);
return i->getType();
}
PtrList<GuiTreeItem> *GuiTree::getList() {
return &list;
}
PtrList<GuiTreeItem> *GuiTree::getGroupList() {
return &groupdefs;
}
SkinItem *GuiTree::getContainerById(const wchar_t *id)
{
containers_by_id.sort(); // doesn't sort if not necessary
return containers_by_id.findLastItem(id);
}
void GuiTree::removeSkinPart(int scriptid) {
for (int i=0;i<list.getNumItems();i++) {
GuiTreeItem *item = list.enumItem(i);
if (item->getScriptId() == scriptid) {
ifc_xmlreaderparams *par = item->getParams();
if (item->getType() == XML_TAG_CONTAINER && par != NULL) {
int p = containers_by_id.searchItem(item);
if (p != -1)
containers_by_id.removeByPos(p);
}
if (item->getType() == XML_TAG_GROUPDEF && par != NULL)
{
const wchar_t *grpid = par->getItemValue(L"id");
if (grpid)
{
int p = groupdefs.searchItem(item);
if (p != -1)
groupdefs.removeByPos(p);
p = xuigroupdefs.searchItem(item);
if (p != -1)
xuigroupdefs.removeByPos(p);
p = wndtypes.searchItem(item);
if (p != -1) {
deferredInvalidateType(item->getWindowType());
wndtypes.removeByPos(p);
}
deferredInvalidateGroup(grpid);
}
}
delete item;
list.removeByPos(i);
i--;
}
}
foreach(list)
list.getfor()->setIdx(foreach_index);
endfor;
}
void GuiTree::reset() {
list.deleteAll();
groupdefs.removeAll();
wndtypes.removeAll();
xuigroupdefs.removeAll();
containers_by_id.removeAll();
}
void GuiTree::deferredInvalidateGroup(const wchar_t *id) {
if (!Skin::isDynamicGroupReloadEnabled()) return;
if (!id || !*id) return;
StringW *s = new StringW;
s->setValue(id);
GuiTreeCB *cb = new GuiTreeCB;
cb->cmd = INVALIDATEGRP;
cb->ptr = s;
timerclient_postDeferredCallback(CB_GUITREE, reinterpret_cast<intptr_t>(cb));
}
void GuiTree::deferredInvalidateType(const wchar_t *type) {
if (!Skin::isDynamicGroupReloadEnabled()) return;
StringW *s = new StringW;
s->setValue(type);
GuiTreeCB *cb = new GuiTreeCB;
cb->cmd = INVALIDATETYPE;
cb->ptr = s;
timerclient_postDeferredCallback(CB_GUITREE, reinterpret_cast<intptr_t>(cb));
}
int GuiTree::timerclient_onDeferredCallback(intptr_t param1, intptr_t param2)
{
if (param1 == CB_GUITREE) {
GuiTreeCB *c = reinterpret_cast<GuiTreeCB *>(param2);
switch (c->cmd) {
case INVALIDATEGRP: {
StringW *s = reinterpret_cast<StringW *>(c->ptr);
WndInfo wi;
wi.groupid = s->getValue();
wi.wndtype = NULL;
wi.guid = INVALID_GUID;
WASABI_API_SYSCB->syscb_issueCallback(SysCallback::WINDOW, WndCallback::GROUPCHANGE, reinterpret_cast<intptr_t>(&wi));
delete s;
break;
}
case INVALIDATETYPE: {
StringW *s = reinterpret_cast<StringW *>(c->ptr);
WndInfo wi;
ZERO(wi);
wi.wndtype = s->getValue();
wi.guid = INVALID_GUID;
WASABI_API_SYSCB->syscb_issueCallback(SysCallback::WINDOW, WndCallback::TYPECHANGE, reinterpret_cast<intptr_t>(&wi));
delete s;
break;
}
}
return 1;
}
return GUITREE_PARENT::timerclient_onDeferredCallback(param1, param2);
}
int GuiTree::getObjectIdx(SkinItem *item) {
if (item == NULL) return -1;
return (static_cast<GuiTreeItem*>(item))->getIdx();
}