Refactor Picture upload
Signed-off-by: Thomas Citharel <tcit@tcit.fr>
This commit is contained in:
parent
5a062ff11b
commit
5e98464017
@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="root">
|
<div class="root">
|
||||||
<figure class="image" v-if="actualImageSrc">
|
<figure class="image" v-if="imageSrc">
|
||||||
<img :src="actualImageSrc" />
|
<img :src="imageSrc" />
|
||||||
</figure>
|
</figure>
|
||||||
<figure class="image is-128x128" v-else>
|
<figure class="image is-128x128" v-else>
|
||||||
<div class="image-placeholder">
|
<div class="image-placeholder">
|
||||||
@ -9,12 +9,19 @@
|
|||||||
</div>
|
</div>
|
||||||
</figure>
|
</figure>
|
||||||
|
|
||||||
<b-upload @input="onFileChanged" :accept="accept">
|
<div class="action-buttons">
|
||||||
<a class="button is-primary">
|
<b-field class="file is-primary">
|
||||||
<b-icon icon="upload"></b-icon>
|
<b-upload @input="onFileChanged" :accept="accept" class="file-label">
|
||||||
|
<span class="file-cta">
|
||||||
|
<b-icon class="file-icon" icon="upload" />
|
||||||
<span>{{ $t("Click to upload") }}</span>
|
<span>{{ $t("Click to upload") }}</span>
|
||||||
</a>
|
</span>
|
||||||
</b-upload>
|
</b-upload>
|
||||||
|
</b-field>
|
||||||
|
<b-button type="is-text" v-if="imageSrc" @click="removeOrClearPicture">
|
||||||
|
{{ $t("Clear") }}
|
||||||
|
</b-button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@ -45,16 +52,22 @@ figure.image {
|
|||||||
color: #eee;
|
color: #eee;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.action-buttons {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
|
import { IPicture } from "@/types/picture.model";
|
||||||
import { Component, Model, Prop, Vue, Watch } from "vue-property-decorator";
|
import { Component, Model, Prop, Vue, Watch } from "vue-property-decorator";
|
||||||
|
|
||||||
@Component
|
@Component
|
||||||
export default class PictureUpload extends Vue {
|
export default class PictureUpload extends Vue {
|
||||||
@Model("change", { type: File }) readonly pictureFile!: File;
|
@Model("change", { type: File }) readonly pictureFile!: File;
|
||||||
|
|
||||||
@Prop({ type: String, required: false }) defaultImageSrc!: string;
|
@Prop({ type: Object, required: false }) defaultImage!: IPicture;
|
||||||
|
|
||||||
@Prop({ type: String, required: false, default: "image/gif,image/png,image/jpeg,image/webp" })
|
@Prop({ type: String, required: false, default: "image/gif,image/png,image/jpeg,image/webp" })
|
||||||
accept!: string;
|
accept!: string;
|
||||||
@ -70,24 +83,40 @@ export default class PictureUpload extends Vue {
|
|||||||
})
|
})
|
||||||
textFallback!: string;
|
textFallback!: string;
|
||||||
|
|
||||||
imageSrc: string | null = null;
|
imageSrc: string | null = this.defaultImage ? this.defaultImage.url : null;
|
||||||
|
|
||||||
|
file!: File | null;
|
||||||
|
|
||||||
mounted(): void {
|
mounted(): void {
|
||||||
|
if (this.pictureFile) {
|
||||||
this.updatePreview(this.pictureFile);
|
this.updatePreview(this.pictureFile);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Watch("pictureFile")
|
@Watch("pictureFile")
|
||||||
onPictureFileChanged(val: File): void {
|
onPictureFileChanged(val: File): void {
|
||||||
|
console.log("onPictureFileChanged", val);
|
||||||
this.updatePreview(val);
|
this.updatePreview(val);
|
||||||
}
|
}
|
||||||
|
|
||||||
onFileChanged(file: File): void {
|
@Watch("defaultImage")
|
||||||
|
onDefaultImageChange(defaultImage: IPicture): void {
|
||||||
|
console.log("onDefaultImageChange", defaultImage);
|
||||||
|
this.imageSrc = defaultImage ? defaultImage.url : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
onFileChanged(file: File | null): void {
|
||||||
this.$emit("change", file);
|
this.$emit("change", file);
|
||||||
|
|
||||||
this.updatePreview(file);
|
this.updatePreview(file);
|
||||||
|
this.file = file;
|
||||||
}
|
}
|
||||||
|
|
||||||
private updatePreview(file?: File) {
|
async removeOrClearPicture(): Promise<void> {
|
||||||
|
this.onFileChanged(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private updatePreview(file?: File | null) {
|
||||||
if (file) {
|
if (file) {
|
||||||
this.imageSrc = URL.createObjectURL(file);
|
this.imageSrc = URL.createObjectURL(file);
|
||||||
return;
|
return;
|
||||||
@ -95,9 +124,5 @@ export default class PictureUpload extends Vue {
|
|||||||
|
|
||||||
this.imageSrc = null;
|
this.imageSrc = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
get actualImageSrc(): string | null {
|
|
||||||
return this.imageSrc || this.defaultImageSrc;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
import gql from "graphql-tag";
|
import gql from "graphql-tag";
|
||||||
|
|
||||||
/* eslint-disable import/prefer-default-export */
|
|
||||||
export const UPLOAD_PICTURE = gql`
|
export const UPLOAD_PICTURE = gql`
|
||||||
mutation UploadPicture($file: Upload!, $alt: String, $name: String!) {
|
mutation UploadPicture($file: Upload!, $alt: String, $name: String!) {
|
||||||
uploadPicture(file: $file, alt: $alt, name: $name) {
|
uploadPicture(file: $file, alt: $alt, name: $name) {
|
||||||
@ -9,3 +8,11 @@ export const UPLOAD_PICTURE = gql`
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
export const REMOVE_PICTURE = gql`
|
||||||
|
mutation RemovePicture($id: ID!) {
|
||||||
|
removePicture(id: $id) {
|
||||||
|
id
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
@ -69,7 +69,7 @@ interface IEventEditJSON {
|
|||||||
visibility: EventVisibility;
|
visibility: EventVisibility;
|
||||||
joinOptions: EventJoinOptions;
|
joinOptions: EventJoinOptions;
|
||||||
draft: boolean;
|
draft: boolean;
|
||||||
picture: IPicture | { pictureId: string } | null;
|
picture?: IPicture | { pictureId: string } | null;
|
||||||
attributedToId: string | null;
|
attributedToId: string | null;
|
||||||
onlineAddress?: string;
|
onlineAddress?: string;
|
||||||
phoneAddress?: string;
|
phoneAddress?: string;
|
||||||
@ -234,7 +234,6 @@ export class EventModel implements IEvent {
|
|||||||
joinOptions: this.joinOptions,
|
joinOptions: this.joinOptions,
|
||||||
draft: this.draft,
|
draft: this.draft,
|
||||||
tags: this.tags.map((t) => t.title),
|
tags: this.tags.map((t) => t.title),
|
||||||
picture: this.picture,
|
|
||||||
onlineAddress: this.onlineAddress,
|
onlineAddress: this.onlineAddress,
|
||||||
phoneAddress: this.phoneAddress,
|
phoneAddress: this.phoneAddress,
|
||||||
physicalAddress: this.physicalAddress,
|
physicalAddress: this.physicalAddress,
|
||||||
|
@ -9,7 +9,7 @@ export async function buildFileFromIPicture(obj: IPicture | null | undefined): P
|
|||||||
return new File([blob], obj.name);
|
return new File([blob], obj.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function buildFileVariable<T>(file: File | null, name: string, alt?: string): Record<string, unknown> {
|
export function buildFileVariable(file: File | null, name: string, alt?: string): Record<string, unknown> {
|
||||||
if (!file) return {};
|
if (!file) return {};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
@ -27,7 +27,7 @@
|
|||||||
<span v-else>{{ $t("I create an identity") }}</span>
|
<span v-else>{{ $t("I create an identity") }}</span>
|
||||||
</h1>
|
</h1>
|
||||||
|
|
||||||
<picture-upload v-model="avatarFile" :defaultImageSrc="avatarUrl" class="picture-upload" />
|
<picture-upload v-model="avatarFile" :defaultImage="identity.avatar" class="picture-upload" />
|
||||||
|
|
||||||
<b-field horizontal :label="$t('Display name')">
|
<b-field horizontal :label="$t('Display name')">
|
||||||
<b-input
|
<b-input
|
||||||
@ -124,6 +124,7 @@ h1 {
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { Component, Prop, Watch } from "vue-property-decorator";
|
import { Component, Prop, Watch } from "vue-property-decorator";
|
||||||
import { mixins } from "vue-class-component";
|
import { mixins } from "vue-class-component";
|
||||||
|
import { IPicture } from "@/types/picture.model";
|
||||||
import {
|
import {
|
||||||
CREATE_PERSON,
|
CREATE_PERSON,
|
||||||
CURRENT_ACTOR_CLIENT,
|
CURRENT_ACTOR_CLIENT,
|
||||||
@ -136,7 +137,7 @@ import { IPerson, Person } from "../../../types/actor";
|
|||||||
import PictureUpload from "../../../components/PictureUpload.vue";
|
import PictureUpload from "../../../components/PictureUpload.vue";
|
||||||
import { MOBILIZON_INSTANCE_HOST } from "../../../api/_entrypoint";
|
import { MOBILIZON_INSTANCE_HOST } from "../../../api/_entrypoint";
|
||||||
import RouteName from "../../../router/name";
|
import RouteName from "../../../router/name";
|
||||||
import { buildFileVariable } from "../../../utils/image";
|
import { buildFileFromIPicture, buildFileVariable } from "../../../utils/image";
|
||||||
import { changeIdentity } from "../../../utils/auth";
|
import { changeIdentity } from "../../../utils/auth";
|
||||||
import identityEditionMixin from "../../../mixins/identityEdition";
|
import identityEditionMixin from "../../../mixins/identityEdition";
|
||||||
|
|
||||||
@ -186,13 +187,6 @@ export default class EditIdentity extends mixins(identityEditionMixin) {
|
|||||||
) as string;
|
) as string;
|
||||||
}
|
}
|
||||||
|
|
||||||
get avatarUrl(): string | null {
|
|
||||||
if (this.identity && this.identity.avatar && this.identity.avatar.url) {
|
|
||||||
return this.identity.avatar.url;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Watch("isUpdate")
|
@Watch("isUpdate")
|
||||||
async isUpdateChanged(): Promise<void> {
|
async isUpdateChanged(): Promise<void> {
|
||||||
this.resetFields();
|
this.resetFields();
|
||||||
@ -286,7 +280,6 @@ export default class EditIdentity extends mixins(identityEditionMixin) {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
this.avatarFile = null;
|
|
||||||
|
|
||||||
this.$notifier.success(
|
this.$notifier.success(
|
||||||
this.$t("Identity {displayName} updated", {
|
this.$t("Identity {displayName} updated", {
|
||||||
|
@ -10,7 +10,11 @@
|
|||||||
|
|
||||||
<form ref="form">
|
<form ref="form">
|
||||||
<subtitle>{{ $t("General information") }}</subtitle>
|
<subtitle>{{ $t("General information") }}</subtitle>
|
||||||
<picture-upload v-model="pictureFile" :textFallback="$t('Headline picture')" />
|
<picture-upload
|
||||||
|
v-model="pictureFile"
|
||||||
|
:textFallback="$t('Headline picture')"
|
||||||
|
:defaultImage="event.picture"
|
||||||
|
/>
|
||||||
|
|
||||||
<b-field :label="$t('Title')" :type="checkTitleLength[0]" :message="checkTitleLength[1]">
|
<b-field :label="$t('Title')" :type="checkTitleLength[0]" :message="checkTitleLength[1]">
|
||||||
<b-input size="is-large" aria-required="true" required v-model="event.title" />
|
<b-input size="is-large" aria-required="true" required v-model="event.title" />
|
||||||
@ -676,6 +680,7 @@ export default class EditEvent extends Vue {
|
|||||||
__typename: "Person",
|
__typename: "Person",
|
||||||
id: organizerActor.id,
|
id: organizerActor.id,
|
||||||
participations: {
|
participations: {
|
||||||
|
__typename: "PaginatedParticipantList",
|
||||||
total: 1,
|
total: 1,
|
||||||
elements: [
|
elements: [
|
||||||
{
|
{
|
||||||
@ -763,11 +768,13 @@ export default class EditEvent extends Vue {
|
|||||||
res.endsOn = null;
|
res.endsOn = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.pictureFile) {
|
||||||
const pictureObj = buildFileVariable(this.pictureFile, "picture");
|
const pictureObj = buildFileVariable(this.pictureFile, "picture");
|
||||||
res = { ...res, ...pictureObj };
|
res = { ...res, ...pictureObj };
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (this.event.picture) {
|
if (this.event.picture && this.pictureFile) {
|
||||||
const oldPictureFile = (await buildFileFromIPicture(this.event.picture)) as File;
|
const oldPictureFile = (await buildFileFromIPicture(this.event.picture)) as File;
|
||||||
const oldPictureFileContent = await readFileAsync(oldPictureFile);
|
const oldPictureFileContent = await readFileAsync(oldPictureFile);
|
||||||
const newPictureFileContent = await readFileAsync(this.pictureFile as File);
|
const newPictureFileContent = await readFileAsync(this.pictureFile as File);
|
||||||
|
@ -31,7 +31,7 @@
|
|||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</nav>
|
</nav>
|
||||||
<section class="container section" v-if="isCurrentActorAGroupAdmin">
|
<section class="container section" v-if="group && isCurrentActorAGroupAdmin">
|
||||||
<form @submit.prevent="updateGroup">
|
<form @submit.prevent="updateGroup">
|
||||||
<b-field :label="$t('Group name')">
|
<b-field :label="$t('Group name')">
|
||||||
<b-input v-model="group.name" />
|
<b-input v-model="group.name" />
|
||||||
@ -43,7 +43,7 @@
|
|||||||
<picture-upload
|
<picture-upload
|
||||||
:textFallback="$t('Avatar')"
|
:textFallback="$t('Avatar')"
|
||||||
v-model="avatarFile"
|
v-model="avatarFile"
|
||||||
:defaultImageSrc="group.avatar ? group.avatar.url : null"
|
:defaultImage="group.avatar"
|
||||||
/>
|
/>
|
||||||
</b-field>
|
</b-field>
|
||||||
|
|
||||||
@ -51,7 +51,7 @@
|
|||||||
<picture-upload
|
<picture-upload
|
||||||
:textFallback="$t('Banner')"
|
:textFallback="$t('Banner')"
|
||||||
v-model="bannerFile"
|
v-model="bannerFile"
|
||||||
:defaultImageSrc="group.banner ? group.banner.url : null"
|
:defaultImage="group.banner"
|
||||||
/>
|
/>
|
||||||
</b-field>
|
</b-field>
|
||||||
<p class="label">{{ $t("Group visibility") }}</p>
|
<p class="label">{{ $t("Group visibility") }}</p>
|
||||||
|
@ -12,7 +12,7 @@
|
|||||||
<picture-upload
|
<picture-upload
|
||||||
v-model="pictureFile"
|
v-model="pictureFile"
|
||||||
:textFallback="$t('Headline picture')"
|
:textFallback="$t('Headline picture')"
|
||||||
:defaultImageSrc="post.picture ? post.picture.url : null"
|
:defaultImage="post.picture"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<b-field
|
<b-field
|
||||||
|
Loading…
Reference in New Issue
Block a user