File Coverage

blib/lib/DDC/PP/CQuery.pm
Criterion Covered Total %
statement 183 305 60.0
branch 13 52 25.0
condition 15 73 20.5
subroutine 71 145 48.9
pod 0 13 0.0
total 282 588 47.9


line stmt bran cond sub pod time code
1             ##-*- Mode: CPerl -*-
2              
3             ##======================================================================
4             ## top-level
5             package DDC::PP::CQuery;
6 20     20   135 use DDC::PP::Object;
  20         45  
  20         651  
7 20     20   100 use DDC::Utils qw();
  20         36  
  20         328  
8 20     20   87 use Carp qw(carp confess);
  20         42  
  20         964  
9 20     20   109 use strict;
  20         41  
  20         494  
10              
11              
12             ##======================================================================
13             ## CQuery
14             package DDC::PP::CQuery;
15 20     20   99 use strict;
  20         53  
  20         8432  
16             our @ISA = qw(DDC::PP::Object);
17              
18             __PACKAGE__->defprop('Label');
19             __PACKAGE__->defprop('Options');
20             sub new {
21 90     90 0 337 my ($that,$label,%opts) = @_;
22 90         299 return $that->SUPER::new(Label=>$label,%opts);
23             }
24              
25             ##-- options only
26             sub _new {
27 317     317   601 my $that = shift;
28 317         981 return $that->SUPER::new(@_);
29             }
30              
31 0     0 0 0 sub Negated { return 0; }
32 0   0 0 0 0 sub Negate { confess((ref($_[0])||$_[0])."::Negate(): attempt to negate non-negatable query"); }
33              
34 0     0 0 0 sub GetMatchId { return 0; }
35 0   0 0 0 0 sub SetMatchId { confess((ref($_[0])||$_[0])."::SetMatchId(): attempt to set match-ID for non-token query"); }
36 2     2 0 10 sub HasMatchId { $_[0]->GetMatchId != 0; }
37              
38 0     0 0 0 sub RootOk { return !$_[0]->Negated; }
39 0     0 0 0 sub ClearOptions { delete $_[0]{Options}; }
40       0 0   sub Clear { ; }
41              
42             sub toString {
43 0     0 0 0 return $_[0]{Label};
44             }
45             sub optionsToString {
46 2 50   2 0 9 return '' if (!$_[0]{Options});
47 2         11 return $_[0]{Options}->toString;
48             }
49             sub toStringFull {
50 2     2 0 13 return $_[0]->toString . $_[0]->optionsToString;
51             }
52              
53             ##-- stringification utility
54             sub sqString {
55 30 50   30 0 117 return DDC::Utils::escapeq(defined($_[1]) ? $_[1] : '');
56             }
57              
58              
59             ##======================================================================
60             ## CQNegatable
61             package DDC::PP::CQNegatable;
62 20     20   141 use strict;
  20         36  
  20         4026  
63             our @ISA = qw(DDC::PP::CQuery);
64              
65             __PACKAGE__->defprop('Negated');
66             sub new {
67 2     2   13 my ($that,$label,$negated,%opts) = @_;
68 2         15 return $that->_new(Label=>$label,Negated=>$negated,%opts);
69             }
70              
71 32 50   32   117 sub Negated { return $_[0]->getNegated ? 1 : 0; }
72 0 0   0   0 sub Negate { return $_[0]->setNegated($_[0]->getNegated ? 0 : 1); }
73              
74             sub NegString {
75 32     32   93 my ($obj,$s) = @_;
76 32 50       122 return $_[0]->Negated ? "!$s" : $s;
77             }
78              
79             ##======================================================================
80             ## CQAtomic
81             package DDC::PP::CQAtomic;
82 20     20   176 use strict;
  20         59  
  20         1161  
83             our @ISA = qw(DDC::PP::CQNegatable);
84              
85              
86             ##======================================================================
87             ## CQBinOp
88             package DDC::PP::CQBinOp;
89 20     20   121 use strict;
  20         41  
  20         6488  
90             our @ISA = qw(DDC::PP::CQNegatable);
91              
92             __PACKAGE__->defprop('Dtr1');
93             __PACKAGE__->defprop('Dtr2');
94             __PACKAGE__->defprop('OpName');
95             sub new {
96 47     47   121 my ($that,$dtr1,$dtr2,$opName,$negated,%opts) = @_;
97 47         199 return $that->_new(Label=>$opName,OpName=>$opName,Dtr1=>$dtr1,Dtr2=>$dtr2,Negated=>$negated,%opts);
98             }
99              
100 6     6   9 sub Children { [grep {defined($_)} @{$_[0]}{qw(Dtr1 Dtr2)}]; }
  12         33  
  6         14  
101              
102             sub GetMatchId {
103 0     0   0 my $obj = shift;
104             return (($obj->{Dtr2} && $obj->{Dtr2}->GetMatchId)
105             || ($obj->{Dtr1} && $obj->{Dtr1}->GetMatchId)
106 0   0     0 || 0);
107             }
108             sub SetMatchId {
109 0     0   0 my ($obj,$id) = @_;
110 0 0       0 $obj->{Dtr1}->SetMatchId($id) if ($obj->{Dtr1});
111 0 0       0 $obj->{Dtr2}->SetMatchId($id) if ($obj->{Dtr2});
112 0         0 return $id;
113             }
114              
115             sub Clear {
116 0     0   0 delete @{$_[0]}{qw(Dtr1 Dtr2)};
  0         0  
117             }
118             sub toString {
119 5     5   2142 my $obj = shift;
120 5         26 return $obj->NegString("(".$obj->{Dtr1}->toString." ".$obj->{OpName}." ".$obj->{Dtr2}->toString.")");
121             }
122              
123             ##======================================================================
124             ## CQAnd
125             package DDC::PP::CQAnd;
126 20     20   157 use strict;
  20         39  
  20         1816  
127             our @ISA = qw(DDC::PP::CQBinOp);
128              
129             sub new {
130 33     33   87 my ($that,$dtr1,$dtr2) = @_;
131 33         147 return $that->SUPER::new($dtr1,$dtr2,"&&");
132             }
133              
134             ##======================================================================
135             ## CQAndImplicit
136             package DDC::PP::CQAndImplicit;
137 20     20   124 use strict;
  20         47  
  20         1100  
138             our @ISA = qw(DDC::PP::CQAnd);
139              
140             ##======================================================================
141             ## CQOr
142             package DDC::PP::CQOr;
143 20     20   133 use strict;
  20         43  
  20         1962  
144             our @ISA = qw(DDC::PP::CQBinOp);
145              
146             sub new {
147 2     2   8 my ($that,$dtr1,$dtr2) = @_;
148 2         23 return $that->SUPER::new($dtr1,$dtr2,"||");
149             }
150              
151             ##======================================================================
152             ## CQWith
153             package DDC::PP::CQWith;
154 20     20   121 use strict;
  20         43  
  20         3995  
155             our @ISA = qw(DDC::PP::CQBinOp);
156              
157             __PACKAGE__->defprop('MatchId');
158             sub new {
159 12     12   32 my ($that,$dtr1,$dtr2,$matchid,%opts) = @_;
160 12   50     77 return $that->SUPER::new($dtr1,$dtr2,"WITH",0,MatchId=>($matchid||0),%opts);
161             }
162              
163             sub GetMatchId {
164 0     0   0 my $obj = shift;
165 0   0     0 return $obj->{MatchId} || $obj->SUPER::GetMatchId();
166             }
167             sub SetMatchId {
168 0     0   0 $_[0]{MatchId} = $_[1];
169             }
170              
171             sub toString {
172 0 0   0   0 return $_[0]->SUPER::toString().($_[0]{MatchId} ? " =$_[0]{MatchId}" : '');
173             }
174              
175             ##======================================================================
176             ## CQWithout
177             package DDC::PP::CQWithout;
178 20     20   130 use strict;
  20         47  
  20         1912  
179             our @ISA = qw(DDC::PP::CQWith);
180              
181             sub new {
182 2     2   8 my ($that,$dtr1,$dtr2,$matchid,%opts) = @_;
183 2         15 return $that->SUPER::new($dtr1,$dtr2,$matchid,OpName=>"WITHOUT",%opts);
184             }
185              
186             ##======================================================================
187             ## CQWithor
188             package DDC::PP::CQWithor;
189 20     20   133 use strict;
  20         63  
  20         1744  
190             our @ISA = qw(DDC::PP::CQWith);
191              
192             sub new {
193 2     2   7 my ($that,$dtr1,$dtr2,$matchid,%opts) = @_;
194 2         16 return $that->SUPER::new($dtr1,$dtr2,$matchid,OpName=>"WITHOR",%opts);
195             }
196              
197             ##======================================================================
198             ## CQToken
199             package DDC::PP::CQToken;
200 20     20   142 use strict;
  20         35  
  20         7010  
201             our @ISA = qw(DDC::PP::CQAtomic);
202              
203             __PACKAGE__->defprop('IndexName');
204             __PACKAGE__->defprop('Value');
205             __PACKAGE__->defprop('MatchId');
206             sub new {
207 268     268   1018 my ($that,$index,$value,$matchid,%opts) = @_;
208 268   100     2355 return $that->_new(Label=>$value,Negated=>0,IndexName=>($index||''),Value=>$value,MatchId=>($matchid||0),%opts);
      50        
209             }
210              
211             *GetMatchId = *getMatchId;
212             *SetMatchId = *setMatchId;
213              
214 0     0   0 sub OperatorKey { return '_'; }
215 27   50 27   205 sub IndexName { return $_[0]{IndexName} || '' }
216 0   0 0   0 sub BreakName { return $_[0]{BreakName} || '' }
217              
218 27 50   27   79 sub IndexString { return $_[0]->IndexName eq '' ? '' : ('$'.$_[0]->IndexName.'='); }
219 12     12   61 sub ValueString { return $_[0]->sqString($_[0]->{Value}); }
220 27 50   27   143 sub MatchIdString { return $_[0]{MatchId} ? " =$_[0]{MatchId}" : ''; }
221              
222             sub toString {
223 27     27   137 my $obj = shift;
224 27         96 return $obj->NegString($obj->IndexString . $obj->ValueString . $obj->MatchIdString);
225             }
226              
227             ##======================================================================
228             ## CQTokExact
229             package DDC::PP::CQTokExact;
230 20     20   150 use strict;
  20         48  
  20         1781  
231             our @ISA = qw(DDC::PP::CQToken);
232              
233 0     0   0 sub OperatorKey { return '@'; }
234 10     10   40 sub ValueString { return '@'.$_[0]->sqString($_[0]->{Value}); }
235              
236             ##======================================================================
237             ## CQTokAny
238             package DDC::PP::CQTokAny;
239 20     20   137 use strict;
  20         37  
  20         1637  
240             our @ISA = qw(DDC::PP::CQToken);
241              
242 0     0   0 sub OperatorKey { return '*'; }
243 0     0   0 sub ValueString { return '*'; }
244              
245             ##======================================================================
246             ## CQTokAnchor
247             package DDC::PP::CQTokAnchor;
248 20     20   176 use strict;
  20         53  
  20         3586  
249             our @ISA = qw(DDC::PP::CQToken);
250              
251             __PACKAGE__->defprop('ValueInt');
252             sub new {
253 0     0   0 my ($that,$index,$value,%opts) = @_;
254 0         0 return $that->SUPER::new($index,$value,0,ValueInt=>$value,%opts);
255             }
256              
257 0     0   0 sub OperatorKey { return '.'; }
258 0     0   0 sub IndexString { return '$.'.$_[0]->IndexName.'='; }
259 0   0 0   0 sub ValueString { return $_[0]{ValueInt} || 0; }
260              
261             ##======================================================================
262             ## CQTokRegex
263             package DDC::PP::CQTokRegex;
264 20     20   128 use strict;
  20         34  
  20         3244  
265             our @ISA = qw(DDC::PP::CQToken);
266              
267             __PACKAGE__->defprop('RegexNegated');
268             sub new {
269 11     11   27 my ($that,$index,$regex,$negated,%opts) = @_;
270 11         48 return $that->SUPER::new($index,$regex,0,%opts);
271             }
272              
273 0     0   0 sub OperatorKey { return '/_/'; }
274             sub ValueString {
275 1 50   1   5 my $re = defined($_[0]{Value}) ? $_[0]{Value} : '';
276             #$re =~ s{/}{\\/}g; ##-- no; see mantis #47973
277 1 50       10 return ($_[0]{RegexNegated} ? '!' : '')."/$re/";
278             }
279              
280             ##======================================================================
281             ## CQTokSet
282             package DDC::PP::CQTokSet;
283 20     20   145 use strict;
  20         45  
  20         4233  
284             our @ISA = qw(DDC::PP::CQToken);
285              
286             __PACKAGE__->defprop('Values');
287             sub new {
288 233     233   833 my ($that,$index,$rawValue,$values,%opts) = @_;
289 233   50     975 return $that->SUPER::new($index,$rawValue,0,Values=>($values||[]),%opts);
290             }
291              
292 0     0   0 sub OperatorKey { return '@_' };
293             sub SetValueString {
294 4     4   10 my ($obj,$values) = @_;
295 4   33     8 $values ||= $obj->{Values};
296 4         9 return join(',', map {$obj->sqString($_)} @$values);
  8         16  
297             }
298             sub ValueString {
299 0     0   0 return '@{' . $_[0]->SetValueString . '}';
300             }
301              
302             ##======================================================================
303             ## CQTokInfl
304             package DDC::PP::CQTokInfl;
305 20     20   135 use strict;
  20         45  
  20         5878  
306             our @ISA = qw(DDC::PP::CQTokSet);
307              
308             __PACKAGE__->defprop('Expanders');
309             sub new {
310 229     229   742 my ($that,$index,$value,$expanders,%opts) = @_;
311 229   100     1088 return $that->SUPER::new($index,$value,[$value],Expanders=>($expanders||[]),Value=>$value,%opts);
312             }
313             sub newSet {
314 0     0   0 my ($that,$index,$values,$expanders,%opts) = @_;
315 0   0     0 return $that->SUPER::new($index,"$values",$values,Expanders=>($expanders||[]),Value=>"$values",%opts);
316             }
317              
318 0     0   0 sub OperatorKey { return '_'; }
319             sub ExpanderString {
320 16     16   31 my $obj = shift;
321 16 50       23 return '' if (!@{$obj->{Expanders}||[]});
  16 100       117  
322 8 50 33     12 return join('|', '', map {!defined($_) || $_ eq '' ? '-' : $_} @{$obj->{Expanders}});
  8         77  
  8         16  
323             }
324             sub ValueString {
325 12     12   40 my $obj = shift;
326 12         50 return $obj->DDC::PP::CQToken::ValueString . $obj->ExpanderString;
327             }
328              
329             ##======================================================================
330             ## CQTokSetInfl
331             package DDC::PP::CQTokSetInfl;
332 20     20   138 use strict;
  20         55  
  20         2908  
333             our @ISA = qw(DDC::PP::CQTokInfl);
334              
335             __PACKAGE__->defprop('RawValues');
336             sub new {
337 4     4   12 my ($that,$index,$values,$expanders,%opts) = @_;
338 4         19 return $that->SUPER::new($index,$values,$expanders,RawValues=>$values,%opts);
339             }
340              
341             sub ValueString {
342 4     4   10 my $obj = shift;
343 4         15 return '{' . $obj->SetValueString($obj->{RawValues}) . '}' . $obj->ExpanderString;
344             }
345              
346             ##======================================================================
347             ## CQTokPrefix
348             package DDC::PP::CQTokPrefix;
349 20     20   151 use strict;
  20         43  
  20         2641  
350             our @ISA = qw(DDC::PP::CQToken);
351              
352             sub new {
353 4     4   12 my ($that,$index,$prefix,%opts) = @_;
354 4         24 return $that->SUPER::new($index,$prefix,0,%opts);
355             }
356 0     0   0 sub OperatorKey { return '/_/'; }
357 0     0   0 sub ValueString { return $_[0]->sqString($_[0]{Value}).'*'; }
358              
359             ##======================================================================
360             ## CQTokSuffix
361             package DDC::PP::CQTokSuffix;
362 20     20   144 use strict;
  20         38  
  20         2733  
363             our @ISA = qw(DDC::PP::CQToken);
364              
365             sub new {
366 0     0   0 my ($that,$index,$suffix,%opts) = @_;
367 0         0 return $that->SUPER::new($index,$suffix,0,%opts);
368             }
369 0     0   0 sub OperatorKey { return '/_/'; }
370 0     0   0 sub ValueString { return '*'.$_[0]->sqString($_[0]{Value}); }
371              
372             ##======================================================================
373             ## CQTokInfix
374             package DDC::PP::CQTokInfix;
375 20     20   133 use strict;
  20         35  
  20         2895  
376             our @ISA = qw(DDC::PP::CQToken);
377              
378             sub new {
379 0     0   0 my ($that,$index,$infix,%opts) = @_;
380 0         0 return $that->SUPER::new($index,$infix,0,%opts);
381             }
382 0     0   0 sub OperatorKey { return '/_/'; }
383 0     0   0 sub ValueString { return '*'.$_[0]->sqString($_[0]{Value}).'*'; }
384              
385             ##======================================================================
386             ## CQTokPrefixSet
387             package DDC::PP::CQTokPrefixSet;
388 20     20   149 use strict;
  20         52  
  20         2732  
389             our @ISA = qw(DDC::PP::CQTokSet);
390              
391             sub new {
392 0     0   0 my ($that,$index,$prefixes,%opts) = @_;
393 0   0     0 return $that->SUPER::new($index,"$prefixes",($prefixes||[]),%opts);
394             }
395              
396 0     0   0 sub OperatorKey { return '/_/'; }
397 0     0   0 sub ValueString { return '{' . $_[0]->SetValueString . '}*'; }
398              
399             ##======================================================================
400             ## CQTokSuffixSet
401             package DDC::PP::CQTokSuffixSet;
402 20     20   131 use strict;
  20         66  
  20         2676  
403             our @ISA = qw(DDC::PP::CQTokSet);
404              
405             sub new {
406 0     0   0 my ($that,$index,$suffixes,%opts) = @_;
407 0   0     0 return $that->SUPER::new($index,"$suffixes",($suffixes||[]),%opts);
408             }
409              
410 0     0   0 sub OperatorKey { return '/_/'; }
411 0     0   0 sub ValueString { return '*{' . $_[0]->SetValueString . '}'; }
412              
413             ##======================================================================
414             ## CQTokInfixSet
415             package DDC::PP::CQTokInfixSet;
416 20     20   122 use strict;
  20         58  
  20         3259  
417             our @ISA = qw(DDC::PP::CQTokSet);
418              
419             sub new {
420 0     0   0 my ($that,$index,$infixes,%opts) = @_;
421 0   0     0 return $that->SUPER::new($index,"$infixes",($infixes||[]),%opts);
422             }
423              
424 0     0   0 sub OperatorKey { return '/_/'; }
425 0     0   0 sub ValueString { return '*{' . $_[0]->SetValueString . '}*'; }
426              
427             ##======================================================================
428             ## CQTokMorph
429             package DDC::PP::CQTokMorph;
430 20     20   170 use strict;
  20         45  
  20         4223  
431             our @ISA = qw(DDC::PP::CQToken);
432              
433             __PACKAGE__->defprop('Items');
434             sub new {
435 0     0   0 my ($that,$index,$items,%opts) = @_;
436 0   0     0 return $that->SUPER::new(($index||'MorphPattern'),"$items",0, Items=>($items||[]), %opts);
      0        
437             }
438              
439             sub Append {
440 0     0   0 my ($obj,$item) = @_;
441 0         0 push(@{$obj->{Items}},$item);
  0         0  
442             }
443              
444 0     0   0 sub OperatorKey { return '[_]'; }
445 0 0   0   0 sub ValueString { return '[' . join(',', map {$_[0]->sqString($_)} @{$_[0]{Items}||[]}). ']'; }
  0         0  
  0         0  
446              
447             ##======================================================================
448             ## CQTokLemma
449             package DDC::PP::CQTokLemma;
450 20     20   132 use strict;
  20         50  
  20         2677  
451             our @ISA = qw(DDC::PP::CQTokMorph);
452              
453             sub new {
454 0     0   0 my ($that,$index,$value,%opts) = @_;
455 0   0     0 return $that->DDC::PP::CQToken::new(($index||'Lemma'),$value,0,%opts);
456             }
457              
458 0     0   0 sub OperatorKey { return '%_'; }
459 0     0   0 sub ValueString { return '%' . $_[0]->sqString($_[0]{Value}); }
460              
461             ##======================================================================
462             ## CQTokThes
463             package DDC::PP::CQTokThes;
464 20     20   156 use strict;
  20         43  
  20         2891  
465             our @ISA = qw(DDC::PP::CQToken);
466              
467             sub new {
468 0     0   0 my ($that,$index,$value,%opts) = @_;
469 0   0     0 return $that->SUPER::new(($index||'Thes'),$value,0,%opts);
470             }
471              
472 0     0   0 sub OperatorKey { return ':{_}'; }
473 0     0   0 sub ValueString { return ':{' . $_[0]->sqString($_[0]{Value}) . '}'; };
474              
475             ##======================================================================
476             ## CQTokChunk
477             package DDC::PP::CQTokChunk;
478 20     20   134 use strict;
  20         52  
  20         2704  
479             our @ISA = qw(DDC::PP::CQToken);
480              
481             sub new {
482 0     0   0 my ($that,$index,$value,%opts) = @_;
483 0   0     0 return $that->SUPER::new(($index||''),$value,0,%opts);
484             }
485              
486 0     0   0 sub OperatorKey { return '^_'; }
487 0     0   0 sub ValueString { return '^' . $_[0]->sqString($_[0]{Value}); }
488              
489             ##======================================================================
490             ## CQTokFile
491             package DDC::PP::CQTokFile;
492 20     20   148 use strict;
  20         46  
  20         2733  
493             our @ISA = qw(DDC::PP::CQToken);
494              
495             sub new {
496 0     0   0 my ($that,$index,$filename,%opts) = @_;
497 0         0 return $that->SUPER::new($index,$filename,0,%opts);
498             }
499              
500 0     0   0 sub OperatorKey { return '<_'; }
501 0     0   0 sub ValueString { return '<' . $_[0]->sqString($_[0]{Value}); }
502              
503             ##======================================================================
504             ## CQNear
505             package DDC::PP::CQNear;
506 20     20   143 use strict;
  20         42  
  20         8357  
507             our @ISA = qw(DDC::PP::CQNegatable);
508              
509             __PACKAGE__->defprop('Dtr1');
510             __PACKAGE__->defprop('Dtr2');
511             __PACKAGE__->defprop('Dtr3');
512             __PACKAGE__->defprop('Dist');
513             sub new {
514 0     0   0 my ($that,$dist,$dtr1,$dtr2,$dtr3,%opts) = @_;
515 0 0       0 return $that->SUPER::new("NEAR",0,Dist=>(defined($dist) ? $dist : 1), Dtr1=>$dtr1, Dtr2=>$dtr2, Dtr3=>$dtr3, %opts);
516             }
517              
518 0     0   0 sub Children { [grep {defined($_)} @{$_[0]}{qw(Dtr1 Dtr2 Dtr3)}]; }
  0         0  
  0         0  
519              
520 0     0   0 sub Clear { delete @{$_[0]}{qw(Dtr1 Dtr2 Dtr3)}; }
  0         0  
521              
522             sub GetMatchId {
523 0     0   0 my $obj = shift;
524             return (($obj->{Dtr3} && $obj->{Dtr3}->GetMatchId)
525             || ($obj->{Dtr2} && $obj->{Dtr2}->GetMatchId)
526             || ($obj->{Dtr1} && $obj->{Dtr2}->GetMatchId)
527 0   0     0 || 0
528             );
529             }
530             sub SetMatchId {
531 0     0   0 my ($obj,$id) = @_;
532 0 0       0 $obj->{Dtr1}->SetMatchId($id) if ($obj->{Dtr1});
533 0 0       0 $obj->{Dtr2}->SetMatchId($id) if ($obj->{Dtr2});
534 0 0       0 $obj->{Dtr3}->SetMatchId($id) if ($obj->{Dtr3});
535 0         0 return $id;
536             }
537              
538             sub toString {
539 0     0   0 return $_[0]->NegString("NEAR(".join(',', (map {$_->toString} grep {defined $_} @{$_[0]}{qw(Dtr1 Dtr2 Dtr3)}), $_[0]{Dist}).")");
  0         0  
  0         0  
  0         0  
540             }
541              
542             ##======================================================================
543             ## CQSeq
544             package DDC::PP::CQSeq;
545 20     20   140 use strict;
  20         34  
  20         13518  
546             our @ISA = qw(DDC::PP::CQAtomic);
547              
548             __PACKAGE__->defprop('Items');
549             __PACKAGE__->defprop('Dists');
550             __PACKAGE__->defprop('DistOps');
551             sub new {
552 2     2   19 my ($that,$items,$dists,$distops,%opts) = @_;
553 2   50     38 return $that->SUPER::new('""',0, Items=>($items||[]), Dists=>($dists||[]), DistOps=>($distops||[]), %opts);
      50        
      50        
554             }
555             sub new1 {
556 0     0   0 my ($that,$item,%opts) = @_;
557 0         0 return $that->new([$item],[],[],%opts);
558             }
559              
560             sub Append {
561 4     4   11 my ($obj,$nextItem,$nextDist,$nextDistOp) = @_;
562 4   100     30 $nextDistOp ||= '<';
563 4 50       6 if (@{$obj->{Items}}) {
  4         29  
564 4         6 push(@{$obj->{Dists}}, $nextDist);
  4         10  
565 4         6 push(@{$obj->{DistOps}}, $nextDistOp);
  4         8  
566             }
567 4         7 push(@{$obj->{Items}}, $nextItem);
  4         12  
568             }
569              
570 0     0     sub Clear { @{$_[0]{Items}} = @{$_[0]{Dists}} = @{$_[0]{DistOps}} = qw(); }
  0            
  0            
  0            
571              
572 0   0 0     sub Children { return $_[0]{Items}||[]; }
573             sub GetMatchId {
574 0     0     my ($id);
575 0 0         foreach (@{$_[0]{Items}||[]}) {
  0            
576 0 0 0       return $id if ($_ && ($id=$_->GetMatchId));
577             }
578 0           return 0;
579             }
580             sub SetMatchId {
581 0     0     my ($obj,$id) = @_;
582 0 0         foreach (@{$_[0]{Items}||[]}) {
  0            
583 0 0         $_->SetMatchId($id) if (UNIVERSAL::can($_,'SetMatchId'));
584             }
585 0           return $id;
586             }
587              
588             sub toString {
589 0     0     my $obj = shift;
590             return $obj->NegString('"'
591             .join(' ',
592             map {
593             ($obj->{Items}[$_]->toString,
594             ($_ < $#{$obj->{Items}} && ($obj->{Dists}[$_] || ($obj->{DistOps}[$_]||'<') ne '<')
595 0 0 0       ? ("#".($obj->{DistOps}[$_]||'<').($obj->{Dists}[$_]||'0'))
      0        
      0        
596             : qw()))
597 0           } (0..$#{$obj->{Items}}))
  0            
598             .'"');
599             }
600              
601              
602              
603             1; ##-- be happy
604              
605             =pod
606              
607             =head1 NAME
608              
609             DDC::PP::CQuery - pure-perl implementation of DDC::XS::CQuery
610              
611             =head1 SYNOPSIS
612              
613             use DDC::PP::CQuery;
614             #... stuff happens ...
615              
616              
617             =head1 DESCRIPTION
618              
619             The DDC::PP::CQuery class is a pure-perl fork of the L class,
620             which see for details.
621              
622             =head1 SEE ALSO
623              
624             perl(1),
625             DDC::PP(3perl),
626             DDC::XS(3perl).
627              
628             =head1 AUTHOR
629              
630             Bryan Jurish Emoocow@cpan.orgE
631              
632             =head1 COPYRIGHT AND LICENSE
633              
634             Copyright (C) 2016 by Bryan Jurish
635              
636             This library is free software; you can redistribute it and/or modify
637             it under the same terms as Perl itself, either Perl version 5.14.2 or,
638             at your option, any later version of Perl 5 you may have available.
639              
640             =cut
641