client brief > my images : upload images working
|
|
@ -18,4 +18,8 @@ Testimage: - file://4xs4UNZhQf3jLvsG
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
Uuid: 6yh1yt2Sk45Y2sOl
|
Uuid: 6yh1yt2Sk45Y2sOl
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Clientbriefimages: - file://ihuGLrw5vll0R4j2
|
||||||
|
|
@ -1 +1,5 @@
|
||||||
Template: blocks/image
|
Template: blocks/image
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Uuid: vHs87W6sM0r43thr
|
||||||
|
After Width: | Height: | Size: 267 KiB |
|
|
@ -0,0 +1,17 @@
|
||||||
|
Description: nouvelle description
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Tags: matériaux & textures
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Date: 2024-10-09 05:10
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Uuid: vs4tgLg5q6QFeb43
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Template: image
|
||||||
|
Before Width: | Height: | Size: 213 KiB After Width: | Height: | Size: 213 KiB |
|
|
@ -1,3 +1,7 @@
|
||||||
|
Date: 2024-10-09 05:10
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
Description:
|
Description:
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
@ -6,7 +10,7 @@ Tags:
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
Uuid: Ke2XIraa5jlvtOdb
|
Uuid: Nlkd6hX666PBBYlk
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
|
Before Width: | Height: | Size: 119 KiB After Width: | Height: | Size: 119 KiB |
|
|
@ -1,3 +1,7 @@
|
||||||
|
Date: 2024-10-09 05:10
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
Description:
|
Description:
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
@ -6,7 +10,7 @@ Tags:
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
Uuid: nqa5cCGHGoWQXulB
|
Uuid: UkFMsBPXVGpQMmXV
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
|
After Width: | Height: | Size: 232 KiB |
|
|
@ -1,3 +1,7 @@
|
||||||
|
Date: 2024-10-09 05:10
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
Description:
|
Description:
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
@ -6,7 +10,7 @@ Tags:
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
Uuid: FWKdZwxSimGOaO2x
|
Uuid: DknJcJ7YnG29JXUN
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
|
Before Width: | Height: | Size: 114 KiB After Width: | Height: | Size: 114 KiB |
|
|
@ -0,0 +1,17 @@
|
||||||
|
Date: 2024-10-09 05:10
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Description:
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Tags:
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Uuid: zGisP0BopebPl2ex
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Template: image
|
||||||
|
Before Width: | Height: | Size: 116 KiB After Width: | Height: | Size: 116 KiB |
|
|
@ -0,0 +1,17 @@
|
||||||
|
Date: 2024-10-09 05:10
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Description:
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Tags:
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Uuid: 2Nt5Vv2SrGcE1iro
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Template: image
|
||||||
|
Before Width: | Height: | Size: 217 KiB After Width: | Height: | Size: 217 KiB |
|
|
@ -0,0 +1,17 @@
|
||||||
|
Date: 2024-10-09 05:10
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Description:
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Tags:
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Uuid: ih7KMM6KjNoQ2umR
|
||||||
|
|
||||||
|
----
|
||||||
|
|
||||||
|
Template: image
|
||||||
|
|
@ -16,12 +16,13 @@ Description: test
|
||||||
|
|
||||||
Clientbriefimages:
|
Clientbriefimages:
|
||||||
|
|
||||||
- file://b4Ywx4ProE0Smc0Q
|
- file://DknJcJ7YnG29JXUN
|
||||||
- file://aCuD18YBmLHg6YaS
|
- file://Nlkd6hX666PBBYlk
|
||||||
- file://Ke2XIraa5jlvtOdb
|
- file://zGisP0BopebPl2ex
|
||||||
- file://QBax5c69DcENi2rt
|
- file://ih7KMM6KjNoQ2umR
|
||||||
- file://nqa5cCGHGoWQXulB
|
- file://UkFMsBPXVGpQMmXV
|
||||||
- file://FWKdZwxSimGOaO2x
|
- file://2Nt5Vv2SrGcE1iro
|
||||||
|
- file://vs4tgLg5q6QFeb43
|
||||||
|
|
||||||
----
|
----
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1,13 +0,0 @@
|
||||||
Description:
|
|
||||||
|
|
||||||
----
|
|
||||||
|
|
||||||
Tags: forme & design, coloris & nuances
|
|
||||||
|
|
||||||
----
|
|
||||||
|
|
||||||
Uuid: QBax5c69DcENi2rt
|
|
||||||
|
|
||||||
----
|
|
||||||
|
|
||||||
Template: image
|
|
||||||
|
|
@ -1,13 +0,0 @@
|
||||||
Description: test de description mais
|
|
||||||
|
|
||||||
----
|
|
||||||
|
|
||||||
Tags: parachèvements
|
|
||||||
|
|
||||||
----
|
|
||||||
|
|
||||||
Uuid: b4Ywx4ProE0Smc0Q
|
|
||||||
|
|
||||||
----
|
|
||||||
|
|
||||||
Template: image
|
|
||||||
|
|
@ -4,19 +4,19 @@ return [
|
||||||
'pattern' => 'upload-images.json',
|
'pattern' => 'upload-images.json',
|
||||||
'method' => 'POST',
|
'method' => 'POST',
|
||||||
'action' => function () {
|
'action' => function () {
|
||||||
if ($uploads = kirby()->request()->files()) {
|
if ($uploads = kirby()->request()->files()) {
|
||||||
$pageUri = kirby()->request()->query()->get('pageUri');
|
$pageUri = kirby()->request()->query()->get('pageUri');
|
||||||
$page = page($pageUri);
|
$page = page($pageUri);
|
||||||
|
|
||||||
|
$allFiles = $page->clientBriefImages()->toFiles()->pluck('uuid', ',');
|
||||||
|
|
||||||
$alerts = [];
|
$alerts = [];
|
||||||
$success = '';
|
|
||||||
$newFiles = [];
|
|
||||||
$allFiles = [];
|
|
||||||
|
|
||||||
foreach ($uploads->get('images') as $upload) {
|
foreach ($uploads->get('images') as $upload) {
|
||||||
// check for duplicate
|
// check for duplicate
|
||||||
$files = $page->files();
|
$files = $page->files();
|
||||||
$duplicates = $files->filter(function ($file) use ($upload) {
|
$duplicates = $files->filter(function ($file) use ($upload) {
|
||||||
|
// get original safename without prefix
|
||||||
$pos = strpos($file->filename(), '_');
|
$pos = strpos($file->filename(), '_');
|
||||||
$originalSafename = substr($file->filename(), $pos + 1);
|
$originalSafename = substr($file->filename(), $pos + 1);
|
||||||
|
|
||||||
|
|
@ -26,9 +26,14 @@ return [
|
||||||
});
|
});
|
||||||
|
|
||||||
if ($duplicates->count() > 0) {
|
if ($duplicates->count() > 0) {
|
||||||
$alerts[$upload['name']] = "The file already exists";
|
$duplicate = $duplicates->first();
|
||||||
|
|
||||||
|
if (in_array($duplicate->uuid(), $allFiles)) {
|
||||||
|
$allFiles[] = $duplicate->uuid();
|
||||||
|
$alerts[$upload['name']] = "The file already exists, added to the field.";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$name = crc32($upload['name'].microtime()). '_' . $upload['name'];
|
$name = crc32($upload['name'].microtime()). '_' . $upload['name'];
|
||||||
$newFile = $page->createFile([
|
$newFile = $page->createFile([
|
||||||
|
|
@ -49,19 +54,24 @@ return [
|
||||||
$alerts[$upload['name']] = $e->getMessage();
|
$alerts[$upload['name']] = $e->getMessage();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$images = array_map(function ($file) {
|
|
||||||
return [
|
|
||||||
'url' => $file->url(),
|
|
||||||
'uuid' => $file->uuid()
|
|
||||||
];
|
|
||||||
}, $newFiles);
|
|
||||||
|
|
||||||
$newPage = $page->update([
|
$newPage = $page->update([
|
||||||
'clientBriefImages' => $allFiles
|
'clientBriefImages' => $allFiles
|
||||||
]);
|
]);
|
||||||
|
|
||||||
return compact('images', 'alerts', 'success');
|
$images = [];
|
||||||
|
|
||||||
|
foreach ($newPage->clientBriefImages()->toFiles() as $image) {
|
||||||
|
$images[] = [
|
||||||
|
'url' => $image->url(),
|
||||||
|
'uuid' => (string) $image->uuid(),
|
||||||
|
'tags' => $image->tags()->split(),
|
||||||
|
'name' => $image->filename(),
|
||||||
|
'description' => $image->description()->value(),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return compact('images', 'alerts');
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
|
|
|
||||||
|
|
@ -12,7 +12,7 @@
|
||||||
>
|
>
|
||||||
Ajouter une ou plusieurs images
|
Ajouter une ou plusieurs images
|
||||||
</button>
|
</button>
|
||||||
<template v-for="image in images" :key="image.uri">
|
<template v-for="image in page.images" :key="image.uri">
|
||||||
<figure
|
<figure
|
||||||
v-if="
|
v-if="
|
||||||
selectedTags.length === 0 ||
|
selectedTags.length === 0 ||
|
||||||
|
|
@ -35,7 +35,6 @@
|
||||||
<AddImagesModal
|
<AddImagesModal
|
||||||
v-if="isAddImagesModalOpen"
|
v-if="isAddImagesModalOpen"
|
||||||
:isAddImagesModalOpen="isAddImagesModalOpen"
|
:isAddImagesModalOpen="isAddImagesModalOpen"
|
||||||
:images="images"
|
|
||||||
@close="isAddImagesModalOpen = false"
|
@close="isAddImagesModalOpen = false"
|
||||||
/>
|
/>
|
||||||
<ImageDetailsModal
|
<ImageDetailsModal
|
||||||
|
|
@ -53,51 +52,23 @@ import { usePageStore } from "../../../stores/page";
|
||||||
import { ref } from "vue";
|
import { ref } from "vue";
|
||||||
import ImageDetailsModal from "./ImageDetailsModal.vue";
|
import ImageDetailsModal from "./ImageDetailsModal.vue";
|
||||||
import AddImagesModal from "./add-images-modal/AddImagesModal.vue";
|
import AddImagesModal from "./add-images-modal/AddImagesModal.vue";
|
||||||
|
import { storeToRefs } from "pinia";
|
||||||
|
|
||||||
const { page } = usePageStore();
|
const { page } = storeToRefs(usePageStore());
|
||||||
|
|
||||||
const selectedTags = ref([]);
|
const selectedTags = ref([]);
|
||||||
const imageDetails = ref(null);
|
const imageDetails = ref(null);
|
||||||
const images = ref(page.images);
|
|
||||||
const isAddImagesModalOpen = ref(false);
|
const isAddImagesModalOpen = ref(false);
|
||||||
|
|
||||||
function onAdvancedUpload(event) {
|
|
||||||
if (event.xhr.status === 200) {
|
|
||||||
toast.add({
|
|
||||||
severity: "success",
|
|
||||||
summary: "Upload réussi",
|
|
||||||
detail: event.xhr.response.success,
|
|
||||||
life: 3000,
|
|
||||||
});
|
|
||||||
const response = JSON.parse(event.xhr.response);
|
|
||||||
console.log(response);
|
|
||||||
images.value = response.images;
|
|
||||||
} else {
|
|
||||||
toast.add({
|
|
||||||
severity: "error",
|
|
||||||
summary: "Échec de l'upload",
|
|
||||||
detail: event.xhr.response.error,
|
|
||||||
life: 3000,
|
|
||||||
});
|
|
||||||
console.error(JSON.parse(event.xhr.response));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function beforeSend(event) {
|
|
||||||
const formData = event.formData;
|
|
||||||
formData.append(
|
|
||||||
"pageUri",
|
|
||||||
"projects/miss-dior-blooming-bouquet/client-brief"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function changeSelectedTags(newTags) {
|
function changeSelectedTags(newTags) {
|
||||||
selectedTags.value = newTags;
|
selectedTags.value = newTags;
|
||||||
}
|
}
|
||||||
|
|
||||||
function removeImage(target) {
|
function removeImage(target) {
|
||||||
console.log("remove", target);
|
console.log("remove", target);
|
||||||
images.value = images.value.filter((image) => image.uuid !== target.uuid);
|
page.value.images = page.value.images.filter(
|
||||||
|
(image) => image.uuid !== target.uuid
|
||||||
|
);
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -26,7 +26,7 @@
|
||||||
<div class="bg-grey-50 | rounded-2xl | p-8 | overflow-y">
|
<div class="bg-grey-50 | rounded-2xl | p-8 | overflow-y">
|
||||||
<component :is="activeTab.component" :params="activeTab.params" />
|
<component :is="activeTab.component" :params="activeTab.params" />
|
||||||
</div>
|
</div>
|
||||||
<ImagesEditPanel :images="images" />
|
<ImagesEditPanel />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
|
|
@ -66,9 +66,8 @@ import MyImages from "./MyImages.vue";
|
||||||
import { usePageStore } from "../../../../stores/page";
|
import { usePageStore } from "../../../../stores/page";
|
||||||
import { ref, watch } from "vue";
|
import { ref, watch } from "vue";
|
||||||
|
|
||||||
const { isAddImagesModalOpen, images } = defineProps({
|
const { isAddImagesModalOpen } = defineProps({
|
||||||
isAddImagesModalOpen: Boolean,
|
isAddImagesModalOpen: Boolean,
|
||||||
images: Array,
|
|
||||||
});
|
});
|
||||||
const emit = defineEmits(["update:isAddImagesModalOpen"]);
|
const emit = defineEmits(["update:isAddImagesModalOpen"]);
|
||||||
|
|
||||||
|
|
@ -83,9 +82,6 @@ const tabs = [
|
||||||
{
|
{
|
||||||
name: "Mes images",
|
name: "Mes images",
|
||||||
component: MyImages,
|
component: MyImages,
|
||||||
params: {
|
|
||||||
images,
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Matériauthèque",
|
name: "Matériauthèque",
|
||||||
|
|
|
||||||
|
|
@ -8,7 +8,7 @@
|
||||||
style="--row-gap: var(--space-8)"
|
style="--row-gap: var(--space-8)"
|
||||||
>
|
>
|
||||||
<AccordionPanel
|
<AccordionPanel
|
||||||
v-for="(image, index) in images"
|
v-for="(image, index) in page.images"
|
||||||
:key="index + 1"
|
:key="index + 1"
|
||||||
:value="index"
|
:value="index"
|
||||||
class="w-full | bg-white | rounded-xl | p-12 pt-8"
|
class="w-full | bg-white | rounded-xl | p-12 pt-8"
|
||||||
|
|
@ -109,18 +109,15 @@ import AccordionContent from "primevue/accordioncontent";
|
||||||
import { usePageStore } from "../../../../stores/page";
|
import { usePageStore } from "../../../../stores/page";
|
||||||
import { toPascalCase } from "../../../../helpers";
|
import { toPascalCase } from "../../../../helpers";
|
||||||
import debounce from "lodash/debounce";
|
import debounce from "lodash/debounce";
|
||||||
|
import { storeToRefs } from "pinia";
|
||||||
|
|
||||||
const { images } = defineProps({
|
const { page } = storeToRefs(usePageStore());
|
||||||
images: Array,
|
|
||||||
});
|
|
||||||
|
|
||||||
const { page } = usePageStore();
|
|
||||||
|
|
||||||
function saveTags(image) {
|
function saveTags(image) {
|
||||||
const headers = {
|
const headers = {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
pageUri: page.uri,
|
pageUri: page.value.uri,
|
||||||
fileName: image.name,
|
fileName: image.name,
|
||||||
properties: [
|
properties: [
|
||||||
{
|
{
|
||||||
|
|
@ -144,7 +141,7 @@ const saveDescription = debounce((image) => {
|
||||||
const headers = {
|
const headers = {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
pageUri: page.uri,
|
pageUri: page.value.uri,
|
||||||
fileName: image.name,
|
fileName: image.name,
|
||||||
properties: [
|
properties: [
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -41,7 +41,7 @@
|
||||||
<div v-if="files.length > 0">Fichiers importés</div>
|
<div v-if="files.length > 0">Fichiers importés</div>
|
||||||
</template>
|
</template>
|
||||||
</FileUpload>
|
</FileUpload>
|
||||||
<figure v-for="image in params.images" class="image">
|
<figure v-for="image in page.images" class="image">
|
||||||
<img :src="image.url" alt="" />
|
<img :src="image.url" alt="" />
|
||||||
</figure>
|
</figure>
|
||||||
<Toast />
|
<Toast />
|
||||||
|
|
@ -53,11 +53,30 @@ import { usePageStore } from "../../../../stores/page";
|
||||||
import Toast from "primevue/toast";
|
import Toast from "primevue/toast";
|
||||||
import { useToast } from "primevue/usetoast";
|
import { useToast } from "primevue/usetoast";
|
||||||
import FileUpload from "primevue/fileupload";
|
import FileUpload from "primevue/fileupload";
|
||||||
|
import { storeToRefs } from "pinia";
|
||||||
|
|
||||||
const { params } = defineProps({
|
const { page } = storeToRefs(usePageStore());
|
||||||
params: Object,
|
|
||||||
});
|
|
||||||
|
|
||||||
const { page } = usePageStore();
|
|
||||||
const toast = useToast();
|
const toast = useToast();
|
||||||
|
|
||||||
|
function onAdvancedUpload(event) {
|
||||||
|
if (event.xhr.status === 200) {
|
||||||
|
toast.add({
|
||||||
|
severity: "success",
|
||||||
|
summary: "Upload réussi",
|
||||||
|
detail: event.xhr.response.success,
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
const response = JSON.parse(event.xhr.response);
|
||||||
|
console.log(response);
|
||||||
|
page.value.images = response.images;
|
||||||
|
} else {
|
||||||
|
toast.add({
|
||||||
|
severity: "error",
|
||||||
|
summary: "Échec de l'upload",
|
||||||
|
detail: event.xhr.response.error,
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
console.error(JSON.parse(event.xhr.response));
|
||||||
|
}
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
|
||||||