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 97 additions and 5 deletions
Showing only changes of commit c8fdfbb09a - Show all commits

View file

@ -0,0 +1,55 @@
<template>
<div>
<!--img src=""-->
data:{{ image_data }}<br>
owner:{{ owner }}<br>
src:{{ src }}<br>
servers:{{ servers }}
</div>
</template>
<style scoped>
img {
width: 190px;
height: 107px;
object-fit: contain;
}
</style>
<script>
import {mapActions, mapGetters} from "vuex";
export default {
name: "AuthenticatedImage",
props: {
src: {
type: String,
required: true
},
owner: {
type: String,
required: true
}
},
data() {
return {
image_data: "",
servers: []
}
},
computed: {
...mapGetters(["signAuth"])
},
methods: {
...mapActions(["getFriendServers"])
},
async mounted() {
try {
this.servers = await this.getFriendServers({username: this.owner});
this.image_data = await this.servers.get(this.signAuth, this.src);
} catch (e) {
console.log(e);
}
}
}
</script>

View file

@ -1,7 +1,8 @@
class NeighborsCache { class NeighborsCache {
constructor() { constructor() {
//this._max_age = 1000 * 60 * 60; // 1 hour //this._max_age = 1000 * 60 * 60; // 1 hour
this._max_age = 1000 * 60 * 5; // 5 minutes //this._max_age = 1000 * 60 * 5; // 5 minutes
this._max_age = 1000 * 15; // 15 seconds
this._cache = JSON.parse(localStorage.getItem('neighbor-cache')) || {}; this._cache = JSON.parse(localStorage.getItem('neighbor-cache')) || {};
} }

View file

@ -2,7 +2,7 @@ import {createStore} from 'vuex';
import router from '@/router'; import router from '@/router';
import FallBackResolver from "@/dns"; import FallBackResolver from "@/dns";
import NeighborsCache from "@/neigbors"; import NeighborsCache from "@/neigbors";
import {createSignAuth, createTokenAuth, createNullAuth, ServerSet, ServerSetUnion} from "@/federation"; import {createNullAuth, createSignAuth, createTokenAuth, ServerSet, ServerSetUnion} from "@/federation";
export default createStore({ export default createStore({
@ -262,6 +262,10 @@ export default createStore({
state.last_load.files = Date.now() state.last_load.files = Date.now()
return data return data
}, },
async fetchFilesByItem({state, commit, dispatch, getters}, {id}) {
const servers = await dispatch('getHomeServers')
return await servers.get(getters.signAuth, '/api/item_files/' + id + '/')
},
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)

View file

@ -32,6 +32,19 @@
<input type="text" class="form-control" id="image" name="image" <input type="text" class="form-control" id="image" name="image"
placeholder="Enter image" v-model="item.image"> placeholder="Enter image" v-model="item.image">
</div--> </div-->
<div class="mb-3">
<label for="image" class="form-label">Image</label>
<div>
<img v-for="file in files" :key="file.id" :alt="file.name"
:src="'https://toolshed.j3d1.de'+file.name" class="img-thumbnail border-info">
<!-- TODO replace dirty hack with proper solution -->
</div>
<div>
<authenticated-image v-for="file in files" :key="file.id" :src="file.name"
:owner="file.owner"></authenticated-image>
</div>
</div>
</div> </div>
</div> </div>
<!-- actions --> <!-- actions -->
@ -59,14 +72,21 @@ import BaseLayout from "@/components/BaseLayout.vue";
import TagField from "@/components/TagField.vue"; import TagField from "@/components/TagField.vue";
import PropertyField from "@/components/PropertyField.vue"; import PropertyField from "@/components/PropertyField.vue";
import {mapActions, mapGetters} from "vuex"; import {mapActions, mapGetters} from "vuex";
import AuthenticatedImage from "@/components/AuthenticatedImage.vue";
export default { export default {
name: "InventoryDetail", name: "InventoryDetail",
components: { components: {
AuthenticatedImage,
PropertyField, TagField, PropertyField, TagField,
BaseLayout, BaseLayout,
...BIcons ...BIcons
}, },
data() {
return {
files: []
}
},
props: { props: {
id: { id: {
type: String, type: String,
@ -80,14 +100,26 @@ export default {
} }
}, },
methods: { methods: {
...mapActions(["fetchInventoryItems", "deleteInventoryItem"]) ...mapActions(["fetchInventoryItems", "deleteInventoryItem", "fetchFilesByItem"]),
}, },
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>
<style scoped> <style scoped>
img {
width: 190px;
height: 107px;
object-fit: contain;
}
</style> </style>

View file

@ -23,7 +23,7 @@ export default defineConfig({
'Content-Security-Policy': 'default-src \'self\';' 'Content-Security-Policy': 'default-src \'self\';'
+ ' script-src \'self\' \'wasm-unsafe-eval\';' + ' script-src \'self\' \'wasm-unsafe-eval\';'
+ ' style-src \'self\' \'unsafe-inline\';' + ' style-src \'self\' \'unsafe-inline\';'
+ ' img-src \'self\' data:; ' + ' img-src \'self\' * data:; '
+ ' connect-src * data:', // TODO: change * to https://* for production + ' connect-src * data:', // TODO: change * to https://* for production
}, },
/*https: { /*https: {