mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-12-09 06:22:08 +00:00
beac740c0f
--HG-- extra : rebase_source : 1c88ebe3a41dee12a982442ad2e711aa287ed55f
661 lines
22 KiB
C++
661 lines
22 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla 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/MPL/
|
|
*
|
|
* 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 MathML Project.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* The University Of Queensland.
|
|
* Portions created by the Initial Developer are Copyright (C) 1999
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Roger B. Sidje <rbs@maths.uq.edu.au>
|
|
* Karl Tomlinson <karlt+@karlt.net>, Mozilla Corporation
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
|
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "nsCOMPtr.h"
|
|
#include "nsString.h"
|
|
#include "nsHashtable.h"
|
|
#include "nsTArray.h"
|
|
|
|
#include "nsIComponentManager.h"
|
|
#include "nsIPersistentProperties2.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsCRT.h"
|
|
|
|
#include "nsMathMLOperators.h"
|
|
|
|
// operator dictionary entry
|
|
struct OperatorData {
|
|
OperatorData(void)
|
|
: mFlags(0),
|
|
mLeftSpace(0.0f),
|
|
mRightSpace(0.0f)
|
|
{
|
|
}
|
|
|
|
// member data
|
|
nsString mStr;
|
|
nsOperatorFlags mFlags;
|
|
float mLeftSpace; // unit is em
|
|
float mRightSpace; // unit is em
|
|
};
|
|
|
|
/*
|
|
The MathML REC says:
|
|
"If the operator does not occur in the dictionary with the specified form,
|
|
the renderer should use one of the forms which is available there, in the
|
|
order of preference: infix, postfix, prefix."
|
|
|
|
The following variable will be used to keep track of all possible forms
|
|
encountered in the Operator Dictionary.
|
|
*/
|
|
static OperatorData* gOperatorFound[4];
|
|
|
|
static PRInt32 gTableRefCount = 0;
|
|
static PRInt32 gOperatorCount = 0;
|
|
static OperatorData* gOperatorArray = nsnull;
|
|
static nsHashtable* gOperatorTable = nsnull;
|
|
static PRBool gInitialized = PR_FALSE;
|
|
static nsTArray<OperatorData*>* gStretchyOperatorArray = nsnull;
|
|
static nsTArray<nsString>* gInvariantCharArray = nsnull;
|
|
|
|
static const PRUnichar kNullCh = PRUnichar('\0');
|
|
static const PRUnichar kDashCh = PRUnichar('#');
|
|
static const PRUnichar kEqualCh = PRUnichar('=');
|
|
static const PRUnichar kColonCh = PRUnichar(':');
|
|
|
|
static const char* const kMathVariant_name[] = {
|
|
"normal",
|
|
"bold",
|
|
"italic",
|
|
"bold-italic",
|
|
"sans-serif",
|
|
"bold-sans-serif",
|
|
"sans-serif-italic",
|
|
"sans-serif-bold-italic",
|
|
"monospace",
|
|
"script",
|
|
"bold-script",
|
|
"fraktur",
|
|
"bold-fraktur",
|
|
"double-struck"
|
|
};
|
|
|
|
void
|
|
SetProperty(OperatorData* aOperatorData,
|
|
nsString aName,
|
|
nsString aValue)
|
|
{
|
|
if (!aName.Length() || !aValue.Length())
|
|
return;
|
|
|
|
// XXX These ones are not kept in the dictionary
|
|
// Support for these requires nsString member variables
|
|
// maxsize (default: infinity)
|
|
// minsize (default: 1)
|
|
|
|
if (aValue.EqualsLiteral("true")) {
|
|
// see if we should enable flags with default value=false
|
|
if (aName.EqualsLiteral("fence"))
|
|
aOperatorData->mFlags |= NS_MATHML_OPERATOR_FENCE;
|
|
else if (aName.EqualsLiteral("accent"))
|
|
aOperatorData->mFlags |= NS_MATHML_OPERATOR_ACCENT;
|
|
else if (aName.EqualsLiteral("largeop"))
|
|
aOperatorData->mFlags |= NS_MATHML_OPERATOR_LARGEOP;
|
|
else if (aName.EqualsLiteral("separator"))
|
|
aOperatorData->mFlags |= NS_MATHML_OPERATOR_SEPARATOR;
|
|
else if (aName.EqualsLiteral("movablelimits"))
|
|
aOperatorData->mFlags |= NS_MATHML_OPERATOR_MOVABLELIMITS;
|
|
}
|
|
else if (aValue.EqualsLiteral("false")) {
|
|
// see if we should disable flags with default value=true
|
|
if (aName.EqualsLiteral("symmetric"))
|
|
aOperatorData->mFlags &= ~NS_MATHML_OPERATOR_SYMMETRIC;
|
|
}
|
|
else if (aName.EqualsLiteral("stretchy") &&
|
|
(1 == aOperatorData->mStr.Length())) {
|
|
if (aValue.EqualsLiteral("vertical"))
|
|
aOperatorData->mFlags |= NS_MATHML_OPERATOR_STRETCHY_VERT;
|
|
else if (aValue.EqualsLiteral("horizontal"))
|
|
aOperatorData->mFlags |= NS_MATHML_OPERATOR_STRETCHY_HORIZ;
|
|
else return; // invalid value
|
|
if (kNotFound == nsMathMLOperators::FindStretchyOperator(aOperatorData->mStr[0])) {
|
|
gStretchyOperatorArray->AppendElement(aOperatorData);
|
|
}
|
|
}
|
|
else {
|
|
PRInt32 i = 0;
|
|
float space = 0.0f;
|
|
PRBool isLeftSpace;
|
|
if (aName.EqualsLiteral("lspace"))
|
|
isLeftSpace = PR_TRUE;
|
|
else if (aName.EqualsLiteral("rspace"))
|
|
isLeftSpace = PR_FALSE;
|
|
else return; // input is not applicable
|
|
|
|
// See if it is a numeric value (unit is assumed to be 'em')
|
|
if (nsCRT::IsAsciiDigit(aValue[0])) {
|
|
PRInt32 error = 0;
|
|
space = aValue.ToFloat(&error);
|
|
if (error) return;
|
|
}
|
|
// See if it is one of the 'namedspace' (ranging 1/18em...7/18em)
|
|
else if (aValue.EqualsLiteral("veryverythinmathspace")) i = 1;
|
|
else if (aValue.EqualsLiteral("verythinmathspace")) i = 2;
|
|
else if (aValue.EqualsLiteral("thinmathspace")) i = 3;
|
|
else if (aValue.EqualsLiteral("mediummathspace")) i = 4;
|
|
else if (aValue.EqualsLiteral("thickmathspace")) i = 5;
|
|
else if (aValue.EqualsLiteral("verythickmathspace")) i = 6;
|
|
else if (aValue.EqualsLiteral("veryverythickmathspace")) i = 7;
|
|
|
|
if (0 != i) // it was a namedspace value
|
|
space = float(i)/float(18);
|
|
|
|
if (isLeftSpace)
|
|
aOperatorData->mLeftSpace = space;
|
|
else
|
|
aOperatorData->mRightSpace = space;
|
|
}
|
|
}
|
|
|
|
PRBool
|
|
SetOperator(OperatorData* aOperatorData,
|
|
nsOperatorFlags aForm,
|
|
const nsCString& aOperator,
|
|
nsString& aAttributes)
|
|
|
|
{
|
|
// aOperator is in the expanded format \uNNNN\uNNNN ...
|
|
// First compress these Unicode points to the internal nsString format
|
|
PRInt32 i = 0;
|
|
nsAutoString name, value;
|
|
PRInt32 len = aOperator.Length();
|
|
PRUnichar c = aOperator[i++];
|
|
PRUint32 state = 0;
|
|
PRUnichar uchar = 0;
|
|
while (i <= len) {
|
|
if (0 == state) {
|
|
if (c != '\\')
|
|
return PR_FALSE;
|
|
if (i < len)
|
|
c = aOperator[i];
|
|
i++;
|
|
if (('u' != c) && ('U' != c))
|
|
return PR_FALSE;
|
|
if (i < len)
|
|
c = aOperator[i];
|
|
i++;
|
|
state++;
|
|
}
|
|
else {
|
|
if (('0' <= c) && (c <= '9'))
|
|
uchar = (uchar << 4) | (c - '0');
|
|
else if (('a' <= c) && (c <= 'f'))
|
|
uchar = (uchar << 4) | (c - 'a' + 0x0a);
|
|
else if (('A' <= c) && (c <= 'F'))
|
|
uchar = (uchar << 4) | (c - 'A' + 0x0a);
|
|
else return PR_FALSE;
|
|
if (i < len)
|
|
c = aOperator[i];
|
|
i++;
|
|
state++;
|
|
if (5 == state) {
|
|
value.Append(uchar);
|
|
uchar = 0;
|
|
state = 0;
|
|
}
|
|
}
|
|
}
|
|
if (0 != state) return PR_FALSE;
|
|
|
|
// Quick return when the caller doesn't care about the attributes and just wants
|
|
// to know if this is a valid operator (this is the case at the first pass of the
|
|
// parsing of the dictionary in InitOperators())
|
|
if (!aForm) return PR_TRUE;
|
|
|
|
// Add operator to hash table (symmetric="true" by default for all operators)
|
|
aOperatorData->mFlags |= aForm | NS_MATHML_OPERATOR_SYMMETRIC;
|
|
aOperatorData->mStr.Assign(value);
|
|
value.AppendInt(aForm, 10);
|
|
nsStringKey key(value);
|
|
gOperatorTable->Put(&key, aOperatorData);
|
|
|
|
#ifdef NS_DEBUG
|
|
NS_LossyConvertUTF16toASCII str(aAttributes);
|
|
#endif
|
|
// Loop over the space-delimited list of attributes to get the name:value pairs
|
|
aAttributes.Append(kNullCh); // put an extra null at the end
|
|
PRUnichar* start = aAttributes.BeginWriting();
|
|
PRUnichar* end = start;
|
|
while ((kNullCh != *start) && (kDashCh != *start)) {
|
|
name.SetLength(0);
|
|
value.SetLength(0);
|
|
// skip leading space, the dash amounts to the end of the line
|
|
while ((kNullCh!=*start) && (kDashCh!=*start) && nsCRT::IsAsciiSpace(*start)) {
|
|
++start;
|
|
}
|
|
end = start;
|
|
// look for ':' or '='
|
|
while ((kNullCh!=*end) && (kDashCh!=*end) && (kColonCh!=*end) && (kEqualCh!=*end)) {
|
|
++end;
|
|
}
|
|
if ((kColonCh!=*end) && (kEqualCh!=*end)) {
|
|
#ifdef NS_DEBUG
|
|
printf("Bad MathML operator: %s\n", str.get());
|
|
#endif
|
|
return PR_TRUE;
|
|
}
|
|
*end = kNullCh; // end segment here
|
|
// this segment is the name
|
|
if (start < end) {
|
|
name.Assign(start);
|
|
}
|
|
start = ++end;
|
|
// look for space or end of line
|
|
while ((kNullCh!=*end) && (kDashCh!=*start) && !nsCRT::IsAsciiSpace(*end)) {
|
|
++end;
|
|
}
|
|
*end = kNullCh; // end segment here
|
|
// this segment is the value
|
|
if (start < end) {
|
|
value.Assign(start);
|
|
}
|
|
SetProperty(aOperatorData, name, value);
|
|
start = ++end;
|
|
}
|
|
return PR_TRUE;
|
|
}
|
|
|
|
nsresult
|
|
InitOperators(void)
|
|
{
|
|
// Load the property file containing the Operator Dictionary
|
|
nsresult rv;
|
|
nsCOMPtr<nsIPersistentProperties> mathfontProp;
|
|
rv = NS_LoadPersistentPropertiesFromURISpec(getter_AddRefs(mathfontProp),
|
|
NS_LITERAL_CSTRING("resource://gre/res/fonts/mathfont.properties"));
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
// Get the list of invariant chars
|
|
for (PRInt32 i = 0; i < eMATHVARIANT_COUNT; ++i) {
|
|
nsCAutoString key(NS_LITERAL_CSTRING("mathvariant."));
|
|
key.Append(kMathVariant_name[i]);
|
|
nsAutoString value;
|
|
mathfontProp->GetStringProperty(key, value);
|
|
gInvariantCharArray->AppendElement(value); // i.e., gInvariantCharArray[i] holds this list
|
|
}
|
|
|
|
// Parse the Operator Dictionary in two passes.
|
|
// The first pass is to count the number of operators; the second pass is to
|
|
// allocate the necessary space for them and to add them in the hash table.
|
|
for (PRInt32 pass = 1; pass <= 2; pass++) {
|
|
OperatorData dummyData;
|
|
OperatorData* operatorData = &dummyData;
|
|
nsCOMPtr<nsISimpleEnumerator> iterator;
|
|
if (NS_SUCCEEDED(mathfontProp->Enumerate(getter_AddRefs(iterator)))) {
|
|
PRBool more;
|
|
PRInt32 index = 0;
|
|
nsCAutoString name;
|
|
nsAutoString attributes;
|
|
while ((NS_SUCCEEDED(iterator->HasMoreElements(&more))) && more) {
|
|
nsCOMPtr<nsIPropertyElement> element;
|
|
if (NS_SUCCEEDED(iterator->GetNext(getter_AddRefs(element)))) {
|
|
if (NS_SUCCEEDED(element->GetKey(name)) &&
|
|
NS_SUCCEEDED(element->GetValue(attributes))) {
|
|
// expected key: operator.\uNNNN.{infix,postfix,prefix}
|
|
if ((21 <= name.Length()) && (0 == name.Find("operator.\\u"))) {
|
|
name.Cut(0, 9); // 9 is the length of "operator.";
|
|
PRInt32 len = name.Length();
|
|
nsOperatorFlags form = 0;
|
|
if (kNotFound != name.RFind(".infix")) {
|
|
form = NS_MATHML_OPERATOR_FORM_INFIX;
|
|
len -= 6; // 6 is the length of ".infix";
|
|
}
|
|
else if (kNotFound != name.RFind(".postfix")) {
|
|
form = NS_MATHML_OPERATOR_FORM_POSTFIX;
|
|
len -= 8; // 8 is the length of ".postfix";
|
|
}
|
|
else if (kNotFound != name.RFind(".prefix")) {
|
|
form = NS_MATHML_OPERATOR_FORM_PREFIX;
|
|
len -= 7; // 7 is the length of ".prefix";
|
|
}
|
|
else continue; // input is not applicable
|
|
name.SetLength(len);
|
|
if (2 == pass) { // allocate space and start the storage
|
|
if (!gOperatorArray) {
|
|
if (0 == gOperatorCount) return NS_ERROR_UNEXPECTED;
|
|
gOperatorArray = new OperatorData[gOperatorCount];
|
|
if (!gOperatorArray) return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
operatorData = &gOperatorArray[index];
|
|
}
|
|
else {
|
|
form = 0; // to quickly return from SetOperator() at pass 1
|
|
}
|
|
// See if the operator should be retained
|
|
if (SetOperator(operatorData, form, name, attributes)) {
|
|
index++;
|
|
if (1 == pass) gOperatorCount = index;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
InitGlobals()
|
|
{
|
|
gInitialized = PR_TRUE;
|
|
nsresult rv = NS_ERROR_OUT_OF_MEMORY;
|
|
gInvariantCharArray = new nsTArray<nsString>();
|
|
gStretchyOperatorArray = new nsTArray<OperatorData*>();
|
|
if (gInvariantCharArray && gStretchyOperatorArray) {
|
|
gOperatorTable = new nsHashtable();
|
|
if (gOperatorTable) {
|
|
rv = InitOperators();
|
|
}
|
|
}
|
|
if (NS_FAILED(rv))
|
|
nsMathMLOperators::CleanUp();
|
|
return rv;
|
|
}
|
|
|
|
void
|
|
nsMathMLOperators::CleanUp()
|
|
{
|
|
if (gInvariantCharArray) {
|
|
delete gInvariantCharArray;
|
|
gInvariantCharArray = nsnull;
|
|
}
|
|
if (gOperatorArray) {
|
|
delete[] gOperatorArray;
|
|
gOperatorArray = nsnull;
|
|
}
|
|
if (gStretchyOperatorArray) {
|
|
delete gStretchyOperatorArray;
|
|
gStretchyOperatorArray = nsnull;
|
|
}
|
|
if (gOperatorTable) {
|
|
delete gOperatorTable;
|
|
gOperatorTable = nsnull;
|
|
}
|
|
}
|
|
|
|
void
|
|
nsMathMLOperators::AddRefTable(void)
|
|
{
|
|
gTableRefCount++;
|
|
}
|
|
|
|
void
|
|
nsMathMLOperators::ReleaseTable(void)
|
|
{
|
|
if (0 == --gTableRefCount) {
|
|
CleanUp();
|
|
}
|
|
}
|
|
|
|
PRBool
|
|
nsMathMLOperators::LookupOperator(const nsString& aOperator,
|
|
const nsOperatorFlags aForm,
|
|
nsOperatorFlags* aFlags,
|
|
float* aLeftSpace,
|
|
float* aRightSpace)
|
|
{
|
|
if (!gInitialized) {
|
|
InitGlobals();
|
|
}
|
|
if (gOperatorTable) {
|
|
NS_ASSERTION(aFlags && aLeftSpace && aRightSpace, "bad usage");
|
|
NS_ASSERTION(aForm>=0 && aForm<4, "*** invalid call ***");
|
|
|
|
OperatorData* found;
|
|
PRInt32 form = NS_MATHML_OPERATOR_GET_FORM(aForm);
|
|
gOperatorFound[NS_MATHML_OPERATOR_FORM_INFIX] = nsnull;
|
|
gOperatorFound[NS_MATHML_OPERATOR_FORM_POSTFIX] = nsnull;
|
|
gOperatorFound[NS_MATHML_OPERATOR_FORM_PREFIX] = nsnull;
|
|
|
|
nsAutoString key(aOperator);
|
|
key.AppendInt(form, 10);
|
|
nsStringKey hkey(key);
|
|
gOperatorFound[form] = found = (OperatorData*)gOperatorTable->Get(&hkey);
|
|
|
|
// If not found, check if the operator exists perhaps in a different form,
|
|
// in the order of preference: infix, postfix, prefix
|
|
if (!found) {
|
|
if (form != NS_MATHML_OPERATOR_FORM_INFIX) {
|
|
form = NS_MATHML_OPERATOR_FORM_INFIX;
|
|
key.Assign(aOperator);
|
|
key.AppendInt(form, 10);
|
|
nsStringKey hashkey(key);
|
|
gOperatorFound[form] = found = (OperatorData*)gOperatorTable->Get(&hashkey);
|
|
}
|
|
if (!found) {
|
|
if (form != NS_MATHML_OPERATOR_FORM_POSTFIX) {
|
|
form = NS_MATHML_OPERATOR_FORM_POSTFIX;
|
|
key.Assign(aOperator);
|
|
key.AppendInt(form, 10);
|
|
nsStringKey hashkey(key);
|
|
gOperatorFound[form] = found = (OperatorData*)gOperatorTable->Get(&hashkey);
|
|
}
|
|
if (!found) {
|
|
if (form != NS_MATHML_OPERATOR_FORM_PREFIX) {
|
|
form = NS_MATHML_OPERATOR_FORM_PREFIX;
|
|
key.Assign(aOperator);
|
|
key.AppendInt(form, 10);
|
|
nsStringKey hashkey(key);
|
|
gOperatorFound[form] = found = (OperatorData*)gOperatorTable->Get(&hashkey);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (found) {
|
|
NS_ASSERTION(found->mStr.Equals(aOperator), "bad setup");
|
|
*aLeftSpace = found->mLeftSpace;
|
|
*aRightSpace = found->mRightSpace;
|
|
*aFlags &= ~NS_MATHML_OPERATOR_FORM; // clear the form bits
|
|
*aFlags |= found->mFlags; // just add bits without overwriting
|
|
return PR_TRUE;
|
|
}
|
|
}
|
|
return PR_FALSE;
|
|
}
|
|
|
|
void
|
|
nsMathMLOperators::LookupOperators(const nsString& aOperator,
|
|
nsOperatorFlags* aFlags,
|
|
float* aLeftSpace,
|
|
float* aRightSpace)
|
|
{
|
|
if (!gInitialized) {
|
|
InitGlobals();
|
|
}
|
|
|
|
aFlags[NS_MATHML_OPERATOR_FORM_INFIX] = 0;
|
|
aLeftSpace[NS_MATHML_OPERATOR_FORM_INFIX] = 0.0f;
|
|
aRightSpace[NS_MATHML_OPERATOR_FORM_INFIX] = 0.0f;
|
|
|
|
aFlags[NS_MATHML_OPERATOR_FORM_POSTFIX] = 0;
|
|
aLeftSpace[NS_MATHML_OPERATOR_FORM_POSTFIX] = 0.0f;
|
|
aRightSpace[NS_MATHML_OPERATOR_FORM_POSTFIX] = 0.0f;
|
|
|
|
aFlags[NS_MATHML_OPERATOR_FORM_PREFIX] = 0;
|
|
aLeftSpace[NS_MATHML_OPERATOR_FORM_PREFIX] = 0.0f;
|
|
aRightSpace[NS_MATHML_OPERATOR_FORM_PREFIX] = 0.0f;
|
|
|
|
if (gOperatorTable) {
|
|
// a lookup with form=0 will put all the variants in gOperatorFound[]
|
|
float dummy;
|
|
nsOperatorFlags flags = 0;
|
|
LookupOperator(aOperator, /*form=*/0, &flags, &dummy, &dummy);
|
|
// if the operator was found, gOperatorFound contains all its variants
|
|
OperatorData* found;
|
|
found = gOperatorFound[NS_MATHML_OPERATOR_FORM_INFIX];
|
|
if (found) {
|
|
aFlags[NS_MATHML_OPERATOR_FORM_INFIX] = found->mFlags;
|
|
aLeftSpace[NS_MATHML_OPERATOR_FORM_INFIX] = found->mLeftSpace;
|
|
aRightSpace[NS_MATHML_OPERATOR_FORM_INFIX] = found->mRightSpace;
|
|
}
|
|
found = gOperatorFound[NS_MATHML_OPERATOR_FORM_POSTFIX];
|
|
if (found) {
|
|
aFlags[NS_MATHML_OPERATOR_FORM_POSTFIX] = found->mFlags;
|
|
aLeftSpace[NS_MATHML_OPERATOR_FORM_POSTFIX] = found->mLeftSpace;
|
|
aRightSpace[NS_MATHML_OPERATOR_FORM_POSTFIX] = found->mRightSpace;
|
|
}
|
|
found = gOperatorFound[NS_MATHML_OPERATOR_FORM_PREFIX];
|
|
if (found) {
|
|
aFlags[NS_MATHML_OPERATOR_FORM_PREFIX] = found->mFlags;
|
|
aLeftSpace[NS_MATHML_OPERATOR_FORM_PREFIX] = found->mLeftSpace;
|
|
aRightSpace[NS_MATHML_OPERATOR_FORM_PREFIX] = found->mRightSpace;
|
|
}
|
|
}
|
|
}
|
|
|
|
PRBool
|
|
nsMathMLOperators::IsMutableOperator(const nsString& aOperator)
|
|
{
|
|
if (!gInitialized) {
|
|
InitGlobals();
|
|
}
|
|
// lookup all the variants of the operator and return true if there
|
|
// is a variant that is stretchy or largeop
|
|
nsOperatorFlags flags[4];
|
|
float lspace[4], rspace[4];
|
|
nsMathMLOperators::LookupOperators(aOperator, flags, lspace, rspace);
|
|
nsOperatorFlags allFlags =
|
|
flags[NS_MATHML_OPERATOR_FORM_INFIX] |
|
|
flags[NS_MATHML_OPERATOR_FORM_POSTFIX] |
|
|
flags[NS_MATHML_OPERATOR_FORM_PREFIX];
|
|
return NS_MATHML_OPERATOR_IS_STRETCHY(allFlags) ||
|
|
NS_MATHML_OPERATOR_IS_LARGEOP(allFlags);
|
|
}
|
|
|
|
PRInt32
|
|
nsMathMLOperators::CountStretchyOperator()
|
|
{
|
|
if (!gInitialized) {
|
|
InitGlobals();
|
|
}
|
|
return (gStretchyOperatorArray) ? gStretchyOperatorArray->Length() : 0;
|
|
}
|
|
|
|
PRInt32
|
|
nsMathMLOperators::FindStretchyOperator(PRUnichar aOperator)
|
|
{
|
|
if (!gInitialized) {
|
|
InitGlobals();
|
|
}
|
|
if (gStretchyOperatorArray) {
|
|
for (PRUint32 k = 0; k < gStretchyOperatorArray->Length(); k++) {
|
|
OperatorData* data = gStretchyOperatorArray->ElementAt(k);
|
|
if (data && (aOperator == data->mStr[0])) {
|
|
return k;
|
|
}
|
|
}
|
|
}
|
|
return kNotFound;
|
|
}
|
|
|
|
nsStretchDirection
|
|
nsMathMLOperators::GetStretchyDirectionAt(PRInt32 aIndex)
|
|
{
|
|
NS_ASSERTION(gStretchyOperatorArray, "invalid call");
|
|
if (gStretchyOperatorArray) {
|
|
NS_ASSERTION(aIndex < PRInt32(gStretchyOperatorArray->Length()),
|
|
"invalid call");
|
|
OperatorData* data = gStretchyOperatorArray->ElementAt(aIndex);
|
|
if (data) {
|
|
if (NS_MATHML_OPERATOR_IS_STRETCHY_VERT(data->mFlags))
|
|
return NS_STRETCH_DIRECTION_VERTICAL;
|
|
else if (NS_MATHML_OPERATOR_IS_STRETCHY_HORIZ(data->mFlags))
|
|
return NS_STRETCH_DIRECTION_HORIZONTAL;
|
|
NS_ASSERTION(PR_FALSE, "*** bad setup ***");
|
|
}
|
|
}
|
|
return NS_STRETCH_DIRECTION_UNSUPPORTED;
|
|
}
|
|
|
|
void
|
|
nsMathMLOperators::DisableStretchyOperatorAt(PRInt32 aIndex)
|
|
{
|
|
NS_ASSERTION(gStretchyOperatorArray, "invalid call");
|
|
if (gStretchyOperatorArray) {
|
|
NS_ASSERTION(aIndex < PRInt32(gStretchyOperatorArray->Length()),
|
|
"invalid call");
|
|
(*gStretchyOperatorArray)[aIndex] = nsnull;
|
|
}
|
|
}
|
|
|
|
/* static */ eMATHVARIANT
|
|
nsMathMLOperators::LookupInvariantChar(const nsAString& aChar)
|
|
{
|
|
if (!gInitialized) {
|
|
InitGlobals();
|
|
}
|
|
if (gInvariantCharArray) {
|
|
for (PRInt32 i = gInvariantCharArray->Length()-1; i >= 0; --i) {
|
|
const nsString& list = gInvariantCharArray->ElementAt(i);
|
|
nsString::const_iterator start, end;
|
|
list.BeginReading(start);
|
|
list.EndReading(end);
|
|
// Style-invariant characters are at offset 3*j + 1.
|
|
if (FindInReadable(aChar, start, end) &&
|
|
start.size_backward() % 3 == 1) {
|
|
return eMATHVARIANT(i);
|
|
}
|
|
}
|
|
}
|
|
return eMATHVARIANT_NONE;
|
|
}
|
|
|
|
/* static */ const nsDependentSubstring
|
|
nsMathMLOperators::TransformVariantChar(const PRUnichar& aChar,
|
|
eMATHVARIANT aVariant)
|
|
{
|
|
if (!gInitialized) {
|
|
InitGlobals();
|
|
}
|
|
if (gInvariantCharArray) {
|
|
nsString list = gInvariantCharArray->ElementAt(aVariant);
|
|
PRInt32 index = list.FindChar(aChar);
|
|
// BMP characters are at offset 3*j
|
|
if (index != kNotFound && index % 3 == 0 && list.Length() - index >= 2 ) {
|
|
// The style-invariant character is the next character
|
|
// (and list should contain padding if the next character is in the BMP).
|
|
++index;
|
|
PRUint32 len = NS_IS_HIGH_SURROGATE(list.CharAt(index)) ? 2 : 1;
|
|
return nsDependentSubstring(list, index, len);
|
|
}
|
|
}
|
|
return nsDependentSubstring(&aChar, &aChar + 1);
|
|
}
|