//----------------------------------------------------------------------------- // This files contains the module name for this mini driver. Each mini driver // must have a unique module name. The module name is used to obtain the // module handle of this Mini Driver. The module handle is used by the // generic library to load in tables from the Mini Driver. // It also contains Install() for upgrading 3.0 driver to 3.1. // //----------------------------------------------------------------------------- #include "strings.h" char *rgchModuleName = "PCL5CH"; char szNone[]=""; // The following are defined to ensure that we upgrade correctly from the // HPPCL5a, HPPCL5e and Win 3.11 HPPCL5MS drivers #define MAX_LJ4_MBMEMSETTING 68 // from HP tech specification #define KB_THRESHOLD 200 // kb range check,needed for conversion #define PRINTDRIVER #include #include "gdidefs.inc" #include "mdevice.h" #include "unidrv.h" #include "minidriv.h" #ifndef _INC_WINDOWSX #include #endif #ifndef WINNT //swpark //#define DEBUG 1 #ifdef DEBUG #include "stdarg.h" void NEAR CDECL DBMSG(LPSTR, ...); #endif #endif //WINNT #ifndef WINNT short NEAR PASCAL MakeAppName(LPSTR,LPCSTR,short); // typedef for atom stuff--what a nuisance! typedef struct tagSFNODE { WORD wIndex; ATOM atom; } SFNODE, FAR *LPSFNODE; // Typedef for Font Installer procedure typedef int (FAR * PASCAL SOFTFONTPROC)(HWND,LPSTR,LPSTR,BOOL,int,int); HINSTANCE hInst; #define DEFAULT_INT 32767 #define SOFT_FONT_THRES 25 // build font summary, if over this limit #define MAX_CART_INDEX 12 #define TMPSIZE 256 // Define these so they happily use the same values as the HPPCL5E driver. #define GS_PHOTO 0 #define GS_LINEART 1 #define GS_SCANJET 2 // map old HPPCL5a's cartindex to unidrv's FONTCART index for newer cartridges. // This mapping table is created based on the old HPPCL5a .rc file. // Note that we do not have "International Collection" cartridge and we // map it to index 0 (arbitrarily). int rgNewCartMap[12] = {0, 8, 7, 2, 3, 0, 5, 6, 1, 4, 9, 10}; // String to determine if we have a member of the LaserJet 4 family char szLJ4[]="HP LaserJet 4"; // Stuff needed for mapping old facenames to new versions #ifndef NOFONTMAP typedef struct tagFACEMAP { char szOldFace[LF_FACESIZE]; char szNewFace[LF_FACESIZE]; } FACEMAP, NEAR * NPFACEMAP; typedef struct tagFACEINDEX { BYTE cFirstChar; BYTE bIndex; } FACEINDEX, NEAR * NPFACEINDEX; FACEMAP FaceMap[]={{"Albertus (W\x01)", "Albertus Medium"}, {"Albertus Xb (W\x01)", "Albertus Extra Bold"}, {"Antique Olv (W\x01)", "Antique Olive"}, {"Antique Olv Cmpct (W\x01)","Antique Olive Compact"}, {"CG Bodoni (W\x01)", "CG Bodoni"}, {"CG Cent Schl (W\x01)", "CG Century Schoolbook"}, {"CG Omega (W\x01)", "CG Omega"}, {"CG Palacio (W\x01)", "CG Palacio"}, {"CG Times (W\x01)", "CG Times"}, {"Clarendon Cd (W\x01)", "Clarendon Condensed"}, {"Cooper Blk (W\x01)", "Cooper Black"}, {"Coronet (W\x01)", "Coronet"}, {"Courier (W\x01)", "Courier"}, {"Garmond (W\x01)", "Garamond"}, {"ITC Benguat (W\x01)", "ITC Benguiat"}, {"ITC Bookman Db (W\x01)", "ITC Bookman Demi"}, {"ITC Bookman Lt (W\x01)", "ITC Bookman Light"}, {"ITC Souvenir Db (W\x01)", "ITC Souvenir Demi"}, {"ITC Souvenir Lt (W\x01)", "ITC Souvenir Light"}, {"Letter Gothic (W\x01)", "Letter Gothic"}, {"Marigold (W\x01)", "Marigold"}, {"Revue Lt (W\x01)", "Revue Light"}, {"Shannon (W\x01)", "Shannon"}, {"Shannon Xb (W\x01)", "Shannon Extra Bold"}, {"Stymie (W\x01)", "Stymie"}, {"Univers (W\x01)", "Univers"}, {"Univers Cd (W\x01)", "Univers Condensed"}}; FACEINDEX FaceIndex[]={{'A',0}, {'C',4}, {'G',13}, {'I',14}, {'L',19}, {'M',20}, {'R',21}, {'S',22}, {'U',25}, {(BYTE)'\xFF',27}}; // Provide an upper limit // to the search for 'U'. #endif //------------------------------------------------------------------------ // Function: LibMain(hInstance,wDataSeg,cbHeapSize,lpszCmdLine) // // Action: Save the hInstance for this DLL // // Return: 1 //------------------------------------------------------------------------ int WINAPI LibMain (HANDLE hInstance, WORD wDataSeg, WORD cbHeapSize, LPSTR lpszCmdLine) { hInst=hInstance; return 1; } //--------------------------*MakeAppName*--------------------------------------- // Action: compose the name for reading the profile data // Return the length of the actual application name. Return -1 if fails. // //------------------------------------------------------------------------------ short NEAR PASCAL MakeAppName(LPSTR lpAppName, LPCSTR lpPortName, short max) { short length, count; LPCSTR lpTmp; LPCSTR lpLastColon = NULL; length = lstrlen(lpAppName); if (!lpPortName) return length; if (length == 0 || length > max - lstrlen(lpPortName)) return -1; // insert the comma lpAppName[length++] = ','; // append the port name but do not want the last ':', if any. for (lpTmp = lpPortName ; *lpTmp; lpTmp++) if (*lpTmp == ':') lpLastColon = lpTmp; if (lpLastColon && lpLastColon == lpTmp - 1) count = lpLastColon - lpPortName; else count = lpTmp - lpPortName; lstrcpy((LPSTR)&lpAppName[length], lpPortName); length += count; lpAppName[length]='\0'; return length; } #define KEY_BUF_SIZE 256 //--------------------------------------------------------------------------- // Function: GetInt(lpSection,lpKey,lpnValue,nDefault,bRemove) // // Action: Load the appropriate string from the resources, then get the // specified integer from the section. Remove the old entry if // it exists and it bRemove is TRUE. // // Return: TRUE if we actually found a value, FALSE if not. //--------------------------------------------------------------------------- BOOL NEAR PASCAL GetInt(LPSTR lpSection, LPCSTR lpKey, LPINT lpnValue, int nDefault, BOOL bRemove) { char szKeyName[60]; int nTest; if(!HIWORD(lpKey)) { if(LoadString(hInst,LOWORD(lpKey),szKeyName,sizeof(szKeyName))) lpKey=szKeyName; else return FALSE; } nTest=GetProfileInt(lpSection,szKeyName,DEFAULT_INT); if(DEFAULT_INT != nTest) { *lpnValue=nTest; if(bRemove) WriteProfileString(lpSection,szKeyName,NULL); return TRUE; } // Section doesn't exist--use default *lpnValue=nDefault; return FALSE; } //------------------------------------------------------------------------- // Function: WriteInt(lpSection,lpKey,nValue) // // Action: Write an integer value to the specified section of win.ini. // // Return: TRUE if successful, FALSE if not. //------------------------------------------------------------------------- BOOL NEAR PASCAL WriteInt(LPSTR lpSection, LPCSTR lpKey, int nValue) { char szKeyName[60]; char szValue[10]; if(!HIWORD(lpKey)) { if(LoadString(hInst,LOWORD(lpKey),szKeyName,sizeof(szKeyName))) lpKey=szKeyName; else return FALSE; } wsprintf(szValue,"%u",nValue); return WriteProfileString(lpSection,szKeyName,szValue); } //---------------------------*MergeFontLists*----------------------------- // Action: Merge the old and new soft fonts. In most cases when we get // called, we really don't do much of anything, because the // font lists are identical. However, we have to do some fun // stuff to merge the lists if they're different. // We know how many soft font entries exist in each section, // via the "SoftFonts" int, but the entries may be non-consecutive. // // Note: This stomps all over the passed in buffer // // Return: TRUE if successfully conpleted, FALSE if not //------------------------------------------------------------------------ BOOL NEAR PASCAL MergeFontLists(LPSTR lpOldSec, LPSTR lpNewSec, LPSTR lpTmp) { WORD wOldFonts; WORD wNewFonts; LPSFNODE lpFonts; WORD wMergedFonts=0; WORD wLoop; WORD wFound; WORD wNewIndex; BYTE szKey[20]; // Get these values outside of the if statement, otherwise the compiler // may optimize out the assignment of wNewFonts GetInt(lpOldSec,MAKEINTRESOURCE(IDS_SOFTFONTS),&wOldFonts,0,FALSE); GetInt(lpNewSec,MAKEINTRESOURCE(IDS_SOFTFONTS),&wNewFonts,0,FALSE); if(wOldFonts || wNewFonts) { char szFormat[30]; // Get a block big enough for the worst case--no common fonts if(!(lpFonts=(LPSFNODE)GlobalAllocPtr(GHND, (DWORD)(wOldFonts+wNewFonts)*sizeof(SFNODE)))) { return FALSE; } // We need a formatting string LoadString(hInst,IDS_SOFTFONTFORMAT,szFormat,sizeof(szFormat)); // Put fonts from lpNew Sec first in the list. This way, if we have // already updated at least one driver from 5A to 5MS and the soft // fonts haven't changed, our old font summary file is still valid. // Unidrv will automatically recreate the font summary file if it // sees that the number of soft fonts has changed. Even though we // know how many soft font entries exist, we don't know that they // will be sequential. (They may not be if one was added, then // deleted). Keep track of the original offset. Even though the // font installer seems to be 1-based, start looking at 0, just // to be safe. for(wLoop=0,wFound=0;wFound25) { nPrinterMB++; // 25=11MB, 26=13MB, so add an extra MB. if(nValue==28) nPrinterMB+=2; // 27=14MB, 28=17MB, so add 2 extra MB. } } nValue=945*nPrinterMB-245; WriteInt(lpSection,MAKEINTRESOURCE(IDS_NEWMEMORY),nValue); return TRUE; } return FALSE; } //----------------------------------------------------------------------- // Function: Convert5eMemory(lpSection) // // Action: Convert the HPPCL5E memory settings from the old to the new values // // Conversion code has to check the win.ini mem setting so that we upgrade // mem setting correctly // // Return: TRUE if the old section existed, FALSE if not //----------------------------------------------------------------------- BOOL NEAR PASCAL Convert5eMemory(LPSTR lpSection) { unsigned nValue; if(GetInt(lpSection,MAKEINTRESOURCE(IDS_NEWMEMORY),&nValue,1,FALSE)) { if (nValue <= KB_THRESHOLD ) { if (nValue > MAX_LJ4_MBMEMSETTING) nValue = MAX_LJ4_MBMEMSETTING; // force it to max value nValue=900*nValue - 450; // convert to KB, using HP formula WriteInt(lpSection,MAKEINTRESOURCE(IDS_NEWMEMORY),nValue); return TRUE; } } return FALSE; } //----------------------------------------------------------------------- // Function: Convert5MSMemory(lpSection) // // Action: Convert the WFW HPPCL5MS memory settings from the old to the new // values // // Conversion code has to check the win.ini mem setting so that we upgrade // mem setting correctly // // Added for backward compatability // // Return: TRUE if the old section existed, FALSE if not //----------------------------------------------------------------------- BOOL NEAR PASCAL Convert5MSMemory(LPSTR lpSection) { unsigned nValue; if(GetInt(lpSection,MAKEINTRESOURCE(IDS_NEWMEMORY),&nValue,1,TRUE)) { nValue = nValue / 900; // convert to MB value nValue = 945*nValue - 245; // formula used in hppcl5a driver // used to convert to available mem WriteInt(lpSection,MAKEINTRESOURCE(IDS_NEWMEMORY),nValue); return TRUE; } return FALSE; } //------------------------------------------------------------------------ // Function: HandleFontCartridges(lpSection,lpOldDrvSec,lpNewDrvSec) // // Action: Handle font cartridge data // // Return: VOID //------------------------------------------------------------------------ VOID NEAR PASCAL HandleFontCartridges(LPSTR lpSection, LPSTR lpOldDrvSec, LPSTR lpNewDrvSec) { int nCount; // Get the count of cartridges--if there are no cartridges, do nothing. if(GetInt(lpSection,MAKEINTRESOURCE(IDS_CARTRIDGECOUNT),&nCount,0,TRUE)) { char szOldCartKey[16]; char szNewCartKey[16]; short nCart = 0; short i; short index; int nLength1; int nLength2; nLength1=LoadString(hInst,IDS_CARTINDEX,szOldCartKey, sizeof(szOldCartKey)); nLength2=LoadString(hInst,IDS_CARTRIDGE,szNewCartKey, sizeof(szNewCartKey)); for (i = 0; i < nCount; i++) { if (i > 0) wsprintf(szOldCartKey+nLength1,"%d",i); // compose cartridge keyname for current driver wsprintf(szNewCartKey+nLength2,"%d",i+1); if ((index = GetProfileInt(lpNewDrvSec, szOldCartKey, 0)) > 0) { WriteProfileString(lpNewDrvSec, szOldCartKey, NULL); nCart++; if (index <= MAX_CART_INDEX) WriteInt(lpSection,szNewCartKey,rgNewCartMap[index-1]); else // external cartridges. Simply copy the id over. WriteInt(lpSection,szNewCartKey,index); } } // Save the # of cartridges WriteInt(lpSection,MAKEINTRESOURCE(IDS_CARTRIDGECOUNT),nCart); } } //-------------------------------------------------------------------------- // Function: HandleFonts(lpSection,lpDevName,lpPort) // // Action: Deal with soft fonts & font cartridges // // Return: VOID //-------------------------------------------------------------------------- VOID NEAR PASCAL HandleFonts(LPSTR lpSection, LPSTR lpDevName, LPSTR lpPort) { char szOldDrvSec[64]; // HPPCL5A, or HPPCL5E, char szNewDrvSec[64]; // HPPCL5MS, int nCount; BOOL bOldExists=FALSE; // Does old section exist? LoadString(hInst,IDS_OLD_5E_DRIVERNAME,szOldDrvSec,sizeof(szOldDrvSec)); MakeAppName((LPSTR)szOldDrvSec,lpPort,sizeof(szOldDrvSec)); // See if the old section exists at all. Temporarily borrow szNewDrvSec. if(GetProfileString(szOldDrvSec,NULL,szNone,szNewDrvSec, sizeof(szNewDrvSec))) { bOldExists=TRUE; } else { // Try the HPPCL5E driver... LoadString(hInst,IDS_OLD_5A_DRIVERNAME,szOldDrvSec,sizeof(szOldDrvSec)); MakeAppName((LPSTR)szOldDrvSec,lpPort,sizeof(szOldDrvSec)); if(GetProfileString(szOldDrvSec,NULL,szNone,szNewDrvSec, sizeof(szNewDrvSec))) { bOldExists=TRUE; } } lstrcpy(szNewDrvSec,rgchModuleName); MakeAppName((LPSTR)szNewDrvSec,lpPort,sizeof(szNewDrvSec)); if(bOldExists) { HandleSoftFonts(szOldDrvSec,szNewDrvSec); HandleFontCartridges(lpSection,szOldDrvSec,szNewDrvSec); } // create UNIDRV's font summary file, if there are many soft fonts. GetInt(szNewDrvSec,MAKEINTRESOURCE(IDS_SOFTFONTS),&nCount,0,FALSE); if(nCount>SOFT_FONT_THRES) { HDC hIC; if(hIC=CreateIC(rgchModuleName,lpDevName,lpPort,NULL)) DeleteDC(hIC); } } //-------------------------*DevInstall*--------------------------------------- // Action: De-install, upgrade or install a device. // //---------------------------------------------------------------------------- int FAR PASCAL DevInstall(HWND hWnd, LPSTR lpDevName, LPSTR lpOldPort, LPSTR lpNewPort) { char szDevSec[64]; // [,] section name if (!lpDevName) return -1; if (!lpOldPort) { char szBuf[10]; if (!lpNewPort) return 0; // install a device for the first time. Convert old HPPCL5a settings, // which are still under [,], into equivalent new // UNIDRV settings under [,], if applicable. // Delete old settings that are linked to the device name, but don't // delete old settings that are liked to the driver and port (softfonts) lstrcpy(szDevSec,lpDevName); MakeAppName((LPSTR)szDevSec,lpNewPort,sizeof(szDevSec)); // check if old settings exist at all if(GetProfileString(szDevSec,NULL,NULL,szBuf,sizeof(szBuf))) { // Do the straight conversions ConvertStraight(szDevSec,IDS_OLDPAPERSIZE,IDS_NEWPAPERSIZE); ConvertStraight(szDevSec,IDS_OLDPAPERSOURCE,IDS_NEWPAPERSOURCE); ConvertStraight(szDevSec,IDS_OLDORIENTATION,IDS_NEWORIENTATION); ConvertStraight(szDevSec,IDS_OLDTRUETYPE,IDS_NEWTRUETYPE); ConvertStraight(szDevSec,IDS_OLDSEPARATION,IDS_NEWSEPARATION); // Convert the simple translations ConvertBool(szDevSec,IDS_OLDPAGEPROTECT,IDS_NEWPAGEPROTECT,1); ConvertBool(szDevSec,IDS_OLDOUTPUT,IDS_NEWOUTPUT,259); // Do the stuff that requires more complicated conversion ConvertResolution(szDevSec,lpDevName); if(!Convert5eHalfTone(szDevSec)) Convert5aHalfTone(szDevSec); if(!Convert5eMemory(szDevSec)) { if(!Convert5MSMemory(szDevSec)) Convert5aMemory(szDevSec); } ConvertVectorMode(szDevSec); // Handle soft fonts & cartridges HandleFonts(szDevSec,lpDevName,lpNewPort); } // Flush the cached settings from win.ini WriteProfileString(NULL,NULL,NULL); } else { int nCount; // move device settings from the old port to the new port, or // de-install a device, i.e. remove its device setttings in order // to compress the profile. // First, check if there is any soft font installed under the // old port. If so, warn the user to copy them over. lstrcpy(szDevSec,rgchModuleName); MakeAppName((LPSTR)szDevSec,lpOldPort,sizeof(szDevSec)); if(GetInt(szDevSec,MAKEINTRESOURCE(IDS_SOFTFONTS),&nCount,0,FALSE) && nCount && lpNewPort) { NPSTR npTemp; if(npTemp=(NPSTR)LocalAlloc(LPTR,TMPSIZE)) { if(LoadString(hInst,IDS_SOFTFONTWARNING,npTemp,TMPSIZE)) { // Use this API so that the M Box is set to the Foreground MSGBOXPARAMS mbp; mbp.cbSize = sizeof(mbp); mbp.hwndOwner = hWnd; mbp.hInstance = hInst; mbp.lpszText = npTemp; mbp.lpszCaption = lpOldPort; mbp.dwStyle = MB_SETFOREGROUND | MB_OK | MB_ICONEXCLAMATION; mbp.lpszIcon = NULL; mbp.dwContextHelpId = 0L; mbp.lpfnMsgBoxCallback = NULL; MessageBoxIndirect(&mbp); } LocalFree((HLOCAL)npTemp); } } } return UniDevInstall(hWnd,lpDevName,lpOldPort,lpNewPort); } // the following 3 definitions MUST be compatible with the // HPPCL font installer #define CLASS_LASERJET 0 #define CLASS_DESKJET 1 #define CLASS_DESKJET_PLUS 2 //---------------------------*InstallExtFonts*--------------------------------- // Action: call the specific font installer to add/delete/modify soft fonts // and/or external cartridges. // // Parameters: // HWND hWnd; handle to the parent windows. // LPSTR lpDeviceName; long pointer to the printer name. // LPSTR lpPortName; long pointer to the associated port name. // BOOL bSoftFonts; flag if supporting soft fonts or not. // // Return Value: // > 0 : if the font information has changed; // == 0 : if nothing has changed; // == -1 : if intending to use the universal font installer // (not available now). //------------------------------------------------------------------------- int FAR PASCAL InstallExtFonts(HWND hWnd, LPSTR lpDeviceName, LPSTR lpPortName, BOOL bSoftFonts) { int fsVers; HANDLE hFIlib; SOFTFONTPROC lpFIns; if ((hFIlib = LoadLibrary((LPSTR)"FINSTALL.DLL")) < 32 || !(lpFIns = (SOFTFONTPROC)GetProcAddress(hFIlib,"InstallSoftFont"))) { if (hFIlib >= 32) FreeLibrary(hFIlib); #ifdef DEBUG MessageBox(0, "Can't load FINSTAL.DLL or can't get InstallSoftFont", NULL, MB_OK); #endif return TRUE; } // FINSTALL.DLL was loaded properly. Now call InstallSoftFont(). // We choose to ignore the returned "fvers". No use of it. fsVers = (*lpFIns)(hWnd,rgchModuleName,lpPortName, (GetKeyState(VK_SHIFT)<0 && GetKeyState(VK_CONTROL)<0), 1, // dummy value for "fvers". bSoftFonts?CLASS_LASERJET:256); FreeLibrary(hFIlib); return fsVers; } // ------------------------------------------------------------------- // // Special case control functions wrt SETCHARSET escape. This is necessary // to avoid breaking Winword and Pagemaker. (note that we don't actually // do anything with SETCHARSET, but apps break unless we say that we do) // // -------------------------------------------------------------------- int FAR PASCAL Control(LPDV lpdv, short function, LPSTR lpInData, LPSTR lpOutData) { // Tell app that SETCHARSET is supported if(QUERYESCSUPPORT == function && *((LPWORD)lpInData) == SETCHARSET) return 1; // Special case SETCHARSET if(SETCHARSET == function) return 1; // General case return UniControl(lpdv, function, lpInData, lpOutData); } #ifndef NOFONTMAP //---------------------------------------------------------------------- // Function: MapFaceName(lplfOld,lplfNew) // // Action: Map old face names to their new counterparts. Do as little // work as we possibly can, since this function gets called // often & we don't want to impact performance. Optimize // the whole search to the case where we don't find a match, // as this will be the most common scenario. // // Return: A pointer to the LOGFONT to actually hand to Unidrv. Cast // the return to a LPSTR, just so the compiler is happy. //---------------------------------------------------------------------- LPSTR NEAR PASCAL MapFaceName(LPLOGFONT lplfOld, LPLOGFONT lplfNew) { LPLOGFONT lpReturn=lplfOld; // By default NPFACEINDEX pIndex; LPSTR lpFace=lplfOld->lfFaceName; BYTE cTest=*lpFace++; // Determine range of possible matches in the table. Since the // table is sorted alphabetically, we may be able to bail out // before we reach the end of the table. for(pIndex=FaceIndex;cTest > pIndex->cFirstChar;pIndex++) ; // Only proceed if the first character matches and this isn't the // firewall (cTest = \xFF). if(cTest==pIndex->cFirstChar && ('\xFF' != cTest)) { WORD wStartIndex=(WORD)(pIndex->bIndex); WORD wStopIndex=(WORD)((pIndex+1)->bIndex); WORD wLoop=wStartIndex; NPSTR npMapFace=&(FaceMap[wStartIndex].szOldFace[1]); BYTE cMapFace; // Check the rest of the string against the table entries. // This search routine takes advantage of the fact that // the old face names in the table are fully sorted alphabetically. // This search routine takes advantage of the fact that our table // is fully sorted, and doesn't do a full string comparison until // we actually think we have a match. Once we think there's a match, // we'll double-check the entire string to prevent false triggering. while(wLoop < wStopIndex) { // Look for a match. At this point, match a wildcard to // anything--we'll do a more stringent check later on, if we // think we have a match. Stop if cTest is NULL. while((((cTest=*lpFace)==(cMapFace=*npMapFace)) || ('\x01'==cMapFace)) && cTest) { npMapFace++; lpFace++; } // We arrive here via two conditions: (1) we've reached the // end of lpFace and cTest is NULL, or (2) cMapFace and cTest // failed to compare. We should only continue searching if // cMapFace is non-NULL and cTest is larger than cMapFace (take // advantage of the fact that FaceMap is sorted alphabetically). // For overall performance, check for the no match case first. // Move to the next table entry as long as there's still a // chance to find a match. if(cTest > cMapFace) { npMapFace+=sizeof(FACEMAP); wLoop++; continue; // Go to the next iteration } // if cTest is non-NULL, then the sorting of the table guarantees // that there are no matches. Bail out now. if(cTest) goto MFN_exit; // cTest is NULL, so we will not make another iteration. The // only thing left to decide is whether or not we have a match // with the current string. if(cMapFace) goto MFN_exit; // The guards above ensure that we only arrive here if both // cTest and cMapFace are NULL, which means that if we're // going to find a match, this string is it. We took shortcuts // in the comparisons above, so do a stringent comparison now // to be sure that this is really a match. The only characters // to match wildcards are '1' and 'N', since these are the only // ones used in previous versions of the driver. for(lpFace=lplfOld->lfFaceName,npMapFace=FaceMap[wLoop].szOldFace; (cMapFace=*npMapFace) && (cTest=*lpFace); npMapFace++,lpFace++) { if(!((cTest==cMapFace) || (('\x01'==cMapFace)&&(('1'==cTest)||('N'==cTest))))) { // False trigger--bail out without changing facename goto MFN_exit; } } // We now know that this really is a match--keep the requested // attributes & change just the face name. *lplfNew=*lplfOld; lstrcpy(lplfNew->lfFaceName,FaceMap[wLoop].szNewFace); lpReturn=lplfNew; goto MFN_exit; } } MFN_exit: return (LPSTR)lpReturn; } #endif //---------------------------------------------------------------------- // Function: RealizeObject(lpdv,sStyle,lpInObj,lpOutObj,lpTextXForm) // // Action: Hook this out to enable font substitution. If the object isn't // a font, do absolutely nothing. // // Return: Save as UniRealizeObject(). //---------------------------------------------------------------------- DWORD FAR PASCAL RealizeObject(LPDV lpdv, short sStyle, LPSTR lpInObj, LPSTR lpOutObj, LPTEXTXFORM lpTextXForm) { #ifndef NOFONTMAP LOGFONT lfNew; if(OBJ_FONT==sStyle) lpInObj=MapFaceName((LPLOGFONT)lpInObj,&lfNew); #endif return UniRealizeObject(lpdv, sStyle, lpInObj, lpOutObj, lpTextXForm); } #endif // WINNT #ifndef WINNT //swpark //--------------------------**fnOEMOutputChar*---------------------------------// // Function: fnOEMOutputChar() // // Action: Output n characters pointed to by lpStr. Translation happens // on any extended characters in lpStr before printing. // // Notes: We go through lpStr printing each character until we need to // translate for an extended character. // We call SubstituteExtendedChar() which will read the device // data for this model if it has not already been retrieved. // We get an offset into the translation table and print // those characters which represent that extended character. // // We then return from SubstituteExtendedChar() and continue // printing the contents of lpStr. // We currently do not call Unidrv to handle this if no change // has been made to the default translation set via the TTY UI. #define OFFSET_HCURRENTCTT 258 static LPWORD lpwTailFirst; static LPWORD lpwTailFirstX; static LPBYTE lpbLeadMap; static LPBYTE lpbLeadMapX; static LPBYTE lpbTailOff; static LPBYTE lpbTailOffX; static LPBYTE lpbTailTable; void near pascal OutputMachine (LPDV lpdv, LPSTR lpStr, WORD n); // VOID _loadds far pascal fnOEMOutputChar(LPDV lpdv, LPSTR lpStr, WORD n, short rcID) VOID FAR PASCAL fnOEMOutputChar(LPDV lpdv, LPSTR lpStr, WORD n, short rcID) { #define OFFSET_1 4 #define OFFSET_2 158 #define OFFSET_3 312 #define OFFSET_4 337 #define OFFSET_5 407 #define OFFSET_6 432 #define OFFSET_7 502 HANDLE hCTT; LPSTR lpCtt; hCTT = *(LPHANDLE)((LPSTR)lpdv+OFFSET_HCURRENTCTT); #ifdef DEBUG DBMSG ("hCTT = %d\n", hCTT); #endif if (hCTT && (lpCtt = (LPSTR)LockResource(hCTT))) { lpwTailFirst = (LPWORD)((LPSTR)lpCtt + OFFSET_1); lpwTailFirstX= (LPWORD)((LPSTR)lpCtt + OFFSET_2); lpbLeadMap = (LPBYTE)lpCtt + OFFSET_3; lpbLeadMapX = (LPBYTE)lpCtt + OFFSET_4; lpbTailOff = (LPBYTE)lpCtt + OFFSET_5; lpbTailOffX = (LPBYTE)lpCtt + OFFSET_6; lpbTailTable = (LPBYTE)lpCtt + OFFSET_7; #ifdef DEBUG DBMSG(" lpwTailFirst[0] = %d\n", lpwTailFirst[0]); DBMSG(" lpwTailFirst[1] = %d\n", lpwTailFirst[1]); DBMSG(" lpwTailFirstX[0]= %d\n", lpwTailFirstX[0]); DBMSG(" lpwTailFirstX[1]= %d\n", lpwTailFirstX[1]); DBMSG(" lpbLeadMap[0] = %d\n", lpbLeadMap[0]); DBMSG(" lpbLeadMap[1] = %d\n", lpbLeadMap[1]); DBMSG(" lpbLeadMapX[0] = %d\n", lpbLeadMapX[0]); DBMSG(" lpbLeadMapX[1] = %d\n", lpbLeadMapX[1]); DBMSG(" lpbTailOff[0] = %d\n", lpbTailOff[0]); DBMSG(" lpbTailOff[1] = %d\n", lpbTailOff[1]); DBMSG(" lpbTailOffX[0] = %d\n", lpbTailOffX[0]); DBMSG(" lpbTailOffX[1] = %d\n", lpbTailOffX[1]); DBMSG(" lpbTailTable[0] = %d\n", lpbTailTable[0]); DBMSG(" lpbTailTable[1] = %d\n", lpbTailTable[1]); #endif #ifdef DEBUG DBMSG ("Sucess Lock CTT\n"); #endif OutputMachine (lpdv, lpStr, n); UnlockResource(hCTT); }else{ #ifdef DEBUG DBMSG ("Fail Lock CTT\n"); #endif WriteSpoolBuf(lpdv, lpStr, n); } } int near pascal IsHangeul( WORD wXW ); WORD near pascal ConvertToChoHap (WORD wCh); void near pascal OutputMachine (LPDV lpdv, LPSTR lpStr, WORD n) { #define S_HANGEUL 1 #define S_REMAINS 2 #define SET_CHOHAP "\033(18H",5 #define SET_WANSUNG "\033(19H",5 int state; short i; WORD wCh; LPSTR lpT = lpStr; state = S_HANGEUL; for (i=n; i > 0;) { if (IsDBCSLeadByte(*lpT) && i>1) { #ifdef DEBUG DBMSG (" IsHangeul() = %d\n", IsHangeul( *(LPWORD)lpT) ); #endif if ( IsHangeul(*(LPWORD)lpT)) { if (state==S_REMAINS) { WriteSpoolBuf(lpdv, SET_CHOHAP); state = S_HANGEUL; } wCh = ConvertToChoHap (*(LPWORD)lpT); WriteSpoolBuf(lpdv, (LPSTR)&wCh, 2); }else{ if (state==S_HANGEUL) { WriteSpoolBuf(lpdv, SET_WANSUNG); state = S_REMAINS; } WriteSpoolBuf(lpdv, lpT, 2); } lpT+=2; i-=2; }else{ if (state==S_HANGEUL) { WriteSpoolBuf(lpdv, SET_WANSUNG); state = S_REMAINS; } WriteSpoolBuf(lpdv, lpT, 1); lpT++; i--; } } if (state==S_REMAINS) WriteSpoolBuf(lpdv, SET_CHOHAP); } int near pascal IsHangeul( WORD wXW ) { #define XWT_INVALID 0xFF #define XWT_EXTENDED 0x01 #define XWT_WANSUNG 0x02 #define XWT_JUNJA 0x03 #define XWT_HANJA 0x04 #define XWT_UDC 0x05 static BYTE iXWType[8][3] = { XWT_EXTENDED, XWT_EXTENDED, XWT_EXTENDED, // Lead = 0x81-0xA0 XWT_EXTENDED, XWT_EXTENDED, XWT_JUNJA, // Lead = 0xA1-0xAC XWT_EXTENDED, XWT_EXTENDED, XWT_INVALID, // Lead = 0xAD-0xAF XWT_EXTENDED, XWT_EXTENDED, XWT_WANSUNG, // Lead = 0xB0-0xC5 XWT_EXTENDED, XWT_INVALID, XWT_WANSUNG, // Lead = 0xC6 XWT_INVALID, XWT_INVALID, XWT_WANSUNG, // Lead = 0xC7-0xC8 XWT_INVALID, XWT_INVALID, XWT_UDC, // Lead = 0xC9, 0xFE XWT_INVALID, XWT_INVALID, XWT_HANJA // Lead = 0xCA-0xFD }; BYTE bT = (BYTE) (( wXW >> 8 ) & 0xFF); BYTE bL = (BYTE) (wXW & 0xFF); int iLType = -1, iTType = -1; WORD wRet; #ifdef DEBUG DBMSG (" In GetXWType, bL =%d\n", bL); DBMSG (" In GetXWType, bT =%d\n", bT); #endif if ( ( bT >= 0x41 ) && ( bT <= 0xFE ) ) { if ( bT <= 0x52 ) iTType = 0; // Tail Range 0x41-0x52 else { if ( bT >= 0xA1 ) iTType = 2; // Tail Range 0xA1-0xFE else if ( ( bT <= 0x5A ) || ( bT >= 0x81 ) || ( ( bT >= 0x61 ) && ( bT <= 0x7A ) ) ) iTType = 1; // Tail Range 0x53-0x5A, 0x61-0x7A, 0x81-0xA0 } } if ( iTType < 0 ) return( 0 ); if ( ( bL >= 0x81 ) && ( bL <= 0xFE ) ) { if ( bL < 0xB0 ) { if ( bL <= 0xA0 ) iLType = 0; // Lead Range 0x81-0xA0 else if ( bL <= 0xAC ) iLType = 1; // Lead Range 0xA1-0xAC else iLType = 2; // Lead Range 0xAD-0xAF } else { if ( bL <= 0xC8 ) { if ( bL < 0xC6 ) iLType = 3; // Lead Range 0xB0-0xC5 else if ( bL == 0xC6 ) iLType = 4; // Lead Range 0xC6 else iLType = 5; // Lead Range 0xC7-0xC8 } else { if ( ( bL == 0xC9 ) || ( bL == 0xFE ) ) iLType = 6; // Lead Range 0xC9, 0xFE else iLType = 7; // Lead Range 0xCA-0xFD } } } wRet = ( iLType < 0 ) ? -1 : iXWType[iLType][iTType]; if (wRet == XWT_WANSUNG || wRet == XWT_EXTENDED) return wRet; else return 0; } WORD near pascal ConvertToChoHap (WORD wCh) { LPWORD lpTF; int iTO; BYTE bL, bT, bJS; WORD wCh1; wCh1 = wCh<<8; wCh1 |= wCh>>8; switch ( IsHangeul(wCh) ) { case XWT_EXTENDED : bL = (BYTE) (( ( wCh1 >> 8 ) & 0xFF ) - 0x81); bT = (BYTE) (wCh1 & 0xFF); if ( ( bT -= 0x41 ) > 0x19 ) if ( ( bT -= 6 ) > 0x33 ) bT -= 6; iTO = bT + lpbTailOffX[bL]; lpTF = lpwTailFirstX + ( bL = lpbLeadMapX[bL] ); goto FindHangeul; case XWT_WANSUNG : #ifdef DEBUG DBMSG(" case XWT_WANSUGN\n"); #endif bL = (BYTE) (( ( wCh1 >> 8 ) & 0xFF ) - 0xB0); iTO = ( wCh1 & 0xFF ) - 0xA1 + lpbTailOff[bL]; lpTF = lpwTailFirst + ( bL = lpbLeadMap[bL] ); goto FindHangeul; default : return (0x6188); } FindHangeul: iTO += *lpTF++; while ( iTO >= *lpTF++ ) bL++; //return ( (WORD)(bL+0x88) * 256 + lpbTailTable[iTO] ); return ( (WORD)lpbTailTable[iTO] * 256 + (bL+0x88) ); FindSymbol: bL = (BYTE) (( wCh1 >> 8 ) & 0xFF); bT = (BYTE) (wCh1 & 0xFF); if ( ( bL & 1 ) == 0 ) bT -= ( ( bT <= 0xEE ) ? 0x70 : 0x5E ); bJS += ( bL / 2 ); //return ((WORD) bJS * 256 + bT ); return ( (WORD)bT * 256 + bJS ); } //-----------------------------**Enable**-----------------------------------// // Function: Enable() // Action: We setup the callback to fnOEMOutPutChar and call UniEnable. // ------------------------------------------------------------------------ short CALLBACK Enable( LPDV lpdv, WORD style, LPSTR lpModel, LPSTR lpPort, LPDM lpStuff) { CUSTOMDATA cd; cd.cbSize = sizeof(CUSTOMDATA); cd.hMd = GetModuleHandle((LPSTR)rgchModuleName); cd.fnOEMDump = NULL; cd.fnOEMOutputChar = (LPFNOEMOUTPUTCHAR)fnOEMOutputChar; return UniEnable(lpdv, style, lpModel, lpPort, lpStuff, &cd); } #ifdef DEBUG static char logbuf[100]; void NEAR CDECL DBMSG(LPSTR lpFmt, BYTE arg) { if (lpFmt) { wvsprintf(logbuf, lpFmt, (LPSTR)&arg); OutputDebugString(logbuf); } } #endif // DEBUG #endif //WINNT