mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-13 15:34:01 +00:00
1090 lines
24 KiB
C
1090 lines
24 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.
|
|
*/
|
|
|
|
#include "pa_parse.h"
|
|
#include <stdio.h>
|
|
|
|
#include "libi18n.h"
|
|
|
|
#ifdef PROFILE
|
|
#pragma profile on
|
|
#endif
|
|
#define PA_REMAP_128_TO_160_ILLEGAL_NCR 1
|
|
|
|
#define NOT_USED 0xfffd
|
|
static uint16 PA_HackTable[] = {
|
|
NOT_USED,
|
|
NOT_USED,
|
|
0x201a, /* SINGLE LOW-9 QUOTATION MARK */
|
|
0x0192, /* LATIN SMALL LETTER F WITH HOOK */
|
|
0x201e, /* DOUBLE LOW-9 QUOTATION MARK */
|
|
0x2026, /* HORIZONTAL ELLIPSIS */
|
|
0x2020, /* DAGGER */
|
|
0x2021, /* DOUBLE DAGGER */
|
|
0x02c6, /* MODIFIER LETTER CIRCUMFLEX ACCENT */
|
|
0x2030, /* PER MILLE SIGN */
|
|
0x0160, /* LATIN CAPITAL LETTER S WITH CARON */
|
|
0x2039, /* SINGLE LEFT-POINTING ANGLE QUOTATION MARK */
|
|
0x0152, /* LATIN CAPITAL LIGATURE OE */
|
|
NOT_USED,
|
|
NOT_USED,
|
|
NOT_USED,
|
|
NOT_USED,
|
|
0x2018, /* LEFT SINGLE QUOTATION MARK */
|
|
0x2019, /* RIGHT SINGLE QUOTATION MARK */
|
|
0x201c, /* LEFT DOUBLE QUOTATION MARK */
|
|
0x201d, /* RIGHT DOUBLE QUOTATION MARK */
|
|
0x2022, /* BULLET */
|
|
0x2013, /* EN DASH */
|
|
0x2014, /* EM DASH */
|
|
0x02dc, /* SMALL TILDE */
|
|
0x2122, /* TRADE MARK SIGN */
|
|
0x0161, /* LATIN SMALL LETTER S WITH CARON */
|
|
0x203a, /* SINGLE RIGHT-POINTING ANGLE QUOTATION MARK */
|
|
0x0153, /* LATIN SMALL LIGATURE OE */
|
|
NOT_USED,
|
|
NOT_USED,
|
|
0x0178 /* LATIN CAPITAL LETTER Y WITH DIAERESIS */
|
|
};
|
|
|
|
|
|
typedef struct PA_N2U_struct {
|
|
char* NE; /* Name Entity, "copy" */
|
|
intn len; /* 4, the length of NE */
|
|
int16 unicode; /* 0x00a4 */
|
|
} PA_N2U; /* Name->Unicode */
|
|
|
|
/* NE LEN UNICODE */
|
|
static PA_N2U PA_Name2Unicode[] = {
|
|
{"quot", 4, 0x0022 },
|
|
{"QUOT", 4, 0x0022 },
|
|
{"amp", 3, 0x0026 },
|
|
{"AMP", 3, 0x0026 },
|
|
{"lt", 2, 0x003c },
|
|
{"LT", 2, 0x003c },
|
|
{"gt", 2, 0x003e },
|
|
{"GT", 2, 0x003e },
|
|
{"nbsp", 4, 0x00a0 },
|
|
{"iexcl", 5, 0x00a1 },
|
|
{"cent", 4, 0x00a2 },
|
|
{"pound", 5, 0x00a3 },
|
|
{"curren", 6, 0x00a4 },
|
|
{"yen", 3, 0x00a5 },
|
|
{"brvbar", 6, 0x00a6 },
|
|
{"sect", 4, 0x00a7 },
|
|
{"uml", 3, 0x00a8 },
|
|
{"copy", 4, 0x00a9 },
|
|
{"COPY", 4, 0x00a9 },
|
|
{"ordf", 4, 0x00aa },
|
|
{"laquo", 5, 0x00ab },
|
|
{"not", 3, 0x00ac },
|
|
{"shy", 3, 0x00ad },
|
|
{"reg", 3, 0x00ae },
|
|
{"REG", 3, 0x00ae },
|
|
{"macr", 4, 0x00af },
|
|
{"deg", 3, 0x00b0 },
|
|
{"plusmn", 6, 0x00b1 },
|
|
{"sup2", 4, 0x00b2 },
|
|
{"sup3", 4, 0x00b3 },
|
|
{"acute", 5, 0x00b4 },
|
|
{"micro", 5, 0x00b5 },
|
|
{"para", 4, 0x00b6 },
|
|
{"middot", 6, 0x00b7 },
|
|
{"cedil", 5, 0x00b8 },
|
|
{"sup1", 4, 0x00b9 },
|
|
{"ordm", 4, 0x00ba },
|
|
{"raquo", 5, 0x00bb },
|
|
{"frac14", 6, 0x00bc },
|
|
{"frac12", 6, 0x00bd },
|
|
{"frac34", 6, 0x00be },
|
|
{"iquest", 6, 0x00bf },
|
|
{"Agrave", 6, 0x00c0 },
|
|
{"Aacute", 6, 0x00c1 },
|
|
{"Acirc", 5, 0x00c2 },
|
|
{"Atilde", 6, 0x00c3 },
|
|
{"Auml", 4, 0x00c4 },
|
|
{"Aring", 5, 0x00c5 },
|
|
{"AElig", 5, 0x00c6 },
|
|
{"Ccedil", 6, 0x00c7 },
|
|
{"Egrave", 6, 0x00c8 },
|
|
{"Eacute", 6, 0x00c9 },
|
|
{"Ecirc", 5, 0x00ca },
|
|
{"Euml", 4, 0x00cb },
|
|
{"Igrave", 6, 0x00cc },
|
|
{"Iacute", 6, 0x00cd },
|
|
{"Icirc", 5, 0x00ce },
|
|
{"Iuml", 4, 0x00cf },
|
|
{"ETH", 3, 0x00d0 },
|
|
{"Ntilde", 6, 0x00d1 },
|
|
{"Ograve", 6, 0x00d2 },
|
|
{"Oacute", 6, 0x00d3 },
|
|
{"Ocirc", 5, 0x00d4 },
|
|
{"Otilde", 6, 0x00d5 },
|
|
{"Ouml", 4, 0x00d6 },
|
|
{"times", 5, 0x00d7 },
|
|
{"Oslash", 6, 0x00d8 },
|
|
{"Ugrave", 6, 0x00d9 },
|
|
{"Uacute", 6, 0x00da },
|
|
{"Ucirc", 5, 0x00db },
|
|
{"Uuml", 4, 0x00dc },
|
|
{"Yacute", 6, 0x00dd },
|
|
{"THORN", 5, 0x00de },
|
|
{"szlig", 5, 0x00df },
|
|
{"agrave", 6, 0x00e0 },
|
|
{"aacute", 6, 0x00e1 },
|
|
{"acirc", 5, 0x00e2 },
|
|
{"atilde", 6, 0x00e3 },
|
|
{"auml", 4, 0x00e4 },
|
|
{"aring", 5, 0x00e5 },
|
|
{"aelig", 5, 0x00e6 },
|
|
{"ccedil", 6, 0x00e7 },
|
|
{"egrave", 6, 0x00e8 },
|
|
{"eacute", 6, 0x00e9 },
|
|
{"ecirc", 5, 0x00ea },
|
|
{"euml", 4, 0x00eb },
|
|
{"igrave", 6, 0x00ec },
|
|
{"iacute", 6, 0x00ed },
|
|
{"icirc", 5, 0x00ee },
|
|
{"iuml", 4, 0x00ef },
|
|
{"eth", 3, 0x00f0 },
|
|
{"ntilde", 6, 0x00f1 },
|
|
{"ograve", 6, 0x00f2 },
|
|
{"oacute", 6, 0x00f3 },
|
|
{"ocirc", 5, 0x00f4 },
|
|
{"otilde", 6, 0x00f5 },
|
|
{"ouml", 4, 0x00f6 },
|
|
{"divide", 6, 0x00f7 },
|
|
{"oslash", 6, 0x00f8 },
|
|
{"ugrave", 6, 0x00f9 },
|
|
{"uacute", 6, 0x00fa },
|
|
{"ucirc", 5, 0x00fb },
|
|
{"uuml", 4, 0x00fc },
|
|
{"yacute", 6, 0x00fd },
|
|
{"thorn", 5, 0x00fe },
|
|
{"yuml", 4, 0x00ff },
|
|
{"OElig", 5, 0x0152 },
|
|
{"oelig", 5, 0x0153 },
|
|
{"Scaron", 6, 0x0160 },
|
|
{"scaron", 6, 0x0161 },
|
|
{"Yuml", 4, 0x0178 },
|
|
{"fnof", 4, 0x0192 },
|
|
{"circ", 4, 0x02c6 },
|
|
{"tilde", 5, 0x02dc },
|
|
{"Alpha", 5, 0x0391 },
|
|
{"Beta", 4, 0x0392 },
|
|
{"Gamma", 5, 0x0393 },
|
|
{"Delta", 5, 0x0394 },
|
|
{"Epsilon", 7, 0x0395 },
|
|
{"Zeta", 4, 0x0396 },
|
|
{"Eta", 3, 0x0397 },
|
|
{"Theta", 5, 0x0398 },
|
|
{"Iota", 4, 0x0399 },
|
|
{"Kappa", 5, 0x039a },
|
|
{"Lambda", 6, 0x039b },
|
|
{"Mu", 2, 0x039c },
|
|
{"Nu", 2, 0x039d },
|
|
{"Xi", 2, 0x039e },
|
|
{"Omicron", 7, 0x039f },
|
|
{"Pi", 2, 0x03a0 },
|
|
{"Rho", 3, 0x03a1 },
|
|
{"Sigma", 5, 0x03a3 },
|
|
{"Tau", 3, 0x03a4 },
|
|
{"Upsilon", 7, 0x03a5 },
|
|
{"Phi", 3, 0x03a6 },
|
|
{"Chi", 3, 0x03a7 },
|
|
{"Psi", 3, 0x03a8 },
|
|
{"Omega", 5, 0x03a9 },
|
|
{"alpha", 5, 0x03b1 },
|
|
{"beta", 4, 0x03b2 },
|
|
{"gamma", 5, 0x03b3 },
|
|
{"delta", 5, 0x03b4 },
|
|
{"epsilon", 7, 0x03b5 },
|
|
{"zeta", 4, 0x03b6 },
|
|
{"eta", 3, 0x03b7 },
|
|
{"theta", 5, 0x03b8 },
|
|
{"iota", 4, 0x03b9 },
|
|
{"kappa", 5, 0x03ba },
|
|
{"lambda", 6, 0x03bb },
|
|
{"mu", 2, 0x03bc },
|
|
{"nu", 2, 0x03bd },
|
|
{"xi", 2, 0x03be },
|
|
{"omicron", 7, 0x03bf },
|
|
{"pi", 2, 0x03c0 },
|
|
{"rho", 3, 0x03c1 },
|
|
{"sigmaf", 6, 0x03c2 },
|
|
{"sigma", 5, 0x03c3 },
|
|
{"tau", 3, 0x03c4 },
|
|
{"upsilon", 7, 0x03c5 },
|
|
{"phi", 3, 0x03c6 },
|
|
{"chi", 3, 0x03c7 },
|
|
{"psi", 3, 0x03c8 },
|
|
{"omega", 5, 0x03c9 },
|
|
{"thetasym", 8, 0x03d1 },
|
|
{"upsih", 5, 0x03d2 },
|
|
{"piv", 3, 0x03d6 },
|
|
{"ensp", 4, 0x2002 },
|
|
{"emsp", 4, 0x2003 },
|
|
{"thinsp", 6, 0x2009 },
|
|
{"zwnj", 4, 0x200c },
|
|
{"zwj", 3, 0x200d },
|
|
{"lrm", 3, 0x200e },
|
|
{"rlm", 3, 0x200f },
|
|
{"ndash", 5, 0x2013 },
|
|
{"mdash", 5, 0x2014 },
|
|
{"lsquo", 5, 0x2018 },
|
|
{"rsquo", 5, 0x2019 },
|
|
{"sbquo", 5, 0x201a },
|
|
{"ldquo", 5, 0x201c },
|
|
{"rdquo", 5, 0x201d },
|
|
{"bdquo", 5, 0x201e },
|
|
{"dagger", 6, 0x2020 },
|
|
{"Dagger", 6, 0x2021 },
|
|
{"bull", 4, 0x2022 },
|
|
{"hellip", 6, 0x2026 },
|
|
{"permil", 6, 0x2030 },
|
|
{"prime", 5, 0x2032 },
|
|
{"Prime", 5, 0x2033 },
|
|
{"lsaquo", 6, 0x2039 },
|
|
{"rsaquo", 6, 0x203a },
|
|
{"oline", 5, 0x203e },
|
|
{"frasl", 5, 0x2044 },
|
|
{"euro", 4, 0x20ac },
|
|
{"image", 5, 0x2111 },
|
|
{"weierp", 6, 0x2118 },
|
|
{"real", 4, 0x211c },
|
|
{"trade", 5, 0x2122 },
|
|
{"alefsym", 7, 0x2135 },
|
|
{"larr", 4, 0x2190 },
|
|
{"uarr", 4, 0x2191 },
|
|
{"rarr", 4, 0x2192 },
|
|
{"darr", 4, 0x2193 },
|
|
{"harr", 4, 0x2194 },
|
|
{"crarr", 5, 0x21b5 },
|
|
{"lArr", 4, 0x21d0 },
|
|
{"uArr", 4, 0x21d1 },
|
|
{"rArr", 4, 0x21d2 },
|
|
{"dArr", 4, 0x21d3 },
|
|
{"hArr", 4, 0x21d4 },
|
|
{"forall", 6, 0x2200 },
|
|
{"part", 4, 0x2202 },
|
|
{"exist", 5, 0x2203 },
|
|
{"empty", 5, 0x2205 },
|
|
{"nabla", 5, 0x2207 },
|
|
{"isin", 4, 0x2208 },
|
|
{"notin", 5, 0x2209 },
|
|
{"ni", 2, 0x220b },
|
|
{"prod", 4, 0x220f },
|
|
{"sum", 3, 0x2211 },
|
|
{"minus", 5, 0x2212 },
|
|
{"lowast", 6, 0x2217 },
|
|
{"radic", 5, 0x221a },
|
|
{"prop", 4, 0x221d },
|
|
{"infin", 5, 0x221e },
|
|
{"ang", 3, 0x2220 },
|
|
{"and", 3, 0x2227 },
|
|
{"or", 2, 0x2228 },
|
|
{"cap", 3, 0x2229 },
|
|
{"cup", 3, 0x222a },
|
|
{"int", 3, 0x222b },
|
|
{"there4", 6, 0x2234 },
|
|
{"sim", 3, 0x223c },
|
|
{"cong", 4, 0x2245 },
|
|
{"asymp", 5, 0x2248 },
|
|
{"ne", 2, 0x2260 },
|
|
{"equiv", 5, 0x2261 },
|
|
{"le", 2, 0x2264 },
|
|
{"ge", 2, 0x2265 },
|
|
{"sub", 3, 0x2282 },
|
|
{"sup", 3, 0x2283 },
|
|
{"nsub", 4, 0x2284 },
|
|
{"sube", 4, 0x2286 },
|
|
{"supe", 4, 0x2287 },
|
|
{"oplus", 5, 0x2295 },
|
|
{"otimes", 6, 0x2297 },
|
|
{"perp", 4, 0x22a5 },
|
|
{"sdot", 4, 0x22c5 },
|
|
{"lceil", 5, 0x2308 },
|
|
{"rceil", 5, 0x2309 },
|
|
{"lfloor", 6, 0x230a },
|
|
{"rfloor", 6, 0x230b },
|
|
{"lang", 4, 0x2329 },
|
|
{"rang", 4, 0x232a },
|
|
{"loz", 3, 0x25ca },
|
|
{"spades", 6, 0x2660 },
|
|
{"clubs", 5, 0x2663 },
|
|
{"hearts", 6, 0x2665 },
|
|
{"diams", 5, 0x2666 },
|
|
|
|
{NULL, 0, 0x0000 },
|
|
};
|
|
|
|
|
|
uint16
|
|
pa_NumericCharacterReference(unsigned char *in, uint32 inlen, uint32 *inread,
|
|
Bool force )
|
|
{
|
|
unsigned char b;
|
|
uint32 i;
|
|
uint32 ncr;
|
|
|
|
|
|
for( i = 0; i < inlen; i++ ){
|
|
b = in[i];
|
|
if ((b < '0') || (b > '9')) break;
|
|
}
|
|
|
|
if( i == 0 ){
|
|
*inread = 0;
|
|
return '?';
|
|
}
|
|
|
|
if( (i >= inlen) && (!force) ){
|
|
*inread = (inlen + 1);
|
|
return '?';
|
|
}
|
|
inlen = i;
|
|
|
|
ncr = 0;
|
|
for (i = 0; i < inlen; i++){
|
|
ncr = 10 * ncr + ( in[i] - '0' );
|
|
}
|
|
*inread = i;
|
|
|
|
/* if it is an illegal NCR, then place the '?' instead of,
|
|
if it is an illegal NE, then treat is as plain text */
|
|
if( ncr & 0xffff0000 ) return '?';
|
|
|
|
return (uint16)ncr;
|
|
}
|
|
|
|
|
|
|
|
/*************************************
|
|
* Function: pa_map_escape
|
|
*
|
|
* Description: This function maps the passed in amperstand
|
|
* escape sequence to a single 8 bit char (ISO 8859-1).
|
|
*
|
|
* Params: Takes a buffer, and a length for that buffer. (buf is NOT
|
|
* a \0 terminated string). Also takes a pointer to an escape length
|
|
* which is home much of the original buffer was used.
|
|
* And an out buffer, and it's outbuflen.
|
|
*
|
|
* Returns: On error, which means it was not passed
|
|
* a valid amperstand escape sequence, it stores the
|
|
* char value '\0' in "out". It also returns the length of
|
|
* the part of the passed buffer that was replaced.
|
|
* Also, the number of bytes written to the out buffer (outlen).
|
|
*************************************/
|
|
static void
|
|
pa_map_escape(char *buf, int32 len, int32 *esc_len, unsigned char *out,
|
|
uint16 outbuflen, uint16 *outlen, Bool force, uint16 win_csid)
|
|
{
|
|
uint16 unicode;
|
|
char val;
|
|
int32 cnt;
|
|
|
|
*esc_len = 0;
|
|
val = 0;
|
|
|
|
/* Navigator View->source does not support Unicode yet */
|
|
if( win_csid == 0 ) win_csid = CS_FE_ASCII;
|
|
|
|
/*
|
|
* Skip the amperstand character at the start.
|
|
*/
|
|
buf++;
|
|
len--;
|
|
|
|
/*
|
|
* Ampersand followed by number sign specifies the decimal
|
|
* value of the character to be placed here.
|
|
*/
|
|
if (*buf == '#')
|
|
{
|
|
unicode = pa_NumericCharacterReference((unsigned char *) (buf + 1),
|
|
(uint32) (len - 1), (uint32 *) esc_len, force );
|
|
*esc_len += 1;
|
|
|
|
#ifdef PA_REMAP_128_TO_160_ILLEGAL_NCR
|
|
/* for some illegal, but popular usage */
|
|
if( unicode >= 0x0080 && unicode <= 0x009f )
|
|
unicode = PA_HackTable[ unicode - 0x0080 ];
|
|
#endif
|
|
}
|
|
/*
|
|
* Else we have to look this escape up in
|
|
* the array of valid escapes.
|
|
*/
|
|
else
|
|
{
|
|
/*
|
|
* Else we have to look this escape up in
|
|
* the array of valid escapes.
|
|
* Since we are searching on the PA_Name2Unicode[].NE string,
|
|
* and not on the escape entered, this lets us partial
|
|
* match if PA_Name2Unicode[].NE matches a prefix of the escape.
|
|
* This means no one PA_Name2Unicode[].NE should be a prefix
|
|
* of any other PA_Name2Unicode[].NE.
|
|
*/
|
|
cnt = 0;
|
|
while( PA_Name2Unicode[cnt].NE != NULL )
|
|
{
|
|
if (*buf == *(PA_Name2Unicode[cnt].NE)
|
|
&& (XP_STRNCMP(buf+1, PA_Name2Unicode[cnt].NE+1,
|
|
PA_Name2Unicode[cnt].len-1) == 0))
|
|
{
|
|
/*
|
|
* Only match a prefix if the next
|
|
* character is NOT an alphanumeric
|
|
*/
|
|
/*
|
|
* The following bug have been related to the following line
|
|
* bug 241429, 57959, 33262, 57912 if you work on this.
|
|
* After we add HTML 4.0 entity table we change the algorithm to prevent
|
|
* &part= be treat as name entity.
|
|
* The algorithm is as following:
|
|
* If the unicode value is in the Latin 1 range
|
|
* We mantain the SGML compliant
|
|
* otherwise
|
|
* We only treat it as name entity if it start from & and end with a ;
|
|
*/
|
|
if(((PA_Name2Unicode[cnt].unicode <= 256) &&
|
|
(!XP_IS_ALPHA(*(buf + PA_Name2Unicode[cnt].len))) &&
|
|
(!XP_IS_DIGIT(*(buf + PA_Name2Unicode[cnt].len)))
|
|
) || (';' == (*(buf + PA_Name2Unicode[cnt].len)))
|
|
)
|
|
{
|
|
unicode = PA_Name2Unicode[cnt].unicode;
|
|
*esc_len = PA_Name2Unicode[cnt].len;
|
|
break;
|
|
}
|
|
}
|
|
cnt++;
|
|
}
|
|
|
|
/*
|
|
* for invalid escapes, return '\0'
|
|
*/
|
|
if( PA_Name2Unicode[cnt].NE == NULL )
|
|
unicode = NOT_USED; /* treated as plain text, just for NE */
|
|
}
|
|
|
|
|
|
if( unicode == NOT_USED )
|
|
{
|
|
out[0] = 0;
|
|
return;
|
|
}
|
|
|
|
|
|
/* if( outbuflen >= INTL_UnicodeToStrLen( win_csid, &unicode, 1 ) )
|
|
I needn't to do that, for the outbuflen is always larget than outlen ,
|
|
the result of generated "out" through NE/NCR is one byte or two bytes */
|
|
INTL_UnicodeToStr( win_csid, &unicode, 1, out, outbuflen );
|
|
|
|
/* Remap NBSP to international version */
|
|
/* We need to do this in any case */
|
|
if(unicode == 0x00a0)
|
|
{
|
|
strcpy((char*)out, INTL_NonBreakingSpace(win_csid));
|
|
}
|
|
|
|
*outlen = XP_STRLEN((char*) out );
|
|
|
|
/* Do fallback for Copyright, Tradmark and Register Sign */
|
|
/* We try those non-standard secret platform specific */
|
|
/* codepoint first. */
|
|
/* If there are no such, do an ASCII fallback */
|
|
if((*outlen == 1) && (*out == '?'))
|
|
{
|
|
char* r = NULL;
|
|
#if defined(XP_WIN) || defined(XP_OS2)
|
|
/*
|
|
For Winodows
|
|
All the CP 125x single byte pages encode
|
|
Copyright Signed as A9
|
|
Register Signed as AE
|
|
Tradmark Signed as 99
|
|
Two Byte Asian Font do not have those sign
|
|
|
|
*/
|
|
switch(unicode)
|
|
{
|
|
case 0x20AC:
|
|
r = "EUR";
|
|
break;
|
|
case 0x00A9:
|
|
switch(win_csid)
|
|
{
|
|
case CS_LATIN1: case CS_CP_1250:
|
|
case CS_CP_1251: case CS_CP_1253:
|
|
case CS_CP_1254: case CS_8859_7:
|
|
case CS_8859_9:
|
|
r = "\xa9";
|
|
break;
|
|
default:
|
|
r = "(C)";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x00AE:
|
|
switch(win_csid)
|
|
{
|
|
case CS_LATIN1: case CS_CP_1250:
|
|
case CS_CP_1251: case CS_CP_1253:
|
|
case CS_CP_1254: case CS_8859_7:
|
|
case CS_8859_9:
|
|
r = "\xae";
|
|
break;
|
|
default:
|
|
r = "(R)";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2122:
|
|
switch(win_csid)
|
|
{
|
|
case CS_LATIN1: case CS_CP_1250:
|
|
case CS_CP_1251: case CS_CP_1253:
|
|
case CS_CP_1254: case CS_8859_7:
|
|
case CS_8859_9:
|
|
r = "\x99";
|
|
break;
|
|
default:
|
|
r = "[tm]";
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
#endif
|
|
#ifdef XP_MAC
|
|
/*
|
|
For Mac:
|
|
Japanese and Chinese font usually have secret glyph
|
|
Copyright Signed at 0xfd
|
|
Tradmark Signed at 0xff
|
|
|
|
*/
|
|
switch(unicode)
|
|
{
|
|
case 0x20AC:
|
|
r = "EUR";
|
|
break;
|
|
case 0x00AD:
|
|
r = "-";
|
|
break;
|
|
case 0x00A9:
|
|
switch(win_csid)
|
|
{
|
|
case CS_MAC_ROMAN:
|
|
case CS_MAC_GREEK:
|
|
r = "\xd9";
|
|
break;
|
|
case CS_MAC_CE:
|
|
case CS_MAC_CYRILLIC:
|
|
r = "\xa9";
|
|
break;
|
|
case CS_KSC_8BIT:
|
|
r = "\x93";
|
|
break;
|
|
case CS_SJIS:
|
|
case CS_GB_8BIT:
|
|
case CS_CNS_8BIT:
|
|
r = "\xfd";
|
|
break;
|
|
case CS_TIS620:
|
|
r = "\xfb";
|
|
break;
|
|
default:
|
|
r = "(C)";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x00AE:
|
|
switch(win_csid)
|
|
{
|
|
case CS_MAC_ROMAN:
|
|
case CS_MAC_GREEK:
|
|
case CS_MAC_CE:
|
|
case CS_MAC_CYRILLIC:
|
|
r = "\xa8";
|
|
break;
|
|
case CS_TIS620:
|
|
r = "\xfa";
|
|
break;
|
|
default:
|
|
r = "(R)";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2122:
|
|
switch(win_csid)
|
|
{
|
|
case CS_MAC_ROMAN:
|
|
case CS_MAC_GREEK:
|
|
case CS_MAC_CE:
|
|
case CS_MAC_CYRILLIC:
|
|
r = "\xaa";
|
|
break;
|
|
case CS_SJIS:
|
|
case CS_GB_8BIT:
|
|
case CS_CNS_8BIT:
|
|
r = "\xfe";
|
|
break;
|
|
case CS_TIS620:
|
|
r = "\xee";
|
|
break;
|
|
default:
|
|
r = "[tm]"; break;
|
|
}
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
#ifdef XP_UNIX
|
|
switch(unicode)
|
|
{
|
|
case 0x20AC:
|
|
r = "EUR";
|
|
break;
|
|
case 0x00A9:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "(C)";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x00AE:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "(R)";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x0152:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "OE";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x0153:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "oe";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x0160:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "S";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x0161:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "s";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x0178:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "Y";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x0192:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "f";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x02C6:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "^";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x02DC:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "~";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2013:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "-";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2014:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "-";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2018:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "`";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2019:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "'";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x201C:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "\"";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x201D:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "\"";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x201A:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = ",";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x201E:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = ",,";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2022:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "*";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2026:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "...";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2039:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "<";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x203A:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = ">";
|
|
break;
|
|
}
|
|
break;
|
|
case 0x2122:
|
|
switch(win_csid)
|
|
{
|
|
default:
|
|
r = "[tm]";
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
#endif
|
|
if(r != NULL )
|
|
{
|
|
strcpy((char*)out, r);
|
|
*outlen = XP_STRLEN((char*)out);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*************************************
|
|
* Function: pa_ExpandEscapes
|
|
*
|
|
* Description: Go through the passed in buffer of text, find and replace
|
|
* all amperstand escape sequences with their character
|
|
* equivilants. The buffer is modified in place,
|
|
* each escape replacement shortens the contents of the
|
|
* buffer.
|
|
*
|
|
* Params: Takes a buffer, and a length for that buffer. (buf is NOT
|
|
* a \0 terminated string). Also takes a pointer to a new length.
|
|
* This is the new (possibly shorter) length of the buffer
|
|
* after the escapes have been replaced. Finally takes a boolean
|
|
* that specifies whether partial escapes are allowed or not.
|
|
*
|
|
* Returns: The new length of the string in the nlen parameter. Also
|
|
* returns a NULL character pointer on success.
|
|
* If the buffer passed contained a possibly truncated
|
|
* amperstand escape, that is considered a partial success.
|
|
* Escapes are replace in all the buffer up to the truncated
|
|
* escape, nlen is the length of the new sub-buffer, and
|
|
* the return code is a pointer to the begining of the truncated
|
|
* escape in the original buffer.
|
|
*************************************/
|
|
|
|
char *
|
|
pa_ExpandEscapes(char *buf, int32 len, int32 *nlen, Bool force, int16 win_csid)
|
|
{
|
|
int32 cnt;
|
|
char *from_ptr;
|
|
char *to_ptr;
|
|
unsigned char out[10];
|
|
uint16 outlen;
|
|
unsigned char *p;
|
|
|
|
*nlen = 0;
|
|
|
|
/*
|
|
* A NULL buffer translates to a NULL buffer.
|
|
*/
|
|
if (buf == NULL)
|
|
{
|
|
return(NULL);
|
|
}
|
|
|
|
/*
|
|
* Look through the passed buf for the beginning of amperstand
|
|
* escapes. They begin with an amperstand followed by a letter
|
|
* or a number sign.
|
|
*/
|
|
from_ptr = buf;
|
|
to_ptr = buf;
|
|
cnt = 0;
|
|
while (cnt < len)
|
|
{
|
|
char *tptr2;
|
|
int32 cnt2;
|
|
int32 esc_len;
|
|
|
|
if (*from_ptr == '&')
|
|
{
|
|
/*
|
|
* If this is the last character in the string
|
|
* we don't know if it is an amperstand escape yet
|
|
* or not. We want to save it until the next buffer
|
|
* to make sure. However, if force is true this
|
|
* can't be a partial escape, so asuume an error
|
|
* and cope as best you can.
|
|
*/
|
|
if ((cnt + 1) == len)
|
|
{
|
|
if (force != FALSE)
|
|
{
|
|
*to_ptr++ = *from_ptr++;
|
|
cnt++;
|
|
continue;
|
|
}
|
|
/*
|
|
* This might be a partial amperstand escape.
|
|
* we will have to wait until we have
|
|
* more data. Break the while loop
|
|
* here.
|
|
*/
|
|
break;
|
|
}
|
|
/*
|
|
* else is the next character is a letter or
|
|
* a '#' , this should be some kind of amperstand
|
|
* escape.
|
|
*/
|
|
else if ((XP_IS_ALPHA(*(from_ptr + 1)))||
|
|
(*(from_ptr + 1) == '#'))
|
|
{
|
|
/*
|
|
* This is an amperstand escape! Drop
|
|
* through to the rest of the while loop.
|
|
*/
|
|
}
|
|
/*
|
|
* else this character does not begin an
|
|
* amperstand escape increment pointer, and go
|
|
* back to the start of the while loop.
|
|
*/
|
|
else
|
|
{
|
|
*to_ptr++ = *from_ptr++;
|
|
cnt++;
|
|
continue;
|
|
}
|
|
}
|
|
/*
|
|
* else this character does not begin an amperstand escape
|
|
* increment pointer, and go back to the start of the
|
|
* while loop.
|
|
*/
|
|
else
|
|
{
|
|
*to_ptr++ = *from_ptr++;
|
|
cnt++;
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* If we got here, it is because we think we have
|
|
* an amperstand escape. Amperstand escapes are of arbitrary
|
|
* length, terminated by either a semi-colon, or white space.
|
|
*
|
|
* Find the end of the amperstand escape.
|
|
*/
|
|
tptr2 = from_ptr;
|
|
cnt2 = cnt;
|
|
while (cnt2 < len)
|
|
{
|
|
if ((*tptr2 == ';')||(XP_IS_SPACE(*tptr2)))
|
|
{
|
|
break;
|
|
}
|
|
tptr2++;
|
|
cnt2++;
|
|
}
|
|
/*
|
|
* We may have failed to find the end because this is
|
|
* only a partial amperstand escape. However, if force is
|
|
* true this can't be a partial escape, so asuume and error
|
|
* and cope as best you can.
|
|
*/
|
|
if ((cnt2 == len)&&(force == FALSE))
|
|
{
|
|
/*
|
|
* This might be a partial amperstand escape.
|
|
* we will have to wait until we have
|
|
* more data. Break the while loop
|
|
* here.
|
|
*/
|
|
break;
|
|
}
|
|
|
|
esc_len = 0;
|
|
pa_map_escape(from_ptr, (intn)(tptr2 - from_ptr + 1),
|
|
&esc_len, out, sizeof(out), &outlen, force, win_csid);
|
|
if (esc_len > (tptr2 - from_ptr + 1))
|
|
{
|
|
/*
|
|
* This is a partial numeric char ref.
|
|
*/
|
|
break;
|
|
}
|
|
/*
|
|
* invalid escape sequences return the end of string
|
|
* character. Treat this not as an amperstand escape, but as
|
|
* normal text.
|
|
*/
|
|
if (out[0] == '\0')
|
|
{
|
|
*to_ptr++ = *from_ptr++;
|
|
cnt++;
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* valid escape sequences are replaced in place, with the
|
|
* rest of the buffer copied forward. For improper escape
|
|
* sequences that we think we can recover from, we may have
|
|
* just used part of the beginning of the escape sequence.
|
|
*/
|
|
p = out;
|
|
while (outlen--)
|
|
{
|
|
*to_ptr++ = *p++;
|
|
}
|
|
tptr2 = (char *)(from_ptr + esc_len + 1);
|
|
cnt2 = cnt + esc_len + 1;
|
|
|
|
/*
|
|
* for semi-colon terminated escapes, eat the semi-colon.
|
|
* The test on cnt2 prevents up walking off the end of the
|
|
* buffer on a broken escape we are coping with.
|
|
*/
|
|
if ((cnt2 < len)&&(*tptr2 == ';'))
|
|
{
|
|
tptr2++;
|
|
cnt2++;
|
|
}
|
|
from_ptr = tptr2;
|
|
cnt = cnt2;
|
|
}
|
|
|
|
/*
|
|
* If we broke the while loop early,
|
|
* it is because we may be holding a
|
|
* partial amperstand escape.
|
|
* return a pointer to that partial escape.
|
|
*/
|
|
if (cnt < len)
|
|
{
|
|
*nlen = (int32)(to_ptr - buf);
|
|
return(from_ptr);
|
|
}
|
|
else
|
|
{
|
|
*nlen = (int32)(to_ptr - buf);
|
|
return(NULL);
|
|
}
|
|
}
|
|
|
|
#ifdef PROFILE
|
|
#pragma profile off
|
|
#endif
|
|
|