]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blobdiff - qcsrc/menu/xonotic/hudskinlist.qc
Avoid loading hud skin list on menu start
[xonotic/xonotic-data.pk3dir.git] / qcsrc / menu / xonotic / hudskinlist.qc
index c06e22535f5d7044f4370bf9bf01cec88a38aada..b3bdc624332ea2cec2991cc4aceece1e18066fc9 100644 (file)
@@ -10,6 +10,9 @@ CLASS(XonoticHUDSkinList, XonoticListBox)
        METHOD(XonoticHUDSkinList, getHUDSkins, void(entity));
        METHOD(XonoticHUDSkinList, setHUDSkin, void(entity));
        METHOD(XonoticHUDSkinList, hudskinName, string(entity, float));
+       METHOD(XonoticHUDSkinList, hudskinPath, string(entity, float));
+       METHOD(XonoticHUDSkinList, hudskinTitle, string(entity, float));
+       METHOD(XonoticHUDSkinList, hudskinAuthor, string(entity, float));
        METHOD(XonoticHUDSkinList, doubleClickListBoxItem, void(entity, float, vector));
        METHOD(XonoticHUDSkinList, keyDown, float(entity, float, float, float));
        METHOD(XonoticHUDSkinList, destroy, void(entity));
@@ -25,6 +28,7 @@ CLASS(XonoticHUDSkinList, XonoticListBox)
 
        ATTRIB(XonoticHUDSkinList, filterString, string, string_null)
        ATTRIB(XonoticHUDSkinList, delayedRefreshTime, float, 0)
+       ATTRIB(XonoticHUDSkinList, savedName, string, string_null)
 ENDCLASS(XonoticHUDSkinList)
 
 #ifndef IMPLEMENTATION
@@ -36,6 +40,7 @@ void SetHUDSkin_Click(entity btn, entity me);
 #endif
 void HUDSkinList_Refresh_Click(entity btn, entity me);
 void HUDSkinList_Filter_Change(entity box, entity me);
+void HUDSkinList_SavedName_Change(entity box, entity me);
 #endif
 
 #ifdef IMPLEMENTATION
@@ -51,21 +56,33 @@ entity makeXonoticHUDSkinList()
 void XonoticHUDSkinList_configureXonoticHUDSkinList(entity me)
 {
        me.configureXonoticListBox(me);
-       me.getHUDSkins(me);
+       me.nItems = 0;
 }
 
+const float HUDSKINPARM_NAME = 0;
+const float HUDSKINPARM_PATH = 1;
+const float HUDSKINPARM_TITLE = 2;
+const float HUDSKINPARM_AUTHOR = 3;
+const float HUDSKINPARM_COUNT = 4;
 string XonoticHUDSkinList_hudskinName(entity me, float i)
 {
-       string s;
-       s = bufstr_get(me.listHUDSkin, i);
-
-       if(substring(s, 0, 1) == "/")
-               s = substring(s, 1, strlen(s) - 1);  // remove the first forward slash
-
-       return s;
+       return bufstr_get(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_NAME);
+}
+string XonoticHUDSkinList_hudskinPath(entity me, float i)
+{
+       return bufstr_get(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_PATH);
+}
+string XonoticHUDSkinList_hudskinTitle(entity me, float i)
+{
+       return bufstr_get(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_TITLE);
+}
+string XonoticHUDSkinList_hudskinAuthor(entity me, float i)
+{
+       return bufstr_get(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_AUTHOR);
 }
 
-void getAllHUDSkins(entity me, string subdir)
+// subdir can be a regular expression
+void getHUDSkinFiles(entity me, int sortbuf, string subdir)
 {
        string s;
        if(me.filterString)
@@ -73,23 +90,75 @@ void getAllHUDSkins(entity me, string subdir)
        else
                s = "*";
        s = strcat(subdir, "hud_", s, ".cfg");
-       float strlen_subdir = strlen(subdir);
 
-       float list, i, n;
-       list = search_begin(s, false, true);
+       int list = search_begin(s, false, true);
        if(list >= 0)
        {
-               n = search_getsize(list);
-               for(i = 0; i < n; ++i)
+               int n = search_getsize(list);
+               for(int i = 0; i < n; ++i)
                {
-                       s = search_getfilename(list, i); // get initial full file name
-                       s = substring(s, strlen_subdir + 4, (strlen(s) - strlen_subdir - 4 - 4)); // remove "hud_" prefix and ".cfg" suffix
-                       bufstr_add(me.listHUDSkin, s, true);
+                       string s = search_getfilename(list, i);
+                       int subdir_ofs = strstrofs(s, "/", 0);
+                       if(subdir_ofs >= 0)
+                       {
+                               int ofs = subdir_ofs;
+                               while(ofs != -1)
+                               {
+                                       subdir_ofs = ofs;
+                                       ofs = strstrofs(s, "/", subdir_ofs + 1);
+                               }
+                       }
+
+                       if(subdir_ofs == -1)
+                               bufstr_add(sortbuf, s, true);
+                       else
+                       {
+                               subdir = substring(s, 0, subdir_ofs);
+                               string filename = substring(s, subdir_ofs + 1, -1);
+                               // invert path and filename position so we can sort sortbuf by filename
+                               bufstr_add(sortbuf, strcat(filename, "/", subdir), true);
+                       }
                }
                search_end(list);
        }
 }
 
+void getAllHUDSkins(entity me, int sortbuf)
+{
+       int n = buf_getsize(sortbuf);
+       for(int i = 0; i < n; ++i)
+       {
+               string entry = bufstr_get(sortbuf, i);
+               int ofs = strstrofs(entry, "/", 0);
+               string s = "";
+               string filename = entry;
+               if(ofs >= 0)
+               {
+                       s = substring(entry, ofs + 1, -1); // skip initial "/"
+                       s = strcat(s, "/");
+                       bufstr_set(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_PATH, s);
+                       filename = strcat(s, substring(entry, 0, ofs));
+               }
+               else
+                       ofs = strlen(entry);
+               s = substring(entry, 4, ofs - 4 - 4); // remove "hud_" prefix and ".cfg" suffix
+               bufstr_set(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_NAME, s);
+
+               int fh = fopen(filename, FILE_READ);
+               if(fh < 0)
+                       continue;
+               while((s = fgets(fh)) && substring(s, 0, 2) == "//")
+               {
+                       tokenize_console(substring(s, 2, -1));
+                       if(argv(0) == "title")
+                               bufstr_set(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_TITLE, argv(1));
+                       else if(argv(0) == "author")
+                               bufstr_set(me.listHUDSkin, i * HUDSKINPARM_COUNT + HUDSKINPARM_AUTHOR, argv(1));
+               }
+               fclose(fh);
+       }
+}
+
 void XonoticHUDSkinList_getHUDSkins(entity me)
 {
        if (me.listHUDSkin >= 0)
@@ -100,11 +169,13 @@ void XonoticHUDSkinList_getHUDSkins(entity me)
                me.nItems = 0;
                return;
        }
-       getAllHUDSkins(me, "");
-       getAllHUDSkins(me, "data/");
-       me.nItems = buf_getsize(me.listHUDSkin);
-       if(me.nItems > 0)
-               buf_sort(me.listHUDSkin, 128, false);
+       int sortbuf = buf_create();
+       getHUDSkinFiles(me, sortbuf, "");
+       getHUDSkinFiles(me, sortbuf, "data/");
+       buf_sort(sortbuf, 128, 0);
+       getAllHUDSkins(me, sortbuf);
+       buf_del(sortbuf);
+       me.nItems = buf_getsize(me.listHUDSkin) / HUDSKINPARM_COUNT;
 }
 
 void XonoticHUDSkinList_destroy(entity me)
@@ -128,7 +199,7 @@ void XonoticHUDSkinList_resizeNotify(entity me, vector relOrigin, vector relSize
 
 void XonoticHUDSkinList_drawListBoxItem(entity me, int i, vector absSize, bool isSelected, bool isFocused)
 {
-       string s;
+       string s, s2;
        if(isSelected)
                draw_Fill('0 0 0', '1 1 0', SKINCOLOR_LISTBOX_SELECTED, SKINALPHA_LISTBOX_SELECTED);
        else if(isFocused)
@@ -137,7 +208,12 @@ void XonoticHUDSkinList_drawListBoxItem(entity me, int i, vector absSize, bool i
                draw_Fill('0 0 0', '1 1 0', SKINCOLOR_LISTBOX_FOCUSED, me.focusedItemAlpha);
        }
 
-       s = me.hudskinName(me,i);
+       s = me.hudskinTitle(me, i);
+       if(s == "")
+               s = me.hudskinName(me, i);
+       s2 = me.hudskinAuthor(me, i);
+       if(s2 != "")
+               s = strcat(s, " (", s2, ")");
        s = draw_TextShortenToWidth(s, me.columnNameSize, 0, me.realFontSize);
        draw_Text(me.realUpperMargin * eY + (me.columnNameOrigin + 0.00 * (me.columnNameSize - draw_TextWidth(s, 0, me.realFontSize))) * eX, s, me.realFontSize, SKINCOLOR_TEXT, SKINALPHA_TEXT, 1);
 }
@@ -153,6 +229,17 @@ void HUDSkinList_Refresh_Click(entity btn, entity me)
        me.setSelected(me, 0); //always select the first element after a list update
 }
 
+void HUDSkinList_SavedName_Change(entity box, entity me)
+{
+       if(me.savedName)
+               strunzone(me.savedName);
+
+       if(box.text != "")
+               me.savedName = strzone(box.text);
+       else
+               me.savedName = string_null;
+}
+
 void HUDSkinList_Filter_Change(entity box, entity me)
 {
        if(me.filterString)
@@ -173,7 +260,10 @@ void HUDSkinList_Filter_Change(entity box, entity me)
 
 void SaveHUDSkin_Click(entity btn, entity me)
 {
-       localcmd("hud save myconfig\n");
+       string s = me.savedName;
+       if(s == "")
+               s = "myconfig";
+       localcmd(sprintf("hud save \"%s\"\n", s));
        me.delayedRefreshTime = time + 1;
 }
 
@@ -189,8 +279,8 @@ void XonoticHUDSkinList_draw(entity me)
 
 void XonoticHUDSkinList_setHUDSkin(entity me)
 {
-       string s = me.hudskinName(me, me.selectedItem);
-       localcmd("exec \"hud_", s, ".cfg\"\n");
+       string cfg = strcat(me.hudskinPath(me, me.selectedItem), "hud_", me.hudskinName(me, me.selectedItem), ".cfg");
+       localcmd("exec \"", cfg, "\"\n");
 }
 
 void SetHUDSkin_Click(entity btn, entity me)