feature: add avatar component
This commit is contained in:
parent
d628e0aaeb
commit
cdf6f35522
|
@ -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');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*判断cover数据格式,做出修改赋值给imgSrc
|
||||||
|
*/
|
||||||
|
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');
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -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>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
},
|
||||||
|
});
|
|
@ -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>;
|
|
@ -0,0 +1,9 @@
|
||||||
|
import { ComputedRef } from 'vue';
|
||||||
|
|
||||||
|
export interface UseImage {
|
||||||
|
acceptTypes: ComputedRef<string>;
|
||||||
|
|
||||||
|
imageSrc: ComputedRef<string>;
|
||||||
|
|
||||||
|
imageTitle: ComputedRef<string>;
|
||||||
|
}
|
|
@ -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);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -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 };
|
||||||
|
}
|
Loading…
Reference in New Issue