2012-11-26 18:58:24 +00:00
|
|
|
/*
|
|
|
|
===========================================================================
|
|
|
|
|
|
|
|
Doom 3 BFG Edition GPL Source Code
|
2012-11-28 15:47:07 +00:00
|
|
|
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
|
2012-11-26 18:58:24 +00:00
|
|
|
|
2012-11-28 15:47:07 +00:00
|
|
|
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
|
2012-11-26 18:58:24 +00:00
|
|
|
|
|
|
|
Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
|
|
|
|
|
|
|
|
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
|
|
|
|
|
|
|
|
===========================================================================
|
|
|
|
*/
|
|
|
|
#pragma hdrstop
|
2012-12-22 15:18:19 +00:00
|
|
|
#include "precompiled.h"
|
2012-11-26 18:58:24 +00:00
|
|
|
|
|
|
|
#pragma warning( disable: 4189 ) // local variable is initialized but not referenced
|
|
|
|
|
|
|
|
/*
|
|
|
|
========================
|
|
|
|
idSWF::DefineFont2
|
|
|
|
========================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idSWF::DefineFont2( idSWFBitStream& bitstream )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint16 characterID = bitstream.ReadU16();
|
2012-11-28 15:47:07 +00:00
|
|
|
idSWFDictionaryEntry* entry = AddDictionaryEntry( characterID, SWF_DICT_FONT );
|
|
|
|
if( entry == NULL )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
uint8 flags = bitstream.ReadU8();
|
|
|
|
uint8 language = bitstream.ReadU8();
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
char fontName[257];
|
|
|
|
uint8 fontNameLength = bitstream.ReadU8();
|
|
|
|
memcpy( fontName, bitstream.ReadData( fontNameLength ), fontNameLength );
|
|
|
|
fontName[ fontNameLength ] = 0;
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
entry->font->fontID = renderSystem->RegisterFont( fontName );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
uint16 numGlyphs = bitstream.ReadU16();
|
|
|
|
entry->font->glyphs.SetNum( numGlyphs );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
|
|
|
if( flags & BIT( 3 ) )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
// 32 bit offsets
|
|
|
|
uint32 offsetTableSize = ( numGlyphs + 1 ) * 4;
|
|
|
|
idSWFBitStream offsetStream( bitstream.ReadData( offsetTableSize ), offsetTableSize, false );
|
2012-11-28 15:47:07 +00:00
|
|
|
if( offsetStream.ReadU32() != offsetTableSize )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
idLib::Warning( "idSWF::DefineFont2: first glyph offset != offsetTableSize" );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
uint32 previousOffset = offsetTableSize;
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < numGlyphs; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint32 nextOffset = offsetStream.ReadU32();
|
|
|
|
uint32 shapeSize = nextOffset - previousOffset;
|
|
|
|
previousOffset = nextOffset;
|
|
|
|
idSWFBitStream shapeStream( bitstream.ReadData( shapeSize ), shapeSize, false );
|
|
|
|
idSWFShapeParser swfShapeParser;
|
|
|
|
swfShapeParser.ParseFont( shapeStream, entry->font->glyphs[i] );
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
// 16 bit offsets
|
|
|
|
uint16 offsetTableSize = ( numGlyphs + 1 ) * 2;
|
|
|
|
idSWFBitStream offsetStream( bitstream.ReadData( offsetTableSize ), offsetTableSize, false );
|
2012-11-28 15:47:07 +00:00
|
|
|
if( offsetStream.ReadU16() != offsetTableSize )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
idLib::Warning( "idSWF::DefineFont2: first glyph offset != offsetTableSize" );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
uint16 previousOffset = offsetTableSize;
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < numGlyphs; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint16 nextOffset = offsetStream.ReadU16();
|
|
|
|
uint16 shapeSize = nextOffset - previousOffset;
|
|
|
|
previousOffset = nextOffset;
|
|
|
|
idSWFBitStream shapeStream( bitstream.ReadData( shapeSize ), shapeSize, false );
|
|
|
|
idSWFShapeParser swfShapeParser;
|
|
|
|
swfShapeParser.ParseFont( shapeStream, entry->font->glyphs[i] );
|
|
|
|
}
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags & BIT( 2 ) )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
// 16 bit codes
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < numGlyphs; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
entry->font->glyphs[i].code = bitstream.ReadU16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
// 8 bit codes
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < numGlyphs; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
entry->font->glyphs[i].code = bitstream.ReadU8();
|
|
|
|
}
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags & BIT( 7 ) )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
entry->font->ascent = bitstream.ReadS16();
|
|
|
|
entry->font->descent = bitstream.ReadS16();
|
|
|
|
entry->font->leading = bitstream.ReadS16();
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < numGlyphs; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
entry->font->glyphs[i].advance = bitstream.ReadS16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < numGlyphs; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
swfRect_t ignored;
|
|
|
|
bitstream.ReadRect( ignored );
|
|
|
|
}
|
|
|
|
uint16 kearningCount = bitstream.ReadU16();
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags & BIT( 2 ) )
|
|
|
|
{
|
|
|
|
for( int i = 0; i < kearningCount; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint16 code1 = bitstream.ReadU16();
|
|
|
|
uint16 code2 = bitstream.ReadU16();
|
|
|
|
int16 adjustment = bitstream.ReadS16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( int i = 0; i < kearningCount; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint16 code1 = bitstream.ReadU8();
|
|
|
|
uint16 code2 = bitstream.ReadU8();
|
|
|
|
int16 adjustment = bitstream.ReadS16();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
========================
|
|
|
|
idSWF::DefineFont3
|
|
|
|
========================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idSWF::DefineFont3( idSWFBitStream& bitstream )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
DefineFont2( bitstream );
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
========================
|
|
|
|
idSWF::DefineTextX
|
|
|
|
========================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idSWF::DefineTextX( idSWFBitStream& bitstream, bool rgba )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint16 characterID = bitstream.ReadU16();
|
2012-11-28 15:47:07 +00:00
|
|
|
idSWFDictionaryEntry* entry = AddDictionaryEntry( characterID, SWF_DICT_TEXT );
|
|
|
|
if( entry == NULL )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
return;
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
idSWFText* text = entry->text;
|
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
bitstream.ReadRect( text->bounds );
|
|
|
|
bitstream.ReadMatrix( text->matrix );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
uint8 glyphBits = bitstream.ReadU8();
|
|
|
|
uint8 advanceBits = bitstream.ReadU8();
|
2012-11-28 15:47:07 +00:00
|
|
|
|
|
|
|
while( true )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint8 flags = bitstream.ReadU8();
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
break;
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
idSWFTextRecord& textRecord = text->textRecords.Alloc();
|
|
|
|
|
|
|
|
if( flags & BIT( 3 ) )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
textRecord.fontID = bitstream.ReadU16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags & BIT( 2 ) )
|
|
|
|
{
|
|
|
|
if( rgba )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
bitstream.ReadColorRGBA( textRecord.color );
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
bitstream.ReadColorRGB( textRecord.color );
|
|
|
|
}
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags & BIT( 0 ) )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
textRecord.xOffset = bitstream.ReadS16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags & BIT( 1 ) )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
textRecord.yOffset = bitstream.ReadS16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( flags & BIT( 3 ) )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
textRecord.textHeight = bitstream.ReadU16();
|
|
|
|
}
|
|
|
|
textRecord.firstGlyph = text->glyphs.Num();
|
|
|
|
textRecord.numGlyphs = bitstream.ReadU8();
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < textRecord.numGlyphs; i++ )
|
|
|
|
{
|
|
|
|
swfGlyphEntry_t& glyph = text->glyphs.Alloc();
|
2012-11-26 18:58:24 +00:00
|
|
|
glyph.index = bitstream.ReadU( glyphBits );
|
|
|
|
glyph.advance = bitstream.ReadS( advanceBits );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
========================
|
|
|
|
idSWF::DefineText
|
|
|
|
========================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idSWF::DefineText( idSWFBitStream& bitstream )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
DefineTextX( bitstream, false );
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
========================
|
|
|
|
idSWF::DefineText2
|
|
|
|
========================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idSWF::DefineText2( idSWFBitStream& bitstream )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
DefineTextX( bitstream, true );
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
========================
|
|
|
|
idSWF::DefineEditText
|
|
|
|
========================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idSWF::DefineEditText( idSWFBitStream& bitstream )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
uint16 characterID = bitstream.ReadU16();
|
2012-11-28 15:47:07 +00:00
|
|
|
idSWFDictionaryEntry* entry = AddDictionaryEntry( characterID, SWF_DICT_EDITTEXT );
|
|
|
|
if( entry == NULL )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
return;
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
idSWFEditText* edittext = entry->edittext;
|
2012-11-26 18:58:24 +00:00
|
|
|
bitstream.ReadRect( edittext->bounds );
|
|
|
|
bitstream.ResetBits();
|
|
|
|
bool hasText = bitstream.ReadBool();
|
|
|
|
bool wordWrap = bitstream.ReadBool();
|
|
|
|
bool multiline = bitstream.ReadBool();
|
|
|
|
bool password = bitstream.ReadBool();
|
|
|
|
bool readonly = bitstream.ReadBool();
|
|
|
|
bool hasTextColor = bitstream.ReadBool();
|
|
|
|
bool hasMaxLength = bitstream.ReadBool();
|
|
|
|
bool hasFont = bitstream.ReadBool();
|
|
|
|
bool hasFontClass = bitstream.ReadBool();
|
|
|
|
bool autoSize = bitstream.ReadBool();
|
|
|
|
bool hasLayout = bitstream.ReadBool();
|
|
|
|
bool noSelect = bitstream.ReadBool();
|
|
|
|
bool border = bitstream.ReadBool();
|
|
|
|
bool wasStatic = bitstream.ReadBool();
|
|
|
|
bool html = bitstream.ReadBool();
|
|
|
|
bool useOutlines = bitstream.ReadBool();
|
2012-11-28 15:47:07 +00:00
|
|
|
if( hasFont )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
edittext->fontID = bitstream.ReadU16();
|
|
|
|
edittext->fontHeight = bitstream.ReadU16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( hasFontClass )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
idStr fontClass = bitstream.ReadString();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( hasTextColor )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
bitstream.ReadColorRGBA( edittext->color );
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( hasMaxLength )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
edittext->maxLength = bitstream.ReadU16();
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( hasLayout )
|
|
|
|
{
|
|
|
|
edittext->align = ( swfEditTextAlign_t )bitstream.ReadU8();
|
2012-11-26 18:58:24 +00:00
|
|
|
edittext->leftMargin = bitstream.ReadU16();
|
|
|
|
edittext->rightMargin = bitstream.ReadU16();
|
|
|
|
edittext->indent = bitstream.ReadU16();
|
|
|
|
edittext->leading = bitstream.ReadS16();
|
|
|
|
}
|
|
|
|
edittext->variable = bitstream.ReadString();
|
2012-11-28 15:47:07 +00:00
|
|
|
if( hasText )
|
|
|
|
{
|
|
|
|
const char* text = bitstream.ReadString();
|
2012-11-26 18:58:24 +00:00
|
|
|
idStr initialText;
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
// convert html tags if necessary
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; text[i] != 0; i++ )
|
|
|
|
{
|
|
|
|
if( text[i] == '<' )
|
|
|
|
{
|
|
|
|
if( i != 0 && text[i + 1] == 'p' )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
initialText.Append( '\n' );
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
for( ; text[i] != 0 && text[i] != '>'; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
byte tc = ( byte )text[i];
|
|
|
|
if( tc == '&' )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
idStr special;
|
2012-11-28 15:47:07 +00:00
|
|
|
for( i++; text[i] != 0 && text[i] != ';'; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
special.Append( text[i] );
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
if( special.Icmp( "amp" ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
tc = '&';
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else if( special.Icmp( "apos" ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
tc = '\'';
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else if( special.Icmp( "lt" ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
tc = '<';
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else if( special.Icmp( "gt" ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
tc = '>';
|
2012-11-28 15:47:07 +00:00
|
|
|
}
|
|
|
|
else if( special.Icmp( "quot" ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
tc = '\"';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
initialText.Append( tc );
|
|
|
|
}
|
|
|
|
edittext->initialText = initialText;
|
|
|
|
}
|
|
|
|
edittext->flags |= wordWrap ? SWF_ET_WORDWRAP : 0;
|
|
|
|
edittext->flags |= multiline ? SWF_ET_MULTILINE : 0;
|
|
|
|
edittext->flags |= password ? SWF_ET_PASSWORD : 0;
|
|
|
|
edittext->flags |= readonly ? SWF_ET_READONLY : 0;
|
|
|
|
edittext->flags |= autoSize ? SWF_ET_AUTOSIZE : 0;
|
|
|
|
edittext->flags |= border ? SWF_ET_BORDER : 0;
|
|
|
|
}
|