зеркало из https://github.com/mozilla/pjs.git
1991 строка
58 KiB
C++
1991 строка
58 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.
|
||
|
*/
|
||
|
/*public APIs for the address books.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include "xp.h"
|
||
|
|
||
|
#include "abcom.h"
|
||
|
#include "abcpane.h"
|
||
|
#include "abpane2.h"
|
||
|
#include "abcinfo.h"
|
||
|
#include "abpicker.h"
|
||
|
#include "addrprsr.h"
|
||
|
#include "prefapi.h"
|
||
|
#include "namecomp.h"
|
||
|
|
||
|
#include "msgurlq.h"
|
||
|
|
||
|
#include "libi18n.h"
|
||
|
#include "intl_csi.h"
|
||
|
|
||
|
extern "C"
|
||
|
{
|
||
|
extern int MK_ADDR_FIRST_LAST_SEP;
|
||
|
}
|
||
|
|
||
|
#if !(defined(SOLARIS) && defined(NS_USE_NATIVE)) && !(defined(HPUX))
|
||
|
inline
|
||
|
#endif
|
||
|
AB_Pane * CastABPane(MSG_Pane * pane)
|
||
|
{
|
||
|
if (pane)
|
||
|
if (pane->GetPaneType() == AB_ABPANE || pane->GetPaneType() == AB_PICKERPANE)
|
||
|
return (AB_Pane *) pane;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
#if !(defined(SOLARIS) && defined(NS_USE_NATIVE)) && !(defined(HPUX))
|
||
|
inline
|
||
|
#endif
|
||
|
AB_ContainerPane * CastABContainerPane(MSG_Pane * pane)
|
||
|
{
|
||
|
if (pane)
|
||
|
if (pane->GetPaneType() == AB_CONTAINERPANE)
|
||
|
return (AB_ContainerPane *) pane;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
#if !(defined(SOLARIS) && defined(NS_USE_NATIVE)) && !(defined(HPUX))
|
||
|
inline
|
||
|
#endif
|
||
|
AB_MailingListPane * CastABMailingListPane(MSG_Pane * pane)
|
||
|
{
|
||
|
if (pane)
|
||
|
if (pane->GetPaneType() == AB_MAILINGLISTPANE)
|
||
|
return (AB_MailingListPane *) pane;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
#if !(defined(SOLARIS) && defined(NS_USE_NATIVE)) && !(defined(HPUX))
|
||
|
inline
|
||
|
#endif
|
||
|
AB_PersonPane * CastABPersonPane(MSG_Pane * pane)
|
||
|
{
|
||
|
if (pane)
|
||
|
if (pane->GetPaneType() == AB_PERSONENTRYPANE)
|
||
|
return (AB_PersonPane * ) pane;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
#if !(defined(SOLARIS) && defined(NS_USE_NATIVE)) && !(defined(HPUX))
|
||
|
inline
|
||
|
#endif
|
||
|
AB_PickerPane * CastABPickerPane(MSG_Pane * pane)
|
||
|
{
|
||
|
if (pane)
|
||
|
if (pane->GetPaneType() == AB_PICKERPANE)
|
||
|
return (AB_PickerPane *) pane;
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// ------------------------ Public API implementation -------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
|
||
|
AB_API int AB_SetShowPropertySheetForDirFunc(MSG_Pane * pane, AB_ShowPropertySheetForDirFunc * func)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
{
|
||
|
abcPane->SetShowPropSheetForDirFunc(func);
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetShowPropertySheetForEntryFunc(MSG_Pane * pane, AB_ShowPropertySheetForEntryFunc * func)
|
||
|
{
|
||
|
if (pane)
|
||
|
{
|
||
|
pane->SetShowPropSheetForEntryFunc(func);
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
#if 0
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
{
|
||
|
abPane->SetShowPropSheetForEntryFunc(func);
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
{
|
||
|
abcPane->SetShowPropSheetForEntryFunc(func);
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return AB_INVALID_PANE;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
|
||
|
/*******************************************************************************
|
||
|
The following are APIs just about all of the panes will respond to.
|
||
|
********************************************************************************/
|
||
|
|
||
|
AB_API int AB_ClosePane(MSG_Pane * pane)
|
||
|
{
|
||
|
// you know, you could really make a good argument that the ab panes should be
|
||
|
// subclassed from an abpane so we could make a virtual method for Close instead
|
||
|
// of this...
|
||
|
|
||
|
// WARNING: the picker pane can be cast as a AB_Pane as well...since Close is static and
|
||
|
// as such cannot be virtual, make sure we cast as a picker pane before trying as an ab_pane....
|
||
|
// oh what a tangled web we weave...
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return AB_PickerPane::Close(abPickerPane);
|
||
|
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return AB_Pane::Close(abPane);
|
||
|
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return AB_ContainerPane::Close(abcPane);
|
||
|
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return AB_MailingListPane::Close(mlPane);
|
||
|
|
||
|
AB_PersonPane * personPane = CastABPersonPane(pane);
|
||
|
if (personPane)
|
||
|
return AB_PersonPane::Close(personPane);
|
||
|
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
/**************************************************************************************
|
||
|
Virtual List View APIs
|
||
|
***************************************************************************************/
|
||
|
AB_API int AB_SetFEPageSizeForPane(MSG_Pane * pane, uint32 pageSize)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane) // valid cast?
|
||
|
return abPane->SetPageSize(pageSize);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
/**************************************************************************************
|
||
|
Selection APIs
|
||
|
***************************************************************************************/
|
||
|
AB_API XP_Bool AB_UseExtendedSelection(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->UseExtendedSelection();
|
||
|
else
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_AddSelection(MSG_Pane * pane, MSG_ViewIndex index)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane && abPane->AddSelection(index))
|
||
|
return AB_SUCCESS;
|
||
|
else
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
AB_API XP_Bool AB_IsSelected(MSG_Pane * pane, MSG_ViewIndex index)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->IsSelected(index);
|
||
|
else
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_RemoveSelection(MSG_Pane * pane, MSG_ViewIndex index)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane && abPane->RemoveSelection(index))
|
||
|
return AB_SUCCESS;
|
||
|
else
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
AB_API void AB_RemoveAllSelections(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane) // valid cast?
|
||
|
abPane->RemoveSelection(MSG_VIEWINDEXNONE);
|
||
|
}
|
||
|
|
||
|
/***************************************************************************************
|
||
|
Type down and name completion APIs. Both actions are asynchronous. Type down generates
|
||
|
a MSG_PaneNotifyTypeDownCompleted pane notification and name completion calls a FE
|
||
|
regsitered call back.
|
||
|
****************************************************************************************/
|
||
|
AB_API int AB_TypedownSearch(MSG_Pane * pane, const char * typedownValue, MSG_ViewIndex startIndex)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane) // valid cast??
|
||
|
return abPane->TypedownSearch(typedownValue, startIndex);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifdef FE_IMPLEMENTS_VISIBLE_NC
|
||
|
AB_API int AB_NameCompletionSearch(MSG_Pane * pane, const char * completionValue, AB_NameCompletionExitFunction * exitFunction, XP_Bool userSeesPane, void * cookie)
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->NameCompletionSearch(completionValue, exitFunction, userSeesPane, cookie);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
#else
|
||
|
AB_API int AB_NameCompletionSearch(MSG_Pane * pane, const char * completionValue, AB_NameCompletionExitFunction * exitFunction, void * cookie)
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->NameCompletionSearch(completionValue, exitFunction, TRUE /* assume pane is visible */, cookie);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
/* Caller must free string!!! */
|
||
|
AB_API char * AB_GetNameCompletionDisplayString(AB_NameCompletionCookie * cookie)
|
||
|
{
|
||
|
if (cookie)
|
||
|
return cookie->GetNameCompletionDisplayString();
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
/* Caller must free string!!! */
|
||
|
AB_API char * AB_GetHeaderString(AB_NameCompletionCookie * cookie)
|
||
|
{
|
||
|
if (cookie)
|
||
|
return cookie->GetHeaderString();
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
/* Caller must free string!!! */
|
||
|
AB_API char * AB_GetExpandedHeaderString(AB_NameCompletionCookie * cookie)
|
||
|
{
|
||
|
if (cookie)
|
||
|
return cookie->GetExpandedHeaderString();
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
AB_API AB_EntryType AB_GetEntryTypeForNCCookie(AB_NameCompletionCookie * cookie)
|
||
|
{
|
||
|
if (cookie)
|
||
|
return cookie->GetEntryType();
|
||
|
else
|
||
|
return AB_NakedAddress; // we need some default value for this...
|
||
|
}
|
||
|
|
||
|
|
||
|
AB_API int AB_FreeNameCompletionCookie(AB_NameCompletionCookie * cookie)
|
||
|
{
|
||
|
if (cookie)
|
||
|
AB_NameCompletionCookie::Close(cookie);
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************************
|
||
|
Name completion picker pane specific APIs.
|
||
|
*****************************************************************************************/
|
||
|
|
||
|
AB_API int AB_CreateABPickerPane(MSG_Pane ** abPickerPane,MWContext * context,MSG_Master * master, uint32 pageSize)
|
||
|
{
|
||
|
return AB_PickerPane::Create(abPickerPane, context, master, pageSize);
|
||
|
}
|
||
|
|
||
|
AB_API AB_ContainerType AB_GetEntryContainerType(MSG_Pane * pane,MSG_ViewIndex index)
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->GetEntryContainerType(index);
|
||
|
else
|
||
|
return AB_UnknownContainer;
|
||
|
}
|
||
|
|
||
|
AB_API AB_NameCompletionCookie * AB_GetNameCompletionCookieForNakedAddress(const char * nakedAddress)
|
||
|
{
|
||
|
AB_NameCompletionCookie * cookie = new AB_NameCompletionCookie(nakedAddress);
|
||
|
if (cookie)
|
||
|
return cookie; // caller must call AB_FreeNameCompletionCookie when done with this cookie
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
AB_API AB_NameCompletionCookie * AB_GetNameCompletionCookieForIndex(MSG_Pane * pane, MSG_ViewIndex index)
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->GetNameCompletionCookieForIndex(index);
|
||
|
else
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
AB_MailingListPane * mList = CastABMailingListPane(pane);
|
||
|
if (abPane || mList)
|
||
|
{
|
||
|
// anyone else, get the ctr for the index and convert the entry..
|
||
|
ABID entryID = AB_ABIDUNKNOWN;
|
||
|
AB_GetABIDForIndex(pane, index, &entryID);
|
||
|
AB_ContainerInfo * ctr = AB_GetContainerForIndex(pane, index);
|
||
|
if (ctr && entryID != AB_ABIDUNKNOWN)
|
||
|
{
|
||
|
AB_NameCompletionCookie * cookie = new AB_NameCompletionCookie(pane, ctr, entryID);
|
||
|
if (cookie)
|
||
|
return cookie;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
/***********************************************************************************
|
||
|
AB_Pane specific APIs
|
||
|
**********************************************************************************/
|
||
|
|
||
|
AB_API int AB_CreateABPane(MSG_Pane ** pane, MWContext * context,MSG_Master * master)
|
||
|
{
|
||
|
return AB_Pane::Create(pane, context, master);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_InitializeABPane(MSG_Pane * pane, AB_ContainerInfo * abContainer)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->LoadContainer(abContainer);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ChangeABContainer(MSG_Pane * pane, AB_ContainerInfo * container)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->LoadContainer(container);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetABIDForIndex(MSG_Pane * pane, MSG_ViewIndex index, ABID * id)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane && id)
|
||
|
{
|
||
|
*id = abPane->GetABIDForIndex(index);
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetEntryIndex(MSG_Pane * pane, ABID id, MSG_ViewIndex * index)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
{
|
||
|
*index = abPane->GetEntryIndexForID(id);
|
||
|
return AB_SUCCESS; // because I changed GetEntryIndexForID...it now returns an index instead of an error code
|
||
|
}
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SearchDirectoryAB2(MSG_Pane * pane, char * searchString)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->SearchDirectory(searchString);
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_LDAPSearchResultsAB2(MSG_Pane * pane, MSG_ViewIndex index, int32 num)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->LDAPSearchResults(index, num);
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FinishSearchAB2(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->FinishSearch();
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_CommandAB2(MSG_Pane * pane, AB_CommandType command, MSG_ViewIndex * indices, int32 numIndices)
|
||
|
{
|
||
|
if (pane)
|
||
|
return pane->DoCommand((MSG_CommandType)command, indices, numIndices);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_CommandStatusAB2(MSG_Pane * srcPane, AB_CommandType command, MSG_ViewIndex * indices, int32 numIndices,
|
||
|
XP_Bool * selectable_p, MSG_COMMAND_CHECK_STATE * selected_p, const char ** displayString,
|
||
|
XP_Bool * plural_p)
|
||
|
{
|
||
|
if (srcPane)
|
||
|
return (int) srcPane->GetCommandStatus((MSG_CommandType)command, indices, numIndices, selectable_p, selected_p, displayString, plural_p);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
|
||
|
}
|
||
|
|
||
|
/****************************************************************************************
|
||
|
AB_ContainerInfo General APIs - adding users and a sender. Does not require a pane
|
||
|
*****************************************************************************************/
|
||
|
|
||
|
AB_API int AB_AddUserAB2(AB_ContainerInfo * abContainer, AB_AttributeValue * valuesArray, uint16 numItems, ABID * entryID /* back end fills this value */)
|
||
|
{
|
||
|
if (abContainer)
|
||
|
return abContainer->AddEntry(valuesArray, numItems, entryID);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_AddUserWithUIAB2(MSG_Pane * srcPane, AB_ContainerInfo * abContainer, AB_AttributeValue * valuesArray, uint16 numItems, XP_Bool lastOneToAdd)
|
||
|
{
|
||
|
if (abContainer)
|
||
|
return abContainer->AddUserWithUI(srcPane, valuesArray, numItems, lastOneToAdd);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_AddSenderAB2(MSG_Pane * srcPane, AB_ContainerInfo * abContainer, const char * author, const char * url)
|
||
|
{
|
||
|
int status = AB_SUCCESS;
|
||
|
|
||
|
int num;
|
||
|
int32 length;
|
||
|
char * ptr;
|
||
|
|
||
|
if (srcPane && url && author && (XP_STRNCASECMP(url, "mailto:?to=", 11) == 0)) /* make sure it is a mailto url */
|
||
|
{
|
||
|
// skip the mailto url info
|
||
|
url += 11;
|
||
|
|
||
|
// find the first address by finding the next section
|
||
|
ptr = XP_STRCHR(url, '&');
|
||
|
length = (ptr ? (ptr - url) : XP_STRLEN(url));
|
||
|
char * buf = (char *) XP_ALLOC(length + 1);
|
||
|
if (buf)
|
||
|
{
|
||
|
XP_MEMCPY(buf, url, length);
|
||
|
buf[length] = 0;
|
||
|
buf = NET_UnEscape(buf);
|
||
|
|
||
|
// check if there is a name and address
|
||
|
char * name = NULL;
|
||
|
char * address = NULL;
|
||
|
num = MSG_ParseRFC822Addresses(buf, &name, &address);
|
||
|
if (num)
|
||
|
{
|
||
|
if (!(*name))
|
||
|
name = XP_STRDUP(author);
|
||
|
status = AB_AddNameAndAddress(srcPane, abContainer, name, address, FALSE);
|
||
|
|
||
|
// advance to the cc part
|
||
|
if ( (XP_STRNCASECMP(ptr, "&cc=", 4) == 0) && (status == AB_SUCCESS))
|
||
|
{
|
||
|
ptr += 4; // skip over &cc=....
|
||
|
char * ptr1 = XP_STRCHR(ptr, '&');
|
||
|
length = (ptr1 ? (ptr1 - ptr) : XP_STRLEN(ptr));
|
||
|
buf = (char *) XP_ALLOC(length + 1);
|
||
|
if (buf)
|
||
|
{
|
||
|
XP_MEMCPY(buf, ptr, length);
|
||
|
buf[length] = 0;
|
||
|
buf = NET_UnEscape(buf);
|
||
|
|
||
|
char * ccNames = NULL;
|
||
|
char * ccAddresses = NULL;
|
||
|
|
||
|
num = MSG_ParseRFC822Addresses(buf, &ccNames, &ccAddresses);
|
||
|
|
||
|
char * currentName = ccNames;
|
||
|
char * currentAddress = ccAddresses;
|
||
|
int loopStatus = AB_SUCCESS;
|
||
|
for (int i = 0; i < num && loopStatus == AB_SUCCESS; i++)
|
||
|
{
|
||
|
loopStatus = AB_AddNameAndAddress(srcPane, abContainer, currentName, currentAddress, FALSE);
|
||
|
// advance name & address to the next pair.
|
||
|
if (*currentName)
|
||
|
currentName += XP_STRLEN(currentName) + 1;
|
||
|
if (*currentAddress)
|
||
|
currentAddress += XP_STRLEN(currentAddress) + 1;
|
||
|
}
|
||
|
|
||
|
status = loopStatus;
|
||
|
XP_FREEIF(buf);
|
||
|
if (ccNames)
|
||
|
XP_FREE(ccNames);
|
||
|
if (ccAddresses)
|
||
|
XP_FREEIF(ccAddresses);
|
||
|
}
|
||
|
} // end of if cc part
|
||
|
} // end of if at least one address to parse
|
||
|
|
||
|
XP_FREE(buf);
|
||
|
if (name)
|
||
|
XP_FREE(name);
|
||
|
if (address)
|
||
|
XP_FREE(address);
|
||
|
}
|
||
|
else
|
||
|
status = AB_OUT_OF_MEMORY;
|
||
|
|
||
|
}
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
/* the following is called by the msgdbview when it wants to add a name and a address to the specified container */
|
||
|
AB_API int AB_AddNameAndAddress(MSG_Pane * pane, AB_ContainerInfo * abContainer, const char * name, const char * address, XP_Bool lastOneToAdd)
|
||
|
{
|
||
|
int status = AB_SUCCESS;
|
||
|
AB_AttribValueArray attribs;
|
||
|
char * tempname = NULL;
|
||
|
if (pane && abContainer)
|
||
|
{
|
||
|
INTL_CharSetInfo c = LO_GetDocumentCharacterSetInfo(pane->GetContext());
|
||
|
|
||
|
if (XP_STRLEN (name))
|
||
|
tempname = XP_STRDUP (name);
|
||
|
else
|
||
|
{
|
||
|
int len = 0;
|
||
|
char * at = NULL;
|
||
|
if (address)
|
||
|
{
|
||
|
// if the mail address doesn't contain @
|
||
|
// then the name is the whole email address
|
||
|
if ((at = XP_STRCHR(address, '@')) == NULL)
|
||
|
tempname = MSG_ExtractRFC822AddressNames (address);
|
||
|
else
|
||
|
{
|
||
|
// otherwise extract everything up to the @
|
||
|
// for the name
|
||
|
len = XP_STRLEN (address) - XP_STRLEN (at);
|
||
|
tempname = (char *) XP_ALLOC (len + 1);
|
||
|
XP_STRNCPY_SAFE (tempname, address, len + 1);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
char *converted = INTL_DecodeMimePartIIStr((const char *) tempname, INTL_DefaultWinCharSetID(0), FALSE);
|
||
|
if (converted && (converted != tempname))
|
||
|
{
|
||
|
XP_FREE(tempname);
|
||
|
tempname = converted;
|
||
|
}
|
||
|
|
||
|
// initialize various attributes
|
||
|
uint16 numAttributes = 6;
|
||
|
AB_AttributeValue * values = (AB_AttributeValue *) XP_ALLOC(sizeof(AB_AttributeValue) * numAttributes);
|
||
|
if (values)
|
||
|
{
|
||
|
// first, break tempname (display name) into components for first/last name
|
||
|
char * firstName = NULL;
|
||
|
char * lastName = NULL;
|
||
|
|
||
|
AB_SplitFullName(tempname, &firstName, &lastName, INTL_GetCSIWinCSID(c));
|
||
|
|
||
|
values[0].attrib = AB_attribGivenName;
|
||
|
if (firstName)
|
||
|
values[0].u.string = firstName;
|
||
|
else
|
||
|
AB_CopyDefaultAttributeValue(AB_attribGivenName, &values[0]);
|
||
|
|
||
|
values[1].attrib = AB_attribEmailAddress;
|
||
|
if (address)
|
||
|
values[1].u.string = XP_STRDUP(address);
|
||
|
else
|
||
|
AB_CopyDefaultAttributeValue(AB_attribEmailAddress, &values[1]);
|
||
|
|
||
|
values[2].attrib = AB_attribEntryType;
|
||
|
values[2].u.entryType = AB_Person;
|
||
|
|
||
|
values[3].attrib = AB_attribWinCSID;
|
||
|
values[3].u.shortValue = INTL_GetCSIWinCSID(c);
|
||
|
|
||
|
values[4].attrib = AB_attribDisplayName;
|
||
|
values[4].u.string = tempname; /* because temp name happens to be the display name */
|
||
|
|
||
|
values[5].attrib = AB_attribFamilyName;
|
||
|
if (lastName)
|
||
|
values[5].u.string = lastName;
|
||
|
else
|
||
|
AB_CopyDefaultAttributeValue(AB_attribFamilyName, &values[5]);
|
||
|
|
||
|
/* AB_BreakApartFirstName (FE_GetAddressBook(NULL), &person); */
|
||
|
status = AB_AddUserWithUIAB2 (pane, abContainer, values, numAttributes, lastOneToAdd);
|
||
|
AB_FreeEntryAttributeValues(values, numAttributes);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
status = ABError_NullPointer;
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************************
|
||
|
Drag and Drop Related APIs - vcards, ab lines, containers, etc.
|
||
|
*****************************************************************************************/
|
||
|
|
||
|
AB_API int AB_DragEntriesIntoContainer(MSG_Pane * srcPane, const MSG_ViewIndex *indices, int32 numIndices,
|
||
|
AB_ContainerInfo * destContainer, AB_DragEffect request)
|
||
|
{
|
||
|
// we'll first try it as an AB_Pane, but it might be an ABContainerPane
|
||
|
AB_Pane * srcABPane = CastABPane(srcPane); // try ab Pane first
|
||
|
if (srcABPane)
|
||
|
return srcABPane->DragEntriesIntoContainer(indices, numIndices, destContainer, request);
|
||
|
else
|
||
|
{
|
||
|
AB_ContainerPane * srcABCPane = CastABContainerPane(srcPane);
|
||
|
if (srcABCPane)
|
||
|
return srcABCPane->DragEntriesIntoContainer(indices, numIndices, destContainer, request);
|
||
|
else
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(srcPane);
|
||
|
if (mlPane)
|
||
|
return mlPane->DragEntriesIntoContainer(indices, numIndices, destContainer, request);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API AB_DragEffect AB_DragEntriesIntoContainerStatus(MSG_Pane * srcPane, const MSG_ViewIndex *indices, int32 numIndices,
|
||
|
AB_ContainerInfo * destContainer, AB_DragEffect request)
|
||
|
{
|
||
|
// we'll first try it as an AB_Pane, but it might be an ABContainerPane
|
||
|
AB_Pane * srcABPane = CastABPane(srcPane); // try ab Pane first
|
||
|
if (srcABPane)
|
||
|
return srcABPane->DragEntriesIntoContainerStatus(indices, numIndices, destContainer, request);
|
||
|
else
|
||
|
{
|
||
|
AB_ContainerPane * srcABCPane = CastABContainerPane(srcPane);
|
||
|
if (srcABCPane)
|
||
|
return srcABCPane->DragEntriesIntoContainerStatus(indices, numIndices, destContainer, request);
|
||
|
else
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(srcPane);
|
||
|
if (mlPane)
|
||
|
return mlPane->DragEntriesIntoContainerStatus(indices, numIndices, destContainer, request);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return AB_Drag_Not_Allowed;
|
||
|
}
|
||
|
|
||
|
/***************************************************************************************
|
||
|
|
||
|
Asynchronous copy APIs. These APIs are used by mkabook.cpp for asycnh copy/move/delete
|
||
|
entry operations between containers.
|
||
|
|
||
|
***************************************************************************************/
|
||
|
|
||
|
AB_API int AB_BeginEntryCopy(AB_ContainerInfo * srcContainer, MWContext * context, AB_AddressBookCopyInfo * copyInfo,
|
||
|
void ** copyCookie, XP_Bool * copyFinished)
|
||
|
{
|
||
|
if (srcContainer)
|
||
|
return srcContainer->BeginCopyEntry(context, copyInfo, copyCookie, copyFinished);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_MoreEntryCopy(AB_ContainerInfo * srcContainer,MWContext * context, AB_AddressBookCopyInfo * copyInfo,
|
||
|
void ** copyCookie, XP_Bool * copyFinished)
|
||
|
{
|
||
|
if (srcContainer)
|
||
|
return srcContainer->MoreCopyEntry(context, copyInfo, copyCookie, copyFinished);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FinishEntryCopy(AB_ContainerInfo * srcContainer, MWContext * context, AB_AddressBookCopyInfo * copyInfo, void ** copyCookie)
|
||
|
{
|
||
|
if (srcContainer)
|
||
|
return srcContainer->FinishCopyEntry(context, copyInfo, copyCookie);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_InterruptEntryCopy(AB_ContainerInfo * srcContainer, MWContext * context,AB_AddressBookCopyInfo * copyInfo, void ** copyCookie)
|
||
|
{
|
||
|
if (srcContainer)
|
||
|
return srcContainer->InterruptCopyEntry(context, copyInfo, copyCookie);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
/****************************************************************************************
|
||
|
Importing and Exporting - ABs from files, vcards...
|
||
|
*****************************************************************************************/
|
||
|
#ifdef FE_IMPLEMENTS_NEW_IMPORT
|
||
|
AB_API int AB_ImportData(MSG_Pane * pane, AB_ContainerInfo * destContainer, const char * buffer, int32 bufSize, AB_ImportExportType dataType)
|
||
|
{
|
||
|
if (destContainer)
|
||
|
return destContainer->ImportData(pane, buffer, bufSize, dataType);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
#else
|
||
|
AB_API int AB_ImportData(AB_ContainerInfo * destContainer, const char * buffer, int32 bufSize, AB_ImportExportType dataType)
|
||
|
{
|
||
|
if (destContainer)
|
||
|
return destContainer->ImportData(NULL, buffer, bufSize, dataType);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
AB_API XP_Bool AB_ImportDataStatus (AB_ContainerInfo * destContainer, AB_ImportExportType dataType)
|
||
|
{
|
||
|
if (destContainer)
|
||
|
return destContainer->ImportDataStatus(dataType);
|
||
|
else
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
#ifdef FE_IMPLEMENTS_NEW_IMPORT
|
||
|
AB_API int AB_ExportData(MSG_Pane * pane, AB_ContainerInfo * srcContainer, char ** buffer, /* filename or NULL. or if type = vcard, then BE allocated & filled vcard */
|
||
|
int32 * bufSize, /* ignored unless Vcard in which case FE allocates, BE fills */
|
||
|
AB_ImportExportType dataType) /* valid types: filename, prompt for filename, VCard */
|
||
|
{
|
||
|
if (srcContainer)
|
||
|
return srcContainer->ExportData(pane, buffer, bufSize, dataType);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
#else
|
||
|
AB_API int AB_ExportData(AB_ContainerInfo * srcContainer, char ** buffer, /* filename or NULL. or if type = vcard, then BE allocated & filled vcard */
|
||
|
int32 * bufSize, /* ignored unless Vcard in which case FE allocates, BE fills */
|
||
|
AB_ImportExportType dataType) /* valid types: filename, prompt for filename, VCard */
|
||
|
{
|
||
|
if (srcContainer)
|
||
|
return srcContainer->ExportData(NULL, buffer, bufSize, dataType);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
AB_API int AB_ImportBegin(AB_ContainerInfo * container, MWContext * context, const char * fileName, void ** importCookie, XP_Bool * importFinished)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ImportBegin(context, fileName, importCookie, importFinished);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ImportMore(AB_ContainerInfo * container, void ** importCookie, XP_Bool * importFinished)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ImportMore(importCookie, importFinished);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ImportInterrupt(AB_ContainerInfo * container, void ** importCookie)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ImportInterrupt(importCookie);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ImportFinish(AB_ContainerInfo * container, void ** importCookie)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ImportFinish(importCookie);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ImportProgress(AB_ContainerInfo * container, void * importCookie, uint32 * position, uint32 * fileLength, uint32 * passCount)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ImportProgress(importCookie, position,fileLength, passCount);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
|
||
|
AB_API int AB_ExportBegin(AB_ContainerInfo * container, MWContext * context, const char * fileName, void ** exportCookie, XP_Bool * exportFinished)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ExportBegin(context, fileName, exportCookie, exportFinished);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ExportMore(AB_ContainerInfo * container, void ** exportCookie, XP_Bool * exportFinished)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ExportMore(exportCookie, exportFinished);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ExportInterrupt(AB_ContainerInfo * container, void ** exportCookie)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ExportInterrupt(exportCookie);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ExportFinish(AB_ContainerInfo * container, void ** exportCookie)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ExportFinish(exportCookie);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ExportProgress(AB_ContainerInfo * container, void * exportCookie, uint32 * numberExported, uint32 * totalEntries)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->ExportProgress(exportCookie, numberExported, totalEntries);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
/* used only in the back end to generate an addbook copy url for copying, moving and deleting entries from/to and address book */
|
||
|
AB_API int AB_CreateAndRunCopyInfoUrl (MSG_Pane * pane, AB_ContainerInfo * srcContainer, ABID * idArray, int32 numItems, AB_ContainerInfo * destContainer, AB_AddressCopyInfoState action)
|
||
|
{
|
||
|
int status = AB_SUCCESS;
|
||
|
if (srcContainer && idArray && pane && numItems)
|
||
|
{
|
||
|
char * url = AB_ContainerInfo::BuildCopyUrl();
|
||
|
AB_AddressBookCopyInfo * copyInfo = (AB_AddressBookCopyInfo *) XP_ALLOC(sizeof(AB_AddressBookCopyInfo));
|
||
|
if (copyInfo)
|
||
|
{
|
||
|
copyInfo->destContainer = destContainer;
|
||
|
if (destContainer)
|
||
|
copyInfo->destContainer->Acquire(); // ref count it...release will occurr when url is finished...
|
||
|
copyInfo->srcContainer = srcContainer;
|
||
|
if (srcContainer)
|
||
|
copyInfo->srcContainer->Acquire();
|
||
|
copyInfo->idArray = idArray;
|
||
|
copyInfo->numItems = numItems;
|
||
|
copyInfo->state = action;
|
||
|
URL_Struct *url_s = NET_CreateURLStruct (url, NET_DONT_RELOAD);
|
||
|
if (url_s)
|
||
|
{
|
||
|
url_s->owner_data = (void *) copyInfo;
|
||
|
MSG_UrlQueue::AddUrlToPane(url_s, AB_ContainerInfo::PostAddressBookCopyUrlExitFunc, pane, FALSE, NET_DONT_RELOAD);
|
||
|
}
|
||
|
else
|
||
|
AB_FreeAddressBookCopyInfo(copyInfo);
|
||
|
} // if copyinfo
|
||
|
else
|
||
|
status = AB_OUT_OF_MEMORY;
|
||
|
XP_FREEIF(url);
|
||
|
}
|
||
|
else
|
||
|
status = ABError_NullPointer;
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
/* used only in the back end to generate add ldap to abook urls and adds them to the url queue...
|
||
|
For each entry, we fire off an ldap url to fetch its attributes..*/
|
||
|
AB_API int AB_ImportLDAPEntriesIntoContainer(MSG_Pane * pane, const MSG_ViewIndex * indices, int32 numIndices, AB_ContainerInfo * destContainer)
|
||
|
{
|
||
|
int status = AB_SUCCESS;
|
||
|
|
||
|
// vars we need to get the distinguished name for each entry....
|
||
|
AB_AttributeValue * value = NULL;
|
||
|
uint16 numItems = 1;
|
||
|
AB_AttribID attrib = AB_attribDistName;
|
||
|
|
||
|
if (pane && destContainer && numIndices)
|
||
|
{
|
||
|
MSG_UrlQueue *pQueue = new MSG_AddLdapToAddressBookQueue (pane);
|
||
|
|
||
|
for (int32 i = 0; i < numIndices; i++)
|
||
|
{
|
||
|
// for each index, get its container...
|
||
|
AB_ContainerInfo * ctrToUse = AB_GetContainerForIndex(pane, indices[i]);
|
||
|
// make sure src ctr is an ldap ctr
|
||
|
AB_LDAPContainerInfo * ldapCtr = NULL;
|
||
|
if (ctrToUse)
|
||
|
ldapCtr = ctrToUse->GetLDAPContainerInfo();
|
||
|
if (ldapCtr)
|
||
|
{
|
||
|
AB_GetEntryAttributesForPane(pane, indices[i], &attrib, &value, &numItems);
|
||
|
if (value && (numItems == 1) && value->u.string && *(value->u.string)) // make sure we have a distinguished name..
|
||
|
{
|
||
|
char * url = ldapCtr->BuildLDAPUrl("addbook-ldap", value->u.string);
|
||
|
AB_AddressBookCopyInfo * copyInfo = (AB_AddressBookCopyInfo *) XP_ALLOC(sizeof(AB_AddressBookCopyInfo));
|
||
|
if (copyInfo)
|
||
|
{
|
||
|
copyInfo->destContainer = destContainer;
|
||
|
copyInfo->destContainer->Acquire(); // ref count it...release will occurr when url is finished...
|
||
|
copyInfo->srcContainer = NULL;
|
||
|
copyInfo->idArray = NULL;
|
||
|
copyInfo->numItems = NULL;
|
||
|
URL_Struct *url_s = NET_CreateURLStruct (url, NET_DONT_RELOAD);
|
||
|
if (url_s)
|
||
|
{
|
||
|
url_s->owner_data = (void *) copyInfo;
|
||
|
pQueue->AddUrlToPane(url_s, AB_ContainerInfo::PostAddressBookCopyUrlExitFunc, pane, FALSE, NET_DONT_RELOAD);
|
||
|
}
|
||
|
} // if copyinfo
|
||
|
else
|
||
|
status = AB_OUT_OF_MEMORY;
|
||
|
XP_FREEIF(url);
|
||
|
} // if value
|
||
|
|
||
|
AB_FreeEntryAttributeValue(value);
|
||
|
value = NULL; // reset for next pass...
|
||
|
} // if ldap ctr
|
||
|
} // for each index loop
|
||
|
|
||
|
pQueue->GetNextUrl(); // Start doing the work. Queue deletes itself when it's done
|
||
|
}
|
||
|
else
|
||
|
status = ABError_NullPointer;
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
|
||
|
/****************************************************************************************
|
||
|
ABContainer Pane --> Creation, Loading, getting line data for each container.
|
||
|
*****************************************************************************************/
|
||
|
AB_API int AB_CreateContainerPane(MSG_Pane ** abContainerPane, MWContext * context, MSG_Master * master)
|
||
|
{
|
||
|
AB_ContainerPane::Create(abContainerPane, context, master);
|
||
|
if (abContainerPane)
|
||
|
return AB_SUCCESS;
|
||
|
else
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_InitializeContainerPane(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->Initialize();
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API MSG_ViewIndex AB_GetIndexForContainer(MSG_Pane * pane /* container pane */, AB_ContainerInfo * container)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->GetIndexForContainer(container);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API AB_ContainerInfo * AB_GetContainerForIndex(MSG_Pane * pane, const MSG_ViewIndex index)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->GetContainerForIndex(index);
|
||
|
else
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->GetContainerForIndex(index);
|
||
|
}
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
/*****************************************************************************************************
|
||
|
Getting/Setting container properties on a pane or directly on the container
|
||
|
*****************************************************************************************************/
|
||
|
|
||
|
AB_API int AB_GetContainerAttributeForPane(MSG_Pane * pane, MSG_ViewIndex index,
|
||
|
AB_ContainerAttribute attrib, AB_ContainerAttribValue ** value)
|
||
|
{
|
||
|
uint16 numItems = 1;
|
||
|
return AB_GetContainerAttributesForPane(pane, index, &attrib, value, &numItems);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetContainerAttribute(AB_ContainerInfo * container, AB_ContainerAttribute attrib, AB_ContainerAttribValue ** value)
|
||
|
{
|
||
|
uint16 numItems = 1;
|
||
|
return AB_GetContainerAttributes(container, &attrib, value, &numItems);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetContainerAttributes(AB_ContainerInfo * container, AB_ContainerAttribute * attribsArray, AB_ContainerAttribValue ** valuesArray,
|
||
|
uint16 * numItems)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->GetAttributes(attribsArray, valuesArray, numItems);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetContainerAttributesForPane(MSG_Pane * pane, MSG_ViewIndex index,AB_ContainerAttribute * attribsArray,
|
||
|
AB_ContainerAttribValue ** valuesArray, uint16 * numItems)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->GetContainerAttributes(index, attribsArray, valuesArray, numItems);
|
||
|
else
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->GetContainerAttributes(index, attribsArray, valuesArray, numItems);
|
||
|
}
|
||
|
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetContainerAttributeForPane(MSG_Pane * pane, MSG_ViewIndex index, AB_ContainerAttribValue * value)
|
||
|
{
|
||
|
return AB_SetContainerAttributesForPane(pane, index, value, 1);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetContainerAttribute(AB_ContainerInfo * container, AB_ContainerAttribValue * value)
|
||
|
{
|
||
|
return AB_SetContainerAttributes(container, value, 1);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetContainerAttributes(AB_ContainerInfo * container, AB_ContainerAttribValue * valuesArray, uint16 numItems)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->SetAttributes(valuesArray, numItems);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
|
||
|
AB_API int AB_SetContainerAttributesForPane(MSG_Pane * pane, MSG_ViewIndex index, AB_ContainerAttribValue * valuesArray,uint16 numItems)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->SetContainerAttributes(index,valuesArray, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FreeContainerAttribValues(AB_ContainerAttribValue * valuesArray, uint16 numItems)
|
||
|
{
|
||
|
if (valuesArray)
|
||
|
{
|
||
|
for (int i = 0; i < numItems; i++)
|
||
|
{
|
||
|
if ( AB_IsStringContainerAttribValue(&valuesArray[i]) && XP_STRLEN(valuesArray[i].u.string) > 0 )
|
||
|
XP_FREEIF(valuesArray[i].u.string); // we can have a string attribute & it doesn't have a value
|
||
|
}
|
||
|
XP_FREE(valuesArray);
|
||
|
}
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FreeContainerAttribValue(AB_ContainerAttribValue * value)
|
||
|
{
|
||
|
return AB_FreeContainerAttribValues(value, 1);
|
||
|
}
|
||
|
|
||
|
AB_API XP_Bool AB_IsStringContainerAttribValue(AB_ContainerAttribValue * value)
|
||
|
{
|
||
|
// only known string attribs are: name
|
||
|
if (value)
|
||
|
if (value->attrib == attribName)
|
||
|
return TRUE;
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetNumRootContainers(MSG_Pane * pane, int32 * numRootContainers) /* FE allocated, BE fills */
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->GetNumRootContainers(numRootContainers);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetOrderedRootContainers(MSG_Pane * pane, AB_ContainerInfo ** ctrArray, /* FE allocated & freed */
|
||
|
int32 * numCtrs)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->GetOrderedRootContainers(ctrArray, numCtrs);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
|
||
|
AB_API DIR_Server * AB_GetDirServerForContainer(AB_ContainerInfo * container)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->GetDIRServer();
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_UpdateDIRServerForContainer(AB_ContainerInfo * abContainer)
|
||
|
{
|
||
|
if (abContainer)
|
||
|
return abContainer->UpdateDIRServer();
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_UpdateDIRServerForContainerPane(MSG_Pane * pane, DIR_Server * directory)
|
||
|
{
|
||
|
AB_ContainerPane * abcPane = CastABContainerPane(pane);
|
||
|
if (abcPane)
|
||
|
return abcPane->UpdateDIRServer(directory);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
/***************************************************************************************************
|
||
|
|
||
|
APIs for getting column information for containers and the picker pane...
|
||
|
|
||
|
****************************************************************************************************/
|
||
|
|
||
|
AB_API AB_ColumnInfo * AB_GetColumnInfo (AB_ContainerInfo * container, AB_ColumnID columnID)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->GetColumnInfo(columnID);
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetColumnAttribIDs(AB_ContainerInfo * container, AB_AttribID * attribIDs, int * numAttribs)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->GetColumnAttribIDs(attribIDs, numAttribs);
|
||
|
else
|
||
|
{
|
||
|
if (numAttribs)
|
||
|
*numAttribs = 0;
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetNumColumnsForContainer(AB_ContainerInfo * container)
|
||
|
{
|
||
|
if (container)
|
||
|
return container->GetNumColumnsForContainer();
|
||
|
else
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* we also have pane versions which are used by the picker pane */
|
||
|
AB_API AB_ColumnInfo * AB_GetColumnInfoForPane(MSG_Pane * pane,AB_ColumnID columnID)
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->GetColumnInfo(columnID);
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetNumColumnsForPane(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->GetNumColumnsForPicker();
|
||
|
else
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetColumnAttribIDsForPane(MSG_Pane * pane, AB_AttribID * attribIDs, int * numAttribs)
|
||
|
{
|
||
|
AB_PickerPane * abPickerPane = CastABPickerPane(pane);
|
||
|
if (abPickerPane)
|
||
|
return abPickerPane->GetColumnAttribIDs(attribIDs, numAttribs);
|
||
|
else
|
||
|
{
|
||
|
if (numAttribs)
|
||
|
*numAttribs = 0;
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FreeColumnInfo(AB_ColumnInfo * columnInfo)
|
||
|
{
|
||
|
/* right now, I'm treating the display string as the resource string..no need to free it. This might change!!! */
|
||
|
if (columnInfo)
|
||
|
{
|
||
|
if (columnInfo->displayString && XP_STRLEN(columnInfo->displayString) > 0)
|
||
|
XP_FREE(columnInfo->displayString);
|
||
|
|
||
|
XP_FREE(columnInfo);
|
||
|
}
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
/*****************************************************************************************************
|
||
|
Getting/Setting entry properties on a pane or directly on the container
|
||
|
*****************************************************************************************************/
|
||
|
|
||
|
AB_API int AB_GetEntryAttributeForPane(MSG_Pane * pane, MSG_ViewIndex index, AB_AttribID attrib, AB_AttributeValue ** value)
|
||
|
{
|
||
|
uint16 numItems = 1;
|
||
|
return AB_GetEntryAttributesForPane(pane, index, &attrib, value, &numItems);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetEntryAttribute(AB_ContainerInfo * abContainer, ABID entryID, AB_AttribID attrib, AB_AttributeValue ** value)
|
||
|
{
|
||
|
uint16 numItems = 1;
|
||
|
return AB_GetEntryAttributes(abContainer,entryID,&attrib, value, &numItems);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetEntryAttributes(AB_ContainerInfo * abContainer, ABID entryID, AB_AttribID * attribs,
|
||
|
AB_AttributeValue ** values, uint16 * numItems)
|
||
|
{
|
||
|
if (abContainer)
|
||
|
return abContainer->GetEntryAttributes(NULL, entryID, attribs, values, numItems);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetEntryAttributesForPane(MSG_Pane * pane, MSG_ViewIndex index, AB_AttribID * attribs,
|
||
|
AB_AttributeValue ** values, uint16 * numItems)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->GetEntryAttributes(index, attribs, values, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetEntryAttributeForPane(MSG_Pane * pane, MSG_ViewIndex index, AB_AttributeValue * value)
|
||
|
{
|
||
|
return AB_SetEntryAttributesForPane(pane, index, value, 1);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetEntryAttribute(AB_ContainerInfo * abContainer, ABID entryID, AB_AttributeValue * value)
|
||
|
{
|
||
|
return AB_SetEntryAttributes(abContainer, entryID, value, 1);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetEntryAttributes(AB_ContainerInfo * abContainer, ABID entryID, AB_AttributeValue * valuesArray, uint16 numItems)
|
||
|
{
|
||
|
if (abContainer)
|
||
|
return abContainer->SetEntryAttributes(entryID, valuesArray, numItems);
|
||
|
else
|
||
|
return AB_INVALID_CONTAINER;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetEntryAttributesForPane(MSG_Pane * pane, MSG_ViewIndex index, AB_AttributeValue * valuesArray, uint16 numItems)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->SetEntryAttributes(index, valuesArray, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FreeEntryAttributeValue(AB_AttributeValue * value)
|
||
|
{
|
||
|
return AB_FreeEntryAttributeValues(value, 1);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FreeEntryAttributeValues(AB_AttributeValue * values, uint16 numItems)
|
||
|
{
|
||
|
if (values)
|
||
|
{
|
||
|
for (int i = 0; i < numItems; i++)
|
||
|
{
|
||
|
if ( AB_IsStringEntryAttributeValue(&values[i]) && values[i].u.string)
|
||
|
XP_FREEIF(values[i].u.string); // we can have a string attribute & it doesn't have a value
|
||
|
}
|
||
|
XP_FREE(values);
|
||
|
}
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
AB_API XP_Bool AB_IsStringEntryAttributeValue(AB_AttributeValue * value)
|
||
|
{
|
||
|
if (value)
|
||
|
{
|
||
|
if (value->attrib == AB_attribUseServer || value->attrib == AB_attribEntryID ||
|
||
|
value->attrib == AB_attribEntryType || value->attrib == AB_attribHTMLMail ||
|
||
|
value->attrib == AB_attribSecurity || value->attrib == AB_attribWinCSID)
|
||
|
return FALSE;
|
||
|
else
|
||
|
return TRUE;
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_CopyEntryAttributeValue(AB_AttributeValue * srcValue, AB_AttributeValue * destValue)
|
||
|
{
|
||
|
if (srcValue && destValue)
|
||
|
{
|
||
|
destValue->attrib = srcValue->attrib;
|
||
|
switch(srcValue->attrib) // maintenance note: add cases for any non-string attribute!!
|
||
|
{
|
||
|
case AB_attribEntryType:
|
||
|
destValue->u.entryType = srcValue->u.entryType;
|
||
|
break;
|
||
|
case AB_attribHTMLMail:
|
||
|
destValue->u.boolValue = srcValue->u.boolValue;
|
||
|
break;
|
||
|
case AB_attribSecurity:
|
||
|
case AB_attribWinCSID:
|
||
|
case AB_attribUseServer:
|
||
|
destValue->u.shortValue = srcValue->u.shortValue;
|
||
|
break;
|
||
|
|
||
|
default: // treat string (most common case) as the default
|
||
|
if (srcValue->u.string && XP_STRLEN(srcValue->u.string) > 0)
|
||
|
destValue->u.string = XP_STRDUP(srcValue->u.string);
|
||
|
else
|
||
|
destValue->u.string = XP_STRDUP("");
|
||
|
}
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
/* FE asked for an attribute we don't have. This routine fills the attribute with an appropriate
|
||
|
default value for the attribut. I expect this to be called only on the XP side of the world */
|
||
|
AB_API int AB_CopyDefaultAttributeValue(AB_AttribID attrib, AB_AttributeValue * value /* already allocated */)
|
||
|
{
|
||
|
if (value)
|
||
|
{
|
||
|
value->attrib = attrib;
|
||
|
switch(attrib)
|
||
|
{
|
||
|
case AB_attribEntryType:
|
||
|
value->u.entryType = AB_Person;
|
||
|
break;
|
||
|
case AB_attribHTMLMail:
|
||
|
value->u.boolValue = FALSE;
|
||
|
break;
|
||
|
case AB_attribSecurity:
|
||
|
case AB_attribWinCSID:
|
||
|
case AB_attribUseServer:
|
||
|
value->u.shortValue = 0;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
XP_ASSERT(AB_IsStringEntryAttributeValue(value)); // Everything else should be a string!
|
||
|
value->u.string = XP_STRDUP("");
|
||
|
}
|
||
|
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
/* caller must free array of values returned */
|
||
|
/* This function is used to generate a set of atributes which can be committed into the database for a naked address */
|
||
|
AB_API int AB_CreateAttributeValuesForNakedAddress(char * nakedAddress, AB_AttributeValue ** valueArray, uint16 * numItems)
|
||
|
{
|
||
|
uint16 numAttributes = 4; // right now we store: entryType, full name, display name, email address
|
||
|
int status = AB_SUCCESS;
|
||
|
char * stringToUse;
|
||
|
|
||
|
if (nakedAddress)
|
||
|
stringToUse = nakedAddress;
|
||
|
else
|
||
|
stringToUse = "";
|
||
|
|
||
|
AB_AttributeValue * values = (AB_AttributeValue *) XP_ALLOC(sizeof(AB_AttributeValue) * numAttributes);
|
||
|
if (values)
|
||
|
{
|
||
|
values[0].attrib = AB_attribEntryType;
|
||
|
values[0].u.entryType = AB_Person;
|
||
|
|
||
|
values[1].attrib = AB_attribFullName;
|
||
|
values[1].u.string = XP_STRDUP(stringToUse);
|
||
|
|
||
|
values[2].attrib = AB_attribDisplayName;
|
||
|
values[2].u.string = XP_STRDUP(stringToUse);
|
||
|
|
||
|
values[3].attrib = AB_attribEmailAddress;
|
||
|
values[3].u.string = XP_STRDUP(stringToUse);
|
||
|
}
|
||
|
else
|
||
|
status = AB_OUT_OF_MEMORY;
|
||
|
|
||
|
if (valueArray)
|
||
|
*valueArray = values;
|
||
|
|
||
|
if (numItems)
|
||
|
*numItems = numAttributes;
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetAttributesForNakedAddress(char * nakedAddress, AB_AttribID * attribs, AB_AttributeValue ** valuesArray,uint16 * numItems)
|
||
|
{
|
||
|
uint16 numItemsAdded = 0;
|
||
|
AB_AttributeValue * values = NULL;
|
||
|
int status = AB_FAILURE;
|
||
|
|
||
|
AB_AddressParser * parser = NULL; // used to produce the email attribute
|
||
|
|
||
|
if (attribs && numItems && *numItems > 0)
|
||
|
{
|
||
|
uint16 numAttributes = *numItems; // right now we store: entryType, full name, display name, email address
|
||
|
values = (AB_AttributeValue *) XP_ALLOC(sizeof(AB_AttributeValue) * numAttributes);
|
||
|
if (values)
|
||
|
{
|
||
|
for (uint16 index = 0; index < numAttributes; index++)
|
||
|
{
|
||
|
if (nakedAddress)
|
||
|
{
|
||
|
values[index].attrib = attribs[index];
|
||
|
|
||
|
switch (attribs[index])
|
||
|
{
|
||
|
// naked addresses only have a couple values that we acknowledge..
|
||
|
case AB_attribEntryType:
|
||
|
values[index].u.entryType = AB_Person;
|
||
|
break;
|
||
|
case AB_attribEmailAddress:
|
||
|
values[index].u.string = NULL;
|
||
|
AB_AddressParser::Create (NULL, &parser);
|
||
|
if (parser)
|
||
|
{
|
||
|
parser->FormatNakedAddress(&values[index].u.string, nakedAddress,TRUE);
|
||
|
parser->Release();
|
||
|
parser = NULL;
|
||
|
}
|
||
|
|
||
|
if (values[index].u.string) // if we actually have a value, break out of the looop...
|
||
|
break;
|
||
|
// we used to use the naked address name as the value for full name and display name.
|
||
|
// However, this doesn't really make sense because the naked address should be treated
|
||
|
// as an email address, NOT the full/disply name. This is most notable in the name completion
|
||
|
// pane where we display a naked address.....
|
||
|
default:
|
||
|
AB_CopyDefaultAttributeValue(attribs[index], &values[index]);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
AB_CopyDefaultAttributeValue(attribs[index], &values[index]);
|
||
|
} // for loop
|
||
|
|
||
|
status = AB_SUCCESS;
|
||
|
numItemsAdded = numAttributes;
|
||
|
}
|
||
|
else
|
||
|
status = AB_OUT_OF_MEMORY;
|
||
|
}
|
||
|
|
||
|
if (numItems)
|
||
|
*numItems = numItemsAdded;
|
||
|
if (valuesArray)
|
||
|
*valuesArray = values;
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
|
||
|
/****************************************************************************************
|
||
|
|
||
|
Sorting - what attribute should we sort by. Are things sorted by first name or last name?
|
||
|
|
||
|
*****************************************************************************************/
|
||
|
|
||
|
AB_API int AB_SortByAttribute(MSG_Pane * pane, AB_AttribID id, XP_Bool sortAscending)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->SortByAttribute(id, sortAscending);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetPaneSortedByAB2(MSG_Pane * pane, AB_AttribID * id)
|
||
|
{
|
||
|
AB_Pane * abPane = CastABPane(pane);
|
||
|
if (abPane)
|
||
|
return abPane->GetPaneSortedBy(id);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
/***************************************************************************************
|
||
|
Mailing List Pane APIs
|
||
|
***************************************************************************************/
|
||
|
|
||
|
AB_API int AB_InitializeMailingListPaneAB2 (MSG_Pane * pane)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->Initialize();
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API AB_ContainerInfo * AB_GetContainerForMailingList(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->GetContainerForMailingList();
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
AB_API ABID AB_GetABIDForMailingListIndex(MSG_Pane * pane, const MSG_ViewIndex index)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->GetABIDForIndex(index);
|
||
|
else
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* Use these two APIs to add entries to the mailing list that are the result of name completion. */
|
||
|
AB_API int AB_AddNakedEntryToMailingList(MSG_Pane * pane,const MSG_ViewIndex index, const char * nakedAddress,XP_Bool ReplaceExisting)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->AddEntry(index, nakedAddress, ReplaceExisting);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_AddNCEntryToMailingList(MSG_Pane * pane, const MSG_ViewIndex index, AB_NameCompletionCookie * cookie, XP_Bool ReplaceExisting)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->AddEntry(index, cookie, ReplaceExisting);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API MSG_ViewIndex AB_GetMailingListIndexForABID(MSG_Pane * pane, ABID entryID)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->GetIndexForABID(entryID);
|
||
|
else
|
||
|
return MSG_VIEWINDEXNONE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetMailingListEntryAttributes(MSG_Pane * pane, const MSG_ViewIndex index, AB_AttributeValue * values,
|
||
|
uint16 numItems)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->SetEntryAttributes(index, values, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetMailingListEntryAttributes(MSG_Pane * pane, const MSG_ViewIndex index, AB_AttribID * attribs,
|
||
|
AB_AttributeValue ** values, uint16 * numItems)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->GetEntryAttributes(index, attribs, values, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetMailingListAttributes(MSG_Pane * pane, AB_AttribID * attribs, AB_AttributeValue ** values,
|
||
|
uint16 * numItems)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->GetAttributes(attribs, values, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetMailingListAttributes(MSG_Pane * pane, AB_AttributeValue * valuesArray, uint16 numItems)
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->SetAttributes(valuesArray, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
/******************************************************************************************
|
||
|
Person Entry Pane aka person property sheet pane. Person entry panes already respond to the
|
||
|
AB_ClosePane API.
|
||
|
*******************************************************************************************/
|
||
|
|
||
|
AB_API AB_ContainerInfo * AB_GetContainerForPerson(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_PersonPane * personPane = CastABPersonPane(pane);
|
||
|
if (personPane)
|
||
|
return personPane->GetContainerForPerson();
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
AB_API ABID AB_GetABIDForPerson(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_PersonPane * personPane = CastABPersonPane(pane);
|
||
|
if (personPane)
|
||
|
return personPane->GetABIDForPerson();
|
||
|
else
|
||
|
return 0; // probably not a good return value. I should change this..but to what???
|
||
|
}
|
||
|
|
||
|
AB_API int AB_SetPersonEntryAttributes(MSG_Pane * pane, AB_AttributeValue * valuesArray, uint16 numItems)
|
||
|
{
|
||
|
AB_PersonPane * personPane = CastABPersonPane(pane);
|
||
|
if (personPane)
|
||
|
return personPane->SetAttributes(valuesArray, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetPersonEntryAttributes(MSG_Pane * pane, AB_AttribID * attribs, AB_AttributeValue ** values, uint16 * numItems)
|
||
|
{
|
||
|
AB_PersonPane * personPane = CastABPersonPane(pane);
|
||
|
if (personPane)
|
||
|
return personPane->GetAttributes(attribs, values, numItems);
|
||
|
else
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_CommitChanges(MSG_Pane * pane)
|
||
|
{
|
||
|
AB_PersonPane * personPane = CastABPersonPane(pane);
|
||
|
if (personPane)
|
||
|
return personPane->CommitChanges();
|
||
|
else
|
||
|
{
|
||
|
AB_MailingListPane * mlPane = CastABMailingListPane(pane);
|
||
|
if (mlPane)
|
||
|
return mlPane->CommitChanges();
|
||
|
}
|
||
|
|
||
|
return AB_INVALID_PANE;
|
||
|
}
|
||
|
|
||
|
/* FEs should use this to obtain a property sheet pane representing the user's identity vcard...*/
|
||
|
AB_API int AB_GetIdentityPropertySheet(MWContext *context, MSG_Master * master, MSG_Pane ** pane)
|
||
|
{
|
||
|
// first, create the vcard identity and convert to attribute values..
|
||
|
AB_AttributeValue * identityValues = NULL;
|
||
|
uint16 numAttributes = 0;
|
||
|
char * identityVCard = NULL;
|
||
|
int status = AB_SUCCESS;
|
||
|
|
||
|
if (pane)
|
||
|
{
|
||
|
status = AB_LoadIdentityVCard(&identityVCard);
|
||
|
if (identityVCard)
|
||
|
{
|
||
|
status = AB_ConvertVCardToAttribValues(identityVCard,&identityValues,&numAttributes);
|
||
|
if (status == AB_SUCCESS) // so far so good...
|
||
|
status = AB_PersonPane::CreateIdentityPane(context, master, identityValues, numAttributes, pane);
|
||
|
|
||
|
AB_FreeEntryAttributeValues(identityValues, numAttributes);
|
||
|
XP_FREE(identityVCard);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************************
|
||
|
Various APIs that don't fit into a category...
|
||
|
|
||
|
******************************************************************************************/
|
||
|
|
||
|
AB_API int AB_SplitFullName (const char *fullName, char **firstName, char **lastName, int16 csid)
|
||
|
{
|
||
|
*firstName = XP_STRDUP (fullName);
|
||
|
if (NULL == *firstName)
|
||
|
return AB_OUT_OF_MEMORY;
|
||
|
|
||
|
char *plastSpace = *firstName;
|
||
|
char *walkName = *firstName;
|
||
|
char *plastName = NULL;
|
||
|
|
||
|
while (walkName && *walkName)
|
||
|
{
|
||
|
if (*walkName == ' ')
|
||
|
{
|
||
|
plastSpace = walkName;
|
||
|
plastName = INTL_NextChar(csid, plastSpace);
|
||
|
}
|
||
|
walkName = INTL_NextChar(csid, walkName);
|
||
|
}
|
||
|
|
||
|
if (plastName)
|
||
|
{
|
||
|
*plastSpace = '\0';
|
||
|
*lastName = XP_STRDUP (plastName);
|
||
|
}
|
||
|
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
/* use the default display name API for generating a display name in the person property sheet card for new users */
|
||
|
AB_API int AB_GenerateDefaultDisplayName(const char * firstName, const char * lastName, char ** defaultDisplayName /* calller must free with XP_FREE */)
|
||
|
{
|
||
|
char * displayName = NULL;
|
||
|
XP_Bool lastFirst = FALSE; /* need to read this in from prefs... */
|
||
|
|
||
|
PREF_GetBoolPref("mail.addr_book.lastnamefirst", &lastFirst);
|
||
|
|
||
|
if (!lastFirst)
|
||
|
{
|
||
|
if (firstName)
|
||
|
{
|
||
|
NET_SACat(&displayName, firstName);
|
||
|
if (lastName) /* if we are going to have a last name, add a space for it..*/
|
||
|
NET_SACat(&displayName, XP_GetString(MK_ADDR_FIRST_LAST_SEP));
|
||
|
}
|
||
|
|
||
|
if (lastName)
|
||
|
NET_SACat(&displayName, lastName);
|
||
|
}
|
||
|
else /* should be treated as Last, First */
|
||
|
{
|
||
|
if (lastName)
|
||
|
{
|
||
|
NET_SACat(&displayName, lastName);
|
||
|
if (firstName)
|
||
|
NET_SACat(&displayName, ", ");
|
||
|
}
|
||
|
|
||
|
if (firstName)
|
||
|
NET_SACat(&displayName, firstName);
|
||
|
}
|
||
|
|
||
|
if (defaultDisplayName)
|
||
|
*defaultDisplayName = displayName;
|
||
|
else
|
||
|
XP_FREEIF(displayName);
|
||
|
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
AB_API XP_List * AB_AcquireAddressBookContainers(MWContext * context)
|
||
|
/* creates a list of reference counted containers for all the address books. No LDAP. */
|
||
|
/* Caller must later release the list using AB_ReleaseContainerList */
|
||
|
{
|
||
|
int i = 0;
|
||
|
|
||
|
XP_List * abServers = XP_ListNew();
|
||
|
XP_List * containers = XP_ListNew();
|
||
|
XP_List * servers = DIR_GetDirServers();
|
||
|
|
||
|
DIR_GetPersonalAddressBooks(servers, abServers);
|
||
|
|
||
|
if (abServers)
|
||
|
{
|
||
|
for (i = 1; i <= XP_ListCount(abServers); i++)
|
||
|
{
|
||
|
DIR_Server * server = (DIR_Server *) XP_ListGetObjectNum(abServers, i);
|
||
|
if (server)
|
||
|
{
|
||
|
AB_ContainerInfo * ctr = NULL;
|
||
|
AB_ContainerInfo::Create(context, server, &ctr);
|
||
|
if (ctr)
|
||
|
XP_ListAddObjectToEnd(containers, ctr);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return containers;
|
||
|
}
|
||
|
|
||
|
|
||
|
AB_API int AB_ReleaseContainersList(XP_List * containers)
|
||
|
/* releases each container and destroys the list */
|
||
|
{
|
||
|
int i = 0;
|
||
|
if (containers)
|
||
|
{
|
||
|
for (i = 1; i <= XP_ListCount(containers); i++)
|
||
|
{
|
||
|
AB_ContainerInfo * ctr = (AB_ContainerInfo *) XP_ListGetObjectNum (containers, i);
|
||
|
if (ctr)
|
||
|
ctr->Release();
|
||
|
}
|
||
|
|
||
|
XP_ListDestroy (containers);
|
||
|
}
|
||
|
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_AcquireContainer(AB_ContainerInfo * container)
|
||
|
{
|
||
|
int status = AB_SUCCESS;
|
||
|
if (container)
|
||
|
container->Acquire();
|
||
|
else
|
||
|
status = ABError_NullPointer;
|
||
|
|
||
|
return status;
|
||
|
|
||
|
}
|
||
|
|
||
|
AB_API int AB_ReleaseContainer(AB_ContainerInfo * container)
|
||
|
{
|
||
|
int status = AB_SUCCESS;
|
||
|
if (container)
|
||
|
container->Release();
|
||
|
else
|
||
|
status = ABError_NullPointer;
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_FreeAddressBookCopyInfo(AB_AddressBookCopyInfo * abCopyInfo) /* use this to free copy info structs */
|
||
|
{
|
||
|
int status = AB_SUCCESS;
|
||
|
|
||
|
if (abCopyInfo)
|
||
|
{
|
||
|
if (abCopyInfo->destContainer)
|
||
|
AB_ReleaseContainer(abCopyInfo->destContainer);
|
||
|
if (abCopyInfo->srcContainer)
|
||
|
AB_ReleaseContainer(abCopyInfo->srcContainer);
|
||
|
if (abCopyInfo->idArray)
|
||
|
XP_FREE(abCopyInfo->idArray);
|
||
|
|
||
|
XP_FREE(abCopyInfo);
|
||
|
}
|
||
|
else
|
||
|
status = ABError_NullPointer;
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
|
||
|
/******************************************************************************************
|
||
|
Replication APIs
|
||
|
*******************************************************************************************/
|
||
|
AB_API AB_ContainerInfo *AB_BeginReplication(MWContext *context, DIR_Server *server)
|
||
|
{
|
||
|
return AB_LDAPContainerInfo::BeginReplication(context, server);
|
||
|
}
|
||
|
|
||
|
AB_API int AB_EndReplication(AB_ContainerInfo *container)
|
||
|
{
|
||
|
AB_LDAPContainerInfo *ldapContainer = container ? container->GetLDAPContainerInfo() : NULL;
|
||
|
if (ldapContainer)
|
||
|
{
|
||
|
ldapContainer->EndReplication();
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_AddReplicaEntry(AB_ContainerInfo *container, char **valueList)
|
||
|
{
|
||
|
AB_LDAPContainerInfo *ldapContainer = container ? container->GetLDAPContainerInfo() : NULL;
|
||
|
if (ldapContainer && ldapContainer->AddReplicaEntry(valueList))
|
||
|
return AB_SUCCESS;
|
||
|
else
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_DeleteReplicaEntry(AB_ContainerInfo *container, char *targetDn)
|
||
|
{
|
||
|
AB_LDAPContainerInfo *ldapContainer = container ? container->GetLDAPContainerInfo() : NULL;
|
||
|
if (ldapContainer)
|
||
|
{
|
||
|
ldapContainer->DeleteReplicaEntry(targetDn);
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_RemoveReplicaEntries(AB_ContainerInfo *container)
|
||
|
{
|
||
|
AB_LDAPContainerInfo *ldapContainer = container ? container->GetLDAPContainerInfo() : NULL;
|
||
|
if (ldapContainer)
|
||
|
{
|
||
|
ldapContainer->RemoveReplicaEntries();
|
||
|
return AB_SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
return AB_FAILURE;
|
||
|
}
|
||
|
|
||
|
AB_API int AB_GetNumReplicaAttributes(AB_ContainerInfo *container)
|
||
|
{
|
||
|
AB_LDAPContainerInfo *ldapContainer = container ? container->GetLDAPContainerInfo() : NULL;
|
||
|
if (ldapContainer)
|
||
|
{
|
||
|
AB_ReplicaAttributeMap *attribMap = ldapContainer->GetReplicaAttributeMap();
|
||
|
|
||
|
if (attribMap)
|
||
|
return attribMap->numAttribs;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
AB_API char **AB_GetReplicaAttributeNames(AB_ContainerInfo *container)
|
||
|
{
|
||
|
AB_LDAPContainerInfo *ldapContainer = container ? container->GetLDAPContainerInfo() : NULL;
|
||
|
if (ldapContainer)
|
||
|
{
|
||
|
AB_ReplicaAttributeMap *attribMap = ldapContainer->GetReplicaAttributeMap();
|
||
|
|
||
|
if (attribMap)
|
||
|
return attribMap->attribNames;
|
||
|
}
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
AB_API XP_Bool AB_ReplicaAttributeMatchesId(AB_ContainerInfo *container, int attribIndex, DIR_AttributeId id)
|
||
|
{
|
||
|
AB_LDAPContainerInfo *ldapContainer = container ? container->GetLDAPContainerInfo() : NULL;
|
||
|
if (ldapContainer)
|
||
|
{
|
||
|
AB_ReplicaAttributeMap *attribMap = ldapContainer->GetReplicaAttributeMap();
|
||
|
|
||
|
if (attribMap && attribIndex < attribMap->numAttribs)
|
||
|
return ldapContainer->ReplicaAttributeMatchesId(attribMap, attribIndex, id);
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|