diff options
author | George Hazan <george.hazan@gmail.com> | 2016-04-08 10:49:29 +0000 |
---|---|---|
committer | George Hazan <george.hazan@gmail.com> | 2016-04-08 10:49:29 +0000 |
commit | 017b7066ab8c82880a62f9208e25c36c43c76a03 (patch) | |
tree | ec1c940848a51bfd77b0ac1c87a53d8e99f9ae7a /plugins/Variables/src/contact.cpp | |
parent | 6cb46f5857189fc42e05981ca2c01bb00e65e59f (diff) |
Variables:
- massive code simplification;
- old shit removed and replaced with lists;
- version bump.
git-svn-id: http://svn.miranda-ng.org/main/trunk@16613 1316c22d-e87f-b044-9b9b-93d7a3e3ba9c
Diffstat (limited to 'plugins/Variables/src/contact.cpp')
-rw-r--r-- | plugins/Variables/src/contact.cpp | 287 |
1 files changed, 121 insertions, 166 deletions
diff --git a/plugins/Variables/src/contact.cpp b/plugins/Variables/src/contact.cpp index 995a4ce610..ecd0da1e65 100644 --- a/plugins/Variables/src/contact.cpp +++ b/plugins/Variables/src/contact.cpp @@ -73,20 +73,31 @@ static builtinCnfs[] = { CCNF_PROTOID, STR_PROTOID }
};
-typedef struct
+/* contact cache entry */
+struct CONTACTCE
{
+ ~CONTACTCE()
+ {
+ mir_free(tszContact);
+ }
+
TCHAR* tszContact;
+ int flags;
MCONTACT hContact;
- DWORD flags;
-} CONTACTCE; /* contact cache entry */
+};
+
+static int SortContactCache(const CONTACTCE *p1, const CONTACTCE *p2)
+{
+ if (p1->flags != p2->flags)
+ return (p1->flags > p2->flags) ? 1 : -1;
+
+ return _tcscmp(p1->tszContact, p2->tszContact);
+}
/* cache for 'getcontactfromstring' service */
-static CONTACTCE *cce = NULL;
-static int cacheSize = 0;
+static OBJLIST<CONTACTCE> arContactCache(20, SortContactCache);
static mir_cs csContactCache;
-static HANDLE hContactSettingChangedHook;
-
/*
converts a string into a CNF_ type
*/
@@ -195,163 +206,134 @@ TCHAR* getContactInfoT(BYTE type, MCONTACT hContact) /*
MS_VARS_GETCONTACTFROMSTRING
*/
-int getContactFromString(CONTACTSINFO *ci)
+MCONTACT getContactFromString(const TCHAR *tszContact, DWORD dwFlags, int nMatch)
{
/* service to retrieve a contact's HANDLE from a given string */
- if (ci == NULL)
- return -1;
-
- TCHAR *tszContact;
- if (ci->flags & CI_UNICODE)
- tszContact = NEWTSTR_ALLOCA(ci->tszContact);
- else {
- WCHAR* tmp = mir_a2t(ci->szContact);
- tszContact = NEWTSTR_ALLOCA(tmp);
- mir_free(tmp);
+ if (tszContact == NULL || *tszContact == 0)
+ return INVALID_CONTACT_ID;
+
+ bool bReturnCount;
+ if (dwFlags & CI_NEEDCOUNT) {
+ dwFlags &= ~CI_NEEDCOUNT;
+ bReturnCount = true;
}
- if ((tszContact == NULL) || (mir_tstrlen(tszContact) == 0))
- return -1;
+ else bReturnCount = false;
- ci->hContacts = NULL;
- int count = 0;
/* search the cache */
{
+ CONTACTCE tmp = { (TCHAR*)tszContact, dwFlags, 0 };
+
mir_cslock lck(csContactCache);
- for (int i = 0; i < cacheSize; i++) {
- if ((!mir_tstrcmp(cce[i].tszContact, tszContact)) && (ci->flags == cce[i].flags)) {
- /* found in cache */
- ci->hContacts = (MCONTACT*)mir_alloc(sizeof(MCONTACT));
- if (ci->hContacts == NULL)
- return -1;
-
- ci->hContacts[0] = cce[i].hContact;
- return 1;
- }
- }
+ CONTACTCE *p = arContactCache.find(&tmp);
+ if (p != NULL)
+ return (bReturnCount) ? 1 : p->hContact; // found in cache
}
/* contact was not in cache, do a search */
- for (MCONTACT hContact = db_find_first(); hContact; hContact = db_find_next(hContact)) {
- bool bMatch = false;
+ CMString tmp;
+ int count = 0;
+ MCONTACT hContact;
+ LIST<void> arResults(0);
+ for (hContact = db_find_first(); hContact; hContact = db_find_next(hContact)) {
// <_HANDLE_:hContact>
- {
- size_t size = mir_tstrlen(_T(PROTOID_HANDLE)) + 36;
- TCHAR *szFind = (TCHAR *)mir_alloc(size * sizeof(TCHAR));
- if (szFind != NULL) {
- mir_sntprintf(szFind, size, _T("<%s:%p>"), _T(PROTOID_HANDLE), hContact);
- if (!_tcsncmp(tszContact, szFind, mir_tstrlen(tszContact)))
- bMatch = true;
-
- mir_free(szFind);
- }
- }
+ tmp.Format(_T("<%s:%d>"), _T(PROTOID_HANDLE), hContact);
+ bool bMatch = (tmp == tszContact);
char *szProto = GetContactProto(hContact);
if (szProto == NULL)
continue;
// <proto:id> (exact)
- if ((ci->flags & CI_PROTOID) && !bMatch) {
- TCHAR *cInfo = getContactInfoT(CNF_UNIQUEID, hContact);
+ if ((dwFlags & CI_PROTOID) && !bMatch) {
+ ptrT cInfo(getContactInfoT(CNF_UNIQUEID, hContact));
if (cInfo) {
- size_t size = mir_tstrlen(cInfo) + mir_strlen(szProto) + 4;
- TCHAR *szFind = (TCHAR *)mir_alloc(size * sizeof(TCHAR));
- if (szFind != NULL) {
- mir_sntprintf(szFind, size, _T("<%S:%s>"), szProto, cInfo);
- mir_free(cInfo);
- if (!_tcsncmp(tszContact, szFind, mir_tstrlen(tszContact)))
- bMatch = true;
- mir_free(szFind);
- }
+ tmp.Format(_T("<%S:%s>"), szProto, cInfo);
+ if (tmp == tszContact)
+ bMatch = true;
}
}
+
// id (exact)
- if ((ci->flags & CI_UNIQUEID) && (!bMatch)) {
- TCHAR *szFind = getContactInfoT(CNF_UNIQUEID, hContact);
- if (szFind != NULL) {
- if (!mir_tstrcmp(tszContact, szFind))
- bMatch = true;
- mir_free(szFind);
- }
+ if ((dwFlags & CI_UNIQUEID) && !bMatch) {
+ ptrT szFind(getContactInfoT(CNF_UNIQUEID, hContact));
+ if (!mir_tstrcmp(tszContact, szFind))
+ bMatch = true;
}
+
// nick (not exact)
- if ((ci->flags & CI_NICK) && (!bMatch)) {
- TCHAR *szFind = getContactInfoT(CNF_NICK, hContact);
- if (szFind != NULL) {
- if (!mir_tstrcmp(tszContact, szFind))
- bMatch = true;
- mir_free(szFind);
- }
+ if ((dwFlags & CI_NICK) && !bMatch) {
+ ptrT szFind(getContactInfoT(CNF_NICK, hContact));
+ if (!mir_tstrcmp(tszContact, szFind))
+ bMatch = true;
}
+
// list name (not exact)
- if ((ci->flags & CI_LISTNAME) && (!bMatch)) {
- TCHAR *szFind = getContactInfoT(CNF_DISPLAY, hContact);
- if (szFind != NULL) {
- if (!mir_tstrcmp(tszContact, szFind))
- bMatch = true;
- mir_free(szFind);
- }
+ if ((dwFlags & CI_LISTNAME) && !bMatch) {
+ ptrT szFind(getContactInfoT(CNF_DISPLAY, hContact));
+ if (!mir_tstrcmp(tszContact, szFind))
+ bMatch = true;
}
+
// firstname (exact)
- if ((ci->flags & CI_FIRSTNAME) && (!bMatch)) {
- TCHAR *szFind = getContactInfoT(CNF_FIRSTNAME, hContact);
- if (szFind != NULL) {
- if (!mir_tstrcmp(tszContact, szFind))
- bMatch = true;
- mir_free(szFind);
- }
+ if ((dwFlags & CI_FIRSTNAME) && !bMatch) {
+ ptrT szFind(getContactInfoT(CNF_FIRSTNAME, hContact));
+ if (!mir_tstrcmp(tszContact, szFind))
+ bMatch = true;
}
+
// lastname (exact)
- if ((ci->flags & CI_LASTNAME) && (!bMatch)) {
- TCHAR *szFind = getContactInfoT(CNF_LASTNAME, hContact);
- if (szFind != NULL) {
- if (!mir_tstrcmp(tszContact, szFind))
- bMatch = true;
- mir_free(szFind);
- }
+ if ((dwFlags & CI_LASTNAME) && !bMatch) {
+ ptrT szFind(getContactInfoT(CNF_LASTNAME, hContact));
+ if (!mir_tstrcmp(tszContact, szFind))
+ bMatch = true;
}
+
// email (exact)
- if ((ci->flags & CI_EMAIL) && (!bMatch)) {
- TCHAR *szFind = getContactInfoT(CNF_EMAIL, hContact);
- if (szFind != NULL) {
- if (!mir_tstrcmp(tszContact, szFind))
- bMatch = true;
- mir_free(szFind);
- }
+ if ((dwFlags & CI_EMAIL) && !bMatch) {
+ ptrT szFind(getContactInfoT(CNF_EMAIL, hContact));
+ if (!mir_tstrcmp(tszContact, szFind))
+ bMatch = true;
}
+
// CNF_ (exact)
- if ((ci->flags & CI_CNFINFO) && (!bMatch)) {
- TCHAR *szFind = getContactInfoT((BYTE)(ci->flags&~(CI_CNFINFO | CI_TCHAR)), hContact);
- if (szFind != NULL) {
- if (!mir_tstrcmp(tszContact, szFind))
- bMatch = true;
- mir_free(szFind);
- }
+ if ((dwFlags & CI_CNFINFO) && !bMatch) {
+ ptrT szFind(getContactInfoT((BYTE)(dwFlags & ~CI_CNFINFO), hContact));
+ if (!mir_tstrcmp(tszContact, szFind))
+ bMatch = true;
}
- if (bMatch) {
- ci->hContacts = (MCONTACT*)mir_realloc(ci->hContacts, (count + 1)*sizeof(MCONTACT));
- if (ci->hContacts == NULL)
- return -1;
- ci->hContacts[count] = hContact;
+ if (bMatch) {
+ if (nMatch == -1)
+ arResults.insert((HANDLE)hContact);
+ else if (nMatch == count)
+ break;
count++;
}
}
- if (count == 1) { /* cache the found result */
+ if (bReturnCount)
+ return count;
+
+ if (hContact == 0)
+ return INVALID_CONTACT_ID;
+
+ // return random contact
+ if (nMatch == -1)
+ return (MCONTACT)arResults[rand() % arResults.getCount()];
+
+ // cache the found result
+ if (count == 1) {
mir_cslock lck(csContactCache);
- cce = (CONTACTCE*)mir_realloc(cce, (cacheSize + 1)*sizeof(CONTACTCE));
- if (cce != NULL) {
- cce[cacheSize].hContact = ci->hContacts[0];
- cce[cacheSize].flags = ci->flags;
- cce[cacheSize].tszContact = mir_tstrdup(tszContact);
- if (cce[cacheSize].tszContact != NULL)
- cacheSize++;
- }
+
+ CONTACTCE *cce = new CONTACTCE();
+ cce->hContact = hContact;
+ cce->flags = dwFlags;
+ cce->tszContact = mir_tstrdup(tszContact);
+ arContactCache.insert(cce);
}
- return count;
+ return hContact;
}
/* keep cache consistent */
@@ -373,29 +355,21 @@ static int contactSettingChanged(WPARAM hContact, LPARAM lParam) bool isUid = (((INT_PTR)uid != CALLSERVICE_NOTFOUND) && (uid != NULL)) && (!strcmp(dbw->szSetting, uid));
mir_cslock lck(csContactCache);
- for (int i = 0; i < cacheSize; i++) {
- if (hContact != cce[i].hContact && (cce[i].flags & CI_CNFINFO) == 0)
+ for (int i = 0; i < arContactCache.getCount(); i++) {
+ CONTACTCE &cce = arContactCache[i];
+ if (hContact != cce.hContact && (cce.flags & CI_CNFINFO) == 0)
continue;
- if ((isNick && (cce[i].flags & CI_NICK)) ||
- (isFirstName && (cce[i].flags & CI_FIRSTNAME)) ||
- (isLastName && (cce[i].flags & CI_LASTNAME)) ||
- (isEmail && (cce[i].flags & CI_EMAIL)) ||
- (isMyHandle && (cce[i].flags & CI_LISTNAME)) ||
- (cce[i].flags & CI_CNFINFO) != 0 || // lazy; always invalidate CNF info cache entries
- (isUid && (cce[i].flags & CI_UNIQUEID))) {
+ if ((isNick && (cce.flags & CI_NICK)) ||
+ (isFirstName && (cce.flags & CI_FIRSTNAME)) ||
+ (isLastName && (cce.flags & CI_LASTNAME)) ||
+ (isEmail && (cce.flags & CI_EMAIL)) ||
+ (isMyHandle && (cce.flags & CI_LISTNAME)) ||
+ (cce.flags & CI_CNFINFO) != 0 || // lazy; always invalidate CNF info cache entries
+ (isUid && (cce.flags & CI_UNIQUEID)))
+ {
/* remove from cache */
- mir_free(cce[i].tszContact);
- if (cacheSize > 1) {
- memmove(&cce[i], &cce[cacheSize - 1], sizeof(CONTACTCE));
- cce = (CONTACTCE*)mir_realloc(cce, (cacheSize - 1)*sizeof(CONTACTCE));
- cacheSize -= 1;
- }
- else {
- mir_free(cce);
- cce = NULL;
- cacheSize = 0;
- }
+ arContactCache.remove(i);
break;
}
}
@@ -404,13 +378,13 @@ static int contactSettingChanged(WPARAM hContact, LPARAM lParam) int initContactModule()
{
- hContactSettingChangedHook = HookEvent(ME_DB_CONTACT_SETTINGCHANGED, contactSettingChanged);
+ HookEvent(ME_DB_CONTACT_SETTINGCHANGED, contactSettingChanged);
return 0;
}
int deinitContactModule()
{
- UnhookEvent(hContactSettingChangedHook);
+ arContactCache.destroy();
return 0;
}
@@ -419,8 +393,11 @@ int deinitContactModule() TCHAR* encodeContactToString(MCONTACT hContact)
{
char *szProto = GetContactProto(hContact);
+ if (szProto == NULL)
+ return NULL;
+
TCHAR *tszUniqueId = getContactInfoT(CNF_UNIQUEID, hContact);
- if (szProto == NULL || tszUniqueId == NULL)
+ if (tszUniqueId == NULL)
return NULL;
size_t size = mir_tstrlen(tszUniqueId) + mir_strlen(szProto) + 4;
@@ -429,25 +406,3 @@ TCHAR* encodeContactToString(MCONTACT hContact) mir_sntprintf(tszResult, size, _T("<%S:%s>"), szProto, tszUniqueId);
return tszResult;
}
-
-// returns a contact from a string in the form <PROTOID:UNIQUEID>
-// returns INVALID_HANDLE_VALUE in case of an error.
-MCONTACT decodeContactFromString(TCHAR *tszContact)
-{
- MCONTACT hContact = INVALID_CONTACT_ID;
- CONTACTSINFO ci = { sizeof(ci) };
- ci.tszContact = tszContact;
- ci.flags = CI_PROTOID | CI_TCHAR;
- int count = getContactFromString(&ci);
- if (count != 1) {
- mir_free(ci.hContacts);
- return hContact;
- }
-
- if (ci.hContacts != NULL) {
- hContact = ci.hContacts[0];
- mir_free(ci.hContacts);
- }
-
- return hContact;
-}
|