feat: Implement WebcamFileSource for life webcam capture #12

Open
busti wants to merge 51 commits from busti/proto/frontend into jedi/proto/frontend
12 changed files with 121 additions and 40 deletions
Showing only changes of commit 297ad7bd99 - Show all commits

View file

@ -6,14 +6,7 @@
<a class="sidebar-toggle d-flex"> <a class="sidebar-toggle d-flex">
<i class="hamburger align-self-center"></i> <i class="hamburger align-self-center"></i>
</a> </a>
<form class="d-none d-sm-inline-block"> <SearchBox/>
<div class="input-group input-group-navbar">
<input type="text" class="form-control" placeholder="Search…" aria-label="Search">
<button class="btn" type="button">
<b-icon-search class="bi-valign-middle"></b-icon-search>
</button>
</div>
</form>
<div class="navbar-collapse collapse"> <div class="navbar-collapse collapse">
<ul class="navbar-nav navbar-align"> <ul class="navbar-nav navbar-align">
<Notifications :notifications="notifications"/> <Notifications :notifications="notifications"/>
@ -30,32 +23,26 @@
<script> <script>
import {mapGetters, mapMutations, mapState} from 'vuex'; import {mapGetters, mapMutations, mapState} from 'vuex';
import * as BIcons from "bootstrap-icons-vue";
import Notifications from "@/components/Notifications.vue"; import Notifications from "@/components/Notifications.vue";
import Messages from "@/components/Messages.vue"; import Messages from "@/components/Messages.vue";
import Footer from "@/components/Footer.vue"; import Footer from "@/components/Footer.vue";
import Sidebar from "@/components/Sidebar.vue"; import Sidebar from "@/components/Sidebar.vue";
import UserDropdown from "@/components/UserDropdown.vue"; import UserDropdown from "@/components/UserDropdown.vue";
import SearchBox from "@/components/SearchBox.vue";
export default { export default {
name: 'BaseLayout', name: 'BaseLayout',
components: { components: {
SearchBox,
UserDropdown, UserDropdown,
Sidebar, Sidebar,
Footer, Footer,
Messages, Messages,
Notifications, Notifications
...BIcons
}, },
computed: { computed: {
...mapState(['messages']), ...mapState(['messages']),
...mapGetters(['notifications']), ...mapGetters(['notifications'])
username() {
return this.$route.params.username
},
},
methods: {
...mapMutations(['logout'])
}, },
async mounted() { async mounted() {
} }

View file

@ -82,7 +82,6 @@
</template> </template>
<script> <script>
import {mapGetters, mapMutations} from 'vuex';
import * as BIcons from "bootstrap-icons-vue"; import * as BIcons from "bootstrap-icons-vue";
export default { export default {

View file

@ -0,0 +1,44 @@
<template>
<form class="d-none d-sm-inline-block">
<div class="input-group input-group-navbar">
<input type="text" class="form-control" placeholder="Search…" aria-label="Search" v-model="query" id="searchText">
<button class="btn" type="button" @click.prevent="search">
<b-icon-search class="bi-valign-middle"></b-icon-search>
</button>
</div>
</form>
</template>
<script>
import * as BIcons from "bootstrap-icons-vue";
export default {
name: "SearchBox",
components: {
...BIcons
},
data() {
return {
query: ""
}
},
methods: {
search() {
if(this.query.length > 0)
this.$router.push("/search/" + encodeURIComponent(this.query));
else
document.getElementById("searchText").focus();
}
},
mounted() {
console.log(this.$route)
console.log(this.$route.params.query)
this.query = decodeURIComponent(this.$route.params.query || encodeURIComponent(""));
}
}
</script>
<style scoped>
</style>

View file

@ -26,9 +26,13 @@
</template> </template>
<script> <script>
import * as BIcons from "bootstrap-icons-vue";
export default { export default {
name: "Sidebar" name: "Sidebar",
components: {
...BIcons
},
} }
</script> </script>

View file

@ -8,10 +8,10 @@
<a class="nav-link dropdown-toggle d-none d-sm-inline-block" href="#" <a class="nav-link dropdown-toggle d-none d-sm-inline-block" href="#"
data-toggle="dropdown"> data-toggle="dropdown">
<!--<img src="/static/assets/img/avatars/avatar.png" class="avatar img-fluid rounded mr-1" <img src="/assets/img/avatars/avatar.png" class="avatar img-fluid rounded mr-1"
alt="Charles Hall"/>--> alt="Charles Hall"/>
<span class="text-dark"> <span class="text-dark">
<!--{{ request.user.username }}--> {{ username }}
</span> </span>
</a> </a>
@ -32,12 +32,22 @@
</template> </template>
<script> <script>
import * as BIcons from "bootstrap-icons-vue";
import Messages from "@/components/Messages.vue"; import {mapMutations} from "vuex";
export default { export default {
name: "UserDropdown", name: "UserDropdown",
components: {Messages} components: {
...BIcons
},
methods: {
...mapMutations(['logout'])
},
computed: {
username() {
return this.$route.params.username
},
}
} }
</script> </script>

View file

@ -7,6 +7,7 @@ import Profile from '@/views/Profile.vue';
import Settings from '@/views/Settings.vue'; import Settings from '@/views/Settings.vue';
import Inventory from '@/views/Inventory.vue'; import Inventory from '@/views/Inventory.vue';
import Friends from "@/views/Friends.vue"; import Friends from "@/views/Friends.vue";
import Search from "@/views/Search.vue";
import InventoryNew from "@/views/InventoryNew.vue"; import InventoryNew from "@/views/InventoryNew.vue";
import InventoryEdit from "@/views/InventoryEdit.vue"; import InventoryEdit from "@/views/InventoryEdit.vue";
import InventoryDetail from "@/views/InventoryDetail.vue"; import InventoryDetail from "@/views/InventoryDetail.vue";
@ -21,6 +22,7 @@ const routes = [
{path: '/inventory/:id/edit', component: InventoryEdit, meta: {requiresAuth: true}}, {path: '/inventory/:id/edit', component: InventoryEdit, meta: {requiresAuth: true}},
{path: '/inventory/new', component: InventoryNew, meta: {requiresAuth: true}}, {path: '/inventory/new', component: InventoryNew, meta: {requiresAuth: true}},
{path: '/friends', component: Friends, meta: {requiresAuth: true}}, {path: '/friends', component: Friends, meta: {requiresAuth: true}},
{path: '/search/:query', component: Search, meta: {requiresAuth: true}},
{path: '/login', component: Login, meta: {requiresAuth: false}}, {path: '/login', component: Login, meta: {requiresAuth: false}},
{path: '/register', component: Register, meta: {requiresAuth: false}}, {path: '/register', component: Register, meta: {requiresAuth: false}},
] ]

View file

@ -84,7 +84,6 @@ export default createStore({
const j = await dispatch('apiLocalGet', {target: '/auth/keys/'}) const j = await dispatch('apiLocalGet', {target: '/auth/keys/'})
const k = j.key const k = j.key
commit('setKey', k) commit('setKey', k)
await router.push({path: '/'});
return true; return true;
} else { } else {
return false; return false;

View file

@ -1,7 +1,3 @@
<script setup>
</script>
<template> <template>
<BaseLayout> <BaseLayout>
<main class="content"> <main class="content">

View file

@ -1,7 +1,3 @@
<script setup>
</script>
<template> <template>
<BaseLayout> <BaseLayout>
<main class="content"> <main class="content">

View file

@ -1,7 +1,3 @@
<script setup>
</script>
<template> <template>
<BaseLayout> <BaseLayout>
<main class="content"> <main class="content">

View file

@ -66,6 +66,7 @@
<script> <script>
import {mapActions, mapMutations} from 'vuex'; import {mapActions, mapMutations} from 'vuex';
import router from "@/router";
export default { export default {
name: 'Login', name: 'Login',
@ -82,7 +83,13 @@ export default {
...mapMutations(['setRemember']), ...mapMutations(['setRemember']),
async do_login(e) { async do_login(e) {
e.preventDefault(); e.preventDefault();
if (!await this.login({username: this.username, password: this.password, remember: this.remember})) { if (await this.login({username: this.username, password: this.password, remember: this.remember})) {
if (this.$route.query.redirect) {
await router.push({path: this.$route.query.redirect});
} else {
await router.push({path: '/'});
}
} else {
this.msg = 'Invalid username or password'; this.msg = 'Invalid username or password';
} }

View file

@ -0,0 +1,41 @@
<template>
<BaseLayout>
<main class="content">
<div class="container-fluid p-0">
<h1 class="h3 mb-3">Blank Page</h1>
<div class="row">
<div class="col-12">
<div class="card">
<div class="card-header">
<h5 class="card-title mb-0">Empty card</h5>
</div>
<div class="card-body">
<div class="logo">
<img src="/src/assets/icons/toolshed-48x48.png" alt="Toolshed logo">
</div>
</div>
</div>
</div>
</div>
</div>
</main>
</BaseLayout>
</template>
<script>
import {mapGetters, mapMutations} from 'vuex';
import * as BIcons from "bootstrap-icons-vue";
import BaseLayout from "@/components/BaseLayout.vue";
export default {
name: 'Search',
components: {
...BIcons,
BaseLayout
},
}
</script>
<style scoped>
</style>