First public contribution.
2 * Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
22 const int KMarkFirstCharacterInTypeface = -1;
23 const int KBytesAddedForMetricIndex = 2;
24 const int KBytesForIndexForFillCharacters = 2;
25 const int KReportRateFrequencyInPercent = 10;
27 // class CroppedValues
40 CroppedValues::CroppedValues()
41 : iTopCrop(0), iBottomCrop(0), iLeftCrop(0), iRightCrop(0)
46 FontReader::FontReader()
50 iCharacterMetrics(NULL),
54 iReadFileFormat(ESymbianGDFFormat),
57 iDefaultXMoveInPixels(KUndefinedInteger),
58 iDefaultYMoveInPixels(KUndefinedInteger)
62 boolean FontReader::Read(const String& aFilename)
64 iFileName = aFilename;
65 boolean state = Open(iFileName.Text());
67 while (!_EOF() && state)
69 if (IdentComp(IdentBDFFileHeader) || IdentComp(IdentBDFComment))
71 state = ReadBDFFontBitmap();
72 iReadFileFormat = EBDFFormat;
74 else if (IdentComp(IdentTypeface))
75 state = ReadTypeface();
76 else if (IdentComp(IdentFontStoreFile))
77 state = ReadFontStoreFile();
80 Error("Resource identifier expected");
89 boolean FontReader::ReadMetricFromBDFCharacter(CharacterMetrics* aCharacterMetrics, CroppedValues* aCropped= NULL)
91 int xMoveInPixels = 0;
92 int yMoveInPixels = 0;
93 int bitmapXOffset = 0;
94 int bitmapYOffset = 0;
98 if (iDefaultXMoveInPixels != KUndefinedInteger)
99 xMoveInPixels = iDefaultXMoveInPixels;
100 if (iDefaultYMoveInPixels != KUndefinedInteger)
101 yMoveInPixels = iDefaultYMoveInPixels;
103 while (!IdentComp(IdentBDFEndChar) && !_EOF() && state)
105 iLexAnal->ReadNextLine(); // Skip to next line
106 if (iLex->iType == ELexIdent)
108 if (IdentComp(IdentBDFCursorMove))
110 state = Number(xMoveInPixels);
112 state = Number(yMoveInPixels);
114 else if (IdentComp(IdentBDFBitmapSizeAndDisplacement))
116 state = Number(iBitmapWidth);
119 state = Number(iBitmapHeight);
120 state = Number(bitmapXOffset);
121 state = Number(bitmapYOffset);
124 else if (IdentComp(IdentBDFStartBitmap) && state)
127 for (line = 0; line < iBitmapHeight; line++)
129 iLexAnal->ReadNextLine(); // Skip to next line
133 for (bits = 0; bits < iBitmapWidth; bits++)
138 strncpy(lex.iText, &iLexAnal->iLine[bits >> 2], 2);
140 byteValue = lex.CovertStringToHex();
145 int bitValue = (byteValue >> bitOffset) & 1;
146 iBitArray[bits][line] = bitValue;
153 Error("Fontbitmap identifier expected");
161 // Make sure that bitmap is fully cropped
169 while (BitmapLineEmpty(line) && line < iBitmapHeight)
174 line = iBitmapHeight - 1;
175 while (BitmapLineEmpty(line) && line > topCrop)
180 int column = iBitmapWidth - 1;
181 while (BitmapColumnEmpty(column) && column >= 0)
187 while (BitmapColumnEmpty(column) && column < iBitmapWidth - 1 - rightCrop)
193 int croppedBitmapHeight = iBitmapHeight - topCrop - bottomCrop;
194 int croppedBitmapWidth = iBitmapWidth - leftCrop - rightCrop;
195 int croppedLeftAdjust = bitmapXOffset + leftCrop;
196 int croppedRightAdjust = (xMoveInPixels - croppedLeftAdjust - croppedBitmapWidth);
197 int croppedAscent = croppedBitmapHeight + bitmapYOffset + bottomCrop;
201 aCharacterMetrics->iAscentInPixels = (chardim) croppedAscent;
202 aCharacterMetrics->iHeightInPixels = (chardim) croppedBitmapHeight;
203 aCharacterMetrics->iLeftAdjustInPixels = (chardim) croppedLeftAdjust;
204 aCharacterMetrics->iMoveInPixels = (chardim) xMoveInPixels;
205 aCharacterMetrics->iRightAdjustInPixels = (chardim) croppedRightAdjust;
208 aCropped->iBottomCrop = bottomCrop;
209 aCropped->iLeftCrop = leftCrop;
210 aCropped->iRightCrop = rightCrop;
211 aCropped->iTopCrop = topCrop;
217 boolean FontReader::ReadBDFCharacter(int /*aCode*/)
219 boolean state = etrue;
220 ObjectList<String*> character;
221 iCharacterMetrics = new CharacterMetrics;
222 CroppedValues* croppedValues = new CroppedValues;
224 state = ReadMetricFromBDFCharacter(iCharacterMetrics, croppedValues);
228 BitmapOffset* offset = new BitmapOffset((uint16)iCodeSection->iCharactersBitmap.iByteList.Length());
231 iCodeSection->iCharactersBitmap.iByteList.NewByte();
232 iCodeSection->iCharacters.iBitmapOffsetList.Add(offset);
234 int index = iFontBitmap->iCharacterMetrics->Index(*iCharacterMetrics);
237 Error("Internal Compiler Error");
240 delete iCharacterMetrics;
244 iCodeSection->iCharactersBitmap.AddIndex(index);
246 int line = croppedValues->iTopCrop;
249 const int bottomCrop = croppedValues->iBottomCrop;
250 while (line < (iBitmapHeight - bottomCrop))
252 if ((line + 1) == (iBitmapHeight - bottomCrop))
254 repeatLines = efalse;
257 else if (!CompareBitmapLines(line, line + 1))
259 repeatLines = efalse;
260 for (countLines = 2; ((line + countLines) < (iBitmapHeight - bottomCrop)) && (countLines < KMaxNumberRepeatedLines); countLines++)
262 if (CompareBitmapLines(line + countLines - 1, line + countLines))
269 for (countLines = 2; ((line + countLines) < (iBitmapHeight - bottomCrop)) && (countLines < KMaxNumberRepeatedLines); countLines++)
271 if (!CompareBitmapLines(line, line + countLines))
280 iCodeSection->iCharactersBitmap.iByteList.AddBit(bit);
281 for (int digit = 0; digit < 4; digit++)
283 bit = char(countLines >> digit);
284 iCodeSection->iCharactersBitmap.iByteList.AddBit(bit);
287 for (lineFromTop = line; lineFromTop < (line + countLines); lineFromTop++)
289 if ((!repeatLines) || (lineFromTop == line))
292 for (column = croppedValues->iLeftCrop;
293 column < iBitmapWidth - croppedValues->iRightCrop; column++)
295 if (iBitArray[column][lineFromTop] == 1)
299 iCodeSection->iCharactersBitmap.iByteList.AddBit(bit);
303 line = line+countLines;
307 delete croppedValues;
311 boolean FontReader::ReadBDFChars(const int aNumberOfGlyphsInFile, const int aMaxConsecutiveFillChars)
313 boolean state = etrue;
314 boolean newCodeSection = etrue;
315 int currentEncodingValue = 0; // each glyph has an unique encoding value
316 int lastEncodingValue = KMarkFirstCharacterInTypeface;
317 const int maxCharacterBitmapsize =
318 ((KMaxBitmapWidth * KMaxBitmapHeight) / KNumberOfBitsInByte) + KBytesAddedForMetricIndex + KBytesForIndexForFillCharacters;
321 for (int numberGlyphsRead = 0; state && numberGlyphsRead < aNumberOfGlyphsInFile && !_EOF(); numberGlyphsRead++)
323 if (state && iCodeSection && iCodeSection->iCharactersBitmap.iByteList.Length() + maxCharacterBitmapsize >= KMaxSizeCodeSectionBitmap)
325 newCodeSection = etrue;
328 state = IdentComp(IdentBDFCharLabel);
330 ErrorIdentifierExpected(IdentBDFCharLabel);
332 iLexAnal->ReadNextLine(); // Skip to next line
334 if (IdentComp(IdentBDFChar))
336 state = Number(currentEncodingValue);
341 ErrorIdentifierExpected(IdentBDFChar);
344 if (KLowestPermittedCharacterEncoding > currentEncodingValue ||
345 KHighestPermittedCharacterEncoding < currentEncodingValue)
347 while (!IdentComp(IdentBDFEndChar) && !_EOF() && state)
348 // Skip fill character.
350 iLexAnal->ReadNextLine(); // Skip to next line
352 iLexAnal->ReadNextLine();
356 if (!iCodeSection || state && (currentEncodingValue > (lastEncodingValue + 1 + aMaxConsecutiveFillChars)))
358 // First character skip to new code section
359 newCodeSection = etrue;
361 else if (state && !newCodeSection && (currentEncodingValue > lastEncodingValue + 1))
363 WriteFillCharacters(currentEncodingValue - (lastEncodingValue + 1));
365 else if (state && currentEncodingValue <= lastEncodingValue)
367 Error("CodeSection out of sequence");
371 if (state && newCodeSection)
375 iCodeSection->iEnd = (uint16) lastEncodingValue;
376 iFontBitmap->iCodeSectionList.Add(iCodeSection);
377 PrintoutCodeSection(iCodeSection);
379 iCodeSection = new BitmapCodeSection;
380 iCodeSection->iStart = (uint16) currentEncodingValue;
384 state = ReadBDFCharacter(currentEncodingValue);
386 newCodeSection = efalse;
387 lastEncodingValue = currentEncodingValue;
392 iCodeSection->iEnd = (uint16) lastEncodingValue;
393 iFontBitmap->iCodeSectionList.Add(iCodeSection);
394 PrintoutCodeSection(iCodeSection);
400 void FontReader::PrintoutCodeSection(const BitmapCodeSection* aCodeSection) const
402 cout << hex << "Codesection 0x" << aCodeSection->iStart << ": 0x" << aCodeSection->iEnd << " read" << endl;
406 boolean FontReader::ReadBDFFontBitmap()
408 boolean state = etrue;
410 iFontBitmap = new FontBitmap;
412 int xresolution; // in dots per inch
413 int yresolution; // in dots per inch
414 int widthBoundingBox = 0; // In pixels
415 int heightBoundingBox = 0; // in pixels
416 int boundingBoxXOffset = 0; // From origin (cursor position at baseline)
417 int boundingBoxYOffset = 0; // From origin (cursor position at baseline)
420 int maxNormalCharWidth = KUndefinedInteger;
421 uid fontUid = KNullUid;
422 int posture = PostureUpright;
423 int strokeWeight = StrokeWeightNormal;
424 int defaultXMoveInPixels = KUndefinedInteger;
425 int defaultYMoveInPixels = KUndefinedInteger;
426 int writingDirection = 0;
427 int maxConsecutiveFillChars = 0; // Max permitted "fill" characters with zero size to prevent
428 // break between code sections.
432 while (!IdentComp(IdentBDFNumChars) && !_EOF() && state)
434 iLexAnal->ReadNextLine(); // Skip to next line
435 if (iLex->iType == ELexIdent)
437 if (IdentComp(IdentBDFFontBitmap))
439 state = IdentCopy(fontLabel);
441 if (IdentComp(IdentBDFPointSize))
443 state = Number(pointSize);
445 state = Number(xresolution);
447 state = Number(yresolution);
449 else if (IdentComp(IdentBDFFontDesignBox))
451 state = Number(widthBoundingBox);
453 state = Number(heightBoundingBox);
455 state = Number(boundingBoxXOffset);
457 state = Number(boundingBoxYOffset);
459 else if (IdentComp(IdentBDFWritingDirection))
461 Number(writingDirection);
462 if (writingDirection != 0)
463 cout << "Warning: Only left to right writing direction supported by EPOC32";
465 else if (IdentComp(IdentBDFCursorMove))
467 state = Number(defaultXMoveInPixels);
469 state = Number(defaultYMoveInPixels);
471 else if (IdentComp(IdentBDFStartProperties)) // Adding additional properties
473 int numberOfProperties;
474 state = Number(numberOfProperties);
477 {for (int properties = 0; properties < numberOfProperties && !_EOF() && state; properties++)
479 if (IdentComp(IdentBDFPropertyUid) || IdentComp(IdentUid))
480 state = Number(fontUid);
481 else if (IdentComp(IdentBDFPropertyBold) || IdentComp(IdentBold))
482 state = Number(strokeWeight);
483 else if (IdentComp(IdentBDFPropertyItalic) || IdentComp(IdentItalic))
484 state = Number(posture);
485 else if (IdentComp(IdentBDFPropertyFontAscent))
486 state = Number(fontAscent);
487 else if (IdentComp(IdentBDFPropertyFontDescent))
488 state = Number(fontDescent);
489 else if (IdentComp(IdentBDFPropertyMaxNormalCharWidth) || IdentComp(IdentMaxNormalCharWidth))
490 state = Number(maxNormalCharWidth);
491 else if (IdentComp(IdentBDFPropertyMaxConsecutiveFillChars) || IdentComp(IdentMaxConsecutiveFillChars))
492 state = Number(maxConsecutiveFillChars);
494 iLexAnal->ReadNextLine(); // Skip to next line
499 state = IdentComp(IdentBDFEndProperties);
501 ErrorIdentifierExpected(IdentBDFEndProperties);
508 Error("Identifier expected");
513 // Check that maximum size of bitmap glyph not exceeded.
515 if (widthBoundingBox>KMaxBitmapWidth || widthBoundingBox>KMaxBitmapWidth)
518 cout << "Error: A font bounding box dimension exceeds maximum permitted in EPOC32";
521 // Check that have everthing that we need and set some bitmap properties.
523 iFontBitmap->iMaxCharWidthInPixels = (chardim) widthBoundingBox;
524 if (fontAscent + fontDescent)
526 iFontBitmap->iCellHeightInPixels =( chardim) (fontAscent+fontDescent);
527 iFontBitmap->iAscentInPixels = (chardim) (fontAscent);
530 { // Old file so use the old calculation
531 iFontBitmap->iCellHeightInPixels =( chardim) heightBoundingBox;
532 iFontBitmap->iAscentInPixels = (chardim) (heightBoundingBox + boundingBoxYOffset);
534 iFontBitmap->iPosture = (boolean) posture;
535 iFontBitmap->iStrokeWeight =( boolean) strokeWeight;
536 iFontBitmap->iLabel = fontLabel;
538 iDefaultXMoveInPixels = defaultXMoveInPixels;
539 iDefaultYMoveInPixels = defaultYMoveInPixels;
541 if (fontUid != KNullUid)
542 iFontBitmap->iUid = fontUid;
545 cerr << "Warning: No font bitmap UID specified in properties\n";
546 iFontBitmap->iUid = rand();
549 if (maxNormalCharWidth != KUndefinedInteger)
550 iFontBitmap->iMaxNormalCharWidthInPixels = (chardim) maxNormalCharWidth;
553 cerr << "Warning: No Maximum Normal Character Width specified in properties\n";
554 iFontBitmap->iMaxNormalCharWidthInPixels = iFontBitmap->iMaxCharWidthInPixels;
558 state = Number(numberChars);
564 // store position of Lex here and then set it back for the reading of the BDF chars!
565 int saveLineNo = iLexAnal->iLineNo;
567 ParseMetricsFromBDF(numberChars, maxConsecutiveFillChars);
568 // now reset the lexical
569 state = Open(iFileName.Text());
574 while(saveLineNo > iLexAnal->iLineNo);
576 ReadBDFChars(numberChars, maxConsecutiveFillChars);
580 state = IdentComp(IdentBDFEndFontBitmap);
582 Error("ENDFONT identifier expected");
585 int globalMove = KUndefinedInteger;
586 int monospaced= etrue;
589 for (int i = 0; i <iFontBitmap->iCodeSectionList.Size(); i++)
591 const int end = iFontBitmap->iCharacterMetrics->iCharacterMetricsList.Size();
592 for (int j = 0; j< end; j++)
594 int move = iFontBitmap->iCharacterMetrics->iCharacterMetricsList[j]->Metric()->iMoveInPixels;
595 if (globalMove == KUndefinedInteger)
597 if (move > iFontBitmap->iMaxCharWidthInPixels)
598 iFontBitmap->iMaxCharWidthInPixels = (chardim) move;
599 if (globalMove!= move)
605 iFontBitmap->iIsProportional = efalse;
607 iFontBitmap->iIsProportional = etrue;
611 iFontStore.AddFontBitmap(iFontBitmap);
612 cout << "FONT read\n";
619 boolean FontReader::ReadTypeface()
621 boolean state = etrue;
622 iTypeface = new FntTypeface;
625 state = IdentCopy(iTypeface->iLabel);
628 while (!IdentComp(IdentEndTypeface) && !_EOF() && state)
630 if (IdentComp(IdentName))
632 state = StringCopy(iTypeface->iName);
633 while ((iLex->iType != ELexNL) && !_EOF() && state)
635 if (IdentComp(IdentTypefaceProportional))
636 iTypeface->iFlags = (boolean) (iTypeface->iFlags | Proportional);
637 else if (IdentComp(IdentSerif))
638 iTypeface->iFlags = (boolean) (iTypeface->iFlags | Serif);
639 else if (IdentComp(IdentSymbol))
640 iTypeface->iFlags = (boolean) (iTypeface->iFlags | Symbol);
643 Error("Typeface identifier or newline expected");
648 else if (IdentComp(IdentFontBitmaps))
651 while (!IdentComp(IdentEndFontBitmaps) && !_EOF() && state)
653 TypefaceFontBitmap* typefacefontbitmap = NULL;
654 if (iLex->iType == ELexIdent)
656 state = IdentCopy(label);
657 FontBitmap* fontbitmap = (FontBitmap*) iFontStore.FindFontBitmap(label);
660 typefacefontbitmap = new TypefaceFontBitmap(fontbitmap);
665 Error("Fontbitmap not found");
672 state = Number(fontbitmapuid);
674 typefacefontbitmap = new TypefaceFontBitmap(fontbitmapuid);
678 while ((iLex->iType != ELexNL) && !_EOF() && state)
680 if (IdentComp(IdentWidthFactor))
683 typefacefontbitmap->iWidthFactor = (char) num;
685 else if (IdentComp(IdentHeightFactor))
688 typefacefontbitmap->iHeightFactor =( char) num;
692 Error("Typefacefontbitmap identifier or newline expected");
697 iTypeface->iTypefaceFontBitmapList.Add(typefacefontbitmap);
699 delete typefacefontbitmap;
707 Error("Typeface identifier expected");
715 iFontStore.AddTypeface(iTypeface);
716 cout << "Typeface read\n";
723 boolean FontReader::ReadFontStoreFile()
725 boolean state = etrue;
730 Error("Fontstorefile already read");
734 iFontStoreFile = new FontStoreFile;
738 while (!IdentComp(IdentEndFontStoreFile) && !_EOF() && state)
740 if (IdentComp(IdentCollectionUid))
742 state = Number(iFontStoreFile->iCollectionUid);
744 else if (IdentComp(IdentKPixelAspectRatio))
748 iFontStoreFile->iKPixelAspectRatio = num;
750 else if (IdentComp(IdentCopyrightInfo))
753 while (!IdentComp(IdentEndCopyrightInfo) && !_EOF() && state)
755 String* string = new String;
756 state = StringCopy(*string);
758 iFontStoreFile->iCopyrightInfo.Add(string);
764 else if (IdentComp(IdentTypefaces))
767 while (!IdentComp(IdentEndTypefaces) && !_EOF() && state)
769 state =IdentCopy(label);
772 typeface = iFontStore.FindTypeface(label);
775 iFontStoreFile->AddTypeface((FntTypeface*) typeface);
779 Error("Typeface not found");
787 else if (IdentComp(IdentExtraFontBitmaps))
790 while (!IdentComp(IdentEndExtraFontBitmaps) && !_EOF() && state)
792 state = IdentCopy(label);
793 FontBitmap* fontbitmap = (FontBitmap*) iFontStore.FindFontBitmap(label);
796 iFontStoreFile->AddFontBitmap((FontBitmap*) fontbitmap);
800 Error("Fontbitmap not found");
809 Error("Fontstorefile identifier expected");
817 iFontStore.AddFontStoreFile(iFontStoreFile);
818 cout << "Fontstorefile read\n";
821 delete iFontStoreFile;
826 int FontReader::Store(const String& aFilename)
828 boolean state = etrue;
832 Error("No fontstore file record");
835 state = iFontStore.Store(aFilename);
839 boolean FontReader::CharLine(String& aCharLine)
841 boolean state = etrue;
842 while (state && (iLex->iType != ELexNL))
845 state = Operator(ch);
848 if ((ch == '.') || (ch == '*'))
853 Error("Operator '.' or '*' expected");
860 void FontReader::ErrorIdentifierExpected(const String& aIdentifier)
862 cerr << "Error: Identifier Expected " << aIdentifier;
864 while ((iLex->iType != ELexNL) && (iLex->iType != ELexEOF))
868 boolean FontReader::CompareBitmapLines(int aLine1, int aLine2)
872 boolean identical = etrue;
873 for (column=0; column < iBitmapWidth; column++)
875 if (iBitArray[column][aLine1] != iBitArray[column][aLine2])
885 boolean FontReader::BitmapLineEmpty(int aLine)
889 boolean empty = etrue;
890 for (column = 0; column < iBitmapWidth; column++)
892 if (iBitArray[column][aLine] != 0)
902 boolean FontReader::BitmapColumnEmpty(int aColumn)
906 boolean empty = etrue;
907 for (line = 0; line < iBitmapHeight; line++)
909 if (iBitArray[aColumn][line] != 0)
919 void FontReader::WriteFillCharacters(int aNumberConsecutive)
921 for (int counter = 0; counter < aNumberConsecutive; counter++)
923 BitmapOffset* offset = new BitmapOffset(KFillCharacterOffset);
924 iCodeSection->iCharacters.iBitmapOffsetList.Add(offset);
928 boolean FontReader::ParseMetricsFromBDF(int aNumberCharsInFile, int aMaxConsecutiveFillChars)
930 boolean state = etrue;
932 int numberCharactersRead;
933 int lastCharacter = KMarkFirstCharacterInTypeface;
934 int reportRate = ((aNumberCharsInFile - 1) / KReportRateFrequencyInPercent) + 1;
936 CharacterMetrics* metric = 0;
938 cout << "Analysing character metrics...\n";
941 for (numberCharactersRead = 0; numberCharactersRead < aNumberCharsInFile && !_EOF() && state; numberCharactersRead++)
943 state = IdentComp(IdentBDFCharLabel);
945 ErrorIdentifierExpected(IdentBDFCharLabel);
947 iLexAnal->ReadNextLine(); // Skip to next line
949 if (IdentComp(IdentBDFChar))
951 state = Number(character);
956 ErrorIdentifierExpected(IdentBDFChar);
959 if (character < KLowestPermittedCharacterEncoding || character > KHighestPermittedCharacterEncoding)
961 while (!IdentComp(IdentBDFEndChar) && !_EOF() && state)
962 // Skip fill character.
964 iLexAnal->ReadNextLine(); // Skip to next line
966 iLexAnal->ReadNextLine();
970 if (character<lastCharacter)
972 Error("CodeSection out of sequence");
976 if ((character > lastCharacter + 1)
977 && (character <= lastCharacter + 1 + aMaxConsecutiveFillChars) && state)
979 // Would result in fill characters being used if not at end of code section!
980 metric = new CharacterMetrics;
981 metric->iLeftAdjustInPixels = (chardim)0;
982 metric->iMoveInPixels = (chardim)0;
983 metric->iRightAdjustInPixels = (chardim)0;
984 metric->iAscentInPixels = (chardim)0;
985 metric->iHeightInPixels = (chardim)0;
986 iFontBitmap->iCharacterMetrics->AddOrIncrementMetric(*metric);
993 metric = new CharacterMetrics;
994 state = ReadMetricFromBDFCharacter(metric);
995 iFontBitmap->iCharacterMetrics->AddOrIncrementMetric(*metric);
1000 lastCharacter = character;
1001 if(numberCharactersRead == ((numberCharactersRead / reportRate) * reportRate))
1003 cout << "Done " << (numberCharactersRead * 100 / aNumberCharsInFile) << "% \n";
1010 iFontBitmap->iCharacterMetrics->SortMetricsByFrequency();