fix(avatar): fix default style and click event

This commit is contained in:
Sagi 2022-10-01 09:55:38 +08:00
parent a1e6a28668
commit d361650012
8 changed files with 208 additions and 115 deletions

View File

@ -1,10 +1,13 @@
import { defineComponent, computed, ref, SetupContext } from 'vue';
import { avatarProps, AvatarProps } from './avatar.props';
import { useImage } from './composition/use-image';
import './avatar.scss';
export default defineComponent({
name: 'Avatar',
props: avatarProps,
emits: ['change'],
emits: ['change', 'update:modelValue'],
setup(props: AvatarProps, context: SetupContext) {
const avatarClass = computed(() => ({
'f-avatar': true,
@ -13,6 +16,8 @@ export default defineComponent({
'f-avatar-square': props.shape === 'square'
}));
const modelValue = ref(props.modelValue);
const avatarStyle = computed(() => ({
width: props.avatarWidth + 'px',
height: props.avatarHeight + 'px'
@ -29,32 +34,30 @@ export default defineComponent({
function getfiledata() {}
const defaultImgSrc =
'';
const errorImgSrc =
'';
const file=ref(null);
const imageType = computed(() => props.type.join());
const { acceptTypes, imageSource,onClickImage } = useImage(props, context, file, modelValue);
return () => {
return (
<div class={avatarClass.value} style={avatarStyle.value}>
<div class={avatarClass.value} style={avatarStyle.value} onClick={onClickImage}>
{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()} />
<img title={props.tile} class="f-avatar-image" src={imageSource.value} onError={errorSrc()} />
{!props.readonly && (
<div class="f-avatar-icon">
<span class="f-icon f-icon-camera"></span>
</div>
)}
<input
ref="file"
name="file-input"
type="file"
class="f-avatar-upload"
accept={imageType.value}
accept={acceptTypes.value}
onChange={getfiledata}
style="display: none;"
/>

View File

@ -27,6 +27,10 @@ export const avatarProps = {
* , MB
*/
maxSize: { type: Number, default: 1 },
/**
*
*/
modelValue: { type: String, default: '' },
/**
*
*/
@ -34,7 +38,7 @@ export const avatarProps = {
/**
*
*/
type: { type: Array<string>, default: [] },
type: { type: Array<string>, default: [] }
};
export type AvatarProps = ExtractPropTypes<typeof avatarProps>;

View File

@ -0,0 +1,69 @@
.f-avatar{
position: relative;
// width: 100%;
// height: 100%;
cursor: pointer;
overflow: hidden;
&.f-avatar-readonly{
cursor: default;
}
&.f-avatar-circle{
border-radius: 100%;
overflow: hidden;
}
&.f-avatar-square{
border-radius: 0;
}
.f-avatar-image,.f-avatar-defult{
display: inline-block;
width: 100%;
height: 100%;
}
.f-avatar-icon{
display: none;
position: absolute;
left: 0;
right: 0;
top: 0;
bottom: 0;
align-items: center;
justify-content: center;
background: rgba(0, 0, 0, 0.3);
.f-icon{
font-size: 24px;
color: #fff;
}
}
&.f-avatar-circle .f-avatar-icon{
border-radius: 100%;
}
&.f-avatar-square .f-avatar-icon{
border-radius: 0;
}
.f-avatar-upload-loading{
position: absolute;
left: 0;
top: 0;
display: inline-block;
width: 100%;
height: 100%;
background-color: rgba(0,0,0,0.15);
.loading-inner{
position: absolute;
width: 100%;
left: 0;
top: 50%;
margin-top: -25px;
height: 50px;
line-height: 50px;
text-align: center;
font-size: 16px;
color: #fff;
}
}
&:hover{
.f-avatar-icon{
display: flex;
}
}
}

View File

@ -3,9 +3,11 @@ import { ComputedRef } from 'vue';
export interface UseImage {
acceptTypes: ComputedRef<string>;
imageSrc: ComputedRef<string>;
imageSource: ComputedRef<string>;
imageTitle: ComputedRef<string>;
onClickImage: () => void;
}
export interface ImageFile {

View File

@ -1,10 +1,15 @@
import { computed, SetupContext } from 'vue';
import { vue } from '@vitejs/plugin-vue';
import { computed, ref, Ref, SetupContext } from 'vue';
import { AvatarProps } from '../avatar.props';
import { UseImage } from './types';
export function useImage(props: AvatarProps, context: SetupContext, fileInput: HTMLInputElement): UseImage {
const defaultImage = '';
const errorImage = '';
export function useImage(props: AvatarProps, context: SetupContext, fileInput: any, modelValue: Ref<string>): UseImage {
const defaultImage =
'';
const errorImage =
'';
const readonly = ref(props.readonly);
// 判断是否是图片路径
function isUrl(url: string) {
@ -39,33 +44,28 @@ export function useImage(props: AvatarProps, context: SetupContext, fileInput: H
return imageTypesArray.join(',');
});
const imageSrc = computed(() => {
if (!props.cover) {
return defaultImage;
const imageSource = computed(() => {
const image = modelValue.value || props.cover || defaultImage;
if (isUrl(image) || isBase64Image(image)) {
return image;
}
if (isUrl(props.cover)) {
return props.cover;
}
if (isBase64Image(props.cover)) {
return props.cover;
}
return appendBase64ImageHeader(props.cover);
return appendBase64ImageHeader(image);
});
const imageTitle = computed(() => {
return props.readonly ? '' : props.tile;
return readonly.value ? '' : props.tile;
});
function onClickImage() {
if (this.readonly) {
if (readonly.value) {
return;
}
fileInput.click();
fileInput && fileInput.value && fileInput.value.click();
}
function getImageFile() {
return this.imgFileObj;
}
return { acceptTypes, imageSrc, imageTitle };
return { acceptTypes, imageSource, imageTitle, onClickImage };
}

View File

@ -3,103 +3,104 @@ import { ButtonEditProps } from '../button-edit.props';
import { UseTextBox } from './types';
export function useTextBox(props: ButtonEditProps, context: SetupContext, modelValue: Ref<string>, displayText: Ref<string>): UseTextBox {
const textBoxTitle = computed(() => (props.enableTitle ? modelValue.value : ''));
const textBoxTitle = computed(() => (props.enableTitle ? modelValue.value : ''));
const textBoxPlaceholder = computed(() => ((props.disable || props.readonly) && !props.forcePlaceholder ? '' : props.placeholder));
const textBoxPlaceholder = computed(() => ((props.disable || props.readonly) && !props.forcePlaceholder ? '' : props.placeholder));
const isTextBoxReadonly = computed(() => props.readonly || !props.editable);
const isTextBoxReadonly = computed(() => props.readonly || !props.editable);
let focusState = false;
let focusState = false;
const hasFocusedTextBox = computed(() => focusState);
const hasFocusedTextBox = computed(() => focusState);
const textBoxClass = computed(() => ({
'text-left': props.textAlign === 'left',
'text-center': props.textAlign === 'center',
'text-right': props.textAlign === 'right',
'form-control': true,
'f-utils-fill': true,
}));
const textBoxClass = computed(() => ({
'text-left': props.textAlign === 'left',
'text-center': props.textAlign === 'center',
'text-right': props.textAlign === 'right',
'form-control': true,
'f-utils-fill': true
}));
function changeTextBoxValue(newValue: string, showEmitChangeEmit = true) {
if (modelValue.value !== newValue) {
modelValue.value = newValue;
if (showEmitChangeEmit) {
context.emit('change', newValue);
}
function changeTextBoxValue(newValue: string, showEmitChangeEmit = true) {
if (modelValue.value !== newValue) {
modelValue.value = newValue;
if (showEmitChangeEmit) {
context.emit('change', newValue);
}
context.emit('update:modelValue', newValue);
}
}
}
watch(
() => props.modelValue,
(value: string) => context.emit('change', value)
);
watch(
() => props.modelValue,
(value: string) => context.emit('change', value)
);
function onBlurTextBox($event: Event) {
focusState = false;
context.emit('blur', $event);
$event.stopPropagation();
}
function onClickTextBox($event: Event) {
context.emit('click', $event);
}
function onFocusTextBox($event: Event) {
if (props.disable) {
return;
function onBlurTextBox($event: Event) {
focusState = false;
context.emit('blur', $event);
$event.stopPropagation();
}
focusState = true;
if (!isTextBoxReadonly.value) {
context.emit('focus', $event);
function onClickTextBox($event: Event) {
context.emit('click', $event);
}
}
function onInput($event: Event) {
context.emit('input', ($event.target as HTMLInputElement).value);
const newValue = ($event.target as HTMLInputElement).value;
displayText.value = newValue;
if (modelValue.value !== newValue) {
changeTextBoxValue(newValue, false);
context.emit('update:modelValue', ($event.target as HTMLInputElement).value);
function onFocusTextBox($event: Event) {
if (props.disable) {
return;
}
focusState = true;
if (!isTextBoxReadonly.value) {
context.emit('focus', $event);
}
}
}
function onMouseDownTextBox($event: MouseEvent) {
const target = $event.target as HTMLElement;
if (target.tagName !== 'INPUT') {
$event.preventDefault();
function onInput($event: Event) {
context.emit('input', ($event.target as HTMLInputElement).value);
const newValue = ($event.target as HTMLInputElement).value;
displayText.value = newValue;
if (modelValue.value !== newValue) {
changeTextBoxValue(newValue, false);
// context.emit('update:modelValue', ($event.target as HTMLInputElement).value);
}
}
$event.stopPropagation();
}
function onKeyDownTextBox($event: Event) {
context.emit('keydown', $event);
}
function onMouseDownTextBox($event: MouseEvent) {
const target = $event.target as HTMLElement;
if (target.tagName !== 'INPUT') {
$event.preventDefault();
}
$event.stopPropagation();
}
function onKeyUpTextBox($event: Event) {
context.emit('keyup', $event);
}
function onKeyDownTextBox($event: Event) {
context.emit('keydown', $event);
}
function onTextBoxValueChange($event: Event) {
const newValue = ($event.target as HTMLInputElement).value;
changeTextBoxValue(newValue);
}
function onKeyUpTextBox($event: Event) {
context.emit('keyup', $event);
}
return {
hasFocusedTextBox,
isTextBoxReadonly,
textBoxClass,
textBoxPlaceholder,
textBoxTitle,
changeTextBoxValue,
onBlurTextBox,
onClickTextBox,
onFocusTextBox,
onInput,
onKeyDownTextBox,
onKeyUpTextBox,
onMouseDownTextBox,
onTextBoxValueChange,
};
function onTextBoxValueChange($event: Event) {
const newValue = ($event.target as HTMLInputElement).value;
changeTextBoxValue(newValue);
}
return {
hasFocusedTextBox,
isTextBoxReadonly,
textBoxClass,
textBoxPlaceholder,
textBoxTitle,
changeTextBoxValue,
onBlurTextBox,
onClickTextBox,
onFocusTextBox,
onInput,
onKeyDownTextBox,
onKeyUpTextBox,
onMouseDownTextBox,
onTextBoxValueChange
};
}

View File

@ -3,8 +3,8 @@
// Check out https://vuejs.org/api/sfc-script-setup.html#script-setup
import { ref } from "vue";
import HelloWorld from './components/hello-world.vue';
import ButtonEdit from "../components/button-edit/src/button-edit.component";
import Avatar from './components/avatar.vue';
import ButtonEdit from './components/button-edit.vue';
const canEdit = ref(true);
const canAutoComplete = ref(false);
@ -20,12 +20,8 @@ const canAutoComplete = ref(false);
<img src="./assets/vue.svg" class="logo vue" alt="Vue logo" />
</a>
</div>
<input type="checkbox" id="checkbox" v-model="canEdit" />
<label for="checkbox">editable:{{ canEdit }}</label>
<input type="checkbox" id="checkbox" v-model="canAutoComplete" />
<label for="checkbox">auto complete:{{ canAutoComplete }}</label>
<HelloWorld msg="Vite + Vue" />
<ButtonEdit :editable="canEdit" :auto-complete="canAutoComplete" :enable-clear="true"></ButtonEdit>
<ButtonEdit></ButtonEdit>
<Avatar></Avatar>
</template>

View File

@ -0,0 +1,18 @@
<script setup lang="ts">
import { ref } from 'vue';
import ButtonEdit from '../../components/button-edit/src/button-edit.component';
const canEdit = ref(true);
const canAutoComplete = ref(false);
const text = ref('');
</script>
<template>
<input type="checkbox" id="checkbox" v-model="canEdit" />
<label for="checkbox">editable:{{ canEdit }}</label>
<input type="checkbox" id="checkbox" v-model="canAutoComplete" />
<label for="checkbox">auto complete:{{ canAutoComplete }}</label>
<ButtonEdit :editable="canEdit" :auto-complete="canAutoComplete" :enable-clear="true" v-model="text"></ButtonEdit>
Text: {{text}}
</template>