/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You under the Apache License, Version 2.0 * (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.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* * $Id: Xlat_Win32RC.cpp 698557 2008-09-24 13:19:01Z amassari $ */ // --------------------------------------------------------------------------- // Includes // --------------------------------------------------------------------------- #include "Xlat.hpp" // --------------------------------------------------------------------------- // Win32RCFormatter: Public Constructors and Destructor // --------------------------------------------------------------------------- Win32RCFormatter::Win32RCFormatter() : fCurDomainName(0) , fMsgOffset(0) , fOutFl(0) { } Win32RCFormatter::~Win32RCFormatter() { XMLString::release(&fCurDomainName); } // --------------------------------------------------------------------------- // Win32RCFormatter: Implementation of the formatter interface // --------------------------------------------------------------------------- void Win32RCFormatter::endDomain(const XMLCh* const domainName , const unsigned int msgCount) { // And close out the message table declaration fwprintf(fOutFl, L"END\n"); } void Win32RCFormatter::endMsgType(const MsgTypes type) { // No-op for us } void Win32RCFormatter::endOutput() { // Close the output file fclose(fOutFl); } void Win32RCFormatter::nextMessage( const XMLCh* const msgText , const XMLCh* const msgId , const unsigned int messageId , const unsigned int curId) { // // We do not transcode to the output format in this case. Instead we // just store the straight Unicode format. Because we cannot assume 'L' // type prefix support, we have to put them out as numeric character // values. // fwprintf(fOutFl, L" %-16d L\"", messageId + fMsgOffset); bool isOnlyASCII=true; const XMLCh* rawData = msgText; while (*rawData) { // don't allow " or \ in the message if(*rawData >= 0x80 || strchr("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789{}[]()',-+ &.;:<>?!|*=/#_", *rawData)==NULL) { isOnlyASCII=false; break; } rawData++; } if(isOnlyASCII) fwprintf(fOutFl, msgText); else { const XMLCh* rawData = msgText; while (*rawData) fwprintf(fOutFl, L"\\x%04lX", *rawData++); } fwprintf(fOutFl, L"\"\n"); } void Win32RCFormatter::startDomain( const XMLCh* const domainName , const XMLCh* const) { // // We have a different array name for each domain, so store that for // later use and for use below. // XMLString::release(&fCurDomainName); if (!XMLString::compareString(XMLUni::fgXMLErrDomain, domainName)) { fCurDomainName = XMLString::transcode("gXMLErrArray"); fMsgOffset = 0; } else if (!XMLString::compareString(XMLUni::fgExceptDomain, domainName)) { fCurDomainName = XMLString::transcode("gXMLExceptArray"); fMsgOffset = 0x2000; } else if (!XMLString::compareString(XMLUni::fgValidityDomain, domainName)) { fCurDomainName = XMLString::transcode("gXMLValidityArray"); fMsgOffset = 0x4000; } else if (!XMLString::compareString(XMLUni::fgXMLDOMMsgDomain, domainName)) { fCurDomainName = XMLString::transcode("gXMLDOMMsgArray"); fMsgOffset = 0x6000; } else { wprintf(L"Unknown message domain: %s\n", domainName); throw ErrReturn_SrcFmtError; } // // Output the leading part of the array declaration. Its just an // array of pointers to Unicode chars. // fwprintf ( fOutFl , L"STRINGTABLE DISCARDABLE\nBEGIN\n" ); } void Win32RCFormatter::startMsgType(const MsgTypes type) { // No-op for us } void Win32RCFormatter::startOutput( const XMLCh* const locale , const XMLCh* const outPath) { // // Ok, lets try to open the the output file. All of the messages for all // the domains are put into a single message tabble in a single RC file, // which can be linked into the program. // // CppErrMsgs_xxxx.RC // // where xxx is the locale suffix passed in. // const unsigned int bufSize = 4095; XMLCh tmpBuf[bufSize + 1]; // make sure the append will work tmpBuf[0] = 0; XMLCh *tmpXMLStr = XMLString::transcode("CppErrMsgs_"); XMLCh *tmpXMLStr2 = XMLString::transcode(".RC"); XMLString::catString(tmpBuf, outPath); XMLString::catString(tmpBuf, tmpXMLStr ); XMLString::catString(tmpBuf, locale); XMLString::catString(tmpBuf, tmpXMLStr2 ); XMLString::release(&tmpXMLStr); XMLString::release(&tmpXMLStr2); char *tmpStr = XMLString::transcode(tmpBuf); fOutFl = fopen(tmpStr, "wt"); XMLString::release(&tmpStr); if ((!fOutFl) || (fwide(fOutFl,1) < 0)) { wprintf(L"Could not open the output file: %s\n\n", xmlStrToPrintable(tmpBuf) ); releasePrintableStr throw ErrReturn_OutFileOpenFailed; } // // Ok, lets output the grunt data at the start of the file. We put out a // comment that indicates its a generated file, and the title string. // fwprintf ( fOutFl , L"// ----------------------------------------------------------------\n" L"// This file was generated from the XML error message source.\n" L"// so do not edit this file directly!!\n" L"// ----------------------------------------------------------------\n\n" ); }