28 #include <type_traits>
55 bool match(
const Properties&,
const Ex::iterator&,
bool ignore_parent_rel=
false,
bool ignore_properties=
false)
const;
70 typedef std::pair<std::string, Ex::iterator>
kvpair_t;
159 virtual void latex(std::ostream&)
const;
161 virtual std::string
name()
const=0;
207 virtual std::string
name()
const
209 return std::string(
"Stay Away");
218 virtual std::string
name()
const
220 return std::string(
"PropertyInherit");
242 typedef internal_property_map_t::iterator
iterator;
262 typedef std::multimap<nset_t::iterator, pat_prop_pair_t, nset_it_less>
property_map_t;
280 template<
class T>
const T*
get(Ex::iterator,
bool ignore_parent_rel=
false)
const;
281 template<
class T>
const T*
get(Ex::iterator,
int& serialnum,
bool doserial=
true,
bool ignore_parent_rel=
false)
const;
283 template<
class T>
const T*
get(Ex::iterator,
const std::string& label)
const;
284 template<
class T>
const T*
get(Ex::iterator,
int& serialnum,
const std::string& label,
bool doserial=
true)
const;
286 template<
class T>
const T*
get(Ex::iterator, Ex::iterator,
bool ignore_parent_rel=
false)
const;
287 template<
class T>
const T*
get(Ex::iterator, Ex::iterator,
int&,
int&,
bool ignore_parent_rel=
false)
const;
293 std::pair<const T*, const pattern *>
get_with_pattern(Ex::iterator,
int& serialnum,
294 const std::string& label,
295 bool doserial=
true,
bool ignore_parent_rel=
false)
const;
299 const std::string& label,
300 bool doserial=
true,
bool ignore_parent_rel=
false)
const;
304 template<
class T> Ex::iterator
head(Ex::iterator,
bool ignore_parent_rel=
false)
const;
335 return get<T>(it, tmp,
false, ignore_parent_rel);
339 const T*
Properties::get(Ex::iterator it,
int& serialnum,
bool doserial,
bool ignore_parent_rel)
const
341 auto ret = get_with_pattern<T>(it, serialnum,
"", doserial, ignore_parent_rel);
347 bool doserial,
bool ignore_parent_rel)
const
351 auto ret = get_with_pattern_ext<T>(it, *compptr, serialnum, label, doserial, ignore_parent_rel);
358 int& serialnum,
const std::string& label,
359 bool doserial,
bool ignore_parent_rel)
const
361 std::pair<const T*, const pattern *> ret;
368 std::pair<property_map_t::const_iterator, property_map_t::const_iterator> pit=
props.equal_range(it->name_only());
373 bool wildcards=
false;
378 bool ignore_properties=
false;
379 if(std::is_same<T, Accent>::value)
380 ignore_properties=
true;
383 property_map_t::const_iterator walk=pit.first;
384 while(walk!=pit.second) {
385 if(wildcards==(*walk).second.first->children_wildcard()) {
387 ret.first=
dynamic_cast<const T *
>((*walk).second.second);
389 if((*walk).second.first->match_ext(*
this, it, comp, ignore_parent_rel, ignore_properties)) {
390 ret.second=(*walk).second.first;
395 serialnum=
serial_number( (*walk).second.second, (*walk).second.first );
403 else if(
dynamic_cast<const Inherit<T> *
>((*walk).second.second))
408 if(!wildcards && !ret.first) {
417 if(std::is_same<T, LaTeXForm>::value)
421 if(!ret.first && inherits) {
423 Ex::sibling_iterator sib=it.begin();
424 while(sib!=it.end()) {
425 std::pair<const T*, const pattern *> tmp=get_with_pattern<T>((Ex::iterator)(sib), serialnum, label, doserial);
439 const T* Properties::get(Ex::iterator it,
const std::string& label)
const
442 return get<T>(it, tmp, label,
false);
446 const T* Properties::get(Ex::iterator it,
int& serialnum,
const std::string& label,
bool doserial)
const
448 auto ret=get_with_pattern<T>(it, serialnum, label, doserial,
false);
453 const T* Properties::get(Ex::iterator it1, Ex::iterator it2,
bool ignore_parent_rel)
const
456 return get<T>(it1,it2,tmp1,tmp2, ignore_parent_rel);
460 const T* Properties::get(Ex::iterator it1, Ex::iterator it2,
int& serialnum1,
int& serialnum2,
bool ignore_parent_rel)
const
466 bool inherits1=
false, inherits2=
false;
467 std::pair<property_map_t::const_iterator, property_map_t::const_iterator> pit1=props.equal_range(it1->name_only());
468 std::pair<property_map_t::const_iterator, property_map_t::const_iterator> pit2=props.equal_range(it2->name_only());
470 property_map_t::const_iterator walk1=pit1.first;
471 while(walk1!=pit1.second) {
472 if((*walk1).second.first->match(*
this, it1, ignore_parent_rel)) {
473 ret1=
dynamic_cast<const T *
>((*walk1).second.second);
475 property_map_t::const_iterator walk2=pit2.first;
476 while(walk2!=pit2.second) {
477 if((*walk2).second.first->match(*
this, it2, ignore_parent_rel)) {
478 ret2=
dynamic_cast<const T *
>((*walk2).second.second);
480 if(ret1==ret2 && walk1!=walk2) {
481 serialnum1=serial_number( (*walk1).second.second, (*walk1).second.first );
482 serialnum2=serial_number( (*walk2).second.second, (*walk2).second.first );
500 if(!found && (inherits1 || inherits2)) {
501 Ex::sibling_iterator sib1, sib2;
502 if(inherits1) sib1=it1.begin();
504 bool keepgoing1=
true;
506 bool keepgoing2=
true;
507 if(inherits2) sib2=it2.begin();
510 const T* tmp=get<T>((Ex::iterator)(sib1), (Ex::iterator)(sib2), serialnum1, serialnum2, ignore_parent_rel);
516 if(!inherits2 || ++sib2==it2.end())
520 if(!inherits1 || ++sib1==it1.end())
532 Ex::iterator Properties::head(Ex::iterator it,
bool ignore_parent_rel)
const
536 if(get<PropertyInherit>(dn, ignore_parent_rel)) {