зеркало из https://github.com/mozilla/pjs.git
7764 строки
244 KiB
C++
7764 строки
244 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public License
|
|
* Version 1.0 (the "NPL"); you may not use this file except in
|
|
* compliance with the NPL. You may obtain a copy of the NPL at
|
|
* http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the NPL is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
|
|
* for the specific language governing rights and limitations under the
|
|
* NPL.
|
|
*
|
|
* The Initial Developer of this code under the NPL is Netscape
|
|
* Communications Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
|
|
* Reserved.
|
|
*/
|
|
|
|
// edprops.cpp : implementation file
|
|
//
|
|
#include "stdafx.h"
|
|
#ifdef EDITOR
|
|
#include "edres2.h"
|
|
#include "property.h"
|
|
#include "edprops.h"
|
|
#include "edttypes.h"
|
|
#include "edt.h"
|
|
#include "pa_tags.h"
|
|
#include "netsvw.h"
|
|
#include "edview.h"
|
|
#include "styles.h" // For WFE_DrawSwatch()
|
|
#ifndef XP_WIN32
|
|
#include "tooltip.h"
|
|
#endif
|
|
#include "nethelp.h"
|
|
#include "xp_help.h"
|
|
#include "prefapi.h"
|
|
#include "prefinfo.h"
|
|
// the dialog box & string resources are in edtrcdll DLL
|
|
#include "edtrcdll\src\resource.h"
|
|
|
|
// For XP Strings
|
|
extern "C" {
|
|
#include "xpgetstr.h"
|
|
#define WANT_ENUM_STRING_IDS
|
|
#include "allxpstr.h"
|
|
#undef WANT_ENUM_STRING_IDS
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
#undef THIS_FILE
|
|
static char BASED_CODE THIS_FILE[] = __FILE__;
|
|
#endif
|
|
|
|
// Font shared by all comboboxes in dialogs and toolbars
|
|
// Should be 1-pixel, 8pt MS Sans Serif or default GUI or font on foreign systems
|
|
CFont * wfe_pFont = 0;
|
|
// BOLD version of above font
|
|
CFont * wfe_pBoldFont = 0;
|
|
int wfe_iFontHeight = 15; // Full Height of the this font
|
|
|
|
// Modification of wfe_iFontHeight for list boxes;
|
|
int wfe_iListItemHeight = 15;
|
|
|
|
|
|
extern void wfe_Progress(MWContext *pMWContext, const char *pMessage);
|
|
|
|
extern BOOL FE_ResolveLinkURL( MWContext* pMWContext,
|
|
CString& csURL, BOOL bAutoAdjustLinks );
|
|
|
|
BOOL bSaveDefaultHRule = TRUE;
|
|
|
|
int32 wfe_iFontSizeMode = ED_FONTSIZE_POINTS;
|
|
|
|
// Convert CString into non-const char*
|
|
#define CHAR_STR(csString) (char*)LPCSTR(csString)
|
|
|
|
#define MAX_AUTO_SAVE_MINUTES 600
|
|
|
|
#define COLORREF_SIZE sizeof(COLORREF)
|
|
|
|
// Array of colors to show in common color dialog
|
|
COLORREF wfe_CustomPalette[16];
|
|
// Colors saved in prefs - the last-picked text and background
|
|
COLORREF wfe_crLastColorPicked;
|
|
COLORREF wfe_crLastBkgrndColorPicked;
|
|
|
|
int wfe_iTrueTypeFontBase = 0;
|
|
int wfe_iTrueTypeFontCount = 0;
|
|
|
|
// Global array of True-Type fonts
|
|
// Neccesitates restarting program after new fonts installed
|
|
char **wfe_ppTrueTypeFonts = NULL;
|
|
|
|
// Global index within wfe_ppTrueTypeFonts if "Other..." was needed (too many fonts)
|
|
int wfe_iFontFaceOtherIndex = 0;
|
|
|
|
int CALLBACK EXPORT
|
|
EnumTrueTypeFonts(LPLOGFONT lpnlf, LPTEXTMETRIC lpntm, int nFontType, LPARAM lParam)
|
|
{
|
|
if (lpntm->tmPitchAndFamily & (/*TMPF_VECTOR|TMPF_DEVICE|*/TMPF_TRUETYPE)){
|
|
if( wfe_iTrueTypeFontCount < MAX_TRUETYPE_FONTS ){
|
|
wfe_ppTrueTypeFonts[wfe_iTrueTypeFontCount] = XP_STRDUP(lpnlf->lfFaceName);
|
|
wfe_iTrueTypeFontCount++;
|
|
} else {
|
|
// Too many fonts -- add "Other..." to use common dialog
|
|
*((BOOL*)lParam) = TRUE;
|
|
return FALSE;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
int CompareStrings( const void *arg1, const void *arg2 )
|
|
{
|
|
// Compare all of both strings:
|
|
// TODO: DO WIDE/MULTIBYTE VERSION
|
|
return _stricmp( * ( char** ) arg1, * ( char** ) arg2 );
|
|
}
|
|
|
|
static BOOL bTooManyFonts = FALSE;
|
|
|
|
BOOL wfe_InitTrueTypeArray(CDC *pDC)
|
|
{
|
|
ASSERT(pDC);
|
|
HDC hdc = pDC->GetSafeHdc();
|
|
if( wfe_ppTrueTypeFonts ){
|
|
return bTooManyFonts;
|
|
}
|
|
|
|
wfe_iTrueTypeFontCount = 0;
|
|
wfe_iFontFaceOtherIndex = 0;
|
|
bTooManyFonts = FALSE;
|
|
wfe_ppTrueTypeFonts = (char**)XP_ALLOC(MAX_TRUETYPE_FONTS*sizeof(char*));
|
|
if( !wfe_ppTrueTypeFonts ){
|
|
ASSERT(wfe_ppTrueTypeFonts);
|
|
bTooManyFonts = TRUE;
|
|
return TRUE;
|
|
}
|
|
memset((void*)wfe_ppTrueTypeFonts, 0, MAX_TRUETYPE_FONTS*sizeof(char*));
|
|
|
|
EnumFontFamilies(hdc, NULL, (FONTENUMPROC)EnumTrueTypeFonts, (LPARAM)&bTooManyFonts);
|
|
|
|
if( wfe_iTrueTypeFontCount ){
|
|
// Sort font names using Quicksort algorithm:
|
|
qsort( (void *)wfe_ppTrueTypeFonts, (size_t)wfe_iTrueTypeFontCount, sizeof( char * ), CompareStrings );
|
|
}
|
|
return bTooManyFonts;
|
|
}
|
|
|
|
void wfe_FreeTrueTypeArray()
|
|
{
|
|
int i;
|
|
if( wfe_ppTrueTypeFonts ){
|
|
for( i = 0; i < MAX_TRUETYPE_FONTS; i++ ){
|
|
XP_FREEIF(wfe_ppTrueTypeFonts[i]);
|
|
}
|
|
XP_FREE(wfe_ppTrueTypeFonts);
|
|
wfe_ppTrueTypeFonts = 0;
|
|
}
|
|
wfe_iTrueTypeFontCount = 0;
|
|
}
|
|
|
|
static char pSepFont1[128] = "_";
|
|
|
|
int wfe_FillFontComboBox(CComboBox * pCombo, int * pMaxWidth)
|
|
{
|
|
char * pFontFaces = EDT_GetFontFaces();
|
|
CDC * pDC = pCombo->GetDC();
|
|
if( !pDC ) return 0;
|
|
HDC hdc = 0;
|
|
if( pMaxWidth ){
|
|
// We need this only if returning iMaxWidth
|
|
hdc = pDC->GetSafeHdc();
|
|
}
|
|
|
|
CSize cSize;
|
|
int iMaxWidth = 0;
|
|
int i = 0;
|
|
|
|
if( pFontFaces ){
|
|
char * pFont = pFontFaces;
|
|
int iLen;
|
|
for ( i = 0; i < 2; i++){
|
|
|
|
iLen = XP_STRLEN(pFont);
|
|
if( iLen ){
|
|
// add to the end of the list
|
|
// LIST MUST BE NOT SORTED for EDT_GetFontFaceIndex to work
|
|
if( i == 1 ){
|
|
// Add signal to draw separator under this string
|
|
strcpy(pSepFont1+1,pFont);
|
|
pCombo->AddString(pSepFont1);
|
|
} else {
|
|
pCombo->AddString(pFont);
|
|
}
|
|
if ( hdc ){
|
|
cSize = CIntlWin::GetTextExtent(0, hdc, pFont, strlen(pFont));
|
|
pDC->LPtoDP(&cSize);
|
|
if ( cSize.cx > iMaxWidth ){
|
|
iMaxWidth = cSize.cx;
|
|
}
|
|
}
|
|
pFont += iLen+1;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
// Done with built-in fonts -- next index must be start of True Type
|
|
wfe_iTrueTypeFontBase = i;
|
|
|
|
// Get pointers to array of font names
|
|
// (Global list should already be built)
|
|
if( wfe_iTrueTypeFontCount && wfe_ppTrueTypeFonts){
|
|
// Add sorted string list to combobox
|
|
for( int i = 0; i < wfe_iTrueTypeFontCount; i++ ){
|
|
char * pFontName = wfe_ppTrueTypeFonts[i];
|
|
pCombo->AddString(pFontName);
|
|
if ( hdc ){
|
|
cSize = CIntlWin::GetTextExtent(0, hdc, pFontName, strlen(pFontName));
|
|
pDC->LPtoDP(&cSize);
|
|
if ( cSize.cx > iMaxWidth ){
|
|
iMaxWidth = cSize.cx;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
// Should never get here if wfe_IntTrueTypeArray was ever called
|
|
bTooManyFonts = TRUE;;
|
|
}
|
|
if( pMaxWidth ){
|
|
// Return max width so dropdown list can be expanded to fit
|
|
*pMaxWidth = iMaxWidth;
|
|
}
|
|
|
|
if( bTooManyFonts ){
|
|
// Add pointer to globalb "Other..." string
|
|
pCombo->AddString(ed_pOther);
|
|
// Calculage the index to the "Other..." item,
|
|
wfe_iFontFaceOtherIndex = wfe_iTrueTypeFontBase + wfe_iTrueTypeFontCount;
|
|
} else {
|
|
wfe_iFontFaceOtherIndex = 0;
|
|
}
|
|
pCombo->ReleaseDC(pDC);
|
|
return wfe_iFontFaceOtherIndex;
|
|
}
|
|
|
|
char wfe_pFontSizeString[64];
|
|
char wfe_pFontSizeList[MAX_FONT_SIZE+1][64];
|
|
|
|
int wfe_FillFontSizeCombo(MWContext *pMWContext, CNSComboBox * pCombo, BOOL bFixedWidth)
|
|
{
|
|
int iFontSize = EDT_GetFontSize(pMWContext);
|
|
if( pMWContext ) {
|
|
// Clear current data
|
|
pCombo->ResetContent();
|
|
|
|
// Format a string like: "+2" for advanced mode, or convert to
|
|
// point size for regular mode
|
|
for ( int i = 1; i < MAX_FONT_SIZE; i++ ){
|
|
pCombo->AddString(wfe_GetFontSizeString(pMWContext, i, bFixedWidth));
|
|
}
|
|
if( wfe_iFontSizeMode == ED_FONTSIZE_ADVANCED ){
|
|
// "_" will signal combobox to draw a separator under this
|
|
pCombo->AddString("_+4");
|
|
// This is "8 pts" so we must allow translating string
|
|
// (others assume all languages can use ascii numbers)
|
|
pCombo->AddString(szLoadString(IDS_8_PTS));
|
|
pCombo->AddString("9");
|
|
pCombo->AddString("10");
|
|
pCombo->AddString("11");
|
|
pCombo->AddString("12");
|
|
pCombo->AddString("14");
|
|
pCombo->AddString("16");
|
|
pCombo->AddString("18");
|
|
pCombo->AddString("20");
|
|
pCombo->AddString("22");
|
|
pCombo->AddString("24");
|
|
pCombo->AddString("26");
|
|
pCombo->AddString("28");
|
|
pCombo->AddString("36");
|
|
pCombo->AddString("48");
|
|
pCombo->AddString("72");
|
|
} else {
|
|
pCombo->AddString(wfe_GetFontSizeString(pMWContext, MAX_FONT_SIZE, bFixedWidth));
|
|
}
|
|
}
|
|
// Index to current size is 1 less than our relative size (range = 1-7)
|
|
// Note that this may be 0 (on startup)
|
|
return (iFontSize - 1);
|
|
}
|
|
|
|
// iSize should be 1 to 7
|
|
// Returns pointer to static string - DON'T FREE RESULT!
|
|
char * wfe_GetFontSizeString(MWContext * pMWContext, int iSize, BOOL bFixedWidth, BOOL bMenu)
|
|
{
|
|
ASSERT(pMWContext);
|
|
int iRelative = iSize - 1 + MIN_FONT_SIZE_RELATIVE;
|
|
|
|
CString csFormat;
|
|
|
|
if( wfe_iFontSizeMode != ED_FONTSIZE_POINTS ){
|
|
if ( iRelative < 0 ){
|
|
// Minus is very narrow - include extra leading space
|
|
csFormat = " -";
|
|
} else if ( iRelative > 0 ){
|
|
csFormat = '+';
|
|
} else {
|
|
// Use 2 spaces befor "0"; has same width as 1 "-" or "+" char
|
|
csFormat = " ";
|
|
}
|
|
|
|
if( bMenu && iSize > 2 ){
|
|
// Add underline for "accelerator" only for "0" and above
|
|
csFormat += '&';
|
|
}
|
|
csFormat += "%i";
|
|
|
|
wsprintf(wfe_pFontSizeList[iSize-1], LPCSTR(csFormat), abs(iRelative));
|
|
} else {
|
|
// Construct strings for Point sizes
|
|
CDCCX *pDC = VOID2CX(pMWContext->fe.cx, CDCCX);
|
|
|
|
EncodingInfo *pEncoding = theApp.m_pIntlFont->GetEncodingInfo(pMWContext);
|
|
int iBaseSize = bFixedWidth ? pEncoding->iFixSize : pEncoding->iPropSize;
|
|
double dPoints = pDC->CalcFontPointSize(iSize, iBaseSize);
|
|
wsprintf(wfe_pFontSizeList[iSize-1], "%d", (int)dPoints);
|
|
}
|
|
return wfe_pFontSizeList[iSize-1];
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// Local string helper functions
|
|
|
|
// Strip out quotes and spaces at the ends
|
|
void CleanupString(CString& csString)
|
|
{
|
|
csString.TrimLeft();
|
|
csString.TrimRight();
|
|
int iQuote;
|
|
while ( (iQuote = csString.Find('\"')) != -1 ){
|
|
csString = csString.Left(iQuote) + csString.Mid(iQuote+1);
|
|
}
|
|
}
|
|
|
|
// Extract separate Name and Value strings
|
|
// Return TRUE if Name string was found
|
|
BOOL GetNameAndValue(CString& csEntry, CString& csName, CString& csValue)
|
|
{
|
|
int iEqual = csEntry.Find('=');
|
|
if ( iEqual != -1 ) {
|
|
csName = csEntry.Left(iEqual);
|
|
CleanupString(csName);
|
|
csValue = csEntry.Mid(iEqual+1);
|
|
CleanupString(csValue);
|
|
return ( !csName.IsEmpty() );
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
// Local function used by Image and Document Properties dialogs
|
|
//
|
|
BOOL wfe_ValidateImage(MWContext * pMWContext, CString& csImageURL, BOOL bCheckIfFileExists)
|
|
{
|
|
BOOL bRetVal = TRUE;
|
|
BOOL bFileExists = FALSE;
|
|
CleanupString(csImageURL);
|
|
char * pAbsoluteImageURL = NULL;
|
|
CString csURL;
|
|
|
|
// Empty is OK
|
|
if ( csImageURL.IsEmpty() ) {
|
|
return TRUE;
|
|
}
|
|
//int iUrlType = NET_URL_Type(csUrl);
|
|
|
|
// We will trust HTTP: URLs to be absolute and valid
|
|
if ( NET_IsHTTP_URL( csImageURL ) ) {
|
|
return TRUE;
|
|
}
|
|
|
|
// Convert to URL format -- just copies if already have URL syntax
|
|
WFE_ConvertFile2Url( csURL, (char*)LPCSTR(csImageURL) );
|
|
|
|
// Convert any user-enterred local strings to an absolute URL
|
|
History_entry * pEntry = SHIST_GetCurrent(&pMWContext->hist);
|
|
if( pEntry && pEntry->address ) {
|
|
pAbsoluteImageURL = NET_MakeAbsoluteURL(
|
|
pEntry->address, (char *)LPCSTR(csURL) );
|
|
}
|
|
|
|
if(bCheckIfFileExists && pAbsoluteImageURL) {
|
|
|
|
if ( NET_IsLocalFileURL((char*)LPCSTR(csImageURL)) ) {
|
|
// We have a local file URL -- test for existence
|
|
char * pLocal = NULL;
|
|
bFileExists = XP_ConvertUrlToLocalFile( csImageURL, &pLocal /*NULL*/);
|
|
if( pLocal ) XP_FREE(pLocal);
|
|
|
|
} else {
|
|
// Assume we have a local file - find it
|
|
XP_StatStruct statinfo;
|
|
if ( -1 != XP_Stat((char*)LPCSTR(csImageURL), &statinfo, xpURL) &&
|
|
statinfo.st_mode & _S_IFREG ) {
|
|
bFileExists = TRUE;
|
|
}
|
|
}
|
|
if ( !bFileExists ) {
|
|
CString csMsg;
|
|
AfxFormatString1(csMsg, IDS_ERR_SRC_NOT_FOUND, csImageURL);
|
|
CWnd *pParent = GetFrame(pMWContext)->GetFrameWnd()->GetLastActivePopup();
|
|
::MessageBox( pParent ? pParent->m_hWnd : NULL,
|
|
csMsg, szLoadString(IDS_VALIDATE_IMAGE_FILE),
|
|
MB_OK | MB_ICONEXCLAMATION);
|
|
}
|
|
}
|
|
|
|
if( bCheckIfFileExists ){
|
|
bRetVal = bFileExists;
|
|
}
|
|
|
|
if ( pAbsoluteImageURL ) {
|
|
if( bRetVal ){
|
|
csImageURL = pAbsoluteImageURL;
|
|
}
|
|
XP_FREE(pAbsoluteImageURL);
|
|
}
|
|
|
|
return bRetVal;
|
|
}
|
|
|
|
// Initialize the strings for percent/pixels combobox
|
|
// used with Width and Height input
|
|
void wfe_InitPixOrPercentCombos(CWnd* pParent)
|
|
{
|
|
CComboBox *pCombo =
|
|
(CComboBox*)(pParent->GetDlgItem(IDC_HEIGHT_PIX_OR_PERCENT));
|
|
if(pCombo){
|
|
pCombo->AddString(szLoadString(IDS_PIXELS));
|
|
pCombo->AddString(szLoadString(IDS_PERCENT_WINDOW));
|
|
}
|
|
pCombo = (CComboBox*)(pParent->GetDlgItem(IDC_WIDTH_PIX_OR_PERCENT));
|
|
if(pCombo){
|
|
pCombo->AddString(szLoadString(IDS_PIXELS));
|
|
pCombo->AddString(szLoadString(IDS_PERCENT_WINDOW));
|
|
}
|
|
}
|
|
|
|
// Get the "100%" Height and "100%" Width that layout uses:
|
|
// this is the current view's dimensions minus margins
|
|
void wfe_GetLayoutViewSize(MWContext * pMWContext, int32 * pWidth, int32 * pHeight)
|
|
{
|
|
if( pMWContext ){
|
|
int32 iXOrigin;
|
|
int32 iYOrigin;
|
|
int32 iHeight;
|
|
int32 iWidth;
|
|
int32 iMarginWidth;
|
|
int32 iMarginHeight;
|
|
|
|
FE_GetDocAndWindowPosition(pMWContext, &iXOrigin, &iYOrigin, &iWidth, &iHeight);
|
|
LO_GetDocumentMargins(pMWContext, &iMarginWidth, &iMarginHeight);
|
|
|
|
if( pWidth ){
|
|
*pWidth = iWidth - (2 * iMarginWidth);
|
|
}
|
|
if( pHeight ){
|
|
*pHeight = iHeight - (2 * iMarginHeight);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
char * wfe_ConvertImage(char *p_fileurl,void *p_parentwindow,MWContext *p_pMWContext)
|
|
{
|
|
if( !wfe_ValidateImage(p_pMWContext, CString(p_fileurl), TRUE) ){
|
|
return NULL;
|
|
}
|
|
|
|
CONVERT_IMGCONTEXT imageContext;
|
|
CONVERT_IMG_INFO imageInfo;
|
|
memset(&imageContext,0,sizeof(CONVERT_IMGCONTEXT));
|
|
imageContext.m_stream.m_type=CONVERT_FILE;
|
|
char *t_charp;
|
|
XP_ConvertUrlToLocalFile(p_fileurl,&t_charp);
|
|
if (!t_charp) //failed to localize file
|
|
{
|
|
// Show same error message as when wfe_ValidateImage fails:
|
|
CString csMsg;
|
|
AfxFormatString1(csMsg, IDS_ERR_SRC_NOT_FOUND, p_fileurl);
|
|
CWnd *pParent = GetFrame(p_pMWContext)->GetFrameWnd()->GetLastActivePopup();
|
|
::MessageBox( pParent ? pParent->m_hWnd : NULL,
|
|
csMsg, szLoadString(IDS_VALIDATE_IMAGE_FILE),
|
|
MB_OK | MB_ICONEXCLAMATION);
|
|
return NULL;
|
|
}
|
|
XP_STRCPY(imageContext.m_filename,t_charp);
|
|
XP_FREE(t_charp);
|
|
imageContext.m_stream.m_file=XP_FileOpen(imageContext.m_filename,xpTemporary,XP_FILE_READ_BIN);
|
|
if (imageContext.m_stream.m_file)
|
|
{
|
|
imageContext.m_imagetype=conv_bmp;
|
|
imageContext.m_callbacks.m_dialogimagecallback=FE_ImageConvertDialog;
|
|
imageContext.m_callbacks.m_displaybuffercallback=FE_ImageConvertDisplayBuffer;
|
|
imageContext.m_callbacks.m_completecallback=NULL;
|
|
|
|
imageContext.m_pMWContext=p_pMWContext;
|
|
imageContext.m_parentwindow=p_parentwindow;
|
|
char *t_outputfilename[1];
|
|
t_outputfilename[0]=NULL;
|
|
CONVERT_IMAGERESULT t_result=convert_stream2image(imageContext,&imageInfo,1,t_outputfilename); //1 for 1 output
|
|
//there is a callback when complete
|
|
XP_FileClose(imageContext.m_stream.m_file);
|
|
if (t_outputfilename[0])
|
|
{
|
|
// If using GIF plugin, wait here until finished
|
|
// (edit buffer is not writeable during plugin use)
|
|
char * pLastDot = strrchr(t_outputfilename[0], '.');
|
|
if(pLastDot && 0 == _stricmp(pLastDot, ".gif")){
|
|
while(!EDT_IsWritableBuffer(p_pMWContext)) FEU_StayingAlive();
|
|
}
|
|
return t_outputfilename[0];
|
|
}
|
|
else
|
|
{
|
|
if (t_result>CONV_OK)//not cancel or ok
|
|
{
|
|
((CWnd *)p_parentwindow)->MessageBox(szLoadString(CNetscapeEditView::m_converrmsg[t_result]));
|
|
}
|
|
return NULL;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
CNSComboBox::CNSComboBox() :
|
|
m_iSearchPos(0),
|
|
m_bAllowSearch(0),
|
|
m_dwButtonDownTime(0),
|
|
m_bCheckTime(0),
|
|
m_pNotInListText(0)
|
|
{
|
|
}
|
|
|
|
BOOL CNSComboBox::Create(RECT& rect, CWnd* pParentWnd, UINT nID)
|
|
{
|
|
if( !CComboBox::Create(CBS_DROPDOWNLIST|CBS_OWNERDRAWFIXED|WS_VISIBLE|WS_TABSTOP|WS_VSCROLL|WS_HSCROLL|CBS_AUTOHSCROLL,
|
|
rect, pParentWnd, nID) ){
|
|
TRACE0("Failed to create Netscape Combo-box\n");
|
|
return FALSE;
|
|
}
|
|
|
|
if( wfe_pFont )
|
|
SetFont(wfe_pFont);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL CNSComboBox::Subclass(CWnd* pParentWnd, UINT nID)
|
|
{
|
|
BOOL bResult = SubclassDlgItem(nID, pParentWnd);
|
|
if( bResult ){
|
|
if( wfe_pFont )
|
|
SetFont(wfe_pFont);
|
|
}
|
|
return bResult;
|
|
}
|
|
|
|
int CNSComboBox::FindSelectedOrSetText(char * pText, int iStartAt)
|
|
{
|
|
int iSel = -1;
|
|
if( pText ){
|
|
// Skip over initial signal for separator
|
|
if( *pText == '_' ){
|
|
pText++;
|
|
}
|
|
for( int i = iStartAt; i < GetCount(); i++ ){
|
|
char * pItem = (char*)GetItemData(i);
|
|
if( *pItem == '_' ){
|
|
pItem++;
|
|
}
|
|
if( 0 == strcmp(pText, pItem) ){
|
|
// We found it
|
|
iSel = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if( iSel == -1 ){
|
|
// Text is not in the list
|
|
XP_FREEIF(m_pNotInListText);
|
|
m_pNotInListText = XP_STRDUP(pText ? pText : " ");
|
|
SetWindowText(m_pNotInListText);
|
|
}
|
|
else {
|
|
SetWindowText(pText);
|
|
}
|
|
// This must be AFTER the SetWindowText()
|
|
// so our setting it to -1 triggers display
|
|
// of m_pNotInListText in the closed combobox
|
|
SetCurSel(iSel);
|
|
|
|
return iSel;
|
|
}
|
|
|
|
void CNSComboBox::MeasureItem(LPMEASUREITEMSTRUCT lpMIS)
|
|
{
|
|
// all items are of fixed size
|
|
lpMIS->itemHeight = wfe_iListItemHeight;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////
|
|
void CNSComboBox::DrawItem(LPDRAWITEMSTRUCT lpDIS)
|
|
{
|
|
CDC* pDC = CDC::FromHandle(lpDIS->hDC);
|
|
|
|
CRect rectItem = lpDIS->rcItem;
|
|
|
|
if( lpDIS->itemState & ODS_SELECTED ){
|
|
pDC->SetTextColor(GetSysColor(COLOR_HIGHLIGHTTEXT));
|
|
pDC->SetBkColor(GetSysColor(COLOR_HIGHLIGHT));
|
|
} else {
|
|
pDC->SetTextColor(RGB(0,0,0));
|
|
pDC->SetBkColor(RGB(255,255,255));
|
|
}
|
|
|
|
char *pData;
|
|
if( lpDIS->itemData == -1){// When is this -1?
|
|
// This is a real neat trick to display text in closed
|
|
// dropdown combobox's main window even though
|
|
// text is not in list (selected item = -1)
|
|
// This text should be set in FindSelectedOrSetText
|
|
pData = m_pNotInListText;
|
|
} else {
|
|
pData = (char*)lpDIS->itemData;
|
|
}
|
|
|
|
if( pData ){
|
|
// Draw a gray line separator
|
|
CPen pen(PS_SOLID, 1, RGB(128,128,128));
|
|
CPen *pOldPen = pDC->SelectObject(&pen);
|
|
|
|
BOOL bDrawSeparator = FALSE;
|
|
if( *pData == '_' ){
|
|
// We have a separator - skip over signal character
|
|
pData++;
|
|
if( GetDroppedState() ) {
|
|
bDrawSeparator = TRUE;
|
|
}
|
|
}
|
|
pDC->ExtTextOut(4, rectItem.top+1, ETO_CLIPPED | ETO_OPAQUE, &lpDIS->rcItem,
|
|
pData, strlen(pData), NULL);
|
|
if( bDrawSeparator ){
|
|
// Draw solid line as separator
|
|
pDC->MoveTo(rectItem.left, rectItem.bottom-1);
|
|
pDC->LineTo(rectItem.right, rectItem.bottom-1);
|
|
}
|
|
pDC->SelectObject(pOldPen);
|
|
}
|
|
}
|
|
|
|
int CNSComboBox::CompareItem(LPCOMPAREITEMSTRUCT lpCIS)
|
|
{
|
|
// We should never sort!
|
|
return 0;
|
|
}
|
|
|
|
// These were initially in CEditFrame:
|
|
BEGIN_MESSAGE_MAP(CNSComboBox, CComboBox)
|
|
//{{AFX_MSG_MAP(CNSComboBox)
|
|
ON_WM_KEYDOWN()
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
void CNSComboBox::InitSearch()
|
|
{
|
|
// Don't allow autoselect while typing
|
|
// when we have a multibyte system
|
|
if( !GetSystemMetrics(SM_DBCSENABLED) ){
|
|
m_bAllowSearch = TRUE;
|
|
m_iSearchPos = 0;
|
|
m_pSearchBuf[m_iSearchPos] = '\0';
|
|
}
|
|
}
|
|
|
|
void CNSComboBox::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
|
|
{
|
|
BOOL bMatch;
|
|
// Ignor shift and control presses
|
|
if( m_bAllowSearch && nChar != VK_SHIFT && nChar != VK_CONTROL ){
|
|
if( nChar != VK_RETURN &&
|
|
(nChar == 4 || XP_IS_ALPHA(nChar) ||
|
|
XP_IS_DIGIT(nChar) || XP_IS_SPACE(nChar)) )
|
|
{
|
|
int iStart = 0;
|
|
if( nChar == VK_BACK ){
|
|
// Backup/delete key in buffer
|
|
if( m_iSearchPos > 0 ){
|
|
m_iSearchPos--;
|
|
}
|
|
} else {
|
|
// Append acceptable character to search buffer
|
|
m_pSearchBuf[m_iSearchPos] = nChar;
|
|
if( m_iSearchPos ){
|
|
// Start search from current item if we already had a string,
|
|
iStart = GetCurSel();
|
|
}
|
|
m_iSearchPos++;
|
|
}
|
|
m_pSearchBuf[m_iSearchPos] = '\0';
|
|
|
|
for( int i = iStart; i < GetCount(); i++ )
|
|
{
|
|
char * pItem = (char*)GetItemData(i);
|
|
if( *pItem == '_' ){
|
|
pItem++;
|
|
}
|
|
if( strlen(pItem) >= (size_t)m_iSearchPos ){
|
|
// Current item is at least as long, so compare
|
|
//
|
|
char save = pItem[m_iSearchPos];
|
|
pItem[m_iSearchPos] = '\0';
|
|
bMatch = ( 0 == stricmp(pItem, m_pSearchBuf) );
|
|
// Restore
|
|
pItem[m_iSearchPos] = save;
|
|
if( bMatch ){
|
|
// We found it - set selection
|
|
SetCurSel(i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if( !bMatch ){
|
|
// We didn't find next char, so backup
|
|
m_iSearchPos--;
|
|
m_pSearchBuf[m_iSearchPos] = '\0';
|
|
}
|
|
} else {
|
|
// Another other key resets the search string
|
|
m_iSearchPos = 0;
|
|
m_pSearchBuf[m_iSearchPos] = '\0';
|
|
}
|
|
}
|
|
CComboBox::OnKeyDown(nChar, nRepCnt, nFlags);
|
|
}
|
|
|
|
BOOL CNSComboBox::PreTranslateMessage(MSG* pMsg)
|
|
{
|
|
return CComboBox::PreTranslateMessage(pMsg);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// Custom Combobox - containing colors
|
|
|
|
CColorComboBox::CColorComboBox() :
|
|
m_crColor(MIXED_COLORREF),
|
|
m_bCustomColor(0),
|
|
m_bMixedColors(0),
|
|
m_hPal(0),
|
|
m_pParent(0)
|
|
{
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CColorComboBox, CComboBox)
|
|
//{{AFX_MSG_MAP(CColorComboBox)
|
|
ON_WM_LBUTTONDOWN()
|
|
ON_WM_LBUTTONUP()
|
|
ON_WM_LBUTTONDBLCLK()
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
void CColorComboBox::OnLButtonDown(UINT nFlags, CPoint cPoint)
|
|
{
|
|
if( m_pParent ){
|
|
// Send message to parent to do the Color Dialog
|
|
m_pParent->SendMessage(WM_COMMAND, ID_GET_COLOR, 0);
|
|
}
|
|
}
|
|
void CColorComboBox::OnLButtonUp(UINT nFlags, CPoint cPoint)
|
|
{
|
|
}
|
|
void CColorComboBox::OnLButtonDblClk(UINT nFlags, CPoint cPoint)
|
|
{
|
|
if( m_pParent ){
|
|
// Send message to parent to do the Color Dialog
|
|
m_pParent->SendMessage(WM_COMMAND, ID_GET_COLOR, 0);
|
|
}
|
|
}
|
|
|
|
int CColorComboBox::SubclassAndFill(CWnd* pParentWnd, UINT nID, COLORREF crDefColor)
|
|
{
|
|
m_pParent = pParentWnd;
|
|
if( SubclassDlgItem(nID, pParentWnd) ){
|
|
return FillList(crDefColor);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
BOOL CColorComboBox::CreateAndFill(RECT& rect, CWnd* pParentWnd, UINT nID, COLORREF crDefColor)
|
|
{
|
|
m_pParent = pParentWnd;
|
|
if( !Create(CBS_DROPDOWNLIST|CBS_OWNERDRAWFIXED|WS_VISIBLE|WS_TABSTOP|WS_VSCROLL,
|
|
rect, pParentWnd, nID) ){
|
|
TRACE0("Failed to create Font Color combo-box\n");
|
|
return FALSE;
|
|
}
|
|
|
|
if( wfe_pFont ){
|
|
SetFont(wfe_pFont);
|
|
}
|
|
return FillList(crDefColor);
|
|
}
|
|
|
|
int CColorComboBox::FillList(COLORREF crDefColor)
|
|
{
|
|
|
|
CDC * pDC = GetDC();
|
|
if( !pDC ) return 0;
|
|
|
|
m_hPal = WFE_GetUIPalette(GetParentFrame());
|
|
|
|
// WE NO LONGER FILL THE LIST - POPUP IS USED INSTEAD
|
|
|
|
ReleaseDC(pDC);
|
|
return TRUE;
|
|
}
|
|
|
|
void CColorComboBox::MeasureItem(LPMEASUREITEMSTRUCT lpMIS)
|
|
{
|
|
// all items are of fixed size
|
|
lpMIS->itemHeight = wfe_iListItemHeight;
|
|
}
|
|
|
|
void CColorComboBox::DrawItem(LPDRAWITEMSTRUCT lpDIS)
|
|
{
|
|
// We need to draw outside the supplied area,
|
|
// so use main combobox DC
|
|
HDC hDC = GetDC()->m_hDC;
|
|
|
|
HPALETTE hOldPal = NULL;
|
|
if( m_hPal )
|
|
{
|
|
hOldPal = ::SelectPalette( hDC, m_hPal, FALSE );
|
|
}
|
|
|
|
RECT rect = lpDIS->rcItem;
|
|
::InflateRect(&rect, 1, 1);
|
|
|
|
// Draw color sample
|
|
if (lpDIS->itemAction & ODA_DRAWENTIRE)
|
|
{
|
|
if( m_bMixedColors )
|
|
{
|
|
rect.top--;
|
|
rect.left--;
|
|
//rect.bottom++;
|
|
::FillRect(hDC, &rect, sysInfo.m_hbrBtnFace);
|
|
if( sysInfo.m_bWin4 )
|
|
{
|
|
// It looks very odd unless we add a line to close off button
|
|
HPEN penShadow = ::CreatePen(PS_SOLID, 1, sysInfo.m_clrBtnShadow);
|
|
HPEN hPenOld = (HPEN)::SelectObject(hDC, penShadow);
|
|
::MoveToEx(hDC, rect.right, rect.top, NULL);
|
|
::LineTo(hDC, rect.right, rect.bottom);
|
|
::SelectObject(hDC, hPenOld);
|
|
::DeleteObject(penShadow);
|
|
}
|
|
} else {
|
|
HBRUSH hBrush = ::CreateSolidBrush(m_crColor|0x02000000); // Like PALETTERGB
|
|
::FillRect(hDC, &rect, hBrush);
|
|
::DeleteObject(hBrush);
|
|
|
|
int32 iColorTotal = GetRValue(m_crColor) + GetGValue(m_crColor) + GetBValue(m_crColor);
|
|
|
|
COLORREF crColorHighlight = 0;
|
|
if( iColorTotal < 153 )
|
|
crColorHighlight = PALETTERGB(255,255,0);
|
|
else if( iColorTotal > 700 )
|
|
crColorHighlight = PALETTERGB(0,0,255);
|
|
|
|
if( sysInfo.m_bWin4 )
|
|
{
|
|
// Extra color highlight if color is too dark or light
|
|
if( crColorHighlight )
|
|
{
|
|
hBrush = ::CreateSolidBrush(crColorHighlight);
|
|
::FrameRect(hDC, &rect, hBrush);
|
|
::DeleteObject(hBrush);
|
|
}
|
|
} else {
|
|
// Older UIs have a flat-look for comboboxes,
|
|
// and need an extra border here
|
|
// And we use color if current color is too dark or bright
|
|
if( crColorHighlight )
|
|
{
|
|
hBrush = ::CreateSolidBrush(crColorHighlight);
|
|
} else {
|
|
hBrush = (HBRUSH)::GetStockObject(BLACK_BRUSH);
|
|
}
|
|
::FrameRect(hDC, &rect, hBrush);
|
|
|
|
if( crColorHighlight )
|
|
::DeleteObject(hBrush);
|
|
}
|
|
}
|
|
}
|
|
if(m_hPal)
|
|
{
|
|
::SelectPalette( hDC, hOldPal, FALSE );
|
|
}
|
|
::ReleaseDC(this->m_hWnd, hDC);
|
|
}
|
|
|
|
int CColorComboBox::CompareItem(LPCOMPAREITEMSTRUCT lpCIS)
|
|
{
|
|
// We should never sort!
|
|
return 0;
|
|
}
|
|
|
|
int CColorComboBox::SetColor(COLORREF cr)
|
|
{
|
|
int iIndex = -1;
|
|
if( cr == MIXED_COLORREF )
|
|
{
|
|
m_crColor = RGB(255,255,255);
|
|
}else if( cr == DEFAULT_COLORREF )
|
|
{
|
|
m_crColor = prefInfo.m_rgbForegroundColor;
|
|
iIndex = 0;
|
|
} else {
|
|
m_crColor = cr;
|
|
LO_Color LoColor;
|
|
LoColor.red = GetRValue(cr);
|
|
LoColor.green = GetGValue(cr);
|
|
LoColor.blue = GetBValue(cr);
|
|
iIndex = EDT_GetMatchingFontColorIndex(&LoColor);
|
|
}
|
|
m_bMixedColors = (cr == MIXED_COLORREF);
|
|
// Custom color is one not in our list
|
|
m_bCustomColor = !m_bMixedColors && iIndex == -1;
|
|
Invalidate(FALSE);
|
|
SetCurSel(iIndex);
|
|
return iIndex;
|
|
}
|
|
|
|
|
|
#define COLOR_ROWS 7
|
|
#define COLOR_COLS 10
|
|
// NOTE: This is same as the wfe_iFontHeight found for
|
|
// our 8-pt Sans Serif font (wfe_pFont)
|
|
// This might be a problem for some Asian fonts if they are taller?
|
|
#define COLOR_SWATCH_HEIGHT 16
|
|
#define COLOR_SWATCH_WIDTH 18
|
|
#define COLOR_SWATCH_SPACING 0
|
|
#define MAIN_COLORS_TOP (2*COLOR_SWATCH_HEIGHT + 4)
|
|
#define LAST_COLOR_TOP (COLOR_SWATCH_HEIGHT + 4)
|
|
#define COLOR_PICKER_HEIGHT (((COLOR_ROWS + 7)*COLOR_SWATCH_HEIGHT) + 26)
|
|
#define COLOR_PICKER_WIDTH (COLOR_COLS*COLOR_SWATCH_WIDTH + 7)
|
|
#define BUTTON_WIDTH (4*COLOR_SWATCH_WIDTH - 4)
|
|
|
|
// Index to the first USER custom color, which follows the Netscape colors
|
|
// (first custom color, the "last-used" is actually 0 in color array)
|
|
// Add 1 for the Last-Used color swatch above main color grid
|
|
// and another for the "Current color" swatch above quick palette
|
|
#define FIRST_CUSTOM_COLOR_INDEX (MAX_NS_COLORS + 2)
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////
|
|
// CColorPicker Widget for picking colors
|
|
// We can't use Pretranslate or SetCapture under Win16,
|
|
// so we must derive from CWnd, not CDialog
|
|
IMPLEMENT_DYNAMIC(CColorPicker,CWnd);
|
|
|
|
CColorPicker::CColorPicker(CWnd * pParent,
|
|
MWContext * pMWContext,
|
|
COLORREF crCurrentColor,
|
|
COLORREF crDefColor,
|
|
UINT nIDCaption,
|
|
RECT * pCallerRect)
|
|
: CWnd(),
|
|
m_pParent(pParent),
|
|
m_pMWContext(pMWContext),
|
|
m_crCurrentColor(crCurrentColor),
|
|
m_crDefColor(crDefColor),
|
|
m_nIDCaption(nIDCaption),
|
|
m_pToolTip(0),
|
|
m_bOtherDown(0),
|
|
m_bAutoDown(0),
|
|
m_bHelpDown(0),
|
|
m_bRunning(TRUE),
|
|
m_bFirstMouseUp(TRUE),
|
|
m_bMouseDown(FALSE),
|
|
m_crDragColor(0)
|
|
{
|
|
// Caption choices are: "Text Color",
|
|
// or "[Page | Table | Cell] Background"
|
|
m_bBackground = (nIDCaption != IDS_TEXT_COLOR);
|
|
|
|
POINT ptOrigin;
|
|
|
|
if( pCallerRect )
|
|
{
|
|
m_CallerRect = *pCallerRect;
|
|
} else {
|
|
m_CallerRect.left = m_CallerRect.right = m_CallerRect.top = m_CallerRect.bottom = 0;
|
|
}
|
|
m_LoColor.red = m_LoColor.green = m_LoColor.blue = 0;
|
|
|
|
// If current is DEFAULT (i.e. "Automatic"), get appropriate default color
|
|
if( m_crCurrentColor == DEFAULT_COLORREF )
|
|
{
|
|
if( crDefColor == DEFAULT_COLORREF || crDefColor == MIXED_COLORREF )
|
|
{
|
|
// Get the "Default" color used by the Browser
|
|
m_crCurrentColor = prefInfo.m_rgbForegroundColor;
|
|
} else if( crDefColor == BACKGROUND_COLORREF )
|
|
{
|
|
// Get current page's background color
|
|
// as "default" for table backgrounds
|
|
CDCCX *pDC = VOID2CX(pMWContext->fe.cx, CDCCX);
|
|
m_crCurrentColor = pDC->m_rgbBackgroundColor;
|
|
} else
|
|
{
|
|
// Use the default color supplied
|
|
m_crCurrentColor = crDefColor;
|
|
}
|
|
}
|
|
|
|
if( m_CallerRect.left == 0 && m_CallerRect.top == 0 )
|
|
{
|
|
// No caller rect supplied - locate top at cursor Y and
|
|
// X so cursor pt. is at horizontal center
|
|
GetCursorPos(&ptOrigin);
|
|
ptOrigin.x -= (ED_TB_BUTTON_WIDTH / 2);
|
|
} else {
|
|
// The -1 lines up light highlight with that of parent
|
|
ptOrigin.x = m_CallerRect.left - 1;
|
|
// We may change this below if near bottom of screen
|
|
ptOrigin.y = m_CallerRect.bottom;
|
|
}
|
|
|
|
if (!CWnd::CreateEx(WS_EX_TOPMOST,
|
|
AfxRegisterWndClass(CS_SAVEBITS | CS_VREDRAW,
|
|
::LoadCursor( NULL, IDC_ARROW),
|
|
sysInfo.m_hbrBtnFace),
|
|
NULL, WS_POPUP /*| WS_VISIBLE*/ | WS_DLGFRAME,
|
|
ptOrigin.x, ptOrigin.y,
|
|
COLOR_PICKER_WIDTH, COLOR_PICKER_HEIGHT,
|
|
pParent->m_hWnd, NULL, NULL))
|
|
{
|
|
TRACE0("Warning: creation of CColorPicker window failed\n");
|
|
return;
|
|
}
|
|
|
|
SetFont(wfe_pFont);
|
|
|
|
// Add a tooltip control
|
|
m_pToolTip = new CNSToolTip2;
|
|
|
|
if(m_pToolTip && !m_pToolTip->Create(this, TTS_ALWAYSTIP) )
|
|
{
|
|
TRACE("Unable To create ToolTip\n");
|
|
delete m_pToolTip;
|
|
m_pToolTip = NULL;
|
|
} else {
|
|
// Lets use speedy tooltips
|
|
m_pToolTip->SetDelayTime(200);
|
|
#ifdef XP_WIN32
|
|
// We MUST do this for MFC tooltips
|
|
EnableToolTips(TRUE);
|
|
#endif // WIN32
|
|
}
|
|
|
|
int iTop = 2;
|
|
int iCustomTop;
|
|
int iLeft = 4;
|
|
int iCol = 0;
|
|
int iRow = 0;
|
|
int iRowLimit = COLOR_ROWS - 1; // Doesn't include custom colors
|
|
//int iXPOffset = 0;
|
|
int iCustomColor = 0; // Counter for custom colors
|
|
|
|
RECT rectOther; // For the "Other..." button
|
|
RECT rect;
|
|
RECT rectLabel;
|
|
char *pLabel; // Holds static pointer for szLoadString (dont free)
|
|
|
|
for( int i = 0; i < MAX_COLORS; i++ )
|
|
{
|
|
LO_Color LoColor;
|
|
|
|
if( i > 0 && i <= MAX_NS_COLORS ) // in main color grid
|
|
{
|
|
// Reset to first column
|
|
if( i == 1 )
|
|
{
|
|
iCol = 0;
|
|
}
|
|
|
|
// Set top to be used for all colors in main region
|
|
iTop = MAIN_COLORS_TOP;
|
|
EDT_GetNSColor(i-1, &LoColor);
|
|
m_crColors[i] = RGB(LoColor.red, LoColor.green, LoColor.blue);
|
|
}
|
|
else
|
|
{
|
|
if( i == 0 )
|
|
{
|
|
// The first swatch is the "last-picked" color
|
|
if( m_bBackground )
|
|
m_crColors[i] = wfe_crLastBkgrndColorPicked;
|
|
else
|
|
m_crColors[i] = wfe_crLastColorPicked;
|
|
|
|
// "Last-picked color" at left margin, below caption area
|
|
pLabel = szLoadString(IDS_LAST_USED_COLOR);
|
|
rectLabel.top = LAST_COLOR_TOP+2;
|
|
rectLabel.left = COLOR_SWATCH_WIDTH + 2;
|
|
rectLabel.right = COLOR_PICKER_WIDTH;
|
|
rectLabel. bottom = rectLabel.top + COLOR_SWATCH_HEIGHT;
|
|
if( !m_LastUsedLabel.Create(pLabel, WS_VISIBLE | WS_CHILD, rectLabel, this) )
|
|
return;
|
|
|
|
// Set the top of the color swatch
|
|
iTop = LAST_COLOR_TOP;
|
|
}
|
|
else if( i == MAX_NS_COLORS+1 ) // Should = FIRST_CUSTOM_COLOR_INDEX-1
|
|
{
|
|
// Current color swatch just below the main colors
|
|
m_crColors[i] = m_crCurrentColor;
|
|
|
|
// Set text label next to this color swatch
|
|
pLabel = szLoadString(IDS_CURRENT_COLOR);
|
|
|
|
iTop = MAIN_COLORS_TOP + (COLOR_ROWS * COLOR_SWATCH_HEIGHT) + 2;
|
|
rectLabel.top = iTop + 1;
|
|
rectLabel.left = COLOR_SWATCH_WIDTH + 2;
|
|
rectLabel.right = COLOR_PICKER_WIDTH - 6 - BUTTON_WIDTH;
|
|
rectLabel. bottom = rectLabel.top + COLOR_SWATCH_HEIGHT;
|
|
if( !m_CurrentLabel.Create(pLabel, WS_VISIBLE | WS_CHILD, rectLabel, this) )
|
|
return;
|
|
iCol = 0;
|
|
iRow = 0;
|
|
}
|
|
else if( i >= FIRST_CUSTOM_COLOR_INDEX )
|
|
{
|
|
// Get custom colors from global palette
|
|
m_crColors[i] = wfe_CustomPalette[iCustomColor++];
|
|
|
|
if( i == FIRST_CUSTOM_COLOR_INDEX )
|
|
{
|
|
// Add the "Automatic" button ( = "Default", i.e., no color in HTML)
|
|
// at the same vertical location as "Current color" swatch
|
|
// (Must create here to get tab order correct)
|
|
rectOther.top = iTop + 2;
|
|
rectOther.bottom = rectOther.top + 22;
|
|
rectOther.right = COLOR_PICKER_WIDTH - 8;
|
|
rectOther.left = rectOther.right - BUTTON_WIDTH;
|
|
|
|
m_OtherButton.Create(szLoadString(IDS_OTHER_BUTTON), BS_PUSHBUTTON|WS_CHILD|WS_VISIBLE|WS_GROUP|WS_TABSTOP,
|
|
rectOther, this, IDC_CHOOSE_COLOR);
|
|
|
|
// TODO: ADD A TOOLTIP FOR "AUTO" BUTTON for added instructions?
|
|
|
|
// Label above custom colors:
|
|
pLabel = szLoadString(IDS_CUSTOM_COLORS_LABEL);
|
|
rectLabel.left = 2;
|
|
rectLabel.right = rectOther.left - 1;
|
|
rectLabel.top = iTop + COLOR_SWATCH_HEIGHT + 2;
|
|
// We cut 2 pixels off font height for tighter fit (TODO: PROBLEM IN OTHER LANGUAGES?)
|
|
rectLabel.bottom = rectLabel.top + wfe_iFontHeight - 2;
|
|
if( !m_CustomColorsLabel.Create(pLabel, WS_VISIBLE | WS_CHILD, rectLabel, this) )
|
|
return;
|
|
|
|
// Top of custom colors row - place just below label
|
|
iTop = iCustomTop = rectLabel.bottom;
|
|
iCol = 0;
|
|
iRow = 0;
|
|
}
|
|
}
|
|
LoColor.red = GetRValue(m_crColors[i]);
|
|
LoColor.green = GetGValue(m_crColors[i]);
|
|
LoColor.blue = GetBValue(m_crColors[i]);
|
|
}
|
|
|
|
rect.left = iCol*COLOR_SWATCH_WIDTH;
|
|
rect.top = iTop + (iRow*COLOR_SWATCH_HEIGHT);
|
|
rect.right = rect.left + COLOR_SWATCH_WIDTH;
|
|
rect.bottom = rect.top + COLOR_SWATCH_HEIGHT;
|
|
|
|
// Create our custom buttons
|
|
m_pColorButtons[i] = new CColorButton(&m_crColors[i], &m_crColor);
|
|
if( !m_pColorButtons[i] ||
|
|
!m_pColorButtons[i]->Create(rect, this, IDC_LAST_USED_COLOR+i, &m_crColors[i], TRUE) )
|
|
{
|
|
return;
|
|
}
|
|
// Add tooltip showing color formated as "R=xxx G=xxx B=xxx HTML: #FFEEAA"
|
|
if( m_pToolTip )
|
|
{
|
|
if( m_crColors[i] == DEFAULT_COLORREF )
|
|
{
|
|
*m_ppTipText[i] = 0;
|
|
} else
|
|
{
|
|
wsprintf(m_ppTipText[i], szLoadString(IDS_COLOR_TIP_FORMAT),
|
|
LoColor.red, LoColor.green, LoColor.blue);
|
|
strcat(m_ppTipText[i], szLoadString(IDS_COLOR_TIP_HTML));
|
|
char * pEnd = m_ppTipText[i] + strlen(m_ppTipText[i]);
|
|
sprintf(pEnd, "#%02X%02X%02X",LoColor.red, LoColor.green, LoColor.blue);
|
|
|
|
m_pToolTip->AddTool(m_pColorButtons[i], m_ppTipText[i], &rect, IDC_LAST_USED_COLOR+i);
|
|
}
|
|
}
|
|
|
|
if( i >= FIRST_CUSTOM_COLOR_INDEX )
|
|
{
|
|
// We are within single custom color row
|
|
iCol++;
|
|
}
|
|
else if( i > 0 )
|
|
{
|
|
// Main color grid
|
|
// Add each button in columns first, add new column after every 7
|
|
if(iRow < iRowLimit)
|
|
{
|
|
iRow++;
|
|
} else {
|
|
iRow = 0;
|
|
iCol++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add Number strings under custom color swatches
|
|
rect.left = 6;
|
|
rect.top = iCustomTop + COLOR_SWATCH_HEIGHT;
|
|
rect.bottom = rect.top + COLOR_SWATCH_HEIGHT;
|
|
|
|
for( i = 0; i < MAX_CUSTOM_COLORS; i++ )
|
|
{
|
|
rect.right = rect.left + COLOR_SWATCH_WIDTH;
|
|
char pNumber[16];
|
|
wsprintf(pNumber, "%d", (i < MAX_CUSTOM_COLORS-1) ? i+1 : 0);
|
|
if( !m_CustomColorNumber[i].Create(pNumber, WS_VISIBLE | WS_CHILD, rect, this) )
|
|
return;
|
|
// Move to next location
|
|
rect.left += COLOR_SWATCH_WIDTH;
|
|
}
|
|
|
|
// Place "Browser default" button in lower left corner:
|
|
rect.top = rect.bottom;
|
|
rect.bottom = rect.top + 22;
|
|
rect.left = 2;
|
|
rect.right = rect.left + BUTTON_WIDTH + 2*COLOR_SWATCH_WIDTH; // BUTTON_WIDTH
|
|
|
|
m_DefaultButton.Create(szLoadString(IDS_DEFAULT_BUTTON), BS_PUSHBUTTON|WS_CHILD|WS_VISIBLE|WS_GROUP|WS_TABSTOP,
|
|
rect, this, IDC_DEFAULT_COLOR);
|
|
|
|
// Now we can determine bottom of entire window
|
|
// Number added to this must be enough for 2 lines of text (this is now 30)
|
|
// This is weird, but adding about 4 results in same place as "Other" button button,
|
|
// seems like button is larger than 24???
|
|
int iTotalHeight = rect.bottom + 8;
|
|
|
|
// Place "Help" color button in lower right corner:
|
|
rect.right = COLOR_PICKER_WIDTH - 8;
|
|
rect.left = rect.right - BUTTON_WIDTH;
|
|
// NOTE: IDS_HELP_BUTTON string is in Browser resources
|
|
m_HelpButton.Create(szLoadString(IDS_HELP_BUTTON), BS_PUSHBUTTON|WS_CHILD|WS_VISIBLE|WS_GROUP|WS_TABSTOP,
|
|
rect, this, IDC_COLOR_HELP);
|
|
|
|
// Set our font for all static strings used
|
|
if( wfe_pFont )
|
|
{
|
|
|
|
// Set the font to our usual 8-pt font
|
|
m_OtherButton.SetFont(wfe_pFont);
|
|
m_HelpButton.SetFont(wfe_pFont);
|
|
m_LastUsedLabel.SetFont(wfe_pFont);
|
|
m_CurrentLabel.SetFont(wfe_pFont);
|
|
m_DefaultButton.SetFont(wfe_pFont);
|
|
m_CustomColorsLabel.SetFont(wfe_pFont);
|
|
for( int i = 0; i < MAX_CUSTOM_COLORS; i++ )
|
|
{
|
|
m_CustomColorNumber[i].SetFont(wfe_pFont);
|
|
}
|
|
}
|
|
|
|
// Check if we need to move up because we would go off the screen
|
|
if( (m_CallerRect.bottom + iTotalHeight) > sysInfo.m_iScreenHeight )
|
|
{
|
|
// Locate toolbar above the caller button
|
|
// if toolbar bottom would extend off bottom of screen
|
|
ptOrigin.y = m_CallerRect.top - iTotalHeight;
|
|
}
|
|
// Move and/or resize the window and show it for the first time
|
|
SetWindowPos(NULL, ptOrigin.x, ptOrigin.y, COLOR_PICKER_WIDTH, iTotalHeight, SWP_SHOWWINDOW );
|
|
|
|
// Capture mouse so we can detect clicking off the dialog
|
|
SetCapture();
|
|
|
|
// Snap focus to the first button
|
|
// Needed to allow Esc key to work
|
|
// after click down/up on the control that launched us
|
|
m_pColorButtons[0]->SetFocus();
|
|
m_crColor = m_crColors[0];
|
|
|
|
// Initialize the array of flags for Quick Palette
|
|
for ( i = 0; i < MAX_CUSTOM_COLORS; i++ )
|
|
m_bColorChanged[i] = FALSE;
|
|
}
|
|
|
|
void CColorPicker::OnPaint()
|
|
{
|
|
CPaintDC dc(this);
|
|
|
|
// Draw the caption text
|
|
int iRight = COLOR_PICKER_WIDTH-8;
|
|
CRect cRect(2,2,iRight-1, COLOR_SWATCH_HEIGHT);
|
|
dc.SetBkMode(TRANSPARENT);
|
|
dc.SetTextColor(sysInfo.m_clrBtnText);
|
|
CFont *pOldFont = dc.SelectObject(wfe_pFont);
|
|
CString cString(szLoadString(m_nIDCaption));
|
|
dc.DrawText(cString, cRect, DT_CENTER);
|
|
dc.SelectObject(pOldFont);
|
|
|
|
// Draw a depressed rect around caption
|
|
|
|
// This doesn't draw very well - it uses right-1 and bottom-1,
|
|
// and the lower-left corner doesn't look good
|
|
// WFE_DrawHighlight( dc.m_hDC, LPRECT(cRect), sysInfo.m_clrBtnShadow, sysInfo.m_clrBtnHilite );
|
|
|
|
CPen cPenShadow(PS_SOLID, 1, sysInfo.m_clrBtnShadow);
|
|
CPen cPenHilite(PS_SOLID, 1, sysInfo.m_clrBtnHilite);
|
|
CPen *pPenOld = (CPen*)dc.SelectObject(&cPenShadow);
|
|
int iBottom = COLOR_SWATCH_HEIGHT+ (wfe_iFontHeight > 16 ? 2: 1);
|
|
dc.MoveTo(1,iBottom);
|
|
dc.LineTo(1,1);
|
|
dc.LineTo(iRight,1);
|
|
dc.SelectObject(&cPenHilite);
|
|
dc.LineTo(iRight,iBottom);
|
|
dc.LineTo(1,iBottom);
|
|
dc.SelectObject(pPenOld);
|
|
}
|
|
|
|
COLORREF crReturnColor;
|
|
|
|
// Call this instead of DoModal() to wait in dialog and get result
|
|
COLORREF CColorPicker::GetColor(LO_Color * pLoColor)
|
|
{
|
|
|
|
// Wait here until user selects a color or
|
|
// cancels the dialog with ESC or by clicking outside window
|
|
while( m_bRunning ) FEU_StayingAlive();
|
|
|
|
// Return selected or custom color
|
|
if( pLoColor )
|
|
{
|
|
pLoColor->red = m_LoColor.red;
|
|
pLoColor->green = m_LoColor.green;
|
|
pLoColor->blue = m_LoColor.blue;
|
|
}
|
|
crReturnColor = m_crColor;
|
|
DestroyWindow();
|
|
return crReturnColor;
|
|
}
|
|
|
|
void CColorPicker::SetColorAndExit()
|
|
{
|
|
if( m_crColor != DEFAULT_COLORREF )
|
|
{
|
|
m_LoColor.red = GetRValue(m_crColor);
|
|
m_LoColor.green = GetGValue(m_crColor);
|
|
m_LoColor.blue = GetBValue(m_crColor);
|
|
}
|
|
// Set flag that will let us return to caller and destroy dialog
|
|
m_bRunning = FALSE;
|
|
}
|
|
|
|
void CColorPicker::CancelAndExit()
|
|
{
|
|
m_crColor = CANCEL_COLORREF;
|
|
m_bRunning = FALSE;
|
|
}
|
|
|
|
CColorPicker::~CColorPicker()
|
|
{
|
|
}
|
|
|
|
void CColorPicker::PostNcDestroy()
|
|
{
|
|
char pPref[32];
|
|
char pColorString[32];
|
|
|
|
// Save the last-picked color to appropriate pref
|
|
if( m_bBackground )
|
|
{
|
|
wsprintf(pColorString, "%d,%d,%d", GetRValue(wfe_crLastBkgrndColorPicked), GetGValue(wfe_crLastBkgrndColorPicked), GetBValue(wfe_crLastBkgrndColorPicked));
|
|
PREF_SetCharPref("editor.last_background_color_picked", pColorString);
|
|
} else {
|
|
wsprintf(pColorString, "%d,%d,%d", GetRValue(wfe_crLastColorPicked), GetGValue(wfe_crLastColorPicked), GetBValue(wfe_crLastColorPicked));
|
|
PREF_SetCharPref("editor.last_color_picked", pColorString);
|
|
}
|
|
|
|
// Save custom colors back to prefs
|
|
for ( int i = 0; i < MAX_CUSTOM_COLORS; i++ )
|
|
{
|
|
// Write new pref only if different than original value
|
|
if( m_bColorChanged[i] )
|
|
{
|
|
COLORREF crColor = wfe_CustomPalette[i]; // should = m_crColors[iColorIndex];
|
|
wsprintf(pColorString, "%d,%d,%d", GetRValue(crColor), GetGValue(crColor), GetBValue(crColor));
|
|
wsprintf(pPref, "editor.custom_color_%d", i);
|
|
PREF_SetCharPref(pPref, pColorString);
|
|
}
|
|
}
|
|
|
|
if( m_pToolTip )
|
|
delete m_pToolTip;
|
|
|
|
ReleaseCapture();
|
|
CWnd::PostNcDestroy();
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CColorPicker, CWnd)
|
|
//{{AFX_MSG_MAP(CColorPicker)
|
|
ON_WM_LBUTTONUP()
|
|
ON_WM_LBUTTONDOWN()
|
|
ON_WM_MOUSEMOVE()
|
|
ON_WM_PAINT()
|
|
//}}AFX_MSG_MAP
|
|
#ifdef XP_WIN32
|
|
ON_NOTIFY_EX( TTN_NEEDTEXT, 0, OnToolTipNotify )
|
|
#endif
|
|
END_MESSAGE_MAP()
|
|
|
|
BOOL CColorPicker::PreTranslateMessage(MSG* pMsg)
|
|
{
|
|
if( m_pToolTip && pMsg->message >= WM_MOUSEFIRST && pMsg->message <= WM_MOUSELAST)
|
|
{
|
|
m_pToolTip->RelayEvent(pMsg);
|
|
}
|
|
if( pMsg->message == WM_KEYDOWN )
|
|
{
|
|
if( pMsg->wParam == VK_ESCAPE )
|
|
{
|
|
CancelAndExit();
|
|
}
|
|
else if( pMsg->wParam == VK_RETURN )
|
|
{
|
|
SetColorAndExit();
|
|
}
|
|
else if( pMsg->wParam == VK_F1 )
|
|
{
|
|
OnColorHelp();
|
|
}
|
|
else
|
|
{
|
|
// Are any number keys pressed?
|
|
// This is 1 less than number on key pressed
|
|
int iNumberIndex = pMsg->wParam - '1';
|
|
// "0" key is AFTER "9" on keyboard
|
|
// so treat it like a 10th color
|
|
if( iNumberIndex == -1 )
|
|
iNumberIndex = 9;
|
|
|
|
if( iNumberIndex >= 0 && iNumberIndex <= 9 ) // This won't work if MAX_CUSTOM_COLORS > 10
|
|
{
|
|
// Set the current focus color at the custom color matching number
|
|
m_crColors[FIRST_CUSTOM_COLOR_INDEX+iNumberIndex] = m_crColor;
|
|
m_pColorButtons[FIRST_CUSTOM_COLOR_INDEX+iNumberIndex]->Invalidate(FALSE);
|
|
}
|
|
}
|
|
}
|
|
return CWnd::PreTranslateMessage(pMsg);
|
|
}
|
|
|
|
BOOL CColorPicker::IsMouseOverDlg(CPoint cPoint)
|
|
{
|
|
CRect cRect;
|
|
GetWindowRect(&cRect);
|
|
ScreenToClient(&cRect);
|
|
return cRect.PtInRect(cPoint);
|
|
}
|
|
|
|
BOOL CColorPicker::IsMouseOverButton(CPoint cPoint, UINT nID)
|
|
{
|
|
CRect cRect;
|
|
((CButton*)GetDlgItem(nID))->GetWindowRect(&cRect);
|
|
ScreenToClient(&cRect);
|
|
return cRect.PtInRect(cPoint);
|
|
}
|
|
|
|
int CColorPicker::GetMouseOverColorIndex(CPoint cPoint)
|
|
{
|
|
CRect cRect;
|
|
for( int i = 0; i < MAX_COLORS; i++){
|
|
GetDlgItem(IDC_LAST_USED_COLOR+i)->GetWindowRect(&cRect);
|
|
ScreenToClient(&cRect);
|
|
if( cRect.PtInRect(cPoint) ){
|
|
// Select the color and return the index
|
|
// unless it is "default" (unknown)
|
|
if( m_crColors[i] != DEFAULT_COLORREF )
|
|
{
|
|
m_crColor = m_crColors[i];
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void CColorPicker::OnLButtonDown(UINT nFlags, CPoint cPoint)
|
|
{
|
|
POINT pt = {cPoint.x,cPoint.y};
|
|
// If user clicks outside of dialog, we are done
|
|
if( !IsMouseOverDlg(cPoint) ){
|
|
CancelAndExit();
|
|
return;
|
|
}
|
|
|
|
if( IsMouseOverButton(cPoint, IDC_CHOOSE_COLOR) && !m_bOtherDown ){
|
|
m_bOtherDown = TRUE;
|
|
((CButton*)GetDlgItem(IDC_CHOOSE_COLOR))->SetState(TRUE);
|
|
} else if( IsMouseOverButton(cPoint, IDC_DEFAULT_COLOR) && !m_bAutoDown ){
|
|
m_bAutoDown = TRUE;
|
|
((CButton*)GetDlgItem(IDC_DEFAULT_COLOR))->SetState(TRUE);
|
|
} else if( IsMouseOverButton(cPoint, IDC_COLOR_HELP) && !m_bHelpDown ){
|
|
m_bHelpDown = TRUE;
|
|
((CButton*)GetDlgItem(IDC_COLOR_HELP))->SetState(TRUE);
|
|
}
|
|
|
|
// Copy color for possible dragging
|
|
int i = GetMouseOverColorIndex(cPoint);
|
|
if( i >= 0 ){
|
|
m_crDragColor = m_crColors[i];
|
|
m_bMouseDown = TRUE;
|
|
m_iMouseDownColorIndex = i;
|
|
|
|
SetCursor(theApp.LoadCursor(IDC_DRAG_COLOR));
|
|
}
|
|
CWnd::OnLButtonDown(nFlags, cPoint);
|
|
}
|
|
|
|
void CColorPicker::OnLButtonUp(UINT nFlags, CPoint cPoint)
|
|
{
|
|
// Simulate how a combobox widget works,
|
|
// quit if mouse up is off the window, except if it comes up
|
|
// over the control that launched us
|
|
if( !IsMouseOverDlg(cPoint) )
|
|
{
|
|
// But check if over the caller rect
|
|
CRect cRect(m_CallerRect);
|
|
ScreenToClient(&cRect);
|
|
if( !cRect.PtInRect(cPoint) )
|
|
{
|
|
CancelAndExit();
|
|
return;
|
|
}
|
|
// Mouse came up over the caller control,
|
|
// so default to last color used
|
|
m_crColor = m_crColors[0];
|
|
} else {
|
|
// Detect mouse up on a color swatch select it
|
|
// or drop copied color
|
|
int i = GetMouseOverColorIndex(cPoint);
|
|
|
|
// Ignore first mouse up on current color
|
|
if( i == 0 && m_bFirstMouseUp )
|
|
return;
|
|
|
|
m_bFirstMouseUp = FALSE;
|
|
|
|
if( i >= 0 )
|
|
{
|
|
if( m_bMouseDown && i >= FIRST_CUSTOM_COLOR_INDEX &&
|
|
m_iMouseDownColorIndex != i) // TRUE when click down/up to select quick color
|
|
{
|
|
// We are dropping a color onto custom palette
|
|
m_crColors[i] = m_crDragColor;
|
|
int iCustomIndex = i - FIRST_CUSTOM_COLOR_INDEX;
|
|
wfe_CustomPalette[iCustomIndex] = m_crDragColor;
|
|
m_bColorChanged[iCustomIndex] = TRUE;
|
|
|
|
// Force redraw
|
|
m_pColorButtons[i]->Invalidate(FALSE);
|
|
} else if( !m_bMouseDown || m_iMouseDownColorIndex == i )
|
|
{
|
|
// Button came up in same color as mouse down,
|
|
// or up when not dragging is not possible
|
|
// because mouse was already down when picker started
|
|
m_crColor = m_crColors[i];
|
|
// Always save selected color as the "last color picked"
|
|
if( m_bBackground )
|
|
wfe_crLastBkgrndColorPicked = m_crColor;
|
|
else
|
|
wfe_crLastColorPicked = m_crColor;
|
|
|
|
SetColorAndExit();
|
|
return;
|
|
}
|
|
// We dropped color or didn't select,
|
|
// but we must be over a color swatch,
|
|
// so set appropriate cursor
|
|
SetCursor(theApp.LoadCursor(IDC_DRAG_ARROW));
|
|
}
|
|
|
|
if( IsMouseOverButton(cPoint, IDC_CHOOSE_COLOR) )
|
|
{
|
|
OnChooseColor();
|
|
return;
|
|
}
|
|
if( IsMouseOverButton(cPoint, IDC_DEFAULT_COLOR) )
|
|
{
|
|
OnDefaultColor();
|
|
return;
|
|
}
|
|
if( IsMouseOverButton(cPoint, IDC_COLOR_HELP) )
|
|
{
|
|
((CButton*)GetDlgItem(IDC_COLOR_HELP))->SetState(FALSE);
|
|
m_bHelpDown = FALSE;
|
|
OnColorHelp();
|
|
}
|
|
}
|
|
m_bMouseDown = FALSE;
|
|
CWnd::OnLButtonUp(nFlags, cPoint);
|
|
}
|
|
|
|
void CColorPicker::SetButtonState(CPoint cPoint, UINT nID, BOOL* pButtonDown )
|
|
{
|
|
BOOL bOverButton = IsMouseOverButton(cPoint, nID);
|
|
|
|
if( !bOverButton && *pButtonDown)
|
|
{
|
|
// Deselect button when mouse moves off
|
|
*pButtonDown = FALSE;
|
|
((CButton*)GetDlgItem(nID))->SetState(FALSE);
|
|
} else if( bOverButton && !*pButtonDown /*&& (nFlags & MK_LBUTTON)*/ )
|
|
{
|
|
*pButtonDown = TRUE;
|
|
((CButton*)GetDlgItem(nID))->SetState(TRUE);
|
|
}
|
|
}
|
|
|
|
void CColorPicker::OnMouseMove(UINT nFlags, CPoint cPoint)
|
|
{
|
|
int i = GetMouseOverColorIndex(cPoint);
|
|
if( i != 0 )
|
|
{
|
|
// If not over Current Color button,
|
|
// kill ignore-on-first-mouseup behavior
|
|
m_bFirstMouseUp = FALSE;
|
|
}
|
|
|
|
// Set button shade state to act like normal dialog button
|
|
// when mouse is held down
|
|
if( nFlags & MK_LBUTTON )
|
|
{
|
|
SetButtonState(cPoint, IDC_CHOOSE_COLOR, &m_bOtherDown);
|
|
SetButtonState(cPoint, IDC_DEFAULT_COLOR, &m_bAutoDown);
|
|
SetButtonState(cPoint, IDC_COLOR_HELP, &m_bHelpDown);
|
|
}
|
|
|
|
if( m_bMouseDown )
|
|
{
|
|
// Give user feedback via cursor
|
|
UINT nIDC;
|
|
if( m_iMouseDownColorIndex == i )
|
|
// Over source swatch - NULL icon is confusing - show arrow + color box
|
|
nIDC = IDC_DRAG_COLOR;
|
|
else if( i >= FIRST_CUSTOM_COLOR_INDEX )
|
|
// Over drop target
|
|
nIDC = IDC_DROP_COLOR;
|
|
else
|
|
// Over anything else - can't drop
|
|
nIDC = IDC_NO_DROP;
|
|
|
|
SetCursor(theApp.LoadCursor(nIDC));
|
|
} else {
|
|
if( i >= 0 )
|
|
{
|
|
// Tells user they can drag a color
|
|
SetCursor(theApp.LoadCursor(IDC_DRAG_ARROW));
|
|
if( GetFocus() != m_pColorButtons[i] )
|
|
{
|
|
// Bring focus to color where mouse is
|
|
m_pColorButtons[i]->SetFocus();
|
|
m_crColor = m_crColors[i];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Not over color or dragging - regular cursor
|
|
SetCursor(theApp.LoadStandardCursor(IDC_ARROW));
|
|
}
|
|
}
|
|
CWnd::OnMouseMove(nFlags, cPoint);
|
|
}
|
|
|
|
//
|
|
#ifdef XP_WIN32
|
|
// This is needed for non-CFrame owners of a CToolTipCtrl
|
|
BOOL CColorPicker::OnToolTipNotify( UINT id, NMHDR * pNMHDR, LRESULT * pResult )
|
|
{
|
|
TOOLTIPTEXT *pTTT = (TOOLTIPTEXT *)pNMHDR;
|
|
UINT nID =pNMHDR->idFrom;
|
|
if (pTTT->uFlags & TTF_IDISHWND)
|
|
{
|
|
// idFrom is actually the HWND of the tool
|
|
nID = ::GetDlgCtrlID((HWND)nID);
|
|
if( nID ) {
|
|
int i = nID - IDC_LAST_USED_COLOR;
|
|
if( i >= 0 && i < MAX_COLORS && m_ppTipText[i] ){
|
|
// Copy string into buffer - limit is 80 characters
|
|
strcpy( pTTT->szText, m_ppTipText[i]);
|
|
return(TRUE);
|
|
}
|
|
}
|
|
}
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
void CColorPicker::OnDefaultColor()
|
|
{
|
|
m_crColor = DEFAULT_COLORREF;
|
|
|
|
SetColorAndExit();
|
|
}
|
|
|
|
// Select any color via Window's color dialog
|
|
void CColorPicker::OnChooseColor()
|
|
{
|
|
// Use CC_PREVENTFULLOPEN in flags (2nd) param if we want to suppress editing
|
|
CColorDialog dlg(wfe_CustomPalette[0], CC_FULLOPEN, this);
|
|
|
|
// Place our current palette into the 16 Custom colors
|
|
// Copy colors so we don't change anything if
|
|
COLORREF crCustomColors[16];
|
|
memcpy((void*)crCustomColors, (void*)wfe_CustomPalette, sizeof(crCustomColors));
|
|
|
|
dlg.m_cc.lpCustColors = crCustomColors;
|
|
dlg.m_cc.lStructSize = sizeof( dlg.m_cc );
|
|
|
|
// TODO: We need to make a derived class to control common dialog palette
|
|
// or do hook for WM_INTDIALOG in m_cc struct
|
|
// so we can set the palette of the Window's CColorDlg
|
|
|
|
UINT nResult = dlg.DoModal();
|
|
|
|
if( nResult == IDOK ){
|
|
m_crColor = dlg.GetColor();
|
|
// Save this color as last color picked
|
|
if( m_bBackground )
|
|
wfe_crLastBkgrndColorPicked = m_crColor;
|
|
else
|
|
wfe_crLastColorPicked = m_crColor;
|
|
|
|
// Set flags for colors that changed
|
|
for ( int i = 0; i < MAX_CUSTOM_COLORS; i++ )
|
|
{
|
|
if( wfe_CustomPalette[i] != crCustomColors[i] )
|
|
m_bColorChanged[i] = TRUE;
|
|
}
|
|
// Copy custom colors back to global palette
|
|
memcpy((void*)wfe_CustomPalette, (void*)crCustomColors, sizeof(crCustomColors));
|
|
|
|
SetColorAndExit();
|
|
} else {
|
|
CancelAndExit();
|
|
}
|
|
}
|
|
|
|
void CColorPicker::OnColorHelp()
|
|
{
|
|
NetHelp(HELP_COLOR_PICKER);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CDropdownToolbar Popup window with thin border to place a vertical row of buttons
|
|
|
|
CDropdownToolbar::CDropdownToolbar(CWnd * pParent,
|
|
MWContext * pMWContext,
|
|
RECT * pCallerRect,
|
|
UINT nCallerID,
|
|
UINT nInitialID)
|
|
: CWnd(),
|
|
m_pParent(pParent),
|
|
m_pMWContext(pMWContext),
|
|
m_nCallerID(nCallerID),
|
|
m_nInitialID(nInitialID),
|
|
m_nButtonCount(0),
|
|
m_nAllocatedCount(0),
|
|
m_pData(NULL),
|
|
m_pToolTip(0),
|
|
m_bFirstButtonUp(TRUE)
|
|
{
|
|
if( pCallerRect ){
|
|
m_CallerRect = *pCallerRect;
|
|
} else {
|
|
m_CallerRect.left = m_CallerRect.right = m_CallerRect.top = m_CallerRect.bottom = 0;
|
|
}
|
|
|
|
if (!CWnd::CreateEx(WS_EX_TOPMOST, AfxRegisterWndClass(CS_SAVEBITS | CS_VREDRAW, ::LoadCursor( NULL, IDC_ARROW)),
|
|
NULL, WS_POPUP | WS_BORDER /*|WS_VISIBLE|WS_CLIPCHILDREN|WS_CLIPSIBLINGS*/,
|
|
0, 0, 0, 0, pParent->m_hWnd, NULL, NULL))
|
|
{
|
|
TRACE0("Warning: creation of CDropdownToolbar window failed\n");
|
|
return;
|
|
}
|
|
|
|
// Add a tooltip control
|
|
m_pToolTip = new CNSToolTip2;
|
|
|
|
if(m_pToolTip && !m_pToolTip->Create(this, TTS_ALWAYSTIP) ){
|
|
TRACE("Unable To create ToolTip\n");
|
|
delete m_pToolTip;
|
|
m_pToolTip = NULL;
|
|
}
|
|
if( m_pToolTip ){
|
|
m_pToolTip->Activate(TRUE);
|
|
// Lets use speedy tooltips
|
|
m_pToolTip->SetDelayTime(50);
|
|
#ifdef XP_WIN32
|
|
// We MUST do this for MFC tooltips
|
|
EnableToolTips(TRUE);
|
|
#endif // WIN32
|
|
}
|
|
}
|
|
|
|
CDropdownToolbar::~CDropdownToolbar()
|
|
{
|
|
if( m_pData ){
|
|
for( UINT i = 0; i < m_nButtonCount; i++ ){
|
|
if( m_pData[i].pButton ){
|
|
if( m_pData[i].pButtonName ) XP_FREE((void*)m_pData[i].pButtonName);
|
|
delete m_pData[i].pButton;
|
|
}
|
|
}
|
|
XP_FREE( m_pData );
|
|
}
|
|
if( m_pToolTip ){
|
|
delete m_pToolTip;
|
|
}
|
|
}
|
|
|
|
void CDropdownToolbar::PostNcDestroy()
|
|
{
|
|
ReleaseCapture();
|
|
CWnd::PostNcDestroy();
|
|
delete this;
|
|
}
|
|
|
|
// Create a CBitmapPushButton - resource name should be correct form for
|
|
// a CBitmapButton: "Button_" will load "Button_D.BMP" and "Button_U.BMP"
|
|
// bitmap files for Down and Up button images.
|
|
BOOL CDropdownToolbar::AddButton(LPSTR pButtonName, UINT nCommandID)
|
|
{
|
|
ASSERT(pButtonName);
|
|
ASSERT(nCommandID);
|
|
if( m_nAllocatedCount < (m_nButtonCount + 1) ){
|
|
m_nAllocatedCount += 8;
|
|
m_pData = (DropdownToolbarData*)XP_REALLOC(m_pData,
|
|
m_nAllocatedCount * sizeof(DropdownToolbarData));
|
|
}
|
|
if( m_pData ){
|
|
DropdownToolbarData *pData = &m_pData[m_nButtonCount];
|
|
|
|
pData->pButton = new CBitmapPushButton(TRUE); // Use "borderless" button style
|
|
if( ! pData->pButton ){
|
|
TRACE0("Failed to allocated new button for DropdownToolbar\n");
|
|
return FALSE;
|
|
}
|
|
pData->nCommandID = nCommandID;
|
|
pData->nBitmapID = 0;
|
|
pData->pButtonName = XP_STRDUP(pButtonName);
|
|
} else {
|
|
TRACE0("Failed to reallocated new DropdownToolbar data\n");
|
|
return FALSE;
|
|
}
|
|
m_nButtonCount++;
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL CDropdownToolbar::AddButton(UINT nBitmapID, UINT nCommandID)
|
|
{
|
|
ASSERT(nBitmapID);
|
|
ASSERT(nCommandID);
|
|
if( m_nAllocatedCount < (m_nButtonCount + 1) ){
|
|
m_nAllocatedCount += 8;
|
|
m_pData = (DropdownToolbarData*)XP_REALLOC(m_pData,
|
|
m_nAllocatedCount * sizeof(DropdownToolbarData));
|
|
}
|
|
if( m_pData ){
|
|
DropdownToolbarData *pData = &m_pData[m_nButtonCount];
|
|
|
|
pData->pButton = new CBitmapPushButton(TRUE); // Use "borderless" button style
|
|
if( ! pData->pButton ){
|
|
TRACE0("Failed to allocated new button for DropdownToolbar\n");
|
|
return FALSE;
|
|
}
|
|
pData->nCommandID = nCommandID;
|
|
pData->pButtonName = NULL;
|
|
pData->nBitmapID = nBitmapID;
|
|
|
|
} else {
|
|
TRACE0("Failed to reallocated new DropdownToolbar data\n");
|
|
return FALSE;
|
|
}
|
|
m_nButtonCount++;
|
|
return TRUE;
|
|
}
|
|
|
|
// Call this AFTER adding all buttons to toolbaar
|
|
void CDropdownToolbar::Show()
|
|
{
|
|
if( m_nButtonCount == 0 )
|
|
return;
|
|
|
|
// Full height of toolbar frame
|
|
int iHeight = (ED_TB_BUTTON_HEIGHT * m_nButtonCount) + 2;
|
|
POINT ptOrigin;
|
|
|
|
if( m_CallerRect.left == 0 && m_CallerRect.top == 0 ){
|
|
// No caller rect supplied - locate top at cursor Y and
|
|
// X so cursor pt. is at horizontal center
|
|
GetCursorPos(&ptOrigin);
|
|
ptOrigin.x -= (ED_TB_BUTTON_WIDTH / 2);
|
|
|
|
} else {
|
|
// Minus 1 for the 1-pixel dialog border around dropdown
|
|
// (makes dropdown look centered under caller button)
|
|
ptOrigin.x = m_CallerRect.left + ((m_CallerRect.right - m_CallerRect.left - ED_TB_BUTTON_WIDTH) / 2);
|
|
|
|
if( (m_CallerRect.bottom + iHeight) > sysInfo.m_iScreenHeight ){
|
|
// Locate toolbar above the caller button
|
|
// if toolbar bottom would extend off bottom of screen
|
|
ptOrigin.y = m_CallerRect.top - iHeight;
|
|
} else {
|
|
ptOrigin.y = m_CallerRect.bottom;
|
|
}
|
|
}
|
|
|
|
// Set location and size (fits tight around all buttons)
|
|
SetWindowPos( &wndTopMost, ptOrigin.x, ptOrigin.y,
|
|
ED_TB_BUTTON_WIDTH + 2, iHeight, SWP_NOACTIVATE );
|
|
|
|
// Convert caller rect to toolbar's coordinates
|
|
// to use for mouse hit testing
|
|
if( m_CallerRect.left != 0 && m_CallerRect.top != 0 ){
|
|
ScreenToClient(&m_CallerRect);
|
|
}
|
|
|
|
// Create all the buttons and load bitmaps etc
|
|
if( m_pData && m_nButtonCount > 0 ){
|
|
DropdownToolbarData *pData = m_pData;
|
|
RECT rect = {0, 0, ED_TB_BUTTON_WIDTH, ED_TB_BUTTON_HEIGHT-1};
|
|
|
|
for( UINT i = 0; i < m_nButtonCount; i++, pData++ ){
|
|
UINT nStyle = BS_PUSHBUTTON | BS_OWNERDRAW | WS_CHILD;
|
|
if( i == 0 ){
|
|
nStyle |= WS_GROUP | WS_TABSTOP;
|
|
}
|
|
rect.bottom++;
|
|
if( ! pData->pButton->Create(pData->pButtonName, nStyle,
|
|
rect, this, pData->nCommandID) ){
|
|
TRACE0("Failed to allocated new button in DropdownToolbar\n");
|
|
|
|
BUTTON_FAILED:
|
|
delete pData->pButton;
|
|
if( pData->pButtonName ) {
|
|
XP_FREE(pData->pButtonName);
|
|
pData->pButtonName = NULL;
|
|
}
|
|
pData->pButton = NULL;
|
|
CWnd::DestroyWindow();
|
|
return;
|
|
}
|
|
rect.bottom--;
|
|
if( pData->pButtonName) {
|
|
CString buttonName(pData->pButtonName);
|
|
if( ! pData->pButton->LoadBitmaps(buttonName + _T("U"),
|
|
buttonName + _T("D"), 0, 0) ){
|
|
TRACE0("Failed to load bitmaps for button in DropdownToolbar\n");
|
|
goto BUTTON_FAILED;
|
|
}
|
|
} else if( pData->nBitmapID ){
|
|
if( ! pData->pButton->LoadBitmap(pData->nBitmapID) ){
|
|
TRACE0("Failed to load bitmaps for button in DropdownToolbar\n");
|
|
goto BUTTON_FAILED;
|
|
}
|
|
} else {
|
|
TRACE0("No bitmap name or ID supplied for button in DropdownToolbar\n");
|
|
goto BUTTON_FAILED;
|
|
}
|
|
// Set pushed-down state for button if requested
|
|
if( m_nInitialID > 0 && pData->nCommandID == m_nInitialID ){
|
|
pData->pButton->SetCheck(TRUE);
|
|
}
|
|
pData->pButton->ShowWindow(SW_SHOW);
|
|
|
|
// Add tooltip for the button
|
|
if( m_pToolTip ){
|
|
#ifdef XP_WIN16
|
|
m_pToolTip->DelTool(pData->pButton->m_hWnd, pData->nCommandID);
|
|
#endif
|
|
m_pToolTip->AddTool(pData->pButton, pData->nCommandID,
|
|
&rect, pData->nCommandID);
|
|
}
|
|
|
|
// Move to next button down
|
|
rect.top += ED_TB_BUTTON_HEIGHT;
|
|
rect.bottom += ED_TB_BUTTON_HEIGHT;
|
|
}
|
|
if( m_pToolTip ){
|
|
m_pToolTip->Activate(TRUE);
|
|
}
|
|
}
|
|
|
|
ShowWindow(SW_SHOW);
|
|
// Grab the mouse so a click outside the dialog closes toolbar
|
|
SetCapture();
|
|
// Restore active caption highlighting to parent frame
|
|
GetParentFrame()->SetActiveWindow();
|
|
}
|
|
|
|
IMPLEMENT_DYNAMIC(CDropdownToolbar,CWnd)
|
|
BEGIN_MESSAGE_MAP(CDropdownToolbar, CWnd)
|
|
//{{AFX_MSG_MAP(CDropdownToolbar)
|
|
ON_WM_LBUTTONUP()
|
|
ON_WM_LBUTTONDOWN()
|
|
ON_WM_MOUSEMOVE()
|
|
//}}AFX_MSG_MAP
|
|
#ifdef XP_WIN32
|
|
ON_NOTIFY_EX( TTN_NEEDTEXT, 0, OnToolTipNotify )
|
|
#endif
|
|
END_MESSAGE_MAP()
|
|
|
|
BOOL CDropdownToolbar::PreTranslateMessage(MSG* pMsg)
|
|
{
|
|
if( m_pToolTip && pMsg->message >= WM_MOUSEFIRST && pMsg->message <= WM_MOUSELAST)
|
|
{
|
|
#ifdef XP_WIN32
|
|
// This is needed to get around an MFC bug
|
|
// where tooltip is disabled after Modal Dialog is called
|
|
m_pToolTip->Activate(TRUE);
|
|
#endif
|
|
m_pToolTip->RelayEvent(pMsg);
|
|
}
|
|
return CWnd::PreTranslateMessage(pMsg);
|
|
}
|
|
|
|
//
|
|
#ifdef XP_WIN32
|
|
// This is needed for non-CFrame owners of a CToolTipCtrl
|
|
BOOL CDropdownToolbar::OnToolTipNotify( UINT id, NMHDR * pNMHDR, LRESULT * pResult )
|
|
{
|
|
TOOLTIPTEXT *pTTT = (TOOLTIPTEXT *)pNMHDR;
|
|
UINT nID =pNMHDR->idFrom;
|
|
if (pTTT->uFlags & TTF_IDISHWND)
|
|
{
|
|
// idFrom is actually the HWND of the tool
|
|
nID = ::GetDlgCtrlID((HWND)nID);
|
|
if(nID) {
|
|
char * pTipText = FEU_GetToolTipText(nID);
|
|
if( pTipText ){
|
|
// Copy string into buffer - limit is 80 characters
|
|
// (use even number for foreign characters
|
|
strncpy( pTTT->szText, pTipText, 78);
|
|
|
|
// Just in case tip was longer than 78 characters!
|
|
pTTT->szText[79] = '\0';
|
|
return(TRUE);
|
|
}
|
|
}
|
|
}
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
void CDropdownToolbar::OnMouseMove(UINT nFlags, CPoint cPoint)
|
|
{
|
|
CWnd::OnMouseMove(nFlags, cPoint);
|
|
|
|
BOOL bStatusSet = FALSE;
|
|
char * szStatus;
|
|
POINT pt = {cPoint.x, cPoint.y};
|
|
|
|
if( m_pMWContext && m_nCallerID && m_CallerRect.top != 0 &&
|
|
::PtInRect(&m_CallerRect, pt) ){
|
|
szStatus = szLoadString(m_nCallerID);
|
|
if( szStatus ){
|
|
// Display status line help for caller button
|
|
// First strip tooltip
|
|
char * pEnter = strchr( szStatus, '\n');
|
|
if( pEnter ){
|
|
*pEnter = '\0';
|
|
}
|
|
wfe_Progress(m_pMWContext, szStatus);
|
|
// We can skip testing toolbar buttons if we are in caller
|
|
bStatusSet = TRUE;
|
|
}
|
|
}
|
|
|
|
// First button rect
|
|
RECT rect = {0, 0, ED_TB_BUTTON_WIDTH - 1,
|
|
ED_TB_BUTTON_HEIGHT - 1};
|
|
|
|
for( UINT i = 0; i < m_nButtonCount; i++ ){
|
|
|
|
BOOL bInButton = ::PtInRect(&rect, pt);
|
|
|
|
// Display status line help if within the toolbar
|
|
if( m_pMWContext && bInButton && !bStatusSet ){
|
|
szStatus = szLoadString(m_pData[i].nCommandID);
|
|
if( szStatus ){
|
|
// WACKY FEATURE: Look for accelerator, eg. (Ctrl+x)
|
|
// in Tooltip portion after a '\n' and move it to
|
|
// show in case tooltip isn't showing
|
|
// (like when mouse button is down)
|
|
char * pEnter = strchr( szStatus, '\n');
|
|
if( pEnter ){
|
|
char * pParen = strchr( pEnter, '(');
|
|
if( pParen ){
|
|
*pEnter = ' ';
|
|
strcpy(pEnter+1, pParen);
|
|
} else {
|
|
// No accerlerator, cut off tooltip
|
|
*pEnter = '\0';
|
|
}
|
|
}
|
|
wfe_Progress(m_pMWContext, szStatus );
|
|
} else {
|
|
// If string is missing, a blank would be less
|
|
// confusing then restoring the previous string!
|
|
wfe_Progress(m_pMWContext, " ");
|
|
}
|
|
}
|
|
|
|
// Set focus to any button if we move over it
|
|
// and set button as selected if left mouse button is down
|
|
if( bInButton ){
|
|
if( (nFlags & MK_LBUTTON ) && !m_pData[i].pButton->m_bSelected ){
|
|
m_pData[i].pButton->SetState(TRUE);
|
|
}
|
|
if( !m_pData[i].pButton->m_bFocus ){
|
|
m_pData[i].pButton->SetFocus();
|
|
}
|
|
break;
|
|
}
|
|
// Note: We are outside of current button
|
|
// Unselect button we moved off of except if its in "pushdown"
|
|
if( !m_pData[i].pButton->m_bDown &&
|
|
m_pData[i].pButton->m_bSelected ) {
|
|
m_pData[i].pButton->SetState(FALSE);
|
|
break;
|
|
}
|
|
// Next button
|
|
rect.top += ED_TB_BUTTON_HEIGHT;
|
|
rect.bottom += ED_TB_BUTTON_HEIGHT;
|
|
}
|
|
}
|
|
|
|
void CDropdownToolbar::OnLButtonDown(UINT nFlags, CPoint cPoint)
|
|
{
|
|
// Find the button pressed and redraw
|
|
CRect rect(0, 0, ED_TB_BUTTON_WIDTH - 1,
|
|
ED_TB_BUTTON_HEIGHT - 1);
|
|
|
|
POINT pt = {cPoint.x, cPoint.y};
|
|
for( UINT i = 0; i < m_nButtonCount; i++ ){
|
|
if( rect.PtInRect(pt) ){
|
|
// Mark button as pressed and has focus
|
|
m_pData[i].pButton->SetState(TRUE);
|
|
m_pData[i].pButton->SetFocus();
|
|
m_pData[i].pButton->Invalidate();
|
|
break;
|
|
}
|
|
rect.top += ED_TB_BUTTON_HEIGHT;
|
|
rect.bottom += ED_TB_BUTTON_HEIGHT;
|
|
}
|
|
CWnd::OnLButtonDown(nFlags, cPoint);
|
|
}
|
|
|
|
// First button up will always destroy toolbar
|
|
// if not in caller button or a toolbar button
|
|
void CDropdownToolbar::OnLButtonUp(UINT nFlags, CPoint cPoint)
|
|
{
|
|
POINT pt = {cPoint.x, cPoint.y};
|
|
BOOL bExit = FALSE;
|
|
|
|
if( m_CallerRect.top != 0 ){
|
|
// Don't destroy toolbar if button came up within the
|
|
// caller's button - this allows tooltips to show up
|
|
// when mouse moves over button when not held down
|
|
// REMOVE THIS FOR TOOLBAR DURING MOUSE-DOWN ONLY
|
|
if( ::PtInRect(&m_CallerRect, pt) ){
|
|
// Ignore the first time button comes up
|
|
// so a simple up+down click doesn't close toolbar
|
|
if( m_bFirstButtonUp ){
|
|
CWnd::OnLButtonUp(nFlags, cPoint);
|
|
m_bFirstButtonUp = FALSE;
|
|
return;
|
|
}
|
|
// Any other time, just exit
|
|
bExit = TRUE;
|
|
}
|
|
}
|
|
|
|
if( !bExit ){
|
|
// Do Hittest for all buttons
|
|
CRect rect(0, 0, ED_TB_BUTTON_WIDTH - 1,
|
|
ED_TB_BUTTON_HEIGHT - 1);
|
|
|
|
for( UINT i = 0; i < m_nButtonCount; i++ ){
|
|
if( rect.PtInRect(pt) ){
|
|
// Post message to parent to do what we're supposed to do
|
|
// only if different from initial state (for pushbutton style)
|
|
if( m_pParent && m_pData[i].nCommandID != m_nInitialID ){
|
|
m_pParent->PostMessage(WM_COMMAND, m_pData[i].nCommandID);
|
|
}
|
|
break;
|
|
}
|
|
rect.top += ED_TB_BUTTON_HEIGHT;
|
|
rect.bottom += ED_TB_BUTTON_HEIGHT;
|
|
}
|
|
}
|
|
|
|
// Restore previous status message
|
|
if( m_pMWContext ){
|
|
wfe_Progress(m_pMWContext, "");
|
|
}
|
|
DestroyWindow();
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CTagDlg dialog - The Arbitrary tag editor
|
|
|
|
|
|
CTagDlg::CTagDlg(CWnd* pParent,
|
|
MWContext* pMWContext, // MUST be supplied!
|
|
char *pTagData )
|
|
: CDialog(CTagDlg::IDD, pParent),
|
|
m_pMWContext(pMWContext),
|
|
m_bInsert(0),
|
|
m_bValidTag(0)
|
|
{
|
|
//{{AFX_DATA_INIT(CTagDlg)
|
|
m_csTagData = pTagData;
|
|
//}}AFX_DATA_INIT
|
|
wfe_GetLayoutViewSize(pMWContext, &m_iFullWidth, &m_iFullHeight);
|
|
}
|
|
|
|
|
|
void CTagDlg::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CDialog::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CTagDlg)
|
|
DDX_Text(pDX, IDC_EDIT_TAG, m_csTagData);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CTagDlg, CDialog)
|
|
//{{AFX_MSG_MAP(CTagDlg)
|
|
ON_BN_CLICKED(ID_HELP, OnHelp)
|
|
ON_BN_CLICKED(IDC_VERIFY_HTML, OnVerifyHtml)
|
|
//}}AFX_MSG_MAP
|
|
#ifdef XP_WIN32
|
|
ON_WM_HELPINFO()
|
|
#endif //XP_WIN32
|
|
END_MESSAGE_MAP()
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CTagDlg message handlers
|
|
|
|
#ifdef XP_WIN32
|
|
BOOL CTagDlg::OnHelpInfo(HELPINFO *)//32bit messagemapping.
|
|
{
|
|
OnHelp();
|
|
return TRUE;
|
|
}
|
|
#endif//XP_WIN32
|
|
|
|
|
|
|
|
BOOL CTagDlg::OnInitDialog()
|
|
{
|
|
// Switch back to NETSCAPE.EXE for resource hInstance
|
|
m_ResourceSwitcher.Reset();
|
|
|
|
if( ED_ELEMENT_UNKNOWN_TAG == EDT_GetCurrentElementType(m_pMWContext) ) {
|
|
m_csTagData = EDT_GetUnknownTagData(m_pMWContext);
|
|
} else {
|
|
m_bInsert = TRUE;
|
|
}
|
|
|
|
CDialog::OnInitDialog();
|
|
return TRUE; // return TRUE unless you set the focus to a control
|
|
// EXCEPTION: OCX Property Pages should return FALSE
|
|
}
|
|
|
|
Bool CTagDlg::DoVerifyTag( char* pTagString ){
|
|
ED_TagValidateResult e = EDT_ValidateTag((char*)LPCSTR(m_csTagData), FALSE );
|
|
UINT nIDS = 0;
|
|
switch( e ){
|
|
case ED_TAG_OK:
|
|
return TRUE;
|
|
case ED_TAG_UNOPENED:
|
|
nIDS = IDS_TAG_UNOPENED;
|
|
break;
|
|
case ED_TAG_UNCLOSED:
|
|
nIDS = IDS_TAG_UNCLOSED;
|
|
break;
|
|
case ED_TAG_UNTERMINATED_STRING:
|
|
nIDS = IDS_TAG_UNTERMINATED_STRING;
|
|
break;
|
|
case ED_TAG_PREMATURE_CLOSE:
|
|
nIDS = IDS_TAG_PREMATURE_CLOSE;
|
|
break;
|
|
case ED_TAG_TAGNAME_EXPECTED:
|
|
nIDS = IDS_TAG_TAGNAME_EXPECTED;
|
|
break;
|
|
default:
|
|
nIDS = IDS_TAG_ERROR;
|
|
}
|
|
|
|
MessageBox( szLoadString(nIDS),
|
|
szLoadString(IDS_ERROR_HTML_CAPTION),
|
|
MB_OK | MB_ICONEXCLAMATION);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
void CTagDlg::OnHelp()
|
|
{
|
|
NetHelp(HELP_HTML_TAG);
|
|
}
|
|
|
|
void CTagDlg::OnOK()
|
|
{
|
|
UpdateData(TRUE);
|
|
if( !DoVerifyTag( (char*)LPCSTR(m_csTagData) ) ){
|
|
return;
|
|
}
|
|
|
|
EDT_BeginBatchChanges(m_pMWContext);
|
|
CDialog::OnOK();
|
|
|
|
if( m_bInsert ){
|
|
EDT_InsertUnknownTag(m_pMWContext,(char*)LPCSTR(m_csTagData));
|
|
} else {
|
|
EDT_SetUnknownTagData(m_pMWContext, (char*)LPCSTR(m_csTagData));
|
|
}
|
|
EDT_EndBatchChanges(m_pMWContext);
|
|
|
|
//Note: For Attributes-only editing(e.g., HREF JavaScript),
|
|
// caller must get data from m_csTagData;
|
|
}
|
|
|
|
void CTagDlg::OnVerifyHtml()
|
|
{
|
|
UpdateData(TRUE);
|
|
DoVerifyTag((char*)LPCSTR(m_csTagData));
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CHRuleDlg dialog Horizontal rule properties in single modal dialog
|
|
|
|
|
|
CHRuleDlg::CHRuleDlg(CWnd* pParent,
|
|
MWContext* pMWContext, // MUST be supplied!
|
|
EDT_HorizRuleData* pData )
|
|
: CDialog(CHRuleDlg::IDD, pParent),
|
|
m_pMWContext(pMWContext)
|
|
{
|
|
ASSERT(pMWContext);
|
|
|
|
if (pData){
|
|
m_pData = pData;
|
|
m_bInsert = FALSE;
|
|
} else {
|
|
m_pData = EDT_NewHorizRuleData();
|
|
m_bInsert = TRUE;
|
|
}
|
|
|
|
//{{AFX_DATA_INIT(CHRuleDlg)
|
|
m_nAlign = 1;
|
|
m_bShading = TRUE;
|
|
m_iWidth = 0;
|
|
m_iHeight = 0;
|
|
m_iWidthType = 0;
|
|
//}}AFX_DATA_INIT
|
|
|
|
wfe_GetLayoutViewSize(pMWContext, &m_iFullWidth, NULL);
|
|
}
|
|
|
|
|
|
void CHRuleDlg::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CDialog::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CHRuleDlg)
|
|
DDX_Radio(pDX, IDC_HR_ALIGN_LEFT, m_nAlign);
|
|
DDX_Check(pDX, IDC_HR_SHADOW, m_bShading);
|
|
DDX_Check(pDX, IDC_HR_SAVE_DEFAULT, bSaveDefaultHRule);
|
|
DDX_CBIndex(pDX, IDC_HR_WIDTH_TYPE, m_iWidthType);
|
|
DDX_Text(pDX, IDC_HR_HEIGHT, m_iHeight);
|
|
DDV_MinMaxInt(pDX, m_iHeight, 1, 10000);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CHRuleDlg, CDialog)
|
|
//{{AFX_MSG_MAP(CHRuleDlg)
|
|
ON_BN_CLICKED(ID_HELP, OnHelp)
|
|
ON_BN_CLICKED(IDC_EXTRA_HTML, OnExtraHTML)
|
|
//}}AFX_MSG_MAP
|
|
#ifdef XP_WIN32
|
|
ON_WM_HELPINFO()
|
|
#endif //XP_WIN32
|
|
END_MESSAGE_MAP()
|
|
|
|
|
|
void CHRuleDlg::OnExtraHTML()
|
|
{
|
|
CExtraHTMLDlg dlg(this, &m_pData->pExtra, IDS_HRULE_TAG);
|
|
dlg.DoModal();
|
|
}
|
|
|
|
BOOL ValidateWidth(CDialog* pDlg, int* iWidth, int iWidthType)
|
|
{
|
|
char szMessage[256];
|
|
CEdit* pWidthControl = (CEdit*)(pDlg->GetDlgItem(IDC_HR_WIDTH));
|
|
ASSERT (pWidthControl);
|
|
char szWidth[16] = "";
|
|
char *szEndWidth;
|
|
pWidthControl->GetWindowText(szWidth, 15);
|
|
|
|
int width = (int)strtol( szWidth, &szEndWidth, 10 );
|
|
|
|
int iMaxWidth = (iWidthType == 0) ? 100 : 10000;
|
|
|
|
if( (width < 1 || width > iMaxWidth) || *szEndWidth != '\0'){
|
|
|
|
// Construct a string showing correct range
|
|
wsprintf( szMessage, szLoadString(IDS_INTEGER_RANGE_ERROR), 1, iMaxWidth );
|
|
|
|
// Notify user with similar message to the DDV_ validation system
|
|
pDlg->MessageBox(szMessage, szLoadString(AFX_IDS_APP_TITLE), MB_ICONEXCLAMATION | MB_OK);
|
|
|
|
// Put focus in the offending control
|
|
// And select all text, just like DDV functions
|
|
pWidthControl->SetFocus();
|
|
pWidthControl->SetSel(0, -1, TRUE);
|
|
return FALSE;
|
|
}
|
|
// Save values if they are good
|
|
if( width > 0 ){
|
|
*iWidth = width;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void UpdateWidth(CDialog* pPage, int iWidth)
|
|
{
|
|
char szBuf[16];
|
|
wsprintf(szBuf, "%d", iWidth);
|
|
pPage->GetDlgItem(IDC_HR_WIDTH)->SetWindowText(szBuf);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CHRuleDlg message handlers
|
|
|
|
BOOL CHRuleDlg::OnInitDialog()
|
|
{
|
|
if ( m_pMWContext == NULL ||
|
|
m_pData == NULL ) {
|
|
// Must have data and context!
|
|
EndDialog(IDCANCEL);
|
|
}
|
|
|
|
// Switch back to NETSCAPE.EXE for resource hInstance
|
|
m_ResourceSwitcher.Reset();
|
|
|
|
switch (m_pData->align) {
|
|
case ED_ALIGN_LEFT:
|
|
m_nAlign = 0;
|
|
break;
|
|
case ED_ALIGN_RIGHT:
|
|
m_nAlign = 2;
|
|
break;
|
|
default:
|
|
// Default is ED_ALIGN_CENTER
|
|
m_nAlign = 1;
|
|
break;
|
|
}
|
|
m_iWidth = CASTINT(m_pData->iWidth);
|
|
m_iWidthType = m_pData->bWidthPercent ? 0 : 1;
|
|
m_iHeight = CASTINT(m_pData->size);
|
|
|
|
// NoShading is HTML tag,
|
|
// but its better to show positive attributes,
|
|
// so we use "3-D Shading", with default = TRUE
|
|
m_bShading = !m_pData->bNoShade;
|
|
|
|
CComboBox * pCombo = (CComboBox*)GetDlgItem(IDC_HR_WIDTH_TYPE);
|
|
ASSERT(pCombo);
|
|
pCombo->AddString(szLoadString(IDS_PERCENT_WINDOW));
|
|
pCombo->AddString(szLoadString(IDS_PIXELS));
|
|
|
|
// Set caption to indicate a new inserted object
|
|
if ( m_bInsert ){
|
|
SetWindowText( szLoadString(IDS_INSERT_HRULE) );
|
|
}
|
|
|
|
UpdateWidth(this, m_iWidth);
|
|
CDialog::OnInitDialog();
|
|
return TRUE; // return TRUE unless you set the focus to a control
|
|
// EXCEPTION: OCX Property Pages should return FALSE
|
|
}
|
|
|
|
void CHRuleDlg::OnHelp()
|
|
{
|
|
NetHelp(HELP_PROPS_HRULE);
|
|
}
|
|
|
|
|
|
|
|
#ifdef XP_WIN32
|
|
BOOL CHRuleDlg::OnHelpInfo(HELPINFO *)//32bit messagemapping.
|
|
{
|
|
OnHelp();
|
|
return TRUE;
|
|
}
|
|
#endif//XP_WIN32
|
|
|
|
|
|
|
|
void CHRuleDlg::OnOK()
|
|
{
|
|
if( !UpdateData(TRUE) ||
|
|
!ValidateWidth(this, &m_iWidth, m_iWidthType ) ){
|
|
return;
|
|
}
|
|
|
|
EDT_BeginBatchChanges(m_pMWContext);
|
|
switch (m_nAlign) {
|
|
case 0:
|
|
m_pData->align = ED_ALIGN_LEFT;
|
|
break;
|
|
case 1:
|
|
m_pData->align = ED_ALIGN_CENTER;
|
|
break;
|
|
case 2:
|
|
m_pData->align = ED_ALIGN_RIGHT;
|
|
break;
|
|
}
|
|
m_pData->iWidth = m_iWidth;
|
|
m_pData->bWidthPercent = (m_iWidthType == 0);
|
|
m_pData->size = m_iHeight;
|
|
m_pData->bNoShade = !m_bShading;
|
|
|
|
if ( m_bInsert ) {
|
|
EDT_InsertHorizRule( m_pMWContext, m_pData );
|
|
} else {
|
|
EDT_SetHorizRuleData( m_pMWContext, m_pData );
|
|
}
|
|
EDT_EndBatchChanges(m_pMWContext);
|
|
|
|
if( bSaveDefaultHRule ){
|
|
// Save current values as default preferences
|
|
PREF_SetIntPref("editor.hrule.height",m_iHeight);
|
|
PREF_SetIntPref("editor.hrule.width",m_iWidth);
|
|
if( m_iWidthType == 0 ){
|
|
PREF_SetBoolPref("editor.hrule.width_percent",TRUE);
|
|
} else {
|
|
PREF_SetBoolPref("editor.hrule.width_percent",FALSE);
|
|
}
|
|
if( m_bShading ){
|
|
PREF_SetBoolPref("editor.hrule.shading",TRUE);
|
|
} else {
|
|
PREF_SetBoolPref("editor.hrule.shading",FALSE);
|
|
}
|
|
PREF_SetIntPref("editor.hrule.align",m_pData->align);
|
|
}
|
|
CDialog::OnOK();
|
|
}
|
|
|
|
BOOL CHRuleDlg::DestroyWindow()
|
|
{
|
|
if (m_pData) {
|
|
EDT_FreeHorizRuleData(m_pData);
|
|
}
|
|
return CDialog::DestroyWindow();
|
|
}
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CTargetDlg dialog Set Target(named anchor) properties in single modal dialog
|
|
|
|
|
|
CTargetDlg::CTargetDlg(CWnd* pParent,
|
|
MWContext* pMWContext, // MUST be supplied!
|
|
char *pName) // Existing target
|
|
: CDialog(CTargetDlg::IDD, pParent),
|
|
m_pMWContext(pMWContext),
|
|
m_bInsert(0)
|
|
{
|
|
ASSERT(pMWContext);
|
|
//{{AFX_DATA_INIT(CTargetDlg)
|
|
m_csName = pName;
|
|
//}}AFX_DATA_INIT
|
|
}
|
|
|
|
|
|
void CTargetDlg::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CDialog::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CTargetDlg)
|
|
DDX_Text(pDX, IDC_TARGET_NAME, m_csName);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CTargetDlg, CDialog)
|
|
//{{AFX_MSG_MAP(CTargetDlg)
|
|
ON_EN_CHANGE(IDC_TARGET_NAME, OnChangeTargetName)
|
|
ON_BN_CLICKED(ID_HELP, OnHelp)
|
|
//}}AFX_MSG_MAP
|
|
#ifdef XP_WIN32
|
|
ON_WM_HELPINFO()
|
|
#endif //XP_WIN32
|
|
END_MESSAGE_MAP()
|
|
|
|
// CTargetDlg message handlers
|
|
|
|
BOOL CTargetDlg::OnInitDialog()
|
|
{
|
|
// Switch back to NETSCAPE.EXE for resource hInstance
|
|
m_ResourceSwitcher.Reset();
|
|
|
|
if( ED_ELEMENT_TARGET == EDT_GetCurrentElementType(m_pMWContext) ) {
|
|
m_csName = EDT_GetTargetData(m_pMWContext);
|
|
} else {
|
|
m_bInsert = TRUE;
|
|
// Use current selected text as suggested target name...
|
|
char *pName = (char*)LO_GetSelectionText(ABSTRACTCX(m_pMWContext)->GetDocumentContext());
|
|
if ( pName ){
|
|
char *pTemp = pName;
|
|
char *pEnd = pName + min( XP_STRLEN(pName), 50);
|
|
// Skip over leading white-space
|
|
while( XP_IS_SPACE(*pTemp) && pTemp < pEnd ) pTemp++;
|
|
if( pTemp < pEnd ){
|
|
char *pStart = pTemp;
|
|
// Stop at any CR/LF and replace
|
|
// other whitespace with real spaces (OR UNDERSCORE?)
|
|
while( pTemp <= pEnd ){
|
|
if( *pTemp == '\n' || *pTemp == '\r' ){
|
|
// Stop at end of line
|
|
pEnd = pTemp;
|
|
break;
|
|
}
|
|
if( XP_IS_SPACE(*pTemp) ) *pTemp = ' ';
|
|
pTemp++;
|
|
}
|
|
pTemp = pEnd;
|
|
// Find last character before the last space from the end
|
|
while( !XP_IS_SPACE(*pTemp) && pTemp != pName ) pTemp--;
|
|
if( pTemp != pName ) *pTemp = '\0';
|
|
m_csName = pStart;
|
|
m_csName.TrimRight();
|
|
}
|
|
XP_FREE(pName);
|
|
}
|
|
}
|
|
|
|
// Get the list of existing targets to
|
|
// warn if target name is already used
|
|
m_pTargetList = EDT_GetAllDocumentTargets(m_pMWContext);
|
|
|
|
CDialog::OnInitDialog();
|
|
|
|
// Enable OK only if we have some text already
|
|
GetDlgItem(IDOK)->EnableWindow(!m_csName.IsEmpty());
|
|
|
|
return TRUE; // return TRUE unless you set the focus to a control
|
|
// EXCEPTION: OCX Property Pages should return FALSE
|
|
}
|
|
|
|
void CTargetDlg::OnHelp()
|
|
{
|
|
NetHelp(HELP_PROPS_TARGET);
|
|
}
|
|
|
|
|
|
|
|
#ifdef XP_WIN32
|
|
BOOL CTargetDlg::OnHelpInfo(HELPINFO *)//32bit messagemapping.
|
|
{
|
|
OnHelp();
|
|
return TRUE;
|
|
}
|
|
#endif//XP_WIN32
|
|
|
|
|
|
|
|
void CTargetDlg::OnOK()
|
|
{
|
|
EDT_BeginBatchChanges(m_pMWContext);
|
|
|
|
CDialog::OnOK();
|
|
// Remove trailing spaces and quote marks
|
|
CleanupString(m_csName);
|
|
|
|
// Strip off hash mark
|
|
if(!m_csName.IsEmpty() && m_csName.GetAt(0) == '#'){
|
|
m_csName = m_csName.Mid(1);;
|
|
}
|
|
if( !m_csName.IsEmpty() ){
|
|
if( m_bInsert ){
|
|
EDT_InsertTarget(m_pMWContext, (char*)LPCSTR(m_csName));
|
|
} else {
|
|
EDT_SetTargetData(m_pMWContext, (char*)LPCSTR(m_csName));
|
|
}
|
|
}
|
|
|
|
EDT_EndBatchChanges(m_pMWContext);
|
|
}
|
|
|
|
void CTargetDlg::OnChangeTargetName()
|
|
{
|
|
UpdateData(TRUE);
|
|
// Remove trailing spaces and quote marks, but
|
|
// use temp string to cleanup and don't send string back with UpdateData
|
|
// This upsets caret location etc.
|
|
CString csTemp = m_csName;
|
|
CleanupString(csTemp);
|
|
GetDlgItem(IDOK)->EnableWindow(!csTemp.IsEmpty());
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
// Special class to control Background color of static control
|
|
//
|
|
// CColorStatic
|
|
|
|
CColorStatic::
|
|
CColorStatic(COLORREF crTextColor, // Default is black text
|
|
COLORREF crBackColor, // on gray Background
|
|
ED_BorderStyle nStyle)
|
|
{
|
|
m_crTextColor = crTextColor;
|
|
m_crBackColor = crBackColor;
|
|
m_nBorderStyle = nStyle;
|
|
|
|
// Create a white brush for the Background
|
|
m_brush.CreateSolidBrush(crBackColor|0x02000000); // Like PALETTERGB
|
|
}
|
|
|
|
CColorStatic::~CColorStatic()
|
|
{
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CColorStatic, CStatic)
|
|
//{{AFX_MSG_MAP(CColorStatic)
|
|
// NOTE - the ClassWizard will add and remove mapping macros here.
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CColorStatic message handlers
|
|
BOOL CColorStatic::OnChildNotify(UINT message, WPARAM wParam, LPARAM lParam,
|
|
LRESULT* pLResult)
|
|
{
|
|
int nCtlType = (int)HIWORD(lParam);
|
|
|
|
// If "message" is not the message you're after, do default processing:
|
|
if (message != WM_CTLCOLOR)
|
|
{
|
|
return CStatic::OnChildNotify(message,wParam,lParam,pLResult);
|
|
}
|
|
|
|
HDC hdcChild = (HDC)wParam;
|
|
SetTextColor(hdcChild, m_crTextColor);
|
|
SetBkColor(hdcChild, m_crTextColor);
|
|
|
|
// Send what would have been the return value of OnCtlColor() - the brush
|
|
// handle - back in pLResult:
|
|
*pLResult = (LRESULT)(m_brush.GetSafeHandle());
|
|
|
|
// Return TRUE to indicate that the message was handled:
|
|
return TRUE;
|
|
}
|
|
|
|
#define MAX_META_NAME_LEN 100
|
|
#define MAX_META_VALUE_LEN 400
|
|
enum {
|
|
EDIT_NONE,
|
|
EDIT_EQUIV,
|
|
EDIT_META
|
|
};
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Document Properties Pages
|
|
//
|
|
// First page = general info
|
|
// "Fixed" Meta Tags we always include
|
|
//
|
|
CDocInfoPage::CDocInfoPage(CWnd* pParent, MWContext * pMWContext,
|
|
CEditorResourceSwitcher * pResourceSwitcher,
|
|
EDT_PageData * pPageData)
|
|
: CNetscapePropertyPage(CDocInfoPage::IDD),
|
|
m_pMWContext(pMWContext),
|
|
m_pResourceSwitcher(pResourceSwitcher),
|
|
m_bActivated(0),
|
|
m_pPageData(pPageData)
|
|
{
|
|
//{{AFX_DATA_INIT(CDocInfoPage)
|
|
m_csAuthor = _T("");
|
|
m_csClassification = _T("");
|
|
m_csDescription = _T("");
|
|
m_csTitle = _T("");
|
|
m_csURL = _T("");
|
|
m_csKeyWords = _T("");
|
|
//}}AFX_DATA_INIT
|
|
ASSERT(pMWContext);
|
|
ASSERT(pPageData);
|
|
|
|
#ifdef XP_WIN32
|
|
// Set the hInstance so we get template from editor's resource DLL
|
|
m_psp.hInstance = AfxGetResourceHandle();
|
|
#endif
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
BOOL CDocInfoPage::OnSetActive()
|
|
{
|
|
if(m_pResourceSwitcher && !m_bActivated){
|
|
// We must be sure we have switched
|
|
// the first time here - before dialog creation
|
|
m_pResourceSwitcher->switchResources();
|
|
}
|
|
if(!CPropertyPage::OnSetActive())
|
|
return(FALSE);
|
|
|
|
if(m_bActivated)
|
|
return(TRUE);
|
|
|
|
// Switch back to EXE's resources
|
|
if( m_pResourceSwitcher ){
|
|
m_pResourceSwitcher->Reset();
|
|
}
|
|
|
|
m_bActivated = TRUE;
|
|
|
|
// This should be same as hist_ent->title??
|
|
m_csTitle = m_pPageData->pTitle;
|
|
|
|
History_entry * hist_ent = SHIST_GetCurrent(&m_pMWContext->hist);
|
|
if(hist_ent && hist_ent->address) {
|
|
m_csURL = hist_ent->address;
|
|
WFE_CondenseURL(m_csURL, 80, FALSE);
|
|
GetDlgItem(IDC_DOC_URL)->SetWindowText(CHAR_STR(m_csURL));
|
|
|
|
// If empty, fill in title with Filename part of URL without extension
|
|
if( m_csTitle.IsEmpty() ){
|
|
char * pTitle = EDT_GetPageTitleFromFilename(hist_ent->address);
|
|
if( pTitle ){
|
|
m_csTitle = pTitle;
|
|
XP_FREE(pTitle);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// Get data from meta tags:
|
|
int count = EDT_MetaDataCount(m_pMWContext);
|
|
for ( int i = 0; i < count; i++ ) {
|
|
EDT_MetaData* pData = EDT_GetMetaData(m_pMWContext, i);
|
|
if ( !pData->bHttpEquiv ) {
|
|
if ( 0 == _stricmp(pData->pName, "Author") ) {
|
|
m_csAuthor = pData->pContent;
|
|
} else if ( 0 == _stricmp(pData->pName, "Classification") ) {
|
|
m_csClassification = pData->pContent;
|
|
} else if ( 0 == _stricmp(pData->pName, "Description") ) {
|
|
m_csDescription = pData->pContent;
|
|
} else if ( 0 == _stricmp(pData->pName, "KeyWords") ) {
|
|
m_csKeyWords = pData->pContent;
|
|
}/* else if ( 0 == _stricmp(pData->pName, "Created") ) {
|
|
m_csCreateDate = pData->pContent;
|
|
} else if ( 0 == _stricmp(pData->pName, "Last-Modified") ) {
|
|
m_csUpdateDate = pData->pContent;
|
|
}*/
|
|
}
|
|
}
|
|
if( m_csAuthor.IsEmpty() ){
|
|
// Get name from preferences if none supplied
|
|
char * pAuthor = NULL;
|
|
PREF_CopyCharPref("editor.author", &pAuthor);
|
|
m_csAuthor = pAuthor;
|
|
XP_FREEIF(pAuthor);
|
|
}
|
|
UpdateData(FALSE);
|
|
return(TRUE);
|
|
}
|
|
|
|
// Create a MetaData structure and save it
|
|
// This commits change - NO BUFFERING
|
|
// Be sure to strip off spaces and quotes before calling this
|
|
void CDocInfoPage::SetMetaData(char * pName, char * pValue)
|
|
{
|
|
EDT_MetaData *pData = EDT_NewMetaData();
|
|
if ( pData ) {
|
|
pData->bHttpEquiv = FALSE;
|
|
if ( pName && XP_STRLEN(pName) > 0 ) {
|
|
pData->pName = XP_STRDUP(pName);
|
|
if ( pValue && XP_STRLEN(pValue) > 0 ) {
|
|
pData->pContent = XP_STRDUP(pValue);
|
|
EDT_SetMetaData(m_pMWContext, pData);
|
|
} else {
|
|
// (Don't really need to do this)
|
|
pData->pContent = NULL;
|
|
// Remove the item
|
|
EDT_DeleteMetaData(m_pMWContext, pData);
|
|
}
|
|
OkToClose();
|
|
}
|
|
EDT_FreeMetaData(pData);
|
|
}
|
|
}
|
|
|
|
void CDocInfoPage::OnHelp()
|
|
{
|
|
NetHelp(HELP_DOC_PROPS_GENERAL);
|
|
}
|
|
|
|
void CDocInfoPage::OnOK()
|
|
{
|
|
CPropertyPage::OnOK();
|
|
|
|
// never visited this page or no change -- don't bother
|
|
if(!m_bActivated ||
|
|
!IS_APPLY_ENABLED(this)){
|
|
return;
|
|
}
|
|
m_bActivated = TRUE;
|
|
|
|
// EDT_BeginBatchChanges(m_pMWContext);
|
|
|
|
// Replace Title and FontDefURL
|
|
XP_FREEIF(m_pPageData->pTitle);
|
|
|
|
CleanupString(m_csTitle);
|
|
|
|
if ( !m_csTitle.IsEmpty() ) {
|
|
m_pPageData->pTitle = XP_STRDUP(m_csTitle);
|
|
}
|
|
|
|
EDT_SetPageData(m_pMWContext, m_pPageData);
|
|
|
|
// Trim quotes and set data meta tags
|
|
CleanupString(m_csAuthor);
|
|
if ( !m_csAuthor.IsEmpty() ) {
|
|
// Don't wipe out author field if empty?
|
|
SetMetaData("Author", CHAR_STR(m_csAuthor));
|
|
}
|
|
|
|
CleanupString(m_csClassification);
|
|
SetMetaData("Classification", CHAR_STR(m_csClassification));
|
|
|
|
CleanupString(m_csDescription);
|
|
SetMetaData("Description", CHAR_STR(m_csDescription));
|
|
|
|
CleanupString(m_csKeyWords);
|
|
SetMetaData("KeyWords", CHAR_STR(m_csKeyWords));
|
|
|
|
// TODO: MAKE THIS STRING IF NEW DOC???
|
|
// SetMetaData(FALSE, "Created", CHAR_STR());
|
|
// SetMetaData(FALSE, "Last-Modified", CHAR_STR());
|
|
|
|
OkToClose();
|
|
//EDT_EndBatchChanges(m_pMWContext);
|
|
}
|
|
|
|
void CDocInfoPage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CDocInfoPage)
|
|
DDX_Text(pDX, IDC_DOC_AUTHOR, m_csAuthor);
|
|
DDX_Text(pDX, IDC_DOC_CLASIFICATION, m_csClassification);
|
|
DDX_Text(pDX, IDC_DOC_DESCRIPTION, m_csDescription);
|
|
DDX_Text(pDX, IDC_DOC_TITLE, m_csTitle);
|
|
DDX_Text(pDX, IDC_DOC_URL, m_csURL);
|
|
DDX_Text(pDX, IDC_KEYWORDS, m_csKeyWords);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CDocInfoPage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CDocInfoPage)
|
|
ON_EN_CHANGE(IDC_DOC_AUTHOR, EnableApply)
|
|
ON_EN_CHANGE(IDC_DOC_CLASIFICATION, EnableApply)
|
|
ON_EN_CHANGE(IDC_DOC_DESCRIPTION, EnableApply)
|
|
ON_EN_CHANGE(IDC_DOC_TITLE, EnableApply)
|
|
ON_EN_CHANGE(IDC_KEYWORDS, EnableApply)
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
|
|
void CDocInfoPage::EnableApply()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// Local helpers for color params converversions
|
|
|
|
// Allocate and initialize a new scheme struct and add it to list
|
|
EDT_ColorSchemeData * AddNewColorData( XP_List *pSchemeData )
|
|
{
|
|
EDT_ColorSchemeData * pColorData = XP_NEW(EDT_ColorSchemeData);
|
|
|
|
if(!pColorData) {
|
|
return NULL;
|
|
}
|
|
memset(pColorData, 0, sizeof(EDT_ColorSchemeData));
|
|
|
|
XP_ListAddObjectToEnd(pSchemeData, pColorData);
|
|
return pColorData;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
/*
|
|
// For reference: the built-in default colors
|
|
LO_DEFAULT_FG_RED 0
|
|
LO_DEFAULT_FG_GREEN 0
|
|
LO_DEFAULT_FG_BLUE 0
|
|
|
|
LO_DEFAULT_BG_RED 192
|
|
LO_DEFAULT_BG_GREEN 192
|
|
LO_DEFAULT_BG_BLUE 192
|
|
|
|
LO_UNVISITED_ANCHOR_RED 0
|
|
LO_UNVISITED_ANCHOR_GREEN 0
|
|
LO_UNVISITED_ANCHOR_BLUE 238
|
|
|
|
LO_VISITED_ANCHOR_RED 85
|
|
LO_VISITED_ANCHOR_GREEN 26
|
|
LO_VISITED_ANCHOR_BLUE 139
|
|
|
|
LO_SELECTED_ANCHOR_RED 255
|
|
LO_SELECTED_ANCHOR_GREEN 0
|
|
LO_SELECTED_ANCHOR_BLUE 0
|
|
// Offsets for getting defaults from lo_master_colors
|
|
// eg:
|
|
red = lo_master_colors[LO_COLOR_FG].red;
|
|
green = lo_master_colors[LO_COLOR_FG].green;
|
|
blue = lo_master_colors[LO_COLOR_FG].blue;
|
|
|
|
LO_COLOR_BG 0
|
|
LO_COLOR_FG 1
|
|
LO_COLOR_LINK 2
|
|
LO_COLOR_VLINK 3
|
|
LO_COLOR_ALINK 4
|
|
*/
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
CDocColorPage::CDocColorPage(CWnd* pParent,
|
|
UINT nIDCaption,
|
|
UINT nIDFocus,
|
|
MWContext * pMWContext,
|
|
CEditorResourceSwitcher * pResourceSwitcher,
|
|
EDT_PageData * pPageData )
|
|
: CNetscapePropertyPage(CDocColorPage::IDD, nIDCaption, nIDFocus),
|
|
m_bActivated(0),
|
|
m_pMWContext(pMWContext),
|
|
m_pResourceSwitcher(pResourceSwitcher),
|
|
m_pPageData(pPageData),
|
|
m_bImageChanged(0),
|
|
m_bValidImage(0),
|
|
m_pSchemeData(0),
|
|
m_hPal(0)
|
|
{
|
|
ASSERT(pMWContext);
|
|
|
|
//{{AFX_DATA_INIT(CDocColorPage)
|
|
m_csBackgroundImage = _T("");
|
|
m_csSelectedScheme = _T("");
|
|
m_bNoSave = 0;
|
|
//}}AFX_DATA_INIT
|
|
|
|
//m_csCustomBackgroundImage = _T("");
|
|
m_csBrowserBackgroundImage = _T("");
|
|
|
|
#ifdef XP_WIN32
|
|
// Set the hInstance so we get template from editor's resource DLL
|
|
m_psp.hInstance = AfxGetResourceHandle();
|
|
#endif
|
|
}
|
|
|
|
CDocColorPage::~CDocColorPage()
|
|
{
|
|
// Destroy the scheme list
|
|
// TODO: Move to XP code
|
|
if ( m_pSchemeData ) {
|
|
EDT_ColorSchemeData *pColorData;
|
|
XP_List * list_ptr = m_pSchemeData;
|
|
while ((pColorData = (EDT_ColorSchemeData *)XP_ListNextObject(list_ptr))) {
|
|
if ( pColorData->pSchemeName ) {
|
|
XP_FREE( pColorData->pSchemeName );
|
|
}
|
|
if ( pColorData->pBackgroundImage ) {
|
|
XP_FREE( pColorData->pBackgroundImage );
|
|
}
|
|
XP_FREE(pColorData);
|
|
}
|
|
XP_ListDestroy(m_pSchemeData);
|
|
}
|
|
}
|
|
|
|
void CDocColorPage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CDocColorPage)
|
|
DDX_Text(pDX, IDC_BKGRND_IMAGE, m_csBackgroundImage);
|
|
DDX_CBString(pDX, IDC_SCHEME_LIST, m_csSelectedScheme);
|
|
DDX_Check(pDX, IDC_NO_SAVE_IMAGE, m_bNoSave);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CDocColorPage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CDocColorPage)
|
|
ON_BN_CLICKED(IDC_BKGRND_USE_IMAGE, OnUseBkgrndImage)
|
|
ON_BN_CLICKED(IDC_CHOOSE_BACKGROUND, OnChooseBkgrndImage)
|
|
ON_CBN_SELCHANGE(IDC_SCHEME_LIST, OnSelchangeSchemeList)
|
|
ON_EN_CHANGE(IDC_BKGRND_IMAGE, OnChangeBkgrndImage)
|
|
ON_EN_KILLFOCUS(IDC_BKGRND_IMAGE, OnKillfocusBkgrndImage)
|
|
ON_WM_PAINT()
|
|
ON_BN_CLICKED(IDC_CHOOSE_TEXT_COLOR, OnChooseTextColor)
|
|
ON_BN_CLICKED(IDC_CHOOSE_LINK_COLOR, OnChooseLinkColor)
|
|
ON_BN_CLICKED(IDC_CHOOSE_ACTIVELINK_COLOR, OnChooseActivelinkColor)
|
|
ON_BN_CLICKED(IDC_CHOOSE_FOLLOWEDLINK_COLOR, OnChooseFollowedlinkColor)
|
|
ON_BN_CLICKED(IDC_CHOOSE_BKGRND_COLOR, OnChooseBkgrndColor)
|
|
ON_BN_CLICKED(IDC_SET_DOC_COLORS, OnColorsRadioButtons)
|
|
ON_BN_CLICKED(IDC_USE_BROWSER_COLORS, OnColorsRadioButtons)
|
|
ON_BN_CLICKED(IDC_USE_AS_DEFAULT, OnUseAsDefault)
|
|
ON_BN_CLICKED(IDC_NO_SAVE_IMAGE, OnNoSave)
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
#include "timer.h" //For nettimer.OnIdle()
|
|
|
|
BOOL CDocColorPage::OnSetActive()
|
|
{
|
|
if(m_pResourceSwitcher && !m_bActivated){
|
|
// We must be sure we have switched
|
|
// the first time here - before dialog creation
|
|
m_pResourceSwitcher->switchResources();
|
|
}
|
|
|
|
if(!CPropertyPage::OnSetActive())
|
|
return(FALSE);
|
|
|
|
if(m_bActivated)
|
|
return(TRUE);
|
|
|
|
// Switch back to EXE's resources
|
|
if( m_pResourceSwitcher ){
|
|
m_pResourceSwitcher->Reset();
|
|
}
|
|
|
|
m_hPal = WFE_GetUIPalette(GetParentFrame());
|
|
|
|
if( !m_TextColorButton.Subclass(this, IDC_CHOOSE_TEXT_COLOR, &m_crText) ||
|
|
!m_LinkColorButton.Subclass(this, IDC_CHOOSE_LINK_COLOR, &m_crLink) ||
|
|
!m_ActiveLinkColorButton.Subclass(this, IDC_CHOOSE_ACTIVELINK_COLOR, &m_crActiveLink) ||
|
|
!m_FollowedLinkColorButton.Subclass(this, IDC_CHOOSE_FOLLOWEDLINK_COLOR, &m_crFollowedLink) ||
|
|
!m_BackgroundColorButton.Subclass(this, IDC_CHOOSE_BKGRND_COLOR, &m_crBackground) ){
|
|
return FALSE;
|
|
}
|
|
|
|
// TODO - do stuff here
|
|
m_bActivated = TRUE;
|
|
|
|
CComboBox * pSchemeListBox = (CComboBox*)GetDlgItem(IDC_SCHEME_LIST);
|
|
|
|
CFont fontPreview;
|
|
// Create a font for the preview window
|
|
if( GetSystemMetrics(SM_DBCSENABLED) ){
|
|
HFONT hFont = NULL;
|
|
|
|
#ifdef _WIN32
|
|
hFont = (HFONT)GetStockObject(DEFAULT_GUI_FONT);
|
|
#endif
|
|
|
|
if (!hFont){
|
|
hFont = (HFONT)GetStockObject(SYSTEM_FONT);
|
|
}
|
|
fontPreview.Attach(hFont);
|
|
GetDlgItem(IDC_DOC_COLOR_PREVIEW)->SetFont(&fontPreview);
|
|
} else {
|
|
// Get a 1-pixel font
|
|
LOGFONT logFont;
|
|
memset(&logFont, 0, sizeof(logFont));
|
|
logFont.lfHeight = -18; // about 14 points?
|
|
logFont.lfWeight = FW_BOLD;
|
|
logFont.lfPitchAndFamily = VARIABLE_PITCH | FF_ROMAN;
|
|
lstrcpy(logFont.lfFaceName,"MS Serif");
|
|
|
|
if( fontPreview.CreateFontIndirect(&logFont) ){
|
|
GetDlgItem(IDC_DOC_COLOR_PREVIEW)->SetFont(&fontPreview);
|
|
} else {
|
|
TRACE0("Could Not create preview for Color Dialog\n");
|
|
}
|
|
}
|
|
|
|
// Get current stuff from preferences
|
|
char * prefStr = NULL;
|
|
PREF_CopyCharPref("editor.color_scheme",&prefStr);
|
|
if (prefStr) {
|
|
m_csSelectedScheme = prefStr;
|
|
XP_FREE(prefStr);
|
|
} else m_csSelectedScheme = "";
|
|
|
|
// Construct a list of Schemes
|
|
// TODO: Make this a global list and move to XP code
|
|
m_pSchemeData = XP_ListNew();
|
|
|
|
EDT_ColorSchemeData * pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
// First scheme is always the Netscape Default Colors
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_COLOR_SCHEME_NS_DEFAULT));
|
|
|
|
COLORREF clr;
|
|
PREF_GetColorPrefDWord("editor.text_color",&clr);
|
|
WFE_SetLO_Color( clr, &pColorData->ColorText );
|
|
|
|
PREF_GetColorPrefDWord("editor.link_color",&clr);
|
|
WFE_SetLO_Color( clr, &pColorData->ColorLink );
|
|
|
|
PREF_GetColorPrefDWord("editor.active_link_color",&clr);
|
|
WFE_SetLO_Color( clr, &pColorData->ColorActiveLink );
|
|
|
|
PREF_GetColorPrefDWord("editor.followed_link_color",&clr);
|
|
WFE_SetLO_Color( clr, &pColorData->ColorFollowedLink );
|
|
|
|
PREF_GetColorPrefDWord("editor.background_color",&clr);
|
|
WFE_SetLO_Color( clr, &pColorData->ColorBackground );
|
|
// Add name to listbox
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_BLACK_ON_WHITE));
|
|
WFE_SetLO_Color( RGB(0,0,0),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(255,255,255),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(255,0,0),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,0,128),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(0,0,255),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_BLACK_ON_OFFWHITE));
|
|
WFE_SetLO_Color( RGB(0,0,0),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(255,240,240),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(255,0,0),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,0,128),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(0,0,255),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_BLACK_ON_LTYELLOW));
|
|
WFE_SetLO_Color( RGB(0,0,0),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(255,255,192),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(0,0,255),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,0,128),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(255,0,255),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_BLACK_ON_YELLOW));
|
|
WFE_SetLO_Color( RGB(64,0,64),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(255,255,128),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(0,0,255),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(0,128, 0),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(255,0,128),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_BLACK_ON_LTBLUE));
|
|
WFE_SetLO_Color( RGB(0,0,0),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(192,192,255),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(0,0,255),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,0,128),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(255,0,128),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_BLACK_ON_MEDBLUE));
|
|
WFE_SetLO_Color( RGB(0,0,0),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(128,128,192),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(255,255,255),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,0,128),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(255,255,0),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_BLUE_ON_ORANGE));
|
|
WFE_SetLO_Color( RGB(0,0,128),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(255,192,64),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(0,0,255),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(0,128,0),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(0,255,255),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_WHITE_ON_BLACK));
|
|
WFE_SetLO_Color( RGB(255,255,255),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(0,0,0),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(255,255,0),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(192,192,192),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(192,255,192),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_WHITE_ON_GREEN));
|
|
WFE_SetLO_Color( RGB(255,255,255),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(0,64,0),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(255,255,0),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,255,128),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(0,255,64),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_WHITE_ON_BLUE));
|
|
WFE_SetLO_Color( RGB(255,255,255),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(0,0,128),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(255,255,0),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,128,255),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(255,0,255),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
pColorData = AddNewColorData( m_pSchemeData );
|
|
if ( pColorData ) {
|
|
pColorData->pSchemeName = XP_STRDUP(szLoadString(IDS_WHITE_ON_VIOLET));
|
|
WFE_SetLO_Color( RGB(255,255,255),
|
|
&pColorData->ColorText );
|
|
WFE_SetLO_Color( RGB(128,0,128),
|
|
&pColorData->ColorBackground );
|
|
WFE_SetLO_Color( RGB(0,255,255),
|
|
&pColorData->ColorLink );
|
|
WFE_SetLO_Color( RGB(128,255,255),
|
|
&pColorData->ColorFollowedLink );
|
|
WFE_SetLO_Color( RGB(0,255,0),
|
|
&pColorData->ColorActiveLink );
|
|
pSchemeListBox->AddString(pColorData->pSchemeName);
|
|
}
|
|
|
|
// Get Browser preference colors
|
|
m_crBrowserText = prefInfo.m_rgbForegroundColor;
|
|
m_crBrowserLink = prefInfo.m_rgbAnchorColor;
|
|
m_crBrowserFollowedLink = prefInfo.m_rgbVisitedColor;
|
|
m_crBrowserBackground = prefInfo.m_rgbBackgroundColor;
|
|
|
|
// m_pPageData may be NULL if we are not an Editor
|
|
// (preferences dialog called from a browser)
|
|
//
|
|
// We are in Document Properties Mode
|
|
// Initialize from current document
|
|
// Note: If the color isn't set,
|
|
// this will get the color from display master colors
|
|
m_crCustomBackground = WFE_LO2COLORREF( m_pPageData->pColorBackground,LO_COLOR_BG );
|
|
m_crCustomText = WFE_LO2COLORREF( m_pPageData->pColorText, LO_COLOR_FG );
|
|
m_crCustomLink = WFE_LO2COLORREF( m_pPageData->pColorLink, LO_COLOR_LINK );
|
|
m_crCustomActiveLink = WFE_LO2COLORREF( m_pPageData->pColorActiveLink, LO_COLOR_ALINK );
|
|
m_crCustomFollowedLink = WFE_LO2COLORREF( m_pPageData->pColorFollowedLink, LO_COLOR_VLINK );
|
|
|
|
if (( m_pPageData->pBackgroundImage )&&( XP_STRLEN(m_pPageData->pBackgroundImage) )) {
|
|
m_csBackgroundImage = m_pPageData->pBackgroundImage;
|
|
// Set checkbox - we have an image
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(1);
|
|
} else {
|
|
// Get the default image from preferences in case they want to use it
|
|
char * szBack = NULL;
|
|
PREF_CopyCharPref("editor.background_image",&szBack);
|
|
m_csBackgroundImage = szBack;
|
|
XP_FREEIF(szBack);
|
|
// But DON'T set the checkbox -- image will not be used
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(0);
|
|
}
|
|
|
|
// If ANY color is set, then we will assume "Custom" mode
|
|
m_bCustomColors =
|
|
( m_pPageData->pColorBackground ||
|
|
m_pPageData->pColorText ||
|
|
m_pPageData->pColorLink ||
|
|
m_pPageData->pColorActiveLink ||
|
|
m_pPageData->pColorFollowedLink );
|
|
|
|
// Save current state so we don't repaint unnecessarily
|
|
m_bWasCustomColors = m_bCustomColors;
|
|
|
|
// Flag to leave image at original location and not save with page
|
|
m_bNoSave = m_pPageData->bBackgroundNoSave;
|
|
|
|
if( m_bCustomColors ){
|
|
// We will start in Custom Colors mode
|
|
// Redraw only if current state is Browser colors
|
|
// Don't unselect scheme
|
|
UseCustomColors(!m_bWasCustomColors, FALSE);
|
|
} else {
|
|
// We will start with Browser colors
|
|
// Don't do update stuff
|
|
UseBrowserColors(FALSE);
|
|
}
|
|
|
|
// Initialize radio buttons and color swatches
|
|
((CButton*)GetDlgItem(IDC_SET_DOC_COLORS))->SetCheck( m_bCustomColors ? 1 : 0 );
|
|
((CButton*)GetDlgItem(IDC_USE_BROWSER_COLORS))->SetCheck( m_bCustomColors ? 0 : 1 );
|
|
|
|
// Disables Apply button (probably don't need to do this)
|
|
SetModified(FALSE);
|
|
UpdateData(FALSE);
|
|
return(TRUE);
|
|
}
|
|
|
|
// Use bUpdateControls = FALSE if initializing,
|
|
// Use TRUE when switching from CustomColor state to Browser colors
|
|
void CDocColorPage::UseBrowserColors(BOOL bRedraw)
|
|
{
|
|
m_bCustomColors = m_bWasCustomColors = FALSE;
|
|
((CButton*)GetDlgItem(IDC_SET_DOC_COLORS))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_USE_BROWSER_COLORS))->SetCheck(1);
|
|
|
|
// Initialize from current preferences
|
|
m_crBackground = m_crBrowserBackground;
|
|
m_crText = m_crBrowserText;
|
|
m_crLink = m_crBrowserLink;
|
|
// Browser doesn't store this color
|
|
m_crActiveLink = m_crBrowserLink;
|
|
m_crFollowedLink = m_crBrowserFollowedLink;
|
|
|
|
if( bRedraw ){
|
|
Invalidate(FALSE);
|
|
}
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
// Redraw dialog to show color swatches -- use bRedraw = FALSE if no colors changing
|
|
void CDocColorPage::UseCustomColors(BOOL bRedraw, BOOL bUnselectScheme)
|
|
{
|
|
m_bCustomColors = m_bWasCustomColors = TRUE;
|
|
((CButton*)GetDlgItem(IDC_SET_DOC_COLORS))->SetCheck(1);
|
|
((CButton*)GetDlgItem(IDC_USE_BROWSER_COLORS))->SetCheck(0);
|
|
|
|
if( bUnselectScheme ){
|
|
// Get selected scheme name
|
|
// (Also gets background image image name, which we ignore here)
|
|
UpdateData(TRUE);
|
|
// TODO: Modify this in the future to
|
|
// reset only if not one of built-in
|
|
// schemes?
|
|
// Get currently-selected scheme
|
|
// to use as default name for saving
|
|
m_csSaveScheme = m_csSelectedScheme;
|
|
|
|
// Unselect scheme list when editing built-in schemes,
|
|
// to remove impression that user can modify them
|
|
((CComboBox*)GetDlgItem(IDC_SCHEME_LIST))->SetCurSel(-1);
|
|
}
|
|
m_crBackground = m_crCustomBackground;
|
|
m_crText = m_crCustomText;
|
|
m_crLink = m_crCustomLink;
|
|
m_crActiveLink = m_crCustomActiveLink;
|
|
m_crFollowedLink = m_crCustomFollowedLink;
|
|
|
|
if( bRedraw ){
|
|
// Redraw color swatches and text samples
|
|
Invalidate(FALSE);
|
|
}
|
|
// Set Apply button states if we are in Properties mode
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
BOOL CDocColorPage::OnKillActive()
|
|
{
|
|
if(m_bActivated){
|
|
if( !UpdateData(TRUE) ){
|
|
return FALSE;
|
|
}
|
|
if ( !m_csBackgroundImage.IsEmpty() ) {
|
|
if ( m_bImageChanged && !m_bValidImage ) {
|
|
if( ((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->GetCheck() &&
|
|
!wfe_ValidateImage( m_pMWContext, m_csBackgroundImage, FALSE /*TRUE*/ ) ){
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(0);
|
|
return FALSE;
|
|
}
|
|
// Send changed image back to editbox in case this is Apply usage
|
|
UpdateData(FALSE);
|
|
}
|
|
}
|
|
if( ((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->GetCheck() &&
|
|
!m_csBackgroundImage.IsEmpty() ){
|
|
int iLastDot = m_csBackgroundImage.ReverseFind('.');
|
|
CString csExt;
|
|
if(iLastDot > 0)
|
|
csExt= m_csBackgroundImage.Mid(iLastDot);
|
|
|
|
//we must check to see if file is a bmp!
|
|
if (0 == csExt.CompareNoCase(".bmp"))
|
|
{
|
|
char *t_outputfilename=wfe_ConvertImage(m_csBackgroundImage.GetBuffer(0),(void *)this,m_pMWContext);
|
|
if (t_outputfilename)
|
|
{
|
|
m_csBackgroundImage=t_outputfilename;
|
|
wfe_ValidateImage( m_pMWContext, m_csBackgroundImage );
|
|
XP_FREE(t_outputfilename);
|
|
UpdateData(FALSE);//we need to update m_csImage!
|
|
}
|
|
else
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void CDocColorPage::OnHelp()
|
|
{
|
|
NetHelp(HELP_DOC_PROPS_APPEARANCE);
|
|
}
|
|
|
|
void CDocColorPage::OnOK()
|
|
{
|
|
//Note: Get and Validate data done in OnKillActive()
|
|
CPropertyPage::OnOK();
|
|
|
|
// never visited this page,
|
|
// or no change when used in Doc. Properties mode -- don't bother
|
|
if( !m_bActivated ||
|
|
!IS_APPLY_ENABLED(this) ){
|
|
return;
|
|
}
|
|
|
|
// Save settings for current document
|
|
// Erase current background image in doc
|
|
if ( m_pPageData->pBackgroundImage ) {
|
|
XP_FREE( m_pPageData->pBackgroundImage );
|
|
m_pPageData->pBackgroundImage = NULL;
|
|
}
|
|
if ( m_bCustomColors ) {
|
|
// Set all custom colors
|
|
WFE_SetLO_ColorPtr( m_crCustomText, &m_pPageData->pColorText );
|
|
WFE_SetLO_ColorPtr( m_crCustomLink, &m_pPageData->pColorLink );
|
|
WFE_SetLO_ColorPtr( m_crCustomActiveLink, &m_pPageData->pColorActiveLink );
|
|
WFE_SetLO_ColorPtr( m_crCustomFollowedLink, &m_pPageData->pColorFollowedLink );
|
|
// Set color even if we use an image
|
|
// (Color tag is written to doc, but image tag overrides)
|
|
WFE_SetLO_ColorPtr( m_crCustomBackground, &m_pPageData->pColorBackground );
|
|
|
|
} else {
|
|
// Use browser colors means storing no tags in document
|
|
// Clear all colors and background image to set the document's tags
|
|
// Note: We ignore the Browser's background image,
|
|
// based on the judgement that if they don't want colors,
|
|
// then the browser's image would be irrelavent (and distracting) while editing
|
|
WFE_FreeLO_Color( &m_pPageData->pColorText );
|
|
WFE_FreeLO_Color( &m_pPageData->pColorLink );
|
|
WFE_FreeLO_Color( &m_pPageData->pColorActiveLink );
|
|
WFE_FreeLO_Color( &m_pPageData->pColorFollowedLink );
|
|
WFE_FreeLO_Color( &m_pPageData->pColorBackground );
|
|
}
|
|
|
|
if( ((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->GetCheck() &&
|
|
!m_csBackgroundImage.IsEmpty() ){
|
|
m_pPageData->pBackgroundImage = XP_STRDUP(m_csBackgroundImage);
|
|
|
|
// Save state of flag from checkbox
|
|
m_pPageData->bBackgroundNoSave = m_bNoSave;
|
|
}
|
|
|
|
// Disable Apply an change "OK" to "Close"
|
|
OkToClose();
|
|
// Clear data-modified flag
|
|
SetModified(FALSE);
|
|
|
|
// This sets tag data for doc and calls front end
|
|
// to setbackground colors and image
|
|
EDT_SetPageData(m_pMWContext, m_pPageData);
|
|
|
|
// This forces redraw of entire window
|
|
// (there is some missing redraw areas at top of window)
|
|
::InvalidateRect(PANECX(m_pMWContext)->GetPane(), NULL, TRUE);
|
|
|
|
if( ((CButton*)GetDlgItem(IDC_USE_AS_DEFAULT))->GetCheck() ){
|
|
// Save current settings to Preferences
|
|
if( ((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->GetCheck() &&
|
|
!m_csBackgroundImage.IsEmpty() ){
|
|
PREF_SetBoolPref("editor.use_background_image",TRUE);
|
|
PREF_SetCharPref("editor.background_image",LPCSTR(m_csBackgroundImage));
|
|
} else {
|
|
PREF_SetBoolPref("editor.use_background_image",FALSE);
|
|
}
|
|
|
|
if ( m_bCustomColors ) {
|
|
PREF_SetBoolPref("editor.use_custom_colors",TRUE);
|
|
PREF_SetCharPref("editor.color_scheme",(char*)LPCSTR(m_csSelectedScheme));
|
|
|
|
PREF_SetColorPrefDWord("editor.text_color",m_crCustomText);
|
|
PREF_SetColorPrefDWord("editor.link_color",m_crCustomLink);
|
|
PREF_SetColorPrefDWord("editor.active_link_color",m_crCustomActiveLink);
|
|
PREF_SetColorPrefDWord("editor.followed_link_color",m_crCustomFollowedLink);
|
|
PREF_SetColorPrefDWord("editor.background_color",m_crCustomBackground);
|
|
} else {
|
|
// Note: don't change previous color preferences
|
|
PREF_SetBoolPref("editor.use_custom_colors",FALSE);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CDocColorPage::OnUseAsDefault()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CDocColorPage::OnColorsRadioButtons()
|
|
{
|
|
// Get checkbox state and change colors appropriately
|
|
// but only if different than previous mode
|
|
m_bCustomColors = IsDlgButtonChecked(IDC_SET_DOC_COLORS);
|
|
if(m_bWasCustomColors != m_bCustomColors){
|
|
if(m_bCustomColors){
|
|
UseCustomColors();
|
|
} else {
|
|
UseBrowserColors();
|
|
}
|
|
// Send the appropriate string to the background image editbox
|
|
UpdateData(FALSE);
|
|
}
|
|
}
|
|
|
|
void CDocColorPage::OnRemoveScheme()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CDocColorPage::OnSaveScheme()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CDocColorPage::OnSelchangeSchemeList()
|
|
{
|
|
// Get data - we want selected color scheme
|
|
UpdateData(TRUE);
|
|
if ( m_csSelectedScheme.IsEmpty() ) {
|
|
return;
|
|
}
|
|
|
|
// Find the scheme in the list and extract color info
|
|
EDT_ColorSchemeData *pColorData;
|
|
XP_List * list_ptr = m_pSchemeData;
|
|
while ((pColorData = (EDT_ColorSchemeData *)XP_ListNextObject(list_ptr))) {
|
|
if ( 0 == XP_STRCMP(pColorData->pSchemeName, m_csSelectedScheme) ) {
|
|
m_crCustomText =
|
|
WFE_LO2COLORREF( &pColorData->ColorText, LO_COLOR_FG );
|
|
m_crCustomLink =
|
|
WFE_LO2COLORREF( &pColorData->ColorLink, LO_COLOR_LINK );
|
|
m_crCustomActiveLink =
|
|
WFE_LO2COLORREF( &pColorData->ColorActiveLink, LO_COLOR_ALINK );
|
|
m_crCustomFollowedLink =
|
|
WFE_LO2COLORREF( &pColorData->ColorFollowedLink, LO_COLOR_VLINK );
|
|
m_crCustomBackground =
|
|
WFE_LO2COLORREF( &pColorData->ColorBackground, LO_COLOR_BG );
|
|
|
|
if ( pColorData->pBackgroundImage ) {
|
|
// Scheme may include a background image...
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(1);
|
|
m_csBackgroundImage = pColorData->pBackgroundImage;
|
|
}
|
|
// ...but if scheme doesn't have image, then leave existing
|
|
// image and checkbox state
|
|
break;
|
|
}
|
|
}
|
|
// Do repaint, but don't unselect the scheme listbox
|
|
UseCustomColors(TRUE, FALSE);
|
|
}
|
|
|
|
BOOL CDocColorPage::ChooseColor(COLORREF * pColor, CColorButton * pButton, COLORREF crDefault, BOOL bBackground )
|
|
{
|
|
RECT rect;
|
|
pButton->GetWindowRect(&rect);
|
|
CColorPicker ColorPicker(this, m_pMWContext, m_bCustomColors ? *pColor : DEFAULT_COLORREF, crDefault,
|
|
bBackground ? IDS_PAGE_BACKGROUND : IDS_TEXT_COLOR, &rect);
|
|
|
|
COLORREF crNew = ColorPicker.GetColor();
|
|
if( crNew != CANCEL_COLORREF ){
|
|
if( crNew == DEFAULT_COLORREF){
|
|
*pColor = crDefault;
|
|
} else {
|
|
*pColor = crNew;
|
|
}
|
|
// Send new color to control
|
|
UpdateData(FALSE);
|
|
// We must be in custom color mode now
|
|
UseCustomColors();
|
|
pButton->Update();
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
void CDocColorPage::OnChooseTextColor()
|
|
{
|
|
ChooseColor(&m_crCustomText, &m_TextColorButton, m_crBrowserText);
|
|
}
|
|
|
|
void CDocColorPage::OnChooseLinkColor()
|
|
{
|
|
ChooseColor(&m_crCustomLink, &m_LinkColorButton, m_crBrowserLink );
|
|
}
|
|
|
|
void CDocColorPage::OnChooseActivelinkColor()
|
|
{
|
|
ChooseColor(&m_crCustomActiveLink, &m_ActiveLinkColorButton, m_crBrowserLink);
|
|
}
|
|
|
|
void CDocColorPage::OnChooseFollowedlinkColor()
|
|
{
|
|
ChooseColor(&m_crCustomFollowedLink, &m_FollowedLinkColorButton, m_crBrowserFollowedLink);
|
|
}
|
|
|
|
void CDocColorPage::OnChooseBkgrndColor()
|
|
{
|
|
if ( ChooseColor(&m_crCustomBackground, &m_BackgroundColorButton, m_crBrowserBackground, TRUE) ) {
|
|
// Change to custom colors if we were Browser mode
|
|
UseCustomColors(!m_bWasCustomColors);
|
|
}
|
|
}
|
|
|
|
void CDocColorPage::OnChangeBkgrndImage()
|
|
{
|
|
// Set flags to trigger validation on Killfocus
|
|
m_bValidImage = FALSE;
|
|
m_bImageChanged = TRUE;
|
|
|
|
UpdateData(TRUE);
|
|
CString csTemp = m_csBackgroundImage;
|
|
csTemp.TrimLeft();
|
|
csTemp.TrimRight();
|
|
|
|
// Set checkbox if there are any characters, clear if empty
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(!csTemp.IsEmpty());
|
|
SetModified(TRUE);
|
|
// NEW: Allow selecting an image if we are in Browser color mode
|
|
// UseCustomColors(!m_bWasCustomColors);
|
|
}
|
|
|
|
void CDocColorPage::OnKillfocusBkgrndImage()
|
|
{
|
|
#if 0
|
|
// Use this if we want to validate as soon as they leave the editbox
|
|
// Get the image name, make it a relative file-type URL,
|
|
// then send it back to editbox
|
|
if( m_bImageChanged &&
|
|
UpdateData(TRUE) ){
|
|
if( !wfe_ValidateImage( m_pMWContext, m_csBackgroundImage, FALSE /*TRUE*/ ) ){
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(0);
|
|
}
|
|
|
|
m_bValidImage = TRUE;
|
|
UpdateData(FALSE);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// Called from the View after saving file to disk -- has new background
|
|
// image URL relative to current document
|
|
void CDocColorPage::SetImageFileSaved(char * pImageURL)
|
|
{
|
|
m_csBackgroundImage = pImageURL;
|
|
UpdateData(FALSE);
|
|
}
|
|
|
|
void CDocColorPage::OnUseBkgrndImage()
|
|
{
|
|
int iUseImageBackground = ((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->GetCheck();;
|
|
SetModified(TRUE);
|
|
if( iUseImageBackground ){
|
|
// User decided to use the image, so do trigger validation
|
|
if( !wfe_ValidateImage( m_pMWContext, m_csBackgroundImage, FALSE /*TRUE*/ ) ){
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(0);
|
|
}
|
|
m_bValidImage = TRUE;
|
|
}
|
|
// TODO: Try to display image in our preview box
|
|
// Need to load file and convert to bitmap
|
|
// Redraw only if current state is Browser colors
|
|
// NEW: Allow selecting an image if we are in Browser color mode
|
|
// UseCustomColors(!m_bWasCustomColors);
|
|
}
|
|
|
|
void CDocColorPage::OnChooseBkgrndImage()
|
|
{
|
|
char * szFilename = wfe_GetExistingImageFileName(this->m_hWnd,
|
|
szLoadString(IDS_SELECT_IMAGE), TRUE);
|
|
if ( szFilename == NULL ){
|
|
return;
|
|
}
|
|
m_csBackgroundImage = szFilename;
|
|
if( !wfe_ValidateImage( m_pMWContext, m_csBackgroundImage, FALSE /*TRUE*/ ) ){
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(0);
|
|
}
|
|
|
|
// Send new name to editbox
|
|
UpdateData(FALSE);
|
|
|
|
XP_FREE( szFilename );
|
|
m_bValidImage = TRUE;
|
|
m_bImageChanged = FALSE;
|
|
|
|
// Note: allow background image without forcing custom colors
|
|
((CButton*)GetDlgItem(IDC_BKGRND_USE_IMAGE))->SetCheck(1);
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CDocColorPage::OnPaint()
|
|
{
|
|
CWnd *pPreview = GetDlgItem(IDC_DOC_COLOR_PREVIEW);
|
|
|
|
// TODO: Quit here if we don't use text preview in "Use Browser Colors" mode
|
|
|
|
// Wierd Windows! We must do this else
|
|
// no painting at all occurs!
|
|
CPaintDC dc(this); // device context for painting
|
|
|
|
// Get the DC to the Preview window
|
|
CDC * pDC = pPreview->GetDC();
|
|
if( !pDC ) return;
|
|
CRect cRect;
|
|
|
|
HPALETTE hOldPal = NULL;
|
|
if( m_hPal ){
|
|
hOldPal = ::SelectPalette( pDC->m_hDC, m_hPal, FALSE );
|
|
}
|
|
|
|
// find out how much area we can draw into
|
|
pPreview->GetClientRect(&cRect);
|
|
|
|
// color for the inside
|
|
CBrush brush(m_crBackground|0x02000000); // Like PALETTERGB
|
|
CBrush * pOldBrush = (CBrush *) pDC->SelectObject(&brush);
|
|
|
|
pDC->LPtoDP(&cRect);
|
|
|
|
cRect.InflateRect(-1,-1);
|
|
|
|
// draw the background
|
|
pDC->FillRect(cRect, &brush);
|
|
|
|
// set the background color
|
|
pDC->SetBkColor(m_crBackground);
|
|
pDC->SetTextColor(m_crText);
|
|
pDC->SetBkMode(TRANSPARENT);
|
|
|
|
// Draw 4 sample text strings
|
|
CString csText = szLoadString(IDS_TEXT);
|
|
|
|
int iNextLine = (cRect.Height() - 2) / 4;
|
|
int iTextTop = 2;
|
|
int iTextLeft = 6;
|
|
|
|
char* csSample = szLoadString(IDS_TEXT);
|
|
CIntlWin::TextOut(0, pDC->GetSafeHdc(), iTextLeft, iTextTop, csSample, strlen(csSample));
|
|
|
|
CSize sizeText = CIntlWin::GetTextExtent(0, pDC->GetSafeHdc(), csSample, strlen(csSample) );
|
|
csSample = szLoadString(IDS_LINK_TEXT);
|
|
pDC->SetTextColor(m_crLink);
|
|
|
|
// This puts "Link text" on 2nd line:
|
|
iTextTop += iNextLine;
|
|
CIntlWin::TextOut(0, pDC->GetSafeHdc(), iTextLeft, iTextTop, csSample, strlen(csSample));
|
|
|
|
// Underline link samples
|
|
sizeText = CIntlWin::GetTextExtent(0, pDC->GetSafeHdc(), csSample, strlen(csSample) );
|
|
CPen penLink(PS_SOLID, 2, m_crLink);
|
|
CPen *pOldPen = pDC->SelectObject(&penLink);
|
|
|
|
pDC->MoveTo(iTextLeft, iTextTop + sizeText.cy + 1);
|
|
pDC->LineTo(iTextLeft + sizeText.cx, iTextTop + sizeText.cy + 1);
|
|
|
|
// Move to a new line of text
|
|
iTextTop += iNextLine;
|
|
csSample = szLoadString(IDS_ACTIVE_LINK_TEXT);
|
|
sizeText = CIntlWin::GetTextExtent(0, pDC->GetSafeHdc(), csSample, strlen(csSample) );
|
|
pDC->SetTextColor(m_crActiveLink);
|
|
CIntlWin::TextOut(0, pDC->GetSafeHdc(), iTextLeft, iTextTop, csSample, strlen(csSample));
|
|
CPen penActive(PS_SOLID, 2, m_crActiveLink);
|
|
pDC->SelectObject(&penActive);
|
|
pDC->MoveTo(iTextLeft, iTextTop + sizeText.cy + 1);
|
|
pDC->LineTo(iTextLeft + sizeText.cx, iTextTop + sizeText.cy + 1);
|
|
|
|
// Next line
|
|
iTextTop += iNextLine;
|
|
csSample = szLoadString(IDS_FOLLOWED_LINK);
|
|
sizeText = CIntlWin::GetTextExtent(0, pDC->GetSafeHdc(), csSample, strlen(csSample) );
|
|
pDC->SetTextColor(m_crFollowedLink);
|
|
CIntlWin::TextOut(0, pDC->GetSafeHdc(), iTextLeft, iTextTop, csSample, strlen(csSample));
|
|
CPen penFollowed(PS_SOLID, 2, m_crFollowedLink);
|
|
pDC->SelectObject(&penFollowed);
|
|
pDC->MoveTo(iTextLeft, iTextTop + sizeText.cy + 1);
|
|
pDC->LineTo(iTextLeft + sizeText.cx, iTextTop + sizeText.cy + 1);
|
|
|
|
pDC->SelectObject(pOldBrush);
|
|
pDC->SelectObject(pOldPen);
|
|
|
|
if(m_hPal){
|
|
::SelectPalette( pDC->m_hDC, hOldPal, FALSE );
|
|
}
|
|
// give the CDC back to the system
|
|
pPreview->ReleaseDC(pDC);
|
|
}
|
|
|
|
|
|
void CDocColorPage::OnNoSave()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
////////////////////////////////////////////////////
|
|
CDocMetaPage::CDocMetaPage(CWnd* pParent, MWContext * pMWContext,
|
|
CEditorResourceSwitcher * pResourceSwitcher)
|
|
: CNetscapePropertyPage(CDocMetaPage::IDD),
|
|
m_bActivated(FALSE),
|
|
m_pMWContext(pMWContext),
|
|
m_pResourceSwitcher(pResourceSwitcher)
|
|
{
|
|
//{{AFX_DATA_INIT(CDocMetaPage)
|
|
//}}AFX_DATA_INIT
|
|
ASSERT(pMWContext);
|
|
|
|
#ifdef XP_WIN32
|
|
// Set the hInstance so we get template from editor's resource DLL
|
|
m_psp.hInstance = AfxGetResourceHandle();
|
|
#endif
|
|
}
|
|
|
|
BOOL CDocMetaPage::OnSetActive()
|
|
{
|
|
if(m_pResourceSwitcher && !m_bActivated){
|
|
// We must be sure we have switched
|
|
// the first time here - before dialog creation
|
|
m_pResourceSwitcher->switchResources();
|
|
}
|
|
if(!CPropertyPage::OnSetActive())
|
|
return(FALSE);
|
|
|
|
if(m_bActivated)
|
|
return(TRUE);
|
|
|
|
// Switch back to EXE's resources
|
|
if( m_pResourceSwitcher ){
|
|
m_pResourceSwitcher->Reset();
|
|
}
|
|
|
|
((CEdit*)GetDlgItem(IDC_VAR_NAME))->LimitText(MAX_META_NAME_LEN);
|
|
((CEdit*)GetDlgItem(IDC_VAR_VALUE))->LimitText(MAX_META_VALUE_LEN);
|
|
|
|
// Get all the items for the lists editbox
|
|
GetMetaData();
|
|
|
|
// Setup list for a new meta item
|
|
((CListBox*)GetDlgItem(IDC_META_LIST))->SetCurSel(m_iMetaCount);
|
|
|
|
m_bActivated = TRUE;
|
|
return(TRUE);
|
|
}
|
|
|
|
void CDocMetaPage::OnSelchangeEquivList()
|
|
{
|
|
CListBox *pEquivList = (CListBox*)GetDlgItem(IDC_EQUIV_LIST);
|
|
CListBox *pMetaList = (CListBox*)GetDlgItem(IDC_META_LIST);
|
|
|
|
int iSelEquiv = pEquivList->GetCurSel();
|
|
int iSelMeta = pMetaList->GetCurSel();
|
|
BOOL bCanSetOrDelete = FALSE;
|
|
|
|
// Only one list can be selected at a time
|
|
// We use selected list as a "radio button" to inform
|
|
// user which list the Name/Value pair applies to
|
|
pMetaList->SetCurSel(-1);
|
|
|
|
// Don't set Name and Value if selecting the last item
|
|
// unless we are switch from other list
|
|
if ( iSelEquiv >= 0 &&
|
|
(-1 != iSelMeta ||
|
|
iSelEquiv != m_iEquivCount) ) {
|
|
CString csEntry;
|
|
CString csName;
|
|
CString csValue;
|
|
pEquivList->GetText(iSelEquiv, csEntry);
|
|
if (GetNameAndValue(csEntry, csName, csValue)) {
|
|
((CEdit*)GetDlgItem(IDC_VAR_NAME))->SetWindowText( LPCSTR(csName) );
|
|
((CEdit*)GetDlgItem(IDC_VAR_VALUE))->SetWindowText( LPCSTR(csValue) );
|
|
bCanSetOrDelete = TRUE;
|
|
} else {
|
|
ClearNameAndValue();
|
|
}
|
|
} else {
|
|
// This will set button states depending on NAME and CONTENT strings
|
|
OnChangeVarNameOrValue();
|
|
return;
|
|
}
|
|
SetModified(TRUE);
|
|
|
|
EnableButtons(bCanSetOrDelete);
|
|
}
|
|
|
|
void CDocMetaPage::OnSelchangeMetaList()
|
|
{
|
|
CListBox *pMetaList = (CListBox*)GetDlgItem(IDC_META_LIST);
|
|
CListBox *pEquivList = (CListBox*)GetDlgItem(IDC_EQUIV_LIST);
|
|
|
|
int iSelMeta = pMetaList->GetCurSel();
|
|
int iSelEquiv = pEquivList->GetCurSel();
|
|
BOOL bCanSetOrDelete = FALSE;
|
|
|
|
// Only one list can be selected at a time
|
|
pEquivList->SetCurSel(-1);
|
|
|
|
// Don't set Name and Value if selecting the last item
|
|
// unless we are switch from other list
|
|
if ( iSelMeta >= 0 &&
|
|
(-1 != iSelEquiv ||
|
|
iSelMeta != m_iMetaCount)) {
|
|
CString csEntry;
|
|
CString csName;
|
|
CString csValue;
|
|
pMetaList->GetText(iSelMeta, csEntry);
|
|
if (GetNameAndValue(csEntry, csName, csValue)) {
|
|
((CEdit*)GetDlgItem(IDC_VAR_NAME))->SetWindowText( LPCSTR(csName) );
|
|
((CEdit*)GetDlgItem(IDC_VAR_VALUE))->SetWindowText( LPCSTR(csValue) );
|
|
bCanSetOrDelete = TRUE;
|
|
} else {
|
|
ClearNameAndValue();
|
|
}
|
|
} else {
|
|
// This will set button states depending on NAME and CONTENT strings
|
|
OnChangeVarNameOrValue();
|
|
return;
|
|
}
|
|
SetModified(TRUE);
|
|
|
|
EnableButtons(bCanSetOrDelete);
|
|
}
|
|
|
|
// Make clicking below list items the same as
|
|
// selecting the last (empty) item so we
|
|
// always change lists when clicked on
|
|
void CDocMetaPage::OnSetfocusEquivList()
|
|
{
|
|
if (((CListBox*)GetDlgItem(IDC_EQUIV_LIST))->GetCurSel() == -1 ) {
|
|
((CListBox*)GetDlgItem(IDC_EQUIV_LIST))->SetCurSel(m_iEquivCount);
|
|
OnSelchangeEquivList();
|
|
}
|
|
}
|
|
|
|
void CDocMetaPage::OnSetfocusMetaList()
|
|
{
|
|
if (((CListBox*)GetDlgItem(IDC_META_LIST))->GetCurSel() == -1 ) {
|
|
((CListBox*)GetDlgItem(IDC_META_LIST))->SetCurSel(m_iMetaCount);
|
|
OnSelchangeMetaList();
|
|
}
|
|
}
|
|
|
|
void CDocMetaPage::OnChangeVarNameOrValue()
|
|
{
|
|
CString csName;
|
|
GetDlgItem(IDC_VAR_NAME)->GetWindowText(csName);
|
|
CString csValue;
|
|
GetDlgItem(IDC_VAR_VALUE)->GetWindowText(csValue);
|
|
CleanupString(csName);
|
|
CleanupString(csValue);
|
|
GetDlgItem(IDC_VAR_SET)->EnableWindow(!csName.IsEmpty() && !csValue.IsEmpty());
|
|
GetDlgItem(IDC_VAR_NEW)->EnableWindow(!csName.IsEmpty() || !csValue.IsEmpty());
|
|
GetDlgItem(IDC_VAR_DELETE)->EnableWindow(!csName.IsEmpty() );
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CDocMetaPage::OnVarSet()
|
|
{
|
|
CListBox *pMetaList = (CListBox*)GetDlgItem(IDC_META_LIST);
|
|
CListBox *pEquivList = (CListBox*)GetDlgItem(IDC_EQUIV_LIST);
|
|
|
|
CString csName;
|
|
GetDlgItem(IDC_VAR_NAME)->GetWindowText(csName);
|
|
CString csValue;
|
|
GetDlgItem(IDC_VAR_VALUE)->GetWindowText(csValue);
|
|
CleanupString(csName);
|
|
CleanupString(csValue);
|
|
|
|
|
|
if ( !csName.IsEmpty() ) {
|
|
int iEqual = csName.Find('=');
|
|
if ( iEqual != -1 ) {
|
|
csName = csName.Left(iEqual);
|
|
CleanupString(csName);
|
|
}
|
|
CString csSearch = csName + "=";
|
|
BOOL bEquiv;
|
|
// We determine which list to use by which was last selected
|
|
if ( pEquivList->GetCurSel() != -1 ) {
|
|
SetMetaData(TRUE, CHAR_STR(csName), CHAR_STR(csValue));
|
|
bEquiv = TRUE;
|
|
} else {
|
|
SetMetaData(FALSE, CHAR_STR(csName), CHAR_STR(csValue));
|
|
bEquiv = FALSE;
|
|
}
|
|
// Get data and rebuild both listboxes
|
|
GetMetaData();
|
|
|
|
// Set the selection to the blank last item in the list
|
|
if (bEquiv) {
|
|
pEquivList->SetCurSel(m_iEquivCount);
|
|
pMetaList->SetCurSel(-1);
|
|
} else {
|
|
pMetaList->SetCurSel(m_iMetaCount);
|
|
pEquivList->SetCurSel(-1);
|
|
}
|
|
// Nothing to delete now
|
|
((CButton*)GetDlgItem(IDC_VAR_DELETE))->EnableWindow(FALSE);
|
|
}
|
|
((CEdit*)GetDlgItem(IDC_VAR_NAME))->SetFocus();
|
|
}
|
|
|
|
void CDocMetaPage::OnVarDelete()
|
|
{
|
|
CListBox *pMetaList = (CListBox*)GetDlgItem(IDC_META_LIST);
|
|
CListBox *pEquivList = (CListBox*)GetDlgItem(IDC_EQUIV_LIST);
|
|
|
|
int nSel = pEquivList->GetCurSel();
|
|
CString csSelString;
|
|
CString csName;
|
|
CString csValue;
|
|
BOOL bEquiv = FALSE;
|
|
|
|
// Get the strings from which ever list has the selected item
|
|
if ( nSel != -1 ) {
|
|
pEquivList->GetText(nSel, csSelString);
|
|
bEquiv = TRUE;
|
|
} else {
|
|
nSel = pMetaList->GetCurSel();
|
|
if ( nSel != -1 ) {
|
|
pMetaList->GetText(nSel, csSelString);
|
|
}
|
|
}
|
|
if ( nSel != -1 && GetNameAndValue(csSelString, csName, csValue ) ) {
|
|
// TRUE = delete entry in EDT data with selected Name-Value pair
|
|
SetMetaData(bEquiv, CHAR_STR(csName), CHAR_STR(csValue), TRUE);
|
|
// Rebuild the list
|
|
GetMetaData();
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
// Set selection to end of list just deleted
|
|
// (or to end of meta list if error deleting)
|
|
if ( bEquiv ) {
|
|
pEquivList->SetCurSel(m_iEquivCount);
|
|
} else {
|
|
pMetaList->SetCurSel(m_iMetaCount);
|
|
}
|
|
((CEdit*)GetDlgItem(IDC_VAR_NAME))->SetFocus();
|
|
}
|
|
|
|
void CDocMetaPage::OnVarNew()
|
|
{
|
|
CListBox *pMetaList = (CListBox*)GetDlgItem(IDC_META_LIST);
|
|
CListBox *pEquivList = (CListBox*)GetDlgItem(IDC_EQUIV_LIST);
|
|
|
|
// Clear the Name-Value edit boxes and
|
|
// remove selection in lists
|
|
ClearNameAndValue();
|
|
|
|
// Reset the selected item to the blank at end
|
|
// of list already selected
|
|
if ( pEquivList->GetCurSel() != -1 ) {
|
|
pEquivList->SetCurSel(m_iEquivCount);
|
|
pMetaList->SetCurSel(-1);
|
|
} else {
|
|
pMetaList->SetCurSel(m_iMetaCount);
|
|
}
|
|
|
|
// Disable all buttons
|
|
EnableButtons(FALSE);
|
|
((CEdit*)GetDlgItem(IDC_VAR_NAME))->SetFocus();
|
|
}
|
|
|
|
void CDocMetaPage::EnableButtons(BOOL bEnable)
|
|
{
|
|
((CButton*)GetDlgItem(IDC_VAR_SET))->EnableWindow(bEnable);
|
|
((CButton*)GetDlgItem(IDC_VAR_DELETE))->EnableWindow(bEnable);
|
|
((CButton*)GetDlgItem(IDC_VAR_NEW))->EnableWindow(bEnable);
|
|
}
|
|
|
|
void CDocMetaPage::ClearNameAndValue()
|
|
{
|
|
((CEdit*)GetDlgItem(IDC_VAR_NAME))->SetWindowText("");
|
|
((CEdit*)GetDlgItem(IDC_VAR_VALUE))->SetWindowText("");
|
|
EnableButtons(FALSE);
|
|
}
|
|
|
|
void CDocMetaPage::GetMetaData()
|
|
{
|
|
CListBox *pMetaList = (CListBox*)GetDlgItem(IDC_META_LIST);
|
|
CListBox *pEquivList = (CListBox*)GetDlgItem(IDC_EQUIV_LIST);
|
|
|
|
// Rebuild our list from the Edit data
|
|
pEquivList->ResetContent();
|
|
pMetaList->ResetContent();
|
|
|
|
// Name = 100, content = 400
|
|
char pName[MAX_META_NAME_LEN+1];
|
|
char pValue[MAX_META_VALUE_LEN+1];
|
|
CString csEntry;
|
|
|
|
m_iEquivCount = 0;
|
|
m_iMetaCount = 0;
|
|
|
|
int count = EDT_MetaDataCount(m_pMWContext);
|
|
for ( int i = 0; i < count; i++ ) {
|
|
EDT_MetaData* pData = EDT_GetMetaData(m_pMWContext, i);
|
|
PR_snprintf(pName, MAX_META_NAME_LEN, "%s", pData->pName);
|
|
PR_snprintf(pValue, MAX_META_VALUE_LEN, "%s", pData->pContent);
|
|
csEntry = pName;
|
|
csEntry += "=";
|
|
csEntry += pValue;
|
|
if ( pData->bHttpEquiv ) {
|
|
pEquivList->AddString(csEntry);
|
|
m_iEquivCount++;
|
|
} else if ( 0 != _stricmp(pName, "Author") &&
|
|
0 != _stricmp(pName, "Description") &&
|
|
0 != _stricmp(pName, "Generator") &&
|
|
0 != _stricmp(pName, "Last-Modified") &&
|
|
0 != _stricmp(pName, "Created") &&
|
|
0 != _stricmp(pName, "Classification") &&
|
|
0 != _stricmp(pName, "Keywords")
|
|
/* && 0 != _stricmp(pName, "SourceURL") */
|
|
) {
|
|
// Skip the fields used in General Info page
|
|
// TODO: PUT META STRINGS IN RESOURCES?
|
|
pMetaList->AddString(csEntry);
|
|
m_iMetaCount++;
|
|
}
|
|
}
|
|
// We add a blank string to each list
|
|
// to ensure a selectable item when none really exists
|
|
// Note: Listbox should NOT be sorted for this to work right
|
|
pEquivList->AddString("");
|
|
pMetaList->AddString("");
|
|
ClearNameAndValue();
|
|
}
|
|
|
|
// Create a MetaData structure and save it
|
|
// This commits change - NO BUFFERING
|
|
// Be sure to strip off spaces and quotes before calling this
|
|
void CDocMetaPage::SetMetaData(BOOL bHttpEquiv, char * pName, char * pValue, BOOL bDelete)
|
|
{
|
|
CListBox *pMetaList = (CListBox*)GetDlgItem(IDC_META_LIST);
|
|
CListBox *pEquivList = (CListBox*)GetDlgItem(IDC_EQUIV_LIST);
|
|
|
|
CString csSelString;
|
|
CString csName;
|
|
CString csValue;
|
|
int nSel = pEquivList->GetCurSel();
|
|
|
|
// Get the strings from which ever list has the selected item
|
|
if ( nSel != -1 )
|
|
{
|
|
pEquivList->GetText(nSel, csSelString);
|
|
} else {
|
|
nSel = pMetaList->GetCurSel();
|
|
if ( nSel != -1 )
|
|
pMetaList->GetText(nSel, csSelString);
|
|
}
|
|
GetNameAndValue(csSelString, csName, csValue);
|
|
|
|
EDT_MetaData *pData = EDT_NewMetaData();
|
|
if ( pData )
|
|
{
|
|
pData->bHttpEquiv = bHttpEquiv;
|
|
if ( pName && XP_STRLEN(pName) > 0 )
|
|
{
|
|
pData->pName = XP_STRDUP(pName);
|
|
if ( pValue && *pValue )
|
|
{
|
|
pData->pContent = XP_STRDUP(pValue);
|
|
if( csName == pName )
|
|
{
|
|
// The NAME in edit box is same as in the selected item,
|
|
// copy the Content value so we replace specifically that item
|
|
pData->pPrevContent = XP_STRDUP(CHAR_STR(csValue));
|
|
// We want to delete this entry
|
|
if( bDelete)
|
|
XP_FREEIF(pData->pContent);
|
|
}
|
|
else
|
|
{
|
|
// We don't match an existing selection
|
|
// Set this so the item is appended to the list
|
|
// rather than replacing the first existing item
|
|
// with the same name. This allows multiple items.
|
|
pData->pPrevContent = pData->pContent;
|
|
}
|
|
EDT_SetMetaData(m_pMWContext, pData);
|
|
}
|
|
OkToClose();
|
|
}
|
|
EDT_FreeMetaData(pData);
|
|
}
|
|
}
|
|
|
|
void CDocMetaPage::OnHelp()
|
|
{
|
|
NetHelp(HELP_DOC_PROPS_ADVANCED);
|
|
}
|
|
|
|
void CDocMetaPage::OnOK()
|
|
{
|
|
CPropertyPage::OnOK();
|
|
|
|
// never visited this page or no change -- don't bother
|
|
if(!m_bActivated ||
|
|
!IS_APPLY_ENABLED(this)){
|
|
return;
|
|
}
|
|
|
|
//EDT_BeginBatchChanges(m_pMWContext);
|
|
|
|
// Set anything in the current Name/Value strings
|
|
OnVarSet();
|
|
|
|
OkToClose();
|
|
|
|
//EDT_EndBatchChanges(m_pMWContext);
|
|
}
|
|
|
|
void CDocMetaPage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CDocMetaPage)
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CDocMetaPage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CDocMetaPage)
|
|
ON_LBN_SELCHANGE(IDC_EQUIV_LIST, OnSelchangeEquivList)
|
|
ON_LBN_SELCHANGE(IDC_META_LIST, OnSelchangeMetaList)
|
|
ON_BN_CLICKED(IDC_VAR_SET, OnVarSet)
|
|
ON_BN_CLICKED(IDC_VAR_DELETE, OnVarDelete)
|
|
ON_BN_CLICKED(IDC_VAR_NEW, OnVarNew)
|
|
ON_EN_CHANGE(IDC_VAR_NAME, OnChangeVarNameOrValue)
|
|
ON_EN_CHANGE(IDC_VAR_VALUE, OnChangeVarNameOrValue)
|
|
ON_LBN_SETFOCUS(IDC_EQUIV_LIST, OnSetfocusEquivList)
|
|
ON_LBN_SETFOCUS(IDC_META_LIST, OnSetfocusMetaList)
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
|
|
// This needs to match contents of Font Size drop-down listbox
|
|
// 0 = "Default", 1 - 7 = font sizes, 8 = "Don't Change"
|
|
static int iDontChangeFontSizeIndex = 8;
|
|
static int iDontChangeFontColorIndex = 0;
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
CCharacterPage::CCharacterPage(CWnd* pParent,
|
|
MWContext *pMWContext,
|
|
CEditorResourceSwitcher * pResourceSwitcher,
|
|
EDT_CharacterData *pData)
|
|
: CNetscapePropertyPage(CCharacterPage::IDD),
|
|
m_pMWContext(pMWContext),
|
|
m_pResourceSwitcher(pResourceSwitcher),
|
|
m_pData(pData),
|
|
m_bActivated(0),
|
|
m_pPageData(0),
|
|
m_crDefault(0),
|
|
m_bCustomColor(0),
|
|
m_iNoChangeFontSize(0),
|
|
m_iNoChangeFontFace(0),
|
|
m_iOtherIndex(0),
|
|
m_bMultipleFonts(0),
|
|
m_bUseDefault(0)
|
|
{
|
|
//{{AFX_DATA_INIT(CCharacterPage)
|
|
//}}AFX_DATA_INIT
|
|
ASSERT(pMWContext);
|
|
// Could we ever get here and have no data?
|
|
// We shouldn't, but maybe we shouldn't assume it here
|
|
// and just gray all controls if no character data.
|
|
ASSERT(pData);
|
|
|
|
#ifdef XP_WIN32
|
|
// Set the hInstance so we get template from editor's resource DLL
|
|
m_psp.hInstance = AfxGetResourceHandle();
|
|
#endif
|
|
}
|
|
|
|
|
|
void CCharacterPage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CCharacterPage)
|
|
// NOTE: the ClassWizard will add DDX and DDV calls here
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CCharacterPage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CCharacterPage)
|
|
ON_BN_CLICKED(IDC_OVERRIDE_COLOR, EnableApply)
|
|
ON_BN_CLICKED(IDC_DONT_CHANGE_COLOR, EnableApply)
|
|
ON_BN_CLICKED(IDC_CHECK_BOLD, EnableApply)
|
|
ON_BN_CLICKED(IDC_CHECK_ITALIC, EnableApply)
|
|
ON_BN_CLICKED(IDC_CHECK_UNDERLINE, EnableApply)
|
|
ON_BN_CLICKED(IDC_CHECK_STRIKETHROUGH, EnableApply)
|
|
ON_BN_CLICKED(IDC_CHECK_BLINK, EnableApply)
|
|
ON_BN_CLICKED(IDC_CHECK_NOBREAK, EnableApply)
|
|
ON_BN_CLICKED(IDC_CHECK_SUBSCRIPT, OnCheckSubscript)
|
|
ON_BN_CLICKED(IDC_CHECK_SUPERSCRIPT, OnCheckSuperscript)
|
|
ON_BN_CLICKED(IDC_CLEAR_ALL_STYLES, OnClearAllStyles)
|
|
ON_BN_CLICKED(IDC_CLEAR_TEXT_STYLES, OnClearTextStyles)
|
|
ON_CBN_SELCHANGE(IDC_FONT_SIZE_COMBO, OnSelchangeFontSize)
|
|
ON_CBN_SELCHANGE(IDC_FONT_SIZE_COMBO2, OnSelchangeFontSize)
|
|
ON_CBN_SELCHANGE(IDC_FONTFACE_COMBO, OnSelchangeFontFace)
|
|
ON_CBN_EDITCHANGE(IDC_FONTFACE_COMBO, OnChangeFontFace)
|
|
ON_CBN_EDITCHANGE(IDC_FONT_SIZE_COMBO2, EnableApply)
|
|
ON_CBN_SELENDOK(IDC_FONTFACE_COMBO, OnSelendokFontFace)
|
|
ON_BN_CLICKED(IDC_CHOOSE_COLOR, OnChooseColor)
|
|
ON_WM_CLOSE()
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
// ON_BN_CLICKED(IDC_CHOOSE_FONT, OnChooseLocalFont)
|
|
|
|
BOOL CCharacterPage::OnSetActive()
|
|
{
|
|
if(m_pResourceSwitcher && !m_bActivated){
|
|
// We must be sure we have switched
|
|
// the first time here - before dialog creation
|
|
m_pResourceSwitcher->switchResources();
|
|
}
|
|
if(!CPropertyPage::OnSetActive())
|
|
return FALSE;
|
|
|
|
if(m_bActivated)
|
|
return TRUE;
|
|
|
|
// Switch back to EXE's resources
|
|
if( m_pResourceSwitcher ){
|
|
m_pResourceSwitcher->Reset();
|
|
}
|
|
|
|
m_pPageData = EDT_GetPageData(m_pMWContext);
|
|
|
|
if( m_pPageData == NULL ){
|
|
return FALSE;
|
|
}
|
|
|
|
// Note: Don't set m_bActivated yet! -- SEE BELOW
|
|
|
|
// Attach color picker button to our color button class
|
|
if( !m_TextColorButton.Subclass(this, IDC_CHOOSE_COLOR, &m_crColor) ){
|
|
return FALSE;
|
|
}
|
|
|
|
// Get the "Default" color used by the Browser
|
|
m_crDefault = prefInfo.m_rgbForegroundColor;
|
|
|
|
// Get current color at cursor or selection
|
|
m_crColor = WFE_GetCurrentFontColor(m_pMWContext);
|
|
|
|
BOOL bUseColor = TRUE;
|
|
if( m_crColor == DEFAULT_COLORREF){
|
|
// Show the default color on the button
|
|
m_crColor = m_crDefault;
|
|
m_bUseDefault = TRUE;
|
|
}
|
|
if( m_crColor == MIXED_COLORREF ){
|
|
// Change the text on radio button to: "Don't change (mixed colors)
|
|
GetDlgItem(IDC_DONT_CHANGE_COLOR)->SetWindowText(szLoadString(IDS_MIXED_COLORS));
|
|
((CButton*)GetDlgItem(IDC_DONT_CHANGE_COLOR))->SetCheck(1);
|
|
} else {
|
|
((CButton*)GetDlgItem(IDC_OVERRIDE_COLOR))->SetCheck(1);
|
|
}
|
|
|
|
// Attach CNSComboBox class to existing control to do user-drawing
|
|
if( !m_FontFaceCombo.Subclass(this, IDC_FONTFACE_COMBO) ){
|
|
return FALSE;
|
|
}
|
|
|
|
// Fill the combobox with global lists of fonts (MUST BE STATIC STRINGS!)
|
|
// This will add "Other..." at bottom of list only if too many fonts are installed
|
|
m_iOtherIndex = wfe_FillFontComboBox(&m_FontFaceCombo);
|
|
|
|
// Get current font face and index to it in font list
|
|
char * pFace = EDT_GetFontFace(m_pMWContext);
|
|
int iFontIndex = m_FontFaceCombo.FindSelectedOrSetText(pFace);
|
|
|
|
// Save whether or not we have mulitple fonts at startup
|
|
m_bMultipleFonts = (pFace == NULL);
|
|
char * pNoChangeText = m_bMultipleFonts ? ed_pMixedFonts : ed_pDontChange;
|
|
// Add the "Dont Change" item at the bottom
|
|
m_iNoChangeFontFace = m_FontFaceCombo.AddString(pNoChangeText);
|
|
|
|
// Set the message explaining the font after selecting it in listbox
|
|
SetFontFaceMessage(iFontIndex);
|
|
|
|
// We need to rebuild font size display if font face changes from
|
|
// Variable to Fixed Width or vice versa, so monitor state
|
|
m_bFixedWidth = (iFontIndex == 1);
|
|
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_BOLD)),TF_BOLD);
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_ITALIC)), TF_ITALIC);
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_UNDERLINE)), TF_UNDERLINE);
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_STRIKETHROUGH)), TF_STRIKEOUT);
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_SUPERSCRIPT)), TF_SUPER);
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_SUBSCRIPT)), TF_SUB);
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_BLINK)), TF_BLINK);
|
|
InitCheckbox(((CButton*)GetDlgItem(IDC_CHECK_NOBREAK)), TF_NOBREAK);
|
|
|
|
// We have 2 comboboxes: 1 droplist for regular (relative size) mode,
|
|
// and combobox with edit style for advanced mode
|
|
if( !m_FontSizeCombo.Subclass(this, wfe_iFontSizeMode == ED_FONTSIZE_ADVANCED ?
|
|
IDC_FONT_SIZE_COMBO2 : IDC_FONT_SIZE_COMBO) ){
|
|
return FALSE;
|
|
}
|
|
// Hide the combobox we will not use
|
|
GetDlgItem(wfe_iFontSizeMode == ED_FONTSIZE_ADVANCED ? IDC_FONT_SIZE_COMBO : IDC_FONT_SIZE_COMBO2)->ShowWindow(SW_HIDE);
|
|
|
|
|
|
// Fill Font Size combo. 3rd param is TRUE if font face = Fixed Width
|
|
wfe_FillFontSizeCombo(m_pMWContext, &m_FontSizeCombo, m_bFixedWidth);
|
|
|
|
// Add the "Dont Change" item at the bottom
|
|
m_iNoChangeFontSize = m_FontSizeCombo.AddString(ed_pDontChange);
|
|
|
|
// Change message to that describing advanced mode
|
|
if( wfe_iFontSizeMode == ED_FONTSIZE_ADVANCED ){
|
|
GetDlgItem(IDC_FONT_SIZE_MSG)->SetWindowText(szLoadString(IDS_ADVANCED_FONTSIZE_MSG));
|
|
}
|
|
|
|
// Set FontSize combobox selected
|
|
int iSel = -1;
|
|
if( m_pData->mask & TF_FONT_SIZE ){
|
|
// If no size, use default instead
|
|
iSel = m_pData->iSize ? m_pData->iSize - 1 : 2;
|
|
m_FontSizeCombo.SetCurSel(iSel);
|
|
// Weird -- we must set string manually
|
|
char * pSelected = (char*)m_FontSizeCombo.GetItemData(iSel);
|
|
if( *pSelected == '_'){
|
|
pSelected++;
|
|
}
|
|
m_FontSizeCombo.SetWindowText(pSelected);
|
|
} else if( (m_pData->mask & TF_FONT_POINT_SIZE) && m_pData->iPointSize ){
|
|
char pSize[16];
|
|
wsprintf(pSize, "%d", m_pData->iPointSize);
|
|
m_FontSizeCombo.FindSelectedOrSetText(pSize, MAX_FONT_SIZE);
|
|
}
|
|
|
|
// We set this last because initialization
|
|
// uses it to know first-time through
|
|
m_bActivated = TRUE;
|
|
|
|
// Send data to controls
|
|
UpdateData(FALSE);
|
|
return(TRUE);
|
|
}
|
|
|
|
void CCharacterPage::OnHelp()
|
|
{
|
|
NetHelp(HELP_PROPS_CHARACTER);
|
|
}
|
|
|
|
BOOL CCharacterPage::OnKillActive()
|
|
|
|
{
|
|
if( ! GetAndValidateFontSizes() ){
|
|
return FALSE;
|
|
}
|
|
// Contrary to MFC help, this does NOT call our OnOK
|
|
return CPropertyPage::OnKillActive();
|
|
}
|
|
|
|
BOOL CCharacterPage::GetAndValidateFontSizes()
|
|
{
|
|
int iSel = m_FontSizeCombo.GetCurSel();
|
|
if( iSel == iDontChangeFontSizeIndex ||
|
|
(iSel == -1 && wfe_iFontSizeMode != ED_FONTSIZE_ADVANCED) ){
|
|
// Ignore this attribute
|
|
FE_CLEAR_BIT(m_pData->mask, TF_FONT_SIZE);
|
|
return TRUE;
|
|
}
|
|
if( wfe_iFontSizeMode == ED_FONTSIZE_ADVANCED){
|
|
CString csAbsoluteSize;
|
|
m_FontSizeCombo.GetWindowText(csAbsoluteSize);
|
|
csAbsoluteSize.TrimLeft();
|
|
csAbsoluteSize.TrimRight();
|
|
|
|
// Empty is OK - same as "Don't change"
|
|
// So get the Relative size
|
|
if( csAbsoluteSize.IsEmpty() ){
|
|
// We may be ignore this attribute if no
|
|
// relative size is selected
|
|
FE_CLEAR_BIT(m_pData->mask, TF_FONT_SIZE);
|
|
m_pData->iPointSize = 0;
|
|
} else if( iSel == -1 || iSel >= MAX_FONT_SIZE ){
|
|
char* pEnd;
|
|
int32 iSize = (int)strtol( LPCSTR(csAbsoluteSize), &pEnd, 10 );
|
|
// Bad conversion if end pointer isn't at terminal null;
|
|
if( *pEnd != '\0' ||
|
|
iSize < MIN_FONT_SIZE_RELATIVE ||
|
|
iSize > ED_FONT_POINT_SIZE_MAX ){
|
|
|
|
char szMessage[256];
|
|
// Construct a string showing correct range
|
|
wsprintf( szMessage, szLoadString(IDS_INTEGER_RANGE_ERROR),
|
|
ED_FONT_POINT_SIZE_MIN, ED_FONT_POINT_SIZE_MAX );
|
|
|
|
// Notify user with similar message to the DDV_ validation system
|
|
MessageBox(szMessage, szLoadString(AFX_IDS_APP_TITLE), MB_ICONEXCLAMATION | MB_OK);
|
|
return FALSE;
|
|
}
|
|
if( iSize >= MIN_FONT_SIZE_RELATIVE && iSize <= 4 ){
|
|
// Assume they really wanted the Relative scale
|
|
iSel = iSize + 3;
|
|
m_pData->iPointSize = 0;
|
|
} else {
|
|
// Size is OK - store it
|
|
m_pData->iPointSize = (int16)iSize;
|
|
|
|
// Set bit mask to indicate we want PointSize, not HTML relative size
|
|
FE_CLEAR_BIT(m_pData->mask, TF_FONT_SIZE);
|
|
FE_SET_BIT(m_pData->mask, TF_FONT_POINT_SIZE);
|
|
FE_SET_BIT(m_pData->values, TF_FONT_POINT_SIZE);
|
|
m_pData->iPointSize = (int16)iSize;
|
|
}
|
|
}
|
|
}
|
|
|
|
if( m_pData->iPointSize == 0){
|
|
if( m_iNoChangeFontSize > 0 && iSel == m_iNoChangeFontSize ){
|
|
// Clear the bit so we will ignore this attribute
|
|
FE_CLEAR_BIT(m_pData->mask, TF_FONT_SIZE);
|
|
} else if( iSel >= 0 && iSel < MAX_FONT_SIZE ){
|
|
// Set the size to the designated value - XP handles default
|
|
// (3rd item = "default", i.e., no SIZE param written)
|
|
FE_SET_BIT(m_pData->mask, TF_FONT_SIZE);
|
|
FE_SET_BIT(m_pData->values, TF_FONT_SIZE);
|
|
m_pData->iSize = iSel+1;
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void CCharacterPage::OnOK()
|
|
{
|
|
CPropertyPage::OnOK();
|
|
|
|
// never visited this page or no change -- don't bother
|
|
if(!m_bActivated ||
|
|
!IS_APPLY_ENABLED(this)){
|
|
return;
|
|
}
|
|
|
|
//EDT_BeginBatchChanges(m_pMWContext);
|
|
if( !GetAndValidateFontSizes() ){
|
|
return;
|
|
}
|
|
|
|
// Set color if checkbox is checked
|
|
if( ((CButton*)GetDlgItem(IDC_OVERRIDE_COLOR))->GetCheck() ){
|
|
if( m_bUseDefault ){
|
|
// Set the color to the default size by
|
|
// setting mask bit and clearing value
|
|
FE_SET_BIT(m_pData->mask, TF_FONT_COLOR);
|
|
FE_CLEAR_BIT(m_pData->values, TF_FONT_COLOR);
|
|
} else {
|
|
// Set the color to the designated value
|
|
FE_SET_BIT(m_pData->mask, TF_FONT_COLOR);
|
|
FE_SET_BIT(m_pData->values, TF_FONT_COLOR);
|
|
WFE_SetLO_ColorPtr( m_crColor, &m_pData->pColor );
|
|
}
|
|
} else {
|
|
// Ignore this attribute
|
|
FE_CLEAR_BIT(m_pData->mask, TF_FONT_COLOR);
|
|
}
|
|
|
|
int iSel = m_FontFaceCombo.GetCurSel();
|
|
CString csFace;
|
|
char * pNewFace = NULL;
|
|
if( iSel == -1 ){
|
|
m_FontFaceCombo.GetWindowText(csFace);
|
|
csFace.TrimLeft();
|
|
csFace.TrimRight();
|
|
if( !csFace.IsEmpty() ){
|
|
// We have a local font name
|
|
pNewFace = (char*)LPCSTR(csFace);
|
|
FE_SET_BIT(m_pData->mask, TF_FONT_FACE);
|
|
}
|
|
} else if( m_iNoChangeFontFace > 0 && iSel == m_iNoChangeFontFace ){
|
|
// Clear the bit so we will ignore this attribute
|
|
FE_CLEAR_BIT(m_pData->mask, TF_FONT_FACE);
|
|
} else {
|
|
// We may be sure of the font and its in our list
|
|
// (EDT_SetFontFace will set default variable or fixed width
|
|
// correctly using the supplied "font face" string
|
|
FE_SET_BIT(m_pData->mask, TF_FONT_FACE);
|
|
// Get pointer to fontface string in combobox
|
|
pNewFace = (char*)m_FontFaceCombo.GetItemData(iSel);
|
|
}
|
|
// Set the bits in our data struct
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_BOLD)), TF_BOLD);
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_ITALIC)), TF_ITALIC);
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_UNDERLINE)), TF_UNDERLINE);
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_STRIKETHROUGH)), TF_STRIKEOUT);
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_SUPERSCRIPT)), TF_SUPER);
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_SUBSCRIPT)), TF_SUB);
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_BLINK)), TF_BLINK);
|
|
SetCharacterStyle(((CButton*)GetDlgItem(IDC_CHECK_NOBREAK)), TF_NOBREAK);
|
|
|
|
// Set font face in the supplied struct and set all character data
|
|
EDT_SetFontFace(m_pMWContext, m_pData, iSel, pNewFace);
|
|
|
|
OkToClose();
|
|
//EDT_EndBatchChanges(m_pMWContext);
|
|
}
|
|
|
|
void CCharacterPage::InitCheckbox(CButton *pButton, ED_TextFormat tf)
|
|
{
|
|
// Get current style but clear all relevant button-style bits
|
|
UINT nStyle = CASTUINT(pButton->GetButtonStyle() & 0xFFFFFFF0L);
|
|
|
|
if( m_pData->mask & tf ) {
|
|
// We know the style, so we can use 2-state boxes
|
|
nStyle |= BS_AUTOCHECKBOX;
|
|
} else {
|
|
// We don't know the style, so we use 3-state boxes
|
|
// so user can return to the "don't change" state
|
|
nStyle |= BS_AUTO3STATE;
|
|
}
|
|
pButton->SetButtonStyle(nStyle);
|
|
SetCheck(pButton, tf);
|
|
}
|
|
|
|
int CCharacterPage::SetCheck(CButton *pButton, ED_TextFormat tf)
|
|
{
|
|
int iCheck = 2; // Uncertain/mixed state
|
|
if( m_pData->mask & tf){
|
|
iCheck = (m_pData->values & tf) ? 1 : 0;
|
|
}
|
|
pButton->SetCheck(iCheck);
|
|
return iCheck;
|
|
}
|
|
|
|
void CCharacterPage::SetCharacterStyle(CButton *pButton, ED_TextFormat tf)
|
|
{
|
|
int iCheck = pButton->GetCheck();
|
|
|
|
if( iCheck == 2 ){
|
|
// Don't change this style - clear mask bit
|
|
FE_CLEAR_BIT(m_pData->mask,tf);
|
|
} else {
|
|
// Set the style -- set mask bit
|
|
FE_SET_BIT(m_pData->mask, tf);
|
|
}
|
|
|
|
if( iCheck == 1){
|
|
// Set the value bit
|
|
FE_SET_BIT(m_pData->values,tf);
|
|
} else {
|
|
FE_CLEAR_BIT(m_pData->values,tf);
|
|
}
|
|
}
|
|
|
|
void CCharacterPage::OnChooseLocalFont()
|
|
{
|
|
CFontDialog dlg(NULL, CF_SCREENFONTS | CF_TTONLY, NULL, this);
|
|
if( dlg.DoModal() ){
|
|
CString csNewFont = dlg.GetFaceName();
|
|
m_FontFaceCombo.SetWindowText(LPCSTR(csNewFont));
|
|
GetDlgItem(IDC_FONTFACE_MSG)->SetWindowText(szLoadString(IDS_TRUE_TYPE));
|
|
SetModified(TRUE);
|
|
}
|
|
}
|
|
|
|
void CCharacterPage::OnChooseColor()
|
|
{
|
|
// Get the combobox location so we popup new dialog just under it
|
|
RECT rect;
|
|
GetDlgItem(IDC_CHOOSE_COLOR)->GetWindowRect(&rect);
|
|
|
|
CColorPicker ColorPicker(this, m_pMWContext, m_crColor, DEFAULT_COLORREF, IDS_TEXT_COLOR, &rect);
|
|
COLORREF crNew = ColorPicker.GetColor();
|
|
|
|
if( crNew != CANCEL_COLORREF ){
|
|
if( crNew == DEFAULT_COLORREF ){
|
|
m_crColor = m_crDefault;
|
|
m_bUseDefault = TRUE;
|
|
} else {
|
|
m_crColor = crNew;
|
|
m_bUseDefault = FALSE;
|
|
}
|
|
// Refresh color button
|
|
m_TextColorButton.Update();
|
|
// Set radio buttons to show "Use color"
|
|
((CButton*)GetDlgItem(IDC_OVERRIDE_COLOR))->SetCheck(1);
|
|
((CButton*)GetDlgItem(IDC_DONT_CHANGE_COLOR))->SetCheck(0);
|
|
SetModified(TRUE);
|
|
}
|
|
}
|
|
|
|
void CCharacterPage::EnableApply()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CCharacterPage::OnCheckSubscript()
|
|
{
|
|
// Super and Sub are mutually exclusive
|
|
// Problem: This wipes out mixed-state items!
|
|
if( ((CButton*)GetDlgItem(IDC_CHECK_SUBSCRIPT))->GetCheck() == 1 ){
|
|
((CButton*)GetDlgItem(IDC_CHECK_SUPERSCRIPT))->SetCheck(0);
|
|
}
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CCharacterPage::OnCheckSuperscript()
|
|
{
|
|
// Super and Sub are mutually exclusive
|
|
// Problem: This wipes out mixed-state items!
|
|
if( ((CButton*)GetDlgItem(IDC_CHECK_SUPERSCRIPT))->GetCheck() == 1 ){
|
|
((CButton*)GetDlgItem(IDC_CHECK_SUBSCRIPT))->SetCheck(0);
|
|
}
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
// Next 2 don't change the doc,
|
|
// just change the control states
|
|
void CCharacterPage::OnClearAllStyles()
|
|
{
|
|
// Remove all styles in the controls
|
|
|
|
// Set to the "default" font size
|
|
m_FontSizeCombo.SetCurSel(2);
|
|
// Update the display
|
|
OnSelchangeFontSize();
|
|
|
|
// Set to default color
|
|
m_bUseDefault = TRUE;
|
|
m_crColor = m_crDefault;
|
|
m_TextColorButton.Update();
|
|
|
|
((CButton*)GetDlgItem(IDC_DONT_CHANGE_COLOR))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_OVERRIDE_COLOR))->SetCheck(1);
|
|
|
|
// Set to default variable font face
|
|
m_FontFaceCombo.SetCurSel(0);
|
|
OnSelchangeFontFace();
|
|
|
|
OnClearTextStyles();
|
|
}
|
|
|
|
void CCharacterPage::OnClearTextStyles()
|
|
{
|
|
((CButton*)GetDlgItem(IDC_CHECK_BOLD))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_CHECK_ITALIC))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_CHECK_UNDERLINE))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_CHECK_STRIKETHROUGH))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_CHECK_SUPERSCRIPT))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_CHECK_SUBSCRIPT))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_CHECK_BLINK))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_CHECK_NOBREAK))->SetCheck(0);
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CCharacterPage::OnChangeFontFace()
|
|
{
|
|
GetDlgItem(IDC_FONTFACE_MSG)->SetWindowText("");
|
|
SetModified(TRUE);
|
|
// Do any autosearch here?
|
|
}
|
|
|
|
void CCharacterPage::SetFontFaceMessage(int iSel)
|
|
{
|
|
UINT nID;
|
|
if( iSel == -1 ||
|
|
(iSel == m_iNoChangeFontFace && !m_bMultipleFonts) ||
|
|
(m_iOtherIndex && iSel == m_iOtherIndex) ){
|
|
GetDlgItem(IDC_FONTFACE_MSG)->SetWindowText("");
|
|
return;
|
|
}
|
|
if( iSel == m_iNoChangeFontFace ){
|
|
nID = IDS_MULTIPLE_FONTS;
|
|
} else if( iSel == 0){
|
|
nID = IDS_DEFAULT_VARIABLE;
|
|
} else if ( iSel == 1 ){
|
|
nID = IDS_DEFAULT_FIXED_WIDTH;
|
|
} else {
|
|
char * pFace = (char*)m_FontFaceCombo.GetItemData(iSel);
|
|
// Give a different message if we find an "XP" font face
|
|
if( pFace != EDT_TranslateToXPFontFace(pFace) ){
|
|
nID = IDS_XP_FONT;
|
|
} else {
|
|
nID = IDS_LOCAL_FONT;
|
|
}
|
|
}
|
|
GetDlgItem(IDC_FONTFACE_MSG)->SetWindowText(szLoadString(nID));
|
|
}
|
|
|
|
void CCharacterPage::OnSelchangeFontFace()
|
|
{
|
|
int iSel = m_FontFaceCombo.GetCurSel();
|
|
if( m_iOtherIndex && iSel == m_iOtherIndex ){
|
|
m_FontFaceCombo.SetCurSel(-1);
|
|
iSel = -1;
|
|
}
|
|
if( iSel >= 0 ){
|
|
char * pSelected = (char*)m_FontFaceCombo.GetItemData(iSel);
|
|
if( *pSelected == '_'){
|
|
pSelected++;
|
|
}
|
|
m_FontFaceCombo.SetWindowText(pSelected);
|
|
}
|
|
// Rebuild font size list if changing from variable to fixed width or vice versa
|
|
if( m_bFixedWidth != (iSel == 1) ){
|
|
m_bFixedWidth = (iSel == 1);
|
|
wfe_FillFontSizeCombo(m_pMWContext, &m_FontSizeCombo, m_bFixedWidth);
|
|
}
|
|
SetFontFaceMessage(iSel);
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CCharacterPage::OnSelendokFontFace()
|
|
{
|
|
int iSel = m_FontFaceCombo.GetCurSel();
|
|
if( m_iOtherIndex &&
|
|
iSel == m_iOtherIndex ){
|
|
m_FontFaceCombo.SetCurSel(-1);
|
|
OnChooseLocalFont();
|
|
return;
|
|
}
|
|
}
|
|
|
|
void CCharacterPage::OnSelchangeFontSize()
|
|
{
|
|
// User can use the last item to not change any sizes,
|
|
// but it looks better to show blank combobox field
|
|
int iSel = m_FontSizeCombo.GetCurSel();
|
|
if( iSel == iDontChangeFontSizeIndex ){
|
|
m_FontSizeCombo.SetCurSel(-1);
|
|
}
|
|
if( wfe_iFontSizeMode == ED_FONTSIZE_ADVANCED ){
|
|
// We must set text manually in edit box
|
|
if( iSel >= 0 ){
|
|
char * pSelected = (char*)m_FontSizeCombo.GetItemData(iSel);
|
|
if( *pSelected == '_'){
|
|
pSelected++;
|
|
}
|
|
m_FontSizeCombo.SetWindowText(pSelected);
|
|
} else {
|
|
m_FontSizeCombo.SetWindowText("");
|
|
}
|
|
} else {
|
|
m_pData->iPointSize = 0;
|
|
}
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CCharacterPage::OnClose()
|
|
{
|
|
// Free extra data we allocated
|
|
if( m_pPageData ){
|
|
EDT_FreePageData(m_pPageData);
|
|
}
|
|
CNetscapePropertyPage::OnClose();
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
/*
|
|
// For quick reference:
|
|
typedef enum {
|
|
ED_LIST_TYPE_DEFAULT,
|
|
ED_LIST_TYPE_DIGIT,
|
|
ED_LIST_TYPE_ROMAN,
|
|
ED_LIST_TYPE_BIG_LETTERS,
|
|
ED_LIST_TYPE_SMALL_LETTERS,
|
|
ED_LIST_TYPE_CIRCLE,
|
|
ED_LIST_TYPE_SQUARE,
|
|
ED_LIST_TYPE_DISC,
|
|
} ED_ListType;
|
|
|
|
|
|
struct _EDT_ListData {
|
|
intn iTagType; P_UNUM_LIST, P_NUM_LIST, P_BLOCKQUOTE, P_DESC_LIST
|
|
Bool bCompact;
|
|
ED_ListType eType;
|
|
int iStart; automatically maps, start is 1
|
|
};
|
|
*/
|
|
|
|
//Container styles:
|
|
enum {
|
|
ED_NO_CONTAINER,
|
|
ED_LIST,
|
|
ED_BLOCKQUOTE
|
|
};
|
|
|
|
|
|
// List Styles
|
|
enum {
|
|
ED_UNUM_LIST,
|
|
ED_NUM_LIST,
|
|
ED_DESC_LIST
|
|
};
|
|
|
|
|
|
//Numbered List Item styles
|
|
enum {
|
|
ED_AUTOMATIC,
|
|
ED_DIGIT,
|
|
ED_BIG_ROMAN,
|
|
ED_SMALL_ROMAN,
|
|
ED_BIG_LETTERS,
|
|
ED_SMALL_LETTERS
|
|
};
|
|
|
|
//Unumbered List Item styles
|
|
enum {
|
|
ED_SOLID_CIRCLE = 1, // We share ED_AUTOMATIC,
|
|
ED_OPEN_CIRCLE,
|
|
ED_SOLID_SQUARE
|
|
};
|
|
|
|
#define ED_DEFAULT -1
|
|
|
|
// This is in edframe.cpp. Its the tag types that map
|
|
// onto the listbox index, just as in the paragraph format toolbar
|
|
extern TagType FEED_nParagraphTags[];
|
|
|
|
// Fixed index values for Paragraph tags list
|
|
#define ED_PARA_NORMAL 0
|
|
#define ED_PARA_LIST 9
|
|
#define ED_PARA_DESC_TITLE 10
|
|
#define ED_PARA_DESC_TEXT 11
|
|
|
|
/////////////////////////////////////////
|
|
// Paragraph / Lists / Alignment
|
|
//
|
|
CParagraphPage::CParagraphPage(CWnd* pParent, MWContext * pMWContext,
|
|
CEditorResourceSwitcher * pResourceSwitcher)
|
|
: CNetscapePropertyPage(CParagraphPage::IDD),
|
|
m_pMWContext(pMWContext),
|
|
m_pResourceSwitcher(pResourceSwitcher),
|
|
m_pListData(NULL),
|
|
m_bActivated(0),
|
|
m_Align(ED_ALIGN_LEFT),
|
|
m_iParagraphStyle(0),
|
|
m_iContainerStyle(0),
|
|
m_iListStyle(-1),
|
|
m_iBulletStyle(0),
|
|
m_iNumberStyle(0),
|
|
m_iMixedStyle(0)
|
|
{
|
|
//{{AFX_DATA_INIT(CParagraphPage)
|
|
m_iStartNumber = 1;
|
|
m_bCompact = FALSE;
|
|
//}}AFX_DATA_INIT
|
|
ASSERT(pMWContext);
|
|
|
|
#ifdef XP_WIN32
|
|
// Set the hInstance so we get template from editor's resource DLL
|
|
m_psp.hInstance = AfxGetResourceHandle();
|
|
#endif
|
|
}
|
|
|
|
|
|
void CParagraphPage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CParagraphPage)
|
|
DDX_Text(pDX, IDC_LIST_START_NUMBER, m_iStartNumber);
|
|
DDV_MinMaxInt(pDX, m_iStartNumber, 1, 10000);
|
|
DDX_Check(pDX, IDC_COMPACT_LIST, m_bCompact);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CParagraphPage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CParagraphPage)
|
|
ON_BN_CLICKED(IDC_ALIGN_CENTER, OnAlignCenter)
|
|
ON_BN_CLICKED(IDC_ALIGN_LEFT, OnAlignLeft)
|
|
ON_BN_CLICKED(IDC_ALIGN_RIGHT, OnAlignRight)
|
|
ON_CBN_SELCHANGE(IDC_CONTAINER_STYLES, OnSelchangeContainerStyle)
|
|
ON_CBN_SELCHANGE(IDC_LIST_ITEM_STYLES, OnSelchangeListItemStyle)
|
|
ON_EN_CHANGE(IDC_LIST_START_NUMBER, OnChangeListStartNumber)
|
|
ON_CBN_SELCHANGE(IDC_LIST_STYLES, OnSelchangeListStyles)
|
|
ON_CBN_SELCHANGE(IDC_PARAGRAPH_STYLES, OnSelchangeParagraphStyles)
|
|
ON_WM_CLOSE()
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
|
|
BOOL CParagraphPage::OnSetActive()
|
|
{
|
|
int i;
|
|
if(m_pResourceSwitcher && !m_bActivated){
|
|
// We must be sure we have switched
|
|
// the first time here - before dialog creation
|
|
m_pResourceSwitcher->switchResources();
|
|
}
|
|
if(!CPropertyPage::OnSetActive())
|
|
return(FALSE);
|
|
|
|
if(m_bActivated)
|
|
return(TRUE);
|
|
|
|
|
|
// Switch back to EXE's resources
|
|
if( m_pResourceSwitcher ){
|
|
m_pResourceSwitcher->Reset();
|
|
}
|
|
|
|
m_bActivated = TRUE;
|
|
CComboBox * pContainerStyles;
|
|
CComboBox * pListStyles;
|
|
|
|
// Initialize alignment param and radio buttons
|
|
m_Align = EDT_GetParagraphAlign( m_pMWContext );
|
|
|
|
switch (m_Align) {
|
|
case ED_ALIGN_CENTER:
|
|
case ED_ALIGN_ABSCENTER:
|
|
((CButton*)GetDlgItem(IDC_ALIGN_CENTER))->SetCheck(1);
|
|
break;
|
|
case ED_ALIGN_RIGHT:
|
|
((CButton*)GetDlgItem(IDC_ALIGN_RIGHT))->SetCheck(1);
|
|
break;
|
|
default:
|
|
((CButton*)GetDlgItem(IDC_ALIGN_LEFT))->SetCheck(1);
|
|
// We end up here for ED_ALIGN_DEFAULT as well as ED_ALIGN_LEFT
|
|
m_Align = ED_ALIGN_LEFT;
|
|
break;
|
|
}
|
|
|
|
CComboBox *pParagraphStyles = (CComboBox*)GetDlgItem(IDC_PARAGRAPH_STYLES);
|
|
|
|
m_ParagraphFormat = EDT_GetParagraphFormatting(m_pMWContext);
|
|
//Try to get Container data
|
|
// (We can't do this just for list because we may be
|
|
// a block quote, which may have any paragraph style)
|
|
m_pListData = EDT_GetListData(m_pMWContext);
|
|
|
|
// Fill the Paragraph styles list:
|
|
for ( i = 0; FEED_nParagraphTags[i] != P_UNKNOWN; i++ ){
|
|
// Major KLUDGE: We want "Description List" to appear in Toolbar Combobox,
|
|
// but not in this combobox - we put it in List Styles combo instead
|
|
// if( FEED_nParagraphTags[i] != P_DESC_LIST)
|
|
pParagraphStyles->AddString(szLoadString(
|
|
CASTUINT(ID_LIST_TEXT_PARAGRAPH_BASE+FEED_nParagraphTags[i])));
|
|
}
|
|
// We will add another item at end to allow user to see
|
|
// "Mixed (don't change)" in listbox
|
|
m_iMixedStyle = i;
|
|
|
|
// Figure out the Paragraph style:
|
|
if( m_ParagraphFormat == P_UNKNOWN ){
|
|
m_iParagraphStyle = m_iMixedStyle;
|
|
pParagraphStyles->AddString(szLoadString(IDS_MIXED_STYLE));
|
|
}
|
|
else for( i = 0; FEED_nParagraphTags[i] != P_UNKNOWN; i++ ){
|
|
if( FEED_nParagraphTags[i] == m_ParagraphFormat ){
|
|
m_iParagraphStyle = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( m_pListData ) {
|
|
m_iContainerStyle = ED_LIST;
|
|
switch( m_pListData->iTagType ){
|
|
case P_UNUM_LIST:
|
|
m_iListStyle = ED_UNUM_LIST;
|
|
break;
|
|
case P_NUM_LIST:
|
|
m_iListStyle = ED_NUM_LIST;
|
|
m_iStartNumber = CASTINT(m_pListData->iStart);
|
|
break;
|
|
case P_BLOCKQUOTE:
|
|
m_iContainerStyle = ED_BLOCKQUOTE;
|
|
m_iListStyle = ED_DEFAULT;
|
|
break;
|
|
case P_DESC_LIST:
|
|
m_iListStyle = ED_DESC_LIST;
|
|
break;
|
|
}
|
|
|
|
m_iBulletStyle = m_iNumberStyle = ED_DEFAULT;
|
|
switch( m_pListData->eType ){
|
|
case ED_LIST_TYPE_DIGIT:
|
|
m_iNumberStyle = ED_DIGIT;
|
|
break;
|
|
case ED_LIST_TYPE_BIG_ROMAN:
|
|
m_iNumberStyle = ED_BIG_ROMAN;
|
|
break;
|
|
case ED_LIST_TYPE_SMALL_ROMAN:
|
|
m_iNumberStyle = ED_SMALL_ROMAN;
|
|
break;
|
|
case ED_LIST_TYPE_BIG_LETTERS:
|
|
m_iNumberStyle = ED_BIG_LETTERS;
|
|
break;
|
|
case ED_LIST_TYPE_SMALL_LETTERS:
|
|
m_iNumberStyle = ED_SMALL_LETTERS;
|
|
break;
|
|
case ED_LIST_TYPE_DISC:
|
|
m_iBulletStyle = ED_SOLID_CIRCLE;
|
|
break;
|
|
case ED_LIST_TYPE_CIRCLE:
|
|
m_iBulletStyle = ED_OPEN_CIRCLE;
|
|
break;
|
|
case ED_LIST_TYPE_SQUARE:
|
|
m_iBulletStyle = ED_SOLID_SQUARE;
|
|
break;
|
|
}
|
|
m_bCompact = m_pListData->bCompact;
|
|
} else {
|
|
m_iContainerStyle = ED_NO_CONTAINER;
|
|
}
|
|
|
|
|
|
VERIFY(pContainerStyles = (CComboBox*)GetDlgItem(IDC_CONTAINER_STYLES));
|
|
VERIFY(pListStyles = (CComboBox*)GetDlgItem(IDC_LIST_STYLES));
|
|
|
|
// Fill the Container styles list
|
|
pContainerStyles->AddString(szLoadString(IDS_DEFAULT));
|
|
pContainerStyles->AddString(szLoadString(IDS_LIST));
|
|
pContainerStyles->AddString(szLoadString(IDS_BLOCK_QUOTE));
|
|
|
|
|
|
// Fill the List styles list
|
|
pListStyles->AddString(szLoadString(IDS_UNUM_LIST));
|
|
pListStyles->AddString(szLoadString(IDS_NUM_LIST));
|
|
pListStyles->AddString(szLoadString(IDS_DESCRIPTION_LIST));
|
|
|
|
// Remove/rebuild item styles depending on current state
|
|
UpdateLists();
|
|
|
|
// TODO - How to get current alignment?
|
|
// (Set align radio button to this value)
|
|
|
|
// Send data to controls
|
|
UpdateData(FALSE);
|
|
return(TRUE);
|
|
}
|
|
|
|
void CParagraphPage::OnHelp()
|
|
{
|
|
NetHelp(HELP_PROPS_PARAGRAPH);
|
|
}
|
|
|
|
void CParagraphPage::OnOK()
|
|
{
|
|
CPropertyPage::OnOK();
|
|
|
|
// never visited this page or no change -- don't bother
|
|
if(!m_bActivated ||
|
|
!IS_APPLY_ENABLED(this)){
|
|
return;
|
|
}
|
|
|
|
//EDT_BeginBatchChanges(m_pMWContext);
|
|
|
|
m_iParagraphStyle = ((CComboBox*)GetDlgItem(IDC_PARAGRAPH_STYLES))->GetCurSel();
|
|
m_iContainerStyle = ((CComboBox*)GetDlgItem(IDC_CONTAINER_STYLES))->GetCurSel();
|
|
m_iListStyle = ((CComboBox*)GetDlgItem(IDC_LIST_STYLES))->GetCurSel();
|
|
|
|
// Set alignment if it was changed
|
|
EDT_SetParagraphAlign( m_pMWContext, m_Align );
|
|
|
|
// Default values
|
|
TagType iTagType = P_UNUM_LIST;
|
|
ED_ListType iListItemType = ED_LIST_TYPE_DEFAULT;
|
|
CComboBox * pListItemStyles = (CComboBox*)GetDlgItem(IDC_LIST_ITEM_STYLES);
|
|
|
|
if( m_iContainerStyle == ED_BLOCKQUOTE ){
|
|
iTagType = P_BLOCKQUOTE;
|
|
} else if( m_iContainerStyle == ED_LIST ){
|
|
switch( m_iListStyle ){
|
|
case ED_NUM_LIST:
|
|
iTagType = P_NUM_LIST;
|
|
break;
|
|
case ED_DESC_LIST:
|
|
iTagType = P_DESC_LIST;
|
|
break;
|
|
//Note: UNUM_LIST is default
|
|
}
|
|
|
|
if( iTagType == P_UNUM_LIST ) {
|
|
m_iBulletStyle = pListItemStyles->GetCurSel();
|
|
switch( m_iBulletStyle ){
|
|
case ED_SOLID_CIRCLE:
|
|
iListItemType = ED_LIST_TYPE_DISC;
|
|
break;
|
|
case ED_OPEN_CIRCLE:
|
|
iListItemType = ED_LIST_TYPE_CIRCLE;
|
|
break;
|
|
case ED_SOLID_SQUARE:
|
|
iListItemType = ED_LIST_TYPE_SQUARE;
|
|
break;
|
|
}
|
|
} else if( iTagType == P_NUM_LIST ) {
|
|
m_iNumberStyle = pListItemStyles->GetCurSel();
|
|
switch( m_iNumberStyle ){
|
|
case ED_DIGIT:
|
|
iListItemType = ED_LIST_TYPE_DIGIT;
|
|
break;
|
|
case ED_BIG_ROMAN:
|
|
iListItemType = ED_LIST_TYPE_BIG_ROMAN;
|
|
break;
|
|
case ED_SMALL_ROMAN:
|
|
iListItemType = ED_LIST_TYPE_SMALL_ROMAN;
|
|
break;
|
|
case ED_BIG_LETTERS:
|
|
iListItemType = ED_LIST_TYPE_BIG_LETTERS;
|
|
break;
|
|
case ED_SMALL_LETTERS:
|
|
iListItemType = ED_LIST_TYPE_SMALL_LETTERS;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// We should have been maintaining paragraph style
|
|
// to match container/list style, so this should work
|
|
// But if "Don't change" style, NEVER set the container style
|
|
// NOTE: This will remove any list if new container format is not appropriate
|
|
if( m_iParagraphStyle != m_iMixedStyle ){
|
|
TagType NewFormat = FEED_nParagraphTags[m_iParagraphStyle];
|
|
if( m_ParagraphFormat != NewFormat ){
|
|
EDT_MorphContainer( m_pMWContext, NewFormat);
|
|
// Get the list data for the new list we might have created
|
|
// or if we destroyed the list by changing paragraph style
|
|
m_pListData = EDT_GetListData(m_pMWContext);
|
|
// Save the new format
|
|
m_ParagraphFormat = NewFormat;
|
|
}
|
|
}
|
|
|
|
// Set List type and attributes
|
|
if ( m_pListData ) {
|
|
// If no container, we Outdented above, so don't set data here
|
|
if( m_iContainerStyle != ED_NO_CONTAINER ) {
|
|
m_pListData->iTagType = iTagType;
|
|
m_pListData->eType = iListItemType;
|
|
m_pListData->bCompact = m_bCompact;
|
|
m_pListData->iStart = m_iStartNumber;
|
|
|
|
// Now we can change the list attribute:
|
|
EDT_SetListData( m_pMWContext, m_pListData );
|
|
}
|
|
}
|
|
OkToClose();
|
|
//EDT_EndBatchChanges(m_pMWContext);
|
|
}
|
|
|
|
// Set listbox selected items depending on current state
|
|
// Primary key is m_iContainerStyle
|
|
// Other style values will be kept but ignored when setting
|
|
// selected item or disabling/enabling dependent listboxes
|
|
// Goal is to show list style and bullet/numbering only if appropriate
|
|
//
|
|
void CParagraphPage::UpdateLists()
|
|
{
|
|
// Set selected items
|
|
((CComboBox*)GetDlgItem(IDC_PARAGRAPH_STYLES))->SetCurSel(m_iParagraphStyle);
|
|
((CComboBox*)GetDlgItem(IDC_CONTAINER_STYLES))->SetCurSel(m_iContainerStyle);
|
|
|
|
// Remove and disable the selection in List Styles if not a List,
|
|
// else show what kind of list it is
|
|
((CComboBox*)GetDlgItem(IDC_LIST_STYLES))->SetCurSel(
|
|
m_iContainerStyle == ED_LIST ? m_iListStyle : -1);
|
|
((CComboBox*)GetDlgItem(IDC_LIST_STYLES))->EnableWindow(m_iContainerStyle == ED_LIST);
|
|
|
|
// Rebuild the List Items styles list
|
|
CComboBox * pListItemStyles = (CComboBox*)GetDlgItem(IDC_LIST_ITEM_STYLES);
|
|
pListItemStyles->ResetContent();
|
|
|
|
CWnd *pListItemLabel = GetDlgItem(IDC_LIST_ITEM_LABEL);
|
|
|
|
BOOL bEnableStartNumber = FALSE;
|
|
BOOL bEnableItemList = FALSE;
|
|
|
|
if ( m_iContainerStyle == ED_LIST ){
|
|
if( m_iListStyle == ED_UNUM_LIST ||
|
|
m_iListStyle == ED_NUM_LIST ){
|
|
|
|
pListItemStyles->EnableWindow(TRUE);
|
|
pListItemStyles->AddString(szLoadString(IDS_AUTOMATIC));
|
|
bEnableItemList = TRUE;
|
|
|
|
if( m_iListStyle == ED_UNUM_LIST ){
|
|
// Be sure we are at least on the first item
|
|
m_iBulletStyle = max(0,m_iBulletStyle);
|
|
pListItemStyles->AddString(szLoadString(IDS_SOLID_CIRCLE));
|
|
pListItemStyles->AddString(szLoadString(IDS_OPEN_CIRCLE));
|
|
pListItemStyles->AddString(szLoadString(IDS_SOLID_SQUARE));
|
|
pListItemStyles->SetCurSel(m_iBulletStyle);
|
|
pListItemLabel->SetWindowText(szLoadString(IDS_BULLET_STYLE));
|
|
} else if( m_iListStyle == ED_NUM_LIST ){
|
|
m_iNumberStyle = max(0,m_iNumberStyle);
|
|
pListItemStyles->AddString(szLoadString(IDS_DIGIT));
|
|
pListItemStyles->AddString(szLoadString(IDS_BIG_ROMAN));
|
|
pListItemStyles->AddString(szLoadString(IDS_SMALL_ROMAN));
|
|
pListItemStyles->AddString(szLoadString(IDS_BIG_LETTERS));
|
|
pListItemStyles->AddString(szLoadString(IDS_SMALL_LETTERS));
|
|
pListItemStyles->SetCurSel(m_iNumberStyle);
|
|
pListItemLabel->SetWindowText(szLoadString(IDS_NUMBER_STYLE));
|
|
bEnableStartNumber = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Set enable depending on bullet or number styles
|
|
pListItemStyles->EnableWindow(bEnableItemList);
|
|
if( !bEnableItemList ) {
|
|
// Remove label if disabled
|
|
pListItemLabel->SetWindowText("");
|
|
}
|
|
|
|
// Disable start number if not a numbered list
|
|
GetDlgItem(IDC_LIST_START_NUMBER)->EnableWindow(bEnableStartNumber);
|
|
|
|
}
|
|
|
|
void CParagraphPage::ChangeParagraphStyle()
|
|
{
|
|
// Selecting one of the list item will force a list container
|
|
if( m_iParagraphStyle == ED_PARA_LIST )
|
|
{
|
|
m_iContainerStyle = ED_LIST;
|
|
// Either numbered or unnumbered list style is OK
|
|
if( m_iListStyle != ED_NUM_LIST )
|
|
m_iListStyle = ED_UNUM_LIST;
|
|
else if( m_iListStyle != ED_UNUM_LIST )
|
|
m_iListStyle = ED_NUM_LIST;
|
|
}
|
|
else if( m_iParagraphStyle == ED_PARA_DESC_TITLE || m_iParagraphStyle == ED_PARA_DESC_TEXT)
|
|
{
|
|
m_iContainerStyle = ED_LIST;
|
|
//if( m_iListStyle == ED_DEFAULT )
|
|
m_iListStyle = ED_DESC_LIST;
|
|
|
|
} else if( m_iContainerStyle == ED_LIST ){
|
|
// Remove List container
|
|
// Note that it is left as is for BLOCK QUOTE
|
|
m_iContainerStyle = ED_NO_CONTAINER;
|
|
}
|
|
UpdateLists();
|
|
}
|
|
|
|
void CParagraphPage::OnSelchangeParagraphStyles()
|
|
{
|
|
SetModified(TRUE);
|
|
m_iParagraphStyle = ((CComboBox*)GetDlgItem(IDC_PARAGRAPH_STYLES))->GetCurSel();
|
|
ChangeParagraphStyle();
|
|
}
|
|
|
|
void CParagraphPage::OnSelchangeContainerStyle()
|
|
{
|
|
SetModified(TRUE);
|
|
m_iContainerStyle = ((CComboBox*)GetDlgItem(IDC_CONTAINER_STYLES))->GetCurSel();
|
|
|
|
// Force List paragraph style
|
|
if( m_iContainerStyle == ED_LIST )
|
|
{
|
|
// If not already a list item, set it now
|
|
if( m_iParagraphStyle < ED_PARA_LIST )
|
|
m_iParagraphStyle = ED_PARA_LIST;
|
|
|
|
}
|
|
else if( m_iParagraphStyle >= ED_PARA_LIST )
|
|
{
|
|
// For no container or BLOCKQUOTE,
|
|
// any list item must be converted to normal
|
|
m_iParagraphStyle = ED_PARA_NORMAL;
|
|
}
|
|
ChangeParagraphStyle();
|
|
}
|
|
|
|
void CParagraphPage::OnSelchangeListStyles()
|
|
{
|
|
SetModified(TRUE);
|
|
m_iListStyle = ((CComboBox*)GetDlgItem(IDC_LIST_STYLES))->GetCurSel();
|
|
|
|
if( m_iListStyle == ED_DESC_LIST )
|
|
{
|
|
// Set appropriate paragraph style for definition list
|
|
if( m_iParagraphStyle < ED_PARA_DESC_TITLE )
|
|
m_iParagraphStyle = ED_PARA_DESC_TITLE;
|
|
}
|
|
else
|
|
{
|
|
// Changing to either numbered or bulleted list
|
|
m_iParagraphStyle = ED_PARA_LIST;
|
|
}
|
|
ChangeParagraphStyle();
|
|
}
|
|
|
|
void CParagraphPage::OnSelchangeListItemStyle()
|
|
{
|
|
SetModified(TRUE);
|
|
m_iListStyle = ((CComboBox*)GetDlgItem(IDC_LIST_STYLES))->GetCurSel();
|
|
int iListItemStyle = ((CComboBox*)GetDlgItem(IDC_LIST_STYLES))->GetCurSel();
|
|
|
|
// Just record index in proper list item category
|
|
if( m_iListStyle == ED_UNUM_LIST){
|
|
m_iBulletStyle = iListItemStyle;
|
|
} else if( m_iListStyle == ED_NUM_LIST){
|
|
m_iNumberStyle = iListItemStyle;
|
|
}
|
|
}
|
|
|
|
void CParagraphPage::OnChangeListStartNumber()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CParagraphPage::OnAlignCenter()
|
|
{
|
|
m_Align = ED_ALIGN_ABSCENTER; // ED_ALIGN_CENTER;
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CParagraphPage::OnAlignLeft()
|
|
{
|
|
m_Align = ED_ALIGN_LEFT;
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CParagraphPage::OnAlignRight()
|
|
{
|
|
m_Align = ED_ALIGN_RIGHT;
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CParagraphPage::OnClose()
|
|
{
|
|
// We can't release data in OnOK
|
|
// since we may stay around after
|
|
// it is called for "Apply" function
|
|
if( m_pListData ){
|
|
EDT_FreeListData(m_pListData);
|
|
}
|
|
CNetscapePropertyPage::OnClose();
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
IMPLEMENT_DYNAMIC(CColorButton, CButton)
|
|
|
|
CColorButton::CColorButton(COLORREF * pColorRef, COLORREF * pSetFocusColor) :
|
|
m_pColorRef(pColorRef),
|
|
m_pSetFocusColor(pSetFocusColor),
|
|
m_hPal(0),
|
|
m_bColorSwatchMode(0),
|
|
m_pToolTip(0)
|
|
{
|
|
}
|
|
|
|
CColorButton::~CColorButton()
|
|
{
|
|
if( m_pToolTip ){
|
|
delete m_pToolTip;
|
|
}
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CColorButton, CButton)
|
|
//{{AFX_MSG_MAP(CColorButton)
|
|
ON_WM_MOUSEMOVE()
|
|
ON_WM_SETFOCUS()
|
|
//}}AFX_MSG_MAP
|
|
#ifdef XP_WIN32
|
|
ON_NOTIFY_EX( TTN_NEEDTEXT, 0, OnToolTipNotify )
|
|
#endif
|
|
END_MESSAGE_MAP()
|
|
|
|
BOOL CColorButton::PreTranslateMessage(MSG* pMsg)
|
|
{
|
|
if( m_pToolTip && pMsg->message >= WM_MOUSEFIRST && pMsg->message <= WM_MOUSELAST)
|
|
{
|
|
#ifdef XP_WIN32
|
|
// This is needed to get around an MFC bug
|
|
// where tooltip is disabled after Modal Dialog is called
|
|
m_pToolTip->Activate(TRUE);
|
|
#endif
|
|
m_pToolTip->RelayEvent(pMsg);
|
|
}
|
|
return CButton::PreTranslateMessage(pMsg);
|
|
}
|
|
|
|
//
|
|
#ifdef XP_WIN32
|
|
// This is needed for non-CFrame owners of a CToolTipCtrl
|
|
BOOL CColorButton::OnToolTipNotify( UINT id, NMHDR * pNMHDR, LRESULT * pResult )
|
|
{
|
|
TOOLTIPTEXT *pTTT = (TOOLTIPTEXT *)pNMHDR;
|
|
UINT nID =pNMHDR->idFrom;
|
|
if (pTTT->uFlags & TTF_IDISHWND)
|
|
{
|
|
if(::GetDlgCtrlID((HWND)nID) == 1) {
|
|
strcpy(pTTT->szText, m_pTipText);
|
|
return(TRUE);
|
|
}
|
|
}
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
BOOL CColorButton::Subclass(CWnd * pParentWnd, UINT nID, COLORREF * pColorRef, BOOL bColorSwatchMode)
|
|
{
|
|
if(!pParentWnd || !pColorRef){
|
|
return FALSE;
|
|
}
|
|
m_pColorRef = pColorRef;
|
|
m_bColorSwatchMode = bColorSwatchMode;
|
|
|
|
// Attach derived class to existing control and route messages to parent dialog
|
|
BOOL bResult = SubclassDlgItem(nID, pParentWnd);
|
|
|
|
// Must do subclass before we can use GetParentFrame()
|
|
m_hPal = WFE_GetUIPalette(GetParentFrame());
|
|
|
|
AddToolTip();
|
|
|
|
return bResult;
|
|
}
|
|
|
|
BOOL CColorButton::Create(RECT& rect, CWnd * pParentWnd, UINT nID, COLORREF * pColorRef, BOOL bColorSwatchMode)
|
|
{
|
|
if(!pParentWnd || !pColorRef){
|
|
return FALSE;
|
|
}
|
|
m_pColorRef = pColorRef;
|
|
m_bColorSwatchMode = bColorSwatchMode;
|
|
|
|
BOOL bResult = CButton::Create(NULL, BS_PUSHBUTTON|BS_OWNERDRAW|WS_CHILD|WS_VISIBLE|WS_GROUP|WS_TABSTOP,
|
|
rect, pParentWnd, nID);
|
|
// Must do Create before we can use GetParentFrame()
|
|
m_hPal = WFE_GetUIPalette(GetParentFrame());
|
|
|
|
AddToolTip();
|
|
|
|
return bResult;
|
|
}
|
|
|
|
void CColorButton::Update()
|
|
{
|
|
// Redraw the button
|
|
Invalidate(FALSE);
|
|
|
|
// Update tooltip for new color
|
|
if( m_pToolTip ){
|
|
// Make a tip with "R=xxx G=xxx B=xxx"
|
|
wsprintf(m_pTipText, szLoadString(IDS_COLOR_TIP_FORMAT),
|
|
GetRValue(*m_pColorRef),GetGValue(*m_pColorRef),GetBValue(*m_pColorRef));
|
|
strcat(m_pTipText, szLoadString(IDS_COLOR_TIP_HTML));
|
|
char * pEnd = m_pTipText + strlen(m_pTipText);
|
|
sprintf(pEnd, "#%02X%02X%02X",GetRValue(*m_pColorRef),GetGValue(*m_pColorRef),GetBValue(*m_pColorRef));
|
|
m_pToolTip->UpdateTipText(m_pTipText, this, 1);
|
|
}
|
|
}
|
|
|
|
void CColorButton::AddToolTip()
|
|
{
|
|
// Add tooltip only if a real button
|
|
// In "color swatch mode", the parent CColorPicker
|
|
// has 1 tooltip associated with all the button controls
|
|
if( !m_bColorSwatchMode ){
|
|
// Add a tooltip control
|
|
m_pToolTip = new CNSToolTip2;
|
|
|
|
if(m_pToolTip && !m_pToolTip->Create(this, TTS_ALWAYSTIP) ){
|
|
TRACE("Unable To create ToolTip\n");
|
|
delete m_pToolTip;
|
|
m_pToolTip = NULL;
|
|
return;
|
|
}
|
|
// Lets use speedy tooltips
|
|
m_pToolTip->SetDelayTime(200);
|
|
#ifdef XP_WIN32
|
|
// We MUST do this for MFC tooltips
|
|
EnableToolTips(TRUE);
|
|
#endif // WIN32
|
|
RECT rect;
|
|
GetClientRect(&rect);
|
|
m_pToolTip->AddTool(this, m_pTipText, &rect, 1);
|
|
// Set the text to show the colors
|
|
Update();
|
|
}
|
|
}
|
|
|
|
void CColorButton::OnMouseMove(UINT nFlags, CPoint point)
|
|
{
|
|
CButton::OnMouseMove(nFlags, point);
|
|
|
|
// For the "Color swatch" mode, automatically pull focus to
|
|
// the button with any mouse over
|
|
if( m_bColorSwatchMode && GetFocus() != this ){
|
|
// Set the color for the caller
|
|
// This allows just mouse move to set the color
|
|
// after user presses Enter key without clicking on a color button
|
|
// and also lets us use OnOK() for all button clicks
|
|
SetFocus();
|
|
}
|
|
}
|
|
|
|
void CColorButton::OnSetFocus(CWnd* pOldWnd)
|
|
{
|
|
// Send the color back to the caller
|
|
if( m_pSetFocusColor ){
|
|
*m_pSetFocusColor = *m_pColorRef;
|
|
}
|
|
CButton::OnSetFocus(pOldWnd);
|
|
}
|
|
|
|
#define COLOR_DROPDOWN_WIDTH 10
|
|
|
|
void CColorButton::DrawItem(LPDRAWITEMSTRUCT lpDIS)
|
|
{
|
|
HPALETTE hOldPal = NULL;
|
|
CDC* pDC = CDC::FromHandle(lpDIS->hDC);
|
|
if( m_hPal )
|
|
{
|
|
hOldPal = ::SelectPalette( pDC->m_hDC, m_hPal, FALSE );
|
|
}
|
|
BOOL bSelected = lpDIS->itemState & ODS_SELECTED;
|
|
|
|
HDC hDC = lpDIS->hDC;
|
|
RECT rect = lpDIS->rcItem;
|
|
if( !m_bColorSwatchMode )
|
|
{
|
|
// We can't draw on last pixels of right and bottom???
|
|
rect.right--;
|
|
rect.bottom--;
|
|
}
|
|
|
|
// Use the button face color as our background
|
|
::FillRect(hDC, &rect, sysInfo.m_hbrBtnFace);
|
|
|
|
HPEN penShadow = ::CreatePen(PS_SOLID, 1, sysInfo.m_clrBtnShadow);
|
|
HPEN penHighlight = ::CreatePen(PS_SOLID, 1, sysInfo.m_clrBtnHilite);
|
|
HPEN penBlack = ::CreatePen(PS_SOLID, 1, RGB(0,0,0));
|
|
HPEN penOld = (HPEN)::SelectObject(hDC, penBlack);
|
|
|
|
RECT rectColor = rect;
|
|
|
|
if( m_bColorSwatchMode )
|
|
{
|
|
// Draw the depressed 3D look like Window's color picker,
|
|
// Note that there is a built-in blank border
|
|
// only focus rect draws on actual window borders
|
|
::InflateRect(&rect, -2, -2);
|
|
rectColor.left += 4;
|
|
rectColor.top += 4;
|
|
rectColor.right -= 3;
|
|
rectColor.bottom -= 3;
|
|
} else {
|
|
rectColor.left += 2;
|
|
rectColor.top += 2;
|
|
rectColor.bottom -= 1;
|
|
rectColor.right -= (COLOR_DROPDOWN_WIDTH + 7);
|
|
}
|
|
|
|
// Draw depressed border for color swatch
|
|
::MoveToEx(hDC, rect.left+1, rect.bottom-1, NULL);
|
|
::LineTo(hDC, rect.left+1, rect.top+1);
|
|
::LineTo(hDC, rect.right, rect.top+1);
|
|
|
|
::SelectObject(hDC, penShadow);
|
|
::MoveToEx(hDC, rect.left, rect.bottom, NULL);
|
|
::LineTo(hDC, rect.left, rect.top);
|
|
::LineTo(hDC, rect.right+1, rect.top);
|
|
|
|
::SelectObject(hDC, penHighlight);
|
|
::MoveToEx(hDC, rect.left+1, rect.bottom, NULL);
|
|
::LineTo(hDC, rect.right, rect.bottom);
|
|
::LineTo(hDC, rect.right, rect.top);
|
|
|
|
::SelectObject(hDC, penBlack);
|
|
|
|
if( !m_bColorSwatchMode )
|
|
{
|
|
::InflateRect(&rect, -2, -2);
|
|
rect.top++;
|
|
rect.left = rect.right - (COLOR_DROPDOWN_WIDTH + 4);
|
|
|
|
// Location of drop-down arrow
|
|
// Width is actually COLOR_DROPDOWN_WIDTH-4,
|
|
// but LineTo needs extra pixel to end where we want
|
|
int iTWidth = COLOR_DROPDOWN_WIDTH - 3;
|
|
int iTLeft = rect.right - iTWidth - 4;
|
|
int iTTop = rect.top + ((rect.bottom - rect.top)/2) - 2;
|
|
|
|
// Offset down and right for pressed-down state
|
|
if( lpDIS->itemState & ODS_SELECTED )
|
|
{
|
|
iTLeft++;
|
|
iTTop++;
|
|
}
|
|
// Draw the black drop-down triangle
|
|
while( iTWidth >= 0 )
|
|
{
|
|
::MoveToEx(hDC, iTLeft, iTTop, NULL);
|
|
::LineTo(hDC, iTLeft + iTWidth, iTTop);
|
|
iTTop++;
|
|
iTLeft++;
|
|
iTWidth -= 2;
|
|
}
|
|
|
|
if( lpDIS->itemState & ODS_SELECTED )
|
|
{
|
|
rect.right++;
|
|
rect.bottom++;
|
|
HBRUSH brushShadow = ::CreateSolidBrush(sysInfo.m_clrBtnShadow|0x02000000);
|
|
::FrameRect(hDC, &rect, brushShadow);
|
|
::DeleteObject(brushShadow);
|
|
} else {
|
|
// Draw the raised 3D button shadows
|
|
::MoveToEx(hDC, rect.left, rect.bottom, NULL);
|
|
::LineTo(hDC, rect.right, rect.bottom);
|
|
::LineTo(hDC, rect.right, rect.top-1);
|
|
|
|
::SelectObject(hDC, penShadow);
|
|
::MoveToEx(hDC, rect.left+1, rect.bottom-1, NULL);
|
|
::LineTo(hDC, rect.right-1, rect.bottom-1);
|
|
::LineTo(hDC, rect.right-1, rect.top);
|
|
::SelectObject(hDC, penHighlight);
|
|
::MoveToEx(hDC, rect.left, rect.bottom-1, NULL);
|
|
::LineTo(hDC, rect.left, rect.top);
|
|
::LineTo(hDC, rect.right, rect.top);
|
|
}
|
|
}
|
|
// Draw the color rectangle unless we have a "no color" or "default" state
|
|
if( *m_pColorRef != NO_COLORREF && *m_pColorRef != DEFAULT_COLORREF)
|
|
{
|
|
|
|
HBRUSH brushColor = ::CreateSolidBrush((*m_pColorRef)|0x02000000); // Like PALETTERGB
|
|
// Fill a rect with supplied color
|
|
HBRUSH brushOld = (HBRUSH)::SelectObject(hDC, brushColor);
|
|
::FillRect(hDC, &rectColor, brushColor);
|
|
|
|
::SelectObject(hDC, brushOld);
|
|
::DeleteObject(brushColor);
|
|
}
|
|
|
|
// Manually draw the focus rect
|
|
if( lpDIS->itemState & ODS_FOCUS )
|
|
{
|
|
RECT rectFocus;
|
|
if( m_bColorSwatchMode )
|
|
{
|
|
rectFocus = lpDIS->rcItem;
|
|
rect.left++;
|
|
rect.top++;
|
|
} else {
|
|
rectFocus = rectColor;
|
|
::InflateRect(&rectFocus, -1, -1);
|
|
}
|
|
::DrawFocusRect(hDC, &rectFocus);
|
|
}
|
|
|
|
// Cleanup
|
|
::SelectObject(hDC, penOld);
|
|
::DeleteObject(penShadow);
|
|
::DeleteObject(penHighlight);
|
|
::DeleteObject(penBlack);
|
|
|
|
if(m_hPal)
|
|
{
|
|
::SelectPalette( pDC->m_hDC, hOldPal, FALSE );
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CBitmapPushButton
|
|
// Allows toolbar-like pushbutton behavior
|
|
|
|
CBitmapPushButton::CBitmapPushButton(BOOL bNoBorder) :
|
|
m_bDown(0),
|
|
m_bFocus(0),
|
|
m_bSelected(0),
|
|
m_hPal(0),
|
|
m_bNoBorder(bNoBorder)
|
|
{
|
|
}
|
|
|
|
int CBitmapPushButton::OnCreate(LPCREATESTRUCT lpCreateStruct )
|
|
{
|
|
if( -1 == CBitmapButton::OnCreate(lpCreateStruct) ){
|
|
return -1;
|
|
}
|
|
// Get the global default palette
|
|
m_hPal = WFE_GetUIPalette(GetParentFrame());
|
|
return 0;
|
|
}
|
|
|
|
CBitmapPushButton::~CBitmapPushButton()
|
|
{
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CBitmapPushButton, CBitmapButton)
|
|
//{{AFX_MSG_MAP(CBitmapPushButton)
|
|
ON_WM_CREATE()
|
|
// NOTE - the ClassWizard will add and remove mapping macros here.
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
|
|
BOOL CBitmapPushButton::LoadBitmap(UINT nBitmapID)
|
|
{
|
|
// delete old bitmaps (if present)
|
|
m_bitmap.DeleteObject();
|
|
m_bitmapSel.DeleteObject();
|
|
m_bitmapFocus.DeleteObject();
|
|
m_bitmapDisabled.DeleteObject();
|
|
|
|
if ( !m_bitmap.LoadBitmap( nBitmapID ) ) {
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void CBitmapPushButton::SetCheck(BOOL bCheck)
|
|
{
|
|
BOOL bChanged = m_bDown != bCheck;
|
|
m_bDown = bCheck;
|
|
SetState(bCheck);
|
|
if(bChanged){
|
|
// We need to force redraw sometimes
|
|
Invalidate(FALSE);
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CBitmapPushButton message handlers
|
|
void CBitmapPushButton::DrawItem(LPDRAWITEMSTRUCT lpDIS)
|
|
{
|
|
// Save current focus state so we know what
|
|
// button is pressed via keyboard in CDropdownToolbar handler
|
|
m_bFocus = (lpDIS->itemState & ODS_FOCUS);
|
|
|
|
// If our state is pushed down ("checked"),
|
|
// then always force selected bit so
|
|
// we don't unselect when losing focus
|
|
if(m_bDown) {
|
|
lpDIS->itemState |= ODS_SELECTED;
|
|
}
|
|
|
|
m_bSelected = lpDIS->itemState & ODS_SELECTED;
|
|
|
|
// Draw the bitmap ourselves to do transparent overlay
|
|
|
|
// Get the bitmap for the image
|
|
HBITMAP hBmpImg = 0;
|
|
|
|
if(m_bSelected && m_bitmapSel.m_hObject ){
|
|
// HBITMAP(m_bitmapSel) doesn't work in old MFC (Win16)
|
|
hBmpImg = (HBITMAP)m_bitmapSel.m_hObject;
|
|
} else if( m_bFocus && m_bitmapFocus.m_hObject ) {
|
|
hBmpImg = (HBITMAP)m_bitmapFocus.m_hObject;
|
|
} else if ( (lpDIS->itemState & ODS_DISABLED) && m_bitmapDisabled.m_hObject ) {
|
|
hBmpImg = (HBITMAP)m_bitmapDisabled.m_hObject;
|
|
} else {
|
|
// Check for no bitmap at all!
|
|
if( !m_bitmap.m_hObject ){
|
|
return;
|
|
}
|
|
hBmpImg = (HBITMAP)m_bitmap.m_hObject;
|
|
}
|
|
|
|
HDC hDC = lpDIS->hDC;
|
|
HDC hBmpDC = ::CreateCompatibleDC(hDC);
|
|
RECT rect = lpDIS->rcItem;
|
|
|
|
// Use the button face color as our background
|
|
::FillRect(hDC, &rect, sysInfo.m_hbrBtnFace);
|
|
|
|
// If no bitmap, we must be disabled and no image supplied - leave blank TODO: GRAY IMAGE
|
|
// WEIRD - m_bitmapDisabled.m_hObject is not NULL
|
|
// if( hBmpImg ){
|
|
if( ! (lpDIS->itemState & ODS_DISABLED) ){
|
|
|
|
HBITMAP hOldBmp = (HBITMAP)::SelectObject(hBmpDC, hBmpImg);
|
|
|
|
// Get this for the real bimap width and height
|
|
BITMAP bmp;
|
|
::GetObject(hBmpImg, sizeof(bmp), &bmp);
|
|
|
|
int x, y;
|
|
// Center the image within the button
|
|
x = rect.left + ((rect.right - rect.left) - bmp.bmWidth) / 2;
|
|
y = rect.top + ((rect.bottom - rect.top) - bmp.bmHeight) / 2;
|
|
|
|
// If button is pushed down, shift image right and down 1 pixel,
|
|
// (but only if not using supplied selected bitmap)
|
|
if(m_bSelected && m_bitmapSel.m_hObject == NULL ){
|
|
x++;
|
|
y++;
|
|
}
|
|
|
|
// Call the handy transparent blit function to paint the bitmap over whatever colors exist.
|
|
::FEU_TransBlt( hDC, x, y , bmp.bmWidth, bmp.bmHeight,
|
|
hBmpDC, 0, 0, m_hPal);
|
|
|
|
::SelectObject(hBmpDC, hOldBmp);
|
|
::DeleteDC(hBmpDC);
|
|
}
|
|
|
|
// Draw the 3D shadow borders
|
|
|
|
// Reduce to use inclusive border
|
|
rect.right --;
|
|
rect.bottom --;
|
|
::InflateRect(&rect, -1, -1);
|
|
HPEN penShadow = ::CreatePen(PS_SOLID, 1, sysInfo.m_clrBtnShadow);
|
|
HPEN penHighlight = ::CreatePen(PS_SOLID, 1, sysInfo.m_clrBtnHilite);
|
|
HPEN penBlack = ::CreatePen(PS_SOLID, 1, RGB(0,0,0));
|
|
HPEN penOld = (HPEN)::SelectObject(hDC, penBlack);
|
|
|
|
if( m_bSelected ){
|
|
// Draw the depressed 3D button shadows
|
|
// only if bitmap wasn't supplied
|
|
if( m_bitmapSel.m_hObject == NULL ){
|
|
::MoveToEx(hDC, rect.left, rect.bottom, NULL);
|
|
::LineTo(hDC, rect.left, rect.top);
|
|
::LineTo(hDC, rect.right+1, rect.top);
|
|
|
|
::SelectObject(hDC, penShadow);
|
|
::MoveToEx(hDC, rect.left+1, rect.bottom-1, NULL);
|
|
::LineTo(hDC, rect.left+1, rect.top+1);
|
|
::LineTo(hDC, rect.right, rect.top+1);
|
|
|
|
::SelectObject(hDC, penHighlight);
|
|
::LineTo(hDC, rect.right, rect.bottom);
|
|
::LineTo(hDC, rect.left, rect.bottom);
|
|
}
|
|
} else if( !m_bNoBorder ){
|
|
// Draw the raised 3D button shadows
|
|
::MoveToEx(hDC, rect.left, rect.bottom, NULL);
|
|
::LineTo(hDC, rect.right, rect.bottom);
|
|
::LineTo(hDC, rect.right, rect.top-1);
|
|
|
|
::SelectObject(hDC, penShadow);
|
|
::MoveToEx(hDC, rect.left+1, rect.bottom-1, NULL);
|
|
::LineTo(hDC, rect.right-1, rect.bottom-1);
|
|
::LineTo(hDC, rect.right-1, rect.top);
|
|
|
|
::SelectObject(hDC, penHighlight);
|
|
::MoveToEx(hDC, rect.left, rect.bottom-1, NULL);
|
|
::LineTo(hDC, rect.left, rect.top);
|
|
::LineTo(hDC, rect.right, rect.top);
|
|
}
|
|
|
|
// Manually draw the focus rect if we don't have an image
|
|
// This is optimized for a 1-pixel highlight bevel
|
|
// and 2-pixel shadow bevel
|
|
// Note that CBitmapButton will NEVER
|
|
// show a distinct Selected+Focus state,
|
|
// so this supplies that feature
|
|
if( m_bFocus && m_bitmapFocus.m_hObject == NULL ){
|
|
if( !m_bSelected ){
|
|
// Draw the raised 3D button shadows if we are not pressing down
|
|
::SelectObject(hDC, penHighlight);
|
|
::MoveToEx(hDC, rect.left, rect.bottom-1, NULL);
|
|
::LineTo(hDC, rect.left, rect.top);
|
|
::LineTo(hDC, rect.right, rect.top);
|
|
|
|
::SelectObject(hDC, penShadow);
|
|
::MoveToEx(hDC, rect.left, rect.bottom, NULL);
|
|
::LineTo(hDC, rect.right, rect.bottom);
|
|
::LineTo(hDC, rect.right, rect.top-1);
|
|
}
|
|
// Draw a solid black frame just outside of 3D shadow rect
|
|
::InflateRect(&rect, 1, 1);
|
|
// Stupid algorithm doesn't include actual right/bottom
|
|
rect.bottom++;
|
|
rect.right++;
|
|
FrameRect(hDC, &rect, (HBRUSH)::GetStockObject(BLACK_BRUSH));
|
|
}
|
|
|
|
// Cleanup
|
|
::SelectObject(hDC, penOld);
|
|
::DeleteObject(penShadow);
|
|
::DeleteObject(penHighlight);
|
|
::DeleteObject(penBlack);
|
|
}
|
|
|
|
// Collection of Alignment/Size/Border controls used by
|
|
// Image, Java, and PlugIn dialogs
|
|
/////////////////////////////////////////////////////////////////////
|
|
CAlignControls::CAlignControls() :
|
|
m_nIDAlign(0),
|
|
m_EdAlign(ED_ALIGN_DEFAULT)
|
|
{
|
|
}
|
|
|
|
BOOL CAlignControls::Init(CWnd *pParent)
|
|
{
|
|
ASSERT(pParent);
|
|
m_pParent = pParent;
|
|
|
|
// Load the bitmaps for our alignment buttons
|
|
VERIFY(m_BtnAlignTop.AutoLoad(IDC_EDAL_T, pParent));
|
|
VERIFY(m_BtnAlignCenter.AutoLoad(IDC_EDAL_C, pParent));
|
|
VERIFY(m_BtnAlignCenterBaseline.AutoLoad(IDC_EDALCB, pParent));
|
|
VERIFY(m_BtnAlignBottomBaseline.AutoLoad(IDC_EDAL_A, pParent));
|
|
VERIFY(m_BtnAlignBottom.AutoLoad(IDC_EDAL_B, pParent));
|
|
VERIFY(m_BtnAlignLeft.AutoLoad(IDC_EDAL_L, pParent));
|
|
VERIFY(m_BtnAlignRight.AutoLoad(IDC_EDAL_R, pParent));
|
|
|
|
// Intialize controls
|
|
SetAlignment();
|
|
return TRUE;
|
|
}
|
|
|
|
// Return if we really changed the align state
|
|
// as compared to the previous state
|
|
BOOL CAlignControls::OnAlignButtonClick(UINT nID)
|
|
{
|
|
if ( nID == 0 ) {
|
|
nID = m_nIDAlign ? m_nIDAlign : IDC_EDAL_B;
|
|
}
|
|
BOOL bChanged = nID != m_nIDAlign;
|
|
m_nIDAlign = nID;
|
|
|
|
m_BtnAlignTop.SetCheck(m_nIDAlign == IDC_EDAL_T);
|
|
m_BtnAlignCenter.SetCheck(m_nIDAlign == IDC_EDAL_C);
|
|
m_BtnAlignCenterBaseline.SetCheck(m_nIDAlign == IDC_EDALCB);
|
|
m_BtnAlignBottomBaseline.SetCheck(m_nIDAlign == IDC_EDAL_A);
|
|
m_BtnAlignBottom.SetCheck(m_nIDAlign == IDC_EDAL_B);
|
|
m_BtnAlignLeft.SetCheck(m_nIDAlign == IDC_EDAL_L);
|
|
m_BtnAlignRight.SetCheck(m_nIDAlign == IDC_EDAL_R);
|
|
|
|
if(bChanged &&
|
|
m_pParent->IsKindOf(RUNTIME_CLASS(CPropertyPage))){
|
|
((CPropertyPage*)m_pParent)->SetModified(TRUE);
|
|
}
|
|
return bChanged;
|
|
}
|
|
|
|
ED_Alignment CAlignControls::GetAlignment()
|
|
{
|
|
// NOTE: ED_ defines are backward for Center and Bottom
|
|
// we should change them in EDTTYPES.H, but too much
|
|
// other code depends on them being wrong!
|
|
switch( m_nIDAlign ){
|
|
case IDC_EDAL_T:
|
|
m_EdAlign = ED_ALIGN_TOP;
|
|
break;
|
|
case IDC_EDAL_C:
|
|
case IDC_EDALCB:
|
|
m_EdAlign = ED_ALIGN_ABSCENTER; // Should write "CENTER" to HTML
|
|
break;
|
|
case IDC_EDAL_L:
|
|
m_EdAlign = ED_ALIGN_LEFT;
|
|
break;
|
|
case IDC_EDAL_R:
|
|
m_EdAlign = ED_ALIGN_RIGHT;
|
|
break;
|
|
default:
|
|
m_EdAlign = ED_ALIGN_BASELINE; // Should write no param (default)
|
|
break;
|
|
}
|
|
return m_EdAlign;
|
|
}
|
|
|
|
void CAlignControls::SetAlignment()
|
|
{
|
|
OnAlignButtonClick(m_nIDAlign);
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////
|
|
// Image dialog page.
|
|
// Note that we must supply Image data since we may be sharing
|
|
// it with Href data.
|
|
// Thus we need a flag to tell us to insert new image.
|
|
/////////////////////////////////////////////////////////////////////
|
|
CImagePage::CImagePage(CWnd* pParent, MWContext * pMWContext,
|
|
CEditorResourceSwitcher * pResourceSwitcher,
|
|
EDT_ImageData * pData, BOOL bInsert)
|
|
: CNetscapePropertyPage(CImagePage::IDD),
|
|
m_pMWContext(pMWContext),
|
|
m_pResourceSwitcher(pResourceSwitcher),
|
|
m_bActivated(0),
|
|
m_bInsert(bInsert),
|
|
m_pData(pData)
|
|
{
|
|
ASSERT(pMWContext);
|
|
ASSERT(pData);
|
|
|
|
//{{AFX_DATA_INIT(CImagePage)
|
|
m_csImage = _T("");
|
|
m_csAltText = _T("");
|
|
m_bNoSave = 0;
|
|
m_bSetAsBackground = 0;
|
|
m_iHeight = 0;
|
|
m_iWidth = 0;
|
|
m_iHSpace = 0;
|
|
m_iVSpace = 0;
|
|
m_iBorder = 0;
|
|
m_bDefaultBorder = FALSE;
|
|
m_iHeightPixOrPercent = 0;
|
|
m_iWidthPixOrPercent = 0;
|
|
m_bLockAspect = 1;
|
|
//}}AFX_DATA_INIT
|
|
m_csHref = _T("");
|
|
m_csImageStart = _T("");
|
|
m_csLastValidImage = _T("");
|
|
// m_csLastValidLowRes = _T("");
|
|
m_bValidImage = FALSE;
|
|
// m_bValidLowRes = FALSE;
|
|
m_bImageChanged = FALSE;
|
|
m_bOriginalButtonPressed = FALSE;
|
|
m_bLockAspect = TRUE;
|
|
|
|
wfe_GetLayoutViewSize(pMWContext, &m_iFullWidth, &m_iFullHeight);
|
|
|
|
#ifdef XP_WIN32
|
|
// Set the hInstance so we get template from editor's resource DLL
|
|
m_psp.hInstance = AfxGetResourceHandle();
|
|
#endif
|
|
}
|
|
|
|
void CImagePage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CImagePage)
|
|
DDX_Text(pDX, IDC_IMAGE_URL, m_csImage);
|
|
DDX_Check(pDX, IDC_NO_SAVE_IMAGE, m_bNoSave);
|
|
DDX_Check(pDX, IDC_LOCK_ASPECT, m_bLockAspect);
|
|
DDX_Check(pDX, IDC_MAKE_IMAGE_BACKGROUND, m_bSetAsBackground);
|
|
DDX_Text(pDX, IDC_IMAGE_HEIGHT, m_iHeight);
|
|
DDV_MinMaxInt(pDX, m_iHeight, 0, 10000);
|
|
DDX_Text(pDX, IDC_IMAGE_WIDTH, m_iWidth);
|
|
DDV_MinMaxInt(pDX, m_iWidth, 0, 10000);
|
|
DDX_Text(pDX, IDC_IMAGE_SPACE_HORIZ, m_iHSpace);
|
|
DDV_MinMaxInt(pDX, m_iHSpace, 0, 1000);
|
|
DDX_Text(pDX, IDC_IMAGE_SPACE_VERT, m_iVSpace);
|
|
DDV_MinMaxInt(pDX, m_iVSpace, 0, 1000);
|
|
DDX_Text(pDX, IDC_IMAGE_BORDER, m_iBorder);
|
|
DDV_MinMaxInt(pDX, m_iBorder, 0, 1000);
|
|
DDX_CBIndex(pDX, IDC_HEIGHT_PIX_OR_PERCENT, m_iHeightPixOrPercent);
|
|
DDX_CBIndex(pDX, IDC_WIDTH_PIX_OR_PERCENT, m_iWidthPixOrPercent);
|
|
DDX_Text(pDX, IDC_IMAGE_ALT_TEXT, m_csAltText);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CImagePage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CImagePage)
|
|
ON_BN_CLICKED(IDC_IMAGE_FILE, OnImageFile)
|
|
ON_EN_CHANGE(IDC_IMAGE_URL, OnChangeImageURL)
|
|
ON_EN_KILLFOCUS(IDC_IMAGE_URL, OnKillfocusImage)
|
|
ON_BN_CLICKED(IDC_IMAGE_ORIGINAL_SIZE, OnImageOriginalSize)
|
|
ON_BN_CLICKED(IDC_EDIT_IMAGE, OnEditImage)
|
|
ON_BN_CLICKED(IDC_NO_SAVE_IMAGE, OnNoSave)
|
|
ON_BN_CLICKED(IDC_REMOVE_ISMAP, OnRemoveIsmap)
|
|
ON_BN_CLICKED(IDC_MAKE_IMAGE_BACKGROUND, OnSetAsBackground)
|
|
ON_BN_CLICKED(IDC_EDAL_A, OnAlignBaseline)
|
|
ON_BN_CLICKED(IDC_EDAL_B, OnAlignBottom)
|
|
ON_BN_CLICKED(IDC_EDAL_C, OnAlignCenter)
|
|
ON_BN_CLICKED(IDC_EDAL_L, OnAlignLeft)
|
|
ON_BN_CLICKED(IDC_EDAL_R, OnAlignRight)
|
|
ON_BN_CLICKED(IDC_EDAL_T, OnAlignTop)
|
|
ON_BN_CLICKED(IDC_EDALCB, OnAlignCenterBaseline)
|
|
ON_EN_CHANGE(IDC_IMAGE_HEIGHT, OnChangeHeight)
|
|
ON_EN_CHANGE(IDC_IMAGE_WIDTH, OnChangeWidth)
|
|
ON_CBN_SELCHANGE(IDC_HEIGHT_PIX_OR_PERCENT, OnSelchangeHeightPixOrPercent)
|
|
ON_CBN_SELCHANGE(IDC_WIDTH_PIX_OR_PERCENT, OnSelchangeWidthPixOrPercent)
|
|
ON_EN_CHANGE(IDC_IMAGE_SPACE_HORIZ, OnChangeSpaceHoriz)
|
|
ON_EN_CHANGE(IDC_IMAGE_SPACE_VERT, OnChangeSpaceVert)
|
|
ON_EN_CHANGE(IDC_IMAGE_BORDER, OnChangeBorder)
|
|
ON_BN_CLICKED(IDC_EXTRA_HTML, OnExtraHTML)
|
|
ON_BN_CLICKED(IDC_LOCK_ASPECT, OnLockAspect)
|
|
ON_EN_CHANGE(IDC_IMAGE_ALT_TEXT, OnChangeAltText)
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
BOOL CImagePage::OnSetActive()
|
|
{
|
|
if(m_pResourceSwitcher && !m_bActivated){
|
|
// We must be sure we have switched
|
|
// the first time here - before dialog creation
|
|
m_pResourceSwitcher->switchResources();
|
|
}
|
|
if(!CPropertyPage::OnSetActive())
|
|
return(FALSE);
|
|
|
|
if(m_bActivated)
|
|
return(TRUE);
|
|
// Switch back to EXE's resources
|
|
if( m_pResourceSwitcher ){
|
|
m_pResourceSwitcher->Reset();
|
|
}
|
|
|
|
if ( m_pData->pSrc && XP_STRLEN(m_pData->pSrc) > 0 ){
|
|
m_csImage = XP_STRDUP(m_pData->pSrc);
|
|
}
|
|
else if ( !m_bInsert ) {
|
|
TRACE0("No Image Filename for Image Properties\n");
|
|
return FALSE;
|
|
}
|
|
|
|
m_bActivated = TRUE;
|
|
|
|
// Translate ImageData defines Align buttons index
|
|
// NOTE: ED_ defines are backward for Center and Bottom
|
|
// we should change them in EDTTYPES.H, but too much
|
|
// other code depends on them being wrong!
|
|
m_AlignControls.m_EdAlign = m_pData->align;
|
|
switch ( m_pData->align ){
|
|
case ED_ALIGN_LEFT:
|
|
m_AlignControls.m_nIDAlign = IDC_EDAL_L;
|
|
break;
|
|
case ED_ALIGN_RIGHT:
|
|
m_AlignControls.m_nIDAlign = IDC_EDAL_R;
|
|
break;
|
|
case ED_ALIGN_TOP:
|
|
case ED_ALIGN_ABSTOP:
|
|
m_AlignControls.m_nIDAlign = IDC_EDAL_T;
|
|
break;
|
|
case ED_ALIGN_ABSCENTER:
|
|
m_AlignControls.m_nIDAlign = IDC_EDALCB;
|
|
break;
|
|
case ED_ALIGN_CENTER:
|
|
m_AlignControls.m_nIDAlign = IDC_EDAL_C;
|
|
break;
|
|
case ED_ALIGN_BOTTOM:
|
|
m_AlignControls.m_nIDAlign = IDC_EDAL_B;
|
|
break;
|
|
case ED_ALIGN_ABSBOTTOM:
|
|
case ED_ALIGN_BASELINE:
|
|
default:
|
|
m_AlignControls.m_nIDAlign = IDC_EDAL_A;
|
|
break;
|
|
}
|
|
|
|
// Intialize the common Alignment / Sizing controls;
|
|
if ( !m_AlignControls.Init(this) ) {
|
|
return FALSE;
|
|
}
|
|
// Use suplied values only if they existed,
|
|
// use our defaults (5 pixels) for new object
|
|
if ( ! m_bInsert ){
|
|
m_iVSpace = CASTINT(m_pData->iVSpace);
|
|
m_iHSpace = CASTINT(m_pData->iHSpace);
|
|
}
|
|
if( m_pData->iBorder >= 0 ){
|
|
m_iBorder = CASTINT(m_pData->iBorder);
|
|
} else {
|
|
// We were given the default border value of -1
|
|
// Set flag to restore -1 if user doesn't change it,
|
|
// but it shows as 0 in the edit box
|
|
m_bDefaultBorder = TRUE;
|
|
m_iBorder = EDT_GetDefaultBorderWidth(m_pMWContext);
|
|
}
|
|
m_iWidth = CASTINT(m_pData->iWidth);
|
|
m_iHeight = CASTINT(m_pData->iHeight);
|
|
m_iHeightPixOrPercent = m_pData->bHeightPercent ? 1 : 0;
|
|
m_iWidthPixOrPercent = m_pData->bWidthPercent ? 1 : 0;
|
|
|
|
// We try to get the "Original" dimensions, or at least
|
|
// those when image was last loaded
|
|
if( m_pData->iOriginalWidth ){
|
|
m_iOriginalWidth = m_pData->iOriginalWidth;
|
|
m_iOriginalHeight = m_pData->iOriginalHeight;
|
|
} else {
|
|
m_iOriginalWidth = m_pData->iWidth;
|
|
m_iOriginalHeight = m_pData->iHeight;
|
|
}
|
|
// Avoid divide by zero
|
|
m_iOriginalWidth = max(1, m_iOriginalWidth);
|
|
m_iOriginalHeight = max(1, m_iOriginalHeight);
|
|
|
|
// Fill drop-lists of units
|
|
wfe_InitPixOrPercentCombos(this);
|
|
|
|
// Controls specific to Image page:
|
|
m_csAltText = m_pData->pAlt;
|
|
|
|
// Get possible HREF for image
|
|
if(m_pData->pHREFData && m_pData->pHREFData->pURL){
|
|
m_csHref = m_pData->pHREFData->pURL;
|
|
}
|
|
|
|
// Save initial image name to test
|
|
// before creating a link
|
|
m_csImageStart = m_csImage;
|
|
// Also save last valid image filenames
|
|
m_csLastValidImage = m_csImage;
|
|
m_bImageChanged = FALSE;
|
|
|
|
// Only allow removing bIsMap on images that already have it,
|
|
// i.e., we can't add it to raw images (YET!)
|
|
(GetDlgItem(IDC_REMOVE_ISMAP))->EnableWindow(m_pData->bIsMap);
|
|
|
|
SetOKEnable();
|
|
|
|
m_bNoSave = m_pData->bNoSave;
|
|
|
|
SetLockAspectEnable();
|
|
|
|
// Send data to controls
|
|
UpdateData(FALSE);
|
|
// Allow Apply button to be active if we are inserting a new object
|
|
SetModified(m_bInsert);
|
|
return(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnHelp()
|
|
{
|
|
NetHelp(HELP_PROPS_IMAGE);
|
|
}
|
|
|
|
void CImagePage::SetOKEnable()
|
|
{
|
|
// Enable OK only if we have both Image URL and Alt text,
|
|
// but Alt text isn't needed if image will be used for background
|
|
(GetParent()->GetDlgItem(IDOK))->EnableWindow( !(m_csImage.IsEmpty()) &&
|
|
(m_bSetAsBackground || !(m_csAltText.IsEmpty())) );
|
|
}
|
|
|
|
void CImagePage::OnOK()
|
|
{
|
|
//EDT_BeginBatchChanges(m_pMWContext);
|
|
|
|
|
|
// Always set HREF data for image if struct exists, even if we didn't visit this page
|
|
// since the value may be changed by CLinkPage
|
|
if( m_pData->pHREFData )
|
|
{
|
|
// If m_pData->pHREFData->pURL is NULL or empty, this clears any existing link
|
|
EDT_SetHREFData(m_pMWContext, m_pData->pHREFData);
|
|
}
|
|
|
|
if(!m_bActivated ||
|
|
// no change
|
|
!IS_APPLY_ENABLED(this) ||
|
|
// or error in data
|
|
!UpdateData(TRUE) )
|
|
{
|
|
//EDT_EndBatchChanges(m_pMWContext);
|
|
return;
|
|
}
|
|
|
|
if ( m_csImage.IsEmpty() )
|
|
{
|
|
// No image -- do nothing
|
|
// WHAT IF THERE IS A LOWRES IMAGE??? ADD MESSAGEBOX?
|
|
if ( m_bInsert )
|
|
return;
|
|
|
|
// TODO: delete current image here?
|
|
// Currently, EDT_SetImageData does not check for m_pImageData = NULL;
|
|
return;
|
|
}
|
|
|
|
// Validate/Relativize images
|
|
// (Shouldn't really need this - validation is done on killfocus of edit boxes)
|
|
if ( m_bImageChanged && !m_bValidImage )
|
|
{
|
|
if ( !wfe_ValidateImage( m_pMWContext, m_csImage ) )
|
|
{
|
|
m_bValidImage = TRUE;
|
|
UpdateData(FALSE);
|
|
return;
|
|
}
|
|
}
|
|
int iLastDot = m_csImage.ReverseFind('.');
|
|
CString csExt;
|
|
if(iLastDot > 0)
|
|
csExt= m_csImage.Mid(iLastDot);
|
|
|
|
//we must check to see if file is a bmp!
|
|
if (0 == csExt.CompareNoCase(".bmp"))
|
|
{
|
|
char *t_outputfilename=wfe_ConvertImage(m_csImage.GetBuffer(0),(void *)this,m_pMWContext);
|
|
if (t_outputfilename)
|
|
{
|
|
m_csImage=t_outputfilename;
|
|
wfe_ValidateImage( m_pMWContext, m_csImage );
|
|
XP_FREE(t_outputfilename);
|
|
UpdateData(FALSE);//we need to update m_csImage!
|
|
}
|
|
else
|
|
return;
|
|
}
|
|
|
|
if( m_bSetAsBackground )
|
|
{
|
|
// Real simple - ignore all data except for image name and save
|
|
EDT_PageData * pPageData = EDT_GetPageData(m_pMWContext);
|
|
if( pPageData )
|
|
{
|
|
XP_FREEIF(pPageData->pBackgroundImage);
|
|
pPageData->pBackgroundImage = XP_STRDUP((char*)LPCSTR(m_csImage));
|
|
pPageData->bBackgroundNoSave = m_bNoSave;
|
|
EDT_SetPageData(m_pMWContext, pPageData);
|
|
EDT_FreePageData(pPageData);
|
|
}
|
|
} else {
|
|
// Get the Alignment/Size data
|
|
m_pData->align = m_AlignControls.GetAlignment();
|
|
if( m_bOriginalButtonPressed )
|
|
{
|
|
// Trick backend into getting size from image,
|
|
// not the values edited
|
|
m_pData->iWidth = 0;
|
|
m_pData->iHeight = 0;
|
|
} else {
|
|
m_pData->iWidth = m_iWidth;
|
|
m_pData->iHeight = m_iHeight;
|
|
}
|
|
m_pData->iHSpace = m_iHSpace;
|
|
m_pData->iVSpace = m_iVSpace;
|
|
|
|
if( m_bDefaultBorder )
|
|
m_pData->iBorder = -1;
|
|
else
|
|
m_pData->iBorder = m_iBorder;
|
|
|
|
m_pData->bWidthPercent = m_iWidthPixOrPercent;
|
|
m_pData->bHeightPercent = m_iHeightPixOrPercent;
|
|
|
|
// Data specific to Image:
|
|
CleanupString(m_csAltText);
|
|
|
|
if ( m_pData->pSrc )
|
|
XP_FREE(m_pData->pSrc);
|
|
|
|
m_pData->pSrc = XP_STRDUP(m_csImage);
|
|
|
|
if ( m_pData->pLowSrc ){
|
|
XP_FREE(m_pData->pLowSrc);
|
|
m_pData->pLowSrc = NULL;
|
|
}
|
|
|
|
// Note: deleting Alt text in editbox to remove Alt Text
|
|
if ( m_pData->pAlt )
|
|
{
|
|
XP_FREE(m_pData->pAlt);
|
|
m_pData->pAlt = NULL;
|
|
}
|
|
if ( !m_csAltText.IsEmpty() )
|
|
{
|
|
m_pData->pAlt = XP_STRDUP(m_csAltText);
|
|
}
|
|
|
|
m_pData->bNoSave = m_bNoSave;
|
|
if ( m_bInsert )
|
|
{
|
|
EDT_InsertImage(m_pMWContext, m_pData, !m_bNoSave);
|
|
// We insert just ONE image (on 1st "Apply" usage)
|
|
// Thus other Apply or OK will modify newly-inserted image
|
|
m_bInsert = FALSE;
|
|
}
|
|
else
|
|
{
|
|
EDT_SetImageData(m_pMWContext, m_pData, !m_bNoSave);
|
|
}
|
|
//Note: ImageData and HrefData should be freed by caller
|
|
}
|
|
|
|
OkToClose();
|
|
//EDT_EndBatchChanges(m_pMWContext);
|
|
CPropertyPage::OnOK();
|
|
}
|
|
|
|
// Get and validate the Image name
|
|
// so it is up to date if we switch to the Link dialog page
|
|
BOOL CImagePage::OnKillActive()
|
|
{
|
|
if( !UpdateData(TRUE) )
|
|
return FALSE;
|
|
|
|
if ( m_bImageChanged && !m_bValidImage )
|
|
wfe_ValidateImage( m_pMWContext, m_csImage );
|
|
|
|
CleanupString(m_csAltText);
|
|
CleanupString(m_csImage);
|
|
|
|
if( m_csImage.IsEmpty() || (!m_bSetAsBackground && m_csAltText.IsEmpty()) )
|
|
{
|
|
// Notify user they must have both image URL and alt text filled in
|
|
MessageBox(szLoadString(IDS_MISSING_IMAGE),
|
|
szLoadString(IDS_IMAGE_PROPS_CAPTION),
|
|
MB_ICONEXCLAMATION | MB_OK);
|
|
|
|
// Put focus in the offending control
|
|
// And select all text, just like DDV functions
|
|
CEdit *pEdit = (CEdit*)GetDlgItem(m_csImage.IsEmpty() ? IDC_IMAGE_URL : IDC_IMAGE_ALT_TEXT);
|
|
pEdit->SetFocus();
|
|
pEdit->SetSel(0, -1, TRUE);
|
|
return FALSE;
|
|
}
|
|
|
|
XP_FREEIF(m_pData->pSrc);
|
|
m_pData->pSrc = XP_STRDUP(m_csImage);
|
|
|
|
// Contrary to MFC help, this does NOT call our OnOK
|
|
return CPropertyPage::OnKillActive();
|
|
}
|
|
|
|
// Called from the View after saving file to disk -- has new image
|
|
// in a URL form relative to current document
|
|
void CImagePage::SetImageFileSaved(char * pImageURL, int iImageNumber )
|
|
{
|
|
UpdateData(TRUE);
|
|
|
|
if( iImageNumber == 1 )
|
|
m_csImage = pImageURL;
|
|
|
|
UpdateData(FALSE);
|
|
}
|
|
|
|
void CImagePage::OnImageFile()
|
|
{
|
|
UpdateData(TRUE);
|
|
char * szFilename = wfe_GetExistingImageFileName(this->m_hWnd,
|
|
szLoadString(IDS_SELECT_IMAGE), TRUE);
|
|
if ( szFilename != NULL )
|
|
{
|
|
m_csImage = szFilename;
|
|
// Note that we don't tell user if file is "bad" since
|
|
// it is difficult to validate in all cases
|
|
wfe_ValidateImage( m_pMWContext, m_csImage );
|
|
XP_FREE( szFilename );
|
|
SetModified(TRUE);
|
|
// Supply a default AltText = image filename
|
|
AutoFillAltText();
|
|
UpdateData(FALSE);
|
|
m_bValidImage = TRUE;
|
|
m_csLastValidImage = m_csImage;
|
|
SetOKEnable();
|
|
}
|
|
}
|
|
|
|
void CImagePage::OnChangeImageURL()
|
|
{
|
|
m_bImageChanged = TRUE;
|
|
m_bValidImage = FALSE;
|
|
SetModified(TRUE);
|
|
// Disable Edit button if no image name
|
|
GetDlgItem(IDC_IMAGE_URL)->GetWindowText(m_csImage);
|
|
m_csImage.TrimLeft();
|
|
m_csImage.TrimRight();
|
|
GetDlgItem(IDC_EDIT_IMAGE)->EnableWindow(!m_csImage.IsEmpty());
|
|
SetOKEnable();
|
|
}
|
|
|
|
void CImagePage::OnKillfocusImage()
|
|
{
|
|
if( m_bImageChanged &&
|
|
UpdateData(TRUE) )
|
|
{
|
|
wfe_ValidateImage( m_pMWContext, m_csImage );
|
|
m_bValidImage = TRUE;
|
|
AutoFillAltText();
|
|
UpdateData(FALSE);
|
|
}
|
|
}
|
|
|
|
// Automatically fill in the ALT text string if none currently
|
|
void CImagePage::AutoFillAltText()
|
|
{
|
|
CleanupString(m_csAltText);
|
|
if( m_csAltText.IsEmpty() )
|
|
{
|
|
char *pName = EDT_GetFilename(CHAR_STR(m_csImage), FALSE);
|
|
if( pName )
|
|
{
|
|
m_csAltText = XP_STRDUP(pName);
|
|
XP_FREE(pName);
|
|
}
|
|
SetOKEnable();
|
|
}
|
|
}
|
|
void CImagePage::OnEditImage()
|
|
{
|
|
UpdateData(TRUE);
|
|
// Get our view from the context and call edit method
|
|
((CNetscapeEditView*)WINCX(m_pMWContext)->GetView())->EditImage((char*)LPCSTR(m_csImage));
|
|
}
|
|
|
|
void CImagePage::SetLockAspectEnable()
|
|
{
|
|
GetDlgItem(IDC_LOCK_ASPECT)->EnableWindow(!m_bSetAsBackground && !m_iHeightPixOrPercent && !m_iWidthPixOrPercent);
|
|
}
|
|
|
|
void CImagePage::OnSetAsBackground()
|
|
{
|
|
m_bSetAsBackground = ((CButton*)GetDlgItem(IDC_MAKE_IMAGE_BACKGROUND))->GetCheck();
|
|
// Enable or Disable all other controls - irrelevant when simply setting background
|
|
GetDlgItem(IDC_EDAL_T)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_EDAL_C)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_EDALCB)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_EDAL_A)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_EDAL_B)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_EDAL_L)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_EDAL_R)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_IMAGE_HEIGHT)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_HEIGHT_PIX_OR_PERCENT)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_IMAGE_WIDTH)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_WIDTH_PIX_OR_PERCENT)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_IMAGE_ORIGINAL_SIZE)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_IMAGE_SPACE_HORIZ)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_IMAGE_SPACE_VERT)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_IMAGE_BORDER)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_EXTRA_HTML)->EnableWindow(!m_bSetAsBackground);
|
|
GetDlgItem(IDC_REMOVE_ISMAP)->EnableWindow(!m_bSetAsBackground && m_pData->bIsMap);
|
|
GetDlgItem(IDC_IMAGE_ALT_TEXT)->EnableWindow(!m_bSetAsBackground);
|
|
|
|
SetLockAspectEnable();
|
|
SetOKEnable();
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnChangeAltText()
|
|
{
|
|
// Get the alt text
|
|
UpdateData(TRUE);
|
|
CleanupString(m_csAltText);
|
|
SetOKEnable();
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnNoSave()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnRemoveIsmap()
|
|
{
|
|
m_pData->bIsMap = FALSE;
|
|
// Once removed, we can't add it back
|
|
(GetDlgItem(IDC_REMOVE_ISMAP))->EnableWindow(FALSE);
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
// Align/Size controls:
|
|
void CImagePage::OnAlignBaseline()
|
|
{
|
|
m_AlignControls.OnAlignButtonClick(IDC_EDAL_A);
|
|
}
|
|
|
|
void CImagePage::OnAlignBottom()
|
|
{
|
|
m_AlignControls.OnAlignButtonClick(IDC_EDAL_B);
|
|
}
|
|
|
|
void CImagePage::OnAlignCenter()
|
|
{
|
|
m_AlignControls.OnAlignButtonClick(IDC_EDAL_C);
|
|
}
|
|
|
|
void CImagePage::OnAlignLeft()
|
|
{
|
|
m_AlignControls.OnAlignButtonClick(IDC_EDAL_L);
|
|
}
|
|
|
|
void CImagePage::OnAlignRight()
|
|
{
|
|
m_AlignControls.OnAlignButtonClick(IDC_EDAL_R);
|
|
}
|
|
|
|
void CImagePage::OnAlignTop()
|
|
{
|
|
m_AlignControls.OnAlignButtonClick(IDC_EDAL_T);
|
|
}
|
|
|
|
void CImagePage::OnAlignCenterBaseline()
|
|
{
|
|
m_AlignControls.OnAlignButtonClick(IDC_EDALCB);
|
|
}
|
|
|
|
|
|
void CImagePage::OnImageOriginalSize()
|
|
{
|
|
// Set flag so we're sure user wants this later
|
|
m_bOriginalButtonPressed = TRUE;
|
|
|
|
UpdateData(TRUE);
|
|
|
|
m_iWidth = CASTINT(m_iOriginalWidth);
|
|
m_iHeight = CASTINT(m_iOriginalHeight);
|
|
|
|
// We must be in pixel mode, not % mode
|
|
m_iHeightPixOrPercent = 0;
|
|
m_iWidthPixOrPercent = 0;
|
|
|
|
UpdateData(FALSE);
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnChangeHeight()
|
|
{
|
|
if( m_bLockAspect && ((CButton*)GetDlgItem(IDC_LOCK_ASPECT))->IsWindowEnabled() )
|
|
{
|
|
// Get value just enterred and set the opposite
|
|
// to a value that keeps aspect ratio of original
|
|
CWnd *pHeightEdit = GetDlgItem(IDC_IMAGE_HEIGHT);
|
|
CWnd *pWidthEdit = GetDlgItem(IDC_IMAGE_WIDTH);
|
|
char pValue[16];
|
|
char* pEnd;
|
|
pHeightEdit->GetWindowText(pValue, 10);
|
|
int32 iHeight = (int)strtol( pValue, &pEnd, 10 );
|
|
|
|
// Bad conversion if end pointer isn't at terminal null;
|
|
if( *pEnd == '\0' )
|
|
{
|
|
m_iHeight = iHeight;
|
|
// Add 0.5 to round off when converting back to int
|
|
m_iWidth = (int)((iHeight * m_iOriginalWidth) / m_iOriginalHeight);
|
|
wsprintf(pValue, "%d", m_iWidth);
|
|
// Avoid bouncing back and forth (and killing stack!)
|
|
// SetWindowText triggers OnChangeWidth
|
|
m_bLockAspect = FALSE;
|
|
pWidthEdit->SetWindowText(pValue);
|
|
m_bLockAspect = TRUE;
|
|
}
|
|
}
|
|
if( m_iHeight != m_iOriginalHeight)
|
|
m_bOriginalButtonPressed = FALSE;
|
|
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnChangeWidth()
|
|
{
|
|
if( m_bLockAspect && ((CButton*)GetDlgItem(IDC_LOCK_ASPECT))->IsWindowEnabled() )
|
|
{
|
|
// Get value just enterred and set the opposite
|
|
// to a value that keeps aspect ratio of original
|
|
CWnd *pWidthEdit = GetDlgItem(IDC_IMAGE_WIDTH);
|
|
CWnd *pHeightEdit = GetDlgItem(IDC_IMAGE_HEIGHT);
|
|
char pValue[16];
|
|
char* pEnd;
|
|
pWidthEdit->GetWindowText(pValue, 10);
|
|
int32 iWidth = (int32)strtol( pValue, &pEnd, 10 );
|
|
|
|
if( *pEnd == '\0' )
|
|
{
|
|
m_iWidth = iWidth;
|
|
m_iHeight = (int)((iWidth * m_iOriginalHeight) / m_iOriginalWidth);
|
|
wsprintf(pValue, "%d", m_iHeight);
|
|
|
|
// Avoid bouncing back and forth (and killing stack!)
|
|
// SetWindowText triggers OnChangeHeight
|
|
m_bLockAspect = FALSE;
|
|
pHeightEdit->SetWindowText(pValue);
|
|
m_bLockAspect = TRUE;
|
|
}
|
|
}
|
|
if( m_iWidth != m_iOriginalWidth)
|
|
m_bOriginalButtonPressed = FALSE;
|
|
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnSelchangeHeightPixOrPercent()
|
|
{
|
|
UpdateData();
|
|
SetLockAspectEnable();
|
|
//TODO: do number conversion if switching state?
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnSelchangeWidthPixOrPercent()
|
|
{
|
|
UpdateData();
|
|
SetLockAspectEnable();
|
|
//TODO: do number conversion if switching state?
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnChangeSpaceHoriz()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnChangeSpaceVert()
|
|
{
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
|
|
void CImagePage::OnChangeBorder()
|
|
{
|
|
SetModified(TRUE);
|
|
// If user changed the border,
|
|
// then use that number instead
|
|
// of the default -1
|
|
m_bDefaultBorder = FALSE;
|
|
}
|
|
|
|
void CImagePage::OnExtraHTML()
|
|
{
|
|
CExtraHTMLDlg dlg(this, &m_pData->pExtra, IDS_IMG_TAG);
|
|
if( dlg.DoModal() && dlg.m_bDataChanged )
|
|
SetModified(TRUE);
|
|
}
|
|
|
|
void CImagePage::OnLockAspect()
|
|
{
|
|
m_bLockAspect = ((CButton*)GetDlgItem(IDC_LOCK_ASPECT))->GetCheck();
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CExtraHTMLDlg dialog (modal popup over CImagePage or CLinkPage for Extra HTML
|
|
|
|
CExtraHTMLDlg::CExtraHTMLDlg(CWnd *pParent, char **ppExtraHTML, UINT nIDTagType)
|
|
: CDialog(CExtraHTMLDlg::IDD, pParent),
|
|
m_ppExtraHTML(ppExtraHTML),
|
|
m_bDataChanged(FALSE),
|
|
m_nIDTagType(nIDTagType)
|
|
{
|
|
ASSERT( ppExtraHTML );
|
|
//{{AFX_DATA_INIT(CExtraHTMLDlg)
|
|
m_csExtraHTML = *ppExtraHTML;
|
|
//}}AFX_DATA_INIT
|
|
}
|
|
|
|
CExtraHTMLDlg::~CExtraHTMLDlg()
|
|
{
|
|
}
|
|
|
|
void CExtraHTMLDlg::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CDialog::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CExtraHTMLDlg)
|
|
DDX_Text(pDX, IDC_EXTRA_HTML_TEXT, m_csExtraHTML);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
|
|
BEGIN_MESSAGE_MAP(CExtraHTMLDlg, CDialog)
|
|
//{{AFX_MSG_MAP(CExtraHTMLDlg)
|
|
ON_BN_CLICKED(ID_HELP, OnHelp)
|
|
//}}AFX_MSG_MAP
|
|
#ifdef XP_WIN32
|
|
ON_WM_HELPINFO()
|
|
#endif //XP_WIN32
|
|
END_MESSAGE_MAP()
|
|
|
|
|
|
BOOL CExtraHTMLDlg::OnInitDialog()
|
|
{
|
|
// Switch back to NETSCAPE.EXE for resource hInstance
|
|
m_ResourceSwitcher.Reset();
|
|
|
|
CDialog::OnInitDialog();
|
|
|
|
// Insert the text describing the tag type into the message and display
|
|
CString csMsg;
|
|
AfxFormatString1( csMsg, IDS_EXTRA_HTML_MSG, szLoadString(m_nIDTagType) );
|
|
GetDlgItem(IDC_EXTRA_HTML_MSG)->SetWindowText(LPCSTR(csMsg));
|
|
|
|
return TRUE; // return TRUE unless you set the focus to a control
|
|
// EXCEPTION: OCX Property Pages should return FALSE
|
|
}
|
|
|
|
void CExtraHTMLDlg::OnHelp()
|
|
{
|
|
NetHelp(HELP_EXTRA_HTML);
|
|
}
|
|
|
|
#ifdef XP_WIN32
|
|
BOOL CExtraHTMLDlg::OnHelpInfo(HELPINFO *)//32bit messagemapping.
|
|
{
|
|
OnHelp();
|
|
return TRUE;
|
|
}
|
|
#endif//XP_WIN32
|
|
|
|
void CExtraHTMLDlg::OnOK()
|
|
{
|
|
CDialog::OnOK();
|
|
// Strip off leading and ending spaces
|
|
m_csExtraHTML.TrimLeft();
|
|
m_csExtraHTML.TrimRight();
|
|
|
|
// Did we already have some data?
|
|
BOOL bHadExtra = 0 != *m_ppExtraHTML;
|
|
|
|
if( !bHadExtra && m_csExtraHTML.IsEmpty() ){
|
|
// We didn't have any before and none now, so we're done
|
|
return;
|
|
}
|
|
|
|
if( bHadExtra && 0 == XP_STRCMP(*m_ppExtraHTML, LPCSTR(m_csExtraHTML)) ){
|
|
// We had data before and it didn't change so we're done
|
|
return;
|
|
}
|
|
|
|
// If here, new data must be different than previous text
|
|
m_bDataChanged = TRUE;
|
|
if( bHadExtra ) XP_FREE(*m_ppExtraHTML);
|
|
|
|
if(m_csExtraHTML.IsEmpty() ){
|
|
// No new text
|
|
*m_ppExtraHTML = NULL;
|
|
} else {
|
|
// Copy new text
|
|
*m_ppExtraHTML = XP_STRDUP(LPCSTR(m_csExtraHTML));
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////
|
|
// Links dialog page.
|
|
// Note that we must supply Href data since we may be sharing
|
|
// it with Image data. If link is an image, *ppImage has name.
|
|
// Thus we need a flag to tell us to insert new link.
|
|
/////////////////////////////////////////////////////////////////////
|
|
CLinkPage::CLinkPage(CWnd* pParent, MWContext * pMWContext,
|
|
CEditorResourceSwitcher * pResourceSwitcher,
|
|
EDT_HREFData *pData, BOOL bInsert,
|
|
BOOL bMayHaveOtherLinks, char **ppImage)
|
|
: CNetscapePropertyPage(CLinkPage::IDD),
|
|
m_bActivated(0),
|
|
m_bInsert(bInsert),
|
|
m_pMWContext(pMWContext),
|
|
m_pResourceSwitcher(pResourceSwitcher),
|
|
m_bMayHaveOtherLinks(bMayHaveOtherLinks),
|
|
m_ppImage(ppImage),
|
|
m_pData(pData),
|
|
m_iTargetCount(0),
|
|
m_bValidHref(0),
|
|
m_bHrefChanged(0),
|
|
m_iCaretMovedBack(0)
|
|
{
|
|
//}}AFX_DATA_INIT
|
|
ASSERT(pMWContext);
|
|
ASSERT(pData);
|
|
|
|
m_szBaseDocument = NULL;
|
|
|
|
// Base URL is the address of current document
|
|
History_entry * hist_ent = SHIST_GetCurrent(&(m_pMWContext->hist));
|
|
if ( hist_ent ){
|
|
m_szBaseDocument = hist_ent->address;
|
|
}
|
|
|
|
// We will use some helper functions from our view
|
|
m_pView = (CNetscapeView*)WINCX(pMWContext)->GetView();
|
|
|
|
//{{AFX_DATA_INIT(CLinkPage)
|
|
m_csHref = _T("");
|
|
m_csAnchorEdit = _T("");
|
|
m_csAnchor = _T("");
|
|
//}}AFX_DATA_INIT
|
|
|
|
m_csLastValidHref = _T("");
|
|
|
|
#ifdef XP_WIN32
|
|
// Set the hInstance so we get template from editor's resource DLL
|
|
m_psp.hInstance = AfxGetResourceHandle();
|
|
#endif
|
|
}
|
|
|
|
CLinkPage::~CLinkPage()
|
|
{
|
|
// Reposition the caret to where it was
|
|
// when new link text was inserted
|
|
while( m_iCaretMovedBack ){
|
|
EDT_NextChar(m_pMWContext, FALSE);
|
|
m_iCaretMovedBack--;
|
|
}
|
|
|
|
}
|
|
|
|
void CLinkPage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CLinkPage)
|
|
// NOTE: the ClassWizard will add DDX and DDV calls here
|
|
DDX_Text(pDX, IDC_HREF_URL, m_csHref);
|
|
DDX_Text(pDX, IDC_ANCHOR_EDIT, m_csAnchorEdit);
|
|
DDX_Text(pDX, IDC_ANCHOR, m_csAnchor);
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CLinkPage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CLinkPage)
|
|
ON_BN_CLICKED(IDC_HREF_FILE, OnHrefFile)
|
|
ON_BN_CLICKED(IDC_HREF_UNLINK, OnHrefUnlink)
|
|
ON_LBN_SELCHANGE(IDC_TARGET_LIST, OnSelchangeTargetList)
|
|
ON_EN_CHANGE(IDC_HREF_URL, OnChangeHrefUrl)
|
|
ON_EN_KILLFOCUS(IDC_HREF_URL, OnKillfocusHrefUrl)
|
|
ON_BN_CLICKED(IDC_TARGETS_IN_CURRENT_DOC, OnTargetsInCurrentDoc)
|
|
ON_BN_CLICKED(IDC_TARGETS_IN_FILE, OnTargetsInFile)
|
|
ON_BN_CLICKED(IDC_EXTRA_HTML, OnExtraHTML)
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
BOOL CLinkPage::OnSetActive()
|
|
{
|
|
if(m_pResourceSwitcher && !m_bActivated){
|
|
// We must be sure we have switched
|
|
// the first time here - before dialog creation
|
|
m_pResourceSwitcher->switchResources();
|
|
}
|
|
if(!CPropertyPage::OnSetActive())
|
|
return(FALSE);
|
|
|
|
BOOL bImageAnchor = FALSE;
|
|
|
|
// Always check the Imagename that might
|
|
// be changed in the Image property page
|
|
if( m_ppImage && *m_ppImage ){
|
|
m_csAnchor = *m_ppImage;
|
|
if(!m_csAnchor.IsEmpty()){
|
|
((CEdit*)GetDlgItem(IDC_ANCHOR))->SetWindowText(m_csAnchor);
|
|
bImageAnchor = TRUE;
|
|
}
|
|
}
|
|
|
|
if(m_bActivated)
|
|
return(TRUE);
|
|
|
|
// Switch back to EXE's resources
|
|
if( m_pResourceSwitcher ){
|
|
m_pResourceSwitcher->Reset();
|
|
}
|
|
|
|
m_bActivated = TRUE;
|
|
|
|
// Get the current URL
|
|
m_csHref = m_pData->pURL;
|
|
|
|
//TODO: GET CURRENT CHARACTER STATE OR PASS IN?
|
|
|
|
|
|
// Fill the "Display text" editbox
|
|
// with selected text
|
|
UINT nIDLabel = 0;
|
|
BOOL bSelected = EDT_IsSelected(m_pMWContext);
|
|
|
|
if (bImageAnchor){
|
|
// Change label for image
|
|
// (Note: We set Image filename string above)
|
|
nIDLabel = IDS_ANCHOR_IMAGE;
|
|
} else {
|
|
if(m_bInsert && !bSelected ){
|
|
// We will create/insert a new Anchor + Href at caret location
|
|
// Use editbox instead of text control
|
|
((CEdit*)GetDlgItem(IDC_ANCHOR))->ShowWindow(SW_HIDE);
|
|
nIDLabel = IDS_NEW_ANCHOR_TEXT;
|
|
}
|
|
else if( EDT_CanSetHREF(m_pMWContext) ) {
|
|
// We have a text link or selected text
|
|
if ( bSelected ) {
|
|
m_csAnchor = (char*)LO_GetSelectionText(ABSTRACTCX(m_pMWContext)->GetDocumentContext());
|
|
} else {
|
|
// We are not selected, but are within a text anchor
|
|
char * pText = EDT_GetHREFText(m_pMWContext);
|
|
m_csAnchor = pText;
|
|
XP_FREE(pText);
|
|
}
|
|
// Replace CR/LF with spaces to avoid ugly break in static display
|
|
for( int i=0; i < m_csAnchor.GetLength(); i++ ){
|
|
if( m_csAnchor.GetAt(i) == '\r' || m_csAnchor.GetAt(i) == '\n' ){
|
|
m_csAnchor.SetAt(i, ' ');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if( nIDLabel ) {
|
|
// Set label above image name or new text editbox
|
|
GetDlgItem(IDC_ANCHOR_LABEL)->SetWindowText(szLoadString(nIDLabel));
|
|
}
|
|
|
|
if( nIDLabel != IDS_NEW_ANCHOR_TEXT ) {
|
|
// Make the existing Anchor Text or Image Filename BOLD by using System Font
|
|
// if( INTL_CharSetType(m_pMWContext->doc_csid) == SINGLEBYTE*/) {
|
|
// This should be OK even in foreign systems
|
|
CFont* pfontBold = CFont::FromHandle((HFONT)GetStockObject(SYSTEM_FONT));
|
|
((CEdit*)GetDlgItem(IDC_ANCHOR))->SetFont(pfontBold);
|
|
// }
|
|
// No editable anchor, so hide editbox
|
|
(GetDlgItem(IDC_ANCHOR_EDIT))->ShowWindow(SW_HIDE);
|
|
}
|
|
|
|
// Save initial HREF to restore string
|
|
// if user changes to something invalid
|
|
m_csLastValidHref = m_csHref;
|
|
|
|
// Init unlink button state. Note that button shows
|
|
// if we may have other links within selection that we can remove
|
|
(GetDlgItem(IDC_HREF_UNLINK))->EnableWindow(!m_csHref.IsEmpty() || m_bMayHaveOtherLinks);
|
|
|
|
// Get the list of Targets (named anchors) in current doc
|
|
m_pTargetList = EDT_GetAllDocumentTargets(m_pMWContext);
|
|
|
|
if( m_pTargetList == NULL ){
|
|
// No targets in current doc, so disable button
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_CURRENT_DOC))->EnableWindow(FALSE);
|
|
}
|
|
|
|
// Try to generate a Targets list for current HREF (if local file),
|
|
// if none found, show the local list if it exists
|
|
if ( 0 == GetTargetsInFile() && m_pTargetList ){
|
|
if ( 0 == GetTargetsInDoc() ){
|
|
(GetDlgItem(IDC_TARGET_LABEL))->SetWindowText(szLoadString(IDS_NO_TARGETS));
|
|
}
|
|
}
|
|
if( m_bMayHaveOtherLinks ){
|
|
(GetDlgItem(IDC_HREF_UNLINK))->SetWindowText(szLoadString(IDS_REMOVE_LINKS));
|
|
}
|
|
|
|
// Send data to controls
|
|
UpdateData(FALSE);
|
|
// Allow Apply button to be active if we are inserting a new object
|
|
SetModified(m_bInsert);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Set the HREF data
|
|
// It will be used by Image property page
|
|
// to set HREF to an image
|
|
BOOL CLinkPage::OnKillActive()
|
|
{
|
|
if( !UpdateData() ){
|
|
return FALSE;
|
|
}
|
|
if ( !m_bValidHref ){
|
|
ValidateHref();
|
|
}
|
|
|
|
return CPropertyPage::OnKillActive();
|
|
}
|
|
|
|
int nTargetIndex = -1;
|
|
|
|
int CLinkPage::GetTargetsInDoc()
|
|
{
|
|
m_iTargetCount = 0;
|
|
CListBox * pListBox = (CListBox*)GetDlgItem(IDC_TARGET_LIST);
|
|
pListBox->ResetContent();
|
|
|
|
if( m_pTargetList ){
|
|
char * pString = m_pTargetList;
|
|
int iLen;
|
|
while( (iLen = XP_STRLEN(pString)) > 0 ) {
|
|
// add to the end of the list
|
|
pListBox->AddString(pString);
|
|
pString += iLen+1;
|
|
m_iTargetCount++;
|
|
}
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_CURRENT_DOC))->SetCheck(1/*m_iTargetCount ? 1 : 0*/);
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_FILE))->SetCheck(0);
|
|
|
|
if( m_iTargetCount ){
|
|
(GetDlgItem(IDC_TARGET_LABEL))->SetWindowText(szLoadString(IDS_TARGETS_IN_CURRENT_DOC));
|
|
}
|
|
}
|
|
|
|
return m_iTargetCount;
|
|
}
|
|
|
|
// File read from HTML to get list of anchors
|
|
int CLinkPage::GetTargetsInFile()
|
|
{
|
|
if( m_csHref.IsEmpty() ){
|
|
return 0;
|
|
}
|
|
m_iTargetCount = 0;
|
|
|
|
// Build list from file
|
|
char * pTargetList = EDT_GetAllDocumentTargetsInFile(
|
|
m_pMWContext, (char*)LPCSTR(m_csHref));
|
|
if( pTargetList ){
|
|
CListBox * pListBox = (CListBox*)GetDlgItem(IDC_TARGET_LIST);
|
|
pListBox->ResetContent();
|
|
char * pString = pTargetList;
|
|
int iLen;
|
|
while( (iLen = XP_STRLEN(pString)) > 0 ) {
|
|
// add to the end of the list
|
|
pListBox->AddString(pString);
|
|
pString += iLen+1;
|
|
m_iTargetCount++;
|
|
}
|
|
|
|
XP_FREE(pTargetList);
|
|
// Save filename so we can avoid rereading file needlessly
|
|
m_csTargetFile = m_csHref;
|
|
} else {
|
|
// HREF wasn't a file or
|
|
// was same as current
|
|
// TODO: How can we know if current URL is same as HREF?
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_FILE))->SetCheck(0);
|
|
return 0;
|
|
}
|
|
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_CURRENT_DOC))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_FILE))->SetCheck( m_iTargetCount ? 1 : 0);
|
|
|
|
// Set appropriate message above targets list
|
|
if ( m_iTargetCount ){
|
|
(GetDlgItem(IDC_TARGET_LABEL))->SetWindowText(szLoadString(IDS_TARGETS_IN_FILE));
|
|
} else {
|
|
(GetDlgItem(IDC_TARGET_LABEL))->SetWindowText(szLoadString(IDS_NO_TARGETS));
|
|
}
|
|
return m_iTargetCount;
|
|
}
|
|
|
|
|
|
void CLinkPage::OnTargetsInCurrentDoc()
|
|
{
|
|
if( m_pTargetList ){
|
|
GetTargetsInDoc();
|
|
}
|
|
}
|
|
|
|
void CLinkPage::OnTargetsInFile()
|
|
{
|
|
UpdateData();
|
|
GetTargetsInFile();
|
|
}
|
|
|
|
void CLinkPage::OnSelchangeTargetList()
|
|
{
|
|
UpdateData(TRUE);
|
|
SetModified(TRUE);
|
|
m_bValidHref = FALSE;
|
|
|
|
// Copy selection text
|
|
int nSel = ((CListBox*)GetDlgItem(IDC_TARGET_LIST))->GetCurSel();
|
|
|
|
nTargetIndex = nSel;
|
|
if( ((CButton*)GetDlgItem(IDC_TARGETS_IN_CURRENT_DOC))->GetCheck() ){
|
|
// For current doc, we need just the target
|
|
m_csHref = '#';
|
|
} else {
|
|
// For file, append target to current HREF,
|
|
// but strip off existing target first
|
|
int iHash = m_csHref.Find('#');
|
|
if( iHash>=0 ){
|
|
m_csHref = m_csHref.Left(iHash+1);
|
|
} else {
|
|
m_csHref += '#';
|
|
}
|
|
}
|
|
CString csTarget;
|
|
((CListBox*)GetDlgItem(IDC_TARGET_LIST))->GetText(nSel, csTarget);
|
|
m_csHref += csTarget;
|
|
(GetDlgItem(IDC_HREF_UNLINK))->EnableWindow(TRUE);
|
|
UpdateData(FALSE);
|
|
}
|
|
|
|
void CLinkPage::OnHrefFile()
|
|
{
|
|
// Get data from dialog
|
|
if ( !UpdateData(TRUE) ){
|
|
return;
|
|
}
|
|
// Note that the file filter is set to *.html,
|
|
// but all other file types are also available
|
|
CString csFile = wfe_GetExistingFileName(this->m_hWnd,
|
|
szLoadString(IDS_LINK_TO_FILE), HTM, TRUE);
|
|
if( !csFile.IsEmpty() ) {
|
|
WFE_ConvertFile2Url(m_csHref, csFile);
|
|
// Convert to a relative URL and validate
|
|
ValidateHref();
|
|
UpdateData(FALSE);
|
|
SetModified(TRUE);
|
|
// Try to get list of targets from local file
|
|
OnTargetsInFile();
|
|
}
|
|
}
|
|
|
|
void CLinkPage::OnHrefUnlink()
|
|
{
|
|
SetModified(TRUE);
|
|
|
|
// Clear link from our member variable and
|
|
// the passed-in data, and the edit box
|
|
m_csHref = "";
|
|
if(m_pData->pURL){
|
|
XP_FREE(m_pData->pURL);
|
|
m_pData->pURL = NULL;
|
|
}
|
|
((CEdit *)GetDlgItem(IDC_HREF_URL))->SetWindowText("");
|
|
|
|
ED_ElementType type = EDT_GetCurrentElementType(m_pMWContext);
|
|
if( (m_bMayHaveOtherLinks && type == ED_ELEMENT_SELECTION /*||
|
|
(type == ED_ELEMENT_SELECTION && EDT_CanSetHREF(m_pMWContext)) */) &&
|
|
IDYES == MessageBox(szLoadString(IDS_REMOVE_OTHER_LINKS),
|
|
szLoadString(IDS_REMOVE_LINKS_CAPTION),
|
|
MB_ICONQUESTION | MB_YESNO) ){
|
|
// Remove just the HREF immediately using masked bits
|
|
EDT_CharacterData *pCharData = EDT_NewCharacterData();
|
|
if( pCharData ){
|
|
pCharData->mask = TF_HREF;
|
|
// "New" should set all values = 0;
|
|
EDT_SetCharacterData(m_pMWContext, pCharData);
|
|
EDT_FreeCharacterData(pCharData);
|
|
// We now have no links to remove
|
|
m_bMayHaveOtherLinks = FALSE;
|
|
// and our action is similar to Apply button
|
|
// in that we commited an action
|
|
OkToClose();
|
|
(GetDlgItem(IDC_HREF_UNLINK))->SetWindowText(szLoadString(IDS_REMOVE_LINK));
|
|
}
|
|
}
|
|
(GetDlgItem(IDC_HREF_UNLINK))->EnableWindow(FALSE);
|
|
}
|
|
|
|
void CLinkPage::OnChangeHrefUrl()
|
|
{
|
|
SetModified(TRUE);
|
|
m_bHrefChanged = TRUE;
|
|
m_bValidHref = FALSE;
|
|
// We only need HREF, so don't bother with UpdateData()
|
|
CString csHREF;
|
|
GetDlgItem(IDC_HREF_URL)->GetWindowText(csHREF);
|
|
CleanupString(csHREF);
|
|
|
|
if( csHREF.IsEmpty() ){
|
|
// No URL, seems safe to redisplay the current list
|
|
GetTargetsInDoc();
|
|
} else if( (m_iTargetCount &&
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_CURRENT_DOC))->GetCheck()) ||
|
|
(((CButton*)GetDlgItem(IDC_TARGETS_IN_FILE))->GetCheck() &&
|
|
m_csHref.Find(LPCSTR(m_csTargetFile)) == -1) ){
|
|
// We had a current target list, or URL was changed
|
|
// so it isn't the same as the last target file used.
|
|
// To remove uncertainty of what the target list refers to, remove it
|
|
m_iTargetCount = 0;
|
|
m_csTargetFile.Empty();
|
|
((CListBox*)GetDlgItem(IDC_TARGET_LIST))->ResetContent();
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_CURRENT_DOC))->SetCheck(0);
|
|
((CButton*)GetDlgItem(IDC_TARGETS_IN_FILE))->SetCheck(0);
|
|
}
|
|
(GetDlgItem(IDC_HREF_UNLINK))->EnableWindow(!csHREF.IsEmpty() || m_bMayHaveOtherLinks);
|
|
|
|
// Update the common data structure with new HREF
|
|
// Can't use SetHrefData() cause we can't change m_csHREF
|
|
if(m_pData->pURL){
|
|
XP_FREE(m_pData->pURL);
|
|
m_pData->pURL = NULL;
|
|
}
|
|
if(!csHREF.IsEmpty()){
|
|
m_pData->pURL = XP_STRDUP(csHREF);
|
|
}
|
|
}
|
|
|
|
// Validate only after user leaves edit box
|
|
void CLinkPage::OnKillfocusHrefUrl()
|
|
{
|
|
if(m_bHrefChanged && UpdateData(TRUE)){
|
|
ValidateHref();
|
|
}
|
|
}
|
|
|
|
// Check for valid Href - convert to relative URL
|
|
// NOTE: Data must be read from control via DDX first
|
|
// This logic assumes that base document is local file,
|
|
// (We force saving a remote document before changing links)
|
|
void CLinkPage::ValidateHref()
|
|
{
|
|
m_bValidHref = TRUE;
|
|
CleanupString(m_csHref);
|
|
|
|
// Empty is OK - this is how we remove links
|
|
if ( m_csHref.IsEmpty() ){
|
|
m_csLastValidHref.Empty();
|
|
return;
|
|
}
|
|
// Strip off "#named_anchor" part and place in
|
|
// separate string???
|
|
|
|
XP_Bool bKeepLinks;
|
|
PREF_GetBoolPref("editor.publish_keep_links",&bKeepLinks);
|
|
if ( FE_ResolveLinkURL(m_pMWContext, m_csHref,bKeepLinks) ) {
|
|
// Save this as a valid reference
|
|
m_csLastValidHref = m_csHref;
|
|
} else {
|
|
// Error or user rejected the Href,
|
|
// restore to previous value
|
|
m_csHref = m_csLastValidHref;
|
|
SetModified(TRUE);
|
|
UpdateData(FALSE);
|
|
}
|
|
// We must always set the HREF data immediately
|
|
// because it will be accessed by
|
|
// CImagePage::OnOK before CLinkPage::OnOK
|
|
SetHrefData();
|
|
}
|
|
|
|
void CLinkPage::SetHrefData()
|
|
{
|
|
if(m_pData->pURL){
|
|
XP_FREE(m_pData->pURL);
|
|
m_pData->pURL = NULL;
|
|
}
|
|
if(!m_csHref.IsEmpty()){
|
|
m_pData->pURL = XP_STRDUP(m_csHref);
|
|
}
|
|
//TODO: Set pTarget and pMocha strings
|
|
#if 0
|
|
// Use this if we supply UI to change target frame
|
|
// For now, just pass through what's there?
|
|
if(m_pData->pTarget){
|
|
XP_FREE(m_pData->pTarget);
|
|
m_pData->pTarget = NULL;
|
|
}
|
|
if(!m_csTarget.IsEmpty()){
|
|
m_pData->pTarget = XP_STRDUP(m_csTarget);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void CLinkPage::OnExtraHTML()
|
|
{
|
|
CExtraHTMLDlg dlg(this, &m_pData->pExtra, IDS_HREF_TAG);
|
|
if( dlg.DoModal() && dlg.m_bDataChanged ){
|
|
SetModified(TRUE);
|
|
}
|
|
}
|
|
|
|
void CLinkPage::OnHelp()
|
|
{
|
|
NetHelp(HELP_PROPS_LINK);
|
|
}
|
|
|
|
void CLinkPage::OnOK()
|
|
{
|
|
CPropertyPage::OnOK();
|
|
|
|
// never visited this page or no change -- don't bother
|
|
if(!m_bActivated ||
|
|
!IS_APPLY_ENABLED(this)){
|
|
return;
|
|
}
|
|
|
|
GetDlgItem(IDC_HREF_URL)->GetWindowText(m_csHref);
|
|
CleanupString(m_csHref);
|
|
|
|
//EDT_BeginBatchChanges(m_pMWContext);
|
|
|
|
CleanupString(m_csAnchorEdit);
|
|
int nResult = 0;
|
|
// TODO: Test nResult for valid URL and add messages to user
|
|
// when there are problems
|
|
|
|
if ( m_iTargetCount ) {
|
|
nTargetIndex = ((CListBox*)GetDlgItem(IDC_TARGET_LIST))->GetCurSel();
|
|
//TODO: Append the "#named_anchor" to the m_csHref;
|
|
}
|
|
|
|
// If we have an image Anchor, the image property page
|
|
// will set HREF data.
|
|
if( ! (m_ppImage && *m_ppImage && **m_ppImage != '\0') ){
|
|
if ( EDT_CanSetHREF(m_pMWContext) ) {
|
|
// Associate a URL with selected text
|
|
// or existing link (text or image anchor)
|
|
// TODO: LLOYD: return error: nResult =
|
|
// EDT_SetHREF( m_pMWContext, szURL );
|
|
EDT_SetHREFData(m_pMWContext, m_pData);
|
|
// Note: This will remove a link if m_pData->pURL is NULL
|
|
}
|
|
else /* if(m_pData->pURL) */{
|
|
// We created a new link -
|
|
// Anchor text should have been typed
|
|
char * szAnchor;
|
|
if ( m_csAnchorEdit.IsEmpty() ) {
|
|
// No anchor text supplied, use URL
|
|
szAnchor = m_pData->pURL;
|
|
} else {
|
|
// Can't do cast of LPCSTR() directly! Gives syntax error!
|
|
const char * szStupidCompiler = LPCSTR(m_csAnchorEdit);
|
|
szAnchor = (char*)szStupidCompiler; // (char*)(LPCSTR(m_csAnchorEdit));
|
|
}
|
|
if( m_bInsert && m_pData->pURL ) {
|
|
// Insert both Anchor text and Href
|
|
// TODO: LLOYD: return error: nResult =
|
|
EDT_PasteHREF( m_pMWContext, &m_pData->pURL, &szAnchor, 1 );
|
|
// Set flag so future Apply actions will not insert another link
|
|
m_bInsert = FALSE;
|
|
// Move back one character so subsequent attributes Applied
|
|
// don't include following text.
|
|
EDT_PreviousChar( m_pMWContext, FALSE );
|
|
m_iCaretMovedBack++;
|
|
|
|
// Get the text just inserted
|
|
char * pNewText = EDT_GetHREFText(m_pMWContext);
|
|
if( pNewText == NULL ){
|
|
// If we insert a space after text, then
|
|
// the link text will be empty.
|
|
// We need to move back another character
|
|
// so current text element is the text, not the space,
|
|
// else subsequent calls to EDT_SetHREFData will
|
|
// attach URL to space and following text, not the
|
|
// newly-inserted text
|
|
EDT_PreviousChar( m_pMWContext, FALSE );
|
|
m_iCaretMovedBack++;
|
|
} else {
|
|
XP_FREE(pNewText);
|
|
}
|
|
|
|
// and set flag so we move forward upon exiting
|
|
|
|
// It would be a pain to allow replacing inserted anchor text,
|
|
// so disable control after 1st insert
|
|
(GetDlgItem(IDC_HREF_UNLINK))->EnableWindow(FALSE);
|
|
} else {
|
|
EDT_SetHREFData(m_pMWContext, m_pData);
|
|
}
|
|
}
|
|
}
|
|
|
|
OkToClose();
|
|
//EDT_EndBatchChanges(m_pMWContext);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
/*
|
|
// **** TEMPLATE for property pages
|
|
/////////////////////////////////////////////////////////////////////
|
|
CPage::CPage(CWnd* pParent, MWContext * pMWContext,
|
|
EDT_<Data> * pData) // EDT data for this property
|
|
: CNetscapePropertyPage(CPage::IDD)
|
|
{
|
|
//{{AFX_DATA_INIT(CPage)
|
|
//}}AFX_DATA_INIT
|
|
ASSERT(pMWContext);
|
|
// ASSERT(pData);
|
|
|
|
m_bActivated = FALSE;
|
|
m_pMWContext = pMWContext;
|
|
EDT_<Data> * m_pData; // EDT data for this property
|
|
}
|
|
|
|
|
|
void CPage::DoDataExchange(CDataExchange* pDX)
|
|
{
|
|
CPropertyPage::DoDataExchange(pDX);
|
|
//{{AFX_DATA_MAP(CPage)
|
|
// NOTE: the ClassWizard will add DDX and DDV calls here
|
|
//}}AFX_DATA_MAP
|
|
}
|
|
|
|
BEGIN_MESSAGE_MAP(CPage, CNetscapePropertyPage)
|
|
//{{AFX_MSG_MAP(CPage)
|
|
// NOTE: the ClassWizard will add message map macros here
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
BOOL CPage::OnSetActive()
|
|
{
|
|
if(!CPropertyPage::OnSetActive())
|
|
return(FALSE);
|
|
|
|
if(m_bActivated)
|
|
return(TRUE);
|
|
|
|
// TODO - Fill in controls here
|
|
m_bActivated = TRUE;
|
|
|
|
// Send data to controls
|
|
UpdateData(FALSE);
|
|
return(TRUE);
|
|
}
|
|
|
|
void CPage::OnOK()
|
|
{
|
|
|
|
CPropertyPage::OnOK();
|
|
|
|
// never visited this page so don't bother
|
|
if(!m_bActivated)
|
|
return;
|
|
|
|
// TODO - Get control data back into pData here
|
|
}
|
|
*/
|
|
#endif // EDITOR
|