1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
|
// SPDX-License-Identifier: GPL-2.0-or-later
/** \file
* LPE <ruler> implementation, see lpe-ruler.cpp.
*/
/*
* Authors:
* Maximilian Albert
*
* Copyright (C) Maximilian Albert 2008 <maximilian.albert@gmail.com>
*
* Released under GNU GPL v2+, read the file 'COPYING' for more information.
*/
#include "live_effects/lpe-ruler.h"
// TODO due to internal breakage in glibmm headers, this must be last:
#include <glibmm/i18n.h>
namespace Inkscape {
namespace LivePathEffect {
static const Util::EnumData<MarkDirType> MarkDirData[] = {
{MARKDIR_LEFT , N_("Left"), "left"},
{MARKDIR_RIGHT , N_("Right"), "right"},
{MARKDIR_BOTH , N_("Both"), "both"},
};
static const Util::EnumDataConverter<MarkDirType> MarkDirTypeConverter(MarkDirData, sizeof(MarkDirData)/sizeof(*MarkDirData));
static const Util::EnumData<BorderMarkType> BorderMarkData[] = {
{BORDERMARK_NONE , NC_("Border mark", "None"), "none"},
{BORDERMARK_START , N_("Start"), "start"},
{BORDERMARK_END , N_("End"), "end"},
{BORDERMARK_BOTH , N_("Both"), "both"},
};
static const Util::EnumDataConverter<BorderMarkType> BorderMarkTypeConverter(BorderMarkData, sizeof(BorderMarkData)/sizeof(*BorderMarkData));
LPERuler::LPERuler(LivePathEffectObject *lpeobject) :
Effect(lpeobject),
mark_distance(_("_Mark distance:"), _("Distance between successive ruler marks"), "mark_distance", &wr, this, 20.0),
unit(_("Unit:"), _("Unit"), "unit", &wr, this),
mark_length(_("Ma_jor length:"), _("Length of major ruler marks"), "mark_length", &wr, this, 14.0),
minor_mark_length(_("Mino_r length:"), _("Length of minor ruler marks"), "minor_mark_length", &wr, this, 7.0),
major_mark_steps(_("Major steps_:"), _("Draw a major mark every ... steps"), "major_mark_steps", &wr, this, 5),
shift(_("Shift marks _by:"), _("Shift marks by this many steps"), "shift", &wr, this, 0),
mark_dir(_("Mark direction:"), _("Direction of marks (when viewing along the path from start to end)"), "mark_dir", MarkDirTypeConverter, &wr, this, MARKDIR_LEFT),
offset(_("_Offset:"), _("Offset of first mark"), "offset", &wr, this, 0.0),
border_marks(_("Border marks:"), _("Choose whether to draw marks at the beginning and end of the path"), "border_marks", BorderMarkTypeConverter, &wr, this, BORDERMARK_BOTH)
{
registerParameter(&unit);
registerParameter(&mark_distance);
registerParameter(&mark_length);
registerParameter(&minor_mark_length);
registerParameter(&major_mark_steps);
registerParameter(&shift);
registerParameter(&offset);
registerParameter(&mark_dir);
registerParameter(&border_marks);
major_mark_steps.param_make_integer();
major_mark_steps.param_set_range(1, 1000);
shift.param_make_integer();
mark_length.param_set_increments(1.0, 10.0);
minor_mark_length.param_set_increments(1.0, 10.0);
offset.param_set_increments(1.0, 10.0);
}
LPERuler::~LPERuler()
= default;
Geom::Point LPERuler::n_major;
Geom::Point LPERuler::n_minor;
Geom::Piecewise<Geom::D2<Geom::SBasis> >
LPERuler::ruler_mark(Geom::Point const &A, Geom::Point const &n, MarkType const &marktype)
{
using namespace Geom;
double real_mark_length = mark_length;
SPDocument *document = getSPDoc();
if (document) {
real_mark_length = Inkscape::Util::Quantity::convert(real_mark_length, unit.get_abbreviation(), document->getDisplayUnit()->abbr.c_str());
}
double real_minor_mark_length = minor_mark_length;
if (document) {
real_minor_mark_length = Inkscape::Util::Quantity::convert(real_minor_mark_length, unit.get_abbreviation(), document->getDisplayUnit()->abbr.c_str());
}
n_major = real_mark_length * n;
n_minor = real_minor_mark_length * n;
if (mark_dir == MARKDIR_BOTH) {
n_major = n_major * 0.5;
n_minor = n_minor * 0.5;
}
Point C, D;
switch (marktype) {
case MARK_MAJOR:
C = A;
D = A + n_major;
if (mark_dir == MARKDIR_BOTH)
C -= n_major;
break;
case MARK_MINOR:
C = A;
D = A + n_minor;
if (mark_dir == MARKDIR_BOTH)
C -= n_minor;
break;
default:
// do nothing
break;
}
Piecewise<D2<SBasis> > seg(D2<SBasis>(SBasis(C[X], D[X]), SBasis(C[Y], D[Y])));
return seg;
}
Geom::Piecewise<Geom::D2<Geom::SBasis> >
LPERuler::doEffect_pwd2 (Geom::Piecewise<Geom::D2<Geom::SBasis> > const & pwd2_in)
{
using namespace Geom;
const int mminterval = static_cast<int>(major_mark_steps);
const int i_shift = static_cast<int>(shift) % mminterval;
int sign = (mark_dir == MARKDIR_RIGHT ? 1 : -1 );
Piecewise<D2<SBasis> >output(pwd2_in);
Piecewise<D2<SBasis> >speed = derivative(pwd2_in);
Piecewise<SBasis> arclength = arcLengthSb(pwd2_in);
double totlength = arclength.lastValue();
//find at which times to draw a mark:
std::vector<double> s_cuts;
double real_mark_distance = mark_distance;
SPDocument *document = getSPDoc();
if (document) {
real_mark_distance = Inkscape::Util::Quantity::convert(real_mark_distance, unit.get_abbreviation(), document->getDisplayUnit()->abbr.c_str());
}
double real_offset = offset;
if (document) {
real_offset = Inkscape::Util::Quantity::convert(real_offset, unit.get_abbreviation(), document->getDisplayUnit()->abbr.c_str());
}
for (double s = real_offset; s<totlength; s+=real_mark_distance){
s_cuts.push_back(s);
}
std::vector<std::vector<double> > roots = multi_roots(arclength, s_cuts);
std::vector<double> t_cuts;
for (auto & root : roots){
//FIXME: 2geom multi_roots solver seem to sometimes "repeat" solutions.
//Here, we are supposed to have one and only one solution for each s.
if(root.size()>0)
t_cuts.push_back(root[0]);
}
//draw the marks
for (size_t i = 0; i < t_cuts.size(); i++) {
Point A = pwd2_in(t_cuts[i]);
Point n = rot90(unit_vector(speed(t_cuts[i])))*sign;
if (static_cast<int>(i % mminterval) == i_shift) {
output.concat (ruler_mark(A, n, MARK_MAJOR));
} else {
output.concat (ruler_mark(A, n, MARK_MINOR));
}
}
//eventually draw a mark at start
if ((border_marks == BORDERMARK_START || border_marks == BORDERMARK_BOTH) && (offset != 0.0 || i_shift != 0)){
Point A = pwd2_in.firstValue();
Point n = rot90(unit_vector(speed.firstValue()))*sign;
output.concat (ruler_mark(A, n, MARK_MAJOR));
}
//eventually draw a mark at end
if (border_marks == BORDERMARK_END || border_marks == BORDERMARK_BOTH){
Point A = pwd2_in.lastValue();
Point n = rot90(unit_vector(speed.lastValue()))*sign;
//speed.lastValue() is sometimes wrong when the path is closed: a tiny line seg might added at the end to fix rounding errors...
//TODO: Find a better fix!! (How do we know if the path was closed?)
if ( A == pwd2_in.firstValue() &&
speed.segs.size() > 1 &&
speed.segs.back()[X].size() <= 1 &&
speed.segs.back()[Y].size() <= 1 &&
speed.segs.back()[X].tailError(0) <= 1e-10 &&
speed.segs.back()[Y].tailError(0) <= 1e-10
){
n = rot90(unit_vector(speed.segs[speed.segs.size()-2].at1()))*sign;
}
output.concat (ruler_mark(A, n, MARK_MAJOR));
}
return output;
}
} //namespace LivePathEffect
} /* namespace Inkscape */
/*
Local Variables:
mode:c++
c-file-style:"stroustrup"
c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +))
indent-tabs-mode:nil
fill-column:99
End:
*/
// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4 :
|