ngx-tethys 14.2.21 → 14.2.23

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.
Files changed (61) hide show
  1. package/CHANGELOG.md +30 -0
  2. package/avatar/avatar-list/avatar-list.component.d.ts +75 -0
  3. package/avatar/avatar.component.d.ts +57 -1
  4. package/avatar/avatar.module.d.ts +5 -4
  5. package/avatar/index.d.ts +1 -0
  6. package/avatar/styles/avatar-list.scss +36 -0
  7. package/avatar/styles/avatar.scss +2 -0
  8. package/avatar/styles/mixin.scss +10 -0
  9. package/date-picker/styles/calendar.scss +0 -2
  10. package/date-picker/styles/picker.scss +1 -1
  11. package/date-picker/styles/range-picker.scss +1 -1
  12. package/esm2020/avatar/avatar-list/avatar-list.component.mjs +175 -0
  13. package/esm2020/avatar/avatar.component.mjs +47 -3
  14. package/esm2020/avatar/avatar.module.mjs +5 -4
  15. package/esm2020/avatar/index.mjs +2 -1
  16. package/esm2020/comment/comment.component.mjs +1 -1
  17. package/esm2020/date-picker/lib/date/date-table.component.mjs +12 -3
  18. package/esm2020/date-picker/lib/popups/date-popup.component.mjs +2 -1
  19. package/esm2020/input-number/input-number.component.mjs +6 -6
  20. package/esm2020/layout/sidebar.component.mjs +18 -7
  21. package/esm2020/list/list-item-meta.component.mjs +1 -1
  22. package/esm2020/popover/header/popover-header.component.mjs +9 -5
  23. package/esm2020/version.mjs +2 -2
  24. package/fesm2015/ngx-tethys-avatar.mjs +221 -7
  25. package/fesm2015/ngx-tethys-avatar.mjs.map +1 -1
  26. package/fesm2015/ngx-tethys-comment.mjs +1 -1
  27. package/fesm2015/ngx-tethys-comment.mjs.map +1 -1
  28. package/fesm2015/ngx-tethys-date-picker.mjs +13 -2
  29. package/fesm2015/ngx-tethys-date-picker.mjs.map +1 -1
  30. package/fesm2015/ngx-tethys-input-number.mjs +5 -5
  31. package/fesm2015/ngx-tethys-input-number.mjs.map +1 -1
  32. package/fesm2015/ngx-tethys-layout.mjs +12 -1
  33. package/fesm2015/ngx-tethys-layout.mjs.map +1 -1
  34. package/fesm2015/ngx-tethys-list.mjs +1 -1
  35. package/fesm2015/ngx-tethys-list.mjs.map +1 -1
  36. package/fesm2015/ngx-tethys-popover.mjs +8 -4
  37. package/fesm2015/ngx-tethys-popover.mjs.map +1 -1
  38. package/fesm2015/ngx-tethys.mjs +1 -1
  39. package/fesm2015/ngx-tethys.mjs.map +1 -1
  40. package/fesm2020/ngx-tethys-avatar.mjs +221 -7
  41. package/fesm2020/ngx-tethys-avatar.mjs.map +1 -1
  42. package/fesm2020/ngx-tethys-comment.mjs +1 -1
  43. package/fesm2020/ngx-tethys-comment.mjs.map +1 -1
  44. package/fesm2020/ngx-tethys-date-picker.mjs +12 -2
  45. package/fesm2020/ngx-tethys-date-picker.mjs.map +1 -1
  46. package/fesm2020/ngx-tethys-input-number.mjs +5 -5
  47. package/fesm2020/ngx-tethys-input-number.mjs.map +1 -1
  48. package/fesm2020/ngx-tethys-layout.mjs +13 -2
  49. package/fesm2020/ngx-tethys-layout.mjs.map +1 -1
  50. package/fesm2020/ngx-tethys-list.mjs +1 -1
  51. package/fesm2020/ngx-tethys-list.mjs.map +1 -1
  52. package/fesm2020/ngx-tethys-popover.mjs +8 -4
  53. package/fesm2020/ngx-tethys-popover.mjs.map +1 -1
  54. package/fesm2020/ngx-tethys.mjs +1 -1
  55. package/fesm2020/ngx-tethys.mjs.map +1 -1
  56. package/layout/sidebar.component.d.ts +6 -4
  57. package/package.json +1 -1
  58. package/popover/header/popover-header.component.d.ts +6 -2
  59. package/schematics/version.d.ts +1 -1
  60. package/schematics/version.js +1 -1
  61. package/styles/variables.scss +1 -0
package/CHANGELOG.md CHANGED
@@ -2,6 +2,36 @@
2
2
 
3
3
  All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
4
4
 
5
+ ## [14.2.23](https://github.com/atinc/ngx-tethys/compare/14.2.21...14.2.23) (2023-03-02)
6
+
7
+
8
+ ### Bug Fixes
9
+
10
+ * **date-picker:** fix quick select label style justify content center #INFR-5621 ([#2557](https://github.com/atinc/ngx-tethys/issues/2557)) ([6f16b1e](https://github.com/atinc/ngx-tethys/commit/6f16b1e314d92638900148f4754373d585921b96)), closes [#INFR-5621](https://github.com/atinc/ngx-tethys/issues/INFR-5621)
11
+ * **date-picker:** same month does not appear twice in week range picker and adjust week range width #INFR-6644 ([#2554](https://github.com/atinc/ngx-tethys/issues/2554)) ([84d1cbe](https://github.com/atinc/ngx-tethys/commit/84d1cbeb945c18dda332f16ff595900232ea4871)), closes [#INFR-6644](https://github.com/atinc/ngx-tethys/issues/INFR-6644) [#INFR-6644](https://github.com/atinc/ngx-tethys/issues/INFR-6644) [#INFR-6644](https://github.com/atinc/ngx-tethys/issues/INFR-6644) [#INFR-6644](https://github.com/atinc/ngx-tethys/issues/INFR-6644) [#INFR-6644](https://github.com/atinc/ngx-tethys/issues/INFR-6644) [#INFR-6644](https://github.com/atinc/ngx-tethys/issues/INFR-6644)
12
+ * **input-number:** fix display null when thyMin or thyMax is null#INFR-6594 ([#2547](https://github.com/atinc/ngx-tethys/issues/2547)) ([3a49083](https://github.com/atinc/ngx-tethys/commit/3a490839d80fcf452beb4dffbe9e4e1c8b307ad3)), closes [null#INFR-6594](https://github.com/null/issues/INFR-6594) [null#INFR-6594](https://github.com/null/issues/INFR-6594)
13
+
14
+
15
+ ### Features
16
+
17
+ * **avatar:** add thy-avatar-list #INFR-5295 [#2525](https://github.com/atinc/ngx-tethys/issues/2525) ([#2526](https://github.com/atinc/ngx-tethys/issues/2526)) ([112eac3](https://github.com/atinc/ngx-tethys/commit/112eac3ff65a78b9a3305195d46464cf5256218e)), closes [#INFR-5295](https://github.com/atinc/ngx-tethys/issues/INFR-5295)
18
+ * **layout:** sidebar support dblclick restore to default width #INFR-6017 ([#2551](https://github.com/atinc/ngx-tethys/issues/2551)) ([06d85ad](https://github.com/atinc/ngx-tethys/commit/06d85ad1fba810837915bf8df688c91003eca8ea)), closes [#INFR-6017](https://github.com/atinc/ngx-tethys/issues/INFR-6017) [#INFR-6017](https://github.com/atinc/ngx-tethys/issues/INFR-6017)
19
+ * **popover:** support popover hader template and add test(#INFR-6586) ([#2549](https://github.com/atinc/ngx-tethys/issues/2549)) ([d5dff36](https://github.com/atinc/ngx-tethys/commit/d5dff368b8cb0f27821ae689f81671b639826ad1)), closes [#INFR-6586](https://github.com/atinc/ngx-tethys/issues/INFR-6586)
20
+ * send message and merge after pub #INFR-6280 ([9353382](https://github.com/atinc/ngx-tethys/commit/9353382acc00bec1545abb6ba7e4daf21935d10b)), closes [#INFR-6280](https://github.com/atinc/ngx-tethys/issues/INFR-6280)
21
+
22
+
23
+
24
+ ## [14.2.22](https://github.com/atinc/ngx-tethys/compare/14.2.21...14.2.22) (2023-02-27)
25
+
26
+
27
+ ### Features
28
+
29
+ * **avatar:** add thy-avatar-list #INFR-5295 [#2525](https://github.com/atinc/ngx-tethys/issues/2525) ([#2526](https://github.com/atinc/ngx-tethys/issues/2526)) ([112eac3](https://github.com/atinc/ngx-tethys/commit/112eac3ff65a78b9a3305195d46464cf5256218e)), closes [#INFR-5295](https://github.com/atinc/ngx-tethys/issues/INFR-5295)
30
+ * **popover:** support popover hader template and add test(#INFR-6586) ([#2549](https://github.com/atinc/ngx-tethys/issues/2549)) ([d5dff36](https://github.com/atinc/ngx-tethys/commit/d5dff368b8cb0f27821ae689f81671b639826ad1)), closes [#INFR-6586](https://github.com/atinc/ngx-tethys/issues/INFR-6586)
31
+ * send message and merge after pub #INFR-6280 ([9353382](https://github.com/atinc/ngx-tethys/commit/9353382acc00bec1545abb6ba7e4daf21935d10b)), closes [#INFR-6280](https://github.com/atinc/ngx-tethys/issues/INFR-6280)
32
+
33
+
34
+
5
35
  ## [14.2.21](https://github.com/atinc/ngx-tethys/compare/14.2.20...14.2.21) (2023-02-22)
6
36
 
7
37
 
@@ -0,0 +1,75 @@
1
+ import { AfterContentInit, AfterViewInit, ElementRef, EventEmitter, NgZone, OnChanges, OnDestroy, SimpleChanges, TemplateRef } from '@angular/core';
2
+ import { ThyAvatarComponent } from '../avatar.component';
3
+ import { SafeAny } from 'ngx-tethys/types';
4
+ import * as i0 from "@angular/core";
5
+ export declare const enum ThyAvatarListMode {
6
+ overlap = "overlap",
7
+ default = "default"
8
+ }
9
+ /**
10
+ * 头像列表组件
11
+ */
12
+ export declare class ThyAvatarListComponent implements OnChanges, OnDestroy, AfterContentInit, AfterViewInit {
13
+ private elementRef;
14
+ private ngZone;
15
+ overlapMode: boolean;
16
+ avatarItems: ThyAvatarComponent[];
17
+ avatarRenderItems: ThyAvatarComponent[];
18
+ avatarSpace: number;
19
+ avatarOverlapSpace: number;
20
+ get more(): number;
21
+ private ngUnsubscribe$;
22
+ private avatarList;
23
+ /**
24
+ * 展示方式
25
+ * @type 'overlap'| 'default'
26
+ * @default default
27
+ */
28
+ set thyMode(value: ThyAvatarListMode);
29
+ /**
30
+ * 响应式,自动计算宽度存放 avatar
31
+ * @default false
32
+ */
33
+ thyResponsive: boolean;
34
+ /**
35
+ * 列表组件允许展示 avatar 最大数量
36
+ */
37
+ thyMax: number;
38
+ /**
39
+ * 头像大小
40
+ * @type 22 | 24 | 28 | 32 | 36 | 44 | 48 | 68 | 110 | 160 | xxs(22px) | xs(24px) | sm(32px) | md(36px) | lg(48px)
41
+ * @default 36
42
+ */
43
+ thyAvatarSize: number | string;
44
+ /**
45
+ * 是否展示移除按钮
46
+ * @type boolean
47
+ * @default false
48
+ */
49
+ thyRemovable: boolean;
50
+ /**
51
+ * avatar 移除按钮事件
52
+ */
53
+ thyRemove: EventEmitter<string>;
54
+ /**
55
+ * append 自定义操作
56
+ */
57
+ append: TemplateRef<SafeAny>;
58
+ /**
59
+ * @private
60
+ */
61
+ private set avatarComponents(value);
62
+ appendContent: ElementRef<HTMLInputElement>;
63
+ constructor(elementRef: ElementRef, ngZone: NgZone);
64
+ ngOnChanges(changes: SimpleChanges): void;
65
+ ngAfterContentInit(): void;
66
+ ngAfterViewInit(): void;
67
+ private setAvatarSize;
68
+ remove(name: string): void;
69
+ private getRenderAvatar;
70
+ private getEndIndex;
71
+ private createResizeObserver;
72
+ ngOnDestroy(): void;
73
+ static ɵfac: i0.ɵɵFactoryDeclaration<ThyAvatarListComponent, never>;
74
+ static ɵcmp: i0.ɵɵComponentDeclaration<ThyAvatarListComponent, "thy-avatar-list", never, { "thyMode": "thyMode"; "thyResponsive": "thyResponsive"; "thyMax": "thyMax"; "thyAvatarSize": "thyAvatarSize"; "thyRemovable": "thyRemovable"; }, { "thyRemove": "thyRemove"; }, ["append", "avatarComponents"], ["*"], false>;
75
+ }
@@ -2,6 +2,7 @@ import { EventEmitter, OnInit } from '@angular/core';
2
2
  import { SafeHtml } from '@angular/platform-browser';
3
3
  import { ThyAvatarService } from './avatar.service';
4
4
  import * as i0 from "@angular/core";
5
+ export declare const DEFAULT_SIZE = 36;
5
6
  export declare const thyAvatarSizeMap: {
6
7
  xxs: number;
7
8
  xs: number;
@@ -13,6 +14,9 @@ export declare const thyAvatarSizeMap: {
13
14
  export declare type ThyAvatarLoading = 'eager' | 'lazy';
14
15
  /** https://wicg.github.io/priority-hints/#idl-index */
15
16
  export declare type ThyAvatarFetchPriority = 'high' | 'low' | 'auto';
17
+ /**
18
+ * 头像组件
19
+ */
16
20
  export declare class ThyAvatarComponent implements OnInit {
17
21
  private thyAvatarService;
18
22
  _src: string;
@@ -23,16 +27,68 @@ export declare class ThyAvatarComponent implements OnInit {
23
27
  avatarName?: string;
24
28
  avatarNameSafeHtml?: SafeHtml;
25
29
  _isAvatar: boolean;
30
+ /**
31
+ * * 已废弃,请使用 thyRemove
32
+ * @deprecated
33
+ */
26
34
  thyOnRemove: EventEmitter<any>;
35
+ /**
36
+ * 移除按钮的事件, 当 thyRemovable 为 true 时起作用
37
+ */
38
+ thyRemove: EventEmitter<any>;
39
+ /**
40
+ * 头像 img 加载 error 时触发
41
+ */
27
42
  thyError: EventEmitter<Event>;
43
+ /**
44
+ * 是否展示人员名称
45
+ * @default false
46
+ */
28
47
  thyShowName: boolean;
48
+ /**
49
+ * 头像路径地址, 默认为全路径,如果不是全路径,可以通过自定义服务 ThyAvatarService,重写 srcTransform 方法实现转换
50
+ *
51
+ */
29
52
  set thySrc(value: string);
53
+ /**
54
+ * 人员名称(可设置自定义名称,需通过自定义服务 ThyAvatarService,重写 nameTransform 方法去实现转换)
55
+ */
30
56
  set thyName(value: string);
57
+ /**
58
+ * 头像大小
59
+ * @type 16 | 22 | 24 | 28 | 32 | 36 | 44 | 48 | 68 | 110 | 160 | xxs(22px) | xs(24px) | sm(32px) | md(36px) | lg(48px)
60
+ * @default md
61
+ */
31
62
  set thySize(value: number | string);
63
+ /**
64
+ * 已废弃,请使用 thyRemovable
65
+ * @deprecated
66
+ * @default false
67
+ */
32
68
  set thyShowRemove(value: boolean);
69
+ /**
70
+ * 是否展示移除按钮
71
+ * @default false
72
+ */
73
+ set thyRemovable(value: boolean);
74
+ /**
75
+ * 图片自定义类
76
+ */
33
77
  thyImgClass: string;
78
+ /**
79
+ * 禁用
80
+ * @default false
81
+ */
34
82
  thyDisabled: boolean;
83
+ /**
84
+ * 图片加载策略
85
+ * @type eager(立即加载) | lazy(延迟加载)
86
+ */
35
87
  thyLoading?: ThyAvatarLoading;
88
+ /**
89
+ * 图片加载优先级
90
+ * @type auto(默认) | high(高) | low(低)
91
+ */
36
92
  thyFetchPriority?: ThyAvatarFetchPriority;
37
93
  private _setAvatarSize;
38
94
  private findClosestSize;
@@ -44,5 +100,5 @@ export declare class ThyAvatarComponent implements OnInit {
44
100
  remove($event: Event): void;
45
101
  avatarImgError($event: Event): void;
46
102
  static ɵfac: i0.ɵɵFactoryDeclaration<ThyAvatarComponent, never>;
47
- static ɵcmp: i0.ɵɵComponentDeclaration<ThyAvatarComponent, "thy-avatar", never, { "thyShowName": "thyShowName"; "thySrc": "thySrc"; "thyName": "thyName"; "thySize": "thySize"; "thyShowRemove": "thyShowRemove"; "thyImgClass": "thyImgClass"; "thyDisabled": "thyDisabled"; "thyLoading": "thyLoading"; "thyFetchPriority": "thyFetchPriority"; }, { "thyOnRemove": "thyOnRemove"; "thyError": "thyError"; }, never, never, false>;
103
+ static ɵcmp: i0.ɵɵComponentDeclaration<ThyAvatarComponent, "thy-avatar", never, { "thyShowName": "thyShowName"; "thySrc": "thySrc"; "thyName": "thyName"; "thySize": "thySize"; "thyShowRemove": "thyShowRemove"; "thyRemovable": "thyRemovable"; "thyImgClass": "thyImgClass"; "thyDisabled": "thyDisabled"; "thyLoading": "thyLoading"; "thyFetchPriority": "thyFetchPriority"; }, { "thyOnRemove": "thyOnRemove"; "thyRemove": "thyRemove"; "thyError": "thyError"; }, never, never, false>;
48
104
  }
@@ -1,10 +1,11 @@
1
1
  import * as i0 from "@angular/core";
2
2
  import * as i1 from "./avatar.component";
3
- import * as i2 from "./avatar.pipe";
4
- import * as i3 from "@angular/common";
5
- import * as i4 from "ngx-tethys/icon";
3
+ import * as i2 from "./avatar-list/avatar-list.component";
4
+ import * as i3 from "./avatar.pipe";
5
+ import * as i4 from "@angular/common";
6
+ import * as i5 from "ngx-tethys/icon";
6
7
  export declare class ThyAvatarModule {
7
8
  static ɵfac: i0.ɵɵFactoryDeclaration<ThyAvatarModule, never>;
8
- static ɵmod: i0.ɵɵNgModuleDeclaration<ThyAvatarModule, [typeof i1.ThyAvatarComponent, typeof i2.AvatarShortNamePipe, typeof i2.AvatarBgColorPipe, typeof i2.AvatarSrcPipe], [typeof i3.CommonModule, typeof i4.ThyIconModule], [typeof i1.ThyAvatarComponent, typeof i2.AvatarShortNamePipe, typeof i2.AvatarBgColorPipe, typeof i2.AvatarSrcPipe]>;
9
+ static ɵmod: i0.ɵɵNgModuleDeclaration<ThyAvatarModule, [typeof i1.ThyAvatarComponent, typeof i2.ThyAvatarListComponent, typeof i3.AvatarShortNamePipe, typeof i3.AvatarBgColorPipe, typeof i3.AvatarSrcPipe], [typeof i4.CommonModule, typeof i5.ThyIconModule], [typeof i1.ThyAvatarComponent, typeof i2.ThyAvatarListComponent, typeof i3.AvatarShortNamePipe, typeof i3.AvatarBgColorPipe, typeof i3.AvatarSrcPipe]>;
9
10
  static ɵinj: i0.ɵɵInjectorDeclaration<ThyAvatarModule>;
10
11
  }
package/avatar/index.d.ts CHANGED
@@ -2,3 +2,4 @@ export * from './avatar.module';
2
2
  export * from './avatar.service';
3
3
  export * from './avatar.pipe';
4
4
  export * from './avatar.component';
5
+ export * from './avatar-list/avatar-list.component';
@@ -0,0 +1,36 @@
1
+ @use "../../styles/variables";
2
+ @use "mixin";
3
+
4
+ .thy-avatar-list {
5
+ display: inline-flex;
6
+ flex: 1;
7
+ line-height: initial;
8
+ width: 100%;
9
+
10
+ .thy-avatar-content {
11
+ position: relative;
12
+ }
13
+ }
14
+
15
+ .thy-avatar-list-overlap {
16
+ @each $size, $value in variables.$avatar-sizes {
17
+ .thy-avatar-#{$size} {
18
+ @include mixin.avatarSize($size, $value);
19
+
20
+ .avatar-default {
21
+ border: 1px solid variables.$white;
22
+ }
23
+ }
24
+ }
25
+ }
26
+
27
+ @each $size, $value in variables.$avatar-sizes {
28
+ .more-#{$size} {
29
+ @include mixin.avatarMoreSize($size, $value);
30
+ background-color: rgba(variables.$black, 0.3);
31
+ color: variables.$white;
32
+ position: absolute;
33
+ left: 0;
34
+ top: 0;
35
+ }
36
+ }
@@ -1,5 +1,7 @@
1
+ @use "sass:meta";
1
2
  @use "../../styles/variables";
2
3
  @use "mixin";
4
+ @use './avatar-list.scss';
3
5
 
4
6
  .thy-avatar {
5
7
  display: inline-block;
@@ -22,3 +22,13 @@
22
22
  }
23
23
  }
24
24
  }
25
+
26
+ @mixin avatarMoreSize($size, $font-size: 12px) {
27
+ width: #{$size}px;
28
+ height: #{$size}px;
29
+ border-radius: 50%;
30
+ line-height: #{$size}px;
31
+ font-size: #{$font-size}px;
32
+ text-align: center;
33
+ display: inline-block;
34
+ }
@@ -26,8 +26,6 @@
26
26
  }
27
27
 
28
28
  &-week-number {
29
- width: 286px;
30
-
31
29
  &-cell {
32
30
  text-align: center;
33
31
  }
@@ -41,7 +41,7 @@
41
41
  &-item {
42
42
  width: 84px;
43
43
  height: 28px;
44
- line-height: 28px;
44
+ line-height: 26px;
45
45
  display: inline-block;
46
46
  text-align: center;
47
47
  border: 1px solid variables.$gray-200;
@@ -135,7 +135,7 @@ $input-box-height: 34px;
135
135
  width:100%;
136
136
 
137
137
  .#{style.$calendar-prefix-cls}-range-part {
138
- width: 286px;
138
+ width: 50%;
139
139
  }
140
140
  }
141
141
 
@@ -0,0 +1,175 @@
1
+ import { __decorate, __metadata } from "tslib";
2
+ import { Component, ContentChild, ContentChildren, ElementRef, EventEmitter, HostBinding, Input, NgZone, Output, QueryList, TemplateRef, ViewChild } from '@angular/core';
3
+ import { InputBoolean, UpdateHostClassService } from 'ngx-tethys/core';
4
+ import { merge, Observable, of, Subject } from 'rxjs';
5
+ import { debounceTime, take, takeUntil } from 'rxjs/operators';
6
+ import { DEFAULT_SIZE, ThyAvatarComponent } from '../avatar.component';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/common";
9
+ import * as i2 from "../avatar.component";
10
+ const AVATAR_ITEM_SPACE = 6;
11
+ const OVERLAP_AVATAR_ITEM_SPACE = -8;
12
+ /**
13
+ * 头像列表组件
14
+ */
15
+ export class ThyAvatarListComponent {
16
+ constructor(elementRef, ngZone) {
17
+ this.elementRef = elementRef;
18
+ this.ngZone = ngZone;
19
+ this.overlapMode = false;
20
+ this.avatarItems = [];
21
+ this.avatarRenderItems = [];
22
+ this.avatarSpace = AVATAR_ITEM_SPACE;
23
+ this.avatarOverlapSpace = OVERLAP_AVATAR_ITEM_SPACE;
24
+ this.ngUnsubscribe$ = new Subject();
25
+ /**
26
+ * 响应式,自动计算宽度存放 avatar
27
+ * @default false
28
+ */
29
+ this.thyResponsive = false;
30
+ /**
31
+ * 头像大小
32
+ * @type 22 | 24 | 28 | 32 | 36 | 44 | 48 | 68 | 110 | 160 | xxs(22px) | xs(24px) | sm(32px) | md(36px) | lg(48px)
33
+ * @default 36
34
+ */
35
+ this.thyAvatarSize = DEFAULT_SIZE;
36
+ /**
37
+ * 是否展示移除按钮
38
+ * @type boolean
39
+ * @default false
40
+ */
41
+ this.thyRemovable = false;
42
+ /**
43
+ * avatar 移除按钮事件
44
+ */
45
+ this.thyRemove = new EventEmitter();
46
+ }
47
+ get more() {
48
+ return this.avatarItems.length - this.avatarRenderItems.length;
49
+ }
50
+ /**
51
+ * 展示方式
52
+ * @type 'overlap'| 'default'
53
+ * @default default
54
+ */
55
+ set thyMode(value) {
56
+ this.overlapMode = value === "overlap" /* ThyAvatarListMode.overlap */;
57
+ }
58
+ /**
59
+ * @private
60
+ */
61
+ set avatarComponents(value) {
62
+ this.avatarItems = value.toArray();
63
+ this.avatarList = value;
64
+ }
65
+ ngOnChanges(changes) {
66
+ if (changes.thyAvatarSize && !changes.thyAvatarSize.firstChange) {
67
+ this.setAvatarSize();
68
+ }
69
+ if (changes.thyMax && !changes.thyMax.firstChange) {
70
+ this.getRenderAvatar();
71
+ }
72
+ }
73
+ ngAfterContentInit() {
74
+ this.setAvatarSize();
75
+ }
76
+ ngAfterViewInit() {
77
+ this.ngZone.onStable.pipe(take(1)).subscribe(() => {
78
+ this.getRenderAvatar();
79
+ });
80
+ if (this.thyResponsive) {
81
+ this.ngZone.runOutsideAngular(() => {
82
+ merge(this.avatarList.changes, this.createResizeObserver(this.elementRef.nativeElement))
83
+ .pipe(debounceTime(100), takeUntil(this.ngUnsubscribe$))
84
+ .subscribe(() => {
85
+ this.getRenderAvatar();
86
+ });
87
+ });
88
+ }
89
+ }
90
+ setAvatarSize() {
91
+ this.avatarItems.forEach((avatar) => {
92
+ avatar.thySize = this.thyAvatarSize;
93
+ });
94
+ this.getRenderAvatar();
95
+ }
96
+ remove(name) {
97
+ this.thyRemove.emit(name);
98
+ }
99
+ getRenderAvatar() {
100
+ const endIndex = this.getEndIndex();
101
+ const max = this.thyMax || this.avatarItems.length;
102
+ const showCount = Math.max(0, Math.min(max, endIndex));
103
+ this.avatarRenderItems = this.avatarItems.slice(0, showCount);
104
+ }
105
+ getEndIndex() {
106
+ if (this.avatarItems.length) {
107
+ const space = this.overlapMode ? this.avatarOverlapSpace : this.avatarSpace;
108
+ const avatarWidth = this.avatarItems[0]._size + space;
109
+ const wrapperWidth = this.elementRef.nativeElement.offsetWidth;
110
+ const appendWidth = this.appendContent ? this.appendContent.nativeElement.offsetWidth : 0;
111
+ const lastAvatarSpecialSpace = this.overlapMode ? space : 0; // overlap 模式下最后一个 avatar 元素占位比其他多 this.avatarSpace 个 px
112
+ return Math.floor((wrapperWidth - appendWidth + lastAvatarSpecialSpace) / avatarWidth);
113
+ }
114
+ else {
115
+ return 0;
116
+ }
117
+ }
118
+ createResizeObserver(element) {
119
+ return typeof ResizeObserver === 'undefined'
120
+ ? of(null)
121
+ : new Observable(observer => {
122
+ const resize = new ResizeObserver(entries => {
123
+ this.ngZone.run(() => {
124
+ observer.next(entries);
125
+ });
126
+ });
127
+ resize.observe(element);
128
+ return () => {
129
+ resize.disconnect();
130
+ };
131
+ });
132
+ }
133
+ ngOnDestroy() {
134
+ this.ngUnsubscribe$.next();
135
+ this.ngUnsubscribe$.complete();
136
+ }
137
+ }
138
+ ThyAvatarListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.0", ngImport: i0, type: ThyAvatarListComponent, deps: [{ token: i0.ElementRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
139
+ ThyAvatarListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.0", type: ThyAvatarListComponent, selector: "thy-avatar-list", inputs: { thyMode: "thyMode", thyResponsive: "thyResponsive", thyMax: "thyMax", thyAvatarSize: "thyAvatarSize", thyRemovable: "thyRemovable" }, outputs: { thyRemove: "thyRemove" }, host: { properties: { "style.margin-left.px": "overlapMode ? -avatarOverlapSpace : 0", "class.thy-avatar-list-overlap": "this.overlapMode" }, classAttribute: "thy-avatar-list" }, providers: [UpdateHostClassService], queries: [{ propertyName: "append", first: true, predicate: ["append"], descendants: true }, { propertyName: "avatarComponents", predicate: ThyAvatarComponent }], viewQueries: [{ propertyName: "appendContent", first: true, predicate: ["appendContent"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<ng-template>\n <ng-content></ng-content>\n</ng-template>\n\n<ng-container *ngFor=\"let avatar of avatarRenderItems; index as index; last as isLast\">\n <div\n class=\"thy-avatar-content\"\n [ngStyle]=\"overlapMode ? { 'margin-left.px': avatarOverlapSpace } : { 'margin-right.px': isLast ? 0 : avatarSpace }\"\n >\n <thy-avatar\n [thySrc]=\"avatar._src\"\n [thyName]=\"avatar.avatarName\"\n [thySize]=\"avatar._size\"\n [thyRemovable]=\"overlapMode ? false : thyRemovable\"\n [thyImgClass]=\"avatar.thyImgClass\"\n [thyDisabled]=\"avatar.thyDisabled\"\n [thyLoading]=\"avatar.thyLoading\"\n [thyFetchPriority]=\"avatar.thyFetchPriority\"\n (thyRemove)=\"remove(avatar.avatarName)\"\n ></thy-avatar>\n <span *ngIf=\"isLast && more && thyResponsive\" ngClass=\"more-{{ avatar._size }}\">+{{ more }} </span>\n </div>\n</ng-container>\n\n<div #appendContent *ngIf=\"append\" [ngStyle]=\"{ 'padding-left.px': avatarSpace }\">\n <ng-container [ngTemplateOutlet]=\"append\"></ng-container>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.ThyAvatarComponent, selector: "thy-avatar", inputs: ["thyShowName", "thySrc", "thyName", "thySize", "thyShowRemove", "thyRemovable", "thyImgClass", "thyDisabled", "thyLoading", "thyFetchPriority"], outputs: ["thyOnRemove", "thyRemove", "thyError"] }] });
140
+ __decorate([
141
+ InputBoolean(),
142
+ __metadata("design:type", Object)
143
+ ], ThyAvatarListComponent.prototype, "thyResponsive", void 0);
144
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.0", ngImport: i0, type: ThyAvatarListComponent, decorators: [{
145
+ type: Component,
146
+ args: [{ selector: 'thy-avatar-list', host: {
147
+ class: 'thy-avatar-list',
148
+ '[style.margin-left.px]': 'overlapMode ? -avatarOverlapSpace : 0'
149
+ }, providers: [UpdateHostClassService], template: "<ng-template>\n <ng-content></ng-content>\n</ng-template>\n\n<ng-container *ngFor=\"let avatar of avatarRenderItems; index as index; last as isLast\">\n <div\n class=\"thy-avatar-content\"\n [ngStyle]=\"overlapMode ? { 'margin-left.px': avatarOverlapSpace } : { 'margin-right.px': isLast ? 0 : avatarSpace }\"\n >\n <thy-avatar\n [thySrc]=\"avatar._src\"\n [thyName]=\"avatar.avatarName\"\n [thySize]=\"avatar._size\"\n [thyRemovable]=\"overlapMode ? false : thyRemovable\"\n [thyImgClass]=\"avatar.thyImgClass\"\n [thyDisabled]=\"avatar.thyDisabled\"\n [thyLoading]=\"avatar.thyLoading\"\n [thyFetchPriority]=\"avatar.thyFetchPriority\"\n (thyRemove)=\"remove(avatar.avatarName)\"\n ></thy-avatar>\n <span *ngIf=\"isLast && more && thyResponsive\" ngClass=\"more-{{ avatar._size }}\">+{{ more }} </span>\n </div>\n</ng-container>\n\n<div #appendContent *ngIf=\"append\" [ngStyle]=\"{ 'padding-left.px': avatarSpace }\">\n <ng-container [ngTemplateOutlet]=\"append\"></ng-container>\n</div>\n" }]
150
+ }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.NgZone }]; }, propDecorators: { overlapMode: [{
151
+ type: HostBinding,
152
+ args: ['class.thy-avatar-list-overlap']
153
+ }], thyMode: [{
154
+ type: Input
155
+ }], thyResponsive: [{
156
+ type: Input
157
+ }], thyMax: [{
158
+ type: Input
159
+ }], thyAvatarSize: [{
160
+ type: Input
161
+ }], thyRemovable: [{
162
+ type: Input
163
+ }], thyRemove: [{
164
+ type: Output
165
+ }], append: [{
166
+ type: ContentChild,
167
+ args: ['append', { static: false }]
168
+ }], avatarComponents: [{
169
+ type: ContentChildren,
170
+ args: [ThyAvatarComponent]
171
+ }], appendContent: [{
172
+ type: ViewChild,
173
+ args: ['appendContent']
174
+ }] } });
175
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avatar-list.component.js","sourceRoot":"","sources":["../../../../../src/avatar/avatar-list/avatar-list.component.ts","../../../../../src/avatar/avatar-list/avatar-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterContentInit,\n    AfterViewInit,\n    Component,\n    ContentChild,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    Input,\n    NgZone,\n    OnChanges,\n    OnDestroy,\n    Output,\n    QueryList,\n    SimpleChanges,\n    TemplateRef,\n    ViewChild\n} from '@angular/core';\nimport { InputBoolean, UpdateHostClassService } from 'ngx-tethys/core';\nimport { merge, Observable, of, Subject } from 'rxjs';\nimport { debounceTime, take, takeUntil } from 'rxjs/operators';\nimport { DEFAULT_SIZE, ThyAvatarComponent } from '../avatar.component';\nimport { SafeAny } from 'ngx-tethys/types';\n\nconst AVATAR_ITEM_SPACE = 6;\n\nconst OVERLAP_AVATAR_ITEM_SPACE = -8;\n\nexport const enum ThyAvatarListMode {\n    overlap = 'overlap',\n    default = 'default'\n}\n/**\n * 头像列表组件\n */\n@Component({\n    selector: 'thy-avatar-list',\n    templateUrl: `./avatar-list.component.html`,\n    host: {\n        class: 'thy-avatar-list',\n        '[style.margin-left.px]': 'overlapMode ? -avatarOverlapSpace : 0'\n    },\n    providers: [UpdateHostClassService]\n})\nexport class ThyAvatarListComponent implements OnChanges, OnDestroy, AfterContentInit, AfterViewInit {\n    @HostBinding('class.thy-avatar-list-overlap') overlapMode = false;\n\n    public avatarItems: ThyAvatarComponent[] = [];\n\n    public avatarRenderItems: ThyAvatarComponent[] = [];\n\n    public avatarSpace = AVATAR_ITEM_SPACE;\n\n    public avatarOverlapSpace = OVERLAP_AVATAR_ITEM_SPACE;\n\n    public get more() {\n        return this.avatarItems.length - this.avatarRenderItems.length;\n    }\n\n    private ngUnsubscribe$ = new Subject<void>();\n\n    private avatarList: QueryList<ThyAvatarComponent>;\n\n    /**\n     * 展示方式\n     * @type  'overlap'| 'default'\n     * @default default\n     */\n    @Input()\n    set thyMode(value: ThyAvatarListMode) {\n        this.overlapMode = value === ThyAvatarListMode.overlap;\n    }\n\n    /**\n     * 响应式，自动计算宽度存放 avatar\n     * @default false\n     */\n    @Input()\n    @InputBoolean()\n    thyResponsive = false;\n\n    /**\n     * 列表组件允许展示 avatar 最大数量\n     */\n    @Input() thyMax: number;\n\n    /**\n     * 头像大小\n     * @type 22 | 24 | 28 | 32 | 36 | 44 | 48 | 68 | 110 | 160 | xxs(22px) | xs(24px) | sm(32px) | md(36px) | lg(48px)\n     * @default 36\n     */\n    @Input() thyAvatarSize: number | string = DEFAULT_SIZE;\n\n    /**\n     * 是否展示移除按钮\n     * @type boolean\n     * @default false\n     */\n    @Input() thyRemovable = false;\n\n    /**\n     * avatar 移除按钮事件\n     */\n    @Output() thyRemove = new EventEmitter<string>();\n\n    /**\n     *  append 自定义操作\n     */\n    @ContentChild('append', { static: false }) append: TemplateRef<SafeAny>;\n\n    /**\n     * @private\n     */\n    @ContentChildren(ThyAvatarComponent)\n    private set avatarComponents(value: QueryList<ThyAvatarComponent>) {\n        this.avatarItems = value.toArray();\n        this.avatarList = value;\n    }\n\n    @ViewChild('appendContent') appendContent: ElementRef<HTMLInputElement>;\n\n    constructor(private elementRef: ElementRef, private ngZone: NgZone) {}\n\n    ngOnChanges(changes: SimpleChanges) {\n        if (changes.thyAvatarSize && !changes.thyAvatarSize.firstChange) {\n            this.setAvatarSize();\n        }\n\n        if (changes.thyMax && !changes.thyMax.firstChange) {\n            this.getRenderAvatar();\n        }\n    }\n\n    ngAfterContentInit() {\n        this.setAvatarSize();\n    }\n\n    ngAfterViewInit() {\n        this.ngZone.onStable.pipe(take(1)).subscribe(() => {\n            this.getRenderAvatar();\n        });\n\n        if (this.thyResponsive) {\n            this.ngZone.runOutsideAngular(() => {\n                merge(this.avatarList.changes, this.createResizeObserver(this.elementRef.nativeElement))\n                    .pipe(debounceTime(100), takeUntil(this.ngUnsubscribe$))\n                    .subscribe(() => {\n                        this.getRenderAvatar();\n                    });\n            });\n        }\n    }\n\n    private setAvatarSize() {\n        this.avatarItems.forEach((avatar: ThyAvatarComponent) => {\n            avatar.thySize = this.thyAvatarSize;\n        });\n        this.getRenderAvatar();\n    }\n\n    public remove(name: string) {\n        this.thyRemove.emit(name);\n    }\n\n    private getRenderAvatar() {\n        const endIndex = this.getEndIndex();\n        const max = this.thyMax || this.avatarItems.length;\n        const showCount = Math.max(0, Math.min(max, endIndex));\n        this.avatarRenderItems = this.avatarItems.slice(0, showCount);\n    }\n\n    private getEndIndex() {\n        if (this.avatarItems.length) {\n            const space = this.overlapMode ? this.avatarOverlapSpace : this.avatarSpace;\n            const avatarWidth = this.avatarItems[0]._size + space;\n            const wrapperWidth = this.elementRef.nativeElement.offsetWidth;\n            const appendWidth = this.appendContent ? this.appendContent.nativeElement.offsetWidth : 0;\n            const lastAvatarSpecialSpace = this.overlapMode ? space : 0; // overlap 模式下最后一个 avatar 元素占位比其他多 this.avatarSpace 个 px\n            return Math.floor((wrapperWidth - appendWidth + lastAvatarSpecialSpace) / avatarWidth);\n        } else {\n            return 0;\n        }\n    }\n\n    private createResizeObserver(element: HTMLElement) {\n        return typeof ResizeObserver === 'undefined'\n            ? of(null)\n            : new Observable(observer => {\n                  const resize = new ResizeObserver(entries => {\n                      this.ngZone.run(() => {\n                          observer.next(entries);\n                      });\n                  });\n                  resize.observe(element);\n                  return () => {\n                      resize.disconnect();\n                  };\n              });\n    }\n\n    ngOnDestroy() {\n        this.ngUnsubscribe$.next();\n        this.ngUnsubscribe$.complete();\n    }\n}\n","<ng-template>\n  <ng-content></ng-content>\n</ng-template>\n\n<ng-container *ngFor=\"let avatar of avatarRenderItems; index as index; last as isLast\">\n  <div\n    class=\"thy-avatar-content\"\n    [ngStyle]=\"overlapMode ? { 'margin-left.px': avatarOverlapSpace } : { 'margin-right.px': isLast ? 0 : avatarSpace }\"\n  >\n    <thy-avatar\n      [thySrc]=\"avatar._src\"\n      [thyName]=\"avatar.avatarName\"\n      [thySize]=\"avatar._size\"\n      [thyRemovable]=\"overlapMode ? false : thyRemovable\"\n      [thyImgClass]=\"avatar.thyImgClass\"\n      [thyDisabled]=\"avatar.thyDisabled\"\n      [thyLoading]=\"avatar.thyLoading\"\n      [thyFetchPriority]=\"avatar.thyFetchPriority\"\n      (thyRemove)=\"remove(avatar.avatarName)\"\n    ></thy-avatar>\n    <span *ngIf=\"isLast && more && thyResponsive\" ngClass=\"more-{{ avatar._size }}\">+{{ more }} </span>\n  </div>\n</ng-container>\n\n<div #appendContent *ngIf=\"append\" [ngStyle]=\"{ 'padding-left.px': avatarSpace }\">\n  <ng-container [ngTemplateOutlet]=\"append\"></ng-container>\n</div>\n"]}