+void set_preference_backend(const char *key, size_t keylen,
+ long lvalue,
+ StrBuf *value,
+ long lPrefType,
+ int save_to_server,
+ PrefDef *PrefType)
+{
+ wcsession *WCC = WC;
+ void *vPrefDef;
+ void *vPrefB;
+ Preference *Pref;
+
+ Pref = (Preference*) malloc(sizeof(Preference));
+ memset(Pref, 0, sizeof(Preference));
+ Pref->Key = NewStrBufPlain(key, keylen);
+
+ if ((PrefType == NULL) &&
+ GetHash(PreferenceHooks, SKEY(Pref->Key), &vPrefDef) &&
+ (vPrefDef != NULL))
+ PrefType = (PrefDef*) vPrefDef;
+
+ if (PrefType != NULL)
+ {
+ Pref->Type = PrefType;
+ Pref->eFlatPrefType = PrefType->eType;
+ if (Pref->Type->eType != lPrefType)
+ syslog(1, "warning: saving preference with wrong type [%s] %d != %ld \n",
+ key, Pref->Type->eType, lPrefType);
+ switch (Pref->Type->eType)
+ {
+ case PRF_UNSET: /* default to string... */
+ case PRF_STRING:
+ Pref->Val = value;
+ Pref->decoded = 1;
+ break;
+ case PRF_INT:
+ Pref->lval = lvalue;
+ Pref->Val = value;
+ if (Pref->Val == NULL)
+ Pref->Val = NewStrBufPlain(NULL, 64);
+ StrBufPrintf(Pref->Val, "%ld", lvalue);
+ Pref->decoded = 1;
+ break;
+ case PRF_QP_STRING:
+ Pref->DeQPed = value;
+ Pref->Val = NewStrBufPlain(NULL, StrLength(Pref->DeQPed) * 3);
+ StrBufEUid_escapize(Pref->Val, Pref->DeQPed);
+ Pref->decoded = 1;
+ break;
+ case PRF_YESNO:
+ Pref->lval = lvalue;
+ if (lvalue)
+ Pref->Val = NewStrBufPlain(HKEY("yes"));
+ else
+ Pref->Val = NewStrBufPlain(HKEY("no"));
+ Pref->decoded = 1;
+ break;
+ }
+ if (Pref->Type->OnLoad != NULL)
+ Pref->Type->OnLoad(Pref->Val, Pref->lval);
+ }
+ else {
+ Pref->eFlatPrefType = lPrefType;
+ switch (lPrefType)
+ {
+ case PRF_STRING:
+ Pref->Val = value;
+ Pref->decoded = 1;
+ break;
+ case PRF_INT:
+ Pref->lval = lvalue;
+ Pref->Val = value;
+ if (Pref->Val == NULL)
+ Pref->Val = NewStrBufPlain(NULL, 64);
+ StrBufPrintf(Pref->Val, "%ld", lvalue);
+ Pref->decoded = 1;
+ break;
+ case PRF_QP_STRING:
+ Pref->DeQPed = value;
+ Pref->Val = NewStrBufPlain(NULL, StrLength(Pref->DeQPed) * 3);
+ StrBufEUid_escapize(Pref->Val, Pref->DeQPed);
+ Pref->decoded = 1;
+ break;
+ case PRF_YESNO:
+ Pref->lval = lvalue;
+ if (lvalue)
+ Pref->Val = NewStrBufPlain(HKEY("yes"));
+ else
+ Pref->Val = NewStrBufPlain(HKEY("no"));
+ Pref->decoded = 1;
+ break;
+ }
+ }
+
+ if ((save_to_server != 0) &&
+ GetHash(WCC->hash_prefs, key, keylen, &vPrefB) &&
+ (vPrefB != NULL) &&
+ (compare_preference (Pref, vPrefB) == 0))
+ save_to_server = 0;
+
+ Put(WCC->hash_prefs, key, keylen, Pref, DestroyPreference);
+
+ if (save_to_server) WCC->SavePrefsToServer = 1;
+}
+
+void set_PREFERENCE(const char *key, size_t keylen, StrBuf *value, int save_to_server)
+{
+ set_preference_backend(key, keylen, 0, value, PRF_STRING, save_to_server, NULL);
+}
+
+int get_PREF_LONG(const char *key, size_t keylen, long *value, long Default)
+{
+ Preference *Pref;
+ int Ret;
+
+ Ret = get_pref_backend(key, keylen, &Pref);
+ if (Ret == 0) {
+ *value = Default;
+ return 0;
+ }
+
+ if (Pref->decoded)
+ *value = Pref->lval;
+ else {
+ *value = Pref->lval = atol(ChrPtr(Pref->Val));
+ Pref->decoded = 1;
+ }
+ return Ret;
+}
+
+
+void set_PREF_LONG(const char *key, size_t keylen, long value, int save_to_server)
+{
+ set_preference_backend(key, keylen, value, NULL, PRF_INT, save_to_server, NULL);
+}
+
+int get_PREF_YESNO(const char *key, size_t keylen, int *value, int Default)
+{
+ Preference *Pref;
+ int Ret;
+
+ Ret = get_pref_backend(key, keylen, &Pref);
+ if (Ret == 0) {
+ *value = Default;
+ return 0;
+ }
+
+ if (Pref->decoded)
+ *value = Pref->lval;
+ else {
+ *value = Pref->lval = strcmp(ChrPtr(Pref->Val), "yes") == 0;
+ Pref->decoded = 1;
+ }
+ return Ret;
+}
+
+void set_PREF_YESNO(const char *key, size_t keylen, long value, int save_to_server)
+{
+ set_preference_backend(key, keylen, value, NULL, PRF_YESNO, save_to_server, NULL);
+}
+
+int get_room_prefs_backend(const char *key, size_t keylen,
+ Preference **Pref)
+{
+ StrBuf *pref_name;
+ int Ret;
+
+ pref_name = NewStrBufPlain (HKEY("ROOM:"));
+ StrBufAppendBuf(pref_name, WC->CurRoom.name, 0);
+ StrBufAppendBufPlain(pref_name, HKEY(":"), 0);
+ StrBufAppendBufPlain(pref_name, key, keylen, 0);
+ Ret = get_pref_backend(SKEY(pref_name), Pref);
+ FreeStrBuf(&pref_name);
+
+ return Ret;
+}
+
+const StrBuf *get_X_PREFS(const char *key, size_t keylen,
+ const char *xkey, size_t xkeylen)
+{
+ int ret;
+ StrBuf *pref_name;
+ Preference *Prf;
+
+ pref_name = NewStrBufPlain (HKEY("XPREF:"));
+ StrBufAppendBufPlain(pref_name, xkey, xkeylen, 0);
+ StrBufAppendBufPlain(pref_name, HKEY(":"), 0);
+ StrBufAppendBufPlain(pref_name, key, keylen, 0);
+
+ ret = get_pref_backend(SKEY(pref_name), &Prf);
+ FreeStrBuf(&pref_name);