This documentation is automatically generated by online-judge-tools/verification-helper
#include "data-structure/link-cut-tree/lazy-reversible-bbst.hpp"
#pragma once
/**
* Author: Teetat Info.
* Date: 2024-04-13
* Description: Lazy Reversible BBST Base.
*/
template<class Tree,class Node,class MonoidAction>
struct LazyReversibleBBST:Tree{
using Tree::merge;
using Tree::split;
using typename Tree::Ptr;
using InfoMonoid = typename MonoidAction::InfoMonoid;
using TagMonoid = typename MonoidAction::TagMonoid;
using Info = typename MonoidAction::Info;
using Tag = typename MonoidAction::Tag;
LazyReversibleBBST()=default;
Info sum(Ptr t){
return t?t->sum:InfoMonoid::unit();
}
void pull(Ptr t){
if(!t)return;
push(t);
t->size=1;
t->sum=t->val;
t->revsum=t->val;
if(t->l){
t->size+=t->l->size;
t->sum=InfoMonoid::op(t->l->sum,t->sum);
t->revsum=InfoMonoid::op(t->revsum,t->l->revsum);
}
if(t->r){
t->size+=t->r->size;
t->sum=InfoMonoid::op(t->sum,t->r->sum);
t->revsum=InfoMonoid::op(t->r->revsum,t->revsum);
}
}
void push(Ptr t){
if(!t)return;
if(t->rev){
toggle(t->l);
toggle(t->r);
t->rev=false;
}
if(t->lz!=TagMonoid::unit()){
propagate(t->l,t->lz);
propagate(t->r,t->lz);
t->lz=TagMonoid::unit();
}
}
void toggle(Ptr t){
if(!t)return;
swap(t->l,t->r);
swap(t->sum,t->revsum);
t->rev^=true;
}
void propagate(Ptr t,const Tag &v){
if(!t)return;
t->val=MonoidAction::op(t->val,v);
t->sum=MonoidAction::op(t->sum,v);
t->revsum=MonoidAction::op(t->revsum,v);
t->lz=TagMonoid::op(t->lz,v);
}
void apply(Ptr &t,int l,int r,const Tag &v){
if(l>r)return;
auto x=split(t,l);
auto y=split(x.second,r-l+1);
propagate(y.first,v);
t=merge(x.first,merge(y.first,y.second));
}
Info query(Ptr &t,int l,int r){
if(l>r)return InfoMonoid::unit();
auto x=split(t,l);
auto y=split(x.second,r-l+1);
Info res=sum(y.first);
t=merge(x.first,merge(y.first,y.second));
return res;
}
void reverse(Ptr &t,int l,int r){
if(l>r)return;
auto x=split(t,l);
auto y=split(x.second,r-l+1);
toggle(y.first);
t=merge(x.first,merge(y.first,y.second));
}
};
#line 2 "data-structure/link-cut-tree/lazy-reversible-bbst.hpp"
/**
* Author: Teetat Info.
* Date: 2024-04-13
* Description: Lazy Reversible BBST Base.
*/
template<class Tree,class Node,class MonoidAction>
struct LazyReversibleBBST:Tree{
using Tree::merge;
using Tree::split;
using typename Tree::Ptr;
using InfoMonoid = typename MonoidAction::InfoMonoid;
using TagMonoid = typename MonoidAction::TagMonoid;
using Info = typename MonoidAction::Info;
using Tag = typename MonoidAction::Tag;
LazyReversibleBBST()=default;
Info sum(Ptr t){
return t?t->sum:InfoMonoid::unit();
}
void pull(Ptr t){
if(!t)return;
push(t);
t->size=1;
t->sum=t->val;
t->revsum=t->val;
if(t->l){
t->size+=t->l->size;
t->sum=InfoMonoid::op(t->l->sum,t->sum);
t->revsum=InfoMonoid::op(t->revsum,t->l->revsum);
}
if(t->r){
t->size+=t->r->size;
t->sum=InfoMonoid::op(t->sum,t->r->sum);
t->revsum=InfoMonoid::op(t->r->revsum,t->revsum);
}
}
void push(Ptr t){
if(!t)return;
if(t->rev){
toggle(t->l);
toggle(t->r);
t->rev=false;
}
if(t->lz!=TagMonoid::unit()){
propagate(t->l,t->lz);
propagate(t->r,t->lz);
t->lz=TagMonoid::unit();
}
}
void toggle(Ptr t){
if(!t)return;
swap(t->l,t->r);
swap(t->sum,t->revsum);
t->rev^=true;
}
void propagate(Ptr t,const Tag &v){
if(!t)return;
t->val=MonoidAction::op(t->val,v);
t->sum=MonoidAction::op(t->sum,v);
t->revsum=MonoidAction::op(t->revsum,v);
t->lz=TagMonoid::op(t->lz,v);
}
void apply(Ptr &t,int l,int r,const Tag &v){
if(l>r)return;
auto x=split(t,l);
auto y=split(x.second,r-l+1);
propagate(y.first,v);
t=merge(x.first,merge(y.first,y.second));
}
Info query(Ptr &t,int l,int r){
if(l>r)return InfoMonoid::unit();
auto x=split(t,l);
auto y=split(x.second,r-l+1);
Info res=sum(y.first);
t=merge(x.first,merge(y.first,y.second));
return res;
}
void reverse(Ptr &t,int l,int r){
if(l>r)return;
auto x=split(t,l);
auto y=split(x.second,r-l+1);
toggle(y.first);
t=merge(x.first,merge(y.first,y.second));
}
};