ngx-fixed-footer 2.0.0 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +16 -15
- package/esm2022/lib/ngx-fixed-footer.directive.mjs +52 -61
- package/esm2022/lib/ngx-fixed-footer.interface.mjs +1 -1
- package/esm2022/lib/ngx-fixed-footer.provider.mjs +1 -1
- package/fesm2022/ngx-fixed-footer.mjs +51 -60
- package/fesm2022/ngx-fixed-footer.mjs.map +1 -1
- package/lib/ngx-fixed-footer.directive.d.ts +12 -12
- package/lib/ngx-fixed-footer.interface.d.ts +2 -2
- package/lib/ngx-fixed-footer.provider.d.ts +1 -1
- package/package.json +4 -4
package/README.md
CHANGED
|
@@ -14,7 +14,7 @@
|
|
|
14
14
|
|
|
15
15
|
> Angular directive that adds fixed footer without overlap
|
|
16
16
|
|
|
17
|
-
> ✓ _Angular
|
|
17
|
+
> ✓ _Angular 18 compatible_
|
|
18
18
|
|
|
19
19
|
Here's the [demo](http://celtian.github.io/ngx-fixed-footer/) or [stackblitz live preview](https://stackblitz.com/edit/ngx-fixed-footer) or [codesandbox live preview](https://codesandbox.io/s/ngx-fixed-footer-m4f21)
|
|
20
20
|
|
|
@@ -22,7 +22,7 @@ Here's the [demo](http://celtian.github.io/ngx-fixed-footer/) or [stackblitz liv
|
|
|
22
22
|
- No dependencies!
|
|
23
23
|
- Directive way
|
|
24
24
|
|
|
25
|
-
## Install
|
|
25
|
+
## 🛠️ Install
|
|
26
26
|
|
|
27
27
|
1. Use yarn (or npm) to install the package
|
|
28
28
|
|
|
@@ -90,15 +90,7 @@ yarn add ngx-fixed-footer
|
|
|
90
90
|
})
|
|
91
91
|
```
|
|
92
92
|
|
|
93
|
-
##
|
|
94
|
-
|
|
95
|
-
| Angular | ngx-fixed-footer | Install |
|
|
96
|
-
| --------- | ---------------- | ----------------------------- |
|
|
97
|
-
| >= 14 | 2.x | `yarn add ngx-fixed-footer` |
|
|
98
|
-
| >= 12 | 1.x | `yarn add ngx-fixed-footer@1` |
|
|
99
|
-
| >= 5 < 13 | 0.x | `yarn add ngx-fixed-footer@0` |
|
|
100
|
-
|
|
101
|
-
## Quick start
|
|
93
|
+
## 🚀 Quick start
|
|
102
94
|
|
|
103
95
|
### Example code
|
|
104
96
|
|
|
@@ -114,7 +106,7 @@ yarn add ngx-fixed-footer
|
|
|
114
106
|
<div ngxFixedFooter [containerSelector]="[role='main']" [cssAttribute]="'padding'">... some content here</div>
|
|
115
107
|
```
|
|
116
108
|
|
|
117
|
-
## Options
|
|
109
|
+
## 🛠️ Options
|
|
118
110
|
|
|
119
111
|
### Root options
|
|
120
112
|
|
|
@@ -134,13 +126,22 @@ _Each directive can override global options._
|
|
|
134
126
|
| **[containerSelector]** | string | value taken from root options | Css selector used for additional padding/margin |
|
|
135
127
|
| **[cssAttribute]** | 'margin' or 'padding' | value taken from root options | Css attribute used on 'containerSelector' |
|
|
136
128
|
|
|
137
|
-
##
|
|
129
|
+
## 🔧 Compatibility
|
|
130
|
+
|
|
131
|
+
| Angular | ngx-fixed-footer | Install |
|
|
132
|
+
| --------- | ---------------- | ----------------------------- |
|
|
133
|
+
| >= 18 | 3.x | `yarn add ngx-fixed-footer` |
|
|
134
|
+
| >= 14 | 2.x | `yarn add ngx-fixed-footer@2` |
|
|
135
|
+
| >= 12 | 1.x | `yarn add ngx-fixed-footer@1` |
|
|
136
|
+
| >= 5 < 13 | 0.x | `yarn add ngx-fixed-footer@0` |
|
|
137
|
+
|
|
138
|
+
## 📦 Dependencies
|
|
138
139
|
|
|
139
140
|
_None_
|
|
140
141
|
|
|
141
|
-
## License
|
|
142
|
+
## 🪪 License
|
|
142
143
|
|
|
143
|
-
Copyright © 2021 -
|
|
144
|
+
Copyright © 2021 - 2024 [Dominik Hladik](https://github.com/Celtian)
|
|
144
145
|
|
|
145
146
|
All contents are licensed under the [MIT license].
|
|
146
147
|
|
|
@@ -1,17 +1,50 @@
|
|
|
1
1
|
import { DOCUMENT } from '@angular/common';
|
|
2
|
-
import { Directive,
|
|
2
|
+
import { Directive, ElementRef, Renderer2, computed, effect, inject, input, signal } from '@angular/core';
|
|
3
3
|
import { DEFAULT_FIXED_FOOTER_OPTIONS } from './ngx-fixed-footer.constants';
|
|
4
4
|
import { APP_FIXED_FOOTER_OPTIONS_TOKEN } from './ngx-fixed-footer.provider';
|
|
5
5
|
import * as i0 from "@angular/core";
|
|
6
6
|
export class NgxFixedFooterDirective {
|
|
7
|
-
constructor(
|
|
8
|
-
this.document =
|
|
9
|
-
this.el =
|
|
10
|
-
this.render =
|
|
7
|
+
constructor() {
|
|
8
|
+
this.document = inject(DOCUMENT);
|
|
9
|
+
this.el = inject(ElementRef);
|
|
10
|
+
this.render = inject(Renderer2);
|
|
11
11
|
this.options = inject(APP_FIXED_FOOTER_OPTIONS_TOKEN, { optional: true }) || DEFAULT_FIXED_FOOTER_OPTIONS;
|
|
12
|
-
this.
|
|
13
|
-
this.
|
|
14
|
-
this.
|
|
12
|
+
this.hasResizeObserver = typeof ResizeObserver !== 'undefined';
|
|
13
|
+
this.offsetHeight = signal(undefined);
|
|
14
|
+
this.resizeObserver = signal(undefined);
|
|
15
|
+
this.prevContainerSelector = signal(undefined);
|
|
16
|
+
this.containerSelector = input(this.options.containerSelector);
|
|
17
|
+
this.cssAttribute = input(this.options.cssAttribute);
|
|
18
|
+
this.container = computed(() => {
|
|
19
|
+
const selector = this.containerSelector() || this.options.containerSelector;
|
|
20
|
+
return this.document.body.querySelector(selector);
|
|
21
|
+
});
|
|
22
|
+
// swap selector
|
|
23
|
+
effect(() => {
|
|
24
|
+
if (!this.hasResizeObserver || !this.document)
|
|
25
|
+
return;
|
|
26
|
+
const cssAttribute = this.cssAttribute();
|
|
27
|
+
const containerSelector = this.containerSelector();
|
|
28
|
+
if (containerSelector) {
|
|
29
|
+
const prevContainerSelector = this.prevContainerSelector();
|
|
30
|
+
if (prevContainerSelector && prevContainerSelector !== containerSelector) {
|
|
31
|
+
this.removeStyle(this.document.body.querySelector(prevContainerSelector), cssAttribute);
|
|
32
|
+
}
|
|
33
|
+
this.setStyle(this.document.body.querySelector(containerSelector), cssAttribute, this.offsetHeight());
|
|
34
|
+
this.prevContainerSelector.set(containerSelector);
|
|
35
|
+
}
|
|
36
|
+
}, { allowSignalWrites: true });
|
|
37
|
+
// swap css attribute
|
|
38
|
+
effect(() => {
|
|
39
|
+
if (!this.hasResizeObserver || !this.document)
|
|
40
|
+
return;
|
|
41
|
+
const cssAttribute = this.cssAttribute();
|
|
42
|
+
if (cssAttribute) {
|
|
43
|
+
const container = this.container();
|
|
44
|
+
this.removeStyle(container, cssAttribute === 'padding' ? 'margin' : 'padding');
|
|
45
|
+
this.setStyle(container, cssAttribute, this.offsetHeight());
|
|
46
|
+
}
|
|
47
|
+
});
|
|
15
48
|
}
|
|
16
49
|
ngOnInit() {
|
|
17
50
|
if (this.hasResizeObserver && this.document) {
|
|
@@ -19,41 +52,17 @@ export class NgxFixedFooterDirective {
|
|
|
19
52
|
resizeObserver.observe(this.html);
|
|
20
53
|
}
|
|
21
54
|
}
|
|
22
|
-
ngOnChanges(changes) {
|
|
23
|
-
if (this.hasResizeObserver && this.document) {
|
|
24
|
-
// swap selector
|
|
25
|
-
if (changes?.containerSelector && !changes?.containerSelector?.firstChange) {
|
|
26
|
-
const prev = changes?.containerSelector?.previousValue;
|
|
27
|
-
const next = changes?.containerSelector?.currentValue;
|
|
28
|
-
if (next !== prev) {
|
|
29
|
-
this.removeStyle(this.document.body.querySelector(prev), this.cssAttribute);
|
|
30
|
-
this.setStyle(this.document.body.querySelector(next), this.cssAttribute, this.offsetHeight);
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
// swap css attribute
|
|
34
|
-
if (changes?.cssAttribute && !changes?.cssAttribute?.firstChange) {
|
|
35
|
-
const prev = changes?.cssAttribute?.previousValue;
|
|
36
|
-
const next = changes?.cssAttribute?.currentValue;
|
|
37
|
-
if (next !== prev) {
|
|
38
|
-
this.removeStyle(this.container, prev);
|
|
39
|
-
this.setStyle(this.container, next, this.offsetHeight);
|
|
40
|
-
}
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
}
|
|
44
55
|
ngOnDestroy() {
|
|
45
|
-
if (this.
|
|
46
|
-
this.removeStyle(this.container, this.cssAttribute);
|
|
47
|
-
|
|
48
|
-
this.resizeObserver.unobserve(this.html);
|
|
49
|
-
}
|
|
56
|
+
if (this.resizeObserver() && this.document) {
|
|
57
|
+
this.removeStyle(this.container(), this.cssAttribute());
|
|
58
|
+
this.resizeObserver().unobserve(this.html);
|
|
50
59
|
}
|
|
51
60
|
}
|
|
52
61
|
checkHeight() {
|
|
53
62
|
const height = this.html.offsetHeight;
|
|
54
|
-
if (this.offsetHeight !== height) {
|
|
55
|
-
this.setStyle(this.container, this.cssAttribute, height);
|
|
56
|
-
this.offsetHeight
|
|
63
|
+
if (this.offsetHeight() !== height) {
|
|
64
|
+
this.setStyle(this.container(), this.cssAttribute(), height);
|
|
65
|
+
this.offsetHeight.set(height);
|
|
57
66
|
}
|
|
58
67
|
}
|
|
59
68
|
removeStyle(container, cssAttribute) {
|
|
@@ -68,35 +77,17 @@ export class NgxFixedFooterDirective {
|
|
|
68
77
|
}
|
|
69
78
|
this.render.setStyle(container, `${cssAttribute}-bottom`, height === 0 ? '' : `${height}px`);
|
|
70
79
|
}
|
|
71
|
-
get container() {
|
|
72
|
-
const selector = this.containerSelector || this.options.containerSelector;
|
|
73
|
-
const container = this.document.body.querySelector(selector);
|
|
74
|
-
if (!container) {
|
|
75
|
-
console.warn(`Container '${selector}' was not found`);
|
|
76
|
-
}
|
|
77
|
-
return container;
|
|
78
|
-
}
|
|
79
80
|
get html() {
|
|
80
81
|
return this.el.nativeElement;
|
|
81
82
|
}
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
}
|
|
85
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: NgxFixedFooterDirective, deps: [{ token: DOCUMENT }, { token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
86
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.0.8", type: NgxFixedFooterDirective, isStandalone: true, selector: "[ngxFixedFooter]", inputs: { containerSelector: "containerSelector", cssAttribute: "cssAttribute" }, usesOnChanges: true, ngImport: i0 }); }
|
|
83
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: NgxFixedFooterDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
84
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.0.1", type: NgxFixedFooterDirective, isStandalone: true, selector: "[ngxFixedFooter]", inputs: { containerSelector: { classPropertyName: "containerSelector", publicName: "containerSelector", isSignal: true, isRequired: false, transformFunction: null }, cssAttribute: { classPropertyName: "cssAttribute", publicName: "cssAttribute", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0 }); }
|
|
87
85
|
}
|
|
88
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
86
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: NgxFixedFooterDirective, decorators: [{
|
|
89
87
|
type: Directive,
|
|
90
88
|
args: [{
|
|
91
89
|
selector: '[ngxFixedFooter]',
|
|
92
90
|
standalone: true
|
|
93
91
|
}]
|
|
94
|
-
}], ctorParameters: () => [
|
|
95
|
-
type: Inject,
|
|
96
|
-
args: [DOCUMENT]
|
|
97
|
-
}] }, { type: i0.ElementRef }, { type: i0.Renderer2 }], propDecorators: { containerSelector: [{
|
|
98
|
-
type: Input
|
|
99
|
-
}], cssAttribute: [{
|
|
100
|
-
type: Input
|
|
101
|
-
}] } });
|
|
102
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-fixed-footer.directive.js","sourceRoot":"","sources":["../../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n  Directive,\n  ElementRef,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Renderer2,\n  SimpleChanges,\n  inject\n} from '@angular/core';\nimport { DEFAULT_FIXED_FOOTER_OPTIONS } from './ngx-fixed-footer.constants';\nimport { NgxFixedFooterCssAttribute, NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\nimport { APP_FIXED_FOOTER_OPTIONS_TOKEN } from './ngx-fixed-footer.provider';\n\n@Directive({\n  selector: '[ngxFixedFooter]',\n  standalone: true\n})\nexport class NgxFixedFooterDirective implements OnDestroy, OnChanges, OnInit {\n  private options: NgxFixedFooterOptions =\n    inject(APP_FIXED_FOOTER_OPTIONS_TOKEN, { optional: true }) || DEFAULT_FIXED_FOOTER_OPTIONS;\n  private offsetHeight: number = undefined;\n  private resizeObserver: ResizeObserver;\n\n  @Input() public containerSelector: string = this.options.containerSelector;\n  @Input() public cssAttribute: NgxFixedFooterCssAttribute = this.options.cssAttribute;\n\n  constructor(\n    @Inject(DOCUMENT) private document: any,\n    private el: ElementRef,\n    private render: Renderer2\n  ) {}\n\n  public ngOnInit(): void {\n    if (this.hasResizeObserver && this.document) {\n      const resizeObserver = new ResizeObserver(() => this.checkHeight());\n      resizeObserver.observe(this.html);\n    }\n  }\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    if (this.hasResizeObserver && this.document) {\n      // swap selector\n      if (changes?.containerSelector && !changes?.containerSelector?.firstChange) {\n        const prev = changes?.containerSelector?.previousValue;\n        const next = changes?.containerSelector?.currentValue;\n        if (next !== prev) {\n          this.removeStyle(this.document.body.querySelector(prev), this.cssAttribute);\n          this.setStyle(this.document.body.querySelector(next), this.cssAttribute, this.offsetHeight);\n        }\n      }\n\n      // swap css attribute\n      if (changes?.cssAttribute && !changes?.cssAttribute?.firstChange) {\n        const prev = changes?.cssAttribute?.previousValue;\n        const next = changes?.cssAttribute?.currentValue;\n        if (next !== prev) {\n          this.removeStyle(this.container, prev);\n          this.setStyle(this.container, next, this.offsetHeight);\n        }\n      }\n    }\n  }\n\n  public ngOnDestroy(): void {\n    if (this.hasResizeObserver && this.document) {\n      this.removeStyle(this.container, this.cssAttribute);\n      if (this.resizeObserver) {\n        this.resizeObserver.unobserve(this.html);\n      }\n    }\n  }\n\n  private checkHeight(): void {\n    const height = this.html.offsetHeight;\n    if (this.offsetHeight !== height) {\n      this.setStyle(this.container, this.cssAttribute, height);\n      this.offsetHeight = height;\n    }\n  }\n\n  private removeStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute): void {\n    if (!container) {\n      throw new Error(`Cannot removeStyle to undefined container`);\n    }\n    this.render.setStyle(container, `${cssAttribute}-bottom`, '');\n  }\n\n  private setStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute, height: number): void {\n    if (!container) {\n      throw new Error(`Cannot setStyle to undefined container`);\n    }\n    this.render.setStyle(container, `${cssAttribute}-bottom`, height === 0 ? '' : `${height}px`);\n  }\n\n  private get container(): HTMLElement {\n    const selector = this.containerSelector || this.options.containerSelector;\n    const container = this.document.body.querySelector(selector);\n    if (!container) {\n      console.warn(`Container '${selector}' was not found`);\n    }\n    return container;\n  }\n\n  private get html(): HTMLElement {\n    return this.el.nativeElement;\n  }\n\n  private get hasResizeObserver(): boolean {\n    return typeof ResizeObserver !== 'undefined';\n  }\n}\n"]}
|
|
92
|
+
}], ctorParameters: () => [] });
|
|
93
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-fixed-footer.directive.js","sourceRoot":"","sources":["../../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n  Directive,\n  ElementRef,\n  OnDestroy,\n  OnInit,\n  Renderer2,\n  computed,\n  effect,\n  inject,\n  input,\n  signal\n} from '@angular/core';\nimport { DEFAULT_FIXED_FOOTER_OPTIONS } from './ngx-fixed-footer.constants';\nimport { NgxFixedFooterCssAttribute, NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\nimport { APP_FIXED_FOOTER_OPTIONS_TOKEN } from './ngx-fixed-footer.provider';\n\n@Directive({\n  selector: '[ngxFixedFooter]',\n  standalone: true\n})\nexport class NgxFixedFooterDirective implements OnDestroy, OnInit {\n  private readonly document = inject(DOCUMENT);\n  private readonly el = inject(ElementRef);\n  private readonly render = inject(Renderer2);\n  private options: NgxFixedFooterOptions =\n    inject(APP_FIXED_FOOTER_OPTIONS_TOKEN, { optional: true }) || DEFAULT_FIXED_FOOTER_OPTIONS;\n  private readonly hasResizeObserver = typeof ResizeObserver !== 'undefined';\n\n  private offsetHeight = signal<number | undefined>(undefined);\n  private resizeObserver = signal<ResizeObserver | undefined>(undefined);\n  private prevContainerSelector = signal<string | undefined>(undefined);\n\n  public containerSelector = input<string>(this.options.containerSelector);\n  public cssAttribute = input<NgxFixedFooterCssAttribute>(this.options.cssAttribute);\n\n  private container = computed<HTMLElement>(() => {\n    const selector = this.containerSelector() || this.options.containerSelector;\n    return this.document.body.querySelector<HTMLElement>(selector);\n  });\n\n  constructor() {\n    // swap selector\n    effect(\n      () => {\n        if (!this.hasResizeObserver || !this.document) return;\n        const cssAttribute = this.cssAttribute();\n        const containerSelector = this.containerSelector();\n        if (containerSelector) {\n          const prevContainerSelector = this.prevContainerSelector();\n          if (prevContainerSelector && prevContainerSelector !== containerSelector) {\n            this.removeStyle(this.document.body.querySelector(prevContainerSelector), cssAttribute);\n          }\n          this.setStyle(this.document.body.querySelector(containerSelector), cssAttribute, this.offsetHeight());\n          this.prevContainerSelector.set(containerSelector);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    // swap css attribute\n    effect(() => {\n      if (!this.hasResizeObserver || !this.document) return;\n      const cssAttribute = this.cssAttribute();\n      if (cssAttribute) {\n        const container = this.container();\n        this.removeStyle(container, cssAttribute === 'padding' ? 'margin' : 'padding');\n        this.setStyle(container, cssAttribute, this.offsetHeight());\n      }\n    });\n  }\n\n  public ngOnInit(): void {\n    if (this.hasResizeObserver && this.document) {\n      const resizeObserver = new ResizeObserver(() => this.checkHeight());\n      resizeObserver.observe(this.html);\n    }\n  }\n\n  public ngOnDestroy(): void {\n    if (this.resizeObserver() && this.document) {\n      this.removeStyle(this.container(), this.cssAttribute());\n      this.resizeObserver().unobserve(this.html);\n    }\n  }\n\n  private checkHeight(): void {\n    const height = this.html.offsetHeight;\n    if (this.offsetHeight() !== height) {\n      this.setStyle(this.container(), this.cssAttribute(), height);\n      this.offsetHeight.set(height);\n    }\n  }\n\n  private removeStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute): void {\n    if (!container) {\n      throw new Error(`Cannot removeStyle to undefined container`);\n    }\n    this.render.setStyle(container, `${cssAttribute}-bottom`, '');\n  }\n\n  private setStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute, height: number): void {\n    if (!container) {\n      throw new Error(`Cannot setStyle to undefined container`);\n    }\n    this.render.setStyle(container, `${cssAttribute}-bottom`, height === 0 ? '' : `${height}px`);\n  }\n\n  private get html(): HTMLElement {\n    return this.el.nativeElement;\n  }\n}\n"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmd4LWZpeGVkLWZvb3Rlci5pbnRlcmZhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtZml4ZWQtZm9vdGVyL3NyYy9saWIvbmd4LWZpeGVkLWZvb3Rlci5pbnRlcmZhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIE5neEZpeGVkRm9vdGVyQ3NzQXR0cmlidXRlID0gJ3BhZGRpbmcnIHwgJ21hcmdpbic7XG5cbmV4cG9ydCBpbnRlcmZhY2UgTmd4Rml4ZWRGb290ZXJPcHRpb25zIHtcbiAgY3NzQXR0cmlidXRlOiBOZ3hGaXhlZEZvb3RlckNzc0F0dHJpYnV0ZTtcbiAgY29udGFpbmVyU2VsZWN0b3I6IHN0cmluZztcbn1cbiJdfQ==
|
|
@@ -10,4 +10,4 @@ export const provideFixedFooter = (options) => {
|
|
|
10
10
|
}
|
|
11
11
|
};
|
|
12
12
|
};
|
|
13
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmd4LWZpeGVkLWZvb3Rlci5wcm92aWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL25neC1maXhlZC1mb290ZXIvc3JjL2xpYi9uZ3gtZml4ZWQtZm9vdGVyLnByb3ZpZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxjQUFjLEVBQVksTUFBTSxlQUFlLENBQUM7QUFDekQsT0FBTyxFQUFFLDBCQUEwQixFQUFFLHFCQUFxQixFQUFFLE1BQU0sOEJBQThCLENBQUM7QUFHakcsTUFBTSxDQUFDLE1BQU0sOEJBQThCLEdBQUcsSUFBSSxjQUFjLENBQXdCLDBCQUEwQixDQUFDLENBQUM7QUFFcEgsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsQ0FBQyxPQUF1QyxFQUFZLEVBQUU7SUFDdEYsT0FBTztRQUNMLE9BQU8sRUFBRSw4QkFBOEI7UUFDdkMsUUFBUSxFQUFFO1lBQ1IsWUFBWSxFQUFFLE9BQU8sQ0FBQyxZQUFZLElBQUkscUJBQXFCO1lBQzNELGlCQUFpQixFQUFFLE9BQU8sQ0FBQyxpQkFBaUIsSUFBSSwwQkFBMEI7U0FDM0U7S0FDRixDQUFDO0FBQ0osQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4sIFByb3ZpZGVyIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBERUZBVUxUX0NPTlRBSU5FUl9TRUxFQ1RPUiwgREVGQVVMVF9DU1NfQVRUUklCVVRFIH0gZnJvbSAnLi9uZ3gtZml4ZWQtZm9vdGVyLmNvbnN0YW50cyc7XG5pbXBvcnQgeyBOZ3hGaXhlZEZvb3Rlck9wdGlvbnMgfSBmcm9tICcuL25neC1maXhlZC1mb290ZXIuaW50ZXJmYWNlJztcblxuZXhwb3J0IGNvbnN0IEFQUF9GSVhFRF9GT09URVJfT1BUSU9OU19UT0tFTiA9IG5ldyBJbmplY3Rpb25Ub2tlbjxOZ3hGaXhlZEZvb3Rlck9wdGlvbnM+KCdbbmd4Rml4ZWRGb290ZXJdIE9wdGlvbnMnKTtcblxuZXhwb3J0IGNvbnN0IHByb3ZpZGVGaXhlZEZvb3RlciA9IChvcHRpb25zOiBQYXJ0aWFsPE5neEZpeGVkRm9vdGVyT3B0aW9ucz4pOiBQcm92aWRlciA9PiB7XG4gIHJldHVybiB7XG4gICAgcHJvdmlkZTogQVBQX0ZJWEVEX0ZPT1RFUl9PUFRJT05TX1RPS0VOLFxuICAgIHVzZVZhbHVlOiB7XG4gICAgICBjc3NBdHRyaWJ1dGU6IG9wdGlvbnMuY3NzQXR0cmlidXRlIHx8IERFRkFVTFRfQ1NTX0FUVFJJQlVURSxcbiAgICAgIGNvbnRhaW5lclNlbGVjdG9yOiBvcHRpb25zLmNvbnRhaW5lclNlbGVjdG9yIHx8IERFRkFVTFRfQ09OVEFJTkVSX1NFTEVDVE9SXG4gICAgfVxuICB9O1xufTtcbiJdfQ==
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { DOCUMENT } from '@angular/common';
|
|
2
2
|
import * as i0 from '@angular/core';
|
|
3
|
-
import { InjectionToken, inject,
|
|
3
|
+
import { InjectionToken, inject, ElementRef, Renderer2, signal, input, computed, effect, Directive } from '@angular/core';
|
|
4
4
|
|
|
5
5
|
const DEFAULT_CSS_ATTRIBUTE = 'padding';
|
|
6
6
|
const DEFAULT_CONTAINER_SELECTOR = '[role="main"]';
|
|
@@ -21,14 +21,47 @@ const provideFixedFooter = (options) => {
|
|
|
21
21
|
};
|
|
22
22
|
|
|
23
23
|
class NgxFixedFooterDirective {
|
|
24
|
-
constructor(
|
|
25
|
-
this.document =
|
|
26
|
-
this.el =
|
|
27
|
-
this.render =
|
|
24
|
+
constructor() {
|
|
25
|
+
this.document = inject(DOCUMENT);
|
|
26
|
+
this.el = inject(ElementRef);
|
|
27
|
+
this.render = inject(Renderer2);
|
|
28
28
|
this.options = inject(APP_FIXED_FOOTER_OPTIONS_TOKEN, { optional: true }) || DEFAULT_FIXED_FOOTER_OPTIONS;
|
|
29
|
-
this.
|
|
30
|
-
this.
|
|
31
|
-
this.
|
|
29
|
+
this.hasResizeObserver = typeof ResizeObserver !== 'undefined';
|
|
30
|
+
this.offsetHeight = signal(undefined);
|
|
31
|
+
this.resizeObserver = signal(undefined);
|
|
32
|
+
this.prevContainerSelector = signal(undefined);
|
|
33
|
+
this.containerSelector = input(this.options.containerSelector);
|
|
34
|
+
this.cssAttribute = input(this.options.cssAttribute);
|
|
35
|
+
this.container = computed(() => {
|
|
36
|
+
const selector = this.containerSelector() || this.options.containerSelector;
|
|
37
|
+
return this.document.body.querySelector(selector);
|
|
38
|
+
});
|
|
39
|
+
// swap selector
|
|
40
|
+
effect(() => {
|
|
41
|
+
if (!this.hasResizeObserver || !this.document)
|
|
42
|
+
return;
|
|
43
|
+
const cssAttribute = this.cssAttribute();
|
|
44
|
+
const containerSelector = this.containerSelector();
|
|
45
|
+
if (containerSelector) {
|
|
46
|
+
const prevContainerSelector = this.prevContainerSelector();
|
|
47
|
+
if (prevContainerSelector && prevContainerSelector !== containerSelector) {
|
|
48
|
+
this.removeStyle(this.document.body.querySelector(prevContainerSelector), cssAttribute);
|
|
49
|
+
}
|
|
50
|
+
this.setStyle(this.document.body.querySelector(containerSelector), cssAttribute, this.offsetHeight());
|
|
51
|
+
this.prevContainerSelector.set(containerSelector);
|
|
52
|
+
}
|
|
53
|
+
}, { allowSignalWrites: true });
|
|
54
|
+
// swap css attribute
|
|
55
|
+
effect(() => {
|
|
56
|
+
if (!this.hasResizeObserver || !this.document)
|
|
57
|
+
return;
|
|
58
|
+
const cssAttribute = this.cssAttribute();
|
|
59
|
+
if (cssAttribute) {
|
|
60
|
+
const container = this.container();
|
|
61
|
+
this.removeStyle(container, cssAttribute === 'padding' ? 'margin' : 'padding');
|
|
62
|
+
this.setStyle(container, cssAttribute, this.offsetHeight());
|
|
63
|
+
}
|
|
64
|
+
});
|
|
32
65
|
}
|
|
33
66
|
ngOnInit() {
|
|
34
67
|
if (this.hasResizeObserver && this.document) {
|
|
@@ -36,41 +69,17 @@ class NgxFixedFooterDirective {
|
|
|
36
69
|
resizeObserver.observe(this.html);
|
|
37
70
|
}
|
|
38
71
|
}
|
|
39
|
-
ngOnChanges(changes) {
|
|
40
|
-
if (this.hasResizeObserver && this.document) {
|
|
41
|
-
// swap selector
|
|
42
|
-
if (changes?.containerSelector && !changes?.containerSelector?.firstChange) {
|
|
43
|
-
const prev = changes?.containerSelector?.previousValue;
|
|
44
|
-
const next = changes?.containerSelector?.currentValue;
|
|
45
|
-
if (next !== prev) {
|
|
46
|
-
this.removeStyle(this.document.body.querySelector(prev), this.cssAttribute);
|
|
47
|
-
this.setStyle(this.document.body.querySelector(next), this.cssAttribute, this.offsetHeight);
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
// swap css attribute
|
|
51
|
-
if (changes?.cssAttribute && !changes?.cssAttribute?.firstChange) {
|
|
52
|
-
const prev = changes?.cssAttribute?.previousValue;
|
|
53
|
-
const next = changes?.cssAttribute?.currentValue;
|
|
54
|
-
if (next !== prev) {
|
|
55
|
-
this.removeStyle(this.container, prev);
|
|
56
|
-
this.setStyle(this.container, next, this.offsetHeight);
|
|
57
|
-
}
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
72
|
ngOnDestroy() {
|
|
62
|
-
if (this.
|
|
63
|
-
this.removeStyle(this.container, this.cssAttribute);
|
|
64
|
-
|
|
65
|
-
this.resizeObserver.unobserve(this.html);
|
|
66
|
-
}
|
|
73
|
+
if (this.resizeObserver() && this.document) {
|
|
74
|
+
this.removeStyle(this.container(), this.cssAttribute());
|
|
75
|
+
this.resizeObserver().unobserve(this.html);
|
|
67
76
|
}
|
|
68
77
|
}
|
|
69
78
|
checkHeight() {
|
|
70
79
|
const height = this.html.offsetHeight;
|
|
71
|
-
if (this.offsetHeight !== height) {
|
|
72
|
-
this.setStyle(this.container, this.cssAttribute, height);
|
|
73
|
-
this.offsetHeight
|
|
80
|
+
if (this.offsetHeight() !== height) {
|
|
81
|
+
this.setStyle(this.container(), this.cssAttribute(), height);
|
|
82
|
+
this.offsetHeight.set(height);
|
|
74
83
|
}
|
|
75
84
|
}
|
|
76
85
|
removeStyle(container, cssAttribute) {
|
|
@@ -85,37 +94,19 @@ class NgxFixedFooterDirective {
|
|
|
85
94
|
}
|
|
86
95
|
this.render.setStyle(container, `${cssAttribute}-bottom`, height === 0 ? '' : `${height}px`);
|
|
87
96
|
}
|
|
88
|
-
get container() {
|
|
89
|
-
const selector = this.containerSelector || this.options.containerSelector;
|
|
90
|
-
const container = this.document.body.querySelector(selector);
|
|
91
|
-
if (!container) {
|
|
92
|
-
console.warn(`Container '${selector}' was not found`);
|
|
93
|
-
}
|
|
94
|
-
return container;
|
|
95
|
-
}
|
|
96
97
|
get html() {
|
|
97
98
|
return this.el.nativeElement;
|
|
98
99
|
}
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
}
|
|
102
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: NgxFixedFooterDirective, deps: [{ token: DOCUMENT }, { token: i0.ElementRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
103
|
-
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.0.8", type: NgxFixedFooterDirective, isStandalone: true, selector: "[ngxFixedFooter]", inputs: { containerSelector: "containerSelector", cssAttribute: "cssAttribute" }, usesOnChanges: true, ngImport: i0 }); }
|
|
100
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: NgxFixedFooterDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
101
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.0.1", type: NgxFixedFooterDirective, isStandalone: true, selector: "[ngxFixedFooter]", inputs: { containerSelector: { classPropertyName: "containerSelector", publicName: "containerSelector", isSignal: true, isRequired: false, transformFunction: null }, cssAttribute: { classPropertyName: "cssAttribute", publicName: "cssAttribute", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0 }); }
|
|
104
102
|
}
|
|
105
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
103
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.1", ngImport: i0, type: NgxFixedFooterDirective, decorators: [{
|
|
106
104
|
type: Directive,
|
|
107
105
|
args: [{
|
|
108
106
|
selector: '[ngxFixedFooter]',
|
|
109
107
|
standalone: true
|
|
110
108
|
}]
|
|
111
|
-
}], ctorParameters: () => [
|
|
112
|
-
type: Inject,
|
|
113
|
-
args: [DOCUMENT]
|
|
114
|
-
}] }, { type: i0.ElementRef }, { type: i0.Renderer2 }], propDecorators: { containerSelector: [{
|
|
115
|
-
type: Input
|
|
116
|
-
}], cssAttribute: [{
|
|
117
|
-
type: Input
|
|
118
|
-
}] } });
|
|
109
|
+
}], ctorParameters: () => [] });
|
|
119
110
|
|
|
120
111
|
/*
|
|
121
112
|
* Public API Surface of ngx-fixed-footer
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ngx-fixed-footer.mjs","sources":["../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.constants.ts","../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.provider.ts","../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.directive.ts","../../../projects/ngx-fixed-footer/src/public-api.ts","../../../projects/ngx-fixed-footer/src/ngx-fixed-footer.ts"],"sourcesContent":["import { NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\n\nexport const DEFAULT_CSS_ATTRIBUTE = 'padding';\nexport const DEFAULT_CONTAINER_SELECTOR = '[role=\"main\"]';\nexport const DEFAULT_FIXED_FOOTER_OPTIONS: NgxFixedFooterOptions = {\n containerSelector: DEFAULT_CONTAINER_SELECTOR,\n cssAttribute: DEFAULT_CSS_ATTRIBUTE\n};\n","import { InjectionToken, Provider } from '@angular/core';\nimport { DEFAULT_CONTAINER_SELECTOR, DEFAULT_CSS_ATTRIBUTE } from './ngx-fixed-footer.constants';\nimport { NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\n\nexport const APP_FIXED_FOOTER_OPTIONS_TOKEN = new InjectionToken<NgxFixedFooterOptions>('[ngxFixedFooter] Options');\n\nexport const provideFixedFooter = (options: NgxFixedFooterOptions): Provider => {\n return {\n provide: APP_FIXED_FOOTER_OPTIONS_TOKEN,\n useValue: {\n cssAttribute: options.cssAttribute || DEFAULT_CSS_ATTRIBUTE,\n containerSelector: options.containerSelector || DEFAULT_CONTAINER_SELECTOR\n }\n };\n};\n","import { DOCUMENT } from '@angular/common';\nimport {\n Directive,\n ElementRef,\n Inject,\n Input,\n OnChanges,\n OnDestroy,\n OnInit,\n Renderer2,\n SimpleChanges,\n inject\n} from '@angular/core';\nimport { DEFAULT_FIXED_FOOTER_OPTIONS } from './ngx-fixed-footer.constants';\nimport { NgxFixedFooterCssAttribute, NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\nimport { APP_FIXED_FOOTER_OPTIONS_TOKEN } from './ngx-fixed-footer.provider';\n\n@Directive({\n selector: '[ngxFixedFooter]',\n standalone: true\n})\nexport class NgxFixedFooterDirective implements OnDestroy, OnChanges, OnInit {\n private options: NgxFixedFooterOptions =\n inject(APP_FIXED_FOOTER_OPTIONS_TOKEN, { optional: true }) || DEFAULT_FIXED_FOOTER_OPTIONS;\n private offsetHeight: number = undefined;\n private resizeObserver: ResizeObserver;\n\n @Input() public containerSelector: string = this.options.containerSelector;\n @Input() public cssAttribute: NgxFixedFooterCssAttribute = this.options.cssAttribute;\n\n constructor(\n @Inject(DOCUMENT) private document: any,\n private el: ElementRef,\n private render: Renderer2\n ) {}\n\n public ngOnInit(): void {\n if (this.hasResizeObserver && this.document) {\n const resizeObserver = new ResizeObserver(() => this.checkHeight());\n resizeObserver.observe(this.html);\n }\n }\n\n public ngOnChanges(changes: SimpleChanges): void {\n if (this.hasResizeObserver && this.document) {\n // swap selector\n if (changes?.containerSelector && !changes?.containerSelector?.firstChange) {\n const prev = changes?.containerSelector?.previousValue;\n const next = changes?.containerSelector?.currentValue;\n if (next !== prev) {\n this.removeStyle(this.document.body.querySelector(prev), this.cssAttribute);\n this.setStyle(this.document.body.querySelector(next), this.cssAttribute, this.offsetHeight);\n }\n }\n\n // swap css attribute\n if (changes?.cssAttribute && !changes?.cssAttribute?.firstChange) {\n const prev = changes?.cssAttribute?.previousValue;\n const next = changes?.cssAttribute?.currentValue;\n if (next !== prev) {\n this.removeStyle(this.container, prev);\n this.setStyle(this.container, next, this.offsetHeight);\n }\n }\n }\n }\n\n public ngOnDestroy(): void {\n if (this.hasResizeObserver && this.document) {\n this.removeStyle(this.container, this.cssAttribute);\n if (this.resizeObserver) {\n this.resizeObserver.unobserve(this.html);\n }\n }\n }\n\n private checkHeight(): void {\n const height = this.html.offsetHeight;\n if (this.offsetHeight !== height) {\n this.setStyle(this.container, this.cssAttribute, height);\n this.offsetHeight = height;\n }\n }\n\n private removeStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute): void {\n if (!container) {\n throw new Error(`Cannot removeStyle to undefined container`);\n }\n this.render.setStyle(container, `${cssAttribute}-bottom`, '');\n }\n\n private setStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute, height: number): void {\n if (!container) {\n throw new Error(`Cannot setStyle to undefined container`);\n }\n this.render.setStyle(container, `${cssAttribute}-bottom`, height === 0 ? '' : `${height}px`);\n }\n\n private get container(): HTMLElement {\n const selector = this.containerSelector || this.options.containerSelector;\n const container = this.document.body.querySelector(selector);\n if (!container) {\n console.warn(`Container '${selector}' was not found`);\n }\n return container;\n }\n\n private get html(): HTMLElement {\n return this.el.nativeElement;\n }\n\n private get hasResizeObserver(): boolean {\n return typeof ResizeObserver !== 'undefined';\n }\n}\n","/*\n * Public API Surface of ngx-fixed-footer\n */\n\nexport * from './lib/ngx-fixed-footer.constants';\nexport * from './lib/ngx-fixed-footer.directive';\nexport * from './lib/ngx-fixed-footer.interface';\nexport * from './lib/ngx-fixed-footer.provider';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"ngx-fixed-footer.mjs","sources":["../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.constants.ts","../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.provider.ts","../../../projects/ngx-fixed-footer/src/lib/ngx-fixed-footer.directive.ts","../../../projects/ngx-fixed-footer/src/public-api.ts","../../../projects/ngx-fixed-footer/src/ngx-fixed-footer.ts"],"sourcesContent":["import { NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\n\nexport const DEFAULT_CSS_ATTRIBUTE = 'padding';\nexport const DEFAULT_CONTAINER_SELECTOR = '[role=\"main\"]';\nexport const DEFAULT_FIXED_FOOTER_OPTIONS: NgxFixedFooterOptions = {\n containerSelector: DEFAULT_CONTAINER_SELECTOR,\n cssAttribute: DEFAULT_CSS_ATTRIBUTE\n};\n","import { InjectionToken, Provider } from '@angular/core';\nimport { DEFAULT_CONTAINER_SELECTOR, DEFAULT_CSS_ATTRIBUTE } from './ngx-fixed-footer.constants';\nimport { NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\n\nexport const APP_FIXED_FOOTER_OPTIONS_TOKEN = new InjectionToken<NgxFixedFooterOptions>('[ngxFixedFooter] Options');\n\nexport const provideFixedFooter = (options: Partial<NgxFixedFooterOptions>): Provider => {\n return {\n provide: APP_FIXED_FOOTER_OPTIONS_TOKEN,\n useValue: {\n cssAttribute: options.cssAttribute || DEFAULT_CSS_ATTRIBUTE,\n containerSelector: options.containerSelector || DEFAULT_CONTAINER_SELECTOR\n }\n };\n};\n","import { DOCUMENT } from '@angular/common';\nimport {\n Directive,\n ElementRef,\n OnDestroy,\n OnInit,\n Renderer2,\n computed,\n effect,\n inject,\n input,\n signal\n} from '@angular/core';\nimport { DEFAULT_FIXED_FOOTER_OPTIONS } from './ngx-fixed-footer.constants';\nimport { NgxFixedFooterCssAttribute, NgxFixedFooterOptions } from './ngx-fixed-footer.interface';\nimport { APP_FIXED_FOOTER_OPTIONS_TOKEN } from './ngx-fixed-footer.provider';\n\n@Directive({\n selector: '[ngxFixedFooter]',\n standalone: true\n})\nexport class NgxFixedFooterDirective implements OnDestroy, OnInit {\n private readonly document = inject(DOCUMENT);\n private readonly el = inject(ElementRef);\n private readonly render = inject(Renderer2);\n private options: NgxFixedFooterOptions =\n inject(APP_FIXED_FOOTER_OPTIONS_TOKEN, { optional: true }) || DEFAULT_FIXED_FOOTER_OPTIONS;\n private readonly hasResizeObserver = typeof ResizeObserver !== 'undefined';\n\n private offsetHeight = signal<number | undefined>(undefined);\n private resizeObserver = signal<ResizeObserver | undefined>(undefined);\n private prevContainerSelector = signal<string | undefined>(undefined);\n\n public containerSelector = input<string>(this.options.containerSelector);\n public cssAttribute = input<NgxFixedFooterCssAttribute>(this.options.cssAttribute);\n\n private container = computed<HTMLElement>(() => {\n const selector = this.containerSelector() || this.options.containerSelector;\n return this.document.body.querySelector<HTMLElement>(selector);\n });\n\n constructor() {\n // swap selector\n effect(\n () => {\n if (!this.hasResizeObserver || !this.document) return;\n const cssAttribute = this.cssAttribute();\n const containerSelector = this.containerSelector();\n if (containerSelector) {\n const prevContainerSelector = this.prevContainerSelector();\n if (prevContainerSelector && prevContainerSelector !== containerSelector) {\n this.removeStyle(this.document.body.querySelector(prevContainerSelector), cssAttribute);\n }\n this.setStyle(this.document.body.querySelector(containerSelector), cssAttribute, this.offsetHeight());\n this.prevContainerSelector.set(containerSelector);\n }\n },\n { allowSignalWrites: true }\n );\n\n // swap css attribute\n effect(() => {\n if (!this.hasResizeObserver || !this.document) return;\n const cssAttribute = this.cssAttribute();\n if (cssAttribute) {\n const container = this.container();\n this.removeStyle(container, cssAttribute === 'padding' ? 'margin' : 'padding');\n this.setStyle(container, cssAttribute, this.offsetHeight());\n }\n });\n }\n\n public ngOnInit(): void {\n if (this.hasResizeObserver && this.document) {\n const resizeObserver = new ResizeObserver(() => this.checkHeight());\n resizeObserver.observe(this.html);\n }\n }\n\n public ngOnDestroy(): void {\n if (this.resizeObserver() && this.document) {\n this.removeStyle(this.container(), this.cssAttribute());\n this.resizeObserver().unobserve(this.html);\n }\n }\n\n private checkHeight(): void {\n const height = this.html.offsetHeight;\n if (this.offsetHeight() !== height) {\n this.setStyle(this.container(), this.cssAttribute(), height);\n this.offsetHeight.set(height);\n }\n }\n\n private removeStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute): void {\n if (!container) {\n throw new Error(`Cannot removeStyle to undefined container`);\n }\n this.render.setStyle(container, `${cssAttribute}-bottom`, '');\n }\n\n private setStyle(container: HTMLElement, cssAttribute: NgxFixedFooterCssAttribute, height: number): void {\n if (!container) {\n throw new Error(`Cannot setStyle to undefined container`);\n }\n this.render.setStyle(container, `${cssAttribute}-bottom`, height === 0 ? '' : `${height}px`);\n }\n\n private get html(): HTMLElement {\n return this.el.nativeElement;\n }\n}\n","/*\n * Public API Surface of ngx-fixed-footer\n */\n\nexport * from './lib/ngx-fixed-footer.constants';\nexport * from './lib/ngx-fixed-footer.directive';\nexport * from './lib/ngx-fixed-footer.interface';\nexport * from './lib/ngx-fixed-footer.provider';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":[],"mappings":"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"}
|
|
@@ -1,25 +1,25 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { OnDestroy, OnInit } from '@angular/core';
|
|
2
2
|
import { NgxFixedFooterCssAttribute } from './ngx-fixed-footer.interface';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
|
-
export declare class NgxFixedFooterDirective implements OnDestroy,
|
|
5
|
-
private document;
|
|
6
|
-
private el;
|
|
7
|
-
private render;
|
|
4
|
+
export declare class NgxFixedFooterDirective implements OnDestroy, OnInit {
|
|
5
|
+
private readonly document;
|
|
6
|
+
private readonly el;
|
|
7
|
+
private readonly render;
|
|
8
8
|
private options;
|
|
9
|
+
private readonly hasResizeObserver;
|
|
9
10
|
private offsetHeight;
|
|
10
11
|
private resizeObserver;
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
12
|
+
private prevContainerSelector;
|
|
13
|
+
containerSelector: import("@angular/core").InputSignal<string>;
|
|
14
|
+
cssAttribute: import("@angular/core").InputSignal<NgxFixedFooterCssAttribute>;
|
|
15
|
+
private container;
|
|
16
|
+
constructor();
|
|
14
17
|
ngOnInit(): void;
|
|
15
|
-
ngOnChanges(changes: SimpleChanges): void;
|
|
16
18
|
ngOnDestroy(): void;
|
|
17
19
|
private checkHeight;
|
|
18
20
|
private removeStyle;
|
|
19
21
|
private setStyle;
|
|
20
|
-
private get container();
|
|
21
22
|
private get html();
|
|
22
|
-
private get hasResizeObserver();
|
|
23
23
|
static ɵfac: i0.ɵɵFactoryDeclaration<NgxFixedFooterDirective, never>;
|
|
24
|
-
static ɵdir: i0.ɵɵDirectiveDeclaration<NgxFixedFooterDirective, "[ngxFixedFooter]", never, { "containerSelector": { "alias": "containerSelector"; "required": false; }; "cssAttribute": { "alias": "cssAttribute"; "required": false; }; }, {}, never, never, true, never>;
|
|
24
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<NgxFixedFooterDirective, "[ngxFixedFooter]", never, { "containerSelector": { "alias": "containerSelector"; "required": false; "isSignal": true; }; "cssAttribute": { "alias": "cssAttribute"; "required": false; "isSignal": true; }; }, {}, never, never, true, never>;
|
|
25
25
|
}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
1
|
import { InjectionToken, Provider } from '@angular/core';
|
|
2
2
|
import { NgxFixedFooterOptions } from './ngx-fixed-footer.interface';
|
|
3
3
|
export declare const APP_FIXED_FOOTER_OPTIONS_TOKEN: InjectionToken<NgxFixedFooterOptions>;
|
|
4
|
-
export declare const provideFixedFooter: (options: NgxFixedFooterOptions) => Provider;
|
|
4
|
+
export declare const provideFixedFooter: (options: Partial<NgxFixedFooterOptions>) => Provider;
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "ngx-fixed-footer",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "3.0.0",
|
|
4
4
|
"author": {
|
|
5
5
|
"name": "Dominik Hladík",
|
|
6
6
|
"email": "dominik.hladik@seznam.cz",
|
|
@@ -10,8 +10,8 @@
|
|
|
10
10
|
"license": "MIT",
|
|
11
11
|
"description": "Angular directive that adds fixed footer without overlap",
|
|
12
12
|
"peerDependencies": {
|
|
13
|
-
"@angular/common": ">=
|
|
14
|
-
"@angular/core": ">=
|
|
13
|
+
"@angular/common": ">=18",
|
|
14
|
+
"@angular/core": ">=18"
|
|
15
15
|
},
|
|
16
16
|
"publishConfig": {
|
|
17
17
|
"registry": "https://registry.npmjs.org"
|
|
@@ -32,7 +32,7 @@
|
|
|
32
32
|
"overlap"
|
|
33
33
|
],
|
|
34
34
|
"engines": {
|
|
35
|
-
"node": ">=
|
|
35
|
+
"node": ">=18"
|
|
36
36
|
},
|
|
37
37
|
"module": "fesm2022/ngx-fixed-footer.mjs",
|
|
38
38
|
"typings": "index.d.ts",
|