ngx-tethys 14.2.21 → 14.2.22
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/CHANGELOG.md +11 -0
- package/avatar/avatar-list/avatar-list.component.d.ts +75 -0
- package/avatar/avatar.component.d.ts +57 -1
- package/avatar/avatar.module.d.ts +5 -4
- package/avatar/index.d.ts +1 -0
- package/avatar/styles/avatar-list.scss +36 -0
- package/avatar/styles/avatar.scss +2 -0
- package/avatar/styles/mixin.scss +10 -0
- package/esm2020/avatar/avatar-list/avatar-list.component.mjs +175 -0
- package/esm2020/avatar/avatar.component.mjs +47 -3
- package/esm2020/avatar/avatar.module.mjs +5 -4
- package/esm2020/avatar/index.mjs +2 -1
- package/esm2020/comment/comment.component.mjs +1 -1
- package/esm2020/list/list-item-meta.component.mjs +1 -1
- package/esm2020/popover/header/popover-header.component.mjs +9 -5
- package/esm2020/version.mjs +2 -2
- package/fesm2015/ngx-tethys-avatar.mjs +221 -7
- package/fesm2015/ngx-tethys-avatar.mjs.map +1 -1
- package/fesm2015/ngx-tethys-comment.mjs +1 -1
- package/fesm2015/ngx-tethys-comment.mjs.map +1 -1
- package/fesm2015/ngx-tethys-list.mjs +1 -1
- package/fesm2015/ngx-tethys-list.mjs.map +1 -1
- package/fesm2015/ngx-tethys-popover.mjs +8 -4
- package/fesm2015/ngx-tethys-popover.mjs.map +1 -1
- package/fesm2015/ngx-tethys.mjs +1 -1
- package/fesm2015/ngx-tethys.mjs.map +1 -1
- package/fesm2020/ngx-tethys-avatar.mjs +221 -7
- package/fesm2020/ngx-tethys-avatar.mjs.map +1 -1
- package/fesm2020/ngx-tethys-comment.mjs +1 -1
- package/fesm2020/ngx-tethys-comment.mjs.map +1 -1
- package/fesm2020/ngx-tethys-list.mjs +1 -1
- package/fesm2020/ngx-tethys-list.mjs.map +1 -1
- package/fesm2020/ngx-tethys-popover.mjs +8 -4
- package/fesm2020/ngx-tethys-popover.mjs.map +1 -1
- package/fesm2020/ngx-tethys.mjs +1 -1
- package/fesm2020/ngx-tethys.mjs.map +1 -1
- package/package.json +1 -1
- package/popover/header/popover-header.component.d.ts +6 -2
- package/schematics/version.d.ts +1 -1
- package/schematics/version.js +1 -1
- package/styles/variables.scss +1 -0
package/CHANGELOG.md
CHANGED
|
@@ -2,6 +2,17 @@
|
|
|
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.22](https://github.com/atinc/ngx-tethys/compare/14.2.21...14.2.22) (2023-02-27)
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
### Features
|
|
9
|
+
|
|
10
|
+
* **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)
|
|
11
|
+
* **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)
|
|
12
|
+
* 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)
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
|
|
5
16
|
## [14.2.21](https://github.com/atinc/ngx-tethys/compare/14.2.20...14.2.21) (2023-02-22)
|
|
6
17
|
|
|
7
18
|
|
|
@@ -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.
|
|
4
|
-
import * as i3 from "
|
|
5
|
-
import * as i4 from "
|
|
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
|
|
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
|
@@ -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
|
+
}
|
package/avatar/styles/mixin.scss
CHANGED
|
@@ -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"]}
|