summaryrefslogtreecommitdiff
path: root/plugins/NewStory/src/history_array.h
blob: f5a39ba5bd95366ffe184d25be6148482baf4786 (plain)
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
#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
{
	MCONTACT hContact;

	bool m_bSelected, m_bHighlighted;
	bool m_bLoaded, m_bIsResult;
	bool m_bOfflineFile;
	uint8_t m_grouping, m_bOfflineDownloaded;
	
	int savedTop, savedHeight;

	DB::EventInfo dbe;
	wchar_t *wtext;
	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);
	void checkCreate();

	int  calcHeight(int top, int width, POINT *pPos = nullptr);
	bool completed() const { return m_bOfflineDownloaded == 100; }
	bool fetch(void);
	void fill(int tmpl);
	void load(bool bLoad = false);
	void setText();

	int getTemplate() const;
	int getCopyTemplate() const;
	void getFontColor(int &fontId, int &colorId) const;

	CMStringA formatHtml(const wchar_t *pwszStr = 0);
	CMStringW formatString() { return TplFormatString(getTemplate(), 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
{
	LIST<wchar_t> strings;
	OBJLIST<ItemBlock> pages;
	int iLastPageCounter = 0;
	MWindow hwndOwner = 0;

	ItemData& allocateItem(void);

public:
	HistoryArray();
	~HistoryArray();

	bool addEvent(NewstoryListData *pOwner, MCONTACT hContact, MEVENT hEvent, int count);
	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__