2022-10-21 16:06:44 +02:00
import {
AfterViewInit ,
ChangeDetectorRef ,
Component ,
ContentChildren ,
ElementRef , EventEmitter ,
Input , OnDestroy , Output ,
QueryList ,
2024-03-28 12:22:21 +01:00
ViewChild ,
Inject ,
PLATFORM_ID
2022-10-21 16:06:44 +02:00
} from "@angular/core" ;
import { SliderTabComponent } from "./slider-tab.component" ;
import { ActivatedRoute , Router } from "@angular/router" ;
import { Subscription } from "rxjs" ;
2024-03-28 12:22:21 +01:00
import { isPlatformServer } from "@angular/common" ;
2022-10-21 16:06:44 +02:00
import Timeout = NodeJS . Timeout ;
declare var UIkit ;
@Component ( {
selector : 'slider-tabs' ,
template : `
2024-09-09 23:25:40 +02:00
< ng - content / >
2024-07-25 09:31:08 +02:00
< div # sliderElement class = "uk-position-relative" [ class.uk - slider ] = " position = = = ' horizontal ' "
[ ngClass ] = "customClass" >
< div [ class.uk - slider - container - tabs ] = " position = = = ' horizontal ' " [ class.uk - border - bottom ] = " border & & position = = = ' horizontal ' " [ ngClass ] = " containerClass " >
< ul # tabsElement [ class.uk - flex - nowrap ] = " position = = = ' horizontal ' "
[ class . uk - slider - items ] = "position === 'horizontal'"
[ class . uk - tab - left ] = "position === 'left'" [ class . uk - tab - right ] = "position === 'right'"
[ attr . uk - switcher ] = "type === 'static'?('connect:' + connect):null"
[ ngClass ] = "'uk-flex-' + flexPosition + ' ' + tabsClass" >
< ng - container * ngIf = "type === 'static'" >
< li * ngFor = "let tab of leftTabs" [ class.uk - invisible ] = " tab.invisible " [ style.max - width ] = " ( position = = = ' horizontal ' ) ? ' 50 % ' : null " class = "uk-text-capitalize uk-text-truncate uk-display-block" >
< a [ class.uk - disabled ] = " tab.invisible | | tab.disabled " >
< span * ngIf = "tab.title" > { { tab . title } } < / span >
< ng - container * ngTemplateOutlet = "tab.tabTemplate" > < / n g - c o n t a i n e r >
< / a >
< / li >
< li * ngFor = "let tab of rightTabs; let i=index;" [ class.uk - invisible ] = " tab.invisible " [ style.max - width ] = " ( position = = = ' horizontal ' ) ? ' 50 % ' : null " [ ngClass ] = " i = = = 0 ? ' uk - flex - 1 uk - flex uk - flex - right ' : ' ' "
class = "uk-text-capitalize uk-text-truncate uk-display-block" >
< a [ class.uk - disabled ] = " tab.invisible | | tab.disabled " [ ngClass ] = " tab.customClass " >
< span * ngIf = "tab.title" > { { tab . title } } < / span >
< ng - container * ngTemplateOutlet = "tab.tabTemplate" > < / n g - c o n t a i n e r >
< / a >
< / li >
< / n g - c o n t a i n e r >
< ng - container * ngIf = "type === 'dynamic'" >
< li * ngFor = "let tab of leftTabs; let i=index;" [ class.uk - invisible ] = " tab.invisible " [ class.uk - active ] = " tab.active " [ style.max - width ] = " ( position = = = ' horizontal ' ) ? ' 50 % ' : null " >
< a [ class.uk - disabled ] = " tab.invisible | | tab.disabled " [ routerLink ] = " tab.routerLink " [ queryParams ] = " tab.queryParams " [ ngClass ] = " tab.customClass " [ relativeTo ] = " tab.relativeTo "
( click ) = "showActive(i)"
class = "uk-text-capitalize uk-text-truncate uk-display-block" >
< span * ngIf = "tab.title" > { { tab . title } } < / span >
< ng - container * ngTemplateOutlet = "tab.tabTemplate" > < / n g - c o n t a i n e r >
< / a >
< / li >
< li * ngFor = "let tab of rightTabs; let i=index;" [ class.uk - invisible ] = " tab.invisible " [ style.max - width ] = " ( position = = = ' horizontal ' ) ? ' 50 % ' : null " [ class.uk - active ] = " tab.active "
[ ngClass ] = "i === 0?'uk-flex-1 uk-flex uk-flex-right':''" >
< a [ class.uk - disabled ] = " tab.invisible | | tab.disabled " [ routerLink ] = " tab.routerLink " [ queryParams ] = " tab.queryParams " [ ngClass ] = " tab.customClass " [ relativeTo ] = " tab.relativeTo "
( click ) = "showActive(i)"
class = "uk-text-capitalize uk-text-truncate uk-display-block" >
< span * ngIf = "tab.title" > { { tab . title } } < / span >
< ng - container * ngTemplateOutlet = "tab.tabTemplate" > < / n g - c o n t a i n e r >
< / a >
< / li >
< / n g - c o n t a i n e r >
< ng - container * ngIf = "type === 'scrollable'" >
< li * ngFor = "let tab of leftTabs" [ class.uk - invisible ] = " tab.invisible " [ style.max - width ] = " ( position = = = ' horizontal ' ) ? ' 50 % ' : null " class = "uk-text-capitalize uk-text-truncate uk-display-block" [ class.uk - active ] = " tab.active " >
< a [ class.uk - disabled ] = " tab.invisible | | tab.disabled " routerLink = "./" [ fragment ] = " tab.id " queryParamsHandling = "merge" [ ngClass ] = " tab.customClass " >
< span * ngIf = "tab.title" > { { tab . title } } < / span >
< ng - container * ngTemplateOutlet = "tab.tabTemplate" > < / n g - c o n t a i n e r >
< / a >
< / li >
< li * ngFor = "let tab of rightTabs; let i=index;" [ class.uk - invisible ] = " tab.invisible " [ style.max - width ] = " ( position = = = ' horizontal ' ) ? ' 50 % ' : null " class = "uk-text-capitalize uk-text-truncate uk-display-block"
[ ngClass ] = "i === 0?'uk-flex-1 uk-flex uk-flex-right':''"
[ class . uk - active ] = "tab.active" >
< a [ class.uk - disabled ] = " tab.invisible | | tab.disabled " routerLink = "./" [ fragment ] = " tab.id " queryParamsHandling = "merge" [ ngClass ] = " tab.customClass " >
< span * ngIf = "tab.title" > { { tab . title } } < / span >
< ng - container * ngTemplateOutlet = "tab.tabTemplate" > < / n g - c o n t a i n e r >
< / a >
< / li >
< / n g - c o n t a i n e r >
< / ul >
< / div >
< ng - container * ngIf = "!isServer" >
< slider - arrow * ngIf = "position === 'horizontal' && arrows" type = "previous" > < / s l i d e r - a r r o w >
< slider - arrow * ngIf = "position === 'horizontal' && arrows" type = "next" > < / s l i d e r - a r r o w >
2023-02-09 11:09:49 +01:00
< / n g - c o n t a i n e r >
2022-10-21 16:06:44 +02:00
< / div >
` ,
} )
export class SliderTabsComponent implements AfterViewInit , OnDestroy {
/ * *
2023-02-09 11:09:49 +01:00
* Type of tabs :
* Static = Uikit tabs with @connect class or selector
* Dynamic = Active is defined by tabComponent ' s active Input
2022-10-21 16:06:44 +02:00
* Scrollable = Active is defined by the active fragment of URL and position of scroll
* * /
@Input ( )
public type : 'static' | 'dynamic' | 'scrollable' = 'static' ;
/ * *
2022-10-24 15:37:05 +02:00
* Connect selector in static type . Default : .uk - switcher
2022-10-21 16:06:44 +02:00
* * /
@Input ( )
2022-10-24 15:37:05 +02:00
public connect = '.uk-switcher' ;
2022-10-21 16:06:44 +02:00
/ * *
* Threshold between 0.0 to 1.0 for Intersection Observer
* * /
@Input ( )
public scrollThreshold = 0.1 ;
/ * *
* Tabs view : Horizontal is the default .
* * /
@Input ( )
public position : 'horizontal' | 'left' | 'right' = 'horizontal' ;
2022-10-24 15:37:05 +02:00
/ * *
* Tabs flex position : Left is the default .
* * /
@Input ( )
public flexPosition : 'center' | 'left' | 'right' = 'left' ;
2024-03-19 11:55:39 +01:00
/ * *
* Set a class for the container
* * /
@Input ( )
public containerClass : string ;
2022-10-21 16:06:44 +02:00
/ * *
2023-02-09 11:09:49 +01:00
* Set a class above tabs
2022-10-21 16:06:44 +02:00
* * /
@Input ( )
2022-10-24 11:57:52 +02:00
public customClass : string ;
2023-02-09 11:09:49 +01:00
/ * *
* Tabs class
* * /
@Input ( )
public tabsClass : string = 'uk-tab' ;
@Input ( )
public border : boolean = true ;
@Input ( )
public arrows : boolean = true ;
2022-10-21 16:06:44 +02:00
@ContentChildren ( SliderTabComponent ) tabs : QueryList < SliderTabComponent > ;
2022-10-24 15:37:05 +02:00
@ViewChild ( 'sliderElement' ) sliderElement : ElementRef ;
2022-10-21 16:06:44 +02:00
@ViewChild ( 'tabsElement' ) tabsElement : ElementRef ;
2023-02-15 10:53:25 +01:00
private slider ;
2022-10-21 16:06:44 +02:00
/ * *
* Notify regarding new active element
* * /
2024-05-30 12:59:07 +02:00
@Output ( ) activeEmitter : EventEmitter < string | number > = new EventEmitter < number > ( ) ;
2022-10-24 15:37:05 +02:00
private activeIndex : number = 0 ;
2022-10-21 16:06:44 +02:00
private subscriptions : any [ ] = [ ] ;
private observer : IntersectionObserver ;
private timeout : Timeout ;
2024-03-28 12:22:21 +01:00
isServer : boolean ;
2024-07-25 09:31:08 +02:00
2022-10-21 16:06:44 +02:00
constructor ( private route : ActivatedRoute ,
private router : Router ,
2024-03-28 12:22:21 +01:00
private cdr : ChangeDetectorRef ,
@Inject ( PLATFORM_ID ) private platform : any ) {
this . isServer = isPlatformServer ( this . platform ) ;
2022-10-21 16:06:44 +02:00
}
2024-07-25 09:31:08 +02:00
2022-10-21 16:06:44 +02:00
ngAfterViewInit() {
2023-07-19 13:09:10 +02:00
this . initTabs ( ) ;
this . tabs . changes . subscribe ( ( ) = > {
2024-07-25 09:31:08 +02:00
this . initTabs ( ) ;
2023-07-19 13:09:10 +02:00
} ) ;
}
public initTabs() {
if ( typeof document !== 'undefined' && this . tabs . length > 0 ) {
setTimeout ( ( ) = > {
if ( this . position === 'horizontal' ) {
this . slider = UIkit . slider ( this . sliderElement . nativeElement , { finite : true } ) ;
this . slider . clsActive = 'uk-slider-active' ;
this . slider . updateActiveClasses ( ) ;
this . slider . slides . forEach ( ( item , index ) = > {
if ( ! this . tabs . get ( index ) . active ) {
item . classList . remove ( 'uk-active' ) ;
2023-02-09 11:09:49 +01:00
}
2023-07-19 13:09:10 +02:00
} ) ;
if ( this . type === 'static' ) {
2024-03-19 11:55:39 +01:00
let tabs = UIkit . switcher ( this . tabsElement . nativeElement , { connect : this.connect } ) ;
2023-07-19 13:09:10 +02:00
tabs . show ( this . activeIndex ) ;
if ( this . connect . includes ( '#' ) ) {
this . scrollToStart ( ) ;
}
} else if ( this . type == 'dynamic' ) {
this . activeIndex = this . tabs . toArray ( ) . findIndex ( tab = > tab . active ) ;
this . slider . show ( this . activeIndex ) ;
} else if ( this . type === 'scrollable' ) {
this . scrollable ( this . slider ) ;
2022-10-21 16:06:44 +02:00
}
2023-07-19 13:09:10 +02:00
} else {
this . scrollable ( ) ;
}
} ) ;
}
2022-10-21 16:06:44 +02:00
}
2024-07-25 09:31:08 +02:00
2022-10-24 15:37:05 +02:00
private scrollToStart() {
this . subscriptions . push ( UIkit . util . on ( this . connect , 'shown' , ( event ) : void = > {
let index = event . detail [ 0 ] . index ( ) ;
if ( index !== this . activeIndex ) {
this . activeIndex = index ;
2023-01-27 12:54:40 +01:00
this . router . navigate ( [ './' ] , { relativeTo : this.route , fragment : this.connect.replace ( '#' , '' ) , queryParamsHandling : "merge" } ) ;
2022-10-24 15:37:05 +02:00
}
} ) ) ;
}
2024-07-25 09:31:08 +02:00
2022-10-21 16:06:44 +02:00
private scrollable ( slider = null ) {
this . activeFragment ( this . route . snapshot . fragment , slider ) ;
this . subscriptions . push ( this . route . fragment . subscribe ( fragment = > {
2022-10-24 15:37:05 +02:00
this . activeFragment ( fragment , slider ) ;
2022-10-21 16:06:44 +02:00
} ) ) ;
this . setObserver ( ) ;
}
2024-07-25 09:31:08 +02:00
2022-10-21 16:06:44 +02:00
private setObserver() {
if ( this . observer ) {
this . observer . disconnect ( ) ;
}
this . observer = new IntersectionObserver ( ( entries ) = > {
entries . forEach ( entry = > {
if ( entry . isIntersecting ) {
if ( this . timeout ) {
clearTimeout ( this . timeout ) ;
}
this . timeout = setTimeout ( ( ) = > {
this . router . navigate ( [ './' ] , {
fragment : entry.target.id ,
relativeTo : this.route ,
2023-01-27 12:54:40 +01:00
state : { disableScroll : true } ,
queryParamsHandling : 'merge'
2022-10-21 16:06:44 +02:00
} ) ;
} , 200 ) ;
}
} ) ;
2024-06-18 13:43:14 +02:00
} , { threshold : this.scrollThreshold } ) ;
2022-10-21 16:06:44 +02:00
this . tabs . forEach ( tab = > {
2024-05-30 12:59:07 +02:00
let element = document . getElementById ( tab . id . toString ( ) ) ;
2022-10-21 16:06:44 +02:00
if ( element ) {
this . observer . observe ( element ) ;
}
} ) ;
}
2023-02-15 10:53:25 +01:00
public showActive ( index ) {
this . activeIndex = index ;
2023-03-27 13:59:17 +02:00
this . activeEmitter . emit ( this . tabs . get ( this . activeIndex ) . id ) ;
2023-02-15 10:53:25 +01:00
if ( this . slider ) {
this . slider . show ( this . activeIndex ) ;
}
}
2024-07-25 09:31:08 +02:00
2022-10-21 16:06:44 +02:00
private activeFragment ( fragment , slider ) {
let index = 0 ;
if ( fragment ) {
index = this . tabs . toArray ( ) . findIndex ( item = > item . id == fragment ) ;
}
2022-10-24 15:37:05 +02:00
if ( slider ) {
2022-10-21 16:06:44 +02:00
slider . show ( index ) ;
}
this . tabs . forEach ( ( tab , i ) = > {
if ( index === i ) {
tab . active = true ;
this . activeEmitter . emit ( tab . id ) ;
} else {
tab . active = false ;
}
} ) ;
this . cdr . detectChanges ( ) ;
}
2024-07-25 09:31:08 +02:00
2023-01-27 12:54:40 +01:00
get leftTabs ( ) : SliderTabComponent [ ] {
return this . tabs . toArray ( ) . filter ( tab = > tab . align === 'left' ) ;
}
2024-07-25 09:31:08 +02:00
2023-01-27 12:54:40 +01:00
get rightTabs ( ) : SliderTabComponent [ ] {
return this . tabs . toArray ( ) . filter ( tab = > tab . align === 'right' ) ;
}
2024-07-25 09:31:08 +02:00
2022-10-21 16:06:44 +02:00
ngOnDestroy() {
this . subscriptions . forEach ( subscription = > {
if ( subscription instanceof Subscription ) {
subscription . unsubscribe ( ) ;
2022-10-24 15:37:05 +02:00
} else if ( subscription instanceof Function ) {
subscription ( ) ;
2022-10-21 16:06:44 +02:00
}
} ) ;
if ( this . observer ) {
this . observer . disconnect ( ) ;
}
}
}