1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
|
#ifndef __history_array__
#define __history_array__
enum
{
GROUPING_NONE = 0,
GROUPING_HEAD = 1,
GROUPING_ITEM = 2,
};
CMStringW TplFormatString(int tpl, MCONTACT hContact, ItemData *item);
struct ItemData
{
bool m_bSelected : 1;
bool m_bHighlighted : 1;
bool m_bDelivered : 1;
bool m_bRemoteRead : 1;
bool m_bNew : 1;
bool m_bLoaded : 1;
bool m_bIsResult : 1;
bool m_bOfflineFile : 1;
uint8_t m_grouping, m_bOfflineDownloaded;
int savedTop, savedHeight, leftOffset;
DB::EventInfo dbe;
wchar_t *wtext, *qtext;
wchar_t *wszNick;
struct NewstoryListData *pOwner;
litehtml::document::ptr m_doc;
ItemData();
~ItemData();
ItemData* checkNext(ItemData *pPrev);
ItemData* checkPrev(ItemData *pPrev);
ItemData* checkPrevGC(ItemData *pPrev);
int calcHeight(int width);
bool completed() const { return m_bOfflineDownloaded == 100; }
bool fetch(void);
void fill(int tmpl);
void load(bool bLoad = false);
void setText(const wchar_t *pwszText = nullptr);
int getTemplate() const;
int getCopyTemplate() const;
void getFontColor(int &fontId, int &colorId) const;
const char* getUrl() const;
CMStringW formatHtml(const wchar_t *pwszStr = 0);
CMStringW formatString() { return TplFormatString(getTemplate(), dbe.hContact, this); }
CMStringW formatStringEx(wchar_t *sztpl);
inline wchar_t* getWBuf()
{
load();
return wtext;
}
};
class Filter
{
uint16_t flags;
ptrW text;
public:
enum
{
INCOMING = 0x001,
OUTGOING = 0x002,
MESSAGES = 0x004,
FILES = 0x008,
STATUS = 0x020,
OTHER = 0x040,
EVENTTEXT = 0x080,
EVENTONLY = 0x100,
};
__forceinline Filter(uint16_t aFlags, const wchar_t *wText) :
flags(aFlags),
text(mir_wstrdup(wText))
{
}
bool check(ItemData *item) const;
};
enum
{
FILTER_TIME = 0x01,
FILTER_TYPE = 0x02,
FILTER_DIRECTION = 0x04,
FILTER_TEXT = 0x08,
FILTER_UNICODE = 0x10,
FTYPE_MESSAGE = 0x01,
FTYPE_FILE = 0x02,
FTYPE_URL = 0x04,
FTYPE_STATUS = 0x08,
FTYPE_OTHER = 0x10,
FTYPE_INCOMING = 0x20,
FTYPE_OUTGOING = 0x40
};
#define HIST_BLOCK_SIZE 1000
struct ItemBlock : public MZeroedObject
{
ItemData data[HIST_BLOCK_SIZE];
};
struct SearchResult
{
SearchResult(MCONTACT _1, MEVENT _2, uint32_t _3) :
hContact(_1),
hEvent(_2),
ts(_3)
{}
MCONTACT hContact;
MEVENT hEvent;
uint32_t ts;
};
class HistoryArray
{
std::map<std::wstring, MCONTACT> gcCache;
LIST<wchar_t> strings;
OBJLIST<ItemBlock> pages;
int iLastPageCounter = 0;
MWindow hwndOwner = 0;
ItemData& allocateItem(void);
public:
HistoryArray();
~HistoryArray();
void addEvent(NewstoryListData *pOwner, MCONTACT hContact, MEVENT hEvent, int count, bool bNew = false);
void addChatEvent(NewstoryListData *pOwner, SESSION_INFO *si, const LOGINFO *pEvent);
void addResults(NewstoryListData *pOwner, const OBJLIST<SearchResult> &pArray);
void addNick(ItemData &pItem, wchar_t *pwszNick);
void clear();
void checkGC(ItemData &pItem, SESSION_INFO *si);
int find(MEVENT hEvent);
int find(int id, int dir, const Filter &filter);
int getCount() const;
void remove(int idx);
void reset()
{
clear();
pages.insert(new ItemBlock());
}
void setOwner(MWindow hwnd) {
hwndOwner = hwnd;
}
ItemData* get(int id, bool bLoad = false) const;
ItemData* insert(int idx);
__forceinline int FindNext(int id, const Filter &filter) { return find(id, +1, filter); }
__forceinline int FindPrev(int id, const Filter &filter) { return find(id, -1, filter); }
};
#endif // __history_array__
|