/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: NPL 1.1/GPL 2.0/LGPL 2.1 * * 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 Communicator client code. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * * Alternatively, the contents of this file may be used under the terms of * either 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 NPL, 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 NPL, 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; }