mirror of
https://github.com/panda3d/panda3d.git
synced 2025-09-29 16:20:11 -04:00
text: Make TextNode pipeline-cycled
Fixes #1070 Note that text property changes are not pipeline-cycled at the moment.
This commit is contained in:
parent
54a4a6abfc
commit
ac4faf3d74
File diff suppressed because it is too large
Load Diff
@ -62,36 +62,9 @@ TextNode::
|
||||
TextNode(const string &name) : PandaNode(name) {
|
||||
set_cull_callback();
|
||||
|
||||
_flags = 0;
|
||||
_max_rows = 0;
|
||||
_usage_hint = GeomEnums::UH_static;
|
||||
_flatten_flags = 0;
|
||||
if (text_flatten) {
|
||||
_flatten_flags |= FF_strong;
|
||||
}
|
||||
if (text_dynamic_merge) {
|
||||
_flatten_flags |= FF_dynamic_merge;
|
||||
}
|
||||
|
||||
if (text_small_caps) {
|
||||
TextProperties::set_small_caps(true);
|
||||
}
|
||||
|
||||
_frame_color.set(1.0f, 1.0f, 1.0f, 1.0f);
|
||||
_card_color.set(1.0f, 1.0f, 1.0f, 1.0f);
|
||||
|
||||
_frame_width = 1.0f;
|
||||
|
||||
_frame_ul.set(0.0f, 0.0f);
|
||||
_frame_lr.set(0.0f, 0.0f);
|
||||
_card_ul.set(0.0f, 0.0f);
|
||||
_card_lr.set(0.0f, 0.0f);
|
||||
|
||||
_transform = LMatrix4::ident_mat();
|
||||
_coordinate_system = CS_default;
|
||||
|
||||
_ul3d.set(0.0f, 0.0f, 0.0f);
|
||||
_lr3d.set(0.0f, 0.0f, 0.0f);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -102,25 +75,6 @@ TextNode::
|
||||
TextNode(const string &name, const TextProperties ©) :
|
||||
PandaNode(name), TextProperties(copy)
|
||||
{
|
||||
_flags = 0;
|
||||
_max_rows = 0;
|
||||
_usage_hint = GeomEnums::UH_static;
|
||||
|
||||
_frame_color.set(1.0f, 1.0f, 1.0f, 1.0f);
|
||||
_card_color.set(1.0f, 1.0f, 1.0f, 1.0f);
|
||||
|
||||
_frame_width = 1.0f;
|
||||
|
||||
_frame_ul.set(0.0f, 0.0f);
|
||||
_frame_lr.set(0.0f, 0.0f);
|
||||
_card_ul.set(0.0f, 0.0f);
|
||||
_card_lr.set(0.0f, 0.0f);
|
||||
|
||||
_transform = LMatrix4::ident_mat();
|
||||
_coordinate_system = CS_default;
|
||||
|
||||
_ul3d.set(0.0f, 0.0f, 0.0f);
|
||||
_lr3d.set(0.0f, 0.0f, 0.0f);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -131,25 +85,9 @@ TextNode(const TextNode ©) :
|
||||
PandaNode(copy),
|
||||
TextEncoder(copy),
|
||||
TextProperties(copy),
|
||||
_card_texture(copy._card_texture),
|
||||
_frame_color(copy._frame_color),
|
||||
_card_color(copy._card_color),
|
||||
_flags(copy._flags),
|
||||
_max_rows(copy._max_rows),
|
||||
_usage_hint(GeomEnums::UH_static),
|
||||
_frame_width(copy._frame_width),
|
||||
_card_border_size(copy._card_border_size),
|
||||
_card_border_uv_portion(copy._card_border_uv_portion),
|
||||
_frame_ul(copy._frame_ul),
|
||||
_frame_lr(copy._frame_lr),
|
||||
_card_ul(copy._card_ul),
|
||||
_card_lr(copy._card_lr),
|
||||
_transform(copy._transform),
|
||||
_coordinate_system(copy._coordinate_system),
|
||||
_ul3d(copy._ul3d),
|
||||
_lr3d(copy._lr3d)
|
||||
_cycler(copy._cycler)
|
||||
{
|
||||
invalidate_with_measure();
|
||||
mark_internal_bounds_stale();
|
||||
}
|
||||
|
||||
/**
|
||||
@ -169,6 +107,82 @@ TextNode::
|
||||
~TextNode() {
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the actual dimensions of the frame around the text. If the frame
|
||||
* was set via set_frame_as_margin(), the result returned by this function
|
||||
* reflects the size of the current text; if the frame was set via
|
||||
* set_frame_actual(), this returns the values actually set.
|
||||
*
|
||||
* If the text has no frame at all, this returns the dimensions of the text
|
||||
* itself, as if the frame were set with a margin of 0, 0, 0, 0.
|
||||
*/
|
||||
LVecBase4 TextNode::
|
||||
get_frame_actual() const {
|
||||
CDLockedReader cdata(_cycler);
|
||||
if ((cdata->_flags & (F_has_frame | F_frame_as_margin)) == F_has_frame) {
|
||||
return LVecBase4(cdata->_frame_ul[0],
|
||||
cdata->_frame_lr[0],
|
||||
cdata->_frame_lr[1],
|
||||
cdata->_frame_ul[1]);
|
||||
}
|
||||
|
||||
if (do_needs_measure(cdata)) {
|
||||
CDWriter cdataw(((TextNode *)this)->_cycler, cdata, false);
|
||||
((TextNode *)this)->do_measure(cdataw);
|
||||
|
||||
LVecBase4 frame(cdataw->_text_ul[0], cdataw->_text_lr[0], cdataw->_text_lr[1], cdataw->_text_ul[1]);
|
||||
if (cdataw->_flags & F_has_frame) {
|
||||
frame += LVecBase4(-cdataw->_frame_ul[0], cdataw->_frame_lr[0], -cdataw->_frame_lr[1], cdataw->_frame_ul[1]);
|
||||
}
|
||||
return frame;
|
||||
}
|
||||
else {
|
||||
LVecBase4 frame(cdata->_text_ul[0], cdata->_text_lr[0], cdata->_text_lr[1], cdata->_text_ul[1]);
|
||||
if (cdata->_flags & F_has_frame) {
|
||||
frame += LVecBase4(-cdata->_frame_ul[0], cdata->_frame_lr[0], -cdata->_frame_lr[1], cdata->_frame_ul[1]);
|
||||
}
|
||||
return frame;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the actual dimensions of the card around the text. If the card was
|
||||
* set via set_card_as_margin(), the result returned by this function reflects
|
||||
* the size of the current text; if the card was set via set_card_actual(),
|
||||
* this returns the values actually set.
|
||||
*
|
||||
* If the text has no card at all, this returns the dimensions of the text
|
||||
* itself, as if the card were set with a margin of 0, 0, 0, 0.
|
||||
*/
|
||||
LVecBase4 TextNode::
|
||||
get_card_actual() const {
|
||||
CDLockedReader cdata(_cycler);
|
||||
if ((cdata->_flags & (F_has_card | F_card_as_margin)) == F_has_card) {
|
||||
return LVecBase4(cdata->_card_ul[0],
|
||||
cdata->_card_lr[0],
|
||||
cdata->_card_lr[1],
|
||||
cdata->_card_ul[1]);
|
||||
}
|
||||
|
||||
if (do_needs_measure(cdata)) {
|
||||
CDWriter cdataw(((TextNode *)this)->_cycler, cdata, false);
|
||||
((TextNode *)this)->do_measure(cdataw);
|
||||
|
||||
LVecBase4 card(cdataw->_text_ul[0], cdataw->_text_lr[0], cdataw->_text_lr[1], cdataw->_text_ul[1]);
|
||||
if (cdataw->_flags & F_has_card) {
|
||||
card += LVecBase4(-cdataw->_card_ul[0], cdataw->_card_lr[0], -cdataw->_card_lr[1], cdataw->_card_ul[1]);
|
||||
}
|
||||
return card;
|
||||
}
|
||||
else {
|
||||
LVecBase4 card(cdata->_text_ul[0], cdata->_text_lr[0], cdata->_text_lr[1], cdata->_text_ul[1]);
|
||||
if (cdata->_flags & F_has_card) {
|
||||
card += LVecBase4(-cdata->_card_ul[0], cdata->_card_lr[0], -cdata->_card_lr[1], cdata->_card_ul[1]);
|
||||
}
|
||||
return card;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the width of a single character of the font, or 0.0 if the
|
||||
* character is not known. This may be a wide character (greater than 255).
|
||||
@ -292,14 +306,15 @@ output(std::ostream &out) const {
|
||||
void TextNode::
|
||||
write(std::ostream &out, int indent_level) const {
|
||||
PandaNode::write(out, indent_level);
|
||||
MutexHolder holder(_lock);
|
||||
|
||||
CDReader cdata(_cycler);
|
||||
TextProperties::write(out, indent_level + 2);
|
||||
indent(out, indent_level + 2)
|
||||
<< "transform is: " << *TransformState::make_mat(_transform) << "\n";
|
||||
<< "transform is: " << *TransformState::make_mat(cdata->_transform) << "\n";
|
||||
indent(out, indent_level + 2)
|
||||
<< "in coordinate system " << _coordinate_system << "\n";
|
||||
<< "in coordinate system " << cdata->_coordinate_system << "\n";
|
||||
indent(out, indent_level + 2)
|
||||
<< "text is " << get_text() << "\n";
|
||||
<< "text is " << cdata->_text << "\n";
|
||||
}
|
||||
|
||||
/**
|
||||
@ -324,8 +339,12 @@ get_internal_geom() const {
|
||||
*/
|
||||
void TextNode::
|
||||
text_changed() {
|
||||
MutexHolder holder(_lock);
|
||||
invalidate_with_measure();
|
||||
// Copy the text to this class, since TextEncoder doesn't store it in a
|
||||
// pipeline-cycled manner.
|
||||
CDWriter cdata(_cycler);
|
||||
cdata->_text = TextEncoder::get_text();
|
||||
cdata->_wtext = TextEncoder::get_wtext();
|
||||
invalidate_with_measure(cdata);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -356,60 +375,65 @@ get_unsafe_to_apply_attribs() const {
|
||||
void TextNode::
|
||||
apply_attribs_to_vertices(const AccumulatedAttribs &attribs, int attrib_types,
|
||||
GeomTransformer &transformer) {
|
||||
MutexHolder holder(_lock);
|
||||
if ((attrib_types & SceneGraphReducer::TT_transform) != 0) {
|
||||
const LMatrix4 &mat = attribs._transform->get_mat();
|
||||
_transform *= mat;
|
||||
|
||||
if ((_flags & F_needs_measure) == 0) {
|
||||
// If we already have a measure, transform it too. We don't need to
|
||||
// invalidate the 2-d parts, since that's not affected by the transform
|
||||
// anyway.
|
||||
_ul3d = _ul3d * mat;
|
||||
_lr3d = _lr3d * mat;
|
||||
}
|
||||
}
|
||||
if ((attrib_types & SceneGraphReducer::TT_color) != 0) {
|
||||
if (attribs._color != nullptr) {
|
||||
const ColorAttrib *ca = DCAST(ColorAttrib, attribs._color);
|
||||
if (ca->get_color_type() == ColorAttrib::T_flat) {
|
||||
const LColor &c = ca->get_color();
|
||||
TextProperties::set_text_color(c);
|
||||
TextProperties::set_shadow_color(c);
|
||||
_frame_color = c;
|
||||
_card_color = c;
|
||||
invalidate_no_measure();
|
||||
Thread *current_thread = Thread::get_current_thread();
|
||||
OPEN_ITERATE_CURRENT_AND_UPSTREAM(_cycler, current_thread) {
|
||||
CDStageWriter cdata(_cycler, pipeline_stage, current_thread);
|
||||
|
||||
if ((attrib_types & SceneGraphReducer::TT_transform) != 0) {
|
||||
const LMatrix4 &mat = attribs._transform->get_mat();
|
||||
cdata->_transform *= mat;
|
||||
|
||||
if ((cdata->_flags & F_needs_measure) == 0) {
|
||||
// If we already have a measure, transform it too. We don't need to
|
||||
// invalidate the 2-d parts, since that's not affected by the transform
|
||||
// anyway.
|
||||
cdata->_ul3d = cdata->_ul3d * mat;
|
||||
cdata->_lr3d = cdata->_lr3d * mat;
|
||||
}
|
||||
}
|
||||
}
|
||||
if ((attrib_types & SceneGraphReducer::TT_color_scale) != 0) {
|
||||
if (attribs._color_scale != nullptr) {
|
||||
const ColorScaleAttrib *csa = DCAST(ColorScaleAttrib, attribs._color_scale);
|
||||
const LVecBase4 &s = csa->get_scale();
|
||||
if (s != LVecBase4(1.0f, 1.0f, 1.0f, 1.0f)) {
|
||||
LVecBase4 tc = get_text_color();
|
||||
tc.componentwise_mult(s);
|
||||
TextProperties::set_text_color(tc);
|
||||
|
||||
LVecBase4 sc = get_shadow_color();
|
||||
sc.componentwise_mult(s);
|
||||
TextProperties::set_shadow_color(sc);
|
||||
|
||||
_frame_color.componentwise_mult(s);
|
||||
_card_color.componentwise_mult(s);
|
||||
|
||||
invalidate_no_measure();
|
||||
if ((attrib_types & SceneGraphReducer::TT_color) != 0) {
|
||||
if (attribs._color != nullptr) {
|
||||
const ColorAttrib *ca = DCAST(ColorAttrib, attribs._color);
|
||||
if (ca->get_color_type() == ColorAttrib::T_flat) {
|
||||
const LColor &c = ca->get_color();
|
||||
TextProperties::set_text_color(c);
|
||||
TextProperties::set_shadow_color(c);
|
||||
cdata->_frame_color = c;
|
||||
cdata->_card_color = c;
|
||||
invalidate_no_measure(cdata);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if ((attrib_types & SceneGraphReducer::TT_color_scale) != 0) {
|
||||
if (attribs._color_scale != nullptr) {
|
||||
const ColorScaleAttrib *csa = DCAST(ColorScaleAttrib, attribs._color_scale);
|
||||
const LVecBase4 &s = csa->get_scale();
|
||||
if (s != LVecBase4(1.0f, 1.0f, 1.0f, 1.0f)) {
|
||||
LVecBase4 tc = get_text_color();
|
||||
tc.componentwise_mult(s);
|
||||
TextProperties::set_text_color(tc);
|
||||
|
||||
// Now propagate the attributes down to our already-generated geometry, if
|
||||
// we have any.
|
||||
if ((_flags & F_needs_rebuild) == 0 &&
|
||||
_internal_geom != nullptr) {
|
||||
SceneGraphReducer gr;
|
||||
gr.apply_attribs(_internal_geom, attribs, attrib_types, transformer);
|
||||
LVecBase4 sc = get_shadow_color();
|
||||
sc.componentwise_mult(s);
|
||||
TextProperties::set_shadow_color(sc);
|
||||
|
||||
cdata->_frame_color.componentwise_mult(s);
|
||||
cdata->_card_color.componentwise_mult(s);
|
||||
|
||||
invalidate_no_measure(cdata);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Now propagate the attributes down to our already-generated geometry, if
|
||||
// we have any.
|
||||
if ((cdata->_flags & F_needs_rebuild) == 0 && cdata->_internal_geom != nullptr) {
|
||||
SceneGraphReducer gr;
|
||||
gr.apply_attribs(cdata->_internal_geom, attribs, attrib_types, transformer);
|
||||
}
|
||||
}
|
||||
CLOSE_ITERATE_CURRENT_AND_UPSTREAM(_cycler);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -501,17 +525,27 @@ compute_internal_bounds(CPT(BoundingVolume) &internal_bounds,
|
||||
// actually generating the text, if we have at least measured it.
|
||||
LPoint3 vertices[8];
|
||||
{
|
||||
MutexHolder holder(_lock);
|
||||
check_measure();
|
||||
LPoint3 ul3d, lr3d;
|
||||
CDLockedReader cdata(_cycler);
|
||||
if (do_needs_measure(cdata)) {
|
||||
CDWriter cdataw(((TextNode *)this)->_cycler, cdata, false);
|
||||
((TextNode *)this)->do_measure(cdataw);
|
||||
|
||||
vertices[0].set(_ul3d[0], _ul3d[1], _ul3d[2]);
|
||||
vertices[1].set(_ul3d[0], _ul3d[1], _lr3d[2]);
|
||||
vertices[2].set(_ul3d[0], _lr3d[1], _ul3d[2]);
|
||||
vertices[3].set(_ul3d[0], _lr3d[1], _lr3d[2]);
|
||||
vertices[4].set(_lr3d[0], _ul3d[1], _ul3d[2]);
|
||||
vertices[5].set(_lr3d[0], _ul3d[1], _lr3d[2]);
|
||||
vertices[6].set(_lr3d[0], _lr3d[1], _ul3d[2]);
|
||||
vertices[7].set(_lr3d[0], _lr3d[1], _lr3d[2]);
|
||||
ul3d = cdataw->_ul3d;
|
||||
lr3d = cdataw->_lr3d;
|
||||
} else {
|
||||
ul3d = cdata->_ul3d;
|
||||
lr3d = cdata->_lr3d;
|
||||
}
|
||||
|
||||
vertices[0].set(ul3d[0], ul3d[1], ul3d[2]);
|
||||
vertices[1].set(ul3d[0], ul3d[1], lr3d[2]);
|
||||
vertices[2].set(ul3d[0], lr3d[1], ul3d[2]);
|
||||
vertices[3].set(ul3d[0], lr3d[1], lr3d[2]);
|
||||
vertices[4].set(lr3d[0], ul3d[1], ul3d[2]);
|
||||
vertices[5].set(lr3d[0], ul3d[1], lr3d[2]);
|
||||
vertices[6].set(lr3d[0], lr3d[1], ul3d[2]);
|
||||
vertices[7].set(lr3d[0], lr3d[1], lr3d[2]);
|
||||
}
|
||||
|
||||
gbv->around(vertices, vertices + 8);
|
||||
@ -542,21 +576,19 @@ r_prepare_scene(GraphicsStateGuardianBase *gsg, const RenderState *node_state,
|
||||
* text instead.
|
||||
*/
|
||||
void TextNode::
|
||||
do_rebuild() {
|
||||
nassertv(_lock.debug_is_locked());
|
||||
_flags &= ~(F_needs_rebuild | F_needs_measure);
|
||||
_internal_geom = do_generate();
|
||||
do_rebuild(CData *cdata) {
|
||||
cdata->_flags &= ~(F_needs_rebuild | F_needs_measure);
|
||||
cdata->_internal_geom = do_generate(cdata);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Can be called in lieu of do_rebuild() to measure the text and set up the
|
||||
* bounding boxes properly without actually assembling it.
|
||||
*/
|
||||
void TextNode::
|
||||
do_measure() {
|
||||
do_measure(CData *cdata) {
|
||||
// We no longer make this a special case.
|
||||
do_rebuild();
|
||||
do_rebuild(cdata);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -565,14 +597,12 @@ do_measure() {
|
||||
* represent it.
|
||||
*/
|
||||
PT(PandaNode) TextNode::
|
||||
do_generate() {
|
||||
nassertr(_lock.debug_is_locked(), nullptr);
|
||||
|
||||
do_generate(CData *cdata) {
|
||||
PStatTimer timer(_text_generate_pcollector);
|
||||
if (text_cat.is_debug()) {
|
||||
text_cat.debug()
|
||||
<< "Rebuilding " << get_type() << " " << get_name()
|
||||
<< " with '" << get_text() << "'\n";
|
||||
<< " with '" << cdata->_text << "'\n";
|
||||
}
|
||||
|
||||
// The strategy here will be to assemble together a bunch of letters,
|
||||
@ -584,18 +614,18 @@ do_generate() {
|
||||
// row, that moves the row into the right place horizontally and vertically,
|
||||
// and for each row, there is another node for each character.
|
||||
|
||||
_ul3d.set(0.0f, 0.0f, 0.0f);
|
||||
_lr3d.set(0.0f, 0.0f, 0.0f);
|
||||
cdata->_ul3d.set(0.0f, 0.0f, 0.0f);
|
||||
cdata->_lr3d.set(0.0f, 0.0f, 0.0f);
|
||||
|
||||
// Now build a new sub-tree for all the text components.
|
||||
string name = get_text();
|
||||
string name = cdata->_text;
|
||||
size_t newline = name.find('\n');
|
||||
if (newline != string::npos) {
|
||||
name = name.substr(0, newline);
|
||||
}
|
||||
PT(PandaNode) root = new PandaNode(name);
|
||||
|
||||
if (!has_text()) {
|
||||
if (cdata->_wtext.empty()) {
|
||||
return root;
|
||||
}
|
||||
|
||||
@ -607,34 +637,32 @@ do_generate() {
|
||||
// Compute the overall text transform matrix. We build the text in a Z-up
|
||||
// coordinate system and then convert it to whatever the user asked for.
|
||||
LMatrix4 mat =
|
||||
LMatrix4::convert_mat(CS_zup_right, _coordinate_system) *
|
||||
_transform;
|
||||
LMatrix4::convert_mat(CS_zup_right, cdata->_coordinate_system) *
|
||||
cdata->_transform;
|
||||
|
||||
CPT(TransformState) transform = TransformState::make_mat(mat);
|
||||
root->set_transform(transform);
|
||||
|
||||
std::wstring wtext = get_wtext();
|
||||
|
||||
// Assemble the text.
|
||||
TextAssembler assembler(this);
|
||||
assembler.set_properties(*this);
|
||||
assembler.set_max_rows(_max_rows);
|
||||
assembler.set_usage_hint(_usage_hint);
|
||||
assembler.set_dynamic_merge((_flatten_flags & FF_dynamic_merge) != 0);
|
||||
bool all_set = assembler.set_wtext(wtext);
|
||||
assembler.set_max_rows(cdata->_max_rows);
|
||||
assembler.set_usage_hint(cdata->_usage_hint);
|
||||
assembler.set_dynamic_merge((cdata->_flatten_flags & FF_dynamic_merge) != 0);
|
||||
bool all_set = assembler.set_wtext(cdata->_wtext);
|
||||
if (all_set) {
|
||||
// No overflow.
|
||||
_flags &= ~F_has_overflow;
|
||||
cdata->_flags &= ~F_has_overflow;
|
||||
} else {
|
||||
// Overflow.
|
||||
_flags |= F_has_overflow;
|
||||
cdata->_flags |= F_has_overflow;
|
||||
}
|
||||
|
||||
PT(PandaNode) text_root = assembler.assemble_text();
|
||||
_text_ul = assembler.get_ul();
|
||||
_text_lr = assembler.get_lr();
|
||||
_num_rows = assembler.get_num_rows();
|
||||
_wordwrapped_wtext = assembler.get_wordwrapped_wtext();
|
||||
cdata->_text_ul = assembler.get_ul();
|
||||
cdata->_text_lr = assembler.get_lr();
|
||||
cdata->_num_rows = assembler.get_num_rows();
|
||||
cdata->_wordwrapped_wtext = assembler.get_wordwrapped_wtext();
|
||||
|
||||
// Parent the text in.
|
||||
PT(PandaNode) text = new PandaNode("text");
|
||||
@ -645,43 +673,45 @@ do_generate() {
|
||||
// the user.
|
||||
const LVector2 &ul = assembler.get_ul();
|
||||
const LVector2 &lr = assembler.get_lr();
|
||||
_ul3d.set(ul[0], 0.0f, ul[1]);
|
||||
_lr3d.set(lr[0], 0.0f, lr[1]);
|
||||
cdata->_ul3d.set(ul[0], 0.0f, ul[1]);
|
||||
cdata->_lr3d.set(lr[0], 0.0f, lr[1]);
|
||||
|
||||
_ul3d = _ul3d * _transform;
|
||||
_lr3d = _lr3d * _transform;
|
||||
cdata->_ul3d = cdata->_ul3d * cdata->_transform;
|
||||
cdata->_lr3d = cdata->_lr3d * cdata->_transform;
|
||||
|
||||
// Incidentally, that means we don't need to measure the text now.
|
||||
_flags &= ~F_needs_measure;
|
||||
cdata->_flags &= ~F_needs_measure;
|
||||
|
||||
// Now flatten our hierarchy to get rid of the transforms we put in,
|
||||
// applying them to the vertices.
|
||||
|
||||
NodePath root_np(root);
|
||||
if (_flatten_flags & FF_strong) {
|
||||
if (cdata->_flatten_flags & FF_strong) {
|
||||
root_np.flatten_strong();
|
||||
} else if (_flatten_flags & FF_medium) {
|
||||
}
|
||||
else if (cdata->_flatten_flags & FF_medium) {
|
||||
root_np.flatten_medium();
|
||||
} else if (_flatten_flags & FF_light) {
|
||||
}
|
||||
else if (cdata->_flatten_flags & FF_light) {
|
||||
root_np.flatten_light();
|
||||
}
|
||||
|
||||
// Now deal with the decorations.
|
||||
|
||||
if (_flags & F_has_card) {
|
||||
if (cdata->_flags & F_has_card) {
|
||||
PT(PandaNode) card_root;
|
||||
if (_flags & F_has_card_border) {
|
||||
card_root = make_card_with_border();
|
||||
if (cdata->_flags & F_has_card_border) {
|
||||
card_root = do_make_card_with_border(cdata);
|
||||
} else {
|
||||
card_root = make_card();
|
||||
card_root = do_make_card(cdata);
|
||||
}
|
||||
card_root->set_transform(transform);
|
||||
card_root->set_attrib(ColorAttrib::make_flat(_card_color));
|
||||
if (_card_color[3] != 1.0f) {
|
||||
card_root->set_attrib(ColorAttrib::make_flat(cdata->_card_color));
|
||||
if (cdata->_card_color[3] != 1.0f) {
|
||||
card_root->set_attrib(TransparencyAttrib::make(TransparencyAttrib::M_alpha));
|
||||
}
|
||||
if (_flags & F_has_card_texture) {
|
||||
card_root->set_attrib(TextureAttrib::make(_card_texture));
|
||||
if (cdata->_flags & F_has_card_texture) {
|
||||
card_root->set_attrib(TextureAttrib::make(cdata->_card_texture));
|
||||
}
|
||||
|
||||
if (has_bin()) {
|
||||
@ -697,17 +727,17 @@ do_generate() {
|
||||
card_root->add_child(root);
|
||||
root = card_root;
|
||||
|
||||
if (_flags & F_card_decal) {
|
||||
if (cdata->_flags & F_card_decal) {
|
||||
card_root->set_effect(DecalEffect::make());
|
||||
}
|
||||
}
|
||||
|
||||
if (_flags & F_has_frame) {
|
||||
PT(PandaNode) frame_root = make_frame();
|
||||
if (cdata->_flags & F_has_frame) {
|
||||
PT(PandaNode) frame_root = do_make_frame(cdata);
|
||||
frame_root->set_transform(transform);
|
||||
root->add_child(frame_root, get_draw_order() + 1);
|
||||
frame_root->set_attrib(ColorAttrib::make_flat(_frame_color));
|
||||
if (_frame_color[3] != 1.0f) {
|
||||
frame_root->set_attrib(ColorAttrib::make_flat(cdata->_frame_color));
|
||||
if (cdata->_frame_color[3] != 1.0f) {
|
||||
frame_root->set_attrib(TransparencyAttrib::make(TransparencyAttrib::M_alpha));
|
||||
}
|
||||
|
||||
@ -728,38 +758,46 @@ do_generate() {
|
||||
*/
|
||||
PT(PandaNode) TextNode::
|
||||
do_get_internal_geom() const {
|
||||
MutexHolder holder(_lock);
|
||||
check_rebuild();
|
||||
return _internal_geom;
|
||||
CDLockedReader cdata(_cycler);
|
||||
if ((cdata->_flags & F_needs_rebuild) != 0) {
|
||||
// Propagate the generated text upstream if the upstream stages have no
|
||||
// changes to the text.
|
||||
CDWriter cdataw(((TextNode *)this)->_cycler, cdata, false);
|
||||
((TextNode *)this)->do_rebuild(cdataw);
|
||||
|
||||
return cdataw->_internal_geom;
|
||||
}
|
||||
else {
|
||||
return cdata->_internal_geom;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates a frame around the text.
|
||||
*/
|
||||
PT(PandaNode) TextNode::
|
||||
make_frame() {
|
||||
nassertr(_lock.debug_is_locked(), nullptr);
|
||||
nassertr((_flags & F_needs_measure) == 0, nullptr);
|
||||
do_make_frame(const CData *cdata) {
|
||||
nassertr((cdata->_flags & F_needs_measure) == 0, nullptr);
|
||||
|
||||
PT(GeomNode) frame_node = new GeomNode("frame");
|
||||
|
||||
PN_stdfloat left = _frame_ul[0];
|
||||
PN_stdfloat right = _frame_lr[0];
|
||||
PN_stdfloat bottom = _frame_lr[1];
|
||||
PN_stdfloat top = _frame_ul[1];
|
||||
PN_stdfloat left = cdata->_frame_ul[0];
|
||||
PN_stdfloat right = cdata->_frame_lr[0];
|
||||
PN_stdfloat bottom = cdata->_frame_lr[1];
|
||||
PN_stdfloat top = cdata->_frame_ul[1];
|
||||
|
||||
if (_flags & F_frame_as_margin) {
|
||||
left = _text_ul[0] - left;
|
||||
right = _text_lr[0] + right;
|
||||
bottom = _text_lr[1] - bottom;
|
||||
top = _text_ul[1] + top;
|
||||
if (cdata->_flags & F_frame_as_margin) {
|
||||
left = cdata->_text_ul[0] - left;
|
||||
right = cdata->_text_lr[0] + right;
|
||||
bottom = cdata->_text_lr[1] - bottom;
|
||||
top = cdata->_text_ul[1] + top;
|
||||
}
|
||||
|
||||
CPT(RenderAttrib) thick = RenderModeAttrib::make(RenderModeAttrib::M_unchanged, _frame_width);
|
||||
CPT(RenderAttrib) thick = RenderModeAttrib::make(RenderModeAttrib::M_unchanged, cdata->_frame_width);
|
||||
CPT(RenderState) state = RenderState::make(thick);
|
||||
|
||||
PT(GeomVertexData) vdata = new GeomVertexData
|
||||
("text", GeomVertexFormat::get_v3(), _usage_hint);
|
||||
("text", GeomVertexFormat::get_v3(), cdata->_usage_hint);
|
||||
vdata->unclean_set_num_rows(4);
|
||||
GeomVertexWriter vertex(vdata, InternalName::get_vertex());
|
||||
|
||||
@ -768,7 +806,7 @@ make_frame() {
|
||||
vertex.set_data3(right, 0.0f, bottom);
|
||||
vertex.set_data3(right, 0.0f, top);
|
||||
|
||||
PT(GeomLinestrips) frame = new GeomLinestrips(_usage_hint);
|
||||
PT(GeomLinestrips) frame = new GeomLinestrips(cdata->_usage_hint);
|
||||
frame->add_consecutive_vertices(0, 4);
|
||||
frame->add_vertex(0);
|
||||
frame->close_primitive();
|
||||
@ -777,8 +815,8 @@ make_frame() {
|
||||
geom->add_primitive(frame);
|
||||
frame_node->add_geom(geom, state);
|
||||
|
||||
if (_flags & F_frame_corners) {
|
||||
PT(GeomPoints) corners = new GeomPoints(_usage_hint);
|
||||
if (cdata->_flags & F_frame_corners) {
|
||||
PT(GeomPoints) corners = new GeomPoints(cdata->_usage_hint);
|
||||
corners->add_consecutive_vertices(0, 4);
|
||||
PT(Geom) geom2 = new Geom(vdata);
|
||||
geom2->add_primitive(corners);
|
||||
@ -792,26 +830,25 @@ make_frame() {
|
||||
* Creates a card behind the text.
|
||||
*/
|
||||
PT(PandaNode) TextNode::
|
||||
make_card() {
|
||||
nassertr(_lock.debug_is_locked(), nullptr);
|
||||
nassertr((_flags & F_needs_measure) == 0, nullptr);
|
||||
do_make_card(const CData *cdata) {
|
||||
nassertr((cdata->_flags & F_needs_measure) == 0, nullptr);
|
||||
|
||||
PT(GeomNode) card_node = new GeomNode("card");
|
||||
|
||||
PN_stdfloat left = _card_ul[0];
|
||||
PN_stdfloat right = _card_lr[0];
|
||||
PN_stdfloat bottom = _card_lr[1];
|
||||
PN_stdfloat top = _card_ul[1];
|
||||
PN_stdfloat left = cdata->_card_ul[0];
|
||||
PN_stdfloat right = cdata->_card_lr[0];
|
||||
PN_stdfloat bottom = cdata->_card_lr[1];
|
||||
PN_stdfloat top = cdata->_card_ul[1];
|
||||
|
||||
if (_flags & F_card_as_margin) {
|
||||
left = _text_ul[0] - left;
|
||||
right = _text_lr[0] + right;
|
||||
bottom = _text_lr[1] - bottom;
|
||||
top = _text_ul[1] + top;
|
||||
if (cdata->_flags & F_card_as_margin) {
|
||||
left = cdata->_text_ul[0] - left;
|
||||
right = cdata->_text_lr[0] + right;
|
||||
bottom = cdata->_text_lr[1] - bottom;
|
||||
top = cdata->_text_ul[1] + top;
|
||||
}
|
||||
|
||||
PT(GeomVertexData) vdata = new GeomVertexData
|
||||
("text", GeomVertexFormat::get_v3t2(), _usage_hint);
|
||||
("text", GeomVertexFormat::get_v3t2(), cdata->_usage_hint);
|
||||
vdata->unclean_set_num_rows(4);
|
||||
GeomVertexWriter vertex(vdata, InternalName::get_vertex());
|
||||
GeomVertexWriter texcoord(vdata, InternalName::get_texcoord());
|
||||
@ -826,7 +863,7 @@ make_card() {
|
||||
texcoord.set_data2(1.0f, 1.0f);
|
||||
texcoord.set_data2(1.0f, 0.0f);
|
||||
|
||||
PT(GeomTristrips) card = new GeomTristrips(_usage_hint);
|
||||
PT(GeomTristrips) card = new GeomTristrips(cdata->_usage_hint);
|
||||
card->add_consecutive_vertices(0, 4);
|
||||
card->close_primitive();
|
||||
|
||||
@ -844,83 +881,88 @@ make_card() {
|
||||
* what have you.
|
||||
*/
|
||||
PT(PandaNode) TextNode::
|
||||
make_card_with_border() {
|
||||
nassertr(_lock.debug_is_locked(), nullptr);
|
||||
nassertr((_flags & F_needs_measure) == 0, nullptr);
|
||||
do_make_card_with_border(const CData *cdata) {
|
||||
nassertr((cdata->_flags & F_needs_measure) == 0, nullptr);
|
||||
|
||||
PT(GeomNode) card_node = new GeomNode("card");
|
||||
|
||||
PN_stdfloat left = _card_ul[0];
|
||||
PN_stdfloat right = _card_lr[0];
|
||||
PN_stdfloat bottom = _card_lr[1];
|
||||
PN_stdfloat top = _card_ul[1];
|
||||
PN_stdfloat left = cdata->_card_ul[0];
|
||||
PN_stdfloat right = cdata->_card_lr[0];
|
||||
PN_stdfloat bottom = cdata->_card_lr[1];
|
||||
PN_stdfloat top = cdata->_card_ul[1];
|
||||
|
||||
if (_flags & F_card_as_margin) {
|
||||
left = _text_ul[0] - left;
|
||||
right = _text_lr[0] + right;
|
||||
bottom = _text_lr[1] - bottom;
|
||||
top = _text_ul[1] + top;
|
||||
if (cdata->_flags & F_card_as_margin) {
|
||||
left = cdata->_text_ul[0] - left;
|
||||
right = cdata->_text_lr[0] + right;
|
||||
bottom = cdata->_text_lr[1] - bottom;
|
||||
top = cdata->_text_ul[1] + top;
|
||||
}
|
||||
|
||||
/*
|
||||
* we now create three tri-strips instead of one with vertices arranged as
|
||||
* follows: 1 3 5 7 - one 2 4 6 8 \ two 9 11
|
||||
* 13 15 \ 10 12 14 16 - three
|
||||
*/
|
||||
PN_stdfloat border_size = cdata->_card_border_size;
|
||||
PN_stdfloat border_uv_portion = cdata->_card_border_uv_portion;
|
||||
|
||||
// we now create three tri-strips instead of one
|
||||
// with vertices arranged as follows:
|
||||
//
|
||||
// 1 3 5 7 - one
|
||||
// 2 4 6 8 / \ two
|
||||
// 9 11 13 15 \ /
|
||||
// 10 12 14 16 - three
|
||||
//
|
||||
|
||||
PT(GeomVertexData) vdata = new GeomVertexData
|
||||
("text", GeomVertexFormat::get_v3t2(), _usage_hint);
|
||||
("text", GeomVertexFormat::get_v3t2(), cdata->_usage_hint);
|
||||
vdata->unclean_set_num_rows(16);
|
||||
GeomVertexWriter vertex(vdata, InternalName::get_vertex());
|
||||
GeomVertexWriter texcoord(vdata, InternalName::get_texcoord());
|
||||
|
||||
// verts 1,2,3,4
|
||||
vertex.set_data3(left, 0.02, top);
|
||||
vertex.set_data3(left, 0.02, top - _card_border_size);
|
||||
vertex.set_data3(left + _card_border_size, 0.02, top);
|
||||
vertex.set_data3(left + _card_border_size, 0.02,
|
||||
top - _card_border_size);
|
||||
vertex.set_data3(left, 0.02, top - border_size);
|
||||
vertex.set_data3(left + border_size, 0.02, top);
|
||||
vertex.set_data3(left + border_size, 0.02,
|
||||
top - border_size);
|
||||
// verts 5,6,7,8
|
||||
vertex.set_data3(right - _card_border_size, 0.02, top);
|
||||
vertex.set_data3(right - _card_border_size, 0.02,
|
||||
top - _card_border_size);
|
||||
vertex.set_data3(right - border_size, 0.02, top);
|
||||
vertex.set_data3(right - border_size, 0.02,
|
||||
top - border_size);
|
||||
vertex.set_data3(right, 0.02, top);
|
||||
vertex.set_data3(right, 0.02, top - _card_border_size);
|
||||
vertex.set_data3(right, 0.02, top - border_size);
|
||||
// verts 9,10,11,12
|
||||
vertex.set_data3(left, 0.02, bottom + _card_border_size);
|
||||
vertex.set_data3(left, 0.02, bottom + border_size);
|
||||
vertex.set_data3(left, 0.02, bottom);
|
||||
vertex.set_data3(left + _card_border_size, 0.02,
|
||||
bottom + _card_border_size);
|
||||
vertex.set_data3(left + _card_border_size, 0.02, bottom);
|
||||
vertex.set_data3(left + border_size, 0.02,
|
||||
bottom + border_size);
|
||||
vertex.set_data3(left + border_size, 0.02, bottom);
|
||||
// verts 13,14,15,16
|
||||
vertex.set_data3(right - _card_border_size, 0.02,
|
||||
bottom + _card_border_size);
|
||||
vertex.set_data3(right - _card_border_size, 0.02, bottom);
|
||||
vertex.set_data3(right, 0.02, bottom + _card_border_size);
|
||||
vertex.set_data3(right - border_size, 0.02,
|
||||
bottom + border_size);
|
||||
vertex.set_data3(right - border_size, 0.02, bottom);
|
||||
vertex.set_data3(right, 0.02, bottom + border_size);
|
||||
vertex.set_data3(right, 0.02, bottom);
|
||||
|
||||
texcoord.set_data2(0.0f, 1.0f); //1
|
||||
texcoord.set_data2(0.0f, 1.0f - _card_border_uv_portion); //2
|
||||
texcoord.set_data2(0.0f + _card_border_uv_portion, 1.0f); //3
|
||||
texcoord.set_data2(0.0f + _card_border_uv_portion,
|
||||
1.0f - _card_border_uv_portion); //4
|
||||
texcoord.set_data2(1.0f -_card_border_uv_portion, 1.0f); //5
|
||||
texcoord.set_data2(1.0f -_card_border_uv_portion,
|
||||
1.0f - _card_border_uv_portion); //6
|
||||
texcoord.set_data2(0.0f, 1.0f - border_uv_portion); //2
|
||||
texcoord.set_data2(0.0f + border_uv_portion, 1.0f); //3
|
||||
texcoord.set_data2(0.0f + border_uv_portion,
|
||||
1.0f - border_uv_portion); //4
|
||||
texcoord.set_data2(1.0f -border_uv_portion, 1.0f); //5
|
||||
texcoord.set_data2(1.0f -border_uv_portion,
|
||||
1.0f - border_uv_portion); //6
|
||||
texcoord.set_data2(1.0f, 1.0f); //7
|
||||
texcoord.set_data2(1.0f, 1.0f - _card_border_uv_portion); //8
|
||||
texcoord.set_data2(1.0f, 1.0f - border_uv_portion); //8
|
||||
|
||||
texcoord.set_data2(0.0f, _card_border_uv_portion); //9
|
||||
texcoord.set_data2(0.0f, border_uv_portion); //9
|
||||
texcoord.set_data2(0.0f, 0.0f); //10
|
||||
texcoord.set_data2(_card_border_uv_portion, _card_border_uv_portion); //11
|
||||
texcoord.set_data2(_card_border_uv_portion, 0.0f); //12
|
||||
texcoord.set_data2(border_uv_portion, border_uv_portion); //11
|
||||
texcoord.set_data2(border_uv_portion, 0.0f); //12
|
||||
|
||||
texcoord.set_data2(1.0f - _card_border_uv_portion, _card_border_uv_portion);//13
|
||||
texcoord.set_data2(1.0f - _card_border_uv_portion, 0.0f);//14
|
||||
texcoord.set_data2(1.0f, _card_border_uv_portion);//15
|
||||
texcoord.set_data2(1.0f - border_uv_portion, border_uv_portion);//13
|
||||
texcoord.set_data2(1.0f - border_uv_portion, 0.0f);//14
|
||||
texcoord.set_data2(1.0f, border_uv_portion);//15
|
||||
texcoord.set_data2(1.0f, 0.0f);//16
|
||||
|
||||
PT(GeomTristrips) card = new GeomTristrips(_usage_hint);
|
||||
PT(GeomTristrips) card = new GeomTristrips(cdata->_usage_hint);
|
||||
card->reserve_num_vertices(24);
|
||||
|
||||
// tristrip #1
|
||||
@ -970,3 +1012,69 @@ count_geoms(PandaNode *node) {
|
||||
|
||||
return num_geoms;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
TextNode::CData::
|
||||
CData() {
|
||||
_flags = 0;
|
||||
_max_rows = 0;
|
||||
_usage_hint = GeomEnums::UH_static;
|
||||
_flatten_flags = 0;
|
||||
if (text_flatten) {
|
||||
_flatten_flags |= FF_strong;
|
||||
}
|
||||
if (text_dynamic_merge) {
|
||||
_flatten_flags |= FF_dynamic_merge;
|
||||
}
|
||||
|
||||
_frame_color.set(1.0f, 1.0f, 1.0f, 1.0f);
|
||||
_card_color.set(1.0f, 1.0f, 1.0f, 1.0f);
|
||||
|
||||
_frame_width = 1.0f;
|
||||
|
||||
_frame_ul.set(0.0f, 0.0f);
|
||||
_frame_lr.set(0.0f, 0.0f);
|
||||
_card_ul.set(0.0f, 0.0f);
|
||||
_card_lr.set(0.0f, 0.0f);
|
||||
|
||||
_transform = LMatrix4::ident_mat();
|
||||
_coordinate_system = CS_default;
|
||||
|
||||
_ul3d.set(0.0f, 0.0f, 0.0f);
|
||||
_lr3d.set(0.0f, 0.0f, 0.0f);
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
TextNode::CData::
|
||||
CData(const CData ©) :
|
||||
_card_texture(copy._card_texture),
|
||||
_frame_color(copy._frame_color),
|
||||
_card_color(copy._card_color),
|
||||
_flags(copy._flags | F_needs_rebuild | F_needs_measure),
|
||||
_max_rows(copy._max_rows),
|
||||
_usage_hint(GeomEnums::UH_static),
|
||||
_frame_width(copy._frame_width),
|
||||
_card_border_size(copy._card_border_size),
|
||||
_card_border_uv_portion(copy._card_border_uv_portion),
|
||||
_frame_ul(copy._frame_ul),
|
||||
_frame_lr(copy._frame_lr),
|
||||
_card_ul(copy._card_ul),
|
||||
_card_lr(copy._card_lr),
|
||||
_transform(copy._transform),
|
||||
_coordinate_system(copy._coordinate_system),
|
||||
_ul3d(copy._ul3d),
|
||||
_lr3d(copy._lr3d)
|
||||
{
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
CycleData *TextNode::CData::
|
||||
make_copy() const {
|
||||
return new CData(*this);
|
||||
}
|
||||
|
@ -24,8 +24,12 @@
|
||||
#include "pandaNode.h"
|
||||
#include "luse.h"
|
||||
#include "geom.h"
|
||||
#include "pmutex.h"
|
||||
#include "mutexHolder.h"
|
||||
#include "cycleData.h"
|
||||
#include "cycleDataLockedReader.h"
|
||||
#include "cycleDataReader.h"
|
||||
#include "cycleDataWriter.h"
|
||||
#include "cycleDataStageReader.h"
|
||||
#include "cycleDataStageWriter.h"
|
||||
|
||||
/**
|
||||
* The primary interface to this module. This class does basic text assembly;
|
||||
@ -99,7 +103,7 @@ PUBLISHED:
|
||||
INLINE bool has_frame() const;
|
||||
INLINE bool is_frame_as_margin() const;
|
||||
INLINE LVecBase4 get_frame_as_set() const;
|
||||
INLINE LVecBase4 get_frame_actual() const;
|
||||
LVecBase4 get_frame_actual() const;
|
||||
|
||||
INLINE void set_frame_line_width(PN_stdfloat line_width);
|
||||
INLINE PN_stdfloat get_frame_line_width() const;
|
||||
@ -116,7 +120,7 @@ PUBLISHED:
|
||||
INLINE bool get_card_decal() const;
|
||||
INLINE bool is_card_as_margin() const;
|
||||
INLINE LVecBase4 get_card_as_set() const;
|
||||
INLINE LVecBase4 get_card_actual() const;
|
||||
LVecBase4 get_card_actual() const;
|
||||
INLINE LVecBase4 get_card_transformed() const;
|
||||
|
||||
INLINE void set_transform(const LMatrix4 &transform);
|
||||
@ -295,30 +299,25 @@ public:
|
||||
Thread *current_thread);
|
||||
|
||||
private:
|
||||
INLINE void invalidate_no_measure();
|
||||
INLINE void invalidate_with_measure();
|
||||
INLINE void check_rebuild() const;
|
||||
INLINE void check_measure() const;
|
||||
class CData;
|
||||
|
||||
void do_rebuild();
|
||||
void do_measure();
|
||||
INLINE void invalidate_no_measure(CData *cdata);
|
||||
INLINE void invalidate_with_measure(CData *cdata);
|
||||
INLINE bool do_needs_rebuild(const CData *cdata) const;
|
||||
INLINE bool do_needs_measure(const CData *cdata) const;
|
||||
|
||||
PT(PandaNode) do_generate();
|
||||
void do_rebuild(CData *cdata);
|
||||
void do_measure(CData *cdata);
|
||||
|
||||
PT(PandaNode) do_generate(CData *cdata);
|
||||
PT(PandaNode) do_get_internal_geom() const;
|
||||
|
||||
PT(PandaNode) make_frame();
|
||||
PT(PandaNode) make_card();
|
||||
PT(PandaNode) make_card_with_border();
|
||||
PT(PandaNode) do_make_frame(const CData *cdata);
|
||||
PT(PandaNode) do_make_card(const CData *cdata);
|
||||
PT(PandaNode) do_make_card_with_border(const CData *cdata);
|
||||
|
||||
static int count_geoms(PandaNode *node);
|
||||
|
||||
Mutex _lock;
|
||||
PT(PandaNode) _internal_geom;
|
||||
|
||||
PT(Texture) _card_texture;
|
||||
LColor _frame_color;
|
||||
LColor _card_color;
|
||||
|
||||
enum Flags {
|
||||
F_has_frame = 0x0001,
|
||||
F_frame_as_margin = 0x0002,
|
||||
@ -334,26 +333,54 @@ private:
|
||||
F_card_decal = 0x0800,
|
||||
};
|
||||
|
||||
int _flags;
|
||||
int _max_rows;
|
||||
GeomEnums::UsageHint _usage_hint;
|
||||
int _flatten_flags;
|
||||
PN_stdfloat _frame_width;
|
||||
PN_stdfloat _card_border_size;
|
||||
PN_stdfloat _card_border_uv_portion;
|
||||
// This is the data that must be cycled between pipeline stages.
|
||||
class EXPCL_PANDA_TEXT CData : public CycleData {
|
||||
public:
|
||||
CData();
|
||||
CData(const CData ©);
|
||||
virtual CycleData *make_copy() const;
|
||||
virtual TypeHandle get_parent_type() const {
|
||||
return TextNode::get_class_type();
|
||||
}
|
||||
|
||||
LVector2 _frame_ul, _frame_lr;
|
||||
LVector2 _card_ul, _card_lr;
|
||||
// We copy these here because they aren't pipeline-cycled on TextEncoder.
|
||||
std::string _text;
|
||||
std::wstring _wtext;
|
||||
|
||||
LMatrix4 _transform;
|
||||
CoordinateSystem _coordinate_system;
|
||||
PT(PandaNode) _internal_geom;
|
||||
|
||||
LPoint3 _ul3d, _lr3d;
|
||||
PT(Texture) _card_texture;
|
||||
LColor _frame_color;
|
||||
LColor _card_color;
|
||||
|
||||
// Returned from TextAssembler:
|
||||
LVector2 _text_ul, _text_lr;
|
||||
int _num_rows;
|
||||
std::wstring _wordwrapped_wtext;
|
||||
int _flags;
|
||||
int _max_rows;
|
||||
GeomEnums::UsageHint _usage_hint;
|
||||
int _flatten_flags;
|
||||
PN_stdfloat _frame_width;
|
||||
PN_stdfloat _card_border_size;
|
||||
PN_stdfloat _card_border_uv_portion;
|
||||
|
||||
LVector2 _frame_ul, _frame_lr;
|
||||
LVector2 _card_ul, _card_lr;
|
||||
|
||||
LMatrix4 _transform;
|
||||
CoordinateSystem _coordinate_system;
|
||||
|
||||
LPoint3 _ul3d, _lr3d;
|
||||
|
||||
// Returned from TextAssembler:
|
||||
LVector2 _text_ul, _text_lr;
|
||||
int _num_rows;
|
||||
std::wstring _wordwrapped_wtext;
|
||||
};
|
||||
|
||||
PipelineCycler<CData> _cycler;
|
||||
typedef CycleDataLockedReader<CData> CDLockedReader;
|
||||
typedef CycleDataReader<CData> CDReader;
|
||||
typedef CycleDataWriter<CData> CDWriter;
|
||||
typedef CycleDataStageReader<CData> CDStageReader;
|
||||
typedef CycleDataStageWriter<CData> CDStageWriter;
|
||||
|
||||
static PStatCollector _text_generate_pcollector;
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user