2018-03-29 01:57:00 -05:00
|
|
|
<template>
|
|
|
|
<div>
|
2018-04-18 17:04:25 -05:00
|
|
|
<h1 class="title">Recipes</h1>
|
2018-03-30 14:31:09 -05:00
|
|
|
|
2018-04-03 18:31:20 -05:00
|
|
|
<router-link v-if="isLoggedIn" :to="{name: 'new_recipe'}" class="button is-primary">Create Recipe</router-link>
|
|
|
|
|
|
|
|
<app-pager :current-page="currentPage" :total-pages="totalPages" paged-item-name="recipe" @changePage="changePage"></app-pager>
|
2018-03-30 14:31:09 -05:00
|
|
|
|
2019-11-10 10:40:26 -06:00
|
|
|
<app-loading v-if="localLoading"></app-loading>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-09-28 20:58:25 -05:00
|
|
|
<table class="table is-fullwidth" :class="{ small: isTouch }">
|
2018-03-30 14:31:09 -05:00
|
|
|
<thead>
|
2019-11-10 10:40:26 -06:00
|
|
|
<tr>
|
|
|
|
<th v-for="h in tableHeader" :key="h.name">
|
|
|
|
<a v-if="h.sort" href="#" @click.prevent="setSort(h.name)">
|
|
|
|
{{h.label}}
|
2020-08-06 20:26:45 -05:00
|
|
|
<app-icon v-if="search.column === h.name" size="sm" :icon="search.direction === 'asc' ? 'caret-bottom' : 'caret-top'"></app-icon>
|
2019-11-10 10:40:26 -06:00
|
|
|
</a>
|
|
|
|
<span v-else>{{h.label}}</span>
|
|
|
|
</th>
|
|
|
|
<th></th>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<td>
|
2020-08-06 20:26:45 -05:00
|
|
|
<app-search-text placeholder="search names" :value="search.name" @input="setSearchName($event)"></app-search-text>
|
2019-11-10 10:40:26 -06:00
|
|
|
</td>
|
|
|
|
<td>
|
2020-08-06 20:26:45 -05:00
|
|
|
<app-search-text placeholder="search tags" :value="search.tags" @input="setSearchTags($event)"></app-search-text>
|
2019-11-10 10:40:26 -06:00
|
|
|
</td>
|
|
|
|
<td colspan="5"></td>
|
|
|
|
</tr>
|
2018-03-30 14:31:09 -05:00
|
|
|
</thead>
|
2019-11-10 10:40:26 -06:00
|
|
|
<transition-group name="fade" tag="tbody">
|
|
|
|
<tr v-for="r in recipes" :key="r.id">
|
|
|
|
<td><router-link :to="{name: 'recipe', params: { id: r.id } }">{{r.name}}</router-link></td>
|
|
|
|
<td>
|
|
|
|
<div class="tags">
|
|
|
|
<span class="tag" v-for="tag in r.tags" :key="tag">{{tag}}</span>
|
|
|
|
</div>
|
|
|
|
</td>
|
|
|
|
<td>
|
|
|
|
<app-rating v-if="r.rating !== null" :value="r.rating" readonly></app-rating>
|
|
|
|
<span v-else>--</span>
|
|
|
|
</td>
|
|
|
|
<td>{{ r.yields }}</td>
|
|
|
|
<td class="recipe-time">{{ formatRecipeTime(r.total_time, r.active_time) }}</td>
|
|
|
|
<td><app-date-time :date-time="r.created_at" :show-time="false"></app-date-time></td>
|
|
|
|
<td>
|
|
|
|
<app-dropdown hover v-if="isLoggedIn" class="is-right">
|
2020-08-06 21:23:31 -05:00
|
|
|
<button slot="button" class="button is-small">
|
2019-11-10 10:40:26 -06:00
|
|
|
<app-icon icon="menu"></app-icon>
|
|
|
|
</button>
|
|
|
|
|
|
|
|
<div class="dropdown-item">
|
|
|
|
<router-link :to="{name: 'new_log', params: { recipeId: r.id } }" class="button is-primary is-fullwidth">
|
|
|
|
<app-icon icon="star" size="md"></app-icon> <span>Add Log Entry</span>
|
|
|
|
</router-link>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="dropdown-item">
|
|
|
|
<router-link :to="{name: 'edit_recipe', params: { id: r.id } }" class="button is-primary is-fullwidth">
|
|
|
|
<app-icon icon="pencil" size="md"></app-icon> <span>Edit Recipe</span>
|
|
|
|
</router-link>
|
2018-04-01 21:43:23 -05:00
|
|
|
</div>
|
2019-11-10 10:40:26 -06:00
|
|
|
|
|
|
|
<div class="dropdown-item">
|
|
|
|
<button type="button" class="button is-danger is-fullwidth" @click="deleteRecipe(r)">
|
|
|
|
<app-icon icon="x" size="md"></app-icon> <span>Delete Recipe</span>
|
2018-09-07 21:56:13 -05:00
|
|
|
</button>
|
2019-11-10 10:40:26 -06:00
|
|
|
</div>
|
2018-09-07 21:56:13 -05:00
|
|
|
|
2019-11-10 10:40:26 -06:00
|
|
|
</app-dropdown>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</transition-group>
|
2018-03-30 14:31:09 -05:00
|
|
|
</table>
|
|
|
|
|
2020-08-06 20:26:45 -05:00
|
|
|
<div v-if="!localLoading && recipes.length === 0">
|
|
|
|
No Recipes
|
|
|
|
</div>
|
|
|
|
|
2018-04-03 18:31:20 -05:00
|
|
|
<app-pager :current-page="currentPage" :total-pages="totalPages" paged-item-name="recipe" @changePage="changePage"></app-pager>
|
2018-05-01 10:55:57 -05:00
|
|
|
<app-confirm :open="showConfirmRecipeDelete" :message="confirmRecipeDeleteMessage" :cancel="recipeDeleteCancel" :confirm="recipeDeleteConfirm"></app-confirm>
|
|
|
|
|
2018-03-29 01:57:00 -05:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
|
2018-03-30 14:31:09 -05:00
|
|
|
import api from "../lib/Api";
|
2024-09-28 20:58:25 -05:00
|
|
|
import { mapWritableState, mapState } from "pinia";
|
2019-11-10 10:40:26 -06:00
|
|
|
import AppLoading from "./AppLoading";
|
2024-09-28 20:58:25 -05:00
|
|
|
import { useAppConfigStore } from "../stores/appConfig";
|
|
|
|
import { useMediaQueryStore } from "../stores/mediaQuery";
|
2018-03-30 14:31:09 -05:00
|
|
|
|
2018-03-29 01:57:00 -05:00
|
|
|
export default {
|
2020-08-06 20:26:45 -05:00
|
|
|
props: {
|
|
|
|
searchQuery: {
|
|
|
|
type: Object,
|
|
|
|
required: false,
|
|
|
|
default: {}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2018-03-30 14:31:09 -05:00
|
|
|
data() {
|
|
|
|
return {
|
2018-04-01 21:43:23 -05:00
|
|
|
recipeData: null,
|
2020-08-06 20:26:45 -05:00
|
|
|
recipeForDeletion: null
|
2018-03-30 14:31:09 -05:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
computed: {
|
2024-09-28 20:58:25 -05:00
|
|
|
...mapState(useMediaQueryStore, { isTouch: store => store.touch }),
|
|
|
|
...mapWritableState(useAppConfigStore, [
|
|
|
|
"initialLoad"
|
2018-09-07 21:56:13 -05:00
|
|
|
]),
|
2020-08-06 20:26:45 -05:00
|
|
|
|
|
|
|
search() {
|
|
|
|
return {
|
|
|
|
name: this.searchQuery.name || null,
|
|
|
|
tags: this.searchQuery.tags || null,
|
|
|
|
column: this.searchQuery.column || "created_at",
|
|
|
|
direction: this.searchQuery.direction || "desc",
|
|
|
|
page: this.searchQuery.page || 1,
|
|
|
|
per: this.searchQuery.per || 25
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2018-03-30 14:31:09 -05:00
|
|
|
recipes() {
|
|
|
|
if (this.recipeData) {
|
|
|
|
return this.recipeData.recipes;
|
|
|
|
} else {
|
|
|
|
return [];
|
|
|
|
}
|
2018-04-01 21:43:23 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
tableHeader() {
|
|
|
|
return [
|
|
|
|
{name: 'name', label: 'Name', sort: true},
|
|
|
|
{name: 'tags', label: 'Tags', sort: false},
|
|
|
|
{name: 'rating', label: 'Rating', sort: true},
|
|
|
|
{name: 'yields', label: 'Yields', sort: false},
|
|
|
|
{name: 'total_time', label: 'Time', sort: true},
|
|
|
|
{name: 'created_at', label: 'Created', sort: true}
|
|
|
|
]
|
2018-04-03 18:31:20 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
totalPages() {
|
|
|
|
if (this.recipeData) {
|
|
|
|
return this.recipeData.total_pages;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
},
|
|
|
|
|
|
|
|
currentPage() {
|
|
|
|
if (this.recipeData) {
|
|
|
|
return this.recipeData.current_page;
|
|
|
|
}
|
|
|
|
return 0;
|
2018-05-01 10:55:57 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
showConfirmRecipeDelete() {
|
|
|
|
return this.recipeForDeletion !== null;
|
|
|
|
},
|
|
|
|
|
|
|
|
confirmRecipeDeleteMessage() {
|
|
|
|
if (this.showConfirmRecipeDelete) {
|
|
|
|
return `Are you sure you want to delete ${this.recipeForDeletion.name}?`;
|
|
|
|
} else {
|
|
|
|
return "??";
|
|
|
|
}
|
2018-03-30 14:31:09 -05:00
|
|
|
}
|
2018-04-01 21:43:23 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
methods: {
|
2020-08-06 20:26:45 -05:00
|
|
|
buildQueryParams() {
|
|
|
|
return {
|
|
|
|
name: this.searchQuery.name,
|
|
|
|
tags: this.searchQuery.tags,
|
|
|
|
column: this.searchQuery.column,
|
|
|
|
direction: this.searchQuery.direction,
|
|
|
|
page: this.searchQuery.page,
|
|
|
|
per: this.searchQuery.per
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
redirectToParams(params) {
|
|
|
|
const rParams = {};
|
|
|
|
|
|
|
|
if (params.name) {
|
|
|
|
rParams.name = params.name;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.tags) {
|
|
|
|
rParams.tags = params.tags;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.column) {
|
|
|
|
rParams.column = params.column;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.direction) {
|
|
|
|
rParams.direction = params.direction;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.page) {
|
|
|
|
rParams.page = params.page;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.per) {
|
|
|
|
rParams.per = params.per;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.$router.push({name: 'recipeList', query: rParams});
|
|
|
|
},
|
2019-11-10 10:40:26 -06:00
|
|
|
|
2018-04-03 18:31:20 -05:00
|
|
|
changePage(idx) {
|
2020-08-06 20:26:45 -05:00
|
|
|
const p = this.buildQueryParams();
|
|
|
|
p.page = idx;
|
|
|
|
this.redirectToParams(p);
|
2018-04-03 18:31:20 -05:00
|
|
|
},
|
|
|
|
|
2018-04-01 21:43:23 -05:00
|
|
|
setSort(col) {
|
2020-08-06 20:26:45 -05:00
|
|
|
const p = this.buildQueryParams();
|
|
|
|
|
|
|
|
if (p.column === col) {
|
|
|
|
p.direction = p.direction === "desc" ? "asc" : "desc";
|
2018-04-01 21:43:23 -05:00
|
|
|
} else {
|
2020-08-06 20:26:45 -05:00
|
|
|
p.column = col;
|
|
|
|
p.direction = "asc";
|
|
|
|
}
|
|
|
|
this.redirectToParams(p);
|
|
|
|
},
|
|
|
|
|
|
|
|
setSearchName(name) {
|
|
|
|
const p = this.buildQueryParams();
|
|
|
|
if (name !== p.name) {
|
|
|
|
p.name = name;
|
|
|
|
p.page = null;
|
|
|
|
this.redirectToParams(p);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
setSearchTags(tags) {
|
|
|
|
const p = this.buildQueryParams();
|
|
|
|
if (tags !== p.tags) {
|
|
|
|
p.tags = tags;
|
|
|
|
p.page = null;
|
|
|
|
this.redirectToParams(p);
|
2018-04-01 21:43:23 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2018-05-01 10:55:57 -05:00
|
|
|
deleteRecipe(recipe) {
|
|
|
|
this.recipeForDeletion = recipe;
|
|
|
|
},
|
|
|
|
|
|
|
|
recipeDeleteConfirm() {
|
|
|
|
if (this.recipeForDeletion !== null) {
|
|
|
|
this.loadResource(
|
|
|
|
api.deleteRecipe(this.recipeForDeletion.id).then(() => {
|
|
|
|
this.recipeForDeletion = null;
|
|
|
|
return this.getList();
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
recipeDeleteCancel() {
|
|
|
|
this.recipeForDeletion = null;
|
|
|
|
},
|
|
|
|
|
2020-08-06 20:26:45 -05:00
|
|
|
getList() {
|
2018-05-01 10:55:57 -05:00
|
|
|
return this.loadResource(
|
2020-08-06 20:26:45 -05:00
|
|
|
api.getRecipeList(this.search.page, this.search.per, this.search.column, this.search.direction, this.search.name, this.search.tags, data => this.recipeData = data)
|
2018-04-01 21:43:23 -05:00
|
|
|
);
|
2020-08-06 20:26:45 -05:00
|
|
|
},
|
2018-04-03 10:29:57 -05:00
|
|
|
|
|
|
|
formatRecipeTime(total, active) {
|
|
|
|
let str = "";
|
|
|
|
|
|
|
|
if (total && total > 0) {
|
|
|
|
str += total;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (active && active > 0) {
|
|
|
|
if (str.length) {
|
|
|
|
str += " (" + active + ")";
|
|
|
|
} else {
|
|
|
|
str += active;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
2018-04-01 21:43:23 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
created() {
|
|
|
|
this.$watch("search",
|
2020-08-06 20:26:45 -05:00
|
|
|
() => {
|
2024-09-28 20:58:25 -05:00
|
|
|
this.getList().then(() => this.initialLoad = true);
|
2020-08-06 20:26:45 -05:00
|
|
|
},
|
2018-04-01 21:43:23 -05:00
|
|
|
{
|
|
|
|
deep: true,
|
|
|
|
immediate: true
|
|
|
|
}
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
components: {
|
2019-11-10 10:40:26 -06:00
|
|
|
AppLoading
|
2018-03-30 14:31:09 -05:00
|
|
|
}
|
2018-03-29 01:57:00 -05:00
|
|
|
}
|
|
|
|
|
2018-07-15 17:00:25 -05:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
.recipe-time {
|
|
|
|
white-space: nowrap;
|
|
|
|
}
|
2018-09-07 21:56:13 -05:00
|
|
|
|
|
|
|
.table th {
|
|
|
|
white-space: nowrap;
|
|
|
|
}
|
|
|
|
|
|
|
|
.table.small {
|
|
|
|
td, th {
|
|
|
|
&:nth-of-type(3), &:nth-of-type(4), &:nth-of-type(5), &:nth-of-type(6) {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-15 17:00:25 -05:00
|
|
|
</style>
|