feature: add avatar component

This commit is contained in:
Sagi 2022-09-22 23:37:52 +08:00
parent d628e0aaeb
commit cdf6f35522
7 changed files with 557 additions and 0 deletions

View File

@ -0,0 +1,308 @@
import { Component, OnInit, ViewChildren, ElementRef, Input, Output, EventEmitter, HostListener, ViewChild, forwardRef } from '@angular/core';
import { Observable, Subject } from 'rxjs';
import { NotifyService } from '@farris/ui-notify';
import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';
import { LocaleService } from '@farris/ui-locale';
export interface upImageFile {
size: number;
name: string;
type: string;
lastModified?: string;
lastModifiedDate?: Date;
state?: string;
base64?: string;
}
@Component({
selector: 'farris-avatar',
templateUrl: './avatar.component.html',
styleUrls: ['./avatar.component.scss'],
providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => AvatarComponent),
multi: true
}
]
})
export class AvatarComponent implements ControlValueAccessor, OnInit {
private defaultImgSrc = ''
private errorImgSrc = '';
public imgSrc;
// public fileBinary: string;
@ViewChild('file') file: ElementRef;
// @Input() cover = '';
tReadOnly: boolean = false;
@Input()
set readonly(value: boolean) {
if (value !== this.tReadOnly) {
let localeTitle = this.localeService.getValue('avatar.imgtitle');
this.imgtitle = value ? '' : (this.imgTitle ? this.imgTitle : localeTitle);
this.tReadOnly = value;
}
};
get readonly(): boolean {
return this.tReadOnly;
}
// @Input() type;
@Input() size: number = 1;
@Input() imgTitle: string;
@Input() avatarWidth: number = 100;
@Input() avatarHeight: number = 100;
@Input() imgShape: string = 'circle';
imgtitle: string = '点击修改';
// @Input() isBase64:boolean = true;
@Output('imgChange') imgChange = new EventEmitter();
_type;
@Input()
set type(val) {
if (val && val.length) {
let types = val;
if (typeof val === 'string') {
types = val.split(',');
}
if (types && types.length) {
this.currentImgType = [];
types.forEach(t => {
if ((typeof t == 'string') && t.constructor == String) {
let tImgtype = 'image/' + t;
if (t === 'jpg') {
let jpgType = 'image/jpeg';
this.currentImgType.push(jpgType);
}
this.currentImgType.push(tImgtype);
}
});
if (this.currentImgType.length > 0) {
this.imgType = this.currentImgType.join(',')
}
}
}
}
get type() {
return this._type;
}
_cover;
@Input()
set cover(val) {
if (val) {
this._cover = val;
this.imgsrcInit(this.cover);
}
else {
this.imgSrc = this.defaultImgSrc;
}
}
get cover() {
return this._cover;
}
private onChangeCallback: Function = () => { }
private onTouchedCallback: Function = () => { }
//是否加载中
loadingImg: boolean;
imgType = 'image/*';
imgFileObj: upImageFile;
currentImgType = ['image/image', 'image/webp', 'image/png', 'image/svg', 'image/gif', 'image/jpg', 'image/jpeg', 'image/bmp'];
constructor(private notifyService: NotifyService, public localeService: LocaleService) {
this.notifyService.config.position = 'top-center';
}
ngOnInit() {
// if(this.cover){
// this.imgsrcInit(this.cover);
// }
// else{
// this.imgSrc = this.defaultImgSrc;
// }
if (this.readonly) {
this.imgtitle = '';
}
else if (this.imgTitle) {
this.imgtitle = this.imgTitle;
}
else {
this.imgtitle = this.localeService.getValue('avatar.imgtitle');
}
}
/*coverimgSrc
*/
imgsrcInit(val) {
let isImg = this.isSrc(val);
if (isImg) {
this.imgSrc = val;
}
else {
let isFullBase64 = this.isBaseSrc(val);
if (isFullBase64) {
this.imgSrc = val
}
else {
this.imgSrc = this.addBase64(val);
}
}
}
@HostListener('click')
onClick(): void {
if (this.readonly) {
return;
}
(this.file.nativeElement as HTMLInputElement).click();
}
getfiledata(event) {
if (this.readonly) {
return;
}
const filetarget = event.target as HTMLInputElement;
let getfile = filetarget.files;
if (!getfile[0]) {
return;
}
let fileType = getfile[0].type;
const isLtSize = getfile[0].size / 1024 / 1024 < this.size;
if (this.currentImgType.indexOf(fileType) < 0) {
let typeerrorText = this.localeService.getValue('avatar.typeError');
this.notifyService.error({
type: 'error',
title: '',
msg: typeerrorText
});
filetarget.value = '';
// this.notifyService.error('上传图片类型不正确');
return;
}
if (!isLtSize) {
let sizeerrorText = this.localeService.getValue('avatar.sizeError');
let errormes: string = sizeerrorText + this.size + "M!";
this.notifyService.error({
type: 'error',
title: '',
msg: errormes
});
filetarget.value = '';
// this.notifyService.error(`上传图片不能大于${this.size}M!`);
return;
}
this.transformFile(getfile[0]);
filetarget.value = '';
}
public getImgFileObj() {
return this.imgFileObj;
}
transformFile(getfile: any) {
// const subject = new Subject();
this.imgFileObj = {
size: getfile.size,
name: getfile.name,
type: getfile.type,
lastModified: getfile.lastModified,
lastModifiedDate: getfile.lastModifiedDate
}
this.do(getfile).subscribe(res => {
this.loadingImg = false;
if (res['state'] === 'done') {
this.imgSrc = res['result'];
//this.onChangeCallback(this.imgSrc);
this.onChangeCallback(this.removeBase64(this.imgSrc));
this.onTouchedCallback();
}
else if (res['state'] === 'error') {
let uploaderrorText = this.localeService.getValue('avatar.uploadError');
this.notifyService.error({
type: 'error',
title: '',
msg: uploaderrorText
})
// this.notifyService.error('图片上传失败,请重试!');
}
this.imgFileObj.state = res['state'];
this.imgFileObj.base64 = res['result'];
this.imgChange.emit(this.imgFileObj);
});
}
read(file: File): Observable<string> {
return Observable.create(observer => {
const reader = new FileReader();
reader.readAsDataURL(file);
//this.loadingImg = true;
// reader.onloadstart=function(){}
reader.onload = () => {
//console.log(reader.result);
observer.next({ state: 'done', 'result': reader.result });
observer.complete();
};
reader.onerror = function () {
observer.next({ state: 'error', 'result': reader.result });
observer.complete();
}
});
}
do(file: any): Observable<string> {
return this.read(file as File);
}
writeValue(val: any): void {
if (val && val.length) {
// if(this.isBase64){
// this.imgSrc = this.addBase64(val);
// }
// else{
// this.imgSrc = val;
// }
this.imgsrcInit(val);
}
else if (this.cover) {
this.imgsrcInit(this.cover);
}
else {
this.imgSrc = this.defaultImgSrc;
}
}
registerOnChange(fn: any): void {
this.onChangeCallback = fn;
}
registerOnTouched(fn: any): void {
this.onTouchedCallback = fn;
}
// setDisabledState?(isDisabled: boolean): void {
// this.disabled = isDisabled;
// }
addBase64(val) {
if (!val) return;
return 'data:image/jpeg;base64,' + val;
}
removeBase64(val) {
if (!val) return;
let img_arr = val.split(',');
if (img_arr.length) {
return img_arr[1];
}
}
//判断是否是图片路径
isSrc(url) {
return (url.match(/\.(jpeg|jpg|gif|png|svg|bmp|webp)$/) != null)
}
//判断是否是完成base64
isBaseSrc(url) {
return (url.indexOf('data:image/') > -1 ? true : false)
}
errorSrc() {
this.imgSrc = this.errorImgSrc;
this.imgtitle = this.localeService.getValue('avatar.loadError');
}
}

View File

@ -0,0 +1,66 @@
import { defineComponent, computed, ref, SetupContext } from 'vue';
import { avatarProps, AvatarProps } from './avatar.props';
export default defineComponent({
name: 'Avatar',
props: avatarProps,
emits: ['change'],
setup(props: AvatarProps, context: SetupContext) {
const avatarClass = computed(() => ({
'f-avatar': true,
'f-avatar-readonly': props.readonly,
'f-avatar-circle': props.shape === 'circle',
'f-avatar-square': props.shape === 'square',
}));
const avatarStyle = computed(() => ({
width: props.avatarWidth + 'px',
height: props.avatarHeight + 'px',
}));
let showLoading = false;
let imgSrc = '';
const currentImgType = ['image/image', 'image/webp', 'image/png', 'image/svg', 'image/gif', 'image/jpg', 'image/jpeg', 'image/bmp'];
function errorSrc() {
return '';
}
function getfiledata() {}
const defaultImgSrc =
'';
const errorImgSrc =
'';
const imageType = computed(() => props.type.join());
return () => {
return (
<div class={avatarClass.value} style={avatarStyle.value}>
{showLoading && (
<div class="f-avatar-upload-loading">
<div class="loading-inner"></div>
</div>
)}
<img title={props.tile} class="f-avatar-image" src={imgSrc} onError={errorSrc()} />
{!props.readonly && (
<div class="f-avatar-icon">
<span class="f-icon f-icon-camera"></span>
</div>
)}
<input
name="file-input"
type="file"
class="f-avatar-upload"
accept={imageType.value}
onChange={getfiledata}
style="display: none;"
/>
</div>
);
};
},
});

View File

@ -0,0 +1,40 @@
import { ExtractPropTypes, PropType } from 'vue';
type AvatarShap = 'square' | 'circle';
export const avatarProps = {
/**
*
*/
avatarWidth: { type: Number, default: 100 },
/**
*
*/
avatarHeight: { type: Number, default: 100 },
/**
*
*/
cover: { type: String },
/**
*
*/
readonly: { type: Boolean, default: false },
/**
*
*/
shape: { type: String as PropType<AvatarShap>, default: 'circle' },
/**
* , MB
*/
maxSize: { type: Number, default: 1 },
/**
*
*/
tile: { type: String, default: '' },
/**
*
*/
type: { type: Array<string>, default: [] },
};
export type AvatarProps = ExtractPropTypes<typeof avatarProps>;

View File

@ -0,0 +1,9 @@
import { ComputedRef } from 'vue';
export interface UseImage {
acceptTypes: ComputedRef<string>;
imageSrc: ComputedRef<string>;
imageTitle: ComputedRef<string>;
}

View File

@ -0,0 +1,74 @@
import { SetupContext } from 'vue';
import { AvatarProps } from '../avatar.props';
export function useFile(props: AvatarProps, context: SetupContext, allowTypes: string[]) {
function getFileData($event: Event) {
if (props.readonly) {
return;
}
const fileInput = $event.target as HTMLInputElement;
const selectedFiles = fileInput.files;
if (!selectedFiles || !selectedFiles[0]) {
return;
}
const fileType = selectedFiles[0].type;
const isLtSize = selectedFiles[0].size / 1024 / 1024 < props.maxSize;
if (allowTypes.indexOf(fileType) < 0) {
const typeErrorMessage = this.localeService.getValue('avatar.typeError');
this.notifyService.error({
type: 'error',
title: '',
msg: typeErrorMessage,
});
fileInput.value = '';
return;
}
if (!isLtSize) {
const sizeErrorMessageTemplate = this.localeService.getValue('avatar.sizeError');
const sizeErrorMessage: string = sizeErrorMessageTemplate + props.maxSize + 'MB!';
this.notifyService.error({
type: 'error',
title: '',
msg: sizeErrorMessage,
});
fileInput.value = '';
return;
}
this.transformFile(selectedFiles[0]);
fileInput.value = '';
}
transformFile(getfile: any) {
// const subject = new Subject();
this.imgFileObj = {
size: getfile.size,
name: getfile.name,
type: getfile.type,
lastModified: getfile.lastModified,
lastModifiedDate: getfile.lastModifiedDate
}
this.do(getfile).subscribe(res => {
this.loadingImg = false;
if (res['state'] === 'done') {
this.imgSrc = res['result'];
//this.onChangeCallback(this.imgSrc);
this.onChangeCallback(this.removeBase64(this.imgSrc));
this.onTouchedCallback();
}
else if (res['state'] === 'error') {
let uploaderrorText = this.localeService.getValue('avatar.uploadError');
this.notifyService.error({
type: 'error',
title: '',
msg: uploaderrorText
})
// this.notifyService.error('图片上传失败,请重试!');
}
this.imgFileObj.state = res['state'];
this.imgFileObj.base64 = res['result'];
this.imgChange.emit(this.imgFileObj);
});
}
}

View File

@ -0,0 +1,60 @@
import { computed, SetupContext } from 'vue';
import { AvatarProps } from '../avatar.props';
import { UseImage } from './types';
export function useImage(props: AvatarProps, context: SetupContext): UseImage {
const defaultImage = '';
const errorImage = '';
// 判断是否是图片路径
function isUrl(url: string) {
return url.match(/\.(jpeg|jpg|gif|png|svg|bmp|webp)$/) != null;
}
// 判断是否是完成base64
function isBase64Image(url: string) {
return url.indexOf('data:image/') > -1;
}
function appendBase64ImageHeader(val) {
if (!val) {
return '';
}
return 'data:image/jpeg;base64,' + val;
}
const acceptTypes = computed(() => {
if (!props.type || !props.type.length) {
return '';
}
const imageTypesArray = props.type.map((fileType: string) => {
if (fileType === 'jpg') {
fileType = 'jpeg';
}
return `image/${fileType}`;
});
if (!imageTypesArray || !imageTypesArray.length) {
return 'image/*';
}
return imageTypesArray.join(',');
});
const imageSrc = computed(() => {
if (!props.cover) {
return defaultImage;
}
if (isUrl(props.cover)) {
return props.cover;
}
if (isBase64Image(props.cover)) {
return props.cover;
}
return appendBase64ImageHeader(props.cover);
});
const imageTitle = computed(() => {
return props.readonly ? '' : props.tile;
});
return { acceptTypes, imageSrc, imageTitle };
}