mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-07 20:17:37 +00:00
1345 lines
42 KiB
C++
1345 lines
42 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.
|
||
*/
|
||
|
||
|
||
|
||
/*====================================================================================*/
|
||
#pragma mark INCLUDE FILES
|
||
/*====================================================================================*/
|
||
|
||
#include "UMailFolderMenus.h"
|
||
#include "CMailNewsContext.h"
|
||
#include "PascalString.h"
|
||
#include "CMessageFolderView.h"
|
||
#include "CMailFolderButtonPopup.h"
|
||
|
||
#include "StSetBroadcasting.h"
|
||
#include "miconutils.h"
|
||
#include "MercutioAPI.h"
|
||
#include "UMenuUtils.h"
|
||
#include "CMessageFolder.h"
|
||
|
||
#pragma mark -
|
||
/*====================================================================================*/
|
||
#pragma mark TYPEDEFS
|
||
/*====================================================================================*/
|
||
|
||
|
||
#pragma mark -
|
||
/*====================================================================================*/
|
||
#pragma mark CONSTANTS
|
||
/*====================================================================================*/
|
||
|
||
static const UInt16 cFolderPopupFlags = MSG_FOLDER_FLAG_MAIL;
|
||
static const UInt8 cSpaceFolderMenuChar = 0x09;
|
||
static const ResIDT cMenuIconIDDiff = 256;
|
||
static const Int16 cItemIdentWidth = 16;
|
||
|
||
#pragma mark -
|
||
/*====================================================================================*/
|
||
#pragma mark INTERNAL FUNCTION PROTOTYPES
|
||
/*====================================================================================*/
|
||
|
||
|
||
#pragma mark -
|
||
/*====================================================================================*/
|
||
#pragma mark INTERNAL CLASS DECLARATIONS
|
||
/*====================================================================================*/
|
||
|
||
#pragma mark -
|
||
/*====================================================================================*/
|
||
#pragma mark CLASS IMPLEMENTATIONS
|
||
/*====================================================================================*/
|
||
|
||
// Static class members
|
||
|
||
LArray CMailFolderMixin::sMailFolderMixinRegisterList;
|
||
Boolean CMailFolderMixin::sMustRebuild = true;
|
||
void* CMailFolderMixin::sMercutioCallback = nil;
|
||
|
||
//-----------------------------------
|
||
CMailFolderMixin::CMailFolderMixin(Boolean inIncludeFolderIcons)
|
||
: mFolderArray(sizeof(CMessageFolder))
|
||
, mDesiredFolderFlags((FolderChoices)(eWantPOP + eWantIMAP))
|
||
, mInitialMenuCount(0)
|
||
//-----------------------------------
|
||
{
|
||
|
||
mUseFolderIcons = inIncludeFolderIcons;
|
||
RegisterMailFolderMixin(this);
|
||
} // CMailFolderMixin::CMailFolderMixin
|
||
|
||
//-----------------------------------
|
||
CMailFolderMixin::~CMailFolderMixin()
|
||
//-----------------------------------
|
||
{
|
||
UnregisterMailFolderMixin(this);
|
||
// Delete all the CMessageFolder objects, or else we'll leak the cache!
|
||
for (ArrayIndexT i = 1; i <= mFolderArray.GetCount(); i++)
|
||
{
|
||
CMessageFolder* f = (CMessageFolder*)mFolderArray.GetItemPtr(i);
|
||
if (f)
|
||
(*(CMessageFolder*)mFolderArray.GetItemPtr(i)).CMessageFolder::~CMessageFolder();
|
||
}
|
||
} // CMailFolderMixin::~CMailFolderMixin
|
||
|
||
//-----------------------------------
|
||
void CMailFolderMixin::UpdateMailFolderMixins()
|
||
//-----------------------------------
|
||
{
|
||
sMustRebuild = true;
|
||
LCommander::SetUpdateCommandStatus(true);
|
||
} // CMailFolderMixin::UpdateMailFolderMixins
|
||
|
||
//-----------------------------------
|
||
void CMailFolderMixin::UpdateMailFolderMixinsNow(CMailFolderMixin *inSingleMailFolderMixin)
|
||
// Notify all registered CMailFolderMixin objects that the mail folder list has changed.
|
||
// This method should be called whenever the list changes, fo example, when the user
|
||
// adds, moves, or deletes mail folders in the Folder View window.
|
||
//
|
||
// If inSingleMailFolderMixin is not nil, only that specific CMailFolderMixin is
|
||
// notified of the change, NOT all CMailFolderMixin objects.
|
||
//
|
||
// IMPORTANT NOTE: CMailFolderMixin::UpdateMailFolderMixinsNow(this) needs to be called
|
||
// for all subclasses after initialization is complete.
|
||
//-----------------------------------
|
||
{
|
||
|
||
if (!sMustRebuild && inSingleMailFolderMixin == nil)
|
||
return;
|
||
if (sMailFolderMixinRegisterList.GetCount() == 0 && inSingleMailFolderMixin == nil)
|
||
return; // No reason to do anything!
|
||
|
||
MSG_Master *master = CMailNewsContext::GetMailMaster();
|
||
FailNIL_(master);
|
||
|
||
// Get a list and count of all current mail folders. You can only ask for one flag
|
||
// at a time (ALL bits must match for positive test in this call).
|
||
Int32 numFoldersMail = ::MSG_GetFoldersWithFlag(master, MSG_FOLDER_FLAG_MAIL, nil, 0);
|
||
Assert_(numFoldersMail > 0); // Should have at least some permanent mail folders!
|
||
Int32 numNewsgroups = 0;
|
||
Int32 numNewsHosts = 0;
|
||
if (
|
||
!inSingleMailFolderMixin
|
||
|| (inSingleMailFolderMixin->mDesiredFolderFlags & (eWantNews|eWantHosts))
|
||
== (eWantNews|eWantHosts)
|
||
)
|
||
numNewsHosts = ::MSG_GetFoldersWithFlag(master, MSG_FOLDER_FLAG_NEWS_HOST, nil, 0);
|
||
if (!inSingleMailFolderMixin
|
||
|| (inSingleMailFolderMixin->mDesiredFolderFlags & eWantNews) != 0)
|
||
numNewsgroups = ::MSG_GetFoldersWithFlag(master, MSG_FOLDER_FLAG_NEWSGROUP|MSG_FOLDER_FLAG_SUBSCRIBED, nil, 0);
|
||
|
||
Int32 numFolders = numFoldersMail + numNewsgroups + numNewsHosts;
|
||
StPointerBlock folderInfoData(sizeof(MSG_FolderInfo *) * numFolders);
|
||
MSG_FolderInfo **folderInfo = (MSG_FolderInfo **) folderInfoData.mPtr;
|
||
|
||
Int32 numFolders2 = ::MSG_GetFoldersWithFlag(master, MSG_FOLDER_FLAG_MAIL, folderInfo, numFoldersMail);
|
||
Assert_(numFolders2 > 0); // Should have at least some permanent mail folders!
|
||
Assert_(numFolders2 == numFoldersMail);
|
||
|
||
// Handle the news hosts and groups, filling in the remaining pointers in the array.
|
||
MSG_FolderInfo** currentInfo = folderInfo + numFoldersMail;
|
||
if (numNewsHosts && numNewsgroups)
|
||
{
|
||
// The difficult case.
|
||
// Get the list of hosts and the list of groups into separate lists.
|
||
StPointerBlock newsHostInfoData(sizeof(MSG_FolderInfo *) * numNewsHosts);
|
||
MSG_FolderInfo **newsHostInfo = (MSG_FolderInfo **) newsHostInfoData.mPtr;
|
||
numFolders2 = ::MSG_GetFoldersWithFlag(
|
||
master,
|
||
MSG_FOLDER_FLAG_NEWS_HOST,
|
||
newsHostInfo,
|
||
numNewsHosts);
|
||
Assert_(numFolders2 == numNewsHosts);
|
||
|
||
StPointerBlock newsgroupInfoData(sizeof(MSG_FolderInfo *) * numNewsgroups);
|
||
MSG_FolderInfo **newsgroupInfo = (MSG_FolderInfo **) newsgroupInfoData.mPtr;
|
||
numFolders2 = ::MSG_GetFoldersWithFlag(
|
||
master,
|
||
MSG_FOLDER_FLAG_NEWSGROUP|MSG_FOLDER_FLAG_SUBSCRIBED,
|
||
newsgroupInfo,
|
||
numNewsgroups);
|
||
Assert_(numFolders2 == numNewsgroups);
|
||
|
||
// Loop over the hosts, copying one host, then all its groups
|
||
if (numNewsHosts)
|
||
{
|
||
for (int i = 0; i < numNewsHosts; i++)
|
||
{
|
||
// Copy over all the news groups for this host
|
||
*currentInfo++ = newsHostInfo[i];
|
||
for (int j = 0; j < numNewsgroups; j++)
|
||
{
|
||
// Are the two hosts the same?
|
||
if (::MSG_GetHostForFolder(newsgroupInfo[j])
|
||
== ::MSG_GetHostForFolder(newsHostInfo[i]))
|
||
{
|
||
*currentInfo++ = newsgroupInfo[j];
|
||
}
|
||
} // inner loop
|
||
} // outer loop
|
||
}
|
||
}
|
||
else if (numNewsgroups)
|
||
{
|
||
numFolders2 = ::MSG_GetFoldersWithFlag(
|
||
master,
|
||
MSG_FOLDER_FLAG_NEWSGROUP|MSG_FOLDER_FLAG_SUBSCRIBED,
|
||
currentInfo,
|
||
numNewsgroups);
|
||
}
|
||
else if (numNewsHosts)
|
||
{
|
||
numFolders2 = ::MSG_GetFoldersWithFlag(
|
||
master,
|
||
MSG_FOLDER_FLAG_NEWS_HOST,
|
||
currentInfo,
|
||
numNewsHosts);
|
||
}
|
||
// Notify all registered CMailFolderMixin objects, or only the specified object
|
||
if ( inSingleMailFolderMixin == nil )
|
||
{
|
||
LArrayIterator iterator(sMailFolderMixinRegisterList);
|
||
while ( iterator.Next(&inSingleMailFolderMixin) )
|
||
inSingleMailFolderMixin->MPopulateWithFolders(folderInfo, numFolders);
|
||
sMustRebuild = false;
|
||
}
|
||
else
|
||
inSingleMailFolderMixin->MPopulateWithFolders(folderInfo, numFolders);
|
||
} // CMailFolderMixin::UpdateMailFolderMixinsNow
|
||
|
||
//-----------------------------------
|
||
Boolean CMailFolderMixin::MSetSelectedFolderName(
|
||
const char *inName,
|
||
Boolean inDoBroadcast)
|
||
// Set the currently selected item in the menu to the item represented by inName. If
|
||
// inName is nil, empty, or cannot be found, the value of the menu is set to 0.
|
||
// Return true if the specified folder name was found in the menu.
|
||
//-----------------------------------
|
||
{
|
||
const ArrayIndexT numItems = mFolderArray.GetCount();
|
||
ArrayIndexT curIndex;
|
||
if ( !inName || !*inName)
|
||
curIndex = numItems + 1;
|
||
else
|
||
{
|
||
for (curIndex = 1; curIndex <= numItems; ++curIndex)
|
||
{
|
||
const char* itemPath = MGetFolderName(curIndex);
|
||
if ( ::strcasecomp(inName, itemPath) == 0 )
|
||
break;
|
||
}
|
||
}
|
||
|
||
if ( curIndex <= numItems )
|
||
{
|
||
MSetSelectedMenuItem(curIndex, inDoBroadcast);
|
||
return true;
|
||
}
|
||
else
|
||
{
|
||
MSetSelectedMenuItem(0, inDoBroadcast);
|
||
return false;
|
||
}
|
||
}
|
||
|
||
//-----------------------------------
|
||
Boolean CMailFolderMixin::MSetSelectedFolder(
|
||
const CMessageFolder& inFolder,
|
||
Boolean /*inDoBroadcast*/)
|
||
//-----------------------------------
|
||
{
|
||
return MSetSelectedFolder(inFolder.GetFolderInfo());
|
||
}
|
||
|
||
//-----------------------------------
|
||
Boolean CMailFolderMixin::MSetSelectedFolder(
|
||
const MSG_FolderInfo* inInfo,
|
||
Boolean inDoBroadcast)
|
||
//-----------------------------------
|
||
{
|
||
const ArrayIndexT numItems = mFolderArray.GetCount();
|
||
Assert_(mFolderArray.GetCount() == (MGetNumMenuItems() - mInitialMenuCount));
|
||
// Not synced otherwise
|
||
ArrayIndexT curIndex;
|
||
if (inInfo == nil)
|
||
curIndex = numItems + 1;
|
||
else
|
||
{
|
||
mFolderArray.Lock();
|
||
for (curIndex = 1; curIndex <= numItems; ++curIndex)
|
||
{
|
||
CMessageFolder* curFolder = (CMessageFolder*)mFolderArray.GetItemPtr(curIndex);
|
||
if (inInfo == curFolder->GetFolderInfo())
|
||
break;
|
||
}
|
||
mFolderArray.Unlock();
|
||
}
|
||
|
||
if ( curIndex <= numItems )
|
||
{
|
||
MSetSelectedMenuItem(curIndex, inDoBroadcast);
|
||
return true;
|
||
}
|
||
else
|
||
{
|
||
MSetSelectedMenuItem(0, inDoBroadcast);
|
||
return false;
|
||
}
|
||
}
|
||
|
||
//-----------------------------------
|
||
const char* CMailFolderMixin::MGetFolderName(ArrayIndexT inItemNumber)
|
||
// Get the currently selected folder name in the menu. Return a reference to head of
|
||
// the returned c-string.
|
||
//-----------------------------------
|
||
{
|
||
CMessageFolder folder = MGetFolder(inItemNumber);
|
||
MSG_FolderInfo* info = folder.GetFolderInfo();
|
||
if (info)
|
||
{
|
||
const char * str = ::MSG_GetFolderNameFromID(info);
|
||
|
||
// Make sure that the name is fully escaped
|
||
static char escapedName[256];
|
||
XP_STRCPY(escapedName, str);
|
||
NET_UnEscape(escapedName);
|
||
char * temp = NET_Escape(escapedName, URL_PATH);
|
||
if (temp)
|
||
{
|
||
XP_STRCPY(escapedName, temp);
|
||
XP_FREE(temp);
|
||
}
|
||
return escapedName;
|
||
}
|
||
return nil;
|
||
}
|
||
|
||
|
||
//-----------------------------------
|
||
const char* CMailFolderMixin::MGetSelectedFolderName()
|
||
// Get the currently selected folder name in the menu. Return a reference to head of
|
||
// the returned c-string.
|
||
//-----------------------------------
|
||
{
|
||
ArrayIndexT index = MGetSelectedMenuItem();
|
||
if (index)
|
||
return MGetFolderName(index);
|
||
return nil;
|
||
}
|
||
|
||
//-----------------------------------
|
||
CMessageFolder CMailFolderMixin::MGetFolder(ArrayIndexT inItemNumber)
|
||
//-----------------------------------
|
||
{
|
||
CMessageFolder result(nil);
|
||
Assert_(mFolderArray.GetCount() == (MGetNumMenuItems() - mInitialMenuCount)); // Not synced otherwise
|
||
if ( (mFolderArray.GetCount() > 0) && (inItemNumber > 0) )
|
||
{
|
||
if ( mFolderArray.ValidIndex(inItemNumber) )
|
||
{
|
||
mFolderArray.Lock();
|
||
result = *(CMessageFolder*) mFolderArray.GetItemPtr(inItemNumber);
|
||
mFolderArray.Unlock();
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
//-----------------------------------
|
||
CMessageFolder CMailFolderMixin::MGetFolderFromMenuItem(ArrayIndexT inItemNumber)
|
||
//-----------------------------------
|
||
{
|
||
// convert the menu item number in to an index into the folder array
|
||
CMessageFolder result(nil);
|
||
if (inItemNumber - mInitialMenuCount > 0)
|
||
{ // If (inItemNumber - mInitialMenuCount) is 0 or less then we are trying to
|
||
// find folder info on a previously existing menu item.
|
||
result = MGetFolder(inItemNumber - mInitialMenuCount);
|
||
}
|
||
return result;
|
||
}
|
||
|
||
|
||
//-----------------------------------
|
||
CMessageFolder CMailFolderMixin::MGetSelectedFolder()
|
||
//-----------------------------------
|
||
{
|
||
return MGetFolder(MGetSelectedMenuItem());
|
||
}
|
||
|
||
//-----------------------------------
|
||
ResIDT CMailFolderMixin::GetMailFolderIconID(UInt16 inFlags)
|
||
//-----------------------------------
|
||
{
|
||
|
||
if ((inFlags & MSG_FOLDER_FLAG_MAIL) != 0)
|
||
return 0;
|
||
UInt16 folderType = inFlags & (
|
||
MSG_FOLDER_FLAG_TRASH | MSG_FOLDER_FLAG_SENTMAIL |
|
||
MSG_FOLDER_FLAG_DRAFTS | MSG_FOLDER_FLAG_QUEUE |
|
||
MSG_FOLDER_FLAG_INBOX
|
||
);
|
||
switch ( folderType )
|
||
{
|
||
case MSG_FOLDER_FLAG_TRASH: return 286;
|
||
case MSG_FOLDER_FLAG_SENTMAIL: return 284;
|
||
case MSG_FOLDER_FLAG_DRAFTS: return 285;
|
||
case MSG_FOLDER_FLAG_QUEUE: return 283;
|
||
case MSG_FOLDER_FLAG_INBOX: return 282;
|
||
default:
|
||
if ( inFlags & MSG_FOLDER_FLAG_IMAPBOX )
|
||
return 287; // Online folder?
|
||
return 281;
|
||
}
|
||
}
|
||
|
||
/*======================================================================================
|
||
Register the specified inMailFolderMixin so that it is automatically notified
|
||
when changes are made to the mail folder list. The specified inMailFolderMixin
|
||
is only added to the list if it is not already there.
|
||
======================================================================================*/
|
||
|
||
void CMailFolderMixin::RegisterMailFolderMixin(CMailFolderMixin *inMailFolderMixin) {
|
||
|
||
Assert_(inMailFolderMixin != nil);
|
||
|
||
if ( sMailFolderMixinRegisterList.FetchIndexOf(&inMailFolderMixin) == LArray::index_Bad ) {
|
||
sMailFolderMixinRegisterList.InsertItemsAt(1, LArray::index_Last, &inMailFolderMixin);
|
||
}
|
||
}
|
||
|
||
|
||
|
||
/*======================================================================================
|
||
Unregister the specified inMailFolderMixin so that it is no longer notified
|
||
when changes are made to the mail folder list. If the specified inMailFolderMixin
|
||
is not in the list, this method does nothing.
|
||
======================================================================================*/
|
||
|
||
void CMailFolderMixin::UnregisterMailFolderMixin(CMailFolderMixin *inMailFolderMixin) {
|
||
|
||
sMailFolderMixinRegisterList.Remove(&inMailFolderMixin);
|
||
}
|
||
|
||
//-----------------------------------
|
||
void CMailFolderMixin::MGetCurrentMenuItemName(Str255 outItemName)
|
||
// Get the name of the menu item that is currently selected, minus any indentation.
|
||
//-----------------------------------
|
||
{
|
||
outItemName[0] = 0;
|
||
MGetMenuItemName(MGetSelectedMenuItem(), outItemName);
|
||
if ( outItemName[0] != 0 )
|
||
{
|
||
UInt8 *curChar = &outItemName[1], *endChar = curChar + outItemName[0];
|
||
do
|
||
{
|
||
if ( *curChar != cSpaceFolderMenuChar ) break;
|
||
} while ( ++curChar < endChar );
|
||
|
||
Int8 numLeading = curChar - (&outItemName[1]);
|
||
|
||
if ( numLeading != 0 )
|
||
{
|
||
outItemName[0] -= numLeading;
|
||
if ( outItemName[0] != 0 )
|
||
{
|
||
::BlockMoveData(&outItemName[numLeading + 1], &outItemName[1], outItemName[0]);
|
||
}
|
||
}
|
||
}
|
||
} // CMailFolderMixin::MGetCurrentMenuItemName
|
||
|
||
//-----------------------------------
|
||
Boolean CMailFolderMixin::FolderFilter(const CMessageFolder& folder)
|
||
// Return true iff you want this folder included in the list. This function should
|
||
// handle all known cases, but we may want to make this virtual and override it in the
|
||
// future.
|
||
//-----------------------------------
|
||
{
|
||
Boolean result;
|
||
|
||
if (folder.GetLevel() > 1)
|
||
{
|
||
UInt32 folderFlags = folder.GetFolderFlags();
|
||
Boolean isPOP, isIMAP, isNews, isInbox, isPublicFolder;
|
||
|
||
isNews = folderFlags & MSG_FOLDER_FLAG_NEWSGROUP ? true: false;
|
||
isIMAP = folderFlags & MSG_FOLDER_FLAG_IMAPBOX ? true: false;
|
||
isPOP = (!isIMAP && (folderFlags & MSG_FOLDER_FLAG_MAIL)) ? true: false;
|
||
isInbox = folderFlags& MSG_FOLDER_FLAG_INBOX ? true : false;
|
||
isPublicFolder = ( isIMAP&& ( folderFlags& MSG_GROUPNAME_FLAG_IMAP_PUBLIC ) ) ? true : false;
|
||
|
||
if (isPOP && (mDesiredFolderFlags & eWantPOP))
|
||
{
|
||
result = true;
|
||
}
|
||
else if (isIMAP && (mDesiredFolderFlags & eWantIMAP))
|
||
{
|
||
result = true;
|
||
}
|
||
else if (isNews && (mDesiredFolderFlags & eWantNews))
|
||
{
|
||
result = true;
|
||
}
|
||
else if ( isInbox && (mDesiredFolderFlags & eWantInbox) )
|
||
{
|
||
result = true;
|
||
}
|
||
else if ( isPublicFolder && (mDesiredFolderFlags & eWantPublicFolder) )
|
||
{
|
||
result = true;
|
||
}
|
||
else
|
||
{
|
||
result = false;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
// level <= 1, we are filtering a host/ server.
|
||
result = (mDesiredFolderFlags & eWantHosts) != 0;
|
||
}
|
||
|
||
return result;
|
||
} // CMailFolderMixin::FolderFilter
|
||
|
||
//-----------------------------------
|
||
pascal void MyMercutioCallback(
|
||
Int16 menuID,
|
||
Int16 /*previousModifiers*/,
|
||
RichItemDataYadaYada& inItemData)
|
||
//-----------------------------------
|
||
{
|
||
// This function is called continuously when the user clicks the
|
||
// folder popup in a Thread window. Static variables have been added
|
||
// for performance reasons.
|
||
|
||
RichItemData& itemData = (RichItemData&)inItemData;
|
||
itemData.flags &= ~kChangedByCallback; // clear bit
|
||
|
||
// Find the mixin object, if any, belonging to this menu
|
||
CMailFolderMixin* mixin = nil;
|
||
Boolean found = false;
|
||
|
||
static Int16 prevMenuID = 0;
|
||
static MenuRef prevMenuH = nil;
|
||
static UInt32 prevTicks = 0;
|
||
MenuRef menuh;
|
||
|
||
Boolean sameMenuAsBefore = (menuID == prevMenuID
|
||
&& prevMenuH != nil
|
||
&& (::TickCount() - prevTicks < 30));
|
||
// Don't reuse the same menu after a short delay because we can't tell
|
||
// whether the user is still clicking the same popup as before: he may
|
||
// have selected another window since the last time (and unfortunately,
|
||
// all the folder popups in the different windows have the same ID).
|
||
prevTicks = ::TickCount();
|
||
|
||
if (sameMenuAsBefore)
|
||
menuh = prevMenuH;
|
||
else
|
||
{
|
||
menuh = ::GetMenuHandle(menuID);
|
||
prevMenuH = menuh;
|
||
prevMenuID = menuID;
|
||
}
|
||
|
||
if (menuh)
|
||
{
|
||
LArrayIterator iterator(CMailFolderMixin::sMailFolderMixinRegisterList);
|
||
while (iterator.Next(&mixin))
|
||
{
|
||
if (mixin->MGetSystemMenuHandle() == menuh)
|
||
{
|
||
found = true;
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
if (!found || !mixin || !mixin->mUseFolderIcons)
|
||
return;
|
||
|
||
static ArrayIndexT prevItemID = 0;
|
||
static CMessageFolder prevFolder(nil);
|
||
CMessageFolder currentFolder(nil);
|
||
if (sameMenuAsBefore && itemData.itemID == prevItemID && prevFolder.GetFolderInfo() != nil)
|
||
currentFolder = prevFolder;
|
||
else
|
||
{
|
||
currentFolder = mixin->MGetFolderFromMenuItem(itemData.itemID);
|
||
prevFolder = currentFolder;
|
||
prevItemID = itemData.itemID;
|
||
}
|
||
if (currentFolder.GetFolderInfo() == NULL)
|
||
return;
|
||
|
||
switch (itemData.cbMsg)
|
||
{
|
||
case cbBasicDataOnlyMsg:
|
||
itemData.flags |=
|
||
(ksameAlternateAsLastTime|kIconIsSmall|kHasIcon/*|kChangedByCallback*/|kdontDisposeIcon);
|
||
itemData.flags &= ~(kIsDynamic);
|
||
|
||
return;
|
||
|
||
case cbIconOnlyMsg:
|
||
break; // look out below!
|
||
|
||
default:
|
||
return;
|
||
}
|
||
|
||
// OK, calculate the icon and check whether it's changed.
|
||
|
||
static ResIDT prevIconID = 0;
|
||
static Handle prevIconSuite = nil;
|
||
ResIDT iconID = currentFolder.GetIconID();
|
||
Handle newIconSuite;
|
||
if (iconID == prevIconID && prevIconSuite != nil)
|
||
newIconSuite = prevIconSuite;
|
||
else
|
||
{
|
||
newIconSuite = CIconList::GetIconSuite(iconID);
|
||
prevIconSuite = newIconSuite;
|
||
prevIconID = iconID;
|
||
}
|
||
if (newIconSuite && newIconSuite != itemData.hIcon)
|
||
{
|
||
// It's changed. Modify the data record for Mercutio.
|
||
itemData.hIcon = newIconSuite;
|
||
itemData.iconType = 'suit';
|
||
if (::LMGetSysFontFam() != systemFont) // no bold for Chicago - it sucketh
|
||
{
|
||
if (currentFolder.CountUnseen() > 0 )
|
||
itemData.textStyle.s |= bold;
|
||
else
|
||
itemData.textStyle.s &= ~bold;
|
||
}
|
||
// itemData.flags |= kChangedByCallback; // set bit
|
||
|
||
// indent the item by shifting the left side
|
||
const Int16 baseLevel
|
||
= (mixin->mDesiredFolderFlags & CMailFolderMixin::eWantHosts)
|
||
? kRootLevel : kRootLevel + 1;
|
||
itemData.itemRect.left
|
||
= itemData.itemRect.right
|
||
- (*menuh)->menuWidth
|
||
+ ((currentFolder.GetLevel() - baseLevel) * cItemIdentWidth);
|
||
}
|
||
} // MyMercutioCallback
|
||
|
||
//-----------------------------------
|
||
void CMailFolderMixin::GetFolderNameForDisplay(char * inName, CMessageFolder& folder)
|
||
//-----------------------------------
|
||
{
|
||
const char* geekName = (char*)folder.GetName();
|
||
const char* prettyName = (char*)folder.GetPrettyName();
|
||
Boolean isNewsFolder = ((folder.GetFolderType() & (MSG_FOLDER_FLAG_NEWSGROUP | MSG_FOLDER_FLAG_NEWS_HOST)) != 0);
|
||
|
||
if (isNewsFolder && prettyName != nil && *prettyName != '\0')
|
||
{
|
||
#if 0
|
||
// sprintf(inName, "%s (%s)", geekName, prettyName);
|
||
Int16 strLen = ::strlen(geekName);
|
||
::strcpy(inName, geekName);
|
||
inName[strLen ++] = ' ';
|
||
inName[strLen ++] = '(';
|
||
::strcpy(&inName[strLen], prettyName);
|
||
strLen += ::strlen(prettyName);
|
||
inName[strLen ++] = ')';
|
||
inName[strLen ++] = '\0';
|
||
#endif
|
||
::strcpy(inName, prettyName);
|
||
}
|
||
else
|
||
::strcpy(inName, geekName);
|
||
} // GetFolderNameForDisplay
|
||
|
||
inline void *operator new(size_t,void *p) { return p; }
|
||
|
||
//-----------------------------------
|
||
Boolean CMailFolderMixin::MPopulateWithFolders(const MSG_FolderInfo **inFolderInfoArray,
|
||
Int32 inNumFolders)
|
||
// Populate the folder menu popup with the specified list of mail folders. This method
|
||
// can be called at anytime after object creation; it does the right thing.
|
||
//
|
||
// The inFolderInfoArray contains a list of (MSG_FolderInfo *) pointer representing
|
||
// the current list of mail folders. inNumFolders is the number of items in the
|
||
// array, 0 if none.
|
||
//
|
||
// This method tries to keep the currently selected folder, but if changed, sets the
|
||
// menu value to 0.
|
||
//-----------------------------------
|
||
{
|
||
MSG_Master *master = CMailNewsContext::GetMailMaster();
|
||
FailNIL_(master);
|
||
|
||
MenuRef menuh = MGetSystemMenuHandle();
|
||
|
||
const Int16 startNumItems = MGetNumMenuItems() - mInitialMenuCount;
|
||
Assert_(mFolderArray.GetCount() == startNumItems); // Should be equal
|
||
|
||
Boolean didChange = false;
|
||
|
||
Int16 addedItems = 0;
|
||
|
||
// Store the currently selected folder
|
||
|
||
CMessageFolder selectedFolder = MGetSelectedFolder();
|
||
|
||
// Set menu items
|
||
MSG_FolderInfo** curInfo = (MSG_FolderInfo**)inFolderInfoArray;
|
||
const Int16 baseLevel = (mDesiredFolderFlags & eWantHosts) ? kRootLevel : kRootLevel + 1;
|
||
const Int16 spacesPerFolderLevel = (cItemIdentWidth / ::CharWidth(cSpaceFolderMenuChar)) + 1;
|
||
CMessageFolder folder(nil);
|
||
for (Int16 curItem = 0; curItem < inNumFolders; curItem++, curInfo++)
|
||
{
|
||
CMessageFolder previousFolder = folder;
|
||
folder = CMessageFolder(*curInfo);
|
||
if (FolderFilter(folder))
|
||
{
|
||
|
||
// Insert a gray line in the menu for host-level stuff we're not inserting.
|
||
if (addedItems && previousFolder.GetLevel() == 1 && !(mDesiredFolderFlags & eWantHosts))
|
||
{
|
||
if ( ++addedItems <= startNumItems )
|
||
{
|
||
if ( MUpdateMenuItem(addedItems + mInitialMenuCount, "\p-", /*iconID*/0) )
|
||
didChange = true;
|
||
new (mFolderArray.GetItemPtr(addedItems)) CMessageFolder(nil);
|
||
}
|
||
else
|
||
{
|
||
didChange = true;
|
||
MAppendMenuItem("\p-", /*iconID*/0);
|
||
mFolderArray.InsertItemsAt(1, addedItems, &folder);
|
||
// Now we have to store it again, this time properly, using the copy
|
||
// constructor. Otherwise, the refcount of CCachedFolderLine is not incremented
|
||
// because the previous line just does a bitwise copy.
|
||
new (mFolderArray.GetItemPtr(addedItems)) CMessageFolder(nil);
|
||
}
|
||
// Ask for a callback so we can set the icon
|
||
if (mUseFolderIcons && MDEF_IsCustomMenu(menuh))
|
||
::SetItemStyle(menuh, addedItems + mInitialMenuCount, condense);
|
||
}
|
||
|
||
// add spaces at the end of the name to make the menu larger
|
||
// and reserve some room for the identation
|
||
char tempString[cMaxMailFolderNameLength + 1];
|
||
GetFolderNameForDisplay(tempString, folder);
|
||
|
||
Int16 nameLen = ::strlen(tempString);
|
||
Int16 numSpaces = ((folder.GetLevel() - baseLevel) * spacesPerFolderLevel);
|
||
if ( numSpaces > (sizeof(tempString) - nameLen) )
|
||
numSpaces = sizeof(tempString) - nameLen;
|
||
if ( numSpaces > 0 )
|
||
::memset(tempString + nameLen, cSpaceFolderMenuChar, numSpaces);
|
||
tempString[nameLen + numSpaces] = '\0';
|
||
NET_UnEscape(tempString);
|
||
|
||
CStr255 menuString(tempString);
|
||
|
||
// Update or append the current menu item
|
||
if ( ++addedItems <= startNumItems )
|
||
{
|
||
if ( MUpdateMenuItem(addedItems + mInitialMenuCount, menuString, /*iconID*/0) )
|
||
didChange = true;
|
||
(*(CMessageFolder*)mFolderArray.GetItemPtr(addedItems)).CMessageFolder::~CMessageFolder();
|
||
new (mFolderArray.GetItemPtr(addedItems)) CMessageFolder(folder);
|
||
}
|
||
else
|
||
{
|
||
didChange = true;
|
||
MAppendMenuItem(menuString, /*iconID*/0);
|
||
mFolderArray.InsertItemsAt(1, addedItems, &folder);
|
||
// Now we have to store it again, this time properly, using the copy
|
||
// constructor. Otherwise, the refcount of CCachedFolderLine is not incremented
|
||
// because the previous line just does a bitwise copy.
|
||
new (mFolderArray.GetItemPtr(addedItems)) CMessageFolder(folder);
|
||
}
|
||
// Ask for a callback so we can set the icon
|
||
if (mUseFolderIcons && MDEF_IsCustomMenu(menuh))
|
||
::SetItemStyle(menuh, addedItems + mInitialMenuCount, condense);
|
||
} // if (FolderFilter(folder))
|
||
} // for
|
||
|
||
// Get rid of any remaining menu items
|
||
if (addedItems < startNumItems)
|
||
{
|
||
didChange = true;
|
||
MClipMenuItems(addedItems + 1 + mInitialMenuCount);
|
||
for (ArrayIndexT i = addedItems + 1; i <= mFolderArray.GetCount(); i++)
|
||
{
|
||
CMessageFolder* f = (CMessageFolder*)mFolderArray.GetItemPtr(i);
|
||
if (f)
|
||
(*(CMessageFolder*)mFolderArray.GetItemPtr(i)).CMessageFolder::~CMessageFolder();
|
||
}
|
||
mFolderArray.RemoveItemsAt(LArray::index_Last, addedItems + 1);
|
||
}
|
||
|
||
// Set the originally selected item (try to keep original folder),
|
||
// or 0 if the folder was deleted
|
||
if (didChange && inNumFolders > 0 )
|
||
{
|
||
// Don't broadcast anything here
|
||
if ( !MSetSelectedFolder(selectedFolder, false) )
|
||
MSetSelectedMenuItem(0, false);
|
||
}
|
||
|
||
// Set up for callback so we can set the icon
|
||
if (mUseFolderIcons && MDEF_IsCustomMenu(menuh))
|
||
{
|
||
if (!sMercutioCallback)
|
||
sMercutioCallback = NewMercutioCallback(MyMercutioCallback);
|
||
FailNIL_(sMercutioCallback);
|
||
MDEF_SetCallbackProc(menuh, (MercutioCallbackUPP)sMercutioCallback);
|
||
MenuPrefsRec myPrefs;
|
||
::memset(&myPrefs, 0, sizeof(myPrefs));
|
||
myPrefs.useCallbackFlag.s = condense;
|
||
MDEF_SetMenuPrefs(menuh, &myPrefs);
|
||
}
|
||
|
||
// Refresh on change
|
||
if (didChange)
|
||
MRefreshMenu();
|
||
return didChange;
|
||
} // MPopulateWithFolders
|
||
|
||
|
||
#pragma mark -
|
||
|
||
/*======================================================================================
|
||
Get the number of menu items.
|
||
======================================================================================*/
|
||
|
||
Int16 CMailFolderPopupMixin::MGetNumMenuItems(void)
|
||
{
|
||
return MGetControl()->GetMaxValue();
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Get the selected menu item.
|
||
======================================================================================*/
|
||
|
||
Int16 CMailFolderPopupMixin::MGetSelectedMenuItem(void)
|
||
{
|
||
return MGetControl()->GetValue();
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Set the selected menu item. Refresh the menu if the selection changes.
|
||
======================================================================================*/
|
||
|
||
void CMailFolderPopupMixin::MSetSelectedMenuItem(Int16 inIndex, Boolean inDoBroadcast)
|
||
{
|
||
|
||
StSetBroadcasting setBroadcasting(
|
||
MGetControl(),
|
||
inDoBroadcast && MGetControl()->IsBroadcasting()
|
||
);
|
||
MGetControl()->SetValue(inIndex);
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Update the current menu item. If any of the input parameters are different from
|
||
the current menu item parameters, return true and reset the menu item; otherwise,
|
||
return false and don't modify the menu. If inIconID is non-zero, set the icon for
|
||
the menu item to that resource. Don't refresh the menu display in anyway.
|
||
======================================================================================*/
|
||
|
||
Boolean CMailFolderPopupMixin::MUpdateMenuItem(Int16 inMenuItem, ConstStr255Param inName,
|
||
ResIDT inIconID)
|
||
{
|
||
LControl *theControl = MGetControl();
|
||
// StEmptyVisRgn emptyRgn(theControl->GetMacPort()); // Make changes invisible
|
||
|
||
Boolean changed = false;
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Assert_(menuH != nil);
|
||
Assert_((inIconID > cMenuIconIDDiff) || (inIconID == 0));
|
||
|
||
Assert_(::CountMItems(menuH) == MGetNumMenuItems());
|
||
Assert_((inMenuItem > 0) && (inMenuItem <= MGetNumMenuItems()));
|
||
|
||
Str255 currentText;
|
||
::GetMenuItemText(menuH, inMenuItem, currentText);
|
||
|
||
changed = !::EqualString(currentText, inName, true, true);
|
||
if ( changed ) ::SetMenuItemText(menuH, inMenuItem, inName);
|
||
|
||
if ( inIconID ) inIconID -= cMenuIconIDDiff;
|
||
Int16 iconID;
|
||
::GetItemIcon(menuH, inMenuItem, &iconID);
|
||
if ( iconID != inIconID ) {
|
||
changed = true;
|
||
::SetItemIcon(menuH, inMenuItem, inIconID);
|
||
}
|
||
|
||
return changed;
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Append the specified menu item to the end of the menu. If inIconID is non-zero, set
|
||
the icon for the menu item to that resource. Don't refresh the menu display in
|
||
anyway.
|
||
======================================================================================*/
|
||
|
||
void CMailFolderPopupMixin::MAppendMenuItem(ConstStr255Param inName, ResIDT inIconID) {
|
||
|
||
LControl *theControl = MGetControl();
|
||
// StEmptyVisRgn emptyRgn(theControl->GetMacPort()); // Make changes invisible
|
||
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Assert_(menuH != nil);
|
||
Assert_((inIconID > cMenuIconIDDiff) || (inIconID == 0));
|
||
Int16 numMenuItems = MGetNumMenuItems() + 1;
|
||
|
||
Assert_(::CountMItems(menuH) == MGetNumMenuItems());
|
||
|
||
UMenuUtils::AppendMenuItem(menuH, inName);
|
||
|
||
if ( inIconID ) {
|
||
::SetItemIcon(menuH, numMenuItems, inIconID - cMenuIconIDDiff);
|
||
}
|
||
theControl->SetMaxValue(numMenuItems);
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Remove menu items from the end of the current menu, beginning with the item at
|
||
inStartClipItem. Don't refresh the menu display in anyway.
|
||
======================================================================================*/
|
||
|
||
void CMailFolderPopupMixin::MClipMenuItems(Int16 inStartClipItem) {
|
||
|
||
LControl *theControl = MGetControl();
|
||
// StEmptyVisRgn emptyRgn(theControl->GetMacPort()); // Make changes invisible
|
||
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Assert_(menuH != nil);
|
||
Int16 numMenuItems = MGetNumMenuItems();
|
||
|
||
Assert_(::CountMItems(menuH) == numMenuItems);
|
||
Assert_((inStartClipItem > 0) && (inStartClipItem <= numMenuItems));
|
||
|
||
while ( numMenuItems >= inStartClipItem ) {
|
||
::DeleteMenuItem(menuH, numMenuItems--);
|
||
}
|
||
inStartClipItem -= 1;
|
||
|
||
if ( MGetSelectedMenuItem() > inStartClipItem ) theControl->SetValue(inStartClipItem);
|
||
theControl->SetMaxValue(inStartClipItem);
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Get the current menu text for the specified item.
|
||
======================================================================================*/
|
||
|
||
void CMailFolderPopupMixin::MGetMenuItemName(Int16 inMenuItem, Str255 outItemName) {
|
||
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Assert_(menuH != nil);
|
||
Assert_(::CountMItems(menuH) == MGetNumMenuItems());
|
||
Assert_((inMenuItem > 0) && (inMenuItem <= MGetNumMenuItems()));
|
||
|
||
::GetMenuItemText(menuH, inMenuItem, outItemName);
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Refresh the menu display.
|
||
======================================================================================*/
|
||
|
||
void CMailFolderPopupMixin::MRefreshMenu(void) {
|
||
|
||
MGetControl()->Refresh();
|
||
}
|
||
|
||
|
||
#pragma mark -
|
||
|
||
/*======================================================================================
|
||
Get a handle to the Mac menu associated with this object.
|
||
======================================================================================*/
|
||
|
||
MenuHandle CGAIconPopupFolderMixin::MGetSystemMenuHandle(void) {
|
||
|
||
return MGetControl()->GetMacMenuH();
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Refresh the menu display.
|
||
======================================================================================*/
|
||
|
||
void CGAIconPopupFolderMixin::MRefreshMenu(void) {
|
||
|
||
MGetControl()->RefreshMenu();
|
||
}
|
||
|
||
|
||
#pragma mark -
|
||
|
||
/*======================================================================================
|
||
Get a handle to the Mac menu associated with this object.
|
||
======================================================================================*/
|
||
|
||
MenuHandle CGAPopupFolderMixin::MGetSystemMenuHandle(void) {
|
||
|
||
return MGetControl()->GetMacMenuH();
|
||
}
|
||
|
||
|
||
#pragma mark -
|
||
|
||
/*======================================================================================
|
||
Get a handle to the Mac menu associated with this object.
|
||
======================================================================================*/
|
||
|
||
MenuHandle CGAIconButtonPopupFolderMixin::MGetSystemMenuHandle(void) {
|
||
|
||
return MGetControl()->GetMacMenuH();
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Refresh the menu display.
|
||
======================================================================================*/
|
||
|
||
void CGAIconButtonPopupFolderMixin::MRefreshMenu(void) {
|
||
|
||
// Don't need to do anything since the menu is not visible unless the user has clicked
|
||
// on it. For a displayed name alongside the button, override in descended class.
|
||
}
|
||
|
||
|
||
#pragma mark -
|
||
|
||
/*======================================================================================
|
||
Get the number of menu items.
|
||
======================================================================================*/
|
||
|
||
Int16 CMenuMailFolderMixin::MGetNumMenuItems(void) {
|
||
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
return ::CountMItems(menuH);
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Get the selected menu item.
|
||
======================================================================================*/
|
||
|
||
Int16 CMenuMailFolderMixin::MGetSelectedMenuItem(void) {
|
||
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Int16 numMenuItems = MGetNumMenuItems();
|
||
|
||
for (Int16 i = 1; i <= numMenuItems; ++i) {
|
||
short mark;
|
||
::GetItemMark(menuH, i, &mark);
|
||
if ( mark ) return i;
|
||
}
|
||
|
||
return 0; // No item selected
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Set the selected menu item. Refresh the menu if the selection changes.
|
||
======================================================================================*/
|
||
|
||
void CMenuMailFolderMixin::MSetSelectedMenuItem(Int16 inIndex, Boolean)
|
||
{
|
||
|
||
Int16 selectedItem = MGetSelectedMenuItem();
|
||
if ( selectedItem != inIndex ) {
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
if ( selectedItem > 0 ) ::SetItemMark(menuH, selectedItem, noMark);
|
||
if ( inIndex > 0 ) ::SetItemMark(menuH, inIndex, checkMark);
|
||
}
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Update the current menu item. If any of the input parameters are different from
|
||
the current menu item parameters, return true and reset the menu item; otherwise,
|
||
return false and don't modify the menu. If inIconID is non-zero, set the icon for
|
||
the menu item to that resource. Don't refresh the menu display in anyway.
|
||
======================================================================================*/
|
||
|
||
Boolean CMenuMailFolderMixin::MUpdateMenuItem(Int16 inMenuItem, ConstStr255Param inName,
|
||
ResIDT inIconID) {
|
||
|
||
Boolean changed = false;
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Assert_(menuH != nil);
|
||
Assert_((inIconID > cMenuIconIDDiff) || (inIconID == 0));
|
||
|
||
Assert_(::CountMItems(menuH) == MGetNumMenuItems());
|
||
Assert_((inMenuItem > 0) && (inMenuItem <= MGetNumMenuItems()));
|
||
|
||
Str255 currentText;
|
||
::GetMenuItemText(menuH, inMenuItem, currentText);
|
||
|
||
changed = !::EqualString(currentText, inName, true, true);
|
||
if ( changed ) ::SetMenuItemText(menuH, inMenuItem, inName);
|
||
|
||
if ( inIconID ) inIconID -= cMenuIconIDDiff;
|
||
Int16 iconID;
|
||
::GetItemIcon(menuH, inMenuItem, &iconID);
|
||
if ( iconID != inIconID ) {
|
||
changed = true;
|
||
::SetItemIcon(menuH, inMenuItem, inIconID);
|
||
}
|
||
|
||
return changed;
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Append the specified menu item to the end of the menu. If inIconID is non-zero, set
|
||
the icon for the menu item to that resource. Don't refresh the menu display in
|
||
anyway.
|
||
======================================================================================*/
|
||
|
||
void CMenuMailFolderMixin::MAppendMenuItem(ConstStr255Param inName, ResIDT inIconID) {
|
||
|
||
LMenu *theMenu = MGetMenu();
|
||
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Assert_(menuH != nil);
|
||
Assert_((inIconID > cMenuIconIDDiff) || (inIconID == 0));
|
||
Int16 numMenuItems = MGetNumMenuItems();
|
||
|
||
Assert_(::CountMItems(menuH) == MGetNumMenuItems());
|
||
|
||
theMenu->InsertCommand(inName, mMenuCommand, numMenuItems);
|
||
|
||
if ( inIconID ) {
|
||
::SetItemIcon(menuH, numMenuItems + 1, inIconID - cMenuIconIDDiff);
|
||
}
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Remove menu items from the end of the current menu, beginning with the item at
|
||
inStartClipItem. Don't refresh the menu display in anyway.
|
||
======================================================================================*/
|
||
|
||
void CMenuMailFolderMixin::MClipMenuItems(Int16 inStartClipItem) {
|
||
|
||
LMenu *theMenu = MGetMenu();
|
||
|
||
Int16 numMenuItems = MGetNumMenuItems();
|
||
|
||
Assert_((inStartClipItem > 0) && (inStartClipItem <= numMenuItems));
|
||
|
||
while ( numMenuItems >= inStartClipItem ) {
|
||
theMenu->RemoveItem(numMenuItems--);
|
||
}
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Get the current menu text for the specified item.
|
||
======================================================================================*/
|
||
|
||
void CMenuMailFolderMixin::MGetMenuItemName(Int16 inMenuItem, Str255 outItemName) {
|
||
|
||
MenuHandle menuH = MGetSystemMenuHandle();
|
||
Assert_(menuH != nil);
|
||
Assert_(::CountMItems(menuH) == MGetNumMenuItems());
|
||
Assert_((inMenuItem > 0) && (inMenuItem <= MGetNumMenuItems()));
|
||
|
||
::GetMenuItemText(menuH, inMenuItem, outItemName);
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Refresh the menu display.
|
||
======================================================================================*/
|
||
|
||
void CMenuMailFolderMixin::MRefreshMenu(void) {
|
||
|
||
// Don't need to do anything since the menu is not visible unless the user has clicked
|
||
// on it.
|
||
}
|
||
|
||
|
||
/*======================================================================================
|
||
Get a handle to the Mac menu associated with this object.
|
||
======================================================================================*/
|
||
|
||
MenuHandle CMenuMailFolderMixin::MGetSystemMenuHandle(void) {
|
||
|
||
return MGetMenu()->GetMacMenuH();
|
||
}
|
||
|
||
|
||
#pragma mark -
|
||
|
||
static const Char16 gsPopup_SmallMark = '<EFBFBD>'; // Mark used for small font popups
|
||
static const Int16 gsPopup_ArrowButtonWidth = 22; // Width used in drawing the arrow only
|
||
|
||
enum
|
||
{
|
||
eSelectAll = 1,
|
||
eUnselectAll = 2
|
||
};
|
||
|
||
CSelectFolderMenu::CSelectFolderMenu(LStream *inStream):
|
||
LGAPopup(inStream),
|
||
mSelectedItemCount(0),
|
||
mTotalItemCount(0),
|
||
mMenuWidth(0)
|
||
{
|
||
mDesiredFolderFlags = (FolderChoices)(eWantIMAP + eWantNews);
|
||
}
|
||
|
||
CSelectFolderMenu::~CSelectFolderMenu()
|
||
{
|
||
|
||
}
|
||
|
||
void CSelectFolderMenu::FinishCreateSelf()
|
||
{
|
||
LGAPopup::FinishCreateSelf();
|
||
mInitialMenuCount = CountMItems(GetMacMenuH());
|
||
CMailFolderMixin::UpdateMailFolderMixinsNow(this);
|
||
// SetArrowOnly(true);
|
||
mTotalItemCount = MGetNumMenuItems() - mInitialMenuCount;
|
||
for (int i = 1; i <= mTotalItemCount; ++i)
|
||
{
|
||
UInt32 folderPrefFlag = MGetFolder(i).GetFolderPrefFlags();
|
||
if (folderPrefFlag & MSG_FOLDER_PREF_OFFLINE)
|
||
{
|
||
mSelectedItemCount++;
|
||
Char16 mark = GetMenuFontSize () < 12 ? gsPopup_SmallMark : checkMark;
|
||
::SetItemMark(GetMacMenuH(), i + mInitialMenuCount, mark);
|
||
}
|
||
}
|
||
UpdateCommandMarks();
|
||
}
|
||
|
||
void
|
||
CSelectFolderMenu::SetupCurrentMenuItem( MenuHandle /*inMenuH*/,
|
||
Int16 /*inCurrentItem*/)
|
||
{
|
||
// Don't muck around with our marks!
|
||
}
|
||
|
||
void
|
||
CSelectFolderMenu::UpdateCommandMarks()
|
||
{
|
||
short mark;
|
||
if (mSelectedItemCount == mTotalItemCount && mTotalItemCount)
|
||
{
|
||
// Mark the SelectAllCommand.
|
||
mark = GetMenuFontSize () < 12 ? gsPopup_SmallMark : checkMark;
|
||
}
|
||
else
|
||
{
|
||
// Unmark the SelectAllCommand.
|
||
mark = 0;
|
||
}
|
||
::SetItemMark(GetMacMenuH(), eSelectAll, mark);
|
||
if (!mSelectedItemCount && mTotalItemCount)
|
||
{
|
||
// Mark the UnselectAllCommand.
|
||
mark = GetMenuFontSize () < 12 ? gsPopup_SmallMark : checkMark;
|
||
}
|
||
else
|
||
{
|
||
// Unmark the UnselectAllCommand.
|
||
mark = 0;
|
||
}
|
||
::SetItemMark(GetMacMenuH(), eUnselectAll, mark);
|
||
if (!mTotalItemCount)
|
||
{
|
||
::DisableItem(GetMacMenuH(), eSelectAll);
|
||
::DisableItem(GetMacMenuH(), eUnselectAll);
|
||
}
|
||
else
|
||
{
|
||
::EnableItem(GetMacMenuH(), eSelectAll);
|
||
::EnableItem(GetMacMenuH(), eUnselectAll);
|
||
}
|
||
}
|
||
|
||
void
|
||
CSelectFolderMenu::CommitCurrentSelections()
|
||
{
|
||
for (int i = 1; i <= mTotalItemCount; ++i)
|
||
{
|
||
CMessageFolder currentFolder = MGetFolder(i);
|
||
UInt32 folderPrefFlag = currentFolder.GetFolderPrefFlags();
|
||
short mark;
|
||
::GetItemMark(GetMacMenuH(), i + mInitialMenuCount, &mark);
|
||
folderPrefFlag = mark? folderPrefFlag | MSG_FOLDER_PREF_OFFLINE:
|
||
folderPrefFlag & ~MSG_FOLDER_PREF_OFFLINE;
|
||
::MSG_SetFolderPrefFlags(currentFolder.GetFolderInfo(), folderPrefFlag);
|
||
}
|
||
}
|
||
|
||
void
|
||
CSelectFolderMenu::SetValue(Int32 inValue)
|
||
{
|
||
short mark;
|
||
if (inValue > eUnselectAll)
|
||
{
|
||
::GetItemMark(GetMacMenuH(), inValue, &mark);
|
||
if (mark)
|
||
{
|
||
mSelectedItemCount--;
|
||
mark = 0;
|
||
}
|
||
else
|
||
{
|
||
mSelectedItemCount++;
|
||
mark = GetMenuFontSize () < 12 ? gsPopup_SmallMark : checkMark;
|
||
}
|
||
::SetItemMark(GetMacMenuH(), inValue, mark);
|
||
}
|
||
else
|
||
{
|
||
// Optimization if this option is already checked, then don't do anything
|
||
if (inValue == eUnselectAll)
|
||
{
|
||
mark = 0;
|
||
mSelectedItemCount = 0;
|
||
}
|
||
if (inValue == eSelectAll)
|
||
{
|
||
mark = GetMenuFontSize () < 12 ? gsPopup_SmallMark : checkMark;
|
||
mSelectedItemCount = mTotalItemCount;
|
||
}
|
||
for (int i = 1; i <= mTotalItemCount; ++i)
|
||
{
|
||
::SetItemMark(GetMacMenuH(), i + mInitialMenuCount, mark);
|
||
}
|
||
}
|
||
UpdateCommandMarks();
|
||
}
|