added foreground service for timer

This commit is contained in:
vishnuraghavb 2021-06-05 23:38:36 +05:30
parent f92e43e216
commit 525d0740f8
3 changed files with 341 additions and 208 deletions

View file

@ -1,15 +1,14 @@
import { TimerNotif } from './shared/utils' import { TimerNotif } from './shared/utils'
const superProto = android.app.Service.prototype const superProto = android.app.Service.prototype
android.app.Service.extend('com.tns.ForegroundService', { android.app.Service.extend('com.tns.ForegroundService', {
onStartCommand: function(intent, flags, startId) { onStartCommand: function(intent, flags, startId) {
console.log('onStartCommand')
superProto.onStartCommand.call(this, intent, flags, startId) superProto.onStartCommand.call(this, intent, flags, startId)
return android.app.Service.START_STICKY return android.app.Service.START_STICKY
}, },
onCreate: function() { onCreate: function() {
console.log('onCreate')
superProto.onCreate.call(this) superProto.onCreate.call(this)
this.startForeground(777, this.getNotification()) this.startForeground(6, this.getNotification())
}, },
onBind: function(intent) { onBind: function(intent) {
return superProto.onBind.call(this, intent) return superProto.onBind.call(this, intent)
@ -18,17 +17,16 @@ android.app.Service.extend('com.tns.ForegroundService', {
return superProto.onUnbind.call(this, intent) return superProto.onUnbind.call(this, intent)
}, },
onDestroy: function() { onDestroy: function() {
console.log('onDestroy')
this.stopForeground(true) this.stopForeground(true)
}, },
getNotification: function() { getNotification: function() {
return TimerNotif.getNotification( return TimerNotif.getNotification(
{ {
bID: 'bringToFront', bID: 'info',
cID: 'cti', cID: 'cti',
cName: 'Cooking Timer info', cName: 'Cooking Timer info',
description: `0 ongoing, 0 paused`, description: `0 ongoing, 0 paused`,
nID: 777, nID: 6,
priority: -2, priority: -2,
sound: null, sound: null,
title: 'EnRecipes is running', title: 'EnRecipes is running',

View file

@ -1,15 +1,98 @@
import {
Application,
AndroidApplication,
ApplicationSettings,
Utils,
Device,
Color,
Frame,
} from '@nativescript/core'
import { import {
localize, localize,
androidLaunchEventLocalizationHandler, androidLaunchEventLocalizationHandler,
} from '@nativescript/localize' } from '@nativescript/localize'
import { Application, AndroidApplication, Utils } from '@nativescript/core' import Vue from 'nativescript-vue'
import EnRecipes from './components/EnRecipes.vue'
import EditRecipe from './components/EditRecipe.vue'
import MealPlanner from './components/MealPlanner.vue'
import CookingTimer from './components/CookingTimer.vue'
import GroceryList from './components/GroceryList.vue'
import store from './store'
import * as utils from '~/shared/utils'
const keepScreenOn = () => { export const EventBus = new Vue()
let renderView: any = EnRecipes
import CollectionView from '@nativescript-community/ui-collectionview/vue'
Vue.use(CollectionView)
import { StackLayout, GridLayout, DockLayout } from '@nativescript-rtl/ui'
Vue.registerElement('RStackLayout', () => StackLayout)
Vue.registerElement('RGridLayout', () => GridLayout)
Vue.registerElement('RDockLayout', () => DockLayout)
import { myMixin } from './shared/mixins'
Vue.mixin(myMixin)
const initFrame = () => {
const vm = store
//MAIN INIT
vm.commit('setTheme', ApplicationSettings.getString('appTheme', 'sysDef'))
if (!vm.state.recipes.length) vm.commit('initRecipes')
vm.commit('initMealPlans')
vm.commit('initListItems')
vm.commit('initTimerPresets')
if (!Object.keys(vm.state.timerSound).length) {
let hasTimerSound = ApplicationSettings.getString('timerSound', null)
vm.commit(
'setTimerSound',
hasTimerSound ? JSON.parse(hasTimerSound) : utils.getTones().defaultTone
)
}
// INIT FRAME
const View = android.view.View as any
const window = Application.android.startActivity.getWindow()
const decorView = window.getDecorView()
let sdkv = parseInt(Device.sdkVersion)
function setColors(color) {
window.setStatusBarColor(new Color(color).android)
sdkv >= 27 && window.setNavigationBarColor(new Color(color).android)
}
switch (vm.state.appTheme) {
case 'Light':
setColors('#f1f3f5')
break
case 'Dark':
setColors('#212529')
break
default:
setColors('#000000')
break
}
if (sdkv >= 27)
decorView.setSystemUiVisibility(
vm.state.appTheme == 'Light'
? View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR |
View.SYSTEM_UI_FLAG_LIGHT_NAVIGATION_BAR
: View.SYSTEM_UI_FLAG_DARK_STATUS_BAR |
View.SYSTEM_UI_FLAG_DARK_NAVIGATION_BAR
)
else
decorView.setSystemUiVisibility(
vm.state.appTheme == 'Light'
? View.SYSTEM_UI_FLAG_LIGHT_STATUS_BAR
: View.SYSTEM_UI_FLAG_DARK_STATUS_BAR
)
Frame.topmost().className = store.state.appTheme
}
const showOverLockscreen = () => {
let ctx = Utils.ad.getApplicationContext() let ctx = Utils.ad.getApplicationContext()
const pm = ctx.getSystemService(android.content.Context.POWER_SERVICE) const pm = ctx.getSystemService(android.content.Context.POWER_SERVICE)
let isScreenOff = !pm.isInteractive() let isScreenOff = !pm.isInteractive()
if (isScreenOff) { if (isScreenOff) {
console.log('keepScreenOn') console.log('showOverLockscreen')
const window = Application.android.startActivity.getWindow() const window = Application.android.startActivity.getWindow()
const windowMgr = android.view.WindowManager const windowMgr = android.view.WindowManager
const flags = const flags =
@ -17,42 +100,63 @@ const keepScreenOn = () => {
windowMgr.LayoutParams.FLAG_TURN_SCREEN_ON | windowMgr.LayoutParams.FLAG_TURN_SCREEN_ON |
windowMgr.LayoutParams.FLAG_KEEP_SCREEN_ON windowMgr.LayoutParams.FLAG_KEEP_SCREEN_ON
window.addFlags(flags) window.addFlags(flags)
function clearFlags(args) {
args.cancel = true
window.clearFlags(flags)
Application.android.off(
AndroidApplication.activityBackPressedEvent,
clearFlags
)
}
Application.android.on(
AndroidApplication.activityBackPressedEvent,
clearFlags
)
} }
} }
Application.on(Application.resumeEvent, keepScreenOn) const intentListener = ({ intent, android }: any) => {
console.log(intent, android)
let action = ((intent || android).getStringExtra('action') ||
(android && android.getAction())) as string
if (action) {
console.log(action)
switch (action) {
case 'new_recipe':
renderView = EditRecipe
break
case 'planner':
renderView = MealPlanner
break
case 'timer':
renderView = CookingTimer
Vue.navigateTo(CookingTimer as any, {
animated: false,
})
break
case 'grocery':
renderView = GroceryList
break
}
}
}
Application.on(Application.resumeEvent, (args) => {
console.log('App Resume')
showOverLockscreen()
})
Application.on(Application.launchEvent, (args) => { Application.on(Application.launchEvent, (args) => {
console.log('launching') console.log('App Launch')
console.log('RTL', store.state.RTL)
store.commit('setRTL')
if (args.android) { if (args.android) {
androidLaunchEventLocalizationHandler() androidLaunchEventLocalizationHandler()
intentListener(args) intentListener(args)
} }
Application.android.on(
AndroidApplication.activityNewIntentEvent,
intentListener
)
Frame.on(Frame.loadedEvent, initFrame)
}) })
import Vue from 'nativescript-vue' Application.on(Application.exitEvent, () => {
import EnRecipes from './components/EnRecipes.vue' console.log('App Exit')
import CookingTimer from './components/CookingTimer.vue' renderView = EnRecipes
import store from './store' Application.android.off(
AndroidApplication.activityNewIntentEvent,
export const EventBus = new Vue() intentListener
)
import CollectionView from '@nativescript-community/ui-collectionview/vue' })
Vue.use(CollectionView)
import { lvMixin } from './shared/mixins'
Vue.mixin(lvMixin)
Vue.config.silent = false Vue.config.silent = false
@ -60,37 +164,5 @@ Vue.filter('L', localize)
new Vue({ new Vue({
store, store,
render: (h) => h(EnRecipes), render: (h) => h('Frame', [h(renderView)]),
}).$start() }).$start()
const intentListener = ({ intent, android }: any) => {
let ct = 'CookingTimer'
let action = (intent || android).getStringExtra('action')
console.log('calling: ', action)
if (action == 'open_timer' && store.state.currentComponent != ct) {
let openTimer = setInterval(() => {
let comp = store.state.currentComponent
if (comp == ct) clearInterval(openTimer)
else {
if (comp == 'CTSettings') Vue.navigateBack()
else {
Vue.navigateTo(CookingTimer)
store.commit('setComponent', 'CookingTimer')
}
}
}, 250)
}
}
Application.on(Application.launchEvent, () => {
Application.android.on(
AndroidApplication.activityNewIntentEvent,
intentListener
)
})
Application.on(Application.exitEvent, () => {
store.commit('setComponent', 'EnRecipes')
Application.android.off(
AndroidApplication.activityNewIntentEvent,
intentListener
)
})

View file

@ -5,11 +5,17 @@ import { openOrCreate } from '@akylas/nativescript-sqlite'
import { import {
getFileAccess, getFileAccess,
File, File,
ApplicationSettings,
Application, Application,
knownFolders, knownFolders,
path, path,
} from '@nativescript/core' } from '@nativescript/core'
import {
getNumber,
setNumber,
getString,
setString,
} from '@nativescript/core/application-settings'
import * as utils from '~/shared/utils'
// OPEN DATABASE FILE // OPEN DATABASE FILE
const db = openOrCreate( const db = openOrCreate(
@ -31,6 +37,11 @@ db.execute(
'CREATE TABLE IF NOT EXISTS mealPlans (date INT, type TEXT, title TEXT)' 'CREATE TABLE IF NOT EXISTS mealPlans (date INT, type TEXT, title TEXT)'
) )
// CREATE timerPresets TABLE
db.execute(
'CREATE TABLE IF NOT EXISTS timerPresets (id INT PRIMARY KEY, label TEXT, time TEXT)'
)
const defaultCuisines = [ const defaultCuisines = [
'American', 'American',
'Brazilian', 'Brazilian',
@ -136,21 +147,22 @@ const defaultUnits = [
'medium', 'medium',
'large', 'large',
] ]
const listItems = { const listItems = {
cuisines: { cuisines: {
sort: true, sort: 1,
defaultItems: defaultCuisines, defaultItems: defaultCuisines,
}, },
categories: { categories: {
sort: true, sort: 1,
defaultItems: defaultCategories, defaultItems: defaultCategories,
}, },
yieldUnits: { yieldUnits: {
sort: false, sort: 0,
defaultItems: defaultYieldUnits, defaultItems: defaultYieldUnits,
}, },
units: { units: {
sort: false, sort: 0,
defaultItems: defaultUnits, defaultItems: defaultUnits,
}, },
} }
@ -227,10 +239,15 @@ export default new Vuex.Store({
trans: '\ue93c', trans: '\ue93c',
delay: '\ue93d', delay: '\ue93d',
ring: '\ue93e', ring: '\ue93e',
print: '\ue93f',
}, },
currentComponent: 'EnRecipes', currentComponent: 'EnRecipes',
sortType: 'Oldest first', sortType: 'random',
language: [ language: [
{
locale: 'ar',
title: 'العربية',
},
{ {
locale: 'da', locale: 'da',
title: 'Dansk', title: 'Dansk',
@ -239,10 +256,18 @@ export default new Vuex.Store({
locale: 'de', locale: 'de',
title: 'Deutsch', title: 'Deutsch',
}, },
{
locale: 'en-IN',
title: 'English (IN)',
},
{ {
locale: 'en-GB', locale: 'en-GB',
title: 'English (UK)', title: 'English (UK)',
}, },
{
locale: 'en-US',
title: 'English (US)',
},
{ {
locale: 'es', locale: 'es',
title: 'Español', title: 'Español',
@ -267,10 +292,6 @@ export default new Vuex.Store({
locale: 'hi', locale: 'hi',
title: 'हिंदी', title: 'हिंदी',
}, },
{
locale: 'ml',
title: 'മലയാളം',
},
{ {
locale: 'id', locale: 'id',
title: 'Indonesia', title: 'Indonesia',
@ -279,6 +300,14 @@ export default new Vuex.Store({
locale: 'it', locale: 'it',
title: 'Italiano', title: 'Italiano',
}, },
{
locale: 'ja',
title: '日本語',
},
{
locale: 'ml',
title: 'മലയാളം',
},
{ {
locale: 'nb-NO', locale: 'nb-NO',
title: 'Norsk bokmål', title: 'Norsk bokmål',
@ -287,10 +316,10 @@ export default new Vuex.Store({
locale: 'nl', locale: 'nl',
title: 'Nederlands', title: 'Nederlands',
}, },
// { {
// locale: 'pt', locale: 'pt',
// title: 'Português', title: 'Português',
// }, },
{ {
locale: 'pt-BR', locale: 'pt-BR',
title: 'Português (BR)', title: 'Português (BR)',
@ -308,97 +337,114 @@ export default new Vuex.Store({
title: 'తెలుగు', title: 'తెలుగు',
}, },
], ],
shakeEnabled: ApplicationSettings.getBoolean('shakeEnabled', true), shakeEnabled: getNumber('shakeEnabled', 1),
importSummary: { importSummary: {
found: 0, found: 0,
imported: 0, imported: 0,
updated: 0, updated: 0,
}, },
layout: ApplicationSettings.getString('layout', 'detailed'), layout: getString('layout', 'detailed'),
selectedCuisine: null, selectedCuisine: null,
selectedCategory: null, selectedCategory: null,
selectedTag: null, selectedTag: null,
appTheme: 'sysDef', appTheme: 'sysDef',
mondayFirst: ApplicationSettings.getBoolean('mondayFirst', false), mondayFirst: getNumber('mondayFirst', 0),
timerDelay: ApplicationSettings.getString('timerDelay', '1 minute'), timerDelay: getString('timerDelay', '1 minute'),
timerSound: {}, timerSound: {},
timerVibrate: ApplicationSettings.getBoolean('timerVibrate', false), timerVibrate: getNumber('timerVibrate', 0),
timerPresets: [ timerPresets: [],
{
id: 534534563,
label: 'Soft Eggs',
recipeID: null,
time: '00:06:00',
timerInterval: null,
isPaused: false,
preset: 1,
},
{
id: 564646,
label: 'Medium Eggs',
recipeID: null,
time: '00:08:00',
timerInterval: null,
isPaused: false,
preset: 1,
},
{
id: 43276767,
label: 'Hard Eggs',
recipeID: null,
time: '00:10:00',
timerInterval: null,
isPaused: false,
preset: 1,
},
],
activeTimers: [], activeTimers: [],
timerIntervals: [], FGService: 0,
RTL: 0,
}, },
mutations: { mutations: {
addTimerPreset(state, timer) { setRTL(state) {
state.timerPresets.push(timer) state.RTL = utils.RTL() as any | 0
}, },
removeTimerPreset(state, index) { setFGService(state, val) {
state.timerPresets.splice(index, 1) state.FGService = val
},
addTimerPreset(state, timer) {
let i = state.timerPresets.findIndex((e) => e.id == timer.id)
if (state.timerPresets.some((e) => e.id == timer.id)) {
state.timerPresets.splice(i, 1, timer)
} else state.timerPresets.push(timer)
db.execute(
`REPLACE INTO timerPresets (id, label, time) VALUES (?, ?, ?)`,
[timer.id, timer.label, timer.time]
)
},
deleteTimerPreset(state, i) {
let id = state.timerPresets[i]
state.timerPresets.splice(i, 1)
db.execute(`DELETE FROM timerPresets WHERE id = ${id}`)
},
initTimerPresets(state) {
if (!state.timerPresets.length)
db.select(`SELECT * FROM timerPresets`).then((res) => {
res.forEach((t) => {
t.recipeID = 0
t.timerInt = 0
t.isPaused = 0
t.preset = 1
t.done = 0
state.timerPresets.push(t)
})
})
},
importTimerPresets(state, timers) {
let newPresets = timers.filter(
(e) => !state.timerPresets.some((f) => f.id === e.id)
)
newPresets.forEach((t) => {
db.execute(
`INSERT INTO timerPresets (id, label, time) VALUES (?, ?, ?)`,
[t.id, t.label, t.time]
)
state.timerPresets.push(t)
})
}, },
clearTimerInterval(state) { clearTimerInterval(state) {
state.activeTimers.forEach((e) => { state.activeTimers.forEach((e) => {
clearInterval(e.timerInterval) clearInterval(e.timerInt)
e.timerInterval = null e.timerInt = 0
}) })
}, },
addActiveTimer(state, { timer, index }) { addActiveTimer(state, { timer, i }) {
state.activeTimers.splice(index, 0, timer) state.activeTimers.splice(i, 0, timer)
},
sortActiveTimers(state) {
let a = state.activeTimers.reduce((acc, e) => {
;(acc[e.recipeID] = acc[e.recipeID] || []).push(e)
return acc
}, {})
state.activeTimers = [...(<any>Object).values(a).flat(2)]
}, },
updateActiveTimer(state, timer) { updateActiveTimer(state, timer) {
let index = state.activeTimers.findIndex((e) => e.id == timer.id) let i = state.activeTimers.findIndex((e) => e.id == timer.id)
state.activeTimers.splice(index, 1, timer) state.activeTimers.splice(i, 1, timer)
}, },
removeActiveTimer(state, index) { removeActiveTimer(state, i) {
state.activeTimers.splice(index, 1) state.activeTimers.splice(i, 1)
}, },
setTimerDelay(state, delay) { setTimerDelay(state, delay) {
state.timerDelay = delay state.timerDelay = delay
ApplicationSettings.setString('timerDelay', delay) setString('timerDelay', delay)
}, },
setTimerSound(state, sound) { setTimerSound(state, sound) {
state.timerSound = { state.timerSound = sound
title: sound.title, setString('timerSound', JSON.stringify(sound))
uri: sound.uri,
}
ApplicationSettings.setString('timerSound', JSON.stringify(sound))
}, },
setTimerVibrate(state, bool) { setTimerVibrate(state, n) {
state.timerVibrate = bool state.timerVibrate = n
ApplicationSettings.setBoolean('timerVibrate', bool) setNumber('timerVibrate', n)
}, },
clearImportSummary(state) { clearImportSummary(state) {
for (const key in state.importSummary) state.importSummary[key] = 0 for (const key in state.importSummary) state.importSummary[key] = 0
}, },
setFirstDay(state, bool) { setFirstDay(state, n) {
state.mondayFirst = bool state.mondayFirst = n | 0
ApplicationSettings.setBoolean('mondayFirst', bool) setNumber('mondayFirst', n | 0)
}, },
setTheme(state, theme) { setTheme(state, theme) {
switch (theme) { switch (theme) {
@ -414,7 +460,7 @@ export default new Vuex.Store({
state.appTheme = theme state.appTheme = theme
break break
} }
ApplicationSettings.setString('appTheme', theme) setString('appTheme', theme)
}, },
clearFilter(state) { clearFilter(state) {
state.selectedCuisine = state.selectedCategory = state.selectedTag = null state.selectedCuisine = state.selectedCategory = state.selectedTag = null
@ -436,8 +482,8 @@ export default new Vuex.Store({
state.recipes.push(e) state.recipes.push(e)
}) })
}) })
state.shakeEnabled = ApplicationSettings.getBoolean('shakeEnabled', true) state.shakeEnabled = getNumber('shakeEnabled', 1)
state.sortType = ApplicationSettings.getString('sortType', 'Oldest first') state.sortType = getString('sortType', 'random')
}, },
importRecipesFromJSON(state, recipes) { importRecipesFromJSON(state, recipes) {
let localRecipesIDs, partition let localRecipesIDs, partition
@ -463,6 +509,12 @@ export default new Vuex.Store({
r.yieldQuantity = r.yield.quantity r.yieldQuantity = r.yield.quantity
r.yieldUnit = r.yield.unit r.yieldUnit = r.yield.unit
delete r.yield delete r.yield
function getTime(d) {
return new Date(d).getTime()
}
r.lastTried = getTime(r.lastTried)
r.lastModified = getTime(r.lastModified)
r.created = getTime(r.created)
return r return r
}) })
} }
@ -492,9 +544,9 @@ export default new Vuex.Store({
JSON.stringify(r.notes), JSON.stringify(r.notes),
r.favorite ? 1 : 0, r.favorite ? 1 : 0,
r.tried ? 1 : 0, r.tried ? 1 : 0,
r.lastTried ? new Date(r.lastTried).getTime() : null, r.lastTried,
r.lastModified ? new Date(r.lastModified).getTime() : null, r.lastModified,
r.created ? new Date(r.created).getTime() : null, r.created,
] ]
) )
}) })
@ -533,9 +585,9 @@ export default new Vuex.Store({
JSON.stringify(r.notes), JSON.stringify(r.notes),
r.favorite ? 1 : 0, r.favorite ? 1 : 0,
r.tried ? 1 : 0, r.tried ? 1 : 0,
r.lastTried ? new Date(r.lastTried).getTime() : null, r.lastTried,
r.lastModified ? new Date(r.lastModified).getTime() : null, r.lastModified,
r.created ? new Date(r.created).getTime() : null, r.created,
] ]
) )
} }
@ -649,9 +701,9 @@ export default new Vuex.Store({
function exist({ id }, index: number) { function exist({ id }, index: number) {
if (id === recipe.id) { if (id === recipe.id) {
i = index i = index
return true return 1
} }
return false return 0
} }
state.recipes.some(exist) state.recipes.some(exist)
? Object.assign(state.recipes[i], recipe) ? Object.assign(state.recipes[i], recipe)
@ -659,9 +711,9 @@ export default new Vuex.Store({
}, },
deleteRecipes(state, ids) { deleteRecipes(state, ids) {
ids.forEach((id: string) => { ids.forEach((id: string) => {
let index = state.recipes.findIndex((e) => e.id === id) let i = state.recipes.findIndex((e) => e.id === id)
getFileAccess().deleteFile(state.recipes[index].image) getFileAccess().deleteFile(state.recipes[i].image)
state.recipes.splice(index, 1) state.recipes.splice(i, 1)
db.execute(`DELETE FROM recipes WHERE id = '${id}'`) db.execute(`DELETE FROM recipes WHERE id = '${id}'`)
state.recipes.forEach((e, i) => { state.recipes.forEach((e, i) => {
if (e.combinations.includes(id)) { if (e.combinations.includes(id)) {
@ -774,11 +826,11 @@ export default new Vuex.Store({
type = 'snacks' type = 'snacks'
break break
} }
return f.title === e.title && f.date === date && f.type === type return f.title == e.title && f.date == date && f.type == type
}) })
} else { } else {
return !state.mealPlans.some( return !state.mealPlans.some(
(f) => f.title === e.title && f.date === e.date && f.type === e.type (f) => f.title == e.title && f.date == e.date && f.type == e.type
) )
} }
}) })
@ -835,47 +887,48 @@ export default new Vuex.Store({
) )
}) })
}, },
addMealPlan(state, { date, type, title }) { addMealPlan(state, { date, type, title, index, inDB }) {
let mealPlan = { let mealPlan = {
date, date,
type, type,
title, title,
} }
state.mealPlans.push(mealPlan) if (inDB) {
const cols = Object.keys(mealPlan).join(', ') const cols = Object.keys(mealPlan).join(', ')
const placeholder = Object.keys(mealPlan) const placeholder = Object.keys(mealPlan)
.fill('?') .fill('?')
.join(', ') .join(', ')
db.execute(
`INSERT INTO mealPlans (${cols}) VALUES (${placeholder})`,
Object.values(mealPlan)
)
},
deleteMealPlan(state, { date, type, title }) {
let index = state.mealPlans.findIndex(
(e) => e.title === title && e.type === type && e.date === date
)
state.mealPlans.splice(index, 1)
state.mealPlans = [...state.mealPlans]
db.execute(
`DELETE FROM mealPlans WHERE date = ${date} AND type = '${type}' AND title = '${title}'`
)
},
toggleState(state, { id, key, setDate }) {
let index = state.recipes.findIndex((e) => e.id == id)
state.recipes[index][key] = state.recipes[index][key] ? 0 : 1
db.execute(
`UPDATE recipes SET ${key} = ${state.recipes[index][key]} WHERE id = '${id}'`
)
if (setDate) {
state.recipes[index].lastTried = new Date().getTime()
db.execute( db.execute(
`UPDATE recipes SET lastTried = ${state.recipes[index].lastTried} WHERE id = '${id}'` `INSERT INTO mealPlans (${cols}) VALUES (${placeholder})`,
Object.values(mealPlan)
) )
if (index === null) state.mealPlans.push(mealPlan)
} else {
state.mealPlans.splice(index, 0, mealPlan)
} }
}, },
setComponent(state, comp) { deleteMealPlan(state, { date, type, title, index, inDB }) {
state.currentComponent = comp if (inDB) {
db.execute(
`DELETE FROM mealPlans WHERE date = ${date} AND type = '${type}' AND title = '${title}'`
)
} else {
state.mealPlans.splice(index, 1)
state.mealPlans = [...state.mealPlans]
}
},
toggleState(state, { id, key, setDate }) {
let i = state.recipes.findIndex((e) => e.id == id)
state.recipes[i][key] = state.recipes[i][key] ? 0 : 1
db.execute(
`UPDATE recipes SET ${key} = ${state.recipes[i][key]} WHERE id = '${id}'`
)
if (setDate) {
state.recipes[i].lastTried = new Date().getTime()
db.execute(
`UPDATE recipes SET lastTried = ${state.recipes[i].lastTried} WHERE id = '${id}'`
)
}
}, },
unSyncCombinations(state, { id, combinations }) { unSyncCombinations(state, { id, combinations }) {
state.recipes.forEach((e, i) => { state.recipes.forEach((e, i) => {
@ -890,19 +943,17 @@ export default new Vuex.Store({
}) })
}, },
setShake(state, shake) { setShake(state, shake) {
state.shakeEnabled = shake state.shakeEnabled = shake | 0
ApplicationSettings.setBoolean('shakeEnabled', shake) setNumber('shakeEnabled', shake | 0)
}, },
setRating(state, { id, rating }) { setRating(state, { id, rating }) {
let index = state.recipes.findIndex((e) => e.id == id) let i = state.recipes.findIndex((e) => e.id == id)
state.recipes[index].rating = rating state.recipes[i].rating = rating
db.execute(`UPDATE recipes SET rating = ${rating} WHERE id = '${id}'`) db.execute(`UPDATE recipes SET rating = ${rating} WHERE id = '${id}'`)
}, },
toggleCart(state, { id }) { toggleCart(state, { id }) {
let index = state.recipes.indexOf( let i = state.recipes.indexOf(state.recipes.filter((e) => e.id === id)[0])
state.recipes.filter((e) => e.id === id)[0] state.recipes[i].inBag = !state.recipes[i].inBag
)
state.recipes[index].inBag = !state.recipes[index].inBag
}, },
unlinkBrokenImages(state) { unlinkBrokenImages(state) {
state.recipes.forEach((r, i) => { state.recipes.forEach((r, i) => {
@ -924,11 +975,26 @@ export default new Vuex.Store({
}, },
}, },
actions: { actions: {
setRTL({ commit }) {
commit('setRTL')
},
sortActiveTimers({ commit }) {
commit('sortActiveTimers')
},
setFGService({ commit }, val) {
commit('setFGService', val)
},
addTimerPreset({ commit }, timer) { addTimerPreset({ commit }, timer) {
commit('addTimerPreset', timer) commit('addTimerPreset', timer)
}, },
removeTimerPreset({ commit }, timer) { deleteTimerPreset({ commit }, timer) {
commit('removeTimerPreset', timer) commit('deleteTimerPreset', timer)
},
initTimerPresets({ commit }) {
commit('initTimerPresets')
},
importTimerPresets({ commit }, timers) {
commit('importTimerPresets', timers)
}, },
clearTimerInterval({ commit }) { clearTimerInterval({ commit }) {
commit('clearTimerInterval') commit('clearTimerInterval')
@ -939,8 +1005,8 @@ export default new Vuex.Store({
updateActiveTimer({ commit }, timer) { updateActiveTimer({ commit }, timer) {
commit('updateActiveTimer', timer) commit('updateActiveTimer', timer)
}, },
removeActiveTimer({ commit }, index) { removeActiveTimer({ commit }, i) {
commit('removeActiveTimer', index) commit('removeActiveTimer', i)
}, },
setTimerDelay({ commit }, delay) { setTimerDelay({ commit }, delay) {
commit('setTimerDelay', delay) commit('setTimerDelay', delay)
@ -948,14 +1014,14 @@ export default new Vuex.Store({
setTimerSound({ commit }, sound) { setTimerSound({ commit }, sound) {
commit('setTimerSound', sound) commit('setTimerSound', sound)
}, },
setTimerVibrate({ commit }, bool) { setTimerVibrate({ commit }, n) {
commit('setTimerVibrate', bool) commit('setTimerVibrate', n)
}, },
clearImportSummary({ commit }) { clearImportSummary({ commit }) {
commit('clearImportSummary') commit('clearImportSummary')
}, },
setFirstDay({ commit }, bool) { setFirstDay({ commit }, n) {
commit('setFirstDay', bool) commit('setFirstDay', n)
}, },
setTheme({ commit }, theme) { setTheme({ commit }, theme) {
commit('setTheme', theme) commit('setTheme', theme)
@ -1017,9 +1083,6 @@ export default new Vuex.Store({
toggleStateAction({ commit }, toggledRecipe) { toggleStateAction({ commit }, toggledRecipe) {
commit('toggleState', toggledRecipe) commit('toggleState', toggledRecipe)
}, },
setComponent({ commit }, comp) {
commit('setComponent', comp)
},
unSyncCombinationsAction({ commit }, combinations) { unSyncCombinationsAction({ commit }, combinations) {
commit('unSyncCombinations', combinations) commit('unSyncCombinations', combinations)
}, },