dmose%mozilla.org 0efb7c174c updated xPL license boilerplate to v1.1, a=chofmann@netscape.com,r=endico@mozilla.org
git-svn-id: svn://10.0.0.236/trunk@52910 18797224-902f-48f8-a5cc-f745e15eee43
1999-11-06 03:43:54 +00:00

1018 lines
23 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.1 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.mozilla.org/NPL/
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code is mozilla.org code.
*
* The Initial Developer of the Original Code is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All
* Rights Reserved.
*
* Contributor(s):
*/
/*
* fb.cpp (FontBrokerObject.cpp)
*
* C++ implementation of the (fb) FontBrokerObject
*
* dp Suresh <dp@netscape.com>
*/
#include "fb.h"
// Since it creates these objects, it uses the local implementation of
// Objects.
// Create : Font
// FontMatchInfo
// RenderingContext
// FontObserver
#include "Mcf.h"
#include "Pcf.h"
#include "f.h"
#include "Mcfmi.h"
#include "Pcfmi.h"
#include "fmi.h"
#include "Mcrc.h"
#include "Pcrc.h"
#include "rc.h"
#include "Pcdoer.h"
// Uses: DoneObserver
#include "Mnfdoer.h"
#include "net.h" // libnet
#include "prio.h" // for reading directories
FontBrokerObject::
FontBrokerObject() : fpPeers(NULL), fpPeersFromCatalog(NULL)
{
// Enable webfonts by default
pref.enableWebfonts = 1;
}
FontBrokerObject::
~FontBrokerObject()
{
}
//
// FontBrokerConsumer Interface specific
//
struct nff * FontBrokerObject::
LookupFont(struct nfrc *rc, struct nffmi *fmi, const char *accessor)
{
struct nff *f = NULL;
// Try to find a Font Object that we already created
// Wondering why the accessor is not involved in the match.
// All webfonts are NOT shared. Only local fonts are shared. And local
// fonts can be accessed by any accessor.
f = fontObjectCache.RcFmi2Font(rc, fmi);
if (f)
{
WF_TRACEMSG(("NF: Found font object 0x%x for fmi (%s) in cache.",
f, nffmi_toString(fmi, NULL)));
return (f);
}
// On each enabled FontDisplayer, do a LookupFont() and use the
// returned FontHandles to make a Font object.
struct wfListElement *tmp = fpPeers.head;
for(; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject *fpp = (FontDisplayerPeerObject *) tmp->item;
void *fh = NULL;
//
// LookupFont() on a displayer will not be done if
// displayer's catalog does not support the {rc, fmi}
// and the displayer is not loaded.
// If the fpp was already loaded, ask it anyway.
// The logic is asking the catalog will save us from loading
// the fpp. But if the fpp is already loaded and in memory,
// then this operation is a relatively cheap operation. Webfonts
// rely on this. Once a font is streamed, anymore lookup fonts
// need to go the webfont displayer too as we dont know that facenames
// the streamed fron contains.
//
// *** The native displayer will always be asked.
//
if (!fpp->isNative())
{
if (pref.enableWebfonts <= 0)
{
// If webfonts are disabled, always believe the catalog.
if (!fpp->queryCatalog(rc, fmi))
{
continue;
}
}
else
{
// If webfonts are enabled, then the special cases are
// 1. fpp already loaded
// In this case, we always ask it. Catalog is not consulted.
// 2. fpp gets loaded in the process of initializing the catalog
// In this case, we ask it since it got loaded.
if (!fpp->isLoaded() && !fpp->queryCatalog(rc, fmi) && !fpp->isLoaded())
{
continue;
}
}
}
fh = fpp->LookupFont(rc, fmi, accessor);
if (fh)
{
if (!f)
{
f = (struct nff *) cfFactory_Create(NULL, rc);
if (!f)
{
// Error.
return (NULL);
}
}
cfImpl *fimpl = cf2cfImpl(f);
FontObject *fobj = (FontObject *) fimpl->object;
XP_ASSERT(fobj);
fobj->addFontHandle(fpp, fh);
}
}
if (f)
{
WF_TRACEMSG(("NF: Created new font object 0x%x for fmi (%s).",
f, nffmi_toString(fmi, NULL)));
// The font needs to get added to the Font<-->Fmi map
// We have to do this as the font object cache is a central list of
// font objects.
fontObjectCache.add(fmi, f);
}
return (f);
}
struct nff *
/*ARGSUSED*/
FontBrokerObject::CreateFontFromUrl(struct nfrc* rc, const char* url_of_font,
const char *url_of_page, jint faux,
struct nfdoer* observer, MWContext *context)
{
struct nff *f = NULL;
// Decide if we want to pull the stream in
// We will not pull the stream under these conditions
// 1. webfonts is OFF in the preference
// 2. a font of this name was already created by us
if (pref.enableWebfonts <= 0)
{
// webfonts is off. Dont do font streaming.
return (NULL);
}
// We should not reuse webfonts. This is because a webfont created for
// one page should not be usable to another page according to
// webfont security. So we will always create these fresh.
//
// Start a stream download of the url.
//
URL_Struct *url = NET_CreateURLStruct(url_of_font, NET_NORMAL_RELOAD);
if (!url)
{
// Error. Cannot create url struct.
WF_TRACEMSG(("NF: Error: Cannot create url struct."));
return (NULL);
}
struct wf_new_stream_data *data = new wf_new_stream_data;
if (!data)
{
// Error. No memory to create wf_new_stream_data.
WF_TRACEMSG(("NF: Error: Cannot create wf_new_stream_data."));
NET_FreeURLStruct(url);
return (NULL);
}
// Create a web FontObject
f = (struct nff *) cfFactory_Create(NULL, rc);
if (!f)
{
// Error.
WF_TRACEMSG(("NF: Error: Cannot create Font nff."));
NET_FreeURLStruct(url);
return (NULL);
}
// The fontobject that was created was wrong. We will destroy it and
// put a new correct fontobject
cfImpl *fimpl = cf2cfImpl(f);
FontObject *fobj = (FontObject *) fimpl->object;
delete fobj;
fobj = new FontObject(f, rc, url->address);
fimpl->object = fobj;
nfdoer_addRef(observer, NULL);
data->observer = observer;
nff_addRef(f, NULL);
data->f = f;
nfrc_addRef(rc, NULL);
data->rc = rc;
data->url_of_page = CopyString(url_of_page);
url->fe_data = data;
//art NET_GetURL(url, FO_CACHE_AND_FONT, NULL, wfUrlExit);
NET_GetURL(url, FO_CACHE_AND_FONT, context,(Net_GetUrlExitFunc*) wfUrlExit);
return (f);
}
const char *
FontBrokerObject::GetMimetype(const char *imimetype, const char *address)
{
// If the mimetype was not known, we will try to figure out the mimetype
// from the extension. This is very tricky. We need to define two things:
// 1. How do we know the mimetype was not known. Let us declare that all
// these mimetypes means the server doesn't know the mimetype
// application/x-unknown
// text/plain
// text/html
//
// 2. What is the extension ?
// We will define it as any string after the last dot in the url.
// That means for http://www.nescape.com/a.tar.gz, we will only
// recognize "gz" as the extension and not "tar.gz"
const char *mimetype = imimetype;
if (mimetype &&
(!wf_strcasecmp(mimetype, "text/plain") ||
!wf_strcasecmp(mimetype, "text/html") ||
!wf_strncasecmp(mimetype, "application/x-unknown", 21)))
{
mimetype = NULL;
}
if (!mimetype)
{
const char *extension = wf_getExtension(address);
struct wfListElement *tmp = fpPeers.head;
for(; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject *fpp = (FontDisplayerPeerObject *) tmp->item;
mimetype = fpp->getMimetypeFromExtension(extension);
if (mimetype && *mimetype)
{
break;
}
}
if (!mimetype)
{
// Noway of finding a mimetype for this url.
// Just use the original and see what happens.
mimetype = imimetype;
}
}
return (mimetype);
}
struct nff *
/*ARGSUSED*/
FontBrokerObject::CreateFontFromFile(struct nfrc* rc, const char *mimetype,
const char* fontFilename,
const char *url_of_page)
{
// Decide if we want to pull the stream in
// We will not pull the stream under these conditions
// 1. webfonts is OFF in the preference
// 2. a font of this name was already created by us
if (pref.enableWebfonts <= 0)
{
// webfonts is off. Dont do font streaming.
return (NULL);
}
struct nff *f = NULL;
// We should not reuse webfonts. This is because a webfont created for
// one page should not be usable to another page according to
// webfont security. So we will always create these fresh.
mimetype = GetMimetype(mimetype, fontFilename);
if (!mimetype || !*mimetype)
{
return (NULL);
}
// Find which font Displayer implements the mimetype
struct wfListElement *tmp = fpPeers.head;
FontDisplayerPeerObject *fpp = NULL;
void *fh = NULL;
for(; tmp; tmp = tmp->next)
{
fpp = (FontDisplayerPeerObject *) tmp->item;
if (fpp->isMimetypeEnabled(mimetype) > 0 &&
(fh = fpp->CreateFontFromFile(rc, mimetype, fontFilename, url_of_page)))
{
f = (struct nff *) cfFactory_Create(NULL, rc);
if (!f)
{
// Error.
return (NULL);
}
// The fontobject that was created was wrong. We will destroy it and
// put a new correct fontobject
cfImpl *fimpl = cf2cfImpl(f);
FontObject *fobj = (FontObject *) fimpl->object;
delete fobj;
fobj = new FontObject(f, rc, fontFilename);
fimpl->object = fobj;
XP_ASSERT(fobj);
fobj->addFontHandle(fpp, fh);
break;
}
}
return (f);
}
//
// Ideally these catalog methods could be cached...
//
struct nffmi ** FontBrokerObject::
ListFonts(struct nfrc *rc, struct nffmi *fmi)
{
struct nffmi ** cumulativeFmiList = NULL;
int n = 0;
struct wfListElement *tmp = fpPeers.head;
for (; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject *fpp = (FontDisplayerPeerObject *) tmp->item;
struct nffmi ** fmiList = fpp->ListFonts(rc, fmi);
if (fmiList && fmiList[0])
{
if (cumulativeFmiList == NULL)
{
// This is the first time. Just initialize the
// cumulativeFmiList with the fmiList.
cumulativeFmiList = fmiList;
while (fmiList[n]) n++;
}
else
{
// Merge this list into the cumulativeFmiList
merge(cumulativeFmiList, n, fmiList);
// WF_FREE is nffbu::free(). So this is ok.
// We dont have to release each fmi as they have been
// copied into the cumulativeFmiList
WF_FREE(fmiList);}
}
}
return(cumulativeFmiList);
}
jdouble * FontBrokerObject::
ListSizes(struct nfrc *rc, struct nffmi *fmi)
{
jdouble * cumulativeSizeList = NULL;
int maxSizes = 0;
struct wfListElement *tmp = fpPeers.head;
for (; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject * fpp = (FontDisplayerPeerObject *) tmp->item;
jdouble* sizeList = fpp->ListSizes(rc, fmi);
if (sizeList && (sizeList[0] >= 0))
{
// Merge this list into the cumulativeFmiList
MergeSizes(cumulativeSizeList, maxSizes, sizeList);
// WF_FREE is nffbu::free(). So this is ok.
WF_FREE(sizeList);
}
}
return(cumulativeSizeList);
}
/*ARGSUSED*/
struct nff *
FontBrokerObject::
GetBaseFont(struct nfrf *rf)
{
return(fontObjectCache.Rf2Font(rf));
}
//
// FontBrokerDisplayer interface specific
//
jint FontBrokerObject::
RegisterFontDisplayer(struct nffp* fp)
{
FontDisplayerPeerObject *fpp = new FontDisplayerPeerObject(fp);
return (registerDisplayer(fpp));
}
//
// All displayer objects get created here. This checks the catalog and
// creates displayer only if the dlm_name isn't in the catalog (or) the
// dlm_name is newer than what we have in the catalog.
//
jint
FontBrokerObject::
CreateFontDisplayerFromDLM(const char* dlm_name)
{
int ret = 0;
FontDisplayerPeerObject * fpp = NULL;
// Check the catalog to see if this dlm_name is present
struct wfListElement *tmp = fpPeersFromCatalog.head;
for (; tmp; tmp = tmp->next)
{
fpp = (FontDisplayerPeerObject *) tmp->item;
int dlmchanged = fpp->dlmChanged(dlm_name);
if (dlmchanged >= 0)
{
// Found the dlm in the catalog. Use it.
// 1. remove the fpp from the fpPeersFromCatalog list
fpPeersFromCatalog.remove(fpp);
// 2. If the fpp has changed, resync it.
if (dlmchanged > 0)
{
fpp->resync();
}
break;
}
}
if (!tmp)
{
// Didn't find the dlm_name in the catalog. Create a new one.
fpp = new FontDisplayerPeerObject(dlm_name);
}
// If the fpp was deleted, this dlm_name doesnt make a displayer
if (fpp->isDeleted())
{
delete fpp;
ret = -1;
}
else
{
ret = registerDisplayer(fpp);
}
return (ret);
}
#ifndef MAXPATHLEN
#define MAXPATHLEN 1024
#define WE_DEFINED_MAXPATHLEN
#endif
jint FontBrokerObject::
ScanForFontDisplayers(const char* dlmDirList)
{
int ndisplayer = 0;
const char *str = dlmDirList;
char directoryName[MAXPATHLEN];
while (*str != '\0')
{
str = wf_scanToken(str, directoryName, sizeof(directoryName), WF_PATH_SEP_STR, 0);
wf_expandFilename(directoryName, MAXPATHLEN);
if (directoryName[0] != '\0')
{
ndisplayer += scanDisplayersFromDir(directoryName);
}
if (*str != '\0')
{
str++;
}
}
return (ndisplayer);
}
int FontBrokerObject::scanDisplayersFromDir(const char *directoryName)
{
int ndisplayer = 0;
PRDir *dir;
PRDirEntry *dp;
char filename[MAXPATHLEN];
char *nameptr;
if ((dir = PR_OpenDir(directoryName)) != NULL)
{
strcpy(filename, directoryName);
nameptr = &filename[strlen(filename)];
// The checking and addtion of '/' is OK on all platforms as the directoryName that
// NSPR uses is all unix style.
if (*(nameptr-1) != '/' )
{
*nameptr++ = '/';
}
while ((dp = PR_ReadDir(dir, PR_SKIP_BOTH)) != NULL)
{
strcpy(nameptr, PR_DirName(dp));
if (wf_isFileDirectory(filename))
{
ndisplayer += scanDisplayersFromDir(filename);
}
else if (wf_stringEndsWith(PR_DirName(dp), ".dll") ||
wf_stringEndsWith(PR_DirName(dp), ".so") ||
wf_stringEndsWith(PR_DirName(dp), ".sl") ||
wf_stringEndsWith(PR_DirName(dp), ".dlm"))
{
// Make a displayer out of this dll.
jint ret = CreateFontDisplayerFromDLM(filename);
if (ret >= 0)
{
ndisplayer++;
}
}
}
PR_CloseDir(dir);
}
return (ndisplayer);
}
#ifdef WE_DEFINED_MAXPATHLEN
#undef MAXPATHLEN
#undef WE_DEFINED_MAXPATHLEN
#endif
void FontBrokerObject::
RfDone(struct nfrf *rf)
{
// Release all references to this rf in the Font
fontObjectCache.releaseRf(rf);
}
//
// FontBrokerUtility interface specific
//
struct nffmi *
FontBrokerObject::
CreateFontMatchInfo(const char* name, const char* charset,
const char* encoding, jint weight, jint pitch, jint style,
jint underline, jint strikeOut,
jint resX, jint resY)
{
// XXX We should implement fmi cache here.
cfmi *fmi = cfmiFactory_Create(NULL, name, charset, encoding, weight,
pitch, style, underline, strikeOut, resX, resY);
return ((struct nffmi *) fmi);
}
struct nfrc*
FontBrokerObject::
CreateRenderingContext(jint majorType, jint minorType,
void **args, jsize nargs)
{
// WARNING: Dont ever think about caching rc
crc *rc = crcFactory_Create(NULL, majorType, minorType, args, nargs);
return ((struct nfrc *) rc);
}
struct nfdoer*
FontBrokerObject::
CreateFontObserver(nfFontObserverCallback callback, void *client_data)
{
struct cdoer *doer = cdoerFactory_Create(NULL, callback, client_data);
return ((struct nfdoer *) doer);
}
// Font preferences
jint
FontBrokerObject::IsWebfontsEnabled(void)
{
return (pref.enableWebfonts);
}
jint
FontBrokerObject::EnableWebfonts(void)
{
pref.enableWebfonts = 1;
return (0);
}
jint
FontBrokerObject::DisableWebfonts(void)
{
pref.enableWebfonts = 0;
return (0);
}
const char **
FontBrokerObject::ListFontDisplayers()
{
int ndisplayers = fpPeers.count();
if (ndisplayers <= 0)
{
return (NULL);
}
const char **displayers =
(const char **) WF_ALLOC(sizeof(char *) * ndisplayers + 1);
if (!displayers)
{
// No memory
return (NULL);
}
int i = 0;
struct wfListElement *tmp = fpPeers.head;
for (; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject * fpp = (FontDisplayerPeerObject *) tmp->item;
if (fpp->name())
{
displayers[i++] = fpp->name();
}
}
displayers[i] = NULL;
return (displayers);
}
jint
FontBrokerObject::IsFontDisplayerEnabled(const char *displayer)
{
FontDisplayerPeerObject *fpp = findDisplayer(displayer);
if (!fpp)
{
return (-2);
}
return(fpp->isDisplayerEnabled());
}
const char **
FontBrokerObject::ListFontDisplayersForMimetype(const char *mimetype)
{
int ndisplayers = fpPeers.count();
if (ndisplayers <= 0)
{
return (NULL);
}
// 16 bit is unhappy to WF_FREE( const char **displayers )
char **displayersBuffer = (char **)WF_ALLOC(sizeof(char *) * ndisplayers + 1);
const char **displayers = (const char **)displayersBuffer;
if (!displayers)
{
// No memory
return (NULL);
}
int i = 0;
struct wfListElement *tmp = fpPeers.head;
for (; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject * fpp = (FontDisplayerPeerObject *) tmp->item;
if (fpp->isMimetypeEnabled(mimetype) > 0)
{
// this requests const char **displayers
if (fpp->name())
{
displayers[i++] = fpp->name();
}
}
}
if (i == 0)
{
// There are no displayers that has this mimetype enabled.
WF_FREE(displayersBuffer);
displayers = NULL;
}
else
{
displayers[i] = NULL;
}
return (displayers);
}
const char *
FontBrokerObject::FontDisplayerForMimetype(const char *mimetype)
{
const char *displayer = NULL;
struct wfListElement *tmp = fpPeers.head;
// The first fpp which has this mimetype disabled
FontDisplayerPeerObject * first_fpp = NULL;
FontDisplayerPeerObject * fpp;
int ret;
for (; tmp; tmp = tmp->next)
{
fpp = (FontDisplayerPeerObject *) tmp->item;
int ret = fpp->isMimetypeEnabled(mimetype);
if (ret > 0)
{
if (fpp->name())
{
displayer = fpp->name();
}
break;
}
else if (!first_fpp && ret == 0)
{
first_fpp = fpp;
}
}
if (!displayer && first_fpp)
{
// No displayer has this mimetype enabled. Enable the first one.
ret = first_fpp->enableMimetype(mimetype);
if (ret >= 0)
{
displayer = first_fpp->name();
}
else
{
// The first fpp refused to enable this mimetype. Find more.
tmp = fpPeers.head;
for (; tmp; tmp = tmp->next)
{
fpp = (FontDisplayerPeerObject *) tmp->item;
int ret = fpp->isMimetypeEnabled(mimetype);
if (ret == 0)
{
// Here is a font Displayer who has this mimetype disabled. Enable it.
ret = fpp->enableMimetype(mimetype);
if (ret >= 0)
{
// Success
displayer = fpp->name();
break;
}
}
}
}
}
return (displayer);
}
jint
FontBrokerObject::EnableFontDisplayer(const char *displayer)
{
FontDisplayerPeerObject *fpp = findDisplayer(displayer);
if (!fpp)
{
return (-2);
}
return((jint)fpp->enableDisplayer());
}
jint
FontBrokerObject::DisableFontDisplayer(const char *displayer)
{
FontDisplayerPeerObject *fpp = findDisplayer(displayer);
if (!fpp)
{
return (-2);
}
return((jint)fpp->disableDisplayer());
}
jint
FontBrokerObject::EnableMimetype(const char *displayer, const char *mimetype)
{
int ret;
FontDisplayerPeerObject *fpp = findDisplayer(displayer);
if (!fpp)
{
// Displayer not found
ret = -2;
}
else if (fpp->enableMimetype(mimetype) < 0)
{
// Mimetype not in displayer.
ret = -1;
}
else
{
// Disable this mimetype for all other displayers
struct wfListElement *tmp = fpPeers.head;
for (; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject * fpp = (FontDisplayerPeerObject *) tmp->item;
if (fpp->name() && strcmp(fpp->name(), displayer))
{
fpp->disableMimetype(mimetype);
}
}
ret = 0;
}
return(ret);
}
jint
FontBrokerObject::DisableMimetype(const char *displayer, const char *mimetype)
{
FontDisplayerPeerObject *fpp = findDisplayer(displayer);
if (!fpp)
{
// Displayer not found
return (-2);
}
return((jint)fpp->disableMimetype(mimetype));
}
jint
FontBrokerObject::LoadCatalog(const char *filename)
{
const char *catfile = filename;
if (!catfile || !*catfile)
{
catfile = catalogFilename;
}
if (!catfile || !*catfile)
{
// Catalog file doens't exist
return (-1);
}
// Read from the catalog file
FontCatalogFile fc(catfile);
if (fc.status() < 0)
{
return (-1);
}
while (!fc.isEof())
{
FontDisplayerPeerObject * fpp = new FontDisplayerPeerObject (fc);
if (!fpp->isDeleted())
{
wfList::ERROR_CODE err = fpPeersFromCatalog.add(fpp);
if (err != wfList::SUCCESS)
{
// Ignore this.
delete fpp;
}
}
else
{
delete fpp;
}
}
return (0);
}
jint
FontBrokerObject::SaveCatalog(const char *filename)
{
const char *catfile = filename;
if (!catfile || !*catfile)
{
catfile = catalogFilename;
}
if (!catfile || !*catfile)
{
// Catalog file doens't exist
return (-1);
}
FontCatalogFile fc(catfile, 1 /* write */);
if (fc.status() < 0)
{
return (-1);
}
struct wfListElement *tmp = fpPeers.head;
while (tmp)
{
FontDisplayerPeerObject * fpp = (FontDisplayerPeerObject *) tmp->item;
fpp->describe(fc);
tmp = tmp->next;
}
return (0);
}
//
// Private methods
//
FontDisplayerPeerObject *
FontBrokerObject::findDisplayer(const char *displayerName)
{
FontDisplayerPeerObject * wffpp = NULL;
struct wfListElement *tmp = fpPeers.head;
for (; tmp; tmp = tmp->next)
{
FontDisplayerPeerObject * fpp = (FontDisplayerPeerObject *) tmp->item;
if (fpp->name() && !strcmp(fpp->name(), displayerName))
{
wffpp = fpp;
break;
}
}
return (wffpp);
}
// Merge fmiList together
int FontBrokerObject::
merge(struct nffmi ** &srcList, int &srcLen,
struct nffmi **newList)
{
int newlen = 0;
if (!newList || !newList[0])
{
return -1;
}
while (newList[newlen]) newlen++;
// Allocate more space to hold
srcList = (struct nffmi **)
WF_REALLOC(srcList, sizeof(*srcList) * (srcLen + newlen + 1));
if (!srcList)
{
// No memory
return -1;
}
//
// Copy the new list into the old one.
//
for(int i=0; i<newlen; i++)
{
srcList[srcLen+i] = newList[i];
}
srcLen += newlen;
srcList[srcLen] = NULL;
return 0;
}
jint FontBrokerObject::registerDisplayer(FontDisplayerPeerObject *fpp)
{
//
// we need to make sure that the native displayer is added to the
// tail. The native displayer will be the last in the queue
// for asking for a font. Reason:
// - The native displayer might be implemented to always return
// a font by doing a closest match.
//
FontDisplayerPeerObject *tmp_fpp;
wfList::ERROR_CODE err = fpPeers.add(fpp);
int ret = 0;
if (err != wfList::SUCCESS)
{
ret = -1;
}
else
{
// Reconfigure the fpPeers list so that the native
// displayer is at the end of the list.
struct wfListElement *tmp = fpPeers.head;
while(tmp)
{
tmp_fpp = (FontDisplayerPeerObject *) tmp->item;
if (tmp_fpp->isNative())
{
break;
}
tmp = tmp->next;
}
// Move native to the tail
if (tmp && tmp != fpPeers.tail)
{
WF_TRACEMSG(("NF: Moving the native displayer to the end of the displayer list."));
tmp_fpp = (FontDisplayerPeerObject *) tmp->item;
fpPeers.remove((void *)tmp_fpp);
// we are banking on the fact that add will add at end.
fpPeers.add(tmp_fpp);
}
}
return(ret);
}