feat: Implement WebcamFileSource for life webcam capture #12

Open
busti wants to merge 51 commits from busti/proto/frontend into jedi/proto/frontend
5 changed files with 45 additions and 33 deletions
Showing only changes of commit ec139531f2 - Show all commits

View file

@ -1,5 +1,5 @@
<template> <template>
<img :src="image_data" :alt="owner + '/' + src"/> <img :src="image_data" :alt="owner + ':' + src"/>
</template> </template>
<style scoped> <style scoped>
@ -42,7 +42,9 @@ export default {
this.servers = await this.getFriendServers({username: this.owner}); this.servers = await this.getFriendServers({username: this.owner});
const response = await this.servers.getRaw(this.signAuth, this.src); const response = await this.servers.getRaw(this.signAuth, this.src);
const mime_type = response.headers.get("content-type"); const mime_type = response.headers.get("content-type");
this.image_data = "data:" + mime_type + ";base64," + btoa(String.fromCharCode(...new Uint8Array(await response.arrayBuffer()))); const base64 = btoa(new Uint8Array(await response.arrayBuffer())
.reduce((data, byte) => data + String.fromCharCode(byte), ""));
this.image_data = "data:" + mime_type + ";base64," + base64;
} catch (e) { } catch (e) {
console.log(e); console.log(e);
} }

View file

@ -11,10 +11,10 @@
</li> </li>
</ul> </ul>
<hr> <hr>
<div> <div style="position: relative;">
<img v-for="file in only_images(files)" :key="file.id" :src="file.name" :alt="file.name" <authenticated-image v-for="file in only_images(item_files).filter(file => file.owner)" :key="file.id"
class="img-thumbnail" :title="file.mime_type"> :owner="file.owner" :src="file.name" class="img-thumbnail"/>
<img v-for="file in only_images(item_files)" :key="file.id" :alt="file.name" <img v-for="file in only_images(item_files).filter(file => file.data)" :key="file.id" :alt="file.name"
:src="'data:' + file.mime_type + ';base64,' + file.data" class="img-thumbnail border-info"> :src="'data:' + file.mime_type + ';base64,' + file.data" class="img-thumbnail border-info">
<fs-file-source @input="addFiles"> <fs-file-source @input="addFiles">
<div class="img-thumbnail btn btn-outline-primary"> <div class="img-thumbnail btn btn-outline-primary">
@ -26,6 +26,15 @@
<b-icon-camera></b-icon-camera> <b-icon-camera></b-icon-camera>
</div> </div>
</camera-file-source> </camera-file-source>
<input type="checkbox" id="file-dropdown" class="invisible-input">
<label class="img-thumbnail btn btn-outline-primary" for="file-dropdown">
<b-icon-plus></b-icon-plus>
</label>
<div class="dropdown-menu">
<authenticated-image v-for="file in only_images(files)" :key="file.id" :src="file.name"
:owner="file.owner"
class="img-thumbnail"/>
</div>
</div> </div>
</drag-drop-file-source> </drag-drop-file-source>
</template> </template>
@ -41,6 +50,25 @@
width: 100%; width: 100%;
height: 100%; height: 100%;
} }
.invisible-input {
display: none;
}
#file-dropdown:checked ~ .dropdown-menu {
display: block;
}
#file-dropdown:checked ~ label {
color: #fff;
background-color: var(--bs-primary);
border-color: var(--bs-primary);
}
#file-dropdown:checked ~ label:hover {
color: var(--bs-primary);
background-color: initial;
}
</style> </style>
<script> <script>
@ -49,11 +77,13 @@ import {mapActions, mapState} from "vuex";
import DragDropFileSource from "@/components/DragDropFileSource.vue"; import DragDropFileSource from "@/components/DragDropFileSource.vue";
import FsFileSource from "@/components/FsFileSource.vue"; import FsFileSource from "@/components/FsFileSource.vue";
import CameraFileSource from "@/components/CameraFileSource.vue"; import CameraFileSource from "@/components/CameraFileSource.vue";
import AuthenticatedImage from "@/components/AuthenticatedImage.vue";
export default { export default {
name: "CombinedFileField", name: "CombinedFileField",
components: { components: {
...BIcons, ...BIcons,
AuthenticatedImage,
DragDropFileSource, DragDropFileSource,
CameraFileSource, CameraFileSource,
FsFileSource FsFileSource
@ -85,7 +115,6 @@ export default {
}); });
}); });
const responses = await Promise.all(jobs); const responses = await Promise.all(jobs);
console.log(responses);
return responses; return responses;
}, },
addFiles(files) { addFiles(files) {
@ -96,11 +125,9 @@ export default {
} }
if (!this.create) { if (!this.create) {
this.uploadFiles(newfiles).then((uploaded) => { this.uploadFiles(newfiles).then((uploaded) => {
console.log(uploaded); this.$emit("change", [...this.item_files, ...uploaded]);
}) })
} }
this.$emit("change", [...this.item_files, ...newfiles]);
console.log(this.item_files);
}, },
only_images(files) { only_images(files) {
@ -108,7 +135,7 @@ export default {
}, },
whithout_images(files) { whithout_images(files) {
return files.filter(file => !file.mime_type.startsWith("image/")); return files.filter(file => !file.mime_type.startsWith("image/"));
}, }
}, },
mounted() { mounted() {
this.fetchFiles(); this.fetchFiles();

View file

@ -176,6 +176,7 @@ export default createStore({
async fetchInventoryItems({commit, dispatch, getters}) { async fetchInventoryItems({commit, dispatch, getters}) {
const servers = await dispatch('getHomeServers') const servers = await dispatch('getHomeServers')
const items = await servers.get(getters.signAuth, '/api/inventory_items/') const items = await servers.get(getters.signAuth, '/api/inventory_items/')
items.map(item => item.files.map(file => file.owner = item.owner))
commit('setInventoryItems', {url: '/', items}) commit('setInventoryItems', {url: '/', items})
return items return items
}, },
@ -258,6 +259,7 @@ export default createStore({
} }
const servers = await dispatch('getHomeServers') const servers = await dispatch('getHomeServers')
const data = await servers.get(getters.signAuth, '/api/files/') const data = await servers.get(getters.signAuth, '/api/files/')
data.map(file => file.owner = state.user)
commit('setFiles', data) commit('setFiles', data)
state.last_load.files = Date.now() state.last_load.files = Date.now()
return data return data
@ -269,7 +271,8 @@ export default createStore({
async pushFile({state, dispatch, getters}, {item_id, file}) { async pushFile({state, dispatch, getters}, {item_id, file}) {
const servers = await dispatch('getHomeServers') const servers = await dispatch('getHomeServers')
const data = await servers.post(getters.signAuth, '/api/item_files/' + item_id + '/', file) const data = await servers.post(getters.signAuth, '/api/item_files/' + item_id + '/', file)
if (data.hash) { if (data.name) {
data.owner = state.user
state.files.push(data) state.files.push(data)
return data return data
} }

View file

@ -35,7 +35,7 @@
<div class="mb-3"> <div class="mb-3">
<label for="image" class="form-label">Image</label> <label for="image" class="form-label">Image</label>
<div> <div>
<authenticated-image v-for="file in files" :key="file.id" :src="file.name" <authenticated-image v-for="file in item.files" :key="file.id" :src="file.name"
:owner="file.owner" class="img-thumbnail border-info"></authenticated-image> :owner="file.owner" class="img-thumbnail border-info"></authenticated-image>
</div> </div>
@ -77,11 +77,6 @@ export default {
BaseLayout, BaseLayout,
...BIcons ...BIcons
}, },
data() {
return {
files: []
}
},
props: { props: {
id: { id: {
type: String, type: String,
@ -99,14 +94,6 @@ export default {
}, },
async mounted() { async mounted() {
await this.fetchInventoryItems() await this.fetchInventoryItems()
console.log(this.id, typeof this.id)
const files = await this.fetchFilesByItem({id: this.id})
this.files = files.map(file => {
return {
...file,
owner: this.item.owner
}
})
} }
} }
</script> </script>

View file

@ -6,13 +6,6 @@
<div class="card"> <div class="card">
<div class="card-header">Edit Item</div> <div class="card-header">Edit Item</div>
<div class="card-body"> <div class="card-body">
<div class="mb-3">
<ul>
<li v-for="file in item.files" :key="file.name">
{{ file.name }} ({{ file.hash }})
</li>
</ul>
</div>
<div class="mb-3"> <div class="mb-3">
<label for="name" class="form-label">Name</label> <label for="name" class="form-label">Name</label>
<input type="text" class="form-control" id="name" name="name" <input type="text" class="form-control" id="name" name="name"