From d361650012324f6ec48df317ebee97325b104906 Mon Sep 17 00:00:00 2001 From: Sagi Date: Sat, 1 Oct 2022 09:55:38 +0800 Subject: [PATCH] fix(avatar): fix default style and click event --- .../avatar/src/avatar.component.tsx | 21 ++- .../components/avatar/src/avatar.props.ts | 6 +- .../ui-vue/components/avatar/src/avatar.scss | 69 ++++++++ .../avatar/src/composition/types.ts | 4 +- .../avatar/src/composition/use-image.ts | 36 ++-- .../src/composition/use-text-box.ts | 161 +++++++++--------- packages/ui-vue/src/App.vue | 8 +- .../ui-vue/src/components/button-edit.vue | 18 ++ 8 files changed, 208 insertions(+), 115 deletions(-) create mode 100644 packages/ui-vue/components/avatar/src/avatar.scss create mode 100644 packages/ui-vue/src/components/button-edit.vue diff --git a/packages/ui-vue/components/avatar/src/avatar.component.tsx b/packages/ui-vue/components/avatar/src/avatar.component.tsx index 120fb7f..ffbbb52 100644 --- a/packages/ui-vue/components/avatar/src/avatar.component.tsx +++ b/packages/ui-vue/components/avatar/src/avatar.component.tsx @@ -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 ( -
+
{showLoading && (
加载中
)} - + {!props.readonly && (
)} diff --git a/packages/ui-vue/components/avatar/src/avatar.props.ts b/packages/ui-vue/components/avatar/src/avatar.props.ts index 2c77616..beb8993 100644 --- a/packages/ui-vue/components/avatar/src/avatar.props.ts +++ b/packages/ui-vue/components/avatar/src/avatar.props.ts @@ -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, default: [] }, + type: { type: Array, default: [] } }; export type AvatarProps = ExtractPropTypes; diff --git a/packages/ui-vue/components/avatar/src/avatar.scss b/packages/ui-vue/components/avatar/src/avatar.scss new file mode 100644 index 0000000..0a0d8e7 --- /dev/null +++ b/packages/ui-vue/components/avatar/src/avatar.scss @@ -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; + } + } +} \ No newline at end of file diff --git a/packages/ui-vue/components/avatar/src/composition/types.ts b/packages/ui-vue/components/avatar/src/composition/types.ts index 79e7e23..79c5010 100644 --- a/packages/ui-vue/components/avatar/src/composition/types.ts +++ b/packages/ui-vue/components/avatar/src/composition/types.ts @@ -3,9 +3,11 @@ import { ComputedRef } from 'vue'; export interface UseImage { acceptTypes: ComputedRef; - imageSrc: ComputedRef; + imageSource: ComputedRef; imageTitle: ComputedRef; + + onClickImage: () => void; } export interface ImageFile { diff --git a/packages/ui-vue/components/avatar/src/composition/use-image.ts b/packages/ui-vue/components/avatar/src/composition/use-image.ts index 92189c9..2a01971 100644 --- a/packages/ui-vue/components/avatar/src/composition/use-image.ts +++ b/packages/ui-vue/components/avatar/src/composition/use-image.ts @@ -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): 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 }; } diff --git a/packages/ui-vue/components/button-edit/src/composition/use-text-box.ts b/packages/ui-vue/components/button-edit/src/composition/use-text-box.ts index 8a31537..aa127ae 100644 --- a/packages/ui-vue/components/button-edit/src/composition/use-text-box.ts +++ b/packages/ui-vue/components/button-edit/src/composition/use-text-box.ts @@ -3,103 +3,104 @@ import { ButtonEditProps } from '../button-edit.props'; import { UseTextBox } from './types'; export function useTextBox(props: ButtonEditProps, context: SetupContext, modelValue: Ref, displayText: Ref): 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 + }; } diff --git a/packages/ui-vue/src/App.vue b/packages/ui-vue/src/App.vue index 369c814..54fe791 100644 --- a/packages/ui-vue/src/App.vue +++ b/packages/ui-vue/src/App.vue @@ -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);
- - - - - + diff --git a/packages/ui-vue/src/components/button-edit.vue b/packages/ui-vue/src/components/button-edit.vue new file mode 100644 index 0000000..760575f --- /dev/null +++ b/packages/ui-vue/src/components/button-edit.vue @@ -0,0 +1,18 @@ + + +