doom3-bfg/neo/swf/SWF_TextInstance.cpp

1618 lines
37 KiB
C++

/*
===========================================================================
Doom 3 BFG Edition GPL Source Code
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
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
#include "../idlib/precompiled.h"
#include "../renderer/Font.h"
idSWFScriptObject_TextInstancePrototype textInstanceScriptObjectPrototype;
idCVar swf_textScrollSpeed( "swf_textScrollSpeed", "80", CVAR_INTEGER, "scroll speed for text" );
idCVar swf_textRndLetterSpeed( "swf_textRndLetterSpeed", "8", CVAR_INTEGER, "scroll speed for text" );
idCVar swf_textRndLetterDelay( "swf_textRndLetterDelay", "100", CVAR_INTEGER, "scroll speed for text" );
idCVar swf_textParagraphSpeed( "swf_textParagraphSpeed", "15", CVAR_INTEGER, "scroll speed for text" );
idCVar swf_textParagraphInc( "swf_textParagraphInc", "1.3", CVAR_FLOAT, "scroll speed for text" );
idCVar swf_subtitleExtraTime( "swf_subtitleExtraTime", "3500", CVAR_INTEGER, "time after subtitles vo is complete" );
idCVar swf_subtitleEarlyTrans( "swf_subtitleEarlyTrans", "3500", CVAR_INTEGER, "early time out to switch the line" );
idCVar swf_subtitleLengthGuess( "swf_subtitleLengthGuess", "10000", CVAR_INTEGER, "early time out to switch the line" );
idCVar swf_textMaxInputLength( "swf_textMaxInputLength", "104", CVAR_INTEGER, "max number of characters that can go into the input line" );
idCVar swf_textStrokeSize( "swf_textStrokeSize", "1.65f", CVAR_FLOAT, "size of font glyph stroke", 0.0f, 2.0f );
idCVar swf_textStrokeSizeGlyphSpacer( "swf_textStrokeSizeGlyphSpacer", "1.5f", CVAR_FLOAT, "additional space for spacing glyphs using stroke" );
/*
========================
idSWFTextInstance::idSWFTextInstance
========================
*/
idSWFTextInstance::idSWFTextInstance()
{
swf = NULL;
}
/*
========================
idSWFTextInstance::~idSWFTextInstance
================== ======
*/
idSWFTextInstance::~idSWFTextInstance()
{
scriptObject.SetText( NULL );
scriptObject.Clear();
scriptObject.Release();
subtitleTimingInfo.Clear();
}
/*
========================
idSWFTextInstance::Init
========================
*/
void idSWFTextInstance::Init( idSWFEditText* _editText, idSWF* _swf )
{
editText = _editText;
swf = _swf;
text = idLocalization::GetString( editText->initialText );
lengthCalculated = false;
variable = editText->variable;
color = editText->color;
visible = true;
selectionStart = -1;
selectionEnd = -1;
scroll = 0;
scrollTime = 0;
maxscroll = 0;
maxLines = 0;
linespacing = 0;
glyphScale = 1.0f;
shiftHeld = false;
tooltip = false;
renderMode = SWF_TEXT_RENDER_NORMAL;
generatingText = false;
triggerGenerate = false;
rndSpotsVisible = 0;
textSpotsVisible = 0;
startRndTime = 0;
charMultiplier = 0;
prevReplaceIndex = 0;
scrollUpdate = false;
ignoreColor = false;
isSubtitle = false;
subLength = 0;
subAlign = 0;
subUpdating = false;
subCharStartIndex = 0;
subNextStartIndex = 0;
subCharEndIndex = 0;
subDisplayTime = 0;
subStartTime = -1;
subSourceID = -1;
subNeedsSwitch = false;
subForceKill = false;
subKillTimeDelay = 0;
subSwitchTime = 0;
subLastWordIndex = 0;
subPrevLastWordIndex = 0;
subInitialLine = true;
textLength = 0;
inputTextStartChar = 0;
renderDelay = swf_textRndLetterDelay.GetInteger();
needsSoundUpdate = false;
useDropShadow = false;
useStroke = false;
strokeStrength = 1.0f;
strokeWeight = swf_textStrokeSize.GetFloat();
scriptObject.SetPrototype( &textInstanceScriptObjectPrototype );
scriptObject.SetText( this );
scriptObject.SetNoAutoDelete( true );
}
/*
========================
idSWFTextInstance::GetTextLength
========================
*/
float idSWFTextInstance::GetTextLength()
{
// CURRENTLY ONLY WORKS FOR SINGLE LINE TEXTFIELDS
if( lengthCalculated && variable.IsEmpty() )
{
return textLength;
}
idStr txtLengthCheck = "";
float len = 0.0f;
if( verify( swf != NULL ) )
{
if( !variable.IsEmpty() )
{
idSWFScriptVar var = swf->GetGlobal( variable );
if( var.IsUndefined() )
{
txtLengthCheck = text;
}
else
{
txtLengthCheck = var.ToString();
}
txtLengthCheck = idLocalization::GetString( txtLengthCheck );
}
else
{
txtLengthCheck = idLocalization::GetString( text );
}
const idSWFEditText* shape = editText;
idSWFDictionaryEntry* fontEntry = swf->FindDictionaryEntry( shape->fontID, SWF_DICT_FONT );
idSWFFont* swfFont = fontEntry->font;
float width = fabs( shape->bounds.br.x - shape->bounds.tl.x );
float postTrans = SWFTWIP( shape->fontHeight );
const idFont* fontInfo = swfFont->fontID;
float glyphScale = postTrans / 48.0f;
int tlen = txtLengthCheck.Length();
int index = 0;
while( index < tlen )
{
scaledGlyphInfo_t glyph;
fontInfo->GetScaledGlyph( glyphScale, txtLengthCheck.UTF8Char( index ), glyph );
len += glyph.xSkip;
if( useStroke )
{
len += ( swf_textStrokeSizeGlyphSpacer.GetFloat() * strokeWeight * glyphScale );
}
if( !( shape->flags & SWF_ET_AUTOSIZE ) && len >= width )
{
len = width;
break;
}
}
}
lengthCalculated = true;
textLength = len;
return textLength;
}
/*
========================
idSWFTextInstance::StartParagraphText
========================
*/
void idSWFTextInstance::StartParagraphText( int time )
{
generatingText = true;
textSpotsVisible = 0;
randomtext = "";
triggerGenerate = false;
startRndTime = time;
rndTime = time;
rnd.SetSeed( time );
prevReplaceIndex = 0;
rndSpotsVisible = text.Length();
indexArray.Clear();
charMultiplier = 0;
text = idLocalization::GetString( text );
lengthCalculated = false;
for( int index = 0; index < text.Length(); ++index )
{
randomtext.Append( " " );
indexArray.Append( index );
}
for( int index = 0; index < indexArray.Num(); ++index )
{
int swapIndex = rnd.RandomInt( indexArray.Num() );
int val = indexArray[index];
indexArray[index] = indexArray[swapIndex];
indexArray[swapIndex] = val;
}
}
/*
========================
idSWFTextInstance::GetParagraphText
========================
*/
idStr idSWFTextInstance::GetParagraphText( int time )
{
if( triggerGenerate )
{
return " ";
}
else if( time - startRndTime < renderDelay )
{
return " ";
}
else if( generatingText )
{
if( time - rndTime >= renderDelay )
{
rndTime = time;
needsSoundUpdate = true;
if( prevReplaceIndex >= text.Length() )
{
generatingText = false;
return text;
}
randomtext[prevReplaceIndex] = text[prevReplaceIndex];
prevReplaceIndex++;
}
}
else
{
scrollUpdate = false;
return text;
}
return randomtext;
}
/*
========================
idSWFTextInstance::StartRandomText
========================
*/
bool idSWFTextInstance::NeedsSoundPlayed()
{
if( soundClip.IsEmpty() )
{
return false;
}
return needsSoundUpdate;
}
/*
========================
idSWFTextInstance::StartRandomText
========================
*/
void idSWFTextInstance::StartRandomText( int time )
{
generatingText = true;
textSpotsVisible = 0;
randomtext = "";
triggerGenerate = false;
startRndTime = time;
rndTime = time;
rnd.SetSeed( time );
rndSpotsVisible = 0;
text = idLocalization::GetString( text );
lengthCalculated = false;
for( int index = 0; index < text.Length(); ++index )
{
if( text[index] == ' ' )
{
randomtext.Append( " " );
}
else
{
randomtext.Append( "." );
rndSpotsVisible++;
}
}
}
/*
========================
idSWFTextInstance::GetRandomText
========================
*/
idStr idSWFTextInstance::GetRandomText( int time )
{
if( triggerGenerate )
{
return " ";
}
else if( time - startRndTime < renderDelay )
{
return " ";
}
else if( generatingText )
{
if( rndSpotsVisible > 0 )
{
int waitTime = swf_textRndLetterSpeed.GetInteger();
if( randomtext.Length() >= 10 )
{
waitTime = waitTime / 3;
}
if( time - rndTime >= waitTime )
{
rndTime = time;
int spotIndex = rnd.RandomInt( rndSpotsVisible );
int cIndex = 0;
for( int c = 0; c < randomtext.Length(); ++ c )
{
if( c >= text.Length() )
{
rndSpotsVisible = 0;
break;
}
if( randomtext[c] == '.' )
{
cIndex++;
}
if( cIndex == spotIndex )
{
bool useCaps = false;
if( c - 1 >= 0 && text[c - 1] == ' ' )
{
useCaps = true;
}
else if( c == 0 )
{
useCaps = true;
}
if( useCaps || renderMode == SWF_TEXT_RENDER_RANDOM_APPEAR_CAPS )
{
randomtext[c] = rnd.RandomInt( 'Z' - 'A' ) + 'A';
}
else
{
randomtext[c] = rnd.RandomInt( 'z' - 'a' ) + 'a';
}
rndSpotsVisible--;
if( !soundClip.IsEmpty() )
{
needsSoundUpdate = true;
}
break;
}
}
}
}
else if( rndSpotsVisible == 0 && textSpotsVisible < text.Length() )
{
if( textSpotsVisible >= randomtext.Length() )
{
textSpotsVisible++;
}
else
{
if( time - rndTime >= swf_textRndLetterSpeed.GetInteger() )
{
rndTime = time;
randomtext[textSpotsVisible] = text[textSpotsVisible];
textSpotsVisible++;
if( !soundClip.IsEmpty() )
{
needsSoundUpdate = true;
}
}
}
}
else
{
return " ";
}
}
else
{
return text;
}
if( rndSpotsVisible == 0 && textSpotsVisible == text.Length() )
{
generatingText = false;
}
return randomtext;
}
/*
==============================================
SUBTITLE FUNCTIONALITY
==============================================
*/
/*
==============================================
idSWFTextInstance::SwitchSubtitleText
==============================================
*/
void idSWFTextInstance::SwitchSubtitleText( int time )
{
subNeedsSwitch = false;
}
void idSWFTextInstance::SetSubNextStartIndex( int value )
{
subNextStartIndex = value;
}
/*
==============================================
idSWFTextInstance::UpdateSubtitle
==============================================
*/
bool idSWFTextInstance::UpdateSubtitle( int time )
{
if( subForceKillQueued )
{
subForceKillQueued = false;
subForceKill = true;
subKillTimeDelay = time + swf_subtitleExtraTime.GetInteger();
}
if( subUpdating && !subForceKill )
{
if( ( time >= subSwitchTime && !subNeedsSwitch ) || ( !subNeedsSwitch && subInitialLine ) )
{
//idLib::Printf( "SWITCH TIME %d / %d \n", time, subSwitchTime );
if( subInitialLine && subtitleTimingInfo.Num() > 0 )
{
if( subStartTime == -1 )
{
subStartTime = time - 600;
}
if( time < subStartTime + subtitleTimingInfo[0].startTime )
{
return true;
}
else
{
text = subtitleText;//subtitleText = "";
subInitialLine = false;
}
}
if( subNextStartIndex + 1 >= text.Length( ) )
{
subForceKillQueued = true;
}
else
{
subCharStartIndex = subNextStartIndex;
//subtitleText.CopyRange( text, subCharStartIndex, subCharEndIndex );
subNeedsSwitch = true;
}
}
}
if( subForceKill )
{
if( time >= subKillTimeDelay )
{
subForceKill = false;
return false;
}
}
return true;
}
/*
==============================================
idSWFTextInstance::SubtitleComplete
==============================================
*/
void idSWFTextInstance::SetSubEndIndex( int endChar, int time )
{
subCharEndIndex = endChar;
if( subCharEndIndex + 1 >= text.Length() )
{
LastWordChanged( subtitleTimingInfo.Num(), time );
}
}
/*
==============================================
idSWFTextInstance::SubtitleComplete
==============================================
*/
void idSWFTextInstance::SubtitleComplete()
{
subInitialLine = true;
subUpdating = false;
isSubtitle = false;
subNeedsSwitch = false;
subCharDisplayTime = 0;
subForceKillQueued = false;
subForceKill = false;
subKillTimeDelay = 0;
subSwitchTime = 0;
subLastWordIndex = 0;
subPrevLastWordIndex = 0;
subStartTime = -1;
subSpeaker = "";
subtitleText = "";
text = "";
subtitleTimingInfo.Clear();
}
/*
==============================================
idSWFTextInstance::LastWordChanged
==============================================
*/
void idSWFTextInstance::LastWordChanged( int wordCount, int time )
{
if( subPrevLastWordIndex + wordCount >= subtitleTimingInfo.Num() )
{
subLastWordIndex = subtitleTimingInfo.Num() - 1;
}
else
{
subLastWordIndex = subPrevLastWordIndex + wordCount - 1;
}
if( subStartTime == -1 )
{
if( subtitleTimingInfo.Num() > 0 )
{
subStartTime = time + subtitleTimingInfo[0].startTime;
}
else
{
subStartTime = time;
}
}
subSwitchTime = subStartTime + subtitleTimingInfo[subLastWordIndex].startTime;// - swf_subtitleEarlyTrans.GetInteger();
//idLib::Printf( "switchtime set 1 %d last word %d\n", subSwitchTime, subLastWordIndex );
}
/*
==============================================
idSWFTextInstance::GetSubtitleBreak
==============================================
*/
int idSWFTextInstance::GetApporoximateSubtitleBreak( int time )
{
int wordIndex = subLastWordIndex;
bool setSwitchTime = false;
if( subStartTime == -1 )
{
subStartTime = time;
}
if( time >= subSwitchTime )
{
subPrevLastWordIndex = subLastWordIndex;
for( int i = wordIndex; i < subtitleTimingInfo.Num(); ++i )
{
if( subtitleTimingInfo[i].forceBreak )
{
if( i + 1 < subtitleTimingInfo.Num() )
{
subSwitchTime = subStartTime + subtitleTimingInfo[i + 1].startTime; // - swf_subtitleEarlyTrans.GetInteger();
//idLib::Printf( "switchtime set 2 %d\n", subSwitchTime );
subLastWordIndex = i;
setSwitchTime = true;
break;
}
else
{
subSwitchTime = subStartTime + subtitleTimingInfo[i].startTime;// - swf_subtitleEarlyTrans.GetInteger();
//idLib::Printf( "switchtime set 3 %d\n", subSwitchTime );
subLastWordIndex = i;
setSwitchTime = true;
break;
}
}
else
{
int timeSpan = subtitleTimingInfo[i].startTime - subtitleTimingInfo[wordIndex].startTime;
if( timeSpan > swf_subtitleLengthGuess.GetInteger() )
{
if( i - 1 >= 0 )
{
subSwitchTime = subStartTime + subtitleTimingInfo[i].startTime;// - swf_subtitleEarlyTrans.GetInteger();
//idLib::Printf( "switchtime set 4 %d\n", subSwitchTime );
subLastWordIndex = i - 1;
}
else
{
subSwitchTime = subStartTime + subtitleTimingInfo[i].startTime;// - swf_subtitleEarlyTrans.GetInteger();
//idLib::Printf( "switchtime set 5 %d\n", subSwitchTime );
subLastWordIndex = i;
}
setSwitchTime = true;
break;
}
}
}
if( !setSwitchTime && subtitleTimingInfo.Num() > 0 )
{
subSwitchTime = subStartTime + subtitleTimingInfo[ subtitleTimingInfo.Num() - 1 ].startTime;// - swf_subtitleEarlyTrans.GetInteger();
//idLib::Printf( "switchtime set 6 %d\n", subSwitchTime );
subLastWordIndex = subtitleTimingInfo.Num();
}
}
return subLastWordIndex;
}
/*
==============================================
idSWFTextInstance::SubtitleComplete
==============================================
*/
void idSWFTextInstance::SubtitleCleanup()
{
subSourceID = -1;
subAlign = -1;
text = "";
}
/*
==============================================
idSWFTextInstance::SetStrokeInfo
==============================================
*/
void idSWFTextInstance::SetStrokeInfo( bool use, float strength, float weight )
{
useStroke = use;
if( use )
{
strokeWeight = weight;
strokeStrength = strength;
}
}
/*
==============================================
idSWFTextInstance::CalcMaxScroll
==============================================
*/
int idSWFTextInstance::CalcMaxScroll( int numLines )
{
if( numLines != -1 )
{
if( numLines < 0 )
{
numLines = 0;
}
maxscroll = numLines;
return maxscroll;
}
const idSWFEditText* shape = editText;
if( !( shape->flags & SWF_ET_MULTILINE ) )
{
return 0;
}
if( swf == NULL )
{
return 0;
}
idSWFDictionaryEntry* fontEntry = swf->FindDictionaryEntry( shape->fontID, SWF_DICT_FONT );
if( fontEntry == NULL )
{
return 0;
}
idSWFFont* swfFont = fontEntry->font;
if( swfFont == NULL )
{
return 0;
}
const idFont* fontInfo = swfFont->fontID;
if( fontInfo == NULL )
{
return 0;
}
idStr textCheck;
if( variable.IsEmpty() )
{
textCheck = idLocalization::GetString( text );
}
else
{
textCheck = idLocalization::GetString( variable );
}
if( textCheck.IsEmpty() )
{
return 0;
}
float x = bounds.tl.x;
float y = bounds.tl.y;
idList< idStr > textLines;
idStr* currentLine = &textLines.Alloc();
// tracks the last breakable character we found
int lastbreak = 0;
float lastbreakX = 0;
int charIndex = 0;
if( IsSubtitle() )
{
charIndex = GetSubStartIndex();
}
while( charIndex < textCheck.Length() )
{
if( textCheck[ charIndex ] == '\n' )
{
if( shape->flags & SWF_ET_MULTILINE )
{
currentLine->Append( '\n' );
x = bounds.tl.x;
y += linespacing;
currentLine = &textLines.Alloc();
lastbreak = 0;
charIndex++;
continue;
}
else
{
break;
}
}
int glyphStart = charIndex;
uint32 tc = textCheck.UTF8Char( charIndex );
scaledGlyphInfo_t glyph;
fontInfo->GetScaledGlyph( glyphScale, tc, glyph );
float glyphSkip = glyph.xSkip;
if( HasStroke() )
{
glyphSkip += ( swf_textStrokeSizeGlyphSpacer.GetFloat() * GetStrokeWeight() * glyphScale );
}
if( x + glyphSkip > bounds.br.x )
{
if( shape->flags & ( SWF_ET_MULTILINE | SWF_ET_WORDWRAP ) )
{
if( lastbreak > 0 )
{
int curLineIndex = currentLine - &textLines[0];
idStr* newline = &textLines.Alloc();
currentLine = &textLines[ curLineIndex ];
if( maxLines == 1 )
{
currentLine->CapLength( currentLine->Length() - 3 );
currentLine->Append( "..." );
break;
}
else
{
*newline = currentLine->c_str() + lastbreak;
currentLine->CapLength( lastbreak );
currentLine = newline;
x -= lastbreakX;
}
}
else
{
currentLine = &textLines.Alloc();
x = bounds.tl.x;
}
lastbreak = 0;
}
else
{
break;
}
}
while( glyphStart < charIndex && glyphStart < text.Length() )
{
currentLine->Append( text[ glyphStart++ ] );
}
x += glyphSkip;
if( tc == ' ' || tc == '-' )
{
lastbreak = currentLine->Length();
lastbreakX = x;
}
}
maxscroll = textLines.Num() - maxLines;
if( maxscroll < 0 )
{
maxscroll = 0;
}
return maxscroll;
}
int idSWFTextInstance::CalcNumLines()
{
const idSWFEditText* shape = editText;
if( !( shape->flags & SWF_ET_MULTILINE ) )
{
return 1;
}
idSWFDictionaryEntry* fontEntry = swf->FindDictionaryEntry( shape->fontID, SWF_DICT_FONT );
if( fontEntry == NULL )
{
return 1;
}
idStr textCheck;
if( variable.IsEmpty() )
{
textCheck = idLocalization::GetString( text );
}
else
{
textCheck = idLocalization::GetString( variable );
}
if( textCheck.IsEmpty() )
{
return 1;
}
if( swf == NULL )
{
return 1;
}
idSWFFont* swfFont = fontEntry->font;
float postTransformHeight = SWFTWIP( shape->fontHeight );
const idFont* fontInfo = swfFont->fontID;
float glyphScale = postTransformHeight / 48.0f;
swfRect_t bounds;
bounds.tl.x = ( shape->bounds.tl.x + SWFTWIP( shape->leftMargin ) );
bounds.br.x = ( shape->bounds.br.x - SWFTWIP( shape->rightMargin ) );
bounds.tl.y = ( shape->bounds.tl.y + ( 1.15f * glyphScale ) );
bounds.br.y = ( shape->bounds.br.y );
float linespacing = fontInfo->GetAscender( 1.15f * glyphScale );
if( shape->leading != 0 )
{
linespacing += ( glyphScale * SWFTWIP( shape->leading ) );
}
float x = bounds.tl.x;
int maxLines = idMath::Ftoi( ( bounds.br.y - bounds.tl.y ) / linespacing );
if( maxLines == 0 )
{
maxLines = 1;
}
// tracks the last breakable character we found
int numLines = 1;
int lastbreak = 0;
int charIndex = 0;
while( charIndex < textCheck.Length() )
{
if( textCheck[ charIndex ] == '\n' )
{
if( numLines == maxLines )
{
return maxLines;
}
numLines++;
lastbreak = 0;
x = bounds.tl.x;
charIndex++;
}
else
{
uint32 tc = textCheck[ charIndex++ ];
scaledGlyphInfo_t glyph;
fontInfo->GetScaledGlyph( glyphScale, tc, glyph );
float glyphSkip = glyph.xSkip;
if( useStroke )
{
glyphSkip += ( swf_textStrokeSizeGlyphSpacer.GetFloat() * strokeWeight * glyphScale );
}
if( x + glyphSkip > bounds.br.x )
{
if( numLines == maxLines )
{
return maxLines;
}
else
{
numLines++;
if( lastbreak != 0 )
{
charIndex = charIndex - ( charIndex - lastbreak );
}
x = bounds.tl.x;
lastbreak = 0;
}
}
else
{
x += glyphSkip;
if( tc == ' ' || tc == '-' )
{
lastbreak = charIndex;
}
}
}
}
return numLines;
}
/*
========================
idSWFScriptObject_TextInstancePrototype
========================
*/
#define SWF_TEXT_FUNCTION_DEFINE( x ) idSWFScriptVar idSWFScriptObject_TextInstancePrototype::idSWFScriptFunction_##x::Call( idSWFScriptObject * thisObject, const idSWFParmList & parms )
#define SWF_TEXT_NATIVE_VAR_DEFINE_GET( x ) idSWFScriptVar idSWFScriptObject_TextInstancePrototype::idSWFScriptNativeVar_##x::Get( class idSWFScriptObject * object )
#define SWF_TEXT_NATIVE_VAR_DEFINE_SET( x ) void idSWFScriptObject_TextInstancePrototype::idSWFScriptNativeVar_##x::Set( class idSWFScriptObject * object, const idSWFScriptVar & value )
#define SWF_TEXT_PTHIS_FUNC( x ) idSWFTextInstance * pThis = thisObject ? thisObject->GetText() : NULL; if ( !verify( pThis != NULL ) ) { idLib::Warning( "SWF: tried to call " x " on NULL edittext" ); return idSWFScriptVar(); }
#define SWF_TEXT_PTHIS_GET( x ) idSWFTextInstance * pThis = object ? object->GetText() : NULL; if ( pThis == NULL ) { return idSWFScriptVar(); }
#define SWF_TEXT_PTHIS_SET( x ) idSWFTextInstance * pThis = object ? object->GetText() : NULL; if ( pThis == NULL ) { return; }
#define SWF_TEXT_FUNCTION_SET( x ) scriptFunction_##x.AddRef(); Set( #x, &scriptFunction_##x );
#define SWF_TEXT_NATIVE_VAR_SET( x ) SetNative( #x, &swfScriptVar_##x );
idSWFScriptObject_TextInstancePrototype::idSWFScriptObject_TextInstancePrototype()
{
SWF_TEXT_FUNCTION_SET( onKey );
SWF_TEXT_FUNCTION_SET( onChar );
SWF_TEXT_FUNCTION_SET( generateRnd );
SWF_TEXT_FUNCTION_SET( calcNumLines );
SWF_TEXT_FUNCTION_SET( clearTimingInfo );
SWF_TEXT_NATIVE_VAR_SET( text );
SWF_TEXT_NATIVE_VAR_SET( _textLength ); // only works on single lines of text not multiline
SWF_TEXT_NATIVE_VAR_SET( autoSize );
SWF_TEXT_NATIVE_VAR_SET( dropShadow );
SWF_TEXT_NATIVE_VAR_SET( _stroke );
SWF_TEXT_NATIVE_VAR_SET( _strokeStrength );
SWF_TEXT_NATIVE_VAR_SET( _strokeWeight );
SWF_TEXT_NATIVE_VAR_SET( variable );
SWF_TEXT_NATIVE_VAR_SET( _alpha );
SWF_TEXT_NATIVE_VAR_SET( textColor );
SWF_TEXT_NATIVE_VAR_SET( _visible );
SWF_TEXT_NATIVE_VAR_SET( selectionStart );
SWF_TEXT_NATIVE_VAR_SET( selectionEnd );
SWF_TEXT_NATIVE_VAR_SET( scroll );
SWF_TEXT_NATIVE_VAR_SET( maxscroll );
SWF_TEXT_NATIVE_VAR_SET( isTooltip );
SWF_TEXT_NATIVE_VAR_SET( mode );
SWF_TEXT_NATIVE_VAR_SET( delay );
SWF_TEXT_NATIVE_VAR_SET( renderSound );
SWF_TEXT_NATIVE_VAR_SET( updateScroll );
SWF_TEXT_NATIVE_VAR_SET( subtitle );
SWF_TEXT_NATIVE_VAR_SET( subtitleAlign );
SWF_TEXT_NATIVE_VAR_SET( subtitleSourceID );
SWF_TEXT_NATIVE_VAR_SET( subtitleSpeaker );
SWF_TEXT_FUNCTION_SET( subtitleSourceCheck );
SWF_TEXT_FUNCTION_SET( subtitleStart );
SWF_TEXT_FUNCTION_SET( subtitleLength );
SWF_TEXT_FUNCTION_SET( killSubtitle );
SWF_TEXT_FUNCTION_SET( forceKillSubtitle );
SWF_TEXT_FUNCTION_SET( subLastLine );
SWF_TEXT_FUNCTION_SET( addSubtitleInfo );
SWF_TEXT_FUNCTION_SET( terminateSubtitle );
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( text )
{
SWF_TEXT_PTHIS_GET( "text" );
return pThis->text;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( text )
{
SWF_TEXT_PTHIS_SET( "text " );
pThis->text = idLocalization::GetString( value.ToString() );
if( pThis->text.IsEmpty() )
{
pThis->selectionEnd = -1;
pThis->selectionStart = -1;
pThis->inputTextStartChar = 0;
}
pThis->lengthCalculated = false;
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( autoSize )
{
SWF_TEXT_PTHIS_GET( "autoSize" );
return ( pThis->editText->flags & SWF_ET_AUTOSIZE ) != 0;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( autoSize )
{
SWF_TEXT_PTHIS_SET( "autoSize" );
if( value.ToBool() )
{
pThis->editText->flags |= SWF_ET_AUTOSIZE;
}
else
{
pThis->editText->flags &= ~SWF_ET_AUTOSIZE;
}
pThis->lengthCalculated = false;
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( dropShadow )
{
SWF_TEXT_PTHIS_GET( "dropShadow" );
return pThis->useDropShadow;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( dropShadow )
{
SWF_TEXT_PTHIS_SET( "dropShadow" );
pThis->useDropShadow = value.ToBool();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( _stroke )
{
SWF_TEXT_PTHIS_GET( "_stroke" );
return pThis->useStroke;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( _stroke )
{
SWF_TEXT_PTHIS_SET( "_stroke" );
pThis->useStroke = value.ToBool();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( _strokeStrength )
{
SWF_TEXT_PTHIS_GET( "_strokeStrength" );
return pThis->strokeStrength;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( _strokeStrength )
{
SWF_TEXT_PTHIS_SET( "_strokeStrength" );
pThis->strokeStrength = value.ToFloat();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( _strokeWeight )
{
SWF_TEXT_PTHIS_GET( "_strokeWeight" );
return pThis->strokeWeight;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( _strokeWeight )
{
SWF_TEXT_PTHIS_SET( "_strokeWeight" );
pThis->strokeWeight = value.ToFloat();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( variable )
{
SWF_TEXT_PTHIS_GET( "variable" );
return pThis->variable;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( variable )
{
SWF_TEXT_PTHIS_SET( "variable" );
pThis->variable = value.ToString();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( _alpha )
{
SWF_TEXT_PTHIS_GET( "_alpha" );
return pThis->color.a / 255.0f;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( _alpha )
{
SWF_TEXT_PTHIS_SET( "_alpha" );
pThis->color.a = idMath::Ftob( value.ToFloat() * 255.0f );
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( _visible )
{
SWF_TEXT_PTHIS_GET( "_visible" );
return pThis->visible;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( _visible )
{
SWF_TEXT_PTHIS_SET( "_visible" );
pThis->visible = value.ToBool();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( selectionStart )
{
SWF_TEXT_PTHIS_GET( "selectionStart" );
return pThis->selectionStart;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( selectionStart )
{
SWF_TEXT_PTHIS_SET( "selectionStart" );
pThis->selectionStart = value.ToInteger();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( selectionEnd )
{
SWF_TEXT_PTHIS_GET( "selectionEnd" );
return pThis->selectionEnd;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( selectionEnd )
{
SWF_TEXT_PTHIS_SET( "selectionEnd" );
pThis->selectionEnd = value.ToInteger();
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( isTooltip )
{
SWF_TEXT_PTHIS_SET( "isTooltip" );
pThis->tooltip = value.ToBool();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( isTooltip )
{
SWF_TEXT_PTHIS_GET( "isTooltip" );
return pThis->tooltip;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( delay )
{
SWF_TEXT_PTHIS_SET( "delay" );
pThis->renderDelay = value.ToInteger();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( delay )
{
SWF_TEXT_PTHIS_GET( "delay" );
return pThis->renderDelay;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( renderSound )
{
SWF_TEXT_PTHIS_SET( "renderSound" );
pThis->soundClip = value.ToString();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( renderSound )
{
SWF_TEXT_PTHIS_GET( "renderSound" );
return pThis->soundClip;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( updateScroll )
{
SWF_TEXT_PTHIS_SET( "updateScroll" );
pThis->scrollUpdate = value.ToBool();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( updateScroll )
{
SWF_TEXT_PTHIS_GET( "updateScroll" );
return pThis->scrollUpdate;
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( mode )
{
SWF_TEXT_PTHIS_GET( "mode" );
return pThis->renderMode;
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( scroll )
{
SWF_TEXT_PTHIS_GET( "scroll" );
return pThis->scroll;
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( maxscroll )
{
SWF_TEXT_PTHIS_GET( "maxscroll" );
return pThis->maxscroll;
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( _textLength )
{
SWF_TEXT_PTHIS_GET( "_textLength" );
return pThis->GetTextLength();
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( mode )
{
SWF_TEXT_PTHIS_SET( "mode" );
int mode = value.ToInteger();
if( mode >= ( int )SWF_TEXT_RENDER_MODE_COUNT || mode < 0 )
{
mode = SWF_TEXT_RENDER_NORMAL;
}
pThis->renderMode = swfTextRenderMode_t( mode );
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( scroll )
{
SWF_TEXT_PTHIS_SET( "scroll" );
int time = Sys_Milliseconds();
if( time >= pThis->scrollTime )
{
pThis->scrollTime = Sys_Milliseconds() + swf_textScrollSpeed.GetInteger();
pThis->scroll = value.ToInteger();
}
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( maxscroll )
{
SWF_TEXT_PTHIS_SET( "maxscroll" );
pThis->maxscroll = value.ToInteger();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( textColor )
{
SWF_TEXT_PTHIS_GET( "textColor" );
int r = ( pThis->color.r << 16 );
int g = ( pThis->color.g << 8 );
int b = pThis->color.b;
int textColor = r | g | b;
return textColor;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( textColor )
{
SWF_TEXT_PTHIS_SET( "textColor" );
int textColor = value.ToInteger();
int r = ( textColor >> 16 ) & 0xFF;
int g = ( textColor >> 8 ) & 0x00FF;
int b = textColor & 0x0000FF;
pThis->color.r = r;
pThis->color.g = g;
pThis->color.b = b;
}
SWF_TEXT_FUNCTION_DEFINE( clearTimingInfo )
{
SWF_TEXT_PTHIS_FUNC( "clearTimingInfo" );
pThis->subtitleTimingInfo.Clear();
return idSWFScriptVar();
}
SWF_TEXT_FUNCTION_DEFINE( generateRnd )
{
SWF_TEXT_PTHIS_FUNC( "generateRnd" );
pThis->triggerGenerate = true;
pThis->rndSpotsVisible = -1;
pThis->generatingText = false;
return idSWFScriptVar();
}
SWF_TEXT_FUNCTION_DEFINE( calcNumLines )
{
SWF_TEXT_PTHIS_FUNC( "calcNumLines" );
return pThis->CalcNumLines();
}
SWF_TEXT_FUNCTION_DEFINE( onKey )
{
SWF_TEXT_PTHIS_FUNC( "onKey" );
int keyCode = parms[0].ToInteger();
bool keyDown = parms[1].ToBool();
if( keyDown )
{
switch( keyCode )
{
case K_LSHIFT:
case K_RSHIFT:
{
pThis->shiftHeld = true;
break;
}
case K_BACKSPACE:
case K_DEL:
{
if( pThis->selectionStart == pThis->selectionEnd )
{
if( keyCode == K_BACKSPACE )
{
pThis->selectionStart = pThis->selectionEnd - 1;
}
else
{
pThis->selectionEnd = pThis->selectionStart + 1;
}
}
int start = Min( pThis->selectionStart, pThis->selectionEnd );
int end = Max( pThis->selectionStart, pThis->selectionEnd );
idStr left = pThis->text.Left( Max( start, 0 ) );
idStr right = pThis->text.Right( Max( pThis->text.Length() - end, 0 ) );
pThis->text = left + right;
pThis->selectionStart = start;
pThis->selectionEnd = start;
break;
}
case K_LEFTARROW:
{
if( pThis->selectionEnd > 0 )
{
pThis->selectionEnd--;
if( !pThis->shiftHeld )
{
pThis->selectionStart = pThis->selectionEnd;
}
}
break;
}
case K_RIGHTARROW:
{
if( pThis->selectionEnd < pThis->text.Length() )
{
pThis->selectionEnd++;
if( !pThis->shiftHeld )
{
pThis->selectionStart = pThis->selectionEnd;
}
}
break;
}
case K_HOME:
{
pThis->selectionEnd = 0;
if( !pThis->shiftHeld )
{
pThis->selectionStart = pThis->selectionEnd;
}
break;
}
case K_END:
{
pThis->selectionEnd = pThis->text.Length();
if( !pThis->shiftHeld )
{
pThis->selectionStart = pThis->selectionEnd;
}
break;
}
}
}
else
{
if( keyCode == K_LSHIFT || keyCode == K_RSHIFT )
{
pThis->shiftHeld = false;
}
}
return true;
}
SWF_TEXT_FUNCTION_DEFINE( onChar )
{
SWF_TEXT_PTHIS_FUNC( "onChar" );
int keyCode = parms[0].ToInteger();
if( keyCode < 32 || keyCode == 127 )
{
return false;
}
char letter = ( char )keyCode;
// assume ` is meant for the console
if( letter == '`' )
{
return false;
}
if( pThis->selectionStart != pThis->selectionEnd )
{
int start = Min( pThis->selectionStart, pThis->selectionEnd );
int end = Max( pThis->selectionStart, pThis->selectionEnd );
idStr left = pThis->text.Left( Max( start, 0 ) );
idStr right = pThis->text.Right( Max( pThis->text.Length() - end, 0 ) );
pThis->text = left + right;
pThis->selectionStart = start;
pThis->text.Clear();
pThis->text.Append( left );
pThis->text.Append( letter );
pThis->text.Append( right );
pThis->selectionStart++;
}
else if( pThis->selectionStart < swf_textMaxInputLength.GetInteger() )
{
if( pThis->selectionStart < 0 )
{
pThis->selectionStart = 0;
}
pThis->text.Insert( letter, pThis->selectionStart++ );
}
pThis->selectionEnd = pThis->selectionStart;
return true;
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( subtitle )
{
SWF_TEXT_PTHIS_GET( "subtitle" );
return pThis->isSubtitle;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( subtitle )
{
SWF_TEXT_PTHIS_SET( "subtitle" );
pThis->isSubtitle = value.ToBool();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( subtitleAlign )
{
SWF_TEXT_PTHIS_GET( "subtitleAlign" );
return pThis->subAlign;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( subtitleAlign )
{
SWF_TEXT_PTHIS_SET( "subtitleAlign" );
pThis->subAlign = value.ToInteger();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( subtitleSourceID )
{
SWF_TEXT_PTHIS_GET( "subtitleSourceID" );
return pThis->subSourceID;
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( subtitleSourceID )
{
SWF_TEXT_PTHIS_SET( "subtitleSourceID" );
pThis->subSourceID = value.ToInteger();
}
SWF_TEXT_NATIVE_VAR_DEFINE_GET( subtitleSpeaker )
{
SWF_TEXT_PTHIS_GET( "subtitleSpeaker" );
return pThis->subSpeaker.c_str();
}
SWF_TEXT_NATIVE_VAR_DEFINE_SET( subtitleSpeaker )
{
SWF_TEXT_PTHIS_SET( "subtitleSpeaker" );
pThis->subSpeaker = value.ToString();
}
SWF_TEXT_FUNCTION_DEFINE( subtitleLength )
{
SWF_TEXT_PTHIS_FUNC( "subtitleLength" );
pThis->subLength = parms[0].ToInteger();
return idSWFScriptVar();
}
SWF_TEXT_FUNCTION_DEFINE( subtitleSourceCheck )
{
SWF_TEXT_PTHIS_FUNC( "subtitleSourceCheck" );
int idCheck = parms[0].ToInteger();
if( pThis->subSourceID == -1 )
{
pThis->subSourceID = idCheck;
return 1;
}
if( idCheck == pThis->subSourceID ) // || pThis->subForceKill ) {
{
pThis->SubtitleComplete();
pThis->subSourceID = idCheck;
return -1;
}
return 0;
}
SWF_TEXT_FUNCTION_DEFINE( subtitleStart )
{
SWF_TEXT_PTHIS_FUNC( "subtitleStart" );
pThis->subUpdating = true;
pThis->subNeedsSwitch = false;
pThis->subForceKillQueued = false;
pThis->subForceKill = false;
pThis->subKillTimeDelay = 0;
// trickery to swap the text so subtitles don't show until they should
pThis->subtitleText = pThis->text;
pThis->text = "";
pThis->subCharStartIndex = 0;
pThis->subNextStartIndex = 0;
pThis->subCharEndIndex = 0;
pThis->subSwitchTime = 0;
pThis->subLastWordIndex = 0;
pThis->subPrevLastWordIndex = 0;
pThis->subStartTime = -1;
pThis->subInitialLine = true;
return idSWFScriptVar();
}
SWF_TEXT_FUNCTION_DEFINE( forceKillSubtitle )
{
SWF_TEXT_PTHIS_FUNC( "forceKillSubtitle" );
pThis->subForceKill = true;
pThis->subKillTimeDelay = 0;
return idSWFScriptVar();
}
SWF_TEXT_FUNCTION_DEFINE( killSubtitle )
{
SWF_TEXT_PTHIS_FUNC( "killSubtitle" );
pThis->subForceKillQueued = true;
//pThis->SubtitleComplete();
return idSWFScriptVar();
}
SWF_TEXT_FUNCTION_DEFINE( terminateSubtitle )
{
SWF_TEXT_PTHIS_FUNC( "terminateSubtitle" );
pThis->SubtitleComplete();
pThis->SubtitleCleanup();
return idSWFScriptVar();
}
SWF_TEXT_FUNCTION_DEFINE( subLastLine )
{
SWF_TEXT_PTHIS_FUNC( "subLastLine" );
idStr lastLine;
int len = pThis->subCharEndIndex - pThis->subCharStartIndex;
pThis->text.Mid( pThis->subCharStartIndex, len, lastLine );
return lastLine;
}
SWF_TEXT_FUNCTION_DEFINE( addSubtitleInfo )
{
SWF_TEXT_PTHIS_FUNC( "addSubtitleInfo" );
if( parms.Num() != 3 )
{
return idSWFScriptVar();
}
subTimingWordData_t info;
info.phrase = parms[0].ToString();
info.startTime = parms[1].ToInteger();
info.forceBreak = parms[2].ToBool();
pThis->subtitleTimingInfo.Append( info );
return idSWFScriptVar();
}