empty states started, minor fixes, try later added
This commit is contained in:
parent
52446b3cd7
commit
399fc6bc20
15 changed files with 915 additions and 545 deletions
|
@ -1,29 +1,28 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<resources xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
|
||||
<!-- Application theme -->
|
||||
<style name="AppTheme" parent="AppThemeBase">
|
||||
<item name="android:datePickerStyle">
|
||||
@style/SpinnerDatePicker
|
||||
</item>
|
||||
<item name="android:timePickerStyle">
|
||||
@style/SpinnerTimePicker
|
||||
</item>
|
||||
<style name="AppThemeBase21" parent="AppThemeBase">
|
||||
<item name="android:windowTranslucentStatus">true</item>
|
||||
<item name="android:datePickerStyle">@style/SpinnerDatePicker</item>
|
||||
<item name="android:timePickerStyle">@style/SpinnerTimePicker</item>
|
||||
</style>
|
||||
|
||||
<style name="AppTheme" parent="AppThemeBase21">
|
||||
</style>
|
||||
|
||||
<!-- Default style for DatePicker - in spinner mode -->
|
||||
<style name="SpinnerDatePicker" parent="android:Widget.Material.Light.DatePicker">
|
||||
<item name="android:datePickerMode">
|
||||
spinner
|
||||
</item>
|
||||
<item name="android:datePickerMode">spinner</item>
|
||||
</style>
|
||||
|
||||
<!-- Default style for TimePicker - in spinner mode -->
|
||||
<style name="SpinnerTimePicker" parent="android:Widget.Material.Light.TimePicker">
|
||||
<item name="android:timePickerMode">
|
||||
spinner
|
||||
</item>
|
||||
<item name="android:timePickerMode">spinner</item>
|
||||
</style>
|
||||
|
||||
<style name="NativeScriptToolbarStyle" parent="NativeScriptToolbarStyleBase">
|
||||
<item name="android:elevation">
|
||||
0dp
|
||||
</item>
|
||||
<item name="android:elevation">4dp</item>
|
||||
<item name="android:paddingTop">24dp</item>
|
||||
</style>
|
||||
</resources>
|
12
app/App_Resources/Android/src/main/res/values-v29/styles.xml
Normal file
12
app/App_Resources/Android/src/main/res/values-v29/styles.xml
Normal file
|
@ -0,0 +1,12 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
|
||||
<!-- Application theme -->
|
||||
<style name="AppThemeBase29" parent="AppThemeBase21">
|
||||
<item name="android:forceDarkAllowed">true</item>
|
||||
</style>
|
||||
|
||||
<style name="AppTheme" parent="AppThemeBase29">
|
||||
</style>
|
||||
|
||||
</resources>
|
|
@ -20,9 +20,7 @@
|
|||
|
||||
<!-- theme to use AFTER launch screen is loaded-->
|
||||
<style name="AppThemeBase" parent="Theme.AppCompat.Light.NoActionBar">
|
||||
<item name="android:forceDarkAllowed">false</item>
|
||||
<item name="toolbarStyle">@style/NativeScriptToolbarStyle</item>
|
||||
|
||||
<item name="colorPrimary">@color/ns_primary</item>
|
||||
<item name="colorPrimaryDark">@color/ns_primaryDark</item>
|
||||
<item name="colorAccent">@color/ns_accent</item>
|
||||
|
|
168
app/app.scss
168
app/app.scss
|
@ -53,6 +53,10 @@ Page {
|
|||
.fieldLabel {
|
||||
background: $grayL4;
|
||||
}
|
||||
.view-reference-text,
|
||||
.view-reference-container {
|
||||
background: white;
|
||||
}
|
||||
.option-highlight {
|
||||
background: $grayL2;
|
||||
}
|
||||
|
@ -108,7 +112,9 @@ Page {
|
|||
background: $grayD4;
|
||||
}
|
||||
.recipe-li,
|
||||
.option-highlight {
|
||||
.option-highlight,
|
||||
.view-reference-text,
|
||||
.view-reference-container {
|
||||
background: $grayD3;
|
||||
}
|
||||
.sd-item,
|
||||
|
@ -151,7 +157,7 @@ TimePickerField {
|
|||
width: 100%;
|
||||
border-width: 1;
|
||||
font-size: 14;
|
||||
padding: 14;
|
||||
padding: 14 14 13;
|
||||
margin: 8 0 0 0;
|
||||
border-radius: 4;
|
||||
border-color: $gray;
|
||||
|
@ -188,7 +194,7 @@ TabView {
|
|||
// ActionBar
|
||||
ActionBar {
|
||||
width: 100%;
|
||||
margin: 0;
|
||||
margin: 24 0 0 0;
|
||||
padding: 0;
|
||||
height: 64;
|
||||
.bx {
|
||||
|
@ -215,6 +221,10 @@ SearchBar {
|
|||
}
|
||||
|
||||
// Side Drawer
|
||||
.sd {
|
||||
margin-top: 24;
|
||||
}
|
||||
|
||||
.sd-item {
|
||||
border-radius: 4;
|
||||
padding: 0 16;
|
||||
|
@ -238,7 +248,7 @@ SearchBar {
|
|||
|
||||
// Home
|
||||
RadListView {
|
||||
margin-bottom: 128;
|
||||
margin: 8 0 128;
|
||||
}
|
||||
.recipe-li {
|
||||
background: white;
|
||||
|
@ -265,12 +275,29 @@ RadListView {
|
|||
color: $orange;
|
||||
}
|
||||
}
|
||||
.emptyState {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
.noResults {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
font-size: 16;
|
||||
line-height: 8;
|
||||
padding: 32 16;
|
||||
padding: 64 16;
|
||||
text-align: center;
|
||||
horizontal-alignment: center;
|
||||
// vertical-alignment: center;
|
||||
.title {
|
||||
font-size: 20;
|
||||
text-align: center;
|
||||
}
|
||||
.bx {
|
||||
font-size: 64;
|
||||
text-align: center;
|
||||
color: $gray;
|
||||
margin-bottom: 16;
|
||||
}
|
||||
}
|
||||
.swipe-item {
|
||||
margin: 0 8;
|
||||
|
@ -336,11 +363,11 @@ RadListView {
|
|||
// View Recipe
|
||||
|
||||
.view-cat {
|
||||
font-size: 14;
|
||||
font-size: 16;
|
||||
color: #ff7043;
|
||||
}
|
||||
.view-other {
|
||||
font-size: 14;
|
||||
font-size: 16;
|
||||
}
|
||||
.view-title {
|
||||
font-size: 22;
|
||||
|
@ -348,46 +375,71 @@ RadListView {
|
|||
margin-bottom: 16;
|
||||
}
|
||||
.view-ingredient {
|
||||
font-size: 14;
|
||||
font-size: 16;
|
||||
line-height: 6;
|
||||
padding-bottom: 16;
|
||||
}
|
||||
.view-favorited {
|
||||
color: #ff7043;
|
||||
padding-bottom: 12;
|
||||
}
|
||||
.activity-indicator {
|
||||
background: #ff7043;
|
||||
}
|
||||
|
||||
.view-count {
|
||||
font-size: 10;
|
||||
width: 20;
|
||||
height: 20;
|
||||
padding-top: 3%;
|
||||
margin: 0 0 0 6;
|
||||
font-size: 12;
|
||||
width: 24;
|
||||
height: 24;
|
||||
padding-top: 4%;
|
||||
margin: 0 0 0 8;
|
||||
text-align: center;
|
||||
border-radius: 100;
|
||||
&.note {
|
||||
clip-path: polygon(
|
||||
5% 0,
|
||||
95% 0,
|
||||
100% 5%,
|
||||
100% 65%,
|
||||
65% 100%,
|
||||
5% 100%,
|
||||
0 95%,
|
||||
0 5%
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
.view-instruction,
|
||||
.view-note,
|
||||
.view-reference {
|
||||
font-size: 14;
|
||||
font-size: 16;
|
||||
line-height: 6;
|
||||
padding: 0 0 16 24;
|
||||
margin: 0 0 0 15;
|
||||
padding: 2 0 14 36;
|
||||
margin: 0 0 0 19;
|
||||
border-width: 0 0 0 2;
|
||||
}
|
||||
.view-instruction.instructionWOBorder {
|
||||
border-color: transparent;
|
||||
}
|
||||
|
||||
.view-note {
|
||||
border-width: 0;
|
||||
}
|
||||
|
||||
.view-reference-container {
|
||||
padding: 14 16;
|
||||
margin: 8 16 8;
|
||||
border-radius: 6;
|
||||
}
|
||||
.view-reference {
|
||||
padding: 0 16 0 0;
|
||||
margin: 0;
|
||||
border-width: 0;
|
||||
}
|
||||
.view-reference-text {
|
||||
font-size: 16;
|
||||
line-height: 6;
|
||||
padding: 16;
|
||||
margin: 8 16 8;
|
||||
border-radius: 6;
|
||||
}
|
||||
// .view-copyReference {
|
||||
// color: #ff7043;
|
||||
// }
|
||||
#btnFabContainer {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
|
@ -398,11 +450,15 @@ RadListView {
|
|||
height: 56;
|
||||
width: 56;
|
||||
background-color: #ff7043;
|
||||
horizontal-align: center;
|
||||
vertical-align: center;
|
||||
vertical-alignment: center;
|
||||
// horizontal-alignment: center;
|
||||
border-radius: 28;
|
||||
padding: 16;
|
||||
margin: 16;
|
||||
android-elevation: 6;
|
||||
&.negative {
|
||||
background-color: #e53935;
|
||||
}
|
||||
}
|
||||
.sectionTitle {
|
||||
font-size: 20;
|
||||
|
@ -421,6 +477,7 @@ RadListView {
|
|||
|
||||
// Dialogs
|
||||
.dialogContainer {
|
||||
width: 100%;
|
||||
&.light {
|
||||
color: $grayD4;
|
||||
background: $grayL4;
|
||||
|
@ -438,6 +495,7 @@ RadListView {
|
|||
}
|
||||
.dialogDescription {
|
||||
font-size: 16;
|
||||
line-height: 4;
|
||||
padding: 0 24 16;
|
||||
}
|
||||
.action {
|
||||
|
@ -456,3 +514,65 @@ RadListView {
|
|||
color: #ff7043;
|
||||
}
|
||||
}
|
||||
ActivityIndicator {
|
||||
color: #ff7043;
|
||||
padding: 16 12;
|
||||
margin: 16 10 16 8;
|
||||
}
|
||||
|
||||
// Vue Transitions
|
||||
.bounce-enter-active {
|
||||
animation-name: bounce-in;
|
||||
animation-duration: 0.5s;
|
||||
animation-fill-mode: forwards;
|
||||
animation-timing-function: ease-out;
|
||||
}
|
||||
.bounce-leave-active {
|
||||
animation-name: bounce-in;
|
||||
animation-duration: 0.1s;
|
||||
animation-fill-mode: forwards;
|
||||
animation-direction: reverse;
|
||||
animation-timing-function: ease-in;
|
||||
}
|
||||
@keyframes bounce-in {
|
||||
0% {
|
||||
transform: scale(0);
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
50% {
|
||||
transform: scale(1.2);
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
100% {
|
||||
transform: scale(1);
|
||||
}
|
||||
}
|
||||
.dolly-enter-active {
|
||||
animation-name: dolly;
|
||||
animation-duration: 1s;
|
||||
animation-delay: 0.25s;
|
||||
animation-fill-mode: forwards;
|
||||
animation-timing-function: ease;
|
||||
}
|
||||
.dolly-leave-active {
|
||||
opacity: 0;
|
||||
}
|
||||
@keyframes dolly {
|
||||
0% {
|
||||
transform: rotate(20deg);
|
||||
}
|
||||
25% {
|
||||
transform: rotate(-20deg);
|
||||
}
|
||||
50% {
|
||||
transform: rotate(10deg);
|
||||
}
|
||||
75% {
|
||||
transform: rotate(-10deg);
|
||||
}
|
||||
100% {
|
||||
transform: rotate(0deg);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -42,10 +42,18 @@
|
|||
<Label class="bx" :text="icon.link" />
|
||||
<Label text="View project on GitHub" class="option-title" />
|
||||
</StackLayout>
|
||||
<StackLayout orientation="horizontal" class="icon-option">
|
||||
<StackLayout
|
||||
orientation="horizontal"
|
||||
class="icon-option"
|
||||
@tap="openURL($event, 'https://t.me/enrecipes')"
|
||||
>
|
||||
<Label class="bx" :text="icon.telegram" />
|
||||
<Label text="Join the telegram group" class="option-title" />
|
||||
</StackLayout>
|
||||
<!-- <StackLayout orientation="horizontal" class="icon-option">
|
||||
<Label class="bx" :text="icon.file" />
|
||||
<Label text="Licenses" class="option-title" />
|
||||
</StackLayout>
|
||||
</StackLayout> -->
|
||||
|
||||
<StackLayout class="hr m-10"></StackLayout>
|
||||
|
||||
|
|
|
@ -34,7 +34,7 @@
|
|||
class="sd-group-header orkm"
|
||||
rows="auto"
|
||||
columns="*, auto"
|
||||
v-if="categories.length"
|
||||
v-if="categoriesWithRecipes.length"
|
||||
>
|
||||
<Label col="0" text="Categories" />
|
||||
<Label
|
||||
|
@ -47,7 +47,7 @@
|
|||
<StackLayout>
|
||||
<GridLayout
|
||||
@tap="navigateTo(item, false, true)"
|
||||
v-for="(item, index) in categories"
|
||||
v-for="(item, index) in categoriesWithRecipes"
|
||||
:key="index"
|
||||
class="sd-item orkm"
|
||||
:class="{
|
||||
|
@ -75,14 +75,14 @@
|
|||
</StackLayout>
|
||||
<StackLayout row="1">
|
||||
<StackLayout class="hr m-10"></StackLayout>
|
||||
<StackLayout
|
||||
<!-- <StackLayout
|
||||
orientation="horizontal"
|
||||
class="sd-item orkm"
|
||||
@tap="donate"
|
||||
>
|
||||
<Label class="bx" :text="icon.donate" margin="0 24 0 0" />
|
||||
<Label text="Donate" />
|
||||
</StackLayout>
|
||||
</StackLayout> -->
|
||||
<StackLayout
|
||||
@tap="navigateTo(item.component, true, false)"
|
||||
v-for="(item, index) in bottommenu"
|
||||
|
@ -105,7 +105,7 @@
|
|||
<EnRecipes
|
||||
ref="enrecipes"
|
||||
:filterFavorites="filterFavorites"
|
||||
:filterMustTry="filterMustTry"
|
||||
:filterTrylater="filterTrylater"
|
||||
:selectedCategory="selectedCategory"
|
||||
:showDrawer="showDrawer"
|
||||
:hijackGlobalBackEvent="hijackGlobalBackEvent"
|
||||
|
@ -145,7 +145,7 @@ export default {
|
|||
return {
|
||||
selectedCategory: null,
|
||||
filterFavorites: false,
|
||||
filterMustTry: false,
|
||||
filterTrylater: false,
|
||||
topmenu: [
|
||||
{
|
||||
title: "Home",
|
||||
|
@ -158,9 +158,9 @@ export default {
|
|||
icon: "heart",
|
||||
},
|
||||
{
|
||||
title: "Must-Try",
|
||||
component: "Must-Try",
|
||||
icon: "musttry",
|
||||
title: "Try later",
|
||||
component: "Try later",
|
||||
icon: "trylater",
|
||||
},
|
||||
],
|
||||
bottommenu: [
|
||||
|
@ -179,8 +179,8 @@ export default {
|
|||
}
|
||||
},
|
||||
computed: {
|
||||
...mapState(["icon", "recipes", "currentComponent"]),
|
||||
categories() {
|
||||
...mapState(["icon", "recipes", "categories", "currentComponent"]),
|
||||
categoriesWithRecipes() {
|
||||
let arr = this.recipes.map((e) => {
|
||||
return e.category
|
||||
})
|
||||
|
@ -197,7 +197,7 @@ export default {
|
|||
toggleCatEdit() {
|
||||
this.catEditMode = !this.catEditMode
|
||||
this.setComponent("EnRecipes")
|
||||
this.filterFavorites = this.filterMustTry = false
|
||||
this.filterFavorites = this.filterTrylater = false
|
||||
this.selectedCategory = null
|
||||
this.$refs.enrecipes.updateFilter()
|
||||
},
|
||||
|
@ -223,9 +223,7 @@ export default {
|
|||
highlight(args) {
|
||||
let temp = args.object.className
|
||||
args.object.className = `${temp} option-highlight`
|
||||
setTimeout(() => {
|
||||
args.object.className = temp
|
||||
}, 100)
|
||||
setTimeout(() => (args.object.className = temp), 100)
|
||||
},
|
||||
// Navigation
|
||||
setSelectedCategory(e) {
|
||||
|
@ -252,8 +250,8 @@ export default {
|
|||
function isFiltered() {
|
||||
vm.filterFavorites
|
||||
? vm.setComponent("Favorites")
|
||||
: vm.filterMustTry
|
||||
? vm.setComponent("Must-Try")
|
||||
: vm.filterTrylater
|
||||
? vm.setComponent("Try later")
|
||||
: vm.selectedCategory
|
||||
? vm.setComponent(vm.selectedCategory)
|
||||
: vm.setComponent("EnRecipes")
|
||||
|
@ -263,13 +261,13 @@ export default {
|
|||
this.closeDrawer()
|
||||
this.catEditMode = false
|
||||
} else if (
|
||||
["Favorites", "Must-Try", this.selectedCategory].includes(
|
||||
["Favorites", "Try later", this.selectedCategory].includes(
|
||||
this.currentComponent
|
||||
)
|
||||
) {
|
||||
preventDefault()
|
||||
this.setComponent("EnRecipes")
|
||||
this.filterFavorites = this.filterMustTry = false
|
||||
this.filterFavorites = this.filterTrylater = false
|
||||
this.selectedCategory = null
|
||||
this.$refs.enrecipes.updateFilter()
|
||||
this.releaseGlobalBackEvent()
|
||||
|
@ -296,7 +294,7 @@ export default {
|
|||
this.setComponent(to)
|
||||
this.$navigateBack({ frame: "main-frame", backstackVisible: false })
|
||||
this.filterFavorites = to === "Favorites" ? true : false
|
||||
this.filterMustTry = to === "Must-Try" ? true : false
|
||||
this.filterTrylater = to === "Try later" ? true : false
|
||||
this.selectedCategory = isCategory ? to : null
|
||||
this.$refs.enrecipes.updateFilter()
|
||||
this.closeDrawer()
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
<template>
|
||||
<Page @unloaded="releaseBackEvent">
|
||||
<Page @loaded="initialize" @unloaded="releaseBackEvent">
|
||||
<ActionBar :flat="viewIsScrolled ? false : true">
|
||||
<GridLayout rows="*" columns="auto, *, auto," class="actionBarContainer">
|
||||
<GridLayout rows="*" columns="auto, *, auto" class="actionBarContainer">
|
||||
<Label
|
||||
class="bx leftAction"
|
||||
:text="icon.back"
|
||||
|
@ -10,7 +10,6 @@
|
|||
@tap="navigateBack"
|
||||
/>
|
||||
<Label class="title orkm" :text="title" col="1" />
|
||||
|
||||
<Label
|
||||
v-if="hasEnoughDetails"
|
||||
class="bx"
|
||||
|
@ -20,13 +19,14 @@
|
|||
/>
|
||||
</GridLayout>
|
||||
</ActionBar>
|
||||
<AbsoluteLayout>
|
||||
<ScrollView
|
||||
width="100%"
|
||||
height="100%"
|
||||
@scroll="onScroll($event)"
|
||||
scrollBarIndicatorVisible="false"
|
||||
>
|
||||
<StackLayout width="100%">
|
||||
<StackLayout width="100%" padding="0 0 128">
|
||||
<!-- Image and camera btn -->
|
||||
<AbsoluteLayout>
|
||||
<StackLayout
|
||||
|
@ -50,34 +50,17 @@
|
|||
:text="icon.image"
|
||||
/>
|
||||
</StackLayout>
|
||||
<StackLayout
|
||||
width="100%"
|
||||
horizontalAlignment="center"
|
||||
:top="screenWidth - 42"
|
||||
>
|
||||
<StackLayout width="100%" :top="screenWidth - 42">
|
||||
<transition :name="recipeContent.imageSrc ? 'null' : 'bounce'">
|
||||
<Label
|
||||
v-if="recipeContent.imageSrc"
|
||||
@tap="removePicture"
|
||||
class="bx fab-button"
|
||||
:text="icon.close"
|
||||
androidElevation="8"
|
||||
/>
|
||||
<GridLayout v-else rows="auto" columns="*, auto, auto, *">
|
||||
<Label
|
||||
col="1"
|
||||
@tap="takePicture"
|
||||
v-if="showFab"
|
||||
horizontalAlignment="right"
|
||||
@tap="photoHandler"
|
||||
class="bx fab-button"
|
||||
:text="icon.camera"
|
||||
androidElevation="8"
|
||||
androidElevation="6"
|
||||
/>
|
||||
<Label
|
||||
col="2"
|
||||
@tap="selectPicture"
|
||||
class="bx fab-button"
|
||||
:text="icon.image"
|
||||
androidElevation="8"
|
||||
/>
|
||||
</GridLayout>
|
||||
</transition>
|
||||
</StackLayout>
|
||||
</AbsoluteLayout>
|
||||
|
||||
|
@ -244,10 +227,11 @@
|
|||
v-for="(reference, index) in recipeContent.references"
|
||||
:key="index"
|
||||
>
|
||||
<TextField
|
||||
<TextView
|
||||
col="0"
|
||||
v-model="recipeContent.references[index]"
|
||||
hint="Website or Video URL"
|
||||
hint="Text or Website/Video URL"
|
||||
editable="true"
|
||||
/>
|
||||
<Label
|
||||
col="2"
|
||||
|
@ -265,6 +249,7 @@
|
|||
</StackLayout>
|
||||
</StackLayout>
|
||||
</ScrollView>
|
||||
</AbsoluteLayout>
|
||||
</Page>
|
||||
</template>
|
||||
|
||||
|
@ -277,14 +262,13 @@ import {
|
|||
getFileAccess,
|
||||
knownFolders,
|
||||
} from "@nativescript/core"
|
||||
import * as imagepicker from "nativescript-imagepicker"
|
||||
import * as camera from "@nativescript/camera"
|
||||
import { Mediafilepicker } from "nativescript-mediafilepicker"
|
||||
|
||||
import { mapState, mapActions } from "vuex"
|
||||
|
||||
import ActionDialog from "./modal/ActionDialog.vue"
|
||||
import PromptDialog from "./modal/PromptDialog.vue"
|
||||
import ConfirmDialog from "./modal/ConfirmDialog.vue"
|
||||
import PromptDialog from "./modal/PromptDialog.vue"
|
||||
|
||||
export default {
|
||||
props: ["recipeIndex", "recipeID", "selectedCategory"],
|
||||
|
@ -294,36 +278,15 @@ export default {
|
|||
viewIsScrolled: false,
|
||||
recipeContent: {
|
||||
imageSrc: null,
|
||||
title: null,
|
||||
category: null,
|
||||
title: undefined,
|
||||
category: "Undefined",
|
||||
prepTime: "00:00",
|
||||
cookTime: "00:00",
|
||||
portionSize: 1,
|
||||
ingredients: [
|
||||
{
|
||||
item: "",
|
||||
quantity: null,
|
||||
unit: "unit",
|
||||
},
|
||||
],
|
||||
instructions: [""],
|
||||
notes: [""],
|
||||
references: [""],
|
||||
isFavorite: false,
|
||||
tried: false,
|
||||
lastModified: null,
|
||||
},
|
||||
tempRecipeContent: {
|
||||
imageSrc: null,
|
||||
title: null,
|
||||
category: null,
|
||||
prepTime: "00:00",
|
||||
cookTime: "00:00",
|
||||
portionSize: 1,
|
||||
ingredients: [
|
||||
{
|
||||
item: "",
|
||||
quantity: null,
|
||||
quantity: undefined,
|
||||
unit: "unit",
|
||||
},
|
||||
],
|
||||
|
@ -334,8 +297,10 @@ export default {
|
|||
tried: false,
|
||||
lastModified: null,
|
||||
},
|
||||
tempRecipeContent: {},
|
||||
blockModal: false,
|
||||
newRecipeID: null,
|
||||
showFab: false,
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
|
@ -344,14 +309,10 @@ export default {
|
|||
return Screen.mainScreen.widthDIPs
|
||||
},
|
||||
hasEnoughDetails() {
|
||||
if (this.recipeID) {
|
||||
return (
|
||||
JSON.stringify(this.recipeContent) !==
|
||||
JSON.stringify(this.tempRecipeContent)
|
||||
)
|
||||
} else {
|
||||
return this.recipeContent.title
|
||||
}
|
||||
},
|
||||
},
|
||||
methods: {
|
||||
|
@ -361,6 +322,9 @@ export default {
|
|||
"overwriteRecipeAction",
|
||||
"addCategoryAction",
|
||||
]),
|
||||
initialize() {
|
||||
this.showFab = true
|
||||
},
|
||||
getRandomID() {
|
||||
let res = ""
|
||||
let chars = "abcdefghijklmnopqrstuvwxyz0123456789"
|
||||
|
@ -380,12 +344,12 @@ export default {
|
|||
}
|
||||
},
|
||||
clearEmptyFields() {
|
||||
if (!this.recipeContent.title) {
|
||||
this.recipeContent.title = "Untitled Recipe"
|
||||
}
|
||||
if (!this.recipeContent.portionSize) {
|
||||
this.recipeContent.portionSize = 1
|
||||
}
|
||||
if (!this.recipeContent.category) {
|
||||
this.recipeContent.category = "Undefined"
|
||||
}
|
||||
this.recipeContent.ingredients.forEach((e, i) => {
|
||||
if (!e.item.length) {
|
||||
this.recipeContent.ingredients.splice(i, 1)
|
||||
|
@ -404,6 +368,10 @@ export default {
|
|||
removeEmpty("references")
|
||||
},
|
||||
saveRecipe() {
|
||||
console.log(
|
||||
JSON.stringify(this.recipeContent),
|
||||
JSON.stringify(this.tempRecipeContent)
|
||||
)
|
||||
this.clearEmptyFields()
|
||||
this.recipeContent.lastModified = new Date()
|
||||
if (this.recipeID) {
|
||||
|
@ -471,15 +439,17 @@ export default {
|
|||
this.blockModal = true
|
||||
this.$showModal(ConfirmDialog, {
|
||||
props: {
|
||||
title: "Discard changes",
|
||||
title: "Unsaved changes",
|
||||
description:
|
||||
"Are you sure you want discard unsaved changes to this recipe?",
|
||||
cancelButtonText: "KEEP EDITING",
|
||||
okButtonText: "DISCARD",
|
||||
"Do you want to save the changes you made in this recipe?",
|
||||
cancelButtonText: "DISCARD",
|
||||
okButtonText: "SAVE",
|
||||
},
|
||||
}).then((action) => {
|
||||
this.blockModal = false
|
||||
if (action) {
|
||||
this.saveRecipe()
|
||||
} else if (action != null) {
|
||||
this.$navigateBack()
|
||||
this.releaseBackEvent()
|
||||
}
|
||||
|
@ -507,19 +477,41 @@ export default {
|
|||
this.navigateBack()
|
||||
}
|
||||
},
|
||||
photoHandler() {
|
||||
if (this.recipeContent.imageSrc) {
|
||||
this.blockModal = true
|
||||
this.$showModal(ConfirmDialog, {
|
||||
props: {
|
||||
title: "Recipe photo",
|
||||
cancelButtonText: "REMOVE",
|
||||
okButtonText: "REPLACE PHOTO",
|
||||
},
|
||||
}).then((action) => {
|
||||
this.blockModal = false
|
||||
if (action) {
|
||||
this.takePicture()
|
||||
} else if (action != null) {
|
||||
this.removePicture()
|
||||
this.releaseBackEvent()
|
||||
}
|
||||
})
|
||||
} else {
|
||||
this.takePicture()
|
||||
}
|
||||
},
|
||||
takePicture() {
|
||||
const vm = this
|
||||
camera.requestPermissions().then(
|
||||
() => {
|
||||
camera
|
||||
.takePicture({
|
||||
width: vm.screenWidth,
|
||||
height: vm.screenWidth,
|
||||
keepAspectRatio: false,
|
||||
saveToGallery: false,
|
||||
const mediafilepicker = new Mediafilepicker()
|
||||
mediafilepicker.openImagePicker({
|
||||
android: {
|
||||
isCaptureMood: false, // if true then camera will open directly.
|
||||
isNeedCamera: true,
|
||||
maxNumberFiles: 1,
|
||||
isNeedFolderList: true,
|
||||
},
|
||||
})
|
||||
.then((imageAsset) => {
|
||||
let result = imageAsset._android
|
||||
mediafilepicker.on("getFiles", (image) => {
|
||||
let result = image.object.get("results")[0].file
|
||||
ImageSource.fromFile(result).then((savedImg) => {
|
||||
let savedImgPath = path.join(
|
||||
knownFolders.documents().getFolder("enrecipes").path,
|
||||
|
@ -529,36 +521,14 @@ export default {
|
|||
vm.recipeContent.imageSrc = savedImgPath
|
||||
})
|
||||
})
|
||||
.catch((err) => {
|
||||
console.log("Error -> " + err.message)
|
||||
mediafilepicker.on("error", function(res) {
|
||||
let msg = res.object.get("msg")
|
||||
console.log(msg)
|
||||
})
|
||||
},
|
||||
() => {
|
||||
console.log("permission request rejected")
|
||||
}
|
||||
)
|
||||
},
|
||||
selectPicture() {
|
||||
let context = imagepicker.create({
|
||||
mode: "single",
|
||||
mediaType: "Image",
|
||||
})
|
||||
context
|
||||
.authorize()
|
||||
.then(() => context.present())
|
||||
.then((selection) => {
|
||||
let result = selection[0]._android
|
||||
ImageSource.fromFile(result).then((savedImg) => {
|
||||
let savedImgPath = path.join(
|
||||
knownFolders.documents().getFolder("enrecipes").path,
|
||||
`${this.getRandomID()}.jpg`
|
||||
)
|
||||
savedImg.saveToFile(savedImgPath, "jpg")
|
||||
this.recipeContent.imageSrc = savedImgPath
|
||||
})
|
||||
})
|
||||
.catch(function(e) {
|
||||
console.log(e)
|
||||
|
||||
mediafilepicker.on("cancel", function(res) {
|
||||
let msg = res.object.get("msg")
|
||||
console.log(msg)
|
||||
})
|
||||
},
|
||||
removePicture() {
|
||||
|
@ -627,9 +597,13 @@ export default {
|
|||
this.title = this.recipeID ? "Edit recipe" : "New recipe"
|
||||
if (this.recipeID) {
|
||||
let recipe = this.recipes.filter((e) => e.id === this.recipeID)[0]
|
||||
Object.assign(this.recipeContent, recipe)
|
||||
Object.assign(this.tempRecipeContent, recipe)
|
||||
Object.assign(this.recipeContent, JSON.parse(JSON.stringify(recipe)))
|
||||
Object.assign(this.tempRecipeContent, JSON.parse(JSON.stringify(recipe)))
|
||||
} else {
|
||||
Object.assign(
|
||||
this.tempRecipeContent,
|
||||
JSON.parse(JSON.stringify(this.recipeContent))
|
||||
)
|
||||
if (this.selectedCategory)
|
||||
this.recipeContent.category = this.selectedCategory
|
||||
this.newRecipeID = this.getRandomID()
|
||||
|
|
|
@ -15,7 +15,6 @@
|
|||
col="0"
|
||||
@tap="closeSearch"
|
||||
/>
|
||||
<!-- @loaded="searchBarLoaded" -->
|
||||
<SearchBar
|
||||
col="1"
|
||||
hint="Search"
|
||||
|
@ -114,41 +113,92 @@
|
|||
<StackLayout height="128"></StackLayout>
|
||||
</v-template>
|
||||
</RadListView>
|
||||
<Label
|
||||
v-if="!recipes.length && !filterFavorites && !filterMustTry"
|
||||
<GridLayout rows="*" columns="*" class="emptyState">
|
||||
<StackLayout
|
||||
col="0"
|
||||
row="0"
|
||||
class="noResults"
|
||||
text='Click the "+" icon to add a new recipe.'
|
||||
v-if="!recipes.length && !filterFavorites && !filterTrylater"
|
||||
verticalAlignment="center"
|
||||
>
|
||||
<Label
|
||||
@tap="addRecipe"
|
||||
class="bx"
|
||||
:text="icon.plusCircle"
|
||||
textWrap="true"
|
||||
/>
|
||||
<Label
|
||||
v-if="!filteredRecipes.length && searchQuery"
|
||||
class="title orkb"
|
||||
text="Start adding your recipes!"
|
||||
textWrap="true"
|
||||
/>
|
||||
<Label text='Tap the "+" icon to add a new recipe' textWrap="true" />
|
||||
</StackLayout>
|
||||
<StackLayout
|
||||
col="0"
|
||||
row="0"
|
||||
class="noResults"
|
||||
v-if="!filteredRecipes.length && searchQuery"
|
||||
verticalAlignment="top"
|
||||
>
|
||||
<Label class="bx" :text="icon.search" textWrap="true" />
|
||||
<Label class="title orkb" text="No recipes found" textWrap="true" />
|
||||
<Label
|
||||
:text="
|
||||
`Your search "${searchQuery}" did not match any recipes in this category.`
|
||||
`Your search "${searchQuery}" did not match any recipes${
|
||||
filterFavorites || filterTrylater || selectedCategory
|
||||
? ' in this category'
|
||||
: ''
|
||||
}`
|
||||
"
|
||||
textWrap="true"
|
||||
/>
|
||||
<Label
|
||||
</StackLayout>
|
||||
<StackLayout
|
||||
col="0"
|
||||
row="0"
|
||||
class="noResults"
|
||||
verticalAlignment="center"
|
||||
v-if="!filteredRecipes.length && filterFavorites && !searchQuery"
|
||||
class="noResults"
|
||||
text="Your favorite recipes will be listed here."
|
||||
>
|
||||
<Label class="bx" :text="icon.heartOutline" textWrap="true" />
|
||||
<Label class="title orkb" text="No favorites yet!" textWrap="true" />
|
||||
<Label
|
||||
text="Your favorited recipes will be listed here"
|
||||
textWrap="true"
|
||||
/>
|
||||
<Label
|
||||
v-if="!filteredRecipes.length && filterMustTry && !searchQuery"
|
||||
</StackLayout>
|
||||
<StackLayout
|
||||
col="0"
|
||||
row="0"
|
||||
class="noResults"
|
||||
text="Your must-try recipes will be listed here."
|
||||
verticalAlignment="center"
|
||||
v-if="!filteredRecipes.length && filterTrylater && !searchQuery"
|
||||
>
|
||||
<Label class="bx" :text="icon.trylaterOutline" textWrap="true" />
|
||||
<Label
|
||||
class="title orkb"
|
||||
text="No recipes here to try!"
|
||||
textWrap="true"
|
||||
/>
|
||||
<GridLayout id="btnFabContainer" rows="*,88" columns="*,88">
|
||||
<!-- text="Your Try later recipes will be listed here" -->
|
||||
<Label
|
||||
text="Your recipes to try later will be listed here"
|
||||
textWrap="true"
|
||||
/>
|
||||
</StackLayout>
|
||||
</GridLayout>
|
||||
<GridLayout id="btnFabContainer" rows="*,auto" columns="*,auto">
|
||||
<transition name="bounce">
|
||||
<Label
|
||||
v-if="showFAB"
|
||||
row="1"
|
||||
col="1"
|
||||
class="bx fab-button"
|
||||
:text="icon.plus"
|
||||
androidElevation="8"
|
||||
@tap="addRecipe"
|
||||
/>
|
||||
</transition>
|
||||
</GridLayout>
|
||||
</AbsoluteLayout>
|
||||
</Page>
|
||||
|
@ -166,7 +216,7 @@ import { mapState, mapActions } from "vuex"
|
|||
export default {
|
||||
props: [
|
||||
"filterFavorites",
|
||||
"filterMustTry",
|
||||
"filterTrylater",
|
||||
"selectedCategory",
|
||||
"showDrawer",
|
||||
"hijackGlobalBackEvent",
|
||||
|
@ -184,6 +234,7 @@ export default {
|
|||
rightAction: false,
|
||||
sortType: "Natural order",
|
||||
deletionDialogActive: false,
|
||||
showFAB: false,
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
|
@ -194,7 +245,7 @@ export default {
|
|||
(e) =>
|
||||
e.isFavorite && e.title.toLowerCase().includes(this.searchQuery)
|
||||
)
|
||||
} else if (this.filterMustTry) {
|
||||
} else if (this.filterTrylater) {
|
||||
return this.recipes.filter(
|
||||
(e) => !e.tried && e.title.toLowerCase().includes(this.searchQuery)
|
||||
)
|
||||
|
@ -216,11 +267,12 @@ export default {
|
|||
initializePage() {
|
||||
this.filterFavorites
|
||||
? this.setComponent("Favorites")
|
||||
: this.filterMustTry
|
||||
? this.setComponent("Must-Try")
|
||||
: this.filterTrylater
|
||||
? this.setComponent("Try later")
|
||||
: this.selectedCategory
|
||||
? this.setComponent(this.selectedCategory)
|
||||
: this.setComponent("EnRecipes")
|
||||
this.showFAB = true
|
||||
},
|
||||
openSearch() {
|
||||
this.showSearch = true
|
||||
|
@ -313,7 +365,7 @@ export default {
|
|||
return item.isFavorite
|
||||
? item.title.toLowerCase().includes(this.searchQuery)
|
||||
: false
|
||||
} else if (this.filterMustTry) {
|
||||
} else if (this.filterTrylater) {
|
||||
return item.tried
|
||||
? false
|
||||
: item.title.toLowerCase().includes(this.searchQuery)
|
||||
|
@ -387,13 +439,14 @@ export default {
|
|||
: (this.viewIsScrolled = false)
|
||||
},
|
||||
addRecipe() {
|
||||
this.showFAB = false
|
||||
this.releaseGlobalBackEvent()
|
||||
this.$navigateTo(EditRecipe, {
|
||||
transition: {
|
||||
name: "slide",
|
||||
duration: 250,
|
||||
curve: "easeIn",
|
||||
},
|
||||
// transition: {
|
||||
// name: "slide",
|
||||
// duration: 250,
|
||||
// curve: "easeIn",
|
||||
// },
|
||||
props: {
|
||||
viewIsScrolled: this.viewIsScrolled,
|
||||
selectedCategory: this.selectedCategory,
|
||||
|
@ -401,13 +454,15 @@ export default {
|
|||
})
|
||||
},
|
||||
viewRecipe({ item, index }) {
|
||||
this.showFAB = false
|
||||
this.$navigateTo(ViewRecipe, {
|
||||
transition: {
|
||||
name: "fade",
|
||||
duration: 250,
|
||||
curve: "easeIn",
|
||||
},
|
||||
// transition: {
|
||||
// name: "fade",
|
||||
// duration: 250,
|
||||
// curve: "easeIn",
|
||||
// },
|
||||
props: {
|
||||
filterTrylater: this.filterTrylater,
|
||||
recipeIndex: index,
|
||||
recipeID: item.id,
|
||||
hijackGlobalBackEvent: this.hijackGlobalBackEvent,
|
||||
|
@ -416,5 +471,8 @@ export default {
|
|||
})
|
||||
},
|
||||
},
|
||||
mounted() {
|
||||
this.showFAB = true
|
||||
},
|
||||
}
|
||||
</script>
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
<StackLayout class="hr m-10"></StackLayout>
|
||||
|
||||
<Label text="Backup/Restore" class="group-header" />
|
||||
<StackLayout
|
||||
<!-- <StackLayout
|
||||
orientation="horizontal"
|
||||
class="option"
|
||||
@tap="selectBackupDir"
|
||||
|
@ -43,7 +43,7 @@
|
|||
<Label text="EnRecipes Backup Directory" class="option-title" />
|
||||
<Label text="/storage/emulated/0/EnRecipes" class="option-info" />
|
||||
</StackLayout>
|
||||
</StackLayout>
|
||||
</StackLayout> -->
|
||||
<StackLayout orientation="horizontal" class="option" @tap="backupData">
|
||||
<Label class="bx" :text="icon.backup" />
|
||||
<Label text="Backup Data" class="option-title" />
|
||||
|
@ -63,7 +63,6 @@ import {
|
|||
path,
|
||||
getFileAccess,
|
||||
knownFolders,
|
||||
Application,
|
||||
} from "@nativescript/core"
|
||||
import * as permissions from "nativescript-permissions"
|
||||
import { Zip } from "nativescript-zip"
|
||||
|
|
|
@ -1,13 +1,13 @@
|
|||
<template>
|
||||
<Page @loaded="initializePage">
|
||||
<ActionBar height="128" margin="0" flat="true">
|
||||
<ActionBar height="152" margin="0" flat="true">
|
||||
<GridLayout
|
||||
rows="64, 64"
|
||||
rows="24, 64, 64"
|
||||
columns="auto, *, auto,auto, auto"
|
||||
class="actionBarContainer"
|
||||
>
|
||||
<Label
|
||||
row="0"
|
||||
row="1"
|
||||
col="0"
|
||||
class="bx leftAction"
|
||||
:text="icon.back"
|
||||
|
@ -15,7 +15,7 @@
|
|||
@tap="$navigateBack()"
|
||||
/>
|
||||
<ScrollView
|
||||
row="1"
|
||||
row="2"
|
||||
col="1"
|
||||
colSpan="3"
|
||||
orientation="horizontal"
|
||||
|
@ -28,28 +28,29 @@
|
|||
/>
|
||||
</ScrollView>
|
||||
<Label
|
||||
row="0"
|
||||
col="2"
|
||||
class="bx"
|
||||
:text="icon.share"
|
||||
@tap="shareRecipe"
|
||||
/>
|
||||
<Label
|
||||
row="0"
|
||||
row="1"
|
||||
col="3"
|
||||
class="bx"
|
||||
:class="{ 'view-favorited': recipe.isFavorite }"
|
||||
:text="recipe.isFavorite ? icon.heart : icon.heartOutline"
|
||||
@tap="toggleFavorite"
|
||||
/>
|
||||
<Label
|
||||
row="0"
|
||||
v-if="!filterTrylater"
|
||||
row="1"
|
||||
col="4"
|
||||
class="bx"
|
||||
:class="{ 'view-favorited': !recipe.tried }"
|
||||
:text="recipe.tried ? icon.musttryOutline : icon.musttry"
|
||||
@tap="toggleMustTry"
|
||||
:text="recipe.tried ? icon.trylaterOutline : icon.trylater"
|
||||
@tap="toggleTrylater"
|
||||
/>
|
||||
<Label
|
||||
v-if="!busy"
|
||||
row="1"
|
||||
col="2"
|
||||
class="bx"
|
||||
:text="icon.edit"
|
||||
@tap="editRecipe"
|
||||
/>
|
||||
<ActivityIndicator v-else row="1" col="4" :busy="busy" />
|
||||
</GridLayout>
|
||||
</ActionBar>
|
||||
<AbsoluteLayout>
|
||||
|
@ -93,10 +94,6 @@
|
|||
class="view-other"
|
||||
:text="`Cooking time: ${getTime(recipe.cookTime)}`"
|
||||
/>
|
||||
<Label
|
||||
class="view-other"
|
||||
:text="`Portion size: ${recipe.portionSize}`"
|
||||
/>
|
||||
</StackLayout>
|
||||
</StackLayout>
|
||||
</ScrollView>
|
||||
|
@ -112,7 +109,7 @@
|
|||
<StackLayout v-else padding="16 16 124">
|
||||
<AbsoluteLayout class="inputField">
|
||||
<TextField
|
||||
width="50%"
|
||||
width="165"
|
||||
v-model="portionScale"
|
||||
keyboardType="number"
|
||||
/>
|
||||
|
@ -207,7 +204,7 @@
|
|||
<Label
|
||||
verticalAlignment="top"
|
||||
horizontalAlignment="center"
|
||||
class="view-count orkb"
|
||||
class="view-count note orkb"
|
||||
col="0"
|
||||
:text="index + 1"
|
||||
/>
|
||||
|
@ -223,52 +220,74 @@
|
|||
text="Click the edit button to add references to this recipe"
|
||||
textWrap="true"
|
||||
/>
|
||||
<StackLayout v-else padding="32 16 132">
|
||||
<GridLayout
|
||||
columns="auto ,*"
|
||||
<StackLayout v-else padding="10 0 132">
|
||||
<StackLayout
|
||||
v-for="(reference, index) in recipe.references"
|
||||
:key="index"
|
||||
@tap="openURL($event, reference)"
|
||||
>
|
||||
<GridLayout
|
||||
v-if="isValidURL(reference)"
|
||||
columns="*, auto"
|
||||
class="view-reference-container"
|
||||
androidElevation="1"
|
||||
>
|
||||
<Label
|
||||
col="0"
|
||||
colSpan="2"
|
||||
verticalAlignment="center"
|
||||
class="view-reference"
|
||||
:text="reference"
|
||||
textWrap="false"
|
||||
@tap="openURL($event, reference)"
|
||||
/>
|
||||
<Label
|
||||
col="1"
|
||||
class="view-copyReference bx"
|
||||
:text="icon.copy"
|
||||
@tap="copyURL($event, reference)"
|
||||
/>
|
||||
</GridLayout>
|
||||
<Label
|
||||
v-else
|
||||
class="view-reference-text"
|
||||
:text="reference"
|
||||
textWrap="true"
|
||||
/>
|
||||
<Label
|
||||
verticalAlignment="top"
|
||||
horizontalAlignment="center"
|
||||
class="orkb view-count"
|
||||
col="0"
|
||||
:text="index + 1"
|
||||
/>
|
||||
</GridLayout>
|
||||
</StackLayout>
|
||||
</StackLayout>
|
||||
</ScrollView>
|
||||
</TabViewItem>
|
||||
</TabView>
|
||||
<GridLayout id="btnFabContainer" rows="*,88" columns="*,88">
|
||||
<GridLayout id="btnFabContainer" rows="*,auto" columns="*,auto">
|
||||
<Label
|
||||
v-if="!busy"
|
||||
@tap="editRecipe"
|
||||
row="1"
|
||||
col="1"
|
||||
class="bx fab-button"
|
||||
:text="icon.edit"
|
||||
androidElevation="8"
|
||||
:text="icon.unchecked"
|
||||
@tap="recipeTried"
|
||||
v-if="filterTrylater"
|
||||
/>
|
||||
<ActivityIndicator v-else row="1" col="1" :busy="busy" />
|
||||
<transition name="dolly" appear>
|
||||
<Label
|
||||
row="1"
|
||||
col="1"
|
||||
class="bx fab-button"
|
||||
:text="icon.share"
|
||||
@tap="shareRecipe"
|
||||
v-if="!filterTrylater && showFab"
|
||||
/>
|
||||
</transition>
|
||||
</GridLayout>
|
||||
</AbsoluteLayout>
|
||||
</Page>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { Screen, Utils, ImageSource, Device } from "@nativescript/core"
|
||||
import { Screen, Utils, ImageSource, Device, File } from "@nativescript/core"
|
||||
import { Feedback, FeedbackType, FeedbackPosition } from "nativescript-feedback"
|
||||
import * as application from "tns-core-modules/application"
|
||||
import * as Toast from "nativescript-toast"
|
||||
import * as SocialShare from "nativescript-social-share"
|
||||
import * as SocialShare from "nativescript-social-share-ns-7"
|
||||
import { setText } from "nativescript-clipboard"
|
||||
|
||||
import { mapState, mapActions } from "vuex"
|
||||
|
||||
|
@ -276,6 +295,7 @@ import EditRecipe from "./EditRecipe.vue"
|
|||
|
||||
export default {
|
||||
props: [
|
||||
"filterTrylater",
|
||||
"recipeIndex",
|
||||
"recipeID",
|
||||
"hijackGlobalBackEvent",
|
||||
|
@ -286,6 +306,7 @@ export default {
|
|||
busy: false,
|
||||
portionScale: 1,
|
||||
recipe: null,
|
||||
showFab: false,
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
|
@ -308,6 +329,47 @@ export default {
|
|||
this.setCurrentComponentAction("ViewRecipe")
|
||||
}, 500)
|
||||
this.portionScale = this.recipe.portionSize
|
||||
this.showFab = true
|
||||
this.showInfo()
|
||||
},
|
||||
niceDates(time) {
|
||||
let date = new Date(time)
|
||||
let diff = (new Date().getTime() - date.getTime()) / 1000
|
||||
console.log(diff)
|
||||
let dayDiff = Math.ceil(diff / 86400)
|
||||
console.log(dayDiff)
|
||||
|
||||
if (isNaN(dayDiff) || dayDiff < 0) return ""
|
||||
|
||||
return (
|
||||
(dayDiff == 0 &&
|
||||
((diff < 60 && "just now") ||
|
||||
(diff < 120 && "1 minute ago") ||
|
||||
(diff < 3600 && Math.floor(diff / 60) + " minutes ago") ||
|
||||
(diff < 7200 && "1 hour ago") ||
|
||||
(diff < 86400 && Math.floor(diff / 3600) + " hours ago"))) ||
|
||||
(dayDiff == 1 && "yesterday") ||
|
||||
(dayDiff < 7 && dayDiff + " days ago") ||
|
||||
(dayDiff < 31 && Math.ceil(dayDiff / 7) + " week(s) ago") ||
|
||||
(dayDiff > 30 &&
|
||||
dayDiff < 365 &&
|
||||
Math.round(dayDiff / 30) + " month(s) ago") ||
|
||||
(dayDiff > 364 && Math.round(dayDiff / 365) + " year(s) ago")
|
||||
)
|
||||
},
|
||||
showInfo() {
|
||||
let feedback = new Feedback()
|
||||
feedback.show({
|
||||
type: FeedbackType.Info,
|
||||
message: `You tried this recipe ${this.niceDates(
|
||||
this.recipe.triedOn
|
||||
)}!`,
|
||||
})
|
||||
},
|
||||
highlight(args) {
|
||||
let temp = args.object.className
|
||||
args.object.className = `${temp} option-highlight`
|
||||
setTimeout(() => (args.object.className = temp), 100)
|
||||
},
|
||||
roundedQuantity(quantity) {
|
||||
return (
|
||||
|
@ -319,13 +381,14 @@ export default {
|
|||
)
|
||||
},
|
||||
editRecipe() {
|
||||
this.showFab = false
|
||||
this.busy = true
|
||||
this.$navigateTo(EditRecipe, {
|
||||
transition: {
|
||||
name: "slide",
|
||||
duration: 250,
|
||||
curve: "easeIn",
|
||||
},
|
||||
// transition: {
|
||||
// name: "slide",
|
||||
// duration: 250,
|
||||
// curve: "easeOut",
|
||||
// },
|
||||
props: {
|
||||
recipeIndex: this.recipeIndex,
|
||||
recipeID: this.recipeID,
|
||||
|
@ -333,6 +396,55 @@ export default {
|
|||
// backstackVisible: false,
|
||||
})
|
||||
},
|
||||
shareRecipe() {
|
||||
let overview = `${
|
||||
this.recipe.title
|
||||
} Recipe\n\nPreparation time: ${this.getTime(
|
||||
this.recipe.prepTime
|
||||
)}\nCooking time: ${this.getTime(this.recipe.cookTime)}\n`
|
||||
let shareContent = overview
|
||||
if (this.recipe.ingredients.length) {
|
||||
let ingredients = `\n\nIngredients for ${this.recipe.portionSize} ${
|
||||
this.recipe.portionSize === 1 ? "postion:" : "portions:"
|
||||
}\n\n`
|
||||
this.recipe.ingredients.forEach((e) => {
|
||||
ingredients += `- ${this.roundedQuantity(e.quantity)} ${e.unit} ${
|
||||
e.item
|
||||
}\n`
|
||||
})
|
||||
shareContent += ingredients
|
||||
}
|
||||
if (this.recipe.instructions.length) {
|
||||
let instructions = `\n\nInstructions:\n\n`
|
||||
this.recipe.instructions.forEach((e, i) => {
|
||||
instructions += `${i + 1}. ${e}\n\n`
|
||||
})
|
||||
shareContent += instructions
|
||||
}
|
||||
if (this.recipe.notes.length) {
|
||||
let notes = `\nNotes:\n\n`
|
||||
this.recipe.notes.forEach((e, i) => {
|
||||
notes += `${i + 1}. ${e}\n\n`
|
||||
})
|
||||
shareContent += notes
|
||||
}
|
||||
if (this.recipe.references.length) {
|
||||
let references = `\nReferences:\n\n`
|
||||
this.recipe.references.forEach((e, i) => {
|
||||
references += `${e}\n\n`
|
||||
})
|
||||
shareContent += references
|
||||
}
|
||||
let sharenote =
|
||||
"\nCreated and shared via EnRecipes.\n\nDownload the app on f-droid:\nhttps://www.vishnuraghav.com/"
|
||||
|
||||
shareContent += sharenote
|
||||
|
||||
SocialShare.shareText(
|
||||
shareContent,
|
||||
"How would you like to share this recipe?"
|
||||
)
|
||||
},
|
||||
toggle(key) {
|
||||
this.toggleStateAction({
|
||||
index: this.recipeIndex,
|
||||
|
@ -341,39 +453,52 @@ export default {
|
|||
key,
|
||||
})
|
||||
},
|
||||
shareRecipe() {
|
||||
// if (this.recipe.imageSrc) {
|
||||
// let image = ImageSource.fromFile(this.recipe.imageSrc)
|
||||
// SocialShare.shareImage(image)
|
||||
// } else {
|
||||
// SocialShare.shareText("Text only")
|
||||
// }
|
||||
alert(Device.sdkVersion)
|
||||
},
|
||||
toggleFavorite() {
|
||||
this.recipe.isFavorite
|
||||
? Toast.makeText("Removed from Favorites").show()
|
||||
: Toast.makeText("Added to Favorites").show()
|
||||
this.toggle("isFavorite")
|
||||
},
|
||||
toggleMustTry() {
|
||||
toggleTrylater() {
|
||||
this.recipe.tried
|
||||
? Toast.makeText("Added to Must-Try").show()
|
||||
: Toast.makeText("Removed from Must-Try").show()
|
||||
? this.filterTrylater
|
||||
? Toast.makeText("Added back to Try later").show()
|
||||
: Toast.makeText("Added to Try later").show()
|
||||
: this.filterTrylater
|
||||
? Toast.makeText("You tried this recipe").show()
|
||||
: Toast.makeText("Removed from Try later").show()
|
||||
// : Toast.makeText("You tried this recipe").show()
|
||||
this.toggle("tried")
|
||||
},
|
||||
recipeTried() {
|
||||
this.toggle("tried")
|
||||
this.$navigateBack()
|
||||
},
|
||||
getTime(time) {
|
||||
let t = time.split(":")
|
||||
let h = t[0]
|
||||
let m = t[1]
|
||||
return h !== "00" ? `${h}h ${m}m` : `${m}m`
|
||||
},
|
||||
isValidURL(string) {
|
||||
let pattern = new RegExp("^https?|www", "ig")
|
||||
return pattern.test(string)
|
||||
},
|
||||
openURL(args, url) {
|
||||
// this.highlight(args)
|
||||
Utils.openUrl(url)
|
||||
},
|
||||
copyURL(args, url) {
|
||||
setText(url).then((e) => {
|
||||
Toast.makeText("URL Copied").show()
|
||||
})
|
||||
},
|
||||
},
|
||||
created() {
|
||||
this.recipe = this.recipes.filter((e) => e.id === this.recipeID)[0]
|
||||
},
|
||||
mounted() {
|
||||
this.showFab = true
|
||||
},
|
||||
}
|
||||
</script>
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
<Page>
|
||||
<StackLayout class="dialogContainer" :class="isLightMode">
|
||||
<Label class="dialogTitle orkm" :text="title" />
|
||||
<Label class="dialogDescription" :text="description" textWrap="true" />
|
||||
<Label v-if="description" class="dialogDescription" :text="description" textWrap="true" />
|
||||
<StackLayout
|
||||
orientation="horizontal"
|
||||
class="actionsContainer"
|
||||
|
|
|
@ -14,9 +14,9 @@ Vue.registerElement(
|
|||
)
|
||||
|
||||
// Vue.registerElement(
|
||||
// "Fab",
|
||||
// () => require("@nstudio/nativescript-floatingactionbutton").Fab
|
||||
// )
|
||||
// 'Fab',
|
||||
// () => require('@nstudio/nativescript-floatingactionbutton').Fab
|
||||
// );
|
||||
|
||||
if (TNS_ENV !== "production") {
|
||||
// Vue.use(VueDevtools)
|
||||
|
|
67
app/store.js
67
app/store.js
|
@ -8,7 +8,59 @@ Vue.use(Vuex)
|
|||
|
||||
export default new Vuex.Store({
|
||||
state: {
|
||||
recipes: [],
|
||||
recipes: [
|
||||
{
|
||||
imageSrc: null,
|
||||
title: "Mushroom & Spinach Risotto",
|
||||
category: "Lunch",
|
||||
prepTime: "00:25",
|
||||
cookTime: "00:45",
|
||||
portionSize: "8",
|
||||
ingredients: [
|
||||
{
|
||||
item: "reduced-sodium chicken broth",
|
||||
unit: "cup",
|
||||
quantity: "5.25",
|
||||
},
|
||||
{ item: "sliced fresh mushrooms", unit: "cup", quantity: "2.5" },
|
||||
{ item: "medium onion, finely chopped", unit: "unit", quantity: "1" },
|
||||
{ item: "butter", unit: "Tbsp", quantity: "3" },
|
||||
{ item: "Garlic", unit: "unit", quantity: "3" },
|
||||
{
|
||||
item: "white wine or reduced-sodium chicken broth",
|
||||
unit: "cup",
|
||||
quantity: ".75",
|
||||
},
|
||||
{ item: "heavy whipping cream", unit: "cup", quantity: "1" },
|
||||
{ item: "uncooked arborio rice", unit: "cup", quantity: "1.75" },
|
||||
{ item: "olive oil", unit: "Tbsp", quantity: "2" },
|
||||
{
|
||||
item: "frozen chopped spinach, thawed and squeezed dry",
|
||||
unit: "cup",
|
||||
quantity: "1.5",
|
||||
},
|
||||
{ item: "pepper", unit: "tsp", quantity: ".5" },
|
||||
{ item: "Salt", unit: "tsp", quantity: ".25" },
|
||||
{ item: "grated Parmesan cheese", unit: "cup", quantity: "1" },
|
||||
],
|
||||
instructions: [
|
||||
"In a large saucepan, heat broth and keep warm. In a large skillet, saute mushrooms and onion in butter until tender. Add garlic; cook 1 minute longer. Stir in wine. Bring to a boil; cook until liquid is reduced by half. Add cream; cook and stir over medium heat until slightly thickened",
|
||||
"In a large saucepan, saute rice in oil for 2-3 minutes or until rice is lightly browned. Stir in 1/2 cup hot broth. Reduce heat; cook and stir for 20 minutes or until broth is absorbed.",
|
||||
"Continue adding hot broth, 1/2 cup at a time, and stirring until all the broth has been absorbed and rice is tender but firm. Add the mushroom mixture, spinach, pepper, salt and grated Parmesan cheese; cook and stir until heated through. If desired, sprinkle with parsley and shaved Parmesan cheese. Serve immediately.",
|
||||
],
|
||||
notes: [
|
||||
"Nutrition Facts\n3/4 cup: 409 calories, 22g fat (12g saturated fat), 61mg cholesterol, 667mg sodium, 41g carbohydrate (3g sugars, 2g fiber), 11g protein.",
|
||||
],
|
||||
references: [
|
||||
"https://www.tasteofhome.com/recipes/mushroom-spinach-risotto/",
|
||||
],
|
||||
isFavorite: false,
|
||||
tried: true,
|
||||
triedOn: "2020-10-28T18:19:06.528Z",
|
||||
lastModified: "2020-10-28T06:19:06.528Z",
|
||||
id: "57qm8oqxdr",
|
||||
},
|
||||
],
|
||||
categories: [],
|
||||
units: [
|
||||
"unit",
|
||||
|
@ -45,10 +97,11 @@ export default new Vuex.Store({
|
|||
plus: "\ueb89",
|
||||
close: "\uec4e",
|
||||
image: "\ueae9",
|
||||
addImage: "\ueae8",
|
||||
back: "\ue988",
|
||||
save: "\uee48",
|
||||
camera: "\uecc2",
|
||||
share: "\uee51",
|
||||
share: "\uee50",
|
||||
edit: "\uee17",
|
||||
theme: "\ued09",
|
||||
folder: "\ued7c",
|
||||
|
@ -59,8 +112,14 @@ export default new Vuex.Store({
|
|||
user: "\uee8e",
|
||||
trash: "\uee83",
|
||||
donate: "\ued41",
|
||||
musttry: "\uec96",
|
||||
musttryOutline: "\ue9bb",
|
||||
trylater: "\uec96",
|
||||
trylaterOutline: "\ue9bb",
|
||||
note: "\uee0a",
|
||||
copy: "\uea51",
|
||||
plusCircle: "\ueb8a",
|
||||
unchecked: "\uea16",
|
||||
checked: "\uece6",
|
||||
telegram: "\ue95e",
|
||||
},
|
||||
currentComponent: "EnRecipes",
|
||||
},
|
||||
|
|
54
package-lock.json
generated
54
package-lock.json
generated
|
@ -1095,20 +1095,17 @@
|
|||
"to-fast-properties": "^2.0.0"
|
||||
}
|
||||
},
|
||||
"@nativescript-community/perms": {
|
||||
"version": "2.1.1",
|
||||
"resolved": "https://registry.npmjs.org/@nativescript-community/perms/-/perms-2.1.1.tgz",
|
||||
"integrity": "sha512-Ay4v1lEGTQ5rYYlYA8CKcCXuxOuyU4633r/JXi9aRG8MgxfOT+rDuQLgSz+LLCYmBK1ndfHHfyUTilkaUj1H8Q=="
|
||||
},
|
||||
"@nativescript/android": {
|
||||
"version": "7.0.1",
|
||||
"resolved": "https://registry.npmjs.org/@nativescript/android/-/android-7.0.1.tgz",
|
||||
"integrity": "sha512-VsZCJ5zfZo0+/lFwKz+S7iFb7MA2jgACB7y8dNje3/cnZl+moKPNjFqitoEP0DY4gLz9LJNbFIIaUt84tMdUSQ==",
|
||||
"dev": true
|
||||
},
|
||||
"@nativescript/camera": {
|
||||
"version": "5.0.2",
|
||||
"resolved": "https://registry.npmjs.org/@nativescript/camera/-/camera-5.0.2.tgz",
|
||||
"integrity": "sha512-frNeCLhdQ+W6oXIv05pALdmZDcwilw/NopLtQILtUwuLS7xhE+UMx6CqQxxCxMYzWKsvET2k9VLAo3mJGAoSeg==",
|
||||
"requires": {
|
||||
"nativescript-permissions": "~1.3.0"
|
||||
}
|
||||
},
|
||||
"@nativescript/core": {
|
||||
"version": "7.0.12",
|
||||
"resolved": "https://registry.npmjs.org/@nativescript/core/-/core-7.0.12.tgz",
|
||||
|
@ -5096,17 +5093,28 @@
|
|||
"to-regex": "^3.0.1"
|
||||
}
|
||||
},
|
||||
"nativescript-clipboard": {
|
||||
"version": "2.0.0",
|
||||
"resolved": "https://registry.npmjs.org/nativescript-clipboard/-/nativescript-clipboard-2.0.0.tgz",
|
||||
"integrity": "sha512-w7qRJiWiBeq55f3IW+pAkbrl+v+yqZf3bhcl1wH6Qm1JqZLWDv7HemzHTM5CvaqQ4gfz5dnGhJW1q1fsr5KSOw=="
|
||||
},
|
||||
"nativescript-couchbase-plugin": {
|
||||
"version": "0.9.6",
|
||||
"resolved": "https://registry.npmjs.org/nativescript-couchbase-plugin/-/nativescript-couchbase-plugin-0.9.6.tgz",
|
||||
"integrity": "sha512-kMA9KHQX82TFaGnGUhY94KLOLss4pb5QmghgoEdu1sLwd94I/f1MQ+kHWbuBOdFmdQJw5oCK+Sey+A22Nd5jgA=="
|
||||
},
|
||||
"nativescript-imagepicker": {
|
||||
"version": "7.1.0",
|
||||
"resolved": "https://registry.npmjs.org/nativescript-imagepicker/-/nativescript-imagepicker-7.1.0.tgz",
|
||||
"integrity": "sha512-YFVwmPz7mv7mNXA7vmnIXmqPZiWxH4RoJPDL3m34egV8Ae9mKJCXZxl2LyPraOP+T4v6iXsxV9NSbjg0kMDuNQ==",
|
||||
"nativescript-feedback": {
|
||||
"version": "2.0.0",
|
||||
"resolved": "https://registry.npmjs.org/nativescript-feedback/-/nativescript-feedback-2.0.0.tgz",
|
||||
"integrity": "sha512-6ZE/g+s1xxez3pMRJa/r0f144VuyapgDjbo8D37nMC4F0WDQLKk8dC0405czhQxD2djVq3GEMfM/n0cuMbY53A=="
|
||||
},
|
||||
"nativescript-mediafilepicker": {
|
||||
"version": "4.0.1",
|
||||
"resolved": "https://registry.npmjs.org/nativescript-mediafilepicker/-/nativescript-mediafilepicker-4.0.1.tgz",
|
||||
"integrity": "sha512-rBrZQR+46dCypIyLrzIlzmHgpmTSMGFR5a6snq8uUhtIqLlc674/nwWlNM1kFOxMh1kKxA+qyk74Of+NCKYoqQ==",
|
||||
"requires": {
|
||||
"nativescript-permissions": "~1.3.0"
|
||||
"@nativescript-community/perms": "^2.1.1",
|
||||
"ts-node": "^9.0.0"
|
||||
}
|
||||
},
|
||||
"nativescript-permissions": {
|
||||
|
@ -5122,10 +5130,10 @@
|
|||
"nativescript-permissions": "~1.3.0"
|
||||
}
|
||||
},
|
||||
"nativescript-social-share": {
|
||||
"version": "1.6.0",
|
||||
"resolved": "https://registry.npmjs.org/nativescript-social-share/-/nativescript-social-share-1.6.0.tgz",
|
||||
"integrity": "sha512-PjSMseCWPGJbW0KPMgQBiTQke6I8cYxf0CGXtuJ0BnRhXrEjF3d+3kAnI8E3O8PeW/BFwNIqLYG4fkoQF4obyA=="
|
||||
"nativescript-social-share-ns-7": {
|
||||
"version": "11.6.0",
|
||||
"resolved": "https://registry.npmjs.org/nativescript-social-share-ns-7/-/nativescript-social-share-ns-7-11.6.0.tgz",
|
||||
"integrity": "sha512-NI6U8iOz3CKKV6nuZYSYUwA5JYFt1uPZAaMAWxySbMVQaKdP7H3lt+Sa/4ENZTvOkaAparD3ZZUxf0PITsrstA=="
|
||||
},
|
||||
"nativescript-toast": {
|
||||
"version": "2.0.0",
|
||||
|
@ -7356,6 +7364,18 @@
|
|||
}
|
||||
}
|
||||
},
|
||||
"ts-node": {
|
||||
"version": "9.0.0",
|
||||
"resolved": "https://registry.npmjs.org/ts-node/-/ts-node-9.0.0.tgz",
|
||||
"integrity": "sha512-/TqB4SnererCDR/vb4S/QvSZvzQMJN8daAslg7MeaiHvD8rDZsSfXmNeNumyZZzMned72Xoq/isQljYSt8Ynfg==",
|
||||
"requires": {
|
||||
"arg": "^4.1.0",
|
||||
"diff": "^4.0.1",
|
||||
"make-error": "^1.1.1",
|
||||
"source-map-support": "^0.5.17",
|
||||
"yn": "3.1.1"
|
||||
}
|
||||
},
|
||||
"tslib": {
|
||||
"version": "2.0.3",
|
||||
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.0.3.tgz",
|
||||
|
|
|
@ -8,16 +8,16 @@
|
|||
"run": "ns run android"
|
||||
},
|
||||
"dependencies": {
|
||||
"@nativescript/camera": "^5.0.2",
|
||||
"@nativescript/core": "~7.0.0",
|
||||
"@nativescript/datetimepicker": "^2.0.4",
|
||||
"@nativescript/theme": "^3.0.0",
|
||||
"@nativescript/webpack": "3.0.0",
|
||||
"nativescript-clipboard": "^2.0.0",
|
||||
"nativescript-couchbase-plugin": "^0.9.6",
|
||||
"nativescript-imagepicker": "^7.1.0",
|
||||
"nativescript-permissions": "^1.3.9",
|
||||
"nativescript-feedback": "^2.0.0",
|
||||
"nativescript-mediafilepicker": "^4.0.1",
|
||||
"nativescript-plugin-filepicker": "^1.0.0",
|
||||
"nativescript-social-share": "^1.6.0",
|
||||
"nativescript-social-share-ns-7": "^11.6.0",
|
||||
"nativescript-toast": "^2.0.0",
|
||||
"nativescript-ui-listview": "^9.0.4",
|
||||
"nativescript-ui-sidedrawer": "^9.0.3",
|
||||
|
|
Loading…
Reference in a new issue