зеркало из https://github.com/mozilla/gecko-dev.git
159 строки
5.4 KiB
C++
159 строки
5.4 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 Communicator client code.
|
|
*
|
|
* The Initial Developer of the Original Code is Netscape Communications Corporation.
|
|
* Portions created by Netscape Communications Corporation are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All
|
|
* Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*
|
|
* 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 LGPL or the GPL. 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 "nsUEscapeToUnicode.h"
|
|
|
|
//----------------------------------------------------------------------
|
|
// Subclassing of nsTableDecoderSupport class [implementation]
|
|
|
|
NS_IMETHODIMP nsUEscapeToUnicode::GetMaxLength(const char * aSrc,
|
|
PRInt32 aSrcLength,
|
|
PRInt32 * aDestLength)
|
|
{
|
|
*aDestLength = (aSrcLength + mBufferLen);
|
|
return NS_OK;
|
|
}
|
|
NS_IMETHODIMP nsUEscapeToUnicode::Convert(
|
|
const char * aSrc, PRInt32 * aSrcLength,
|
|
PRUnichar * aDest, PRInt32 * aDestLength)
|
|
{
|
|
const char* srcPtr = aSrc;
|
|
const char* srcEnd = aSrc + *aSrcLength;
|
|
PRUnichar* destPtr = aDest;
|
|
PRUnichar* destEnd = aDest + *aDestLength;
|
|
while((mBufferLen >0) && (destPtr < destEnd))
|
|
*destPtr++ = mBuffer[--mBufferLen];
|
|
|
|
for(;(srcPtr < srcEnd) && (destPtr < destEnd); srcPtr++)
|
|
{
|
|
switch(mState)
|
|
{
|
|
case 0:
|
|
if('\\' == *srcPtr) {
|
|
mState++;
|
|
} else {
|
|
if(0x80 == (0x80 & *srcPtr))
|
|
{
|
|
*destPtr++ = (PRUnichar)0xFFFD;
|
|
} else {
|
|
*destPtr++ = (PRUnichar)*srcPtr;
|
|
}
|
|
}
|
|
break;
|
|
case 1: // got a '\'
|
|
if(('u' == *srcPtr) || ('U' == *srcPtr)) {
|
|
mState++;
|
|
mData=0;
|
|
} else if(('n' == *srcPtr) || ('r' == *srcPtr) || ('t' == *srcPtr)) {
|
|
mState =0;
|
|
if((destPtr+2) >= destEnd) {
|
|
// notice the data in mBuffer is stored in reversed order
|
|
mBufferLen = 2;
|
|
mBuffer[1] = PRUnichar('\\') ;
|
|
mBuffer[0] = (PRUnichar)*srcPtr;
|
|
goto error;
|
|
}
|
|
*destPtr++ = PRUnichar('\\');
|
|
*destPtr++ = (PRUnichar)*srcPtr;
|
|
} else {
|
|
mState =0;
|
|
*destPtr++ = (PRUnichar)*srcPtr;
|
|
}
|
|
break;
|
|
case 2: // got \u
|
|
case 3: // got \u1
|
|
case 4: // got \u12
|
|
case 5: // got \u123
|
|
if(('0' <= *srcPtr) && (*srcPtr <='9')) {
|
|
mData = (mData << 4) | (*srcPtr - '0');
|
|
mState++;
|
|
} else if(('a' <= *srcPtr) && (*srcPtr <='f')) {
|
|
mData = (mData << 4) | ((*srcPtr - 'a') + 0x0a);
|
|
mState++;
|
|
} else if(('A' <= *srcPtr) && (*srcPtr <='F')) {
|
|
mData = (mData << 4) | ((*srcPtr - 'A') + 0x0a);
|
|
mState++;
|
|
} else {
|
|
// if we got \u , \u2 , \u34
|
|
if((destPtr+2) >= destEnd) {
|
|
// notice the data in mBuffer is stored in reversed order
|
|
mBufferLen = 2;
|
|
mBuffer[1] = (PRUnichar) mData;
|
|
mBuffer[0] = (PRUnichar)*srcPtr;
|
|
mState=0;
|
|
goto error;
|
|
}
|
|
*destPtr++ = (PRUnichar) mData;
|
|
*destPtr++ = (PRUnichar)*srcPtr;
|
|
mState=0;
|
|
}
|
|
if(6==mState) // got \u1234
|
|
{
|
|
*destPtr++ = (PRUnichar) mData;
|
|
mState = 0;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
};
|
|
}
|
|
|
|
*aDestLength = destPtr - aDest;
|
|
*aSrcLength = srcPtr - aSrc;
|
|
return NS_OK;
|
|
|
|
error:
|
|
*aDestLength = destPtr - aDest;
|
|
*aSrcLength = srcPtr - aSrc;
|
|
return NS_OK_UDEC_MOREOUTPUT;
|
|
}
|
|
|
|
NS_IMETHODIMP nsUEscapeToUnicode::Reset()
|
|
{
|
|
mState =0;
|
|
mData=0;
|
|
mBufferLen =0;
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
nsresult NEW_UEscapeToUnicode(nsISupports **aResult)
|
|
{
|
|
*aResult = new nsUEscapeToUnicode();
|
|
return (NULL == *aResult) ? NS_ERROR_OUT_OF_MEMORY : NS_OK;
|
|
}
|