~centralelyon2010/inkscape/imagelinks2

« back to all changes in this revision

Viewing changes to src/sp-tspan.cpp

  • Committer: JazzyNico
  • Date: 2011-08-29 20:25:30 UTC
  • Revision ID: nicoduf@yahoo.fr-20110829202530-6deuoz11q90usldv
Code refactoring and merging with trunk (revision 10599).

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
#define __SP_TSPAN_C__
2
 
 
3
1
/*
4
2
 * SVG <text> and <tspan> implementation
5
3
 *
6
4
 * Author:
7
5
 *   Lauris Kaplinski <lauris@kaplinski.com>
8
6
 *   bulia byak <buliabyak@users.sf.net>
 
7
 *   Jon A. Cruz <jon@joncruz.org>
 
8
 *   Abhishek Sharma
9
9
 *
10
10
 * Copyright (C) 1999-2002 Lauris Kaplinski
11
11
 * Copyright (C) 2000-2001 Ximian, Inc.
40
40
#include "sp-textpath.h"
41
41
#include "text-editing.h"
42
42
#include "style.h"
43
 
#include "libnr/nr-matrix-fns.h"
44
43
#include "xml/repr.h"
45
44
#include "document.h"
46
45
 
57
56
static void sp_tspan_set(SPObject *object, unsigned key, gchar const *value);
58
57
static void sp_tspan_update(SPObject *object, SPCtx *ctx, guint flags);
59
58
static void sp_tspan_modified(SPObject *object, unsigned flags);
60
 
static void sp_tspan_bbox(SPItem const *item, NRRect *bbox, Geom::Matrix const &transform, unsigned const flags);
 
59
static Geom::OptRect sp_tspan_bbox(SPItem const *item, Geom::Affine const &transform, SPItem::BBoxType type);
61
60
static Inkscape::XML::Node *sp_tspan_write(SPObject *object, Inkscape::XML::Document *doc, Inkscape::XML::Node *repr, guint flags);
62
61
static char *sp_tspan_description (SPItem *item);
63
62
 
133
132
{
134
133
    //SPTSpan *tspan = SP_TSPAN(object);
135
134
        
136
 
    sp_object_read_attr(object, "x");
137
 
    sp_object_read_attr(object, "y");
138
 
    sp_object_read_attr(object, "dx");
139
 
    sp_object_read_attr(object, "dy");
140
 
    sp_object_read_attr(object, "rotate");
141
 
    sp_object_read_attr(object, "sodipodi:role");
 
135
    object->readAttr( "x" );
 
136
    object->readAttr( "y" );
 
137
    object->readAttr( "dx" );
 
138
    object->readAttr( "dy" );
 
139
    object->readAttr( "rotate" );
 
140
    object->readAttr( "sodipodi:role" );
142
141
        
143
142
    if (((SPObjectClass *) tspan_parent_class)->build)
144
143
        ((SPObjectClass *) tspan_parent_class)->build(object, doc, repr);
168
167
    }
169
168
}
170
169
 
171
 
static void
172
 
sp_tspan_update(SPObject *object, SPCtx *ctx, guint flags)
 
170
static void sp_tspan_update(SPObject *object, SPCtx *ctx, guint flags)
173
171
{
174
 
    if (((SPObjectClass *) tspan_parent_class)->update)
 
172
    if (((SPObjectClass *) tspan_parent_class)->update) {
175
173
        ((SPObjectClass *) tspan_parent_class)->update(object, ctx, flags);
 
174
    }
176
175
        
177
 
    if (flags & SP_OBJECT_MODIFIED_FLAG) flags |= SP_OBJECT_PARENT_MODIFIED_FLAG;
 
176
    if (flags & SP_OBJECT_MODIFIED_FLAG) {
 
177
        flags |= SP_OBJECT_PARENT_MODIFIED_FLAG;
 
178
    }
178
179
    flags &= SP_OBJECT_MODIFIED_CASCADE;
179
180
        
180
 
    SPObject *ochild;
181
 
    for ( ochild = sp_object_first_child(object) ; ochild ; ochild = SP_OBJECT_NEXT(ochild) ) {
 
181
    for ( SPObject *ochild = object->firstChild() ; ochild ; ochild = ochild->getNext() ) {
182
182
        if ( flags || ( ochild->uflags & SP_OBJECT_MODIFIED_FLAG )) {
183
183
            ochild->updateDisplay(ctx, flags);
184
184
        }
185
185
    }
186
186
}
187
187
 
188
 
static void
189
 
sp_tspan_modified(SPObject *object, unsigned flags)
 
188
static void sp_tspan_modified(SPObject *object, unsigned flags)
190
189
{
191
 
    if (((SPObjectClass *) tspan_parent_class)->modified)
 
190
    if (((SPObjectClass *) tspan_parent_class)->modified) {
192
191
        ((SPObjectClass *) tspan_parent_class)->modified(object, flags);
 
192
    }
193
193
        
194
 
    if (flags & SP_OBJECT_MODIFIED_FLAG)
 
194
    if (flags & SP_OBJECT_MODIFIED_FLAG) {
195
195
        flags |= SP_OBJECT_PARENT_MODIFIED_FLAG;
 
196
    }
196
197
    flags &= SP_OBJECT_MODIFIED_CASCADE;
197
198
        
198
 
    SPObject *ochild;
199
 
    for ( ochild = sp_object_first_child(object) ; ochild ; ochild = SP_OBJECT_NEXT(ochild) ) {
 
199
    for ( SPObject *ochild = object->firstChild() ; ochild ; ochild = ochild->getNext() ) {
200
200
        if (flags || (ochild->mflags & SP_OBJECT_MODIFIED_FLAG)) {
201
201
            ochild->emitModified(flags);
202
202
        }
203
203
    }
204
204
}
205
205
 
206
 
static void sp_tspan_bbox(SPItem const *item, NRRect *bbox, Geom::Matrix const &transform, unsigned const /*flags*/)
 
206
static Geom::OptRect
 
207
sp_tspan_bbox(SPItem const *item, Geom::Affine const &transform, SPItem::BBoxType type)
207
208
{
 
209
    Geom::OptRect bbox;
208
210
    // find out the ancestor text which holds our layout
209
 
    SPObject *parent_text = SP_OBJECT(item);
210
 
    for (; parent_text != NULL && !SP_IS_TEXT(parent_text); parent_text = SP_OBJECT_PARENT (parent_text)){};
211
 
    if (parent_text == NULL) return;
 
211
    SPObject const *parent_text = item;
 
212
    while (parent_text && !SP_IS_TEXT(parent_text)) {
 
213
        parent_text = parent_text->parent;
 
214
    }
 
215
    if (parent_text == NULL) {
 
216
        return bbox;
 
217
    }
212
218
 
213
219
    // get the bbox of our portion of the layout
214
 
    SP_TEXT(parent_text)->layout.getBoundingBox(bbox, transform, sp_text_get_length_upto(parent_text, item), sp_text_get_length_upto(item, NULL) - 1);
 
220
    bbox = SP_TEXT(parent_text)->layout.bounds(transform, sp_text_get_length_upto(parent_text, item), sp_text_get_length_upto(item, NULL) - 1);
 
221
    if (!bbox) return bbox;
215
222
 
216
223
    // Add stroke width
217
 
    SPStyle* style=SP_OBJECT_STYLE (item);
218
 
    if (!style->stroke.isNone()) {
219
 
        double const scale = transform.descrim();
220
 
        if ( fabs(style->stroke_width.computed * scale) > 0.01 ) { // sinon c'est 0=oon veut pas de bord
221
 
            double const width = MAX(0.125, style->stroke_width.computed * scale);
222
 
            if ( fabs(bbox->x1 - bbox->x0) > -0.00001 && fabs(bbox->y1 - bbox->y0) > -0.00001 ) {
223
 
                bbox->x0-=0.5*width;
224
 
                bbox->x1+=0.5*width;
225
 
                bbox->y0-=0.5*width;
226
 
                bbox->y1+=0.5*width;
227
 
            }
228
 
        }
 
224
    // FIXME this code is incorrect
 
225
    if (type == SPItem::VISUAL_BBOX && !item->style->stroke.isNone()) {
 
226
        double scale = transform.descrim();
 
227
        bbox->expandBy(0.5 * item->style->stroke_width.computed * scale);
229
228
    }
 
229
    return bbox;
230
230
}
231
231
 
232
232
static Inkscape::XML::Node *
242
242
        
243
243
    if ( flags&SP_OBJECT_WRITE_BUILD ) {
244
244
        GSList *l = NULL;
245
 
        for (SPObject* child = sp_object_first_child(object) ; child != NULL ; child = SP_OBJECT_NEXT(child) ) {
 
245
        for (SPObject* child = object->firstChild() ; child ; child = child->getNext() ) {
246
246
            Inkscape::XML::Node* c_repr=NULL;
247
247
            if ( SP_IS_TSPAN(child) || SP_IS_TREF(child) ) {
248
248
                c_repr = child->updateRepr(xml_doc, NULL, flags);
251
251
            } else if ( SP_IS_STRING(child) ) {
252
252
                c_repr = xml_doc->createTextNode(SP_STRING(child)->string.c_str());
253
253
            }
254
 
            if ( c_repr ) l = g_slist_prepend(l, c_repr);
 
254
            if ( c_repr ) {
 
255
                l = g_slist_prepend(l, c_repr);
 
256
            }
255
257
        }
256
258
        while ( l ) {
257
259
            repr->addChild((Inkscape::XML::Node *) l->data, NULL);
259
261
            l = g_slist_remove(l, l->data);
260
262
        }
261
263
    } else {
262
 
        for (SPObject* child = sp_object_first_child(object) ; child != NULL ; child = SP_OBJECT_NEXT(child) ) {
 
264
        for (SPObject* child = object->firstChild() ; child ; child = child->getNext() ) {
263
265
            if ( SP_IS_TSPAN(child) || SP_IS_TREF(child) ) {
264
266
                child->updateRepr(flags);
265
267
            } else if ( SP_IS_TEXTPATH(child) ) {
266
268
                //c_repr = child->updateRepr(xml_doc, NULL, flags); // shouldn't happen
267
269
            } else if ( SP_IS_STRING(child) ) {
268
 
                SP_OBJECT_REPR(child)->setContent(SP_STRING(child)->string.c_str());
 
270
                child->getRepr()->setContent(SP_STRING(child)->string.c_str());
269
271
            }
270
272
        }
271
273
    }
272
274
        
273
 
    if (((SPObjectClass *) tspan_parent_class)->write)
 
275
    if (((SPObjectClass *) tspan_parent_class)->write) {
274
276
        ((SPObjectClass *) tspan_parent_class)->write(object, xml_doc, repr, flags);
 
277
    }
275
278
        
276
279
    return repr;
277
280
}
361
364
    textpath->originalPath = NULL;
362
365
    textpath->isUpdating=false;
363
366
    // set up the uri reference
364
 
    textpath->sourcePath = new SPUsePath(SP_OBJECT(textpath));
 
367
    textpath->sourcePath = new SPUsePath(textpath);
365
368
    textpath->sourcePath->user_unlink = sp_textpath_to_text;
366
369
}
367
370
 
387
390
        ((SPObjectClass *) textpath_parent_class)->release(object);
388
391
}
389
392
 
390
 
static void
391
 
sp_textpath_build(SPObject *object, SPDocument *doc, Inkscape::XML::Node *repr)
 
393
static void sp_textpath_build(SPObject *object, SPDocument *doc, Inkscape::XML::Node *repr)
392
394
{
393
 
    //SPTextPath *textpath = SP_TEXTPATH(object);
394
 
        
395
 
    sp_object_read_attr(object, "x");
396
 
    sp_object_read_attr(object, "y");
397
 
    sp_object_read_attr(object, "dx");
398
 
    sp_object_read_attr(object, "dy");
399
 
    sp_object_read_attr(object, "rotate");
400
 
    sp_object_read_attr(object, "startOffset");
401
 
    sp_object_read_attr(object, "xlink:href");
402
 
        
403
 
    bool  no_content=true;
 
395
    object->readAttr( "x" );
 
396
    object->readAttr( "y" );
 
397
    object->readAttr( "dx" );
 
398
    object->readAttr( "dy" );
 
399
    object->readAttr( "rotate" );
 
400
    object->readAttr( "startOffset" );
 
401
    object->readAttr( "xlink:href" );
 
402
        
 
403
    bool  no_content = true;
404
404
    for (Inkscape::XML::Node* rch = repr->firstChild() ; rch != NULL; rch = rch->next()) {
405
 
        if ( rch->type() == Inkscape::XML::TEXT_NODE ) {no_content=false;break;}
 
405
        if ( rch->type() == Inkscape::XML::TEXT_NODE )
 
406
        {
 
407
            no_content = false;
 
408
            break;
 
409
        }
406
410
    }
407
411
        
408
412
    if ( no_content ) {
409
 
        Inkscape::XML::Document *xml_doc = sp_document_repr_doc(doc);
 
413
        Inkscape::XML::Document *xml_doc = doc->getReprDoc();
410
414
        Inkscape::XML::Node* rch = xml_doc->createTextNode("");
411
415
        repr->addChild(rch, NULL);
412
416
    }
413
417
        
414
 
    if (((SPObjectClass *) textpath_parent_class)->build)
 
418
    if (((SPObjectClass *) textpath_parent_class)->build) {
415
419
        ((SPObjectClass *) textpath_parent_class)->build(object, doc, repr);
 
420
    }
416
421
}
417
422
 
418
423
static void
439
444
    }
440
445
}
441
446
 
442
 
static void
443
 
sp_textpath_update(SPObject *object, SPCtx *ctx, guint flags)
 
447
static void sp_textpath_update(SPObject *object, SPCtx *ctx, guint flags)
444
448
{
445
449
    SPTextPath *textpath = SP_TEXTPATH(object);
446
450
        
447
 
    textpath->isUpdating=true;
448
 
    if ( textpath->sourcePath->sourceDirty ) refresh_textpath_source(textpath);
449
 
    textpath->isUpdating=false;
 
451
    textpath->isUpdating = true;
 
452
    if ( textpath->sourcePath->sourceDirty ) {
 
453
        refresh_textpath_source(textpath);
 
454
    }
 
455
    textpath->isUpdating = false;
450
456
                
451
 
    if (((SPObjectClass *) textpath_parent_class)->update)
 
457
    if (((SPObjectClass *) textpath_parent_class)->update) {
452
458
        ((SPObjectClass *) textpath_parent_class)->update(object, ctx, flags);
 
459
    }
453
460
                
454
 
    if (flags & SP_OBJECT_MODIFIED_FLAG) flags |= SP_OBJECT_PARENT_MODIFIED_FLAG;
 
461
    if (flags & SP_OBJECT_MODIFIED_FLAG) {
 
462
        flags |= SP_OBJECT_PARENT_MODIFIED_FLAG;
 
463
    }
455
464
    flags &= SP_OBJECT_MODIFIED_CASCADE;
456
465
                        
457
 
    SPObject *ochild;
458
 
    for ( ochild = sp_object_first_child(object) ; ochild ; ochild = SP_OBJECT_NEXT(ochild) ) {
 
466
    for ( SPObject *ochild = object->firstChild() ; ochild ; ochild = ochild->getNext() ) {
459
467
        if ( flags || ( ochild->uflags & SP_OBJECT_MODIFIED_FLAG )) {
460
468
            ochild->updateDisplay(ctx, flags);
461
469
        }
483
491
    }
484
492
}
485
493
 
486
 
static void
487
 
sp_textpath_modified(SPObject *object, unsigned flags)
 
494
static void sp_textpath_modified(SPObject *object, unsigned flags)
488
495
{
489
 
    if (((SPObjectClass *) textpath_parent_class)->modified)
 
496
    if (((SPObjectClass *) textpath_parent_class)->modified) {
490
497
        ((SPObjectClass *) textpath_parent_class)->modified(object, flags);
 
498
    }
491
499
        
492
 
    if (flags & SP_OBJECT_MODIFIED_FLAG)
 
500
    if (flags & SP_OBJECT_MODIFIED_FLAG) {
493
501
        flags |= SP_OBJECT_PARENT_MODIFIED_FLAG;
 
502
    }
494
503
    flags &= SP_OBJECT_MODIFIED_CASCADE;
495
504
        
496
 
    SPObject *ochild;
497
 
    for ( ochild = sp_object_first_child(object) ; ochild ; ochild = SP_OBJECT_NEXT(ochild) ) {
 
505
    for ( SPObject *ochild = object->firstChild() ; ochild ; ochild = ochild->getNext() ) {
498
506
        if (flags || (ochild->mflags & SP_OBJECT_MODIFIED_FLAG)) {
499
507
            ochild->emitModified(flags);
500
508
        }
514
522
        if (textpath->startOffset.unit == SVGLength::PERCENT) {
515
523
                Inkscape::SVGOStringStream os;
516
524
            os << (textpath->startOffset.computed * 100.0) << "%";
517
 
            SP_OBJECT_REPR(textpath)->setAttribute("startOffset", os.str().c_str());
 
525
            textpath->getRepr()->setAttribute("startOffset", os.str().c_str());
518
526
        } else {
519
527
            /* FIXME: This logic looks rather undesirable if e.g. startOffset is to be
520
528
               in ems. */
526
534
        
527
535
    if ( flags&SP_OBJECT_WRITE_BUILD ) {
528
536
        GSList *l = NULL;
529
 
        for (SPObject* child = sp_object_first_child(object) ; child != NULL ; child = SP_OBJECT_NEXT(child) ) {
 
537
        for (SPObject* child = object->firstChild() ; child ; child = child->getNext() ) {
530
538
            Inkscape::XML::Node* c_repr=NULL;
531
539
            if ( SP_IS_TSPAN(child) || SP_IS_TREF(child) ) {
532
540
                c_repr = child->updateRepr(xml_doc, NULL, flags);
535
543
            } else if ( SP_IS_STRING(child) ) {
536
544
                c_repr = xml_doc->createTextNode(SP_STRING(child)->string.c_str());
537
545
            }
538
 
            if ( c_repr ) l = g_slist_prepend(l, c_repr);
 
546
            if ( c_repr ) {
 
547
                l = g_slist_prepend(l, c_repr);
 
548
            }
539
549
        }
540
550
        while ( l ) {
541
551
            repr->addChild((Inkscape::XML::Node *) l->data, NULL);
543
553
            l = g_slist_remove(l, l->data);
544
554
        }
545
555
    } else {
546
 
        for (SPObject* child = sp_object_first_child(object) ; child != NULL ; child = SP_OBJECT_NEXT(child) ) {
 
556
        for (SPObject* child = object->firstChild() ; child ; child = child->getNext() ) {
547
557
            if ( SP_IS_TSPAN(child) || SP_IS_TREF(child) ) {
548
558
                child->updateRepr(flags);
549
559
            } else if ( SP_IS_TEXTPATH(child) ) {
550
560
                //c_repr = child->updateRepr(xml_doc, NULL, flags); // shouldn't happen
551
561
            } else if ( SP_IS_STRING(child) ) {
552
 
                SP_OBJECT_REPR(child)->setContent(SP_STRING(child)->string.c_str());
 
562
                child->getRepr()->setContent(SP_STRING(child)->string.c_str());
553
563
            }
554
564
        }
555
565
    }
556
566
        
557
 
    if (((SPObjectClass *) textpath_parent_class)->write)
 
567
    if (((SPObjectClass *) textpath_parent_class)->write) {
558
568
        ((SPObjectClass *) textpath_parent_class)->write(object, xml_doc, repr, flags);
 
569
    }
559
570
        
560
571
    return repr;
561
572
}
575
586
void
576
587
sp_textpath_to_text(SPObject *tp)
577
588
{
578
 
    SPObject *text = SP_OBJECT_PARENT(tp);
 
589
    SPObject *text = tp->parent;
579
590
 
580
 
    NRRect bbox;
581
 
    sp_item_invoke_bbox(SP_ITEM(text), &bbox, sp_item_i2doc_affine(SP_ITEM(text)), TRUE);
582
 
    Geom::Point xy(bbox.x0, bbox.y0);
 
591
    Geom::OptRect bbox = SP_ITEM(text)->geometricBounds(SP_ITEM(text)->i2doc_affine());
 
592
    if (!bbox) return;
 
593
    Geom::Point xy = bbox->min();
583
594
 
584
595
    // make a list of textpath children
585
596
    GSList *tp_reprs = NULL;
586
 
    for (SPObject *o = SP_OBJECT(tp)->firstChild() ; o != NULL; o = o->next) {
587
 
        tp_reprs = g_slist_prepend(tp_reprs, SP_OBJECT_REPR(o));
 
597
    for (SPObject *o = tp->firstChild() ; o != NULL; o = o->next) {
 
598
        tp_reprs = g_slist_prepend(tp_reprs, o->getRepr());
588
599
    }
589
600
 
590
601
    for ( GSList *i = tp_reprs ; i ; i = i->next ) {
591
602
        // make a copy of each textpath child
592
 
        Inkscape::XML::Node *copy = ((Inkscape::XML::Node *) i->data)->duplicate(SP_OBJECT_REPR(text)->document());
 
603
        Inkscape::XML::Node *copy = ((Inkscape::XML::Node *) i->data)->duplicate(text->getRepr()->document());
593
604
        // remove the old repr from under textpath
594
 
        SP_OBJECT_REPR(tp)->removeChild((Inkscape::XML::Node *) i->data);
 
605
        tp->getRepr()->removeChild((Inkscape::XML::Node *) i->data);
595
606
        // put its copy under text
596
 
        SP_OBJECT_REPR(text)->addChild(copy, NULL); // fixme: copy id
 
607
        text->getRepr()->addChild(copy, NULL); // fixme: copy id
597
608
    }
598
609
 
599
610
    //remove textpath
603
614
    // set x/y on text
604
615
    /* fixme: Yuck, is this really the right test? */
605
616
    if (xy[Geom::X] != 1e18 && xy[Geom::Y] != 1e18) {
606
 
        sp_repr_set_svg_double(SP_OBJECT_REPR(text), "x", xy[Geom::X]);
607
 
        sp_repr_set_svg_double(SP_OBJECT_REPR(text), "y", xy[Geom::Y]);
 
617
        sp_repr_set_svg_double(text->getRepr(), "x", xy[Geom::X]);
 
618
        sp_repr_set_svg_double(text->getRepr(), "y", xy[Geom::Y]);
608
619
    }
609
620
}
610
621
 
618
629
  fill-column:99
619
630
  End:
620
631
*/
621
 
// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99 :
 
632
// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:fileencoding=utf-8:textwidth=99 :