2011-01-28 16:07:46 -08:00

522 lines
16 KiB
C++

/////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2002 Ultr@VNC Team Members. All Rights Reserved.
//
// 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.
//
// If the source code for the program is not available from the place from
// which you received this file, check
// http://ultravnc.sourceforge.net/
//
////////////////////////////////////////////////////////////////////////////
#include "stdhdrs.h"
#include "vncviewer.h"
#include "TextChat.h"
#include "Exception.h"
#include "commctrl.h"
#include "richedit.h"
#define TEXTMAXSIZE 16384
#define MAXNAMESIZE 128 // MAX_COMPUTERNAME_LENGTH+1 (32)
#define CHAT_OPEN -1 // Todo; put these codes in rfbproto.h
#define CHAT_CLOSE -2
#define CHAT_FINISHED -3
extern char sz_E1[64];
extern char sz_E2[64];
//
//
//
TextChat::TextChat(VNCviewerApp *pApp, ClientConnection *pCC)
{
m_pApp = pApp;
m_pCC = pCC;
m_fTextChatRunning = false;
m_fVisible = true;
m_fPersistentTexts = false;
m_szLocalText = new char [TEXTMAXSIZE];
memset(m_szLocalText, 0, TEXTMAXSIZE);
m_szLastLocalText = new char [TEXTMAXSIZE];
memset(m_szLastLocalText, 0, TEXTMAXSIZE);
m_szRemoteText = new char [TEXTMAXSIZE];
memset(m_szRemoteText, 0, TEXTMAXSIZE);
m_szRemoteName = new char[MAXNAMESIZE];
memset(m_szRemoteName,0,MAXNAMESIZE);
m_szLocalName = new char[MAXNAMESIZE];
memset(m_szLocalName,0,MAXNAMESIZE);
m_szTextBoxBuffer = new char[TEXTMAXSIZE]; // History Text (containing all chat messages from everybody)
memset(m_szTextBoxBuffer,0,TEXTMAXSIZE);
unsigned long pcSize=MAXNAMESIZE;
if (GetComputerName(m_szLocalName,&pcSize))
{
if( pcSize >= MAXNAMESIZE)
{
m_szLocalName[MAXNAMESIZE-4]='.';
m_szLocalName[MAXNAMESIZE-3]='.';
m_szLocalName[MAXNAMESIZE-2]='.';
m_szLocalName[MAXNAMESIZE-1]=0x00;
}
}
// Load the Rich Edit control DLL
m_hRichEdit = LoadLibrary( "RICHED32.DLL" );
if (!m_hRichEdit)
{
MessageBox( NULL, sz_E1,
sz_E2, MB_OK | MB_ICONEXCLAMATION );
// Todo: do normal edit instead (no colors)
}
}
//
//
//
TextChat::~TextChat()
{
if (m_szLocalText != NULL) delete [] m_szLocalText;
if (m_szLastLocalText != NULL) delete [] m_szLastLocalText;
if (m_szRemoteText != NULL) delete [] m_szRemoteText;
if (m_szRemoteName != NULL) delete [] m_szRemoteName;
if (m_szLocalName != NULL) delete [] m_szLocalName;
if (m_szTextBoxBuffer != NULL) delete [] m_szTextBoxBuffer;
m_fTextChatRunning = false;
SendMessage(m_hDlg, WM_COMMAND, IDOK, 0L);
if (m_hRichEdit != NULL) FreeLibrary(m_hRichEdit);
}
//
// Set text format to a selection in the Chat area
//
void TextChat::SetTextFormat(bool bBold /*= false */, bool bItalic /*= false*/
, long nSize /*= 0x75*/, const char* szFaceName /*= "MS Sans Serif"*/, DWORD dwColor /*= BLACK*/)
{
if ( GetDlgItem( m_hDlg, IDC_CHATAREA_EDIT ) ) // Sanity Check
{
CHARFORMAT cf;
memset( &cf, 0, sizeof(CHARFORMAT) ); // Initialize structure
cf.cbSize = sizeof(CHARFORMAT);
cf.dwMask = CFM_COLOR | CFM_FACE | CFM_SIZE;
if (bBold)
{
cf.dwMask |= CFM_BOLD;
cf.dwEffects |= CFE_BOLD;
}
if (bItalic)
{
cf.dwMask |= CFM_ITALIC;
cf.dwEffects |= CFE_ITALIC;
}
cf.crTextColor = dwColor; // set color in AABBGGRR mode (alpha-RGB)
cf.yHeight = nSize; // set size in points
strcpy( cf.szFaceName, szFaceName);
SendDlgItemMessage( m_hDlg, IDC_CHATAREA_EDIT, EM_SETCHARFORMAT, SCF_SELECTION, (LPARAM)&cf );
}
}
//
//
//
void TextChat::ProcessTextChatMsg()
{
rfbTextChatMsg tcm;
m_pCC->ReadExact(((char *) &tcm) + m_pCC->m_nTO, sz_rfbTextChatMsg - m_pCC->m_nTO);
int len = Swap32IfLE(tcm.length);
if (len == CHAT_OPEN)
{
// Open TextChat Dialog
PostMessage(m_pCC->m_hwndMain, WM_SYSCOMMAND, ID_TEXTCHAT, (LPARAM)0L);
return;
}
else if (len == CHAT_CLOSE)
{
// Close TextChat Dialog
if (!m_fTextChatRunning) return;
PostMessage(m_hDlg, WM_COMMAND, IDOK, 0);
return;
}
else if (len == CHAT_FINISHED)
{
// On Server notif, request FullScreen update
if (!m_fTextChatRunning) return;
m_fTextChatRunning = false;
m_pCC->SendAppropriateFramebufferUpdateRequest();
return;
}
else
{
// Read the incoming text
if (len > TEXTMAXSIZE) return; // Todo: Improve this...
if (len == 0)
{
SetDlgItemText(m_hDlg, IDC_REMOTETEXT_EDIT, "");
memset(m_szRemoteText, 0, TEXTMAXSIZE);
}
else
{
memset(m_szRemoteText, 0, TEXTMAXSIZE);
m_pCC->ReadExact(m_szRemoteText, len);
PrintMessage(m_szRemoteText, m_szRemoteName, RED);
ShowChatWindow(true); // Show the chat window on new message reception
}
}
}
//
//
//
void TextChat::SendTextChatRequest(int nMsg)
{
rfbTextChatMsg tcm;
tcm.type = rfbTextChat;
tcm.length = Swap32IfLE(nMsg);
m_pCC->WriteExact((char *)&tcm, sz_rfbTextChatMsg, rfbTextChat);
return;
}
//
// Output messages in the chat area
//
//
void TextChat::PrintMessage(const char* szMessage,const char* szSender,DWORD dwColor /*= BLACK*/)
{
// char szTextBoxBuffer[TEXTMAXSIZE];
// memset(szTextBoxBuffer,0,TEXTMAXSIZE);
CHARRANGE cr;
GETTEXTLENGTHEX lpG;
lpG.flags = GTL_NUMBYTES;
lpG.codepage = CP_ACP; // ANSI
int nLen = SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT, EM_GETTEXTLENGTHEX, (WPARAM)&lpG, 0L);
if (nLen + 32 > TEXTMAXSIZE )
{
memset(m_szTextBoxBuffer, 0, TEXTMAXSIZE);
strcpy(m_szTextBoxBuffer, "------------------------------------------------------------------------------------------------------------------------\n");
SetDlgItemText(m_hDlg, IDC_CHATAREA_EDIT, m_szTextBoxBuffer);
}
// Todo: test if chat text + sender + message > TEXTMAXSIZE -> Modulo display
if (szSender != NULL) //print the sender's name
{
/*
// Time
char tbuffer[9];
_tzset();
_strtime(tbuffer);
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT,WM_GETTEXT, TEXTMAXSIZE-1,(LONG)m_szTextBoxBuffer);
cr.cpMax = strlen(m_szTextBoxBuffer); // Select the last caracter to make the text insertion
cr.cpMin = cr.cpMax;
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT,EM_EXSETSEL,0,(LONG) &cr);
SetTextFormat(false, false, 0x75, "MS Sans Serif", GREY);
_snprintf(m_szTextBoxBuffer, TEXTMAXSIZE-1, "[%s]", tbuffer);
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT,EM_REPLACESEL,FALSE,(LONG)m_szTextBoxBuffer); // Replace the selection with the message
*/
// Sender
// SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT,WM_GETTEXT, TEXTMAXSIZE-1,(LONG)m_szTextBoxBuffer);
cr.cpMax = nLen; //strlen(m_szTextBoxBuffer); // Select the last caracter to make the text insertion
cr.cpMin = cr.cpMax;
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT, EM_EXSETSEL, 0, (LONG)&cr);
/***/
SetTextFormat(false, false, 0x75, "MS Sans Serif", dwColor);
_snprintf(m_szTextBoxBuffer, MAXNAMESIZE-1 + 4, "<%s>: ", szSender);
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT,EM_REPLACESEL,FALSE,(LONG)m_szTextBoxBuffer); // Replace the selection with the message
}
nLen = SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT, EM_GETTEXTLENGTHEX, (WPARAM)&lpG, 0L);
if (szMessage != NULL) //print the sender's message
{
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT,WM_GETTEXT, TEXTMAXSIZE-1,(LONG)m_szTextBoxBuffer);
cr.cpMax = nLen; //strlen(m_szTextBoxBuffer); // Select the last caracter to make the text insertion
cr.cpMin = cr.cpMax;
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT,EM_EXSETSEL,0,(LONG) &cr);
/***/
SetTextFormat(false, false, 0x75, "MS Sans Serif", dwColor != GREY ? BLACK : GREY);
_snprintf(m_szTextBoxBuffer, TEXTMAXSIZE-1, "%s", szMessage);
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT, EM_REPLACESEL, FALSE, (LONG)m_szTextBoxBuffer);
}
// Scroll down the chat window
// The following seems necessary under W9x & Me if we want the Edit to scroll to bottom...
SCROLLINFO si;
ZeroMemory(&si, sizeof(SCROLLINFO));
si.cbSize = sizeof(SCROLLINFO);
si.fMask = SIF_RANGE|SIF_PAGE;
GetScrollInfo(GetDlgItem(m_hDlg, IDC_CHATAREA_EDIT), SB_VERT, &si);
si.nPos = si.nMax - max(si.nPage - 1, 0);
SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT, WM_VSCROLL, MAKELONG(SB_THUMBPOSITION, si.nPos), 0L); // Scroll down the ch
// This line does the bottom scrolling correctly under NT4,W2K, XP...
// SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT, WM_VSCROLL, SB_BOTTOM, 0L);
}
//
// Send local text content
//
void TextChat::SendLocalText(void)
{
// We keep it because we could use it
// for future retype functionality. (F3)
memcpy(m_szLastLocalText, m_szLocalText, strlen(m_szLocalText));
PrintMessage(m_szLocalText, m_szLocalName, BLUE);
rfbTextChatMsg tcm;
tcm.type = rfbTextChat;
tcm.length = Swap32IfLE(strlen(m_szLocalText));
m_pCC->WriteExact((char *)&tcm, sz_rfbTextChatMsg, rfbTextChat);
m_pCC->WriteExact((char *)m_szLocalText, strlen(m_szLocalText));
//and we clear the input box
SetDlgItemText(m_hDlg, IDC_INPUTAREA_EDIT, "");
return;
}
//
//
//
int TextChat::DoDialog()
{
#ifndef _ULTRAVNCAX_
return DialogBoxParam(pApp->m_instance, DIALOG_MAKEINTRESOURCE(IDD_TEXTCHAT_DLG),
NULL, (DLGPROC) TextChatDlgProc, (LONG) this);
#else
HWND h = CreateDialogParam(pApp->m_instance, DIALOG_MAKEINTRESOURCE(IDD_TEXTCHAT_DLG),
NULL, (DLGPROC) TextChatDlgProc, (LONG) this);
ShowWindow( h, SW_SHOW );
return 0;
#endif
}
//
//
//
void TextChat::KillDialog()
{
// DestroyWindow(m_hDlg);
}
//
//
//
void TextChat::ShowChatWindow(bool fVisible)
{
ShowWindow(m_hDlg, fVisible ? SW_RESTORE : SW_MINIMIZE);
SetForegroundWindow(m_hDlg);
// Put the Chat Windows always on Top if fullscreen
if (fVisible && m_pCC->InFullScreenMode())
{
RECT Rect;
GetWindowRect(m_hDlg, &Rect);
SetWindowPos(m_hDlg,
HWND_TOPMOST,
Rect.left,
Rect.top,
Rect.right - Rect.left,
Rect.bottom - Rect.top,
SWP_SHOWWINDOW);
}
m_fVisible = fVisible; // This enables screen updates to be processed in ClientConnection
// Refresh screen view if Chat window has been hidden
if (!fVisible)
m_pCC->SendAppropriateFramebufferUpdateRequest();
}
//
//
//
BOOL CALLBACK TextChat::TextChatDlgProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam )
{
TextChat* _this = (TextChat *) GetWindowLong(hWnd, GWL_USERDATA);
switch (uMsg)
{
case WM_INITDIALOG:
{
SetWindowLong(hWnd, GWL_USERDATA, lParam);
TextChat *_this = (TextChat *) lParam;
if (_this->m_szLocalText == NULL || _this->m_szRemoteText == NULL)
EndDialog(hWnd, FALSE);
// Window always on top if Fullscreen On
if (_this->m_pCC->InFullScreenMode())
{
RECT Rect;
GetWindowRect(hWnd, &Rect);
SetWindowPos(hWnd,
HWND_TOPMOST,
Rect.left,
Rect.top,
Rect.right - Rect.left,
Rect.bottom - Rect.top,
SWP_SHOWWINDOW);
}
// CentreWindow(hWnd);
_this->m_hWnd = hWnd;
_this->m_hDlg = hWnd;
if (_snprintf(_this->m_szRemoteName,MAXNAMESIZE-1,"%s", _this->m_pCC->m_desktopName) < 0 )
{
_this->m_szRemoteName[MAXNAMESIZE-4]='.';
_this->m_szRemoteName[MAXNAMESIZE-3]='.';
_this->m_szRemoteName[MAXNAMESIZE-2]='.';
_this->m_szRemoteName[MAXNAMESIZE-1]=0x00;
}
const long lTitleBufSize=256;
char szTitle[lTitleBufSize];
_snprintf(szTitle,lTitleBufSize-1," Chat with <%s> - VMOpsVNC",_this->m_szRemoteName);
SetWindowText(hWnd, szTitle);
// Trunc the remote name for display in Chat Area before the first parenthesis, if any.
char *p = strchr(_this->m_szRemoteName, '(');
if (p != NULL) *(p - 1) = '\0';
// Chat area
_this->SetTextFormat(); // Set character formatting and background color
SendDlgItemMessage( hWnd, IDC_CHATAREA_EDIT, EM_SETBKGNDCOLOR, FALSE, 0xFFFFFF );
memset(_this->m_szLocalText, 0, TEXTMAXSIZE);
// if (!_this->m_fPersistentTexts)
{
memset(_this->m_szLastLocalText, 0, TEXTMAXSIZE);
memset(_this->m_szTextBoxBuffer, 0, TEXTMAXSIZE); // Clear Chat area
// Load and display diclaimer message
// sf@2005 - Only if the DSMplugin is used
if (!_this->m_pCC->m_fUsePlugin)
{
if (LoadString(_this->m_pApp->m_instance, IDS_WARNING, _this->m_szRemoteText, TEXTMAXSIZE -1) )
_this->PrintMessage(_this->m_szRemoteText, NULL ,GREY);
}
}
SetDlgItemText(hWnd, IDC_INPUTAREA_EDIT, _this->m_szLocalText);
SetDlgItemText(hWnd, IDC_CHATAREA_EDIT, _this->m_szTextBoxBuffer); // Chat area
// Scroll down the chat window
// The following seems necessary under W9x & Me if we want the Edit to scroll to bottom...
SCROLLINFO si;
ZeroMemory(&si, sizeof(SCROLLINFO));
si.cbSize = sizeof(SCROLLINFO);
si.fMask = SIF_RANGE|SIF_PAGE;
GetScrollInfo(GetDlgItem(hWnd, IDC_CHATAREA_EDIT), SB_VERT, &si);
si.nPos = si.nMax - max(si.nPage - 1, 0);
SendDlgItemMessage(hWnd, IDC_CHATAREA_EDIT, WM_VSCROLL, MAKELONG(SB_THUMBPOSITION, si.nPos), 0L);
// This line does the bottom scrolling correctly under NT4,W2K, XP...
// SendDlgItemMessage(m_hDlg, IDC_CHATAREA_EDIT, WM_VSCROLL, SB_BOTTOM, 0L);
// SendDlgItemMessage(hWnd, IDC_PERSISTENT_CHECK, BM_SETCHECK, _this->m_fPersistentTexts, 0);
// Tell the other side to open the TextChat Window
_this->SendTextChatRequest(CHAT_OPEN);
SetForegroundWindow(hWnd);
return TRUE;
}
// break;
case WM_COMMAND:
switch (LOWORD(wParam))
{
/*
case IDC_PERSISTENT_CHECK:
_this->m_fPersistentTexts = (SendDlgItemMessage(hWnd, IDC_PERSISTENT_CHECK, BM_GETCHECK, 0, 0) == BST_CHECKED);
return TRUE;
*/
case IDOK:
// Server orders to close TextChat
EndDialog(hWnd, FALSE);
return TRUE;
case IDCANCEL:
_this->SendTextChatRequest(CHAT_CLOSE); // Server must close TextChat
EndDialog(hWnd, FALSE);
return TRUE;
case IDC_SEND_B:
{
memset(_this->m_szLocalText,0,TEXTMAXSIZE);
UINT nRes = GetDlgItemText( hWnd, IDC_INPUTAREA_EDIT, _this->m_szLocalText, TEXTMAXSIZE-1);
strcat(_this->m_szLocalText, "\n");
_this->SendLocalText();
SetFocus(GetDlgItem(hWnd, IDC_INPUTAREA_EDIT));
}
return TRUE;
case IDC_HIDE_B:
_this->ShowChatWindow(false);
return TRUE;
case IDC_INPUTAREA_EDIT:
if(HIWORD(wParam) == EN_UPDATE)
{
memset(_this->m_szLocalText,0,TEXTMAXSIZE);
UINT nRes = GetDlgItemText( hWnd, IDC_INPUTAREA_EDIT, _this->m_szLocalText, TEXTMAXSIZE);
if (strstr(_this->m_szLocalText,"\n") >0 )
{
_this->SendLocalText();
}
}
return TRUE;
}
break;
case WM_SYSCOMMAND:
switch (LOWORD(wParam))
{
case SC_RESTORE:
_this->ShowChatWindow(true);
//SetFocus(GetDlgItem(hWnd, IDC_INPUTAREA_EDIT));
return TRUE;
}
break;
case WM_DESTROY:
// _this->SendTextChatRequest(_this, CHAT_FINISHED);
EndDialog(hWnd, FALSE);
return TRUE;
}
return 0;
}