summaryrefslogtreecommitdiff
path: root/plugins/NewAwaySysMod/src
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/NewAwaySysMod/src')
-rw-r--r--plugins/NewAwaySysMod/src/AwayOpt.cpp1686
-rw-r--r--plugins/NewAwaySysMod/src/AwaySys.cpp1004
-rw-r--r--plugins/NewAwaySysMod/src/CString.cpp308
-rw-r--r--plugins/NewAwaySysMod/src/CString.h237
-rw-r--r--plugins/NewAwaySysMod/src/Client.cpp232
-rw-r--r--plugins/NewAwaySysMod/src/Common.h399
-rw-r--r--plugins/NewAwaySysMod/src/ContactList.cpp898
-rw-r--r--plugins/NewAwaySysMod/src/ContactList.h142
-rw-r--r--plugins/NewAwaySysMod/src/MsgEventAdded.cpp291
-rw-r--r--plugins/NewAwaySysMod/src/MsgTree.cpp790
-rw-r--r--plugins/NewAwaySysMod/src/MsgTree.h95
-rw-r--r--plugins/NewAwaySysMod/src/Notification.cpp285
-rw-r--r--plugins/NewAwaySysMod/src/Options.cpp963
-rw-r--r--plugins/NewAwaySysMod/src/Options.h483
-rw-r--r--plugins/NewAwaySysMod/src/Path.h43
-rw-r--r--plugins/NewAwaySysMod/src/Properties.cpp411
-rw-r--r--plugins/NewAwaySysMod/src/Properties.h597
-rw-r--r--plugins/NewAwaySysMod/src/ReadAwayMsg.cpp158
-rw-r--r--plugins/NewAwaySysMod/src/Services.cpp231
-rw-r--r--plugins/NewAwaySysMod/src/Services.h27
-rw-r--r--plugins/NewAwaySysMod/src/SetAwayMsg.cpp1698
-rw-r--r--plugins/NewAwaySysMod/src/TMyArray.h353
-rw-r--r--plugins/NewAwaySysMod/src/resource.h261
-rw-r--r--plugins/NewAwaySysMod/src/version.h5
24 files changed, 11597 insertions, 0 deletions
diff --git a/plugins/NewAwaySysMod/src/AwayOpt.cpp b/plugins/NewAwaySysMod/src/AwayOpt.cpp
new file mode 100644
index 0000000000..7ba87f43ab
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/AwayOpt.cpp
@@ -0,0 +1,1686 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (C) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "MsgTree.h"
+#include "Properties.h"
+#include "Path.h"
+#include "m_button.h"
+#include "m_clc.h"
+#include "GroupCheckbox.h"
+#include "ThemedImageCheckbox.h"
+
+//NightFox
+#include <m_modernopt.h>
+
+int g_Messages_RecentRootID, g_Messages_PredefinedRootID;
+CIconList g_IconList;
+
+
+void MySetPos(HWND hwndParent)
+// Set window size and center its controls
+{
+ HWND hWndTab = FindWindowEx(GetParent(hwndParent), NULL, _T("SysTabControl32"), _T(""));
+ if (!hWndTab)
+ {
+ _ASSERT(0);
+ return;
+ }
+ RECT rcDlg;
+ GetClientRect(hWndTab, &rcDlg);
+ TabCtrl_AdjustRect(hWndTab, false, &rcDlg);
+ rcDlg.right -= rcDlg.left - 2;
+ rcDlg.bottom -= rcDlg.top;
+ rcDlg.top += 2;
+ if (hwndParent)
+ {
+ RECT OldSize;
+ GetClientRect(hwndParent, &OldSize);
+ MoveWindow(hwndParent, rcDlg.left, rcDlg.top, rcDlg.right, rcDlg.bottom, true);
+ int dx = (rcDlg.right - OldSize.right) >> 1;
+ int dy = (rcDlg.bottom - OldSize.bottom) >> 1;
+ HWND hCurWnd = GetWindow(hwndParent, GW_CHILD);
+ while (hCurWnd)
+ {
+ RECT CWOldPos;
+ GetWindowRect(hCurWnd, &CWOldPos);
+ POINT pt;
+ pt.x = CWOldPos.left;
+ pt.y = CWOldPos.top;
+ ScreenToClient(hwndParent, &pt);
+ SetWindowPos(hCurWnd, NULL, pt.x + dx, pt.y + dy, 0, 0, SWP_NOSIZE | SWP_NOZORDER);
+ hCurWnd = GetNextWindow(hCurWnd, GW_HWNDNEXT);
+ }
+ }
+}
+
+// ================================================ Message options ================================================
+
+COptPage g_MessagesOptPage(MOD_NAME, NULL);
+
+void EnableMessagesOptDlgControls(CMsgTree* MsgTree)
+{
+ int I;
+ int IsNotGroup = false;
+ int Selected = false;
+ CBaseTreeItem* TreeItem = MsgTree->GetSelection();
+ if (TreeItem && !(TreeItem->Flags & TIF_ROOTITEM))
+ {
+ IsNotGroup = !(TreeItem->Flags & TIF_GROUP);
+ Selected = true;
+ }
+ g_MessagesOptPage.Enable(IDC_MESSAGEDLG_DEL, Selected);
+ g_MessagesOptPage.Enable(IDC_MESSAGEDLG_MSGTITLE, Selected);
+ for (I = 0; I < g_MessagesOptPage.Items.GetSize(); I++)
+ {
+ if (g_MessagesOptPage.Items[I]->GetParam() == IDC_MESSAGEDLG_MSGTREE)
+ {
+ g_MessagesOptPage.Items[I]->Enable(IsNotGroup);
+ }
+ }
+ SendDlgItemMessage(g_MessagesOptPage.GetWnd(), IDC_MESSAGEDLG_MSGDATA, EM_SETREADONLY, !IsNotGroup, 0);
+ g_MessagesOptPage.MemToPage(true);
+}
+
+static WNDPROC g_OrigDefStatusButtonMsgProc;
+
+static LRESULT CALLBACK DefStatusButtonSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ if (Msg == WM_LBUTTONUP && IsDlgButtonChecked(GetParent(hWnd), GetDlgCtrlID(hWnd)))
+ {
+ return 0;
+ }
+ return CallWindowProc(g_OrigDefStatusButtonMsgProc, hWnd, Msg, wParam, lParam);
+}
+
+
+INT_PTR CALLBACK MessagesOptDlg(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
+{
+ static int ChangeLock = 0;
+ static CMsgTree* MsgTree = NULL;
+ static struct {
+ int DlgItem, Status, IconIndex;
+ } DefMsgDlgItems[] = {
+ IDC_MESSAGEDLG_DEF_ONL, ID_STATUS_ONLINE, ILI_PROTO_ONL,
+ IDC_MESSAGEDLG_DEF_AWAY, ID_STATUS_AWAY, ILI_PROTO_AWAY,
+ IDC_MESSAGEDLG_DEF_NA, ID_STATUS_NA, ILI_PROTO_NA,
+ IDC_MESSAGEDLG_DEF_OCC, ID_STATUS_OCCUPIED, ILI_PROTO_OCC,
+ IDC_MESSAGEDLG_DEF_DND, ID_STATUS_DND, ILI_PROTO_DND,
+ IDC_MESSAGEDLG_DEF_FFC, ID_STATUS_FREECHAT, ILI_PROTO_FFC,
+ IDC_MESSAGEDLG_DEF_INV, ID_STATUS_INVISIBLE, ILI_PROTO_INV,
+ IDC_MESSAGEDLG_DEF_OTP, ID_STATUS_ONTHEPHONE, ILI_PROTO_OTP,
+ IDC_MESSAGEDLG_DEF_OTL, ID_STATUS_OUTTOLUNCH, ILI_PROTO_OTL
+ };
+ static struct {
+ int DlgItem, IconIndex;
+ TCHAR* Text;
+ } Buttons[] = {
+ IDC_MESSAGEDLG_NEWMSG, ILI_NEWMESSAGE, LPGENT("Create new message"),
+ IDC_MESSAGEDLG_NEWCAT, ILI_NEWCATEGORY, LPGENT("Create new category"),
+ IDC_MESSAGEDLG_DEL, ILI_DELETE, LPGENT("Delete"),
+ IDC_MESSAGEDLG_VARS, ILI_NOICON, LPGENT("Open Variables help dialog"),
+ };
+ switch (msg)
+ {
+ case WM_INITDIALOG:
+ {
+ TranslateDialogDefault(hwndDlg);
+ MySetPos(hwndDlg);
+ ChangeLock++;
+ g_MessagesOptPage.SetWnd(hwndDlg);
+ SendMessage(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTITLE), EM_LIMITTEXT, TREEITEMTITLE_MAXLEN, 0);
+ SendMessage(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGDATA), EM_LIMITTEXT, AWAY_MSGDATA_MAX, 0);
+ // init image buttons
+ int I;
+ for (I = 0; I < lengthof(Buttons); I++)
+ {
+ HWND hButton = GetDlgItem(hwndDlg, Buttons[I].DlgItem);
+ SendMessage(hButton, BUTTONADDTOOLTIP, (WPARAM)TranslateTS(Buttons[I].Text), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ }
+ // now default status message buttons
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ HWND hButton = GetDlgItem(hwndDlg, DefMsgDlgItems[I].DlgItem);
+ SendMessage(hButton, BUTTONADDTOOLTIP, CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, DefMsgDlgItems[I].Status, GSMDF_TCHAR), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASPUSHBTN, 0, 0);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ g_OrigDefStatusButtonMsgProc = (WNDPROC)SetWindowLongPtr(hButton, GWLP_WNDPROC, (LONG_PTR)DefStatusButtonSubclassProc);
+ }
+ SendMessage(hwndDlg, UM_ICONSCHANGED, 0, 0);
+ g_MessagesOptPage.DBToMemToPage();
+ _ASSERT(!MsgTree);
+ MsgTree = new CMsgTree(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTREE));
+ if (!MsgTree->SetSelection(MsgTree->GetDefMsg(ID_STATUS_AWAY), MTSS_BYID))
+ {
+ MsgTree->SetSelection(g_Messages_PredefinedRootID, MTSS_BYID);
+ }
+ ChangeLock--;
+ return true;
+ } break;
+ case UM_ICONSCHANGED:
+ {
+ int I;
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ SendDlgItemMessage(hwndDlg, DefMsgDlgItems[I].DlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[DefMsgDlgItems[I].IconIndex]);
+ }
+ for (I = 0; I < lengthof(Buttons); I++)
+ {
+ if (Buttons[I].IconIndex != ILI_NOICON)
+ {
+ SendDlgItemMessage(hwndDlg, Buttons[I].DlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[Buttons[I].IconIndex]);
+ }
+ }
+ my_variables_skin_helpbutton(hwndDlg, IDC_MESSAGEDLG_VARS);
+ } break;
+ case WM_NOTIFY:
+ {
+ switch (((NMHDR*)lParam)->code)
+ {
+ case PSN_APPLY:
+ {
+ HWND hTreeView = GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTREE);
+ HTREEITEM hSelectedItem = TreeView_GetSelection(hTreeView);
+ ChangeLock++;
+ TreeView_SelectItem(hTreeView, NULL);
+ TreeView_SelectItem(hTreeView, hSelectedItem);
+ ChangeLock--;
+ MsgTree->Save();
+ return true;
+ } break;
+ }
+ if (((LPNMHDR)lParam)->idFrom == IDC_MESSAGEDLG_MSGTREE)
+ {
+ switch (((LPNMHDR)lParam)->code)
+ {
+ case MTN_SELCHANGED:
+ {
+ PNMMSGTREE pnm = (PNMMSGTREE)lParam;
+ int I;
+ if (pnm->ItemOld && !(pnm->ItemOld->Flags & (TIF_ROOTITEM | TIF_GROUP)))
+ {
+ TCString Msg;
+ GetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, Msg.GetBuffer(AWAY_MSGDATA_MAX), AWAY_MSGDATA_MAX);
+ Msg.ReleaseBuffer();
+ if (((CTreeItem*)pnm->ItemOld)->User_Str1 != (const TCHAR*)Msg)
+ {
+ ((CTreeItem*)pnm->ItemOld)->User_Str1 = Msg;
+ MsgTree->SetModified(true);
+ }
+ }
+ if (pnm->ItemNew)
+ {
+ ChangeLock++;
+ if (!(pnm->ItemNew->Flags & TIF_ROOTITEM))
+ {
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, pnm->ItemNew->Title);
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, (pnm->ItemNew->Flags & TIF_GROUP) ? _T("") : ((CTreeItem*)pnm->ItemNew)->User_Str1);
+ } else
+ {
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, _T(""));
+ if (pnm->ItemNew->ID == g_Messages_RecentRootID)
+ {
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, TranslateT("Your most recent status messages are placed in this category. It's not recommended to put your messages manually here, as they'll be replaced by your recent messages."));
+ } else
+ {
+ _ASSERT(pnm->ItemNew->ID == g_Messages_PredefinedRootID);
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, TranslateT("You can put your frequently used and favorite messages in this category."));
+ }
+ }
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ COptItem_Checkbox *Checkbox = (COptItem_Checkbox*)g_MessagesOptPage.Find(DefMsgDlgItems[I].DlgItem);
+ Checkbox->SetWndValue(g_MessagesOptPage.GetWnd(), MsgTree->GetDefMsg(DefMsgDlgItems[I].Status) == pnm->ItemNew->ID);
+ }
+ ChangeLock--;
+ }
+ EnableMessagesOptDlgControls(MsgTree);
+ return 0;
+ } break;
+ case MTN_DEFMSGCHANGED:
+ {
+ PNMMSGTREE pnm = (PNMMSGTREE)lParam;
+ if (!ChangeLock)
+ {
+ CBaseTreeItem* SelectedItem = MsgTree->GetSelection();
+ _ASSERT(SelectedItem);
+ if ((pnm->ItemOld && pnm->ItemOld->ID == SelectedItem->ID) || (pnm->ItemNew && pnm->ItemNew->ID == SelectedItem->ID))
+ { // SelectedItem contains the same info as one of ItemOld or ItemNew - so we'll just use SelectedItem and won't bother with identifying which of ItemOld or ItemNew is currently selected
+ int I;
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ COptItem_Checkbox *Checkbox = (COptItem_Checkbox*)g_MessagesOptPage.Find(DefMsgDlgItems[I].DlgItem);
+ Checkbox->SetWndValue(g_MessagesOptPage.GetWnd(), MsgTree->GetDefMsg(DefMsgDlgItems[I].Status) == SelectedItem->ID);
+ }
+ }
+ }
+ if (!ChangeLock)
+ {
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ return 0;
+ } break;
+ case MTN_ITEMRENAMED:
+ {
+ PNMMSGTREE pnm = (PNMMSGTREE)lParam;
+ CBaseTreeItem* SelectedItem = MsgTree->GetSelection();
+ _ASSERT(SelectedItem);
+ if (pnm->ItemNew->ID == SelectedItem->ID && !ChangeLock)
+ {
+ ChangeLock++;
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, pnm->ItemNew->Title);
+ ChangeLock--;
+ }
+ } // go through
+ case MTN_ENDDRAG:
+ case MTN_NEWCATEGORY:
+ case MTN_NEWMESSAGE:
+ case MTN_DELETEITEM:
+ case TVN_ITEMEXPANDED:
+ {
+ if (!ChangeLock)
+ {
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ return 0;
+ } break;
+ }
+ }
+ } break;
+ case WM_COMMAND:
+ {
+ switch (HIWORD(wParam))
+ {
+ case BN_CLICKED:
+ {
+ switch (LOWORD(wParam))
+ {
+ case IDC_MESSAGEDLG_DEF_ONL:
+ case IDC_MESSAGEDLG_DEF_AWAY:
+ case IDC_MESSAGEDLG_DEF_NA:
+ case IDC_MESSAGEDLG_DEF_OCC:
+ case IDC_MESSAGEDLG_DEF_DND:
+ case IDC_MESSAGEDLG_DEF_FFC:
+ case IDC_MESSAGEDLG_DEF_INV:
+ case IDC_MESSAGEDLG_DEF_OTP:
+ case IDC_MESSAGEDLG_DEF_OTL:
+ {
+ int I;
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ if (LOWORD(wParam) == DefMsgDlgItems[I].DlgItem)
+ {
+ MsgTree->SetDefMsg(DefMsgDlgItems[I].Status, MsgTree->GetSelection()->ID); // PSM_CHANGED is sent here through MTN_DEFMSGCHANGED, so we don't need to send it once more
+ break;
+ }
+ }
+ } break;
+ case IDC_MESSAGEDLG_VARS:
+ {
+ my_variables_showhelp(hwndDlg, IDC_MESSAGEDLG_MSGDATA);
+ } break;
+ case IDC_MESSAGEDLG_DEL:
+ {
+ MsgTree->EnsureVisible(MsgTree->GetSelection()->hItem);
+ MsgTree->DeleteSelectedItem();
+ } break;
+ case IDC_MESSAGEDLG_NEWCAT:
+ {
+ MsgTree->AddCategory();
+ SetFocus(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTITLE));
+ } break;
+ case IDC_MESSAGEDLG_NEWMSG:
+ {
+ MsgTree->AddMessage();
+ SetFocus(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTITLE));
+ } break;
+ }
+ } break;
+ case EN_CHANGE:
+ {
+ if (LOWORD(wParam) == IDC_MESSAGEDLG_MSGDATA || LOWORD(wParam) == IDC_MESSAGEDLG_MSGTITLE)
+ {
+ if (!ChangeLock)
+ {
+ if (LOWORD(wParam) == IDC_MESSAGEDLG_MSGTITLE)
+ {
+ CBaseTreeItem* TreeItem = MsgTree->GetSelection();
+ if (TreeItem && !(TreeItem->Flags & TIF_ROOTITEM))
+ {
+ GetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, TreeItem->Title.GetBuffer(TREEITEMTITLE_MAXLEN), TREEITEMTITLE_MAXLEN);
+ TreeItem->Title.ReleaseBuffer();
+ ChangeLock++;
+ MsgTree->UpdateItem(TreeItem->ID);
+ ChangeLock--;
+ }
+ }
+ MsgTree->SetModified(true);
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ }
+ } break;
+ }
+ } break;
+ case WM_DESTROY:
+ {
+ delete MsgTree;
+ MsgTree = NULL;
+ g_MessagesOptPage.SetWnd(NULL);
+ } break;
+ }
+ return 0;
+}
+
+// ================================================ Main options ================================================
+
+
+COptPage g_MoreOptPage(MOD_NAME, NULL);
+
+
+void EnableMoreOptDlgControls()
+{
+ g_MoreOptPage.Enable(IDC_MOREOPTDLG_PERSTATUSPERSONAL, g_MoreOptPage.GetWndValue(IDC_MOREOPTDLG_SAVEPERSONALMSGS));
+ int Enabled = g_MoreOptPage.GetWndValue(IDC_MOREOPTDLG_RECENTMSGSCOUNT);
+ g_MoreOptPage.Enable(IDC_MOREOPTDLG_PERSTATUSMRM, Enabled);
+ g_MoreOptPage.Enable(IDC_MOREOPTDLG_USELASTMSG, Enabled);
+ g_MoreOptPage.Enable(IDC_MOREOPTDLG_USEDEFMSG, Enabled);
+ g_MoreOptPage.Enable(IDC_MOREOPTDLG_PERSTATUSPERSONAL, g_MoreOptPage.GetWndValue(IDC_MOREOPTDLG_SAVEPERSONALMSGS));
+ g_MoreOptPage.Enable(IDC_MOREOPTDLG_UPDATEMSGSPERIOD, g_MoreOptPage.GetWndValue(IDC_MOREOPTDLG_UPDATEMSGS));
+ InvalidateRect(GetDlgItem(g_MoreOptPage.GetWnd(), IDC_MOREOPTDLG_UPDATEMSGSPERIOD_SPIN), NULL, false); // update spin control
+ g_MoreOptPage.MemToPage(true);
+}
+
+
+INT_PTR CALLBACK MoreOptDlg(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
+{
+ static int ChangeLock = 0;
+ static struct {
+ int DlgItem, Status, IconIndex;
+ } StatusButtons[] = {
+ IDC_MOREOPTDLG_DONTPOPDLG_ONL, ID_STATUS_ONLINE, ILI_PROTO_ONL,
+ IDC_MOREOPTDLG_DONTPOPDLG_AWAY, ID_STATUS_AWAY, ILI_PROTO_AWAY,
+ IDC_MOREOPTDLG_DONTPOPDLG_NA, ID_STATUS_NA, ILI_PROTO_NA,
+ IDC_MOREOPTDLG_DONTPOPDLG_OCC, ID_STATUS_OCCUPIED, ILI_PROTO_OCC,
+ IDC_MOREOPTDLG_DONTPOPDLG_DND, ID_STATUS_DND, ILI_PROTO_DND,
+ IDC_MOREOPTDLG_DONTPOPDLG_FFC, ID_STATUS_FREECHAT, ILI_PROTO_FFC,
+ IDC_MOREOPTDLG_DONTPOPDLG_INV, ID_STATUS_INVISIBLE, ILI_PROTO_INV,
+ IDC_MOREOPTDLG_DONTPOPDLG_OTP, ID_STATUS_ONTHEPHONE, ILI_PROTO_OTP,
+ IDC_MOREOPTDLG_DONTPOPDLG_OTL, ID_STATUS_OUTTOLUNCH, ILI_PROTO_OTL
+ };
+ switch (msg)
+ {
+ case WM_INITDIALOG:
+ {
+ TranslateDialogDefault(hwndDlg);
+ MySetPos(hwndDlg);
+ ChangeLock++;
+ g_MoreOptPage.SetWnd(hwndDlg);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_WAITFORMSG, EM_LIMITTEXT, 4, 0);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_RECENTMSGSCOUNT, EM_LIMITTEXT, 2, 0);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_WAITFORMSG_SPIN, UDM_SETRANGE32, -1, 9999);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_RECENTMSGSCOUNT_SPIN, UDM_SETRANGE32, 0, 99);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_UPDATEMSGSPERIOD_SPIN, UDM_SETRANGE32, 30, 99999);
+ int I;
+ for (I = 0; I < lengthof(StatusButtons); I++)
+ {
+ HWND hButton = GetDlgItem(hwndDlg, StatusButtons[I].DlgItem);
+ SendMessage(hButton, BUTTONADDTOOLTIP, CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, StatusButtons[I].Status, GSMDF_TCHAR), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASPUSHBTN, TRUE, 0);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ }
+ SendMessage(hwndDlg, UM_ICONSCHANGED, 0, 0);
+ g_MoreOptPage.DBToMemToPage();
+ EnableMoreOptDlgControls();
+ ChangeLock--;
+ return true;
+ } break;
+ case UM_ICONSCHANGED:
+ {
+ int I;
+ for (I = 0; I < lengthof(StatusButtons); I++)
+ {
+ SendDlgItemMessage(hwndDlg, StatusButtons[I].DlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[StatusButtons[I].IconIndex]);
+ }
+ } break;
+ case WM_NOTIFY:
+ {
+ switch (((NMHDR*)lParam)->code)
+ {
+ case PSN_APPLY:
+ {
+ g_MoreOptPage.PageToMemToDB();
+ InitUpdateMsgs();
+ return true;
+ } break;
+ }
+ } break;
+ case WM_COMMAND:
+ {
+ switch (HIWORD(wParam))
+ {
+ case BN_CLICKED:
+ {
+ switch (LOWORD(wParam))
+ {
+
+ case IDC_MOREOPTDLG_RESETPROTOMSGS:
+ case IDC_MOREOPTDLG_SAVEPERSONALMSGS:
+ case IDC_MOREOPTDLG_UPDATEMSGS:
+ {
+ EnableMoreOptDlgControls();
+ } // go through
+ case IDC_MOREOPTDLG_PERSTATUSMRM:
+ case IDC_MOREOPTDLG_PERSTATUSPROTOMSGS:
+ case IDC_MOREOPTDLG_PERSTATUSPROTOSETTINGS:
+ case IDC_MOREOPTDLG_PERSTATUSPERSONAL:
+ case IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS:
+ case IDC_MOREOPTDLG_USEMENUITEM:
+ case IDC_MOREOPTDLG_MYNICKPERPROTO:
+ case IDC_MOREOPTDLG_USEDEFMSG:
+ case IDC_MOREOPTDLG_USELASTMSG:
+ case IDC_MOREOPTDLG_DONTPOPDLG_ONL:
+ case IDC_MOREOPTDLG_DONTPOPDLG_AWAY:
+ case IDC_MOREOPTDLG_DONTPOPDLG_NA:
+ case IDC_MOREOPTDLG_DONTPOPDLG_OCC:
+ case IDC_MOREOPTDLG_DONTPOPDLG_DND:
+ case IDC_MOREOPTDLG_DONTPOPDLG_FFC:
+ case IDC_MOREOPTDLG_DONTPOPDLG_INV:
+ case IDC_MOREOPTDLG_DONTPOPDLG_OTP:
+ case IDC_MOREOPTDLG_DONTPOPDLG_OTL:
+ {
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ return 0;
+ } break;
+ }
+ } break;
+ case EN_CHANGE:
+ {
+ if (!ChangeLock && g_MoreOptPage.GetWnd())
+ {
+ if (LOWORD(wParam) == IDC_MOREOPTDLG_RECENTMSGSCOUNT)
+ {
+ EnableMoreOptDlgControls();
+ }
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ } break;
+ }
+ } break;
+ case WM_DESTROY:
+ {
+ g_MoreOptPage.SetWnd(NULL);
+ } break;
+ }
+ return 0;
+}
+
+
+// ================================================ Autoreply options ================================================
+
+
+COptPage g_AutoreplyOptPage(MOD_NAME, NULL);
+
+
+void EnableAutoreplyOptDlgControls()
+{
+ int I;
+ g_AutoreplyOptPage.PageToMem();
+ int Autoreply = g_AutoreplyOptPage.GetValue(IDC_REPLYDLG_ENABLEREPLY);
+
+ for (I = 0; I < g_AutoreplyOptPage.Items.GetSize(); I++)
+ {
+ switch (g_AutoreplyOptPage.Items[I]->GetParam())
+ {
+ case IDC_REPLYDLG_ENABLEREPLY:
+ {
+ g_AutoreplyOptPage.Items[I]->Enable(Autoreply);
+ } break;
+ case IDC_REPLYDLG_SENDCOUNT:
+ {
+ g_AutoreplyOptPage.Items[I]->Enable(Autoreply && g_AutoreplyOptPage.GetValue(IDC_REPLYDLG_SENDCOUNT) > 0);
+ } break;
+ }
+ }
+ g_AutoreplyOptPage.MemToPage(true);
+ InvalidateRect(GetDlgItem(g_AutoreplyOptPage.GetWnd(), IDC_REPLYDLG_SENDCOUNT_SPIN), NULL, 0); // update spin control
+}
+
+INT_PTR CALLBACK AutoreplyOptDlg(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
+{
+ static int ChangeLock = 0;
+ static HWND hWndTooltips;
+ static HFONT s_hDrawFont;
+ static struct {
+ int DlgItem, Status, IconIndex;
+ } StatusButtons[] = {
+ IDC_REPLYDLG_DISABLE_ONL, ID_STATUS_ONLINE, ILI_PROTO_ONL,
+ IDC_REPLYDLG_DISABLE_AWAY, ID_STATUS_AWAY, ILI_PROTO_AWAY,
+ IDC_REPLYDLG_DISABLE_NA, ID_STATUS_NA, ILI_PROTO_NA,
+ IDC_REPLYDLG_DISABLE_OCC, ID_STATUS_OCCUPIED, ILI_PROTO_OCC,
+ IDC_REPLYDLG_DISABLE_DND, ID_STATUS_DND, ILI_PROTO_DND,
+ IDC_REPLYDLG_DISABLE_FFC, ID_STATUS_FREECHAT, ILI_PROTO_FFC,
+ IDC_REPLYDLG_DISABLE_INV, ID_STATUS_INVISIBLE, ILI_PROTO_INV,
+ IDC_REPLYDLG_DISABLE_OTP, ID_STATUS_ONTHEPHONE, ILI_PROTO_OTP,
+ IDC_REPLYDLG_DISABLE_OTL, ID_STATUS_OUTTOLUNCH, ILI_PROTO_OTL
+ };
+ switch (msg)
+ {
+ case WM_INITDIALOG:
+ {
+ ChangeLock++;
+ TranslateDialogDefault(hwndDlg);
+ MySetPos(hwndDlg);
+ g_AutoreplyOptPage.SetWnd(hwndDlg);
+
+ LOGFONT logFont;
+ HFONT hFont = (HFONT)SendDlgItemMessage(hwndDlg, IDC_REPLYDLG_STATIC_DISABLEWHENSTATUS, WM_GETFONT, 0, 0); // try getting the font used by the control
+ if (!hFont)
+ {
+ hFont = (HFONT)GetStockObject(DEFAULT_GUI_FONT);
+ }
+ GetObject(hFont, sizeof(logFont), &logFont);
+ logFont.lfWeight = FW_BOLD;
+ s_hDrawFont = CreateFontIndirect(&logFont); // recreate the font
+ SendDlgItemMessage(hwndDlg, IDC_REPLYDLG_STATIC_DISABLEWHENSTATUS, WM_SETFONT, (WPARAM)s_hDrawFont, true);
+ SendDlgItemMessage(hwndDlg, IDC_REPLYDLG_STATIC_FORMAT, WM_SETFONT, (WPARAM)s_hDrawFont, true);
+
+ SendDlgItemMessage(hwndDlg, IDC_REPLYDLG_SENDCOUNT, EM_LIMITTEXT, 3, 0);
+ SendDlgItemMessage(hwndDlg, IDC_REPLYDLG_SENDCOUNT_SPIN, UDM_SETRANGE32, -1, 999);
+
+ HWND hCombo = GetDlgItem(hwndDlg, IDC_REPLYDLG_ONLYIDLEREPLY_COMBO);
+ static struct {
+ TCHAR *Text;
+ int Meaning;
+ } IdleComboValues[] = {
+ LPGENT("Windows"), AUTOREPLY_IDLE_WINDOWS,
+ LPGENT("Miranda"), AUTOREPLY_IDLE_MIRANDA
+ };
+ int I;
+ for (I = 0; I < lengthof(IdleComboValues); I++)
+ {
+ SendMessage(hCombo, CB_SETITEMDATA, SendMessage(hCombo, CB_ADDSTRING, 0, (LPARAM)TranslateTS(IdleComboValues[I].Text)), IdleComboValues[I].Meaning);
+ }
+
+ for (I = 0; I < lengthof(StatusButtons); I++)
+ {
+ HWND hButton = GetDlgItem(hwndDlg, StatusButtons[I].DlgItem);
+ SendMessage(hButton, BUTTONADDTOOLTIP, CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, StatusButtons[I].Status, GSMDF_TCHAR), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASPUSHBTN, TRUE, 0);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ }
+ HWND hButton = GetDlgItem(hwndDlg, IDC_REPLYDLG_VARS);
+ SendMessage(hButton, BUTTONADDTOOLTIP, (WPARAM)TranslateT("Open Variables help dialog"), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ MakeThemedImageCheckbox(GetDlgItem(hwndDlg, IDC_MOREOPTDLG_EVNTMSG));
+ MakeThemedImageCheckbox(GetDlgItem(hwndDlg, IDC_MOREOPTDLG_EVNTURL));
+ MakeThemedImageCheckbox(GetDlgItem(hwndDlg, IDC_MOREOPTDLG_EVNTFILE));
+ SendMessage(hwndDlg, UM_ICONSCHANGED, 0, 0);
+
+ // init tooltips
+ struct {
+ int DlgItemID;
+ TCHAR* Text;
+ } Tooltips[] = {
+ IDC_REPLYDLG_RESETCOUNTERWHENSAMEICON, LPGENT("When this checkbox is ticked, NewAwaySys counts \"send times\" starting from the last status message change, even if status mode didn't change.\nWhen the checkbox isn't ticked, \"send times\" are counted from last status mode change (i.e. disabled state is more restrictive)."),
+ IDC_MOREOPTDLG_EVNTMSG, LPGENT("Message"),
+ IDC_MOREOPTDLG_EVNTURL, LPGENT("URL"),
+ IDC_MOREOPTDLG_EVNTFILE, LPGENT("File")
+ };
+ TOOLINFO ti = {0};
+ hWndTooltips = CreateWindowEx(WS_EX_TOPMOST, TOOLTIPS_CLASS, _T(""), WS_POPUP | TTS_NOPREFIX, 0, 0, 0, 0, NULL, NULL, GetModuleHandle(NULL), NULL);
+ ti.cbSize = sizeof(ti);
+ ti.uFlags = TTF_IDISHWND | TTF_SUBCLASS;
+ ti.hwnd = hwndDlg;
+ for (I = 0; I < lengthof(Tooltips); I++)
+ {
+ ti.uId = (UINT)GetDlgItem(hwndDlg, Tooltips[I].DlgItemID);
+ ti.lpszText = TranslateTS(Tooltips[I].Text);
+ SendMessage(hWndTooltips, TTM_ADDTOOL, 0, (LPARAM)&ti);
+ }
+ SendMessage(hWndTooltips, TTM_SETMAXTIPWIDTH, 0, 500);
+ SendMessage(hWndTooltips, TTM_SETDELAYTIME, TTDT_AUTOPOP, 32767); // tooltip hide time; looks like 32 seconds is the maximum
+
+ g_AutoreplyOptPage.DBToMemToPage();
+ EnableAutoreplyOptDlgControls();
+ ChangeLock--;
+ MakeGroupCheckbox(GetDlgItem(hwndDlg, IDC_REPLYDLG_ENABLEREPLY));
+ return true;
+ } break;
+ case UM_ICONSCHANGED:
+ {
+ int I;
+ for (I = 0; I < lengthof(StatusButtons); I++)
+ {
+ SendDlgItemMessage(hwndDlg, StatusButtons[I].DlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[StatusButtons[I].IconIndex]);
+ }
+ my_variables_skin_helpbutton(hwndDlg, IDC_REPLYDLG_VARS);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_EVNTMSG, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[ILI_EVENT_MESSAGE]);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_EVNTURL, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[ILI_EVENT_URL]);
+ SendDlgItemMessage(hwndDlg, IDC_MOREOPTDLG_EVNTFILE, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[ILI_EVENT_FILE]);
+ } break;
+ case WM_NOTIFY:
+ {
+ switch (((NMHDR*)lParam)->code)
+ {
+ case PSN_APPLY:
+ {
+ g_AutoreplyOptPage.PageToMemToDB();
+ //UpdateSOEButtons();
+ return true;
+ } break;
+ }
+ } break;
+ case WM_COMMAND:
+ {
+ switch (HIWORD(wParam))
+ {
+ case BN_CLICKED:
+ {
+ switch (LOWORD(wParam))
+ {
+ case IDC_REPLYDLG_ENABLEREPLY:
+ {
+ EnableAutoreplyOptDlgControls();
+ } // go through
+ case IDC_REPLYDLG_DONTSENDTOICQ:
+ case IDC_REPLYDLG_DONTREPLYINVISIBLE:
+ case IDC_REPLYDLG_ONLYCLOSEDDLGREPLY:
+ case IDC_REPLYDLG_ONLYIDLEREPLY:
+ case IDC_REPLYDLG_RESETCOUNTERWHENSAMEICON:
+ case IDC_REPLYDLG_EVENTMSG:
+ case IDC_REPLYDLG_EVENTURL:
+ case IDC_REPLYDLG_EVENTFILE:
+ case IDC_REPLYDLG_LOGREPLY:
+ case IDC_REPLYDLG_DISABLE_ONL:
+ case IDC_REPLYDLG_DISABLE_AWAY:
+ case IDC_REPLYDLG_DISABLE_NA:
+ case IDC_REPLYDLG_DISABLE_OCC:
+ case IDC_REPLYDLG_DISABLE_DND:
+ case IDC_REPLYDLG_DISABLE_FFC:
+ case IDC_REPLYDLG_DISABLE_INV:
+ case IDC_REPLYDLG_DISABLE_OTP:
+ case IDC_REPLYDLG_DISABLE_OTL:
+ {
+ if (!ChangeLock)
+ {
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ } break;
+ case IDC_REPLYDLG_VARS:
+ {
+ my_variables_showhelp(hwndDlg, IDC_REPLYDLG_PREFIX);
+ } break;
+ }
+ } break;
+ case EN_CHANGE:
+ {
+ if ((LOWORD(wParam) == IDC_REPLYDLG_SENDCOUNT) || (LOWORD(wParam) == IDC_REPLYDLG_PREFIX))
+ {
+ if (!ChangeLock && g_AutoreplyOptPage.GetWnd())
+ {
+ if (LOWORD(wParam) == IDC_REPLYDLG_SENDCOUNT)
+ {
+ EnableAutoreplyOptDlgControls();
+ }
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ }
+ } break;
+ case CBN_SELCHANGE:
+ {
+ if (LOWORD(wParam) == IDC_REPLYDLG_ONLYIDLEREPLY_COMBO)
+ {
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ } break;
+ }
+ } break;
+ case WM_DESTROY:
+ {
+ g_AutoreplyOptPage.SetWnd(NULL);
+ if (s_hDrawFont)
+ {
+ DeleteObject(s_hDrawFont);
+ }
+ DestroyWindow(hWndTooltips);
+ } break;
+ }
+ return 0;
+}
+// ================================================ Modern options ==============================================
+
+INT_PTR CALLBACK MessagesModernOptDlg(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
+{
+ static int ChangeLock = 0;
+ static CMsgTree* MsgTree = NULL;
+ static struct {
+ int DlgItem, Status, IconIndex;
+ } DefMsgDlgItems[] = {
+ IDC_MESSAGEDLG_DEF_ONL, ID_STATUS_ONLINE, ILI_PROTO_ONL,
+ IDC_MESSAGEDLG_DEF_AWAY, ID_STATUS_AWAY, ILI_PROTO_AWAY,
+ IDC_MESSAGEDLG_DEF_NA, ID_STATUS_NA, ILI_PROTO_NA,
+ IDC_MESSAGEDLG_DEF_OCC, ID_STATUS_OCCUPIED, ILI_PROTO_OCC,
+ IDC_MESSAGEDLG_DEF_DND, ID_STATUS_DND, ILI_PROTO_DND,
+ IDC_MESSAGEDLG_DEF_FFC, ID_STATUS_FREECHAT, ILI_PROTO_FFC,
+ IDC_MESSAGEDLG_DEF_INV, ID_STATUS_INVISIBLE, ILI_PROTO_INV,
+ IDC_MESSAGEDLG_DEF_OTP, ID_STATUS_ONTHEPHONE, ILI_PROTO_OTP,
+ IDC_MESSAGEDLG_DEF_OTL, ID_STATUS_OUTTOLUNCH, ILI_PROTO_OTL
+ };
+ static struct {
+ int DlgItem, IconIndex;
+ TCHAR* Text;
+ } Buttons[] = {
+ IDC_MESSAGEDLG_NEWMSG, ILI_NEWMESSAGE, LPGENT("Create new message"),
+ IDC_MESSAGEDLG_NEWCAT, ILI_NEWCATEGORY, LPGENT("Create new category"),
+ IDC_MESSAGEDLG_DEL, ILI_DELETE, LPGENT("Delete"),
+ IDC_MESSAGEDLG_VARS, ILI_NOICON, LPGENT("Open Variables help dialog"),
+ };
+ switch (msg)
+ {
+ case WM_INITDIALOG:
+ {
+ TranslateDialogDefault(hwndDlg);
+ MySetPos(hwndDlg);
+ ChangeLock++;
+ g_MessagesOptPage.SetWnd(hwndDlg);
+ SendMessage(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTITLE), EM_LIMITTEXT, TREEITEMTITLE_MAXLEN, 0);
+ SendMessage(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGDATA), EM_LIMITTEXT, AWAY_MSGDATA_MAX, 0);
+ // init image buttons
+ int I;
+ for (I = 0; I < lengthof(Buttons); I++)
+ {
+ HWND hButton = GetDlgItem(hwndDlg, Buttons[I].DlgItem);
+ SendMessage(hButton, BUTTONADDTOOLTIP, (WPARAM)TranslateTS(Buttons[I].Text), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ }
+ // now default status message buttons
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ HWND hButton = GetDlgItem(hwndDlg, DefMsgDlgItems[I].DlgItem);
+ SendMessage(hButton, BUTTONADDTOOLTIP, CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, DefMsgDlgItems[I].Status, GSMDF_TCHAR), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASPUSHBTN, TRUE, 0);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ g_OrigDefStatusButtonMsgProc = (WNDPROC)SetWindowLongPtr(hButton, GWLP_WNDPROC, (LONG_PTR)DefStatusButtonSubclassProc);
+ }
+ SendMessage(hwndDlg, UM_ICONSCHANGED, 0, 0);
+ g_MessagesOptPage.DBToMemToPage();
+ _ASSERT(!MsgTree);
+ MsgTree = new CMsgTree(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTREE));
+ if (!MsgTree->SetSelection(MsgTree->GetDefMsg(ID_STATUS_AWAY), MTSS_BYID))
+ {
+ MsgTree->SetSelection(g_Messages_PredefinedRootID, MTSS_BYID);
+ }
+ ChangeLock--;
+ //MakeGroupCheckbox(GetDlgItem(hwndDlg, IDC_REPLYDLG_ENABLEREPLY));
+ return true;
+ } break;
+ case UM_ICONSCHANGED:
+ {
+ int I;
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ SendDlgItemMessage(hwndDlg, DefMsgDlgItems[I].DlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[DefMsgDlgItems[I].IconIndex]);
+ }
+ for (I = 0; I < lengthof(Buttons); I++)
+ {
+ if (Buttons[I].IconIndex != ILI_NOICON)
+ {
+ SendDlgItemMessage(hwndDlg, Buttons[I].DlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[Buttons[I].IconIndex]);
+ }
+ }
+ my_variables_skin_helpbutton(hwndDlg, IDC_MESSAGEDLG_VARS);
+ } break;
+ case WM_NOTIFY:
+ {
+ switch (((NMHDR*)lParam)->code)
+ {
+ case PSN_APPLY:
+ {
+ HWND hTreeView = GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTREE);
+ HTREEITEM hSelectedItem = TreeView_GetSelection(hTreeView);
+ ChangeLock++;
+ TreeView_SelectItem(hTreeView, NULL);
+ TreeView_SelectItem(hTreeView, hSelectedItem);
+ ChangeLock--;
+ MsgTree->Save();
+ return true;
+ } break;
+ }
+ if (((LPNMHDR)lParam)->idFrom == IDC_MESSAGEDLG_MSGTREE)
+ {
+ switch (((LPNMHDR)lParam)->code)
+ {
+ case MTN_SELCHANGED:
+ {
+ PNMMSGTREE pnm = (PNMMSGTREE)lParam;
+ int I;
+ if (pnm->ItemOld && !(pnm->ItemOld->Flags & (TIF_ROOTITEM | TIF_GROUP)))
+ {
+ TCString Msg;
+ GetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, Msg.GetBuffer(AWAY_MSGDATA_MAX), AWAY_MSGDATA_MAX);
+ Msg.ReleaseBuffer();
+ if (((CTreeItem*)pnm->ItemOld)->User_Str1 != (const TCHAR*)Msg)
+ {
+ ((CTreeItem*)pnm->ItemOld)->User_Str1 = Msg;
+ MsgTree->SetModified(true);
+ }
+ }
+ if (pnm->ItemNew)
+ {
+ ChangeLock++;
+ if (!(pnm->ItemNew->Flags & TIF_ROOTITEM))
+ {
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, pnm->ItemNew->Title);
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, (pnm->ItemNew->Flags & TIF_GROUP) ? _T("") : ((CTreeItem*)pnm->ItemNew)->User_Str1);
+ } else
+ {
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, _T(""));
+ if (pnm->ItemNew->ID == g_Messages_RecentRootID)
+ {
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, TranslateT("Your most recent status messages are placed in this category. It's not recommended to put your messages manually here, as they'll be replaced by your recent messages."));
+ } else
+ {
+ _ASSERT(pnm->ItemNew->ID == g_Messages_PredefinedRootID);
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGDATA, TranslateT("You can put your frequently used and favorite messages in this category."));
+ }
+ }
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ COptItem_Checkbox *Checkbox = (COptItem_Checkbox*)g_MessagesOptPage.Find(DefMsgDlgItems[I].DlgItem);
+ Checkbox->SetWndValue(g_MessagesOptPage.GetWnd(), MsgTree->GetDefMsg(DefMsgDlgItems[I].Status) == pnm->ItemNew->ID);
+ }
+ ChangeLock--;
+ }
+ EnableMessagesOptDlgControls(MsgTree);
+ return 0;
+ } break;
+ case MTN_DEFMSGCHANGED:
+ {
+ PNMMSGTREE pnm = (PNMMSGTREE)lParam;
+ if (!ChangeLock)
+ {
+ CBaseTreeItem* SelectedItem = MsgTree->GetSelection();
+ _ASSERT(SelectedItem);
+ if ((pnm->ItemOld && pnm->ItemOld->ID == SelectedItem->ID) || (pnm->ItemNew && pnm->ItemNew->ID == SelectedItem->ID))
+ { // SelectedItem contains the same info as one of ItemOld or ItemNew - so we'll just use SelectedItem and won't bother with identifying which of ItemOld or ItemNew is currently selected
+ int I;
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ COptItem_Checkbox *Checkbox = (COptItem_Checkbox*)g_MessagesOptPage.Find(DefMsgDlgItems[I].DlgItem);
+ Checkbox->SetWndValue(g_MessagesOptPage.GetWnd(), MsgTree->GetDefMsg(DefMsgDlgItems[I].Status) == SelectedItem->ID);
+ }
+ }
+ }
+ if (!ChangeLock)
+ {
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ return 0;
+ } break;
+ case MTN_ITEMRENAMED:
+ {
+ PNMMSGTREE pnm = (PNMMSGTREE)lParam;
+ CBaseTreeItem* SelectedItem = MsgTree->GetSelection();
+ _ASSERT(SelectedItem);
+ if (pnm->ItemNew->ID == SelectedItem->ID && !ChangeLock)
+ {
+ ChangeLock++;
+ SetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, pnm->ItemNew->Title);
+ ChangeLock--;
+ }
+ } // go through
+ case MTN_ENDDRAG:
+ case MTN_NEWCATEGORY:
+ case MTN_NEWMESSAGE:
+ case MTN_DELETEITEM:
+ case TVN_ITEMEXPANDED:
+ {
+ if (!ChangeLock)
+ {
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ return 0;
+ } break;
+ }
+ }
+ } break;
+ case WM_COMMAND:
+ {
+ switch (HIWORD(wParam))
+ {
+ case BN_CLICKED:
+ {
+
+
+ switch (LOWORD(wParam))
+ {
+
+ case IDC_MESSAGEDLG_DEF_ONL:
+ case IDC_MESSAGEDLG_DEF_AWAY:
+ case IDC_MESSAGEDLG_DEF_NA:
+ case IDC_MESSAGEDLG_DEF_OCC:
+ case IDC_MESSAGEDLG_DEF_DND:
+ case IDC_MESSAGEDLG_DEF_FFC:
+ case IDC_MESSAGEDLG_DEF_INV:
+ case IDC_MESSAGEDLG_DEF_OTP:
+ case IDC_MESSAGEDLG_DEF_OTL:
+ {
+ int I;
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ if (LOWORD(wParam) == DefMsgDlgItems[I].DlgItem)
+ {
+ MsgTree->SetDefMsg(DefMsgDlgItems[I].Status, MsgTree->GetSelection()->ID); // PSM_CHANGED is sent here through MTN_DEFMSGCHANGED, so we don't need to send it once more
+ break;
+ }
+ }
+ } break;
+ case IDC_MESSAGEDLG_VARS:
+ {
+ my_variables_showhelp(hwndDlg, IDC_MESSAGEDLG_MSGDATA);
+ } break;
+ case IDC_MESSAGEDLG_DEL:
+ {
+ MsgTree->EnsureVisible(MsgTree->GetSelection()->hItem);
+ MsgTree->DeleteSelectedItem();
+ } break;
+ case IDC_MESSAGEDLG_NEWCAT:
+ {
+ MsgTree->AddCategory();
+ SetFocus(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTITLE));
+ } break;
+ case IDC_MESSAGEDLG_NEWMSG:
+ {
+ MsgTree->AddMessage();
+ SetFocus(GetDlgItem(hwndDlg, IDC_MESSAGEDLG_MSGTITLE));
+ } break;
+
+ //NightFox:
+ case IDC_LNK_AUTOAWAY:
+ {
+ OPENOPTIONSDIALOG ood = {0};
+ ood.cbSize = sizeof(ood);
+ ood.pszPage = "Status";
+ ood.pszTab = "Autoreply";
+ Options_OpenPage(&ood);
+ break;
+ }
+
+ }
+
+
+ } break;
+ case EN_CHANGE:
+ {
+ if (LOWORD(wParam) == IDC_MESSAGEDLG_MSGDATA || LOWORD(wParam) == IDC_MESSAGEDLG_MSGTITLE)
+ {
+ if (!ChangeLock)
+ {
+ if (LOWORD(wParam) == IDC_MESSAGEDLG_MSGTITLE)
+ {
+ CBaseTreeItem* TreeItem = MsgTree->GetSelection();
+ if (TreeItem && !(TreeItem->Flags & TIF_ROOTITEM))
+ {
+ GetDlgItemText(hwndDlg, IDC_MESSAGEDLG_MSGTITLE, TreeItem->Title.GetBuffer(TREEITEMTITLE_MAXLEN), TREEITEMTITLE_MAXLEN);
+ TreeItem->Title.ReleaseBuffer();
+ ChangeLock++;
+ MsgTree->UpdateItem(TreeItem->ID);
+ ChangeLock--;
+ }
+ }
+ MsgTree->SetModified(true);
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, (WPARAM)hwndDlg, 0);
+ }
+ }
+ } break;
+ }
+ } break;
+ case WM_DESTROY:
+ {
+ delete MsgTree;
+ MsgTree = NULL;
+ g_MessagesOptPage.SetWnd(NULL);
+ } break;
+ }
+ return 0;
+}
+
+
+
+// ================================================ Contact list ================================================
+// Based on the code from built-in Miranda ignore module
+
+#define UM_CONTACTSDLG_RESETLISTOPTIONS (WM_USER + 20)
+
+#define EXTRACOLUMNSCOUNT 3
+#define IGNORECOLUMN 2
+#define AUTOREPLYCOLUMN 1
+#define NOTIFYCOLUMN 0
+
+#define EXTRAICON_DOT 0
+#define EXTRAICON_IGNORE 1
+#define EXTRAICON_AUTOREPLYON 2
+#define EXTRAICON_AUTOREPLYOFF 3
+//#define EXTRAICON_ENABLENOTIFY 4
+//#define EXTRAICON_DISABLENOTIFY 5
+//#define EXTRAICON_INDEFINITE 6
+#define EXTRAICON_INDEFINITE 4
+
+#define VAL_INDEFINITE (-2)
+
+static WNDPROC g_OrigContactsProc;
+
+__inline int DBValueToIgnoreIcon(int Value)
+{
+ switch (Value)
+ {
+ case VAL_INDEFINITE: return EXTRAICON_INDEFINITE;
+ case 0: return EXTRAICON_DOT;
+ default: return EXTRAICON_IGNORE;
+ }
+}
+
+__inline int IgnoreIconToDBValue(int Value)
+{
+ switch (Value)
+ {
+ case EXTRAICON_DOT: return 0;
+ case EXTRAICON_IGNORE: return 1;
+ default: return VAL_INDEFINITE; // EXTRAICON_INDEFINITE and 0xFF
+ }
+}
+
+__inline int DBValueToOptReplyIcon(int Value)
+{
+ switch (Value)
+ {
+ case VAL_INDEFINITE: return EXTRAICON_INDEFINITE;
+ case VAL_USEDEFAULT: return EXTRAICON_DOT;
+ case 0: return EXTRAICON_AUTOREPLYOFF;
+ default: return EXTRAICON_AUTOREPLYON;
+ }
+}
+
+__inline int ReplyIconToDBValue(int Value)
+{
+ switch (Value)
+ {
+ case EXTRAICON_DOT: return VAL_USEDEFAULT;
+ case EXTRAICON_AUTOREPLYOFF: return 0;
+ case EXTRAICON_AUTOREPLYON: return 1;
+ default: return VAL_INDEFINITE; // EXTRAICON_INDEFINITE and 0xFF
+ }
+}
+/*
+__inline int DBValueToNotifyIcon(int Value)
+{
+ switch (Value)
+ {
+ case VAL_INDEFINITE: return EXTRAICON_INDEFINITE;
+ case VAL_USEDEFAULT: return EXTRAICON_DOT;
+ case 0: return EXTRAICON_DISABLENOTIFY;
+ default: return EXTRAICON_ENABLENOTIFY;
+ }
+}
+
+__inline int NotifyIconToDBValue(int Value)
+{
+ switch (Value)
+ {
+ case EXTRAICON_DOT: return VAL_USEDEFAULT;
+ case EXTRAICON_DISABLENOTIFY: return 0;
+ case EXTRAICON_ENABLENOTIFY: return 1;
+ default: return VAL_INDEFINITE; // EXTRAICON_INDEFINITE and 0xFF
+ }
+}*/
+
+static void SetListGroupIcons(HWND hwndList, HANDLE hFirstItem, HANDLE hParentItem)
+{
+ int Icons[EXTRACOLUMNSCOUNT] = {0xFF, 0xFF, 0xFF};
+ int I;
+ HANDLE hItem;
+ int FirstItemType = SendMessage(hwndList, CLM_GETITEMTYPE, (WPARAM)hFirstItem, 0);
+ hItem = (FirstItemType == CLCIT_GROUP) ? hFirstItem : (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTGROUP, (LPARAM)hFirstItem);
+ while (hItem)
+ {
+ HANDLE hChildItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_CHILD, (LPARAM)hItem);
+ if (hChildItem)
+ {
+ SetListGroupIcons(hwndList, hChildItem, hItem);
+ }
+ for (I = 0; I < lengthof(Icons); I++)
+ {
+ int Icon = SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(I, 0));
+ if (Icons[I] == 0xFF)
+ {
+ Icons[I] = Icon;
+ } else if (Icon != 0xFF && Icons[I] != Icon)
+ {
+ Icons[I] = EXTRAICON_INDEFINITE;
+ }
+ }
+ hItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTGROUP, (LPARAM)hItem);
+ }
+ hItem = (FirstItemType == CLCIT_CONTACT) ? hFirstItem : (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTCONTACT, (LPARAM)hFirstItem);
+ while (hItem)
+ {
+ for (I = 0; I < lengthof(Icons); I++)
+ {
+ int Icon = SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(I, 0));
+ if (Icons[I] == 0xFF)
+ {
+ Icons[I] = Icon;
+ } else if (Icon != 0xFF && Icons[I] != Icon)
+ {
+ Icons[I] = EXTRAICON_INDEFINITE;
+ }
+ }
+ hItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTCONTACT, (LPARAM)hItem);
+ }
+// set icons
+ for (I = 0; I < lengthof(Icons); I++)
+ {
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hParentItem, MAKELPARAM(I, Icons[I]));
+ }
+}
+
+static void SetAllChildIcons(HWND hwndList, HANDLE hFirstItem, int iColumn, int iImage)
+{
+ int typeOfFirst, iOldIcon;
+ HANDLE hItem, hChildItem;
+ typeOfFirst = SendMessage(hwndList, CLM_GETITEMTYPE, (WPARAM)hFirstItem, 0);
+ //check groups
+ if (typeOfFirst == CLCIT_GROUP)
+ {
+ hItem = hFirstItem;
+ } else
+ {
+ hItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTGROUP, (LPARAM)hFirstItem);
+ }
+ while (hItem)
+ {
+ hChildItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_CHILD, (LPARAM)hItem);
+ if (hChildItem)
+ {
+ SetAllChildIcons(hwndList, hChildItem, iColumn, iImage);
+ }
+ hItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTGROUP, (LPARAM)hItem);
+ }
+ //check contacts
+ if (typeOfFirst == CLCIT_CONTACT)
+ {
+ hItem = hFirstItem;
+ } else
+ {
+ hItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTCONTACT, (LPARAM)hFirstItem);
+ }
+ while (hItem)
+ {
+ iOldIcon = SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, iColumn);
+ if (iOldIcon != 0xFF && iOldIcon != iImage)
+ {
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(iColumn, iImage));
+ }
+ hItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_NEXTCONTACT, (LPARAM)hItem);
+ }
+}
+
+static void SetIconsForColumn(HWND hwndList, HANDLE hItem, HANDLE hItemAll, int iColumn, int iImage)
+{
+ int itemType = SendMessage(hwndList, CLM_GETITEMTYPE, (WPARAM)hItem, 0);
+ switch (itemType)
+ {
+ case CLCIT_CONTACT:
+ {
+ int oldiImage = SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, iColumn);
+ if (oldiImage != 0xFF && oldiImage != iImage)
+ {
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(iColumn, iImage));
+ }
+ } break;
+ case CLCIT_INFO:
+ {
+ if (hItem == hItemAll)
+ {
+ SetAllChildIcons(hwndList, hItem, iColumn, iImage);
+ } else
+ {
+// _ASSERT(hItem == hItemUnknown);
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(iColumn, iImage));
+ }
+ } break;
+ case CLCIT_GROUP:
+ {
+ hItem = (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_CHILD, (LPARAM)hItem);
+ if (hItem)
+ {
+ SetAllChildIcons(hwndList, hItem, iColumn, iImage);
+ }
+ } break;
+ }
+}
+
+static void SaveItemState(HWND hwndList, MCONTACT hContact, HANDLE hItem)
+{ // hContact == INVALID_HANDLE_VALUE means that hItem is hItemUnknown
+ int Ignore = IgnoreIconToDBValue(SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(IGNORECOLUMN, 0)));
+ int Reply = ReplyIconToDBValue(SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(AUTOREPLYCOLUMN, 0)));
+// int Notify = NotifyIconToDBValue(SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(NOTIFYCOLUMN, 0)));
+ if (Ignore != VAL_INDEFINITE)
+ {
+ CContactSettings(ID_STATUS_ONLINE, hContact).Ignore = Ignore;
+ }
+ if (Reply != VAL_INDEFINITE)
+ {
+ CContactSettings(ID_STATUS_ONLINE, hContact).Autoreply = Reply;
+ }
+/* if (Notify != VAL_INDEFINITE)
+ {
+ CContactSettings(ID_STATUS_ONLINE, hContact).PopupNotify = Notify;
+ }*/
+ if (hContact != INVALID_CONTACT_ID && g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS))
+ {
+ int iMode;
+ for (iMode = ID_STATUS_AWAY; iMode < ID_STATUS_OUTTOLUNCH; iMode++)
+ {
+ if (Ignore != VAL_INDEFINITE)
+ {
+ CContactSettings(iMode, hContact).Ignore = Ignore;
+ }
+ if (Reply != VAL_INDEFINITE)
+ {
+ CContactSettings(iMode, hContact).Autoreply = Reply;
+ } // Notify is not per-status, so we're not setting it here
+ }
+ }
+}
+
+static void SetAllContactIcons(HWND hwndList, HANDLE hItemUnknown)
+{
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hItemUnknown, MAKELPARAM(IGNORECOLUMN, DBValueToIgnoreIcon(CContactSettings(ID_STATUS_ONLINE, INVALID_CONTACT_ID).Ignore)));
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hItemUnknown, MAKELPARAM(AUTOREPLYCOLUMN, DBValueToOptReplyIcon(CContactSettings(ID_STATUS_ONLINE, INVALID_CONTACT_ID).Autoreply)));
+
+ MCONTACT hContact = db_find_first();
+ do {
+ HANDLE hItem = (HANDLE)SendMessage(hwndList, CLM_FINDCONTACT, (WPARAM)hContact, 0);
+ if (hItem) {
+ char *szProto = GetContactProto(hContact);
+ int Ignore = CContactSettings(ID_STATUS_ONLINE, hContact).Ignore;
+ int Reply = CContactSettings(ID_STATUS_ONLINE, hContact).Autoreply;
+// int Notify = CContactSettings(ID_STATUS_ONLINE, hContact).PopupNotify;
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS)) {
+ int iMode;
+ for (iMode = ID_STATUS_AWAY; iMode < ID_STATUS_OUTTOLUNCH; iMode++) {
+ if (CContactSettings(iMode, hContact).Ignore != Ignore)
+ Ignore = VAL_INDEFINITE;
+
+ if (CContactSettings(iMode, hContact).Autoreply != Reply)
+ Reply = VAL_INDEFINITE;
+ }
+ }
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(IGNORECOLUMN, DBValueToIgnoreIcon(Ignore)));
+ SendMessage(hwndList, CLM_SETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(AUTOREPLYCOLUMN, DBValueToOptReplyIcon(Reply)));
+ }
+ }
+ while (hContact = db_find_next(hContact));
+}
+
+static LRESULT CALLBACK ContactsSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ switch (Msg)
+ {
+ case WM_LBUTTONDBLCLK:
+ {
+ DWORD hitFlags;
+ HANDLE hItem = (HANDLE)SendMessage(hWnd, CLM_HITTEST, (WPARAM)&hitFlags, lParam);
+ if (hItem && (hitFlags & CLCHT_ONITEMEXTRA))
+ {
+ Msg = WM_LBUTTONDOWN; // may be considered as a hack, but it's needed to make clicking on extra icons more convenient
+ }
+ } break;
+ }
+ return CallWindowProc(g_OrigContactsProc, hWnd, Msg, wParam, lParam);
+}
+
+INT_PTR CALLBACK ContactsOptDlg(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
+{
+ static HANDLE hItemAll, hItemUnknown;
+ switch (msg)
+ {
+ case WM_INITDIALOG:
+ {
+ TranslateDialogDefault(hwndDlg);
+ MySetPos(hwndDlg);
+ HWND hwndList = GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST);
+ HIMAGELIST hIml = ImageList_Create(GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON), ILC_COLOR32 | ILC_MASK, 5, 2);
+ ImageList_AddIcon(hIml, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_DOT)));
+ ImageList_AddIcon(hIml, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_IGNORE)));
+ ImageList_AddIcon(hIml, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_ENABLED)));
+ ImageList_AddIcon(hIml, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_DISABLED)));
+ ImageList_AddIcon(hIml, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_INDEFINITE)));
+ SendMessage(hwndList, CLM_SETEXTRAIMAGELIST, 0, (LPARAM)hIml);
+ SendMessage(hwndDlg, UM_CONTACTSDLG_RESETLISTOPTIONS, 0, 0);
+ SendMessage(hwndList, CLM_SETEXTRACOLUMNS, EXTRACOLUMNSCOUNT, 0);
+ CLCINFOITEM cii={0};
+ cii.cbSize = sizeof(cii);
+ cii.flags = CLCIIF_GROUPFONT;
+ cii.pszText = TranslateT("** All contacts **");
+ hItemAll = (HANDLE)SendMessage(hwndList, CLM_ADDINFOITEM, 0, (LPARAM)&cii);
+ cii.pszText = TranslateT("** Not-on-list contacts **"); // == Unknown contacts
+ hItemUnknown = (HANDLE)SendMessage(hwndList, CLM_ADDINFOITEM, 0, (LPARAM)&cii);
+ MCONTACT hContact = db_find_first();
+ do
+ {
+ char *szProto = GetContactProto(hContact);
+ if (szProto)
+ {
+ int Flag1 = CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_1, 0);
+ if ((Flag1 & PF1_IM) != PF1_IM && !(Flag1 & PF1_INDIVMODEMSG))
+ { // does contact's protocol supports message sending/receiving or individual status messages?
+ SendMessage(hwndList, CLM_DELETEITEM, SendMessage(hwndList, CLM_FINDCONTACT, (WPARAM)hContact, 0), 0);
+ }
+ }
+ }
+ while (hContact = db_find_next(hContact));
+ SetAllContactIcons(hwndList, hItemUnknown);
+ SetListGroupIcons(hwndList, (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_ROOT, 0), hItemAll);
+ g_OrigContactsProc = (WNDPROC)SetWindowLongPtr(hwndList, GWLP_WNDPROC, (LONG_PTR)ContactsSubclassProc);
+ } break;
+ case UM_CONTACTSDLG_RESETLISTOPTIONS:
+ {
+ HWND hwndList = GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST);
+ SendMessage(hwndList, CLM_SETBKBITMAP, 0, NULL);
+ SendMessage(hwndList, CLM_SETBKCOLOR, GetSysColor(COLOR_WINDOW), 0);
+ SendMessage(hwndList, CLM_SETGREYOUTFLAGS, 0, 0);
+ SendMessage(hwndList, CLM_SETLEFTMARGIN, 4, 0);
+ SendMessage(hwndList, CLM_SETINDENT, 10, 0);
+ SendMessage(hwndList, CLM_SETHIDEEMPTYGROUPS, 1, 0);
+ int I;
+ for (I = 0; I <= FONTID_MAX; I++)
+ {
+ SendMessage(hwndList, CLM_SETTEXTCOLOR, I, GetSysColor(COLOR_WINDOWTEXT));
+ }
+ } break;
+ case WM_SETFOCUS:
+ {
+ SetFocus(GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST));
+ } break;
+ case WM_NOTIFY:
+ {
+ switch (((LPNMHDR)lParam)->idFrom)
+ {
+ case IDC_CONTACTSDLG_LIST:
+ {
+ switch (((LPNMHDR)lParam)->code)
+ {
+ case CLN_NEWCONTACT:
+ case CLN_LISTREBUILT:
+ {
+ SetAllContactIcons(GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST), hItemUnknown);
+ } // fall through
+ case CLN_CONTACTMOVED:
+ {
+ SetListGroupIcons(GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST), (HANDLE)SendDlgItemMessage(hwndDlg, IDC_CONTACTSDLG_LIST, CLM_GETNEXTITEM, CLGN_ROOT, 0), hItemAll);
+ } break;
+ case CLN_OPTIONSCHANGED:
+ {
+ SendMessage(hwndDlg, UM_CONTACTSDLG_RESETLISTOPTIONS, 0, 0);
+ } break;
+ case NM_CLICK:
+ case NM_DBLCLK:
+ {
+ NMCLISTCONTROL *nm = (NMCLISTCONTROL*)lParam;
+ HWND hwndList = GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST);
+ HANDLE hItem;
+ DWORD hitFlags;
+ if (nm->iColumn == -1)
+ {
+ break;
+ }
+ hItem = (HANDLE)SendMessage(hwndList, CLM_HITTEST, (WPARAM)&hitFlags, MAKELPARAM(nm->pt.x, nm->pt.y));
+ if (!hItem || !(hitFlags & CLCHT_ONITEMEXTRA))
+ {
+ break;
+ }
+ int iImage = SendMessage(hwndList, CLM_GETEXTRAIMAGE, (WPARAM)hItem, MAKELPARAM(nm->iColumn, 0));
+ switch (nm->iColumn)
+ {
+ case AUTOREPLYCOLUMN:
+ {
+ switch (iImage)
+ {
+ case EXTRAICON_DOT: iImage = EXTRAICON_AUTOREPLYOFF; break;
+ case EXTRAICON_AUTOREPLYOFF: iImage = EXTRAICON_AUTOREPLYON; break;
+ default: iImage = EXTRAICON_DOT; // EXTRAICON_AUTOREPLYON and EXTRAICON_INDEFINITE
+ }
+ } break;
+ case IGNORECOLUMN:
+ {
+ iImage = (iImage == EXTRAICON_DOT) ? EXTRAICON_IGNORE : EXTRAICON_DOT;
+ } break;
+ /*case NOTIFYCOLUMN:
+ {
+ switch (iImage)
+ {
+ case EXTRAICON_DOT: iImage = EXTRAICON_DISABLENOTIFY; break;
+ case EXTRAICON_DISABLENOTIFY: iImage = EXTRAICON_ENABLENOTIFY; break;
+ default: iImage = EXTRAICON_DOT; // EXTRAICON_ENABLENOTIFY and EXTRAICON_INDEFINITE
+ }
+ }*/ break;
+ }
+ SetIconsForColumn(hwndList, hItem, hItemAll, nm->iColumn, iImage);
+ SetListGroupIcons(hwndList, (HANDLE)SendMessage(hwndList, CLM_GETNEXTITEM, CLGN_ROOT, 0), hItemAll);
+ SendMessage(GetParent(hwndDlg), PSM_CHANGED, 0, 0);
+ } break;
+ }
+ }
+ break;
+ case 0:
+ switch (((LPNMHDR)lParam)->code) {
+ case PSN_APPLY:
+ for (MCONTACT hContact = db_find_first(); hContact; hContact = db_find_next(hContact)) {
+ HANDLE hItem = (HANDLE)SendDlgItemMessage(hwndDlg, IDC_CONTACTSDLG_LIST, CLM_FINDCONTACT, (WPARAM)hContact, 0);
+ if (hItem)
+ SaveItemState(GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST), hContact, hItem);
+ }
+
+ SaveItemState(GetDlgItem(hwndDlg, IDC_CONTACTSDLG_LIST), INVALID_CONTACT_ID, hItemUnknown);
+ return true;
+ }
+ break;
+ }
+ } break;
+ case WM_DESTROY:
+ HIMAGELIST hIml = (HIMAGELIST)SendDlgItemMessage(hwndDlg, IDC_CONTACTSDLG_LIST, CLM_GETEXTRAIMAGELIST, 0, 0);
+ _ASSERT(hIml);
+ ImageList_Destroy(hIml);
+ break;
+ }
+ return 0;
+}
+
+
+int OptsDlgInit(WPARAM wParam, LPARAM lParam)
+{
+ OPTIONSDIALOGPAGE optDi = {0};
+ optDi.cbSize = sizeof(optDi);
+ optDi.position = 920000000;
+ optDi.hInstance = g_hInstance;
+ optDi.flags = ODPF_BOLDGROUPS;
+
+ optDi.pszTitle = OPT_MAINGROUP;
+ optDi.pfnDlgProc = MessagesOptDlg;
+ optDi.pszTemplate = MAKEINTRESOURCEA(IDD_MESSAGES);
+ optDi.pszTab = LPGEN("Statuses messages");
+ Options_AddPage(wParam, &optDi);
+
+ optDi.pfnDlgProc = MoreOptDlg;
+ optDi.pszTemplate = MAKEINTRESOURCEA(IDD_MOREOPTDIALOG);
+ optDi.pszTab = LPGEN("Main options");
+ Options_AddPage(wParam, &optDi);
+
+ optDi.pfnDlgProc = AutoreplyOptDlg;
+ optDi.pszTemplate = MAKEINTRESOURCEA(IDD_AUTOREPLY);
+ optDi.pszTab = LPGEN("Autoreply");
+ Options_AddPage(wParam, &optDi);
+
+ optDi.pfnDlgProc = ContactsOptDlg;
+ optDi.pszTemplate = MAKEINTRESOURCEA(IDD_CONTACTSOPTDLG);
+ optDi.pszTab = LPGEN("Contacts");
+ Options_AddPage(wParam, &optDi);
+ return 0;
+}
+
+
+COptPage g_SetAwayMsgPage(MOD_NAME, NULL);
+COptPage g_MsgTreePage(MOD_NAME, NULL);
+
+void InitOptions()
+{
+ g_MessagesOptPage.Items.AddElem(new COptItem_Generic(IDC_MESSAGEDLG_VARS, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Generic(IDC_MESSAGEDLG_DEL));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Generic(IDC_MESSAGEDLG_MSGTITLE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Generic(IDC_MESSAGEDLG_MSGDATA));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_ONL, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_AWAY, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_NA, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_OCC, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_DND, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_FFC, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_INV, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_OTP, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ g_MessagesOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MESSAGEDLG_DEF_OTL, NULL, DBVT_BYTE, 0, 0, IDC_MESSAGEDLG_MSGTREE));
+ TreeItemArray DefMsgTree;
+ int ParentID1;
+ int ID = 0;
+ TreeRootItemArray RootItems;
+ RootItems.AddElem(CTreeRootItem(TranslateT("Predefined messages"), g_Messages_PredefinedRootID = ID++, TIF_EXPANDED));
+ RootItems.AddElem(CTreeRootItem(TranslateT("Recent messages"), g_Messages_RecentRootID = ID++, TIF_EXPANDED));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Gone fragging"), g_Messages_PredefinedRootID, ID++, 0, TranslateTS(_T("Been fragging since %") _T(VAR_AWAYSINCE_TIME) _T("%, I'll msg you later when the adrenaline wears off."))));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Creepy"), g_Messages_PredefinedRootID, ID++, 0, TranslateT("Your master, %nas_mynick%, has been %nas_statdesc% since the day that is only known as ?nas_awaysince_date(dddd)... When he gets back, I'll tell him you dropped by...")));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Default messages"), g_Messages_PredefinedRootID, ParentID1 = ID++, TIF_GROUP | TIF_EXPANDED));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_ONL, StatusToDBSetting(ID_STATUS_ONLINE, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Online"), ParentID1, ID++, 0, TranslateT("Yep, I'm here.")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_AWAY, StatusToDBSetting(ID_STATUS_AWAY, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Away"), ParentID1, ID++, 0, TranslateT("Been gone since %nas_awaysince_time%, will be back later.")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_NA, StatusToDBSetting(ID_STATUS_NA, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("NA"), ParentID1, ID++, 0, TranslateT("Give it up, I'm not in!")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_OCC, StatusToDBSetting(ID_STATUS_OCCUPIED, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Occupied"), ParentID1, ID++, 0, TranslateT("Not right now.")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_DND, StatusToDBSetting(ID_STATUS_DND, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("DND"), ParentID1, ID++, 0, TranslateT("Give a guy some peace, would ya?")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_FFC, StatusToDBSetting(ID_STATUS_FREECHAT, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Free for chat"), ParentID1, ID++, 0, TranslateT("I'm a chatbot!")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_INV, StatusToDBSetting(ID_STATUS_INVISIBLE, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Invisible"), ParentID1, ID++, 0, TranslateT("I'm hiding from the mafia.")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_OTP, StatusToDBSetting(ID_STATUS_ONTHEPHONE, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("On the phone"), ParentID1, ID++, 0, TranslateT("I've been on the phone since %nas_awaysince_time%, give me a sec!")));
+ g_MsgTreePage.Items.AddElem(new COptItem_IntDBSetting(IDS_MESSAGEDLG_DEF_OTL, StatusToDBSetting(ID_STATUS_OUTTOLUNCH, MESSAGES_DB_MSGTREEDEF), DBVT_WORD, false, ID));
+ DefMsgTree.AddElem(CTreeItem(TranslateT("Out to lunch"), ParentID1, ID++, 0, TranslateT("Been having ?ifgreater(?ctime(H),2,?ifgreater(?ctime(H),10,?ifgreater(?ctime(H),16,supper,dinner),breakfast),supper) since %nas_awaysince_time%.")));
+ g_MsgTreePage.Items.AddElem(new COptItem_TreeCtrl(IDV_MSGTREE, "MsgTree", DefMsgTree, RootItems, 0, "Text"));
+
+ g_SetAwayMsgPage.Items.AddElem(new COptItem_BitDBSetting(IDS_SAWAYMSG_SHOWMSGTREE, "SAMDlgFlags", DBVT_BYTE, DF_SAM_DEFDLGFLAGS, DF_SAM_SHOWMSGTREE));
+ g_SetAwayMsgPage.Items.AddElem(new COptItem_BitDBSetting(IDS_SAWAYMSG_SHOWCONTACTTREE, "SAMDlgFlags", DBVT_BYTE, DF_SAM_DEFDLGFLAGS, DF_SAM_SHOWCONTACTTREE));
+ g_SetAwayMsgPage.Items.AddElem(new COptItem_BitDBSetting(IDS_SAWAYMSG_AUTOSAVEDLGSETTINGS, "AutoSaveDlgSettings", DBVT_BYTE, 1));
+ g_SetAwayMsgPage.Items.AddElem(new COptItem_BitDBSetting(IDS_SAWAYMSG_DISABLEVARIABLES, "DisableVariables", DBVT_BYTE, 0));
+
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_PERSTATUSMRM, "PerStatusMRM", DBVT_BYTE, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_RESETPROTOMSGS, "ResetProtoMsgs", DBVT_BYTE, 1));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_PERSTATUSPROTOMSGS, "PerStatusProtoMsgs", DBVT_BYTE, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_PERSTATUSPROTOSETTINGS, "PerStatusProtoSettings", DBVT_BYTE, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_SAVEPERSONALMSGS, "SavePersonalMsgs", DBVT_BYTE, 1));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_PERSTATUSPERSONAL, "PerStatusPersonal", DBVT_BYTE, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS, "PerStatusPersonalSettings", DBVT_BYTE, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_USEMENUITEM, "UseMenuItem", DBVT_BYTE, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_MYNICKPERPROTO, "MyNickPerProto", DBVT_BYTE, 1));
+ g_MoreOptPage.Items.AddElem(new COptItem_IntEdit(IDC_MOREOPTDLG_WAITFORMSG, "WaitForMsg", DBVT_WORD, TRUE, 5));
+ g_MoreOptPage.Items.AddElem(new COptItem_IntEdit(IDC_MOREOPTDLG_RECENTMSGSCOUNT, "MRMCount", DBVT_WORD, TRUE, 5));
+ g_MoreOptPage.Items.AddElem(new COptItem_Radiobutton(IDC_MOREOPTDLG_USEDEFMSG, "UseByDefault", DBVT_BYTE, MOREOPTDLG_DEF_USEBYDEFAULT, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_Radiobutton(IDC_MOREOPTDLG_USELASTMSG, "UseByDefault", DBVT_BYTE, MOREOPTDLG_DEF_USEBYDEFAULT, 1));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_UPDATEMSGS, "UpdateMsgs", DBVT_BYTE, 0));
+ g_MoreOptPage.Items.AddElem(new COptItem_IntEdit(IDC_MOREOPTDLG_UPDATEMSGSPERIOD, "UpdateMsgsPeriod", DBVT_WORD, FALSE, 300));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_ONL, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_ONL));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_AWAY, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_AWAY));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_NA, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_NA));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_OCC, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_OCC));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_DND, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_DND));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_FFC, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_FFC));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_INV, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_INV));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_OTP, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_OTP));
+ g_MoreOptPage.Items.AddElem(new COptItem_Checkbox(IDC_MOREOPTDLG_DONTPOPDLG_OTL, "DontPopDlg", DBVT_WORD, MOREOPTDLG_DEF_DONTPOPDLG, SF_OTL));
+
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_ENABLEREPLY, DB_ENABLEREPLY, DBVT_BYTE, AUTOREPLY_DEF_REPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Generic(IDC_REPLYDLG_STATIC_ONEVENT, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_EVENTMSG, "ReplyOnEvent", DBVT_BYTE, AUTOREPLY_DEF_REPLYONEVENT, EF_MSG, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_EVENTURL, "ReplyOnEvent", DBVT_BYTE, AUTOREPLY_DEF_REPLYONEVENT, EF_URL, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_EVENTFILE, "ReplyOnEvent", DBVT_BYTE, AUTOREPLY_DEF_REPLYONEVENT, EF_FILE, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DONTSENDTOICQ, "DontSendToICQ", DBVT_BYTE, 0, 0, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DONTREPLYINVISIBLE, "DontReplyInvisible", DBVT_BYTE, 1, 0, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_LOGREPLY, "LogReply", DBVT_BYTE, 1, 0, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_ONLYIDLEREPLY, "OnlyIdleReply", DBVT_BYTE, 0, 0, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_ONLYCLOSEDDLGREPLY, "OnlyClosedDlgReply", DBVT_BYTE, 1, 0, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Generic(IDC_REPLYDLG_STATIC_SEND, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_IntEdit(IDC_REPLYDLG_SENDCOUNT, "ReplyCount", DBVT_WORD, TRUE, -1, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Generic(IDC_REPLYDLG_STATIC_TIMES, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_RESETCOUNTERWHENSAMEICON, "ResetReplyCounterWhenSameIcon", DBVT_BYTE, 1, 0, IDC_REPLYDLG_SENDCOUNT));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Generic(IDC_REPLYDLG_STATIC_DISABLEWHENSTATUS, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_ONL, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_ONL, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_AWAY, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_AWAY, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_NA, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_NA, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_OCC, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_OCC, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_DND, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_DND, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_FFC, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_FFC, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_INV, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_INV, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_OTP, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_OTP, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Checkbox(IDC_REPLYDLG_DISABLE_OTL, "DisableReply", DBVT_WORD, AUTOREPLY_DEF_DISABLEREPLY, SF_OTL, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Generic(IDC_REPLYDLG_STATIC_FORMAT, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Edit(IDC_REPLYDLG_PREFIX, "ReplyPrefix", AWAY_MSGDATA_MAX, AUTOREPLY_DEF_PREFIX, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Generic(IDC_REPLYDLG_VARS, IDC_REPLYDLG_ENABLEREPLY));
+ g_AutoreplyOptPage.Items.AddElem(new COptItem_Generic(IDC_REPLYDLG_STATIC_EXTRATEXT, IDC_REPLYDLG_ENABLEREPLY));
+}
+
+//NightFox
+int ModernOptInitialise(WPARAM wParam, LPARAM lParam)
+{
+ static int iBoldControls[] =
+ {
+ IDC_TXT_TITLE1,
+ IDC_TXT_TITLE2,
+ IDC_TXT_TITLE3,
+ MODERNOPT_CTRL_LAST
+ };
+
+ MODERNOPTOBJECT obj = {0};
+
+ obj.cbSize = sizeof(obj);
+ obj.dwFlags = MODEROPT_FLG_TCHAR|MODEROPT_FLG_NORESIZE;
+ obj.hInstance = g_hInstance;
+ obj.iSection = MODERNOPT_PAGE_STATUS;
+ obj.iType = MODERNOPT_TYPE_SECTIONPAGE;
+ obj.iBoldControls = iBoldControls;
+// obj.lpzClassicGroup = NULL;
+ obj.lpzClassicPage = "Status";
+ obj.lpzClassicTab = "Main options";
+// obj.lpzHelpUrl = "http://wiki.miranda-im.org/";
+
+ obj.lpzTemplate = MAKEINTRESOURCEA(IDD_MODERNOPT_MESSAGES);
+ obj.pfnDlgProc = MessagesModernOptDlg;
+ CallService(MS_MODERNOPT_ADDOBJECT, wParam, (LPARAM)&obj);
+ return 0;
+} \ No newline at end of file
diff --git a/plugins/NewAwaySysMod/src/AwaySys.cpp b/plugins/NewAwaySysMod/src/AwaySys.cpp
new file mode 100644
index 0000000000..1032780f79
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/AwaySys.cpp
@@ -0,0 +1,1004 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+ Copyright (c) 2004-2005 Iksaif Entertainment
+ Copyright (c) 2002-2003 Goblineye Entertainment
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+/*
+ Thanx to Faith Healer for icons and help, Unregistered for his plugins (variables, AAA :p),
+ mistag for GamerStatus, BigMuscle for his code to use AAA
+ Thanx to Tornado, orignal developer of AwaySys.
+ Please note that some code from the Miranda's original away module (SRAway) is used around
+ AwaySys. i tried to mention it wherever possible, but i might have forgotten a few. Kudos to Miranda's authors.
+ The Read-Away-Msg part was practically copied from Miranda, not proud of it, but since i really can't see how can i make it better, there
+ was no point in rewriting it all.
+*/
+
+#include "Common.h"
+#include "m_genmenu.h"
+#include "m_idle.h"
+#include "m_statusplugins.h"
+#include "m_NewAwaySys.h"
+#include "m_ContactSettings.h"
+#include "MsgTree.h"
+#include "ContactList.h"
+#include "Properties.h"
+#include "Path.h"
+#include "Services.h"
+#include "version.h"
+
+//NightFox
+#include <m_modernopt.h>
+#include <process.h> // needed for MSVC 7 msvcr7*.dll patch
+
+HINSTANCE g_hInstance;
+
+int hLangpack = 0;
+LIST<void> hHooks(5), hServices(5);
+HANDLE g_hContactMenuItem = NULL, g_hReadStatMenuItem = NULL, /*g_hTopToolbarbutton = NULL, */g_hToggleSOEMenuItem = NULL, g_hToggleSOEContactMenuItem = NULL, g_hAutoreplyOnContactMenuItem = NULL, g_hAutoreplyOffContactMenuItem = NULL, g_hAutoreplyUseDefaultContactMenuItem = NULL;
+bool g_fNoProcessing = false; // tells the status change proc not to do anything
+int g_bIsIdle = false;
+HANDLE hMainThread;
+int g_CSProtoCount = 0; // CommonStatus - StartupStatus and AdvancedAutoAway
+VAR_PARSE_DATA VarParseData;
+INT_PTR (*g_OldCallService)(const char *, WPARAM, LPARAM) = NULL;
+
+static struct
+{
+ int Status, DisableReplyCtlID, DontShowDialogCtlID;
+} StatusModeList[] = {
+ ID_STATUS_ONLINE, IDC_REPLYDLG_DISABLE_ONL, IDC_MOREOPTDLG_DONTPOPDLG_ONL,
+ ID_STATUS_AWAY, IDC_REPLYDLG_DISABLE_AWAY, IDC_MOREOPTDLG_DONTPOPDLG_AWAY,
+ ID_STATUS_NA, IDC_REPLYDLG_DISABLE_NA, IDC_MOREOPTDLG_DONTPOPDLG_NA,
+ ID_STATUS_OCCUPIED, IDC_REPLYDLG_DISABLE_OCC, IDC_MOREOPTDLG_DONTPOPDLG_OCC,
+ ID_STATUS_DND, IDC_REPLYDLG_DISABLE_DND, IDC_MOREOPTDLG_DONTPOPDLG_DND,
+ ID_STATUS_FREECHAT, IDC_REPLYDLG_DISABLE_FFC, IDC_MOREOPTDLG_DONTPOPDLG_FFC,
+ ID_STATUS_INVISIBLE, IDC_REPLYDLG_DISABLE_INV, IDC_MOREOPTDLG_DONTPOPDLG_INV,
+ ID_STATUS_ONTHEPHONE, IDC_REPLYDLG_DISABLE_OTP, IDC_MOREOPTDLG_DONTPOPDLG_OTP,
+ ID_STATUS_OUTTOLUNCH, IDC_REPLYDLG_DISABLE_OTL, IDC_MOREOPTDLG_DONTPOPDLG_OTL
+};
+
+// took this nice idea from MetaContacts plugin ;)
+// my_make_version is required to break up #define PRODUCTVER from VersionNo.h
+DWORD my_make_version(const int a, const int b, const int c, const int d)
+{
+ return PLUGIN_MAKE_VERSION(a, b, c, d);
+}
+
+PLUGININFOEX pluginInfo = {
+ sizeof(PLUGININFOEX),
+ "New Away System Mod",
+ my_make_version(PRODUCTVER), // see VersionNo.h
+ "New Away System Mod plugin for Miranda IM.",
+ "NightFox; Deathdemon; XF007; Goblineye Entertainment",
+ "NightFox@myied.org",
+ "© 2010 NightFox; © 2005-2007 Chervov Dmitry; © 2004-2005 Iksaif; © 2002-2003 Goblineye Entertainment",
+ "http://MyiEd.org/packs",
+ UNICODE_AWARE,
+ {0xb2dd9270, 0xce5e, 0x11df, {0xbd, 0x3d, 0x8, 0x0, 0x20, 0xc, 0x9a, 0x66}}
+};
+
+BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
+{
+ g_hInstance = hinstDLL;
+ return TRUE;
+}
+
+extern "C" __declspec(dllexport) const MUUID MirandaInterfaces[] = {MIID_SRAWAY, MIID_LAST}; // TODO: add MIID_WHOISREADING here if there'll be any some time in future..
+
+extern "C" __declspec(dllexport) PLUGININFOEX *MirandaPluginInfoEx(DWORD mirandaVersion)
+{
+ return &pluginInfo;
+}
+
+//NightFox
+int ModernOptInitialise(WPARAM wParam,LPARAM lParam);
+
+TCString GetDynamicStatMsg(MCONTACT hContact, char *szProto, DWORD UIN, int iStatus)
+{
+// hContact is the contact that requests the status message
+ if (hContact != INVALID_CONTACT_ID)
+ VarParseData.Message = CContactSettings(iStatus, hContact).GetMsgFormat(GMF_ANYCURRENT, NULL, szProto);
+ else // contact is unknown
+ VarParseData.Message = CProtoSettings(szProto, iStatus).GetMsgFormat(iStatus ? GMF_LASTORDEFAULT : GMF_ANYCURRENT);
+
+ TCString sTime;
+ VarParseData.szProto = szProto ? szProto : ((hContact && hContact != INVALID_CONTACT_ID) ? (char *)CallService(MS_PROTO_GETCONTACTBASEPROTO, (WPARAM)hContact, 0) : NULL);
+ VarParseData.UIN = UIN;
+ VarParseData.Flags = 0;
+ if (ServiceExists(MS_VARS_FORMATSTRING) && !g_SetAwayMsgPage.GetDBValueCopy(IDS_SAWAYMSG_DISABLEVARIABLES))
+ {
+ FORMATINFO fi = {0};
+ fi.cbSize = sizeof(fi);
+ fi.tszFormat = VarParseData.Message;
+ fi.hContact = hContact;
+ fi.flags = FIF_TCHAR;
+ TCHAR *szResult = (TCHAR*)CallService(MS_VARS_FORMATSTRING, (WPARAM)&fi, 0);
+ if (szResult)
+ {
+ VarParseData.Message = szResult;
+ mir_free(szResult);
+ }
+ }
+ return VarParseData.Message = VarParseData.Message.Left(AWAY_MSGDATA_MAX);
+}
+
+
+int StatusMsgReq(WPARAM wParam, LPARAM lParam, CString &szProto)
+{
+ _ASSERT(szProto != NULL);
+ LogMessage("ME_ICQ_STATUSMSGREQ called. szProto=%s, Status=%d, UIN=%d", (char*)szProto, wParam, lParam);
+// find the contact
+ char *szFoundProto;
+ MCONTACT hFoundContact = NULL; // if we'll find the contact only on some other protocol, but not on szProto, then we'll use that hContact.
+ for (MCONTACT hContact = db_find_first(); hContact; hContact = db_find_next(hContact)) {
+ char *szCurProto = (char *)CallService(MS_PROTO_GETCONTACTBASEPROTO, (WPARAM)hContact, 0);
+ if (db_get_dw(hContact, szCurProto, "UIN", 0) == lParam) {
+ szFoundProto = szCurProto;
+ hFoundContact = hContact;
+ if (!strcmp(szCurProto, szProto))
+ break;
+ }
+ }
+
+ int iMode = ICQStatusToGeneralStatus(wParam);
+ if (!hFoundContact)
+ hFoundContact = INVALID_CONTACT_ID;
+ else if (iMode >= ID_STATUS_ONLINE && iMode <= ID_STATUS_OUTTOLUNCH)
+ // don't count xstatus requests
+ db_set_w(hFoundContact, MOD_NAME, DB_REQUESTCOUNT, db_get_w(hFoundContact, MOD_NAME, DB_REQUESTCOUNT, 0) + 1);
+
+ MCONTACT hContactForSettings = hFoundContact; // used to take into account not-on-list contacts when getting contact settings, but at the same time allows to get correct contact info for contacts that are in the DB
+ if (hContactForSettings != INVALID_CONTACT_ID && db_get_b(hContactForSettings, "CList", "NotOnList", 0))
+ hContactForSettings = INVALID_CONTACT_ID; // INVALID_HANDLE_VALUE means the contact is not-on-list
+
+ if (g_SetAwayMsgPage.GetWnd()) {
+ CallAllowedPS_SETAWAYMSG(szProto, iMode, NULL); // we can set status messages to NULL here, as they'll be changed again when the SAM dialog closes.
+ return 0;
+ }
+ if (CContactSettings(iMode, hContactForSettings).Ignore)
+ {
+ CallAllowedPS_SETAWAYMSG(szProto, iMode, ""); // currently NULL makes ICQ to ignore _any_ further status message requests until the next PS_SETAWAYMSG, so i can't use it here..
+ return 0; // move along, sir
+ }
+
+ if (iMode)
+ { // if it's not an xstatus message request
+ CallAllowedPS_SETAWAYMSG(szProto, iMode, (char*)_T2A(GetDynamicStatMsg(hFoundContact, szProto, lParam)));
+ }
+// COptPage PopupNotifyData(g_PopupOptPage);
+// PopupNotifyData.DBToMem();
+ VarParseData.szProto = szProto;
+ VarParseData.UIN = lParam;
+ VarParseData.Flags = 0;
+ if (!iMode)
+ {
+ VarParseData.Flags |= VPF_XSTATUS;
+ }
+/*
+ int ShowPopup = ((iMode == ID_STATUS_ONLINE && PopupNotifyData.GetValue(IDC_POPUPOPTDLG_ONLNOTIFY)) ||
+ (iMode == ID_STATUS_AWAY && PopupNotifyData.GetValue(IDC_POPUPOPTDLG_AWAYNOTIFY)) ||
+ (iMode == ID_STATUS_DND && PopupNotifyData.GetValue(IDC_POPUPOPTDLG_DNDNOTIFY)) ||
+ (iMode == ID_STATUS_NA && PopupNotifyData.GetValue(IDC_POPUPOPTDLG_NANOTIFY)) ||
+ (iMode == ID_STATUS_OCCUPIED && PopupNotifyData.GetValue(IDC_POPUPOPTDLG_OCCNOTIFY)) ||
+ (iMode == ID_STATUS_FREECHAT && PopupNotifyData.GetValue(IDC_POPUPOPTDLG_FFCNOTIFY)) ||
+ (!iMode && PopupNotifyData.GetValue(IDC_POPUPOPTDLG_OTHERNOTIFY))) &&
+ CContactSettings(iMode, hContactForSettings).PopupNotify.IncludingParents();
+
+// Show popup and play a sound
+ if (ShowPopup)
+ {
+ ShowPopup = ShowPopupNotification(PopupNotifyData, hFoundContact, iMode); // we need ShowPopup also to determine whether to log to file or not
+ }
+*/
+// Log status message request to a file
+// if (!PopupNotifyData.GetValue(IDC_POPUPOPTDLG_LOGONLYWITHPOPUP) || ShowPopup)
+// {
+ TCString LogMsg;
+ if (!iMode)
+ { // if it's an xstatus message request
+ LogMsg = db_get_s(NULL, szProto, "XStatusName", _T(""));
+ TCString XMsg(db_get_s(NULL, szProto, "XStatusMsg", _T("")));
+ if (XMsg.GetLen())
+ {
+ if (LogMsg.GetLen())
+ {
+ LogMsg += _T("\r\n");
+ }
+ LogMsg += XMsg;
+ }
+ } else
+ {
+ LogMsg = VarParseData.Message;
+ }
+ if (ServiceExists(MS_VARS_FORMATSTRING))
+ {
+ logservice_log(LOG_ID, hFoundContact, LogMsg);
+ } else
+ {
+ TCString szUIN;
+ _ultot(lParam, szUIN.GetBuffer(16), 10);
+ szUIN.ReleaseBuffer();
+ TCHAR *szStatDesc = iMode ? (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, iMode, 0) : STR_XSTATUSDESC;
+ if (!szStatDesc)
+ {
+ _ASSERT(0);
+ szStatDesc = _T("");
+ }
+ logservice_log(LOG_ID, hFoundContact, TCString((TCHAR*)CallService(MS_CLIST_GETCONTACTDISPLAYNAME, (WPARAM)hFoundContact, GCDNF_TCHAR)) + _T(" (") + szUIN + TranslateT(") read your ") + szStatDesc + TranslateT(" message:\r\n") + LogMsg);
+ }
+// }
+ return 0;
+}
+
+// Here is an ugly workaround to support multiple ICQ accounts
+// hope 5 icq accounts will be sufficient for everyone ;)
+#define MAXICQACCOUNTS 5
+CString ICQProtoList[MAXICQACCOUNTS];
+#define StatusMsgReqN(N) int StatusMsgReq##N(WPARAM wParam, LPARAM lParam) {return StatusMsgReq(wParam, lParam, ICQProtoList[N - 1]);}
+StatusMsgReqN(1) StatusMsgReqN(2) StatusMsgReqN(3) StatusMsgReqN(4) StatusMsgReqN(5)
+MIRANDAHOOK StatusMsgReqHooks[] = {StatusMsgReq1, StatusMsgReq2, StatusMsgReq3, StatusMsgReq4, StatusMsgReq5};
+
+int IsAnICQProto(char *szProto)
+{
+ int i;
+ for (i = 0; i < MAXICQACCOUNTS; i++)
+ {
+ if (ICQProtoList[i] == (const char*)szProto)
+ {
+ return true;
+ }
+ }
+ return false;
+}
+
+
+int StatusChanged(WPARAM wParam, LPARAM lParam)
+{
+// wParam = iMode
+// lParam = (char*)szProto
+ LogMessage("MS_CLIST_SETSTATUSMODE called. szProto=%s, Status=%d", lParam ? (char*)lParam : "NULL", wParam);
+ g_ProtoStates[(char*)lParam].Status = wParam;
+// let's check if we handle this thingy
+ if (g_fNoProcessing) { // we're told not to do anything
+ g_fNoProcessing = false; // take it off
+ return 0;
+ }
+ DWORD Flag1 = 0;
+ DWORD Flag3 = 0;
+ if (lParam) {
+ Flag1 = CallProtoService((char*)lParam, PS_GETCAPS, PFLAGNUM_1, 0);
+ Flag3 = CallProtoService((char*)lParam, PS_GETCAPS, PFLAGNUM_3, 0);
+ }
+ else {
+ PROTOACCOUNT **proto;
+ int iProtoCount = 0;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&iProtoCount, (LPARAM)&proto);
+ int i;
+ for (i = 0; i < iProtoCount; i++) {
+ Flag1 |= CallProtoService(proto[i]->szModuleName, PS_GETCAPS, PFLAGNUM_1, 0);
+ Flag3 |= CallProtoService(proto[i]->szModuleName, PS_GETCAPS, PFLAGNUM_3, 0);
+ }
+ }
+ if (!(Flag1 & PF1_MODEMSGSEND || Flag3 & Proto_Status2Flag(wParam) || (Flag1 & PF1_IM) == PF1_IM))
+ {
+ return 0; // there are no protocols with changed status that support autoreply or away messages for this status
+ }
+ if (g_SetAwayMsgPage.GetWnd())
+ {
+ SetForegroundWindow(g_SetAwayMsgPage.GetWnd());
+ return 0;
+ }
+ int i;
+ for (i = lengthof(StatusModeList) - 1; i >= 0; i--)
+ {
+ if (wParam == StatusModeList[i].Status)
+ {
+ break;
+ }
+ }
+ if (i < 0)
+ {
+ return 0;
+ }
+ BOOL bScreenSaverRunning;
+ SystemParametersInfo(SPI_GETSCREENSAVERRUNNING, 0, &bScreenSaverRunning, 0);
+ if (bScreenSaverRunning || g_MoreOptPage.GetDBValueCopy(StatusModeList[i].DontShowDialogCtlID))
+ {
+ CProtoSettings((char*)lParam).SetMsgFormat(SMF_PERSONAL, CProtoSettings((char*)lParam).GetMsgFormat(GMF_LASTORDEFAULT));
+ ChangeProtoMessages((char*)lParam, wParam, TCString());
+ } else
+ {
+ SetAwayMsgData *dat = new SetAwayMsgData;
+ ZeroMemory(dat, sizeof(SetAwayMsgData));
+ dat->szProtocol = (char*)lParam;
+ dat->IsModeless = false;
+ DialogBoxParam(g_hInstance, MAKEINTRESOURCE(IDD_SETAWAYMSG), NULL, SetAwayMsgDlgProc, (LPARAM)dat);
+ }
+ return 0;
+}
+
+
+#define ID_STATUS_LAST 40081 // yes, 40081 means internal CommonStatus' ID_STATUS_LAST here, not ID_STATUS_IDLE :-S
+#define ID_STATUS_CURRENT 40082
+#define ID_STATUS_DISABLED 41083
+int CSStatusChange(WPARAM wParam, LPARAM lParam) // CommonStatus plugins (StartupStatus and AdvancedAutoAway)
+{
+// wParam = PROTOCOLSETTINGEX** protoSettings
+ PROTOCOLSETTINGEX** ps = *(PROTOCOLSETTINGEX***)wParam;
+ if (!ps)
+ {
+ return -1;
+ }
+ LogMessage("ME_CS_STATUSCHANGEEX event:");
+ int i;
+ for (i = 0; i < g_CSProtoCount; i++)
+ {
+ LogMessage("%d: cbSize=%d, szProto=%s, status=%d, lastStatus=%d, szMsg:", i + 1, ps[i]->cbSize, ps[i]->szName ? (char*)ps[i]->szName : "NULL", ps[i]->status, ps[i]->lastStatus, ps[i]->szMsg ? ps[i]->szMsg : _T("NULL"));
+ if (ps[i]->status != ID_STATUS_DISABLED)
+ {
+ if (ps[i]->status != ID_STATUS_CURRENT)
+ {
+ g_ProtoStates[ps[i]->szName].Status = (ps[i]->status == ID_STATUS_LAST) ? ps[i]->lastStatus : ps[i]->status;
+ }
+ CProtoSettings(ps[i]->szName).SetMsgFormat(SMF_TEMPORARY, ps[i]->szMsg ? ps[i]->szMsg : CProtoSettings(ps[i]->szName).GetMsgFormat(GMF_LASTORDEFAULT));
+ }
+ }
+ return 0;
+}
+
+
+static int IdleChangeEvent(WPARAM wParam, LPARAM lParam)
+{
+ LogMessage("ME_IDLE_CHANGED event. lParam=0x%x", lParam); // yes, we don't do anything with status message changes on idle.. there seems to be no any good solution for the wrong status message issue :(
+ g_bIsIdle = lParam & IDF_ISIDLE;
+ return 0;
+}
+
+
+int CSModuleLoaded(WPARAM wParam, LPARAM lParam) // StartupStatus and AdvancedAutoAway
+{
+// wParam = ProtoCount
+ g_CSProtoCount = wParam;
+ return 0;
+}
+
+
+int PreBuildContactMenu(WPARAM hContact, LPARAM lParam)
+{
+ char *szProto = GetContactProto(hContact);
+ CLISTMENUITEM miSetMsg = {0};
+ miSetMsg.cbSize = sizeof(miSetMsg);
+ miSetMsg.flags = CMIM_FLAGS | CMIF_TCHAR | CMIF_HIDDEN;
+ CLISTMENUITEM miReadMsg = {0};
+ miReadMsg.cbSize = sizeof(miReadMsg);
+ miReadMsg.flags = CMIM_FLAGS | CMIF_TCHAR | CMIF_HIDDEN;
+ int iMode = szProto ? CallProtoService(szProto, PS_GETSTATUS, 0, 0) : 0;
+ int Flag1 = szProto ? CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_1, 0) : 0;
+ int iContactMode = db_get_w(hContact, szProto, "Status", ID_STATUS_OFFLINE);
+ TCHAR szSetStr[256], szReadStr[256];
+ if (szProto)
+ {
+ int i;
+ for (i = lengthof(StatusModeList) - 1; i >= 0; i--)
+ {
+ if (iMode == StatusModeList[i].Status)
+ {
+ break;
+ }
+ }
+ if ((Flag1 & PF1_MODEMSGSEND && CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_3, 0) & Proto_Status2Flag(iMode)) || ((Flag1 & PF1_IM) == PF1_IM && (i < 0 || !g_AutoreplyOptPage.GetDBValueCopy(StatusModeList[i].DisableReplyCtlID))))
+ { // the protocol supports status message sending for current status, or autoreplying
+ mir_sntprintf(szSetStr, SIZEOF(szSetStr), TranslateT("Set %s message for the contact"), CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, iMode, GSMDF_TCHAR), CallService(MS_CLIST_GETCONTACTDISPLAYNAME, (WPARAM)hContact, GCDNF_TCHAR));
+ miSetMsg.ptszName = szSetStr;
+ miSetMsg.flags = CMIM_FLAGS | CMIF_TCHAR | CMIM_NAME;
+ }
+ if (Flag1 & PF1_MODEMSGRECV && CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_3, 0) & Proto_Status2Flag(iContactMode))
+ { // the protocol supports status message reading for contact's status
+ mir_sntprintf(szReadStr, SIZEOF(szReadStr), TranslateT("Re&ad %s Message"), CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, iContactMode, GSMDF_TCHAR));
+ miReadMsg.ptszName = szReadStr;
+ miReadMsg.flags = CMIM_FLAGS | CMIF_TCHAR | CMIM_NAME | CMIM_ICON;
+ miReadMsg.hIcon = LoadSkinnedProtoIcon(szProto, iContactMode);
+ }
+ }
+ if (g_hContactMenuItem)
+ {
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hContactMenuItem, (LPARAM)&miSetMsg);
+
+ if ((Flag1 & PF1_IM) == PF1_IM)
+ { // if this contact supports sending/receiving messages
+ int iAutoreply = CContactSettings(g_ProtoStates[szProto].Status, hContact).Autoreply;
+ CLISTMENUITEM mi = {0};
+ mi.cbSize = sizeof(mi);
+ mi.flags = CMIM_ICON | CMIM_FLAGS | CMIF_TCHAR;
+ switch (iAutoreply)
+ {
+ case VAL_USEDEFAULT: mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_DOT)); break;
+ case 0: mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_DISABLED)); break;
+ default: iAutoreply = 1; mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_ENABLED)); break;
+ }
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hToggleSOEContactMenuItem, (LPARAM)&mi);
+ mi.flags = CMIM_FLAGS | CMIF_TCHAR | (iAutoreply == 1 ? CMIF_CHECKED : 0);
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hAutoreplyOnContactMenuItem, (LPARAM)&mi);
+ mi.flags = CMIM_FLAGS | CMIF_TCHAR | (iAutoreply == 0 ? CMIF_CHECKED : 0);
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hAutoreplyOffContactMenuItem, (LPARAM)&mi);
+ mi.flags = CMIM_FLAGS | CMIF_TCHAR | (iAutoreply == VAL_USEDEFAULT ? CMIF_CHECKED : 0);
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hAutoreplyUseDefaultContactMenuItem, (LPARAM)&mi);
+ } else
+ { // hide the Autoreply menu item
+ CLISTMENUITEM mi = {0};
+ mi.cbSize = sizeof(mi);
+ mi.flags = CMIM_FLAGS | CMIF_TCHAR | CMIF_HIDDEN;
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hToggleSOEContactMenuItem, (LPARAM)&mi);
+ }
+ }
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hReadStatMenuItem, (LPARAM)&miReadMsg);
+ return 0;
+}
+
+
+static INT_PTR SetContactStatMsg(WPARAM hContact, LPARAM lParam)
+{
+ if (g_SetAwayMsgPage.GetWnd()) // already setting something
+ {
+ SetForegroundWindow(g_SetAwayMsgPage.GetWnd());
+ return 0;
+ }
+ SetAwayMsgData *dat = new SetAwayMsgData;
+ ZeroMemory(dat, sizeof(SetAwayMsgData));
+ dat->hInitContact = hContact;
+ dat->szProtocol = GetContactProto(hContact);
+ dat->IsModeless = false;
+ DialogBoxParam(g_hInstance, MAKEINTRESOURCE(IDD_SETAWAYMSG), NULL, SetAwayMsgDlgProc, (LPARAM)dat);
+ return 0;
+}
+
+/* //NightFox: deleted used-to-be support
+void UpdateSOEButtons(MCONTACT hContact)
+{
+ if (!hContact)
+ {
+ int SendOnEvent = CContactSettings(g_ProtoStates[(char*)NULL].Status).Autoreply;
+ CallService(MS_TTB_SETBUTTONSTATE, (WPARAM)g_hTopToolbarbutton, SendOnEvent ? TTBST_PUSHED : TTBST_RELEASED);
+ CLISTMENUITEM mi = {0};
+ mi.cbSize = sizeof(mi);
+ mi.position = 1000020000;
+ mi.flags = CMIF_TCHAR | CMIM_NAME | CMIM_ICON; // strange, but CMIF_TCHAR is still necessary even without CMIM_FLAGS
+ mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(SendOnEvent ? IDI_SOE_ENABLED : IDI_SOE_DISABLED));
+ mi.ptszName = SendOnEvent ? DISABLE_SOE_COMMAND : ENABLE_SOE_COMMAND;
+ CallService(MS_CLIST_MODIFYMENUITEM, (WPARAM)g_hToggleSOEMenuItem, (LPARAM)&mi);
+ }
+ if (g_SetAwayMsgPage.GetWnd())
+ {
+ SendMessage(g_SetAwayMsgPage.GetWnd(), UM_SAM_REPLYSETTINGCHANGED, (WPARAM)hContact, 0);
+ }
+}
+*/
+
+INT_PTR ToggleSendOnEvent(WPARAM hContact, LPARAM lParam)
+{ // used only for the global setting
+ CContactSettings(g_ProtoStates[hContact ? GetContactProto(hContact) : NULL].Status, hContact).Autoreply.Toggle();
+ //UpdateSOEButtons();
+ return 0;
+}
+
+
+INT_PTR srvAutoreplyOn(WPARAM hContact, LPARAM lParam)
+{
+ CContactSettings(g_ProtoStates[GetContactProto(hContact)].Status, hContact).Autoreply = 1;
+ //UpdateSOEButtons(hContact);
+ return 0;
+}
+
+
+INT_PTR srvAutoreplyOff(WPARAM hContact, LPARAM lParam)
+{
+ CContactSettings(g_ProtoStates[GetContactProto(hContact)].Status, hContact).Autoreply = 0;
+ //UpdateSOEButtons(hContact);
+ return 0;
+}
+
+
+INT_PTR srvAutoreplyUseDefault(WPARAM hContact, LPARAM lParam)
+{
+ CContactSettings(g_ProtoStates[GetContactProto(hContact)].Status, hContact).Autoreply = VAL_USEDEFAULT;
+ //UpdateSOEButtons(hContact);
+ return 0;
+}
+
+/* //NightFox: deleted used-to-be support
+int Create_TopToolbar(WPARAM wParam, LPARAM lParam)
+{
+ int SendOnEvent = CContactSettings(g_ProtoStates[(char*)NULL].Status).Autoreply;
+ if (ServiceExists(MS_TTB_ADDBUTTON))
+ {
+ TTBButton ttbb = {0};
+ ttbb.cbSize = sizeof(ttbb);
+ ttbb.hbBitmapUp = LoadBitmap(g_hInstance, MAKEINTRESOURCE(IDB_SOE_DISABLED));
+ ttbb.hbBitmapDown = LoadBitmap(g_hInstance, MAKEINTRESOURCE(IDB_SOE_ENABLED));
+ ttbb.pszServiceUp = MS_AWAYSYS_AUTOREPLY_TOGGLE;
+ ttbb.pszServiceDown = MS_AWAYSYS_AUTOREPLY_TOGGLE;
+ ttbb.dwFlags = TTBBF_VISIBLE | TTBBF_SHOWTOOLTIP;
+ ttbb.name = Translate("Toggle autoreply on/off");
+ g_hTopToolbarbutton = (HANDLE)CallService(MS_TTB_ADDBUTTON, (WPARAM)&ttbb, 0);
+ CallService(MS_TTB_SETBUTTONSTATE, (WPARAM)g_hTopToolbarbutton, SendOnEvent ? TTBST_PUSHED : TTBST_RELEASED);
+ }
+ return 0;
+}
+*/
+
+static int IconsChanged(WPARAM wParam, LPARAM lParam)
+{
+ g_IconList.ReloadIcons();
+ if (g_MessagesOptPage.GetWnd())
+ {
+ SendMessage(g_MessagesOptPage.GetWnd(), UM_ICONSCHANGED, 0, 0);
+ }
+ if (g_MoreOptPage.GetWnd())
+ {
+ SendMessage(g_MoreOptPage.GetWnd(), UM_ICONSCHANGED, 0, 0);
+ }
+ if (g_AutoreplyOptPage.GetWnd())
+ {
+ SendMessage(g_AutoreplyOptPage.GetWnd(), UM_ICONSCHANGED, 0, 0);
+ }
+/* if (g_PopupOptPage.GetWnd())
+ {
+ SendMessage(g_PopupOptPage.GetWnd(), UM_ICONSCHANGED, 0, 0);
+ }*/
+ return 0;
+}
+
+
+static int ContactSettingsInit(WPARAM wParam, LPARAM lParam)
+{
+ CONTACTSETTINGSINIT *csi = (CONTACTSETTINGSINIT*)wParam;
+ char *szProto = (csi->Type == CSIT_CONTACT) ? GetContactProto(csi->hContact) : NULL;
+ if ((csi->Type == CSIT_GROUP) || szProto)
+ {
+ int Flag1 = (csi->Type == CSIT_CONTACT) ? CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_1, 0) : PF1_IM; // we assume that there can be some contacts in the group with PF1_IM capability
+ if ((Flag1 & PF1_IM) == PF1_IM || Flag1 & PF1_INDIVMODEMSG)
+ { // does contact's protocol supports message sending/receiving or individual status messages?
+ CONTACTSETTINGSCONTROL csc = {0};
+ csc.cbSize = sizeof(csc);
+ csc.cbStateSize = sizeof(CSCONTROLSTATE);
+ csc.Position = CSPOS_SORTBYALPHABET;
+ csc.ControlType = CSCT_CHECKBOX;
+ csc.szModule = MOD_NAME;
+ csc.StateNum = 3;
+ csc.DefState = 2; // these settings are used for all controls below
+
+ /*if ((csi->Type == CSIT_GROUP) || IsAnICQProto(szProto))
+ {
+ csc.Flags = CSCF_TCHAR;
+ csc.ptszTitle = LPGENT("New Away System: Status message request notifications");
+ csc.ptszGroup = CSGROUP_NOTIFICATIONS;
+ csc.ptszTooltip = NULL;
+ csc.szSetting = DB_POPUPNOTIFY;
+ CallService(MS_CONTACTSETTINGS_ADDCONTROL, wParam, (LPARAM)&csc);
+ }*/
+ int StatusMode = 0;
+ if (csi->Type == CSIT_CONTACT)
+ {
+ CContactSettings CSettings(0, csi->hContact);
+ StatusMode = CSettings.Status;
+ } else
+ {
+ _ASSERT(csi->Type == CSIT_GROUP);
+ StatusMode = g_ProtoStates[(char*)NULL].Status;
+ }
+ if (StatusMode == ID_STATUS_OFFLINE)
+ {
+ StatusMode = ID_STATUS_AWAY;
+ }
+ CString Setting;
+ TCHAR Title[128];
+ csc.Flags = CSCF_TCHAR | CSCF_DONT_TRANSLATE_STRINGS; // these Flags and ptszGroup are used for both controls below
+ csc.ptszGroup = TranslateT("New Away System");
+
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS))
+ {
+ mir_sntprintf(Title, SIZEOF(Title), TranslateT("Enable autoreply when you are %s"), (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, StatusMode, GSMDF_TCHAR));
+ csc.ptszTitle = Title;
+ csc.ptszTooltip = TranslateT("\"Store contact autoreply/ignore settings for each status separately\" is enabled, so this setting is per-contact AND per-status.");
+ } else
+ {
+ csc.ptszTitle = TranslateT("Enable autoreply");
+ csc.ptszTooltip = NULL;
+ }
+ Setting = StatusToDBSetting(StatusMode, DB_ENABLEREPLY, IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS);
+ csc.szSetting = Setting;
+ CallService(MS_CONTACTSETTINGS_ADDCONTROL, wParam, (LPARAM)&csc);
+
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS))
+ {
+ mir_sntprintf(Title, SIZEOF(Title), TranslateT("Don't send status message when you are %s"), (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, StatusMode, GSMDF_TCHAR));
+ csc.ptszTitle = Title;
+ csc.ptszTooltip = TranslateT("Ignore status message requests from this contact and don't send an autoreply.\r\n\"Store contact autoreply/ignore settings for each status separately\" is enabled, so this setting is per-contact AND per-status.");
+ } else
+ {
+ csc.ptszTitle = TranslateT("Don't send status message");
+ csc.ptszTooltip = TranslateT("Ignore status message requests from this contact and don't send an autoreply");
+ }
+ Setting = StatusToDBSetting(StatusMode, DB_IGNOREREQUESTS, IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS);
+ csc.szSetting = Setting;
+ CallService(MS_CONTACTSETTINGS_ADDCONTROL, wParam, (LPARAM)&csc);
+ }
+ }
+ return 0;
+}
+
+
+INT_PTR srvVariablesHandler(WPARAM wParam, LPARAM lParam)
+{
+ ARGUMENTSINFO *ai = (ARGUMENTSINFO*)lParam;
+ ai->flags = AIF_DONTPARSE;
+ TCString Result;
+ if (!lstrcmp(ai->targv[0], _T(VAR_AWAYSINCE_TIME)))
+ {
+ GetTimeFormat(MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT), 0, g_ProtoStates[VarParseData.szProto].AwaySince, (ai->argc > 1 && *ai->targv[1]) ? ai->targv[1] : _T("H:mm"), Result.GetBuffer(256), 256);
+ Result.ReleaseBuffer();
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_AWAYSINCE_DATE)))
+ {
+ GetDateFormat(MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT), 0, g_ProtoStates[VarParseData.szProto].AwaySince, (ai->argc > 1 && *ai->targv[1]) ? ai->targv[1] : NULL, Result.GetBuffer(256), 256);
+ Result.ReleaseBuffer();
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_STATDESC)))
+ {
+ Result = (VarParseData.Flags & VPF_XSTATUS) ? STR_XSTATUSDESC : (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, g_ProtoStates[VarParseData.szProto].Status, GSMDF_TCHAR);
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_MYNICK)))
+ {
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_MYNICKPERPROTO) && VarParseData.szProto)
+ {
+ Result = db_get_s(NULL, VarParseData.szProto, "Nick", (TCHAR*)NULL);
+ }
+ if (Result == NULL)
+ {
+ Result = (TCHAR*)CallService(MS_CLIST_GETCONTACTDISPLAYNAME, NULL, GCDNF_TCHAR);
+ }
+ if (Result == NULL)
+ {
+ Result = TranslateT("Stranger");
+ }
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_REQUESTCOUNT)))
+ {
+ mir_sntprintf(Result.GetBuffer(16), 16, _T("%d"), db_get_w(ai->fi->hContact, MOD_NAME, DB_REQUESTCOUNT, 0));
+ Result.ReleaseBuffer();
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_MESSAGENUM)))
+ {
+ mir_sntprintf(Result.GetBuffer(16), 16, _T("%d"), db_get_w(ai->fi->hContact, MOD_NAME, DB_MESSAGECOUNT, 0));
+ Result.ReleaseBuffer();
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_TIMEPASSED)))
+ {
+ ULARGE_INTEGER ul_AwaySince, ul_Now;
+ SYSTEMTIME st;
+ GetLocalTime(&st);
+ SystemTimeToFileTime(&st, (LPFILETIME)&ul_Now);
+ SystemTimeToFileTime(g_ProtoStates[VarParseData.szProto].AwaySince, (LPFILETIME)&ul_AwaySince);
+ ul_Now.QuadPart -= ul_AwaySince.QuadPart;
+ ul_Now.QuadPart /= 10000000; // now it's in seconds
+ Result.GetBuffer(256);
+ if (ul_Now.LowPart >= 7200) // more than 2 hours
+ {
+ mir_sntprintf(Result, 256, TranslateT("%d hours"), ul_Now.LowPart / 3600);
+ } else if (ul_Now.LowPart >= 120) // more than 2 minutes
+ {
+ mir_sntprintf(Result, 256, TranslateT("%d minutes"), ul_Now.LowPart / 60);
+ } else
+ {
+ mir_sntprintf(Result, 256, TranslateT("%d seconds"), ul_Now.LowPart);
+ }
+ Result.ReleaseBuffer();
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_PREDEFINEDMESSAGE)))
+ {
+ ai->flags = 0; // reset AIF_DONTPARSE flag
+ if (ai->argc != 2)
+ {
+ return NULL;
+ }
+ COptPage MsgTreeData(g_MsgTreePage);
+ COptItem_TreeCtrl *TreeCtrl = (COptItem_TreeCtrl*)MsgTreeData.Find(IDV_MSGTREE);
+ TreeCtrl->DBToMem(CString(MOD_NAME));
+ int i;
+ for (i = 0; i < TreeCtrl->Value.GetSize(); i++)
+ {
+ if (!(TreeCtrl->Value[i].Flags & TIF_GROUP) && !_tcsicmp(TreeCtrl->Value[i].Title, ai->targv[1]))
+ {
+ Result = TreeCtrl->Value[i].User_Str1;
+ break;
+ }
+ }
+ if (Result == NULL)
+ { // if we didn't find a message with specified title
+ return NULL; // return it now, as later we change NULL to ""
+ }
+ } else if (!lstrcmp(ai->targv[0], _T(VAR_PROTOCOL)))
+ {
+ if (VarParseData.szProto)
+ {
+ CString AnsiResult;
+ CallProtoService(VarParseData.szProto, PS_GETNAME, 256, (LPARAM)AnsiResult.GetBuffer(256));
+ AnsiResult.ReleaseBuffer();
+ Result = _A2T(AnsiResult);
+ }
+ if (Result == NULL)
+ { // if we didn't find a message with specified title
+ return NULL; // return it now, as later we change NULL to ""
+ }
+ }
+ TCHAR *szResult;
+ if (!(szResult = (TCHAR*)malloc((Result.GetLen() + 1) * sizeof(TCHAR))))
+ {
+ return NULL;
+ }
+ _tcscpy(szResult, (Result != NULL) ? Result : _T(""));
+ return (int)szResult;
+}
+
+
+INT_PTR srvFreeVarMem(WPARAM wParam, LPARAM lParam)
+{
+ if (!lParam)
+ {
+ return -1;
+ }
+ free((void*)lParam);
+ return 0;
+}
+
+
+static INT_PTR MyCallService(const char *name, WPARAM wParam, LPARAM lParam)
+{
+ if (name && wParam <= ID_STATUS_OUTTOLUNCH && wParam >= ID_STATUS_OFFLINE) // wParam conditions here are distinctive "features" of PS_SETSTATUS and PS_SETAWAYMSG services, so if wParam does not suit them, we'll pass the control to the old CallService function as soon as possible
+ {
+ const char *pProtoNameEnd = strrchr(name, '/');
+ if (pProtoNameEnd)
+ {
+ if (!lstrcmpA(pProtoNameEnd, PS_SETSTATUS))
+ {
+ // it's PS_SETSTATUS service; wParam = status; lParam = 0
+ // returns 0 on success, nonzero on failure
+ CString Proto("");
+ Proto.DiffCat(name, pProtoNameEnd);
+ if (wParam != g_ProtoStates[Proto].Status)
+ {
+ g_ProtoStates[Proto].Status = wParam;
+ TCString Msg(CProtoSettings(Proto).GetMsgFormat(GMF_LASTORDEFAULT));
+ LogMessage("Detected a PS_SETSTATUS call with Status different from the one known to NAS. szProto=%s, NewStatus=%d, NewMsg:\n%s", (char*)Proto, wParam, (Msg != NULL) ? _T2A(Msg) : "NULL");
+ CProtoSettings(Proto).SetMsgFormat(SMF_TEMPORARY, Msg);
+ }
+ }
+ else if (!lstrcmpA(pProtoNameEnd, PS_SETAWAYMSG)) {
+ // PS_SETAWAYMSG service; wParam = status; lParam = (const char*)szMessage
+ // returns 0 on success, nonzero on failure
+ CString Proto("");
+ Proto.DiffCat(name, pProtoNameEnd);
+ char *param = (char*)lParam;
+ LogMessage("Someone else than NAS called PS_SETAWAYMSG. szProto=%s, Status=%d, Msg:\n%s", (char*)Proto, wParam, param ? param : "NULL");
+ CProtoSettings(Proto).SetMsgFormat(SMF_TEMPORARY, lParam ? ((ServiceExists(MS_VARS_FORMATSTRING) && !g_SetAwayMsgPage.GetDBValueCopy(IDS_SAWAYMSG_DISABLEVARIABLES)) ? VariablesEscape((TCHAR*)_A2T(param)) : (TCHAR*)_A2T(param)) : TCString(_T("")));
+ ChangeProtoMessages(Proto, wParam, TCString());
+ return 0;
+ }
+ }
+ }
+ return g_OldCallService(name, wParam, lParam);
+}
+
+
+int MirandaLoaded(WPARAM wParam, LPARAM lParam)
+{
+ LoadMsgTreeModule();
+ LoadCListModule();
+ InitUpdateMsgs();
+ g_IconList.ReloadIcons();
+
+ PROTOCOLDESCRIPTOR **proto;
+ DuplicateHandle(GetCurrentProcess(), GetCurrentThread(), GetCurrentProcess(), &hMainThread, THREAD_SET_CONTEXT, false, 0);
+ int iProtoCount = 0;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&iProtoCount, (LPARAM)&proto);
+ int i;
+ int CurProtoIndex;
+ for (i = 0, CurProtoIndex = 0; i < iProtoCount && CurProtoIndex < MAXICQACCOUNTS; i++)
+ {
+ if (proto[i]->type == PROTOTYPE_PROTOCOL)
+ {
+ HANDLE hHook = HookEvent(CString(proto[i]->szName) + ME_ICQ_STATUSMSGREQ, StatusMsgReqHooks[CurProtoIndex]);
+ if (hHook)
+ {
+ hHooks.insert(hHook);
+ ICQProtoList[CurProtoIndex] = proto[i]->szName;
+ CurProtoIndex++;
+ }
+ }
+ }
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_SETCONTACTSTATMSG, SetContactStatMsg));
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_AUTOREPLY_TOGGLE, ToggleSendOnEvent));
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_AUTOREPLY_ON, srvAutoreplyOn));
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_AUTOREPLY_OFF, srvAutoreplyOff));
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_AUTOREPLY_USEDEFAULT, srvAutoreplyUseDefault));
+ hServices.insert(CreateServiceFunction(MS_NAS_GETSTATEA, GetStateA));
+ hServices.insert(CreateServiceFunction(MS_NAS_SETSTATEA, SetStateA));
+ hServices.insert(CreateServiceFunction(MS_NAS_GETSTATEW, GetStateW));
+ hServices.insert(CreateServiceFunction(MS_NAS_SETSTATEW, SetStateW));
+ hServices.insert(CreateServiceFunction(MS_NAS_INVOKESTATUSWINDOW, InvokeStatusWindow));
+ hServices.insert(CreateServiceFunction(MS_AWAYMSG_GETSTATUSMSG, GetStatusMsg));
+// and old AwaySysMod service, for compatibility reasons
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_SETSTATUSMODE, SetStatusMode));
+//NightFox: none;
+// hHooks.insert(HookEvent(ME_TTB_MODULELOADED, Create_TopToolbar));
+ hHooks.insert(HookEvent(ME_OPT_INITIALISE, OptsDlgInit));
+ hHooks.insert(HookEvent(ME_CLIST_STATUSMODECHANGE, StatusChanged));
+ hHooks.insert(HookEvent(ME_CS_STATUSCHANGEEX, CSStatusChange)); // for compatibility with StartupStatus and AdvancedAutoAway
+ hHooks.insert(HookEvent(ME_DB_EVENT_FILTER_ADD, MsgEventAdded));
+ hHooks.insert(HookEvent(ME_CLIST_PREBUILDCONTACTMENU, PreBuildContactMenu));
+ hHooks.insert(HookEvent(ME_SKIN_ICONSCHANGED, IconsChanged));
+ hHooks.insert(HookEvent(ME_IDLE_CHANGED, IdleChangeEvent));
+ hHooks.insert(HookEvent(ME_CONTACTSETTINGS_INITIALISE, ContactSettingsInit));
+ g_hReadWndList = (HANDLE)CallService(MS_UTILS_ALLOCWINDOWLIST, 0, 0);
+ int SendOnEvent = CContactSettings(g_ProtoStates[(char*)NULL].Status).Autoreply;
+
+ CLISTMENUITEM mi = {0};
+ mi.cbSize = sizeof(mi);
+ mi.position = 1000020000;
+ mi.flags = CMIF_TCHAR | CMIF_NOTOFFLINE;
+ mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(SendOnEvent ? IDI_SOE_ENABLED : IDI_SOE_DISABLED));
+ mi.ptszName = SendOnEvent ? DISABLE_SOE_COMMAND : ENABLE_SOE_COMMAND;
+ mi.pszService = MS_AWAYSYS_AUTOREPLY_TOGGLE;
+ g_hToggleSOEMenuItem = Menu_AddMainMenuItem(&mi);
+
+ ZeroMemory(&mi, sizeof(mi));
+ mi.cbSize = sizeof(mi);
+ mi.position = -2000005000;
+ mi.flags = CMIF_TCHAR | CMIF_NOTOFFLINE | CMIF_HIDDEN;
+ mi.ptszName = LPGENT("Read status message"); // never seen...
+ mi.pszService = MS_AWAYMSG_SHOWAWAYMSG;
+ g_hReadStatMenuItem = Menu_AddContactMenuItem(&mi);
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_USEMENUITEM)) {
+ ZeroMemory(&mi, sizeof(mi));
+ mi.cbSize = sizeof(mi);
+ mi.flags = CMIF_TCHAR | CMIF_HIDDEN;
+ mi.ptszName = LPGENT("Set status message"); // will never be shown
+ mi.position = 1000020000;
+ mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_MSGICON));
+ mi.pszService = MS_AWAYSYS_SETCONTACTSTATMSG;
+ g_hContactMenuItem = Menu_AddContactMenuItem(&mi);
+
+ ZeroMemory(&mi, sizeof(mi));
+ mi.cbSize = sizeof(mi);
+ mi.flags = CMIF_TCHAR | CMIF_ROOTPOPUP;
+ mi.hIcon = NULL;
+ mi.pszPopupName = (char*)-1;
+ mi.position = 1000020000;
+ mi.ptszName = LPGENT("Autoreply");
+ g_hToggleSOEContactMenuItem = Menu_AddContactMenuItem(&mi);
+
+ mi.flags = CMIF_TCHAR | CMIF_CHILDPOPUP;
+ mi.pszPopupName = (char*)g_hToggleSOEContactMenuItem;
+ mi.popupPosition = 1000020000;
+ mi.position = 1000020000;
+
+ mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_ENABLED));
+ mi.ptszName = LPGENT("On");
+ mi.pszService = MS_AWAYSYS_AUTOREPLY_ON;
+ g_hAutoreplyOnContactMenuItem = Menu_AddContactMenuItem(&mi);
+
+ mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_DISABLED));
+ mi.ptszName = LPGENT("Off");
+ mi.pszService = MS_AWAYSYS_AUTOREPLY_OFF;
+ g_hAutoreplyOffContactMenuItem = Menu_AddContactMenuItem(&mi);
+
+ mi.hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_DOT));
+ mi.ptszName = LPGENT("Use the default setting");
+ mi.pszService = MS_AWAYSYS_AUTOREPLY_USEDEFAULT;
+ g_hAutoreplyUseDefaultContactMenuItem = Menu_AddContactMenuItem(&mi);
+ }
+ // add that funky thingy (just tweaked a bit, was spotted in Miranda's src code)
+ // we have to read the status message from contacts too... err
+ hServices.insert(CreateServiceFunction(MS_AWAYMSG_SHOWAWAYMSG, GetContactStatMsg));
+
+ SkinAddNewSoundEx(AWAYSYS_STATUSMSGREQUEST_SOUND, NULL, LPGEN("NewAwaySys: Incoming status message request"));
+
+ if (ServiceExists(MS_VARS_REGISTERTOKEN)) {
+ struct
+ {
+ TCHAR *Name;
+ char *Descr;
+ int Flags;
+ } Variables[] = {
+ _T(VAR_AWAYSINCE_TIME), LPGEN("New Away System\t(x)\tAway since time in default format; ?nas_awaysince_time(x) in format x"), TRF_FIELD | TRF_FUNCTION,
+ _T(VAR_AWAYSINCE_DATE), LPGEN("New Away System\t(x)\tAway since date in default format; ?nas_awaysince_date(x) in format x"), TRF_FIELD | TRF_FUNCTION,
+ _T(VAR_STATDESC), LPGEN("New Away System\tStatus description"), TRF_FIELD | TRF_FUNCTION,
+ _T(VAR_MYNICK), LPGEN("New Away System\tYour nick for current protocol"), TRF_FIELD | TRF_FUNCTION,
+ _T(VAR_REQUESTCOUNT), LPGEN("New Away System\tNumber of status message requests from the contact"), TRF_FIELD | TRF_FUNCTION,
+ _T(VAR_MESSAGENUM), LPGEN("New Away System\tNumber of messages from the contact"), TRF_FIELD | TRF_FUNCTION,
+ _T(VAR_TIMEPASSED), LPGEN("New Away System\tTime passed until request"), TRF_FIELD | TRF_FUNCTION,
+ _T(VAR_PREDEFINEDMESSAGE), LPGEN("New Away System\t(x)\tReturns one of your predefined messages by its title: ?nas_predefinedmessage(creepy)"), TRF_FUNCTION,
+ _T(VAR_PROTOCOL), LPGEN("New Away System\tCurrent protocol name"), TRF_FIELD | TRF_FUNCTION
+ };
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_FREEVARMEM, srvFreeVarMem));
+ hServices.insert(CreateServiceFunction(MS_AWAYSYS_VARIABLESHANDLER, srvVariablesHandler));
+ TOKENREGISTER tr = {0};
+ tr.cbSize = sizeof(TOKENREGISTER);
+ tr.szService = MS_AWAYSYS_VARIABLESHANDLER;
+ tr.szCleanupService = MS_AWAYSYS_FREEVARMEM;
+ tr.memType = TR_MEM_OWNER;
+ int i;
+ for (i = 0; i < lengthof(Variables); i++)
+ {
+ tr.flags = Variables[i].Flags | TRF_CALLSVC | TRF_TCHAR;
+ tr.tszTokenString = Variables[i].Name;
+ tr.szHelpText = Variables[i].Descr;
+ CallService(MS_VARS_REGISTERTOKEN, 0, (LPARAM)&tr);
+ }
+ }
+
+ //NightFox
+ HookEvent(ME_MODERNOPT_INITIALIZE, ModernOptInitialise);
+
+
+ return 0;
+}
+
+
+extern "C" int __declspec(dllexport) Load(void)
+{
+ mir_getLP( &pluginInfo );
+
+ hHooks.insert(HookEvent(ME_SYSTEM_MODULESLOADED, MirandaLoaded));
+ if (db_get_s(NULL, "KnownModules", "New Away System", (char*)NULL) == NULL)
+ db_set_s(NULL, "KnownModules", "New Away System", MOD_NAME);
+
+ InitCommonControls();
+ InitOptions(); // must be called before we hook CallService
+
+ logservice_register(LOG_ID, LPGENT("New Away System"), _T("NewAwaySys?puts(p,?dbsetting(%subject%,Protocol,p))?if2(_?dbsetting(,?get(p),?pinfo(?get(p),uidsetting)),).log"), TranslateTS(_T("`[`!cdate()-!ctime()`]` ?cinfo(%subject%,display) (?cinfo(%subject%,id)) read your %") _T(VAR_STATDESC) _T("% message:\r\n%extratext%\r\n\r\n")));
+
+ if (db_get_b(NULL, MOD_NAME, DB_SETTINGSVER, 0) < 1)
+ { // change all %nas_message% variables to %extratext% if it wasn't done before
+ TCString Str;
+ Str = db_get_s(NULL, MOD_NAME, "PopupsFormat", _T(""));
+ if (Str.GetLen())
+ {
+ db_set_ts(NULL, MOD_NAME, "PopupsFormat", Str.Replace(_T("nas_message"), _T("extratext")));
+ }
+ Str = db_get_s(NULL, MOD_NAME, "ReplyPrefix", _T(""));
+ if (Str.GetLen())
+ {
+ db_set_ts(NULL, MOD_NAME, "ReplyPrefix", Str.Replace(_T("nas_message"), _T("extratext")));
+ }
+ }
+ if (db_get_b(NULL, MOD_NAME, DB_SETTINGSVER, 0) < 2)
+ { // disable autoreply for not-on-list contacts, as such contact may be a spam bot
+ db_set_b(NULL, MOD_NAME, ContactStatusToDBSetting(0, DB_ENABLEREPLY, 0, INVALID_CONTACT_ID), 0);
+ db_set_b(NULL, MOD_NAME, DB_SETTINGSVER, 2);
+ }
+ return 0;
+}
+
+
+extern "C" int __declspec(dllexport) Unload()
+{
+ CloseHandle(hMainThread);
+ int i;
+ for (i = 0; i < hHooks.getCount(); i++)
+ UnhookEvent(hHooks[i]);
+
+ for (i = 0; i < hServices.getCount(); i++)
+ DestroyServiceFunction(hServices[i]);
+
+ return 0;
+}
diff --git a/plugins/NewAwaySysMod/src/CString.cpp b/plugins/NewAwaySysMod/src/CString.cpp
new file mode 100644
index 0000000000..da36abd4d6
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/CString.cpp
@@ -0,0 +1,308 @@
+/*
+ TCString.cpp - TCString class
+ Copyright (c) 2005-2008 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+
+#define STR_GROWBY 64
+
+#define min(a,b) (((a) < (b)) ? (a) : (b))
+
+
+template <class T>
+void TString<T>::Empty()
+{
+ nBufSize = 1;
+ SetAllocSize(STR_GROWBY);
+ pBuf[0] = 0;
+}
+
+
+template <class T>
+void TString<T>::Free()
+{
+// HeapFree(GetProcessHeap(), 0, pBuf);
+ free(pBuf);
+ pBuf = NULL;
+ nBufSize = 0;
+ nAllocSize = 0;
+}
+
+
+template <class T>
+T *TString<T>::GetBuffer(int nNewLen)
+{
+ if (nNewLen != -1)
+ {
+ SetBufSize(nNewLen + 1);
+ }
+ _ASSERT(pBuf);
+ return pBuf;
+}
+
+
+template <class T>
+void TString<T>::ReleaseBuffer(int nNewLen)
+{
+ if (nNewLen == -1)
+ {
+ nBufSize = My_lstrlen(pBuf) + 1;
+ } else
+ {
+ nBufSize = nNewLen + 1;
+ pBuf[nNewLen] = 0;
+ _ASSERT(My_lstrlen(pBuf) == nNewLen);
+ }
+ _ASSERT(nBufSize <= nAllocSize); // prevent buffer overruns
+}
+
+
+template <class T>
+void TString<T>::SetAllocSize(int nNewAllocSize)
+{
+ _ASSERT(nNewAllocSize > 0);
+ T *pNewBuf = /*(char *)HeapAlloc(GetProcessHeap(), 0, sizeof(char) * nNewAllocSize);*/
+(T *)malloc(sizeof(T) * nNewAllocSize);
+ if (pBuf)
+ {
+ memcpy(pNewBuf, pBuf, sizeof(T) * min(nBufSize, nNewAllocSize));
+ //HeapFree(GetProcessHeap(), 0, pBuf);
+ free(pBuf);
+ }
+ pBuf = pNewBuf;
+ nAllocSize = nNewAllocSize;
+}
+
+
+template <class T>
+void TString<T>::SetBufSize(int nNewBufSize)
+{
+ _ASSERT(nNewBufSize >= 0);
+ if (nNewBufSize < nBufSize)
+ {
+ _ASSERT(pBuf);
+ pBuf[nNewBufSize - 1] = 0;
+ }
+ if ((unsigned)(nAllocSize - nNewBufSize) / STR_GROWBY)
+ {
+ SetAllocSize((nNewBufSize + STR_GROWBY - 1) - (nNewBufSize + STR_GROWBY - 1) % STR_GROWBY);
+ }
+ nBufSize = nNewBufSize;
+}
+
+
+template <class T>
+TString<T>& TString<T>::Cat(const T *pStr)
+{
+ _ASSERT(pBuf && pStr);
+ int StrLen = My_lstrlen(pStr);
+ SetAllocSize(nBufSize + StrLen);
+ My_lstrcpy(GetBuffer() + GetLen(), pStr);
+ ReleaseBuffer(nBufSize + StrLen - 1);
+ return *this;
+}
+
+
+template <class T>
+TString<T>& TString<T>::Cat(const T c)
+{
+ _ASSERT(pBuf);
+ SetAllocSize(nBufSize + 1);
+ int CurLen = GetLen();
+ T *p = GetBuffer();
+ p[CurLen] = c;
+ p[CurLen + 1] = '\0';
+ ReleaseBuffer(nBufSize);
+ return *this;
+}
+
+
+template <class T>
+TString<T>& TString<T>::DiffCat(const T *pStart, const T *pEnd)
+{
+ _ASSERT(pBuf && pStart && pEnd);
+ int StrLen = pEnd - pStart;
+ SetAllocSize(nBufSize + StrLen);
+ My_strncpy(GetBuffer() + GetLen(), pStart, StrLen);
+ ReleaseBuffer(nBufSize + StrLen - 1);
+ return *this;
+}
+
+
+template <class T>
+TString<T>& TString<T>::Replace(const T *szFind, const T *szReplaceBy)
+{
+ if (!pBuf)
+ {
+ return *this;
+ }
+ T *pCurPos = pBuf;
+ int FindLen = My_lstrlen(szFind);
+ T *p;
+ TString<T> Result;
+ Result.GetBuffer(1)[0] = '\0';
+ Result.ReleaseBuffer(0); // set the string to ""; we can't do it in a usual way (using a constructor or an assignment) because we don't know whether "" needs to be unicode or ansi
+ while (p = ( T* )My_strstr(pCurPos, szFind))
+ {
+ Result.DiffCat(pCurPos, p);
+ Result += szReplaceBy;
+ pCurPos = p + FindLen;
+ }
+ Result += pCurPos;
+ *this = Result;
+ return *this;
+}
+
+
+template <class T>
+TString<T>& TString<T>::Replace(int nIndex, int nCount, const T *szReplaceBy)
+{
+ if (!pBuf || !szReplaceBy || nIndex < 0 || nCount < 0)
+ {
+ return *this;
+ }
+
+ TString<T> Result;
+ Result.GetBuffer(1)[0] = '\0';
+ Result.ReleaseBuffer(0); // set the string to ""; we can't do it in a usual way (using a constructor or an assignment) because we don't know whether "" needs to be unicode or ansi
+ if (nIndex > GetLen())
+ {
+ nIndex = GetLen();
+ }
+ if (nIndex + nCount > GetLen())
+ {
+ nCount = GetLen() - nIndex;
+ }
+ Result.DiffCat(pBuf, pBuf + nIndex);
+ Result += szReplaceBy;
+ Result += pBuf + nIndex + nCount;
+ *this = Result;
+ return *this;
+}
+
+
+template <class T>
+TString<T> TString<T>::Left(int nCount) const
+{
+ _ASSERT(nCount >= 0);
+ TString<T> Result(*this);
+ Result.SetBufSize(nCount + 1);
+ return Result;
+}
+
+
+template <class T>
+TString<T> TString<T>::Right(int nCount) const
+{
+ _ASSERT(nCount >= 0);
+ if (nCount < GetLen())
+ {
+ return &pBuf[GetLen() - nCount];
+ } else
+ {
+ return *this;
+ }
+}
+
+
+template <class T>
+TString<T> TString<T>::SubStr(int nIndex, int nCount) const
+{
+ _ASSERT(nIndex >= 0 && nCount >= 0);
+ TString<T> Result;
+ if (nIndex < GetLen())
+ {
+ My_strncpy(Result.GetBuffer(nCount), &pBuf[nIndex], nCount);
+ Result.ReleaseBuffer();
+ } else
+ {
+ Result.GetBuffer(1)[0] = '\0';
+ Result.ReleaseBuffer(0);
+ }
+ return Result;
+}
+
+
+template <class T>
+TString<T> TString<T>::ToLower() const
+{
+ TString<T> Result(*this);
+ if (!pBuf)
+ {
+ return Result; // return NULL string
+ }
+ My_strlwr((T*)Result);
+ return Result;
+}
+
+
+template <class T>
+TString<T>& TString<T>::operator = (const T *pStr)
+{
+ if (pStr)
+ {
+ int StrLen = My_lstrlen(pStr);
+ SetBufSize(StrLen + 1);
+ My_lstrcpy(GetBuffer(), pStr);
+ ReleaseBuffer(StrLen);
+ } else
+ {
+ Free();
+ }
+ return *this;
+}
+
+template class TString<TCHAR>;
+template class TString<char>;
+template class TString<WCHAR>;
+
+CString db_get_s(MCONTACT hContact, const char *szModule, const char *szSetting, const char *szDefaultValue)
+{
+ ptrA p( db_get_sa(hContact, szModule, szSetting));
+ return CString(p == NULL ? szDefaultValue : p);
+}
+
+TCString db_get_s(MCONTACT hContact, const char *szModule, const char *szSetting, const TCHAR *szDefaultValue)
+{
+ ptrT p( db_get_tsa(hContact, szModule, szSetting));
+ return TCString(p == NULL ? szDefaultValue : p);
+}
+
+TCString DBGetContactSettingAsString(MCONTACT hContact, const char *szModule, const char *szSetting, const TCHAR *szDefaultValue)
+{ // also converts numeric values to a string
+ DBVARIANT dbv = {0};
+ int iRes = db_get_ws(hContact, szModule, szSetting, &dbv);
+
+ TCString Result;
+ if (!iRes && (dbv.type == DBVT_ASCIIZ || dbv.type == DBVT_WCHAR))
+ {
+ Result = dbv.ptszVal;
+ }
+ else if (dbv.type == DBVT_BYTE || dbv.type == DBVT_WORD || dbv.type == DBVT_DWORD)
+ {
+ long value = (dbv.type == DBVT_DWORD) ? dbv.dVal : (dbv.type == DBVT_WORD ? dbv.wVal : dbv.bVal);
+ _ultot(value, Result.GetBuffer(64), 10);
+ Result.ReleaseBuffer();
+ }
+ else Result = szDefaultValue;
+
+ if (!iRes)
+ db_free(&dbv);
+
+ return Result;
+}
diff --git a/plugins/NewAwaySysMod/src/CString.h b/plugins/NewAwaySysMod/src/CString.h
new file mode 100644
index 0000000000..0d81a4f7c3
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/CString.h
@@ -0,0 +1,237 @@
+/*
+ TCString.h - TCString class
+ Copyright (c) 2005-2008 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#pragma once
+
+#include <windows.h>
+#include <tchar.h>
+#include <crtdbg.h>
+#ifdef CHARARRAY_CONVERT
+#include "TMyArray.h"
+#endif
+#include "newpluginapi.h"
+#include "m_system.h"
+#include "m_database.h"
+
+__inline int My_lstrlen(LPCSTR lpString) {return lstrlenA(lpString);}
+__inline int My_lstrlen(LPCWSTR lpString) {return lstrlenW(lpString);}
+__inline int My_lstrcmp(LPCSTR lpString1, LPCSTR lpString2) {return lstrcmpA(lpString1, lpString2);}
+__inline int My_lstrcmp(LPCWSTR lpString1, LPCWSTR lpString2) {return lstrcmpW(lpString1, lpString2);}
+__inline LPCSTR My_strstr(LPCSTR lpString1, LPCSTR lpString2) {return strstr(lpString1, lpString2);}
+__inline LPWSTR My_strstr(LPCWSTR lpString1, LPCWSTR lpString2) {return (LPWSTR)wcsstr(lpString1, lpString2);}
+__inline LPSTR My_lstrcpy(LPSTR lpString1, LPCSTR lpString2) {return lstrcpyA(lpString1, lpString2);}
+__inline LPWSTR My_lstrcpy(LPWSTR lpString1, LPCWSTR lpString2) {return lstrcpyW(lpString1, lpString2);}
+__inline LPSTR My_strncpy(LPSTR lpString1, LPCSTR lpString2, int Len) {return strncpy(lpString1, lpString2, Len);}
+__inline LPWSTR My_strncpy(LPWSTR lpString1, LPCWSTR lpString2, int Len) {return wcsncpy(lpString1, lpString2, Len);}
+__inline LPSTR My_strlwr(LPSTR lpString) {return _strlwr(lpString);}
+__inline LPWSTR My_strlwr(LPWSTR lpString) {return _wcslwr(lpString);}
+
+template <class T>
+class TString
+{
+public:
+ TString(): pBuf(NULL), nBufSize(0), nAllocSize(0) {}
+ TString(const T *pStr): pBuf(NULL), nBufSize(0), nAllocSize(0) {*this = pStr;}
+ TString(const TString<T> &Str): pBuf(NULL), nBufSize(0), nAllocSize(0) {*this = Str.pBuf;}
+ ~TString() {Free();}
+
+ int GetLen() const {return (nBufSize) ? (nBufSize - 1) : 0;};
+ int IsEmpty() const {return (!GetLen());};
+ T *GetBuffer(int nNewLen = -1);
+ void ReleaseBuffer(int nNewLen = -1);
+ TString<T>& Cat(const T *pStr);
+ TString<T>& Cat(const T c);
+ TString<T>& DiffCat(const T *pStart, const T *pEnd);
+ TString<T>& Replace(const T *szFind, const T *szReplaceBy);
+ TString<T>& Replace(int nIndex, int nCount, const T *szReplaceBy);
+ TString<T> Left(int nCount) const;
+ TString<T> Right(int nCount) const;
+ TString<T> SubStr(int nIndex, int nCount) const;
+ TString<T> ToLower() const;
+ void Empty();
+ void Free();
+ T& operator [] (int nIndex) {_ASSERT(nIndex >= 0 && nIndex <= GetLen()); return pBuf[nIndex];}
+ operator const T*() const {return pBuf;}
+ operator T*() {return pBuf;}
+ TString<T>& operator = (const T *pStr);
+ TString<T>& operator = (const TString<T> &Str) {return *this = Str.pBuf;}
+// TCString& operator + (const char *pStr)
+// {_ASSERT(pBuf && pStr); TCString Result(*this); return Result.Cat(pStr);}
+ friend TString<T> operator + (const TString<T> &Str1, const T *Str2)
+ {_ASSERT(Str1.pBuf && Str2); TString<T> Result(Str1); return Result.Cat(Str2);}
+/* friend TCString operator + (const char *Str1, const TCString &Str2)
+ {_ASSERT(Str1 && Str2.pBuf); TCString Result(Str1); return Result.Cat(Str2);}*/
+ TString<T>& operator += (const T *pStr) {_ASSERT(pBuf && pStr); return this->Cat(pStr);}
+ TString<T>& operator += (const T c) {_ASSERT(pBuf); return this->Cat(c);}
+ int operator == (const T *pStr) const {return (!pBuf || !pStr) ? (pBuf == pStr) : !My_lstrcmp(pBuf, pStr);}
+ int operator != (const T *pStr) const {return (!pBuf || !pStr) ? (pBuf != pStr) : My_lstrcmp(pBuf, pStr);}
+ int operator < (const T *pStr) const {_ASSERT(pBuf && pStr); return My_lstrcmp(pBuf, pStr) > 0;}
+ int operator > (const T *pStr) const {_ASSERT(pBuf && pStr); return My_lstrcmp(pBuf, pStr) < 0;}
+ int operator <= (const T *pStr) const {_ASSERT(pBuf && pStr); return My_lstrcmp(pBuf, pStr) >= 0;}
+ int operator >= (const T *pStr) const {_ASSERT(pBuf && pStr); return My_lstrcmp(pBuf, pStr) <= 0;}
+// TCString& Format(char *pszFormat, ...);
+
+private:
+ void SetBufSize(int nNewBufSize);
+ void SetAllocSize(int nNewAllocSize);
+
+ T *pBuf;
+ int nBufSize; // current string length + 1 (including 0 at the end)
+ int nAllocSize; // allocated memory size
+};
+
+
+typedef TString<TCHAR> TCString;
+typedef TString<char> CString;
+typedef TString<WCHAR> WCString;
+
+#ifdef CHARARRAY_CONVERT
+
+__inline CHARARRAY WCHAR2ANSI_ARRAY(CHARARRAY &c)
+{
+ CHARARRAY Result;
+ int Len = WideCharToMultiByte(CP_ACP, 0, (WCHAR*)c.GetData(), c.GetSize() / sizeof(WCHAR), NULL, 0, NULL, NULL);
+ if (Len)
+ {
+ Result.SetAtGrow(Len - 1);
+ if (!WideCharToMultiByte(CP_ACP, 0, (WCHAR*)c.GetData(), c.GetSize() / sizeof(WCHAR), Result.GetData(), Len, NULL, NULL))
+ {
+ Result.RemoveAll();
+ }
+ if (Result.GetSize())
+ {
+ Result.RemoveElem(Result.GetSize() - 1); // remove the null terminator
+ }
+ }
+ return Result;
+}
+
+__inline CHARARRAY ANSI2WCHAR_ARRAY(CHARARRAY &c)
+{
+ CHARARRAY Result;
+ int Len = MultiByteToWideChar(CP_ACP, 0, c.GetData(), c.GetSize(), NULL, 0);
+ if (Len)
+ {
+ Result.SetAtGrow(Len * sizeof(WCHAR) - 1);
+ if (!MultiByteToWideChar(CP_ACP, 0, c.GetData(), c.GetSize(), (WCHAR*)Result.GetData(), Len))
+ {
+ Result.RemoveAll();
+ }
+ if (Result.GetSize())
+ {
+ Result.RemoveElem(Result.GetSize() - 1);
+ Result.RemoveElem(Result.GetSize() - 1); // remove the null terminator
+ }
+ }
+ return Result;
+}
+
+
+__inline CHARARRAY WCHAR2UTF8(WCString Str)
+{
+ CHARARRAY Result;
+ int Len = WideCharToMultiByte(CP_UTF8, 0, Str, -1, NULL, 0, NULL, NULL);
+ if (Len)
+ {
+ Result.SetAtGrow(Len - 1);
+ if (!WideCharToMultiByte(CP_UTF8, 0, Str, -1, Result.GetData(), Len, NULL, NULL))
+ {
+ Result.RemoveAll();
+ }
+ }
+ return Result;
+}
+
+
+#endif // CHARARRAY_CONVERT
+
+
+#undef db_get_s
+CString db_get_s(MCONTACT hContact, const char *szModule, const char *szSetting, const char *szDefaultValue);
+TCString db_get_s(MCONTACT hContact, const char *szModule, const char *szSetting, const TCHAR *szDefaultValue);
+int db_get_s(MCONTACT hContact, const char *szModule, const char *szSetting, DBVARIANT *dbv);
+TCString DBGetContactSettingAsString(MCONTACT hContact, const char *szModule, const char *szSetting, const TCHAR *szDefaultValue); // also converts numeric values to a string
+
+// various string helpers. their return values are valid only while the class is visible
+class UTF8Encode
+{
+public:
+ UTF8Encode(const char *str) { p = mir_utf8encode(str); }
+ UTF8Encode(const wchar_t *str) { p = mir_utf8encodeW(str); }
+ ~UTF8Encode() { mir_free(p); }
+ operator char*() { return p; }
+
+private:
+ char *p;
+};
+
+class UTF8DecodeA
+{
+public:
+ UTF8DecodeA(const char *str) { p = mir_strdup(str); mir_utf8decode(p, NULL); }
+ ~UTF8DecodeA() { mir_free(p); }
+ operator char*() { return p; }
+
+private:
+ char *p;
+};
+
+class UTF8DecodeW
+{
+public:
+ UTF8DecodeW(const char *str) { p = mir_utf8decodeW(str); }
+ ~UTF8DecodeW() { mir_free(p); }
+ operator wchar_t*() { return p; }
+
+private:
+ wchar_t *p;
+};
+
+
+#define UTF8Decode UTF8DecodeW
+
+
+/*class mallocStrA
+{
+public:
+ mallocStrA(int n) { p = (char*)malloc((n + 1) * sizeof(char)); }
+ mallocStrA(const char *str) { p = str ? strdup(str) : NULL; }
+ ~mallocStrA() { if (p) free(p); }
+ operator char*() { return p; }
+
+private:
+ char *p;
+};
+
+class mallocStrW
+{
+public:
+ mallocStrW(int n) { p = (wchar_t*)malloc((n + 1) * sizeof(wchar_t)); }
+ mallocStrW(const wchar_t *str) { p = str ? _wcsdup(str) : NULL; }
+ ~mallocStrW() { if (p) free(p); }
+ operator wchar_t*() { return p; }
+
+private:
+ wchar_t *p;
+};
+
+
+#define mallocStr mallocStrW
+
+*/ \ No newline at end of file
diff --git a/plugins/NewAwaySysMod/src/Client.cpp b/plugins/NewAwaySysMod/src/Client.cpp
new file mode 100644
index 0000000000..9d8c520f7c
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Client.cpp
@@ -0,0 +1,232 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+ Copyright (c) 2004-2005 Iksaif Entertainment
+ Copyright (c) 2002-2003 Goblineye Entertainment
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "MsgTree.h"
+#include "Properties.h"
+
+#define PARSE_INTERVAL 10000
+
+HANDLE g_hTerminateUpdateMsgsThread = NULL;
+HANDLE g_hUpdateMsgsThread = NULL;
+
+void __cdecl UpdateMsgsThreadProc(void *)
+{
+ int ProtoCount;
+ PROTOCOLDESCRIPTOR **proto;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&ProtoCount, (LPARAM)&proto);
+ int I;
+ while (WaitForSingleObject(g_hTerminateUpdateMsgsThread, 0) == WAIT_TIMEOUT && !Miranda_Terminated())
+ {
+ DWORD MinUpdateTimeDifference = g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_UPDATEMSGSPERIOD) * 1000; // in milliseconds
+ for (I = 0; I < ProtoCount; I++)
+ {
+ if (proto[I]->type == PROTOTYPE_PROTOCOL && CallProtoService(proto[I]->szName, PS_GETCAPS, PFLAGNUM_1, 0) & PF1_MODEMSGSEND && !IsAnICQProto(proto[I]->szName))
+ {
+ int Status = CallProtoService(proto[I]->szName, PS_GETSTATUS, 0, 0);
+ if (Status < ID_STATUS_OFFLINE || Status > ID_STATUS_OUTTOLUNCH)
+ {
+ Status = g_ProtoStates[proto[I]->szName].Status;
+ }
+ if (CallProtoService(proto[I]->szName, PS_GETCAPS, PFLAGNUM_3, 0) & Proto_Status2Flag(Status) && g_ProtoStates[proto[I]->szName].CurStatusMsg.GetUpdateTimeDifference() >= MinUpdateTimeDifference)
+ {
+ TCString CurMsg(GetDynamicStatMsg(INVALID_CONTACT_ID, proto[I]->szName));
+ if ((TCString)g_ProtoStates[proto[I]->szName].CurStatusMsg != (const TCHAR*)CurMsg) // if the message has changed
+ {
+ g_ProtoStates[proto[I]->szName].CurStatusMsg = CurMsg;
+ CallAllowedPS_SETAWAYMSG(proto[I]->szName, Status, (char*)_T2A(CurMsg));
+ }
+ }
+ }
+ }
+ SleepEx(PARSE_INTERVAL, true);
+ }
+}
+
+
+static void __stdcall DummyAPCFunc(ULONG_PTR)
+{
+ return;
+}
+
+
+void InitUpdateMsgs()
+{
+ int UpdateMsgs = g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_UPDATEMSGS);
+ if (g_hUpdateMsgsThread && !UpdateMsgs)
+ {
+ _ASSERT(WaitForSingleObject(g_hUpdateMsgsThread, 0) == WAIT_TIMEOUT);
+ SetEvent(g_hTerminateUpdateMsgsThread);
+ QueueUserAPC(DummyAPCFunc, g_hUpdateMsgsThread, 0); // wake up the thread, as it's most probably in SleepEx() now
+ WaitForSingleObject(g_hUpdateMsgsThread, INFINITE);
+ g_hUpdateMsgsThread = NULL;
+ CloseHandle(g_hTerminateUpdateMsgsThread);
+ } else if (!g_hUpdateMsgsThread && UpdateMsgs)
+ {
+ g_hTerminateUpdateMsgsThread = CreateEvent(NULL, TRUE, FALSE, NULL);
+ g_hUpdateMsgsThread = (HANDLE)mir_forkthread(UpdateMsgsThreadProc, NULL);
+ }
+}
+
+
+void ChangeProtoMessages(char* szProto, int iMode, TCString &Msg)
+{
+ TCString CurMsg(Msg);
+ if (szProto)
+ {
+ if (Msg == NULL)
+ {
+ CurMsg = GetDynamicStatMsg(INVALID_CONTACT_ID, szProto);
+ }
+ CallAllowedPS_SETAWAYMSG(szProto, iMode, (char*)_T2A(CurMsg));
+ g_ProtoStates[szProto].CurStatusMsg = CurMsg;
+ } else // change message of all protocols
+ {
+ int ProtoCount;
+ PROTOCOLDESCRIPTOR **proto;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&ProtoCount, (LPARAM)&proto);
+ int I;
+ for (I = 0; I < ProtoCount; I++)
+ {
+ if (proto[I]->type == PROTOTYPE_PROTOCOL && !db_get_b(NULL, proto[I]->szName, "LockMainStatus", 0))
+ {
+ if (Msg == NULL)
+ {
+ CurMsg = GetDynamicStatMsg(INVALID_CONTACT_ID, proto[I]->szName);
+ }
+ CallAllowedPS_SETAWAYMSG(proto[I]->szName, iMode, (char*)_T2A(CurMsg));
+ g_ProtoStates[proto[I]->szName].CurStatusMsg = CurMsg;
+ }
+ }
+ }
+ static struct
+ {
+ int Status;
+ char *Setting;
+ } StatusSettings[] = {
+ ID_STATUS_OFFLINE, "Off",
+ ID_STATUS_ONLINE, "On",
+ ID_STATUS_AWAY, "Away",
+ ID_STATUS_NA, "Na",
+ ID_STATUS_DND, "Dnd",
+ ID_STATUS_OCCUPIED, "Occupied",
+ ID_STATUS_FREECHAT, "FreeChat",
+ ID_STATUS_INVISIBLE, "Inv",
+ ID_STATUS_ONTHEPHONE, "Otp",
+ ID_STATUS_OUTTOLUNCH, "Otl",
+ ID_STATUS_IDLE, "Idl"
+ };
+ int I;
+ for (I = 0; I < lengthof(StatusSettings); I++)
+ {
+ if (iMode == StatusSettings[I].Status)
+ {
+ db_set_ts(NULL, "SRAway", CString(StatusSettings[I].Setting) + "Msg", CurMsg);
+ db_set_ts(NULL, "SRAway", CString(StatusSettings[I].Setting) + "Default", CurMsg); // TODO: make it more accurate, and change not only here, but when changing status messages through UpdateMsgsTimerFunc too; and when changing messages through AutoAway() ?
+ break;
+ }
+ }
+// InitUpdateMsgs();
+}
+
+
+int GetRecentGroupID(int iMode)
+{ // returns an ID of a group where recent messages are stored, accordingly to current settings and status mode.
+ // -1 if the group is not found
+// COptPage MoreOptData(g_MoreOptPage);
+ COptPage MsgTreeData(g_MsgTreePage);
+ COptItem_TreeCtrl *TreeCtrl = (COptItem_TreeCtrl*)MsgTreeData.Find(IDV_MSGTREE);
+ TreeCtrl->DBToMem(CString(MOD_NAME));
+ int Order;
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_PERSTATUSMRM))
+ {
+ for (Order = 0; Order < TreeCtrl->Value.GetSize(); Order++) // find a group named accordingly to the current status
+ {
+ if (TreeCtrl->Value[Order].ParentID == g_Messages_RecentRootID && TreeCtrl->Value[Order].Flags & TIF_GROUP && !_tcsicmp(TreeCtrl->Value[Order].Title, iMode ? (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, iMode, GSMDF_TCHAR) : MSGTREE_RECENT_OTHERGROUP))
+ {
+ return TreeCtrl->Value[Order].ID;
+ }
+ }
+ } else // simply use Recent Messages category
+ {
+ return g_Messages_RecentRootID;
+ }
+ return -1;
+}
+
+
+int ICQStatusToGeneralStatus(int bICQStat)
+{
+ switch (bICQStat)
+ {
+ case MTYPE_AUTOONLINE: return ID_STATUS_ONLINE;
+ case MTYPE_AUTOAWAY: return ID_STATUS_AWAY;
+ case MTYPE_AUTONA: return ID_STATUS_NA;
+ case MTYPE_AUTODND: return ID_STATUS_DND;
+ case MTYPE_AUTOBUSY: return ID_STATUS_OCCUPIED;
+ case MTYPE_AUTOFFC: return ID_STATUS_FREECHAT;
+ default: return 0;
+ }
+}
+
+
+TCString VariablesEscape(TCString Str)
+{
+ if (!Str.GetLen())
+ {
+ return _T("");
+ }
+ enum eState
+ {
+ ST_TEXT, ST_QUOTE
+ };
+ eState State = ST_QUOTE;
+ TCString Result(_T("`"));
+ const TCHAR *p = Str;
+ while (*p)
+ {
+ if (*p == '`')
+ {
+ if (State == ST_TEXT)
+ {
+ Result += _T("````");
+ State = ST_QUOTE;
+ } else
+ {
+ Result += _T("``");
+ }
+ } else
+ {
+ Result += *p;
+ State = ST_TEXT;
+ }
+ p++;
+ }
+ if (State == ST_QUOTE)
+ {
+ Result.GetBuffer()[Result.GetLen() - 1] = '\0';
+ Result.ReleaseBuffer();
+ } else
+ {
+ Result += '`';
+ }
+ return Result;
+}
diff --git a/plugins/NewAwaySysMod/src/Common.h b/plugins/NewAwaySysMod/src/Common.h
new file mode 100644
index 0000000000..8449efea0a
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Common.h
@@ -0,0 +1,399 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+ Copyright (c) 2004-2005 Iksaif Entertainment
+ Copyright (c) 2002-2003 Goblineye Entertainment
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#pragma once
+
+#define _CRT_SECURE_NO_WARNINGS
+#define _WIN32_WINNT 0x0500
+
+#define MIRANDA_VER 0x0A00
+
+#include <windows.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <CommCtrl.h>
+#include <commdlg.h>
+#include <Uxtheme.h>
+#include <vsstyle.h>
+
+#include <time.h>
+#include <shellapi.h>
+#include <crtdbg.h>
+#include <tchar.h>
+#include <stdarg.h>
+
+#include "newpluginapi.h"
+#include "m_clist.h"
+#include "m_system.h"
+#include "m_database.h"
+#include "m_clui.h"
+#include "m_langpack.h"
+#include "m_protosvc.h"
+#include "m_options.h"
+#include "..\..\protocols\IcqOscarJ\src\icq_constants.h"
+#include "m_skin.h"
+#include "m_awaymsg.h"
+#include "m_utils.h"
+#include "m_system_cpp.h"
+#include "m_history.h"
+#include "m_message.h"
+#include "m_userinfo.h"
+#include "m_icq.h"
+#include "win2k.h"
+
+#include "resource.h"
+
+#include "m_variables.h"
+//#include "m_toptoolbar.h"
+#include "m_popup.h"
+//#include "m_popupw.h"
+#include "m_metacontacts.h"
+#include "m_LogService.h"
+
+#pragma comment(lib,"comctl32.lib")
+
+#include "CString.h"
+#include "Options.h"
+
+#define CBSCHECK_UNCHECKED 1
+#define CBSCHECK_CHECKED 5
+#define CBSCHECK_MIXED 9
+#define CBSSTATE_NORMAL 0
+#define CBSSTATE_HOT 1
+#define CBSSTATE_PRESSED 2
+#define CBSSTATE_DISABLED 3
+
+#define VAR_AWAYSINCE_TIME "nas_awaysince_time"
+#define VAR_AWAYSINCE_DATE "nas_awaysince_date"
+#define VAR_STATDESC "nas_statdesc"
+#define VAR_MYNICK "nas_mynick"
+#define VAR_REQUESTCOUNT "nas_requestcount"
+#define VAR_MESSAGENUM "nas_messagecount"
+#define VAR_TIMEPASSED "nas_timepassed"
+#define VAR_PREDEFINEDMESSAGE "nas_predefinedmessage"
+#define VAR_PROTOCOL "nas_protocol"
+
+#define SENDSMSG_EVENT_MSG 0x1
+#define SENDSMSG_EVENT_URL 0x2
+#define SENDSMSG_EVENT_FILE 0x4
+
+#define AWAY_MSGDATA_MAX 8000
+
+// Flags for status database settings
+#define SF_OFF 0x1
+#define SF_ONL 0x2
+#define SF_AWAY 0x4
+#define SF_NA 0x8
+#define SF_OCC 0x10
+#define SF_DND 0x20
+#define SF_FFC 0x40
+#define SF_INV 0x80
+#define SF_OTP 0x100
+#define SF_OTL 0x200
+#define SF_OTHER 0x80000000
+/*
+// Actions on popup click
+#define PCA_OPENMESSAGEWND 0 // open message window
+#define PCA_CLOSEPOPUP 1 // close popup
+#define PCA_OPENDETAILS 2 // open contact details window
+#define PCA_OPENMENU 3 // open contact menu
+#define PCA_OPENHISTORY 4 // open contact history
+#define PCA_OPENLOG 5 // open log file
+#define PCA_DONOTHING 6 // do nothing
+
+// Notification options defaults
+#define POPUP_DEF_POPUP_FORMAT TranslateT("?cinfo(%subject%,display) (?cinfo(%subject%,id)) is reading your %nas_statdesc% message:\r\n%extratext%")
+#define POPUP_DEF_USEPOPUPS 0
+#define POPUP_DEF_LCLICKACTION PCA_OPENMESSAGEWND
+#define POPUP_DEF_RCLICKACTION PCA_CLOSEPOPUP
+#define POPUP_DEF_POPUP_BGCOLOUR 0xFFB5BC
+#define POPUP_DEF_POPUP_TEXTCOLOUR 0
+#define POPUP_DEF_USEDEFBGCOLOUR 0
+#define POPUP_DEF_USEDEFTEXTCOLOUR 0
+#define POPUP_DEF_POPUPNOTIFYFLAGS (SF_ONL | SF_AWAY | SF_NA | SF_OCC | SF_DND | SF_FFC | SF_INV | SF_OTP | SF_OTL)
+#define POPUP_DEF_POPUPDELAY 0
+
+#define POPUP_MAXPOPUPDELAY 9999
+*/
+#define MOREOPTDLG_DEF_DONTPOPDLG (SF_ONL | SF_INV)
+#define MOREOPTDLG_DEF_USEBYDEFAULT 0
+
+// Event flags (used for "reply on event" options)
+#define EF_MSG 1
+#define EF_URL 2
+#define EF_FILE 4
+
+#define AUTOREPLY_DEF_REPLY 0
+#define AUTOREPLY_DEF_REPLYONEVENT (EF_MSG | EF_URL | EF_FILE)
+#define AUTOREPLY_DEF_PREFIX TranslateT("Miranda IM autoreply >\r\n%extratext%")
+#define AUTOREPLY_DEF_DISABLEREPLY (SF_ONL | SF_INV)
+
+#define AUTOREPLY_IDLE_WINDOWS 0
+#define AUTOREPLY_IDLE_MIRANDA 1
+#define AUTOREPLY_DEF_IDLEREPLYVALUE AUTOREPLY_IDLE_WINDOWS
+
+#define AUTOREPLY_MAXPREFIXLEN 8000
+
+#define VAL_USEDEFAULT 2 // undefined value for ignore/autoreply/notification settings in the db; must be 2 for proper ContactSettings support
+
+// Set Away Message dialog flags
+#define DF_SAM_SHOWMSGTREE 1
+#define DF_SAM_SHOWCONTACTTREE 2
+#define DF_SAM_DEFDLGFLAGS DF_SAM_SHOWMSGTREE
+
+// WriteAwayMsgInDB option flags:
+#define WRITE_LMSG 1
+#define WRITE_RMSG 2
+#define WRITE_INTERPRET 4
+#define WRITE_CMSG 8
+
+#define TOGGLE_SOE_COMMAND LPGENT("Toggle autoreply on/off")
+#define DISABLE_SOE_COMMAND LPGENT("Toggle autoreply off")
+#define ENABLE_SOE_COMMAND LPGENT("Toggle autoreply on")
+
+#define STR_XSTATUSDESC TranslateT("extended status")
+
+#define MOD_NAME "NewAwaySys"
+#define LOG_ID MOD_NAME // LogService log ID
+#define LOG_PREFIX MOD_NAME ": " // netlib.log prefix for all NAS' messages
+
+#define DB_SETTINGSVER "SettingsVer"
+
+#ifndef lengthof
+#define lengthof(s) (sizeof(s) / sizeof(*s))
+#endif
+
+#define MS_NETLIB_LOG "Netlib/Log"
+
+#define UM_ICONSCHANGED (WM_USER + 121)
+
+// IDD_READAWAYMSG user-defined message constants
+#define UM_RAM_AWAYMSGACK (WM_USER + 10)
+
+// IDD_SETAWAYMSG user-defined message constants
+#define UM_SAM_SPLITTERMOVED (WM_USER + 1)
+#define UM_SAM_SAVEDLGSETTINGS (WM_USER + 2)
+#define UM_SAM_APPLYANDCLOSE (WM_USER + 3)
+#define UM_SAM_KILLTIMER (WM_USER + 4)
+#define UM_SAM_REPLYSETTINGCHANGED (WM_USER + 5)
+#define UM_SAM_PROTOSTATUSCHANGED (WM_USER + 6) // wParam = (char*)szProto
+
+#define UM_CLICK (WM_USER + 100)
+
+#define SAM_DB_DLGPOSX "SAMDlgPosX"
+#define SAM_DB_DLGPOSY "SAMDlgPosY"
+#define SAM_DB_DLGSIZEX "SAMDlgSizeX"
+#define SAM_DB_DLGSIZEY "SAMDlgSizeY"
+#define SAM_DB_MSGSPLITTERPOS "SAMMsgSplitterPos"
+#define SAM_DB_CONTACTSPLITTERPOS "SAMContactSplitterPos"
+
+#define DB_MESSAGECOUNT "MessageCount"
+#define DB_REQUESTCOUNT "RequestCount"
+#define DB_SENDCOUNT "SendCount"
+#define MESSAGES_DB_MSGTREEDEF "MsgTreeDef"
+
+#define MSGTREE_RECENT_OTHERGROUP _T("Other")
+
+// GetMsgFormat flags
+#define GMF_PERSONAL 1 // is also used to get global status message, when hContact = NULL (szProto = NULL)
+#define GMF_PROTOORGLOBAL 2
+#define GMF_LASTORDEFAULT 4 // this flag doesn't require hContact or szProto
+#define GMF_TEMPORARY 8 // doesn't require status
+#define GMF_ANYCURRENT (GMF_TEMPORARY | GMF_PERSONAL | GMF_PROTOORGLOBAL)
+
+// SetMsgFormat flags
+#define SMF_PERSONAL 1 // is also used to set global status message, when hContact = NULL (szProto = NULL)
+#define SMF_LAST 2
+#define SMF_TEMPORARY 4 // doesn't require status
+
+// VAR_PARSE_DATA flags
+#define VPF_XSTATUS 1 // use "extended status" instead of the usual status description in %nas_statdesc%, and XStatus message in %nas_message%
+
+// options dialog
+#define OPT_TITLE LPGENT("Away System")
+#define OPT_MAINGROUP LPGEN("Status")
+#define OPT_POPUPGROUP LPGEN("Popups")
+
+#define MRM_MAX_GENERATED_TITLE_LEN 35 // maximum length of automatically generated title for recent messages
+
+int ICQStatusToGeneralStatus(int bICQStat); // TODO: get rid of these protocol-specific functions, if possible
+
+#define MS_AWAYSYS_SETCONTACTSTATMSG "AwaySys/SetContactStatMsg"
+
+#define MS_AWAYSYS_AUTOREPLY_TOGGLE "AwaySys/AutoreplyToggle"
+#define MS_AWAYSYS_AUTOREPLY_ON "AwaySys/AutoreplyOn"
+#define MS_AWAYSYS_AUTOREPLY_OFF "AwaySys/AutoreplyOff"
+#define MS_AWAYSYS_AUTOREPLY_USEDEFAULT "AwaySys/AutoreplyUseDefault"
+
+#define MS_AWAYSYS_VARIABLESHANDLER "AwaySys/VariablesHandler"
+#define MS_AWAYSYS_FREEVARMEM "AwaySys/FreeVarMem"
+// these are obsolete AwaySysMod services, though they're still here for compatibility with old plugins
+#define MS_AWAYSYS_SETSTATUSMODE "AwaySys/SetStatusMode" // change the status mode. wParam is new mode, lParam is new status message (AwaySys will interpret variables out of it), may be NULL.
+#define MS_AWAYSYS_IGNORENEXT "AwaySys/IgnoreNextStatusChange" //ignore nest status change
+
+typedef struct SetAwayMsgData_type
+{
+ CString szProtocol;
+ MCONTACT hInitContact; // initial contact (filled by caller)
+ TCString Message; // initial message, NULL means default
+ bool IsModeless; // means the dialog was created with the CreateDialogParam function, not DialogBoxParam
+ int ISW_Flags; // InvokeStatusWindow service flags
+} SetAwayMsgData;
+
+typedef struct READAWAYMSGDATA_TYPE
+{
+ MCONTACT hContact; // contact
+ HANDLE hSeq; // sequence for stat msg request
+ HANDLE hAwayMsgEvent; // hooked
+} READAWAYMSGDATA;
+
+typedef struct
+{
+ char *szProto;
+ TCString Message;
+ DWORD UIN;
+ int Flags; // a combination of VPF_ flags
+} VAR_PARSE_DATA;
+
+typedef struct
+{
+ MCONTACT hContact;
+ int iStatusMode;
+ TCString Proto;
+} DYNAMIC_NOTIFY_DATA;
+
+typedef struct
+{
+ BYTE PopupLClickAction, PopupRClickAction;
+ MCONTACT hContact;
+ HICON hStatusIcon; // needed here to destroy its handle on UM_FREEPLUGINDATA
+} PLUGIN_DATA;
+
+typedef struct
+{
+ int cbSize;
+ char *szProto;
+ MCONTACT hContact;
+ char *szMsg;
+ WORD status;
+} NAS_ISWINFOv1;
+
+#define MTYPE_AUTOONLINE 0xE7 // required to support ICQ Plus online status messages
+
+// Beware of conflicts between two different windows trying to use the same page at a time!
+// Other windows than the owner of the Page must copy the page to their own memory,
+// or use GetDBValueCopy to retrieve values
+extern COptPage g_MessagesOptPage;
+extern COptPage g_AutoreplyOptPage;
+//extern COptPage g_PopupOptPage;
+extern COptPage g_MoreOptPage;
+extern COptPage g_SetAwayMsgPage;
+
+extern HINSTANCE g_hInstance;
+extern HANDLE hMainThread;
+extern int g_Messages_RecentRootID, g_Messages_PredefinedRootID;
+extern VAR_PARSE_DATA VarParseData;
+extern bool g_fNoProcessing;
+extern int g_bIsIdle;
+
+// AwaySys.cpp
+TCString GetDynamicStatMsg(MCONTACT hContact, char *szProto = NULL, DWORD UIN = 0, int iStatus = 0);
+int IsAnICQProto(char *szProto);
+
+// Client.cpp
+void InitUpdateMsgs();
+void ChangeProtoMessages(char* szProto, int iMode, TCString &Msg);
+int GetRecentGroupID(int iMode);
+TCString VariablesEscape(TCString Str);
+
+// SetAwayMsg.cpp
+INT_PTR CALLBACK SetAwayMsgDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
+
+// ReadAwayMsg.cpp
+extern HANDLE g_hReadWndList;
+INT_PTR GetContactStatMsg(WPARAM wParam, LPARAM lParam);
+
+// AwayOpt.cpp
+int OptsDlgInit(WPARAM wParam, LPARAM lParam); // called on opening of the options dialog
+void InitOptions(); // called once when plugin is loaded
+
+//int ShowPopupNotification(COptPage &PopupNotifyData, MCONTACT hContact, int iStatusMode);
+void ShowLog(TCString &LogFilePath);
+void ShowMsg(TCHAR *szFirstLine, TCHAR *szSecondLine = _T(""), bool IsErrorMsg = false, int Timeout = 0);
+
+#define AWAYSYS_STATUSMSGREQUEST_SOUND "AwaySysStatusMsgRequest"
+#define ME_AWAYSYS_WORKAROUND "AwaySys/_CallService"
+int _Workaround_CallService(const char *name, WPARAM wParam, LPARAM lParam);
+
+// MsgEventAdded.cpp
+int MsgEventAdded(WPARAM wParam, LPARAM lParam);
+
+// buttons
+//void UpdateSOEButtons(MCONTACT hContact = NULL);
+INT_PTR ToggleSendOnEvent(WPARAM wParam, LPARAM lParam);
+//int Create_TopToolbar(WPARAM wParam, LPARAM lParam);
+
+
+static __inline int LogMessage(const char *Format, ...)
+{
+ va_list va;
+ char szText[8096];
+ strcpy(szText, LOG_PREFIX);
+ va_start(va, Format);
+ mir_vsnprintf(szText + (lengthof(LOG_PREFIX) - 1), sizeof(szText) - (lengthof(LOG_PREFIX) - 1), Format, va);
+ va_end(va);
+ return CallService(MS_NETLIB_LOG, NULL, (LPARAM)szText);
+}
+
+__inline int CallAllowedPS_SETAWAYMSG(const char *szProto, int iMode, const char *szMsg)
+{ // we must use this function everywhere we want to call PS_SETAWAYMSG, otherwise NAS won't allow to change the message!
+ LogMessage("PS_SETAWAYMSG called by NAS. szProto=%s, Status=%d, Msg:\n%s", szProto, iMode, szMsg ? szMsg : "NULL");
+ char str[MAXMODULELABELLENGTH];
+ strcpy(str, szProto);
+ strcat(str, PS_SETAWAYMSG);
+ return CallService(str, (WPARAM)iMode, (LPARAM)szMsg);
+}
+
+static __inline void my_variables_skin_helpbutton(HWND hwndDlg, UINT uIDButton)
+{
+ HICON hIcon = ServiceExists(MS_VARS_GETSKINITEM) ? (HICON)CallService(MS_VARS_GETSKINITEM, 0, (LPARAM)VSI_HELPICON) : NULL;
+ if (hIcon)
+ SendDlgItemMessage(hwndDlg, uIDButton, BM_SETIMAGE, IMAGE_ICON, (LPARAM)hIcon);
+}
+
+static __inline int my_variables_showhelp(HWND hwndDlg, UINT uIDEdit, int flags = 0, char *szSubjectDesc = NULL, char *szExtraDesc = NULL)
+{
+ if (ServiceExists(MS_VARS_SHOWHELPEX))
+ {
+ VARHELPINFO vhi = {0};
+ vhi.cbSize = sizeof(VARHELPINFO);
+ vhi.flags = flags ? flags : (VHF_FULLDLG | VHF_SETLASTSUBJECT);
+ vhi.hwndCtrl = GetDlgItem(hwndDlg, uIDEdit);
+ vhi.szSubjectDesc = szSubjectDesc;
+ vhi.szExtraTextDesc = szExtraDesc;
+ return CallService(MS_VARS_SHOWHELPEX, (WPARAM)hwndDlg, (LPARAM)&vhi);
+ }
+ else
+ {
+ ShowMsg(TranslateT("New Away System"), TranslateT("Variables plugin is not installed"), true);
+ return -1;
+ }
+}
diff --git a/plugins/NewAwaySysMod/src/ContactList.cpp b/plugins/NewAwaySysMod/src/ContactList.cpp
new file mode 100644
index 0000000000..f9a710532f
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/ContactList.cpp
@@ -0,0 +1,898 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "ContactList.h"
+#include "Properties.h"
+
+#define INTM_CONTACTDELETED (WM_USER + 1)
+#define INTM_ICONCHANGED (WM_USER + 2)
+#define INTM_INVALIDATE (WM_USER + 3)
+
+#define HCONTACT_ISGROUP 0x80000000
+#define HCONTACT_ISINFO 0xFFFF0000
+#define IsHContactInfo(h) (((unsigned)(h) & HCONTACT_ISINFO) == HCONTACT_ISINFO)
+#define IsHContactGroup(h) (!IsHContactInfo(h) && ((unsigned)(h) & HCONTACT_ISGROUP))
+#define IsHContactContact(h) (((unsigned)(h) & HCONTACT_ISGROUP) == 0)
+
+#define EXTRAICON_XSTEP (GetSystemMetrics(SM_CXSMICON) + 1)
+
+static HANDLE hCLWindowList;
+
+
+static int CLContactDeleted(WPARAM wParam, LPARAM lParam)
+{
+ WindowList_Broadcast(hCLWindowList, INTM_CONTACTDELETED, wParam, lParam);
+ return 0;
+}
+
+static int CLContactIconChanged(WPARAM wParam, LPARAM lParam)
+{
+ WindowList_Broadcast(hCLWindowList, INTM_ICONCHANGED, wParam, lParam);
+ return 0;
+}
+
+static int CLIconsChanged(WPARAM wParam, LPARAM lParam)
+{
+ WindowList_Broadcast(hCLWindowList, INTM_INVALIDATE, 0, 0);
+ return 0;
+}
+
+void LoadCListModule()
+{
+ hCLWindowList = (HANDLE)CallService(MS_UTILS_ALLOCWINDOWLIST, 0, 0);
+ HookEvent(ME_DB_CONTACT_DELETED, CLContactDeleted);
+ HookEvent(ME_CLIST_CONTACTICONCHANGED, CLContactIconChanged);
+ HookEvent(ME_SKIN_ICONSCHANGED, CLIconsChanged);
+}
+
+
+static LRESULT CALLBACK ParentSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ CCList *dat = CWndUserData(hWnd).GetCList();
+ switch (Msg)
+ {
+ case WM_NOTIFY:
+ {
+ LPNMHDR pnmh = (LPNMHDR)lParam;
+ if (pnmh->hwndFrom == dat->hTreeView)
+ {
+ switch (pnmh->code)
+ {
+ case TVN_ITEMEXPANDED: // just set an appropriate group image
+ {
+ LPNMTREEVIEW pnmtv = (LPNMTREEVIEW)lParam;
+ TVITEM tvItem;
+ tvItem.hItem = pnmtv->itemNew.hItem;
+ tvItem.mask = TVIF_HANDLE | TVIF_IMAGE | TVIF_SELECTEDIMAGE;
+ tvItem.iImage = tvItem.iSelectedImage = (pnmtv->itemNew.state & TVIS_EXPANDED) ? IMAGE_GROUPOPEN : IMAGE_GROUPSHUT;
+ TreeView_SetItem(dat->hTreeView, &tvItem);
+ } break;
+ case TVN_SELCHANGED:
+ {
+ LPNMTREEVIEW pnmtv = (LPNMTREEVIEW)lParam;
+ TREEITEMARRAY OldSelection = dat->SelectedItems;
+ int I;
+ for (I = 0; I < dat->SelectedItems.GetSize(); I++)
+ {
+ if (dat->SelectedItems[I] != pnmtv->itemNew.hItem)
+ {
+ TreeView_SetItemState(dat->hTreeView, dat->SelectedItems[I], 0, TVIS_SELECTED);
+ }
+ }
+ dat->SelectedItems.RemoveAll();
+ if (pnmtv->itemNew.hItem)
+ {
+ dat->SelectedItems.AddElem(pnmtv->itemNew.hItem);
+ dat->SelectGroups(pnmtv->itemNew.hItem, true);
+ }
+ NMCLIST nm;
+ nm.hdr.code = MCLN_SELCHANGED;
+ nm.hdr.hwndFrom = dat->hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(dat->hTreeView);
+ nm.OldSelection = &OldSelection;
+ nm.NewSelection = &dat->SelectedItems;
+ SendMessage(hWnd, WM_NOTIFY, 0, (LPARAM)&nm);
+ } break;
+ case TVN_DELETEITEM:
+ {
+ if (dat->Items.GetSize()) // if Items size = 0, then this TVN_DELETEITEM came after WM_DESTROY, so there is no need to do anything
+ {
+ LPNMTREEVIEW pnmtv = (LPNMTREEVIEW)lParam;
+ TREEITEMARRAY OldSelection = dat->SelectedItems;
+ int Index = dat->SelectedItems.Find(pnmtv->itemOld.hItem);
+ if (Index != -1)
+ {
+ dat->SelectedItems.RemoveElem(Index);
+ }
+ // find an item to pass to SelectGroups()
+ HTREEITEM hItem = TreeView_GetNextSibling(dat->hTreeView, pnmtv->itemOld.hItem);
+ if (!hItem)
+ {
+ hItem = TreeView_GetPrevSibling(dat->hTreeView, pnmtv->itemOld.hItem);
+ if (!hItem)
+ {
+ hItem = TreeView_GetParent(dat->hTreeView, pnmtv->itemOld.hItem);
+ }
+ }
+ if (hItem) // if it wasn't one of the root items
+ {
+ dat->SelectGroups(hItem, dat->SelectedItems.Find(hItem) != -1);
+ }
+ NMCLIST nm;
+ nm.hdr.code = MCLN_SELCHANGED;
+ nm.hdr.hwndFrom = dat->hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(dat->hTreeView);
+ nm.OldSelection = &OldSelection;
+ nm.NewSelection = &dat->SelectedItems;
+ SendMessage(hWnd, WM_NOTIFY, 0, (LPARAM)&nm);
+ dat->Items[pnmtv->itemOld.lParam].hContact = INVALID_CONTACT_ID;
+ }
+ } break;
+ case NM_CUSTOMDRAW:
+ {
+ LPNMTVCUSTOMDRAW lpNMCD = (LPNMTVCUSTOMDRAW)lParam;
+ switch (lpNMCD->nmcd.dwDrawStage)
+ {
+ case CDDS_PREPAINT: // the control is about to start painting
+ {
+ return CDRF_NOTIFYITEMDRAW; // instruct the control to return information when it draws items
+ } break;
+ case CDDS_ITEMPREPAINT:
+ {
+ return CDRF_NOTIFYPOSTPAINT;
+ } break;
+ case CDDS_ITEMPOSTPAINT:
+ {
+ RECT rc;
+ if (TreeView_GetItemRect(dat->hTreeView, (HTREEITEM)lpNMCD->nmcd.dwItemSpec, &rc, false))
+ {
+ int I;
+ for (I = 0; I < MAXEXTRAICONS; I++)
+ {
+ BYTE nIndex = dat->Items[lpNMCD->nmcd.lItemlParam].ExtraIcons[I];
+ if (nIndex != CLC_EXTRAICON_EMPTY)
+ {
+ ImageList_DrawEx(dat->ExtraImageList, nIndex, lpNMCD->nmcd.hdc, rc.right - EXTRAICON_XSTEP * (I + 1), rc.top, 0, 0, /*GetSysColor(COLOR_WINDOW)*/CLR_NONE, CLR_NONE, ILD_NORMAL);
+ }
+ }
+ }
+ } break;
+ }
+ } break;
+ }
+ }
+ }
+ }
+ return CallWindowProc(dat->OrigParentProc, hWnd, Msg, wParam, lParam);
+}
+
+
+static LRESULT CALLBACK ContactListSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ CCList *dat = CWndUserData(GetParent(hWnd)).GetCList();
+ switch (Msg)
+ {
+ case INTM_CONTACTDELETED: // wParam = (HANDLE)hContact
+ {
+ HTREEITEM hItem = dat->FindContact(wParam);
+ if (hItem)
+ TreeView_DeleteItem(hWnd, hItem);
+ } break;
+ case INTM_ICONCHANGED: // wParam = (HANDLE)hContact, lParam = IconID
+ {
+ TVITEM tvi;
+ tvi.hItem = dat->FindContact(wParam);
+ if (tvi.hItem)
+ {
+ tvi.mask = TVIF_HANDLE | TVIF_IMAGE | TVIF_SELECTEDIMAGE;
+ tvi.iImage = tvi.iSelectedImage = lParam;
+ TreeView_SetItem(hWnd, &tvi);
+ dat->SortContacts();
+ InvalidateRect(hWnd, NULL, false);
+ }
+ } break;
+ case INTM_INVALIDATE:
+ {
+ InvalidateRect(hWnd, NULL, true);
+ } break;
+ case WM_RBUTTONDOWN:
+ {
+ SetFocus(hWnd);
+ TVHITTESTINFO hitTest;
+ hitTest.pt.x = (short)LOWORD(lParam);
+ hitTest.pt.y = (short)HIWORD(lParam);
+ TreeView_HitTest(hWnd, &hitTest);
+ if (hitTest.hItem && hitTest.flags & TVHT_ONITEM)
+ {
+ TreeView_SelectItem(hWnd, hitTest.hItem);
+ }
+ return DefWindowProc(hWnd, Msg, wParam, lParam);
+ } break;
+ case WM_LBUTTONDOWN:
+ {
+ POINT pt = {(short)LOWORD(lParam), (short)HIWORD(lParam)};
+ DWORD hitFlags;
+ HTREEITEM hItem = dat->HitTest(&pt, &hitFlags);
+ if (!hItem)
+ {
+ break;
+ }
+ if (hitFlags & MCLCHT_ONITEMICON)
+ {
+ if (TreeView_GetChild(hWnd, hItem)) // if it's a group, then toggle its state
+ {
+ NMTREEVIEW nmtv;
+ nmtv.hdr.hwndFrom = hWnd;
+ nmtv.hdr.idFrom = GetDlgCtrlID(hWnd);
+ nmtv.hdr.code = TVN_ITEMEXPANDING;
+ nmtv.action = TVE_TOGGLE;
+ nmtv.itemNew.hItem = hItem;
+ nmtv.itemNew.mask = TVIF_HANDLE | TVIF_STATE | TVIF_PARAM;
+ TreeView_GetItem(hWnd, &nmtv.itemNew);
+ nmtv.ptDrag = pt;
+ if (SendMessage(GetParent(hWnd), WM_NOTIFY, 0, (LPARAM)&nmtv))
+ {
+ return 0;
+ }
+ HTREEITEM hOldSelItem = TreeView_GetSelection(hWnd);
+ TreeView_Expand(hWnd, hItem, TVE_TOGGLE);
+ HTREEITEM hNewSelItem = TreeView_GetSelection(hWnd);
+ if (hNewSelItem != hOldSelItem)
+ {
+ TreeView_SetItemState(hWnd, hOldSelItem, (dat->SelectedItems.Find(hOldSelItem) == -1) ? 0 : TVIS_SELECTED, TVIS_SELECTED);
+ TreeView_SetItemState(hWnd, hNewSelItem, (dat->SelectedItems.Find(hNewSelItem) == -1) ? 0 : TVIS_SELECTED, TVIS_SELECTED);
+ }
+ nmtv.hdr.code = TVN_ITEMEXPANDED;
+ TreeView_GetItem(hWnd, &nmtv.itemNew);
+ SendMessage(GetParent(hWnd), WM_NOTIFY, 0, (LPARAM)&nmtv);
+ return 0;
+ }
+ }
+ if (hitFlags & MCLCHT_ONITEM)
+ {
+ if (wParam & MK_CONTROL)
+ {
+ SetFocus(hWnd);
+ TREEITEMARRAY OldSelection = dat->SelectedItems;
+ int nIndex = dat->SelectedItems.Find(hItem);
+ if (nIndex == -1)
+ {
+ TreeView_SetItemState(hWnd, hItem, TVIS_SELECTED, TVIS_SELECTED);
+ dat->SelectedItems.AddElem(hItem);
+ } else
+ {
+ TreeView_SetItemState(hWnd, hItem, 0, TVIS_SELECTED);
+ dat->SelectedItems.RemoveElem(nIndex);
+ }
+ dat->SelectGroups(hItem, nIndex == -1);
+ NMCLIST nm;
+ nm.hdr.code = MCLN_SELCHANGED;
+ nm.hdr.hwndFrom = hWnd;
+ nm.hdr.idFrom = GetDlgCtrlID(hWnd);
+ nm.OldSelection = &OldSelection;
+ nm.NewSelection = &dat->SelectedItems;
+ SendMessage(GetParent(hWnd), WM_NOTIFY, 0, (LPARAM)&nm);
+ return 0;
+ } else
+ {
+ if (hItem == TreeView_GetSelection(hWnd) && (dat->SelectedItems.GetSize() != 1 || (dat->SelectedItems.GetSize() == 1 && dat->SelectedItems[0] != hItem))) // if it was a click on the selected item and there's need to do something in this case, then send SELCHANGED notification by ourselves, as the tree control doesn't do anything
+ {
+ TreeView_SetItemState(hWnd, hItem, TVIS_SELECTED, TVIS_SELECTED);
+ NMTREEVIEW nm = {0};
+ nm.hdr.code = TVN_SELCHANGED;
+ nm.hdr.hwndFrom = hWnd;
+ nm.hdr.idFrom = GetDlgCtrlID(hWnd);
+ nm.itemOld.hItem = TreeView_GetSelection(hWnd);
+ nm.itemOld.mask = TVIF_HANDLE | TVIF_STATE | TVIF_PARAM;
+ TreeView_GetItem(hWnd, &nm.itemOld);
+ nm.itemNew = nm.itemOld;
+ SendMessage(GetParent(hWnd), WM_NOTIFY, 0, (LPARAM)&nm);
+ }
+ }
+ }
+ } break;
+ case WM_SETFOCUS:
+ case WM_KILLFOCUS:
+ {
+ int I;
+ for (I = 0; I < dat->SelectedItems.GetSize(); I++)
+ {
+ RECT rc;
+ if (TreeView_GetItemRect(hWnd, dat->SelectedItems[I], &rc, false))
+ {
+ InvalidateRect(hWnd, &rc, false);
+ }
+ }
+ } break;
+ case WM_SIZE:
+ case WM_HSCROLL:
+ {
+ InvalidateRect(hWnd, NULL, false);
+ } break;
+ case WM_MEASUREITEM:
+ {
+ if (!wParam) // if the message was sent by a menu
+ {
+ return CallService(MS_CLIST_MENUMEASUREITEM, wParam, lParam);
+ }
+ } break;
+ case WM_DRAWITEM:
+ {
+ if (!wParam) // if the message was sent by a menu
+ {
+ return CallService(MS_CLIST_MENUDRAWITEM, wParam, lParam);
+ }
+ } break;
+ case WM_CONTEXTMENU:
+ {
+ POINT pt;
+ pt.x = (short)LOWORD(lParam);
+ pt.y = (short)HIWORD(lParam);
+ HTREEITEM hItem = NULL;
+ if (pt.x == -1 && pt.y == -1)
+ {
+ if (dat->SelectedItems.GetSize() == 1)
+ {
+ hItem = dat->SelectedItems[0];
+ TreeView_EnsureVisible(hWnd, hItem);
+ RECT rc;
+ TreeView_GetItemRect(hWnd, hItem, &rc, true);
+ pt.x = rc.left;
+ pt.y = rc.bottom;
+ }
+ } else
+ {
+ DWORD hitFlags;
+ ScreenToClient(hWnd, &pt);
+ hItem = dat->HitTest(&pt, &hitFlags);
+ if (!(hitFlags & MCLCHT_ONITEM))
+ {
+ hItem = NULL;
+ }
+ }
+ if (hItem)
+ {
+ MCONTACT hContact = dat->GetItemData(hItem).hContact;
+ if (IsHContactContact(hContact))
+ {
+ HMENU hMenu = (HMENU)CallService(MS_CLIST_MENUBUILDCONTACT, (WPARAM)hContact, 0);
+ if (hMenu)
+ {
+ ClientToScreen(hWnd, &pt);
+ CallService(MS_CLIST_MENUPROCESSCOMMAND, MAKEWPARAM(TrackPopupMenu(hMenu, TPM_LEFTALIGN | TPM_RIGHTBUTTON | TPM_RETURNCMD, pt.x, pt.y, 0, hWnd, NULL), MPCF_CONTACTMENU), (LPARAM)hContact);
+ DestroyMenu(hMenu);
+ return 0;
+ }
+ }
+ }
+ } break;
+ case WM_DESTROY:
+ {
+ if (dat->ExtraImageList)
+ {
+ ImageList_Destroy(dat->ExtraImageList);
+ }
+ dat->SelectedItems.RemoveAll();
+ dat->Items.RemoveAll();
+ } break;
+ }
+ return CallWindowProc(dat->OrigTreeViewProc, hWnd, Msg, wParam, lParam);
+}
+
+
+CCList::CCList(HWND hTreeView): hTreeView(hTreeView), ExtraImageList(NULL)
+{
+ CWndUserData(GetParent(hTreeView)).SetCList(this);
+ OrigTreeViewProc = (WNDPROC)SetWindowLongPtr(hTreeView, GWLP_WNDPROC, (LONG_PTR)ContactListSubclassProc);
+ OrigParentProc = (WNDPROC)SetWindowLongPtr(GetParent(hTreeView), GWLP_WNDPROC, (LONG_PTR)ParentSubclassProc);
+ TreeView_SetImageList(hTreeView, CallService(MS_CLIST_GETICONSIMAGELIST, 0, 0), TVSIL_NORMAL);
+ WindowList_Add(hCLWindowList, hTreeView, NULL);
+ TreeView_SetIndent(hTreeView, 5); // doesn't set it less than the initial value on my system, and I guess it's because of icons... but who knows - maybe it will work somewhere
+}
+
+
+CCList::~CCList()
+{
+ WindowList_Remove(hCLWindowList, hTreeView);
+ _ASSERT(GetWindowLongPtr(GetParent(hTreeView), GWLP_WNDPROC) == (LONG_PTR)ParentSubclassProc); // we won't allow anyone to change our WNDPROC. otherwise we're not sure that we're setting the right WNDPROC back
+ SetWindowLongPtr(hTreeView, GWLP_WNDPROC, (LONG_PTR)OrigTreeViewProc);
+ SetWindowLongPtr(GetParent(hTreeView), GWLP_WNDPROC, (LONG_PTR)OrigParentProc);
+ CWndUserData(GetParent(hTreeView)).SetCList(NULL);
+}
+
+
+HTREEITEM CCList::AddContact(MCONTACT hContact)
+// adds a new contact if it doesn't exist yet; returns its hItem
+{
+ _ASSERT(IsHContactContact(hContact));
+ HTREEITEM hContactItem = FindContact(hContact);
+ if (hContactItem)
+ {
+ return hContactItem;
+ }
+ TVINSERTSTRUCT tvIns;
+ ZeroMemory(&tvIns, sizeof(tvIns));
+ tvIns.hParent = AddGroup(db_get_s(hContact, "CList", "Group", _T("")));
+/* if (!tvIns.hParent)
+ {
+ return NULL;
+ }*/ // <- place hidden contacts in the root anyway, as otherwise we won't see icq contacts that are hidden beneath metacontacts; TODO: show metacontacts as groups??
+ tvIns.item.pszText = (TCHAR*)CallService(MS_CLIST_GETCONTACTDISPLAYNAME, (WPARAM)hContact, GCDNF_TCHAR);
+ tvIns.hInsertAfter = TVI_ROOT;
+ tvIns.item.mask = TVIF_TEXT | TVIF_IMAGE | TVIF_SELECTEDIMAGE | TVIF_PARAM;
+ tvIns.item.iImage = tvIns.item.iSelectedImage = CallService(MS_CLIST_GETCONTACTICON, (WPARAM)hContact, 0);
+ tvIns.item.lParam = Items.AddElem(CCLItemData(hContact));
+ return TreeView_InsertItem(hTreeView, &tvIns);
+}
+
+
+typedef struct
+{
+ HANDLE hGroup;
+ TCString GroupName;
+} sGroupEnumData;
+
+int GroupEnum(const char *szSetting, LPARAM lParam)
+{
+ sGroupEnumData *GroupEnumData = (sGroupEnumData*)lParam;
+ TCString GroupName = db_get_s(NULL, "CListGroups", szSetting, _T(" "));
+ if (!lstrcmp(GroupEnumData->GroupName, &GroupName[1]))
+ {
+ GroupEnumData->hGroup = (HANDLE)(atol(szSetting) | HCONTACT_ISGROUP);
+ }
+ return 0;
+}
+
+HTREEITEM CCList::AddGroup(TCString GroupName)
+// adds a new group if it doesn't exist yet; returns its hItem
+{
+ if (GroupName == _T(""))
+ {
+ return TVI_ROOT;
+ }
+ sGroupEnumData GroupEnumData;
+ GroupEnumData.GroupName = GroupName;
+ GroupEnumData.hGroup = NULL;
+ DBCONTACTENUMSETTINGS dbEnum;
+ ZeroMemory(&dbEnum, sizeof(dbEnum));
+ dbEnum.lParam = (LPARAM)&GroupEnumData;
+ dbEnum.pfnEnumProc = GroupEnum;
+ dbEnum.szModule = "CListGroups";
+ CallService(MS_DB_CONTACT_ENUMSETTINGS, NULL, (LPARAM)&dbEnum);
+ if (!GroupEnumData.hGroup) // means there is no such group in the groups list
+ {
+ return NULL;
+ }
+ HTREEITEM hGroupItem = FindContact((MCONTACT)GroupEnumData.hGroup);
+ if (hGroupItem)
+ {
+ return hGroupItem; // exists already, just return its handle
+ }
+ TVINSERTSTRUCT tvIns = {0};
+ tvIns.hParent = TVI_ROOT;
+ tvIns.item.pszText = _tcsrchr(GroupName, '\\');
+ if (tvIns.item.pszText)
+ {
+ TCString ParentGroupName(_T(""));
+ tvIns.hParent = AddGroup(ParentGroupName.DiffCat(GroupName, tvIns.item.pszText));
+ tvIns.item.pszText++;
+ } else
+ {
+ tvIns.item.pszText = GroupName;
+ }
+ tvIns.hInsertAfter = TVI_ROOT;
+ tvIns.item.mask = TVIF_TEXT | TVIF_STATE | TVIF_IMAGE | TVIF_SELECTEDIMAGE | TVIF_PARAM;
+ tvIns.item.state = tvIns.item.stateMask = TVIS_BOLD | TVIS_EXPANDED;
+ tvIns.item.iImage = tvIns.item.iSelectedImage = IMAGE_GROUPOPEN;
+ tvIns.item.lParam = Items.AddElem(CCLItemData((MCONTACT)GroupEnumData.hGroup));
+ return TreeView_InsertItem(hTreeView, &tvIns);
+}
+
+
+HTREEITEM CCList::AddInfo(TCString Title, HTREEITEM hParent, HTREEITEM hInsertAfter, LPARAM lParam, HICON hIcon)
+{
+ TVINSERTSTRUCT tvi = {0};
+ tvi.item.mask = TVIF_TEXT | TVIF_STATE | TVIF_PARAM;
+ tvi.item.pszText = Title;
+ tvi.hParent = hParent;
+ tvi.hInsertAfter = hInsertAfter;
+ tvi.item.lParam = Items.AddElem(CCLItemData());
+ Items[tvi.item.lParam].lParam = lParam;
+ tvi.item.state = tvi.item.stateMask = TVIS_BOLD | TVIS_EXPANDED;
+ if (hIcon)
+ {
+ HIMAGELIST iml = TreeView_GetImageList(hTreeView, TVSIL_NORMAL);
+ tvi.item.mask |= TVIF_IMAGE | TVIF_SELECTEDIMAGE;
+ tvi.item.iImage = tvi.item.iSelectedImage = ImageList_AddIcon(iml, hIcon); // we don't check for duplicate icons, but I think that's ok, judging that the check will require some pretty significant amount of additional coding
+ TreeView_SetImageList(hTreeView, iml, TVSIL_NORMAL);
+ }
+ return TreeView_InsertItem(hTreeView, &tvi);
+}
+
+
+void CCList::SetInfoIcon(HTREEITEM hItem, HICON hIcon)
+{
+ _ASSERT(hItem && hIcon && GetItemType(hItem) == MCLCIT_INFO);
+ TVITEM tvi = {0};
+ tvi.mask = TVIF_HANDLE | TVIF_IMAGE | TVIF_SELECTEDIMAGE;
+ tvi.hItem = hItem;
+ HIMAGELIST iml = TreeView_GetImageList(hTreeView, TVSIL_NORMAL);
+ tvi.iImage = tvi.iSelectedImage = ImageList_AddIcon(iml, hIcon); // again, we don't check for duplicate icons
+ TreeView_SetImageList(hTreeView, iml, TVSIL_NORMAL);
+ TreeView_SetItem(hTreeView, &tvi);
+}
+
+
+static int CALLBACK CompareItemsCallback(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
+{
+ CCList *dat = (CCList*)lParamSort;
+ if (IsHContactInfo(dat->Items[lParam1].hContact)) // Info items precede all other items
+ {
+ return (IsHContactInfo(dat->Items[lParam2].hContact)) ? 0 : -1;
+ } else if (IsHContactInfo(dat->Items[lParam2].hContact))
+ {
+ return 1;
+ }
+ if (IsHContactGroup(dat->Items[lParam1].hContact)) // groups precede contacts
+ {
+ if (IsHContactGroup(dat->Items[lParam2].hContact))
+ {
+ return (unsigned)dat->Items[lParam1].hContact - (unsigned)dat->Items[lParam2].hContact;
+ } else
+ {
+ return -1;
+ }
+ } else if (IsHContactGroup(dat->Items[lParam2].hContact))
+ {
+ return 1;
+ }
+ return CallService(MS_CLIST_CONTACTSCOMPARE, (WPARAM)dat->Items[lParam1].hContact, (LPARAM)dat->Items[lParam2].hContact);
+}
+
+void CCList::SortContacts()
+{
+ TVSORTCB tvSort;
+ ZeroMemory(&tvSort, sizeof(tvSort));
+ tvSort.lpfnCompare = CompareItemsCallback;
+ tvSort.hParent = TVI_ROOT;
+ tvSort.lParam = (LPARAM)this;
+ while (tvSort.hParent)
+ {
+ TreeView_SortChildrenCB(hTreeView, &tvSort, 0);
+ tvSort.hParent = GetNextItem(MCLGN_NEXT | MCLGN_GROUP | MCLGN_MULTILEVEL, tvSort.hParent);
+ }
+}
+
+
+int CCList::GetExtraImage(HTREEITEM hItem, int iColumn) // returns iImage, or CLC_EXTRAICON_EMPTY
+{
+ _ASSERT(iColumn < MAXEXTRAICONS);
+ return GetItemData(hItem).ExtraIcons[iColumn];
+}
+
+
+void CCList::SetExtraImage(HTREEITEM hItem, int iColumn, int iImage) // set iImage to CLC_EXTRAICON_EMPTY to reset image
+{
+ _ASSERT(iColumn < MAXEXTRAICONS);
+ GetItemData(hItem).ExtraIcons[iColumn] = iImage;
+ RECT rc;
+ if (TreeView_GetItemRect(hTreeView, hItem, &rc, false))
+ {
+ InvalidateRect(hTreeView, &rc, true);
+ }
+}
+
+
+void CCList::SetExtraImageList(HIMAGELIST hImgList)
+{
+ ExtraImageList = hImgList;
+ InvalidateRect(hTreeView, NULL, false);
+}
+
+
+int CCList::GetItemType(HTREEITEM hItem) // returns a MCLCIT_ (see below)
+{
+ MCONTACT hContact = GetItemData(hItem).hContact;
+ return (IsHContactInfo(hContact)) ? MCLCIT_INFO : ((IsHContactGroup(hContact)) ? MCLCIT_GROUP : MCLCIT_CONTACT);
+}
+
+
+DWORD CCList::GetItemTypeAsCLGNFlag(HTREEITEM hItem)
+{
+ MCONTACT hContact = GetItemData(hItem).hContact;
+ return (IsHContactInfo(hContact)) ? MCLGN_INFO : ((IsHContactGroup(hContact)) ? MCLGN_GROUP : MCLGN_CONTACT);
+}
+
+
+HTREEITEM CCList::GetNextItem(DWORD Flags, HTREEITEM hItem)
+{
+ switch (Flags & ~(MCLGN_MULTILEVEL | MCLGN_NOTCHILD | MCLGN_ANY))
+ {
+ case MCLGN_ROOT:
+ {
+ return TreeView_GetRoot(hTreeView);
+ } break;
+ case MCLGN_LAST:
+ {
+ HTREEITEM hNextItem = TVI_ROOT;
+ do
+ {
+ hItem = hNextItem;
+ hNextItem = TreeView_GetLastChild(hTreeView, hNextItem);
+ } while (hNextItem);
+ return (hItem == TVI_ROOT) ? NULL : hItem;
+ } break;
+ case MCLGN_CHILD:
+ {
+ return TreeView_GetChild(hTreeView, hItem);
+ } break;
+ case MCLGN_LASTCHILD:
+ {
+ return TreeView_GetLastChild(hTreeView, hItem);
+ } break;
+ case MCLGN_PARENT:
+ {
+ return TreeView_GetParent(hTreeView, hItem);
+ } break;
+ case MCLGN_NEXT:
+ {
+ do
+ {
+ if (Flags & MCLGN_MULTILEVEL)
+ {
+ HTREEITEM hNextItem = NULL;
+ if ((Flags & MCLGN_NOTCHILD) != MCLGN_NOTCHILD)
+ {
+ hNextItem = TreeView_GetChild(hTreeView, hItem);
+ }
+ if (!hNextItem)
+ {
+ hNextItem = TreeView_GetNextSibling(hTreeView, hItem);
+ while (!hNextItem) // move back until we find next sibling of the item or one of its parents
+ {
+ hItem = TreeView_GetParent(hTreeView, hItem);
+ if (!hItem) // means it was the root, there are no items left.
+ {
+ break; // returns NULL as the next item
+ }
+ hNextItem = TreeView_GetNextSibling(hTreeView, hItem);
+ }
+ }
+ hItem = hNextItem;
+ } else
+ {
+ hItem = TreeView_GetNextSibling(hTreeView, hItem);
+ }
+ Flags &= ~(MCLGN_NOTCHILD & ~MCLGN_MULTILEVEL); // clear MCLGN_NOTCHILD flag
+ } while (hItem && !(Flags & GetItemTypeAsCLGNFlag(hItem)));
+ return hItem;
+ } break;
+ case MCLGN_PREV:
+ {
+ do
+ {
+ if (Flags & MCLGN_MULTILEVEL)
+ {
+ HTREEITEM hNextItem = TreeView_GetPrevSibling(hTreeView, hItem);
+ if (hNextItem)
+ {
+ if ((Flags & MCLGN_NOTCHILD) != MCLGN_NOTCHILD)
+ {
+ while (hNextItem)
+ {
+ hItem = hNextItem;
+ hNextItem = TreeView_GetLastChild(hTreeView, hItem);
+ }
+ } else
+ {
+ hItem = hNextItem;
+ }
+ } else
+ {
+ hItem = TreeView_GetParent(hTreeView, hItem);
+ }
+ } else
+ {
+ hItem = TreeView_GetPrevSibling(hTreeView, hItem);
+ }
+ Flags &= ~(MCLGN_NOTCHILD & ~MCLGN_MULTILEVEL); // clear MCLGN_NOTCHILD flag
+ } while (hItem && !(Flags & GetItemTypeAsCLGNFlag(hItem)));
+ return hItem;
+ } break;
+ default:
+ {
+ _ASSERT(0);
+ } break;
+ }
+ return NULL;
+}
+
+
+MCONTACT CCList::GethContact(HTREEITEM hItem) // returns hContact, hGroup or hInfo
+{
+ MCONTACT hContact = GetItemData(hItem).hContact;
+ if (IsHContactContact(hContact))
+ return hContact;
+ if (IsHContactGroup(hContact))
+ return hContact & ~HCONTACT_ISGROUP;
+ return hContact & ~HCONTACT_ISINFO;
+}
+
+
+HTREEITEM CCList::HitTest(LPPOINT pt, PDWORD hitFlags) // pt is relative to control; returns hItem or NULL
+{
+ TVHITTESTINFO hti;
+ hti.pt = *pt;
+ TreeView_HitTest(hTreeView, &hti);
+ *hitFlags = 0;
+ if (hti.flags & TVHT_ABOVE)
+ {
+ *hitFlags |= MCLCHT_ABOVE;
+ }
+ if (hti.flags & TVHT_BELOW)
+ {
+ *hitFlags |= MCLCHT_BELOW;
+ }
+ if (hti.flags & TVHT_TOLEFT)
+ {
+ *hitFlags |= MCLCHT_TOLEFT;
+ }
+ if (hti.flags & TVHT_TORIGHT)
+ {
+ *hitFlags |= MCLCHT_TORIGHT;
+ }
+ if (hti.flags & TVHT_NOWHERE)
+ {
+ *hitFlags |= MCLCHT_NOWHERE;
+ }
+ if (hti.flags & TVHT_ONITEMINDENT)
+ {
+ *hitFlags |= MCLCHT_ONITEMINDENT;
+ }
+ if (hti.flags & (TVHT_ONITEMICON | TVHT_ONITEMSTATEICON))
+ {
+ *hitFlags |= MCLCHT_ONITEMICON;
+ }
+ if (hti.flags & TVHT_ONITEMLABEL)
+ {
+ *hitFlags |= MCLCHT_ONITEMLABEL;
+ }
+ if (hti.flags & TVHT_ONITEMRIGHT)
+ {
+ *hitFlags |= MCLCHT_ONITEMRIGHT;
+ }
+ if (hti.flags & (TVHT_ONITEMINDENT | TVHT_ONITEM | TVHT_ONITEMRIGHT))
+ {
+ // extraicon tests
+ RECT rc;
+ if (TreeView_GetItemRect(hTreeView, hti.hItem, &rc, false))
+ {
+ int nIndex = (rc.right - pt->x - 1) / EXTRAICON_XSTEP;
+ if (nIndex >= 0 && nIndex < MAXEXTRAICONS && GetItemData(hti.hItem).ExtraIcons[nIndex] != CLC_EXTRAICON_EMPTY)
+ {
+ *hitFlags |= MCLCHT_ONITEMEXTRA | (nIndex << 24);
+ }
+ }
+ }
+ return hti.hItem;
+}
+
+
+int CCList::Array_SetItemState(HTREEITEM hItem, bool bSelected)
+{
+ _ASSERT(hItem);
+ int nIndex = SelectedItems.Find(hItem);
+ if (nIndex == -1 && bSelected)
+ {
+ return SelectedItems.AddElem(hItem);
+ } else if (nIndex != -1 && !bSelected)
+ {
+ SelectedItems.RemoveElem(nIndex);
+ return -1;
+ }
+ return nIndex;
+}
+
+
+CCLItemData& CCList::GetItemData(HTREEITEM hItem)
+{
+ _ASSERT(hItem && hItem != INVALID_HANDLE_VALUE);
+ TVITEM tvi;
+ tvi.mask = TVIF_HANDLE | TVIF_PARAM;
+ tvi.hItem = hItem;
+ int Res = TreeView_GetItem(hTreeView, &tvi);
+ _ASSERT(Res);
+ return Items[tvi.lParam];
+}
+
+
+HTREEITEM CCList::TreeView_GetLastChild(HWND hTreeView, HTREEITEM hItem)
+{
+ HTREEITEM hPrevItem = TreeView_GetChild(hTreeView, hItem);
+ hItem = hPrevItem;
+ while (hItem) // find last sibling
+ {
+ hPrevItem = hItem;
+ hItem = TreeView_GetNextSibling(hTreeView, hPrevItem);
+ }
+ return hPrevItem;
+}
+
+
+HTREEITEM CCList::FindContact(MCONTACT hContact)
+{
+ TVITEM tvi;
+ tvi.mask = TVIF_HANDLE | TVIF_PARAM;
+ tvi.hItem = TreeView_GetRoot(hTreeView);
+ while (tvi.hItem)
+ {
+ TreeView_GetItem(hTreeView, &tvi);
+ if (Items[tvi.lParam].hContact == hContact)
+ {
+ return tvi.hItem;
+ }
+ tvi.hItem = GetNextItem(MCLGN_NEXT | MCLGN_ANY | MCLGN_MULTILEVEL, tvi.hItem);
+ }
+ return NULL;
+}
+
+
+void CCList::SelectGroups(HTREEITEM hCurItem, bool bSelected)
+{
+// select/deselect all child items
+ HTREEITEM hItem = TreeView_GetChild(hTreeView, hCurItem);
+ HTREEITEM hLimitItem = GetNextItem(MCLGN_NEXT | MCLGN_ANY | MCLGN_NOTCHILD, hCurItem);
+ while (hItem && hItem != hLimitItem)
+ {
+ TreeView_SetItemState(hTreeView, hItem, bSelected ? TVIS_SELECTED : 0, TVIS_SELECTED);
+ Array_SetItemState(hItem, bSelected);
+ hItem = GetNextItem(MCLGN_NEXT | MCLGN_ANY | MCLGN_MULTILEVEL, hItem);
+ }
+// select/deselect all parent groups
+ hCurItem = TreeView_GetParent(hTreeView, hCurItem);
+ if (bSelected)
+ {
+ while (hCurItem) // select until we'll find an unselected item or until we'll reach the root
+ {
+ hItem = TreeView_GetChild(hTreeView, hCurItem);
+ while (hItem) // walk through all siblings
+ {
+ if (!(TreeView_GetItemState(hTreeView, hItem, TVIS_SELECTED) & TVIS_SELECTED))
+ {
+ break;
+ }
+ hItem = TreeView_GetNextSibling(hTreeView, hItem);
+ }
+ if (hItem) // means there was at least one unselected item
+ {
+ break;
+ }
+ TreeView_SetItemState(hTreeView, hCurItem, TVIS_SELECTED, TVIS_SELECTED);
+ Array_SetItemState(hCurItem, true);
+ hCurItem = TreeView_GetParent(hTreeView, hCurItem);
+ }
+ }
+ while (hCurItem) // and deselect all remaining parent groups
+ {
+ TreeView_SetItemState(hTreeView, hCurItem, 0, TVIS_SELECTED);
+ Array_SetItemState(hCurItem, false);
+ hCurItem = TreeView_GetParent(hTreeView, hCurItem);
+ }
+}
diff --git a/plugins/NewAwaySysMod/src/ContactList.h b/plugins/NewAwaySysMod/src/ContactList.h
new file mode 100644
index 0000000000..8b9ce2f845
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/ContactList.h
@@ -0,0 +1,142 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+// Miranda's built-in CListControl is very slow and it has too limited functionality... 8( So I had to use my own control instead.
+
+#pragma once
+
+typedef TMyArray<HTREEITEM> TREEITEMARRAY;
+typedef TREEITEMARRAY* PTREEITEMARRAY;
+
+
+#define CLC_EXTRAICON_EMPTY 0xFF
+#define CLC_ROOT TVI_ROOT
+
+#define MAXEXTRAICONS 16
+
+class CCLItemData // internal CCList's class
+{
+public:
+ CCLItemData(MCONTACT hContact = INVALID_CONTACT_ID): hContact(hContact) {FillMemory(ExtraIcons, sizeof(ExtraIcons), CLC_EXTRAICON_EMPTY);};
+
+ BYTE ExtraIcons[MAXEXTRAICONS];
+ MCONTACT hContact;
+ LPARAM lParam;
+};
+
+typedef TMyArray<CCLItemData> TREEITEMDATAARRAY;
+typedef TREEITEMDATAARRAY* PTREEITEMDATAARRAY;
+
+class CCList
+{
+public:
+ CCList(HWND hTreeView);
+ ~CCList();
+
+ HTREEITEM AddContact(MCONTACT hContact);
+ HTREEITEM AddGroup(TCString GroupName);
+ HTREEITEM AddInfo(TCString Title, HTREEITEM hParent, HTREEITEM hInsertAfter, LPARAM lParam = NULL, HICON hIcon = NULL);
+ void SetInfoIcon(HTREEITEM hItem, HICON hIcon);
+ int GetExtraImage(HTREEITEM hItem, int iColumn); // returns iImage, or CLC_EXTRAICON_EMPTY
+ void SetExtraImage(HTREEITEM hItem, int iColumn, int iImage); // set iImage to CLC_EXTRAICON_EMPTY to reset image
+ void SetExtraImageList(HIMAGELIST hImgList);
+ int GetItemType(HTREEITEM hItem); // returns a MCLCIT_ (see below)
+ HTREEITEM GetNextItem(DWORD Flags, HTREEITEM hItem);
+ void SortContacts();
+ MCONTACT GethContact(HTREEITEM hItem); // returns hContact, hGroup or hInfo
+ HTREEITEM HitTest(LPPOINT pt, PDWORD hitFlags); // pt is relative to control; returns hItem or NULL
+ void EnsureVisible(HTREEITEM hItem) {TreeView_EnsureVisible(hTreeView, hItem); InvalidateRect(hTreeView, NULL, false);} // sometimes horizontal scrollbar position changes too, so we must redraw extra icons - that's why here is InvalidateRect. TODO: try to find a way to invalidate it on _every_ horizontal scrollbar position change, instead of just here - unfortunately the scrollbar doesn't notify the tree control of its position change through WM_HSCROLL in some cases
+ int SelectItem(HTREEITEM hItem) {return TreeView_SelectItem(hTreeView, hItem);}
+ void SetItemParam(HTREEITEM hItem, LPARAM lParam) {GetItemData(hItem).lParam = lParam;}
+ LPARAM GetItemParam(HTREEITEM hItem) {return GetItemData(hItem).lParam;}
+ PTREEITEMARRAY GetSelection() {return &SelectedItems;}
+ void SetRedraw(bool bRedraw) {SendMessage(hTreeView, WM_SETREDRAW, bRedraw, 0);}
+
+ friend LRESULT CALLBACK ParentSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam);
+ friend LRESULT CALLBACK ContactListSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam);
+ friend int CALLBACK CompareItemsCallback(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort);
+
+private:
+ int Array_SetItemState(HTREEITEM hItem, bool bSelected);
+ CCLItemData& GetItemData(HTREEITEM hItem);
+ HTREEITEM TreeView_GetLastChild(HWND hTreeView, HTREEITEM hItem);
+ HTREEITEM FindContact(MCONTACT hContact); // returns NULL if not found
+ void SelectGroups(HTREEITEM hCurItem, bool bSelected);
+ DWORD GetItemTypeAsCLGNFlag(HTREEITEM hItem); // returns MCLGN_CONTACT, MCLGN_GROUP or MCLGN_INFO
+
+ HWND hTreeView;
+ WNDPROC OrigTreeViewProc;
+ WNDPROC OrigParentProc;
+ TREEITEMDATAARRAY Items; // array that stores info for every tree item; array items are left even if corresponding tree item was deleted, so treeitem's lParam can be used as an index of an item in this array.
+ TREEITEMARRAY SelectedItems; // contains HTREEITEMs of all selected items
+ HIMAGELIST ExtraImageList;
+};
+
+
+// HitTest constants
+#define MCLCHT_ABOVE 0x0001 // above the client area
+#define MCLCHT_BELOW 0x0002 // below the client area
+#define MCLCHT_TOLEFT 0x0004 // to the left of the client area
+#define MCLCHT_TORIGHT 0x0008 // to the right of the client area
+#define MCLCHT_NOWHERE 0x0010 // in the client area, but below the last item
+#define MCLCHT_ONITEMINDENT 0x0020 // to the left of an item icon
+#define MCLCHT_ONITEMICON 0x0040
+#define MCLCHT_ONITEMLABEL 0x0080
+#define MCLCHT_ONITEMRIGHT 0x0100 // in the area to the right of an item
+#define MCLCHT_ONITEMEXTRA 0x0200 // on an extra icon, HIBYTE(HIWORD(hitFlags)) says which
+#define MCLCHT_ONITEM (MCLCHT_ONITEMICON | MCLCHT_ONITEMLABEL)
+
+// item types
+#define MCLCIT_GROUP 0
+#define MCLCIT_CONTACT 1
+#define MCLCIT_INFO 3
+
+// GetNextItem constants
+#define MCLGN_ROOT 0
+#define MCLGN_LAST 1
+#define MCLGN_CHILD 2
+#define MCLGN_LASTCHILD 3
+#define MCLGN_PARENT 4
+#define MCLGN_NEXT 5
+#define MCLGN_PREV 6
+// flags for use with MCLGN_NEXT and MCLGN_PREV:
+#define MCLGN_CONTACT 0x20 // you need to specify at least one of these three! otherwise GetNextItem will not find anything
+#define MCLGN_GROUP 0x40
+#define MCLGN_INFO 0x80
+#define MCLGN_ANY (MCLGN_CONTACT | MCLGN_GROUP | MCLGN_INFO)
+#define MCLGN_MULTILEVEL 0x100 // walk through items of different levels too (ex.: MCLGN_NEXT | MCLGN_MULTILEVEL)
+#define MCLGN_NOTCHILD (0x200 | MCLGN_MULTILEVEL) // when this flag is set, child items of specified hItem are ignored. for example, MCLGN_NEXT | MCLGN_CONTACT | MCLGN_NOTCHILD retrieves next contact that is not a child of hItem.
+// GetNextItem(MCLGN_PREV | MCLGN_ALL | MCLGN_NOTCHILD, CLM_GETNEXTITEM(MCLGN_NEXT | MCLGN_ALL | MCLGN_NOTCHILD, hItem) === hItem
+
+
+// notifications
+
+typedef struct
+{
+ NMHDR hdr;
+ PTREEITEMARRAY OldSelection, NewSelection;
+} NMCLIST;
+typedef NMCLIST* PNMCLIST;
+
+#define MCLN_FIRST (0U - 100U)
+
+#define MCLN_SELCHANGED (MCLN_FIRST - 20) // lParam = &NMCLIST; OldSelection and NewSelection contain selection info
+
+
+void LoadCListModule();
diff --git a/plugins/NewAwaySysMod/src/MsgEventAdded.cpp b/plugins/NewAwaySysMod/src/MsgEventAdded.cpp
new file mode 100644
index 0000000000..6995a6c683
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/MsgEventAdded.cpp
@@ -0,0 +1,291 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (C) 2005-2007 Chervov Dmitry
+ Copyright (C) 2004-2005 Iksaif Entertainment
+ Copyright (C) 2002-2003 Goblineye Entertainment
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "Properties.h"
+
+// stupid compiler.. kept returning me an "INTERNAL COMPILER ERROR" on almost every line of MsgEventAdded function with Release mode and unicode enabled >:( even disabling optimization through #pragma optimize didn't help.. The only acceptable solution I found is to move this function to a separate file and to disable optimization for the whole file in the project properties.
+
+static struct
+{
+ int Status, DisableReplyCtlID, DontShowDialogCtlID;
+} StatusModeList[] = {
+ ID_STATUS_ONLINE, IDC_REPLYDLG_DISABLE_ONL, IDC_MOREOPTDLG_DONTPOPDLG_ONL,
+ ID_STATUS_AWAY, IDC_REPLYDLG_DISABLE_AWAY, IDC_MOREOPTDLG_DONTPOPDLG_AWAY,
+ ID_STATUS_NA, IDC_REPLYDLG_DISABLE_NA, IDC_MOREOPTDLG_DONTPOPDLG_NA,
+ ID_STATUS_OCCUPIED, IDC_REPLYDLG_DISABLE_OCC, IDC_MOREOPTDLG_DONTPOPDLG_OCC,
+ ID_STATUS_DND, IDC_REPLYDLG_DISABLE_DND, IDC_MOREOPTDLG_DONTPOPDLG_DND,
+ ID_STATUS_FREECHAT, IDC_REPLYDLG_DISABLE_FFC, IDC_MOREOPTDLG_DONTPOPDLG_FFC,
+ ID_STATUS_INVISIBLE, IDC_REPLYDLG_DISABLE_INV, IDC_MOREOPTDLG_DONTPOPDLG_INV,
+ ID_STATUS_ONTHEPHONE, IDC_REPLYDLG_DISABLE_OTP, IDC_MOREOPTDLG_DONTPOPDLG_OTP,
+ ID_STATUS_OUTTOLUNCH, IDC_REPLYDLG_DISABLE_OTL, IDC_MOREOPTDLG_DONTPOPDLG_OTL
+};
+
+
+class CAutoreplyData
+{
+public:
+ CAutoreplyData(MCONTACT hContact, TCString Reply): hContact(hContact), Reply(Reply) {}
+
+ MCONTACT hContact;
+ TCString Reply;
+};
+
+
+void __cdecl AutoreplyDelayThread(void *_ad)
+{ // _ad must be allocated using "new CAutoreplyData()"
+ CAutoreplyData *ad = (CAutoreplyData*)_ad;
+ _ASSERT(ad && ad->hContact && ad->Reply.GetLen());
+ char *szProto = GetContactProto(ad->hContact);
+ if (!szProto)
+ {
+ _ASSERT(0);
+ return;
+ }
+
+ int ReplyLen = (ad->Reply.GetLen() + 1) * (sizeof(char) + sizeof(WCHAR));
+ PBYTE pBuf = (PBYTE)malloc(ReplyLen);
+ memcpy(pBuf, _T2A(ad->Reply), ad->Reply.GetLen() + 1);
+ memcpy(pBuf + ad->Reply.GetLen() + 1, ad->Reply, (ad->Reply.GetLen() + 1) * sizeof(WCHAR));
+ CallContactService(ad->hContact, ServiceExists(CString(szProto) + PSS_MESSAGE "W") ? (PSS_MESSAGE "W") : PSS_MESSAGE, PREF_UNICODE, (LPARAM)pBuf);
+
+ if (g_AutoreplyOptPage.GetDBValueCopy(IDC_REPLYDLG_LOGREPLY))
+ { // store in the history
+ DBEVENTINFO dbeo = {0};
+ dbeo.cbSize = sizeof(dbeo);
+ dbeo.eventType = EVENTTYPE_MESSAGE;
+ dbeo.flags = DBEF_SENT;
+ dbeo.szModule = szProto;
+ dbeo.timestamp = time(NULL);
+
+ dbeo.cbBlob = ReplyLen;
+ dbeo.pBlob = pBuf;
+
+ SleepEx(1000, true); // delay before sending the reply, as we need it to be later than the message we're replying to (without this delay, srmm puts the messages in a wrong order)
+ db_event_add(ad->hContact, &dbeo);
+ }
+
+ free(pBuf);
+ delete ad;
+}
+
+
+int IsSRMsgWindowOpen(MCONTACT hContact, int DefaultRetVal)
+{
+ if (ServiceExists(MS_MSG_GETWINDOWDATA))
+ {
+ MessageWindowData mwd = {0};
+ mwd.cbSize = sizeof(mwd);
+ MessageWindowInputData mwid = {0};
+ mwid.cbSize = sizeof(mwid);
+ mwid.uFlags = MSG_WINDOW_UFLAG_MSG_BOTH;
+ mwid.hContact = hContact;
+ return !CallService(MS_MSG_GETWINDOWDATA, (WPARAM)&mwid, (LPARAM)&mwd) && mwd.hwndWindow;
+ } else
+ {
+ return DefaultRetVal;
+ }
+}
+
+
+#define MAX_REPLY_TIMEDIFF 5 // maximum "age" of an event to remain unfiltered; in seconds
+#define MSGWNDOPEN_UNDEFINED (-1)
+
+class CMetacontactEvent
+{
+public:
+ CMetacontactEvent(MCONTACT hMetaContact, DWORD timestamp, int bMsgWindowIsOpen): hMetaContact(hMetaContact), timestamp(timestamp), bMsgWindowIsOpen(bMsgWindowIsOpen) {};
+
+ MCONTACT hMetaContact;
+ DWORD timestamp;
+ int bMsgWindowIsOpen;
+};
+
+TMyArray<CMetacontactEvent> MetacontactEvents;
+
+
+int MsgEventAdded(WPARAM hContact, LPARAM lParam)
+{
+ DBEVENTINFO *dbei = (DBEVENTINFO*)lParam;
+ if (!hContact)
+ return 0;
+
+ if (dbei->flags & DBEF_SENT || (dbei->eventType != EVENTTYPE_MESSAGE && dbei->eventType != EVENTTYPE_URL && dbei->eventType != EVENTTYPE_FILE))
+ return 0;
+
+ if (time(NULL) - dbei->timestamp > MAX_REPLY_TIMEDIFF) // don't reply to offline messages
+ return 0;
+
+ char *szProto = GetContactProto(hContact);
+ if (!szProto)
+ return 0;
+
+ DWORD Flags1 = CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_1, 0);
+ if (!(Flags1 & PF1_IMSEND)) // don't reply to protocols that don't support outgoing messages
+ return 0;
+
+ int bMsgWindowIsOpen = MSGWNDOPEN_UNDEFINED;
+ if (dbei->flags & DBEF_READ) {
+ MCONTACT hMetaContact;
+ if (ServiceExists(MS_MC_GETMETACONTACT) && (hMetaContact = CallService(MS_MC_GETMETACONTACT, (WPARAM)hContact, 0))) // if it's a subcontact of a metacontact
+ { // ugly workaround for metacontacts, part II
+ // remove outdated events first
+ DWORD CurTime = time(NULL);
+ int I;
+ for (I = MetacontactEvents.GetSize() - 1; I >= 0; I--)
+ {
+ if (CurTime - MetacontactEvents[I].timestamp > MAX_REPLY_TIMEDIFF)
+ {
+ MetacontactEvents.RemoveElem(I);
+ }
+ }
+ // we compare only event timestamps, and do not look at the message itself. it's unlikely that there'll be two events from a contact at the same second, so it's a trade-off between speed and reliability
+ for (I = MetacontactEvents.GetSize() - 1; I >= 0; I--)
+ {
+ if (MetacontactEvents[I].timestamp == dbei->timestamp && MetacontactEvents[I].hMetaContact == hMetaContact)
+ {
+ bMsgWindowIsOpen = MetacontactEvents[I].bMsgWindowIsOpen;
+ break;
+ }
+ }
+ if (I < 0)
+ {
+ _ASSERT(0);
+ return 0;
+ }
+ } else
+ {
+ return 0;
+ }
+ }
+ if (ServiceExists(MS_MC_GETPROTOCOLNAME) && !lstrcmpA(szProto, (char*)CallService(MS_MC_GETPROTOCOLNAME, 0, 0)))
+ { // ugly workaround for metacontacts, part I; store all metacontacts' events to a temporary array, so we'll be able to get the 'source' protocol when subcontact event happens later. we need the protocol to get its status and per-status settings properly
+ // remove outdated events first
+ DWORD CurTime = time(NULL);
+ int I;
+ for (I = MetacontactEvents.GetSize() - 1; I >= 0; I--)
+ {
+ if (CurTime - MetacontactEvents[I].timestamp > MAX_REPLY_TIMEDIFF)
+ {
+ MetacontactEvents.RemoveElem(I);
+ }
+ }
+ // add the new event and wait for a subcontact's event
+ MetacontactEvents.AddElem(CMetacontactEvent(hContact, dbei->timestamp, IsSRMsgWindowOpen(hContact, false)));
+ return 0;
+ }
+ unsigned int iMode = CallProtoService(szProto, PS_GETSTATUS, 0, 0);
+ int I;
+ for (I = lengthof(StatusModeList) - 1; I >= 0; I--)
+ {
+ if (iMode == StatusModeList[I].Status)
+ {
+ break;
+ }
+ }
+ if (I < 0)
+ {
+ return 0;
+ }
+ COptPage AutoreplyOptData(g_AutoreplyOptPage);
+ AutoreplyOptData.DBToMem();
+ if (dbei->eventType == EVENTTYPE_MESSAGE)
+ {
+ db_set_w(hContact, MOD_NAME, DB_MESSAGECOUNT, db_get_w(hContact, MOD_NAME, DB_MESSAGECOUNT, 0) + 1); // increment message counter
+ }
+ if (AutoreplyOptData.GetValue(StatusModeList[I].DisableReplyCtlID))
+ {
+ return 0;
+ }
+ MCONTACT hContactForSettings = hContact; // used to take into account not-on-list contacts when getting contact settings, but at the same time allows to get correct contact info for contacts that are in the DB
+ if (hContactForSettings != INVALID_CONTACT_ID && db_get_b(hContactForSettings, "CList", "NotOnList", 0))
+ {
+ hContactForSettings = INVALID_CONTACT_ID; // INVALID_HANDLE_VALUE means the contact is not-on-list
+ }
+ if (!CContactSettings(iMode, hContactForSettings).Autoreply.IncludingParents(szProto) || CContactSettings(iMode, hContactForSettings).Ignore)
+ {
+ return 0;
+ }
+ if (AutoreplyOptData.GetValue(IDC_REPLYDLG_DONTREPLYINVISIBLE))
+ {
+ WORD ApparentMode = db_get_w(hContact, szProto, "ApparentMode", 0);
+ if ((iMode == ID_STATUS_INVISIBLE && (!(Flags1 & PF1_INVISLIST) || ApparentMode != ID_STATUS_ONLINE)) ||
+ (Flags1 & PF1_VISLIST && ApparentMode == ID_STATUS_OFFLINE))
+ {
+ return 0;
+ }
+ }
+ if (AutoreplyOptData.GetValue(IDC_REPLYDLG_ONLYCLOSEDDLGREPLY))
+ {
+ if (bMsgWindowIsOpen && bMsgWindowIsOpen != MSGWNDOPEN_UNDEFINED)
+ {
+ return 0;
+ }
+ // we never get here for a metacontact; we did check for metacontact's window earlier, and here we need to check only for subcontact's window
+ if (IsSRMsgWindowOpen(hContact, false))
+ {
+ return 0;
+ }
+ }
+ if (AutoreplyOptData.GetValue(IDC_REPLYDLG_ONLYIDLEREPLY) && !g_bIsIdle)
+ {
+ return 0;
+ }
+ int UIN = 0;
+ if (IsAnICQProto(szProto))
+ {
+ UIN = db_get_dw(hContact, szProto, "UIN", 0);
+ }
+ int SendCount = AutoreplyOptData.GetValue(IDC_REPLYDLG_SENDCOUNT);
+ if ((AutoreplyOptData.GetValue(IDC_REPLYDLG_DONTSENDTOICQ) && UIN) || // an icq contact
+ (SendCount != -1 && db_get_b(hContact, MOD_NAME, DB_SENDCOUNT, 0) >= SendCount))
+ {
+ return 0;
+ }
+ if ((dbei->eventType == EVENTTYPE_MESSAGE && !AutoreplyOptData.GetValue(IDC_REPLYDLG_EVENTMSG)) || (dbei->eventType == EVENTTYPE_URL && !AutoreplyOptData.GetValue(IDC_REPLYDLG_EVENTURL)) || (dbei->eventType == EVENTTYPE_FILE && !AutoreplyOptData.GetValue(IDC_REPLYDLG_EVENTFILE)))
+ {
+ return 0;
+ }
+ db_set_b(hContact, MOD_NAME, DB_SENDCOUNT, db_get_b(hContact, MOD_NAME, DB_SENDCOUNT, 0) + 1);
+ GetDynamicStatMsg(hContact); // it updates VarParseData.Message needed for %extratext% in the format
+ TCString Reply(*(TCString*)AutoreplyOptData.GetValue(IDC_REPLYDLG_PREFIX));
+ if (Reply != NULL && ServiceExists(MS_VARS_FORMATSTRING) && !g_SetAwayMsgPage.GetDBValueCopy(IDS_SAWAYMSG_DISABLEVARIABLES))
+ {
+ FORMATINFO fi = {0};
+ fi.cbSize = sizeof(FORMATINFO);
+ fi.tszFormat = Reply;
+ fi.hContact = hContact;
+ fi.flags = FIF_TCHAR;
+ fi.tszExtraText = VarParseData.Message;
+ TCHAR *szResult = (TCHAR *)CallService(MS_VARS_FORMATSTRING, (WPARAM)&fi, 0);
+ if (szResult != NULL)
+ {
+ Reply = szResult;
+ mir_free(szResult);
+ }
+ }
+ if (Reply.GetLen())
+ {
+ CAutoreplyData *ad = new CAutoreplyData(hContact, Reply);
+ mir_forkthread(AutoreplyDelayThread, ad);
+ }
+ return 0;
+}
diff --git a/plugins/NewAwaySysMod/src/MsgTree.cpp b/plugins/NewAwaySysMod/src/MsgTree.cpp
new file mode 100644
index 0000000000..e53ea75cfc
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/MsgTree.cpp
@@ -0,0 +1,790 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "MsgTree.h"
+#include "Properties.h"
+
+#define UM_MSGTREE_INIT (WM_USER + 0x2103)
+#define UM_MSGTREE_UPDATE (WM_USER + 0x2104)
+
+#define MSGTREE_TIMER_ID 0x2103
+#define MSGTREE_DRAGANDDROP_GROUPEXPANDTIME 600
+
+#define IMGLIST_NEWMESSAGE 0
+#define IMGLIST_NEWCATEGORY 1
+#define IMGLIST_DELETE 2
+
+struct {
+ int DBSetting, Status, MenuItemID;
+} StatusModeList[] = {
+ IDS_MESSAGEDLG_DEF_ONL, ID_STATUS_ONLINE, IDR_MSGTREEMENU_DEF_ONL,
+ IDS_MESSAGEDLG_DEF_AWAY, ID_STATUS_AWAY, IDR_MSGTREEMENU_DEF_AWAY,
+ IDS_MESSAGEDLG_DEF_NA, ID_STATUS_NA, IDR_MSGTREEMENU_DEF_NA,
+ IDS_MESSAGEDLG_DEF_OCC, ID_STATUS_OCCUPIED, IDR_MSGTREEMENU_DEF_OCC,
+ IDS_MESSAGEDLG_DEF_DND, ID_STATUS_DND, IDR_MSGTREEMENU_DEF_DND,
+ IDS_MESSAGEDLG_DEF_FFC, ID_STATUS_FREECHAT, IDR_MSGTREEMENU_DEF_FFC,
+ IDS_MESSAGEDLG_DEF_INV, ID_STATUS_INVISIBLE, IDR_MSGTREEMENU_DEF_INV,
+ IDS_MESSAGEDLG_DEF_OTP, ID_STATUS_ONTHEPHONE, IDR_MSGTREEMENU_DEF_OTP,
+ IDS_MESSAGEDLG_DEF_OTL, ID_STATUS_OUTTOLUNCH, IDR_MSGTREEMENU_DEF_OTL
+};
+
+static HANDLE hMTWindowList;
+static WNDPROC g_OrigEditProc;
+
+
+void LoadMsgTreeModule()
+{
+ hMTWindowList = (HANDLE)CallService(MS_UTILS_ALLOCWINDOWLIST, 0, 0);
+}
+
+
+static LRESULT CALLBACK EditSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ switch (Msg)
+ {
+ case WM_GETDLGCODE:
+ {
+ return CallWindowProc(g_OrigEditProc, hWnd, Msg, wParam, lParam) | DLGC_WANTALLKEYS;
+ } break;
+ }
+ return CallWindowProc(g_OrigEditProc, hWnd, Msg, wParam, lParam);
+}
+
+static LRESULT CALLBACK ParentSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ CMsgTree *dat = CWndUserData(hWnd).GetMsgTree();
+ switch (Msg)
+ {
+ case WM_NOTIFY:
+ {
+ if (((LPNMHDR)lParam)->hwndFrom == dat->hTreeView)
+ {
+ switch (((LPNMHDR)lParam)->code)
+ {
+ case TVN_BEGINDRAGA:
+ case TVN_BEGINDRAGW:
+ {
+ LPNMTREEVIEW pnmtv = (LPNMTREEVIEW)lParam;
+ NMMSGTREE nm = {0};
+ COptItem_TreeCtrl *TreeCtrl = dat->GetTreeCtrl();
+ int Order = TreeCtrl->hItemToOrder(pnmtv->itemNew.hItem);
+ _ASSERT(Order != -1);
+ if (Order != -1)
+ {
+ nm.ItemOld = (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ nm.hdr.code = MTN_BEGINDRAG;
+ nm.hdr.hwndFrom = dat->hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(dat->hTreeView);
+ if (!SendMessage(hWnd, WM_NOTIFY, 0, (LPARAM)&nm))
+ {
+ SetCapture(hWnd);
+ dat->hPrevDropTarget = dat->hDragItem = pnmtv->itemNew.hItem;
+ SetFocus(dat->hTreeView);
+ TreeView_SelectItem(dat->hTreeView, dat->hDragItem);
+ }
+ }
+ } break;
+ case TVN_SELCHANGEDA:
+ case TVN_SELCHANGEDW:
+ {
+ if (dat->UpdateLock)
+ {
+ return 0;
+ }
+ LPNMTREEVIEW pnmtv = (LPNMTREEVIEW)lParam;
+ NMMSGTREE nm = {0};
+ COptItem_TreeCtrl *TreeCtrl = dat->GetTreeCtrl();
+ if (pnmtv->itemOld.hItem)
+ {
+ int Order = TreeCtrl->IDToOrder(pnmtv->itemOld.lParam);
+ if (Order != -1)
+ {
+ nm.ItemOld = (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ }
+ }
+ if (pnmtv->itemNew.hItem)
+ {
+ int Order = TreeCtrl->IDToOrder(pnmtv->itemNew.lParam);
+ if (Order != -1)
+ {
+ nm.ItemNew = (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ }
+ }
+ nm.hdr.code = MTN_SELCHANGED;
+ nm.hdr.hwndFrom = dat->hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(dat->hTreeView);
+ SendMessage(hWnd, WM_NOTIFY, 0, (LPARAM)&nm);
+ } break;
+ case TVN_BEGINLABELEDITA:
+ case TVN_BEGINLABELEDITW:
+ {
+ if (dat->GetTreeCtrl()->IDToOrder(((LPNMTVDISPINFO)lParam)->item.lParam) < 0)
+ {
+ return true; // cancel editing
+ }
+ g_OrigEditProc = (WNDPROC)SetWindowLongPtr(TreeView_GetEditControl(dat->hTreeView), GWLP_WNDPROC, (LONG_PTR)EditSubclassProc);
+ } break;
+// case TVN_ENDLABELEDITA: // stupid miranda options.. how am I supposed to get ptvdi->item.pszText if it's in ANSI??
+ case TVN_ENDLABELEDIT:
+ {
+ LPNMTVDISPINFO ptvdi = (LPNMTVDISPINFO)lParam;
+ if (ptvdi->item.pszText)
+ {
+ COptItem_TreeCtrl *TreeCtrl = dat->GetTreeCtrl();
+ int Order = TreeCtrl->IDToOrder(ptvdi->item.lParam);
+ if (Order >= 0)
+ {
+ TreeCtrl->Value[Order].Title = ptvdi->item.pszText;
+ TreeCtrl->SetModified(true);
+ NMMSGTREE nm = {0};
+ nm.ItemNew = &TreeCtrl->Value[Order];
+ nm.hdr.code = MTN_ITEMRENAMED;
+ nm.hdr.hwndFrom = dat->hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(dat->hTreeView);
+ SendMessage(GetParent(dat->hTreeView), WM_NOTIFY, 0, (LPARAM)&nm);
+ return true; // commit new text
+ }
+ }
+ } break;
+ case NM_CLICK:
+ case NM_RCLICK:
+ {
+ TVHITTESTINFO hitTest;
+ hitTest.pt.x = (short)LOWORD(GetMessagePos());
+ hitTest.pt.y = (short)HIWORD(GetMessagePos());
+ ScreenToClient(dat->hTreeView, &hitTest.pt);
+ TreeView_HitTest(dat->hTreeView, &hitTest);
+ if (hitTest.hItem)
+ {
+ if (TreeView_GetSelection(dat->hTreeView) == hitTest.hItem)
+ { // make sure TVN_SELCHANGED notification is sent always, even if previous selected item was the same as new
+ TreeView_SelectItem(dat->hTreeView, NULL);
+ }
+ TreeView_SelectItem(dat->hTreeView, hitTest.hItem);
+ }
+ } break;
+ case NM_CUSTOMDRAW:
+ {
+ NMTVCUSTOMDRAW *lpNMCD = (NMTVCUSTOMDRAW*)lParam;
+ switch (lpNMCD->nmcd.dwDrawStage)
+ {
+ case CDDS_PREPAINT: // the control is about to start painting
+ {
+ return CDRF_NOTIFYITEMDRAW; // instruct the control to return information when it draws items
+ } break;
+ case CDDS_ITEMPREPAINT:
+ {
+ return CDRF_NOTIFYPOSTPAINT;
+ } break;
+ case CDDS_ITEMPOSTPAINT:
+ {
+ RECT rc;
+ TreeView_GetItemRect(lpNMCD->nmcd.hdr.hwndFrom, (HTREEITEM)lpNMCD->nmcd.dwItemSpec, &rc, true);
+ int iSize = GetSystemMetrics(SM_CXSMICON);
+ int I;
+ int x = rc.left - iSize - 5;
+ for (I = 0; I < lengthof(StatusModeList); I++)
+ {
+ if (lpNMCD->nmcd.lItemlParam == dat->MsgTreePage.GetValue(StatusModeList[I].DBSetting))
+ {
+ DrawIconEx(lpNMCD->nmcd.hdc, x, rc.top, LoadSkinnedProtoIcon(NULL, StatusModeList[I].Status), iSize, iSize, 0, GetSysColorBrush(COLOR_WINDOW), DI_NORMAL);
+ x -= iSize + 1;
+ }
+/* if (lpNMCD->nmcd.lItemlParam == GetRecentGroupID(StatusModeList[I].Status))
+ {
+ DrawIconEx(lpNMCD->nmcd.hdc, 3, rc.top, LoadSkinnedProtoIcon(NULL, StatusModeList[I].Status), iSize, iSize, 0, GetSysColorBrush(COLOR_WINDOW), DI_NORMAL);
+ }*/
+ }
+ } break;
+ }
+ } break;
+ }
+ }
+ } break;
+ case WM_MOUSEMOVE:
+ {
+ if (dat->hDragItem)
+ {
+ TVHITTESTINFO hti;
+ hti.pt.x = (short)LOWORD(lParam);
+ hti.pt.y = (short)HIWORD(lParam);
+ ClientToScreen(hWnd, &hti.pt);
+ ScreenToClient(dat->hTreeView, &hti.pt);
+ TreeView_HitTest(dat->hTreeView, &hti);
+ if (hti.hItem)
+ {
+ TreeView_SelectDropTarget(dat->hTreeView, hti.hItem);
+ SetTimer(hWnd, MSGTREE_TIMER_ID, MSGTREE_DRAGANDDROP_GROUPEXPANDTIME, NULL);
+ } else
+ {
+ if (hti.flags & TVHT_ABOVE)
+ {
+ SendMessage(dat->hTreeView, WM_VSCROLL, MAKEWPARAM(SB_LINEUP, 0), 0);
+ }
+ if (hti.flags & TVHT_BELOW)
+ {
+ SendMessage(dat->hTreeView, WM_VSCROLL, MAKEWPARAM(SB_LINEDOWN, 0), 0);
+ }
+ TreeView_SelectDropTarget(dat->hTreeView, NULL);
+ KillTimer(hWnd, MSGTREE_TIMER_ID);
+ }
+ }
+ } break;
+ case WM_LBUTTONUP:
+ {
+ if (dat->hDragItem)
+ {
+ TreeView_SelectDropTarget(dat->hTreeView, NULL);
+ KillTimer(hWnd, MSGTREE_TIMER_ID);
+ ReleaseCapture();
+ TVHITTESTINFO hti;
+ hti.pt.x = (short)LOWORD(lParam);
+ hti.pt.y = (short)HIWORD(lParam);
+ ClientToScreen(hWnd, &hti.pt);
+ ScreenToClient(dat->hTreeView, &hti.pt);
+ TreeView_HitTest(dat->hTreeView, &hti);
+ if (hti.hItem && dat->hDragItem != hti.hItem)
+ {
+ NMMSGTREE nm = {0};
+ COptItem_TreeCtrl *TreeCtrl = dat->GetTreeCtrl();
+ int OrderOld = TreeCtrl->hItemToOrder(dat->hDragItem);
+ int OrderNew = TreeCtrl->hItemToOrder(hti.hItem);
+ _ASSERT(OrderOld != -1 && OrderNew != -1);
+ nm.ItemOld = (OrderOld <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(OrderOld)] : (CBaseTreeItem*)&TreeCtrl->Value[OrderOld];
+ nm.ItemNew = (OrderNew <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(OrderNew)] : (CBaseTreeItem*)&TreeCtrl->Value[OrderNew];
+ nm.hdr.code = MTN_ENDDRAG;
+ nm.hdr.hwndFrom = dat->hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(dat->hTreeView);
+ if (!SendMessage(hWnd, WM_NOTIFY, 0, (LPARAM)&nm))
+ {
+ dat->UpdateLock++;
+ dat->GetTreeCtrl()->MoveItem(hWnd, dat->hDragItem, hti.hItem);
+ dat->UpdateLock--;
+ }
+ }
+ dat->hDragItem = NULL;
+ }
+ } break;
+ case WM_TIMER:
+ {
+ if (wParam == MSGTREE_TIMER_ID)
+ {
+ KillTimer(hWnd, MSGTREE_TIMER_ID);
+ TVHITTESTINFO hti;
+ hti.pt.x = (short)LOWORD(GetMessagePos());
+ hti.pt.y = (short)HIWORD(GetMessagePos());
+ ScreenToClient(dat->hTreeView, &hti.pt);
+ TreeView_HitTest(dat->hTreeView, &hti);
+ if (hti.hItem && dat->hDragItem != hti.hItem && TreeView_GetChild(dat->hTreeView, hti.hItem)) // target is a group and is not the same item that we're dragging
+ {
+ TreeView_Expand(dat->hTreeView, hti.hItem, TVE_EXPAND);
+ }
+ }
+ } break;
+ }
+ return CallWindowProc(dat->OrigParentProc, hWnd, Msg, wParam, lParam);
+}
+
+
+static LRESULT CALLBACK MsgTreeSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ CMsgTree *dat = CWndUserData(GetParent(hWnd)).GetMsgTree();
+ switch (Msg)
+ {
+ case UM_MSGTREE_UPDATE: // returns TRUE if updated
+ {
+ bool Modified = dat->MsgTreePage.GetModified();
+ TCString WndTitle;
+ if (Modified)
+ {
+ WndTitle.GetBuffer(256);
+ HWND hCurWnd = hWnd;
+ do
+ {
+ hCurWnd = GetParent(hCurWnd);
+ } while (hCurWnd && !GetWindowText(hCurWnd, WndTitle, 256));
+ WndTitle.ReleaseBuffer();
+ }
+ if (!Modified || MessageBox(GetParent(hWnd), TCString(TranslateT("You've made changes to multiple Message trees at a time.\r\nDo you want to leave changes in \"")) + WndTitle + TranslateT("\" dialog?\r\nPress Yes to leave changes in this dialog, or No to discard its changes and save changes of the other Message tree instead."), WndTitle + _T(" - ") + TranslateT("New Away System"), MB_ICONQUESTION | MB_YESNO) == IDNO)
+ {
+ COptItem_TreeCtrl *TreeCtrl = dat->GetTreeCtrl();
+ TCString OldTitle, OldMsg, NewTitle, NewMsg;
+ int OldOrder = TreeCtrl->IDToOrder(TreeCtrl->GetSelectedItemID(GetParent(hWnd)));
+ if (OldOrder != -1)
+ {
+ CBaseTreeItem* ItemOld = (OldOrder <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(OldOrder)] : (CBaseTreeItem*)&TreeCtrl->Value[OldOrder];
+ OldTitle = ItemOld->Title;
+ if (!(ItemOld->Flags & TIF_ROOTITEM))
+ {
+ OldMsg = ((CTreeItem*)ItemOld)->User_Str1;
+ }
+ }
+ dat->UpdateLock++;
+ dat->MsgTreePage.DBToMemToPage();
+ dat->UpdateLock--;
+ NMMSGTREE nm = {0};
+ int Order = TreeCtrl->IDToOrder(TreeCtrl->GetSelectedItemID(GetParent(hWnd)));
+ if (Order != -1)
+ {
+ nm.ItemNew = (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ NewTitle = nm.ItemNew->Title;
+ if (!(nm.ItemNew->Flags & TIF_ROOTITEM))
+ {
+ NewMsg = ((CTreeItem*)nm.ItemNew)->User_Str1;
+ }
+ }
+ if (OldTitle.IsEmpty())
+ {
+ OldTitle = _T(""); // to be sure that NULL will be equal to "" in the latter comparisons
+ }
+ if (OldMsg.IsEmpty())
+ {
+ OldMsg = _T("");
+ }
+ if (NewTitle.IsEmpty())
+ {
+ NewTitle = _T("");
+ }
+ if (NewMsg.IsEmpty())
+ {
+ NewMsg = _T("");
+ }
+ if (OldTitle != (const TCHAR*)NewTitle || OldMsg != (const TCHAR*)NewMsg)
+ {
+ // probably it's better to leave nm.ItemOld = NULL, to prevent accidental rewriting of it with old data from an edit control etc.
+ nm.hdr.code = MTN_SELCHANGED;
+ nm.hdr.hwndFrom = hWnd;
+ nm.hdr.idFrom = GetDlgCtrlID(hWnd);
+ SendMessage(GetParent(hWnd), WM_NOTIFY, 0, (LPARAM)&nm);
+ }
+ return true;
+ }
+ return false;
+ } break;
+ case WM_KEYDOWN:
+ {
+ switch (wParam)
+ {
+ case VK_DELETE:
+ {
+ dat->DeleteSelectedItem();
+ } break;
+ case VK_INSERT:
+ {
+ dat->AddMessage();
+ } break;
+ }
+ } break;
+ case WM_RBUTTONDOWN:
+ {
+ SetFocus(hWnd);
+ TVHITTESTINFO hitTest;
+ hitTest.pt.x = (short)LOWORD(lParam);
+ hitTest.pt.y = (short)HIWORD(lParam);
+ TreeView_HitTest(hWnd, &hitTest);
+ if (hitTest.hItem && hitTest.flags & TVHT_ONITEM)
+ {
+ TreeView_SelectItem(hWnd, hitTest.hItem);
+ }
+ return DefWindowProc(hWnd, Msg, wParam, lParam);
+ } break;
+ case WM_CONTEXTMENU:
+ {
+ TVHITTESTINFO ht;
+ ht.pt.x = (short)LOWORD(lParam);
+ ht.pt.y = (short)HIWORD(lParam);
+ TVITEM tvi = {0};
+ if (ht.pt.x == -1 && ht.pt.y == -1) // use selected item
+ {
+ if (tvi.hItem = TreeView_GetSelection(hWnd))
+ {
+ TreeView_EnsureVisible(hWnd, tvi.hItem);
+ RECT rc;
+ TreeView_GetItemRect(hWnd, tvi.hItem, &rc, true);
+ ht.pt.x = rc.left;
+ ht.pt.y = rc.bottom;
+ }
+ } else
+ {
+ ScreenToClient(hWnd, &ht.pt);
+ TreeView_HitTest(hWnd, &ht);
+ if (ht.hItem && ht.flags & TVHT_ONITEM)
+ {
+ tvi.hItem = ht.hItem;
+ }
+ }
+ if (tvi.hItem)
+ {
+ COptItem_TreeCtrl *TreeCtrl = dat->GetTreeCtrl();
+ tvi.mask = TVIF_HANDLE | TVIF_PARAM;
+ TreeView_GetItem(hWnd, &tvi);
+ int Order = TreeCtrl->IDToOrder(tvi.lParam);
+ if (Order >= 0)
+ {
+ HMENU hMenu;
+ if (TreeCtrl->Value[Order].Flags & TIF_GROUP)
+ {
+ hMenu = LoadMenu(g_hInstance, MAKEINTRESOURCE(IDR_MSGTREE_CATEGORYMENU));
+ } else
+ {
+ hMenu = LoadMenu(g_hInstance, MAKEINTRESOURCE(IDR_MSGTREE_MESSAGEMENU));
+ }
+ _ASSERT(hMenu);
+ HMENU hPopupMenu = GetSubMenu(hMenu, 0);
+ TranslateMenu(hPopupMenu);
+ ClientToScreen(hWnd, &ht.pt);
+ struct
+ {
+ int ItemID, IconID;
+ } MenuItems[] = {
+ IDM_MSGTREEMENU_NEWMESSAGE, IMGLIST_NEWMESSAGE,
+ IDM_MSGTREEMENU_NEWCATEGORY, IMGLIST_NEWCATEGORY,
+ IDM_MSGTREEMENU_DELETE, IMGLIST_DELETE
+ };
+ MENUITEMINFO mii = {0};
+ mii.cbSize = sizeof(mii);
+ mii.fMask = MIIM_BITMAP | MIIM_DATA | MIIM_STATE | MIIM_CHECKMARKS;
+ mii.hbmpItem = HBMMENU_CALLBACK;
+ int I;
+ for (I = 0; I < lengthof(MenuItems); I++) // set icons
+ {
+ mii.dwItemData = MenuItems[I].IconID;
+ SetMenuItemInfo(hPopupMenu, MenuItems[I].ItemID, false, &mii);
+ }
+ mii.fMask = MIIM_STATE;
+ mii.fState = MFS_CHECKED;
+ for (I = 0; I < lengthof(StatusModeList); I++) // set checkmarks
+ {
+ if (TreeCtrl->Value[Order].ID == dat->MsgTreePage.GetValue(StatusModeList[I].DBSetting))
+ {
+ SetMenuItemInfo(hPopupMenu, StatusModeList[I].MenuItemID, false, &mii);
+ }
+ }
+ int MenuResult = TrackPopupMenu(hPopupMenu, TPM_RIGHTBUTTON | TPM_RETURNCMD, ht.pt.x, ht.pt.y, 0, hWnd, NULL);
+ switch (MenuResult)
+ {
+ case IDM_MSGTREEMENU_NEWMESSAGE:
+ {
+ dat->AddMessage();
+ } break;
+ case IDM_MSGTREEMENU_NEWCATEGORY:
+ {
+ dat->AddCategory();
+ } break;
+ case IDM_MSGTREEMENU_RENAME:
+ {
+ TreeView_EditLabel(hWnd, tvi.hItem);
+ } break;
+ case IDM_MSGTREEMENU_DELETE:
+ {
+ dat->DeleteSelectedItem();
+ } break;
+ case IDR_MSGTREEMENU_DEF_ONL:
+ case IDR_MSGTREEMENU_DEF_AWAY:
+ case IDR_MSGTREEMENU_DEF_NA:
+ case IDR_MSGTREEMENU_DEF_OCC:
+ case IDR_MSGTREEMENU_DEF_DND:
+ case IDR_MSGTREEMENU_DEF_FFC:
+ case IDR_MSGTREEMENU_DEF_INV:
+ case IDR_MSGTREEMENU_DEF_OTP:
+ case IDR_MSGTREEMENU_DEF_OTL:
+ {
+ int I;
+ for (I = 0; I < lengthof(StatusModeList); I++)
+ {
+ if (StatusModeList[I].MenuItemID == MenuResult)
+ {
+ dat->SetDefMsg(StatusModeList[I].Status, tvi.lParam);
+ break;
+ }
+ }
+ } break;
+ }
+ DestroyMenu(hMenu);
+ return 0;
+ }
+ }
+ } break;
+ case WM_MEASUREITEM:
+ {
+ LPMEASUREITEMSTRUCT lpmi = (LPMEASUREITEMSTRUCT)lParam;
+ if (lpmi->CtlType == ODT_MENU)
+ {
+ lpmi->itemWidth = max(0, GetSystemMetrics(SM_CXSMICON) - GetSystemMetrics(SM_CXMENUCHECK) + 4);
+ lpmi->itemHeight = GetSystemMetrics(SM_CYSMICON) + 2;
+ return true;
+ }
+ } break;
+ case WM_DRAWITEM:
+ {
+ LPDRAWITEMSTRUCT dis = (LPDRAWITEMSTRUCT)lParam;
+ if (dis->CtlType == ODT_MENU)
+ {
+ ImageList_DrawEx(dat->hImageList, dis->itemData, dis->hDC, 2, (dis->rcItem.bottom + dis->rcItem.top - GetSystemMetrics(SM_CYSMICON)) / 2 + 1, 0, 0, GetSysColor(COLOR_WINDOW), CLR_NONE, ILD_NORMAL);
+ return true;
+ }
+ } break;
+ }
+ return CallWindowProc(dat->OrigTreeViewProc, hWnd, Msg, wParam, lParam);
+}
+
+
+CMsgTree::CMsgTree(HWND hTreeView): MsgTreePage(g_MsgTreePage), hTreeView(hTreeView), hDragItem(NULL), hPrevDropTarget(NULL), UpdateLock(0)
+{
+ CWndUserData(GetParent(hTreeView)).SetMsgTree(this);
+ OrigParentProc = (WNDPROC)SetWindowLongPtr(GetParent(hTreeView), GWLP_WNDPROC, (LONG_PTR)ParentSubclassProc);
+ OrigTreeViewProc = (WNDPROC)SetWindowLongPtr(hTreeView, GWLP_WNDPROC, (LONG_PTR)MsgTreeSubclassProc);
+ MsgTreePage.SetWnd(GetParent(hTreeView));
+ COptItem_TreeCtrl* TreeCtrl = (COptItem_TreeCtrl*)MsgTreePage.Find(IDV_MSGTREE);
+ TreeCtrl->SetDlgItemID(GetDlgCtrlID(hTreeView));
+ hImageList = ImageList_Create(GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON), ILC_COLOR32 | ILC_MASK, 5, 2);
+ ImageList_AddIcon(hImageList, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_NEWMESSAGE)));
+ ImageList_AddIcon(hImageList, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_NEWCATEGORY)));
+ ImageList_AddIcon(hImageList, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_DELETE)));
+ MsgTreePage.DBToMemToPage();
+ if (!g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_RECENTMSGSCOUNT))
+ { // show "Recent messages" group only when RECENTMSGSCOUNT is not set to 0.
+ TreeView_DeleteItem(hTreeView, TreeCtrl->RootItems[g_Messages_RecentRootID].hItem);
+ }
+ WindowList_Add(hMTWindowList, hTreeView, NULL);
+}
+
+CMsgTree::~CMsgTree()
+{
+ _ASSERT(GetWindowLongPtr(GetParent(hTreeView), GWLP_WNDPROC) == (LONG_PTR)ParentSubclassProc); // we won't allow anyone to change our WNDPROC. otherwise we're not sure that we're setting the right WNDPROC back
+ SetWindowLongPtr(hTreeView, GWLP_WNDPROC, (GetWindowLongPtr(GetParent(hTreeView), GWLP_WNDPROC) == (LONG_PTR)ParentSubclassProc) ? (LONG_PTR)OrigTreeViewProc : (LONG_PTR)DefDlgProc); // yeah, if that crazy Help plugin substituted MY WndProc again, he won't get his WndProc back.. he-he >:)
+ SetWindowLongPtr(GetParent(hTreeView), GWLP_WNDPROC, (LONG_PTR)OrigParentProc);
+ CWndUserData(GetParent(hTreeView)).SetMsgTree(NULL);
+ WindowList_Remove(hMTWindowList, hTreeView);
+ ImageList_Destroy(hImageList);
+}
+
+CBaseTreeItem* CMsgTree::GetSelection() // returns NULL if there's nothing selected
+{
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ int Order = TreeCtrl->IDToOrder(TreeCtrl->GetSelectedItemID(GetParent(hTreeView)));
+ if (Order != -1)
+ {
+ return (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ }
+ return NULL;
+}
+
+bool CMsgTree::SetSelection(int ID, int Flags) // set ID = -1 to unselect; returns TRUE on unselect and on successful select
+{
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ int Order = (Flags & MTSS_BYORDER) ? ID : TreeCtrl->IDToOrder(ID);
+ if (Order == -1 && ID != -1)
+ {
+ return false;
+ }
+ TreeView_SelectItem(hTreeView, (Order == -1) ? NULL : ((Order <= TREECTRL_ROOTORDEROFFS) ? TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)].hItem : TreeCtrl->Value[Order].hItem));
+ return true;
+}
+
+int CMsgTree::GetDefMsg(int iMode)
+{
+ int I;
+ for (I = 0; I < lengthof(StatusModeList); I++)
+ {
+ if (StatusModeList[I].Status == iMode)
+ {
+ return MsgTreePage.GetValue(StatusModeList[I].DBSetting);
+ }
+ }
+ return 0;
+}
+
+void CMsgTree::SetDefMsg(int iMode, int ID)
+{
+ int I;
+ for (I = 0; I < lengthof(StatusModeList); I++)
+ {
+ if (StatusModeList[I].Status == iMode)
+ {
+ if (MsgTreePage.GetValue(StatusModeList[I].DBSetting) != ID)
+ {
+ RECT rc;
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ int OrderOld = TreeCtrl->IDToOrder(MsgTreePage.GetValue(StatusModeList[I].DBSetting));
+ if (OrderOld >= 0 && TreeView_GetItemRect(hTreeView, TreeCtrl->Value[OrderOld].hItem, &rc, false))
+ {
+ InvalidateRect(hTreeView, &rc, true); // refresh icons of previous default tree item
+ }
+ int OrderNew = TreeCtrl->IDToOrder(ID);
+ if (OrderNew >= 0 && TreeView_GetItemRect(hTreeView, TreeCtrl->Value[OrderNew].hItem, &rc, false))
+ {
+ InvalidateRect(hTreeView, &rc, true); // refresh new default item icons
+ }
+ MsgTreePage.SetValue(StatusModeList[I].DBSetting, ID);
+ NMMSGTREE nm = {0};
+ if (OrderOld >= 0)
+ {
+ nm.ItemOld = &TreeCtrl->Value[OrderOld];
+ }
+ if (OrderNew >= 0)
+ {
+ nm.ItemNew = &TreeCtrl->Value[OrderNew];
+ }
+ nm.hdr.code = MTN_DEFMSGCHANGED;
+ nm.hdr.hwndFrom = hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(hTreeView);
+ SendMessage(GetParent(hTreeView), WM_NOTIFY, 0, (LPARAM)&nm);
+ }
+ break;
+ }
+ }
+}
+
+void CMsgTree::Save()
+{
+ if (MsgTreePage.GetModified())
+ {
+ MsgTreePage.PageToMemToDB();
+ WindowList_BroadcastAsync(hMTWindowList, UM_MSGTREE_UPDATE, 0, 0);
+ }
+}
+
+void CMsgTree::UpdateItem(int ID) // updates item title, and expanded/collapsed state for groups
+{
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ int Order = TreeCtrl->IDToOrder(ID);
+ if (Order != -1)
+ {
+ CBaseTreeItem* TreeItem = (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ TCString NewTitle;
+ TVITEM tvi;
+ tvi.mask = TVIF_HANDLE | TVIF_TEXT;
+ tvi.hItem = TreeItem->hItem;
+ tvi.pszText = NewTitle.GetBuffer(TREEITEMTITLE_MAXLEN);
+ tvi.cchTextMax = TREEITEMTITLE_MAXLEN;
+ TreeView_GetItem(hTreeView, &tvi);
+ if (TreeItem->Title != (const TCHAR*)tvi.pszText)
+ {
+ TreeCtrl->SetModified(true);
+ NMMSGTREE nm = {0};
+ nm.ItemNew = TreeItem;
+ nm.hdr.code = MTN_ITEMRENAMED;
+ nm.hdr.hwndFrom = hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(hTreeView);
+ SendMessage(GetParent(hTreeView), WM_NOTIFY, 0, (LPARAM)&nm);
+ }
+ tvi.mask = TVIF_HANDLE | TVIF_TEXT;
+ tvi.pszText = TreeItem->Title;
+ TreeView_SetItem(hTreeView, &tvi);
+ TreeView_Expand(hTreeView, tvi.hItem, (TreeItem->Flags & TIF_EXPANDED) ? TVE_EXPAND : TVE_COLLAPSE);
+ }
+}
+
+bool CMsgTree::DeleteSelectedItem() // returns true if the item was deleted
+{
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ int Order = TreeCtrl->IDToOrder(TreeCtrl->GetSelectedItemID(GetParent(hTreeView)));
+ _ASSERT(Order >= 0);
+ CTreeItem *SelectedItem = &TreeCtrl->Value[Order];
+ //NightFox: fix for langpack and fix cut char space in text
+ //if (MessageBox(GetParent(hTreeView), TCString(TranslateT("Do you really want to delete this ")) + ((SelectedItem->Flags & TIF_GROUP) ? TranslateT("category with its messages?") : TranslateT("message?")), TranslateT("New Away System"), MB_ICONQUESTION | MB_YESNO | MB_DEFBUTTON2) == IDYES)
+ if (MessageBox(GetParent(hTreeView),
+ ((SelectedItem->Flags & TIF_GROUP) ?
+ TranslateT("Do you really want to delete this category with its messages?")
+ :
+ TranslateT("Do you really want to delete this message?"))
+ ,
+ TranslateT("New Away System"), MB_ICONQUESTION | MB_YESNO | MB_DEFBUTTON2) == IDYES)
+ {
+ NMMSGTREE nm = {0};
+ nm.ItemOld = SelectedItem;
+ nm.hdr.code = MTN_DELETEITEM;
+ nm.hdr.hwndFrom = hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(hTreeView);
+ if (!SendMessage(GetParent(hTreeView), WM_NOTIFY, 0, (LPARAM)&nm))
+ {
+ TreeCtrl->Delete(GetParent(hTreeView), TreeCtrl->GetSelectedItemID(GetParent(hTreeView)));
+ return true;
+ }
+ }
+ return false;
+}
+
+CTreeItem* CMsgTree::AddCategory()
+{
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ CTreeItem* TreeItem = TreeCtrl->InsertItem(GetParent(hTreeView), CTreeItem(_T(""), 0, 0, TIF_GROUP));
+ TVITEM tvi;
+ tvi.mask = TVIF_HANDLE | TVIF_TEXT;
+ tvi.hItem = TreeItem->hItem;
+ TreeItem->Title = tvi.pszText = TranslateT("New category");
+ TreeView_SetItem(hTreeView, &tvi);
+ TreeView_EditLabel(hTreeView, TreeItem->hItem);
+ NMMSGTREE nm = {0};
+ nm.ItemNew = TreeItem;
+ nm.hdr.code = MTN_NEWCATEGORY;
+ nm.hdr.hwndFrom = hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(hTreeView);
+ SendMessage(GetParent(hTreeView), WM_NOTIFY, 0, (LPARAM)&nm);
+ return TreeItem;
+}
+
+CTreeItem* CMsgTree::AddMessage()
+{
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ CTreeItem* TreeItem = TreeCtrl->InsertItem(GetParent(hTreeView), CTreeItem(_T(""), 0, 0));
+ TVITEM tvi;
+ tvi.mask = TVIF_HANDLE | TVIF_TEXT;
+ tvi.hItem = TreeItem->hItem;
+ TreeItem->Title = tvi.pszText = TranslateT("New message");
+ TreeView_SetItem(hTreeView, &tvi);
+ TreeView_EditLabel(hTreeView, TreeItem->hItem);
+ NMMSGTREE nm = {0};
+ nm.ItemNew = TreeItem;
+ nm.hdr.code = MTN_NEWMESSAGE;
+ nm.hdr.hwndFrom = hTreeView;
+ nm.hdr.idFrom = GetDlgCtrlID(hTreeView);
+ SendMessage(GetParent(hTreeView), WM_NOTIFY, 0, (LPARAM)&nm);
+ return TreeItem;
+}
+
+CBaseTreeItem* CMsgTree::GetNextItem(int Flags, CBaseTreeItem* Item) // Item is 'int ID' if MTGN_BYID flag is set; returns CBaseTreeItem* or NULL
+{
+ COptItem_TreeCtrl *TreeCtrl = GetTreeCtrl();
+ CBaseTreeItem* TreeItem = Item;
+ if (Flags & MTGN_BYID)
+ {
+ int Order = TreeCtrl->IDToOrder((int)Item);
+ _ASSERT(Order != -1);
+ TreeItem = (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ }
+ int TVFlag = 0;
+ switch (Flags & ~MTGN_BYID)
+ {
+ case MTGN_ROOT: TVFlag = TVGN_ROOT; break;
+ case MTGN_CHILD: TVFlag = TVGN_CHILD; break;
+ case MTGN_PARENT: TVFlag = TVGN_PARENT; break;
+ case MTGN_NEXT: TVFlag = TVGN_NEXT; break;
+ case MTGN_PREV: TVFlag = TVGN_PREVIOUS; break;
+ default: _ASSERT(0);
+ }
+ int Order = TreeCtrl->hItemToOrder(TreeView_GetNextItem(hTreeView, TreeItem ? TreeItem->hItem : NULL, TVFlag));
+ if (Order != -1)
+ {
+ return (Order <= TREECTRL_ROOTORDEROFFS) ? (CBaseTreeItem*)&TreeCtrl->RootItems[ROOT_ORDER_TO_INDEX(Order)] : (CBaseTreeItem*)&TreeCtrl->Value[Order];
+ }
+ return NULL;
+}
diff --git a/plugins/NewAwaySysMod/src/MsgTree.h b/plugins/NewAwaySysMod/src/MsgTree.h
new file mode 100644
index 0000000000..98b2f9d6cc
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/MsgTree.h
@@ -0,0 +1,95 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#pragma once
+
+extern COptPage g_MsgTreePage;
+
+class CMsgTree
+{
+public:
+ CMsgTree(HWND hTreeView); // warning: it changes GWLP_USERDATA of the parent window
+ ~CMsgTree();
+
+ CBaseTreeItem* GetSelection(); // returns NULL if there's nothing selected
+ bool SetSelection(int ID, int Flags); // set ID = -1 to unselect; ID specifies an order of an item if Flags = MTSS_BYORDER
+ int GetDefMsg(int iMode);
+ void SetDefMsg(int iMode, int ID);
+ void Save();
+ void UpdateItem(int ID); // updates item title, and expanded/collapsed state for groups; set SetModified to TRUE to change Modified flag of the tree
+ bool DeleteSelectedItem(); // returns true if the item was deleted
+ CTreeItem* AddCategory();
+ CTreeItem* AddMessage();
+ CBaseTreeItem* GetNextItem(int Flags, CBaseTreeItem* Item); // Item is 'int ID' if MTGN_BYID flag is set; returns CBaseTreeItem* or NULL
+ void EnsureVisible(HTREEITEM hItem) {TreeView_EnsureVisible(hTreeView, hItem);}
+ bool GetModified() {return MsgTreePage.GetModified();}
+ void SetModified(bool Modified) {MsgTreePage.SetModified(Modified);}
+
+ friend LRESULT CALLBACK ParentSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam);
+ friend LRESULT CALLBACK MsgTreeSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam);
+
+private:
+ COptItem_TreeCtrl* GetTreeCtrl() {return (COptItem_TreeCtrl*)MsgTreePage.Find(GetDlgCtrlID(hTreeView));}
+
+ HWND hTreeView;
+ WNDPROC OrigTreeViewProc;
+ WNDPROC OrigParentProc;
+ COptPage MsgTreePage;
+ HTREEITEM hDragItem;
+ HTREEITEM hPrevDropTarget;
+ int UpdateLock;
+ HIMAGELIST hImageList;
+};
+
+// SetSelection constants
+#define MTSS_BYID 0
+#define MTSS_BYORDER 1
+
+// GetNextItem constants
+#define MTGN_ROOT 0
+#define MTGN_CHILD 1
+#define MTGN_PARENT 2
+#define MTGN_NEXT 3
+#define MTGN_PREV 4
+#define MTGN_BYID 0x8000 // means that lParam is ID of an item, not a pointer to its data
+
+
+// notifications
+
+typedef struct
+{
+ NMHDR hdr;
+ CBaseTreeItem* ItemOld;
+ CBaseTreeItem* ItemNew;
+} NMMSGTREE;
+typedef NMMSGTREE* PNMMSGTREE;
+
+#define MTN_FIRST (0U - 100U)
+
+#define MTN_SELCHANGED (MTN_FIRST - 0) // lParam = &NMMSGTREE; ItemOld and ItemNew contain old and new item info
+#define MTN_BEGINDRAG (MTN_FIRST - 1) // lParam = &NMMSGTREE; ItemOld points to the item info; return TRUE to cancel dragging
+#define MTN_ENDDRAG (MTN_FIRST - 2) // lParam = &NMMSGTREE; ItemOld points to the item info; ItemNew points to the item on which ItemOld was dropped; return TRUE to cancel moving
+#define MTN_NEWCATEGORY (MTN_FIRST - 3) // lParam = &NMMSGTREE; ItemNew points to new category info
+#define MTN_NEWMESSAGE (MTN_FIRST - 4) // lParam = &NMMSGTREE; ItemNew points to new message info
+#define MTN_ITEMRENAMED (MTN_FIRST - 5) // lParam = &NMMSGTREE; ItemNew points to renamed item info
+#define MTN_DELETEITEM (MTN_FIRST - 6) // lParam = &NMMSGTREE; ItemOld points to the item info; return TRUE to prevent item from being deleted
+#define MTN_DEFMSGCHANGED (MTN_FIRST - 7) // lParam = &NMMSGTREE; ItemOld and ItemNew point to old and new default item info correspondingly
+
+
+void LoadMsgTreeModule();
diff --git a/plugins/NewAwaySysMod/src/Notification.cpp b/plugins/NewAwaySysMod/src/Notification.cpp
new file mode 100644
index 0000000000..03ca926f94
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Notification.cpp
@@ -0,0 +1,285 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (C) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "Path.h"
+#include "Properties.h"
+
+
+void ShowMsg(TCHAR *FirstLine, TCHAR *SecondLine, bool IsErrorMsg, int Timeout)
+{
+ if (ServiceExists(MS_POPUP_ADDPOPUPT))
+ {
+ POPUPDATAT ppd = {0};
+ ppd.lchIcon = LoadIcon(NULL, IsErrorMsg ? IDI_EXCLAMATION : IDI_INFORMATION);
+ lstrcpy(ppd.lptzContactName, FirstLine);
+ lstrcpy(ppd.lptzText, SecondLine);
+ ppd.colorBack = IsErrorMsg ? 0x0202E3 : 0xE8F1FD;
+ ppd.colorText = IsErrorMsg ? 0xE8F1FD : 0x000000;
+ ppd.iSeconds = Timeout;
+ CallService(MS_POPUP_ADDPOPUPT, (WPARAM)&ppd, 0);
+ }
+ else MessageBox(NULL, SecondLine, FirstLine, MB_OK | (IsErrorMsg ? MB_ICONEXCLAMATION : MB_ICONINFORMATION));
+}
+
+
+static int CALLBACK MenuWndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
+{
+ switch (uMsg)
+ {
+ case WM_MEASUREITEM:
+ {
+ return CallService(MS_CLIST_MENUMEASUREITEM, wParam, lParam);
+ }
+ case WM_DRAWITEM:
+ {
+ return CallService(MS_CLIST_MENUDRAWITEM, wParam, lParam);
+ }
+ }
+ return DefWindowProc(hWnd, uMsg, wParam, lParam);
+}
+
+
+static VOID CALLBACK ShowContactMenu(DWORD wParam)
+// wParam = hContact
+{
+ POINT pt;
+ HWND hMenuWnd = CreateWindowEx(WS_EX_TOOLWINDOW, _T("static"), _T(MOD_NAME)_T("_MenuWindow"), 0, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, HWND_DESKTOP, NULL, g_hInstance, NULL);
+ SetWindowLongPtr(hMenuWnd, GWLP_WNDPROC, (LONG)(WNDPROC)MenuWndProc);
+ HMENU hMenu = (HMENU)CallService(MS_CLIST_MENUBUILDCONTACT, (WPARAM)wParam, 0);
+ GetCursorPos(&pt);
+ SetForegroundWindow(hMenuWnd);
+ CallService(MS_CLIST_MENUPROCESSCOMMAND, MAKEWPARAM(TrackPopupMenu(hMenu, TPM_LEFTALIGN | TPM_RIGHTBUTTON | TPM_RETURNCMD, pt.x, pt.y, 0, hMenuWnd, NULL), MPCF_CONTACTMENU), (LPARAM)wParam);
+ PostMessage(hMenuWnd, WM_NULL, 0, 0);
+ DestroyMenu(hMenu);
+ DestroyWindow(hMenuWnd);
+}
+
+/*
+void Popup_DoAction(HWND hWnd, BYTE Action, PLUGIN_DATA *pdata)
+{
+ MCONTACT hContact = (HANDLE)CallService(MS_POPUP_GETCONTACT, (WPARAM)hWnd, 0);
+ switch (Action)
+ {
+ case PCA_OPENMESSAGEWND: // open message window
+ {
+ if (hContact && hContact != INVALID_HANDLE_VALUE)
+ {
+ CallServiceSync(ServiceExists("SRMsg/LaunchMessageWindow") ? "SRMsg/LaunchMessageWindow" : MS_MSG_SENDMESSAGE, (WPARAM)hContact, 0);
+ }
+ } break;
+ case PCA_OPENMENU: // open contact menu
+ {
+ if (hContact && hContact != INVALID_HANDLE_VALUE)
+ {
+ QueueUserAPC(ShowContactMenu, hMainThread, (ULONG_PTR)hContact);
+ }
+ } break;
+ case PCA_OPENDETAILS: // open contact details window
+ {
+ if (hContact != INVALID_HANDLE_VALUE)
+ {
+ CallServiceSync(MS_USERINFO_SHOWDIALOG, (WPARAM)hContact, 0);
+ }
+ } break;
+ case PCA_OPENHISTORY: // open contact history
+ {
+ if (hContact != INVALID_HANDLE_VALUE)
+ {
+ CallServiceSync(MS_HISTORY_SHOWCONTACTHISTORY, (WPARAM)hContact, 0);
+ }
+ } break;
+ case PCA_OPENLOG: // open log file
+ {
+ TCString LogFilePath;
+ LS_LOGINFO li = {0};
+ li.cbSize = sizeof(li);
+ li.szID = LOG_ID;
+ li.hContact = hContact;
+ li.Flags = LSLI_TCHAR;
+ li.tszLogPath = LogFilePath.GetBuffer(MAX_PATH);
+ if (!CallService(MS_LOGSERVICE_GETLOGINFO, (WPARAM)&li, 0))
+ {
+ LogFilePath.ReleaseBuffer();
+ ShowLog(LogFilePath);
+ } else
+ {
+ LogFilePath.ReleaseBuffer();
+ }
+ } break;
+ case PCA_CLOSEPOPUP: // close popup
+ {
+ PUDeletePopup(hWnd);
+ } break;
+ case PCA_DONOTHING: // do nothing
+ break;
+ }
+}
+
+
+static int CALLBACK ReqNotifyPopupDlgProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
+{
+ PLUGIN_DATA *pdata;
+ switch (message)
+ {
+ case WM_COMMAND:
+ {
+ if (HIWORD(wParam) == STN_CLICKED) // left mouse button
+ {
+ pdata = (PLUGIN_DATA*)CallService(MS_POPUP_GETPLUGINDATA, (WPARAM)hWnd, 0);
+ if (pdata)
+ {
+ Popup_DoAction(hWnd, pdata->PopupLClickAction, pdata);
+ }
+ return true;
+ }
+ } break;
+ case WM_CONTEXTMENU: // right mouse button
+ {
+ pdata = (PLUGIN_DATA*)CallService(MS_POPUP_GETPLUGINDATA, (WPARAM)hWnd, 0);
+ if (pdata)
+ {
+ Popup_DoAction(hWnd, pdata->PopupRClickAction, pdata);
+ }
+ return true;
+ } break;
+ case UM_FREEPLUGINDATA:
+ {
+ pdata = (PLUGIN_DATA*)CallService(MS_POPUP_GETPLUGINDATA, (WPARAM)hWnd, 0);
+ if (pdata)
+ {
+ if (pdata->hStatusIcon)
+ {
+ DestroyIcon(pdata->hStatusIcon);
+ }
+ free(pdata);
+ }
+ return true;
+ } break;
+ }
+ return DefWindowProc(hWnd, message, wParam, lParam);
+}
+
+
+int ShowPopupNotification(COptPage &PopupNotifyData, MCONTACT hContact, int iStatusMode)
+{ // returns TRUE if popup was shown
+// we take szProto, UIN and Message from VarParseData
+ POPUPDATAT ppd = {0};
+ ppd.lchContact = hContact;
+ TCString ExtraText;
+ if (!iStatusMode)
+ { // if it's an xstatus message request
+ ExtraText = db_get_s(NULL, VarParseData.szProto, "XStatusName", _T(""));
+ TCString XMsg(db_get_s(NULL, VarParseData.szProto, "XStatusMsg", _T("")));
+ if (XMsg.GetLen())
+ {
+ if (ExtraText.GetLen())
+ {
+ ExtraText += _T("\r\n");
+ }
+ ExtraText += XMsg;
+ }
+ } else
+ {
+ ExtraText = VarParseData.Message;
+ }
+ TCString PopupMsg(*(TCString*)PopupNotifyData.GetValue(IDC_POPUPOPTDLG_POPUPFORMAT));
+ if (ServiceExists(MS_VARS_FORMATSTRING))
+ {
+ FORMATINFO fi = {0};
+ fi.cbSize = sizeof(fi);
+ fi.flags = FIF_TCHAR;
+ fi.tszFormat = PopupMsg;
+ fi.hContact = hContact;
+ fi.tszExtraText = ExtraText;
+ TCHAR *szResult = (TCHAR*)CallService(MS_VARS_FORMATSTRING, (WPARAM)&fi, 0);
+ if (szResult)
+ {
+ PopupMsg = szResult;
+ mir_free(szResult);
+ }
+ } else
+ {
+ TCString szUIN;
+ _ultot(VarParseData.UIN, szUIN.GetBuffer(16), 10);
+ szUIN.ReleaseBuffer();
+ TCHAR *szStatDesc = iStatusMode ? (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, iStatusMode, GSMDF_TCHAR) : STR_XSTATUSDESC;
+ _ASSERT(szStatDesc);
+ PopupMsg = TCString((TCHAR*)CallService(MS_CLIST_GETCONTACTDISPLAYNAME, (WPARAM)hContact, GCDNF_TCHAR)) + _T(" (") + szUIN + TranslateT(") is reading your ") + szStatDesc + TranslateT(" message:\r\n") + ExtraText;
+ }
+ if (PopupMsg.GetLen())
+ {
+ TCHAR *pLineBreak = _tcsstr(PopupMsg, _T("\n"));
+ if (pLineBreak)
+ {
+ PopupMsg.GetBuffer();
+ TCHAR *pNextLine = pLineBreak + 1;
+ while ((*pNextLine == '\r' || *pNextLine == '\n') && *pNextLine)
+ {
+ *pNextLine++;
+ }
+ _tcsncpy(ppd.lpzText, pNextLine, 499);
+ while ((*pLineBreak == '\r' || *pLineBreak == '\n') && (pLineBreak >= PopupMsg))
+ {
+ *pLineBreak-- = 0;
+ }
+ PopupMsg.ReleaseBuffer();
+ } else
+ {
+ lstrcpy(ppd.lpzText, _T(""));
+ }
+ _tcsncpy(ppd.lpzContactName, PopupMsg, 499);
+ ppd.colorBack = (PopupNotifyData.GetValue(IDC_POPUPOPTDLG_DEFBGCOLOUR) ? 0 : PopupNotifyData.GetValue(IDC_POPUPOPTDLG_BGCOLOUR));
+ ppd.colorText = (PopupNotifyData.GetValue(IDC_POPUPOPTDLG_DEFTEXTCOLOUR) ? 0 : PopupNotifyData.GetValue(IDC_POPUPOPTDLG_TEXTCOLOUR));
+ ppd.PluginWindowProc = (WNDPROC)ReqNotifyPopupDlgProc;
+ PLUGIN_DATA *pdata = (PLUGIN_DATA*)calloc(1, sizeof(PLUGIN_DATA));
+ if (!iStatusMode)
+ { // it's an xstatus message request
+ ppd.lchIcon = pdata->hStatusIcon = (HICON)CallProtoService(VarParseData.szProto, PS_ICQ_GETCUSTOMSTATUSICON, 0, 0);
+ }
+ if (!pdata->hStatusIcon || (DWORD)pdata->hStatusIcon == CALLSERVICE_NOTFOUND)
+ {
+ pdata->hStatusIcon = NULL;
+ ppd.lchIcon = LoadSkinnedProtoIcon(VarParseData.szProto, iStatusMode);
+ }
+ pdata->PopupLClickAction = PopupNotifyData.GetValue(IDC_POPUPOPTDLG_LCLICK_ACTION);
+ pdata->PopupRClickAction = PopupNotifyData.GetValue(IDC_POPUPOPTDLG_RCLICK_ACTION);
+ ppd.PluginData = pdata;
+ ppd.iSeconds = PopupNotifyData.GetValue(IDC_POPUPOPTDLG_POPUPDELAY);
+ CallService(MS_POPUP_ADDPOPUPT, (WPARAM)&ppd, 0);
+ if (hContact)
+ {
+ SkinPlaySound(AWAYSYS_STATUSMSGREQUEST_SOUND);
+ }
+ return true;
+ }
+ return false;
+}
+*/
+
+void ShowLog(TCString &LogFilePath)
+{
+ int Result = (int)ShellExecute(NULL, _T("open"), LogFilePath, NULL, NULL, SW_SHOW);
+ if (Result <= 32) // Error
+ {
+ TCHAR szError[64];
+ mir_sntprintf(szError, SIZEOF(szError), TranslateT("Error #%d"), Result);
+ ShowMsg(szError, TranslateT("Can't open log file ") + LogFilePath, true);
+ }
+}
diff --git a/plugins/NewAwaySysMod/src/Options.cpp b/plugins/NewAwaySysMod/src/Options.cpp
new file mode 100644
index 0000000000..e0d585d991
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Options.cpp
@@ -0,0 +1,963 @@
+/*
+ Options.cpp
+ Copyright (c) 2005-2008 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "Options.h"
+
+static CString sEmptyString("");
+
+
+COptPage::COptPage(const COptPage &Item)
+{
+ *this = Item;
+}
+
+COptPage::~COptPage()
+{
+ int I;
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ delete Items[I];
+ }
+ Items.RemoveAll();
+}
+
+void COptPage::MemToPage(int OnlyEnable)
+{
+ int I;
+ _ASSERT(hWnd);
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ if (OnlyEnable)
+ {
+ Items[I]->COptItem::MemToWnd(hWnd);
+ } else
+ {
+ Items[I]->MemToWnd(hWnd);
+ }
+ }
+}
+
+void COptPage::PageToMem()
+{
+ int I;
+ _ASSERT(hWnd);
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ Items[I]->WndToMem(hWnd);
+ }
+}
+
+void COptPage::DBToMem()
+{
+ int I;
+ _ASSERT(sModule != "");
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ Items[I]->DBToMem(sModule, &sDBSettingPrefix);
+ }
+}
+
+void COptPage::MemToDB()
+{
+ int I;
+ _ASSERT(sModule != "");
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ Items[I]->MemToDB(sModule, &sDBSettingPrefix);
+ }
+}
+
+void COptPage::CleanDBSettings()
+{
+ int I;
+ _ASSERT(sModule != "");
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ Items[I]->CleanDBSettings(sModule, &sDBSettingPrefix);
+ }
+}
+
+bool COptPage::GetModified()
+{
+ int I;
+ _ASSERT(sModule != "");
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ if (Items[I]->GetModified())
+ {
+ return true;
+ }
+ }
+ return false;
+}
+
+void COptPage::SetModified(bool Modified)
+{
+ int I;
+ _ASSERT(sModule != "");
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ Items[I]->SetModified(Modified);
+ }
+}
+
+COptItem *COptPage::Find(int DlgItemID)
+{
+ int I;
+ for (I = 0; I < Items.GetSize(); I++)
+ {
+ if (Items[I]->GetID() == DlgItemID)
+ {
+ return Items[I];
+ }
+ }
+ _ASSERT(0);
+ return 0;
+}
+
+COptPage& COptPage::operator = (const COptPage& Page)
+{
+ int I;
+ hWnd = Page.hWnd;
+ sModule = Page.sModule;
+ sDBSettingPrefix = Page.sDBSettingPrefix;
+ Items.RemoveAll();
+ for (I = 0; I < Page.Items.GetSize(); I++)
+ {
+ Items.AddElem(Page.Items[I]->Copy());
+ }
+ return *this;
+}
+
+
+int COptItem::GetIntDBVal(CString &sModule, int bSigned, CString *sDBSettingPrefix)
+{ // default procedure for reading value from DB; used only for integral types
+ if (sDBSetting != NULL)
+ {
+ _ASSERT(nValueSize == DBVT_BYTE || nValueSize == DBVT_WORD || nValueSize == DBVT_DWORD);
+ DBVARIANT dbv;
+ if (db_get(NULL, sModule, sDBSetting, &dbv))
+ return GetDefValue();
+
+ return (nValueSize == DBVT_BYTE) ? (bSigned ? (signed char)dbv.bVal : (unsigned char)dbv.bVal) : ((nValueSize == DBVT_WORD) ? (bSigned ? (signed short)dbv.wVal : (unsigned short)dbv.wVal) : dbv.dVal);
+ }
+ return GetDefValue();
+}
+
+void COptItem::SetIntDBVal(CString &sModule, int Value, CString *sDBSettingPrefix)
+{ // default procedure for writing value to the DB; used only for integral types
+ if (sDBSetting != NULL && !ReadOnly)
+ {
+ _ASSERT(nValueSize == DBVT_BYTE || nValueSize == DBVT_WORD || nValueSize == DBVT_DWORD);
+
+ DBVARIANT dbv;
+ dbv.type = nValueSize;
+ dbv.dVal = Value;
+ db_set(NULL, sModule, sDBSetting, &dbv);
+ }
+}
+
+TCString COptItem::GetStrDBVal(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (sDBSetting != NULL)
+ {
+ _ASSERT(GetDefValue());
+ return db_get_s(NULL, sModule, sDBSettingPrefix ? (*sDBSettingPrefix + sDBSetting) : sDBSetting, *(TCString*)GetDefValue());
+ }
+ return *(TCString*)GetDefValue();
+}
+
+void COptItem::SetStrDBVal(CString &sModule, TCString &Str, CString *sDBSettingPrefix)
+{
+ if (sDBSetting != NULL && !ReadOnly)
+ {
+ db_set_ts(NULL, sModule, sDBSettingPrefix ? (*sDBSettingPrefix + sDBSetting) : sDBSetting, Str);
+ }
+}
+
+
+
+void COptItem_Checkbox::DBToMem(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (ValueMask)
+ {
+ Value = (GetIntDBVal(sModule, false, sDBSettingPrefix) & ValueMask) ? BST_CHECKED : BST_UNCHECKED;
+ } else
+ {
+ Value = GetIntDBVal(sModule, false, sDBSettingPrefix);
+ }
+ COptItem::DBToMem(sModule, sDBSettingPrefix);
+}
+
+void COptItem_Checkbox::MemToDB(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (ValueMask)
+ {
+ if (Value == BST_CHECKED)
+ {
+ SetIntDBVal(sModule, GetIntDBVal(sModule, false, sDBSettingPrefix) | ValueMask, sDBSettingPrefix);
+ } else
+ {
+ SetIntDBVal(sModule, GetIntDBVal(sModule, false, sDBSettingPrefix) & ~ValueMask, sDBSettingPrefix);
+ }
+ } else
+ {
+ SetIntDBVal(sModule, Value, sDBSettingPrefix);
+ }
+ COptItem::MemToDB(sModule, sDBSettingPrefix);
+}
+
+void COptItem_Checkbox::WndToMem(HWND hWnd)
+{
+ Value = IsDlgButtonChecked(hWnd, DlgItemID);
+// tri-state checkboxes in combination with ValueMask != 0 are not supported ;)
+ _ASSERT(!ValueMask || Value != BST_INDETERMINATE);
+ COptItem::WndToMem(hWnd);
+}
+
+void COptItem_Checkbox::MemToWnd(HWND hWnd)
+{
+ CheckDlgButton(hWnd, DlgItemID, Value);
+ COptItem::MemToWnd(hWnd);
+}
+
+
+
+void COptItem_BitDBSetting::DBToMem(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (ValueMask)
+ {
+ Value = (GetIntDBVal(sModule, false, sDBSettingPrefix) & ValueMask) != 0;
+ } else
+ {
+ Value = GetIntDBVal(sModule, false, sDBSettingPrefix);
+ }
+ COptItem::DBToMem(sModule, sDBSettingPrefix);
+}
+
+void COptItem_BitDBSetting::MemToDB(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (ValueMask)
+ {
+ if (Value)
+ {
+ SetIntDBVal(sModule, GetIntDBVal(sModule, false, sDBSettingPrefix) | ValueMask, sDBSettingPrefix);
+ } else
+ {
+ SetIntDBVal(sModule, GetIntDBVal(sModule, false, sDBSettingPrefix) & ~ValueMask, sDBSettingPrefix);
+ }
+ } else
+ {
+ SetIntDBVal(sModule, Value, sDBSettingPrefix);
+ }
+ COptItem::MemToDB(sModule, sDBSettingPrefix);
+}
+
+
+// ================================================ COptItem_TreeCtrl ================================================
+
+int COptItem_TreeCtrl::IDToOrder(int ID)
+{
+ int I;
+ for (I = 0; I < RootItems.GetSize(); I++)
+ {
+ if (RootItems[I].ID == ID)
+ {
+ return ROOT_INDEX_TO_ORDER(I);
+ }
+ }
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ if (Value[I].ID == ID)
+ {
+ return I;
+ }
+ }
+ return -1;
+}
+
+int COptItem_TreeCtrl::hItemToOrder(HTREEITEM hItem)
+{
+ int I;
+ for (I = 0; I < RootItems.GetSize(); I++)
+ {
+ if (RootItems[I].hItem == hItem)
+ {
+ return ROOT_INDEX_TO_ORDER(I);
+ }
+ }
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ if (Value[I].hItem == hItem)
+ {
+ return I;
+ }
+ }
+ return -1;
+}
+
+int COptItem_TreeCtrl::GenerateID()
+{
+ int ID = 0;
+ while (IDToOrder(ID) != -1)
+ {
+ ID++;
+ }
+ return ID;
+}
+
+typedef struct
+{
+ COptItem_TreeCtrl *TreeCtrl;
+ CString *sModule;
+ CString *sDBSettingPrefix;
+} sTreeReadEnumData;
+
+int TreeReadEnum(const char *szSetting, LPARAM lParam)
+{
+ sTreeReadEnumData *TreeReadEnumData = (sTreeReadEnumData*)lParam;
+ int Len = TreeReadEnumData->TreeCtrl->sDBSetting.GetLen() + lengthof(TREEITEM_DBSTR_TITLE) - 1;
+ if (!strncmp(szSetting, TreeReadEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_TITLE, Len) && isdigit(szSetting[Len]))
+ {
+ int ID = atol(szSetting + Len);
+ short ParentID = (TreeReadEnumData->TreeCtrl->TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL) ? 0 : db_get_w(NULL, *TreeReadEnumData->sModule,
+ *TreeReadEnumData->sDBSettingPrefix + TreeReadEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_PARENT + (szSetting + Len), -1);
+ short Order = db_get_w(NULL, *TreeReadEnumData->sModule,
+ *TreeReadEnumData->sDBSettingPrefix + TreeReadEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_ORDER + (szSetting + Len), -1);
+ char Flags = (TreeReadEnumData->TreeCtrl->TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL && !(TreeReadEnumData->TreeCtrl->TreeFlags & TREECTRL_FLAG_HAS_CHECKBOXES)) ? 0 : db_get_b(NULL, *TreeReadEnumData->sModule,
+ *TreeReadEnumData->sDBSettingPrefix + TreeReadEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_FLAGS + (szSetting + Len), 0);
+ if (ParentID >= 0 && Order >= 0)
+ {
+ TreeReadEnumData->TreeCtrl->Value.SetAtGrow(Order).ID = ID;
+ TreeReadEnumData->TreeCtrl->Value.SetAtGrow(Order).ParentID = ParentID;
+ TreeReadEnumData->TreeCtrl->Value.SetAtGrow(Order).Flags = Flags;
+ TreeReadEnumData->TreeCtrl->Value.SetAtGrow(Order).hItem = NULL;
+ TreeReadEnumData->TreeCtrl->Value.SetAtGrow(Order).Title = db_get_s(NULL, *TreeReadEnumData->sModule, *TreeReadEnumData->sDBSettingPrefix + szSetting, _T(""));
+ TreeReadEnumData->TreeCtrl->Value.SetAtGrow(Order).User_Str1 = (TreeReadEnumData->TreeCtrl->User_Str1_DBName == NULL) ? NULL :
+ db_get_s(NULL, *TreeReadEnumData->sModule,
+ *TreeReadEnumData->sDBSettingPrefix + TreeReadEnumData->TreeCtrl->sDBSetting + TreeReadEnumData->TreeCtrl->User_Str1_DBName + (szSetting + Len), (TCHAR*)NULL);
+ }
+ }
+ return 0;
+}
+
+void COptItem_TreeCtrl::DBToMem(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (!sDBSettingPrefix)
+ {
+ sDBSettingPrefix = &sEmptyString;
+ }
+ Value.RemoveAll();
+ sTreeReadEnumData TreeReadEnumData;
+ TreeReadEnumData.TreeCtrl = this;
+ TreeReadEnumData.sModule = &sModule;
+ TreeReadEnumData.sDBSettingPrefix = sDBSettingPrefix;
+ DBCONTACTENUMSETTINGS dbEnum;
+ dbEnum.lParam = (LPARAM)&TreeReadEnumData;
+ dbEnum.ofsSettings = 0;
+ dbEnum.pfnEnumProc = TreeReadEnum;
+ dbEnum.szModule = sModule;
+ CallService(MS_DB_CONTACT_ENUMSETTINGS, NULL, (LPARAM)&dbEnum);
+ if (!Value.GetSize())
+ {
+ Value = DefValue;
+ } else
+ {
+ int I;
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ if (Value[I].Title == NULL)
+ {
+ Value.RemoveElem(I);
+ I--;
+ }
+ }
+ }
+ COptItem::DBToMem(sModule, sDBSettingPrefix);
+}
+
+void COptItem_TreeCtrl::MemToDB(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (!ReadOnly && Modified)
+ {
+ if (!sDBSettingPrefix)
+ {
+ sDBSettingPrefix = &sEmptyString;
+ }
+ CleanDBSettings(sModule, sDBSettingPrefix);
+ int I;
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ CString StrID;
+ _itoa(Value[I].ID, StrID.GetBuffer(64), 10);
+ StrID.ReleaseBuffer();
+ db_set_ts(NULL, sModule, *sDBSettingPrefix + sDBSetting + TREEITEM_DBSTR_TITLE + StrID, Value[I].Title);
+ if (!(TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL))
+ db_set_w(NULL, sModule, *sDBSettingPrefix + sDBSetting + TREEITEM_DBSTR_PARENT + StrID, Value[I].ParentID);
+
+ db_set_w(NULL, sModule, *sDBSettingPrefix + sDBSetting + TREEITEM_DBSTR_ORDER + StrID, I);
+ if (!(TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL) || TreeFlags & TREECTRL_FLAG_HAS_CHECKBOXES)
+ db_set_b(NULL, sModule, *sDBSettingPrefix + sDBSetting + TREEITEM_DBSTR_FLAGS + StrID, Value[I].Flags);
+
+ if (User_Str1_DBName != NULL && Value[I].User_Str1 != NULL)
+ db_set_ts(NULL, sModule, *sDBSettingPrefix + sDBSetting + User_Str1_DBName + StrID, Value[I].User_Str1);
+ }
+ COptItem::MemToDB(sModule, sDBSettingPrefix);
+ }
+}
+
+void COptItem_TreeCtrl::WndToMem(HWND hWnd)
+{ // only need to gather info of items state (expanded/collapsed, checked/unchecked)
+ HWND hTreeView = GetDlgItem(hWnd, DlgItemID);
+ int I;
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ DWORD State = TreeView_GetItemState(hTreeView, Value[I].hItem, TVIS_EXPANDED | TVIS_STATEIMAGEMASK);
+ int OldFlags = Value[I].Flags;
+ if (State & TVIS_EXPANDED)
+ {
+ Value[I].Flags |= TIF_EXPANDED;
+ } else
+ {
+ Value[I].Flags &= ~TIF_EXPANDED;
+ }
+ if (TreeFlags & TREECTRL_FLAG_HAS_CHECKBOXES && (State >> 12) - 1)
+ {
+ Value[I].Flags |= TIF_ENABLED;
+ } else
+ {
+ Value[I].Flags &= ~TIF_ENABLED;
+ }
+ if (Value[I].Flags != OldFlags)
+ {
+ Modified = true;
+ }
+ }
+ COptItem::WndToMem(hWnd);
+}
+
+void COptItem_TreeCtrl::MemToWnd(HWND hWnd)
+{
+ HWND hTreeView = GetDlgItem(hWnd, DlgItemID);
+ if (TreeFlags & TREECTRL_FLAG_HAS_CHECKBOXES)
+ { // have to set this in run-time as it's specified in MSDN
+ LONG Style = GetWindowLongPtr(hTreeView, GWL_STYLE);
+ SetWindowLongPtr(hTreeView, GWL_STYLE, Style & ~TVS_CHECKBOXES);
+ SetWindowLongPtr(hTreeView, GWL_STYLE, Style | TVS_CHECKBOXES);
+ }
+ TVINSERTSTRUCT tvIn = {0};
+ int ScrollPos = GetScrollPos(hTreeView, SB_VERT);
+ int SelectOrder = IDToOrder(GetSelectedItemID(hWnd));
+ SendMessage(hTreeView, WM_SETREDRAW, false, 0);
+ TreeView_DeleteAllItems(hTreeView);
+ _ASSERT(RootItems.GetSize());
+ int I;
+ if (!(TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL))
+ {
+ for (I = 0; I < RootItems.GetSize(); I++)
+ {
+ tvIn.item.mask = TVIF_TEXT | TVIF_STATE | TVIF_PARAM;
+ RootItems[I].Flags |= TIF_GROUP;
+ tvIn.item.state = tvIn.item.stateMask = TVIS_BOLD | ((RootItems[I].Flags & TIF_EXPANDED) ? TVIS_EXPANDED : 0);
+ tvIn.item.pszText = RootItems[I].Title;
+ tvIn.hParent = TVI_ROOT;
+ tvIn.hInsertAfter = TVI_LAST;
+ tvIn.item.lParam = RootItems[I].ID;
+ RootItems[I].hItem = TreeView_InsertItem(hTreeView, &tvIn);
+ }
+ }
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ Value[I].hItem = RootItems[0].hItem; // put an item to first group in case of some strange error
+ }
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ tvIn.item.mask = TVIF_TEXT | TVIF_STATE | TVIF_PARAM;
+ tvIn.item.state = tvIn.item.stateMask = (Value[I].Flags & TIF_GROUP) ? (TVIS_BOLD | ((Value[I].Flags & TIF_EXPANDED) ? TVIS_EXPANDED : 0)) : 0;
+ if (TreeFlags & TREECTRL_FLAG_HAS_CHECKBOXES)
+ {
+ tvIn.item.stateMask |= TVIS_STATEIMAGEMASK;
+ tvIn.item.state |= INDEXTOSTATEIMAGEMASK((Value[I].Flags & TIF_ENABLED) ? 2 : 1);
+ }
+ tvIn.item.pszText = Value[I].Title;
+ int Order = IDToOrder(Value[I].ParentID);
+ if (Order != -1)
+ {
+ tvIn.hParent = (Order <= TREECTRL_ROOTORDEROFFS) ? RootItems[ROOT_ORDER_TO_INDEX(Order)].hItem : Value[Order].hItem;
+ tvIn.hInsertAfter = TVI_LAST;
+ tvIn.item.lParam = Value[I].ID;
+ Value[I].hItem = TreeView_InsertItem(hTreeView, &tvIn);
+ } else
+ { // found an orphan item; probably it's better just to delete it
+ Value.RemoveElem(I);
+ I--;
+ }
+ }
+ TreeView_SelectItem(hTreeView, (SelectOrder >= 0) ? Value[SelectOrder].hItem : ((SelectOrder <= TREECTRL_ROOTORDEROFFS) ? RootItems[ROOT_ORDER_TO_INDEX(SelectOrder)].hItem : NULL));
+ SendMessage(hTreeView, WM_SETREDRAW, true, 0);
+ SCROLLBARINFO sbi;
+ sbi.cbSize = sizeof(sbi);
+ GetScrollBarInfo(hTreeView, OBJID_VSCROLL, &sbi);
+ if (!(sbi.rgstate[0] & STATE_SYSTEM_INVISIBLE))
+ {
+ int MinPos, MaxPos;
+ GetScrollRange(hTreeView, SB_VERT, &MinPos, &MaxPos);
+ if (ScrollPos < MinPos)
+ {
+ ScrollPos = MinPos;
+ } else if (ScrollPos > MaxPos)
+ {
+ ScrollPos = MaxPos;
+ }
+ SetScrollPos(hTreeView, SB_VERT, ScrollPos, true);
+ }
+ COptItem::MemToWnd(hWnd);
+}
+
+
+typedef struct
+{
+ TMyArray<CString> TreeSettings;
+ COptItem_TreeCtrl *TreeCtrl;
+ CString *sDBSettingPrefix;
+} sTreeDeleteEnumData;
+
+int TreeDeleteEnum(const char *szSetting, LPARAM lParam)
+{
+ sTreeDeleteEnumData *TreeDeleteEnumData = (sTreeDeleteEnumData*)lParam;
+ CString CurSetting;
+ CurSetting = *TreeDeleteEnumData->sDBSettingPrefix + TreeDeleteEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_TITLE;
+ if (!strncmp(szSetting, CurSetting, CurSetting.GetLen()))
+ {
+ TreeDeleteEnumData->TreeSettings.AddElem(szSetting);
+ }
+ CurSetting = *TreeDeleteEnumData->sDBSettingPrefix + TreeDeleteEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_PARENT;
+ if (!strncmp(szSetting, CurSetting, CurSetting.GetLen()))
+ {
+ TreeDeleteEnumData->TreeSettings.AddElem(szSetting);
+ }
+ CurSetting = *TreeDeleteEnumData->sDBSettingPrefix + TreeDeleteEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_ORDER;
+ if (!strncmp(szSetting, CurSetting, CurSetting.GetLen()))
+ {
+ TreeDeleteEnumData->TreeSettings.AddElem(szSetting);
+ }
+ CurSetting = *TreeDeleteEnumData->sDBSettingPrefix + TreeDeleteEnumData->TreeCtrl->sDBSetting + TREEITEM_DBSTR_FLAGS;
+ if (!strncmp(szSetting, CurSetting, CurSetting.GetLen()))
+ {
+ TreeDeleteEnumData->TreeSettings.AddElem(szSetting);
+ }
+ if (TreeDeleteEnumData->TreeCtrl->User_Str1_DBName != NULL)
+ {
+ CurSetting = *TreeDeleteEnumData->sDBSettingPrefix + TreeDeleteEnumData->TreeCtrl->sDBSetting + TreeDeleteEnumData->TreeCtrl->User_Str1_DBName;
+ if (!strncmp(szSetting, CurSetting, CurSetting.GetLen()))
+ {
+ TreeDeleteEnumData->TreeSettings.AddElem(szSetting);
+ }
+ }
+ return 0;
+}
+
+void COptItem_TreeCtrl::CleanDBSettings(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (!sDBSettingPrefix)
+ {
+ sDBSettingPrefix = &sEmptyString;
+ }
+ sTreeDeleteEnumData TreeDeleteEnumData;
+ TreeDeleteEnumData.TreeCtrl = this;
+ TreeDeleteEnumData.sDBSettingPrefix = sDBSettingPrefix;
+ DBCONTACTENUMSETTINGS dbEnum;
+ dbEnum.lParam = (LPARAM)&TreeDeleteEnumData;
+ dbEnum.ofsSettings = 0;
+ dbEnum.pfnEnumProc = TreeDeleteEnum;
+ dbEnum.szModule = sModule;
+ CallService(MS_DB_CONTACT_ENUMSETTINGS, NULL, (LPARAM)&dbEnum);
+ int I;
+ for (I = 0; I < TreeDeleteEnumData.TreeSettings.GetSize(); I++)
+ {
+ db_unset(NULL, sModule, TreeDeleteEnumData.TreeSettings[I]);
+ }
+}
+
+
+int COptItem_TreeCtrl::GetSelectedItemID(HWND hWnd)
+{
+ HWND hTreeView = GetDlgItem(hWnd, DlgItemID);
+ TVITEM tvi = {0};
+ tvi.hItem = TreeView_GetSelection(hTreeView);
+ if (!tvi.hItem)
+ {
+ return -1;
+ }
+ tvi.mask = TVIF_HANDLE | TVIF_PARAM;
+ TreeView_GetItem(hTreeView, &tvi);
+ return tvi.lParam;
+}
+
+void COptItem_TreeCtrl::Delete(HWND hWnd, int ID)
+{
+ int SelectedOrder = IDToOrder(ID);
+ _ASSERT(SelectedOrder >= 0);
+ RecursiveDelete(hWnd, SelectedOrder);
+ Modified = true;
+}
+
+void COptItem_TreeCtrl::RecursiveDelete(HWND hWnd, int I)
+{
+ if (Value[I].Flags & TIF_GROUP)
+ {
+ int J;
+ for (J = I + 1; J < Value.GetSize(); J++)
+ {
+ if (Value[J].ParentID == Value[I].ID)
+ {
+ RecursiveDelete(hWnd, J--);
+ }
+ }
+ }
+ HWND hTreeView = GetDlgItem(hWnd, DlgItemID);
+ TreeView_DeleteItem(hTreeView, Value[I].hItem);
+ Value.RemoveElem(I);
+}
+
+CTreeItem* COptItem_TreeCtrl::InsertItem(HWND hWnd, CTreeItem &Item)
+// Item's ID and ParentID are not used (the new item position is determined by current selection in the tree)
+// returns a pointer to the newly inserted item info
+{
+ _ASSERT(!(TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL) || !(Item.Flags & TIF_GROUP));
+ HWND hTreeView = GetDlgItem(hWnd, DlgItemID);
+ TVITEM tvi;
+ int SelOrder = -1;
+ Item.ParentID = RootItems[0].ID;
+ TVINSERTSTRUCT tvIn = {0};
+ tvIn.hParent = RootItems[0].hItem;
+ tvIn.hInsertAfter = TVI_FIRST;
+ if (tvi.hItem = TreeView_GetSelection(hTreeView))
+ {
+ tvi.mask = TVIF_HANDLE | TVIF_PARAM;
+ TreeView_GetItem(hTreeView, &tvi);
+ SelOrder = IDToOrder(tvi.lParam);
+ if (SelOrder <= TREECTRL_ROOTORDEROFFS)
+ {
+ Item.ParentID = RootItems[ROOT_ORDER_TO_INDEX(SelOrder)].ID;
+ tvIn.hParent = RootItems[ROOT_ORDER_TO_INDEX(SelOrder)].hItem;
+ SelOrder = -1;
+ } else
+ {
+ if (Value[SelOrder].Flags & TIF_GROUP)
+ {
+ Item.ParentID = Value[SelOrder].ID;
+ tvIn.hParent = Value[SelOrder].hItem;
+ } else
+ {
+ Item.ParentID = Value[SelOrder].ParentID;
+ int Order = IDToOrder(Value[SelOrder].ParentID);
+ tvIn.hParent = (Order <= TREECTRL_ROOTORDEROFFS) ? RootItems[ROOT_ORDER_TO_INDEX(Order)].hItem : Value[Order].hItem;
+ tvIn.hInsertAfter = Value[SelOrder].hItem;
+ }
+ }
+ }
+ tvIn.item.mask = TVIF_TEXT | TVIF_STATE | TVIF_PARAM;
+ tvIn.item.state = tvIn.item.stateMask = (Item.Flags & TIF_GROUP) ? (TVIS_BOLD |
+ ((Item.Flags & TIF_EXPANDED) ? TVIS_EXPANDED : 0)) : 0;
+ if (TreeFlags & TREECTRL_FLAG_HAS_CHECKBOXES)
+ {
+ tvIn.item.stateMask |= TVIS_STATEIMAGEMASK;
+ tvIn.item.state |= INDEXTOSTATEIMAGEMASK((Item.Flags & TIF_ENABLED) ? 2 : 1);
+ }
+ tvIn.item.pszText = Item.Title;
+ tvIn.item.lParam = Item.ID = GenerateID();
+ Value.InsertElem(Item, SelOrder + 1);
+ Value[SelOrder + 1].hItem = TreeView_InsertItem(hTreeView, &tvIn);
+ TreeView_SelectItem(hTreeView, Value[SelOrder + 1].hItem);
+ Modified = true;
+ return &Value[SelOrder + 1];
+}
+
+int COptItem_TreeCtrl::RecursiveMove(int ItemOrder, int ParentID, int InsertAtOrder)
+// ItemOrder must be a movable item (i.e. ItemOrder >= 0)
+// InsertAtOrder must be >= 0 too.
+{
+ int ItemsMoved = 1;
+ Value.MoveElem(ItemOrder, InsertAtOrder);
+ Value[InsertAtOrder].ParentID = ParentID;
+ if (Value[InsertAtOrder].Flags & TIF_GROUP) // need to ensure that no items were left before their group by an order.
+ {
+ int GroupID = Value[InsertAtOrder].ID;
+ int I;
+ for (I = ItemOrder; I < InsertAtOrder; I++) // if ItemOrder > InsertAtOrder then there is simply nothing to do
+ {
+ if (Value[I].ParentID == GroupID)
+ {
+ int CurrentItemsMoved = RecursiveMove(I, GroupID, InsertAtOrder);
+ ItemsMoved += CurrentItemsMoved;
+ InsertAtOrder -= CurrentItemsMoved;
+ I--;
+ }
+ }
+ }
+ return ItemsMoved;
+}
+
+void COptItem_TreeCtrl::MoveItem(HWND hWnd, HTREEITEM hItem, HTREEITEM hMoveTo)
+{ // hMoveTo can be NULL and it means that we must move hItem to the beginning of the list
+ _ASSERT(hItem && (hMoveTo || TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL));
+ if (hItem == hMoveTo)
+ {
+ return;
+ }
+ HWND hTreeView = GetDlgItem(hWnd, DlgItemID);
+ TVITEM tvi;
+ tvi.mask = TVIF_HANDLE | TVIF_PARAM;
+ tvi.hItem = hItem;
+ TreeView_GetItem(hTreeView, &tvi);
+ int ItemOrder = IDToOrder(tvi.lParam);
+ _ASSERT(ItemOrder != -1);
+ int MoveToOrder;
+ if (hMoveTo)
+ {
+ tvi.hItem = hMoveTo;
+ TreeView_GetItem(hTreeView, &tvi);
+ MoveToOrder = IDToOrder(tvi.lParam);
+ _ASSERT(MoveToOrder != -1);
+ } else
+ {
+ MoveToOrder = -1;
+ }
+ if (ItemOrder <= TREECTRL_ROOTORDEROFFS)
+ {
+ return; // can't move root items
+ }
+ if (Value[ItemOrder].Flags & TIF_GROUP)
+ { // need to check for a case when trying to move a group to its own subgroup.
+ int Order = MoveToOrder;
+ while (Order >= 0)
+ {
+ Order = IDToOrder(Value[Order].ParentID);
+ if (Order == ItemOrder)
+ {
+ return;
+ }
+ }
+ }
+// well, everything is ok, we really can move that item.
+ WndToMem(hWnd); // save groups state (expanded/collapsed)
+ if (MoveToOrder != -1 && ((MoveToOrder <= TREECTRL_ROOTORDEROFFS) ? RootItems[ROOT_ORDER_TO_INDEX(MoveToOrder)].Flags : Value[MoveToOrder].Flags) & TIF_GROUP)
+ { // if the destination is a group, then move the item to that group
+ RecursiveMove(ItemOrder, (MoveToOrder <= TREECTRL_ROOTORDEROFFS) ? RootItems[ROOT_ORDER_TO_INDEX(MoveToOrder)].ID : Value[MoveToOrder].ID, (MoveToOrder >= 0) ? ((ItemOrder < MoveToOrder) ? MoveToOrder : (MoveToOrder + 1)) : 0);
+ } else
+ { // else place the item after the destination item
+ RecursiveMove(ItemOrder, (MoveToOrder == -1) ? 0 : Value[MoveToOrder].ParentID, (ItemOrder < MoveToOrder) ? MoveToOrder : (MoveToOrder + 1)); // when TREECTRL_FLAG_IS_SINGLE_LEVEL, we always have a root item with ID = 0.
+ }
+ MemToWnd(hWnd); // update the tree
+ Modified = true;
+}
+
+
+// ================================================ COptItem_ListCtrl ================================================
+
+typedef struct
+{
+ COptItem_ListCtrl *ListCtrl;
+ CString *sModule;
+ CString *sDBSettingPrefix;
+} sListReadEnumData;
+
+int ListReadEnum(const char *szSetting, LPARAM lParam)
+{
+ sListReadEnumData *ListReadEnumData = (sListReadEnumData*)lParam;
+ int Len = ListReadEnumData->sDBSettingPrefix->GetLen() + ListReadEnumData->ListCtrl->sDBSetting.GetLen() + lengthof(LISTITEM_DBSTR_TEXT) - 1;
+ if (!strncmp(szSetting, *ListReadEnumData->sDBSettingPrefix + ListReadEnumData->ListCtrl->sDBSetting + LISTITEM_DBSTR_TEXT, Len) && isdigit(szSetting[Len]))
+ {
+ int ID = atol(szSetting + Len);
+ ListReadEnumData->ListCtrl->Value.SetAtGrow(ID).Text = db_get_s(NULL, *ListReadEnumData->sModule, *ListReadEnumData->sDBSettingPrefix + szSetting, _T(""));
+ }
+ return 0;
+}
+
+void COptItem_ListCtrl::DBToMem(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (!sDBSettingPrefix)
+ {
+ sDBSettingPrefix = &sEmptyString;
+ }
+ Value.RemoveAll();
+ sListReadEnumData ListReadEnumData;
+ ListReadEnumData.ListCtrl = this;
+ ListReadEnumData.sModule = &sModule;
+ ListReadEnumData.sDBSettingPrefix = sDBSettingPrefix;
+ DBCONTACTENUMSETTINGS dbEnum;
+ dbEnum.lParam = (LPARAM)&ListReadEnumData;
+ dbEnum.ofsSettings = 0;
+ dbEnum.pfnEnumProc = ListReadEnum;
+ dbEnum.szModule = sModule;
+ CallService(MS_DB_CONTACT_ENUMSETTINGS, NULL, (LPARAM)&dbEnum);
+ if (!Value.GetSize())
+ {
+ Value = DefValue;
+ } else
+ {
+ int I;
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ if (Value[I].Text == NULL) // NULL is not ""!
+ {
+ Value.RemoveElem(I);
+ I--;
+ }
+ }
+ }
+ COptItem::DBToMem(sModule, sDBSettingPrefix);
+}
+
+void COptItem_ListCtrl::MemToDB(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (!ReadOnly && Modified)
+ {
+ if (!sDBSettingPrefix)
+ {
+ sDBSettingPrefix = &sEmptyString;
+ }
+ CleanDBSettings(sModule, sDBSettingPrefix);
+ int I;
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ CString StrID;
+ _itoa(I, StrID.GetBuffer(64), 10);
+ StrID.ReleaseBuffer();
+ db_set_ts(NULL, sModule, *sDBSettingPrefix + sDBSetting + LISTITEM_DBSTR_TEXT + StrID, Value[I].Text);
+ }
+ COptItem::MemToDB(sModule, sDBSettingPrefix);
+ }
+}
+
+void COptItem_ListCtrl::WndToMem(HWND hWnd)
+{
+// nothing to do
+ COptItem::WndToMem(hWnd);
+}
+
+void COptItem_ListCtrl::MemToWnd(HWND hWnd)
+{
+ HWND hListView = GetDlgItem(hWnd, DlgItemID);
+ SendMessage(hListView, WM_SETREDRAW, false, 0);
+ SendMessage(hListView, LB_RESETCONTENT, 0, 0);
+ int I;
+ for (I = 0; I < Value.GetSize(); I++)
+ {
+ SendMessage(hListView, LB_INSERTSTRING, -1, (LPARAM)(TCHAR*)Value[I].Text);
+ }
+ SendMessage(hListView, WM_SETREDRAW, true, 0);
+ COptItem::MemToWnd(hWnd);
+}
+
+
+typedef struct
+{
+ TMyArray<CString> ListSettings;
+ COptItem_ListCtrl *ListCtrl;
+ CString *sDBSettingPrefix;
+} sListDeleteEnumData;
+
+int ListDeleteEnum(const char *szSetting, LPARAM lParam)
+{
+ sListDeleteEnumData *ListDeleteEnumData = (sListDeleteEnumData*)lParam;
+ CString CurSetting = *ListDeleteEnumData->sDBSettingPrefix + ListDeleteEnumData->ListCtrl->sDBSetting + LISTITEM_DBSTR_TEXT;
+ if (!strncmp(szSetting, CurSetting, CurSetting.GetLen()))
+ {
+ ListDeleteEnumData->ListSettings.AddElem(szSetting);
+ }
+ return 0;
+}
+
+void COptItem_ListCtrl::CleanDBSettings(CString &sModule, CString *sDBSettingPrefix)
+{
+ if (!sDBSettingPrefix)
+ {
+ sDBSettingPrefix = &sEmptyString;
+ }
+ sListDeleteEnumData ListDeleteEnumData;
+ ListDeleteEnumData.ListCtrl = this;
+ ListDeleteEnumData.sDBSettingPrefix = sDBSettingPrefix;
+ DBCONTACTENUMSETTINGS dbEnum;
+ dbEnum.lParam = (LPARAM)&ListDeleteEnumData;
+ dbEnum.ofsSettings = 0;
+ dbEnum.pfnEnumProc = ListDeleteEnum;
+ dbEnum.szModule = sModule;
+ CallService(MS_DB_CONTACT_ENUMSETTINGS, NULL, (LPARAM)&dbEnum);
+ int I;
+ for (I = 0; I < ListDeleteEnumData.ListSettings.GetSize(); I++)
+ {
+ db_unset(NULL, sModule, ListDeleteEnumData.ListSettings[I]);
+ }
+}
+
+
+int COptItem_ListCtrl::GetSelectedItemID(HWND hWnd)
+{
+ int Res = SendDlgItemMessage(hWnd, DlgItemID, LB_GETCURSEL, 0, 0);
+ return (Res == LB_ERR) ? -1 : Res; // I know that LB_ERR = -1 ;)
+}
+
+int COptItem_ListCtrl::SetSelectedItemID(HWND hWnd, int ID)
+{
+ int Res = SendDlgItemMessage(hWnd, DlgItemID, LB_SETCURSEL, ID, 0);
+ return (Res == LB_ERR) ? -1 : Res;
+}
+
+void COptItem_ListCtrl::Delete(HWND hWnd, int ID)
+{
+ _ASSERT(ID >= 0);
+ HWND hListView = GetDlgItem(hWnd, DlgItemID);
+ int Res = SendMessage(hListView, LB_DELETESTRING, ID, 0);
+ _ASSERT(Res != LB_ERR);
+ Value.RemoveElem(ID);
+ Modified = true;
+}
+
+void COptItem_ListCtrl::ModifyItem(HWND hWnd, int ID, CListItem &Item)
+{ // changes the text of item with the specified ID
+ _ASSERT(ID >= 0);
+ HWND hListView = GetDlgItem(hWnd, DlgItemID);
+ SendMessage(hListView, WM_SETREDRAW, false, 0);
+ int CurSel = SendMessage(hListView, LB_GETCURSEL, 0, 0);
+ int TopIndex = SendMessage(hListView, LB_GETTOPINDEX, 0, 0);
+ int Res = SendMessage(hListView, LB_DELETESTRING, ID, 0);
+ _ASSERT(Res != LB_ERR);
+ Res = SendMessage(hListView, LB_INSERTSTRING, ID, (LPARAM)(TCHAR*)(Item.Text));
+ _ASSERT(Res != LB_ERR && Res != LB_ERRSPACE);
+ SendMessage(hListView, LB_SETCURSEL, CurSel, 0);
+ SendMessage(hListView, LB_SETTOPINDEX, TopIndex, 0);
+ SendMessage(hListView, WM_SETREDRAW, true, 0);
+ Value[ID].Text = Item.Text;
+ Modified = true;
+}
+
+CListItem* COptItem_ListCtrl::InsertItem(HWND hWnd, int ID, CListItem &Item)
+// returns a pointer to the newly inserted item info
+// ID is position at which to insert the item; -1 = add to the end of the list
+{
+ HWND hListView = GetDlgItem(hWnd, DlgItemID);
+ int Res = SendMessage(hListView, LB_INSERTSTRING, ID, (LPARAM)(TCHAR*)(Item.Text)); // LB_INSERTSTRING doesn't sort the lists even with LBS_SORT style
+ _ASSERT(Res != LB_ERR && Res != LB_ERRSPACE);
+ int I = Value.AddElem(Item);
+ Modified = true;
+ return &Value[I];
+}
diff --git a/plugins/NewAwaySysMod/src/Options.h b/plugins/NewAwaySysMod/src/Options.h
new file mode 100644
index 0000000000..666162c50a
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Options.h
@@ -0,0 +1,483 @@
+/*
+ Options.h
+ Copyright (c) 2005-2008 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#pragma once
+
+#include "CString.h"
+#include "TMyArray.h"
+
+#ifndef lengthof
+#define lengthof(s) (sizeof(s) / sizeof(*s))
+#endif
+
+
+class COptItem
+{
+public:
+ COptItem() {}
+ COptItem(int DlgItemID, char *szDBSetting, int nValueSize, int lParam = 0, bool ReadOnly = false):
+ DlgItemID(DlgItemID), nValueSize(nValueSize), sDBSetting(szDBSetting), lParam(lParam), Enabled(true), ReadOnly(ReadOnly), Modified(false) {}
+/* COptItem(const COptItem &Item): DlgItemID(Item.DlgItemID), nValueSize(Item.nValueSize),
+ sDBSetting(Item.szDBSetting), lParam(Item.lParam), Enabled(Item.Enabled) {};*/
+ virtual ~COptItem() {}
+
+ virtual void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {Modified = false;}
+ virtual void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {Modified = false;}
+ virtual void WndToMem(HWND hWnd) {}
+ virtual void MemToWnd(HWND hWnd) {EnableWindow(GetDlgItem(hWnd, DlgItemID), Enabled);}
+ void DBToMemToWnd(CString &sModule, HWND hWnd, CString *sDBSettingPrefix = NULL) {DBToMem(sModule, sDBSettingPrefix); MemToWnd(hWnd);}
+ void WndToMemToDB(HWND hWnd, CString &sModule, CString *sDBSettingPrefix = NULL) {WndToMem(hWnd); MemToDB(sModule, sDBSettingPrefix);}
+ virtual void CleanDBSettings(CString &sModule, CString *sDBSettingPrefix = NULL) {db_unset(NULL, sModule, sDBSettingPrefix ? (*sDBSettingPrefix + sDBSetting) : sDBSetting);}; // TODO: also set Value to DefValue?
+
+ virtual void SetValue(int Value) {Modified = true;}
+ virtual void SetDefValue(int DefValue) {}
+ virtual int GetValue() {return 0;}
+ virtual int GetDefValue() {return 0;}
+ int GetDBValue(CString &sModule, CString *sDBSettingPrefix = NULL) {DBToMem(sModule, sDBSettingPrefix); return GetValue();}
+ void SetDBValue(CString &sModule, int Value, CString *sDBSettingPrefix = NULL) {SetValue(Value); MemToDB(sModule, sDBSettingPrefix);}
+ int GetDBValueCopy(CString &sModule, CString *sDBSettingPrefix = NULL) {COptItem* Item = Copy(); Item->DBToMem(sModule, sDBSettingPrefix); int Value = Item->GetValue(); delete Item; return Value;} // retrieves DB value, but doesn't affect current page/item state; beware! it doesn't work with string values / other dynamic pointers
+ void SetDBValueCopy(CString &sModule, int Value, CString *sDBSettingPrefix = NULL) {COptItem* Item = Copy(); Item->SetValue(Value); Item->MemToDB(sModule, sDBSettingPrefix); delete Item;}
+ int GetWndValue(HWND hWnd) {WndToMem(hWnd); return GetValue();}
+ void SetWndValue(HWND hWnd, int Value) {SetValue(Value); MemToWnd(hWnd);}
+ void SetDlgItemID(int DlgItemID) {this->DlgItemID = DlgItemID;}
+ bool GetModified() {return Modified;}
+ void SetModified(bool Modified) {this->Modified = Modified;}
+
+ void Enable(int Enabled) {this->Enabled = Enabled;}
+ int GetParam() {return lParam;}
+ int GetID() {return DlgItemID;}
+
+// virtual COptItem& operator = (const COptItem& Item) {return *this;};
+ virtual COptItem* Copy() {_ASSERT(0); return NULL;} // Attention! Free Copy() result when it's not needed anymore!
+
+ CString sDBSetting;
+
+protected:
+ int GetIntDBVal(CString &sModule, int bSigned = false, CString *sDBSettingPrefix = NULL);
+ void SetIntDBVal(CString &sModule, int Value, CString *sDBSettingPrefix = NULL);
+ TCString GetStrDBVal(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void SetStrDBVal(CString &sModule, TCString &Str, CString *sDBSettingPrefix = NULL);
+
+ int DlgItemID;
+ int Enabled;
+ bool ReadOnly;
+ bool Modified;
+ int nValueSize; // maximum pValue size in bytes
+ int lParam;
+};
+
+
+class COptItem_Generic : public COptItem
+{
+public:
+ COptItem_Generic() {}
+ COptItem_Generic(int DlgItemID, int lParam = 0): COptItem(DlgItemID, NULL, 0, lParam) {}
+ virtual COptItem* Copy() {return new COptItem_Generic(*this);}
+};
+
+
+class COptItem_Edit : public COptItem
+{
+public:
+ COptItem_Edit() {}
+ COptItem_Edit(int DlgItemID, char *szDBSetting, int nMaxLen, TCHAR *szDefValue, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nMaxLen, lParam, ReadOnly), sDefValue(szDefValue) {}
+// COptItem_Edit(const COptItem_Edit &Item): sDefValue(Item.sDefValue), sValue(Item.sValue) {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {sValue = GetStrDBVal(sModule, sDBSettingPrefix); COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {SetStrDBVal(sModule, sValue, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {GetDlgItemText(hWnd, DlgItemID, sValue.GetBuffer(nValueSize), nValueSize); sValue.ReleaseBuffer(); COptItem::MemToWnd(hWnd);}
+ void MemToWnd(HWND hWnd) {SetDlgItemText(hWnd, DlgItemID, sValue); COptItem::MemToWnd(hWnd);}
+ void SetValue(int Value) {sValue = *(TCString*)Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {sDefValue = *(TCString*)DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return (int)&sValue;}
+ int GetDefValue() {return (int)&sDefValue;}
+
+// COptItem_Edit& operator = (const COptItem_Edit& Item) {return *this;};
+ virtual COptItem* Copy() {return new COptItem_Edit(*this);}
+
+ TCString sDefValue;
+ TCString sValue;
+};
+
+
+class COptItem_IntEdit : public COptItem
+{
+public:
+ COptItem_IntEdit() {}
+ COptItem_IntEdit(int DlgItemID, char *szDBSetting, int nValueSize = DBVT_BYTE, int bSigned = true, int DefValue = 0, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nValueSize, lParam, ReadOnly), DefValue(DefValue), Value(0), bSigned(bSigned) {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {Value = GetIntDBVal(sModule, bSigned, sDBSettingPrefix); COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {SetIntDBVal(sModule, Value, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {Value = GetDlgItemInt(hWnd, DlgItemID, NULL, bSigned); COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {SetDlgItemInt(hWnd, DlgItemID, Value, bSigned); COptItem::MemToWnd(hWnd);}
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_IntEdit(*this);}
+
+ int DefValue;
+ int Value;
+ int bSigned;
+};
+
+
+class COptItem_Checkbox : public COptItem
+{
+public:
+ COptItem_Checkbox() {}
+ COptItem_Checkbox(int DlgItemID, char *szDBSetting, int nValueSize = DBVT_BYTE, int DefValue = 0, int ValueMask = 0, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nValueSize, lParam, ReadOnly), DefValue(DefValue), Value(0), ValueMask(ValueMask) {}
+
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void WndToMem(HWND hWnd);
+ void MemToWnd(HWND hWnd);
+
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_Checkbox(*this);}
+
+ int Value;
+ int DefValue;
+ int ValueMask;
+};
+
+
+class COptItem_Radiobutton : public COptItem
+{
+public:
+ COptItem_Radiobutton() {}
+ COptItem_Radiobutton(int DlgItemID, char *szDBSetting, int nValueSize, int DefValue, int ValueMask, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nValueSize, lParam, ReadOnly), DefValue(DefValue), Value(0), ValueMask(ValueMask) {}
+
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {Value = (GetIntDBVal(sModule, false, sDBSettingPrefix) == ValueMask) ? BST_CHECKED : BST_UNCHECKED; COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {if ((Value == BST_CHECKED)) SetIntDBVal(sModule, ValueMask, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {Value = IsDlgButtonChecked(hWnd, DlgItemID); COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {CheckDlgButton(hWnd, DlgItemID, Value); COptItem::MemToWnd(hWnd);}
+
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_Radiobutton(*this);}
+
+ int Value;
+ int DefValue;
+ int ValueMask;
+};
+
+
+class COptItem_Combobox : public COptItem
+{
+public:
+ COptItem_Combobox() {}
+ COptItem_Combobox(int DlgItemID, char *szDBSetting, int nValueSize = DBVT_BYTE, int DefValue = 0, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nValueSize, lParam, ReadOnly), DefValue(DefValue), Value(0) {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {Value = GetIntDBVal(sModule, false, sDBSettingPrefix); COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {SetIntDBVal(sModule, Value, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {Value = SendDlgItemMessage(hWnd, DlgItemID, CB_GETITEMDATA, (WPARAM)SendDlgItemMessage(hWnd, DlgItemID, CB_GETCURSEL, 0, 0), 0); COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {SendDlgItemMessage(hWnd, DlgItemID, CB_SETCURSEL, Value, 0); COptItem::MemToWnd(hWnd);}
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_Combobox(*this);}
+
+ int DefValue;
+ int Value;
+};
+
+
+class COptItem_Colourpicker : public COptItem
+{
+public:
+ COptItem_Colourpicker() {}
+ COptItem_Colourpicker(int DlgItemID, char *szDBSetting, int DefValue = 0, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, DBVT_DWORD, lParam, ReadOnly), DefValue(DefValue), Value(0) {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {Value = GetIntDBVal(sModule, false, sDBSettingPrefix); COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {SetIntDBVal(sModule, Value, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {Value = SendDlgItemMessage(hWnd, DlgItemID, CPM_GETCOLOUR, 0, 0); COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {SendDlgItemMessage(hWnd, DlgItemID, CPM_SETCOLOUR, 0, Value); COptItem::MemToWnd(hWnd);}
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_Colourpicker(*this);}
+
+ DWORD DefValue;
+ DWORD Value;
+};
+
+
+class COptItem_Slider : public COptItem
+{
+public:
+ COptItem_Slider() {}
+ COptItem_Slider(int DlgItemID, char *szDBSetting, int nValueSize = DBVT_BYTE, int DefValue = 0, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nValueSize, lParam, ReadOnly), DefValue(DefValue), Value(0) {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {Value = GetIntDBVal(sModule, false, sDBSettingPrefix); COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {SetIntDBVal(sModule, Value, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {Value = SendDlgItemMessage(hWnd, DlgItemID, TBM_GETPOS, 0, 0); COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {SendDlgItemMessage(hWnd, DlgItemID, TBM_SETPOS, true, Value); COptItem::MemToWnd(hWnd);}
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_Slider(*this);}
+
+ int DefValue;
+ int Value;
+};
+
+
+class COptItem_IntDBSetting : public COptItem
+{
+public:
+ COptItem_IntDBSetting() {}
+ COptItem_IntDBSetting(int DlgItemID, char *szDBSetting, int nValueSize = DBVT_BYTE, int bSigned = true, int DefValue = 0, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nValueSize, lParam, ReadOnly), DefValue(DefValue), Value(0), bSigned(bSigned) {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {Value = GetIntDBVal(sModule, bSigned, sDBSettingPrefix); COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {SetIntDBVal(sModule, Value, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {COptItem::MemToWnd(hWnd);}
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_IntDBSetting(*this);}
+
+ int Value;
+ int DefValue;
+ int bSigned;
+};
+
+
+class COptItem_BitDBSetting : public COptItem
+{
+public:
+ COptItem_BitDBSetting() {}
+ COptItem_BitDBSetting(int DlgItemID, char *szDBSetting, int nValueSize = DBVT_BYTE, int DefValue = 0, int ValueMask = 0, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nValueSize, lParam, ReadOnly), DefValue(DefValue), Value(0), ValueMask(ValueMask) {}
+
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void WndToMem(HWND hWnd) {COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {COptItem::MemToWnd(hWnd);}
+
+ void SetValue(int Value) {this->Value = Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return Value;}
+ int GetDefValue() {return DefValue;}
+ virtual COptItem* Copy() {return new COptItem_BitDBSetting(*this);}
+
+ int Value;
+ int DefValue;
+ int ValueMask;
+};
+
+
+class COptItem_StrDBSetting : public COptItem
+{
+public:
+ COptItem_StrDBSetting() {}
+ COptItem_StrDBSetting(int DlgItemID, char *szDBSetting, int nMaxLen, TCHAR *szDefValue, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, nMaxLen, lParam, ReadOnly), sDefValue(szDefValue) {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL) {sValue = GetStrDBVal(sModule, sDBSettingPrefix); COptItem::DBToMem(sModule, sDBSettingPrefix);}
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL) {SetStrDBVal(sModule, sValue, sDBSettingPrefix); COptItem::MemToDB(sModule, sDBSettingPrefix);}
+ void WndToMem(HWND hWnd) {COptItem::WndToMem(hWnd);}
+ void MemToWnd(HWND hWnd) {COptItem::MemToWnd(hWnd);}
+ void SetValue(int Value) {sValue = *(TCString*)Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {sDefValue = *(TCString*)DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return (int)&sValue;}
+ int GetDefValue() {return (int)&sDefValue;}
+ virtual COptItem* Copy() {return new COptItem_StrDBSetting(*this);}
+
+ TCString sDefValue;
+ TCString sValue;
+};
+
+
+// Tree item flags
+#define TIF_GROUP 1 // is a group
+#define TIF_EXPANDED 2 // item is expanded
+#define TIF_ENABLED 4 // item is checked (has sense when the tree has checkboxes)
+#define TIF_ROOTITEM 0x80 // item is a root item
+
+class CBaseTreeItem
+{
+public:
+ CBaseTreeItem();
+ CBaseTreeItem(TCString Title, int ID, int Flags): Title(Title), ID(ID), Flags(Flags), hItem(NULL) {}
+
+ TCString Title;
+ int ID;
+ int Flags;
+ HTREEITEM hItem;
+};
+
+class CTreeItem : public CBaseTreeItem
+{
+public:
+ CTreeItem();
+ CTreeItem(TCString Title, int ParentID, int ID, int Flags = 0, TCString User_Str1 = NULL):
+ CBaseTreeItem(Title, ID, Flags & ~TIF_ROOTITEM), ParentID(ParentID), User_Str1(User_Str1) {}
+
+ int ParentID;
+ TCString User_Str1;
+};
+
+class CTreeRootItem : public CBaseTreeItem
+{
+public:
+ CTreeRootItem();
+ CTreeRootItem(TCString Title, int ID, int Flags): CBaseTreeItem(Title, ID, Flags | TIF_ROOTITEM) {}
+};
+
+typedef TMyArray<CTreeItem> TreeItemArray;
+typedef TMyArray<CTreeRootItem> TreeRootItemArray;
+
+#define TREECTRL_ROOTORDEROFFS -2
+#define ROOT_INDEX_TO_ORDER(i) (TREECTRL_ROOTORDEROFFS - (i))
+#define ROOT_ORDER_TO_INDEX(i) (TREECTRL_ROOTORDEROFFS - (i))
+
+#define TREEITEMTITLE_MAXLEN 128
+#define TREEITEM_DBSTR_TITLE "Title"
+#define TREEITEM_DBSTR_PARENT "Parent"
+#define TREEITEM_DBSTR_ORDER "Order"
+#define TREEITEM_DBSTR_FLAGS "Flags"
+
+// Tree control flags
+#define TREECTRL_FLAG_IS_SINGLE_LEVEL 1 // means that the tree items can't have children, i.e. the tree is a plain list of items
+#define TREECTRL_FLAG_HAS_CHECKBOXES 2
+//#define TREECTRL_FLAG_UNORDERED 4 TODO?
+
+class COptItem_TreeCtrl : public COptItem
+{
+public:
+ COptItem_TreeCtrl() {}
+ COptItem_TreeCtrl(int DlgItemID, char *szDBSetting, TreeItemArray &DefValue, TreeRootItemArray RootItems, int lParam = 0, CString User_Str1_DBName = NULL, bool ReadOnly = false, int TreeFlags = 0): COptItem(DlgItemID, szDBSetting, DBVT_DWORD, lParam, ReadOnly), DefValue(DefValue), RootItems(RootItems), User_Str1_DBName(User_Str1_DBName), TreeFlags(TreeFlags)
+ {
+ if (TreeFlags & TREECTRL_FLAG_IS_SINGLE_LEVEL)
+ {
+ _ASSERT(!RootItems.GetSize()); // there can't be any root items when the tree is a plain list
+ this->RootItems.AddElem(CTreeRootItem(_T(""), 0, TIF_EXPANDED)); // TODO??
+ this->RootItems[0].hItem = TVI_ROOT;
+ }
+ }
+ ~COptItem_TreeCtrl() {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void WndToMem(HWND hWnd);
+ void MemToWnd(HWND hWnd);
+ void CleanDBSettings(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void SetValue(int Value) {this->Value = *(TreeItemArray*)Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = *(TreeItemArray*)DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return (int)&Value;}
+ int GetDefValue() {return (int)&DefValue;}
+ virtual COptItem* Copy() {return new COptItem_TreeCtrl(*this);}
+
+ int IDToOrder(int ID);
+ int hItemToOrder(HTREEITEM hItem);
+ int GenerateID();
+ int GetSelectedItemID(HWND hWnd);
+ void Delete(HWND hWnd, int ID);
+ CTreeItem* InsertItem(HWND hWnd, CTreeItem &Item);
+ void MoveItem(HWND hWnd, HTREEITEM hItem, HTREEITEM hMoveTo);
+
+ TreeItemArray DefValue, Value;
+ TreeRootItemArray RootItems;
+ CString User_Str1_DBName;
+ int TreeFlags;
+
+protected:
+ void RecursiveDelete(HWND hWnd, int I);
+ int RecursiveMove(int ItemOrder, int ParentID, int InsertAtOrder);
+};
+
+
+class CListItem
+{
+public:
+ CListItem();
+ CListItem(TCString Text): Text(Text) {}
+
+ TCString Text;
+};
+
+typedef TMyArray<CListItem> ListItemArray;
+
+#define LISTITEM_DBSTR_TEXT "Text"
+
+class COptItem_ListCtrl : public COptItem
+{
+public:
+ COptItem_ListCtrl() {}
+ COptItem_ListCtrl(int DlgItemID, char *szDBSetting, ListItemArray &DefValue, int lParam = 0, bool ReadOnly = false): COptItem(DlgItemID, szDBSetting, DBVT_DWORD, lParam, ReadOnly), DefValue(DefValue) {}
+ ~COptItem_ListCtrl() {}
+ void DBToMem(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void MemToDB(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void WndToMem(HWND hWnd);
+ void MemToWnd(HWND hWnd);
+ void CleanDBSettings(CString &sModule, CString *sDBSettingPrefix = NULL);
+ void SetValue(int Value) {this->Value = *(ListItemArray*)Value; COptItem::SetValue(Value);}
+ void SetDefValue(int DefValue) {this->DefValue = *(ListItemArray*)DefValue; COptItem::SetDefValue(DefValue);}
+ int GetValue() {return (int)&Value;}
+ int GetDefValue() {return (int)&DefValue;}
+ virtual COptItem* Copy() {return new COptItem_ListCtrl(*this);}
+
+ int GetSelectedItemID(HWND hWnd); // returns -1 if there's no selection
+ int SetSelectedItemID(HWND hWnd, int ID);
+ void Delete(HWND hWnd, int ID);
+ CListItem* InsertItem(HWND hWnd, int ID, CListItem &Item);
+ void ModifyItem(HWND hWnd, int ID, CListItem &Item);
+
+ ListItemArray DefValue, Value;
+};
+
+
+class COptPage
+{
+public:
+ COptPage(): hWnd(NULL), sDBSettingPrefix("") {}
+ COptPage(char *szModule, HWND hWnd, CString sDBSettingPrefix = ""): sModule(szModule), hWnd(hWnd), sDBSettingPrefix(sDBSettingPrefix) {}
+ COptPage(const COptPage &Item);
+ ~COptPage();
+
+ void DBToMem();
+ void MemToDB();
+ void MemToPage(int OnlyEnable = 0);
+ void PageToMem();
+ void DBToMemToPage() {DBToMem(); MemToPage();}
+ void PageToMemToDB() {PageToMem(); MemToDB();}
+ void CleanDBSettings();
+
+ COptItem *Find(int DlgItemID);
+ int GetValue(int DlgItemID) {return Find(DlgItemID)->GetValue();}
+ void SetValue(int DlgItemID, int Value) {Find(DlgItemID)->SetValue(Value);}
+ int GetDBValue(int DlgItemID) {return Find(DlgItemID)->GetDBValue(sModule, &sDBSettingPrefix);}
+ void SetDBValue(int DlgItemID, int Value) {Find(DlgItemID)->SetDBValue(sModule, Value, &sDBSettingPrefix);}
+ int GetDBValueCopy(int DlgItemID) {return Find(DlgItemID)->GetDBValueCopy(sModule, &sDBSettingPrefix);}
+ void SetDBValueCopy(int DlgItemID, int Value) {Find(DlgItemID)->SetDBValueCopy(sModule, Value, &sDBSettingPrefix);}
+ int GetWndValue(int DlgItemID) {return Find(DlgItemID)->GetWndValue(hWnd);}
+ void SetWndValue(int DlgItemID, int Value) {Find(DlgItemID)->SetWndValue(hWnd, Value);}
+ HWND GetWnd() {return hWnd;}
+ void SetWnd(HWND hWnd) {_ASSERT(!this->hWnd || !hWnd); this->hWnd = hWnd;}
+ void Enable(int DlgItemID, int Enabled) {Find(DlgItemID)->Enable(Enabled);}
+ bool GetModified();
+ void SetModified(bool Modified);
+
+ COptPage& operator = (const COptPage& Page);
+
+ HWND hWnd;
+ CString sModule, sDBSettingPrefix;
+ TMyArray<COptItem*, COptItem*> Items;
+};
diff --git a/plugins/NewAwaySysMod/src/Path.h b/plugins/NewAwaySysMod/src/Path.h
new file mode 100644
index 0000000000..e108a919e5
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Path.h
@@ -0,0 +1,43 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#pragma once
+
+#include "m_utils.h"
+#include "CString.h"
+
+
+__inline TCString Path_ToRelative(TCString &Path)
+{
+ TCString Str;
+ Str.GetBuffer(MAX_PATH);
+ CallService(MS_UTILS_PATHTORELATIVET, (WPARAM)(TCHAR*)Path, (LPARAM)(TCHAR*)Str);
+ Str.ReleaseBuffer();
+ return Str;
+}
+
+
+__inline TCString Path_ToAbsolute(TCString &Path)
+{
+ TCString Str;
+ Str.GetBuffer(MAX_PATH);
+ CallService(MS_UTILS_PATHTOABSOLUTET, (WPARAM)(TCHAR*)Path, (LPARAM)(TCHAR*)Str);
+ Str.ReleaseBuffer();
+ return Str;
+}
diff --git a/plugins/NewAwaySysMod/src/Properties.cpp b/plugins/NewAwaySysMod/src/Properties.cpp
new file mode 100644
index 0000000000..d1e87d0b8c
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Properties.cpp
@@ -0,0 +1,411 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (C) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+#include "Properties.h"
+
+CProtoStates g_ProtoStates;
+
+void ResetContactSettingsOnStatusChange(MCONTACT hContact)
+{
+ db_unset(hContact, MOD_NAME, DB_REQUESTCOUNT);
+ db_unset(hContact, MOD_NAME, DB_SENDCOUNT);
+ db_unset(hContact, MOD_NAME, DB_MESSAGECOUNT);
+}
+
+
+void ResetSettingsOnStatusChange(const char *szProto = NULL, int bResetPersonalMsgs = false, int Status = 0)
+{
+ if (bResetPersonalMsgs)
+ { // bResetPersonalMsgs &&= g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_SAVEPERSONALMSGS);
+ bResetPersonalMsgs = !g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_SAVEPERSONALMSGS);
+ }
+ MCONTACT hContact = db_find_first();
+ while (hContact)
+ {
+ const char *szCurProto;
+ if (!szProto || ((szCurProto = (const char*)CallService(MS_PROTO_GETCONTACTBASEPROTO, (WPARAM)hContact, 0)) && !lstrcmpA(szProto, szCurProto)))
+ {
+ ResetContactSettingsOnStatusChange(hContact);
+ if (bResetPersonalMsgs)
+ {
+ CContactSettings(Status, hContact).SetMsgFormat(SMF_PERSONAL, NULL); // TODO: delete only when SAM dialog opens?
+ }
+ }
+ hContact = db_find_next(hContact);
+ }
+}
+
+
+CProtoState::CStatus& CProtoState::CStatus::operator = (int Status)
+{
+ _ASSERT(Status >= ID_STATUS_OFFLINE && Status <= ID_STATUS_OUTTOLUNCH);
+ if (Status < ID_STATUS_OFFLINE || Status > ID_STATUS_OUTTOLUNCH)
+ {
+ return *this; // ignore status change if the new status is unknown
+ }
+ bool bModified = false;
+ if (szProto)
+ {
+ if (this->Status != Status)
+ {
+ this->Status = Status;
+ (*GrandParent)[szProto].AwaySince.Reset();
+ ResetSettingsOnStatusChange(szProto, true, Status);
+ bModified = true;
+ }
+ if ((*GrandParent)[szProto].TempMsg.IsSet())
+ {
+ (*GrandParent)[szProto].TempMsg.Unset();
+ bModified = true;
+ }
+ } else // global status change
+ {
+ int I;
+ int bStatusModified = false;
+ for (I = 0; I < GrandParent->GetSize(); I++)
+ {
+ CProtoState &State = (*GrandParent)[I];
+ if (!db_get_b(NULL, State.GetProto(), "LockMainStatus", 0)) // if the protocol isn't locked
+ {
+ if (State.Status != Status)
+ {
+ State.Status.Status = Status; // "Status.Status" - changing Status directly to prevent recursive calls to the function
+ State.AwaySince.Reset();
+ bModified = true;
+ bStatusModified = true;
+ }
+ if (State.TempMsg.IsSet())
+ {
+ State.TempMsg.Unset();
+ bModified = true;
+ }
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_RESETPROTOMSGS))
+ {
+ CProtoSettings(State.szProto, Status).SetMsgFormat(SMF_PERSONAL, NULL);
+ }
+ }
+ }
+ if (bStatusModified)
+ {
+ ResetSettingsOnStatusChange(NULL, true, Status);
+ }
+ }
+ if (bModified && g_SetAwayMsgPage.GetWnd())
+ {
+ SendMessage(g_SetAwayMsgPage.GetWnd(), UM_SAM_PROTOSTATUSCHANGED, (WPARAM)(char*)szProto, 0);
+ }
+ return *this;
+}
+
+
+void CProtoState::CAwaySince::Reset()
+{
+ GetLocalTime(&AwaySince);
+ if (GrandParent && !szProto)
+ {
+ int I;
+ for (I = 0; I < GrandParent->GetSize(); I++)
+ {
+ GetLocalTime((*GrandParent)[I].AwaySince);
+ }
+ }
+}
+
+
+void CContactSettings::SetMsgFormat(int Flags, TCString Message)
+{
+ if (Flags & SMF_PERSONAL)
+ { // set a personal message for a contact. also it's used to set global status message (hContact = NULL).
+ // if Message == NULL, then the function deletes the message.
+ CString DBSetting(StatusToDBSetting(Status, DB_STATUSMSG, IDC_MOREOPTDLG_PERSTATUSPERSONAL));
+ if (g_AutoreplyOptPage.GetDBValueCopy(IDC_REPLYDLG_RESETCOUNTERWHENSAMEICON) && GetMsgFormat(SMF_PERSONAL) != (const TCHAR*)Message)
+ {
+ ResetContactSettingsOnStatusChange(hContact);
+ }
+ if (Message != NULL)
+ {
+ db_set_ts(hContact, MOD_NAME, DBSetting, Message);
+ } else
+ {
+ db_unset(hContact, MOD_NAME, DBSetting);
+ }
+ }
+ if (Flags & (SMF_LAST | SMF_TEMPORARY))
+ {
+ _ASSERT(!hContact);
+ CProtoSettings().SetMsgFormat(Flags & (SMF_LAST | SMF_TEMPORARY), Message);
+ }
+}
+
+
+TCString CContactSettings::GetMsgFormat(int Flags, int *pOrder, char *szProtoOverride)
+// returns the requested message; sets Order to the order of the message in the message tree, if available; or to -1 otherwise.
+// szProtoOverride is needed only to get status message of the right protocol when the ICQ contact is on list, but not with the same protocol on which it requests the message - this way we can still get contact details.
+{
+ TCString Message = NULL;
+ if (pOrder)
+ {
+ *pOrder = -1;
+ }
+ if (Flags & GMF_PERSONAL)
+ { // try getting personal message (it overrides global)
+ Message = db_get_s(hContact, MOD_NAME, StatusToDBSetting(Status, DB_STATUSMSG, IDC_MOREOPTDLG_PERSTATUSPERSONAL), (TCHAR*)NULL);
+ }
+ if (Flags & (GMF_LASTORDEFAULT | GMF_PROTOORGLOBAL | GMF_TEMPORARY) && Message.IsEmpty())
+ {
+ char *szProto = szProtoOverride ? szProtoOverride : (hContact ? GetContactProto(hContact) : NULL);
+ if (Flags & (GMF_LASTORDEFAULT | GMF_PROTOORGLOBAL))
+ { // we mustn't pass here by GMF_TEMPORARY flag, as otherwise we'll handle GMF_TEMPORARY | GMF_PERSONAL combination incorrectly, which is supposed to get only per-contact messages, and at the same time also may be used with NULL contact to get the global status message
+ Message = CProtoSettings(szProto).GetMsgFormat(Flags, pOrder);
+ } else if (!hContact)
+ { // handle global temporary message too
+ if (g_ProtoStates[szProto].TempMsg.IsSet())
+ {
+ Message = g_ProtoStates[szProto].TempMsg;
+ }
+ }
+ }
+ return Message;
+}
+
+
+void CProtoSettings::SetMsgFormat(int Flags, TCString Message)
+{
+ if (Flags & (SMF_TEMPORARY | SMF_PERSONAL) && g_AutoreplyOptPage.GetDBValueCopy(IDC_REPLYDLG_RESETCOUNTERWHENSAMEICON) && GetMsgFormat(Flags & (SMF_TEMPORARY | SMF_PERSONAL)) != (const TCHAR*)Message)
+ {
+ ResetSettingsOnStatusChange(szProto);
+ }
+ if (Flags & SMF_TEMPORARY)
+ {
+ _ASSERT(!Status || Status == g_ProtoStates[szProto].Status);
+ g_ProtoStates[szProto].TempMsg = (szProto || Message != NULL) ? Message : CProtoSettings(NULL, Status).GetMsgFormat(GMF_LASTORDEFAULT);
+ }
+ if (Flags & SMF_PERSONAL)
+ { // set a "personal" message for a protocol. also it's used to set global status message (hContact = NULL).
+ // if Message == NULL, then we'll use the "default" message - i.e. it's either the global message for szProto != NULL (we delete the per-proto DB setting), or it's just a default message for a given status for szProto == NULL.
+ g_ProtoStates[szProto].TempMsg.Unset();
+ CString DBSetting(ProtoStatusToDBSetting(DB_STATUSMSG, IDC_MOREOPTDLG_PERSTATUSPROTOMSGS));
+ if (Message != NULL)
+ {
+ db_set_ts(NULL, MOD_NAME, DBSetting, Message);
+ } else
+ {
+ if (!szProto)
+ {
+ db_set_ts(NULL, MOD_NAME, DBSetting, CProtoSettings(NULL, Status).GetMsgFormat(GMF_LASTORDEFAULT)); // global message can't be NULL; we can use an empty string instead if it's really necessary
+ } else
+ {
+ db_unset(NULL, MOD_NAME, DBSetting);
+ }
+ }
+ }
+ if (Flags & SMF_LAST)
+ {
+ COptPage MsgTreeData(g_MsgTreePage);
+ COptItem_TreeCtrl *TreeCtrl = (COptItem_TreeCtrl*)MsgTreeData.Find(IDV_MSGTREE);
+ TreeCtrl->DBToMem(CString(MOD_NAME));
+ int RecentGroupID = GetRecentGroupID(Status);
+ if (RecentGroupID == -1)
+ { // we didn't find the group, it also means that we're using per status messages; so we need to create it
+ TreeCtrl->Value.AddElem(CTreeItem(Status ? (const TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, Status, GSMDF_TCHAR) : MSGTREE_RECENT_OTHERGROUP, g_Messages_RecentRootID, RecentGroupID = TreeCtrl->GenerateID(), TIF_GROUP));
+ TreeCtrl->SetModified(true);
+ }
+ int I;
+ TCString Title(_T(""));
+ for (I = 0; I < TreeCtrl->Value.GetSize(); I++) // try to find an identical message in the same group (to prevent saving multiple identical messages), or at least if we'll find an identical message somewhere else, then we'll use its title for our new message
+ {
+ if (!(TreeCtrl->Value[I].Flags & TIF_GROUP) && TreeCtrl->Value[I].User_Str1 == (const TCHAR*)Message)
+ {
+ if (TreeCtrl->Value[I].ParentID == RecentGroupID)
+ { // found it in the same group
+ int GroupOrder = TreeCtrl->IDToOrder(RecentGroupID);
+ TreeCtrl->Value.MoveElem(I, (GroupOrder >= 0) ? (GroupOrder + 1) : 0); // now move it to the top of recent messages list
+ TreeCtrl->SetModified(true);
+ break; // no reason to search for anything else
+ } else if (Title.IsEmpty())
+ { // it's not in the same group, but at least we'll use its title
+ Title = TreeCtrl->Value[I].Title;
+ }
+ }
+ }
+ if (I == TreeCtrl->Value.GetSize())
+ { // we didn't find an identical message in the same group, so we'll add our new message here
+ if (Title.IsEmpty())
+ { // didn't find a title for our message either
+ if (Message.GetLen() > MRM_MAX_GENERATED_TITLE_LEN)
+ {
+ Title = Message.Left(MRM_MAX_GENERATED_TITLE_LEN - 3) + _T("...");
+ } else
+ {
+ Title = Message;
+ }
+ TCHAR *p = Title.GetBuffer();
+ while (*p) // remove "garbage"
+ {
+ if (!(p = _tcspbrk(p, _T("\r\n\t"))))
+ {
+ break;
+ }
+ *p++ = ' ';
+ }
+ Title.ReleaseBuffer();
+ }
+ int GroupOrder = TreeCtrl->IDToOrder(RecentGroupID);
+ TreeCtrl->Value.InsertElem(CTreeItem(Title, RecentGroupID, TreeCtrl->GenerateID(), 0, Message), (GroupOrder >= 0) ? (GroupOrder + 1) : 0);
+ TreeCtrl->SetModified(true);
+ }
+ // now clean up here
+ int MRMNum = 0;
+ int MaxMRMNum = g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_RECENTMSGSCOUNT);
+ for (I = 0; I < TreeCtrl->Value.GetSize(); I++)
+ {
+ if (TreeCtrl->Value[I].ParentID == RecentGroupID)
+ { // found a child of our group
+ if (TreeCtrl->Value[I].Flags & TIF_GROUP || ++MRMNum > MaxMRMNum) // what groups are doing here?! :))
+ {
+ TreeCtrl->Value.RemoveElem(I);
+ TreeCtrl->SetModified(true);
+ I--;
+ }
+ }
+ }
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_PERSTATUSMRM))
+ { // if we're saving recent messages per status, then remove any messages that were left at the recent messages' root
+ for (I = 0; I < TreeCtrl->Value.GetSize(); I++)
+ {
+ if (TreeCtrl->Value[I].ParentID == g_Messages_RecentRootID)
+ {
+ if (!(TreeCtrl->Value[I].Flags & TIF_GROUP))
+ {
+ TreeCtrl->Value.RemoveElem(I);
+ TreeCtrl->SetModified(true);
+ I--;
+ }
+ }
+ }
+ }
+ TreeCtrl->MemToDB(CString(MOD_NAME));
+ }
+}
+
+
+TCString CProtoSettings::GetMsgFormat(int Flags, int *pOrder)
+// returns the requested message; sets Order to the order of the message in the message tree, if available; or to -1 otherwise.
+{
+ TCString Message = NULL;
+ if (pOrder)
+ {
+ *pOrder = -1;
+ }
+ if (Flags & GMF_TEMPORARY)
+ {
+ _ASSERT(!Status || Status == g_ProtoStates[szProto].Status);
+ if (g_ProtoStates[szProto].TempMsg.IsSet())
+ {
+ Message = g_ProtoStates[szProto].TempMsg;
+ Flags &= ~GMF_PERSONAL; // don't allow personal message to overwrite our NULL temporary message
+ }
+ }
+ if (Flags & GMF_PERSONAL && Message == NULL)
+ { // try getting personal message (it overrides global)
+ Message = db_get_s(NULL, MOD_NAME, ProtoStatusToDBSetting(DB_STATUSMSG, IDC_MOREOPTDLG_PERSTATUSPROTOMSGS), (TCHAR*)NULL);
+ }
+ if (Flags & GMF_PROTOORGLOBAL && Message == NULL)
+ {
+ Message = CProtoSettings().GetMsgFormat(GMF_PERSONAL | (Flags & GMF_TEMPORARY), pOrder);
+ return (Message == NULL) ? _T("") : Message; // global message can't be NULL
+ }
+ if (Flags & GMF_LASTORDEFAULT && Message == NULL)
+ { // try to get the last or default message, depending on current settings
+ COptPage MsgTreeData(g_MsgTreePage);
+ COptItem_TreeCtrl *TreeCtrl = (COptItem_TreeCtrl*)MsgTreeData.Find(IDV_MSGTREE);
+ TreeCtrl->DBToMem(CString(MOD_NAME));
+ Message = NULL;
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_USELASTMSG)) // if using last message by default...
+ {
+ Message = db_get_s(NULL, MOD_NAME, ProtoStatusToDBSetting(DB_STATUSMSG, IDC_MOREOPTDLG_PERSTATUSPROTOMSGS), (TCHAR*)NULL); // try per-protocol message first
+ if (Message.IsEmpty())
+ {
+ Message = NULL; // to be sure it's NULL, not "" - as we're checking 'Message == NULL' later
+ int RecentGroupID = GetRecentGroupID(Status);
+ if (RecentGroupID != -1)
+ {
+ int I;
+ for (I = 0; I < TreeCtrl->Value.GetSize(); I++) // find first message in the group
+ {
+ if (TreeCtrl->Value[I].ParentID == RecentGroupID && !(TreeCtrl->Value[I].Flags & TIF_GROUP))
+ {
+ Message = TreeCtrl->Value[I].User_Str1;
+ if (pOrder)
+ {
+ *pOrder = I;
+ }
+ break;
+ }
+ }
+ }
+ }
+ } // else, if using default message by default...
+ if (Message == NULL) // ...or we didn't succeed retrieving the last message
+ { // get default message for this status
+ int DefMsgID = -1;
+ static struct {
+ int DBSetting, Status;
+ } DefMsgDlgItems[] = {
+ IDS_MESSAGEDLG_DEF_ONL, ID_STATUS_ONLINE,
+ IDS_MESSAGEDLG_DEF_AWAY, ID_STATUS_AWAY,
+ IDS_MESSAGEDLG_DEF_NA, ID_STATUS_NA,
+ IDS_MESSAGEDLG_DEF_OCC, ID_STATUS_OCCUPIED,
+ IDS_MESSAGEDLG_DEF_DND, ID_STATUS_DND,
+ IDS_MESSAGEDLG_DEF_FFC, ID_STATUS_FREECHAT,
+ IDS_MESSAGEDLG_DEF_INV, ID_STATUS_INVISIBLE,
+ IDS_MESSAGEDLG_DEF_OTP, ID_STATUS_ONTHEPHONE,
+ IDS_MESSAGEDLG_DEF_OTL, ID_STATUS_OUTTOLUNCH
+ };
+ int I;
+ for (I = 0; I < lengthof(DefMsgDlgItems); I++)
+ {
+ if (DefMsgDlgItems[I].Status == Status)
+ {
+ DefMsgID = MsgTreeData.GetDBValue(DefMsgDlgItems[I].DBSetting);
+ break;
+ }
+ }
+ if (DefMsgID == -1)
+ {
+ DefMsgID = MsgTreeData.GetDBValue(IDS_MESSAGEDLG_DEF_AWAY); // use away message for unknown statuses
+ }
+ int Order = TreeCtrl->IDToOrder(DefMsgID); // this will return -1 in any case if something goes wrong
+ if (Order >= 0)
+ {
+ Message = TreeCtrl->Value[Order].User_Str1;
+ }
+ if (pOrder)
+ {
+ *pOrder = Order;
+ }
+ }
+ if (Message == NULL)
+ {
+ Message = _T(""); // last or default message can't be NULL.. otherwise ICQ won't reply to status message requests and won't notify us of status message requests at all
+ }
+ }
+ return Message;
+}
diff --git a/plugins/NewAwaySysMod/src/Properties.h b/plugins/NewAwaySysMod/src/Properties.h
new file mode 100644
index 0000000000..b484f4c2d1
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Properties.h
@@ -0,0 +1,597 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#pragma once
+
+#include "Common.h"
+#include "MsgTree.h"
+#include "ContactList.h"
+#include "statusmodes.h"
+
+#define DB_STATUSMSG "StatusMsg"
+#define DB_ENABLEREPLY "EnableReply"
+#define DB_IGNOREREQUESTS "IgnoreRequests"
+//#define DB_POPUPNOTIFY "UsePopups"
+#define DB_UNK_CONTACT_PREFIX "Unk" // DB_ENABLEREPLY, DB_IGNOREREQUESTS and DB_POPUPNOTIFY settings prefix for not-on-list contacts
+
+class _CWndUserData
+{
+public:
+ _CWndUserData(): MsgTree(NULL), CList(NULL) {}
+
+ CMsgTree *MsgTree;
+ CCList *CList;
+};
+
+
+class CWndUserData
+{
+public:
+ CWndUserData(HWND hWnd): hWnd(hWnd)
+ {
+ _ASSERT(hWnd);
+ dat = (_CWndUserData*)GetWindowLongPtr(hWnd, GWLP_USERDATA);
+ if (!dat)
+ {
+ dat = new _CWndUserData;
+ SetWindowLongPtr(hWnd, GWLP_USERDATA, (LONG_PTR)dat);
+ }
+ }
+
+ ~CWndUserData()
+ {
+ _ASSERT(dat == (_CWndUserData*)GetWindowLongPtr(hWnd, GWLP_USERDATA));
+ if (!dat->MsgTree && !dat->CList) // TODO: Uninitialized Memory Read on closing the options dialog - fix it
+ {
+ SetWindowLongPtr(hWnd, GWLP_USERDATA, NULL);
+ delete dat; // TODO: memory leak - this is never executed - fix it
+ }
+ }
+
+ CMsgTree *GetMsgTree() {return dat->MsgTree;}
+ void SetMsgTree(CMsgTree *MsgTree) {dat->MsgTree = MsgTree;}
+ CCList *GetCList() {return dat->CList;}
+ void SetCList(CCList *CList) {dat->CList = CList;}
+
+private:
+ HWND hWnd;
+ _CWndUserData *dat;
+};
+
+#define IL_SKINICON 0x80000000
+#define IL_PROTOICON 0x40000000
+
+#define ILI_NOICON (-1)
+#define ILI_EVENT_MESSAGE 0
+#define ILI_EVENT_URL 1
+#define ILI_EVENT_FILE 2
+#define ILI_PROTO_ONL 3
+#define ILI_PROTO_AWAY 4
+#define ILI_PROTO_NA 5
+#define ILI_PROTO_OCC 6
+#define ILI_PROTO_DND 7
+#define ILI_PROTO_FFC 8
+#define ILI_PROTO_INV 9
+#define ILI_PROTO_OTP 10
+#define ILI_PROTO_OTL 11
+#define ILI_DOT 12
+#define ILI_MSGICON 13
+#define ILI_IGNORE 14
+#define ILI_SOE_DISABLED 15
+#define ILI_SOE_ENABLED 16
+#define ILI_NEWMESSAGE 17
+#define ILI_NEWCATEGORY 18
+#define ILI_SAVE 19
+#define ILI_SAVEASNEW 20
+#define ILI_DELETE 21
+#define ILI_SETTINGS 22
+#define ILI_STATUS_OTHER 23
+
+static int Icons[] = {
+ SKINICON_EVENT_MESSAGE | IL_SKINICON, SKINICON_EVENT_URL | IL_SKINICON, SKINICON_EVENT_FILE | IL_SKINICON,
+ ID_STATUS_ONLINE | IL_PROTOICON, ID_STATUS_AWAY | IL_PROTOICON, ID_STATUS_NA | IL_PROTOICON, ID_STATUS_OCCUPIED | IL_PROTOICON, ID_STATUS_DND | IL_PROTOICON, ID_STATUS_FREECHAT | IL_PROTOICON, ID_STATUS_INVISIBLE | IL_PROTOICON, ID_STATUS_ONTHEPHONE | IL_PROTOICON, ID_STATUS_OUTTOLUNCH | IL_PROTOICON,
+ IDI_DOT, IDI_MSGICON, IDI_IGNORE, IDI_SOE_ENABLED, IDI_SOE_DISABLED, IDI_NEWMESSAGE, IDI_NEWCATEGORY, IDI_SAVE, IDI_SAVEASNEW, IDI_DELETE, IDI_SETTINGS, IDI_STATUS_OTHER
+};
+
+
+class CIconList
+{
+public:
+ ~CIconList()
+ {
+ int I;
+ for (I = 0; I < IconList.GetSize(); I++)
+ {
+ if (IconList[I])
+ {
+ DestroyIcon(IconList[I]);
+ }
+ }
+ }
+
+ HICON& operator [] (int nIndex) {return IconList[nIndex];}
+ void ReloadIcons()
+ {
+ int cxIcon = GetSystemMetrics(SM_CXSMICON);
+ int cyIcon = GetSystemMetrics(SM_CYSMICON);
+ int I;
+ for (I = 0; I < lengthof(Icons); I++)
+ {
+ if (IconList.GetSize() > I && IconList[I])
+ {
+ DestroyIcon(IconList[I]);
+ }
+ if (Icons[I] & IL_SKINICON)
+ {
+ IconList.SetAtGrow(I) = (HICON)CopyImage(LoadSkinnedIcon(Icons[I] & ~IL_SKINICON), IMAGE_ICON, cxIcon, cyIcon, LR_COPYFROMRESOURCE);
+ } else if (Icons[I] & IL_PROTOICON)
+ {
+ IconList.SetAtGrow(I) = (HICON)CopyImage(LoadSkinnedProtoIcon(NULL, Icons[I] & ~IL_PROTOICON), IMAGE_ICON, cxIcon, cyIcon, LR_COPYFROMRESOURCE);
+ } else
+ {
+ IconList.SetAtGrow(I) = (HICON)LoadImage(g_hInstance, MAKEINTRESOURCE(Icons[I]), IMAGE_ICON, cxIcon, cyIcon, 0);
+ }
+ }
+ }
+
+private:
+ TMyArray<HICON> IconList;
+};
+
+extern CIconList g_IconList;
+
+
+class CProtoStates;
+
+class CProtoState
+{
+public:
+ CProtoState(const char* szProto, CProtoStates* Parent): szProto(szProto), Parent(Parent), Status(szProto, Parent), AwaySince(szProto, Parent) {}
+
+ class CStatus
+ {
+ public:
+ CStatus(const char* szProto, CProtoStates* GrandParent): szProto(szProto), GrandParent(GrandParent), Status(ID_STATUS_OFFLINE) {}
+ CStatus& operator = (int Status);
+ operator int() {return Status;}
+ friend class CProtoState;
+ private:
+ int Status;
+ CString szProto;
+ CProtoStates* GrandParent;
+ } Status;
+
+ class CAwaySince
+ {
+ public:
+ CAwaySince(const char* szProto, CProtoStates* GrandParent): szProto(szProto), GrandParent(GrandParent) {Reset();}
+ void Reset();
+ operator LPSYSTEMTIME() {return &AwaySince;}
+ friend class CProtoState;
+ private:
+ SYSTEMTIME AwaySince;
+ CString szProto;
+ CProtoStates* GrandParent;
+ } AwaySince;
+
+ class CCurStatusMsg
+ {
+ public:
+ CCurStatusMsg() {*this = NULL;}
+ CCurStatusMsg& operator = (TCString Msg)
+ {
+ CurStatusMsg = Msg;
+ SYSTEMTIME st;
+ GetLocalTime(&st);
+ SystemTimeToFileTime(&st, (LPFILETIME)&LastUpdateTime); // I'm too lazy to declare FILETIME structure and then copy its data to absolutely equivalent ULARGE_INTEGER structure, so we'll just pass a pointer to the ULARGE_INTEGER directly ;-P
+ return *this;
+ }
+ operator TCString() {return CurStatusMsg;}
+ DWORD GetUpdateTimeDifference()
+ {
+ ULARGE_INTEGER CurTime;
+ SYSTEMTIME st;
+ GetLocalTime(&st);
+ SystemTimeToFileTime(&st, (LPFILETIME)&CurTime);
+ return (DWORD)((CurTime.QuadPart - LastUpdateTime.QuadPart) / 10000); // in milliseconds
+ }
+ private:
+ TCString CurStatusMsg;
+ ULARGE_INTEGER LastUpdateTime;
+ } CurStatusMsg;
+
+ class CTempMsg
+ { // we use temporary messages to keep user-defined per-protocol messages intact, when changing messages through MS_NAS_SETSTATE, or when autoaway becomes active etc.. temporary messages are automatically resetted when protocol status changes
+ public:
+ CTempMsg(): iIsSet(0) {}
+ CTempMsg& operator = (TCString Msg) {this->Msg = Msg; iIsSet = true; return *this;}
+ operator TCString()
+ {
+ _ASSERT(iIsSet);
+ return Msg;
+ }
+ void Unset() {iIsSet = false;}
+ int IsSet() {return iIsSet;}
+
+ private:
+ int iIsSet; // as we need TempMsg to support every possible value, including NULL and "", we'll use this variable to specify whether TempMsg is set
+ TCString Msg;
+ } TempMsg;
+
+ void SetParent(CProtoStates* Parent)
+ {
+ this->Parent = Parent;
+ Status.GrandParent = Parent;
+ AwaySince.GrandParent = Parent;
+ }
+
+ CString &GetProto() {return szProto;}
+
+//NightFox: fix?
+//private:
+public:
+ CString szProto;
+ CProtoStates* Parent;
+};
+
+
+class CProtoStates // this class stores all protocols' dynamic data
+{
+public:
+ CProtoStates() {}
+
+ CProtoStates(const CProtoStates &States) {*this = States;}
+ CProtoStates& operator = (const CProtoStates& States)
+ {
+ ProtoStates = States.ProtoStates;
+ int I;
+ for (I = 0; I < ProtoStates.GetSize(); I++)
+ {
+ ProtoStates[I].SetParent(this);
+ }
+ return *this;
+ }
+
+ CProtoState& operator[](const char *szProto)
+ {
+ int I;
+ for (I = 0; I < ProtoStates.GetSize(); I++)
+ {
+ if (ProtoStates[I].GetProto() == szProto)
+ {
+ return ProtoStates[I];
+ }
+ }
+ if (!szProto) // we need to be sure that we have _all_ protocols in the list, before dealing with global status, so we're adding them here.
+ {
+ int ProtoCount;
+ PROTOCOLDESCRIPTOR **proto;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&ProtoCount, (LPARAM)&proto);
+ int I;
+ for (I = 0; I < ProtoCount; I++)
+ {
+ if (proto[I]->type == PROTOTYPE_PROTOCOL)
+ {
+ (*this)[proto[I]->szName]; // add a protocol if it isn't in the list yet
+ }
+ }
+ }
+ return ProtoStates[ProtoStates.AddElem(CProtoState(szProto, this))];
+ }
+
+ friend class CProtoState;
+ friend class CProtoState::CStatus;
+ friend class CProtoState::CAwaySince;
+
+private:
+ CProtoState& operator[](int nIndex) {return ProtoStates[nIndex];}
+ int GetSize() {return ProtoStates.GetSize();}
+
+ TMyArray<CProtoState> ProtoStates;
+};
+
+extern CProtoStates g_ProtoStates;
+
+
+static struct
+{
+ int Status;
+ char *Setting;
+} StatusSettings[] = {
+ ID_STATUS_OFFLINE, "Off",
+ ID_STATUS_ONLINE, "Onl",
+ ID_STATUS_AWAY, "Away",
+ ID_STATUS_NA, "Na",
+ ID_STATUS_DND, "Dnd",
+ ID_STATUS_OCCUPIED, "Occ",
+ ID_STATUS_FREECHAT, "Ffc",
+ ID_STATUS_INVISIBLE, "Inv",
+ ID_STATUS_ONTHEPHONE, "Otp",
+ ID_STATUS_OUTTOLUNCH, "Otl",
+ ID_STATUS_IDLE, "Idle"
+};
+
+
+class CProtoSettings
+{
+public:
+ CProtoSettings(const char *szProto = NULL, int iStatus = 0): szProto(szProto), Status(iStatus, szProto)
+ {
+ Autoreply.Parent = this;
+ }
+
+ CString ProtoStatusToDBSetting(const char *Prefix, int MoreOpt_PerStatusID = 0)
+ {
+ if (!MoreOpt_PerStatusID || g_MoreOptPage.GetDBValueCopy(MoreOpt_PerStatusID))
+ {
+ int I;
+ for (I = 0; I < lengthof(StatusSettings); I++)
+ {
+ if (Status == StatusSettings[I].Status)
+ {
+ return szProto ? (CString(Prefix) + "_" + szProto + "_" + StatusSettings[I].Setting) : (CString(Prefix) + StatusSettings[I].Setting);
+ }
+ }
+ }
+ return szProto ? (CString(Prefix) + "_" + szProto) : CString(Prefix);
+ }
+
+ class CAutoreply
+ {
+ public:
+ CAutoreply& operator = (const int Value)
+ {
+ CString Setting(Parent->szProto ? Parent->ProtoStatusToDBSetting(DB_ENABLEREPLY, IDC_MOREOPTDLG_PERSTATUSPROTOSETTINGS) : DB_ENABLEREPLY);
+ if (db_get_b(NULL, MOD_NAME, Setting, VAL_USEDEFAULT) == Value)
+ {
+ return *this; // prevent deadlocks when calling UpdateSOEButtons
+ }
+ if (Value != VAL_USEDEFAULT)
+ {
+ db_set_b(NULL, MOD_NAME, Setting, Value != 0);
+ } else
+ {
+ db_unset(NULL, MOD_NAME, Setting);
+ }
+ /*if (!Parent->szProto)
+ {
+ UpdateSOEButtons();
+ }*/
+ return *this;
+ }
+ operator int() {return db_get_b(NULL, MOD_NAME, Parent->szProto ? Parent->ProtoStatusToDBSetting(DB_ENABLEREPLY, IDC_MOREOPTDLG_PERSTATUSPROTOSETTINGS) : DB_ENABLEREPLY, Parent->szProto ? VAL_USEDEFAULT : AUTOREPLY_DEF_REPLY);}
+ int IncludingParents() // takes into account global data also, if per-protocol setting is not defined
+ {
+ _ASSERT(Parent->szProto);
+ int Value = *this;
+ return (Value == VAL_USEDEFAULT) ? CProtoSettings(NULL).Autoreply : Value;
+ }
+ friend class CProtoSettings;
+ private:
+ CProtoSettings *Parent;
+ } Autoreply;
+
+ class CStatus
+ {
+ public:
+ CStatus(int iStatus = 0, const char *szProto = NULL): Status(iStatus), szProto(szProto) {}
+ CStatus& operator = (int Status) {this->Status = Status; return *this;}
+ operator int()
+ {
+ if (!Status)
+ {
+ Status = g_ProtoStates[szProto].Status;
+ }
+ return Status;
+ }
+ private:
+ int Status;
+ const char *szProto;
+ } Status;
+
+ void SetMsgFormat(int Flags, TCString Message);
+ TCString GetMsgFormat(int Flags, int *pOrder = NULL);
+
+//NightFox: fix?
+//private:
+public:
+ const char *szProto;
+};
+
+
+__inline CString StatusToDBSetting(int Status, const char *Prefix, int MoreOpt_PerStatusID = 0)
+{
+ if (!MoreOpt_PerStatusID || g_MoreOptPage.GetDBValueCopy(MoreOpt_PerStatusID))
+ {
+ int I;
+ for (I = 0; I < lengthof(StatusSettings); I++)
+ {
+ if (Status == StatusSettings[I].Status)
+ {
+ return CString(Prefix) + StatusSettings[I].Setting;
+ }
+ }
+ }
+ return CString(Prefix);
+}
+
+
+__inline CString ContactStatusToDBSetting(int Status, const char *Prefix, int MoreOpt_PerStatusID, MCONTACT hContact)
+{
+ if (hContact == INVALID_CONTACT_ID)
+ { // it's a not-on-list contact
+ return CString(DB_UNK_CONTACT_PREFIX) + Prefix;
+ }
+ if (hContact)
+ {
+ StatusToDBSetting(Status, Prefix, MoreOpt_PerStatusID);
+ }
+ return CString(Prefix);
+}
+
+
+class CContactSettings
+{
+public:
+ CContactSettings(int iStatus = 0, MCONTACT _hContact = NULL): Status(iStatus, hContact), hContact(_hContact)
+ {
+ Ignore.Parent = this;
+ Autoreply.Parent = this;
+// PopupNotify.Parent = this;
+ }
+
+ CString ContactStatusToDBSetting(const char *Prefix, int MoreOpt_PerStatusID = 0)
+ {
+ return ::ContactStatusToDBSetting((hContact != INVALID_CONTACT_ID) ? Status : NULL, Prefix, MoreOpt_PerStatusID, hContact);
+ }
+
+ class CIgnore
+ {
+ public:
+ CIgnore& operator = (const int Value)
+ {
+ CString Setting(Parent->ContactStatusToDBSetting(DB_IGNOREREQUESTS, IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS));
+ MCONTACT hContact = (Parent->hContact != INVALID_CONTACT_ID) ? Parent->hContact : NULL;
+ if (Value)
+ {
+ db_set_b(hContact, MOD_NAME, Setting, 1);
+ } else
+ {
+ db_unset(hContact, MOD_NAME, Setting);
+ }
+ return *this;
+ }
+ operator int() {return db_get_b((Parent->hContact != INVALID_CONTACT_ID) ? Parent->hContact : NULL, MOD_NAME, Parent->ContactStatusToDBSetting(DB_IGNOREREQUESTS, IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS), 0);}
+ friend class CContactSettings;
+ private:
+ CContactSettings *Parent;
+ } Ignore;
+
+ class CAutoreply
+ {
+ public:
+ CAutoreply& operator = (const int Value)
+ {
+ CString Setting(Parent->ContactStatusToDBSetting(DB_ENABLEREPLY, IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS));
+ MCONTACT hContact = (Parent->hContact != INVALID_CONTACT_ID) ? Parent->hContact : NULL;
+ if (db_get_b(hContact, MOD_NAME, Setting, VAL_USEDEFAULT) == Value)
+ return *this; // prevent deadlocks when calling UpdateSOEButtons
+
+ if (Value != VAL_USEDEFAULT)
+ db_set_b(hContact, MOD_NAME, Setting, Value != 0);
+ else
+ db_unset(hContact, MOD_NAME, Setting);
+ return *this;
+ }
+ operator int() {return db_get_b((Parent->hContact != INVALID_CONTACT_ID) ? Parent->hContact : NULL, MOD_NAME, Parent->ContactStatusToDBSetting(DB_ENABLEREPLY, IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS), Parent->hContact ? VAL_USEDEFAULT : AUTOREPLY_DEF_REPLY);}
+ int IncludingParents(const char *szProtoOverride = NULL) // takes into account protocol and global data also, if per-contact setting is not defined
+ {
+ _ASSERT((Parent->hContact && Parent->hContact != INVALID_CONTACT_ID) || szProtoOverride); // we need either correct protocol or a correct hContact to determine its protocol
+ int Value = *this;
+ if (Value == VAL_USEDEFAULT)
+ {
+ const char *szProto = (Parent->hContact && Parent->hContact != INVALID_CONTACT_ID) ? (const char*)CallService(MS_PROTO_GETCONTACTBASEPROTO, (WPARAM)Parent->hContact, 0) : szProtoOverride;
+ return CProtoSettings(szProto).Autoreply.IncludingParents();
+ }
+ return Value;
+ }
+ int GetNextToggleValue()
+ {
+ switch ((int)*this)
+ {
+ case VAL_USEDEFAULT: return 0; break;
+ case 0: return 1; break;
+ default: return Parent->hContact ? VAL_USEDEFAULT : AUTOREPLY_DEF_REPLY; break;
+ }
+ }
+ int Toggle() {return *this = GetNextToggleValue();}
+ friend class CContactSettings;
+ private:
+ CContactSettings *Parent;
+ } Autoreply;
+
+/* class CPopupNotify
+ {
+ public:
+ CPopupNotify& operator = (const int Value)
+ {
+ //CString Setting(Parent->ContactStatusToDBSetting(DB_POPUPNOTIFY, 0)); //IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS
+ MCONTACT hContact = (Parent->hContact != INVALID_HANDLE_VALUE) ? Parent->hContact : NULL;
+ if (db_get_b(hContact, MOD_NAME, Setting, VAL_USEDEFAULT) == Value)
+ {
+ return *this; // prevent deadlocks when calling UpdateSOEButtons
+ }
+ if (Value != VAL_USEDEFAULT)
+ {
+ db_set_b(hContact, MOD_NAME, Setting, Value != 0);
+ } else
+ {
+ db_unset(hContact, MOD_NAME, Setting);
+ }
+ if (!Parent->hContact)
+ {
+ //UpdateSOEButtons();
+ }
+ return *this;
+ }
+ operator int() {return db_get_b((Parent->hContact != INVALID_HANDLE_VALUE) ? Parent->hContact : NULL, MOD_NAME, Parent->ContactStatusToDBSetting(DB_POPUPNOTIFY, 0), Parent->hContact ? VAL_USEDEFAULT : POPUP_DEF_USEPOPUPS);} //IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS
+ int IncludingParents() // takes into account protocol and global data also, if per-contact setting is not defined
+ {
+ int Value = *this;
+ if (Value == VAL_USEDEFAULT)
+ { // PopupNotify setting is not per-status
+ return CContactSettings(ID_STATUS_ONLINE).PopupNotify;
+ }
+ return Value;
+ }
+ friend class CContactSettings;
+ private:
+ CContactSettings *Parent;
+ } PopupNotify;
+*/
+ class CStatus
+ {
+ public:
+ CStatus(int iStatus = 0, MCONTACT _hContact = NULL): Status(iStatus), hContact(_hContact) {}
+ CStatus& operator = (int Status) {this->Status = Status; return *this;}
+ operator int()
+ {
+ if (!Status)
+ {
+ _ASSERT(hContact != INVALID_CONTACT_ID);
+ char *szProto = hContact ? GetContactProto(hContact) : NULL;
+ Status = (szProto || !hContact) ? g_ProtoStates[szProto].Status : ID_STATUS_AWAY;
+ }
+ return Status;
+ }
+ friend class CPopupNotify;
+ friend class CAutoreply;
+ private:
+ int Status;
+ MCONTACT hContact;
+ } Status;
+
+ void SetMsgFormat(int Flags, TCString Message);
+ TCString GetMsgFormat(int Flags, int *pOrder = NULL, char *szProtoOverride = NULL);
+
+//NightFox: fix?
+//private:
+public:
+ MCONTACT hContact;
+};
diff --git a/plugins/NewAwaySysMod/src/ReadAwayMsg.cpp b/plugins/NewAwaySysMod/src/ReadAwayMsg.cpp
new file mode 100644
index 0000000000..2464c38f4c
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/ReadAwayMsg.cpp
@@ -0,0 +1,158 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (C) 2005-2007 Chervov Dmitry
+ Copyright (C) 2004-2005 Iksaif Entertainment
+ Copyright (C) 2002-2003 Goblineye Entertainment
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include "Common.h"
+
+#define RAMDLGSIZESETTING "ReadAwayMsgDlg"
+
+HANDLE g_hReadWndList = NULL;
+
+
+static int ReadAwayMsgDlgResize(HWND hwndDlg, LPARAM lParam, UTILRESIZECONTROL *urc)
+{
+ switch (urc->wId)
+ {
+ case IDC_READAWAYMSG_MSG:
+ {
+ return RD_ANCHORX_WIDTH | RD_ANCHORY_HEIGHT;
+ }
+ case IDC_READAWAYMSG_RETRIEVE:
+ {
+ return RD_ANCHORX_CENTRE | RD_ANCHORY_CENTRE;
+ }
+ case IDOK:
+ {
+ return RD_ANCHORX_CENTRE | RD_ANCHORY_BOTTOM;
+ }
+ }
+ return RD_ANCHORX_LEFT | RD_ANCHORY_TOP;
+}
+
+
+static INT_PTR CALLBACK ReadAwayMsgDlgProc(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
+{
+ switch (msg)
+ {
+ case WM_INITDIALOG:
+ {
+ TranslateDialogDefault(hwndDlg);
+ HICON hTitleIcon = LoadSkinnedIcon(SKINICON_OTHER_MIRANDA);
+ SendMessage(hwndDlg, WM_SETICON, ICON_BIG, (LPARAM)hTitleIcon);
+ SendMessage(hwndDlg, WM_SETICON, ICON_SMALL, (LPARAM)hTitleIcon);
+ Utils_RestoreWindowPosition(hwndDlg, NULL, MOD_NAME, RAMDLGSIZESETTING);
+ READAWAYMSGDATA *awayData = new READAWAYMSGDATA;
+ SetWindowLongPtr(hwndDlg, GWLP_USERDATA, (LONG_PTR)awayData);
+ awayData->hContact = lParam;
+ awayData->hAwayMsgEvent = HookEventMessage(ME_PROTO_ACK, hwndDlg, UM_RAM_AWAYMSGACK);
+ awayData->hSeq = (HANDLE)CallContactService(awayData->hContact, PSS_GETAWAYMSG, 0, 0);
+ WindowList_Add(g_hReadWndList, hwndDlg, awayData->hContact);
+
+ TCHAR str[256], format[128];
+ TCHAR *status, *contactName;
+ contactName = (TCHAR*)CallService(MS_CLIST_GETCONTACTDISPLAYNAME, (WPARAM)awayData->hContact, GCDNF_TCHAR);
+ char *szProto = GetContactProto(awayData->hContact);
+ status = (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, db_get_w(awayData->hContact, szProto, "Status", ID_STATUS_OFFLINE), GSMDF_TCHAR);
+ GetWindowText(hwndDlg, format, lengthof(format));
+ _sntprintf(str, lengthof(str), format, status, contactName);
+ SetWindowText(hwndDlg, str);
+ GetDlgItemText(hwndDlg, IDC_READAWAYMSG_RETRIEVE, format, sizeof(format));
+ _sntprintf(str, lengthof(str), format, status);
+ SetDlgItemText(hwndDlg, IDC_READAWAYMSG_RETRIEVE, str);
+ return true;
+ } break;
+ case UM_RAM_AWAYMSGACK: // got away msg
+ {
+ ACKDATA *ack = (ACKDATA*)lParam;
+ READAWAYMSGDATA *awayData = (READAWAYMSGDATA*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
+ // make sure everything is ok, and this is our ack
+ if (ack->hContact != awayData->hContact || ack->type != ACKTYPE_AWAYMSG || ack->hProcess != awayData->hSeq || ack->result != ACKRESULT_SUCCESS)
+ {
+ break;
+ }
+ // done with the event
+ if (awayData->hAwayMsgEvent)
+ {
+ UnhookEvent(awayData->hAwayMsgEvent);
+ awayData->hAwayMsgEvent = NULL;
+ }
+ SetDlgItemText(hwndDlg, IDC_READAWAYMSG_MSG, (const TCHAR*)ack->lParam);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_READAWAYMSG_RETRIEVE), SW_HIDE);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_READAWAYMSG_MSG), SW_SHOW);
+ SetDlgItemText(hwndDlg, IDOK, TranslateT("&Close"));
+ db_set_s(awayData->hContact, "CList", "StatusMsg", (const char*)ack->lParam);
+ } break;
+ case WM_COMMAND:
+ {
+ switch (HIWORD(wParam))
+ {
+ case BN_CLICKED:
+ {
+ switch(LOWORD(wParam))
+ {
+ case IDCANCEL:
+ case IDOK:
+ {
+ DestroyWindow(hwndDlg);
+ } break;
+ }
+ } break;
+ }
+ } break;
+ case WM_SIZE:
+ {
+ UTILRESIZEDIALOG urd = {0};
+ urd.cbSize = sizeof(urd);
+ urd.hInstance = g_hInstance;
+ urd.hwndDlg = hwndDlg;
+ urd.lpTemplate = MAKEINTRESOURCEA(IDD_READAWAYMSG);
+ urd.pfnResizer = ReadAwayMsgDlgResize;
+ CallService(MS_UTILS_RESIZEDIALOG, 0, (LPARAM)&urd);
+ break;
+ }
+ case WM_CLOSE:
+ {
+ DestroyWindow(hwndDlg);
+ } break;
+ case WM_DESTROY:
+ {
+ READAWAYMSGDATA *awayData = (READAWAYMSGDATA*)GetWindowLongPtr(hwndDlg, GWLP_USERDATA);
+ if (awayData->hAwayMsgEvent)
+ {
+ UnhookEvent(awayData->hAwayMsgEvent);
+ awayData->hAwayMsgEvent = NULL;
+ }
+ delete awayData;
+ Utils_SaveWindowPosition(hwndDlg, NULL, MOD_NAME, RAMDLGSIZESETTING);
+ WindowList_Remove(g_hReadWndList, hwndDlg);
+ } break;
+ }
+ return false;
+}
+
+
+INT_PTR GetContactStatMsg(WPARAM wParam, LPARAM lParam)
+{
+ if (HWND hWnd = WindowList_Find(g_hReadWndList, wParam)) // already have it
+ SetForegroundWindow(hWnd);
+ else
+ CreateDialogParam(g_hInstance, MAKEINTRESOURCE(IDD_READAWAYMSG), NULL, ReadAwayMsgDlgProc, wParam);
+ return 0;
+}
diff --git a/plugins/NewAwaySysMod/src/Services.cpp b/plugins/NewAwaySysMod/src/Services.cpp
new file mode 100644
index 0000000000..392d318222
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Services.cpp
@@ -0,0 +1,231 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+
+#include "Common.h"
+#include "Properties.h"
+#include "m_NewAwaySys.h"
+
+
+typedef struct {
+ int cbSize;
+ char *szProto;
+ union
+ {
+ char *szMsg;
+ WCHAR *wszMsg;
+ TCHAR *tszMsg;
+ };
+ WORD status;
+} NAS_PROTOINFOv1;
+
+
+__inline void PSSetStatus(char *szProto, WORD Status, int bNoClistSetStatusMode = false) // just a helper function that sets the status and handles szProto==NULL correctly
+{
+ g_ProtoStates[szProto].Status = Status;
+ if (szProto)
+ {
+ CallProtoService(szProto, PS_SETSTATUS, Status, 0);
+ } else if (!bNoClistSetStatusMode) // global status
+ {
+/* int ProtoCount;
+ PROTOCOLDESCRIPTOR **proto;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&ProtoCount, (LPARAM)&proto);
+ int i;
+ for (i = 0; i < ProtoCount; i++)
+ {
+ if (proto[i]->type == PROTOTYPE_PROTOCOL)
+ {
+ CallProtoService(proto[i]->szName, PS_SETSTATUS, Status, 0);
+ }
+ }*/
+ g_fNoProcessing = true;
+ CallService(MS_CLIST_SETSTATUSMODE, Status, 0);
+ _ASSERT(!g_fNoProcessing && g_ProtoStates[(char*)NULL].Status == Status);
+ g_fNoProcessing = false;
+ }
+}
+
+
+INT_PTR GetStatusMsg(WPARAM wParam, LPARAM lParam) // called by GamerStatus and built-in miranda autoaway module
+// wParam = (int)status, lParam = 0
+// MS_AWAYMSG_GETSTATUSMSG "SRAway/GetStatusMessage"
+{
+ LogMessage("MS_AWAYMSG_GETSTATUSMSG called. status=%d", wParam);
+ CString Msg(_T2A(GetDynamicStatMsg(INVALID_CONTACT_ID, NULL, 0, wParam)));
+ char *szMsg;
+ if (Msg == NULL)
+ { // it's ok to return NULL, so we'll do it
+ szMsg = NULL;
+ } else
+ {
+ szMsg = (char*)mir_alloc(Msg.GetLen() + 1);
+ lstrcpyA(szMsg, Msg);
+ }
+ LogMessage("returned szMsg:\n%s", szMsg ? szMsg : "NULL");
+ return (int)szMsg;
+}
+
+
+INT_PTR SetStatusMode(WPARAM wParam, LPARAM lParam) // called by GamerStatus and StatCtrl
+{
+// wParam = int iMode
+// lParam = char* szMsg, may be null - then we need to use the default message
+/* if (CallService(MS_CLIST_GETSTATUSMODE, 0, 0) != wParam) // not same status
+ {*/ // try this way; global status may be not the same as current status of some separate protocol
+ LogMessage("MS_AWAYSYS_SETSTATUSMODE called. status=%d, szMsg:\n%s", wParam, lParam ? (char*)lParam : "NULL");
+ g_fNoProcessing = true;
+ CallService(MS_CLIST_SETSTATUSMODE, wParam, 0);
+// }
+ _ASSERT(!g_fNoProcessing && g_ProtoStates[(char*)NULL].Status == wParam);
+ g_fNoProcessing = false;
+ CProtoSettings(NULL, wParam).SetMsgFormat(SMF_TEMPORARY, lParam ? (TCHAR*)_A2T((char*)lParam) : CProtoSettings(NULL, wParam).GetMsgFormat(GMF_LASTORDEFAULT));
+ ChangeProtoMessages(NULL, wParam, TCString());
+ return 0;
+}
+
+
+int GetState(WPARAM wParam, LPARAM lParam, int Widechar)
+// wParam = (WPARAM)(NAS_PROTOINFO*)pi - pointer to an array of NAS_PROTOINFO items to be filled.
+// lParam = (LPARAM)(int)protoCount - number of items in pi.
+// returns 0 on success
+{
+ NAS_PROTOINFO *pi = (NAS_PROTOINFO*)wParam;
+ LogMessage("MS_NAS_GETSTATE called with %d items and Widechar=%d:", lParam, Widechar);
+ for (int i = 0; i < lParam; i++)
+ {
+ if (pi->cbSize < sizeof(NAS_PROTOINFO) && pi->cbSize != sizeof(NAS_PROTOINFOv1))
+ return 1;
+
+ int Flags = (pi->cbSize > sizeof(NAS_PROTOINFOv1)) ? pi->Flags : 0;
+ LogMessage("%d (received): cbSize=%d, status=%d, szProto=%s, Flags=0x%x", i + 1, pi->cbSize, pi->status, pi->szProto ? pi->szProto : "NULL", Flags);
+ if ((pi->status >= ID_STATUS_ONLINE && pi->status <= ID_STATUS_OUTTOLUNCH) || !pi->status)
+ {
+ TCString Msg(pi->status ? CProtoSettings(pi->szProto, pi->status).GetMsgFormat(GMF_LASTORDEFAULT) : CProtoSettings(pi->szProto).GetMsgFormat(((Flags & PIF_NOTTEMPORARY) ? 0 : GMF_TEMPORARY) | GMF_PERSONAL));
+ if (Msg != NULL)
+ {
+ pi->szMsg = (char*)mir_alloc(Msg.GetLen() + 1);
+ _ASSERT(pi->szMsg);
+ if (Widechar)
+ lstrcpyW(pi->wszMsg, Msg);
+ else
+ lstrcpyA(pi->szMsg, _T2A(Msg));
+ }
+ else pi->szMsg = NULL;
+
+ if (!pi->status)
+ pi->status = g_ProtoStates[pi->szProto].Status;
+ }
+ else pi->szMsg = NULL;
+
+ LogMessage("%d (returned): status=%d, Flags=0x%x, szMsg:\n%s", i + 1, pi->status, (pi->cbSize > sizeof(NAS_PROTOINFOv1)) ? pi->Flags : 0, pi->szMsg ? (Widechar ? _T2A(pi->wszMsg) : pi->szMsg) : "NULL");
+ *(char**)&pi += pi->cbSize;
+ }
+ return 0;
+}
+
+
+INT_PTR GetStateA(WPARAM wParam, LPARAM lParam)
+{
+ return GetState(wParam, lParam, false);
+}
+
+
+INT_PTR GetStateW(WPARAM wParam, LPARAM lParam)
+{
+ return GetState(wParam, lParam, true);
+}
+
+
+int SetState(WPARAM wParam, LPARAM lParam, int Widechar)
+// wParam = (WPARAM)(NAS_PROTOINFO*)pi - pointer to an array of NAS_PROTOINFO items.
+// lParam = (LPARAM)(int)protoCount - number of items in pi.
+// returns 0 on success
+{
+ NAS_PROTOINFO *pi = (NAS_PROTOINFO*)wParam;
+ LogMessage("MS_NAS_SETSTATE called with %d items and Widechar=%d:", lParam, Widechar);
+ for (int i = 0; i < lParam; i++)
+ {
+ _ASSERT(pi->szMsg != (char*)(-1));
+ if (pi->cbSize < sizeof(NAS_PROTOINFO) && pi->cbSize != sizeof(NAS_PROTOINFOv1))
+ return 1;
+
+ int Flags = (pi->cbSize > sizeof(NAS_PROTOINFOv1)) ? pi->Flags : 0;
+ LogMessage("%d: cbSize=%d, status=%d, szProto=%s, Flags=0x%x, szMsg:\n%s", i + 1, pi->cbSize, pi->status, pi->szProto ? pi->szProto : "NULL", Flags, pi->szMsg ? (Widechar ? _T2A(pi->wszMsg) : pi->szMsg) : "NULL");
+ if (pi->status)
+ PSSetStatus(pi->szProto, pi->status, Flags & PIF_NO_CLIST_SETSTATUSMODE);
+ else
+ pi->status = g_ProtoStates[pi->szProto].Status;
+
+ CProtoSettings(pi->szProto).SetMsgFormat((Flags & PIF_NOTTEMPORARY) ? SMF_PERSONAL : SMF_TEMPORARY, Widechar ? pi->wszMsg : _A2T(pi->szMsg));
+ if (pi->szMsg || !(Flags & PIF_NO_CLIST_SETSTATUSMODE))
+ ChangeProtoMessages(pi->szProto, pi->status, TCString());
+
+ mir_free(pi->szMsg);
+ pi->szMsg = (char*)(-1);
+ *(char**)&pi += pi->cbSize;
+ }
+ return 0;
+}
+
+
+INT_PTR SetStateA(WPARAM wParam, LPARAM lParam)
+{
+ return SetState(wParam, lParam, false);
+}
+
+
+INT_PTR SetStateW(WPARAM wParam, LPARAM lParam)
+{
+ return SetState(wParam, lParam, true);
+}
+
+
+INT_PTR InvokeStatusWindow(WPARAM wParam, LPARAM lParam)
+// wParam = (WPARAM)(NAS_ISWINFO*)iswi - pointer to a NAS_ISWINFO structure.
+// lParam = 0
+// returns HWND of the window on success, or NULL on failure.
+{
+ NAS_ISWINFO *iswi = (NAS_ISWINFO*)wParam;
+ if (iswi->cbSize != sizeof(NAS_ISWINFOv1) && iswi->cbSize < sizeof(NAS_ISWINFO))
+ {
+ return NULL;
+ }
+ LogMessage("MS_NAS_INVOKESTATUSWINDOW called. cbSize=%d, status=%d, szProto=%s, hContact=0x%08x, Flags=0x%x, szMsg:\n%s", iswi->cbSize, iswi->status, iswi->szProto ? iswi->szProto : "NULL", iswi->hContact, (iswi->cbSize < sizeof(NAS_ISWINFO)) ? 0 : iswi->Flags, iswi->szMsg ? ((iswi->Flags & ISWF_UNICODE) ? _T2A(iswi->wszMsg) : iswi->szMsg) : "NULL");
+ if (iswi->status)
+ {
+ PSSetStatus(iswi->szProto, iswi->status);
+ }
+ if (g_SetAwayMsgPage.GetWnd()) // already setting something
+ {
+ SetForegroundWindow(g_SetAwayMsgPage.GetWnd());
+ return (int)g_SetAwayMsgPage.GetWnd();
+ }
+ SetAwayMsgData *dat = new SetAwayMsgData;
+ ZeroMemory(dat, sizeof(SetAwayMsgData));
+ dat->hInitContact = iswi->hContact;
+ dat->szProtocol = iswi->szProto;
+ dat->Message = (iswi->Flags & ISWF_UNICODE) ? iswi->wszMsg : _A2T(iswi->szMsg);
+ dat->IsModeless = true;
+ if (iswi->cbSize > sizeof(NAS_ISWINFOv1))
+ {
+ dat->ISW_Flags = iswi->Flags;
+ }
+ return (int)CreateDialogParam(g_hInstance, MAKEINTRESOURCE(IDD_SETAWAYMSG), NULL, SetAwayMsgDlgProc, (LPARAM)dat);
+}
diff --git a/plugins/NewAwaySysMod/src/Services.h b/plugins/NewAwaySysMod/src/Services.h
new file mode 100644
index 0000000000..903bbd3784
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/Services.h
@@ -0,0 +1,27 @@
+/*
+ New Away System - plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+INT_PTR GetStatusMsg(WPARAM wParam, LPARAM lParam);
+INT_PTR SetStatusMode(WPARAM wParam, LPARAM lParam);
+// int IgnoreNextStatusChange(WPARAM wParam, LPARAM lParam);
+INT_PTR GetStateA(WPARAM wParam, LPARAM lParam);
+INT_PTR SetStateA(WPARAM wParam, LPARAM lParam);
+INT_PTR GetStateW(WPARAM wParam, LPARAM lParam);
+INT_PTR SetStateW(WPARAM wParam, LPARAM lParam);
+INT_PTR InvokeStatusWindow(WPARAM wParam, LPARAM lParam);
diff --git a/plugins/NewAwaySysMod/src/SetAwayMsg.cpp b/plugins/NewAwaySysMod/src/SetAwayMsg.cpp
new file mode 100644
index 0000000000..9b918be898
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/SetAwayMsg.cpp
@@ -0,0 +1,1698 @@
+/*
+ New Away System plugin for Miranda IM
+ Copyright (c) 2005-2007 Chervov Dmitry
+ Copyright (c) 2004-2005 Iksaif Entertainment
+ Copyright (c) 2002-2003 Goblineye Entertainment
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+
+#include "Common.h"
+#include "m_button.h"
+#include "m_icolib.h"
+#include "ContactList.h"
+#include "MsgTree.h"
+#include "Properties.h"
+#include "m_NewAwaySys.h"
+
+#define EXTRAIMGLIST_DOT 0
+#define EXTRAIMGLIST_IGNORE 1
+#define EXTRAIMGLIST_MSG 2
+#define EXTRAIMGLIST_AUTOREPLY_ON 3
+#define EXTRAIMGLIST_AUTOREPLY_OFF 4
+
+#define EXTRACOLUMN_IGNORE 0
+#define EXTRACOLUMN_REPLY 1
+
+#define MINSPLITTERPOS 176
+#define MINYDLGSIZE 100
+
+#define SAM_TIMER_ID 1
+
+#define UM_SAM_INITCLIST (WM_USER + 100)
+
+#define CLSEL_DAT_CONTACT ((PTREEITEMARRAY)(-1)) // these constants are used to specify selection when the contact list is not loaded
+#define CLSEL_DAT_NOTHING ((PTREEITEMARRAY)(-2))
+
+static WNDPROC g_OrigEditMsgProc;
+static WNDPROC g_OrigSplitterProc;
+static WNDPROC g_OrigCListProc;
+static int g_VariablesButtonDX;
+static int g_MsgSplitterX, g_ContactSplitterX;
+static int MinMsgSplitterX, MinContactSplitterX;
+static int MinMsgEditSize; // used to calculate minimal X size of the dialog
+
+
+
+static LRESULT CALLBACK MsgEditSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ switch (Msg)
+ {
+ case WM_CHAR:
+ {
+ if (GetKeyState(VK_CONTROL) & 0x8000)
+ {
+ if (wParam == '\n')
+ { // ctrl-enter
+ PostMessage(GetParent(hWnd), WM_COMMAND, IDC_OK, 0);
+ return 0;
+ }
+ if (wParam == 1)
+ { // ctrl-a
+ SendMessage(hWnd, EM_SETSEL, 0, -1);
+ return 0;
+ }
+ if (wParam == 23)
+ { // ctrl-w
+ SendMessage(GetParent(hWnd), WM_CLOSE, 0, 0);
+ return 0;
+ }
+ if (wParam == 127)
+ { // ctrl-backspace
+ DWORD start, end;
+ SendMessage(hWnd, EM_GETSEL, (WPARAM)&end, NULL);
+ SendMessage(hWnd, WM_KEYDOWN, VK_LEFT, 0);
+ SendMessage(hWnd, EM_GETSEL, (WPARAM)&start, NULL);
+ int nLen = GetWindowTextLength(hWnd);
+ TCHAR *text = (TCHAR*)malloc((nLen + 1) * sizeof(TCHAR));
+ GetWindowText(hWnd, text, nLen + 1);
+ MoveMemory(text + start, text + end, sizeof(TCHAR) * (_tcslen(text) + 1 - end));
+ SetWindowText(hWnd, text);
+ free(text);
+ SendMessage(hWnd, EM_SETSEL, start, start);
+ SendMessage(GetParent(hWnd), WM_COMMAND, MAKEWPARAM(GetDlgCtrlID(hWnd), EN_CHANGE), (LPARAM)hWnd);
+ return 0;
+ }
+ }
+ } break;
+ case WM_KEYDOWN:
+ {
+ SendMessage(GetParent(hWnd), UM_SAM_KILLTIMER, 0, 0);
+ } break;
+ }
+ return CallWindowProc(g_OrigEditMsgProc, hWnd, Msg, wParam, lParam);
+}
+
+// used splitter code from TabSRMM as a base
+static LRESULT CALLBACK SplitterSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ switch (Msg)
+ {
+ case WM_NCHITTEST:
+ {
+ return HTCLIENT;
+ }
+ case WM_SETCURSOR:
+ {
+ SetCursor(LoadCursor(NULL, IDC_SIZEWE));
+ return true;
+ }
+ case WM_LBUTTONDOWN:
+ {
+ SetCapture(hWnd);
+ return false;
+ }
+ case WM_MOUSEMOVE:
+ {
+ if (GetCapture() == hWnd)
+ {
+ RECT rc;
+ GetClientRect(hWnd, &rc);
+ SendMessage(GetParent(hWnd), UM_SAM_SPLITTERMOVED, (short)LOWORD(GetMessagePos()) + rc.right / 2, (LPARAM)GetDlgCtrlID(hWnd));
+ }
+ return 0;
+ }
+ case WM_LBUTTONUP:
+ {
+ ReleaseCapture();
+ return false;
+ }
+ }
+ return CallWindowProc(g_OrigSplitterProc, hWnd, Msg, wParam, lParam);
+}
+
+static LRESULT CALLBACK CListSubclassProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
+{
+ CCList *dat = CWndUserData(GetParent(hWnd)).GetCList();
+ switch (Msg)
+ {
+ case WM_MOUSEMOVE:
+ {
+ DWORD hitFlags;
+ POINT pt = {(short)LOWORD(lParam), (short)HIWORD(lParam)};
+ if (dat->HitTest(&pt, &hitFlags) && hitFlags & MCLCHT_ONITEMEXTRA)
+ {
+ lParam = 0; // reset mouse coordinates, so TreeView's wndproc will not draw any item in a hovered state
+ }
+ } break;
+ case WM_LBUTTONDOWN:
+ case WM_LBUTTONDBLCLK:
+ {
+ DWORD hitFlags;
+ POINT pt = {(short)LOWORD(lParam), (short)HIWORD(lParam)};
+ if (dat->HitTest(&pt, &hitFlags) && hitFlags & MCLCHT_ONITEMEXTRA)
+ {
+ SetFocus(hWnd);
+ NMHDR nmhdr;
+ nmhdr.code = NM_CLICK;
+ nmhdr.hwndFrom = hWnd;
+ nmhdr.idFrom = GetDlgCtrlID(hWnd);
+ SendMessage(GetParent(hWnd), WM_NOTIFY, 0, (LPARAM)&nmhdr);
+ return DefWindowProc(hWnd, Msg, wParam, lParam);
+ }
+ } break;
+ }
+ return CallWindowProc(g_OrigCListProc, hWnd, Msg, wParam, lParam);
+}
+
+
+static int SetAwayMsgDlgResize(HWND hwndDlg, LPARAM lParam, UTILRESIZECONTROL *urc)
+{
+ COptPage *SetAwayMsgPage = (COptPage*)lParam;
+ int bShowMsgTree = SetAwayMsgPage->GetValue(IDS_SAWAYMSG_SHOWMSGTREE);
+ int bShowContactTree = SetAwayMsgPage->GetValue(IDS_SAWAYMSG_SHOWCONTACTTREE);
+ switch (urc->wId)
+ {
+ case IDC_SAWAYMSG_MSGDATA:
+ {
+ urc->rcItem.right = urc->dlgOriginalSize.cx - 2;
+ if (bShowContactTree)
+ {
+ urc->rcItem.right -= g_ContactSplitterX;
+ }
+ urc->rcItem.left = (bShowMsgTree) ? (g_MsgSplitterX + 2) : 2;
+ return RD_ANCHORX_WIDTH | RD_ANCHORY_HEIGHT;
+ }
+ case IDC_SAWAYMSG_TREE:
+ {
+ urc->rcItem.right = g_MsgSplitterX - 2;
+ return RD_ANCHORX_LEFT | RD_ANCHORY_HEIGHT;
+ }
+ case IDC_SAWAYMSG_CONTACTSTREE:
+ {
+ urc->rcItem.left = urc->dlgOriginalSize.cx - g_ContactSplitterX + 2;
+ return RD_ANCHORX_RIGHT | RD_ANCHORY_HEIGHT;
+ }
+ case IDC_OK:
+ {
+ int X = bShowMsgTree ? min(max(g_MsgSplitterX, MinMsgSplitterX), urc->dlgNewSize.cx - ((bShowContactTree ? MinContactSplitterX : 0) + MinMsgEditSize)) : 0;
+ OffsetRect(&urc->rcItem, X + 2 - urc->rcItem.left, 0);
+ return RD_ANCHORX_LEFT | RD_ANCHORY_BOTTOM;
+ }
+ case IDC_SAWAYMSG_EDITMSGS:
+ case IDC_SAWAYMSG_SAVEMSG:
+ case IDC_SAWAYMSG_SAVEASNEW:
+ case IDC_SAWAYMSG_NEWCATEGORY:
+ case IDC_SAWAYMSG_DELETE:
+ {
+ return RD_ANCHORX_LEFT | RD_ANCHORY_BOTTOM;
+ }
+ case IDC_SAWAYMSG_VARS:
+ case IDC_SAWAYMSG_OPTIONS:
+ {
+ int X = bShowContactTree ? max(min(g_ContactSplitterX, urc->dlgNewSize.cx - (bShowMsgTree ? max(g_MsgSplitterX, MinMsgSplitterX) : 0) - MinMsgEditSize), MinContactSplitterX) : 0;
+ OffsetRect(&urc->rcItem, urc->dlgOriginalSize.cx - X - 2 - urc->rcItem.right - ((urc->wId == IDC_SAWAYMSG_VARS) ? g_VariablesButtonDX : 0), 0);
+ return RD_ANCHORX_RIGHT | RD_ANCHORY_BOTTOM;
+ }
+ case IDC_SAWAYMSG_STATIC_IGNOREICON:
+ case IDC_SAWAYMSG_STATIC_REPLYICON:
+ {
+ urc->rcItem.left = urc->rcItem.right - 16;
+ urc->rcItem.top = urc->rcItem.bottom - 16;
+ } // go through
+ case IDC_SAWAYMSG_IGNOREREQ:
+ case IDC_SAWAYMSG_SENDMSG:
+ {
+ return RD_ANCHORX_RIGHT | RD_ANCHORY_BOTTOM;
+ }
+ case IDC_SAWAYMSG_MSGSPLITTER:
+ {
+ OffsetRect(&urc->rcItem, g_MsgSplitterX - (urc->rcItem.left + ++urc->rcItem.right) / 2, 0);
+ return RD_ANCHORX_LEFT | RD_ANCHORY_HEIGHT;
+ }
+ case IDC_SAWAYMSG_CONTACTSPLITTER:
+ {
+ OffsetRect(&urc->rcItem, urc->dlgOriginalSize.cx - g_ContactSplitterX - (urc->rcItem.left + urc->rcItem.right) / 2, 0);
+ return RD_ANCHORX_RIGHT | RD_ANCHORY_HEIGHT;
+ }
+ }
+
+
+ return RD_ANCHORX_LEFT | RD_ANCHORY_BOTTOM;
+}
+
+
+__inline int DBValueToReplyIcon(int Value)
+{
+ switch (Value)
+ {
+ case VAL_USEDEFAULT: return EXTRAIMGLIST_DOT;
+ case 0: return EXTRAIMGLIST_AUTOREPLY_OFF;
+ default: return EXTRAIMGLIST_AUTOREPLY_ON;
+ }
+}
+
+
+int GetRealReplyIcon(CCList *CList, HTREEITEM hItem)
+{
+ _ASSERT(CList);
+ MCONTACT hContact = CList->GethContact(hItem);
+ int ItemType = CList->GetItemType(hItem);
+ char *szProto = (char*)CList->GetItemParam(hItem);
+ return (ItemType == MCLCIT_GROUP) ? CList->GetItemParam(hItem) : DBValueToReplyIcon((ItemType == MCLCIT_CONTACT) ? (int)CContactSettings(0, hContact).Autoreply : CProtoSettings(szProto).Autoreply);
+}
+
+
+void SetExtraIcon(CCList *CList, int nColumn, HTREEITEM hItem, int nIcon)
+{
+ _ASSERT(CList);
+ int ItemType = CList->GetItemType(hItem);
+ MCONTACT hContact = CList->GethContact(hItem);
+ if (ItemType == MCLCIT_CONTACT)
+ {
+ if (nIcon == -1) // means we need to retrieve it from the db by ourselves
+ {
+ if (nColumn == EXTRACOLUMN_IGNORE)
+ {
+ nIcon = CContactSettings(0, hContact).Ignore ? EXTRAIMGLIST_IGNORE : EXTRAIMGLIST_DOT;
+ } else
+ {
+ _ASSERT(nColumn == EXTRACOLUMN_REPLY);
+ nIcon = DBValueToReplyIcon(CContactSettings(0, hContact).Autoreply);
+ }
+ } else // save it back to the db
+ {
+ if (nColumn == EXTRACOLUMN_IGNORE)
+ {
+ CContactSettings(0, hContact).Ignore = nIcon == EXTRAIMGLIST_IGNORE;
+ } else
+ {
+ _ASSERT(nColumn == EXTRACOLUMN_REPLY);
+ CContactSettings(0, hContact).Autoreply = (nIcon == EXTRAIMGLIST_DOT) ? VAL_USEDEFAULT : (nIcon == EXTRAIMGLIST_AUTOREPLY_ON);
+ }
+ }
+ if (nColumn == EXTRACOLUMN_IGNORE && nIcon != EXTRAIMGLIST_IGNORE)
+ {
+ nIcon = (CContactSettings(0, hContact).GetMsgFormat(GMF_PERSONAL) == NULL) ? EXTRAIMGLIST_DOT : EXTRAIMGLIST_MSG;
+ }
+ } else if (ItemType == MCLCIT_INFO)
+ {
+ char *szProto = (char*)CList->GetItemParam(hItem);
+ if (nColumn == EXTRACOLUMN_REPLY)
+ {
+ if (nIcon == -1)
+ {
+ nIcon = DBValueToReplyIcon(CProtoSettings(szProto).Autoreply);
+ } else
+ {
+ CProtoSettings(szProto).Autoreply = (nIcon == EXTRAIMGLIST_DOT) ? VAL_USEDEFAULT : (nIcon == EXTRAIMGLIST_AUTOREPLY_ON);
+ }
+ if (!szProto && nIcon == EXTRAIMGLIST_DOT)
+ {
+ nIcon = EXTRAIMGLIST_AUTOREPLY_OFF;
+ }
+ } else
+ {
+ nIcon = (CProtoSettings(szProto).GetMsgFormat(GMF_TEMPORARY | GMF_PERSONAL) == NULL) ? EXTRAIMGLIST_DOT : EXTRAIMGLIST_MSG;
+ }
+ }
+ int Ignore = (nColumn == EXTRACOLUMN_IGNORE) ? (nIcon == EXTRAIMGLIST_IGNORE) : ((ItemType == MCLCIT_CONTACT) ? CContactSettings(0, hContact).Ignore : ((ItemType == MCLCIT_GROUP) ? CList->GetExtraImage(hItem, EXTRACOLUMN_IGNORE) : false));
+ if (Ignore)
+ {
+ if (nColumn == EXTRACOLUMN_IGNORE)
+ {
+ CList->SetExtraImage(hItem, EXTRACOLUMN_REPLY, CLC_EXTRAICON_EMPTY);
+ } else
+ {
+ nIcon = CLC_EXTRAICON_EMPTY;
+ }
+ } else
+ {
+ int nReplyIcon;
+ if (ItemType == MCLCIT_CONTACT)
+ {
+ nReplyIcon = DBValueToReplyIcon(CContactSettings(0, hContact).Autoreply);
+ } else if (ItemType == MCLCIT_GROUP)
+ {
+ nReplyIcon = GetRealReplyIcon(CList, hItem);
+ } else
+ {
+ _ASSERT(ItemType == MCLCIT_INFO);
+ char *szProto = (char*)CList->GetItemParam(hItem);
+ nReplyIcon = DBValueToReplyIcon(CProtoSettings(szProto).Autoreply);
+ }
+ if (nColumn == EXTRACOLUMN_IGNORE)
+ {
+ CList->SetExtraImage(hItem, EXTRACOLUMN_REPLY, nReplyIcon);
+ } else if (nIcon == CLC_EXTRAICON_EMPTY)
+ {
+ nIcon = nReplyIcon;
+ }
+ }
+ CList->SetExtraImage(hItem, nColumn, nIcon);
+}
+
+
+void SetCListGroupIcons(SetAwayMsgData *dat, CCList *CList)
+{
+ _ASSERT(CList);
+ HTREEITEM hItem = CList->GetNextItem(MCLGN_LAST, NULL); // start from last item, so every item is processed before its parents
+ if (!hItem)
+ {
+ return;
+ }
+ if (CList->GetItemType(hItem) != MCLCIT_GROUP)
+ {
+ hItem = CList->GetNextItem(MCLGN_PREV | MCLGN_GROUP | MCLGN_MULTILEVEL, hItem);
+ if (!hItem)
+ {
+ return;
+ }
+ }
+ do
+ {
+ HTREEITEM hCurItem = CList->GetNextItem(MCLGN_CHILD, hItem);
+ if (hCurItem)
+ {
+ int IgnoreIcon = CList->GetExtraImage(hCurItem, EXTRACOLUMN_IGNORE);
+ int AutoreplyIcon = GetRealReplyIcon(CList, hCurItem);
+ if (IgnoreIcon == EXTRAIMGLIST_MSG)
+ {
+ IgnoreIcon = EXTRAIMGLIST_DOT;
+ }
+ while ((hCurItem = CList->GetNextItem(MCLGN_NEXT | MCLGN_ANY, hCurItem)) && (IgnoreIcon != EXTRAIMGLIST_DOT || AutoreplyIcon != EXTRAIMGLIST_DOT))
+ {
+ if (CList->GetExtraImage(hCurItem, EXTRACOLUMN_IGNORE) != EXTRAIMGLIST_IGNORE)
+ {
+ IgnoreIcon = EXTRAIMGLIST_DOT;
+ }
+ int CurReplyIcon = GetRealReplyIcon(CList, hCurItem);
+ if (CurReplyIcon != AutoreplyIcon)
+ {
+ AutoreplyIcon = EXTRAIMGLIST_DOT;
+ }
+ }
+ CList->SetItemParam(hItem, AutoreplyIcon); // store Reply icon in item's Param, so that we can always get real reply icon for groups later, even if CLC_EXTRAICON_EMPTY is set instead of it
+ SetExtraIcon(CList, EXTRACOLUMN_IGNORE, hItem, IgnoreIcon);
+ SetExtraIcon(CList, EXTRACOLUMN_REPLY, hItem, AutoreplyIcon);
+ }
+ } while (hItem = CList->GetNextItem(MCLGN_PREV | MCLGN_GROUP | MCLGN_MULTILEVEL, hItem));
+}
+
+
+int GetSelContactsNum(CCList *CList, PTREEITEMARRAY Selection = NULL, bool *bOnlyInfo = NULL) // "SelContacts" mean not only contacts, but everything with "personal" status messages and settings - i.e. "All contacts" and protocol items are counted too.
+{
+ if (!CList)
+ {
+ return (Selection == CLSEL_DAT_NOTHING) ? 0 : 1; // Selection == NULL means we need to retrieve current selection by ourselves, and current selection is always CLSEL_DAT_CONTACT in this case
+ }
+ if (!Selection)
+ {
+ Selection = CList->GetSelection();
+ }
+ int I;
+ int nContacts = 0;
+ if (bOnlyInfo)
+ {
+ *bOnlyInfo = true;
+ }
+ for (I = 0; I < Selection->GetSize(); I++)
+ {
+ int ItemType = CList->GetItemType((*Selection)[I]);
+ if (bOnlyInfo && ItemType != MCLCIT_INFO)
+ {
+ *bOnlyInfo = false;
+ }
+ nContacts += ItemType == MCLCIT_CONTACT || ItemType == MCLCIT_INFO;
+ }
+ return nContacts;
+}
+
+
+void ApplySelContactsMessage(SetAwayMsgData* dat, CCList *CList, PTREEITEMARRAY Selection = NULL)
+{
+ TCString Message;
+ HWND hwndDlg = g_SetAwayMsgPage.GetWnd();
+ GetDlgItemText(hwndDlg, IDC_SAWAYMSG_MSGDATA, Message.GetBuffer(AWAY_MSGDATA_MAX), AWAY_MSGDATA_MAX);
+ Message.ReleaseBuffer();
+ if (!lstrlen(Message))
+ {
+ Message = NULL; // delete personal message if it's empty
+ }
+ if (CList)
+ {
+ if (!Selection)
+ {
+ Selection = CList->GetSelection();
+ }
+ int I;
+ for (I = 0; I < Selection->GetSize(); I++)
+ {
+ HTREEITEM hItem = (*Selection)[I];
+ int ItemType = CList->GetItemType(hItem);
+ if (ItemType == MCLCIT_CONTACT)
+ {
+ MCONTACT hContact = CList->GethContact(hItem);
+ CContactSettings(0, hContact).SetMsgFormat(SMF_PERSONAL, Message);
+ } else if (ItemType == MCLCIT_INFO)
+ {
+ char *szProto = (char*)CList->GetItemParam(hItem);
+ CProtoSettings(szProto).SetMsgFormat(SMF_PERSONAL, (szProto || Message != NULL) ? Message : _T("")); // "szProto || Message != NULL..." means that we'll set an empty message instead of NULL for the global status, if the message is empty (NULL for the global status has a special meaning - SetMsgFormat will set the default message instead of NULL)
+ } else
+ {
+ continue;
+ }
+ SetExtraIcon(CList, EXTRACOLUMN_IGNORE, hItem, -1); // refresh contact's personal message icon (MSG or DOT)
+ }
+ } else if (Selection != CLSEL_DAT_NOTHING)
+ {
+ if (dat->hInitContact)
+ CContactSettings(0, dat->hInitContact).SetMsgFormat(SMF_PERSONAL, Message);
+ else
+ CProtoSettings(dat->szProtocol).SetMsgFormat(SMF_PERSONAL, (dat->szProtocol || Message != NULL) ? Message : _T(""));
+ }
+ SendDlgItemMessage(hwndDlg, IDC_SAWAYMSG_MSGDATA, EM_SETMODIFY, false, 0);
+ SetDlgItemText(hwndDlg, IDC_OK, TranslateT("OK"));
+}
+
+
+void UpdateCheckboxesState(CCList *CList)
+{
+ _ASSERT(CList);
+ PTREEITEMARRAY Selection = CList->GetSelection();
+ int I;
+ int ReplyIcon = -1;
+ int IgnoreIcon = -1;
+ for (I = 0; I < Selection->GetSize() && (IgnoreIcon != EXTRAIMGLIST_DOT || ReplyIcon != EXTRAIMGLIST_DOT); I++)
+ {
+ HTREEITEM hItem = (*Selection)[I];
+ int ItemType = CList->GetItemType(hItem);
+ if (ItemType == MCLCIT_CONTACT || ItemType == MCLCIT_INFO)
+ {
+ int CurIgnoreIcon = CList->GetExtraImage(hItem, EXTRACOLUMN_IGNORE);
+ if (IgnoreIcon == -1)
+ {
+ IgnoreIcon = CurIgnoreIcon;
+ } else if (CurIgnoreIcon != EXTRAIMGLIST_IGNORE)
+ {
+ IgnoreIcon = EXTRAIMGLIST_DOT;
+ }
+ int CurReplyIcon = CList->GetExtraImage(hItem, EXTRACOLUMN_REPLY);
+ if (ReplyIcon == -1)
+ {
+ ReplyIcon = CurReplyIcon;
+ } else if (CurReplyIcon != ReplyIcon)
+ {
+ ReplyIcon = EXTRAIMGLIST_DOT;
+ }
+ }
+ }
+ HWND hwndDlg = g_SetAwayMsgPage.GetWnd();
+ CheckDlgButton(hwndDlg, IDC_SAWAYMSG_IGNOREREQ, (IgnoreIcon == EXTRAIMGLIST_IGNORE) ? BST_CHECKED : BST_UNCHECKED);
+ EnableWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SENDMSG), IgnoreIcon != EXTRAIMGLIST_IGNORE);
+ CheckDlgButton(hwndDlg, IDC_SAWAYMSG_SENDMSG, (ReplyIcon == EXTRAIMGLIST_AUTOREPLY_ON) ? BST_CHECKED : ((ReplyIcon == EXTRAIMGLIST_AUTOREPLY_OFF) ? BST_UNCHECKED : BST_INDETERMINATE));
+}
+
+HICON g_LoadIconEx( const char* name, bool big )
+{
+ char szSettingName[100];
+ mir_snprintf( szSettingName, sizeof( szSettingName ), "%s_%s", "", name );
+ return Skin_GetIcon(szSettingName, big);
+}
+INT_PTR CALLBACK SetAwayMsgDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
+{
+ static int SetMsgSplitterX, SetContactSplitterX;
+ static SetAwayMsgData *dat;
+ static HWND hWndTooltips;
+ static int UpdateLock = 0;
+ static int Countdown;
+ static CMsgTree *MsgTree = NULL;
+ static CCList *CList = NULL;
+ static struct {
+ int DlgItem, IconIndex;
+ TCHAR* Text;
+ } Buttons[] = {
+ IDC_SAWAYMSG_SAVEMSG, ILI_SAVE, LPGENT("Save, replacing the selected message"),
+ IDC_SAWAYMSG_SAVEASNEW, ILI_SAVEASNEW, LPGENT("Save as a new message"),
+ IDC_SAWAYMSG_NEWCATEGORY, ILI_NEWCATEGORY, LPGENT("Create new category"),
+ IDC_SAWAYMSG_DELETE, ILI_DELETE, LPGENT("Delete"),
+ IDC_SAWAYMSG_VARS, ILI_NOICON, LPGENT("Open Variables help dialog"),
+ IDC_SAWAYMSG_OPTIONS, ILI_SETTINGS, LPGENT("Show settings menu")
+ };
+
+
+ switch (uMsg)
+ {
+ case WM_INITDIALOG:
+ {
+ dat = (SetAwayMsgData*)lParam;
+ TranslateDialogDefault(hwndDlg);
+ g_SetAwayMsgPage.SetWnd(hwndDlg);
+ g_SetAwayMsgPage.DBToMemToPage();
+
+ HICON hTitleIconBigElse = LoadSkinnedIconBig(SKINICON_OTHER_MIRANDA);
+
+ char *szProto = dat->hInitContact ? GetContactProto(dat->hInitContact) : dat->szProtocol;
+ int Status = 0;
+ Status = g_ProtoStates[dat->szProtocol].Status;
+ HICON hTitleIcon = LoadSkinnedProtoIcon(szProto, Status);
+ HICON hTitleIconBig = LoadSkinnedProtoIconBig(szProto, Status);
+
+
+
+
+ if(hTitleIconBig == NULL || (HICON)CALLSERVICE_NOTFOUND == hTitleIconBig) {
+ SendMessage(hwndDlg, WM_SETICON, ICON_BIG, (LPARAM)hTitleIconBigElse);
+ } else {
+ SendMessage(hwndDlg, WM_SETICON, ICON_BIG, (LPARAM)hTitleIconBig);
+ }
+
+ SendMessage(hwndDlg, WM_SETICON, ICON_SMALL, (LPARAM)hTitleIcon);
+
+
+ SendDlgItemMessage(hwndDlg, IDC_SAWAYMSG_MSGDATA, EM_LIMITTEXT, AWAY_MSGDATA_MAX, 0);
+
+
+
+
+ // init window size variables / resize the window
+ RECT rc;
+ POINT pt;
+ GetWindowRect(GetDlgItem(hwndDlg, IDC_SAWAYMSG_MSGSPLITTER), &rc);
+ pt.x = (rc.left + rc.right) / 2;
+ pt.y = 0;
+ ScreenToClient(hwndDlg, &pt);
+ g_MsgSplitterX = pt.x;
+ RECT rcContactSplitter;
+ GetWindowRect(GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSPLITTER), &rcContactSplitter);
+ RECT rcMsgDlg;
+ GetWindowRect(hwndDlg, &rcMsgDlg);
+ g_ContactSplitterX = rcMsgDlg.right - rcContactSplitter.left - (rcContactSplitter.right - rcContactSplitter.left) / 2;
+ RECT rcVars;
+ GetWindowRect(GetDlgItem(hwndDlg, IDC_SAWAYMSG_VARS), &rcVars);
+ g_VariablesButtonDX = rcContactSplitter.left - rcVars.right;
+// GetWindowRect(GetDlgItem(hwndDlg, IDC_OK), &rc);
+ RECT rcOK;
+ GetWindowRect(GetDlgItem(hwndDlg, IDC_OK), &rcOK);
+ //MinMsgEditSize = rc.right - rc.left + rcContactSplitter.right - rcVars.left + 3;
+ MinMsgEditSize = rcOK.right - rc.left + rcContactSplitter.right - rcVars.left + 3;
+ rc.left = MINSPLITTERPOS;
+
+ //NightFox: use MINSPLITTERPOS as min size ;
+// GetWindowRect(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SAVEMSG), &rc);
+// GetWindowRect(GetDlgItem(hwndDlg, IDC_SAWAYMSG_DELETE), &rc);
+// rc.left = MINSPLITTERPOS;
+ //GetWindowRect(GetDlgItem(hwndDlg, IDC_OK), &rc);
+ //GetWindowRect(GetDlgItem(hwndDlg, IDC_OK), &rc);
+
+ pt.x = rc.right;
+ pt.y = 0;
+ ScreenToClient(hwndDlg, &pt);
+ MinMsgSplitterX = pt.x;
+ GetWindowRect(GetDlgItem(hwndDlg, IDC_SAWAYMSG_IGNOREREQ), &rc);
+ MinContactSplitterX = rcMsgDlg.right - rc.left + 1;
+
+ /*
+
+ //NightFox: add status bar
+
+ PROTOACCOUNT * acc = ProtoGetAccount(szProto);
+
+ //HWND hWndStatusBar = CreateStatusWindow(WS_CHILD | WS_VISIBLE | WS_CLIPCHILDREN | SBARS_SIZEGRIP, NULL, hwndDlg, 999);//IDC_STATUSBAR);
+ SetWindowPos(hWndStatusBar, HWND_BOTTOM, 0, 0, 0, 0, SWP_NOMOVE|SWP_NOSIZE);
+
+
+ SIZE sz;
+
+ HDC hdc = GetDC(hWndStatusBar);
+ HFONT hFntSave = (HFONT)SelectObject(hdc, GetStockObject(DEFAULT_GUI_FONT));
+ GetTextExtentPoint32(hdc, acc->tszAccountName, lstrlen(acc->tszAccountName), &sz);
+ sz.cx += GetSystemMetrics(SM_CXSMICON) * 3;
+ SelectObject(hdc, hFntSave);
+ ReleaseDC(hWndStatusBar, hdc);
+
+ RECT rcStatus; GetWindowRect(hWndStatusBar, &rcStatus);
+ int parts[] = { rcStatus.right-rcStatus.left - sz.cx, -1 };
+ SendMessage(hWndStatusBar, SB_SETPARTS, 2, (LPARAM)parts);
+ SendMessage(hWndStatusBar, SB_SETICON, 1, (LPARAM)LoadSkinnedProtoIcon(szProto, Status));
+ SendMessage(hWndStatusBar, SB_SETTEXT, 1, (LPARAM)acc->tszAccountName);
+ */
+
+
+
+
+
+
+
+
+
+
+ // [try] getting dialog position
+ int DlgPosX = db_get_dw(NULL, MOD_NAME, SAM_DB_DLGPOSX, -1);
+ int DlgPosY = db_get_dw(NULL, MOD_NAME, SAM_DB_DLGPOSY, -1);
+ int DlgSizeX = db_get_dw(NULL, MOD_NAME, SAM_DB_DLGSIZEX, -1);
+ int DlgSizeY = db_get_dw(NULL, MOD_NAME, SAM_DB_DLGSIZEY, -1);
+ int MsgSplitterX = db_get_dw(NULL, MOD_NAME, SAM_DB_MSGSPLITTERPOS, -1);
+ int ContactSplitterX = db_get_dw(NULL, MOD_NAME, SAM_DB_CONTACTSPLITTERPOS, -1);
+ if (DlgPosX >= 0 && DlgPosY >= 0 && DlgSizeX > 0 && DlgSizeY > 0 && MsgSplitterX > 0 && ContactSplitterX > 0)
+ {
+ RECT rcWorkArea, rcIntersect;
+ SystemParametersInfo(SPI_GETWORKAREA, 0, &rcWorkArea, 0);
+ RECT rcDlgPos = {DlgPosX, DlgPosY, DlgPosX + DlgSizeX, DlgPosY + DlgSizeY};
+ if (!IntersectRect(&rcIntersect, &rcDlgPos, &rcWorkArea)) // make sure the window will be visible
+ {
+ DlgPosX = rcWorkArea.left;
+ DlgPosY = rcWorkArea.top;
+ }
+ SetWindowPos(hwndDlg, NULL, DlgPosX, DlgPosY, DlgSizeX, DlgSizeY, SWP_NOZORDER);
+ g_MsgSplitterX = MsgSplitterX;
+ g_ContactSplitterX = ContactSplitterX;
+ }
+
+ SetMsgSplitterX = g_MsgSplitterX;
+ SetContactSplitterX = g_ContactSplitterX;
+ SendMessage(hwndDlg, WM_SIZE, 0, 0); // show/hide dialog controls accordingly to the settings
+
+ MsgTree = new CMsgTree(GetDlgItem(hwndDlg, IDC_SAWAYMSG_TREE)); // Attention: it's important to call NEW and DELETE in a proper order, as CMsgTree and CCList are setting their own WNDPROCs for the parent window, so we must prevent WNDPROC conflicts.
+ CList = NULL;
+ if (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWCONTACTTREE))
+ {
+ SendMessage(hwndDlg, UM_SAM_INITCLIST, 0, 0);
+ } else
+ {
+ NMCLIST nm;
+ nm.hdr.code = MCLN_SELCHANGED;
+ nm.hdr.hwndFrom = GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE);
+ nm.hdr.idFrom = IDC_SAWAYMSG_CONTACTSTREE;
+ nm.OldSelection = CLSEL_DAT_NOTHING;
+ nm.NewSelection = CLSEL_DAT_CONTACT;
+ SendMessage(hwndDlg, WM_NOTIFY, 0, (LPARAM)&nm);
+ }
+
+ // init message tree
+ if (g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_RECENTMSGSCOUNT) && g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_PERSTATUSMRM))
+ {
+ char *szProto = dat->hInitContact ? GetContactProto(dat->hInitContact) : dat->szProtocol;
+ int ID = GetRecentGroupID((szProto || !dat->hInitContact) ? g_ProtoStates[szProto].Status : ID_STATUS_AWAY);
+ CBaseTreeItem* pTreeItem = MsgTree->GetNextItem(MTGN_CHILD | MTGN_BYID, (CBaseTreeItem*)g_Messages_RecentRootID);
+ while (pTreeItem)
+ {
+ if (pTreeItem->Flags & TIF_GROUP)
+ {
+ if (pTreeItem->ID == ID)
+ {
+ pTreeItem->Flags |= TIF_EXPANDED; // leave expanded only one appropriate subgroup of Recent Messages group
+ } else
+ {
+ pTreeItem->Flags &= ~TIF_EXPANDED;
+ }
+ MsgTree->UpdateItem(pTreeItem->ID);
+ }
+ pTreeItem = MsgTree->GetNextItem(MTGN_NEXT, pTreeItem);
+ }
+// InvalidateRect(GetDlgItem(hwndDlg, IDC_SAWAYMSG_TREE), NULL, true);
+ }
+ int Order;
+ CProtoSettings(dat->szProtocol).GetMsgFormat(GMF_LASTORDEFAULT, &Order);
+ if (Order >= 0)
+ { // so just select an appropriate message tree item; MSGDATA text is filled automatically through SELCHANGED notification
+ MsgTree->SetSelection(Order, MTSS_BYORDER);
+ }
+
+ if (dat->Message != NULL) // this allows to override the default message
+ {
+ SetDlgItemText(hwndDlg, IDC_SAWAYMSG_MSGDATA, dat->Message);
+ SendDlgItemMessage(hwndDlg, IDC_SAWAYMSG_MSGDATA, EM_SETMODIFY, true, 0);
+ }
+ SendDlgItemMessage(hwndDlg, IDC_SAWAYMSG_MSGDATA, EM_SETSEL, 0, -1); // select all the text in the edit control
+
+ // init timer
+ Countdown = (dat->ISW_Flags & ISWF_NOCOUNTDOWN) ? -1 : g_MoreOptPage.GetDBValueCopy(IDC_MOREOPTDLG_WAITFORMSG);
+ if (Countdown == -1) // infinite
+ {
+ SetDlgItemText(hwndDlg, IDC_OK, TranslateT("OK"));
+ } else
+ {
+ SendMessage(hwndDlg, WM_TIMER, SAM_TIMER_ID, NULL);
+ SetTimer(hwndDlg, SAM_TIMER_ID, 1000, NULL);
+ }
+
+ // init image buttons
+ int I;
+ for (I = 0; I < lengthof(Buttons); I++)
+ {
+ HWND hButton = GetDlgItem(hwndDlg, Buttons[I].DlgItem);
+ SendMessage(hButton, BUTTONADDTOOLTIP, (WPARAM)TranslateTS(Buttons[I].Text), BATF_TCHAR);
+ SendMessage(hButton, BUTTONSETASFLATBTN, TRUE, 0);
+ }
+
+ // init tooltips
+ struct {
+ int DlgItemID;
+ TCHAR* Text;
+ } Tooltips[] = {
+ IDC_SAWAYMSG_IGNOREREQ, LPGENT("Don't send the status message to selected contact(s)"),
+ IDC_SAWAYMSG_SENDMSG, LPGENT("Send an autoreply to selected contact(s)"),
+ };
+ TOOLINFO ti = {0};
+ hWndTooltips = CreateWindowEx(WS_EX_TOPMOST, TOOLTIPS_CLASS, _T(""), WS_POPUP, 0, 0, 0, 0, NULL, NULL, GetModuleHandle(NULL), NULL);
+ ti.cbSize = sizeof(ti);
+ ti.uFlags = TTF_IDISHWND | TTF_SUBCLASS;
+ for (I = 0; I < lengthof(Tooltips); I++)
+ {
+ ti.uId = (UINT)GetDlgItem(hwndDlg, Tooltips[I].DlgItemID);
+ ti.lpszText = TranslateTS(Tooltips[I].Text);
+ SendMessage(hWndTooltips, TTM_ADDTOOL, 0, (LPARAM)&ti);
+ }
+ SendMessage(hwndDlg, UM_ICONSCHANGED, 0, 0);
+
+ g_OrigCListProc = (WNDPROC)SetWindowLongPtr(GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE), GWLP_WNDPROC, (LONG_PTR)CListSubclassProc);
+ g_OrigEditMsgProc = (WNDPROC)SetWindowLongPtr(GetDlgItem(hwndDlg, IDC_SAWAYMSG_MSGDATA), GWLP_WNDPROC, (LONG_PTR)MsgEditSubclassProc);
+ g_OrigSplitterProc = (WNDPROC)SetWindowLongPtr(GetDlgItem(hwndDlg, IDC_SAWAYMSG_MSGSPLITTER), GWLP_WNDPROC, (LONG_PTR)SplitterSubclassProc);
+ SetWindowLongPtr(GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSPLITTER), GWLP_WNDPROC, (LONG_PTR)SplitterSubclassProc);
+ return false;
+ } break;
+ case WM_NOTIFY:
+ {
+ switch (((NMHDR*)lParam)->idFrom)
+ {
+ case IDC_SAWAYMSG_CONTACTSTREE:
+ {
+ switch (((NMHDR*)lParam)->code) // check the notification code
+ {
+ case NM_CLICK:
+ {
+ _ASSERT(CList);
+ HWND hTreeView = GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE);
+ POINT pt = {(short)LOWORD(GetMessagePos()), (short)HIWORD(GetMessagePos())};
+ ScreenToClient(hTreeView, &pt);
+ DWORD hitFlags;
+ HTREEITEM hItem = CList->HitTest(&pt, &hitFlags);
+ if (hitFlags & MCLCHT_ONITEMEXTRA)
+ {
+ int nColumn = HIBYTE(HIWORD(hitFlags));
+ int CurIcon = CList->GetExtraImage(hItem, nColumn);
+ int ItemType = CList->GetItemType(hItem);
+ if (nColumn == EXTRACOLUMN_IGNORE)
+ {
+ CurIcon = (CurIcon == EXTRAIMGLIST_IGNORE) ? EXTRAIMGLIST_DOT : EXTRAIMGLIST_IGNORE;
+ } else
+ {
+ _ASSERT(nColumn == EXTRACOLUMN_REPLY);
+ switch (CurIcon)
+ {
+ case EXTRAIMGLIST_DOT: CurIcon = EXTRAIMGLIST_AUTOREPLY_OFF; break;
+ case EXTRAIMGLIST_AUTOREPLY_OFF: CurIcon = EXTRAIMGLIST_AUTOREPLY_ON; break;
+ case EXTRAIMGLIST_AUTOREPLY_ON: CurIcon = EXTRAIMGLIST_DOT; break;
+ }
+ }
+ SetExtraIcon(CList, nColumn, hItem, CurIcon);
+ if (ItemType == MCLCIT_GROUP) // set all child items
+ {
+ HTREEITEM hCurItem = CList->GetNextItem(MCLGN_CHILD, hItem);
+ HTREEITEM hLimitItem = CList->GetNextItem(MCLGN_NEXT | MCLGN_ANY | MCLGN_NOTCHILD, hItem);
+ while (hCurItem && hCurItem != hLimitItem)
+ {
+ SetExtraIcon(CList, nColumn, hCurItem, CurIcon);
+ hCurItem = CList->GetNextItem(MCLGN_NEXT | MCLGN_ANY | MCLGN_MULTILEVEL, hCurItem);
+ }
+ }
+ SetCListGroupIcons(dat, CList);
+ UpdateCheckboxesState(CList);
+ }
+ } break;
+ case MCLN_SELCHANGED:
+ {
+ PNMCLIST nm = (PNMCLIST)lParam;
+ TCString BtnTitle(TranslateT("OK"));
+ if (CList)
+ {
+ UpdateCheckboxesState(CList);
+ }
+ bool bOnlyInfo;
+ bool bLeaveOldMessage = nm->OldSelection == nm->NewSelection; // OldSelection == NewSelection when we send MCLN_SELCHANGED from UM_SAM_PROTOSTATUSCHANGED; seems that it's better to leave old message then
+ int nOldContacts = GetSelContactsNum(CList, nm->OldSelection);
+ int nNewContacts = GetSelContactsNum(CList, nm->NewSelection, &bOnlyInfo);
+ EnableWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_IGNOREREQ), !bOnlyInfo);
+ if (CList)
+ {
+ if (SendDlgItemMessage(hwndDlg, IDC_SAWAYMSG_MSGDATA, EM_GETMODIFY, 0, 0))
+ {
+ int I, J;
+ for (I = nm->OldSelection->GetSize() - 1; I >= 0; I--)
+ {
+ for (J = nm->NewSelection->GetSize() - 1; J >= 0; J--)
+ {
+ if ((*nm->NewSelection)[J] == (*nm->OldSelection)[I])
+ {
+ break;
+ }
+ }
+ if (J < 0)
+ {
+ break;
+ }
+ }
+ if (nNewContacts > 1)
+ {
+ BtnTitle = TranslateT("Apply");
+ }
+ if (I >= 0 && nOldContacts) // there's at least one item in the old selection that is not present in the new one
+ {
+ if ((nOldContacts == 1) || MessageBox(hwndDlg, TranslateT("Do you want to apply the message?"), TranslateT("New Away System"), MB_ICONQUESTION | MB_YESNO) == IDYES)
+ {
+ ApplySelContactsMessage(dat, CList, nm->OldSelection);
+ }
+ } else if (nOldContacts)
+ {
+ bLeaveOldMessage = true; // don't change the edit control text yet - we're still writing...
+ }
+ }
+ } else
+ {
+ if (nOldContacts && !nNewContacts)
+ {
+ ApplySelContactsMessage(dat, CList, nm->OldSelection);
+ }
+ }
+
+ // determine the right new message and window title for the edit control now
+ TCString Message;
+ int Status = 0;
+ if (CList)
+ {
+ bool MessageDetermined = false;
+ bool StatusDetermined = false;
+ int I;
+ for (I = 0; I < nm->NewSelection->GetSize(); I++)
+ {
+ HTREEITEM hItem = (*nm->NewSelection)[I];
+ MCONTACT hContact;
+ char *szProto;
+ int ItemType = CList->GetItemType(hItem);
+ if (ItemType == MCLCIT_CONTACT)
+ {
+ hContact = CList->GethContact(hItem);
+ _ASSERT(hContact);
+ szProto = GetContactProto(hContact);
+ _ASSERT(szProto);
+ } else if (ItemType == MCLCIT_INFO)
+ {
+ szProto = (char*)CList->GetItemParam(hItem);
+ }
+ if (ItemType == MCLCIT_CONTACT || ItemType == MCLCIT_INFO)
+ {
+ int CurStatus = g_ProtoStates[szProto].Status;
+ if (!MessageDetermined)
+ {
+ TCString CurMessage((ItemType == MCLCIT_CONTACT) ? CContactSettings(0, hContact).GetMsgFormat(GMF_PERSONAL) : CProtoSettings(szProto).GetMsgFormat(GMF_TEMPORARY | GMF_PERSONAL));
+ if (CurMessage == NULL)
+ {
+ CurMessage = _T("");
+ }
+ if (Message == NULL)
+ {
+ Message = CurMessage;
+ } else if (CurMessage != (const TCHAR*)Message)
+ {
+ Message = _T("");
+ BtnTitle = TranslateT("Apply");
+ MessageDetermined = true;
+ }
+ }
+ if (!StatusDetermined)
+ {
+ if (!Status)
+ {
+ Status = CurStatus;
+ } else if (CurStatus != Status)
+ {
+ Status = 0;
+ StatusDetermined = true;
+ }
+ }
+ }
+ }
+ } else if (nNewContacts) // if (!CList)
+ {
+ Status = g_ProtoStates[dat->szProtocol].Status;
+ Message = dat->hInitContact ? CContactSettings(0, dat->hInitContact).GetMsgFormat(GMF_PERSONAL) : CProtoSettings(dat->szProtocol).GetMsgFormat(GMF_TEMPORARY | GMF_PERSONAL);
+ }
+ if (!bLeaveOldMessage)
+ {
+ SetDlgItemText(hwndDlg, IDC_SAWAYMSG_MSGDATA, (Message == NULL) ? _T("") : Message);
+ }
+ SetDlgItemText(hwndDlg, IDC_OK, BtnTitle);
+ //NightFox: fix titlebar
+ //TCString WindowTitle(TranslateT("Set "));
+ TCString WindowTitle(TranslateT("Set message for"));
+ WindowTitle += _T(" \"");
+ WindowTitle += Status ? (TCHAR*)CallService(MS_CLIST_GETSTATUSMODEDESCRIPTION, Status, GSMDF_TCHAR) : TranslateT("Statuses");
+ //WindowTitle += TranslateT(" message");
+ if (nNewContacts == 1)
+ {
+ WindowTitle += TCString(TranslateT("\" ("));
+ MCONTACT hContact = NULL;
+ char *szProto = NULL;
+ if (CList)
+ {
+ int I;
+ for (I = 0; I < nm->NewSelection->GetSize(); I++)
+ {
+ HTREEITEM hItem = (*nm->NewSelection)[I];
+ int ItemType = CList->GetItemType(hItem);
+ if (ItemType == MCLCIT_CONTACT)
+ {
+ hContact = CList->GethContact((*nm->NewSelection)[I]);
+ break;
+ } else if (ItemType == MCLCIT_INFO)
+ {
+ szProto = (char*)CList->GetItemParam(hItem);
+ break;
+ }
+ }
+ } else
+ {
+ hContact = dat->hInitContact;
+ szProto = dat->szProtocol;
+ }
+ if (hContact)
+ {
+
+ if (IsAnICQProto(GetContactProto(hContact))) {
+ WindowTitle += TranslateT("message for");
+ WindowTitle += _T(" ");
+ } else {
+ WindowTitle += TranslateT("for");
+ WindowTitle += _T(" ");
+ }
+
+
+ WindowTitle += (TCHAR*)CallService(MS_CLIST_GETCONTACTDISPLAYNAME, (WPARAM)hContact, GCDNF_TCHAR);
+ if (!IsAnICQProto(GetContactProto(hContact)))
+ {
+ //WindowTitle += TranslateT(" (autoreply only)");
+ WindowTitle += _T(" ");
+ WindowTitle += TranslateT("available autoreply only");
+ WindowTitle += _T(")");
+ } else {
+ WindowTitle += _T(")");
+ }
+ } else
+ {
+ if (!szProto)
+ {
+ //WindowTitle += TranslateT("all contacts");
+ WindowTitle += TranslateT("all accounts");
+ WindowTitle += _T(")");
+ } else
+ {
+ //CString ProtoTitle;
+
+ /*if (CallProtoService(szProto, PS_GETNAME, 256, (LPARAM)ProtoTitle.GetBuffer(256)))
+ { // on a failure, set ProtoTitle to an empty string
+ //ProtoTitle[0] = '\0';
+ }*/
+ //ProtoTitle.ReleaseBuffer();
+ //WindowTitle += _A2T(ProtoTitle) + TranslateT(" protocol");
+
+ PROTOACCOUNT * acc = ProtoGetAccount(szProto);
+
+ WindowTitle += acc->tszAccountName;
+ //WindowTitle += _T(" ");
+ //WindowTitle += TranslateT("account");
+ WindowTitle += _T(")");
+ }
+ }
+ } else {
+ WindowTitle += _T("\"");
+ }
+ SetWindowText(hwndDlg, WindowTitle);
+ } break;
+ }
+ } break;
+ case IDC_SAWAYMSG_TREE:
+ {
+ switch (((NMHDR*)lParam)->code)
+ {
+ case MTN_SELCHANGED:
+ {
+ if (!UpdateLock)
+ {
+ PNMMSGTREE pnm = (PNMMSGTREE)lParam;
+ if (pnm->ItemNew && !(pnm->ItemNew->Flags & (TIF_ROOTITEM | TIF_GROUP)))
+ {
+ SetDlgItemText(hwndDlg, IDC_SAWAYMSG_MSGDATA, ((CTreeItem*)pnm->ItemNew)->User_Str1);
+ SendMessage(hwndDlg, WM_COMMAND, MAKEWPARAM(IDC_SAWAYMSG_MSGDATA, EN_CHANGE), (LPARAM)GetDlgItem(hwndDlg, IDC_SAWAYMSG_MSGDATA));
+ SendDlgItemMessage(hwndDlg, IDC_SAWAYMSG_MSGDATA, EM_SETMODIFY, true, 0);
+ EnableWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SAVEMSG), true);
+ } else
+ {
+ EnableWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SAVEMSG), false);
+ }
+ EnableWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_DELETE), pnm->ItemNew && !(pnm->ItemNew->Flags & TIF_ROOTITEM));
+ }
+ } break;
+ }
+ } break;
+ }
+ return true;
+ } break;
+ case WM_TIMER:
+ {
+ if (wParam == SAM_TIMER_ID)
+ {
+ if (!Countdown)
+ {
+ SendMessage(hwndDlg, UM_SAM_APPLYANDCLOSE, 0, 0);
+ return true;
+ }
+ TCHAR BtnTitle[64];
+ _sntprintf(BtnTitle, sizeof(BtnTitle), TranslateT("Closing in %d"), Countdown);
+ SetDlgItemText(hwndDlg, IDC_OK, BtnTitle);
+ Countdown--;
+ }
+ } break;
+ case UM_SAM_INITCLIST:
+ {
+ _ASSERT(!CList);
+ CList = new CCList(GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE));
+
+ // init contact tree
+ HIMAGELIST hil;
+ hil = ImageList_Create(GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON), ILC_COLOR32 | ILC_MASK, 5, 2);
+ ImageList_AddIcon(hil, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_DOT)));
+ ImageList_AddIcon(hil, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_IGNORE)));
+ ImageList_AddIcon(hil, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_MSGICON)));
+ ImageList_AddIcon(hil, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_ENABLED)));
+ ImageList_AddIcon(hil, LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SOE_DISABLED)));
+ CList->SetExtraImageList(hil);
+
+ HTREEITEM hSelItem;
+ HTREEITEM hItem = hSelItem = CList->AddInfo(TranslateT("** All contacts **"), CLC_ROOT, CLC_ROOT, NULL, LoadSkinnedProtoIcon(NULL, g_ProtoStates[(char*)NULL].Status));
+ int ProtoCount;
+ PROTOCOLDESCRIPTOR **proto;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&ProtoCount, (LPARAM)&proto);
+ int I;
+ for (I = 0; I < ProtoCount; I++)
+ {
+ if (proto[I]->type == PROTOTYPE_PROTOCOL && CallProtoService(proto[I]->szName, PS_GETCAPS, PFLAGNUM_1, 0) & PF1_MODEMSGSEND) // don't forget to change Recent Message Save loop in the UM_SAM_APPLYANDCLOSE if you're changing something here
+ {
+ //CString ProtoTitle;
+ //CallProtoService(proto[I]->szName, PS_GETNAME, 256, (LPARAM)ProtoTitle.GetBuffer(256));
+ //ProtoTitle.ReleaseBuffer();
+
+ PROTOACCOUNT * acc = ProtoGetAccount(proto[I]->szName);
+ //NightFox: protocols -> accounts
+ //hItem = CList->AddInfo(TCString(_T("* ")) + _A2T(ProtoTitle) + TranslateT(" contacts *"), CLC_ROOT, hItem, (LPARAM)proto[I]->szName, LoadSkinnedProtoIcon(proto[I]->szName, g_ProtoStates[proto[I]->szName].Status));
+ hItem = CList->AddInfo(TCString(_T("* ")) + acc->tszAccountName/* + TranslateT(" contacts *")*/ + _T(" *"), CLC_ROOT, hItem, (LPARAM)proto[I]->szName, LoadSkinnedProtoIcon(proto[I]->szName, g_ProtoStates[proto[I]->szName].Status));
+ if (dat->szProtocol && !strcmp(proto[I]->szName, dat->szProtocol))
+ {
+ hSelItem = hItem;
+ }
+ }
+ }
+
+ MCONTACT hContact = db_find_first();
+ CList->SetRedraw(false);
+ do
+ {
+ char *szProto = GetContactProto(hContact);
+ if (szProto)
+ {
+ int Flag1 = CallProtoService(szProto, PS_GETCAPS, PFLAGNUM_1, 0);
+ if ((Flag1 & PF1_IM) == PF1_IM || Flag1 & PF1_INDIVMODEMSG)
+ { // check if the protocol supports message sending/receiving or individual status messages before adding this contact
+ HTREEITEM hItem = CList->AddContact(hContact);
+ if (dat->hInitContact == hContact)
+ {
+ hSelItem = hItem;
+ }
+ }
+ }
+ } while (hContact = db_find_next(hContact));
+ CList->SortContacts();
+ hItem = CLC_ROOT;
+ while (hItem = CList->GetNextItem(MCLGN_NEXT | MCLGN_CONTACT | MCLGN_INFO | MCLGN_MULTILEVEL, hItem))
+ {
+ SetExtraIcon(CList, EXTRACOLUMN_IGNORE, hItem, -1);
+ SetExtraIcon(CList, EXTRACOLUMN_REPLY, hItem, -1);
+ }
+ SetCListGroupIcons(dat, CList);
+ CList->SetRedraw(true); // SetRedraw must be before SelectItem - otherwise SelectItem won't change scrollbar position to make the selected contact visible
+ CList->SelectItem(hSelItem); // must be selected after setting all extra icons, to set checkboxes properly
+ } break;
+ case UM_SAM_APPLYANDCLOSE:
+ {
+ KillTimer(hwndDlg, SAM_TIMER_ID);
+ if (CList)
+ {
+ CList->SelectItem(NULL);
+ } else
+ {
+ NMCLIST nm;
+ nm.hdr.code = MCLN_SELCHANGED;
+ nm.hdr.hwndFrom = GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE);
+ nm.hdr.idFrom = IDC_SAWAYMSG_CONTACTSTREE;
+ nm.OldSelection = CLSEL_DAT_CONTACT;
+ nm.NewSelection = CLSEL_DAT_NOTHING;
+ SendMessage(hwndDlg, WM_NOTIFY, 0, (LPARAM)&nm);
+ }
+ MsgTree->Save();
+ // save Recent Messages
+ int ProtoCount;
+ PROTOCOLDESCRIPTOR **proto;
+ CallService(MS_PROTO_ENUMACCOUNTS, (WPARAM)&ProtoCount, (LPARAM)&proto);
+ int I;
+ for (I = 0; I < ProtoCount; I++)
+ {
+ if (proto[I]->type == PROTOTYPE_PROTOCOL && CallProtoService(proto[I]->szName, PS_GETCAPS, PFLAGNUM_1, 0) & PF1_MODEMSGSEND)
+ {
+ TCString Message(CProtoSettings(proto[I]->szName).GetMsgFormat(GMF_PERSONAL)); // yes, we don't specify GMF_TEMPORARY here, because we don't need to save it
+ if (Message != NULL)
+ {
+ CProtoSettings(proto[I]->szName).SetMsgFormat(SMF_LAST, Message); // if the user set a message for this protocol, save it to the recent messages
+ }
+ ChangeProtoMessages(proto[I]->szName, g_ProtoStates[proto[I]->szName].Status, TCString(NULL)); // and actual setting of a status message for the protocol
+ }
+ }
+ TCString Message(CProtoSettings().GetMsgFormat(GMF_PERSONAL));
+ if (Message != NULL)
+ {
+ CProtoSettings().SetMsgFormat(SMF_LAST, Message); // save the global message to the recent messages
+ }
+ if (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_AUTOSAVEDLGSETTINGS))
+ {
+ SendMessage(hwndDlg, UM_SAM_SAVEDLGSETTINGS, 0, 0);
+ }
+ if (dat->IsModeless)
+ {
+ DestroyWindow(hwndDlg);
+ } else
+ {
+ EndDialog(hwndDlg, 2);
+ }
+ return true;
+ } break;
+ case UM_SAM_SAVEDLGSETTINGS:
+ {
+ RECT rcRect;
+ GetWindowRect(hwndDlg, &rcRect);
+ db_set_dw(NULL, MOD_NAME, SAM_DB_DLGPOSX, rcRect.left);
+ db_set_dw(NULL, MOD_NAME, SAM_DB_DLGPOSY, rcRect.top);
+ db_set_dw(NULL, MOD_NAME, SAM_DB_DLGSIZEX, rcRect.right - rcRect.left);
+ db_set_dw(NULL, MOD_NAME, SAM_DB_DLGSIZEY, rcRect.bottom - rcRect.top);
+ db_set_dw(NULL, MOD_NAME, SAM_DB_MSGSPLITTERPOS, g_MsgSplitterX);
+ db_set_dw(NULL, MOD_NAME, SAM_DB_CONTACTSPLITTERPOS, g_ContactSplitterX);
+ g_SetAwayMsgPage.PageToMemToDB();
+ } break;
+ case UM_SAM_REPLYSETTINGCHANGED:
+ { // wParam = (HANDLE)hContact
+ if (CList)
+ {
+ HTREEITEM hItem = CLC_ROOT;
+ if (!wParam)
+ { // it's the global autoreply setting
+ while (hItem = CList->GetNextItem(MCLGN_NEXT | MCLGN_INFO | MCLGN_MULTILEVEL, hItem))
+ {
+ if (!CList->GetItemParam(hItem))
+ { // we found the item
+ SetExtraIcon(CList, EXTRACOLUMN_REPLY, hItem, -1); // update it
+ break;
+ }
+ }
+ } else
+ { // it's a contact's autoreply setting
+ while (hItem = CList->GetNextItem(MCLGN_NEXT | MCLGN_CONTACT | MCLGN_MULTILEVEL, hItem))
+ {
+ MCONTACT hContact = CList->GethContact(hItem);
+ if (CList->GethContact(hItem) == wParam)
+ { // we found the item
+ SetExtraIcon(CList, EXTRACOLUMN_REPLY, hItem, -1); // update it
+ break;
+ }
+ }
+ }
+ UpdateCheckboxesState(CList);
+ }
+ } break;
+ case UM_SAM_PROTOSTATUSCHANGED:
+ { // wParam = (char*)szProto
+ if (CList)
+ {
+ HTREEITEM hItem = CLC_ROOT;
+ while (hItem = CList->GetNextItem(MCLGN_NEXT | MCLGN_CONTACT | MCLGN_INFO | MCLGN_MULTILEVEL, hItem))
+ {
+ if (CList->GetItemType(hItem) == MCLCIT_INFO)
+ {
+ char *szProto = (char*)CList->GetItemParam(hItem);
+ if (!wParam || !lstrcmpA(szProto, (char*)wParam))
+ {
+ CList->SetInfoIcon(hItem, LoadSkinnedProtoIcon(szProto, g_ProtoStates[szProto].Status));
+ }
+ }
+ SetExtraIcon(CList, EXTRACOLUMN_IGNORE, hItem, -1);
+ SetExtraIcon(CList, EXTRACOLUMN_REPLY, hItem, -1);
+ }
+ SetCListGroupIcons(dat, CList);
+ }
+/* dat->hInitContact = NULL; // TODO: test it
+ dat->szProtocol = (char*)wParam;*/
+ NMCLIST nm;
+ nm.hdr.code = MCLN_SELCHANGED;
+ nm.hdr.hwndFrom = GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE);
+ nm.hdr.idFrom = IDC_SAWAYMSG_CONTACTSTREE;
+ nm.OldSelection = nm.NewSelection = CList ? CList->GetSelection() : CLSEL_DAT_CONTACT;
+ SendMessage(hwndDlg, WM_NOTIFY, 0, (LPARAM)&nm); // everything else is in the MCLN_SELCHANGED handler, so we'll just call it from here.
+ } break;
+ case UM_ICONSCHANGED:
+ {
+ int I;
+ for (I = 0; I < lengthof(Buttons); I++)
+ {
+ if (Buttons[I].IconIndex != ILI_NOICON)
+ {
+ SendDlgItemMessage(hwndDlg, Buttons[I].DlgItem, BM_SETIMAGE, IMAGE_ICON, (LPARAM)g_IconList[Buttons[I].IconIndex]);
+ }
+ }
+ my_variables_skin_helpbutton(hwndDlg, IDC_SAWAYMSG_VARS);
+ } break;
+ case WM_COMMAND:
+ {
+ switch (HIWORD(wParam))
+ {
+ case EN_CHANGE:
+ {
+ if (LOWORD(wParam) == IDC_SAWAYMSG_MSGDATA)
+ {
+ SetDlgItemText(hwndDlg, IDC_OK, TranslateTS((GetSelContactsNum(CList) > 1) ? LPGENT("Apply") : LPGENT("OK")));
+ }
+ } break;
+ case EN_KILLFOCUS:
+ {
+ if (LOWORD(wParam) == IDC_SAWAYMSG_MSGDATA && GetForegroundWindow() == hwndDlg)
+ {
+ SendMessage(hwndDlg, UM_SAM_KILLTIMER, 0, 0);
+ }
+ } break;
+ case BN_CLICKED:
+ {
+ switch (LOWORD(wParam))
+ {
+ case IDC_SAWAYMSG_IGNOREREQ:
+ {
+ _ASSERT(CList);
+ int Ignore = IsDlgButtonChecked(hwndDlg, IDC_SAWAYMSG_IGNOREREQ) == BST_CHECKED;
+ EnableWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SENDMSG), !Ignore);
+ PTREEITEMARRAY Selection = CList->GetSelection();
+ int I;
+ for (I = 0; I < Selection->GetSize(); I++)
+ {
+ if (CList->GetItemType((*Selection)[I]) == MCLCIT_CONTACT)
+ {
+ SetExtraIcon(CList, EXTRACOLUMN_IGNORE, (*Selection)[I], Ignore ? EXTRAIMGLIST_IGNORE : EXTRAIMGLIST_DOT);
+ }
+ }
+ if (Selection->GetSize() == 1)
+ {
+ CList->EnsureVisible((*Selection)[0]);
+ }
+ SetCListGroupIcons(dat, CList);
+ } break;
+ case IDC_SAWAYMSG_SENDMSG:
+ {
+ _ASSERT(CList);
+ int Reply = IsDlgButtonChecked(hwndDlg, IDC_SAWAYMSG_SENDMSG);
+ int ReplyIcon = (Reply == BST_CHECKED) ? EXTRAIMGLIST_AUTOREPLY_ON : ((Reply == BST_UNCHECKED) ? EXTRAIMGLIST_AUTOREPLY_OFF : EXTRAIMGLIST_DOT);
+ PTREEITEMARRAY Selection = CList->GetSelection();
+ int I;
+ for (I = 0; I < Selection->GetSize(); I++)
+ {
+ SetExtraIcon(CList, EXTRACOLUMN_REPLY, (*Selection)[I], ReplyIcon);
+ }
+ if (Selection->GetSize() == 1)
+ {
+ CList->EnsureVisible((*Selection)[0]);
+ }
+ SetCListGroupIcons(dat, CList);
+ } break;
+ case IDC_SAWAYMSG_OPTIONS:
+ {
+ HMENU hMenu = LoadMenu(g_hInstance, MAKEINTRESOURCE(IDR_SAM_OPTIONS));
+ _ASSERT(hMenu);
+ HMENU hPopupMenu = GetSubMenu(hMenu, 0);
+ TranslateMenu(hPopupMenu);
+ CheckMenuItem(hPopupMenu, IDM_SAM_OPTIONS_SHOWMSGTREE, MF_BYCOMMAND | (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWMSGTREE) ? MF_CHECKED : MF_UNCHECKED));
+ CheckMenuItem(hPopupMenu, IDM_SAM_OPTIONS_SHOWCONTACTTREE, MF_BYCOMMAND | (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWCONTACTTREE) ? MF_CHECKED : MF_UNCHECKED));
+ CheckMenuItem(hPopupMenu, IDM_SAM_OPTIONS_AUTOSAVEDLGSETTINGS, MF_BYCOMMAND | (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_AUTOSAVEDLGSETTINGS) ? MF_CHECKED : MF_UNCHECKED));
+ int VariablesInstalled = ServiceExists(MS_VARS_FORMATSTRING);
+ CheckMenuItem(hPopupMenu, IDM_SAM_OPTIONS_DISABLEVARIABLES, MF_BYCOMMAND | ((g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_DISABLEVARIABLES) || !VariablesInstalled) ? MF_CHECKED : MF_UNCHECKED));
+ if (!VariablesInstalled)
+ {
+ EnableMenuItem(hPopupMenu, IDM_SAM_OPTIONS_DISABLEVARIABLES, MF_BYCOMMAND | MF_GRAYED);
+ }
+ switch (TrackPopupMenu(hPopupMenu, TPM_LEFTALIGN | TPM_RIGHTBUTTON | TPM_NONOTIFY | TPM_RETURNCMD, (short)LOWORD(GetMessagePos()), (short)HIWORD(GetMessagePos()), 0, hwndDlg, NULL))
+ {
+ case IDM_SAM_OPTIONS_SHOWMSGTREE:
+ {
+ int bShow = !g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWMSGTREE);
+ g_SetAwayMsgPage.SetValue(IDS_SAWAYMSG_SHOWMSGTREE, bShow);
+ RECT rcDlg;
+ GetWindowRect(hwndDlg, &rcDlg);
+ rcDlg.left -= bShow ? g_MsgSplitterX : -g_MsgSplitterX;
+ SendMessage(hwndDlg, WM_SIZING, WMSZ_LEFT, (LPARAM)&rcDlg);
+ SetWindowPos(hwndDlg, NULL, rcDlg.left, rcDlg.top, rcDlg.right - rcDlg.left, rcDlg.bottom - rcDlg.top, SWP_NOZORDER);
+ SendMessage(hwndDlg, WM_SIZE, 0, 0); // show/hide dialog controls accordingly
+ } break;
+ case IDM_SAM_OPTIONS_SHOWCONTACTTREE:
+ {
+ int bShow = !g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWCONTACTTREE);
+ if (bShow && !CList)
+ {
+ NMCLIST nm;
+ nm.hdr.code = MCLN_SELCHANGED;
+ nm.hdr.hwndFrom = GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE);
+ nm.hdr.idFrom = IDC_SAWAYMSG_CONTACTSTREE;
+ nm.OldSelection = CLSEL_DAT_CONTACT;
+ nm.NewSelection = CLSEL_DAT_NOTHING;
+ SendMessage(hwndDlg, WM_NOTIFY, 0, (LPARAM)&nm); // save current message
+ SendMessage(hwndDlg, UM_SAM_INITCLIST, 0, 0);
+ }
+ g_SetAwayMsgPage.SetValue(IDS_SAWAYMSG_SHOWCONTACTTREE, bShow);
+ RECT rcDlg;
+ GetWindowRect(hwndDlg, &rcDlg);
+ rcDlg.right += bShow ? g_ContactSplitterX : -g_ContactSplitterX;
+ SendMessage(hwndDlg, WM_SIZING, WMSZ_RIGHT, (LPARAM)&rcDlg);
+ SetWindowPos(hwndDlg, NULL, 0, 0, rcDlg.right - rcDlg.left, rcDlg.bottom - rcDlg.top, SWP_NOZORDER | SWP_NOMOVE);
+ SendMessage(hwndDlg, WM_SIZE, 0, 0); // show/hide dialog controls accordingly
+ } break;
+ case IDM_SAM_OPTIONS_AUTOSAVEDLGSETTINGS:
+ {
+ g_SetAwayMsgPage.SetDBValue(IDS_SAWAYMSG_AUTOSAVEDLGSETTINGS, !g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_AUTOSAVEDLGSETTINGS));
+ } break;
+ case IDM_SAM_OPTIONS_SAVEDLGSETTINGSNOW:
+ {
+ SendMessage(hwndDlg, UM_SAM_SAVEDLGSETTINGS, 0, 0);
+ } break;
+ case IDM_SAM_OPTIONS_DISABLEVARIABLES:
+ {
+ g_SetAwayMsgPage.SetDBValue(IDS_SAWAYMSG_DISABLEVARIABLES, !g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_DISABLEVARIABLES));
+ } break;
+ }
+ DestroyMenu(hMenu);
+ } break;
+ case IDC_SAWAYMSG_SAVEMSG:
+ {
+ CBaseTreeItem* TreeItem = MsgTree->GetSelection();
+ if (TreeItem && !(TreeItem->Flags & TIF_ROOTITEM))
+ {
+ MsgTree->EnsureVisible(TreeItem->hItem);
+ TCString NewMsg;
+ GetDlgItemText(hwndDlg, IDC_SAWAYMSG_MSGDATA, NewMsg.GetBuffer(AWAY_MSGDATA_MAX), AWAY_MSGDATA_MAX);
+ NewMsg.ReleaseBuffer();
+ if (((CTreeItem*)TreeItem)->User_Str1 != (const TCHAR*)NewMsg)
+ {
+ ((CTreeItem*)TreeItem)->User_Str1 = NewMsg;
+ MsgTree->SetModified(true);
+ }
+ }
+ } break;
+ case IDC_SAWAYMSG_SAVEASNEW:
+ {
+ TCString Text;
+ GetDlgItemText(hwndDlg, IDC_SAWAYMSG_MSGDATA, Text.GetBuffer(AWAY_MSGDATA_MAX), AWAY_MSGDATA_MAX);
+ Text.ReleaseBuffer();
+ UpdateLock++;
+ CTreeItem* TreeItem = MsgTree->AddMessage();
+ UpdateLock--;
+ TreeItem->User_Str1 = Text;
+ } break;
+ case IDC_SAWAYMSG_NEWCATEGORY:
+ {
+ MsgTree->AddCategory();
+ } break;
+ case IDC_SAWAYMSG_DELETE:
+ {
+ MsgTree->EnsureVisible(MsgTree->GetSelection()->hItem);
+ MsgTree->DeleteSelectedItem();
+ } break;
+ case IDC_SAWAYMSG_VARS:
+ {
+ my_variables_showhelp(hwndDlg, IDC_SAWAYMSG_MSGDATA);
+ } break;
+ case IDC_OK:
+ {
+ // save OK button title before resetting it in SetSelContactsMessage
+ TCString BtnTitle;
+ GetDlgItemText(hwndDlg, IDC_OK, BtnTitle.GetBuffer(64), 64);
+ BtnTitle.ReleaseBuffer();
+ ApplySelContactsMessage(dat, CList);
+ if (BtnTitle != (const TCHAR*)TranslateT("Apply"))
+ {
+ SendMessage(hwndDlg, UM_SAM_APPLYANDCLOSE, 0, 0);
+ }
+ } break;
+ }
+ } break;
+ }
+ } break;
+ case UM_SAM_SPLITTERMOVED:
+ {
+ switch (lParam)
+ {
+ RECT rc;
+ POINT pt;
+ case IDC_SAWAYMSG_MSGSPLITTER:
+ {
+ GetClientRect(hwndDlg, &rc);
+ pt.x = wParam;
+ pt.y = 0;
+ ScreenToClient(hwndDlg, &pt);
+ g_MsgSplitterX = (pt.x < MINSPLITTERPOS) ? MINSPLITTERPOS : pt.x;
+ int MaxSetSplitterX = rc.right - MINSPLITTERPOS + 2;
+ int MaxSplitterX = MaxSetSplitterX;
+ if (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWCONTACTTREE))
+ {
+ MaxSetSplitterX -= g_ContactSplitterX;
+ MaxSplitterX -= MINSPLITTERPOS;
+ }
+ if (g_MsgSplitterX > MaxSetSplitterX)
+ {
+ if (g_MsgSplitterX > MaxSplitterX)
+ {
+ g_ContactSplitterX = MINSPLITTERPOS;
+ g_MsgSplitterX = MaxSplitterX;
+ } else
+ {
+ g_ContactSplitterX = MINSPLITTERPOS + MaxSplitterX - g_MsgSplitterX;
+ }
+ }
+ } break;
+ case IDC_SAWAYMSG_CONTACTSPLITTER:
+ {
+ GetClientRect(hwndDlg, &rc);
+ pt.x = wParam;
+ pt.y = 0;
+ ScreenToClient(hwndDlg, &pt);
+ g_ContactSplitterX = (rc.right - pt.x < MINSPLITTERPOS) ? MINSPLITTERPOS : (rc.right - pt.x);
+ int MaxSetSplitterX = rc.right - MINSPLITTERPOS + 2;
+ int MaxSplitterX = MaxSetSplitterX;
+ if (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWMSGTREE))
+ {
+ MaxSetSplitterX -= g_MsgSplitterX;
+ MaxSplitterX -= MINSPLITTERPOS;
+ }
+ if (g_ContactSplitterX > MaxSetSplitterX)
+ {
+ if (g_ContactSplitterX > MaxSplitterX)
+ {
+ g_MsgSplitterX = MINSPLITTERPOS;
+ g_ContactSplitterX = MaxSplitterX;
+ } else
+ {
+ g_MsgSplitterX = MINSPLITTERPOS + MaxSplitterX - g_ContactSplitterX;
+ }
+ }
+ } break;
+ }
+ SetMsgSplitterX = g_MsgSplitterX;
+ SetContactSplitterX = g_ContactSplitterX;
+ SendMessage(hwndDlg, WM_SIZE, 0, 0);
+ } break;
+ case WM_SIZING:
+ {
+ RECT *prcDlg = (RECT*)lParam;
+ int MinSetXSize = MINSPLITTERPOS + 7;
+ int MinXSize = MinMsgEditSize + 7;
+ if (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWMSGTREE))
+ {
+ MinSetXSize += SetMsgSplitterX - 1;
+ MinXSize += MinMsgSplitterX - 1;
+ }
+ if (g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWCONTACTTREE))
+ {
+ MinSetXSize += SetContactSplitterX - 1;
+ MinXSize += MinContactSplitterX - 1;
+ }
+ if (prcDlg->right - prcDlg->left < MinXSize)
+ {
+ if (wParam != WMSZ_LEFT && wParam != WMSZ_TOPLEFT && wParam != WMSZ_BOTTOMLEFT)
+ {
+ prcDlg->right = prcDlg->left + MinXSize;
+ } else
+ {
+ prcDlg->left = prcDlg->right - MinXSize;
+ }
+ }
+ if (prcDlg->right - prcDlg->left < MinSetXSize)
+ {
+ int Delta = MinSetXSize - (prcDlg->right - prcDlg->left);
+ if (SetMsgSplitterX > MinMsgSplitterX)
+ {
+ int D2 = min(Delta, SetMsgSplitterX - MinMsgSplitterX);
+ g_MsgSplitterX = SetMsgSplitterX - D2;
+ Delta -= D2;
+ }
+ g_ContactSplitterX = SetContactSplitterX - Delta;
+ } else
+ {
+ g_MsgSplitterX = SetMsgSplitterX;
+ g_ContactSplitterX = SetContactSplitterX;
+ }
+ if (prcDlg->bottom - prcDlg->top < MINYDLGSIZE)
+ {
+ if (wParam != WMSZ_TOP && wParam != WMSZ_TOPLEFT && wParam != WMSZ_TOPRIGHT)
+ {
+ prcDlg->bottom = prcDlg->top + MINYDLGSIZE;
+ } else
+ {
+ prcDlg->top = prcDlg->bottom - MINYDLGSIZE;
+ }
+ }
+ return true;
+ } break;
+ case WM_SIZE:
+ {
+ UTILRESIZEDIALOG urd = {0};
+ urd.cbSize = sizeof(urd);
+ urd.hInstance = g_hInstance;
+ urd.hwndDlg = hwndDlg;
+ urd.lParam = (LPARAM)&g_SetAwayMsgPage;
+ urd.lpTemplate = MAKEINTRESOURCEA(IDD_SETAWAYMSG);
+ urd.pfnResizer = SetAwayMsgDlgResize;
+ CallService(MS_UTILS_RESIZEDIALOG, 0, (LPARAM)&urd);
+ if (!wParam && !lParam) // means that we sent WM_SIZE message to apply new settings to the dialog; probably it's somewhat a misuse, but who cares ;-P
+ {
+ int bShow;
+ bShow = g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWMSGTREE) ? SW_SHOW : SW_HIDE;
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_TREE), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SAVEMSG), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SAVEASNEW), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_NEWCATEGORY), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_DELETE), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_MSGSPLITTER), bShow);
+ bShow = g_SetAwayMsgPage.GetValue(IDS_SAWAYMSG_SHOWCONTACTTREE) ? SW_SHOW : SW_HIDE;
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_IGNOREREQ), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_SENDMSG), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSPLITTER), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_STATIC_IGNOREICON), bShow);
+ ShowWindow(GetDlgItem(hwndDlg, IDC_SAWAYMSG_STATIC_REPLYICON), bShow);
+ }
+ /*if (hWndStatusBar)
+ {
+ RECT rcStatus; GetWindowRect(hWndStatusBar, &rcStatus);
+ RECT rcClient; GetClientRect(hwndDlg, &rcClient);
+ SetWindowPos(hWndStatusBar, NULL, 0, rcClient.bottom-(rcStatus.bottom-rcStatus.top), rcClient.right, (rcStatus.bottom-rcStatus.top), SWP_NOZORDER);
+ }*/
+
+
+ } // go through
+ case UM_SAM_KILLTIMER:
+ case WM_LBUTTONDOWN:
+ case WM_MOUSEACTIVATE:
+ case WM_MOVING: // stops counting
+ {
+ if (Countdown != -1) // still counting
+ {
+ KillTimer(hwndDlg, SAM_TIMER_ID);
+ SetDlgItemText(hwndDlg, IDC_OK, TranslateT("OK"));
+ Countdown = -1;
+ }
+ } break;
+ case WM_SETCURSOR:
+ {
+ HWND hTreeView = GetDlgItem(hwndDlg, IDC_SAWAYMSG_CONTACTSTREE);
+ if ((HWND)wParam == hTreeView && LOWORD(lParam) == HTCLIENT)
+ {
+ _ASSERT(CList);
+ DWORD hitFlags;
+ POINT pt;
+ pt.x = (short)LOWORD(GetMessagePos());
+ pt.y = (short)HIWORD(GetMessagePos());
+ ScreenToClient(hTreeView, &pt);
+ CList->HitTest(&pt, &hitFlags);
+ HCURSOR hCursor = NULL;
+ if (hitFlags & (MCLCHT_ONITEM | MCLCHT_ONITEMEXTRA))
+ {
+ SetClassLong(hTreeView, GCLP_HCURSOR, NULL);
+ hCursor = LoadCursor(NULL, IDC_HAND); // set mouse cursor to a hand when hovering over items or their extra images
+ } else
+ {
+ SetClassLong(hTreeView, GCLP_HCURSOR, (LONG)LoadCursor(NULL, IDC_ARROW));
+ }
+ if (!hCursor)
+ {
+ hCursor = LoadCursor(NULL, IDC_ARROW);
+ }
+ SetCursor(hCursor);
+ return true;
+ } else
+ {
+ SetClassLong(hTreeView, GCLP_HCURSOR, (LONG)LoadCursor(NULL, IDC_ARROW));
+ }
+ } break;
+ case WM_CLOSE:
+ {
+ KillTimer(hwndDlg, SAM_TIMER_ID);
+ if (dat->IsModeless)
+ {
+ DestroyWindow(hwndDlg);
+ } else
+ {
+ EndDialog(hwndDlg, 2);
+ }
+ } break;
+ case WM_DESTROY:
+ {
+ if (dat)
+ {
+ delete dat;
+ }
+ if (CList)
+ {
+ delete CList;
+ CList = NULL;
+ }
+ delete MsgTree;
+ MsgTree = NULL;
+ g_SetAwayMsgPage.SetWnd(NULL);
+ DestroyWindow(hWndTooltips);
+ return false;
+ } break;
+ }
+ return false;
+}
diff --git a/plugins/NewAwaySysMod/src/TMyArray.h b/plugins/NewAwaySysMod/src/TMyArray.h
new file mode 100644
index 0000000000..68ca908fe1
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/TMyArray.h
@@ -0,0 +1,353 @@
+/*
+ TMyArray.h - TMyArray template
+ Copyright (c) 2005-2008 Chervov Dmitry
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#pragma once
+
+#include <stdlib.h>
+
+#define ARRAY_GROWBY 4
+// if there is more than ARRAY_FREETHRESHOLD free array elements, then we're reallocating the array
+#define ARRAY_FREETHRESHOLD 16
+
+template <class T, class ARG_T = const T&, int GrowBy = ARRAY_GROWBY, int FreeThreshold = ARRAY_FREETHRESHOLD>
+class TMyArray
+{
+public:
+ TMyArray();
+ TMyArray(const TMyArray<T, ARG_T, GrowBy, FreeThreshold> &A);
+ ~TMyArray();
+
+ int GetSize() const;
+ T* GetData() const;
+ int AddElem(ARG_T pElem);
+ int Add(const T *pItems, int nItems);
+ void InsertElem(ARG_T pElem, int nInsertAt);
+ void MoveElem(int nIndex, int nMoveTo);
+ T& SetAtGrow(int nIndex);
+ int Find(ARG_T pElem); // returns an index of the specified item, or -1 if the array doesn't contain the item
+ int BinarySearch(int (*CmpFunc)(ARG_T pItem, LPARAM lParam), LPARAM lParam, int *pIndex = NULL); // returns an index of the specified item, or -1 if the array doesn't contain the item;
+ // also it's possible to specify pIndex so that even if the array doesn't contain the item, the function will set *pIndex to a position where the item can be inserted;
+ // CmpFunc must return -1, 0 and 1 depending whether pItem is lesser, equal or greater than needed;
+ // BinarySearch() requires the array to be sorted in an ascending order
+ void RemoveElem(int nIndex, int nItems = 1);
+ void RemoveAll();
+ const T& operator[](int nIndex) const;
+ T& operator[](int nIndex);
+ TMyArray<T, ARG_T, GrowBy, FreeThreshold>& operator = (const TMyArray<T, ARG_T, GrowBy, FreeThreshold> &A);
+ TMyArray<T, ARG_T, GrowBy, FreeThreshold>& operator += (const TMyArray<T, ARG_T, GrowBy, FreeThreshold> &A);
+
+private:
+ int SetAllocNum(int nNewAllocNum);
+
+ T* pData;
+ int nElemNum;
+ int nAllocNum;
+};
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+TMyArray<T, ARG_T, GrowBy, FreeThreshold>::TMyArray()
+{
+ nElemNum = 0;
+ nAllocNum = 0;
+ pData = NULL;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+TMyArray<T, ARG_T, GrowBy, FreeThreshold>::TMyArray(const TMyArray<T, ARG_T, GrowBy, FreeThreshold> &A)//: TMyArray<T, ARG_T>()
+{
+ nElemNum = 0;
+ nAllocNum = 0;
+ pData = NULL;
+ *this = A;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+TMyArray<T, ARG_T, GrowBy, FreeThreshold>::~TMyArray()
+{
+ RemoveAll();
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline int TMyArray<T, ARG_T, GrowBy, FreeThreshold>::GetSize() const
+{
+ return nElemNum;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline T* TMyArray<T, ARG_T, GrowBy, FreeThreshold>::GetData() const
+{
+ return pData;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline int TMyArray<T, ARG_T, GrowBy, FreeThreshold>::SetAllocNum(int nNewAllocNum)
+{
+ _ASSERT(nNewAllocNum >= nElemNum);
+ T*pNewData = (nNewAllocNum) ? (T*)malloc(sizeof(T) * nNewAllocNum) : NULL;
+ if (pData)
+ {
+ if (pNewData)
+ {
+ memcpy(pNewData, pData, sizeof(T) * nElemNum);
+ }
+ free(pData);
+ }
+ pData = pNewData;
+ if (!pNewData)
+ {
+ nAllocNum = 0;
+ return -1; // not enough memory?
+ } else
+ {
+ nAllocNum = nNewAllocNum;
+ return 0; // everything's ok
+ }
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline int TMyArray<T, ARG_T, GrowBy, FreeThreshold>::AddElem(ARG_T pElem)
+{
+ if (nElemNum >= nAllocNum)
+ { // reallocate memory to fit new element
+ SetAllocNum(nAllocNum + GrowBy);
+ }
+ memset(pData + nElemNum, 0, sizeof(T));
+ pData[nElemNum] = pElem;
+ return nElemNum++;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline int TMyArray<T, ARG_T, GrowBy, FreeThreshold>::Add(const T *pItems, int nItems)
+{
+ if (nElemNum + nItems > nAllocNum)
+ { // reallocate memory to fit new items
+ SetAllocNum(nAllocNum + nElemNum + nItems - 1 - ((nElemNum + nItems - 1) % GrowBy) + GrowBy);
+ }
+ memset(pData + nElemNum, 0, sizeof(T) * nItems);
+ int I;
+ for (I = 0; I < nItems; I++)
+ {
+ pData[nElemNum++] = pItems[I];
+ }
+ return nElemNum - nItems; // returns an index of the first item added
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline void TMyArray<T, ARG_T, GrowBy, FreeThreshold>::InsertElem(ARG_T pElem, int nInsertAt)
+{
+ _ASSERT(nInsertAt >= 0 && nInsertAt <= nElemNum);
+ if (nElemNum >= nAllocNum)
+ { // reallocate memory to fit new items
+ SetAllocNum(nAllocNum + GrowBy);
+ }
+ memmove(pData + nInsertAt + 1, pData + nInsertAt, sizeof(T) * (nElemNum - nInsertAt));
+ memset(pData + nInsertAt, 0, sizeof(T));
+ pData[nInsertAt] = pElem;
+ nElemNum++;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline void TMyArray<T, ARG_T, GrowBy, FreeThreshold>::MoveElem(int nIndex, int nMoveTo)
+{
+ _ASSERT(nIndex >= 0 && nIndex < nElemNum && nMoveTo >= 0 && nMoveTo < nElemNum);
+ if (nIndex == nMoveTo)
+ {
+ return; // nothing to do
+ }
+ char Elem[sizeof(T)];
+ memmove(Elem, pData + nIndex, sizeof(T));
+ if (nIndex < nMoveTo)
+ {
+ memmove(pData + nIndex, pData + nIndex + 1, sizeof(T) * (nMoveTo - nIndex));
+ } else
+ {
+ memmove(pData + nMoveTo + 1, pData + nMoveTo, sizeof(T) * (nIndex - nMoveTo));
+ }
+ memmove(pData + nMoveTo, Elem, sizeof(T));
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline T& TMyArray<T, ARG_T, GrowBy, FreeThreshold>::SetAtGrow(int nIndex)
+{
+ _ASSERT(nIndex >= 0);
+ if (nIndex < nElemNum)
+ {
+ return pData[nIndex];
+ }
+ if (nIndex >= nAllocNum)
+ {
+ if (SetAllocNum(nIndex - (nIndex % GrowBy) + GrowBy))
+ { // if there was an error
+ nElemNum = 0;
+ return *pData;
+ }
+ }
+ memset(pData + nElemNum, 0, sizeof(T) * (nIndex - nElemNum + 1));
+ nElemNum = nIndex + 1;
+ return pData[nIndex];
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline int TMyArray<T, ARG_T, GrowBy, FreeThreshold>::Find(ARG_T pElem)
+{
+ int I;
+ for (I = 0; I < nElemNum; I++)
+ {
+ if (pData[I] == pElem)
+ {
+ return I;
+ }
+ }
+ return -1;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline int TMyArray<T, ARG_T, GrowBy, FreeThreshold>::BinarySearch(int (*CmpFunc)(ARG_T pItem, LPARAM lParam), LPARAM lParam, int *pIndex)
+{
+ int L, R;
+ int CmpResult;
+ if (!nElemNum)
+ {
+ if (pIndex)
+ {
+ *pIndex = -1;
+ }
+ return -1;
+ }
+ for (L = 0, R = nElemNum; R - L > 1;)
+ {
+ int C = (L + R) >> 1; // rounds always to a lesser index if L + R is odd
+ CmpResult = CmpFunc(pData[C], lParam); // usually, CmpFunc = pData[C] - lParam; i.e. CmpFunc > 0 when pData[C] is greater than necessary, < 0 when pData[C] is less, and = 0 when pData[C] is the item we search for
+ if (CmpResult < 0)
+ {
+ L = C;
+ } else if (CmpResult > 0)
+ {
+ R = C;
+ } else
+ { // CmpResult == 0
+ if (pIndex)
+ {
+ *pIndex = C;
+ }
+ return C;
+ }
+ }
+ if (pIndex)
+ {
+ *pIndex = L;
+ }
+ CmpResult = CmpFunc(pData[L], lParam);
+ if (!CmpResult)
+ {
+ return L;
+ }
+ if (CmpResult > 0)
+ { // we don't need to check pData[R], as pData[R] > pData[L] > lParam, i.e. there are no suitable item in the array
+ return -1;
+ }
+// CmpResult < 0, we have to check pData[R] too
+ if (pIndex)
+ {
+ *pIndex = R;
+ }
+ if (R >= nElemNum)
+ {
+ return -1;
+ }
+ return CmpFunc(pData[R], lParam) ? -1 : R;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline void TMyArray<T, ARG_T, GrowBy, FreeThreshold>::RemoveElem(int nIndex, int nItems)
+{
+ _ASSERT(nIndex >= 0 && nIndex + nItems <= nElemNum);
+ if (!nItems)
+ {
+ return;
+ }
+// delete pData[nIndex];
+// ~pData[nIndex];
+ int I;
+ for (I = nIndex; I < nIndex + nItems; I++)
+ {
+ (pData + I)->~T();
+ }
+ memmove(pData + nIndex, pData + nIndex + nItems, sizeof(T) * (nElemNum - nIndex - nItems));
+ nElemNum -= nItems;
+ if (nAllocNum - nElemNum >= FreeThreshold)
+ {
+ SetAllocNum(nAllocNum - FreeThreshold);
+ }/* else
+ {
+ memset(pData + nElemNum, 0, sizeof(T));
+ }*/
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline void TMyArray<T, ARG_T, GrowBy, FreeThreshold>::RemoveAll()
+{
+ int I;
+ for (I = 0; I < nElemNum; I++)
+ {
+ //delete pData[I];
+ (pData + I)->~T();
+ }
+ nElemNum = 0;
+ SetAllocNum(0);
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline const T& TMyArray<T, ARG_T, GrowBy, FreeThreshold>::operator[](int nIndex) const
+{
+ _ASSERT(nIndex >= 0 && nIndex < nElemNum);
+ return pData[nIndex];
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline T& TMyArray<T, ARG_T, GrowBy, FreeThreshold>::operator[](int nIndex)
+{
+ _ASSERT(nIndex >= 0 && nIndex < nElemNum);
+ return pData[nIndex];
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline TMyArray<T, ARG_T, GrowBy, FreeThreshold>& TMyArray<T, ARG_T, GrowBy, FreeThreshold>::operator = (const TMyArray<T, ARG_T, GrowBy, FreeThreshold> &A)
+{
+ RemoveAll();
+ int I;
+ for (I = 0; I < A.GetSize(); I++)
+ {
+ AddElem(A[I]);
+ }
+ return *this;
+}
+
+template <class T, class ARG_T, int GrowBy, int FreeThreshold>
+__forceinline TMyArray<T, ARG_T, GrowBy, FreeThreshold>& TMyArray<T, ARG_T, GrowBy, FreeThreshold>::operator += (const TMyArray<T, ARG_T, GrowBy, FreeThreshold> &A)
+{
+ int I;
+ for (I = 0; I < A.GetSize(); I++)
+ {
+ AddElem(A[I]);
+ }
+ return *this;
+}
+
+typedef TMyArray<char, const char&, 1024, 4096> CHARARRAY;
diff --git a/plugins/NewAwaySysMod/src/resource.h b/plugins/NewAwaySysMod/src/resource.h
new file mode 100644
index 0000000000..1c6190554b
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/resource.h
@@ -0,0 +1,261 @@
+//{{NO_DEPENDENCIES}}
+// Microsoft Developer Studio generated include file.
+// Used by NewAwaySys.rc
+//
+#define IDD_OPTDIALOGMAIN 101
+#define IDD_MESSAGES 101
+#define IDD_MODERNOPT_MESSAGES 102
+#define IDI_MSGICON 109
+#define IDI_SOE_ENABLED 110
+#define IDD_READAWAYMSG 111
+#define IDI_SOE_DISABLED 111
+#define IDB_SOE_ENABLED 112
+#define IDB_SOE_DISABLED 114
+#define IDD_NOTIFYOPTDLG 115
+#define IDD_NOTIFYVARSDLG 116
+#define IDD_AUTOREPLY 121
+#define IDR_SAM_OPTIONS 122
+#define IDI_DOT 123
+#define IDI_IGNORE 124
+#define IDI_SAVEMSG 127
+#define IDI_SAVE 127
+#define IDI_SAVEASNEW 128
+#define IDI_DELETE 130
+#define IDI_NEWCATEGORY 133
+#define IDR_MSGTREE_CATEGORYMENU 134
+#define IDR_MSGTREE_MESSAGEMENU 135
+#define IDI_NEWMESSAGE 136
+#define IDI_SETTINGS 137
+#define IDI_STATUS_OTHER 138
+#define IDD_CONTACTSOPTDLG 139
+#define IDI_DISABLENOTIFY 140
+#define IDI_ENABLENOTIFY 141
+#define IDI_INDEFINITE 142
+#define IDD_SETAWAYMSG_BAK 144
+#define IDD_SETAWAYMSG 216
+#define IDD_MOREOPTDIALOG 217
+#define IDD_POPUPOPTDLG 218
+#define IDC_STATUSBAR 999
+#define IDC_SAWAYMSG_MSGNAME 1001
+#define IDC_MESSAGEDLG_DEL 1002
+#define IDC_OPTDLG_MSGDATA 1003
+#define IDC_MESSAGEDLG_MSGDATA 1003
+#define IDC_OPTDLG_MSGNAME 1004
+#define IDC_OPTDLG_MAKEDEFAULT 1005
+#define IDC_OPTDLG_USELASTMSG 1007
+#define IDC_MOREOPTDLG_USELASTMSG 1007
+#define IDC_OPTDLG_USEDEFAULT 1008
+#define IDC_SAWAYMSG_MSGDATA 1008
+#define IDC_MOREOPTDLG_USEDEFMSG 1008
+#define IDC_OPTDLG_SAVE 1010
+#define IDC_LNK_AUTOAWAY 1010
+#define IDC_OPTDLG_RESET 1012
+#define IDC_OPTDLG_VARSLIST 1013
+#define IDC_MESSAGEDLG_VARS 1013
+#define IDC_SAWAYMSG_STATIC_CHARSTXT 1014
+#define IDC_STATIC_BYDEFAULT 1017
+#define IDC_STATIC_MSG 1021
+#define IDC_MOREOPTDLG_NEVERSTATDLG 1028
+#define IDC_MOREOPTDLG_WAITFORMSG 1029
+#define IDC_STATIC_MOREOPTDLG_WAIT 1030
+#define IDC_STATIC_MOREOPTDLG_SECONDSREMAIN 1031
+#define IDC_MOREOPTDLG_LMESTAT 1032
+#define IDC_MOREOPTDLG_PERSTATUSMRM 1032
+#define IDC_MOREOPTDLG_RMDLGPOS 1033
+#define IDC_MOREOPTDLG_SAVEDLGPOS 1033
+#define IDC_MOREOPTDLG_SETLASTCONTCAT 1034
+#define IDC_MOREOPTDLG_SAVECONTACTMSGS 1034
+#define IDC_MOREOPTDLG_SAVEPERSONALMSGS 1034
+#define IDC_MOREOPTDLG_USEMENUITEM 1035
+#define IDC_MOREOPTDLG_PERSTATUSPERSONAL 1036
+#define IDC_MOREOPTDLG_RESETPROTOMSGS 1037
+#define IDC_MOREOPTDLG_PERSTATUSPROTOMSGS 1038
+#define IDC_MOREOPTDLG_PERSTATUSPROTOSETTINGS 1039
+#define IDC_SAWAYMSG_CONTACTSTREE 1040
+#define IDC_MOREOPTDLG_PERSTATUSPERSONALSETTINGS 1040
+#define IDC_SAWAYMSG_STATIC_VARS 1041
+#define IDC_SAWAYMSG_SHOWCONTACTS 1042
+#define IDC_SAWAYMSG_STATIC_RESETCHNG 1043
+#define IDC_SAWAYMSG_OPTIONS 1043
+#define IDC_MOREOPTDLG_SENDNONICQ 1049
+#define IDC_REPLYDLG_DONTSENDTOICQ 1049
+#define IDC_MOREOPTDLG_EVNTMSG 1050
+#define IDC_REPLYDLG_EVENTMSG 1050
+#define IDC_MOREOPTDLG_EVNTURL 1051
+#define IDC_REPLYDLG_EVENTURL 1051
+#define IDC_MOREOPTDLG_EVNTFILE 1052
+#define IDC_REPLYDLG_EVENTFILE 1052
+#define IDC_MOREOPTDLG_SENDONCE 1053
+#define IDC_SAWAYMSG_SENDMSG 1054
+#define IDC_MOREOPTDLG_SENDALL 1054
+#define IDC_SAWAYMSG_IGNOREREQ 1055
+#define IDC_MOREOPTDLG_STATIC_SEND 1057
+#define IDC_REPLYDLG_STATIC_SEND 1057
+#define IDC_MOREOPTDLG_STATIC_ONEVENT 1058
+#define IDC_REPLYDLG_STATIC_ONEVENT 1058
+#define IDC_VARSLIST_STATIC_TIME 1061
+#define IDC_MOREOPTDLG_NOPOPLIST 1062
+#define IDC_MOREOPTDLG_NOPOP 1063
+#define IDC_MOREOPTDLG_NOSSMLIST 1064
+#define IDC_MOREOPTDLG_NOSSM 1065
+#define IDC_OPTDLG_PREFIX 1065
+#define IDC_REPLYDLG_PREFIX 1065
+#define IDC_MOREOPTDLG_LOGMSG 1066
+#define IDC_REPLYDLG_LOGREPLY 1066
+#define IDC_OPTDLG_USEDEFNICK 1068
+#define IDC_MOREOPTDLG_MYNICKPERPROTO 1068
+#define IDC_MOREOPTDLG_SHOWRMSG 1069
+#define IDC_MOREOPTDLG_SENDMSG 1070
+#define IDC_REPLYDLG_ENABLEREPLY 1070
+#define IDC_POPUPOPTDLG_POPUPFORMAT 1071
+#define IDC_POPUPOPTDLG_USEPOPUPS 1072
+#define IDC_POPUPOPTDLG_BGCOLOUR 1073
+#define IDC_POPUPOPTDLG_TEXTCOLOUR 1074
+#define IDC_POPUPOPTDLG_LCLICK_ACTION 1075
+#define IDC_POPUPOPTDLG_RCLICK_ACTION 1077
+#define IDC_POPUPOPTDLG_POPUPPREVIEW 1078
+#define IDC_POPUPOPTDLG_AWAYNOTIFY 1084
+#define IDC_POPUPOPTDLG_NANOTIFY 1085
+#define IDC_POPUPOPTDLG_OCCNOTIFY 1086
+#define IDC_POPUPOPTDLG_DNDNOTIFY 1087
+#define IDC_POPUPOPTDLG_FFCNOTIFY 1088
+#define IDC_POPUPOPTDLG_DEFBGCOLOUR 1089
+#define IDC_POPUPOPTDLG_DEFTEXTCOLOUR 1090
+#define IDC_NOTIFYVARS_STATIC_TIME 1091
+#define IDC_VARS_STATIC_TIME 1091
+#define IDC_POPUPOPTDLG_OTHERNOTIFY 1091
+#define IDC_POPUPOPTDLG_STATIC_LCLICK 1092
+#define IDC_POPUPOPTDLG_STATIC_RCLICK 1093
+#define IDC_POPUPOPTDLG_STATIC_BGCOLOUR 1094
+#define IDC_POPUPOPTDLG_STATIC_TEXTCOLOUR 1095
+#define IDC_POPUPOPTDLG_ONLNOTIFY 1096
+#define IDC_POPUPOPTDLG_POPUPDELAY 1099
+#define IDC_POPUPOPTDLG_STATIC_SEC 1100
+#define IDC_POPUPOPTDLG_STATIC_DEFAULT 1101
+#define IDC_POPUPOPTDLG_STATIC_INFINITE 1102
+#define IDC_MESSAGEDLG_DEF_AWAY 1109
+#define IDC_SAWAYMSG_SAVEMSG 1109
+#define IDC_REPLYDLG_ONLYIDLEREPLY 1109
+#define IDC_POPUPOPTDLG_VARS 1110
+#define IDC_REPLYDLG_DISABLE_AWAY 1110
+#define IDC_SAWAYMSG_SAVEASNEW 1110
+#define IDC_MOREOPTDLG_PANEL 1111
+#define IDC_SAWAYMSG_NEWCATEGORY 1111
+#define IDC_SAWAYMSG_DELETE 1112
+#define IDC_MESSAGEDLG_DEF_ONL 1113
+#define IDC_MESSAGEDLG_DEF_NA 1114
+#define IDC_MESSAGEDLG_DEF_OCC 1115
+#define IDC_MESSAGEDLG_DEF_DND 1116
+#define IDC_MESSAGEDLG_DEF_FFC 1117
+#define IDC_MESSAGEDLG_DEF_OTP 1118
+#define IDC_MESSAGEDLG_DEF_OTL 1119
+#define IDC_MESSAGEDLG_MSGTITLE 1120
+#define IDC_MOREOPTDLG_DONTPOPDLG_INV 1120
+#define IDC_MESSAGEDLG_MSGTREE 1121
+#define IDC_REPLYDLG_DISABLE_ONL 1121
+#define IDC_MESSAGEDLG_NEWCAT 1122
+#define IDC_REPLYDLG_DISABLE_NA 1122
+#define IDC_MESSAGEDLG_NEWMSG 1123
+#define IDC_REPLYDLG_DISABLE_OCC 1123
+#define IDC_REPLYDLG_DISABLE_DND 1124
+#define IDC_REPLYDLG_DISABLE_FFC 1125
+#define IDC_REPLYDLG_DISABLE_OTP 1126
+#define IDC_REPLYDLG_DISABLE_OTL 1127
+#define IDC_REPLYDLG_DISABLE_INV 1128
+#define IDC_MOREOPTDLG_RECENTMSGSCOUNT 1129
+#define IDC_MOREOPTDLG_UPDATEMSGSPERIOD 1130
+#define IDC_MESSAGEDLG_DEF_INV 1131
+#define IDC_REPLYDLG_SENDCOUNT 1132
+#define IDC_MOREOPTDLG_DONTPOPDLG_ONL 1133
+#define IDC_MOREOPTDLG_DONTPOPDLG_AWAY 1134
+#define IDC_MOREOPTDLG_DONTPOPDLG_NA 1135
+#define IDC_MOREOPTDLG_DONTPOPDLG_OCC 1136
+#define IDC_MOREOPTDLG_DONTPOPDLG_DND 1137
+#define IDC_MOREOPTDLG_DONTPOPDLG_FFC 1138
+#define IDC_MOREOPTDLG_DONTPOPDLG_OTP 1139
+#define IDC_MOREOPTDLG_DONTPOPDLG_OTL 1140
+#define IDC_REPLYDLG_STATIC_TIMES 1141
+#define IDC_MESSAGEDLG_TEST 1142
+#define IDC_MOREOPTDLG_EDITMESSAGES 1142
+#define IDC_SAWAYMSG_VARS 1142
+#define IDC_SAWAYMSG_TREE 1149
+#define IDC_SAWAYMSG_EDITMSGS 1150
+#define IDC_SAWAYMSG_RESET 1151
+#define IDC_OK 1152
+#define IDC_MSGSPLITTER 1154
+#define IDC_SAWAYMSG_MSGSPLITTER 1154
+#define IDC_CONTACTSPLITTER 1155
+#define IDC_SAWAYMSG_CONTACTSPLITTER 1155
+#define IDC_SAWAYMSG_STATIC_IGNOREICON 1157
+#define IDC_SAWAYMSG_STATIC_REPLYICON 1158
+#define IDC_MOREOPTDLG_WAITFORMSG_SPIN 1161
+#define IDC_MOREOPTDLG_RECENTMSGSCOUNT_SPIN 1162
+#define IDC_REPLYDLG_SENDCOUNT_SPIN 1163
+#define IDC_MOREOPTDLG_UPDATEMSGSPERIOD_SPIN 1163
+#define IDC_POPUPOPTDLG_POPUPDELAY_SPIN 1164
+#define IDC_REPLYDLG_VARS 1165
+#define IDC_MOREOPTDLG_UPDATEMSGS 1166
+#define IDC_CONTACTSDLG_LIST 1168
+#define IDC_REPLYDLG_DONTREPLYINVISIBLE 1169
+#define IDC_REPLYDLG_ONLYIDLEREPLY_COMBO 1170
+#define IDC_REPLYDLG_ONLYCLOSEDDLGREPLY 1171
+#define IDC_REPLYDLG_RESETCOUNTERWHENSAMEICON 1172
+#define IDC_REPLYDLG_STATIC_DISABLEWHENSTATUS 1174
+#define IDC_REPLYDLG_STATIC_FORMAT 1175
+#define IDC_POPUPOPTDLG_GROUP_NOTIFICATION 1176
+#define IDC_POPUPOPTDLG_LOGONLYWITHPOPUP 1177
+#define IDC_POPUPOPTDLG_STATIC_EXTRATEXT 1178
+#define IDC_REPLYDLG_STATIC_EXTRATEXT 1179
+#define IDC_TXT_TITLE1 1181
+#define IDC_TXT_TITLE2 1182
+#define IDC_TXT_TITLE3 1183
+#define IDC_SAWAYMSG_STATIC_COUNTER 1202
+#define IDC_READAWAYMSG_RETRIEVE 3453
+#define IDS_MESSAGEDLG_DEF_ONL 20000
+#define IDS_MESSAGEDLG_DEF_AWAY 20001
+#define IDS_MESSAGEDLG_DEF_NA 20002
+#define IDS_MESSAGEDLG_DEF_OCC 20003
+#define IDS_MESSAGEDLG_DEF_DND 20004
+#define IDS_MESSAGEDLG_DEF_FFC 20005
+#define IDS_MESSAGEDLG_DEF_INV 20006
+#define IDS_MESSAGEDLG_DEF_OTP 20007
+#define IDS_MESSAGEDLG_DEF_OTL 20008
+#define IDV_MSGTREE 20009
+#define IDS_SAWAYMSG_SHOWMSGTREE 20100
+#define IDS_SAWAYMSG_SHOWCONTACTTREE 20101
+#define IDS_SAWAYMSG_AUTOSAVEDLGSETTINGS 20102
+#define IDS_SAWAYMSG_DISABLEVARIABLES 20103
+#define IDM_SAM_OPTIONS_SHOWMSGTREE 40007
+#define IDM_SAM_OPTIONS_SHOWCONTACTTREE 40008
+#define IDM_SAM_OPTIONS_EDITMSGS 40009
+#define IDM_SAM_OPTIONS_AUTOSAVEDLGSETTINGS 40014
+#define IDM_SAM_OPTIONS_SAVEDLGSETTINGSNOW 40015
+#define IDM_MSGTREEMENU_RENAME 40017
+#define IDM_MSGTREEMENU_NEWMESSAGE 40022
+#define IDM_MSGTREEMENU_NEWCATEGORY 40024
+#define IDM_MSGTREEMENU_DELETE 40026
+#define ID_MESSAGETREECATEGORYMENU_SETASDEFAULTFOR 40027
+#define IDR_MSGTREEMENU_DEF_ONL 40037
+#define IDR_MSGTREEMENU_DEF_AWAY 40038
+#define IDR_MSGTREEMENU_DEF_NA 40039
+#define IDR_MSGTREEMENU_DEF_OCC 40040
+#define IDR_MSGTREEMENU_DEF_DND 40041
+#define IDR_MSGTREEMENU_DEF_FFC 40042
+#define IDR_MSGTREEMENU_DEF_INV 40043
+#define IDR_MSGTREEMENU_DEF_OTP 40044
+#define IDR_MSGTREEMENU_DEF_OTL 40045
+#define IDM_SAM_OPTIONS_DISABLEVARIABLES 40047
+#define IDC_READAWAYMSG_MSG 54765
+
+// Next default values for new objects
+//
+#ifdef APSTUDIO_INVOKED
+#ifndef APSTUDIO_READONLY_SYMBOLS
+#define _APS_NO_MFC 1
+#define _APS_3D_CONTROLS 1
+#define _APS_NEXT_RESOURCE_VALUE 145
+#define _APS_NEXT_COMMAND_VALUE 40048
+#define _APS_NEXT_CONTROL_VALUE 1185
+#define _APS_NEXT_SYMED_VALUE 101
+#endif
+#endif
diff --git a/plugins/NewAwaySysMod/src/version.h b/plugins/NewAwaySysMod/src/version.h
new file mode 100644
index 0000000000..3634e99662
--- /dev/null
+++ b/plugins/NewAwaySysMod/src/version.h
@@ -0,0 +1,5 @@
+#define STRSPECIALBUILD "2636"
+#define FILEVER 0, 3, 8, 4
+#define STRFILEVER "0, 3, 8, 4"
+#define PRODUCTVER FILEVER
+#define STRPRODUCTVER STRFILEVER