thug/Code/Gfx/2D/TextElement.cpp
2016-02-14 08:39:12 +11:00

1375 lines
43 KiB
C++

#include <core/defines.h>
#include <gel/scripting/checksum.h>
#include <gel/scripting/script.h>
#include <gel/scripting/struct.h>
#include <gel/scripting/array.h>
#include <gel/scripting/vecpair.h>
#include <gfx/NxFontMan.h>
#include <gfx/NxFont.h>
#include <gfx/2D/ScreenElement2.h>
#include <gfx/2D/ScreenElemMan.h>
#include <gfx/2D/TextElement.h>
#include <gfx/2D/BlurEffect.h>
#include <gfx/2D/Window.h>
#define BLUR_EFFECT_ON 1
namespace Front
{
/******************************************************************/
/* */
/* */
/******************************************************************/
CTextElement::CTextElement() :
m_shadow_rgba(0, 0, 0, 128)
{
//Ryan("I am a new CTextElement\n");
m_font_checksum = 0;
mp_font = NULL;
mp_text = NULL;
mp_blur_effect = NULL;
m_use_shadow = false;
create_text_instances(1);
m_override_encoded_rgba = false;
m_previous_override_rgba_state = false;
SetType(CScreenElement::TYPE_TEXT_ELEMENT);
}
/******************************************************************/
/* */
/* */
/******************************************************************/
CTextElement::~CTextElement()
{
//Ryan("Destroying CTextElement\n");
DetachBlurEffect();
destroy_text_instances();
if (mp_text)
delete mp_text;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::SetProperties(Script::CStruct *pProps)
{
CScreenElement::SetProperties(pProps);
uint32 font_crc;
if (pProps->GetChecksum(CRCD(0x2f6bf72d,"font"), &font_crc))
SetFont(font_crc);
Dbg_MsgAssert(m_font_checksum, ("no font loaded"));
const char *p_text;
if (pProps->GetText(CRCD(0xc4745838,"text"), &p_text))
SetText(p_text);
Dbg_MsgAssert(m_id != Obj::CBaseManager::vNO_OBJECT_ID, ("what, no ID"));
if (pProps->ContainsFlag(CRCD(0xd1653c6c,"blur_effect")))
AttachBlurEffect();
if (pProps->ContainsFlag(CRCD(0x4b73e4f8,"no_blur_effect")))
DetachBlurEffect();
Image::RGBA blur_rgba;
if (resolve_rgba(pProps, CRCD(0x4f0e1182,"blur_rgba"), &blur_rgba) && mp_blur_effect)
mp_blur_effect->SetRGBA(blur_rgba);
Script::CPair shadow_offs;
if (pProps->GetPair(CRCD(0x2a1fe0cc,"shadow_offs"), &shadow_offs))
{
SetShadowOff(shadow_offs.mX, shadow_offs.mY);
}
if (pProps->ContainsFlag(CRCD(0x8a897dd2,"shadow")))
SetShadowState(true);
if (pProps->ContainsFlag(CRCD(0x95b5b9a0,"no_shadow")))
SetShadowState(false);
resolve_rgba(pProps, CRCD(0x1e7bb1f5,"shadow_rgba"), &m_shadow_rgba);
if ( pProps->ContainsFlag( CRCD(0xbf51a0c,"remember_override_rgba_state") ) )
{
Dbg_MsgAssert( !pProps->ContainsFlag( CRCD(0x61b7b58b,"restore_override_rgba_state") ), ( "restore_override_rgba_state and remember_override_rgba_state?" ) );
m_previous_override_rgba_state = m_override_encoded_rgba;
}
if ( pProps->ContainsFlag( CRCD(0x61b7b58b,"restore_override_rgba_state") ) )
{
Dbg_MsgAssert( !pProps->ContainsFlag( CRCD(0xbf51a0c,"remember_override_rgba_state") ), ( "restore_override_rgba_state and remember_override_rgba_state?" ) );
m_override_encoded_rgba = m_previous_override_rgba_state;
}
if (pProps->ContainsFlag(CRCD(0xb7686f92,"override_encoded_rgba")))
m_override_encoded_rgba = true;
if (pProps->ContainsFlag(CRCD(0xd25bc6cd,"dont_override_encoded_rgba")))
m_override_encoded_rgba = false;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::SetMorph(Script::CStruct *pProps)
{
if (mp_blur_effect)
{
mp_blur_effect->SetMorph(pProps);
}
CScreenElement::SetMorph(pProps);
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::SetFont(uint32 font_checksum)
{
m_font_checksum = font_checksum;
mp_font = Nx::CFontManager::sGetFont(m_font_checksum);
Dbg_MsgAssert(Nx::CFontManager::sTestFontLoaded(m_font_checksum), ("font %s isn't loaded", Script::FindChecksumName(m_font_checksum)));
m_object_flags |= CScreenElement::vCHANGED_STATIC_PROPS;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::SetText(const char *pText)
{
int new_length = strlen(pText) + 1;
Dbg_MsgAssert(new_length < vMAX_TEXT_LENGTH, ("string too long %d", new_length));
// if the old string is not big enough, then we need to allocate a new one
// otherwize, overwrite it, as allocations are slow
if (!mp_text || (int)(strlen(mp_text)) < new_length-1) // note, does not account for actual allocated size, but will work is string are same size
{
if (mp_text)
delete mp_text;
Mem::Manager::sHandle().PushContext(Mem::Manager::sHandle().FrontEndHeap());
mp_text = new char[new_length];
Mem::Manager::sHandle().PopContext();
}
// copy pText to mp_text, replacing '\m' tags with '\b' tags
const char *p_in = pText;
char *p_out = mp_text;
while(*p_in)
{
if (*p_in == '\\' && *(p_in+1) == '\\')
{
// output "\\" tag unchanged
*p_out++ = *p_in++;
*p_out++ = *p_in++;
}
// ignore sticky space
else if (*p_in == '\\' && *(p_in+1) == '_')
{
p_in++;
p_in++;
*p_out++ = ' ';
}
else if (*p_in == '\\' && *(p_in+1) == 'm')
{
*p_out++ = *p_in++;
*p_out++ = 'b';
p_in++;
*p_out++ = Nx::CFontManager::sMapMetaCharacterToButton(p_in++);
}
else
{
*p_out++ = *p_in++;
}
}
*p_out = '\0';
if( mp_font )
{
float w, h;
mp_font->QueryString(mp_text, w, h); // width and height in pixels
if (!(m_object_flags & vFORCED_DIMS))
{
SetDims(w, h);
}
}
m_object_flags |= CScreenElement::vCHANGED_STATIC_PROPS;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
int CTextElement::GetLength()
{
return strlen( mp_text );
}
/******************************************************************/
/* */
/* */
/******************************************************************/
bool CTextElement::Concatenate( const char *pText )
{
int old_length = 0;
if ( mp_text )
old_length += strlen( mp_text );
int length = old_length + strlen( pText );
if ( length + 1 >= vMAX_TEXT_LENGTH )
{
// new string would be too long...
return false;
}
if ( length > 0 )
{
char new_string[vMAX_TEXT_LENGTH];
// initialize string
strcpy( new_string, "" );
if ( mp_text )
strcpy( new_string, mp_text );
// add new text
strcat( new_string, pText );
SetText( new_string );
return true;
}
return false;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
bool CTextElement::Backspace()
{
if ( !mp_text )
return false;
int length = strlen( mp_text );
if ( length == 0 )
return false;
// copy string
char new_string[vMAX_TEXT_LENGTH];
strcpy( new_string, mp_text );
// take care of escaped backslash char (ASCII code 92)
if ( length > 1 && new_string[length - 1] == 92 && new_string[length - 2] == 92 )
new_string[length - 2] = '\0';
// terminate
new_string[length - 1] = '\0';
SetText( new_string );
return true;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::AttachBlurEffect()
{
#if BLUR_EFFECT_ON
if (!mp_blur_effect)
{
destroy_text_instances();
// create blur effect according to specs
Mem::Manager::sHandle().PushContext(Mem::Manager::sHandle().FrontEndHeap());
mp_blur_effect = new CBlurEffect();
// mp_blur_effect->SetRGBA(m_rgba);
mp_blur_effect->SetRGBA( m_local_props.GetRGBA() );
Mem::Manager::sHandle().PopContext();
// get number of CText objects needed
// create 'em
create_text_instances(mp_blur_effect->GetNumEntries() + 1);
}
#endif
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::DetachBlurEffect()
{
if (mp_blur_effect)
{
destroy_text_instances();
delete mp_blur_effect;
mp_blur_effect = NULL;
create_text_instances(1);
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::SetShadowState(bool shadowOn)
{
if (shadowOn && !m_use_shadow)
{
m_use_shadow = true;
create_text_instances(m_num_tab_entries, true);
}
else if (!shadowOn && m_use_shadow)
{
m_use_shadow = false;
destroy_text_instances(true);
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::SetShadowOff(float offX, float offY)
{
m_shadow_off_x = offX;
m_shadow_off_y = offY;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::update()
{
//Dbg_Message("I am drawing text element at (%.1f,%.1f), scale %.1f\n", m_summed_props.ulx, m_summed_props.uly, m_summed_props.scalex);
#if 0
if( mp_font )
{
mp_font->BeginText( *((uint32 *) &m_local_props.GetRGBA() ), m_summed_props.scalex);
mp_font->DrawString(mp_text, m_summed_props.ulx, m_summed_props.uly);
mp_font->EndText();
}
#else
if (!mp_text) return;
bool offscreen = false;
// HACK
if (m_summed_props.GetScreenUpperLeftY() < -200.0f || m_summed_props.GetScreenUpperLeftY() > 648.0f)
offscreen = true;
if (m_num_tab_entries)
{
Image::RGBA true_rgba = m_local_props.GetRGBA();
if (m_summed_props.alpha >= .0001f)
true_rgba.a = (uint8) ((float) m_local_props.GetRGBA().a * m_summed_props.alpha);
else
true_rgba.a = 0;
/*
-pass current pos, scale, alpha (local) to blur effect
-for each entry in table
-get pos, scale, alpa (in local, relative to current)
*/
mpp_text_req_tab[0]->SetFont(mp_font);
mpp_text_req_tab[0]->SetPos(m_summed_props.GetScreenUpperLeftX(), m_summed_props.GetScreenUpperLeftY());
mpp_text_req_tab[0]->SetRGBA(true_rgba, m_override_encoded_rgba);
mpp_text_req_tab[0]->SetScale(m_summed_props.GetScaleX(), m_summed_props.GetScaleY());
mpp_text_req_tab[0]->SetString(mp_text);
if (m_object_flags & CScreenElement::v3D_POS)
{
mpp_text_req_tab[0]->SetZValue(m_summed_props.GetScreenPosZ());
mpp_text_req_tab[0]->SetHidden( ( ( m_object_flags & CScreenElement::v3D_CULLED ) || IsHidden() ) );
} else {
mpp_text_req_tab[0]->SetPriority(m_z_priority);
mpp_text_req_tab[0]->SetHidden( ( offscreen || IsHidden() ) );
}
#if 0 // Garrett: Can't find window
// Update the clip window. This should only be done at init time, instead.
CWindowElement *p_window = get_window();
mpp_text_req_tab[0]->SetWindow(p_window->GetClipWindow());
#endif
if (m_use_shadow)
{
Image::RGBA true_rgba = m_shadow_rgba;
if (m_summed_props.alpha >= .0001f)
true_rgba.a = (uint8) ((float) m_shadow_rgba.a * m_summed_props.alpha);
else
true_rgba.a = 0;
mpp_shadow_req_tab[0]->SetFont(mp_font);
mpp_shadow_req_tab[0]->SetPos(m_summed_props.GetScreenUpperLeftX() + m_shadow_off_x, m_summed_props.GetScreenUpperLeftY() + m_shadow_off_y);
mpp_shadow_req_tab[0]->SetRGBA(true_rgba, m_override_encoded_rgba);
mpp_shadow_req_tab[0]->SetScale(m_summed_props.GetScaleX(), m_summed_props.GetScaleY());
mpp_shadow_req_tab[0]->SetString(mp_text);
if (m_object_flags & CScreenElement::v3D_POS)
{
mpp_shadow_req_tab[0]->SetZValue(m_summed_props.GetScreenPosZ());
mpp_shadow_req_tab[0]->SetHidden( ( ( m_object_flags & CScreenElement::v3D_CULLED ) || IsHidden() ) );
} else {
mpp_shadow_req_tab[0]->SetPriority(m_z_priority - .005f);
mpp_shadow_req_tab[0]->SetHidden( ( offscreen || IsHidden() ) );
}
}
}
if(mp_blur_effect)
{
/*
-pass current center pos, scale, alpha (local) to blur effect
-for each entry in table
-get center pos, scale, alpa (in local, relative to current)
*/
Dbg_MsgAssert(!(m_object_flags & CScreenElement::v3D_POS), ("Can't use motion blur on 3D positioned text"));
mp_blur_effect->Update();
// work out absolute center of this element
float center_x = m_summed_props.GetScreenUpperLeftX() + m_summed_props.GetScaleX() * m_base_w / 2.0f;
float center_y = m_summed_props.GetScreenUpperLeftY() + m_summed_props.GetScaleY() * m_base_h / 2.0f;
for (int i = 1; i < m_num_tab_entries; i++)
{
ConcatProps &blur_entry = mp_blur_effect->GetInfo(i-1);
Image::RGBA true_rgba = mp_blur_effect->GetRGBA();
true_rgba.a = (uint8) ((float) true_rgba.a * blur_entry.alpha);
float draw_pos_x = center_x - (blur_entry.GetScreenPosX() + m_base_w * blur_entry.GetScaleX() / 2.0f) * m_summed_props.GetScaleX();
float draw_pos_y = center_y - (blur_entry.GetScreenPosY() + m_base_h * blur_entry.GetScaleY() / 2.0f) * m_summed_props.GetScaleY();
float draw_scale_x = m_summed_props.GetScaleX() * blur_entry.GetScaleX();
float draw_scale_y = m_summed_props.GetScaleY() * blur_entry.GetScaleY();
mpp_text_req_tab[i]->SetPos(draw_pos_x, draw_pos_y);
mpp_text_req_tab[i]->SetRGBA(true_rgba, m_override_encoded_rgba);
mpp_text_req_tab[i]->SetScale(draw_scale_x, draw_scale_y);
mpp_text_req_tab[i]->SetPriority(m_z_priority - i * .01f);
mpp_text_req_tab[i]->SetFont(mp_font);
mpp_text_req_tab[i]->SetString(mp_text);
mpp_text_req_tab[i]->SetHidden( ( offscreen || IsHidden() ) );
if (m_use_shadow)
{
Image::RGBA true_rgba = m_shadow_rgba;
if (m_summed_props.alpha >= .0001f)
true_rgba.a = (uint8) ((float) m_shadow_rgba.a * m_summed_props.alpha);
else
true_rgba.a = 0;
mpp_shadow_req_tab[i]->SetPos(draw_pos_x + m_shadow_off_x * draw_scale_x,
draw_pos_y + m_shadow_off_y * draw_scale_y);
mpp_shadow_req_tab[i]->SetRGBA(true_rgba, m_override_encoded_rgba);
mpp_shadow_req_tab[i]->SetScale(draw_scale_x, draw_scale_y);
mpp_shadow_req_tab[i]->SetPriority(m_z_priority - i * .01f - .005f);
mpp_shadow_req_tab[i]->SetFont(mp_font);
mpp_shadow_req_tab[i]->SetString(mp_text);
mpp_shadow_req_tab[i]->SetHidden( ( offscreen || IsHidden() ) );
}
}
}
#endif
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::create_text_instances(int numEntries, bool shadow_only)
{
Dbg_Assert(!mpp_shadow_req_tab);
if (m_use_shadow)
{
Mem::Manager::sHandle().PushContext(Mem::Manager::sHandle().FrontEndHeap());
mpp_shadow_req_tab = new Nx::CText*[numEntries];
Mem::Manager::sHandle().PopContext();
for (int i = 0; i < numEntries; i++)
{
mpp_shadow_req_tab[i] = Nx::CTextMan::sGetTextInstance();
}
}
if (!shadow_only)
{
Dbg_Assert(!m_num_tab_entries);
m_num_tab_entries = numEntries;
Mem::Manager::sHandle().PushContext(Mem::Manager::sHandle().FrontEndHeap());
mpp_text_req_tab = new Nx::CText*[numEntries];
Mem::Manager::sHandle().PopContext();
for (int i = 0; i < numEntries; i++)
{
mpp_text_req_tab[i] = Nx::CTextMan::sGetTextInstance();
}
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextElement::destroy_text_instances(bool shadow_only)
{
if (mpp_shadow_req_tab)
{
for (int i = 0; i < m_num_tab_entries; i++)
{
Nx::CTextMan::sFreeTextInstance(mpp_shadow_req_tab[i]);
}
delete mpp_shadow_req_tab;
mpp_shadow_req_tab = NULL;
}
if (!shadow_only)
{
Dbg_Assert(m_num_tab_entries);
for (int i = 0; i < m_num_tab_entries; i++)
{
Nx::CTextMan::sFreeTextInstance(mpp_text_req_tab[i]);
}
delete mpp_text_req_tab;
m_num_tab_entries = 0;
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
CTextBlockElement::CTextBlockElement() :
m_shadow_rgba(0, 0, 0, 128)
{
//Ryan("I am a new CTextBlockElement\n");
m_font = 0;
m_internal_scale = 1.0f;
m_line_spacing_scale = 1.0f;
m_total_height = 0.0f;
m_total_out_lines = 0;
mp_blur_effect = NULL;
m_allow_expansion = false;
m_override_encoded_rgba = false;
m_previous_override_rgba_state = false;
SetType(CScreenElement::TYPE_TEXT_BLOCK_ELEMENT);
}
/******************************************************************/
/* */
/* */
/******************************************************************/
CTextBlockElement::~CTextBlockElement()
{
//Ryan("Destroying CTextBlockElement\n");
if (mp_blur_effect)
delete mp_blur_effect;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::SetProperties(Script::CStruct *pProps)
{
CScreenElement::SetProperties(pProps);
// must find this before setting font
if (pProps->GetFloat(CRCD(0x1fe341d2,"internal_scale"), &m_internal_scale))
m_object_flags |= CScreenElement::vCHANGED_STATIC_PROPS;
pProps->GetFloat(CRCD(0xe3fa22fc,"line_spacing"), &m_line_spacing_scale);
uint32 font_crc;
if (pProps->GetChecksum(CRCD(0x2f6bf72d,"font"), &font_crc))
SetFont(font_crc);
Dbg_MsgAssert(m_font, ("no font loaded"));
if (resolve_just(pProps, CRCD(0x67e093e4,"internal_just"), &m_internal_just_x, &m_internal_just_y))
m_object_flags |= CScreenElement::vCHANGED_STATIC_PROPS;
const char *pp_line_tab[32];
Script::CArray *p_text_array;
if (pProps->ContainsFlag( CRCD(0x322839a2,"allow_expansion") ))
m_allow_expansion = true;
if (pProps->GetText(CRCD(0xc4745838,"text"), pp_line_tab))
{
SetText(pp_line_tab, 1);
}
else if (pProps->GetArray(CRCD(0xc4745838,"text"), &p_text_array))
{
int num_entries = p_text_array->GetSize();
for (int i = 0; i < num_entries; i++)
pp_line_tab[i] = p_text_array->GetString(i);
SetText(pp_line_tab, num_entries);
}
/*
Property changes to be forwared to children follow:
*/
if (pProps->ContainsFlag(CRCD(0xd1653c6c,"blur_effect")) && !mp_blur_effect)
{
#if BLUR_EFFECT_ON
Mem::Manager::sHandle().PushContext(Mem::Manager::sHandle().FrontEndHeap());
mp_blur_effect = new CBlurEffect();
// mp_blur_effect->SetRGBA(m_rgba);
mp_blur_effect->SetRGBA( m_local_props.GetRGBA() );
Mem::Manager::sHandle().PopContext();
#endif
}
else if (pProps->ContainsFlag(CRCD(0x4b73e4f8,"no_blur_effect")) && mp_blur_effect)
{
delete mp_blur_effect;
mp_blur_effect = NULL;
}
Image::RGBA blur_rgba;
if (resolve_rgba(pProps, CRCD(0x4f0e1182,"blur_rgba"), &blur_rgba) && mp_blur_effect)
{
mp_blur_effect->SetRGBA(blur_rgba);
}
Script::CPair shadow_offs;
if (pProps->GetPair(CRCD(0x2a1fe0cc,"shadow_offs"), &shadow_offs))
{
m_shadow_off_x = shadow_offs.mX;
m_shadow_off_y = shadow_offs.mY;
}
if (pProps->ContainsFlag(CRCD(0x8a897dd2,"shadow")))
m_use_shadow = true;
if (pProps->ContainsFlag(CRCD(0x95b5b9a0,"no_shadow")))
m_use_shadow = false;
resolve_rgba(pProps, CRCD(0x1e7bb1f5,"shadow_rgba"), &m_shadow_rgba);
if ( pProps->ContainsFlag( CRCD(0xbf51a0c,"remember_override_rgba_state") ) )
{
Dbg_MsgAssert( !pProps->ContainsFlag( CRCD(0x61b7b58b,"restore_override_rgba_state") ), ( "restore_override_rgba_state and remember_override_rgba_state?" ) );
m_previous_override_rgba_state = m_override_encoded_rgba;
}
if ( pProps->ContainsFlag( CRCD(0x61b7b58b,"restore_override_rgba_state") ) )
{
Dbg_MsgAssert( !pProps->ContainsFlag( CRCD(0xbf51a0c,"remember_override_rgba_state") ), ( "restore_override_rgba_state and remember_override_rgba_state?" ) );
m_override_encoded_rgba = m_previous_override_rgba_state;
}
if (pProps->ContainsFlag(CRCD(0xb7686f92,"override_encoded_rgba")))
m_override_encoded_rgba = true;
if (pProps->ContainsFlag(CRCD(0xd25bc6cd,"dont_override_encoded_rgba")))
m_override_encoded_rgba = false;
forward_properties_to_children();
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::SetMorph(Script::CStruct *pProps)
{
if (mp_blur_effect)
{
// Distribute blur effect settings to children, account for different
// vertical positions of children relative to center of this
// cheap hack, update self so that contained elements have proper position
update();
Tmr::Time target_time;
const CBlurEffect::Props &blur_target = mp_blur_effect->SetMorph(pProps, &target_time);
//printf("%f\n", blur_target.topAlpha);
// hook up blur effect to children
CScreenElementPtr p_child = CScreenElement::GetFirstChild();
while(p_child)
{
Dbg_MsgAssert(p_child->GetType() == CScreenElement::TYPE_TEXT_ELEMENT,("TextBlockElement %s has child %s that is NOT a CTextElement (it's %s)\n",
Script::FindChecksumName(GetID()), Script::FindChecksumName(p_child->GetID()), Script::FindChecksumName(p_child->GetType())));
// "child blur target"
CBlurEffect::Props cbt = blur_target;
// find center of child relative to center of this
// (we know we'll be getting top of child, since that's the just this class sets)
float child_center_x, child_center_y;
p_child->GetLocalPos(&child_center_x, &child_center_y);
child_center_y += p_child->GetBaseH() / 2.0f;
float disp_y = m_base_h / 2.0f - child_center_y;
cbt.maxDisplacementY += blur_target.bottomScaleY * disp_y - disp_y;
CBlurEffect *p_child_blur = ((CTextElement *) ((CScreenElement *) p_child))->GetBlurEffect();
Dbg_Assert(p_child_blur);
p_child_blur->SetAllTargetProps(cbt, target_time);
p_child = p_child->GetNextSibling();
}
}
CScreenElement::SetMorph(pProps);
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::SetFont(uint32 font_checksum)
{
Dbg_MsgAssert(!m_font, ("font already set"));
m_font = font_checksum;
Dbg_MsgAssert(Nx::CFontManager::sTestFontLoaded(m_font), ("font %s isn't loaded", Script::FindChecksumName(m_font)));
// how many lines will fit?
Nx::CFont* p_font = Nx::CFontManager::sGetFont(m_font);
Dbg_MsgAssert(p_font, ("no font found"));
// see if word wraps past end of element
float word_w, word_h;
p_font->QueryString("AAA", word_w, word_h); // width and height in pixels
m_num_visible_lines = (int) m_base_h / (int) (word_h * m_internal_scale);
m_object_flags |= CScreenElement::vCHANGED_STATIC_PROPS;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::SetText(const char **ppTextLines, int numLines)
{
Dbg_MsgAssert(m_object_flags & vFORCED_DIMS, ("no dimensions have been set"));
Dbg_MsgAssert(numLines < 256, ("large number of lines, probably not good"));
// char parsed_lines[MAX_LINES][MAX_CHARS];
CScreenElementManager* p_manager = CScreenElementManager::Instance();
// destroy any children that are present
SetChildLockState( UNLOCK );
CScreenElementPtr p_child = GetFirstChild();
while ( p_child )
{
CScreenElementPtr p_next = p_child->GetNextSibling();
p_manager->DestroyElement(p_child->GetID());
p_child = p_next;
}
Dbg_Assert( !GetFirstChild() );
Mem::Manager::sHandle().PushContext( Mem::Manager::sHandle().FrontEndHeap() );
mpp_parsed_lines = new char*[MAX_LINES];
// (Mick) in order to avoid doing multiple allocations, we just
// allocate a single array big enough for all the lines
mpp_parsed_lines[0] = new char[MAX_CHARS * MAX_LINES];
Mem::Manager::sHandle().PopContext();
// then calculate the pointers for all the other lines, as an offset to this
for ( int l = 1; l < MAX_LINES; l++ )
{
mpp_parsed_lines[l] = mpp_parsed_lines[0] + MAX_CHARS*l;
}
for( int i = 0; i < MAX_LINES; i++ )
{
mpp_parsed_lines[i][0] = '\0';
}
m_out_char = 0;
m_virtual_out_line = 0;
m_out_line = 0;
m_current_line_width = 0.0f;
// reset number of visible lines
if ( m_allow_expansion )
{
m_base_h = 0.0f;
m_num_visible_lines = 0;
}
for ( int in_line = 0; in_line < numLines; in_line++ )
read_in_text_line( ppTextLines[in_line] );
// printf("line %d: %s\n", m_out_line, mpp_parsed_lines[m_out_line]);
// want to handle the "no text" case
if (numLines)
{
m_virtual_out_line++;
m_out_line++;
}
// we only to want to use the N most recent lines,
// where N is the number of lines that will fit
int buf_use_line = 0;
m_total_out_lines = m_out_line;
if ( m_virtual_out_line >= m_num_visible_lines )
{
// expand the element if we're supposed to
if ( m_allow_expansion )
{
Nx::CFont* p_font = Nx::CFontManager::sGetFont(m_font);
Dbg_MsgAssert(p_font, ("no font found"));
// change the height of the element
float word_w, word_h;
p_font->QueryString("AAA", word_w, word_h ); // width and height in pixels
m_base_h = m_virtual_out_line * (word_h * m_internal_scale * m_line_spacing_scale );
m_num_visible_lines = (int) m_base_h / (int) (word_h * m_internal_scale * m_line_spacing_scale );
// re-parse the text now that m_num_visible_lines has changed
// Mick: no need for this, since they are fixed size, we just overwrite them
// for (int l = 0; l < MAX_LINES; l++)
// {
// delete mpp_parsed_lines[l];
// mpp_parsed_lines[l] = new char[MAX_CHARS];
// }
m_out_char = 0;
m_virtual_out_line = 0;
m_out_line = 0;
m_current_line_width = 0.0f;
for (int i = 0; i < numLines; i++)
read_in_text_line(ppTextLines[i]);
// printf("changing m_total_out_lines from %i to %i\n", m_total_out_lines, m_num_visible_lines);
m_total_out_lines = m_num_visible_lines;
}
else
{
Dbg_MsgAssert( m_num_visible_lines, ("Text block element too small for one line") );
buf_use_line = m_virtual_out_line % m_num_visible_lines;
m_total_out_lines = m_num_visible_lines;
}
}
// create new children
m_total_height = 0.0f;
for (int i = 0; i < m_total_out_lines; i++)
{
Mem::Manager::sHandle().PushContext(Mem::Manager::sHandle().FrontEndHeap());
CTextElement *p_new_elem = new CTextElement();
Mem::Manager::sHandle().PopContext();
p_manager->RegisterObject(*p_new_elem);
p_manager->SetParent(this, p_new_elem);
p_new_elem->SetFont(m_font);
p_new_elem->SetText(&mpp_parsed_lines[buf_use_line][0]);
p_new_elem->SetScale(m_internal_scale, m_internal_scale);
m_total_height += p_new_elem->GetBaseH() * m_internal_scale * m_line_spacing_scale;
buf_use_line++;
if (buf_use_line >= m_num_visible_lines)
buf_use_line = 0;
}
// delete the lines, now we've used them
// Mick: Now we just delete the array of chars, and the array of pointers
// for (int l = 0; l < MAX_LINES; l++)
// {
// delete mpp_parsed_lines[l];
// mpp_parsed_lines[l] = NULL;
// }
delete mpp_parsed_lines[0];
delete mpp_parsed_lines;
SetChildLockState(LOCK);
m_object_flags |= CScreenElement::vCHANGED_STATIC_PROPS;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::SetText(const char *pTextLine)
{
SetText(&pTextLine, 1);
}
/******************************************************************/
/* */
/* */
/******************************************************************/
bool CTextBlockElement::GetText( char* p_text, int size )
{
strcpy( p_text, "" );
int total_length = 0;
CScreenElementPtr p_child = GetFirstChild();
while( p_child )
{
Dbg_MsgAssert(p_child->GetType() == CScreenElement::TYPE_TEXT_ELEMENT,("TextBlockElement %s has child %s that is NOT a CTextElement (it's %s)\n",
Script::FindChecksumName(GetID()), Script::FindChecksumName(p_child->GetID()), Script::FindChecksumName(p_child->GetType())));
CScreenElementPtr p_next = p_child->GetNextSibling();
CTextElement* p_text_element = (CTextElement*)p_child.Convert();
char* p_text_element_string = p_text_element->GetText();
total_length += strlen( p_text_element_string );
if ( p_next )
total_length += 1;
if ( total_length < size )
{
strcat( p_text, p_text_element_string );
// add a space between lines if this isn't the last line
if ( p_next )
strcat( p_text, " " );
}
else
{
Dbg_MsgAssert( 0, ( "TextBlockElement::GetText - text too long" ) );
return false;
}
p_child = p_next;
}
return true;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
int CTextBlockElement::GetLength()
{
// printf( "CTextBlockElement::GetLength called on %s\n", Script::FindChecksumName( m_id ) );
int total_length = 0;
CScreenElementPtr p_child = GetFirstChild();
while( p_child )
{
Dbg_MsgAssert(p_child->GetType() == CScreenElement::TYPE_TEXT_ELEMENT,("TextBlockElement %s has child %s that is NOT a CTextElement (it's %s)\n",
Script::FindChecksumName(GetID()), Script::FindChecksumName(p_child->GetID()), Script::FindChecksumName(p_child->GetType())));
CScreenElementPtr p_next = p_child->GetNextSibling();
CTextElement* p_text_element = (CTextElement*)p_child.Convert();
total_length += p_text_element->GetLength();
if ( p_next )
total_length += 1;
p_child = p_next;
}
return total_length;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
bool CTextBlockElement::Backspace()
{
int length = GetLength();
if ( length == 0 )
return false;
Dbg_MsgAssert( length < MAX_EDITABLE_TEXT_BLOCK_LENGTH, ( "Backspace failed - string too long" ) );
char new_string[MAX_EDITABLE_TEXT_BLOCK_LENGTH];
GetText( new_string, MAX_EDITABLE_TEXT_BLOCK_LENGTH );
if ( !new_string )
return false;
// take care of escaped backslash char (ASCII code 92)
if ( length > 1 && new_string[length - 1] == 92 && new_string[length - 2] == 92 )
new_string[length - 2] = '\0';
// terminate
new_string[length - 1] = '\0';
SetText( new_string );
return true;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
/*
Called from read_in_text_line()
*/
CTextBlockElement::EParseResult CTextBlockElement::parse_next_word(char *pWordBuf, const char **ppSource)
{
bool keep_scanning = true;
EParseResult result = NORMAL;
char *p_out = pWordBuf;
while (keep_scanning)
{
switch(**ppSource)
{
case ' ':
*p_out++ = **ppSource;
(*ppSource)++;
keep_scanning = false;
break;
case '\\':
{
(*ppSource)++;
if (**ppSource == '_')
{
*p_out++ = ' ';
(*ppSource)++;
}
else if (**ppSource == 'n')
{
(*ppSource)++;
result = NEXT_LINE;
keep_scanning = false;
}
else
{
// is \?, where ? is some character
// output both
*p_out++ = '\\';
*p_out++ = **ppSource;
(*ppSource)++;
}
break;
} // end case
case '\0':
result = END_OF_BUFFER;
keep_scanning = false;
break;
default:
{
*p_out++ = **ppSource;
(*ppSource)++;
break;
}
} // end switch
Dbg_Assert((int) p_out - (int) pWordBuf < MAX_CHARS - 1);
}
*p_out++ = '\0';
return result;
}
/*
Called from SetText()
*/
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::read_in_text_line(const char *pText)
{
Nx::CFont* p_font = Nx::CFontManager::sGetFont(m_font);
Dbg_MsgAssert(p_font, ("no font found"));
const char *p_in = pText;
char p_word[MAX_CHARS];
EParseResult last_result = NORMAL;
// printf("reading in: %s\n", pText);
while ( last_result != END_OF_BUFFER )
{
EParseResult result = parse_next_word( p_word, &p_in );
// see if word wraps past end of element
float word_w, word_h;
p_font->QueryString(p_word, word_w, word_h); // width and height in pixels
word_w *= m_internal_scale;
// printf("result = %d, word = '%s', screen_w = %.2f\n", result, p_word, word_w);
// if adding this word makes the line too long
// or we had a newline char or similar
// or this word will overflow the buffer (unlikely, but possible)
if ( ( m_current_line_width > 0.0f && m_current_line_width + word_w > m_base_w )
|| ( last_result == NEXT_LINE )
|| ( m_out_char + strlen( p_word ) >= MAX_CHARS )
)
{
// printf("m_current_line_width = %f, word_w = %f, m_base_w = %f\n", m_current_line_width, word_w, m_base_w);
// printf("line %d: %s\n", m_out_line, mpp_parsed_lines[m_out_line]);
// if last character on line is a space, do away with it
if ( mpp_parsed_lines[m_out_line][m_out_char-1] == ' ' )
mpp_parsed_lines[m_out_line][m_out_char-1] = '\0';
// Wrap to next line.
// Once we fill all visible lines, we start recycling the buffer,
// keeping only the most recent lines, hence the virtual line stuff.
m_virtual_out_line++;
m_out_line++;
if ( m_out_line >= m_num_visible_lines )
m_out_line = 0;
m_out_char = 0;
m_current_line_width = 0.0f;
}
Dbg_Assert(m_out_line < MAX_LINES); // need to increase MAX_LINES if triggered
Dbg_MsgAssert(m_out_char + strlen(p_word) < MAX_CHARS, ("overflow, max allowed = %d, needed = %d\n%s%s\n", MAX_CHARS, m_out_char + strlen(p_word),&mpp_parsed_lines[m_out_line][0],p_word));
char *p_out = &mpp_parsed_lines[m_out_line][0] + m_out_char;
strcpy(p_out, p_word);
m_out_char += strlen(p_word);
m_current_line_width += word_w;
last_result = result;
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::update()
{
if (m_object_flags & CScreenElement::vCHANGED_STATIC_PROPS)
{
Dbg_Assert(m_total_height > .00001f || !m_total_out_lines);
// position children
float y = (m_internal_just_y + 1.0f) * (m_base_h - m_total_height) / 2.0f;
CScreenElementPtr p_child = GetFirstChild();
for (int i = 0; i < m_total_out_lines; i++)
{
Dbg_MsgAssert(p_child->GetType() == CScreenElement::TYPE_TEXT_ELEMENT,("TextBlockElement %s has child %s that is NOT a CTextElement (it's %s)\n",
Script::FindChecksumName(GetID()), Script::FindChecksumName(p_child->GetID()), Script::FindChecksumName(p_child->GetType())));
p_child->SetJust(m_internal_just_x, -1.0f);
// p_child->SetRGBA(m_rgba);
p_child->SetRGBA( m_local_props.GetRGBA() );
p_child->SetPos((m_internal_just_x + 1.0f) * m_base_w / 2.0f, y);
y += p_child->GetBaseH() * m_internal_scale * m_line_spacing_scale;
p_child = p_child->GetNextSibling();
}
forward_properties_to_children();
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void CTextBlockElement::forward_properties_to_children()
{
CScreenElementPtr p_child = CScreenElement::GetFirstChild();
while(p_child)
{
Dbg_MsgAssert(p_child->GetType() == CScreenElement::TYPE_TEXT_ELEMENT,("TextBlockElement %s has child %s that is NOT a CTextElement (it's %s)\n",
Script::FindChecksumName(GetID()), Script::FindChecksumName(p_child->GetID()), Script::FindChecksumName(p_child->GetType())));
if (mp_blur_effect)
((CTextElement *) ((CScreenElement *) p_child))->AttachBlurEffect();
else
((CTextElement *) ((CScreenElement *) p_child))->DetachBlurEffect();
if (m_use_shadow)
((CTextElement *) ((CScreenElement *) p_child))->SetShadowState(true);
else
((CTextElement *) ((CScreenElement *) p_child))->SetShadowState(false);
((CTextElement *) ((CScreenElement *) p_child))->SetShadowOff(m_shadow_off_x, m_shadow_off_y);
((CTextElement *) ((CScreenElement *) p_child))->SetShadowRGBA(m_shadow_rgba);
((CTextElement *) ((CScreenElement *) p_child))->SetEncodedRGBAOverride(m_override_encoded_rgba);
p_child = p_child->GetNextSibling();
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
// if enforce_max_width is true, the function will fail if a single word is
// bigger than the width of a line. That is, if the text would go outside
// the specified width and there's no way to wrap, concatenation will fail.
// Only the end of the string is checked. The text already in the element
// is assumed to be safe. Thus it's possible to have a textblock element go
// outside it's box by using SetText directly, rather than using concatenate
bool CTextBlockElement::Concatenate( const char* pText, bool enforce_max_width, bool last_line )
{
#ifdef __NOPT_ASSERT__
int length = GetLength();
Dbg_MsgAssert( length < (int)( MAX_EDITABLE_TEXT_BLOCK_LENGTH - strlen( pText ) ), ( "TextBlock too long to concatenate" ) );
#endif
// setup new, unwrapped string
char* p_new_string = new char[MAX_EDITABLE_TEXT_BLOCK_LENGTH];
GetText( p_new_string, Front::MAX_EDITABLE_TEXT_BLOCK_LENGTH );
strcat( p_new_string, pText );
// enforce max width but allow them to add a space.
if ( (enforce_max_width && strcmp( pText, " " ) != 0) || last_line)
{
Nx::CFont* p_font = Nx::CFontManager::sGetFont( m_font );
Dbg_MsgAssert( p_font, ( "no font found" ) );
// parse through the new text word by word
const char *p_in = p_new_string;
char p_word[MAX_CHARS];
// skip up to the last word
EParseResult last_result = NORMAL;
while ( last_result != END_OF_BUFFER )
{
last_result = parse_next_word( p_word, &p_in );
}
// check that each word is not longer than the max width
float word_w, word_h;
p_font->QueryString( p_word, word_w, word_h ); // width and height in pixels
word_w *= m_internal_scale;
if ( word_w > m_base_w )
{
// too long!
delete p_new_string;
return false;
}
}
SetText( p_new_string );
delete p_new_string;
return true;
}
}