translations, set name props as unique, set child data as deleted on deleting parents
This commit is contained in:
parent
38f43a48ec
commit
0182bf463b
52
TODO
52
TODO
@ -1,48 +1,45 @@
|
||||
MAIN TASKS:
|
||||
Database:
|
||||
☐ set name properties as unique (and add checks to forms)
|
||||
☐ implement users
|
||||
☐ check objectbox docs on how to make users
|
||||
☐ tie all data to users
|
||||
☐ add user filters to all getters
|
||||
☐ change settings to not be a singleton, but only one settings instance per user and one default entry
|
||||
☐ add login and authentification
|
||||
☐ enable restoring data from sync
|
||||
☐ find a solution for storage
|
||||
☐ hosting
|
||||
☐ objectbox sync (commercial use is not free)
|
||||
☐ implement alternative data export for now?
|
||||
☐ create default datasets for configuration (meal categories, portion types, accuracies, event types, possibly meal source)
|
||||
☐ cleanup unneeded models (and make sure the app still runs)
|
||||
Features:
|
||||
☐ app icon
|
||||
☐ add explanations to each section
|
||||
☐ german language support
|
||||
☐ indicate nested creation process (creating from dropdown etc)
|
||||
☐ indicate read only fields
|
||||
☐ indicate nested creation process (creating from dropdown etc)
|
||||
☐ app info/credits screen
|
||||
☐ add explanations to each section
|
||||
☐ app icon
|
||||
Components/Framework:
|
||||
☐ check through all detail forms and set required fields/according messages
|
||||
☐ show indicator and make all fields readonly if user somehow gets to a deleted record detail view
|
||||
☐ check for changes before navigating as well (not just on cancel)
|
||||
☐ fix bug when navigating while creating/editing a record
|
||||
☐ change placement of delete and floating button because its very easy to accidentally hit delete
|
||||
☐ dropdown tweaks
|
||||
☐ edit item -> cancel: shouldn't clear dropdwon
|
||||
☐ account for deleted/disabled elements
|
||||
☐ change app id from "com.example.diameter" to "at.sarahziesel.diameter"
|
||||
☐ come up with new concept for duration component
|
||||
☐ update duration fields to use corresponding component
|
||||
☐ log event type detail (reminder duration)
|
||||
☐ log event detail (reminder duration)
|
||||
☐ meal (bolus delay)
|
||||
☐ log bolus (delay)
|
||||
☐ check through all detail forms and set required fields/according messages
|
||||
☐ change placement of delete and floating button because its very easy to accidentally hit delete
|
||||
☐ check for changes before navigating as well (not just on cancel)
|
||||
☐ show indicator and make all fields readonly if user somehow gets to a deleted record detail view
|
||||
☐ dropdown tweaks
|
||||
☐ edit item -> cancel: shouldn't clear dropdwon
|
||||
☐ account for deleted/disabled elements
|
||||
Log:
|
||||
☐ add filters
|
||||
☐ check if there is still an active bolus when suggesting glucose bolus
|
||||
☐ remember/display setting for correction boli correctly
|
||||
Log Events:
|
||||
☐ add filters
|
||||
☐ don't show warning for existing event if it's the one being edited
|
||||
☐ add create button to event type dropdown
|
||||
Categorization:
|
||||
☐ add colors to event types as indicators for log entries and graphs in reports
|
||||
☐ implement reminders as push notifications
|
||||
Settings:
|
||||
☐ export functionality
|
||||
☐ implement loading from google drive
|
||||
☐ find out pricing for google apis
|
||||
☐ add some options (which data to export, including dead records...)
|
||||
☐ option to switch theme
|
||||
☐ add fields for glucose target tiers (as map of cutoff glucose and colors)
|
||||
☐ add field for active insulin duration
|
||||
@ -50,6 +47,15 @@ MAIN TASKS:
|
||||
☐ add functionality to delete dead records (meaning: set deleted flag and no relations to undeleted records)
|
||||
|
||||
Archive:
|
||||
✔ set child data as deleted on deleting parent (ie. log boli for log entry etc) @done(22-06-19 23:44) @project(MAIN TASKS.Database)
|
||||
✔ set name properties as unique @done(22-06-17 01:34) @project(MAIN TASKS.Database)
|
||||
✔ german language support @done(22-06-17 01:19) @project(MAIN TASKS.Features)
|
||||
✔ add translations for all sections @done(22-06-17 01:19) @project(MAIN TASKS.Features)
|
||||
✔ add data source field to all models @done(22-04-15 00:24) @project(MAIN TASKS.Database)
|
||||
✔ add export methods to all models @done(22-04-12 21:38) @project(MAIN TASKS.Database)
|
||||
✔ add import methods to all models @done(22-04-15 00:24) @project(MAIN TASKS.Database)
|
||||
✔ implement export to json @done(22-04-12 21:43) @project(MAIN TASKS.Settings)
|
||||
✔ implement saving export to google drive @done(22-04-13 15:01) @project(MAIN TASKS.Settings)
|
||||
✔ switch day on swipe @done(22-03-19 23:20) @project(MAIN TASKS.Log)
|
||||
✔ switch day on swipe @done(22-03-19 23:23) @project(MAIN TASKS.Log Events)
|
||||
✔ switch day on swipe @done(22-03-19 23:20) @project(MAIN TASKS.Reports)
|
||||
|
@ -0,0 +1 @@
|
||||
{"installed":{"client_id":"988592836243-pmdkvghnvd6fdeo4qm0sjgstcqvrbhqs.apps.googleusercontent.com","project_id":"coastal-fiber-347020","auth_uri":"https://accounts.google.com/o/oauth2/auth","token_uri":"https://oauth2.googleapis.com/token","auth_provider_x509_cert_url":"https://www.googleapis.com/oauth2/v1/certs"}}
|
501
assets/i18n/de.json
Normal file
501
assets/i18n/de.json
Normal file
@ -0,0 +1,501 @@
|
||||
{
|
||||
"accuracy": {
|
||||
"confirmDelete": "Willst du diese Präzision wirklich löschen?",
|
||||
"deleted": "Präzision gelöscht",
|
||||
"detail": {
|
||||
"title": "{status} Präzision"
|
||||
},
|
||||
"empty": "Du hast noch keine Präzisionen erstellt!",
|
||||
"fields": {
|
||||
"confidenceRating": "Reihung",
|
||||
"fürCarbsRatio": "für KH-Verhältnis",
|
||||
"fürPortionSize": "für Portionsgröße",
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"new": "Neue",
|
||||
"saved": {
|
||||
"1": "{status}Präzision gespeichert",
|
||||
"else": "{status}Präzisionen gespeichert"
|
||||
},
|
||||
"title": "Präzision"
|
||||
},
|
||||
"basal": {
|
||||
"confirmDelete": "Willst du diese Basalrate wirklich löschen?",
|
||||
"deleted": "Basalrate gelöscht",
|
||||
"empty": "Du hast noch keine Basalraten erstellt!",
|
||||
"fields": {
|
||||
"endTime": "Endzeit",
|
||||
"startTime": "Startzeit",
|
||||
"units": "Einheiten"
|
||||
},
|
||||
"new": "Neue",
|
||||
"saved": {
|
||||
"1": "{status}Basalrate gespeichert",
|
||||
"else": "{status}Basalraten gespeichert"
|
||||
},
|
||||
"title": "{status} Basalrate für {profileName}",
|
||||
"warnings": {
|
||||
"duplicate": "Es gibt bereits eine Rate mit dieser Startzeit.",
|
||||
"endTimeLast": "Letzter Basal des Tages muss um 00:00 enden",
|
||||
"gap": "Lücke zwischen dieser und der vorigen Rate",
|
||||
"overlap": "Zeitraum der Rate überlappt mit einer anderen",
|
||||
"startTimeFirst": "Erster Basal des Tages muss um 00:00 beginnen"
|
||||
}
|
||||
},
|
||||
"basalProfile": {
|
||||
"activated": "{profileName} wurde als aktives Profil ausgewählt",
|
||||
"active": "Derzeit aktives Profil",
|
||||
"confirmDelete": "Willst du dieseBasal Profile?",
|
||||
"copied": "Kopie von {profileName} hinzugefügt",
|
||||
"copyOf": "Kopie von {profileName}",
|
||||
"default": "Standard-Profil",
|
||||
"deleted": "Basalprofil gelöscht",
|
||||
"detail": {
|
||||
"tabs": {
|
||||
"profile": "Profil",
|
||||
"rates": "Raten"
|
||||
},
|
||||
"title": "{status} Basalprofil {profileName}"
|
||||
},
|
||||
"empty": "Du hast noch kein Basalprofil erstellt!",
|
||||
"fields": {
|
||||
"active": "aktiv",
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"new": "Neues",
|
||||
"saved": "{status}Basalprofil gespeichert",
|
||||
"title": "Basalprofile",
|
||||
"warnings": {
|
||||
"activeAlreadySet": "Es gibt bereits ein oder mehrere aktive Profile. Was möchtest du tun?",
|
||||
"multipleActive": "Mehr als ein aktives Profil.",
|
||||
"noActive": "Kein aktives Basalprofil.",
|
||||
"noActiveOnCreate": "Im Moment ist kein Profil aktiv. Möchtest du dieses aktivieren?",
|
||||
"resolve": {
|
||||
"activate": "Profil aktivieren",
|
||||
"activateCurrent": "Dieses Profil aktivieren",
|
||||
"create": "Profil erstellen",
|
||||
"createInstead": "Stattdessen neues Profil erstellen",
|
||||
"deactivateOthers": "Alle anderen deaktivieren",
|
||||
"deactivateProfile": "{profileName} deaktivieren",
|
||||
"ignore": "Ignorierem",
|
||||
"pick": "Wähle ein Profil"
|
||||
}
|
||||
}
|
||||
},
|
||||
"bolus": {
|
||||
"confirmDelete": "Willst du diese Bolusrate wirklich löschen?",
|
||||
"deleted": "Bolusrate gelöscht",
|
||||
"empty": "Du hast noch keine Bolusraten erstellt!",
|
||||
"fields": {
|
||||
"endTime": "Endzeit",
|
||||
"perCarbs": "pro KH",
|
||||
"perGlucose": "pro {glucoseMeasurementSuffix}",
|
||||
"startTime": "Startzeit",
|
||||
"units": "Einheiten"
|
||||
},
|
||||
"new": "Neue",
|
||||
"saved": {
|
||||
"1": "{status}Bolusrate gespeichert",
|
||||
"else": "{status}Bolusraten gespeichert"
|
||||
},
|
||||
"title": "{status} Bolusrate für {profileName}",
|
||||
"warnings": {
|
||||
"duplicate": "Es gibt bereits eine Rate mit dieser Startzeit.",
|
||||
"endTimeLast": "Letzter Bolus des Tages muss um 00:00 enden",
|
||||
"gap": "Lücke zwischen dieser und der vorigen Rate",
|
||||
"overlap": "Zeitraum der Rate überlappt mit einer anderen",
|
||||
"startTimeFirst": "Erster Bolus des Tages muss um 00:00 beginnen"
|
||||
}
|
||||
},
|
||||
"bolusProfile": {
|
||||
"activated": "{profileName} wurde als aktives Profil ausgewählt",
|
||||
"active": "Derzeit aktives Profil",
|
||||
"confirmDelete": "Willst du dieses Boluspofil wirklich löschen?",
|
||||
"copied": "Kopie von {profileName} hinzugefügt",
|
||||
"copyOf": "Kopie von {profileName}",
|
||||
"default": "Standardprofil",
|
||||
"deleted": "Bolusprofil gelöscht",
|
||||
"detail": {
|
||||
"tabs": {
|
||||
"profile": "Profil",
|
||||
"rates": "Raten"
|
||||
},
|
||||
"title": "{status} Bolusprofil {profileName}"
|
||||
},
|
||||
"empty": "Du hast noch keine Bolusprofile erstellt!",
|
||||
"fields": {
|
||||
"active": "aktiv",
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"validators": {
|
||||
"name": "Bezeichnung ist leer"
|
||||
}
|
||||
},
|
||||
"new": "Neues",
|
||||
"saved": "{status}Bolusprofil gespeichert",
|
||||
"title": "Bolusprofile",
|
||||
"warnings": {
|
||||
"activeAlreadySet": "Es gibt bereits ein oder mehrere aktive Profile. Was möchtest du tun?",
|
||||
"multipleActive": "Mehr als ein aktives Profil.",
|
||||
"noActive": "Kein aktives Bolusprofil.",
|
||||
"noActiveOnCreate": "Im Moment ist kein Profil aktiv. Möchtest du dieses aktivieren?",
|
||||
"resolve": {
|
||||
"activate": "Profil aktivieren",
|
||||
"activateCurrent": "Dieses Profil aktivieren",
|
||||
"create": "Profil erstellen",
|
||||
"createInstead": "Stattdessen neues Profil erstellen",
|
||||
"deactivateOthers": "Alle anderen deaktivieren",
|
||||
"deactivateProfile": "{profileName} deaktivieren",
|
||||
"ignore": "Ignorierem",
|
||||
"pick": "Wähle ein Profil"
|
||||
}
|
||||
}
|
||||
},
|
||||
"categories": "Kategorien",
|
||||
"event": {
|
||||
"confirmDelete": "Willst du dieses Ereignis wirklich löschen?",
|
||||
"confirmEnd": "Willst du dieses Ereignis wirklich beenden?",
|
||||
"deleted": "Log-Ereignis gelöscht",
|
||||
"detail": {
|
||||
"title": "{status} Log-Ereignis {name}"
|
||||
},
|
||||
"empty": "Kkeine Ereigniss für dieses Datum!",
|
||||
"emptyActive": "Keine aktiven Ereignisse!",
|
||||
"end": "Ereignis beenden",
|
||||
"ended": "Log-Ereignis beendet",
|
||||
"fields": {
|
||||
"basalProfile": "Basalprofil",
|
||||
"bolusProfile": "Bolusprofil",
|
||||
"date": "Datum",
|
||||
"endDate": "Enddatum",
|
||||
"endTime": "Endzeit",
|
||||
"eventType": "Ereignis-Typ",
|
||||
"hasEndTime": "hat Endzeitpunkt",
|
||||
"notes": "Bemerkung",
|
||||
"reminderDuration": "Dauer für Erinnerung",
|
||||
"startDate": "Startdatum",
|
||||
"startTime": "Startzeit",
|
||||
"time": "Zeit"
|
||||
},
|
||||
"new": "Neues",
|
||||
"saved": "{status}Log-Ereignis gespeichert",
|
||||
"title": "Log-Ereignisse",
|
||||
"titleActive": "Aktive Ereignisse",
|
||||
"warnings": {
|
||||
"duplicate": "Im angegebenen Zeitraum gibt es bereits ein aktives Ereignis vom selben Typ. Was möchtest du tun?"
|
||||
}
|
||||
},
|
||||
"eventType": {
|
||||
"deleted": "Log-Ereignis-Typ gelöscht",
|
||||
"detail": {
|
||||
"title": "{status} Log-Ereignis-Typ {name}"
|
||||
},
|
||||
"empty": "Du hast noch keine Log-Ereignis-Typen erstellt!",
|
||||
"fields": {
|
||||
"basalProfile": "Basalprofil",
|
||||
"bolusProfile": "Bolusprofil",
|
||||
"defaultReminderDuration": "Standarddauer für Erinnerung",
|
||||
"hasEndTime": "hat Endzeitpunkt",
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"new": "Neuer",
|
||||
"saved": "{status}Log-Ereignis-Typ gespeichert",
|
||||
"title": "Log-Ereignis-Typen"
|
||||
},
|
||||
"export": {
|
||||
"error": "Error beim Import der folgenden Daten: {data}"
|
||||
},
|
||||
"general": {
|
||||
"apply": "Anwenden",
|
||||
"cancel": "Abbrechen",
|
||||
"close": "Schließen",
|
||||
"confirm": "Bestätigen",
|
||||
"confirmDelete": "Willst du diesen Eintrag wirklich löschen?",
|
||||
"confirmDiscard": "Änderungen wurden bereits vorgenommen. Eingaben verwerfen?",
|
||||
"discard": "Verwerfen",
|
||||
"edit": "Bearbeiten",
|
||||
"example": "Beispiel",
|
||||
"keepEditing": "Weiter bearbeiten",
|
||||
"next": "Weiter",
|
||||
"per": "pro",
|
||||
"save": "Speichern",
|
||||
"saveAndClose": "Speichern & Schließen",
|
||||
"saveAsIs": "Aktuellen Stand speichern",
|
||||
"suffixes": {
|
||||
"carbs": "{nutritionMeasurement} KH",
|
||||
"carbsPerU": "{nutritionMeasurementSuffix} KH pro E",
|
||||
"mins": " min",
|
||||
"perDay": "am Tag",
|
||||
"units": "E",
|
||||
"uPerBreadUnit": "E pro BE",
|
||||
"uPerGlucose": "{glucoseMeasurementSuffix} pro Einheit"
|
||||
}
|
||||
},
|
||||
"log": {
|
||||
"confirmDelete": "Willst du diesen Logeintrag wirklich löschen?",
|
||||
"deleted": "Logeintrag gelöscht",
|
||||
"empty": "Du hast noch keine Logeinträge für dieses Datum erstellt!",
|
||||
"fields": {
|
||||
"date": "Datum",
|
||||
"glucose": "Blutzucker",
|
||||
"time": "Zeit"
|
||||
},
|
||||
"filter": {
|
||||
"endDate": "Endzeit",
|
||||
"maxGlucose": "max. {glucoseMeasurement}",
|
||||
"meal": "Mahlzeit",
|
||||
"mealNameContains": "Name der Mahlzeit enthält",
|
||||
"minGlucose": "min. {glucoseMeasurement}",
|
||||
"noteContains": "Bemerkung enthält",
|
||||
"startDate": "Startzeit"
|
||||
},
|
||||
"saved": "{status}Logeintrag gespeichert",
|
||||
"tabs": {
|
||||
"bolus": {
|
||||
"confirmDelete": "Willst du diesen Bolus wirklich löschen?",
|
||||
"delayedBy": "(um {delay} verzögert)",
|
||||
"deleted": "Bolus gelöscht",
|
||||
"detail": {
|
||||
"fields": {
|
||||
"carbs": "KH",
|
||||
"correction": "Korrektur",
|
||||
"current": "Aktuell",
|
||||
"delayedBolus": "Verzögerter Bolus",
|
||||
"delayedBolusDuration": "Dauer Verzögerter Bolus",
|
||||
"forGlucose": "für Blutzucker",
|
||||
"forMeal": "für Mahlzeit",
|
||||
"immediateBolus": "Sofortiger Bolus",
|
||||
"meal": "Mahlzeit",
|
||||
"setManually": "manuall einstellen",
|
||||
"target": "Zielwert",
|
||||
"units": "Bolus-Einheiten"
|
||||
},
|
||||
"title": "{status} Bolus"
|
||||
},
|
||||
"empty": "Du hast noch keine Boli für diesen Logeintrag erstellt!",
|
||||
"forMeal": "für {meal}",
|
||||
"new": "Neuer",
|
||||
"saved": "{status}Bolus gespeichert",
|
||||
"title": "Boli",
|
||||
"toCorrect": "zur Korrektur von {correction}"
|
||||
},
|
||||
"general": "Allgemein",
|
||||
"meal": {
|
||||
"confirmDelete": "Willst du diese Mahlzeit wirklich löschen?",
|
||||
"deleted": "Mahlzeit gelöscht",
|
||||
"detail": {
|
||||
"fields": {
|
||||
"amount": "Menge",
|
||||
"carbsRatio": "KH-Verhältnis",
|
||||
"carbsRatioAccuracy": "Präzision KH-Verhältnis",
|
||||
"meal": "Mahlzeit",
|
||||
"mealCategory": "Kategorie",
|
||||
"mealPortionType": "Portionstyp",
|
||||
"mealSource": "Herkunft",
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"portionSize": "Portionsgröße",
|
||||
"portionSizeAccuracy": "Präzision Portionsgröße",
|
||||
"setManually": "KH-Verhältnis manuell einstellen",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"title": "{status} Mahlzeit"
|
||||
},
|
||||
"empty": "Du hast noch keine Mahlzeiten für diesen Logeintrag erstellt!",
|
||||
"new": "Neue",
|
||||
"saved": "{status}Mahlzeit gespeichert",
|
||||
"title": "Mahlzeiten"
|
||||
}
|
||||
},
|
||||
"title": "Logeinträge"
|
||||
},
|
||||
"meal": {
|
||||
"confirmDelete": "Willst du diese Mahlzeit wirklich löschen?",
|
||||
"deleted": "Mahlzeit gelöscht",
|
||||
"detail": {
|
||||
"title": "{status} Mahlzeit {name}"
|
||||
},
|
||||
"empty": "Du hast noch keine Mahlzeiten erstellt!",
|
||||
"fields": {
|
||||
"additional": {
|
||||
"carbsRatioAccuracy": "Präzision KH-Verhältnis",
|
||||
"mealCategory": "Kategorie",
|
||||
"portionSizeAccuracy": "Präzision Portionsgröße",
|
||||
"title": "Zusätzliche Felder"
|
||||
},
|
||||
"carbsPerPortion": "KH pro Portion",
|
||||
"carbsRatio": "KH-Verhältnis",
|
||||
"delay": {
|
||||
"duration": "Dauer",
|
||||
"title": "Verzögerter Bolus"
|
||||
},
|
||||
"mealPortionType": "Portionstyp",
|
||||
"mealSource": "Herkunft",
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"portionSize": "Portionsgröße",
|
||||
"setManually": "KH-Verhältnis manuell einstellen",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"new": "Neue",
|
||||
"saved": "{status}Mahlzeit gespeichert",
|
||||
"title": "Mahlzeiten"
|
||||
},
|
||||
"mealCategory": {
|
||||
"deleted": "Mahlzeiten-Kategorie gelöscht",
|
||||
"detail": {
|
||||
"title": "{status} Mahlzeiten-Kategorie {name}"
|
||||
},
|
||||
"empty": "Du hast noch keine Mahlzeiten-Kategorien erstellt!",
|
||||
"fields": {
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Mahlzeiten-Kategorie gespeichert",
|
||||
"title": "Mahlzeiten-Kategorien"
|
||||
},
|
||||
"mealSource": {
|
||||
"deleted": "Mahlzeiten-Herkunft gelöscht",
|
||||
"detail": {
|
||||
"title": "{status} Mahlzeiten-Herkunft {name}"
|
||||
},
|
||||
"empty": "Du hast noch keine Mahlzeiten-Herkünfte erstellt!",
|
||||
"fields": {
|
||||
"defaultCarbsRatioAccuracy": "Standard-Präzision KH-Verhältnis",
|
||||
"defaultMealCategory": "Standard Mahlzeiten-Kategorie",
|
||||
"defaultMealPortionType": "Standard-Portionstyp",
|
||||
"defaultPortionSizeAccuracy": "Standard-Präzision Portionsgröße",
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Mahlzeiten-Herkunft gespeichert",
|
||||
"title": "Mahlzeiten-Herkünfte"
|
||||
},
|
||||
"navigation": {
|
||||
"basalProfiles": "Basalprofile",
|
||||
"bolusProfiles": "Bolusprofile",
|
||||
"categorization": "Kategorisierung",
|
||||
"log": "Log",
|
||||
"logEvent": "Log-Ereignisse",
|
||||
"meals": "Mahlzeiten",
|
||||
"reports": "Berichte",
|
||||
"settings": "Einstellungen"
|
||||
},
|
||||
"portionType": {
|
||||
"deleted": "Mahlzeiten-Portionstyp gelöscht",
|
||||
"detail": {
|
||||
"title": "{status} Mahlzeiten-Portionstyp {name}"
|
||||
},
|
||||
"empty": "Du hast noch keine Mahlzeiten-Portionstypen erstellt!",
|
||||
"fields": {
|
||||
"name": "Name",
|
||||
"notes": "Bemerkung",
|
||||
"validators": {
|
||||
"name": "Name ist leer"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Mahlzeiten-Portionstyp gespeichert",
|
||||
"title": "Mahlzeiten-Portionstypen"
|
||||
},
|
||||
"reports": {
|
||||
"dailyChart": {
|
||||
"empty": "Du hast noch keine Logeinträge für dieses Datum erstellt!",
|
||||
"showBasal": "zeige Basal",
|
||||
"showBolus": "zeige Bolus",
|
||||
"showChart": "zeige Grafik",
|
||||
"showMeals": "zeige Mahlzeiten",
|
||||
"title": "Tagesgrafik"
|
||||
},
|
||||
"export": {
|
||||
"date": "Datum",
|
||||
"delayedBy": "(um {delay} verzögert)",
|
||||
"endDate": "Enddatum",
|
||||
"export": "exportieren",
|
||||
"range": "Zeitraum",
|
||||
"showBasal": "zeige Basal",
|
||||
"showBolus": "zeige Bolus",
|
||||
"showChart": "zeige Grafik",
|
||||
"showMahlzeits": "zeige Mahlzeiten",
|
||||
"showTable": "zeige Tabelle",
|
||||
"singleDate": "Bestimmtes Datum",
|
||||
"tableHeaders": {
|
||||
"bolus": "Bolus",
|
||||
"carbs": "Kohlenhydrate",
|
||||
"glucose": "Blutzucker",
|
||||
"mealBolus": "Mahlzeit Bolus",
|
||||
"mealBemerkung": "Mahlzeit Bemerkung",
|
||||
"meals": "Mahlzeit",
|
||||
"notes": "Bemerkung",
|
||||
"portionSize": "Portionsgröße",
|
||||
"time": "Zeit"
|
||||
},
|
||||
"title": "Log-Übersicht"
|
||||
},
|
||||
"ids": {
|
||||
"basal": "Basal",
|
||||
"bolus": "Bolus",
|
||||
"carbs": "Kohlenhydrate",
|
||||
"glucose": "Blutzucker"
|
||||
},
|
||||
"sections": {
|
||||
"dailyReport": "Tagesbericht",
|
||||
"pdfReport": "PDF-Übersicht"
|
||||
},
|
||||
"title": "Berichte"
|
||||
},
|
||||
"settings": {
|
||||
"confirmReset": "Bist du sicher, dass du alle Einstellungen zurücksetzen möchtest?",
|
||||
"fields": {
|
||||
"confirmOnCancel": "beim Abbrechen der Bearbeitung eines Eintrags, wenn bereits Änderungen vorgenommen wurden",
|
||||
"confirmOnDelete": "beim Löschen eines Eintrags",
|
||||
"confirmOnEndEreignis": "beim Beenden eines Ereignisses",
|
||||
"dateformat": "Datumsformat",
|
||||
"displayBothDetail": "zeige beide Blutzucker-Maßeinheiten in der Detailansicht",
|
||||
"displayBothList": "display both glucose measurements in list view",
|
||||
"glucoseMeasurement": "Bevorzugte Maßeinheit für Blutzuckerwerte",
|
||||
"insulinIncrement": "Insulin-Dosierungsschritte",
|
||||
"longDateformat": "Datumsformat lang",
|
||||
"longTimeformat": "Zeitformat lang",
|
||||
"mmolLIncrement": "Schritte Mmol/l",
|
||||
"nutritionIncrement": "Schritte Maßeinheit für Mahlzeiten",
|
||||
"nutritionMeasurement": "Bevorzugte Maßeinheit für Mahlzeiten",
|
||||
"onlyDisplayActive": "zeige nur aktive Blutzucker-Maßeinheit",
|
||||
"targetGlucose": "Ziel-Blutzucker",
|
||||
"timeformat": "Zeitformat"
|
||||
},
|
||||
"reset": "Einstellungen wurden auf Standardeinstellungen zurückgesetzt",
|
||||
"resetAll": "Alle zurücksetzen",
|
||||
"sections": {
|
||||
"confirmation": "Bestätigungen",
|
||||
"dateTimeformat": "Zeit- und Datumsformat",
|
||||
"measurements": "Maßeinheiten"
|
||||
},
|
||||
"title": "Applikationseinstellungen",
|
||||
"updated": "Einstellungen gespeichert"
|
||||
}
|
||||
}
|
501
assets/i18n/en.json
Normal file
501
assets/i18n/en.json
Normal file
@ -0,0 +1,501 @@
|
||||
{
|
||||
"accuracy": {
|
||||
"confirmDelete": "Are you sure you want to delete this Accuracy?",
|
||||
"deleted": "Accuracy deleted",
|
||||
"detail": {
|
||||
"title": "{status} Accuracy"
|
||||
},
|
||||
"empty": "You have not created any Accuracies yet!",
|
||||
"fields": {
|
||||
"confidenceRating": "Confidence Rating",
|
||||
"forCarbsRatio": "for Carbs Ratio",
|
||||
"forPortionSize": "for Portion Size",
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"validators": {
|
||||
"name": "Empty Name"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": {
|
||||
"1": "{status}Accuracy saved",
|
||||
"else": "{status}Accuracy saved"
|
||||
},
|
||||
"title": "Accuracies"
|
||||
},
|
||||
"basal": {
|
||||
"confirmDelete": "Are you sure you want to delete this Basal Rate?",
|
||||
"deleted": "Basal Rate deleted",
|
||||
"empty": "You have not created any Basal Rates yet!",
|
||||
"fields": {
|
||||
"endTime": "End Time",
|
||||
"startTime": "Start Time",
|
||||
"units": "Units"
|
||||
},
|
||||
"new": "New",
|
||||
"saved": {
|
||||
"1": "{status}Basal Rate saved",
|
||||
"else": "{status}Basal Rates saved"
|
||||
},
|
||||
"title": "{status} Basal Rate for {profileName}",
|
||||
"warnings": {
|
||||
"duplicate": "There's already a rate with this start time.",
|
||||
"endTimeLast": "Last Basal of the day needs to end at 00:00",
|
||||
"gap": "There's a time gap between this and the previous rate",
|
||||
"overlap": "This rate's time period overlaps with another one.",
|
||||
"startTimeFirst": "First Basal of the day needs to start at 00:00"
|
||||
}
|
||||
},
|
||||
"basalProfile": {
|
||||
"activated": "{profileName} has been set as your active Profile",
|
||||
"active": "Current Active Profile",
|
||||
"confirmDelete": "Are you sure you want to delete this Basal Profile?",
|
||||
"copied": "Added copy of {profileName}",
|
||||
"copyOf": "Copy of {profileName}",
|
||||
"default": "Default Profile",
|
||||
"deleted": "Basal Profile deleted",
|
||||
"detail": {
|
||||
"tabs": {
|
||||
"profile": "Profile",
|
||||
"rates": "Rates"
|
||||
},
|
||||
"title": "{status} Basal Profile {profileName}"
|
||||
},
|
||||
"empty": "You have not created any Basal Profiles yet!",
|
||||
"fields": {
|
||||
"active": "active",
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"validators": {
|
||||
"name": "Empty title"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Basal Profile saved",
|
||||
"title": "Basal Profiles",
|
||||
"warnings": {
|
||||
"activeAlreadySet": "There are already one or more active profiles. What would you like to do?",
|
||||
"multipleActive": "More than one active Basal Profile has been found.",
|
||||
"noActive": "You currently do not have an active Basal Profile.",
|
||||
"noActiveOnCreate": "There is currently no active profile. Would you like to set this one as active?",
|
||||
"resolve": {
|
||||
"activate": "Activate a Profile",
|
||||
"activateCurrent": "Activate This Profile",
|
||||
"create": "Create a Profile",
|
||||
"createInstead": "Create a New Profile Instead",
|
||||
"deactivateOthers": "Deactivate All Others",
|
||||
"deactivateProfile": "Deactivate {profileName}",
|
||||
"ignore": "Ignore",
|
||||
"pick": "Pick a Profile"
|
||||
}
|
||||
}
|
||||
},
|
||||
"bolus": {
|
||||
"confirmDelete": "Are you sure you want to delete this Bolus Rate?",
|
||||
"deleted": "Bolus Rate deleted",
|
||||
"empty": "You have not created any Bolus Rates yet!",
|
||||
"fields": {
|
||||
"endTime": "End Time",
|
||||
"perCarbs": "per carns",
|
||||
"perGlucose": "per {glucoseMeasurementSuffix}",
|
||||
"startTime": "Start Time",
|
||||
"units": "Units"
|
||||
},
|
||||
"new": "New",
|
||||
"saved": {
|
||||
"1": "{status}Bolus Rate saved",
|
||||
"else": "{status}Bolus Rates saved"
|
||||
},
|
||||
"title": "{status} Bolus Rate for {profileName}",
|
||||
"warnings": {
|
||||
"duplicate": "There's already a rate with this start time.",
|
||||
"endTimeLast": "Last Bolus of the day needs to end at 00:00",
|
||||
"gap": "There's a time gap between this and the previous rate",
|
||||
"overlap": "This rate's time period overlaps with another one.",
|
||||
"startTimeFirst": "First Bolus of the day needs to start at 00:00"
|
||||
}
|
||||
},
|
||||
"bolusProfile": {
|
||||
"activated": "{profileName} has been set as your active Profile",
|
||||
"active": "Current Active Profile",
|
||||
"confirmDelete": "Are you sure you want to delete this Bolus Profile?",
|
||||
"copied": "Added copy of {profileName}",
|
||||
"copyOf": "Copy of {profileName}",
|
||||
"default": "Default Profile",
|
||||
"deleted": "Bolus Profile deleted",
|
||||
"detail": {
|
||||
"tabs": {
|
||||
"profile": "Profile",
|
||||
"rates": "Rates"
|
||||
},
|
||||
"title": "{status} Bolus Profile {profileName}"
|
||||
},
|
||||
"empty": "You have not created any Bolus Profiles yet!",
|
||||
"fields": {
|
||||
"active": "active",
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"validators": {
|
||||
"name": "Empty title"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Bolus Profile saved",
|
||||
"title": "Bolus Profiles",
|
||||
"warnings": {
|
||||
"activeAlreadySet": "There are already one or more active profiles. What would you like to do?",
|
||||
"multipleActive": "More than one active Bolus Profile has been found.",
|
||||
"noActive": "You currently do not have an active Bolus Profile.",
|
||||
"noActiveOnCreate": "There is currently no active profile. Would you like to set this one as active?",
|
||||
"resolve": {
|
||||
"activate": "Activate a Profile",
|
||||
"activateCurrent": "Activate This Profile",
|
||||
"create": "Create a Profile",
|
||||
"createInstead": "Create a New Profile Instead",
|
||||
"deactivateOthers": "Deactivate All Others",
|
||||
"deactivateProfile": "Deactivate {profileName}",
|
||||
"ignore": "Ignore",
|
||||
"pick": "Pick a Profile"
|
||||
}
|
||||
}
|
||||
},
|
||||
"categories": "Categories",
|
||||
"event": {
|
||||
"confirmDelete": "Are you sure you want to delete this Log Event?",
|
||||
"confirmEnd": "Are you sure you want to end this Event?",
|
||||
"deleted": "Log Event deleted",
|
||||
"detail": {
|
||||
"title": "{status} Log Event {name}"
|
||||
},
|
||||
"empty": "There are no Events for that date!",
|
||||
"emptyActive": "There are no Active Events!",
|
||||
"end": "End Event",
|
||||
"ended": "Log Event ended",
|
||||
"fields": {
|
||||
"basalProfile": "Basal Profile",
|
||||
"bolusProfile": "Bolus Profile",
|
||||
"date": "Date",
|
||||
"endDate": "End Date",
|
||||
"endTime": "End Time",
|
||||
"eventType": "Event Type",
|
||||
"hasEndTime": "has end time",
|
||||
"notes": "Notes",
|
||||
"reminderDuration": "Reminder Duration",
|
||||
"startDate": "Start Date",
|
||||
"startTime": "Start Time",
|
||||
"time": "Time"
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Log Event saved",
|
||||
"title": "Log Events",
|
||||
"titleActive": "Active Events",
|
||||
"warnings": {
|
||||
"duplicate": "An Event of this type is already active within the set time frame. What would you like to do?"
|
||||
}
|
||||
},
|
||||
"eventType": {
|
||||
"deleted": "Log Event Type deleted",
|
||||
"detail": {
|
||||
"title": "{status} Log Event Type {name}"
|
||||
},
|
||||
"empty": "You have not created any Log Event Types yet!",
|
||||
"fields": {
|
||||
"basalProfile": "Basal Profile",
|
||||
"bolusProfile": "Bolus Profile",
|
||||
"defaultReminderDuration": "Default Reminder Duration",
|
||||
"hasEndTime": "has end time",
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"validators": {
|
||||
"name": "Empty name"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Log Event Type saved",
|
||||
"title": "Log Event Types"
|
||||
},
|
||||
"export": {
|
||||
"error": "Error importing the following data: {data}"
|
||||
},
|
||||
"general": {
|
||||
"apply": "Apply",
|
||||
"cancel": "Cancel",
|
||||
"close": "Close",
|
||||
"confirm": "Confirm",
|
||||
"confirmDelete": "Are you sure you want to delete this record?",
|
||||
"confirmDiscard": "You already made some changes. Discard your input?",
|
||||
"discard": "Discard",
|
||||
"edit": "Edit",
|
||||
"example": "Example",
|
||||
"keepEditing": "Keep Editing",
|
||||
"next": "Next",
|
||||
"per": "per",
|
||||
"save": "Save",
|
||||
"saveAndClose": "Save & Close",
|
||||
"saveAsIs": "Save As Is",
|
||||
"suffixes": {
|
||||
"carbs": "{nutritionMeasurement} carbs",
|
||||
"carbsPerU": "{nutritionMeasurementSuffix} carbs per U",
|
||||
"mins": " min",
|
||||
"perDay": "per day",
|
||||
"units": "U",
|
||||
"uPerBreadUnit": "U per bread unit",
|
||||
"uPerGlucose": "{glucoseMeasurementSuffix} per unit"
|
||||
}
|
||||
},
|
||||
"log": {
|
||||
"confirmDelete": "Are you sure you want to delete this Log Entry?",
|
||||
"deleted": "Log Entry deleted",
|
||||
"empty": "You have not created any Log Entries for this date yet!",
|
||||
"fields": {
|
||||
"date": "Date",
|
||||
"glucose": "Blood Glucose",
|
||||
"time": "Time"
|
||||
},
|
||||
"filter": {
|
||||
"endDate": "End Date",
|
||||
"maxGlucose": "max {glucoseMeasurement}",
|
||||
"meal": "Meal",
|
||||
"mealNameContains": "Meal Name Contains",
|
||||
"minGlucose": "min {glucoseMeasurement}",
|
||||
"noteContains": "Note Contains",
|
||||
"startDate": "Start Date"
|
||||
},
|
||||
"saved": "{status}Log Entry saved",
|
||||
"tabs": {
|
||||
"bolus": {
|
||||
"confirmDelete": "Are you sure you want to delete this Bolus?",
|
||||
"delayedBy": "(delayed by {delay})",
|
||||
"deleted": "Bolus deleted",
|
||||
"detail": {
|
||||
"fields": {
|
||||
"carbs": "Carbs",
|
||||
"correction": "Correction",
|
||||
"current": "Current",
|
||||
"delayedBolus": "Delayed Bolus",
|
||||
"delayedBolusDuration": "Delayed Bolus Duration",
|
||||
"forGlucose": "for glucose",
|
||||
"forMeal": "for meal",
|
||||
"immediateBolus": "Immediate Bolus",
|
||||
"meal": "Meal",
|
||||
"setManually": "set manually",
|
||||
"target": "Target",
|
||||
"units": "Bolus Units"
|
||||
},
|
||||
"title": "{status} Bolus"
|
||||
},
|
||||
"empty": "You have not added any Boli to this Log Entry yet!",
|
||||
"forMeal": "for {meal}",
|
||||
"new": "New",
|
||||
"saved": "{status}Bolus saved",
|
||||
"title": "Boli",
|
||||
"toCorrect": "to correct {correction}"
|
||||
},
|
||||
"general": "General",
|
||||
"meal": {
|
||||
"confirmDelete": "Are you sure you want to delete this Meal?",
|
||||
"deleted": "Meal deleted",
|
||||
"detail": {
|
||||
"fields": {
|
||||
"amount": "Amount",
|
||||
"carbsRatio": "Carbs Ratio",
|
||||
"carbsRatioAccuracy": "Carbs Ratio Accuracy",
|
||||
"meal": "Meal",
|
||||
"mealCategory": "Meal Category",
|
||||
"mealPortionType": "Meal Portion Type",
|
||||
"mealSource": "Meal Source",
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"portionSize": "Portion Size",
|
||||
"portionSizeAccuracy": "Portion Size Accuracy",
|
||||
"setManually": "set carbs ratio manually",
|
||||
"validators": {
|
||||
"name": "Empty Name"
|
||||
}
|
||||
},
|
||||
"title": "{status} Meal"
|
||||
},
|
||||
"empty": "You have not added any Meals to this Log Entry yet!",
|
||||
"new": "New",
|
||||
"saved": "{status}Meal saved",
|
||||
"title": "Meals"
|
||||
}
|
||||
},
|
||||
"title": "Log Entries"
|
||||
},
|
||||
"meal": {
|
||||
"confirmDelete": "Are you sure you want to delete this Meal?",
|
||||
"deleted": "Meal deleted",
|
||||
"detail": {
|
||||
"title": "{status} Meal {name}"
|
||||
},
|
||||
"empty": "You have not created any Meals yet!",
|
||||
"fields": {
|
||||
"additional": {
|
||||
"carbsRatioAccuracy": "Carbs Ratio Accuracy",
|
||||
"mealCategory": "Meal Category",
|
||||
"portionSizeAccuracy": "Portion Size Accuracy",
|
||||
"title": "Additional Fields"
|
||||
},
|
||||
"carbsPerPortion": "Carbs Per Portion",
|
||||
"carbsRatio": "Carbs Ratio",
|
||||
"delay": {
|
||||
"duration": "Duration",
|
||||
"title": "Bolus Delay"
|
||||
},
|
||||
"mealPortionType": "Meal Portion Type",
|
||||
"mealSource": "Meal Source",
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"portionSize": "Portion Size",
|
||||
"setManually": "set carbs ratio manually",
|
||||
"validators": {
|
||||
"name": "Empty Name"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Meal saved",
|
||||
"title": "Meals"
|
||||
},
|
||||
"mealCategory": {
|
||||
"deleted": "Meal Category deleted",
|
||||
"detail": {
|
||||
"title": "{status} Meal Category {name}"
|
||||
},
|
||||
"empty": "You have not created any Meal Categories yet!",
|
||||
"fields": {
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"validators": {
|
||||
"name": "Empty name"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Meal Category saved",
|
||||
"title": "Meal Categories"
|
||||
},
|
||||
"mealSource": {
|
||||
"deleted": "Meal Source deleted",
|
||||
"detail": {
|
||||
"title": "{status} Meal Source {name}"
|
||||
},
|
||||
"empty": "You have not created any Meal Sources yet!",
|
||||
"fields": {
|
||||
"defaultCarbsRatioAccuracy": "Default Carbs Ratio Accuracy",
|
||||
"defaultMealCategory": "Default Meal Category",
|
||||
"defaultMealPortionType": "Default Meal Portion Type",
|
||||
"defaultPortionSizeAccuracy": "Default Portion Size Accuracy",
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"validators": {
|
||||
"name": "Empty name"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Meal Source saved",
|
||||
"title": "Meal Sources"
|
||||
},
|
||||
"navigation": {
|
||||
"basalProfiles": "Basal Profiles",
|
||||
"bolusProfiles": "Bolus Profiles",
|
||||
"categorization": "Categorization",
|
||||
"log": "Log",
|
||||
"logEvents": "Log Events",
|
||||
"meals": "Meals",
|
||||
"reports": "Reports",
|
||||
"settings": "Settings"
|
||||
},
|
||||
"portionType": {
|
||||
"deleted": "Meal Portion Type deleted",
|
||||
"detail": {
|
||||
"title": "{status} Meal Portion Type {name}"
|
||||
},
|
||||
"empty": "You have not created any Meal Portion Types yet!",
|
||||
"fields": {
|
||||
"name": "Name",
|
||||
"notes": "Notes",
|
||||
"validators": {
|
||||
"name": "Empty name"
|
||||
}
|
||||
},
|
||||
"new": "New",
|
||||
"saved": "{status}Meal Portion Type saved",
|
||||
"title": "Meal Portion Types"
|
||||
},
|
||||
"reports": {
|
||||
"dailyChart": {
|
||||
"empty": "You have not created any Log Entries for this date yet!",
|
||||
"showBasal": "show Basal",
|
||||
"showBolus": "show Bolus",
|
||||
"showChart": "show Chart",
|
||||
"showMeals": "show Meals",
|
||||
"title": "Daily Chart"
|
||||
},
|
||||
"export": {
|
||||
"date": "Date",
|
||||
"delayedBy": "(delayed by {delay})",
|
||||
"endDate": "End Date",
|
||||
"export": "export",
|
||||
"range": "Range",
|
||||
"showBasal": "show Basal",
|
||||
"showBolus": "show Bolus",
|
||||
"showChart": "show Chart",
|
||||
"showMeals": "show Meals",
|
||||
"showTable": "show Table",
|
||||
"singleDate": "Single Date",
|
||||
"tableHeaders": {
|
||||
"bolus": "Bolus",
|
||||
"carbs": "Carbohydrates",
|
||||
"glucose": "Glucose",
|
||||
"mealBolus": "Meal Bolus",
|
||||
"mealNotes": "Meal Notes",
|
||||
"meals": "Meals",
|
||||
"notes": "Notes",
|
||||
"portionSize": "Portion Size",
|
||||
"time": "Time"
|
||||
},
|
||||
"title": "Log Report"
|
||||
},
|
||||
"ids": {
|
||||
"basal": "Basal",
|
||||
"bolus": "Bolus",
|
||||
"carbs": "Carbohydrates",
|
||||
"glucose": "Glucose"
|
||||
},
|
||||
"sections": {
|
||||
"dailyReport": "Daily Report",
|
||||
"pdfReport": "PDF Report"
|
||||
},
|
||||
"title": "Reports"
|
||||
},
|
||||
"settings": {
|
||||
"confirmReset": "Are you sure you want to reset all settings?",
|
||||
"fields": {
|
||||
"confirmOnCancel": "on cancelling edit or creation of a record if changes have already been made",
|
||||
"confirmOnDelete": "on deleting a record",
|
||||
"confirmOnEndEvent": "on stopping (ending) an event",
|
||||
"dateFormat": "Date Format",
|
||||
"displayBothDetail": "display both glucose measurements in detail view",
|
||||
"displayBothList": "display both glucose measurements in list view",
|
||||
"glucoseMeasurement": "Preferred Glucose Measurement",
|
||||
"insulinIncrement": "Insulin Increment",
|
||||
"longDateFormat": "Long Date Format",
|
||||
"longTimeFormat": "Long Time Format",
|
||||
"mmolLIncrement": "Mmol/l Increment",
|
||||
"nutritionIncrement": "Nutrition Increment",
|
||||
"nutritionMeasurement": "Preferred Nutrition Measurement",
|
||||
"onlyDisplayActive": "only display active glucose measurement",
|
||||
"targetGlucose": "Target Glucose",
|
||||
"timeFormat": "Time Format"
|
||||
},
|
||||
"reset": "Settings have been reset to default",
|
||||
"resetAll": "Reset All",
|
||||
"sections": {
|
||||
"confirmation": "Confirmation Prompts",
|
||||
"dateTimeFormat": "Time & Date Format",
|
||||
"measurements": "Measurements"
|
||||
},
|
||||
"title": "Application Settings",
|
||||
"updated": "Settings updated"
|
||||
}
|
||||
}
|
6
build.yaml
Normal file
6
build.yaml
Normal file
@ -0,0 +1,6 @@
|
||||
targets:
|
||||
$default:
|
||||
sources:
|
||||
include:
|
||||
- assets/i18n/**
|
||||
- lib/**
|
@ -291,7 +291,7 @@
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.tide;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.diameter;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||
SWIFT_VERSION = 5.0;
|
||||
@ -415,7 +415,7 @@
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.tide;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.diameter;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
|
||||
@ -434,7 +434,7 @@
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = Runner/Info.plist;
|
||||
LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.tide;
|
||||
PRODUCT_BUNDLE_IDENTIFIER = com.example.diameter;
|
||||
PRODUCT_NAME = "$(TARGET_NAME)";
|
||||
SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h";
|
||||
SWIFT_VERSION = 5.0;
|
||||
|
@ -11,7 +11,7 @@
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>tide</string>
|
||||
<string>diameter</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>APPL</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
|
@ -5,7 +5,7 @@ class AppTheme {
|
||||
AppTheme._();
|
||||
|
||||
static ThemeData lightTheme = FlexColorScheme.light(
|
||||
surfaceStyle: FlexSurface.medium,
|
||||
// surfaceMode: FlexSurfaceMode.level,
|
||||
scheme: FlexScheme.aquaBlue,
|
||||
fontFamily: 'Roboto',
|
||||
).toTheme;
|
||||
|
@ -1,11 +1,13 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class DetailBottomRow extends StatefulWidget {
|
||||
final void Function()? onCancel;
|
||||
final void Function()? onAction;
|
||||
final void Function()? onMiddleAction;
|
||||
final String actionText;
|
||||
final String middleActionText;
|
||||
final String actionTextKey;
|
||||
final String middleActionTextKey;
|
||||
final IconData actionIcon;
|
||||
final IconData middleActionIcon;
|
||||
|
||||
@ -14,9 +16,9 @@ class DetailBottomRow extends StatefulWidget {
|
||||
required this.onCancel,
|
||||
required this.onAction,
|
||||
this.onMiddleAction,
|
||||
this.actionText = 'SAVE',
|
||||
this.actionTextKey = LocalizationKeys.general_save,
|
||||
this.actionIcon = Icons.save,
|
||||
this.middleActionText = 'SAVE & CLOSE',
|
||||
this.middleActionTextKey = LocalizationKeys.general_saveAndClose,
|
||||
this.middleActionIcon = Icons.done})
|
||||
: super(key: key);
|
||||
|
||||
@ -39,7 +41,7 @@ class _DetailBottomRowState<T> extends State<DetailBottomRow> {
|
||||
Icons.close,
|
||||
size: 18.0,
|
||||
),
|
||||
label: const Text('CANCEL'),
|
||||
label: Text(translate(LocalizationKeys.general_cancel).toUpperCase()),
|
||||
),
|
||||
widget.onMiddleAction != null
|
||||
? ElevatedButton.icon(
|
||||
@ -48,7 +50,7 @@ class _DetailBottomRowState<T> extends State<DetailBottomRow> {
|
||||
widget.middleActionIcon,
|
||||
size: 18.0,
|
||||
),
|
||||
label: Text(widget.middleActionText),
|
||||
label: Text(translate(widget.middleActionTextKey)),
|
||||
)
|
||||
: const Spacer(),
|
||||
ElevatedButton.icon(
|
||||
@ -57,7 +59,7 @@ class _DetailBottomRowState<T> extends State<DetailBottomRow> {
|
||||
widget.actionIcon,
|
||||
size: 18.0,
|
||||
),
|
||||
label: Text(widget.actionText),
|
||||
label: Text(translate(widget.actionTextKey)),
|
||||
),
|
||||
],
|
||||
),
|
||||
|
@ -6,13 +6,15 @@ class TimeOfDayFormField extends StatefulWidget {
|
||||
final TextEditingController controller;
|
||||
final String label;
|
||||
final void Function(TimeOfDay?) onChanged;
|
||||
final String? Function(String?)? validator;
|
||||
|
||||
const TimeOfDayFormField(
|
||||
{Key? key,
|
||||
required this.time,
|
||||
required this.controller,
|
||||
required this.label,
|
||||
required this.onChanged})
|
||||
required this.onChanged,
|
||||
this.validator})
|
||||
: super(key: key);
|
||||
|
||||
@override
|
||||
@ -35,6 +37,7 @@ class _TimeOfDayFormFieldState extends State<TimeOfDayFormField> {
|
||||
);
|
||||
widget.onChanged(newTime);
|
||||
},
|
||||
validator: widget.validator,
|
||||
);
|
||||
}
|
||||
}
|
366
lib/data_export.dart
Normal file
366
lib/data_export.dart
Normal file
@ -0,0 +1,366 @@
|
||||
import 'dart:convert';
|
||||
import 'dart:io';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/models/accuracy.dart';
|
||||
import 'package:diameter/models/basal.dart';
|
||||
import 'package:diameter/models/basal_profile.dart';
|
||||
import 'package:diameter/models/bolus.dart';
|
||||
import 'package:diameter/models/bolus_profile.dart';
|
||||
import 'package:diameter/models/glucose_target.dart';
|
||||
import 'package:diameter/models/log_bolus.dart';
|
||||
import 'package:diameter/models/log_entry.dart';
|
||||
import 'package:diameter/models/log_event.dart';
|
||||
import 'package:diameter/models/log_event_type.dart';
|
||||
import 'package:diameter/models/log_meal.dart';
|
||||
import 'package:diameter/models/meal.dart';
|
||||
import 'package:diameter/models/meal_category.dart';
|
||||
import 'package:diameter/models/meal_portion_type.dart';
|
||||
import 'package:diameter/models/meal_source.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
// import 'package:flutter/material.dart';
|
||||
import 'package:google_sign_in/google_sign_in.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
import 'package:path_provider/path_provider.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
import 'package:googleapis/drive/v3.dart' as drive;
|
||||
|
||||
class GoogleAuthClient extends http.BaseClient {
|
||||
final Map<String, String> _headers;
|
||||
|
||||
final http.Client _client = http.Client();
|
||||
|
||||
GoogleAuthClient(this._headers);
|
||||
|
||||
@override
|
||||
Future<http.StreamedResponse> send(http.BaseRequest request) {
|
||||
return _client.send(request..headers.addAll(_headers));
|
||||
}
|
||||
}
|
||||
|
||||
class DataExport {
|
||||
static Map<String, dynamic> appDataToJson(DateTime timestamp) {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
|
||||
data['exportDate'] = timestamp.toString();
|
||||
data['accuracies'] =
|
||||
Accuracy.box.getAll().map((accuracy) => accuracy.toJson()).toList();
|
||||
data['basalProfiles'] = BasalProfile.box
|
||||
.getAll()
|
||||
.map((basalProfile) => basalProfile.toJson())
|
||||
.toList();
|
||||
data['basalRates'] =
|
||||
Basal.box.getAll().map((basal) => basal.toJson()).toList();
|
||||
data['bolusProfiles'] = BolusProfile.box
|
||||
.getAll()
|
||||
.map((bolusProfile) => bolusProfile.toJson())
|
||||
.toList();
|
||||
data['bolusRates'] =
|
||||
Bolus.box.getAll().map((bolusRates) => bolusRates.toJson()).toList();
|
||||
data['glucoseTargets'] = GlucoseTarget.box
|
||||
.getAll()
|
||||
.map((glucoseTarget) => glucoseTarget.toJson())
|
||||
.toList();
|
||||
data['logBoli'] =
|
||||
LogBolus.box.getAll().map((logBolus) => logBolus.toJson()).toList();
|
||||
data['logEntries'] =
|
||||
LogEntry.box.getAll().map((logEntry) => logEntry.toJson()).toList();
|
||||
data['logEventTypes'] = LogEventType.box
|
||||
.getAll()
|
||||
.map((logEventType) => logEventType.toJson())
|
||||
.toList();
|
||||
data['logEvents'] =
|
||||
LogEvent.box.getAll().map((logEvent) => logEvent.toJson()).toList();
|
||||
data['logMeals'] =
|
||||
LogMeal.box.getAll().map((logMeal) => logMeal.toJson()).toList();
|
||||
data['mealCategories'] = MealCategory.box
|
||||
.getAll()
|
||||
.map((mealCategory) => mealCategory.toJson())
|
||||
.toList();
|
||||
data['mealPortionTypes'] = MealPortionType.box
|
||||
.getAll()
|
||||
.map((mealPortionType) => mealPortionType.toJson())
|
||||
.toList();
|
||||
data['mealSources'] = MealSource.box
|
||||
.getAll()
|
||||
.map((mealSource) => mealSource.toJson())
|
||||
.toList();
|
||||
data['meals'] = Meal.box.getAll().map((meal) => meal.toJson()).toList();
|
||||
data['settings'] = Settings.toJson();
|
||||
|
||||
return data;
|
||||
}
|
||||
|
||||
List<String> appDataFromJson(File file,
|
||||
{String source = '', bool overrideExisting = true, bool skipDeleted = false}) {
|
||||
final Map<String, dynamic> data = json.decode(file.openRead().toString());
|
||||
final List<String> errors = [];
|
||||
|
||||
final exportDate = DateTime.tryParse(data['exportDate']);
|
||||
if (exportDate != null && Settings.lastExportTimeStamp != null && exportDate.isAfter(Settings.lastExportTimeStamp!)) {
|
||||
if (data.keys.contains('accuracies')) {
|
||||
for (var entry in data['accuracies']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = Accuracy.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('basalProfiles')) {
|
||||
for (var entry in data['basalProfiles']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = BasalProfile.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('basalRates')) {
|
||||
for (var entry in data['basalRates']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = Basal.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('bolusRates')) {
|
||||
for (var entry in data['bolusRates']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = Bolus.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('glucoseTargets')) {
|
||||
for (var entry in data['glucoseTargets']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = GlucoseTarget.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('logBoli')) {
|
||||
for (var entry in data['logBoli']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = LogBolus.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('logEntries')) {
|
||||
for (var entry in data['logEntries']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = LogEntry.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('logEvents')) {
|
||||
for (var entry in data['logEvents']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = LogEvent.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('logMeals')) {
|
||||
for (var entry in data['logMeals']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = LogMeal.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('mealCategories')) {
|
||||
for (var entry in data['mealCategories']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = MealCategory.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('mealPortionTypes')) {
|
||||
for (var entry in data['mealPortionTypes']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = MealPortionType.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('mealSources')) {
|
||||
for (var entry in data['mealSources']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = MealSource.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.keys.contains('meals')) {
|
||||
for (var entry in data['meals']) {
|
||||
if (!skipDeleted || entry['deleted'] == false) {
|
||||
final error = Meal.putFromJson(entry, overrideExisting, source);
|
||||
if (error != null) {
|
||||
errors.add(translate(LocalizationKeys.export_error, args: {"data": entry}));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// show confirmation dialog because export is newer than last saved timestamp
|
||||
}
|
||||
|
||||
return errors;
|
||||
}
|
||||
|
||||
static Future<File> exportDataToFile(DateTime timestamp) async {
|
||||
final appDocDir = await getApplicationDocumentsDirectory();
|
||||
final appDocPath = appDocDir.path;
|
||||
|
||||
final DateFormat formatter = DateFormat(DateFormat.YEAR_MONTH_DAY);
|
||||
final date = DateTime.now();
|
||||
final file = File('$appDocPath/diameter_${formatter.format(date)}.json');
|
||||
|
||||
file.writeAsStringSync(json.encode(appDataToJson(timestamp)));
|
||||
return file;
|
||||
}
|
||||
|
||||
static Future<void> exportToGoogleDrive() async {
|
||||
final login = GoogleSignIn.standard(scopes: [drive.DriveApi.driveScope]);
|
||||
final GoogleSignInAccount? account = await login.signIn();
|
||||
|
||||
if (account != null) {
|
||||
final authenticateClient = GoogleAuthClient(await account.authHeaders);
|
||||
final driveApi = drive.DriveApi(authenticateClient);
|
||||
|
||||
final timestamp = DateTime.now();
|
||||
var localFile = await DataExport.exportDataToFile(timestamp);
|
||||
var media = drive.Media(localFile.openRead(), localFile.lengthSync());
|
||||
|
||||
final settings = Settings.get();
|
||||
settings.lastExportTimestamp = timestamp;
|
||||
Settings.put(settings);
|
||||
|
||||
drive.File driveFile = drive.File()..name = 'diameter.json';
|
||||
await driveApi.files.create(driveFile, uploadMedia: media);
|
||||
}
|
||||
}
|
||||
|
||||
static Future<void> importFromGoogleDrive() async {
|
||||
final login = GoogleSignIn.standard(scopes: [drive.DriveApi.driveScope]);
|
||||
final GoogleSignInAccount? account = await login.signIn();
|
||||
|
||||
if (account != null) {
|
||||
final authenticateClient = GoogleAuthClient(await account.authHeaders);
|
||||
final driveApi = drive.DriveApi(authenticateClient);
|
||||
|
||||
final timestamp = DateTime.now();
|
||||
var localFile = await DataExport.exportDataToFile(timestamp);
|
||||
var media = drive.Media(localFile.openRead(), localFile.lengthSync());
|
||||
|
||||
final settings = Settings.get();
|
||||
settings.lastExportTimestamp = timestamp;
|
||||
Settings.put(settings);
|
||||
|
||||
drive.File driveFile = drive.File()..name = 'diameter.json';
|
||||
await driveApi.files.create(driveFile, uploadMedia: media);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// class DataExportDialog extends StatefulWidget {
|
||||
// static const String routeName = '/data-export';
|
||||
// const DataExportDialog({Key? key}) : super(key: key);
|
||||
|
||||
// @override
|
||||
// _DataExportDialogState createState() => _DataExportDialogState();
|
||||
// }
|
||||
|
||||
// class _DataExportDialogState extends State<DataExportDialog> {
|
||||
// final ScrollController _scrollController = ScrollController();
|
||||
|
||||
// bool _isSaving = false;
|
||||
|
||||
// @override
|
||||
// void dispose() {
|
||||
// _scrollController.dispose();
|
||||
// super.dispose();
|
||||
// }
|
||||
|
||||
// @override
|
||||
// void initState() {
|
||||
// super.initState();
|
||||
// }
|
||||
|
||||
// void onExport(BuildContext context) async {
|
||||
// setState(() {
|
||||
// _isSaving = true;
|
||||
// });
|
||||
|
||||
// Navigator.pop(context);
|
||||
// setState(() {
|
||||
// _isSaving = false;
|
||||
// });
|
||||
// }
|
||||
|
||||
// @override
|
||||
// Widget build(BuildContext context) {
|
||||
// return AlertDialog(
|
||||
// content: Column(
|
||||
// mainAxisSize: MainAxisSize.min,
|
||||
// crossAxisAlignment: CrossAxisAlignment.center,
|
||||
// children: const [
|
||||
// // CheckboxListTile(
|
||||
// // value: showChart,
|
||||
// // onChanged: (_) => setState(() => showChart = !showChart),
|
||||
// // title: const Text('show Chart'),
|
||||
// // controlAffinity: ListTileControlAffinity.leading,
|
||||
// // ),
|
||||
// ],
|
||||
// ),
|
||||
// actions: <Widget>[
|
||||
// TextButton(
|
||||
// onPressed: () => Navigator.pop(context),
|
||||
// child: const Text('CANCEL'),
|
||||
// ),
|
||||
// ElevatedButton(
|
||||
// onPressed: !_isSaving ? () => onExport(context) : null,
|
||||
// child: const Text('EXPORT'),
|
||||
// ),
|
||||
// ]);
|
||||
// }
|
||||
// }
|
453
lib/localization_keys.dart
Normal file
453
lib/localization_keys.dart
Normal file
@ -0,0 +1,453 @@
|
||||
|
||||
// ignore_for_file: constant_identifier_names
|
||||
|
||||
class LocalizationKeys {
|
||||
static const String general = "general";
|
||||
static const String general_apply = "$general.apply";
|
||||
static const String general_cancel = "$general.cancel";
|
||||
static const String general_confirm = "$general.confirm";
|
||||
static const String general_discard = "$general.discard";
|
||||
static const String general_confirmDiscard = "$general.confirmDiscard";
|
||||
static const String general_delete = "$general.delete";
|
||||
static const String general_confirmDelete = "$general.confirmDelete";
|
||||
static const String general_save = "$general.save";
|
||||
static const String general_close = "$general.close";
|
||||
static const String general_saveAndClose = "$general.saveAndClose";
|
||||
static const String general_saveAsIs = "$general.saveAsIs";
|
||||
static const String general_keepEditing = "$general.keepEditing";
|
||||
static const String general_next = "$general.next";
|
||||
static const String general_edit = "$general.edit";
|
||||
static const String general_per = "$general.per";
|
||||
static const String general_example = "$general.example";
|
||||
static const String general_suffixes = "$general.suffixes";
|
||||
static const String general_suffixes_units = "$general_suffixes.units";
|
||||
static const String general_suffixes_carbs = "$general_suffixes.carbs";
|
||||
static const String general_suffixes_mins = "$general_suffixes.mins";
|
||||
static const String general_suffixes_carbsPerU = "$general_suffixes.carbsPerU";
|
||||
static const String general_suffixes_uPerBreadUnit = "$general_suffixes.uPerBreadUnit";
|
||||
static const String general_suffixes_uPerGlucose = "$general_suffixes.uPerGlucose";
|
||||
static const String general_suffixes_perDay = "$general_suffixes.perDay";
|
||||
|
||||
static const String navigation = "navigation";
|
||||
static const String navigation_log = "$navigation.log";
|
||||
static const String navigation_logEvents = "$navigation.logEvents";
|
||||
static const String navigation_reports = "$navigation.reports";
|
||||
static const String navigation_meals = "$navigation.meals";
|
||||
static const String navigation_basalProfiles = "$navigation.basalProfiles";
|
||||
static const String navigation_bolusProfiles = "$navigation.bolusProfiles";
|
||||
static const String navigation_categorization = "$navigation.categorization";
|
||||
static const String navigation_settings = "$navigation.settings";
|
||||
|
||||
static const String accuracy = "accuracy";
|
||||
static const String accuracy_empty = "$accuracy.empty";
|
||||
static const String accuracy_saved = "$accuracy.saved";
|
||||
static const String accuracy_saved_1 = "$accuracy_saved.1";
|
||||
static const String accuracy_saved_else = "$accuracy_saved.else";
|
||||
static const String accuracy_deleted = "$accuracy.deleted";
|
||||
static const String accuracy_confirmDelete = "$accuracy.confirmDelete";
|
||||
static const String accuracy_new = "$accuracy.new";
|
||||
static const String accuracy_fields = "$accuracy.fields";
|
||||
static const String accuracy_fields_name = "$accuracy_fields.name";
|
||||
static const String accuracy_fields_forPortionSize = "$accuracy_fields.forPortionSize";
|
||||
static const String accuracy_fields_forCarbsRatio = "$accuracy_fields.forCarbsRatio";
|
||||
static const String accuracy_fields_confidenceRating = "$accuracy_fields.confidenceRating";
|
||||
static const String accuracy_fields_notes = "$accuracy_fields.notes";
|
||||
static const String accuracy_fields_validators = "$accuracy_fields.validators";
|
||||
static const String accuracy_fields_validators_name = "$accuracy_fields_validators.name";
|
||||
static const String accuracy_title = "$accuracy.title";
|
||||
static const String accuracy_detail = "$accuracy.detail";
|
||||
static const String accuracy_detail_title = "$accuracy_detail.title";
|
||||
|
||||
static const String basal = "basal";
|
||||
static const String basal_empty = "$basal.empty";
|
||||
static const String basal_warnings = "$basal.warnings";
|
||||
static const String basal_warnings_duplicate = "$basal_warnings.duplicate";
|
||||
static const String basal_warnings_overlap = "$basal_warnings.overlap";
|
||||
static const String basal_warnings_startTimeFirst = "$basal_warnings.startTimeFirst";
|
||||
static const String basal_warnings_gap = "$basal_warnings.gap";
|
||||
static const String basal_warnings_endTimeLast = "$basal_warnings.endTimeLast";
|
||||
static const String basal_saved = "$basal.saved";
|
||||
static const String basal_saved_1 = "$basal_saved.1";
|
||||
static const String basal_saved_else = "$basal_saved.else";
|
||||
static const String basal_deleted = "$basal.deleted";
|
||||
static const String basal_confirmDelete = "$basal.confirmDelete";
|
||||
static const String basal_new = "$basal.new";
|
||||
static const String basal_fields = "$basal.fields";
|
||||
static const String basal_fields_startTime = "$basal_fields.startTime";
|
||||
static const String basal_fields_endTime = "$basal_fields.endTime";
|
||||
static const String basal_fields_units = "$basal_fields.units";
|
||||
static const String basal_title = "$basal.title";
|
||||
|
||||
static const String basalProfile = "basalProfile";
|
||||
static const String basalProfile_empty = "$basalProfile.empty";
|
||||
static const String basalProfile_warnings = "$basalProfile.warnings";
|
||||
static const String basalProfile_warnings_noActive = "$basalProfile_warnings.noActive";
|
||||
static const String basalProfile_warnings_noActiveOnCreate = "$basalProfile_warnings.noActiveOnCreate";
|
||||
static const String basalProfile_warnings_multipleActive = "$basalProfile_warnings.multipleActive";
|
||||
static const String basalProfile_warnings_activeAlreadySet = "$basalProfile_warnings.activeAlreadySet";
|
||||
static const String basalProfile_warnings_resolve = "$basalProfile_warnings.resolve";
|
||||
static const String basalProfile_warnings_resolve_activate = "$basalProfile_warnings_resolve.activate";
|
||||
static const String basalProfile_warnings_resolve_activateCurrent = "$basalProfile_warnings_resolve.activateCurrent";
|
||||
static const String basalProfile_warnings_resolve_create = "$basalProfile_warnings_resolve.create";
|
||||
static const String basalProfile_warnings_resolve_createInstead = "$basalProfile_warnings_resolve.createInstead";
|
||||
static const String basalProfile_warnings_resolve_pick = "$basalProfile_warnings_resolve.pick";
|
||||
static const String basalProfile_warnings_resolve_ignore = "$basalProfile_warnings_resolve.ignore";
|
||||
static const String basalProfile_warnings_resolve_deactivateProfile = "$basalProfile_warnings_resolve.deactivateProfile";
|
||||
static const String basalProfile_warnings_resolve_deactivateOthers = "$basalProfile_warnings_resolve.deactivateOthers";
|
||||
static const String basalProfile_saved = "$basalProfile.saved";
|
||||
static const String basalProfile_default = "$basalProfile.default";
|
||||
static const String basalProfile_active = "$basalProfile.active";
|
||||
static const String basalProfile_copied = "$basalProfile.copied";
|
||||
static const String basalProfile_copyOf = "$basalProfile.copyOf";
|
||||
static const String basalProfile_deleted = "$basalProfile.deleted";
|
||||
static const String basalProfile_confirmDelete = "$basalProfile.confirmDelete";
|
||||
static const String basalProfile_activated = "$basalProfile.activated";
|
||||
static const String basalProfile_new = "$basalProfile.new";
|
||||
static const String basalProfile_fields = "$basalProfile.fields";
|
||||
static const String basalProfile_fields_name = "$basalProfile_fields.name";
|
||||
static const String basalProfile_fields_notes = "$basalProfile_fields.notes";
|
||||
static const String basalProfile_fields_active = "$basalProfile_fields.active";
|
||||
static const String basalProfile_fields_validators = "$basalProfile_fields.validators";
|
||||
static const String basalProfile_fields_validators_name = "$basalProfile_fields_validators.name";
|
||||
static const String basalProfile_title = "$basalProfile.title";
|
||||
static const String basalProfile_detail = "$basalProfile.detail";
|
||||
static const String basalProfile_detail_title = "$basalProfile_detail.title";
|
||||
static const String basalProfile_detail_tabs = "$basalProfile_detail.tabs";
|
||||
static const String basalProfile_detail_tabs_profile = "$basalProfile_detail_tabs.profile";
|
||||
static const String basalProfile_detail_tabs_rates = "$basalProfile_detail_tabs.rates";
|
||||
|
||||
static const String bolus = "bolus";
|
||||
static const String bolus_empty = "$bolus.empty";
|
||||
static const String bolus_warnings = "$bolus.warnings";
|
||||
static const String bolus_warnings_duplicate = "$bolus_warnings.duplicate";
|
||||
static const String bolus_warnings_overlap = "$bolus_warnings.overlap";
|
||||
static const String bolus_warnings_startTimeFirst = "$bolus_warnings.startTimeFirst";
|
||||
static const String bolus_warnings_gap = "$bolus_warnings.gap";
|
||||
static const String bolus_warnings_endTimeLast = "$bolus_warnings.endTimeLast";
|
||||
static const String bolus_saved = "$bolus.saved";
|
||||
static const String bolus_saved_1 = "$bolus_saved.1";
|
||||
static const String bolus_saved_else = "$bolus_saved.else";
|
||||
static const String bolus_deleted = "$bolus.deleted";
|
||||
static const String bolus_confirmDelete = "$bolus.confirmDelete";
|
||||
static const String bolus_new = "$bolus.new";
|
||||
static const String bolus_fields = "$bolus.fields";
|
||||
static const String bolus_fields_startTime = "$bolus_fields.startTime";
|
||||
static const String bolus_fields_endTime = "$bolus_fields.endTime";
|
||||
static const String bolus_fields_units = "$bolus_fields.units";
|
||||
static const String bolus_fields_perCarbs = "$bolus_fields.perCarbs";
|
||||
static const String bolus_fields_perGlucose = "$bolus_fields.perGlucose";
|
||||
static const String bolus_title = "$bolus.title";
|
||||
|
||||
static const String bolusProfile = "bolusProfile";
|
||||
static const String bolusProfile_empty = "$bolusProfile.empty";
|
||||
static const String bolusProfile_warnings = "$bolusProfile.warnings";
|
||||
static const String bolusProfile_warnings_noActive = "$bolusProfile_warnings.noActive";
|
||||
static const String bolusProfile_warnings_noActiveOnCreate = "$bolusProfile_warnings.noActiveOnCreate";
|
||||
static const String bolusProfile_warnings_multipleActive = "$bolusProfile_warnings.multipleActive";
|
||||
static const String bolusProfile_warnings_activeAlreadySet = "$bolusProfile_warnings.activeAlreadySet";
|
||||
static const String bolusProfile_warnings_resolve = "$bolusProfile_warnings.resolve";
|
||||
static const String bolusProfile_warnings_resolve_activate = "$bolusProfile_warnings_resolve.activate";
|
||||
static const String bolusProfile_warnings_resolve_activateCurrent = "$bolusProfile_warnings_resolve.activateCurrent";
|
||||
static const String bolusProfile_warnings_resolve_create = "$bolusProfile_warnings_resolve.create";
|
||||
static const String bolusProfile_warnings_resolve_createInstead = "$bolusProfile_warnings_resolve.createInstead";
|
||||
static const String bolusProfile_warnings_resolve_pick = "$bolusProfile_warnings_resolve.pick";
|
||||
static const String bolusProfile_warnings_resolve_ignore = "$bolusProfile_warnings_resolve.ignore";
|
||||
static const String bolusProfile_warnings_resolve_deactivateProfile = "$bolusProfile_warnings_resolve.deactivateProfile";
|
||||
static const String bolusProfile_warnings_resolve_deactivateOthers = "$bolusProfile_warnings_resolve.deactivateOthers";
|
||||
static const String bolusProfile_saved = "$bolusProfile.saved";
|
||||
static const String bolusProfile_default = "$bolusProfile.default";
|
||||
static const String bolusProfile_active = "$bolusProfile.active";
|
||||
static const String bolusProfile_copied = "$bolusProfile.copied";
|
||||
static const String bolusProfile_copyOf = "$bolusProfile.copyOf";
|
||||
static const String bolusProfile_deleted = "$bolusProfile.deleted";
|
||||
static const String bolusProfile_confirmDelete = "$bolusProfile.confirmDelete";
|
||||
static const String bolusProfile_activated = "$bolusProfile.activated";
|
||||
static const String bolusProfile_new = "$bolusProfile.new";
|
||||
static const String bolusProfile_fields = "$bolusProfile.fields";
|
||||
static const String bolusProfile_fields_name = "$bolusProfile_fields.name";
|
||||
static const String bolusProfile_fields_notes = "$bolusProfile_fields.notes";
|
||||
static const String bolusProfile_fields_active = "$bolusProfile_fields.active";
|
||||
static const String bolusProfile_fields_validators = "$bolusProfile_fields.validators";
|
||||
static const String bolusProfile_fields_validators_name = "$bolusProfile_fields_validators.name";
|
||||
static const String bolusProfile_title = "$bolusProfile.title";
|
||||
static const String bolusProfile_detail = "$bolusProfile.detail";
|
||||
static const String bolusProfile_detail_title = "$bolusProfile_detail.title";
|
||||
static const String bolusProfile_detail_tabs = "$bolusProfile_detail.tabs";
|
||||
static const String bolusProfile_detail_tabs_profile = "$bolusProfile_detail_tabs.profile";
|
||||
static const String bolusProfile_detail_tabs_rates = "$bolusProfile_detail_tabs.rates";
|
||||
|
||||
static const String categories = "categories";
|
||||
|
||||
static const String log = "log";
|
||||
static const String log_title = "$log.title";
|
||||
static const String log_empty = "$log.empty";
|
||||
static const String log_confirmDelete = "$log.confirmDelete";
|
||||
static const String log_saved = "$log.saved";
|
||||
static const String log_deleted = "$log.deleted";
|
||||
static const String log_new = "$log.new";
|
||||
static const String log_fields = "$log.fields";
|
||||
static const String log_fields_date = "$log_fields.date";
|
||||
static const String log_fields_time = "$log_fields.time";
|
||||
static const String log_fields_glucose = "$log_fields.glucose";
|
||||
static const String log_fields_notes = "$log_fields.notes";
|
||||
static const String log_filter = "$log.filter";
|
||||
static const String log_filter_startDate = "$log_filter.startDate";
|
||||
static const String log_filter_endDate = "$log_filter.endDate";
|
||||
static const String log_filter_minGlucose = "$log_filter.minGlucose";
|
||||
static const String log_filter_maxGlucose = "$log_filter.maxGlucose";
|
||||
static const String log_filter_meal = "$log_filter.meal";
|
||||
static const String log_filter_mealNameContains = "$log_filter.mealNameContains";
|
||||
static const String log_filter_noteContains = "$log_filter.noteContains";
|
||||
static const String log_detail = "$log.detail";
|
||||
static const String log_detail_title = "$log_detail.title";
|
||||
static const String log_detail_tabs = "$log_detail.tabs";
|
||||
static const String log_detail_tabs_general = "$log_detail_tabs.general";
|
||||
static const String log_detail_tabs_meal = "$log_detail_tabs.meal";
|
||||
static const String log_detail_tabs_meal_title = "$log_detail_tabs_meal.title";
|
||||
static const String log_detail_tabs_meal_saved = "$log_detail_tabs_meal.saved";
|
||||
static const String log_detail_tabs_meal_new = "$log_detail_tabs_meal.new";
|
||||
static const String log_detail_tabs_meal_empty = "$log_detail_tabs_meal.empty";
|
||||
static const String log_detail_tabs_meal_confirmDelete = "$log_detail_tabs_meal.confirmDelete";
|
||||
static const String log_detail_tabs_meal_deleted = "$log_detail_tabs_meal.deleted";
|
||||
static const String log_detail_tabs_meal_detail = "$log_detail_tabs_meal.detail";
|
||||
static const String log_detail_tabs_meal_detail_title = "$log_detail_tabs_meal_detail.title";
|
||||
static const String log_detail_tabs_meal_detail_additionalFields = "$log_detail_tabs_meal_detail.additionalFields";
|
||||
static const String log_detail_tabs_meal_detail_fields = "$log_detail_tabs_meal_detail.fields";
|
||||
static const String log_detail_tabs_meal_detail_fields_meal = "$log_detail_tabs_meal_detail_fields.meal";
|
||||
static const String log_detail_tabs_meal_detail_fields_name = "$log_detail_tabs_meal_detail_fields.name";
|
||||
static const String log_detail_tabs_meal_detail_fields_amount = "$log_detail_tabs_meal_detail_fields.amount";
|
||||
static const String log_detail_tabs_meal_detail_fields_portionSize = "$log_detail_tabs_meal_detail_fields.portionSize";
|
||||
static const String log_detail_tabs_meal_detail_fields_carbsRatio = "$log_detail_tabs_meal_detail_fields.carbsRatio";
|
||||
static const String log_detail_tabs_meal_detail_fields_totalCarbs = "$log_detail_tabs_meal_detail_fields.totalCarbs";
|
||||
static const String log_detail_tabs_meal_detail_fields_setManually = "$log_detail_tabs_meal_detail_fields.setManually";
|
||||
static const String log_detail_tabs_meal_detail_fields_notes = "$log_detail_tabs_meal_detail_fields.notes";
|
||||
static const String log_detail_tabs_meal_detail_fields_mealSource = "$log_detail_tabs_meal_detail_fields.mealSource";
|
||||
static const String log_detail_tabs_meal_detail_fields_mealCategory = "$log_detail_tabs_meal_detail_fields.mealCategory";
|
||||
static const String log_detail_tabs_meal_detail_fields_mealPortionType = "$log_detail_tabs_meal_detail_fields.mealPortionType";
|
||||
static const String log_detail_tabs_meal_detail_fields_portionSizeAccuracy = "$log_detail_tabs_meal_detail_fields.portionSizeAccuracy";
|
||||
static const String log_detail_tabs_meal_detail_fields_carbsRatioAccuracy = "$log_detail_tabs_meal_detail_fields.carbsRatioAccuracy";
|
||||
static const String log_detail_tabs_meal_detail_fields_validators = "$log_detail_tabs_meal_detail_fields.validators";
|
||||
static const String log_detail_tabs_meal_detail_fields_validators_name = "$log_detail_tabs_meal_detail_fields_validators.name";
|
||||
static const String log_detail_tabs_bolus = "$log_detail_tabs.bolus";
|
||||
static const String log_detail_tabs_bolus_title = "$log_detail_tabs_bolus.title";
|
||||
static const String log_detail_tabs_bolus_saved = "$log_detail_tabs_bolus.saved";
|
||||
static const String log_detail_tabs_bolus_new = "$log_detail_tabs_bolus.new";
|
||||
static const String log_detail_tabs_bolus_deleted = "$log_detail_tabs_bolus.deleted";
|
||||
static const String log_detail_tabs_bolus_confirmDelete = "$log_detail_tabs_bolus.confirmDelete";
|
||||
static const String log_detail_tabs_bolus_delayedBy = "$log_detail_tabs_bolus.delayedBy";
|
||||
static const String log_detail_tabs_bolus_forMeal = "$log_detail_tabs_bolus.forMeal";
|
||||
static const String log_detail_tabs_bolus_toCorrect = "$log_detail_tabs_bolus.toCorrect";
|
||||
static const String log_detail_tabs_bolus_empty = "$log_detail_tabs_bolus.empty";
|
||||
static const String log_detail_tabs_bolus_detail = "$log_detail_tabs_bolus.detail";
|
||||
static const String log_detail_tabs_bolus_detail_title = "$log_detail_tabs_bolus_detail.title";
|
||||
static const String log_detail_tabs_bolus_detail_fields = "$log_detail_tabs_bolus_detail.fields";
|
||||
static const String log_detail_tabs_bolus_detail_fields_units = "$log_detail_tabs_bolus_detail_fields.units";
|
||||
static const String log_detail_tabs_bolus_detail_fields_setManually = "$log_detail_tabs_bolus_detail_fields.setManually";
|
||||
static const String log_detail_tabs_bolus_detail_fields_forGlucose = "$log_detail_tabs_bolus_detail_fields.forGlucose";
|
||||
static const String log_detail_tabs_bolus_detail_fields_forMeal = "$log_detail_tabs_bolus_detail_fields.forMeal";
|
||||
static const String log_detail_tabs_bolus_detail_fields_current = "$log_detail_tabs_bolus_detail_fields.current";
|
||||
static const String log_detail_tabs_bolus_detail_fields_target = "$log_detail_tabs_bolus_detail_fields.target";
|
||||
static const String log_detail_tabs_bolus_detail_fields_correction = "$log_detail_tabs_bolus_detail_fields.correction";
|
||||
static const String log_detail_tabs_bolus_detail_fields_meal = "$log_detail_tabs_bolus_detail_fields.meal";
|
||||
static const String log_detail_tabs_bolus_detail_fields_carbs = "$log_detail_tabs_bolus_detail_fields.carbs";
|
||||
static const String log_detail_tabs_bolus_detail_fields_delayedBolusDuration = "$log_detail_tabs_bolus_detail_fields.delayedBolusDuration";
|
||||
static const String log_detail_tabs_bolus_detail_fields_immediateBolus = "$log_detail_tabs_bolus_detail_fields.immediateBolus";
|
||||
static const String log_detail_tabs_bolus_detail_fields_delayedBolus = "$log_detail_tabs_bolus_detail_fields.delayedBolus";
|
||||
|
||||
static const String meal = "meal";
|
||||
static const String meal_title = "$meal.title";
|
||||
static const String meal_empty = "$meal.empty";
|
||||
static const String meal_confirmDelete = "$meal.confirmDelete";
|
||||
static const String meal_saved = "$meal.saved";
|
||||
static const String meal_deleted = "$meal.deleted";
|
||||
static const String meal_new = "$meal.new";
|
||||
static const String meal_fields = "$meal.fields";
|
||||
static const String meal_fields_name = "$meal_fields.name";
|
||||
static const String meal_fields_mealSource = "$meal_fields.mealSource";
|
||||
static const String meal_fields_mealPortionType = "$meal_fields.mealPortionType";
|
||||
static const String meal_fields_notes = "$meal_fields.notes";
|
||||
static const String meal_fields_carbsRatio = "$meal_fields.carbsRatio";
|
||||
static const String meal_fields_setManually = "$meal_fields.setManually";
|
||||
static const String meal_fields_portionSize = "$meal_fields.portionSize";
|
||||
static const String meal_fields_carbsPerPortion = "$meal_fields.carbsPerPortion";
|
||||
static const String meal_fields_delay = "$meal_fields.delay";
|
||||
static const String meal_fields_delay_title = "$meal_fields_delay.title";
|
||||
static const String meal_fields_delay_duration = "$meal_fields_delay.duration";
|
||||
static const String meal_fields_additional = "$meal_fields.additional";
|
||||
static const String meal_fields_additional_title = "$meal_fields_additional.title";
|
||||
static const String meal_fields_additional_mealCategory = "$meal_fields_additional.mealCategory";
|
||||
static const String meal_fields_additional_portionSizeAccuracy = "$meal_fields_additional.portionSizeAccuracy";
|
||||
static const String meal_fields_additional_carbsRatioAccuracy = "$meal_fields_additional.carbsRatioAccuracy";
|
||||
static const String meal_fields_validators = "$meal_fields.validators";
|
||||
static const String meal_fields_validators_name = "$meal_fields_validators.name";
|
||||
static const String meal_detail = "$meal.detail";
|
||||
static const String meal_detail_title = "$meal_detail.title";
|
||||
|
||||
static const String mealSource = "mealSource";
|
||||
static const String mealSource_title = "$mealSource.title";
|
||||
static const String mealSource_empty = "$mealSource.empty";
|
||||
static const String mealSource_confirmDelete = "$mealSource.confirmDelete";
|
||||
static const String mealSource_saved = "$mealSource.saved";
|
||||
static const String mealSource_deleted = "$mealSource.deleted";
|
||||
static const String mealSource_new = "$mealSource.new";
|
||||
static const String mealSource_fields = "$mealSource.fields";
|
||||
static const String mealSource_fields_name = "$mealSource_fields.name";
|
||||
static const String mealSource_fields_notes = "$mealSource_fields.notes";
|
||||
static const String mealSource_fields_defaultCarbsRatioAccuracy = "$mealSource_fields.defaultCarbsRatioAccuracy";
|
||||
static const String mealSource_fields_defaultPortionSizeAccuracy = "$mealSource_fields.defaultPortionSizeAccuracy";
|
||||
static const String mealSource_fields_defaultMealCategory = "$mealSource_fields.defaultMealCategory";
|
||||
static const String mealSource_fields_defaultMealPortionType = "$mealSource_fields.defaultMealPortionType";
|
||||
static const String mealSource_fields_validators = "$mealSource_fields.validators";
|
||||
static const String mealSource_fields_validators_name = "$mealSource_fields_validators.name";
|
||||
static const String mealSource_detail = "$mealSource.detail";
|
||||
static const String mealSource_detail_title = "$mealSource_detail.title";
|
||||
|
||||
static const String mealCategory = "mealCategory";
|
||||
static const String mealCategory_title = "$mealCategory.title";
|
||||
static const String mealCategory_empty = "$mealCategory.empty";
|
||||
static const String mealCategory_confirmDelete = "$mealCategory.confirmDelete";
|
||||
static const String mealCategory_saved = "$mealCategory.saved";
|
||||
static const String mealCategory_deleted = "$mealCategory.deleted";
|
||||
static const String mealCategory_new = "$mealCategory.new";
|
||||
static const String mealCategory_fields = "$mealCategory.fields";
|
||||
static const String mealCategory_fields_name = "$mealCategory_fields.name";
|
||||
static const String mealCategory_fields_notes = "$mealCategory_fields.notes";
|
||||
static const String mealCategory_fields_validators = "$mealCategory_fields.validators";
|
||||
static const String mealCategory_fields_validators_name = "$mealCategory_fields_validators.name";
|
||||
static const String mealCategory_detail = "$mealCategory.detail";
|
||||
static const String mealCategory_detail_title = "$mealCategory_detail.title";
|
||||
|
||||
static const String portionType = "portionType";
|
||||
static const String portionType_title = "$portionType.title";
|
||||
static const String portionType_empty = "$portionType.empty";
|
||||
static const String portionType_confirmDelete = "$portionType.confirmDelete";
|
||||
static const String portionType_saved = "$portionType.saved";
|
||||
static const String portionType_deleted = "$portionType.deleted";
|
||||
static const String portionType_new = "$portionType.new";
|
||||
static const String portionType_fields = "$portionType.fields";
|
||||
static const String portionType_fields_name = "$portionType_fields.name";
|
||||
static const String portionType_fields_notes = "$portionType_fields.notes";
|
||||
static const String portionType_fields_validators = "$portionType_fields.validators";
|
||||
static const String portionType_fields_validators_name = "$portionType_fields_validators.name";
|
||||
static const String portionType_detail = "$portionType.detail";
|
||||
static const String portionType_detail_title = "$portionType_detail.title";
|
||||
|
||||
static const String eventType = "eventType";
|
||||
static const String eventType_title = "$eventType.title";
|
||||
static const String eventType_empty = "$eventType.empty";
|
||||
static const String eventType_saved = "$eventType.saved";
|
||||
static const String eventType_deleted = "$eventType.deleted";
|
||||
static const String eventType_new = "$eventType.new";
|
||||
static const String eventType_fields = "$eventType.fields";
|
||||
static const String eventType_fields_name = "$eventType_fields.name";
|
||||
static const String eventType_fields_notes = "$eventType_fields.notes";
|
||||
static const String eventType_fields_hasEndTime = "$eventType_fields.hasEndTime";
|
||||
static const String eventType_fields_defaultReminderDuration = "$eventType_fields.defaultReminderDuration";
|
||||
static const String eventType_fields_bolusProfile = "$eventType_fields.bolusProfile";
|
||||
static const String eventType_fields_basalProfile = "$eventType_fields.basalProfile";
|
||||
static const String eventType_fields_validators = "$eventType_fields.validators";
|
||||
static const String eventType_fields_validators_name = "$eventType_fields_validators.name";
|
||||
static const String eventType_detail = "$eventType.detail";
|
||||
static const String eventType_detail_title = "$eventType_detail.title";
|
||||
|
||||
static const String event = "event";
|
||||
static const String event_warnings = "$event.warnings";
|
||||
static const String event_warnings_duplicate = "$event_warnings.duplicate";
|
||||
static const String event_title = "$event.title";
|
||||
static const String event_titleActive = "$event.titleAcive";
|
||||
static const String event_empty = "$event.empty";
|
||||
static const String event_emptyActive = "$event.emptyActive";
|
||||
static const String event_saved = "$event.saved";
|
||||
static const String event_deleted = "$event.deleted";
|
||||
static const String event_confirmDelete = "$event.confirmDelete";
|
||||
static const String event_end = "$event.end";
|
||||
static const String event_ended = "$event.ended";
|
||||
static const String event_confirmEnd = "$event.confirmEnd";
|
||||
static const String event_new = "$event.new";
|
||||
static const String event_fields = "$event.fields";
|
||||
static const String event_fields_eventType = "$event_fields.eventType";
|
||||
static const String event_fields_startDate = "$event_fields.startDate";
|
||||
static const String event_fields_endDate = "$event_fields.endDate";
|
||||
static const String event_fields_date = "$event_fields.date";
|
||||
static const String event_fields_startTime = "$event_fields.startTime";
|
||||
static const String event_fields_endTime = "$event_fields.endTime";
|
||||
static const String event_fields_time = "$event_fields.time";
|
||||
static const String event_fields_notes = "$event_fields.notes";
|
||||
static const String event_fields_hasEndTime = "$event_fields.hasEndTime";
|
||||
static const String event_fields_reminderDuration = "$event_fields.reminderDuration";
|
||||
static const String event_fields_bolusProfile = "$event_fields.bolusProfile";
|
||||
static const String event_fields_basalProfile = "$event_fields.basalProfile";
|
||||
static const String event_detail = "$event.detail";
|
||||
static const String event_detail_title = "$event_detail.title";
|
||||
|
||||
static const String settings = "settings";
|
||||
static const String settings_title = "$settings.title";
|
||||
static const String settings_reset = "$settings.reset";
|
||||
static const String settings_resetAll = "$settings.resetAll";
|
||||
static const String settings_confirmReset = "$settings.confirmReset";
|
||||
static const String settings_updated = "$settings.updated";
|
||||
static const String settings_sections = "$settings.sections";
|
||||
static const String settings_sections_measurements = "$settings_sections.measurements";
|
||||
static const String settings_sections_confirmation = "$settings_sections.confirmation";
|
||||
static const String settings_sections_dateTimeFormat = "$settings_sections.dateTimeFormat";
|
||||
static const String settings_fields = "$settings.fields";
|
||||
static const String settings_fields_nutritionMeasurement = "$settings_fields.nutritionMeasurement";
|
||||
static const String settings_fields_glucoseMeasurement = "$settings_fields.glucoseMeasurement";
|
||||
static const String settings_fields_targetGlucose = "$settings_fields.targetGlucose";
|
||||
static const String settings_fields_insulinIncrement = "$settings_fields.insulinIncrement";
|
||||
static const String settings_fields_nutritionIncrement = "$settings_fields.nutritionIncrement";
|
||||
static const String settings_fields_mmolLIncrement = "$settings_fields.mmolLIncrement";
|
||||
static const String settings_fields_onlyDisplayActive = "$settings_fields.onlyDisplayActive";
|
||||
static const String settings_fields_displayBothDetail = "$settings_fields.displayBothDetail";
|
||||
static const String settings_fields_displayBothList = "$settings_fields.displayBothList";
|
||||
static const String settings_fields_confirmOnCancel = "$settings_fields.confirmOnCancel";
|
||||
static const String settings_fields_confirmOnDelete = "$settings_fields.confirmOnDelete";
|
||||
static const String settings_fields_confirmOnEndEvent = "$settings_fields.confirmOnEndEvent";
|
||||
static const String settings_fields_dateFormat = "$settings_fields.dateFormat";
|
||||
static const String settings_fields_longDateFormat = "$settings_fields.longDateFormat";
|
||||
static const String settings_fields_timeFormat = "$settings_fields.timeFormat";
|
||||
static const String settings_fields_longTimeFormat = "$settings_fields.longTimeFormat";
|
||||
|
||||
static const String reports = "reports";
|
||||
static const String reports_title = "$reports.title";
|
||||
static const String reports_ids = "$reports.ids";
|
||||
static const String reports_ids_glucose = "$reports_ids.glucose";
|
||||
static const String reports_ids_carbs = "$reports_ids.carbs";
|
||||
static const String reports_ids_basal = "$reports_ids.basal";
|
||||
static const String reports_ids_bolus = "$reports_ids.bolus";
|
||||
static const String reports_dailyCharts = "$reports.dailyCharts";
|
||||
static const String reports_dailyCharts_showChart = "$reports_dailyCharts.showChart";
|
||||
static const String reports_dailyCharts_showBolus = "$reports_dailyCharts.showBolus";
|
||||
static const String reports_dailyCharts_showBasal = "$reports_dailyCharts.showBasal";
|
||||
static const String reports_dailyCharts_showMeals = "$reports_dailyCharts.showMeals";
|
||||
static const String reports_dailyCharts_empty = "$reports_dailyCharts.empty";
|
||||
static const String reports_sections = "$reports.sections";
|
||||
static const String reports_sections_dailyReport = "$reports_sections.dailyReport";
|
||||
static const String reports_sections_pdfReport = "$reports_sections.pdfReport";
|
||||
static const String reports_export = "$reports.export";
|
||||
static const String reports_export_title = "$reports_export.title";
|
||||
static const String reports_export_singleDate = "$reports_export.singleDate";
|
||||
static const String reports_export_range = "$reports_export.range";
|
||||
static const String reports_export_date = "$reports_export.date";
|
||||
static const String reports_export_endDate = "$reports_export.endDate";
|
||||
static const String reports_export_showChart = "$reports_export.showChart";
|
||||
static const String reports_export_showBolus = "$reports_export.showBolus";
|
||||
static const String reports_export_showBasal = "$reports_export.showBasal";
|
||||
static const String reports_export_showMeals = "$reports_export.showMeals";
|
||||
static const String reports_export_showTable = "$reports_export.showTable";
|
||||
static const String reports_export_export = "$reports_export.export";
|
||||
static const String reports_export_tableHeaders = "$reports_export.tableHeaders";
|
||||
static const String reports_export_tableHeaders_time = "$reports_export_tableHeaders.time";
|
||||
static const String reports_export_tableHeaders_glucose = "$reports_export_tableHeaders.glucose";
|
||||
static const String reports_export_tableHeaders_bolus = "$reports_export_tableHeaders.bolus";
|
||||
static const String reports_export_tableHeaders_notes = "$reports_export_tableHeaders.notes";
|
||||
static const String reports_export_tableHeaders_meals = "$reports_export_tableHeaders.meals";
|
||||
static const String reports_export_tableHeaders_portionSize = "$reports_export_tableHeaders.portionSize";
|
||||
static const String reports_export_tableHeaders_carbs = "$reports_export_tableHeaders.carbs";
|
||||
static const String reports_export_tableHeaders_mealBolus = "$reports_export_tableHeaders.mealBolus";
|
||||
static const String reports_export_tableHeaders_mealNotes = "$reports_export_tableHeaders.mealNotes";
|
||||
|
||||
static const String export = "export";
|
||||
static const String export_error = "$export.error";
|
||||
|
||||
}
|
@ -25,30 +25,62 @@ import 'package:diameter/screens/meal/meal_list.dart';
|
||||
import 'package:diameter/screens/reports/export.dart';
|
||||
import 'package:diameter/screens/reports/reports.dart';
|
||||
import 'package:diameter/settings.dart';
|
||||
import 'package:diameter/data_export.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/screens/basal/basal_profile_list.dart';
|
||||
import 'package:diameter/screens/bolus/bolus_profile_list.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:objectbox/objectbox.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
late ObjectBox objectBox;
|
||||
|
||||
Future<void> main() async {
|
||||
WidgetsFlutterBinding.ensureInitialized();
|
||||
|
||||
DataExport.exportToGoogleDrive();
|
||||
|
||||
objectBox = await ObjectBox.create();
|
||||
|
||||
Sync.isAvailable();
|
||||
SyncClient syncClient = Sync.client(
|
||||
objectBox.store,
|
||||
'ws://192.168.1.184:9999',
|
||||
SyncCredentials.sharedSecretString(secret)
|
||||
);
|
||||
SyncClient syncClient = Sync.client(objectBox.store,
|
||||
'ws://192.168.1.184:9999', SyncCredentials.sharedSecretString(secret));
|
||||
syncClient.start();
|
||||
syncClient.requestUpdates(subscribeForFuturePushes: false);
|
||||
|
||||
DataExport.exportToGoogleDrive();
|
||||
|
||||
var delegate = await LocalizationDelegate.create(
|
||||
fallbackLocale: 'en_US',
|
||||
supportedLocales: ['en_US', 'de'],
|
||||
);
|
||||
|
||||
runApp(
|
||||
GestureDetector(
|
||||
LocalizedApp(delegate, const App()),
|
||||
);
|
||||
}
|
||||
|
||||
class App extends StatelessWidget {
|
||||
const App({Key? key}) : super(key: key);
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
var localizationDelegate = LocalizedApp.of(context).delegate;
|
||||
|
||||
return LocalizationProvider(
|
||||
state: LocalizationProvider.of(context).state,
|
||||
child: GestureDetector(
|
||||
onTap: () => FocusManager.instance.primaryFocus?.unfocus(),
|
||||
child: MaterialApp(
|
||||
localizationsDelegates: [
|
||||
localizationDelegate,
|
||||
],
|
||||
supportedLocales: const [
|
||||
Locale('en', 'US'),
|
||||
Locale('de', 'DE'),
|
||||
],
|
||||
// localizationDelegate.supportedLocales,
|
||||
locale: localizationDelegate.currentLocale,
|
||||
theme: AppTheme.makeTheme(AppTheme.lightTheme),
|
||||
darkTheme: AppTheme.makeTheme(AppTheme.darkTheme),
|
||||
themeMode: Settings.themeMode,
|
||||
@ -87,3 +119,4 @@ Future<void> main() async {
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
@ -10,11 +10,13 @@ class Accuracy {
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String value;
|
||||
bool forCarbsRatio;
|
||||
bool forPortionSize;
|
||||
int? confidenceRating;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
Accuracy({
|
||||
@ -25,6 +27,7 @@ class Accuracy {
|
||||
this.forPortionSize = false,
|
||||
this.confidenceRating,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -60,10 +63,14 @@ class Accuracy {
|
||||
}
|
||||
|
||||
static void reorder(Accuracy accuracy, int? newPosition) {
|
||||
QueryBuilder<Accuracy> all = box.query(Accuracy_.deleted.equals(false).and(Accuracy_.id.notEquals(accuracy.id)))
|
||||
QueryBuilder<Accuracy> all = box.query(Accuracy_.deleted
|
||||
.equals(false)
|
||||
.and(Accuracy_.id.notEquals(accuracy.id)))
|
||||
..order(Accuracy_.confidenceRating);
|
||||
List<Accuracy> accuracies = all.build().find();
|
||||
newPosition == null || newPosition >= accuracies.length ? accuracies.add(accuracy) : accuracies.insert(newPosition, accuracy);
|
||||
newPosition == null || newPosition >= accuracies.length
|
||||
? accuracies.add(accuracy)
|
||||
: accuracies.insert(newPosition, accuracy);
|
||||
box.putMany(accuracies.map((item) {
|
||||
item.confidenceRating = accuracies.indexOf(item);
|
||||
return item;
|
||||
@ -74,4 +81,31 @@ class Accuracy {
|
||||
String toString() {
|
||||
return value;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['value'] = value;
|
||||
data['forCarbsRatio'] = forCarbsRatio;
|
||||
data['forPortionSize'] = forPortionSize;
|
||||
data['confidenceRating'] = confidenceRating;
|
||||
data['notes'] = notes;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final accuracy = Accuracy(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
value: json['value'],
|
||||
forCarbsRatio: json['forCarbsRatio'],
|
||||
forPortionSize: json['forPortionSize'],
|
||||
confidenceRating: json['confidenceRating'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
Accuracy.put(accuracy);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -18,6 +18,7 @@ class Basal {
|
||||
@Property(type: PropertyType.date)
|
||||
DateTime endTime;
|
||||
double units;
|
||||
String? source;
|
||||
|
||||
// relations
|
||||
final basalProfile = ToOne<BasalProfile>();
|
||||
@ -29,6 +30,7 @@ class Basal {
|
||||
required this.startTime,
|
||||
required this.endTime,
|
||||
this.units = 0,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -43,6 +45,13 @@ class Basal {
|
||||
}
|
||||
}
|
||||
|
||||
static void removeAllForProfile(int id) {
|
||||
box.putMany(getAllForProfile(id).map((item) {
|
||||
item.deleted = true;
|
||||
return item;
|
||||
}).toList());
|
||||
}
|
||||
|
||||
static List<Basal> getAllForProfile(int id) {
|
||||
QueryBuilder<Basal> builder = box.query(Basal_.deleted.equals(false))
|
||||
..order(Basal_.startTime);
|
||||
@ -94,4 +103,42 @@ class Basal {
|
||||
String toString() {
|
||||
return DateTimeUtils.displayTime(startTime);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['startTime'] = startTime.toIso8601String();
|
||||
data['endTime'] = endTime.toIso8601String();
|
||||
data['units'] = units;
|
||||
data['basalProfile'] = basalProfile.targetId;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
DateTime? startTime = DateTime.tryParse(json['startTime']);
|
||||
DateTime? endTime = DateTime.tryParse(json['endTime']);
|
||||
|
||||
if (startTime == null || endTime == null) {
|
||||
return startTime == null
|
||||
? endTime == null
|
||||
? 'start and end time are missing'
|
||||
: 'start time is missing'
|
||||
: 'end time is missing';
|
||||
}
|
||||
|
||||
final basal = Basal(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
startTime: startTime,
|
||||
endTime: endTime,
|
||||
units: json['units'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
basal.basalProfile.targetId = json['basalProfile'];
|
||||
Basal.put(basal);
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -1,4 +1,5 @@
|
||||
import 'package:diameter/main.dart';
|
||||
import 'package:diameter/models/basal.dart';
|
||||
import 'package:diameter/models/log_event.dart';
|
||||
import 'package:objectbox/objectbox.dart';
|
||||
import 'package:diameter/objectbox.g.dart' show BasalProfile_;
|
||||
@ -11,9 +12,11 @@ class BasalProfile {
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String name;
|
||||
bool active;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
BasalProfile({
|
||||
@ -22,6 +25,7 @@ class BasalProfile {
|
||||
this.name = '',
|
||||
this.active = false,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -29,7 +33,8 @@ class BasalProfile {
|
||||
static void put(BasalProfile basalProfile) => box.put(basalProfile);
|
||||
|
||||
static List<BasalProfile> getAll() {
|
||||
QueryBuilder<BasalProfile> all = box.query(BasalProfile_.deleted.equals(false))
|
||||
QueryBuilder<BasalProfile> all = box
|
||||
.query(BasalProfile_.deleted.equals(false))
|
||||
..order(BasalProfile_.name);
|
||||
return all.build().find();
|
||||
}
|
||||
@ -38,13 +43,16 @@ class BasalProfile {
|
||||
final item = box.get(id);
|
||||
if (item != null) {
|
||||
item.deleted = true;
|
||||
Basal.removeAllForProfile(id);
|
||||
box.put(item);
|
||||
}
|
||||
}
|
||||
|
||||
static int activeCount() {
|
||||
Query<BasalProfile> query = box
|
||||
.query(BasalProfile_.active.equals(true) & BasalProfile_.deleted.equals(false)).build();
|
||||
.query(BasalProfile_.active.equals(true) &
|
||||
BasalProfile_.deleted.equals(false))
|
||||
.build();
|
||||
return query.find().length;
|
||||
}
|
||||
|
||||
@ -58,13 +66,15 @@ class BasalProfile {
|
||||
static BasalProfile? getActive(DateTime? dateTime) {
|
||||
if (dateTime != null) {
|
||||
List<LogEvent> activeEvents = LogEvent.getAllActiveForTime(dateTime)
|
||||
.where((event) => event.basalProfile.target != null).toList();
|
||||
.where((event) => event.basalProfile.target != null)
|
||||
.toList();
|
||||
if (activeEvents.length > 1) {
|
||||
final now = DateTime.now();
|
||||
activeEvents =
|
||||
activeEvents.where((item) => !activeEvents.any((other) =>
|
||||
item.time.isBefore(other.time) || (item.endTime ?? now).isAfter(other.endTime ?? now)
|
||||
)).toList();
|
||||
activeEvents = activeEvents
|
||||
.where((item) => !activeEvents.any((other) =>
|
||||
item.time.isBefore(other.time) ||
|
||||
(item.endTime ?? now).isAfter(other.endTime ?? now)))
|
||||
.toList();
|
||||
}
|
||||
if (activeEvents.length == 1) {
|
||||
return activeEvents.single.basalProfile.target;
|
||||
@ -84,4 +94,28 @@ class BasalProfile {
|
||||
String toString() {
|
||||
return name;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['name'] = name;
|
||||
data['active'] = active;
|
||||
data['notes'] = notes;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final basalProfile = BasalProfile(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'] == 'true',
|
||||
name: json['name'],
|
||||
active: json['active'] == 'true',
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
BasalProfile.put(basalProfile);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -21,6 +21,7 @@ class Bolus {
|
||||
double carbs;
|
||||
int? mgPerDl;
|
||||
double? mmolPerL;
|
||||
String? source;
|
||||
|
||||
// relations
|
||||
final bolusProfile = ToOne<BolusProfile>();
|
||||
@ -35,6 +36,7 @@ class Bolus {
|
||||
this.carbs = 0,
|
||||
this.mgPerDl,
|
||||
this.mmolPerL,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -56,6 +58,13 @@ class Bolus {
|
||||
}
|
||||
}
|
||||
|
||||
static void removeAllForProfile(int id) {
|
||||
box.putMany(getAllForProfile(id).map((item) {
|
||||
item.deleted = true;
|
||||
return item;
|
||||
}).toList());
|
||||
}
|
||||
|
||||
static Bolus? getRateForTime(DateTime? dateTime) {
|
||||
if (dateTime != null) {
|
||||
final bolusProfile = BolusProfile.getActive(dateTime);
|
||||
@ -81,4 +90,48 @@ class Bolus {
|
||||
String toString() {
|
||||
return DateTimeUtils.displayTime(startTime);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['startTime'] = startTime.toIso8601String();
|
||||
data['endTime'] = endTime.toIso8601String();
|
||||
data['units'] = units;
|
||||
data['carbs'] = carbs;
|
||||
data['mgPerDl'] = mgPerDl;
|
||||
data['mmolPerL'] = mmolPerL;
|
||||
data['bolusProfile'] = bolusProfile.targetId;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
DateTime? startTime = DateTime.tryParse(json['startTime']);
|
||||
DateTime? endTime = DateTime.tryParse(json['endTime']);
|
||||
|
||||
if (startTime == null || endTime == null) {
|
||||
return startTime == null
|
||||
? endTime == null
|
||||
? 'start and end time are missing'
|
||||
: 'start time is missing'
|
||||
: 'end time is missing';
|
||||
}
|
||||
|
||||
final bolus = Bolus(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
startTime: startTime,
|
||||
endTime: endTime,
|
||||
units: json['units'],
|
||||
carbs: json['carbs'],
|
||||
mgPerDl: json['mgPerDl'],
|
||||
mmolPerL: json['mmolPerL'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
bolus.bolusProfile.targetId = json['bolusProfile'];
|
||||
Bolus.put(bolus);
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -1,4 +1,5 @@
|
||||
import 'package:diameter/main.dart';
|
||||
import 'package:diameter/models/bolus.dart';
|
||||
import 'package:diameter/models/log_event.dart';
|
||||
import 'package:objectbox/objectbox.dart';
|
||||
import 'package:diameter/objectbox.g.dart' show BolusProfile_;
|
||||
@ -11,9 +12,11 @@ class BolusProfile {
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String name;
|
||||
bool active;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
BolusProfile({
|
||||
@ -22,6 +25,7 @@ class BolusProfile {
|
||||
this.name = '',
|
||||
this.active = false,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -38,6 +42,7 @@ class BolusProfile {
|
||||
final item = box.get(id);
|
||||
if (item != null) {
|
||||
item.deleted = true;
|
||||
Bolus.removeAllForProfile(id);
|
||||
box.put(item);
|
||||
}
|
||||
}
|
||||
@ -87,4 +92,29 @@ class BolusProfile {
|
||||
String toString() {
|
||||
return name;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['name'] = name;
|
||||
data['active'] = active;
|
||||
data['notes'] = notes;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final bolusProfile = BolusProfile(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'] == 'true',
|
||||
name: json['name'],
|
||||
active: json['active'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
BolusProfile.put(bolusProfile);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -17,6 +17,7 @@ class GlucoseTarget {
|
||||
double fromMmolPerL;
|
||||
double toMmolPerL;
|
||||
int color;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
GlucoseTarget({
|
||||
@ -27,6 +28,7 @@ class GlucoseTarget {
|
||||
required this.fromMmolPerL,
|
||||
required this.toMmolPerL,
|
||||
required this.color,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -49,11 +51,15 @@ class GlucoseTarget {
|
||||
if (mgPerDl > 0 &&
|
||||
(mmolPerL == 0 ||
|
||||
Settings.glucoseMeasurement == GlucoseMeasurement.mgPerDl)) {
|
||||
condition = GlucoseTarget_.fromMgPerDL.lessOrEqual(mgPerDl).and(GlucoseTarget_.toMgPerDl.greaterOrEqual(mgPerDl));
|
||||
condition = GlucoseTarget_.fromMgPerDL
|
||||
.lessOrEqual(mgPerDl)
|
||||
.and(GlucoseTarget_.toMgPerDl.greaterOrEqual(mgPerDl));
|
||||
} else if (mmolPerL > 0 &&
|
||||
(mgPerDl == 0 ||
|
||||
Settings.glucoseMeasurement == GlucoseMeasurement.mmolPerL)) {
|
||||
condition = GlucoseTarget_.fromMmolPerL.lessOrEqual(mmolPerL).and(GlucoseTarget_.toMmolPerL.greaterOrEqual(mmolPerL));
|
||||
condition = GlucoseTarget_.fromMmolPerL
|
||||
.lessOrEqual(mmolPerL)
|
||||
.and(GlucoseTarget_.toMmolPerL.greaterOrEqual(mmolPerL));
|
||||
} else {
|
||||
return Colors.black;
|
||||
}
|
||||
@ -119,4 +125,33 @@ class GlucoseTarget {
|
||||
];
|
||||
box.putMany(defaultTargets);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['fromMgPerDL'] = fromMgPerDL;
|
||||
data['toMgPerDl'] = toMgPerDl;
|
||||
data['fromMmolPerL'] = fromMmolPerL;
|
||||
data['toMmolPerL'] = toMmolPerL;
|
||||
data['color'] = color;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final glucoseTarget = GlucoseTarget(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
fromMgPerDL: json['fromMgPerDL'],
|
||||
toMgPerDl: json['toMgPerDl'],
|
||||
fromMmolPerL: json['fromMmolPerL'],
|
||||
toMmolPerL: json['toMmolPerL'],
|
||||
color: json['color'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
GlucoseTarget.put(glucoseTarget);
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -24,6 +24,7 @@ class LogBolus {
|
||||
double? mmolPerLCorrection;
|
||||
bool setManually;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// relations
|
||||
final logEntry = ToOne<LogEntry>();
|
||||
@ -45,6 +46,7 @@ class LogBolus {
|
||||
this.mmolPerLCorrection,
|
||||
this.setManually = false,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -72,8 +74,7 @@ class LogBolus {
|
||||
}
|
||||
|
||||
static bool bolusForMealExists(int id) {
|
||||
QueryBuilder<LogBolus> builder = box.query(LogBolus_.deleted
|
||||
.equals(false));
|
||||
QueryBuilder<LogBolus> builder = box.query(LogBolus_.deleted.equals(false));
|
||||
builder.link(LogBolus_.meal, LogMeal_.id.equals(id));
|
||||
return builder.build().find().isNotEmpty;
|
||||
}
|
||||
@ -86,8 +87,63 @@ class LogBolus {
|
||||
}
|
||||
}
|
||||
|
||||
static void removeAllForEntry(int id) {
|
||||
box.putMany(getAllForEntry(id).map((item) {
|
||||
item.deleted = true;
|
||||
return item;
|
||||
}).toList());
|
||||
}
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return units.toString();
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['units'] = units;
|
||||
data['carbs'] = carbs;
|
||||
data['delay'] = delay;
|
||||
data['mgPerDlCurrent'] = mgPerDlCurrent;
|
||||
data['mgPerDlTarget'] = mgPerDlTarget;
|
||||
data['mgPerDlCorrection'] = mgPerDlCorrection;
|
||||
data['mmolPerLCurrent'] = mmolPerLCurrent;
|
||||
data['mmolPerLTarget'] = mmolPerLTarget;
|
||||
data['mmolPerLCorrection'] = mmolPerLCorrection;
|
||||
data['setManually'] = setManually;
|
||||
data['notes'] = notes;
|
||||
data['logEntry'] = logEntry.targetId;
|
||||
data['rate'] = rate.targetId;
|
||||
data['meal'] = meal.targetId;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final logBolus = LogBolus(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
units: json['units'],
|
||||
carbs: json['carbs'],
|
||||
delay: json['delay'],
|
||||
mgPerDlCurrent: json['mgPerDlCurrent'],
|
||||
mgPerDlTarget: json['mgPerDlTarget'],
|
||||
mgPerDlCorrection: json['mgPerDlCorrection'],
|
||||
mmolPerLCurrent: json['mmolPerLCurrent'],
|
||||
mmolPerLTarget: json['mmolPerLTarget'],
|
||||
mmolPerLCorrection: json['mmolPerLCorrection'],
|
||||
setManually: json['setManually'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
logBolus.logEntry.targetId = json['logEntry'];
|
||||
logBolus.rate.targetId = json['rate'];
|
||||
logBolus.meal.targetId = json['meal'];
|
||||
|
||||
LogBolus.put(logBolus);
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,6 @@
|
||||
import 'package:diameter/main.dart';
|
||||
import 'package:diameter/models/log_bolus.dart';
|
||||
import 'package:diameter/models/log_meal.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
import 'package:objectbox/objectbox.dart';
|
||||
@ -19,6 +20,7 @@ class LogEntry {
|
||||
double? mmolPerL;
|
||||
double? glucoseTrend;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
LogEntry({
|
||||
@ -29,6 +31,7 @@ class LogEntry {
|
||||
this.mmolPerL,
|
||||
this.glucoseTrend,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -40,6 +43,8 @@ class LogEntry {
|
||||
final item = box.get(id);
|
||||
if (item != null) {
|
||||
item.deleted = true;
|
||||
LogMeal.removeAllForEntry(id);
|
||||
LogBolus.removeAllForEntry(id);
|
||||
box.put(item);
|
||||
}
|
||||
}
|
||||
@ -64,8 +69,77 @@ class LogEntry {
|
||||
}).toList();
|
||||
}
|
||||
|
||||
static List<LogEntry> getAllByFilter({
|
||||
DateTime? startDate,
|
||||
DateTime? endDate,
|
||||
int? minMgPerDl,
|
||||
int? maxMgPerDl,
|
||||
double? minMmolPerL,
|
||||
double? maxMmolPerL,
|
||||
int? mealId,
|
||||
String? mealName,
|
||||
String? note,
|
||||
}) {
|
||||
DateTime start = startDate ?? DateTime(2000, 1, 1);
|
||||
DateTime end = endDate ?? DateTime.now();
|
||||
QueryBuilder<LogEntry> builder = box.query(LogEntry_.deleted.equals(false) &
|
||||
(Settings.glucoseMeasurement == GlucoseMeasurement.mgPerDl
|
||||
? LogEntry_.mgPerDl.between(
|
||||
minMgPerDl ?? 0, maxMgPerDl ?? double.maxFinite.toInt())
|
||||
: LogEntry_.mmolPerL
|
||||
.between(minMmolPerL ?? 0, maxMmolPerL ?? double.maxFinite)))
|
||||
..order(LogEntry_.time, flags: Order.descending);
|
||||
|
||||
return builder.build().find().where((entry) {
|
||||
return (note == null || (entry.notes ?? '').contains(note)) &&
|
||||
(mealId == null ||
|
||||
LogMeal.getAllForEntry(entry.id)
|
||||
.any((LogMeal logMeal) => logMeal.meal.targetId == mealId)) &&
|
||||
(mealName == null ||
|
||||
LogMeal.getAllForEntry(entry.id).any(
|
||||
(LogMeal logMeal) => logMeal.value.contains(mealName))) &&
|
||||
(entry.time.compareTo(start) >= 0 && entry.time.isBefore(end));
|
||||
}).toList();
|
||||
}
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return DateTimeUtils.displayDateTime(time);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['time'] = time.toIso8601String();
|
||||
data['mgPerDl'] = mgPerDl;
|
||||
data['mmolPerL'] = mmolPerL;
|
||||
data['glucoseTrend'] = glucoseTrend;
|
||||
data['notes'] = notes;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
DateTime? time = DateTime.tryParse(json['time']);
|
||||
|
||||
if (time == null) {
|
||||
return 'time is missing';
|
||||
}
|
||||
|
||||
final logEntry = LogEntry(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
time: time,
|
||||
mgPerDl: json['mgPerDl'],
|
||||
mmolPerL: json['mmolPerL'],
|
||||
glucoseTrend: json['glucoseTrend'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
LogEntry.put(logEntry);
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -20,6 +20,7 @@ class LogEvent {
|
||||
bool hasEndTime;
|
||||
int? reminderDuration;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
@Transient()
|
||||
String? title;
|
||||
@ -40,6 +41,7 @@ class LogEvent {
|
||||
this.hasEndTime = false,
|
||||
this.reminderDuration,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -143,4 +145,46 @@ class LogEvent {
|
||||
String toString() {
|
||||
return eventType.target?.value ?? '';
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['time'] = time.toIso8601String();
|
||||
data['endTime'] = endTime?.toIso8601String();
|
||||
data['hasEndTime'] = hasEndTime;
|
||||
data['reminderDuration'] = reminderDuration;
|
||||
data['notes'] = notes;
|
||||
data['eventType'] = eventType.targetId;
|
||||
data['bolusProfile'] = bolusProfile.targetId;
|
||||
data['basalProfile'] = basalProfile.targetId;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
DateTime? time = DateTime.tryParse(json['time']);
|
||||
|
||||
if (time == null) {
|
||||
return 'time is missing';
|
||||
}
|
||||
|
||||
final logEvent = LogEvent(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
time: time,
|
||||
endTime: DateTime.tryParse(json['endTime']),
|
||||
hasEndTime: json['hasEndTime'],
|
||||
reminderDuration: json['reminderDuration'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
logEvent.eventType.targetId = json['eventType'];
|
||||
logEvent.bolusProfile.targetId = json['bolusProfile'];
|
||||
logEvent.basalProfile.targetId = json['basalProfile'];
|
||||
|
||||
LogEvent.put(logEvent);
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -12,10 +12,12 @@ class LogEventType {
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String value;
|
||||
bool hasEndTime;
|
||||
int? defaultReminderDuration;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
LogEventType({
|
||||
@ -25,6 +27,7 @@ class LogEventType {
|
||||
this.hasEndTime = false,
|
||||
this.defaultReminderDuration,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// relations
|
||||
@ -52,4 +55,32 @@ class LogEventType {
|
||||
String toString() {
|
||||
return value;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['value'] = value;
|
||||
data['hasEndTime'] = hasEndTime;
|
||||
data['defaultReminderDuration'] = defaultReminderDuration;
|
||||
data['notes'] = notes;
|
||||
return data;
|
||||
}
|
||||
|
||||
|
||||
static String? putFromJson(Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final logEventType = LogEventType(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'],
|
||||
value: json['value'],
|
||||
hasEndTime: json['hasEndTime'],
|
||||
defaultReminderDuration: json['defaultReminderDuration'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
LogEventType.put(logEventType);
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -22,8 +22,8 @@ class LogMeal {
|
||||
double? portionSize;
|
||||
double? totalCarbs;
|
||||
String? notes;
|
||||
double? bolus;
|
||||
double amount;
|
||||
String? source;
|
||||
|
||||
// relations
|
||||
final logEntry = ToOne<LogEntry>();
|
||||
@ -44,6 +44,7 @@ class LogMeal {
|
||||
this.portionSize,
|
||||
this.totalCarbs,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -57,6 +58,13 @@ class LogMeal {
|
||||
}
|
||||
}
|
||||
|
||||
static void removeAllForEntry(int id) {
|
||||
box.putMany(getAllForEntry(id).map((item) {
|
||||
item.deleted = true;
|
||||
return item;
|
||||
}).toList());
|
||||
}
|
||||
|
||||
static List<LogMeal> getAllForEntry(int id) {
|
||||
QueryBuilder<LogMeal> builder = box.query(LogMeal_.deleted.equals(false));
|
||||
builder.link(LogMeal_.logEntry, LogEntry_.id.equals(id));
|
||||
@ -86,4 +94,50 @@ class LogMeal {
|
||||
String toString() {
|
||||
return value;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['value'] = value;
|
||||
data['carbsRatio'] = carbsRatio;
|
||||
data['portionSize'] = portionSize;
|
||||
data['totalCarbs'] = totalCarbs;
|
||||
data['notes'] = notes;
|
||||
data['amount'] = amount;
|
||||
data['logEntry'] = logEntry.targetId;
|
||||
data['meal'] = meal.targetId;
|
||||
data['mealSource'] = mealSource.targetId;
|
||||
data['mealCategory'] = mealCategory.targetId;
|
||||
data['mealPortionType'] = mealPortionType.targetId;
|
||||
data['portionSizeAccuracy'] = portionSizeAccuracy.targetId;
|
||||
data['carbsRatioAccuracy'] = carbsRatioAccuracy.targetId;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final logMeal = LogMeal(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'] == 'true',
|
||||
value: json['value'],
|
||||
carbsRatio: json['carbsRatio'],
|
||||
portionSize: json['portionSize'],
|
||||
totalCarbs: json['totalCarbs'],
|
||||
amount: json['amount'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
logMeal.logEntry.targetId = json['logEntry'];
|
||||
logMeal.meal.targetId = json['meal'];
|
||||
logMeal.mealSource.targetId = json['mealSource'];
|
||||
logMeal.mealCategory.targetId = json['mealCategory'];
|
||||
logMeal.mealPortionType.targetId = json['mealPortionType'];
|
||||
logMeal.portionSizeAccuracy.targetId = json['portionSizeAccuracy'];
|
||||
logMeal.carbsRatioAccuracy.targetId = json['carbsRatioAccuracy'];
|
||||
|
||||
LogMeal.put(logMeal);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -16,6 +16,7 @@ class Meal {
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String value;
|
||||
double? carbsRatio;
|
||||
double? portionSize;
|
||||
@ -23,6 +24,7 @@ class Meal {
|
||||
int? delayedBolusDuration;
|
||||
double? delayedBolusPercentage;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// relations
|
||||
final mealSource = ToOne<MealSource>();
|
||||
@ -42,6 +44,7 @@ class Meal {
|
||||
this.delayedBolusDuration,
|
||||
this.delayedBolusPercentage,
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -65,4 +68,48 @@ class Meal {
|
||||
String toString() {
|
||||
return value;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['value'] = value;
|
||||
data['carbsRatio'] = carbsRatio;
|
||||
data['portionSize'] = portionSize;
|
||||
data['carbsPerPortion'] = carbsPerPortion;
|
||||
data['delayedBolusDuration'] = delayedBolusDuration;
|
||||
data['delayedBolusPercentage'] = delayedBolusPercentage;
|
||||
data['notes'] = notes;
|
||||
data['mealSource'] = mealSource.targetId;
|
||||
data['mealCategory'] = mealCategory.targetId;
|
||||
data['mealPortionType'] = mealPortionType.targetId;
|
||||
data['portionSizeAccuracy'] = portionSizeAccuracy.targetId;
|
||||
data['carbsRatioAccuracy'] = carbsRatioAccuracy.targetId;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final meal = Meal(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'] == 'true',
|
||||
value: json['value'],
|
||||
carbsRatio: json['carbsRatio'],
|
||||
portionSize: json['portionSize'],
|
||||
carbsPerPortion: json['carbsPerPortion'],
|
||||
delayedBolusDuration: json['delayedBolusDuration'],
|
||||
delayedBolusPercentage: json['delayedBolusPercentage'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
meal.mealSource.targetId = json['mealSource'];
|
||||
meal.mealCategory.targetId = json['mealCategory'];
|
||||
meal.mealPortionType.targetId = json['mealPortionType'];
|
||||
meal.portionSizeAccuracy.targetId = json['portionSizeAccuracy'];
|
||||
meal.carbsRatioAccuracy.targetId = json['carbsRatioAccuracy'];
|
||||
|
||||
Meal.put(meal);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -10,8 +10,10 @@ class MealCategory {
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String value;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
MealCategory({
|
||||
@ -19,6 +21,7 @@ class MealCategory {
|
||||
this.deleted = false,
|
||||
this.value = '',
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -42,4 +45,27 @@ class MealCategory {
|
||||
String toString() {
|
||||
return value;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['value'] = value;
|
||||
data['notes'] = notes;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final mealCategory = MealCategory(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'] == 'true',
|
||||
value: json['value'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
MealCategory.put(mealCategory);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -5,13 +5,16 @@ import 'package:diameter/objectbox.g.dart' show MealPortionType_;
|
||||
@Entity(uid: 2111511899235985637)
|
||||
@Sync()
|
||||
class MealPortionType {
|
||||
static final Box<MealPortionType> box = objectBox.store.box<MealPortionType>();
|
||||
static final Box<MealPortionType> box =
|
||||
objectBox.store.box<MealPortionType>();
|
||||
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String value;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// constructor
|
||||
MealPortionType({
|
||||
@ -19,6 +22,7 @@ class MealPortionType {
|
||||
this.deleted = false,
|
||||
this.value = '',
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -26,7 +30,9 @@ class MealPortionType {
|
||||
static void put(MealPortionType mealPortionType) => box.put(mealPortionType);
|
||||
|
||||
static List<MealPortionType> getAll() {
|
||||
QueryBuilder<MealPortionType> builder = box.query(MealPortionType_.deleted.equals(false))..order(MealPortionType_.value);
|
||||
QueryBuilder<MealPortionType> builder = box
|
||||
.query(MealPortionType_.deleted.equals(false))
|
||||
..order(MealPortionType_.value);
|
||||
return builder.build().find();
|
||||
}
|
||||
|
||||
@ -42,4 +48,27 @@ class MealPortionType {
|
||||
String toString() {
|
||||
return value;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['value'] = value;
|
||||
data['notes'] = notes;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final mealPortionType = MealPortionType(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'] == 'true',
|
||||
value: json['value'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
MealPortionType.put(mealPortionType);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -13,8 +13,10 @@ class MealSource {
|
||||
// properties
|
||||
int id;
|
||||
bool deleted;
|
||||
@Unique()
|
||||
String value;
|
||||
String? notes;
|
||||
String? source;
|
||||
|
||||
// relations
|
||||
final defaultMealCategory = ToOne<MealCategory>();
|
||||
@ -28,6 +30,7 @@ class MealSource {
|
||||
this.deleted = false,
|
||||
this.value = '',
|
||||
this.notes,
|
||||
this.source,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -35,7 +38,8 @@ class MealSource {
|
||||
static void put(MealSource mealSource) => box.put(mealSource);
|
||||
|
||||
static List<MealSource> getAll() {
|
||||
QueryBuilder<MealSource> builder = box.query(MealSource_.deleted.equals(false))..order(MealSource_.value);
|
||||
QueryBuilder<MealSource> builder =
|
||||
box.query(MealSource_.deleted.equals(false))..order(MealSource_.value);
|
||||
return builder.build().find();
|
||||
}
|
||||
|
||||
@ -51,4 +55,36 @@ class MealSource {
|
||||
String toString() {
|
||||
return value;
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = id;
|
||||
data['deleted'] = deleted;
|
||||
data['value'] = value;
|
||||
data['notes'] = notes;
|
||||
data['defaultMealCategory'] = defaultMealCategory.targetId;
|
||||
data['defaultMealPortionType'] = defaultMealPortionType.targetId;
|
||||
data['defaultCarbsRatioAccuracy'] = defaultCarbsRatioAccuracy.targetId;
|
||||
data['defaultPortionSizeAccuracy'] = defaultPortionSizeAccuracy.targetId;
|
||||
return data;
|
||||
}
|
||||
|
||||
static String? putFromJson(
|
||||
Map<String, dynamic> json, bool overrideExisting, String? source) {
|
||||
final mealSource = MealSource(
|
||||
id: overrideExisting ? json['id'] : 0,
|
||||
deleted: json['deleted'] == 'true',
|
||||
value: json['value'],
|
||||
notes: json['notes'],
|
||||
source: source,
|
||||
);
|
||||
|
||||
mealSource.defaultMealCategory.targetId = json['defaultMealCategory'];
|
||||
mealSource.defaultMealPortionType.targetId = json['defaultMealPortionType'];
|
||||
mealSource.defaultCarbsRatioAccuracy.targetId = json['defaultCarbsRatioAccuracy'];
|
||||
mealSource.defaultPortionSizeAccuracy.targetId = json['defaultPortionSizeAccuracy'];
|
||||
|
||||
MealSource.put(mealSource);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -69,6 +69,8 @@ class Settings {
|
||||
|
||||
bool useDarkTheme;
|
||||
|
||||
DateTime? lastExportTimestamp;
|
||||
|
||||
// constructor
|
||||
Settings({
|
||||
this.id = 0,
|
||||
@ -89,6 +91,7 @@ class Settings {
|
||||
this.targetGlucoseMgPerDl = 100,
|
||||
this.targetGlucoseMmolPerL = 5.5,
|
||||
this.useDarkTheme = false,
|
||||
this.lastExportTimestamp,
|
||||
});
|
||||
|
||||
// methods
|
||||
@ -101,7 +104,8 @@ class Settings {
|
||||
|
||||
static NutritionMeasurement get nutritionMeasurement =>
|
||||
NutritionMeasurement.values[get().nutritionMeasurementIndex];
|
||||
static GlucoseMeasurement get glucoseMeasurement => GlucoseMeasurement.values[get().glucoseMeasurementIndex];
|
||||
static GlucoseMeasurement get glucoseMeasurement =>
|
||||
GlucoseMeasurement.values[get().glucoseMeasurementIndex];
|
||||
static GlucoseDisplayMode get glucoseDisplayMode =>
|
||||
GlucoseDisplayMode.values[get().glucoseDisplayModeIndex];
|
||||
|
||||
@ -120,10 +124,39 @@ class Settings {
|
||||
static ThemeMode get themeMode =>
|
||||
get().useDarkTheme ? ThemeMode.dark : ThemeMode.light;
|
||||
|
||||
static DateTime? get lastExportTimeStamp => get().lastExportTimestamp;
|
||||
|
||||
static void put(Settings settings) => box.put(settings);
|
||||
|
||||
static void reset() {
|
||||
box.removeAll();
|
||||
box.put(Settings(useDarkTheme: ThemeMode.system == ThemeMode.dark));
|
||||
}
|
||||
|
||||
static Map<String, dynamic> toJson() {
|
||||
Settings settings = get();
|
||||
final Map<String, dynamic> data = <String, dynamic>{};
|
||||
data['id'] = settings.id;
|
||||
data['nutritionMeasurementIndex'] = settings.nutritionMeasurementIndex;
|
||||
data['glucoseDisplayModeIndex'] = settings.glucoseDisplayModeIndex;
|
||||
data['glucoseMeasurementIndex'] = settings.glucoseMeasurementIndex;
|
||||
data['targetGlucoseMgPerDl'] = settings.targetGlucoseMgPerDl;
|
||||
data['targetGlucoseMmolPerL'] = settings.targetGlucoseMmolPerL;
|
||||
data['insulinIncrements'] = settings.insulinIncrements;
|
||||
data['nutritionIncrements'] = settings.nutritionIncrements;
|
||||
data['mmolPerLIncrements'] = settings.mmolPerLIncrements;
|
||||
data['amountIncrements'] = settings.amountIncrements;
|
||||
data['dateFormat'] = settings.dateFormat;
|
||||
data['longDateFormat'] = settings.longDateFormat;
|
||||
data['timeFormat'] = settings.timeFormat;
|
||||
data['longTimeFormat'] = settings.longTimeFormat;
|
||||
data['showConfirmationDialogOnCancel'] =
|
||||
settings.showConfirmationDialogOnCancel;
|
||||
data['showConfirmationDialogOnDelete'] =
|
||||
settings.showConfirmationDialogOnDelete;
|
||||
data['showConfirmationDialogOnStopEvent'] =
|
||||
settings.showConfirmationDialogOnStopEvent;
|
||||
data['useDarkTheme'] = settings.useDarkTheme;
|
||||
return data;
|
||||
}
|
||||
}
|
||||
|
@ -1,12 +1,12 @@
|
||||
import 'package:diameter/main.dart';
|
||||
import 'package:diameter/models/meal.dart';
|
||||
import 'package:diameter/models/recipe.dart';
|
||||
import 'package:diameter/models/x_recipe.dart';
|
||||
import 'package:diameter/utils/utils.dart';
|
||||
import 'package:objectbox/objectbox.dart';
|
||||
import 'package:diameter/objectbox.g.dart' show Ingredient_, Recipe_;
|
||||
|
||||
@Entity(uid: 6950311793136068892)
|
||||
@Sync()
|
||||
// @Entity(uid: 6950311793136068892)
|
||||
// @Sync()
|
||||
class Ingredient {
|
||||
static final Box<Ingredient> box = objectBox.store.box<Ingredient>();
|
||||
|
@ -1,12 +1,12 @@
|
||||
import 'package:diameter/main.dart';
|
||||
import 'package:diameter/models/ingredient.dart';
|
||||
import 'package:diameter/models/x_ingredient.dart';
|
||||
import 'package:diameter/models/meal.dart';
|
||||
import 'package:diameter/utils/utils.dart';
|
||||
import 'package:objectbox/objectbox.dart';
|
||||
import 'package:diameter/objectbox.g.dart' show Recipe_;
|
||||
|
||||
@Entity(uid: 6497942314956341514)
|
||||
@Sync()
|
||||
// @Entity(uid: 6497942314956341514)
|
||||
// @Sync()
|
||||
class Recipe {
|
||||
static final Box<Recipe> box = objectBox.store.box<Recipe>();
|
||||
|
@ -1,8 +1,8 @@
|
||||
import 'package:diameter/main.dart';
|
||||
import 'package:objectbox/objectbox.dart';
|
||||
|
||||
@Entity()
|
||||
@Sync()
|
||||
// @Entity()
|
||||
// @Sync()
|
||||
class User {
|
||||
static final Box<User> box = objectBox.store.box<User>();
|
||||
// properties
|
@ -1,3 +1,4 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/screens/category/categories.dart';
|
||||
import 'package:diameter/screens/category/accuracy_detail.dart';
|
||||
import 'package:diameter/screens/category/accuracy_list.dart';
|
||||
@ -28,6 +29,7 @@ import 'package:diameter/screens/reports/export.dart';
|
||||
import 'package:diameter/screens/reports/reports.dart';
|
||||
import 'package:diameter/settings.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class Routes {
|
||||
static const String basal = BasalDetailScreen.routeName;
|
||||
@ -119,7 +121,7 @@ class _NavigationState extends State<Navigation> {
|
||||
),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Log'),
|
||||
title: Text(translate(LocalizationKeys.navigation_log)),
|
||||
leading: const Icon(Icons.dashboard),
|
||||
onTap: () {
|
||||
selectDestination(Routes.log);
|
||||
@ -127,7 +129,7 @@ class _NavigationState extends State<Navigation> {
|
||||
selected: widget.currentLocation == Routes.log,
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Log Events'),
|
||||
title: Text(translate(LocalizationKeys.navigation_logEvents)),
|
||||
leading: const Icon(Icons.event),
|
||||
onTap: () {
|
||||
selectDestination(Routes.logEvents);
|
||||
@ -135,7 +137,7 @@ class _NavigationState extends State<Navigation> {
|
||||
selected: widget.currentLocation == Routes.logEvents,
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Reports'),
|
||||
title: Text(translate(LocalizationKeys.navigation_reports)),
|
||||
leading: const Icon(Icons.show_chart),
|
||||
onTap: () {
|
||||
selectDestination(Routes.reports);
|
||||
@ -143,7 +145,7 @@ class _NavigationState extends State<Navigation> {
|
||||
selected: Routes.reportRoutes.contains(widget.currentLocation),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Meals'),
|
||||
title: Text(translate(LocalizationKeys.navigation_meals)),
|
||||
leading: const Icon(Icons.dinner_dining),
|
||||
onTap: () {
|
||||
selectDestination(Routes.meals);
|
||||
@ -151,7 +153,7 @@ class _NavigationState extends State<Navigation> {
|
||||
selected: Routes.mealRoutes.contains(widget.currentLocation),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Basal Profiles'),
|
||||
title: Text(translate(LocalizationKeys.navigation_basalProfiles)),
|
||||
leading: const Icon(Icons.access_time),
|
||||
onTap: () {
|
||||
selectDestination(Routes.basalProfiles);
|
||||
@ -159,7 +161,7 @@ class _NavigationState extends State<Navigation> {
|
||||
selected: Routes.basalRoutes.contains(widget.currentLocation),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Bolus Profiles'),
|
||||
title: Text(translate(LocalizationKeys.navigation_bolusProfiles)),
|
||||
leading: const Icon(Icons.medication),
|
||||
onTap: () {
|
||||
selectDestination(Routes.bolusProfiles);
|
||||
@ -167,7 +169,7 @@ class _NavigationState extends State<Navigation> {
|
||||
selected: Routes.bolusRoutes.contains(widget.currentLocation),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Categorization'),
|
||||
title: Text(translate(LocalizationKeys.navigation_categorization)),
|
||||
leading: const Icon(Icons.category),
|
||||
onTap: () {
|
||||
selectDestination(Routes.category);
|
||||
@ -175,7 +177,7 @@ class _NavigationState extends State<Navigation> {
|
||||
selected: Routes.categoryRoutes.contains(widget.currentLocation),
|
||||
),
|
||||
ListTile(
|
||||
title: const Text('Settings'),
|
||||
title: Text(translate(LocalizationKeys.navigation_settings)),
|
||||
leading: const Icon(Icons.settings),
|
||||
onTap: () {
|
||||
selectDestination(Routes.settings);
|
||||
|
@ -5,7 +5,7 @@
|
||||
"entities": [
|
||||
{
|
||||
"id": "2:1467758525778521891",
|
||||
"lastPropertyId": "6:3409466778841164684",
|
||||
"lastPropertyId": "7:3194552447464951410",
|
||||
"name": "Basal",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -42,13 +42,18 @@
|
||||
"id": "6:3409466778841164684",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "7:3194552447464951410",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "3:3613736032926903785",
|
||||
"lastPropertyId": "5:8140071977687660397",
|
||||
"lastPropertyId": "6:3153627070507809184",
|
||||
"name": "BasalProfile",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -77,13 +82,18 @@
|
||||
"id": "5:8140071977687660397",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "6:3153627070507809184",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "4:3417770529060202389",
|
||||
"lastPropertyId": "9:7440090146687096977",
|
||||
"lastPropertyId": "10:6138236789530596038",
|
||||
"name": "Bolus",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -135,13 +145,18 @@
|
||||
"id": "9:7440090146687096977",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "10:6138236789530596038",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "5:8812452529027052317",
|
||||
"lastPropertyId": "5:8082994824481464395",
|
||||
"lastPropertyId": "6:8746199841403936632",
|
||||
"name": "BolusProfile",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -170,13 +185,18 @@
|
||||
"id": "5:8082994824481464395",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "6:8746199841403936632",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "6:752131069307970560",
|
||||
"lastPropertyId": "10:2505303363495348118",
|
||||
"lastPropertyId": "11:8454251731096762477",
|
||||
"name": "LogEntry",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -215,13 +235,18 @@
|
||||
"id": "10:2505303363495348118",
|
||||
"name": "glucoseTrend",
|
||||
"type": 8
|
||||
},
|
||||
{
|
||||
"id": "11:8454251731096762477",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "7:4303325892753185970",
|
||||
"lastPropertyId": "12:3041952167628926163",
|
||||
"lastPropertyId": "13:9206766629540069341",
|
||||
"name": "LogEvent",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -284,13 +309,18 @@
|
||||
"id": "12:3041952167628926163",
|
||||
"name": "reminderDuration",
|
||||
"type": 6
|
||||
},
|
||||
{
|
||||
"id": "13:9206766629540069341",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "8:8362795406595606110",
|
||||
"lastPropertyId": "8:1869014400856897151",
|
||||
"lastPropertyId": "9:7377683740652293217",
|
||||
"name": "LogEventType",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -340,13 +370,18 @@
|
||||
"flags": 520,
|
||||
"indexId": "28:4563029809754152081",
|
||||
"relationTarget": "BasalProfile"
|
||||
},
|
||||
{
|
||||
"id": "9:7377683740652293217",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "9:411177866700467286",
|
||||
"lastPropertyId": "19:8965198821438347033",
|
||||
"lastPropertyId": "20:3296734778930341954",
|
||||
"name": "LogMeal",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -371,11 +406,6 @@
|
||||
"name": "portionSize",
|
||||
"type": 8
|
||||
},
|
||||
{
|
||||
"id": "6:8074052538574863399",
|
||||
"name": "bolus",
|
||||
"type": 8
|
||||
},
|
||||
{
|
||||
"id": "9:1920579694098037947",
|
||||
"name": "notes",
|
||||
@ -451,13 +481,18 @@
|
||||
"id": "19:8965198821438347033",
|
||||
"name": "totalCarbs",
|
||||
"type": 8
|
||||
},
|
||||
{
|
||||
"id": "20:3296734778930341954",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "10:382130101578692012",
|
||||
"lastPropertyId": "15:8283810711091063880",
|
||||
"lastPropertyId": "16:8612850434699732822",
|
||||
"name": "Meal",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -546,13 +581,18 @@
|
||||
"id": "15:8283810711091063880",
|
||||
"name": "delayedBolusPercentage",
|
||||
"type": 8
|
||||
},
|
||||
{
|
||||
"id": "16:8612850434699732822",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "11:3158200688796904913",
|
||||
"lastPropertyId": "4:824435977543069541",
|
||||
"lastPropertyId": "5:2221810999445049020",
|
||||
"name": "MealCategory",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -576,13 +616,18 @@
|
||||
"id": "4:824435977543069541",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "5:2221810999445049020",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "12:2111511899235985637",
|
||||
"lastPropertyId": "4:5680236937391945907",
|
||||
"lastPropertyId": "5:6807468448392267469",
|
||||
"name": "MealPortionType",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -606,13 +651,18 @@
|
||||
"id": "4:5680236937391945907",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "5:6807468448392267469",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "13:1283034494527412242",
|
||||
"lastPropertyId": "8:4547899751779962180",
|
||||
"lastPropertyId": "9:6873017370661053783",
|
||||
"name": "MealSource",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -668,13 +718,18 @@
|
||||
"id": "8:4547899751779962180",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "9:6873017370661053783",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "14:8033487006694871160",
|
||||
"lastPropertyId": "18:7503231998671134983",
|
||||
"lastPropertyId": "19:3683854670538916324",
|
||||
"name": "LogBolus",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -767,13 +822,18 @@
|
||||
"id": "18:7503231998671134983",
|
||||
"name": "mmolPerLCorrection",
|
||||
"type": 8
|
||||
},
|
||||
{
|
||||
"id": "19:3683854670538916324",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "15:291512798403320400",
|
||||
"lastPropertyId": "7:6675647182186603076",
|
||||
"lastPropertyId": "8:1800866627092978542",
|
||||
"name": "Accuracy",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -812,13 +872,18 @@
|
||||
"id": "7:6675647182186603076",
|
||||
"name": "deleted",
|
||||
"type": 1
|
||||
},
|
||||
{
|
||||
"id": "8:1800866627092978542",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "16:3989341091218179227",
|
||||
"lastPropertyId": "27:3553639710779248831",
|
||||
"lastPropertyId": "28:7196743106880728684",
|
||||
"name": "Settings",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -912,13 +977,18 @@
|
||||
"id": "27:3553639710779248831",
|
||||
"name": "amountIncrements",
|
||||
"type": 8
|
||||
},
|
||||
{
|
||||
"id": "28:7196743106880728684",
|
||||
"name": "lastExportTimestamp",
|
||||
"type": 10
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
},
|
||||
{
|
||||
"id": "17:5041265995704044399",
|
||||
"lastPropertyId": "7:1333487551279074696",
|
||||
"lastPropertyId": "8:1091046738846336945",
|
||||
"name": "GlucoseTarget",
|
||||
"flags": 2,
|
||||
"properties": [
|
||||
@ -957,6 +1027,11 @@
|
||||
"id": "7:1333487551279074696",
|
||||
"name": "color",
|
||||
"type": 6
|
||||
},
|
||||
{
|
||||
"id": "8:1091046738846336945",
|
||||
"name": "source",
|
||||
"type": 9
|
||||
}
|
||||
],
|
||||
"relations": []
|
||||
@ -1125,7 +1200,8 @@
|
||||
4678123663117222609,
|
||||
780211923138281722,
|
||||
763575433624979013,
|
||||
1225271130099322691
|
||||
1225271130099322691,
|
||||
8074052538574863399
|
||||
],
|
||||
"retiredRelationUids": [],
|
||||
"version": 1
|
||||
|
@ -19,7 +19,7 @@ import 'models/basal_profile.dart';
|
||||
import 'models/bolus.dart';
|
||||
import 'models/bolus_profile.dart';
|
||||
import 'models/glucose_target.dart';
|
||||
import 'models/ingredient.dart';
|
||||
import 'models/x_ingredient.dart';
|
||||
import 'models/log_bolus.dart';
|
||||
import 'models/log_entry.dart';
|
||||
import 'models/log_event.dart';
|
||||
@ -29,9 +29,9 @@ import 'models/meal.dart';
|
||||
import 'models/meal_category.dart';
|
||||
import 'models/meal_portion_type.dart';
|
||||
import 'models/meal_source.dart';
|
||||
import 'models/recipe.dart';
|
||||
import 'models/x_recipe.dart';
|
||||
import 'models/settings.dart';
|
||||
import 'models/user.dart';
|
||||
import 'models/x_user.dart';
|
||||
|
||||
export 'package:objectbox/objectbox.dart'; // so that callers only have to import this file
|
||||
|
||||
@ -39,7 +39,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(2, 1467758525778521891),
|
||||
name: 'Basal',
|
||||
lastPropertyId: const IdUid(6, 3409466778841164684),
|
||||
lastPropertyId: const IdUid(7, 3194552447464951410),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -73,6 +73,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(6, 3409466778841164684),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(7, 3194552447464951410),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -80,7 +85,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(3, 3613736032926903785),
|
||||
name: 'BasalProfile',
|
||||
lastPropertyId: const IdUid(5, 8140071977687660397),
|
||||
lastPropertyId: const IdUid(6, 3153627070507809184),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -107,6 +112,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(5, 8140071977687660397),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(6, 3153627070507809184),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -114,7 +124,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(4, 3417770529060202389),
|
||||
name: 'Bolus',
|
||||
lastPropertyId: const IdUid(9, 7440090146687096977),
|
||||
lastPropertyId: const IdUid(10, 6138236789530596038),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -163,6 +173,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(9, 7440090146687096977),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(10, 6138236789530596038),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -170,7 +185,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(5, 8812452529027052317),
|
||||
name: 'BolusProfile',
|
||||
lastPropertyId: const IdUid(5, 8082994824481464395),
|
||||
lastPropertyId: const IdUid(6, 8746199841403936632),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -197,6 +212,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(5, 8082994824481464395),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(6, 8746199841403936632),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -204,7 +224,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(6, 752131069307970560),
|
||||
name: 'LogEntry',
|
||||
lastPropertyId: const IdUid(10, 2505303363495348118),
|
||||
lastPropertyId: const IdUid(11, 8454251731096762477),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -241,6 +261,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(10, 2505303363495348118),
|
||||
name: 'glucoseTrend',
|
||||
type: 8,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(11, 8454251731096762477),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -248,7 +273,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(7, 4303325892753185970),
|
||||
name: 'LogEvent',
|
||||
lastPropertyId: const IdUid(12, 3041952167628926163),
|
||||
lastPropertyId: const IdUid(13, 9206766629540069341),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -306,6 +331,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(12, 3041952167628926163),
|
||||
name: 'reminderDuration',
|
||||
type: 6,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(13, 9206766629540069341),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -313,7 +343,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(8, 8362795406595606110),
|
||||
name: 'LogEventType',
|
||||
lastPropertyId: const IdUid(8, 1869014400856897151),
|
||||
lastPropertyId: const IdUid(9, 7377683740652293217),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -359,14 +389,19 @@ final _entities = <ModelEntity>[
|
||||
type: 11,
|
||||
flags: 520,
|
||||
indexId: const IdUid(28, 4563029809754152081),
|
||||
relationTarget: 'BasalProfile')
|
||||
relationTarget: 'BasalProfile'),
|
||||
ModelProperty(
|
||||
id: const IdUid(9, 7377683740652293217),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
backlinks: <ModelBacklink>[]),
|
||||
ModelEntity(
|
||||
id: const IdUid(9, 411177866700467286),
|
||||
name: 'LogMeal',
|
||||
lastPropertyId: const IdUid(19, 8965198821438347033),
|
||||
lastPropertyId: const IdUid(20, 3296734778930341954),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -389,11 +424,6 @@ final _entities = <ModelEntity>[
|
||||
name: 'portionSize',
|
||||
type: 8,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(6, 8074052538574863399),
|
||||
name: 'bolus',
|
||||
type: 8,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(9, 1920579694098037947),
|
||||
name: 'notes',
|
||||
@ -462,6 +492,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(19, 8965198821438347033),
|
||||
name: 'totalCarbs',
|
||||
type: 8,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(20, 3296734778930341954),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -469,7 +504,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(10, 382130101578692012),
|
||||
name: 'Meal',
|
||||
lastPropertyId: const IdUid(15, 8283810711091063880),
|
||||
lastPropertyId: const IdUid(16, 8612850434699732822),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -551,6 +586,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(15, 8283810711091063880),
|
||||
name: 'delayedBolusPercentage',
|
||||
type: 8,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(16, 8612850434699732822),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -558,7 +598,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(11, 3158200688796904913),
|
||||
name: 'MealCategory',
|
||||
lastPropertyId: const IdUid(4, 824435977543069541),
|
||||
lastPropertyId: const IdUid(5, 2221810999445049020),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -580,6 +620,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(4, 824435977543069541),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(5, 2221810999445049020),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -587,7 +632,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(12, 2111511899235985637),
|
||||
name: 'MealPortionType',
|
||||
lastPropertyId: const IdUid(4, 5680236937391945907),
|
||||
lastPropertyId: const IdUid(5, 6807468448392267469),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -609,6 +654,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(4, 5680236937391945907),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(5, 6807468448392267469),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -616,7 +666,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(13, 1283034494527412242),
|
||||
name: 'MealSource',
|
||||
lastPropertyId: const IdUid(8, 4547899751779962180),
|
||||
lastPropertyId: const IdUid(9, 6873017370661053783),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -666,6 +716,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(8, 4547899751779962180),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(9, 6873017370661053783),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -673,7 +728,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(14, 8033487006694871160),
|
||||
name: 'LogBolus',
|
||||
lastPropertyId: const IdUid(18, 7503231998671134983),
|
||||
lastPropertyId: const IdUid(19, 3683854670538916324),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -761,6 +816,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(18, 7503231998671134983),
|
||||
name: 'mmolPerLCorrection',
|
||||
type: 8,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(19, 3683854670538916324),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -768,7 +828,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(15, 291512798403320400),
|
||||
name: 'Accuracy',
|
||||
lastPropertyId: const IdUid(7, 6675647182186603076),
|
||||
lastPropertyId: const IdUid(8, 1800866627092978542),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -805,6 +865,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(7, 6675647182186603076),
|
||||
name: 'deleted',
|
||||
type: 1,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(8, 1800866627092978542),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -812,7 +877,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(16, 3989341091218179227),
|
||||
name: 'Settings',
|
||||
lastPropertyId: const IdUid(27, 3553639710779248831),
|
||||
lastPropertyId: const IdUid(28, 7196743106880728684),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -904,6 +969,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(27, 3553639710779248831),
|
||||
name: 'amountIncrements',
|
||||
type: 8,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(28, 7196743106880728684),
|
||||
name: 'lastExportTimestamp',
|
||||
type: 10,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -911,7 +981,7 @@ final _entities = <ModelEntity>[
|
||||
ModelEntity(
|
||||
id: const IdUid(17, 5041265995704044399),
|
||||
name: 'GlucoseTarget',
|
||||
lastPropertyId: const IdUid(7, 1333487551279074696),
|
||||
lastPropertyId: const IdUid(8, 1091046738846336945),
|
||||
flags: 2,
|
||||
properties: <ModelProperty>[
|
||||
ModelProperty(
|
||||
@ -948,6 +1018,11 @@ final _entities = <ModelEntity>[
|
||||
id: const IdUid(7, 1333487551279074696),
|
||||
name: 'color',
|
||||
type: 6,
|
||||
flags: 0),
|
||||
ModelProperty(
|
||||
id: const IdUid(8, 1091046738846336945),
|
||||
name: 'source',
|
||||
type: 9,
|
||||
flags: 0)
|
||||
],
|
||||
relations: <ModelRelation>[],
|
||||
@ -1124,7 +1199,8 @@ ModelDefinition getObjectBoxModel() {
|
||||
4678123663117222609,
|
||||
780211923138281722,
|
||||
763575433624979013,
|
||||
1225271130099322691
|
||||
1225271130099322691,
|
||||
8074052538574863399
|
||||
],
|
||||
retiredRelationUids: const [],
|
||||
modelVersion: 5,
|
||||
@ -1141,13 +1217,16 @@ ModelDefinition getObjectBoxModel() {
|
||||
object.id = id;
|
||||
},
|
||||
objectToFB: (Basal object, fb.Builder fbb) {
|
||||
fbb.startTable(7);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(8);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addInt64(1, object.startTime.millisecondsSinceEpoch);
|
||||
fbb.addInt64(2, object.endTime.millisecondsSinceEpoch);
|
||||
fbb.addFloat64(3, object.units);
|
||||
fbb.addInt64(4, object.basalProfile.targetId);
|
||||
fbb.addBool(5, object.deleted);
|
||||
fbb.addOffset(6, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1163,8 +1242,10 @@ ModelDefinition getObjectBoxModel() {
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 6, 0)),
|
||||
endTime: DateTime.fromMillisecondsSinceEpoch(
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 8, 0)),
|
||||
units: const fb.Float64Reader()
|
||||
.vTableGet(buffer, rootOffset, 10, 0));
|
||||
units:
|
||||
const fb.Float64Reader().vTableGet(buffer, rootOffset, 10, 0),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 16));
|
||||
object.basalProfile.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 12, 0);
|
||||
object.basalProfile.attach(store);
|
||||
@ -1182,12 +1263,15 @@ ModelDefinition getObjectBoxModel() {
|
||||
final nameOffset = fbb.writeString(object.name);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(6);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(7);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, nameOffset);
|
||||
fbb.addBool(2, object.active);
|
||||
fbb.addOffset(3, notesOffset);
|
||||
fbb.addBool(4, object.deleted);
|
||||
fbb.addOffset(5, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1204,7 +1288,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
active:
|
||||
const fb.BoolReader().vTableGet(buffer, rootOffset, 8, false),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 10));
|
||||
.vTableGetNullable(buffer, rootOffset, 10),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 14));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -1217,7 +1303,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
object.id = id;
|
||||
},
|
||||
objectToFB: (Bolus object, fb.Builder fbb) {
|
||||
fbb.startTable(10);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(11);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addInt64(1, object.startTime.millisecondsSinceEpoch);
|
||||
fbb.addInt64(2, object.endTime.millisecondsSinceEpoch);
|
||||
@ -1227,6 +1315,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addFloat64(6, object.mmolPerL);
|
||||
fbb.addInt64(7, object.bolusProfile.targetId);
|
||||
fbb.addBool(8, object.deleted);
|
||||
fbb.addOffset(9, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1249,7 +1338,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
mgPerDl: const fb.Int64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 14),
|
||||
mmolPerL: const fb.Float64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 16));
|
||||
.vTableGetNullable(buffer, rootOffset, 16),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 22));
|
||||
object.bolusProfile.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 18, 0);
|
||||
object.bolusProfile.attach(store);
|
||||
@ -1267,12 +1358,15 @@ ModelDefinition getObjectBoxModel() {
|
||||
final nameOffset = fbb.writeString(object.name);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(6);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(7);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, nameOffset);
|
||||
fbb.addBool(2, object.active);
|
||||
fbb.addOffset(3, notesOffset);
|
||||
fbb.addBool(4, object.deleted);
|
||||
fbb.addOffset(5, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1289,7 +1383,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
active:
|
||||
const fb.BoolReader().vTableGet(buffer, rootOffset, 8, false),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 10));
|
||||
.vTableGetNullable(buffer, rootOffset, 10),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 14));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -1304,7 +1400,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
objectToFB: (LogEntry object, fb.Builder fbb) {
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(11);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(12);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addInt64(1, object.time.millisecondsSinceEpoch);
|
||||
fbb.addInt64(2, object.mgPerDl);
|
||||
@ -1312,6 +1410,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addOffset(7, notesOffset);
|
||||
fbb.addBool(8, object.deleted);
|
||||
fbb.addFloat64(9, object.glucoseTrend);
|
||||
fbb.addOffset(10, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1332,7 +1431,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
glucoseTrend: const fb.Float64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 22),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 18));
|
||||
.vTableGetNullable(buffer, rootOffset, 18),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 24));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -1348,7 +1449,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
objectToFB: (LogEvent object, fb.Builder fbb) {
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(13);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(14);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addInt64(1, object.time.millisecondsSinceEpoch);
|
||||
fbb.addInt64(2, object.endTime?.millisecondsSinceEpoch);
|
||||
@ -1359,6 +1462,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addInt64(9, object.bolusProfile.targetId);
|
||||
fbb.addInt64(10, object.basalProfile.targetId);
|
||||
fbb.addInt64(11, object.reminderDuration);
|
||||
fbb.addOffset(12, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1381,7 +1485,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
reminderDuration: const fb.Int64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 26),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 12));
|
||||
.vTableGetNullable(buffer, rootOffset, 12),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 28));
|
||||
object.eventType.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 18, 0);
|
||||
object.eventType.attach(store);
|
||||
@ -1406,7 +1512,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
final valueOffset = fbb.writeString(object.value);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(9);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(10);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, valueOffset);
|
||||
fbb.addBool(2, object.hasEndTime);
|
||||
@ -1415,6 +1523,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addBool(5, object.deleted);
|
||||
fbb.addInt64(6, object.bolusProfile.targetId);
|
||||
fbb.addInt64(7, object.basalProfile.targetId);
|
||||
fbb.addOffset(8, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1433,7 +1542,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
defaultReminderDuration: const fb.Int64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 10),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 12));
|
||||
.vTableGetNullable(buffer, rootOffset, 12),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 20));
|
||||
object.bolusProfile.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 16, 0);
|
||||
object.bolusProfile.attach(store);
|
||||
@ -1462,12 +1573,13 @@ ModelDefinition getObjectBoxModel() {
|
||||
final valueOffset = fbb.writeString(object.value);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(20);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(21);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, valueOffset);
|
||||
fbb.addFloat64(2, object.carbsRatio);
|
||||
fbb.addFloat64(3, object.portionSize);
|
||||
fbb.addFloat64(5, object.bolus);
|
||||
fbb.addOffset(8, notesOffset);
|
||||
fbb.addInt64(9, object.logEntry.targetId);
|
||||
fbb.addInt64(10, object.meal.targetId);
|
||||
@ -1479,6 +1591,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addBool(16, object.deleted);
|
||||
fbb.addFloat64(17, object.amount);
|
||||
fbb.addFloat64(18, object.totalCarbs);
|
||||
fbb.addOffset(19, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1501,9 +1614,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
totalCarbs: const fb.Float64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 40),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 20))
|
||||
..bolus = const fb.Float64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 14);
|
||||
.vTableGetNullable(buffer, rootOffset, 20),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 42));
|
||||
object.logEntry.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 22, 0);
|
||||
object.logEntry.attach(store);
|
||||
@ -1545,7 +1658,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
final valueOffset = fbb.writeString(object.value);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(16);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(17);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, valueOffset);
|
||||
fbb.addFloat64(2, object.carbsRatio);
|
||||
@ -1560,6 +1675,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addInt64(12, object.carbsRatioAccuracy.targetId);
|
||||
fbb.addBool(13, object.deleted);
|
||||
fbb.addFloat64(14, object.delayedBolusPercentage);
|
||||
fbb.addOffset(15, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1584,7 +1700,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
delayedBolusPercentage: const fb.Float64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 32),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 18));
|
||||
.vTableGetNullable(buffer, rootOffset, 18),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 34));
|
||||
object.mealSource.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 20, 0);
|
||||
object.mealSource.attach(store);
|
||||
@ -1614,11 +1732,14 @@ ModelDefinition getObjectBoxModel() {
|
||||
final valueOffset = fbb.writeString(object.value);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(5);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(6);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, valueOffset);
|
||||
fbb.addOffset(2, notesOffset);
|
||||
fbb.addBool(3, object.deleted);
|
||||
fbb.addOffset(4, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1633,7 +1754,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
value: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGet(buffer, rootOffset, 6, ''),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 8));
|
||||
.vTableGetNullable(buffer, rootOffset, 8),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 12));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -1649,11 +1772,14 @@ ModelDefinition getObjectBoxModel() {
|
||||
final valueOffset = fbb.writeString(object.value);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(5);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(6);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, valueOffset);
|
||||
fbb.addOffset(2, notesOffset);
|
||||
fbb.addBool(3, object.deleted);
|
||||
fbb.addOffset(4, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1668,7 +1794,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
value: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGet(buffer, rootOffset, 6, ''),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 8));
|
||||
.vTableGetNullable(buffer, rootOffset, 8),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 12));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -1689,7 +1817,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
final valueOffset = fbb.writeString(object.value);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(9);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(10);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, valueOffset);
|
||||
fbb.addOffset(2, notesOffset);
|
||||
@ -1698,6 +1828,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addInt64(5, object.defaultCarbsRatioAccuracy.targetId);
|
||||
fbb.addInt64(6, object.defaultPortionSizeAccuracy.targetId);
|
||||
fbb.addBool(7, object.deleted);
|
||||
fbb.addOffset(8, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1712,7 +1843,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
value: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGet(buffer, rootOffset, 6, ''),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 8));
|
||||
.vTableGetNullable(buffer, rootOffset, 8),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 20));
|
||||
object.defaultMealCategory.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 10, 0);
|
||||
object.defaultMealCategory.attach(store);
|
||||
@ -1739,7 +1872,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
objectToFB: (LogBolus object, fb.Builder fbb) {
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(19);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(20);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addFloat64(1, object.units);
|
||||
fbb.addFloat64(2, object.carbs);
|
||||
@ -1756,6 +1891,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addFloat64(15, object.mmolPerLCurrent);
|
||||
fbb.addFloat64(16, object.mmolPerLTarget);
|
||||
fbb.addFloat64(17, object.mmolPerLCorrection);
|
||||
fbb.addOffset(18, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1788,7 +1924,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
setManually: const fb.BoolReader()
|
||||
.vTableGet(buffer, rootOffset, 16, false),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 18));
|
||||
.vTableGetNullable(buffer, rootOffset, 18),
|
||||
source:
|
||||
const fb.StringReader(asciiOptimization: true).vTableGetNullable(buffer, rootOffset, 40));
|
||||
object.logEntry.targetId =
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 20, 0);
|
||||
object.logEntry.attach(store);
|
||||
@ -1812,7 +1950,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
final valueOffset = fbb.writeString(object.value);
|
||||
final notesOffset =
|
||||
object.notes == null ? null : fbb.writeString(object.notes!);
|
||||
fbb.startTable(8);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(9);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, valueOffset);
|
||||
fbb.addBool(2, object.forCarbsRatio);
|
||||
@ -1820,6 +1960,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addInt64(4, object.confidenceRating);
|
||||
fbb.addOffset(5, notesOffset);
|
||||
fbb.addBool(6, object.deleted);
|
||||
fbb.addOffset(7, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1840,7 +1981,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
confidenceRating: const fb.Int64Reader()
|
||||
.vTableGetNullable(buffer, rootOffset, 12),
|
||||
notes: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 14));
|
||||
.vTableGetNullable(buffer, rootOffset, 14),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 18));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -1861,7 +2004,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
final longTimeFormatOffset = object.longTimeFormat == null
|
||||
? null
|
||||
: fbb.writeString(object.longTimeFormat!);
|
||||
fbb.startTable(28);
|
||||
fbb.startTable(29);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addOffset(1, dateFormatOffset);
|
||||
fbb.addOffset(2, longDateFormatOffset);
|
||||
@ -1880,13 +2023,15 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addFloat64(24, object.nutritionIncrements);
|
||||
fbb.addFloat64(25, object.mmolPerLIncrements);
|
||||
fbb.addFloat64(26, object.amountIncrements);
|
||||
fbb.addInt64(27, object.lastExportTimestamp?.millisecondsSinceEpoch);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
objectFromFB: (Store store, ByteData fbData) {
|
||||
final buffer = fb.BufferContext(fbData);
|
||||
final rootOffset = buffer.derefObject(0);
|
||||
|
||||
final lastExportTimestampValue =
|
||||
const fb.Int64Reader().vTableGetNullable(buffer, rootOffset, 58);
|
||||
final object = Settings(
|
||||
id: const fb.Int64Reader().vTableGet(buffer, rootOffset, 4, 0),
|
||||
nutritionMeasurementIndex:
|
||||
@ -1916,7 +2061,8 @@ ModelDefinition getObjectBoxModel() {
|
||||
showConfirmationDialogOnStopEvent: const fb.BoolReader().vTableGet(buffer, rootOffset, 18, false),
|
||||
targetGlucoseMgPerDl: const fb.Int64Reader().vTableGet(buffer, rootOffset, 44, 0),
|
||||
targetGlucoseMmolPerL: const fb.Float64Reader().vTableGet(buffer, rootOffset, 46, 0),
|
||||
useDarkTheme: const fb.BoolReader().vTableGet(buffer, rootOffset, 48, false));
|
||||
useDarkTheme: const fb.BoolReader().vTableGet(buffer, rootOffset, 48, false),
|
||||
lastExportTimestamp: lastExportTimestampValue == null ? null : DateTime.fromMillisecondsSinceEpoch(lastExportTimestampValue));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -1929,7 +2075,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
object.id = id;
|
||||
},
|
||||
objectToFB: (GlucoseTarget object, fb.Builder fbb) {
|
||||
fbb.startTable(8);
|
||||
final sourceOffset =
|
||||
object.source == null ? null : fbb.writeString(object.source!);
|
||||
fbb.startTable(9);
|
||||
fbb.addInt64(0, object.id);
|
||||
fbb.addBool(1, object.deleted);
|
||||
fbb.addInt64(2, object.fromMgPerDL);
|
||||
@ -1937,6 +2085,7 @@ ModelDefinition getObjectBoxModel() {
|
||||
fbb.addFloat64(4, object.fromMmolPerL);
|
||||
fbb.addFloat64(5, object.toMmolPerL);
|
||||
fbb.addInt64(6, object.color);
|
||||
fbb.addOffset(7, sourceOffset);
|
||||
fbb.finish(fbb.endTable());
|
||||
return object.id;
|
||||
},
|
||||
@ -1957,7 +2106,9 @@ ModelDefinition getObjectBoxModel() {
|
||||
toMmolPerL:
|
||||
const fb.Float64Reader().vTableGet(buffer, rootOffset, 14, 0),
|
||||
color:
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 16, 0));
|
||||
const fb.Int64Reader().vTableGet(buffer, rootOffset, 16, 0),
|
||||
source: const fb.StringReader(asciiOptimization: true)
|
||||
.vTableGetNullable(buffer, rootOffset, 18));
|
||||
|
||||
return object;
|
||||
}),
|
||||
@ -2105,6 +2256,9 @@ class Basal_ {
|
||||
/// see [Basal.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<Basal>(_entities[0].properties[5]);
|
||||
|
||||
/// see [Basal.source]
|
||||
static final source = QueryStringProperty<Basal>(_entities[0].properties[6]);
|
||||
}
|
||||
|
||||
/// [BasalProfile] entity fields to define ObjectBox queries.
|
||||
@ -2128,6 +2282,10 @@ class BasalProfile_ {
|
||||
/// see [BasalProfile.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<BasalProfile>(_entities[1].properties[4]);
|
||||
|
||||
/// see [BasalProfile.source]
|
||||
static final source =
|
||||
QueryStringProperty<BasalProfile>(_entities[1].properties[5]);
|
||||
}
|
||||
|
||||
/// [Bolus] entity fields to define ObjectBox queries.
|
||||
@ -2164,6 +2322,9 @@ class Bolus_ {
|
||||
/// see [Bolus.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<Bolus>(_entities[2].properties[8]);
|
||||
|
||||
/// see [Bolus.source]
|
||||
static final source = QueryStringProperty<Bolus>(_entities[2].properties[9]);
|
||||
}
|
||||
|
||||
/// [BolusProfile] entity fields to define ObjectBox queries.
|
||||
@ -2187,6 +2348,10 @@ class BolusProfile_ {
|
||||
/// see [BolusProfile.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<BolusProfile>(_entities[3].properties[4]);
|
||||
|
||||
/// see [BolusProfile.source]
|
||||
static final source =
|
||||
QueryStringProperty<BolusProfile>(_entities[3].properties[5]);
|
||||
}
|
||||
|
||||
/// [LogEntry] entity fields to define ObjectBox queries.
|
||||
@ -2217,6 +2382,10 @@ class LogEntry_ {
|
||||
/// see [LogEntry.glucoseTrend]
|
||||
static final glucoseTrend =
|
||||
QueryDoubleProperty<LogEntry>(_entities[4].properties[6]);
|
||||
|
||||
/// see [LogEntry.source]
|
||||
static final source =
|
||||
QueryStringProperty<LogEntry>(_entities[4].properties[7]);
|
||||
}
|
||||
|
||||
/// [LogEvent] entity fields to define ObjectBox queries.
|
||||
@ -2259,6 +2428,10 @@ class LogEvent_ {
|
||||
/// see [LogEvent.reminderDuration]
|
||||
static final reminderDuration =
|
||||
QueryIntegerProperty<LogEvent>(_entities[5].properties[9]);
|
||||
|
||||
/// see [LogEvent.source]
|
||||
static final source =
|
||||
QueryStringProperty<LogEvent>(_entities[5].properties[10]);
|
||||
}
|
||||
|
||||
/// [LogEventType] entity fields to define ObjectBox queries.
|
||||
@ -2294,6 +2467,10 @@ class LogEventType_ {
|
||||
/// see [LogEventType.basalProfile]
|
||||
static final basalProfile = QueryRelationToOne<LogEventType, BasalProfile>(
|
||||
_entities[6].properties[7]);
|
||||
|
||||
/// see [LogEventType.source]
|
||||
static final source =
|
||||
QueryStringProperty<LogEventType>(_entities[6].properties[8]);
|
||||
}
|
||||
|
||||
/// [LogMeal] entity fields to define ObjectBox queries.
|
||||
@ -2312,51 +2489,52 @@ class LogMeal_ {
|
||||
static final portionSize =
|
||||
QueryDoubleProperty<LogMeal>(_entities[7].properties[3]);
|
||||
|
||||
/// see [LogMeal.bolus]
|
||||
static final bolus = QueryDoubleProperty<LogMeal>(_entities[7].properties[4]);
|
||||
|
||||
/// see [LogMeal.notes]
|
||||
static final notes = QueryStringProperty<LogMeal>(_entities[7].properties[5]);
|
||||
static final notes = QueryStringProperty<LogMeal>(_entities[7].properties[4]);
|
||||
|
||||
/// see [LogMeal.logEntry]
|
||||
static final logEntry =
|
||||
QueryRelationToOne<LogMeal, LogEntry>(_entities[7].properties[6]);
|
||||
QueryRelationToOne<LogMeal, LogEntry>(_entities[7].properties[5]);
|
||||
|
||||
/// see [LogMeal.meal]
|
||||
static final meal =
|
||||
QueryRelationToOne<LogMeal, Meal>(_entities[7].properties[7]);
|
||||
QueryRelationToOne<LogMeal, Meal>(_entities[7].properties[6]);
|
||||
|
||||
/// see [LogMeal.mealSource]
|
||||
static final mealSource =
|
||||
QueryRelationToOne<LogMeal, MealSource>(_entities[7].properties[8]);
|
||||
QueryRelationToOne<LogMeal, MealSource>(_entities[7].properties[7]);
|
||||
|
||||
/// see [LogMeal.mealCategory]
|
||||
static final mealCategory =
|
||||
QueryRelationToOne<LogMeal, MealCategory>(_entities[7].properties[9]);
|
||||
QueryRelationToOne<LogMeal, MealCategory>(_entities[7].properties[8]);
|
||||
|
||||
/// see [LogMeal.mealPortionType]
|
||||
static final mealPortionType =
|
||||
QueryRelationToOne<LogMeal, MealPortionType>(_entities[7].properties[10]);
|
||||
QueryRelationToOne<LogMeal, MealPortionType>(_entities[7].properties[9]);
|
||||
|
||||
/// see [LogMeal.portionSizeAccuracy]
|
||||
static final portionSizeAccuracy =
|
||||
QueryRelationToOne<LogMeal, Accuracy>(_entities[7].properties[11]);
|
||||
QueryRelationToOne<LogMeal, Accuracy>(_entities[7].properties[10]);
|
||||
|
||||
/// see [LogMeal.carbsRatioAccuracy]
|
||||
static final carbsRatioAccuracy =
|
||||
QueryRelationToOne<LogMeal, Accuracy>(_entities[7].properties[12]);
|
||||
QueryRelationToOne<LogMeal, Accuracy>(_entities[7].properties[11]);
|
||||
|
||||
/// see [LogMeal.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<LogMeal>(_entities[7].properties[13]);
|
||||
QueryBooleanProperty<LogMeal>(_entities[7].properties[12]);
|
||||
|
||||
/// see [LogMeal.amount]
|
||||
static final amount =
|
||||
QueryDoubleProperty<LogMeal>(_entities[7].properties[14]);
|
||||
QueryDoubleProperty<LogMeal>(_entities[7].properties[13]);
|
||||
|
||||
/// see [LogMeal.totalCarbs]
|
||||
static final totalCarbs =
|
||||
QueryDoubleProperty<LogMeal>(_entities[7].properties[15]);
|
||||
QueryDoubleProperty<LogMeal>(_entities[7].properties[14]);
|
||||
|
||||
/// see [LogMeal.source]
|
||||
static final source =
|
||||
QueryStringProperty<LogMeal>(_entities[7].properties[15]);
|
||||
}
|
||||
|
||||
/// [Meal] entity fields to define ObjectBox queries.
|
||||
@ -2413,6 +2591,9 @@ class Meal_ {
|
||||
/// see [Meal.delayedBolusPercentage]
|
||||
static final delayedBolusPercentage =
|
||||
QueryDoubleProperty<Meal>(_entities[8].properties[13]);
|
||||
|
||||
/// see [Meal.source]
|
||||
static final source = QueryStringProperty<Meal>(_entities[8].properties[14]);
|
||||
}
|
||||
|
||||
/// [MealCategory] entity fields to define ObjectBox queries.
|
||||
@ -2432,6 +2613,10 @@ class MealCategory_ {
|
||||
/// see [MealCategory.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<MealCategory>(_entities[9].properties[3]);
|
||||
|
||||
/// see [MealCategory.source]
|
||||
static final source =
|
||||
QueryStringProperty<MealCategory>(_entities[9].properties[4]);
|
||||
}
|
||||
|
||||
/// [MealPortionType] entity fields to define ObjectBox queries.
|
||||
@ -2451,6 +2636,10 @@ class MealPortionType_ {
|
||||
/// see [MealPortionType.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<MealPortionType>(_entities[10].properties[3]);
|
||||
|
||||
/// see [MealPortionType.source]
|
||||
static final source =
|
||||
QueryStringProperty<MealPortionType>(_entities[10].properties[4]);
|
||||
}
|
||||
|
||||
/// [MealSource] entity fields to define ObjectBox queries.
|
||||
@ -2487,6 +2676,10 @@ class MealSource_ {
|
||||
/// see [MealSource.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<MealSource>(_entities[11].properties[7]);
|
||||
|
||||
/// see [MealSource.source]
|
||||
static final source =
|
||||
QueryStringProperty<MealSource>(_entities[11].properties[8]);
|
||||
}
|
||||
|
||||
/// [LogBolus] entity fields to define ObjectBox queries.
|
||||
@ -2553,6 +2746,10 @@ class LogBolus_ {
|
||||
/// see [LogBolus.mmolPerLCorrection]
|
||||
static final mmolPerLCorrection =
|
||||
QueryDoubleProperty<LogBolus>(_entities[12].properties[15]);
|
||||
|
||||
/// see [LogBolus.source]
|
||||
static final source =
|
||||
QueryStringProperty<LogBolus>(_entities[12].properties[16]);
|
||||
}
|
||||
|
||||
/// [Accuracy] entity fields to define ObjectBox queries.
|
||||
@ -2583,6 +2780,10 @@ class Accuracy_ {
|
||||
/// see [Accuracy.deleted]
|
||||
static final deleted =
|
||||
QueryBooleanProperty<Accuracy>(_entities[13].properties[6]);
|
||||
|
||||
/// see [Accuracy.source]
|
||||
static final source =
|
||||
QueryStringProperty<Accuracy>(_entities[13].properties[7]);
|
||||
}
|
||||
|
||||
/// [Settings] entity fields to define ObjectBox queries.
|
||||
@ -2657,6 +2858,10 @@ class Settings_ {
|
||||
/// see [Settings.amountIncrements]
|
||||
static final amountIncrements =
|
||||
QueryDoubleProperty<Settings>(_entities[14].properties[17]);
|
||||
|
||||
/// see [Settings.lastExportTimestamp]
|
||||
static final lastExportTimestamp =
|
||||
QueryIntegerProperty<Settings>(_entities[14].properties[18]);
|
||||
}
|
||||
|
||||
/// [GlucoseTarget] entity fields to define ObjectBox queries.
|
||||
@ -2688,6 +2893,10 @@ class GlucoseTarget_ {
|
||||
/// see [GlucoseTarget.color]
|
||||
static final color =
|
||||
QueryIntegerProperty<GlucoseTarget>(_entities[15].properties[6]);
|
||||
|
||||
/// see [GlucoseTarget.source]
|
||||
static final source =
|
||||
QueryStringProperty<GlucoseTarget>(_entities[15].properties[7]);
|
||||
}
|
||||
|
||||
/// [Recipe] entity fields to define ObjectBox queries.
|
||||
|
@ -1,6 +1,7 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/components/forms/time_of_day_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
@ -10,6 +11,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/basal.dart';
|
||||
import 'package:diameter/models/basal_profile.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BasalDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/basal';
|
||||
@ -130,7 +132,7 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
_startTime.hour == other.startTime.hour &&
|
||||
_startTime.minute == other.startTime.minute)
|
||||
.isNotEmpty) {
|
||||
error = 'There\'s already a rate with this start time.';
|
||||
error = translate(LocalizationKeys.basal_warnings_duplicate);
|
||||
}
|
||||
|
||||
if (basalRates
|
||||
@ -141,7 +143,7 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
DateTimeUtils.convertTimeOfDayToDateTime(_endTime)
|
||||
.isAfter(other.startTime))
|
||||
.isNotEmpty) {
|
||||
error = 'This rate\'s time period overlaps with another one.';
|
||||
error = translate(LocalizationKeys.basal_warnings_overlap);
|
||||
}
|
||||
|
||||
return error == null
|
||||
@ -154,11 +156,11 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
actions: <Widget>[
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 'CANCEL'),
|
||||
child: const Text('GO BACK TO EDITING'),
|
||||
child: Text(translate(LocalizationKeys.general_keepEditing).toUpperCase()),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 'CONFIRM'),
|
||||
child: const Text('SAVE AS IS'),
|
||||
child: Text(translate(LocalizationKeys.general_saveAsIs).toUpperCase()),
|
||||
),
|
||||
],
|
||||
);
|
||||
@ -196,13 +198,26 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
).then((result) {
|
||||
Navigator.pop(
|
||||
context,
|
||||
['New Basal Rate${result[1] != null ? 's' : ''} saved', basal] +
|
||||
[result[1]],
|
||||
[
|
||||
translatePlural(
|
||||
LocalizationKeys.basal_saved,
|
||||
result.length,
|
||||
args: {
|
||||
"status": _isNew ? '${LocalizationKeys.basal_new} ' : ''
|
||||
},
|
||||
),
|
||||
basal] + [result[1]],
|
||||
);
|
||||
});
|
||||
} else {
|
||||
Navigator.pop(
|
||||
context, ['${_isNew ? 'New' : ''} Basal Rate saved', basal]);
|
||||
context, [translatePlural(
|
||||
LocalizationKeys.basal_saved,
|
||||
1,
|
||||
args: {
|
||||
"status": _isNew ? '${LocalizationKeys.basal_new} ' : ''
|
||||
},
|
||||
), basal]);
|
||||
}
|
||||
}
|
||||
});
|
||||
@ -241,7 +256,14 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(
|
||||
'${_isNew ? 'New' : 'Edit'} Basal Rate for ${BasalProfile.get(widget.basalProfileId)?.name}'),
|
||||
translate(
|
||||
LocalizationKeys.basal_title,
|
||||
args: {
|
||||
"status": _isNew ? LocalizationKeys.basal_new : LocalizationKeys.general_edit,
|
||||
"profileName": BasalProfile.get(widget.basalProfileId)?.name,
|
||||
}
|
||||
),
|
||||
),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: BasalDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
@ -259,7 +281,7 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(right: 5),
|
||||
child: TimeOfDayFormField(
|
||||
label: 'Start Time',
|
||||
label: translate(LocalizationKeys.basal_fields_startTime),
|
||||
controller: _startTimeController,
|
||||
time: _startTime,
|
||||
onChanged: updateStartTime,
|
||||
@ -270,7 +292,7 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 5),
|
||||
child: TimeOfDayFormField(
|
||||
label: 'End Time',
|
||||
label: translate(LocalizationKeys.basal_fields_endTime),
|
||||
controller: _endTimeController,
|
||||
time: _endTime,
|
||||
onChanged: updateEndTime,
|
||||
@ -281,8 +303,8 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
),
|
||||
NumberFormField(
|
||||
controller: _unitsController,
|
||||
label: 'Units',
|
||||
suffix: 'U',
|
||||
label: translate(LocalizationKeys.basal_fields_units),
|
||||
suffix: translate(LocalizationKeys.general_suffixes_units),
|
||||
autoRoundToMultipleOfStep: true,
|
||||
step: Settings.insulinSteps,
|
||||
onChanged: (value) {
|
||||
@ -291,7 +313,7 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
Utils.toStringMatchingTemplateFractionPrecision(
|
||||
value, Settings.insulinSteps);
|
||||
}
|
||||
}),
|
||||
})
|
||||
],
|
||||
),
|
||||
],
|
||||
@ -305,8 +327,8 @@ class _BasalDetailScreenState extends State<BasalDetailScreen> {
|
||||
onMiddleAction: _isSaving || _isFinalRate
|
||||
? null
|
||||
: () => handleSaveAction(next: false),
|
||||
actionText: _isFinalRate ? 'SAVE & CLOSE' : 'NEXT',
|
||||
middleActionText: 'SAVE & CLOSE',
|
||||
actionTextKey: translate(_isFinalRate ? LocalizationKeys.general_saveAndClose : LocalizationKeys.general_next).toUpperCase(),
|
||||
middleActionTextKey: translate(LocalizationKeys.general_saveAndClose).toUpperCase(),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
@ -1,3 +1,4 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
@ -5,6 +6,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/basal.dart';
|
||||
import 'package:diameter/models/basal_profile.dart';
|
||||
import 'package:diameter/screens/basal/basal_detail.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BasalListScreen extends StatefulWidget {
|
||||
final BasalProfile basalProfile;
|
||||
@ -61,7 +63,7 @@ class _BasalListScreenState extends State<BasalListScreen> {
|
||||
|
||||
void onDelete(Basal basal) {
|
||||
Basal.remove(basal.id);
|
||||
reload(message: 'Basal Rate deleted');
|
||||
reload(message: translate(LocalizationKeys.basal_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(Basal basal) async {
|
||||
@ -69,7 +71,7 @@ class _BasalListScreenState extends State<BasalListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(basal),
|
||||
message: 'Are you sure you want to delete this Basal Rate?',
|
||||
message: translate(LocalizationKeys.basal_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(basal);
|
||||
@ -83,26 +85,26 @@ class _BasalListScreenState extends State<BasalListScreen> {
|
||||
// check for gaps
|
||||
if (index == 0 &&
|
||||
(basal.startTime.hour != 0 || basal.startTime.minute != 0)) {
|
||||
return 'First Basal of the day needs to start at 00:00';
|
||||
return translate(LocalizationKeys.basal_warnings_startTimeFirst);
|
||||
}
|
||||
|
||||
if (index > 0) {
|
||||
var lastEndTime = basalRates[index - 1].endTime;
|
||||
if (basal.startTime.isAfter(lastEndTime)) {
|
||||
return 'There\'s a time gap between this and the previous rate';
|
||||
return translate(LocalizationKeys.basal_warnings_gap);
|
||||
}
|
||||
}
|
||||
|
||||
if (index == basalRates.length - 1 &&
|
||||
(basal.endTime.hour != 0 || basal.endTime.minute != 0)) {
|
||||
return 'Last Basal of the day needs to end at 00:00';
|
||||
return translate(LocalizationKeys.basal_warnings_endTimeLast);
|
||||
}
|
||||
|
||||
// check for duplicates
|
||||
if (basalRates
|
||||
.where((other) => basal != other && basal.startTime == other.startTime)
|
||||
.isNotEmpty) {
|
||||
return 'There are multiple rates with this start time';
|
||||
return translate(LocalizationKeys.basal_warnings_duplicate);
|
||||
}
|
||||
|
||||
if (basalRates
|
||||
@ -110,7 +112,7 @@ class _BasalListScreenState extends State<BasalListScreen> {
|
||||
basal.startTime.isBefore(other.startTime) &&
|
||||
basal.endTime.isAfter(other.startTime))
|
||||
.isNotEmpty) {
|
||||
return 'This rate\'s time period overlaps with another one';
|
||||
return translate(LocalizationKeys.basal_warnings_overlap);
|
||||
}
|
||||
|
||||
return null;
|
||||
@ -158,7 +160,7 @@ class _BasalListScreenState extends State<BasalListScreen> {
|
||||
child: Text(
|
||||
'${DateTimeUtils.displayTime(basal.startTime)} - ${DateTimeUtils.displayTime(basal.endTime)}')),
|
||||
const Spacer(),
|
||||
Expanded(child: Text('${basal.units} U')),
|
||||
Expanded(child: Text('${basal.units} ${translate(LocalizationKeys.general_suffixes_units)}')),
|
||||
],
|
||||
),
|
||||
trailing: Row(
|
||||
@ -180,8 +182,8 @@ class _BasalListScreenState extends State<BasalListScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
child: Text('You have not created any Basal Rates yet!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.basal_empty)),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,6 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/basal.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
@ -9,6 +10,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/basal_profile.dart';
|
||||
import 'package:diameter/screens/basal/basal_list.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BasalProfileDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/basal-profile';
|
||||
@ -133,21 +135,30 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
content: const Text(
|
||||
'There are already one or more active profiles. What would you like to do?'),
|
||||
content: Text(translate(LocalizationKeys.basalProfile_warnings_activeAlreadySet)),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 0),
|
||||
child: const Text('IGNORE'),
|
||||
child: Text(translate(LocalizationKeys.basalProfile_warnings_resolve_ignore).toUpperCase()),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 1),
|
||||
child:
|
||||
Text('DEACTIVATE ${_nameController.text.toUpperCase()}'),
|
||||
Text(
|
||||
translate(
|
||||
LocalizationKeys.basalProfile_warnings_resolve_deactivateProfile,
|
||||
args: {
|
||||
"profileName": _nameController.text,
|
||||
}
|
||||
).toUpperCase()
|
||||
),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 2),
|
||||
child: const Text('DEACTIVATE ALL OTHERS'),
|
||||
child: Text(
|
||||
translate(
|
||||
LocalizationKeys.basalProfile_warnings_resolve_deactivateOthers,
|
||||
).toUpperCase()),
|
||||
)
|
||||
],
|
||||
);
|
||||
@ -167,16 +178,25 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
content: const Text(
|
||||
'There is currently no active profile. Would you like to set this one as active?'),
|
||||
content: Text(
|
||||
translate(
|
||||
LocalizationKeys.basalProfile_warnings_noActiveOnCreate,
|
||||
).toUpperCase()),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 0),
|
||||
child: const Text('IGNORE'),
|
||||
child: Text(
|
||||
translate(
|
||||
LocalizationKeys.basalProfile_warnings_resolve_ignore,
|
||||
).toUpperCase()),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 1),
|
||||
child: const Text('ACTIVATE THIS PROFILE'),
|
||||
child: Text(
|
||||
translate(
|
||||
LocalizationKeys.basalProfile_warnings_resolve_activateCurrent,
|
||||
).toUpperCase()
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
@ -249,7 +269,11 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
|
||||
if (close) {
|
||||
Navigator.pop(context,
|
||||
['${_isNew ? 'New' : ''} Basal Profile saved', basalProfile]);
|
||||
[
|
||||
translate(LocalizationKeys.basalProfile_saved, args: {
|
||||
"status": _isNew ? '${translate(LocalizationKeys.basalProfile_new)} ' : ''
|
||||
}),
|
||||
]);
|
||||
} else {
|
||||
if (_isNew) {
|
||||
Navigator.push(
|
||||
@ -260,7 +284,7 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
),
|
||||
).then((result) => Navigator.pop(context, result));
|
||||
} else {
|
||||
reload(message: 'Basal Profile saved');
|
||||
reload(message: translate(LocalizationKeys.basalProfile_saved));
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -329,12 +353,12 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
fields: [
|
||||
TextFormField(
|
||||
controller: _nameController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.basalProfile_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty title';
|
||||
return translate(LocalizationKeys.basalProfile_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
@ -342,8 +366,8 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
TextFormField(
|
||||
keyboardType: TextInputType.multiline,
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.basalProfile_fields_notes),
|
||||
),
|
||||
minLines: 2,
|
||||
maxLines: 5,
|
||||
@ -355,7 +379,7 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
_active = value;
|
||||
});
|
||||
},
|
||||
label: 'active',
|
||||
label: translate(LocalizationKeys.basalProfile_fields_active),
|
||||
),
|
||||
],
|
||||
),
|
||||
@ -374,13 +398,24 @@ class _BasalProfileDetailScreenState extends State<BasalProfileDetailScreen> {
|
||||
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Basal Profile' : _basalProfile!.name),
|
||||
title: Text(translate(LocalizationKeys.basalProfile_detail_title, args: {
|
||||
"status": _isNew ? '${translate(LocalizationKeys.basalProfile_new)} ' : '',
|
||||
"profileName": _basalProfile!.name,
|
||||
})),
|
||||
bottom: _isNew
|
||||
? PreferredSize(child: Container(), preferredSize: Size.zero)
|
||||
: const TabBar(
|
||||
: TabBar(
|
||||
tabs: [
|
||||
Tab(text: 'PROFILE'),
|
||||
Tab(text: 'RATES'),
|
||||
Tab(
|
||||
text: translate(
|
||||
LocalizationKeys.basalProfile_detail_tabs_profile,
|
||||
).toUpperCase(),
|
||||
),
|
||||
Tab(
|
||||
text: translate(
|
||||
LocalizationKeys.basalProfile_detail_tabs_rates,
|
||||
).toUpperCase(),
|
||||
),
|
||||
],
|
||||
),
|
||||
actions: appBarActions,
|
||||
|
@ -1,3 +1,4 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/models/basal.dart';
|
||||
@ -6,6 +7,7 @@ import 'package:diameter/navigation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/basal_profile.dart';
|
||||
import 'package:diameter/screens/basal/basal_profile_detail.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BasalProfileListScreen extends StatefulWidget {
|
||||
static const String routeName = '/basal-profiles';
|
||||
@ -59,26 +61,33 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
banner = activeProfileCount != 1
|
||||
? MaterialBanner(
|
||||
content: Text(activeProfileCount == 0
|
||||
? 'You currently do not have an active Basal Profile.'
|
||||
: 'More than one active Basal Profile has been found.'),
|
||||
? translate(LocalizationKeys.basalProfile_warnings_noActive)
|
||||
: translate(
|
||||
LocalizationKeys.basalProfile_warnings_multipleActive)),
|
||||
leading: const CircleAvatar(child: Icon(Icons.warning)),
|
||||
forceActionsBelow: true,
|
||||
actions: activeProfileCount == 0
|
||||
? [
|
||||
_basalProfiles.isNotEmpty
|
||||
? TextButton(
|
||||
child: const Text('ACTIVATE A PROFILE'),
|
||||
child: Text(translate(LocalizationKeys
|
||||
.basalProfile_warnings_resolve_activate)
|
||||
.toUpperCase()),
|
||||
onPressed: handlePickActiveProfileAction,
|
||||
)
|
||||
: Container(),
|
||||
TextButton(
|
||||
child: const Text('CREATE A NEW PROFILE'),
|
||||
child: Text(translate(LocalizationKeys
|
||||
.basalProfile_warnings_resolve_create)
|
||||
.toUpperCase()),
|
||||
onPressed: () => onNew(true),
|
||||
),
|
||||
]
|
||||
: [
|
||||
TextButton(
|
||||
child: const Text('PICK A PROFILE'),
|
||||
child: Text(translate(LocalizationKeys
|
||||
.basalProfile_warnings_resolve_pick)
|
||||
.toUpperCase()),
|
||||
onPressed: handlePickActiveProfileAction,
|
||||
),
|
||||
],
|
||||
@ -90,8 +99,9 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
void handleDuplicateAction(BasalProfile basalProfile) async {
|
||||
final copy = BasalProfile(
|
||||
active: false,
|
||||
name: 'Copy of ${basalProfile.name}',
|
||||
);
|
||||
name: translate(LocalizationKeys.basalProfile_copyOf, args: {
|
||||
"profileName": basalProfile.name,
|
||||
}));
|
||||
BasalProfile.put(copy);
|
||||
|
||||
final rates = Basal.getAllForProfile(basalProfile.id);
|
||||
@ -105,12 +115,19 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
Basal.put(basal);
|
||||
}
|
||||
|
||||
reload(message: 'Added copy of ${basalProfile.name}');
|
||||
reload(
|
||||
message: translate(
|
||||
LocalizationKeys.basalProfile_warnings_resolve_ignore,
|
||||
args: {"profileName": basalProfile.name},
|
||||
));
|
||||
}
|
||||
|
||||
void onDelete(BasalProfile basalProfile) {
|
||||
BasalProfile.remove(basalProfile.id);
|
||||
reload(message: 'Basal Profile deleted');
|
||||
reload(
|
||||
message: translate(
|
||||
LocalizationKeys.basalProfile_deleted,
|
||||
));
|
||||
}
|
||||
|
||||
void handleDeleteAction(BasalProfile basalProfile) async {
|
||||
@ -118,7 +135,7 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(basalProfile),
|
||||
message: 'Are you sure you want to delete this Basal Profile?',
|
||||
message: translate(LocalizationKeys.basalProfile_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(basalProfile);
|
||||
@ -131,7 +148,8 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
basalProfile.active = true;
|
||||
BasalProfile.put(basalProfile);
|
||||
reload(
|
||||
message: '${basalProfile.name} has been set as your active Profile');
|
||||
message: translate(LocalizationKeys.basalProfile_activated,
|
||||
args: {"profileName": basalProfile.name}));
|
||||
}
|
||||
}
|
||||
|
||||
@ -141,14 +159,16 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
content: AutoCompleteDropdownButton(
|
||||
controller: TextEditingController(text: ''),
|
||||
items: _basalProfiles,
|
||||
label: 'Default Basal Profile',
|
||||
label: translate(LocalizationKeys.basalProfile_default),
|
||||
onChanged: onPickActive,
|
||||
),
|
||||
leading: const CircleAvatar(child: Icon(Icons.info)),
|
||||
forceActionsBelow: true,
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('CREATE A NEW PROFILE INSTEAD'),
|
||||
child: Text(translate(LocalizationKeys
|
||||
.basalProfile_warnings_resolve_createInstead)
|
||||
.toUpperCase()),
|
||||
onPressed: () => onNew(true),
|
||||
),
|
||||
],
|
||||
@ -178,7 +198,7 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Basal Profiles'),
|
||||
title: Text(translate(LocalizationKeys.basalProfile_title)),
|
||||
actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
],
|
||||
@ -202,9 +222,9 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
double dailyTotal =
|
||||
Basal.getDailyTotalForProfile(basalProfile.id);
|
||||
String activeProfileText = basalProfile.active
|
||||
? ' (Default Profile)'
|
||||
? '(${translate(LocalizationKeys.basalProfile_default)})'
|
||||
: basalProfile.id == _activeProfile?.id
|
||||
? ' (Current Active Profile)'
|
||||
? ' (${translate(LocalizationKeys.basalProfile_active)})'
|
||||
: '';
|
||||
return Card(
|
||||
child: ListTile(
|
||||
@ -228,7 +248,9 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
? [
|
||||
Text(dailyTotal
|
||||
.toStringAsPrecision(3)),
|
||||
const Text('U/day',
|
||||
Text(
|
||||
translate(LocalizationKeys
|
||||
.general_suffixes_perDay),
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
: [],
|
||||
@ -263,8 +285,8 @@ class _BasalProfileListScreenState extends State<BasalProfileListScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
child: Text('You have not created any Basal Profiles yet!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.basalProfile_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,6 +1,7 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/components/forms/time_of_day_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
@ -10,6 +11,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/bolus.dart';
|
||||
import 'package:diameter/models/bolus_profile.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BolusDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/bolus';
|
||||
@ -140,7 +142,7 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
_startTime.hour == other.startTime.hour &&
|
||||
_startTime.minute == other.startTime.minute)
|
||||
.isNotEmpty) {
|
||||
error = 'There\'s already a rate with this start time.';
|
||||
error = translate(LocalizationKeys.bolus_warnings_duplicate);
|
||||
}
|
||||
|
||||
if (bolusRates
|
||||
@ -151,7 +153,7 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
DateTimeUtils.convertTimeOfDayToDateTime(_endTime)
|
||||
.isAfter(other.startTime))
|
||||
.isNotEmpty) {
|
||||
error = 'This rate\'s time period overlaps with another one.';
|
||||
error = translate(LocalizationKeys.bolus_warnings_overlap);
|
||||
}
|
||||
|
||||
return error == null
|
||||
@ -164,11 +166,11 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
actions: <Widget>[
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 'CANCEL'),
|
||||
child: const Text('GO BACK TO EDITING'),
|
||||
child: Text(translate(LocalizationKeys.general_keepEditing).toUpperCase()),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 'CONFIRM'),
|
||||
child: const Text('SAVE AS IS'),
|
||||
child: Text(translate(LocalizationKeys.general_saveAsIs).toUpperCase()),
|
||||
),
|
||||
],
|
||||
);
|
||||
@ -210,13 +212,27 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
).then((result) {
|
||||
Navigator.pop(
|
||||
context,
|
||||
['New Bolus Rate${result[1] != null ? 's' : ''} saved', bolus] +
|
||||
[
|
||||
translatePlural(
|
||||
LocalizationKeys.bolus_saved,
|
||||
result.length,
|
||||
args: {
|
||||
"status": _isNew ? '${LocalizationKeys.bolus_new} ' : ''
|
||||
},
|
||||
),
|
||||
bolus] +
|
||||
[result[1]],
|
||||
);
|
||||
});
|
||||
} else {
|
||||
Navigator.pop(
|
||||
context, ['${_isNew ? 'New' : ''} Bolus Rate saved', bolus]);
|
||||
context, [translatePlural(
|
||||
LocalizationKeys.bolus_saved,
|
||||
1,
|
||||
args: {
|
||||
"status": _isNew ? '${LocalizationKeys.bolus_new} ' : ''
|
||||
},
|
||||
), bolus]);
|
||||
}
|
||||
}
|
||||
});
|
||||
@ -288,7 +304,14 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(
|
||||
'${_isNew ? 'New' : 'Edit'} Bolus Rate for ${BolusProfile.get(widget.bolusProfileId)?.name}'),
|
||||
translate(
|
||||
LocalizationKeys.basal_title,
|
||||
args: {
|
||||
"status": _isNew ? LocalizationKeys.bolus_new : LocalizationKeys.general_edit,
|
||||
"profileName": BolusProfile.get(widget.bolusProfileId)?.name,
|
||||
}
|
||||
),
|
||||
),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: BolusDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
@ -306,7 +329,7 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(right: 5),
|
||||
child: TimeOfDayFormField(
|
||||
label: 'Start Time',
|
||||
label: translate(LocalizationKeys.bolus_fields_startTime),
|
||||
controller: _startTimeController,
|
||||
time: _startTime,
|
||||
onChanged: updateStartTime,
|
||||
@ -317,7 +340,7 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 5),
|
||||
child: TimeOfDayFormField(
|
||||
label: 'End Time',
|
||||
label: translate(LocalizationKeys.bolus_fields_endTime),
|
||||
controller: _endTimeController,
|
||||
time: _endTime,
|
||||
onChanged: updateEndTime,
|
||||
@ -328,8 +351,8 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
),
|
||||
NumberFormField(
|
||||
controller: _unitsController,
|
||||
label: 'Units',
|
||||
suffix: 'U',
|
||||
label: translate(LocalizationKeys.bolus_fields_units),
|
||||
suffix: translate(LocalizationKeys.general_suffixes_units),
|
||||
autoRoundToMultipleOfStep: true,
|
||||
step: Settings.insulinSteps,
|
||||
onChanged: (value) {
|
||||
@ -342,7 +365,7 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
),
|
||||
NumberFormField(
|
||||
controller: _carbsController,
|
||||
label: 'per carbs',
|
||||
label: translate(LocalizationKeys.bolus_fields_perCarbs),
|
||||
suffix: Settings.nutritionMeasurementSuffix,
|
||||
autoRoundToMultipleOfStep: true,
|
||||
step: Settings.nutritionSteps,
|
||||
@ -365,8 +388,10 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
? Expanded(
|
||||
flex: Settings.glucoseMeasurement == GlucoseMeasurement.mgPerDl ? 2 : 1,
|
||||
child: NumberFormField(
|
||||
label: 'per mg/dl',
|
||||
suffix: 'mg/dl',
|
||||
label: translate(LocalizationKeys.bolus_fields_perGlucose, args: {
|
||||
"glucoseMeasurement": Settings.glucoseMeasurementSuffix
|
||||
}),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
readOnly: Settings.glucoseMeasurement ==
|
||||
GlucoseMeasurement.mmolPerL,
|
||||
showSteppers: Settings.glucoseMeasurement == GlucoseMeasurement.mgPerDl,
|
||||
@ -384,8 +409,10 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
? Expanded(
|
||||
flex: Settings.glucoseMeasurement == GlucoseMeasurement.mmolPerL ? 2 : 1,
|
||||
child: NumberFormField(
|
||||
label: 'per mmol/l',
|
||||
suffix: 'mmol/l',
|
||||
label: translate(LocalizationKeys.bolus_fields_perGlucose, args: {
|
||||
"glucoseMeasurement": Settings.glucoseMeasurementSuffix
|
||||
}),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
readOnly: Settings.glucoseMeasurement ==
|
||||
GlucoseMeasurement.mgPerDl,
|
||||
showSteppers: Settings.glucoseMeasurement == GlucoseMeasurement.mmolPerL,
|
||||
@ -410,8 +437,8 @@ class _BolusDetailScreenState extends State<BolusDetailScreen> {
|
||||
onMiddleAction: _isSaving || _isFinalRate
|
||||
? null
|
||||
: () => handleSaveAction(next: false),
|
||||
actionText: _isFinalRate ? 'SAVE & CLOSE' : 'NEXT',
|
||||
middleActionText: 'SAVE & CLOSE',
|
||||
actionTextKey: _isFinalRate ? translate(LocalizationKeys.general_saveAndClose) : translate(LocalizationKeys.general_next),
|
||||
middleActionTextKey: translate(LocalizationKeys.general_saveAndClose),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
@ -1,3 +1,4 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
@ -5,6 +6,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/bolus.dart';
|
||||
import 'package:diameter/models/bolus_profile.dart';
|
||||
import 'package:diameter/screens/bolus/bolus_detail.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BolusListScreen extends StatefulWidget {
|
||||
final BolusProfile bolusProfile;
|
||||
@ -61,7 +63,7 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
|
||||
void onDelete(Bolus bolus) {
|
||||
Bolus.remove(bolus.id);
|
||||
reload(message: 'Bolus Rate deleted');
|
||||
reload(message: translate(LocalizationKeys.bolus_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(Bolus bolus) async {
|
||||
@ -69,7 +71,7 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(bolus),
|
||||
message: 'Are you sure you want to delete this Bolus Rate?',
|
||||
message: translate(LocalizationKeys.bolus_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(bolus);
|
||||
@ -82,26 +84,26 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
|
||||
if (index == 0 &&
|
||||
(bolus.startTime.toLocal().hour != 0 || bolus.startTime.minute != 0)) {
|
||||
return 'First Bolus of the day needs to start at 00:00';
|
||||
return translate(LocalizationKeys.bolus_warnings_startTimeFirst);
|
||||
}
|
||||
|
||||
if (index > 0) {
|
||||
var lastEndTime = bolusRates[index - 1].endTime;
|
||||
if (bolus.startTime.isAfter(lastEndTime)) {
|
||||
return 'There\'s a time gap between this and the previous rate';
|
||||
return translate(LocalizationKeys.bolus_warnings_gap);
|
||||
}
|
||||
}
|
||||
|
||||
if (index == bolusRates.length - 1 &&
|
||||
(bolus.endTime.toLocal().hour != 0 || bolus.endTime.minute != 0)) {
|
||||
return 'Last Bolus of the day needs to end at 00:00';
|
||||
return translate(LocalizationKeys.bolus_warnings_endTimeLast);
|
||||
}
|
||||
|
||||
// check for duplicates
|
||||
if (bolusRates
|
||||
.where((other) => bolus != other && bolus.startTime == other.startTime)
|
||||
.isNotEmpty) {
|
||||
return 'There are multiple rates with this start time';
|
||||
return translate(LocalizationKeys.bolus_warnings_duplicate);
|
||||
}
|
||||
|
||||
if (bolusRates
|
||||
@ -109,7 +111,7 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
bolus.startTime.isBefore(other.startTime) &&
|
||||
bolus.endTime.isAfter(other.startTime))
|
||||
.isNotEmpty) {
|
||||
return 'This rate\'s time period overlaps with another one';
|
||||
return translate(LocalizationKeys.bolus_warnings_overlap);
|
||||
}
|
||||
|
||||
return null;
|
||||
@ -162,8 +164,9 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
? [
|
||||
Text((bolus.carbs / bolus.units)
|
||||
.toStringAsPrecision(2)),
|
||||
Text(
|
||||
'${Settings.nutritionMeasurementSuffix} carbs per U',
|
||||
Text(translate(LocalizationKeys.general_suffixes_carbsPerU, args: {
|
||||
"nutritionMeasurementSuffix": Settings.nutritionMeasurementSuffix
|
||||
}),
|
||||
textAlign: TextAlign.center,
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
@ -176,7 +179,7 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
? [
|
||||
Text((bolus.units / bolus.carbs * 12)
|
||||
.toStringAsPrecision(2)),
|
||||
const Text('U per bread unit',
|
||||
Text(translate(LocalizationKeys.general_suffixes_uPerBreadUnit),
|
||||
textAlign: TextAlign.center,
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
@ -196,8 +199,9 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
: bolus.mmolPerL ?? 0)! /
|
||||
bolus.units))
|
||||
.toString()),
|
||||
Text(
|
||||
'${Settings.glucoseMeasurementSuffix} per unit',
|
||||
Text(translate(LocalizationKeys.general_suffixes_uPerGlucose, args: {
|
||||
"glucoseMeasurementSuffix": Settings.glucoseMeasurementSuffix
|
||||
}),
|
||||
textAlign: TextAlign.center,
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
@ -225,8 +229,8 @@ class _BolusListScreenState extends State<BolusListScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
child: Text('You have not created any Bolus Rates yet!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.bolus_title)),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,6 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/bolus.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
@ -9,6 +10,7 @@ import 'package:flutter/material.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/bolus_profile.dart';
|
||||
import 'package:diameter/screens/bolus/bolus_list.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BolusProfileDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/bolus-profile';
|
||||
@ -131,20 +133,19 @@ class _BolusProfileDetailScreenState extends State<BolusProfileDetailScreen> {
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
content: const Text(
|
||||
'There are already one or more active profiles. What would you like to do?'),
|
||||
content: Text(translate(LocalizationKeys.bolusProfile_warnings_activeAlreadySet)),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 0),
|
||||
child: const Text('IGNORE'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_ignore).toUpperCase()),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 1),
|
||||
child: const Text('DEACTIVATE THIS PROFILE'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_deactivateProfile).toUpperCase()),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 2),
|
||||
child: const Text('DEACTIVATE ALL OTHERS'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_deactivateOthers).toUpperCase()),
|
||||
)
|
||||
],
|
||||
);
|
||||
@ -164,16 +165,15 @@ class _BolusProfileDetailScreenState extends State<BolusProfileDetailScreen> {
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
content: const Text(
|
||||
'There is currently no active profile. Would you like to set this one as active?'),
|
||||
content: Text(translate(LocalizationKeys.bolusProfile_warnings_noActiveOnCreate)),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 0),
|
||||
child: const Text('IGNORE'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_ignore).toUpperCase()),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 1),
|
||||
child: const Text('ACTIVATE THIS PROFILE'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_activateCurrent).toUpperCase()),
|
||||
),
|
||||
],
|
||||
);
|
||||
@ -246,8 +246,15 @@ class _BolusProfileDetailScreenState extends State<BolusProfileDetailScreen> {
|
||||
BolusProfile.put(bolusProfile);
|
||||
|
||||
if (close) {
|
||||
Navigator.pop(context,
|
||||
['${_isNew ? 'New' : ''} Bolus Profile saved', bolusProfile]);
|
||||
Navigator.pop(context, [
|
||||
translate(
|
||||
LocalizationKeys.bolusProfile_saved,
|
||||
args: {
|
||||
"status": _isNew ? '${translate(LocalizationKeys.bolusProfile_new)} ' : ''
|
||||
},
|
||||
), bolusProfile
|
||||
],
|
||||
);
|
||||
} else {
|
||||
if (_isNew) {
|
||||
Navigator.push(
|
||||
@ -258,7 +265,7 @@ class _BolusProfileDetailScreenState extends State<BolusProfileDetailScreen> {
|
||||
),
|
||||
).then((result) => Navigator.pop(context, result));
|
||||
} else {
|
||||
reload(message: 'Bolus Profile saved');
|
||||
reload(message: translate(LocalizationKeys.bolusProfile_saved));
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -329,19 +336,19 @@ class _BolusProfileDetailScreenState extends State<BolusProfileDetailScreen> {
|
||||
fields: [
|
||||
TextFormField(
|
||||
controller: _nameController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.bolusProfile_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty title';
|
||||
return translate(LocalizationKeys.bolusProfile_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.bolusProfile_fields_notes),
|
||||
),
|
||||
controller: _notesController,
|
||||
keyboardType: TextInputType.multiline,
|
||||
@ -355,7 +362,7 @@ class _BolusProfileDetailScreenState extends State<BolusProfileDetailScreen> {
|
||||
_active = value;
|
||||
});
|
||||
},
|
||||
label: 'active',
|
||||
label: translate(LocalizationKeys.bolusProfile_fields_active),
|
||||
),
|
||||
],
|
||||
),
|
||||
@ -374,13 +381,16 @@ class _BolusProfileDetailScreenState extends State<BolusProfileDetailScreen> {
|
||||
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Bolus Profile' : _bolusProfile!.name),
|
||||
title: Text(translate(LocalizationKeys.bolusProfile_detail_title, args: {
|
||||
"status": _isNew ? '${translate(LocalizationKeys.bolusProfile_new)} ' : '',
|
||||
"profileName": _bolusProfile!.name,
|
||||
})),
|
||||
bottom: _isNew
|
||||
? PreferredSize(child: Container(), preferredSize: Size.zero)
|
||||
: const TabBar(
|
||||
: TabBar(
|
||||
tabs: [
|
||||
Tab(text: 'PROFILE'),
|
||||
Tab(text: 'RATES'),
|
||||
Tab(text: translate(LocalizationKeys.bolusProfile_detail_tabs_profile).toUpperCase()),
|
||||
Tab(text: translate(LocalizationKeys.bolusProfile_detail_tabs_rates).toUpperCase()),
|
||||
],
|
||||
),
|
||||
actions: appBarActions,
|
||||
|
@ -1,3 +1,4 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/models/bolus.dart';
|
||||
@ -6,6 +7,7 @@ import 'package:diameter/navigation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/bolus_profile.dart';
|
||||
import 'package:diameter/screens/bolus/bolus_profile_detail.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class BolusProfileListScreen extends StatefulWidget {
|
||||
static const String routeName = '/bolus-profiles';
|
||||
@ -62,26 +64,26 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
banner = activeProfileCount != 1
|
||||
? MaterialBanner(
|
||||
content: Text(activeProfileCount == 0
|
||||
? 'You currently do not have an active Bolus Profile.'
|
||||
: 'More than one active Bolus Profile has been found.'),
|
||||
? translate(LocalizationKeys.bolusProfile_warnings_noActive)
|
||||
: translate(LocalizationKeys.bolusProfile_warnings_multipleActive)),
|
||||
leading: const CircleAvatar(child: Icon(Icons.warning)),
|
||||
forceActionsBelow: true,
|
||||
actions: activeProfileCount == 0
|
||||
? [
|
||||
_bolusProfiles.isNotEmpty
|
||||
? TextButton(
|
||||
child: const Text('ACTIVATE A PROFILE'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_activate).toUpperCase()),
|
||||
onPressed: handlePickActiveProfileAction,
|
||||
)
|
||||
: Container(),
|
||||
TextButton(
|
||||
child: const Text('CREATE A NEW PROFILE'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_create).toUpperCase()),
|
||||
onPressed: () => onNew(true),
|
||||
),
|
||||
]
|
||||
: [
|
||||
TextButton(
|
||||
child: const Text('PICK A PROFILE'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_pick).toUpperCase()),
|
||||
onPressed: handlePickActiveProfileAction,
|
||||
),
|
||||
],
|
||||
@ -93,7 +95,12 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
void handleDuplicateAction(BolusProfile bolusProfile) async {
|
||||
final copy = BolusProfile(
|
||||
active: false,
|
||||
name: 'Copy of ${bolusProfile.name}',
|
||||
name: translate(
|
||||
LocalizationKeys.bolusProfile_copyOf,
|
||||
args: {
|
||||
"profileName": bolusProfile.name
|
||||
},
|
||||
),
|
||||
);
|
||||
BolusProfile.put(copy);
|
||||
|
||||
@ -111,12 +118,19 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
Bolus.put(bolus);
|
||||
}
|
||||
|
||||
reload(message: 'Added copy of ${bolusProfile.name}');
|
||||
reload(
|
||||
message: translate(
|
||||
LocalizationKeys.bolusProfile_copied,
|
||||
args: {
|
||||
"profileName": bolusProfile.name
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
void onDelete(BolusProfile bolusProfile) {
|
||||
BolusProfile.remove(bolusProfile.id);
|
||||
reload(message: 'Bolus Profile deleted');
|
||||
reload(message: translate(LocalizationKeys.bolusProfile_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(BolusProfile bolusProfile) async {
|
||||
@ -124,7 +138,7 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(bolusProfile),
|
||||
message: 'Are you sure you want to delete this Bolus Profile?',
|
||||
message: translate(LocalizationKeys.bolusProfile_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(bolusProfile);
|
||||
@ -137,7 +151,13 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
bolusProfile.active = true;
|
||||
BolusProfile.put(bolusProfile);
|
||||
reload(
|
||||
message: '${bolusProfile.name} has been set as your active Profile');
|
||||
message: translate(
|
||||
LocalizationKeys.bolusProfile_activated,
|
||||
args: {
|
||||
"profileName": bolusProfile.name
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@ -147,14 +167,14 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
content: AutoCompleteDropdownButton(
|
||||
controller: TextEditingController(text: ''),
|
||||
items: _bolusProfiles,
|
||||
label: 'Default Basal Profile',
|
||||
label: translate(LocalizationKeys.bolusProfile_default),
|
||||
onChanged: onPickActive,
|
||||
),
|
||||
leading: const CircleAvatar(child: Icon(Icons.info)),
|
||||
forceActionsBelow: true,
|
||||
actions: [
|
||||
TextButton(
|
||||
child: const Text('CREATE A NEW PROFILE INSTEAD'),
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_warnings_resolve_createInstead).toUpperCase()),
|
||||
onPressed: () => onNew(true),
|
||||
),
|
||||
],
|
||||
@ -184,7 +204,7 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Bolus Profiles'),
|
||||
title: Text(translate(LocalizationKeys.bolusProfile_title)),
|
||||
actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
],
|
||||
@ -206,9 +226,9 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
itemBuilder: (context, index) {
|
||||
final bolusProfile = _bolusProfiles[index];
|
||||
String activeProfileText = bolusProfile.active
|
||||
? ' (Default Profile)'
|
||||
? ' (${translate(LocalizationKeys.bolusProfile_default)})'
|
||||
: bolusProfile.id == _activeProfile?.id
|
||||
? ' (Current Active Profile)'
|
||||
? ' (${translate(LocalizationKeys.bolusProfile_active)})'
|
||||
: '';
|
||||
return Card(
|
||||
child: ListTile(
|
||||
@ -250,8 +270,8 @@ class _BolusProfileListScreenState extends State<BolusProfileListScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
child: Text('You have not created any Bolus Profiles yet!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.bolusProfile_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,12 +1,14 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/accuracy.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class AccuracyDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/accuracy';
|
||||
@ -93,7 +95,12 @@ class _AccuracyDetailScreenState extends State<AccuracyDetailScreen> {
|
||||
Accuracy.reorder(
|
||||
accuracy, int.tryParse(_confidenceRatingController.text));
|
||||
Navigator.pop(
|
||||
context, ['${_isNew ? 'New' : ''} Accuracy saved', accuracy]);
|
||||
context, [translate(
|
||||
LocalizationKeys.accuracy_saved,
|
||||
args: {
|
||||
"status": _isNew ? '${translate(LocalizationKeys.accuracy_new)} ' : ''
|
||||
},
|
||||
), accuracy]);
|
||||
}
|
||||
setState(() {
|
||||
_isSaving = false;
|
||||
@ -130,7 +137,9 @@ class _AccuracyDetailScreenState extends State<AccuracyDetailScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Accuracy' : _accuracy!.value),
|
||||
title: Text(translate(LocalizationKeys.accuracy_detail_title, args: {
|
||||
"status": _isNew ? '${translate(LocalizationKeys.accuracy_new)} ' : '',
|
||||
})),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: AccuracyDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
@ -145,12 +154,12 @@ class _AccuracyDetailScreenState extends State<AccuracyDetailScreen> {
|
||||
fields: [
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.accuracy_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
return translate(LocalizationKeys.accuracy_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
@ -158,7 +167,7 @@ class _AccuracyDetailScreenState extends State<AccuracyDetailScreen> {
|
||||
BooleanFormField(
|
||||
icon: const Icon(Icons.square_foot),
|
||||
value: _forPortionSize,
|
||||
label: 'for portion size',
|
||||
label: translate(LocalizationKeys.accuracy_fields_forPortionSize),
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_forPortionSize = value;
|
||||
@ -168,7 +177,7 @@ class _AccuracyDetailScreenState extends State<AccuracyDetailScreen> {
|
||||
BooleanFormField(
|
||||
icon: const Icon(Icons.pie_chart),
|
||||
value: _forCarbsRatio,
|
||||
label: 'for carbs ratio',
|
||||
label: translate(LocalizationKeys.accuracy_fields_forCarbsRatio),
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_forCarbsRatio = value;
|
||||
@ -177,7 +186,7 @@ class _AccuracyDetailScreenState extends State<AccuracyDetailScreen> {
|
||||
),
|
||||
NumberFormField(
|
||||
controller: _confidenceRatingController,
|
||||
label: 'Confidence Rating',
|
||||
label: translate(LocalizationKeys.accuracy_fields_confidenceRating),
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_confidenceRatingController.text =
|
||||
@ -188,8 +197,8 @@ class _AccuracyDetailScreenState extends State<AccuracyDetailScreen> {
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
keyboardType: TextInputType.multiline,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.accuracy_fields_notes),
|
||||
),
|
||||
minLines: 2,
|
||||
maxLines: 5,
|
||||
|
@ -1,9 +1,11 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/category/accuracy_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/accuracy.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class AccuracyListScreen extends StatefulWidget {
|
||||
static const String routeName = '/accuracies';
|
||||
@ -58,7 +60,7 @@ class _AccuracyListScreenState extends State<AccuracyListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(accuracy),
|
||||
message: 'Are you sure you want to delete this Accuracy?',
|
||||
message: translate(LocalizationKeys.accuracy_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(accuracy);
|
||||
@ -81,7 +83,7 @@ class _AccuracyListScreenState extends State<AccuracyListScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Accuracies'),
|
||||
title: Text(translate(LocalizationKeys.accuracy_detail)),
|
||||
actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
],
|
||||
@ -164,8 +166,8 @@ class _AccuracyListScreenState extends State<AccuracyListScreen> {
|
||||
);
|
||||
}),
|
||||
)
|
||||
: const Center(
|
||||
child: Text('You have not created any Accuracies yet!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.accuracy_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,5 +1,7 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class CategoryOverviewScreen extends StatefulWidget {
|
||||
static const String routeName = '/category';
|
||||
@ -22,7 +24,7 @@ class _CategoryOverviewScreenState extends State<CategoryOverviewScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Reports'),
|
||||
title: Text(translate(LocalizationKeys.categories)),
|
||||
),
|
||||
drawer:
|
||||
const Navigation(currentLocation: CategoryOverviewScreen.routeName),
|
||||
@ -45,7 +47,7 @@ class _CategoryOverviewScreenState extends State<CategoryOverviewScreen> {
|
||||
child: Icon(Icons.local_grocery_store, size: 50, color: Theme.of(context).textTheme.subtitle2?.color),
|
||||
),
|
||||
Text(
|
||||
'MEAL SOURCES',
|
||||
translate(LocalizationKeys.mealSource_title).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center
|
||||
),
|
||||
@ -66,7 +68,7 @@ class _CategoryOverviewScreenState extends State<CategoryOverviewScreen> {
|
||||
child: Icon(Icons.category, size: 50, color: Theme.of(context).textTheme.subtitle2?.color),
|
||||
),
|
||||
Text(
|
||||
'MEAL CATEGORIES',
|
||||
translate(LocalizationKeys.mealCategory_title).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center
|
||||
),
|
||||
@ -86,7 +88,7 @@ class _CategoryOverviewScreenState extends State<CategoryOverviewScreen> {
|
||||
child: Icon(Icons.pie_chart, size: 50, color: Theme.of(context).textTheme.subtitle2?.color),
|
||||
),
|
||||
Text(
|
||||
'PORTION TYPES',
|
||||
translate(LocalizationKeys.portionType_title).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center
|
||||
),
|
||||
@ -107,7 +109,7 @@ class _CategoryOverviewScreenState extends State<CategoryOverviewScreen> {
|
||||
child: Icon(Icons.architecture, size: 50, color: Theme.of(context).textTheme.subtitle2?.color),
|
||||
),
|
||||
Text(
|
||||
'ACCURACIES',
|
||||
translate(LocalizationKeys.accuracy_title).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center
|
||||
),
|
||||
@ -127,7 +129,7 @@ class _CategoryOverviewScreenState extends State<CategoryOverviewScreen> {
|
||||
child: Icon(Icons.event, size: 50, color: Theme.of(context).textTheme.subtitle2?.color),
|
||||
),
|
||||
Text(
|
||||
'EVENT TYPES',
|
||||
translate(LocalizationKeys.eventType_title).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center
|
||||
),
|
||||
|
@ -1,6 +1,7 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/duration_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
@ -12,6 +13,7 @@ import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/basal/basal_profile_detail.dart';
|
||||
import 'package:diameter/screens/bolus/bolus_profile_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class EventTypeDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-event-type';
|
||||
@ -126,7 +128,9 @@ class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
eventType.bolusProfile.target = _bolusProfile;
|
||||
LogEventType.put(eventType);
|
||||
Navigator.pop(
|
||||
context, ['${_isNew ? 'New' : ''} Log Event Type Saved', eventType]);
|
||||
context, [translate(LocalizationKeys.eventType_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.eventType_new : '',
|
||||
}), eventType]);
|
||||
}
|
||||
setState(() {
|
||||
_isSaving = false;
|
||||
@ -161,7 +165,10 @@ class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Log Event Type' : _logEventType!.value),
|
||||
title: Text(translate(LocalizationKeys.eventType_detail_title, args: {
|
||||
"status": _isNew ? LocalizationKeys.eventType_new : LocalizationKeys.general_edit,
|
||||
"name": _isNew ? '' : _logEventType!.value,
|
||||
}))
|
||||
),
|
||||
drawer:
|
||||
const Navigation(currentLocation: EventTypeDetailScreen.routeName),
|
||||
@ -175,19 +182,19 @@ class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
FormWrapper(formState: _logEventTypeForm, fields: [
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.eventType_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
return translate(LocalizationKeys.eventType_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
BooleanFormField(
|
||||
value: _hasEndTime,
|
||||
label: 'has end time',
|
||||
label: translate(LocalizationKeys.eventType_fields_hasEndTime),
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_hasEndTime = value;
|
||||
@ -201,7 +208,7 @@ class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
padding: const EdgeInsets.only(bottom: 10.0),
|
||||
child: DurationFormField(
|
||||
minutes: _defaultReminderDuration,
|
||||
label: 'Default Reminder Duration',
|
||||
label: translate(LocalizationKeys.eventType_fields_defaultReminderDuration),
|
||||
onChanged: (value) => _defaultReminderDuration = value ?? 0,
|
||||
showSteppers: true,
|
||||
),
|
||||
@ -215,7 +222,7 @@ class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
BolusProfile>(
|
||||
selectedItem: _bolusProfile,
|
||||
controller: _bolusProfileController,
|
||||
label: 'Bolus Profile',
|
||||
label: translate(LocalizationKeys.eventType_fields_bolusProfile),
|
||||
items: _bolusProfiles,
|
||||
onChanged: updateBolusProfile,
|
||||
),
|
||||
@ -252,7 +259,7 @@ class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
AutoCompleteDropdownButton<BasalProfile>(
|
||||
controller: _basalProfileController,
|
||||
selectedItem: _basalProfile,
|
||||
label: 'Basal Profile',
|
||||
label: translate(LocalizationKeys.eventType_fields_basalProfile),
|
||||
items: _basalProfiles,
|
||||
onChanged: updateBasalProfile,
|
||||
),
|
||||
@ -283,8 +290,8 @@ class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
: []),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.eventType_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
|
@ -1,7 +1,9 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/models/log_event_type.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/category/event_type_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class EventTypeListScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-event-types';
|
||||
@ -48,7 +50,7 @@ class _EventTypeListScreenState extends State<EventTypeListScreen> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(title: const Text('Log Event Types'), actions: <Widget>[
|
||||
appBar: AppBar(title: Text(translate(LocalizationKeys.eventType_title)), actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
]),
|
||||
drawer:
|
||||
@ -88,7 +90,7 @@ class _EventTypeListScreenState extends State<EventTypeListScreen> {
|
||||
IconButton(
|
||||
onPressed: () async {
|
||||
LogEventType.remove(logEventType.id);
|
||||
reload(message: 'Log Event Type deleted');
|
||||
reload(message: translate(LocalizationKeys.eventType_deleted));
|
||||
},
|
||||
icon: const Icon(Icons.delete,
|
||||
color: Colors.blue),
|
||||
@ -100,9 +102,9 @@ class _EventTypeListScreenState extends State<EventTypeListScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
: Center(
|
||||
child:
|
||||
Text('You have not created any Log Event Types yet!'),
|
||||
Text(translate(LocalizationKeys.eventType_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,10 +1,12 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/meal_category.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealCategoryDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/meal-category';
|
||||
@ -76,7 +78,9 @@ class _MealCategoryDetailScreenState extends State<MealCategoryDetailScreen> {
|
||||
);
|
||||
MealCategory.put(mealCategory);
|
||||
Navigator.pop(context, [
|
||||
'${_isNew ? 'New' : ''} Meal Category saved', mealCategory
|
||||
translate(LocalizationKeys.mealCategory_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.mealCategory_new : '',
|
||||
}), mealCategory
|
||||
]);
|
||||
}
|
||||
}
|
||||
@ -102,7 +106,11 @@ class _MealCategoryDetailScreenState extends State<MealCategoryDetailScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Meal Category' : _mealCategory!.value),
|
||||
title: Text(translate(LocalizationKeys.mealCategory_detail_title, args: {
|
||||
"status": _isNew ? LocalizationKeys.mealCategory_new : '',
|
||||
"name": _mealCategory!.value
|
||||
}),
|
||||
),
|
||||
),
|
||||
drawer:
|
||||
const Navigation(currentLocation: MealCategoryDetailScreen.routeName),
|
||||
@ -118,20 +126,20 @@ class _MealCategoryDetailScreenState extends State<MealCategoryDetailScreen> {
|
||||
fields: [
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.mealCategory_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
return translate(LocalizationKeys.mealCategory_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.mealCategory_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
|
@ -1,9 +1,11 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/category/meal_category_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/meal_category.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealCategoryListScreen extends StatefulWidget {
|
||||
static const String routeName = '/meal-categories';
|
||||
@ -50,7 +52,7 @@ class _MealCategoryListScreenState extends State<MealCategoryListScreen> {
|
||||
|
||||
void onDelete(MealCategory mealCategory) {
|
||||
MealCategory.remove(mealCategory.id);
|
||||
reload(message: 'Meal Category deleted');
|
||||
reload(message: translate(LocalizationKeys.mealCategory_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(MealCategory mealCategory) async {
|
||||
@ -58,7 +60,7 @@ class _MealCategoryListScreenState extends State<MealCategoryListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(mealCategory),
|
||||
message: 'Are you sure you want to delete this Meal Category?',
|
||||
message: translate(LocalizationKeys.mealCategory_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(mealCategory);
|
||||
@ -69,7 +71,7 @@ class _MealCategoryListScreenState extends State<MealCategoryListScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Meal Categories'),
|
||||
title: Text(translate(LocalizationKeys.mealCategory_title)),
|
||||
actions: <Widget>[
|
||||
IconButton(
|
||||
onPressed: reload,
|
||||
@ -125,8 +127,8 @@ class _MealCategoryListScreenState extends State<MealCategoryListScreen> {
|
||||
);
|
||||
},
|
||||
),
|
||||
): const Center(
|
||||
child: Text('You have not created any Meal Categories yet!'),
|
||||
): Center(
|
||||
child: Text(translate(LocalizationKeys.mealCategory_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,10 +1,12 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/meal_portion_type.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealPortionTypeDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/meal-portion-type';
|
||||
@ -78,7 +80,9 @@ class _MealPortionTypeDetailScreenState
|
||||
);
|
||||
MealPortionType.put(mealPortionType);
|
||||
Navigator.pop(context,
|
||||
['${_isNew ? 'New' : ''} Meal Portion Type saved', mealPortionType]);
|
||||
[translate(LocalizationKeys.portionType_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.portionType_new : '',
|
||||
}), mealPortionType]);
|
||||
}
|
||||
}
|
||||
|
||||
@ -102,10 +106,12 @@ class _MealPortionTypeDetailScreenState
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
bool isNew = _mealPortionType == null;
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(isNew ? 'New Meal Portion Type' : _mealPortionType!.value),
|
||||
title: Text(translate(LocalizationKeys.portionType_detail_title, args: {
|
||||
"status": translate(_isNew ? LocalizationKeys.portionType_new : LocalizationKeys.general_edit),
|
||||
"name": _mealPortionType!.value,
|
||||
})),
|
||||
),
|
||||
drawer: const Navigation(
|
||||
currentLocation: MealPortionTypeDetailScreen.routeName),
|
||||
@ -121,20 +127,20 @@ class _MealPortionTypeDetailScreenState
|
||||
fields: [
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.portionType_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
return translate(LocalizationKeys.portionType_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.portionType_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
|
@ -1,9 +1,11 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/category/meal_portion_type_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/models/meal_portion_type.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealPortionTypeListScreen extends StatefulWidget {
|
||||
static const String routeName = '/meal-portion-types';
|
||||
@ -51,7 +53,7 @@ class _MealPortionTypeListScreenState extends State<MealPortionTypeListScreen> {
|
||||
|
||||
void onDelete(MealPortionType mealPortionType) {
|
||||
MealPortionType.remove(mealPortionType.id);
|
||||
reload(message: 'Meal Portion Type deleted');
|
||||
reload(message: translate(LocalizationKeys.portionType_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(MealPortionType mealPortionType) async {
|
||||
@ -59,7 +61,7 @@ class _MealPortionTypeListScreenState extends State<MealPortionTypeListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(mealPortionType),
|
||||
message: 'Are you sure you want to delete this Meal Portion Type?',
|
||||
message: translate(LocalizationKeys.portionType_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(mealPortionType);
|
||||
@ -70,7 +72,7 @@ class _MealPortionTypeListScreenState extends State<MealPortionTypeListScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Meal Portion Types'),
|
||||
title: Text(translate(LocalizationKeys.portionType_title)),
|
||||
actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
],
|
||||
@ -124,8 +126,8 @@ class _MealPortionTypeListScreenState extends State<MealPortionTypeListScreen> {
|
||||
);
|
||||
},
|
||||
),
|
||||
) : const Center(
|
||||
child: Text('You have not created any Meal Portion Types yet!'),
|
||||
) : Center(
|
||||
child: Text(translate(LocalizationKeys.portionType_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,4 +1,5 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
@ -12,6 +13,7 @@ import 'package:diameter/screens/category/accuracy_detail.dart';
|
||||
import 'package:diameter/screens/category/meal_category_detail.dart';
|
||||
import 'package:diameter/screens/category/meal_portion_type_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealSourceDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/meal-source';
|
||||
@ -121,7 +123,9 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
mealSource.defaultMealCategory.target = _defaultMealCategory;
|
||||
mealSource.defaultMealPortionType.target = _defaultMealPortionType;
|
||||
MealSource.put(mealSource);
|
||||
Navigator.pop(context, ['${_isNew ? 'New' : ''} Meal Source saved', mealSource]);
|
||||
Navigator.pop(context, [translate(LocalizationKeys.mealSource_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.mealSource_new : ''
|
||||
}), mealSource]);
|
||||
}
|
||||
|
||||
void handleCancelAction() {
|
||||
@ -158,7 +162,12 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Meal Source' : _mealSource!.value),
|
||||
title: Text(
|
||||
translate(LocalizationKeys.mealSource_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.mealSource_new : LocalizationKeys.general_edit,
|
||||
"name": _mealSource!.value,
|
||||
})
|
||||
),
|
||||
),
|
||||
drawer:
|
||||
const Navigation(currentLocation: MealSourceDetailScreen.routeName),
|
||||
@ -174,12 +183,12 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
fields: [
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.mealSource_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
return translate(LocalizationKeys.mealSource_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
@ -190,7 +199,7 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<Accuracy>(
|
||||
selectedItem: _defaultCarbsRatioAccuracy,
|
||||
controller: _defaultCarbsRatioAccuracyController,
|
||||
label: 'Default Carbs Ratio Accuracy',
|
||||
label: translate(LocalizationKeys.mealSource_fields_defaultCarbsRatioAccuracy),
|
||||
items: _carbsRatioAccuracies,
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
@ -233,7 +242,7 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<Accuracy>(
|
||||
selectedItem: _defaultPortionSizeAccuracy,
|
||||
controller: _defaultPortionSizeAccuracyController,
|
||||
label: 'Default Portion Size Accuracy',
|
||||
label: translate(LocalizationKeys.mealSource_fields_defaultPortionSizeAccuracy),
|
||||
items: _portionSizeAccuracies,
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
@ -278,7 +287,7 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<MealCategory>(
|
||||
selectedItem: _defaultMealCategory,
|
||||
controller: _defaultMealCategoryController,
|
||||
label: 'Default Meal Category',
|
||||
label: translate(LocalizationKeys.mealSource_fields_defaultMealCategory),
|
||||
items: _mealCategories,
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
@ -320,7 +329,7 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<MealPortionType>(
|
||||
selectedItem: _defaultMealPortionType,
|
||||
controller: _defaultMealPortionTypeController,
|
||||
label: 'Default Meal Portion Type',
|
||||
label: translate(LocalizationKeys.mealSource_fields_defaultMealPortionType),
|
||||
items: _mealPortionTypes,
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
@ -359,8 +368,8 @@ class _MealSourceDetailScreenState extends State<MealSourceDetailScreen> {
|
||||
),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.mealSource_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
|
@ -1,9 +1,11 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/meal_source.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/category/meal_source_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealSourceListScreen extends StatefulWidget {
|
||||
static const String routeName = '/meal-sources';
|
||||
@ -50,7 +52,7 @@ class _MealSourceListScreenState extends State<MealSourceListScreen> {
|
||||
|
||||
void onDelete(MealSource mealSource) {
|
||||
MealSource.remove(mealSource.id);
|
||||
reload(message: 'Meal Source deleted');
|
||||
reload(message: translate(LocalizationKeys.mealSource_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(MealSource mealSource) async {
|
||||
@ -58,7 +60,7 @@ class _MealSourceListScreenState extends State<MealSourceListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(mealSource),
|
||||
message: 'Are you sure you want to delete this Meal Source?',
|
||||
message: translate(LocalizationKeys.mealSource_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(mealSource);
|
||||
@ -69,7 +71,7 @@ class _MealSourceListScreenState extends State<MealSourceListScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Meal Sources'),
|
||||
title: Text(translate(LocalizationKeys.mealSource_title)),
|
||||
actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
],
|
||||
@ -122,8 +124,8 @@ class _MealSourceListScreenState extends State<MealSourceListScreen> {
|
||||
);
|
||||
}
|
||||
),
|
||||
) : const Center(
|
||||
child: Text('You have not created any Meal Sources yet!'),
|
||||
) : Center(
|
||||
child: Text(translate(LocalizationKeys.mealSource_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -3,6 +3,7 @@ import 'dart:math';
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
@ -15,6 +16,7 @@ import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/log/log_entry/log_meal_detail.dart';
|
||||
import 'package:diameter/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
enum BolusType {
|
||||
meal,
|
||||
@ -453,7 +455,9 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
}
|
||||
|
||||
Navigator.pop(context,
|
||||
['${_isNew ? 'New' : ''} Bolus Saved', logBolus, delayedBolus]);
|
||||
[translate(LocalizationKeys.log_detail_tabs_bolus_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.log_detail_tabs_bolus_new : ''
|
||||
}), logBolus, delayedBolus]);
|
||||
}
|
||||
setState(() {
|
||||
_isSaving = false;
|
||||
@ -509,7 +513,9 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Bolus' : 'Edit Bolus'),
|
||||
title: Text(translate(LocalizationKeys.log_detail_tabs_bolus_detail_title, args: {
|
||||
"status": _isNew ? LocalizationKeys.log_detail_tabs_bolus_new : LocalizationKeys.general_edit
|
||||
})),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: LogBolusDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
@ -544,7 +550,7 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
contentPadding: const EdgeInsets.only(
|
||||
left: 10.0, right: 10.0, top: 10.0),
|
||||
value: _setManually,
|
||||
label: 'set manually',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_setManually),
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_setManually = value;
|
||||
@ -559,7 +565,7 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: RadioListTile(
|
||||
title: const Text('for glucose'),
|
||||
title: Text(translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_forGlucose)),
|
||||
groupValue: _bolusType,
|
||||
value: BolusType.glucose,
|
||||
onChanged: (_) {
|
||||
@ -571,7 +577,7 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
),
|
||||
Expanded(
|
||||
child: RadioListTile(
|
||||
title: const Text('for meal'),
|
||||
title: Text(translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_forMeal)),
|
||||
groupValue: _bolusType,
|
||||
value: BolusType.meal,
|
||||
onChanged: (value) {
|
||||
@ -599,8 +605,8 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
padding:
|
||||
const EdgeInsets.only(right: 5.0),
|
||||
child: NumberFormField(
|
||||
label: 'Current',
|
||||
suffix: 'mg/dl',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_current),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
controller:
|
||||
_mgPerDlCurrentController,
|
||||
onChanged: (_) => onChangeGlucose(),
|
||||
@ -613,8 +619,8 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
padding: const EdgeInsets.symmetric(
|
||||
horizontal: 5.0),
|
||||
child: NumberFormField(
|
||||
label: 'Target',
|
||||
suffix: 'mg/dl',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_target),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
controller:
|
||||
_mgPerDlTargetController,
|
||||
onChanged: (_) => onChangeGlucose(),
|
||||
@ -627,9 +633,9 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
padding:
|
||||
const EdgeInsets.only(left: 5.0),
|
||||
child: TextFormField(
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Correction',
|
||||
suffixText: 'mg/dl',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_correction),
|
||||
suffixText: Settings.glucoseMeasurementSuffix,
|
||||
),
|
||||
controller:
|
||||
_mgPerDlCorrectionController,
|
||||
@ -661,8 +667,8 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
padding: const EdgeInsets.only(
|
||||
right: 5.0),
|
||||
child: NumberFormField(
|
||||
label: 'Current',
|
||||
suffix: 'mmol/l',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_current),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
controller:
|
||||
_mmolPerLCurrentController,
|
||||
onChanged: (_) =>
|
||||
@ -676,8 +682,8 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
padding: const EdgeInsets.symmetric(
|
||||
horizontal: 5.0),
|
||||
child: NumberFormField(
|
||||
label: 'Target',
|
||||
suffix: 'mmol/l',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_target),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
controller:
|
||||
_mmolPerLTargetController,
|
||||
onChanged: (_) =>
|
||||
@ -691,9 +697,9 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
padding: const EdgeInsets.only(
|
||||
left: 5.0),
|
||||
child: TextFormField(
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Correction',
|
||||
suffixText: 'mmol/l',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_correction),
|
||||
suffixText: Settings.glucoseMeasurementSuffix,
|
||||
),
|
||||
controller:
|
||||
_mmolPerLCorrectionController,
|
||||
@ -713,7 +719,7 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<LogMeal>(
|
||||
controller: _mealController,
|
||||
selectedItem: _meal,
|
||||
label: 'Meal',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_meal),
|
||||
items: _logMeals,
|
||||
onChanged: onSelectMeal,
|
||||
),
|
||||
@ -758,9 +764,9 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: TextFormField(
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Delayed Bolus Duration',
|
||||
suffixText: ' min',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_delayedBolusDuration),
|
||||
suffixText: translate(LocalizationKeys.general_suffixes_mins),
|
||||
),
|
||||
controller: _delayController,
|
||||
onChanged: (value) => setState(() {}),
|
||||
@ -789,8 +795,8 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(right: 5.0),
|
||||
child: NumberFormField(
|
||||
label: 'Immediate Bolus',
|
||||
suffix: ' U',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_immediateBolus),
|
||||
suffix: translate(LocalizationKeys.general_suffixes_units),
|
||||
controller: _immediateUnitsController,
|
||||
max: double.tryParse(_unitsController.text),
|
||||
step: Settings.insulinSteps,
|
||||
@ -804,8 +810,8 @@ class _LogBolusDetailScreenState extends State<LogBolusDetailScreen> {
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 5.0),
|
||||
child: NumberFormField(
|
||||
label: 'Delayed Bolus',
|
||||
suffix: ' U',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_bolus_detail_fields_delayedBolus),
|
||||
suffix: translate(LocalizationKeys.general_suffixes_units),
|
||||
controller: _delayedUnitsController,
|
||||
max: double.tryParse(_unitsController.text),
|
||||
step: Settings.insulinSteps,
|
||||
|
@ -1,3 +1,4 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/log_bolus.dart';
|
||||
import 'package:diameter/models/log_entry.dart';
|
||||
@ -5,6 +6,7 @@ import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/screens/log/log_entry/log_bolus_detail.dart';
|
||||
import 'package:diameter/screens/log/log_entry/log_meal_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class LogBolusListScreen extends StatefulWidget {
|
||||
final LogEntry logEntry;
|
||||
@ -61,7 +63,7 @@ class _LogBolusListScreenState extends State<LogBolusListScreen> {
|
||||
|
||||
void onDelete(LogBolus logBolus) {
|
||||
LogBolus.remove(logBolus.id);
|
||||
reload(message: 'Bolus deleted');
|
||||
reload(message: translate(LocalizationKeys.log_detail_tabs_bolus_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(LogBolus logBolus) async {
|
||||
@ -69,7 +71,8 @@ class _LogBolusListScreenState extends State<LogBolusListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(logBolus),
|
||||
message: 'Are you sure you want to delete this Bolus?',
|
||||
message:
|
||||
translate(LocalizationKeys.log_detail_tabs_bolus_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(logBolus);
|
||||
@ -100,9 +103,14 @@ class _LogBolusListScreenState extends State<LogBolusListScreen> {
|
||||
itemCount: widget.logBoli.length,
|
||||
itemBuilder: (context, index) {
|
||||
final bolus = widget.logBoli[index];
|
||||
String titleText = '${bolus.units} U ${(bolus.delay ?? 0) != 0
|
||||
? ' (delayed by ${bolus.delay} min)'
|
||||
: ''}';
|
||||
String titleText =
|
||||
'${bolus.units} ${translate(LocalizationKeys.general_suffixes_units)} ${(bolus.delay ?? 0) != 0 ?
|
||||
translate(
|
||||
LocalizationKeys.log_detail_tabs_bolus_delayedBy,
|
||||
args: {
|
||||
"delay": '${bolus.delay} ${translate(LocalizationKeys.general_suffixes_mins)}'
|
||||
}
|
||||
) : ''}';
|
||||
return Card(
|
||||
child: ListTile(
|
||||
onTap: () => handleEditAction(bolus),
|
||||
@ -110,8 +118,8 @@ class _LogBolusListScreenState extends State<LogBolusListScreen> {
|
||||
titleText.toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
subtitle: Text(bolus.carbs != null ?
|
||||
'for ${(bolus.meal.target ?? '').toString()} (${bolus.carbs}${Settings.nutritionMeasurementSuffix} carbs)'
|
||||
subtitle: Text(bolus.carbs != null
|
||||
? 'for ${(bolus.meal.target ?? '').toString()} (${bolus.carbs}${Settings.nutritionMeasurementSuffix} carbs)'
|
||||
: 'to correct ${Settings.glucoseMeasurement == GlucoseMeasurement.mgPerDl ? bolus.mgPerDlCorrection : bolus.mmolPerLCorrection} ${Settings.glucoseMeasurementSuffix}'),
|
||||
trailing: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
@ -138,9 +146,9 @@ class _LogBolusListScreenState extends State<LogBolusListScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
child: Text(
|
||||
'You have not added any Boli to this Log Entry yet!'),
|
||||
: Center(
|
||||
child:
|
||||
Text(translate(LocalizationKeys.log_detail_tabs_bolus_empty)),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
@ -2,6 +2,7 @@ import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/date_time_form_field.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/components/forms/time_of_day_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/log_bolus.dart';
|
||||
@ -18,6 +19,8 @@ import 'package:diameter/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'dart:math' as math;
|
||||
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class LogEntryScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-entry';
|
||||
final int id;
|
||||
@ -192,7 +195,9 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
|
||||
if (close) {
|
||||
Navigator.pop(
|
||||
context, ['${_isNew ? 'New' : ''} Log Entry Saved', logEntry]);
|
||||
context, [translate(LocalizationKeys.log_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.log_new : ''
|
||||
}), logEntry]);
|
||||
} else {
|
||||
if (_isNew) {
|
||||
Navigator.push(
|
||||
@ -202,7 +207,7 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
),
|
||||
).then((result) => Navigator.pop(context, result));
|
||||
} else {
|
||||
reload(message: 'Log Entry Saved');
|
||||
reload(message: translate(LocalizationKeys.log_saved));
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -306,7 +311,7 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
padding: const EdgeInsets.only(right: 5),
|
||||
child: DateTimeFormField(
|
||||
date: _time,
|
||||
label: 'Date',
|
||||
label: translate(LocalizationKeys.log_fields_date),
|
||||
controller: _dateController,
|
||||
onChanged: (newTime) {
|
||||
if (newTime != null) {
|
||||
@ -329,7 +334,7 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
padding: const EdgeInsets.only(left: 5),
|
||||
child: TimeOfDayFormField(
|
||||
time: TimeOfDay.fromDateTime(_time),
|
||||
label: 'Time',
|
||||
label: translate(LocalizationKeys.log_fields_time),
|
||||
controller: _timeController,
|
||||
onChanged: (newTime) {
|
||||
if (newTime != null) {
|
||||
@ -363,8 +368,8 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
? 2
|
||||
: 1,
|
||||
child: NumberFormField(
|
||||
label: 'Blood Glucose',
|
||||
suffix: 'mg/dl',
|
||||
label: translate(LocalizationKeys.log_fields_glucose),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
readOnly: Settings.glucoseMeasurement ==
|
||||
GlucoseMeasurement.mmolPerL,
|
||||
showSteppers:
|
||||
@ -388,8 +393,8 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
? 2
|
||||
: 1,
|
||||
child: NumberFormField(
|
||||
label: 'Blood Glucose',
|
||||
suffix: 'mmol/l',
|
||||
label: translate(LocalizationKeys.log_fields_glucose),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
readOnly: Settings.glucoseMeasurement ==
|
||||
GlucoseMeasurement.mgPerDl,
|
||||
showSteppers:
|
||||
@ -421,8 +426,8 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
@ -444,14 +449,16 @@ class _LogEntryScreenState extends State<LogEntryScreen> {
|
||||
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Log Entry' : 'Edit Log Entry'),
|
||||
title: Text(translate(LocalizationKeys.log_detail_title, args: {
|
||||
"status": _isNew ? LocalizationKeys.log_new : LocalizationKeys.general_edit
|
||||
})),
|
||||
bottom: _isNew
|
||||
? PreferredSize(child: Container(), preferredSize: Size.zero)
|
||||
: const TabBar(
|
||||
: TabBar(
|
||||
tabs: [
|
||||
Tab(text: 'GENERAL'),
|
||||
Tab(text: 'MEALS'),
|
||||
Tab(text: 'BOLI'),
|
||||
Tab(text: translate(LocalizationKeys.log_detail_tabs_general).toUpperCase()),
|
||||
Tab(text: translate(LocalizationKeys.log_detail_tabs_meal_title).toUpperCase()),
|
||||
Tab(text: translate(LocalizationKeys.log_detail_tabs_bolus_title).toUpperCase()),
|
||||
],
|
||||
),
|
||||
actions: appBarActions,
|
||||
|
@ -1,6 +1,7 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
@ -19,6 +20,7 @@ import 'package:diameter/screens/category/meal_source_detail.dart';
|
||||
import 'package:diameter/screens/meal/meal_detail.dart';
|
||||
import 'package:diameter/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class LogMealDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-meal';
|
||||
@ -230,7 +232,9 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
logMeal.carbsRatioAccuracy.target = _carbsRatioAccuracy;
|
||||
|
||||
LogMeal.put(logMeal);
|
||||
Navigator.pop(context, ['${_isNew ? 'New' : ''} Meal Saved', logMeal]);
|
||||
Navigator.pop(context, [translate(LocalizationKeys.log_detail_tabs_meal_saved, args: {
|
||||
"status": _isNew ? translate(LocalizationKeys.log_detail_tabs_bolus_new) : ''
|
||||
}), logMeal]);
|
||||
}
|
||||
setState(() {
|
||||
_isSaving = false;
|
||||
@ -405,7 +409,9 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Meal for Log Entry' : _logMeal!.value),
|
||||
title: Text(translate(LocalizationKeys.log_detail_tabs_meal_detail_title, args: {
|
||||
"status": _isNew ? translate(LocalizationKeys.log_detail_tabs_bolus_new) : LocalizationKeys.general_edit
|
||||
})),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: LogMealDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
@ -424,7 +430,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<Meal>(
|
||||
controller: _mealController,
|
||||
selectedItem: _meal,
|
||||
label: 'Meal',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_meal),
|
||||
items: _meals,
|
||||
onChanged: onSelectMeal,
|
||||
),
|
||||
@ -449,12 +455,12 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
),
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
return translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
@ -465,7 +471,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
flex: 10,
|
||||
child: NumberFormField(
|
||||
controller: _amountController,
|
||||
label: 'Amount',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_amount),
|
||||
suffix: _mealPortionType?.value,
|
||||
onChanged: updateAmount,
|
||||
),
|
||||
@ -524,7 +530,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: 'Portion size',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_portionSize),
|
||||
suffix: Settings.nutritionMeasurementSuffix,
|
||||
controller: _portionSizeController,
|
||||
showSteppers: false,
|
||||
@ -548,7 +554,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
const SizedBox(width: 10),
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: 'Carbs ratio',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_carbsRatio),
|
||||
suffix: '%',
|
||||
controller: _carbsRatioController,
|
||||
showSteppers: false,
|
||||
@ -570,7 +576,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
const SizedBox(width: 10),
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: 'Total carbs',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_totalCarbs),
|
||||
suffix: Settings.nutritionMeasurementSuffix,
|
||||
controller: _totalCarbsController,
|
||||
showSteppers: false,
|
||||
@ -595,7 +601,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
),
|
||||
BooleanFormField(
|
||||
value: _setManually,
|
||||
label: 'set carbs ratio manually',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_setManually),
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_setManually = value;
|
||||
@ -605,8 +611,8 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
@ -621,7 +627,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
mainAxisSize: MainAxisSize.max,
|
||||
children: [
|
||||
Text(
|
||||
'ADDITIONAL FIELDS',
|
||||
translate(LocalizationKeys.log_detail_tabs_meal_detail_additionalFields).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
const Spacer(),
|
||||
@ -642,7 +648,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
AutoCompleteDropdownButton<MealSource>(
|
||||
controller: _mealSourceController,
|
||||
selectedItem: _mealSource,
|
||||
label: 'Meal Source',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_mealSource),
|
||||
items: _mealSources,
|
||||
onChanged: updateMealSource,
|
||||
),
|
||||
@ -677,7 +683,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
AutoCompleteDropdownButton<MealCategory>(
|
||||
controller: _mealCategoryController,
|
||||
selectedItem: _mealCategory,
|
||||
label: 'Meal Category',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_mealCategory),
|
||||
items: _mealCategories,
|
||||
onChanged: updateMealCategory,
|
||||
),
|
||||
@ -712,7 +718,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
AutoCompleteDropdownButton<MealPortionType>(
|
||||
controller: _mealPortionTypeController,
|
||||
selectedItem: _mealPortionType,
|
||||
label: 'Meal Portion Type',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_mealPortionType),
|
||||
items: _mealPortionTypes,
|
||||
onChanged: updateMealPortionType,
|
||||
),
|
||||
@ -747,7 +753,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
AutoCompleteDropdownButton<Accuracy>(
|
||||
controller: _portionSizeAccuracyController,
|
||||
selectedItem: _portionSizeAccuracy,
|
||||
label: 'Portion Size Accuracy',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_portionSizeAccuracy),
|
||||
items: _portionSizeAccuracies,
|
||||
onChanged: updatePortionSizeAccuracy,
|
||||
),
|
||||
@ -783,7 +789,7 @@ class _LogMealDetailScreenState extends State<LogMealDetailScreen> {
|
||||
AutoCompleteDropdownButton<Accuracy>(
|
||||
controller: _carbsRatioAccuracyController,
|
||||
selectedItem: _carbsRatioAccuracy,
|
||||
label: 'Carbs Ratio Accuracy',
|
||||
label: translate(LocalizationKeys.log_detail_tabs_meal_detail_fields_carbsRatioAccuracy),
|
||||
items: _carbsRatioAccuracies,
|
||||
onChanged: updateCarbsRatioAccuracy,
|
||||
),
|
||||
|
@ -1,9 +1,11 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/log_entry.dart';
|
||||
import 'package:diameter/models/log_meal.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/screens/log/log_entry/log_meal_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class LogMealListScreen extends StatefulWidget {
|
||||
final LogEntry logEntry;
|
||||
@ -57,7 +59,7 @@ class _LogMealListScreenState extends State<LogMealListScreen> {
|
||||
|
||||
void onDelete(LogMeal logMeal) {
|
||||
LogMeal.remove(logMeal.id);
|
||||
reload(message: 'Meal deleted');
|
||||
reload(message: translate(LocalizationKeys.log_detail_tabs_meal_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(LogMeal meal) async {
|
||||
@ -65,7 +67,7 @@ class _LogMealListScreenState extends State<LogMealListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(meal),
|
||||
message: 'Are you sure you want to delete this Meal?',
|
||||
message: translate(LocalizationKeys.log_detail_tabs_meal_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(meal);
|
||||
@ -99,7 +101,7 @@ class _LogMealListScreenState extends State<LogMealListScreen> {
|
||||
? [
|
||||
Text(meal.totalCarbs!.toStringAsPrecision(3)),
|
||||
Text(
|
||||
'${Settings.nutritionMeasurementSuffix} carbs',
|
||||
'${Settings.nutritionMeasurementSuffix} ${translate(LocalizationKeys.general_suffixes_carbs)}',
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
: [],
|
||||
@ -119,9 +121,9 @@ class _LogMealListScreenState extends State<LogMealListScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
: Center(
|
||||
child: Text(
|
||||
'You have not added any Meals to this Log Entry yet!'),
|
||||
translate(LocalizationKeys.log_detail_tabs_meal_empty)),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
@ -2,6 +2,7 @@ import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/date_time_form_field.dart';
|
||||
import 'package:diameter/components/forms/time_of_day_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
@ -15,6 +16,7 @@ import 'package:diameter/screens/basal/basal_profile_detail.dart';
|
||||
import 'package:diameter/screens/bolus/bolus_profile_detail.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class LogEventDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-event';
|
||||
@ -194,20 +196,19 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
content: const Text(
|
||||
'An Event of this type is already active within the set time frame. What would you like to do?'),
|
||||
content: Text(translate(LocalizationKeys.event_warnings_duplicate)),
|
||||
actions: [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 'DISCARD'),
|
||||
child: const Text('DISCARD'),
|
||||
child: Text(translate(LocalizationKeys.general_discard).toUpperCase()),
|
||||
),
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 'EDIT'),
|
||||
child: const Text('KEEP EDITING'),
|
||||
child: Text(translate(LocalizationKeys.general_keepEditing).toUpperCase()),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 'SAVE'),
|
||||
child: const Text('SAVE'),
|
||||
child: Text(translate(LocalizationKeys.general_save).toUpperCase()),
|
||||
)
|
||||
],
|
||||
);
|
||||
@ -236,7 +237,9 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
event.basalProfile.target = _basalProfile;
|
||||
event.bolusProfile.target = _bolusProfile;
|
||||
LogEvent.put(event);
|
||||
Navigator.pop(context, ['${_isNew ? 'New' : ''} Event Saved', event]);
|
||||
Navigator.pop(context, [translate(LocalizationKeys.event_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.event_new : "",
|
||||
}), event]);
|
||||
}
|
||||
|
||||
void handleSaveAction() async {
|
||||
@ -276,7 +279,10 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
final now = DateTime.now();
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Event' : 'Edit Event'),
|
||||
title: Text(translate(LocalizationKeys.event_saved, args: {
|
||||
"status": _isNew ? LocalizationKeys.event_new : LocalizationKeys.general_edit,
|
||||
"name": _isNew ? '' : _logEvent?.eventType.target?.value
|
||||
})),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: LogEventDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
@ -292,7 +298,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
AutoCompleteDropdownButton<LogEventType>(
|
||||
controller: _eventTypeController,
|
||||
selectedItem: _eventType,
|
||||
label: 'Event Type',
|
||||
label: translate(LocalizationKeys.event_fields_eventType),
|
||||
items: _logEventTypes,
|
||||
onChanged: onSelectEventType,
|
||||
),
|
||||
@ -303,7 +309,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
padding: const EdgeInsets.only(right: 5),
|
||||
child: DateTimeFormField(
|
||||
date: _time,
|
||||
label: _hasEndTime ? 'Start Date' : 'Date',
|
||||
label: translate(_hasEndTime ? LocalizationKeys.event_fields_startDate : LocalizationKeys.event_fields_date),
|
||||
controller: _dateController,
|
||||
onChanged: (newTime) {
|
||||
if (newTime != null) {
|
||||
@ -322,7 +328,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
padding: const EdgeInsets.only(left: 5),
|
||||
child: TimeOfDayFormField(
|
||||
time: TimeOfDay.fromDateTime(_time),
|
||||
label: _hasEndTime ? 'Start Time' : 'Time',
|
||||
label: translate(_hasEndTime ? LocalizationKeys.event_fields_startTime : LocalizationKeys.event_fields_time),
|
||||
controller: _timeController,
|
||||
onChanged: (newTime) {
|
||||
if (newTime != null) {
|
||||
@ -345,7 +351,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
_hasEndTime = value;
|
||||
});
|
||||
},
|
||||
label: 'has end time',
|
||||
label: translate(LocalizationKeys.event_fields_hasEndTime),
|
||||
),
|
||||
Column(
|
||||
children: _hasEndTime
|
||||
@ -357,7 +363,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
padding: const EdgeInsets.only(right: 5),
|
||||
child: DateTimeFormField(
|
||||
date: _endTime ?? now,
|
||||
label: 'End Date',
|
||||
label: translate(LocalizationKeys.event_fields_endDate),
|
||||
controller: _endDateController,
|
||||
onChanged: (newTime) {
|
||||
if (newTime != null) {
|
||||
@ -381,7 +387,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
child: TimeOfDayFormField(
|
||||
time: TimeOfDay.fromDateTime(
|
||||
_endTime ?? now),
|
||||
label: 'End Time',
|
||||
label: translate(LocalizationKeys.event_fields_endTime),
|
||||
controller: _endTimeController,
|
||||
onChanged: (newTime) {
|
||||
if (newTime != null) {
|
||||
@ -408,8 +414,8 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
keyboardType:
|
||||
const TextInputType.numberWithOptions(),
|
||||
decoration: InputDecoration(
|
||||
labelText: 'Default Reminder Duration',
|
||||
suffixText: ' min',
|
||||
labelText: translate(LocalizationKeys.event_fields_reminderDuration),
|
||||
suffixText: translate(LocalizationKeys.general_suffixes_mins),
|
||||
enabled: _hasEndTime,
|
||||
),
|
||||
),
|
||||
@ -421,7 +427,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
BolusProfile>(
|
||||
controller: _bolusProfileController,
|
||||
selectedItem: _bolusProfile,
|
||||
label: 'Bolus Profile',
|
||||
label: translate(LocalizationKeys.event_fields_bolusProfile),
|
||||
items: _bolusProfiles,
|
||||
onChanged: updateBolusProfile,
|
||||
),
|
||||
@ -457,7 +463,7 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
BasalProfile>(
|
||||
controller: _basalProfileController,
|
||||
selectedItem: _basalProfile,
|
||||
label: 'Basal Profile',
|
||||
label: translate(LocalizationKeys.event_fields_basalProfile),
|
||||
items: _basalProfiles,
|
||||
onChanged: updateBasalProfile,
|
||||
),
|
||||
@ -489,8 +495,8 @@ class _LogEventDetailScreenState extends State<LogEventDetailScreen> {
|
||||
: []),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.event_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
|
@ -1,3 +1,4 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/log_event.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
@ -5,6 +6,7 @@ import 'package:diameter/screens/log/log_event/log_event_detail.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class LogEventListScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-events';
|
||||
@ -88,7 +90,7 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
|
||||
void onDelete(LogEvent logEvent) {
|
||||
LogEvent.remove(logEvent.id);
|
||||
reload(message: 'Event deleted');
|
||||
reload(message: translate(LocalizationKeys.event_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(LogEvent logEvent) async {
|
||||
@ -96,7 +98,7 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(logEvent),
|
||||
message: 'Are you sure you want to delete this Event?',
|
||||
message: translate(LocalizationKeys.event_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(logEvent);
|
||||
@ -106,7 +108,7 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
void onStop(LogEvent event) async {
|
||||
event.endTime = DateTime.now();
|
||||
LogEvent.put(event);
|
||||
reload(message: 'Event ended');
|
||||
reload(message: translate(LocalizationKeys.event_ended));
|
||||
}
|
||||
|
||||
void handleStopAction(LogEvent event) async {
|
||||
@ -114,8 +116,8 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onStop(event),
|
||||
message: 'Are you sure you want to end this Event?',
|
||||
confirmationLabel: 'END EVENT',
|
||||
message: translate(LocalizationKeys.event_confirmEnd),
|
||||
confirmationLabel: translate(LocalizationKeys.event_end),
|
||||
);
|
||||
} else {
|
||||
onStop(event);
|
||||
@ -136,7 +138,7 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Log Events'),
|
||||
title: Text(translate(LocalizationKeys.event_title)),
|
||||
actions: <Widget>[
|
||||
IconButton(
|
||||
onPressed: () => onChangeDate(DateTime.now()),
|
||||
@ -159,7 +161,7 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: Text(
|
||||
'ACTIVE EVENTS',
|
||||
translate(LocalizationKeys.event_titleActive).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
@ -235,8 +237,8 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
);
|
||||
},
|
||||
)
|
||||
: const Center(
|
||||
child: Text('There are no Active Events!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.event_emptyActive)),
|
||||
),
|
||||
const Padding(
|
||||
padding: EdgeInsets.all(10.0),
|
||||
@ -351,8 +353,8 @@ class _LogEventListScreenState extends State<LogEventListScreen> {
|
||||
);
|
||||
},
|
||||
))
|
||||
: const Center(
|
||||
child: Text('There are no Events for that date!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.event_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,304 +0,0 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/duration_form_field.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/basal_profile.dart';
|
||||
import 'package:diameter/models/bolus_profile.dart';
|
||||
import 'package:diameter/models/log_event_type.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/basal/basal_profile_detail.dart';
|
||||
import 'package:diameter/screens/bolus/bolus_profile_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class EventTypeDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-event-type';
|
||||
final int id;
|
||||
const EventTypeDetailScreen({Key? key, this.id = 0}) : super(key: key);
|
||||
|
||||
@override
|
||||
_EventTypeDetailScreenState createState() => _EventTypeDetailScreenState();
|
||||
}
|
||||
|
||||
class _EventTypeDetailScreenState extends State<EventTypeDetailScreen> {
|
||||
LogEventType? _logEventType;
|
||||
bool _isNew = true;
|
||||
bool _isSaving = false;
|
||||
|
||||
List<BolusProfile> _bolusProfiles = [];
|
||||
List<BasalProfile> _basalProfiles = [];
|
||||
|
||||
final GlobalKey<FormState> _logEventTypeForm = GlobalKey<FormState>();
|
||||
final ScrollController _scrollController = ScrollController();
|
||||
|
||||
final _valueController = TextEditingController(text: '');
|
||||
final _notesController = TextEditingController(text: '');
|
||||
|
||||
bool _hasEndTime = false;
|
||||
int _defaultReminderDuration = 0;
|
||||
BolusProfile? _bolusProfile;
|
||||
BasalProfile? _basalProfile;
|
||||
final _bolusProfileController = TextEditingController(text: '');
|
||||
final _basalProfileController = TextEditingController(text: '');
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
|
||||
reload();
|
||||
|
||||
_bolusProfiles = BolusProfile.getAll();
|
||||
_basalProfiles = BasalProfile.getAll();
|
||||
|
||||
if (_logEventType != null) {
|
||||
_valueController.text = _logEventType!.value;
|
||||
_defaultReminderDuration =
|
||||
_logEventType!.defaultReminderDuration ?? 0;
|
||||
_hasEndTime = _logEventType!.hasEndTime;
|
||||
_notesController.text = _logEventType!.notes ?? '';
|
||||
_basalProfile = _logEventType!.basalProfile.target;
|
||||
_basalProfileController.text = (_basalProfile ?? '').toString();
|
||||
_bolusProfile = _logEventType!.bolusProfile.target;
|
||||
_bolusProfileController.text = (_bolusProfile ?? '').toString();
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_scrollController.dispose();
|
||||
_valueController.dispose();
|
||||
_notesController.dispose();
|
||||
_bolusProfileController.dispose();
|
||||
_basalProfileController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
void reload({String? message}) {
|
||||
if (widget.id != 0) {
|
||||
setState(() {
|
||||
_logEventType = LogEventType.get(widget.id);
|
||||
});
|
||||
}
|
||||
_isNew = _logEventType == null;
|
||||
|
||||
setState(() {
|
||||
if (message != null) {
|
||||
var snackBar = SnackBar(
|
||||
content: Text(message),
|
||||
duration: const Duration(seconds: 2),
|
||||
);
|
||||
ScaffoldMessenger.of(context)
|
||||
..removeCurrentSnackBar()
|
||||
..showSnackBar(snackBar);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
void updateBasalProfile(BasalProfile? value) {
|
||||
setState(() {
|
||||
_basalProfile = value;
|
||||
_basalProfileController.text = (_basalProfile ?? '').toString();
|
||||
});
|
||||
}
|
||||
|
||||
void updateBolusProfile(BolusProfile? value) {
|
||||
setState(() {
|
||||
_bolusProfile = value;
|
||||
_bolusProfileController.text = (_bolusProfile ?? '').toString();
|
||||
});
|
||||
}
|
||||
|
||||
void handleSaveAction() async {
|
||||
setState(() {
|
||||
_isSaving = true;
|
||||
});
|
||||
if (_logEventTypeForm.currentState!.validate()) {
|
||||
LogEventType eventType = LogEventType(
|
||||
id: widget.id,
|
||||
value: _valueController.text,
|
||||
notes: _notesController.text,
|
||||
defaultReminderDuration: _defaultReminderDuration,
|
||||
hasEndTime: _hasEndTime,
|
||||
);
|
||||
eventType.basalProfile.target = _basalProfile;
|
||||
eventType.bolusProfile.target = _bolusProfile;
|
||||
LogEventType.put(eventType);
|
||||
Navigator.pop(
|
||||
context, ['${_isNew ? 'New' : ''} Log Event Type Saved', eventType]);
|
||||
}
|
||||
setState(() {
|
||||
_isSaving = false;
|
||||
});
|
||||
}
|
||||
|
||||
void handleCancelAction() {
|
||||
bool isNew = _logEventType == null;
|
||||
if (Settings.get().showConfirmationDialogOnCancel &&
|
||||
((isNew &&
|
||||
(_valueController.text != '' ||
|
||||
_defaultReminderDuration != 0 ||
|
||||
_notesController.text != '' ||
|
||||
_hasEndTime)) ||
|
||||
(!isNew &&
|
||||
(_valueController.text != _logEventType!.value ||
|
||||
_defaultReminderDuration !=
|
||||
_logEventType!.defaultReminderDuration ||
|
||||
_notesController.text != (_logEventType!.notes ?? '') ||
|
||||
_hasEndTime != _logEventType!.hasEndTime)))) {
|
||||
DialogUtils.showCancelConfirmationDialog(
|
||||
context: context,
|
||||
isNew: isNew,
|
||||
onSave: handleSaveAction,
|
||||
);
|
||||
} else {
|
||||
Navigator.pop(context);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Log Event Type' : _logEventType!.value),
|
||||
),
|
||||
drawer:
|
||||
const Navigation(currentLocation: EventTypeDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
controller: _scrollController,
|
||||
child: SingleChildScrollView(
|
||||
controller: _scrollController,
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||
children: <Widget>[
|
||||
FormWrapper(formState: _logEventTypeForm, fields: [
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
BooleanFormField(
|
||||
value: _hasEndTime,
|
||||
label: 'has end time',
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_hasEndTime = value;
|
||||
});
|
||||
},
|
||||
),
|
||||
Column(
|
||||
children: _hasEndTime
|
||||
? [
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(bottom: 10.0),
|
||||
child: DurationFormField(
|
||||
minutes: _defaultReminderDuration,
|
||||
label: 'Default Reminder Duration',
|
||||
onChanged: (value) => _defaultReminderDuration = value ?? 0,
|
||||
showSteppers: true,
|
||||
),
|
||||
),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(bottom: 10.0),
|
||||
child: Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: AutoCompleteDropdownButton<
|
||||
BolusProfile>(
|
||||
selectedItem: _bolusProfile,
|
||||
controller: _bolusProfileController,
|
||||
label: 'Bolus Profile',
|
||||
items: _bolusProfiles,
|
||||
onChanged: updateBolusProfile,
|
||||
),
|
||||
),
|
||||
IconButton(
|
||||
onPressed: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => _bolusProfile ==
|
||||
null
|
||||
? const BolusProfileDetailScreen()
|
||||
: BolusProfileDetailScreen(
|
||||
id: _bolusProfile!.id),
|
||||
),
|
||||
).then((result) {
|
||||
setState(() {
|
||||
updateBolusProfile(result?[1]);
|
||||
});
|
||||
reload(message: result?[0]);
|
||||
});
|
||||
},
|
||||
icon: Icon(_bolusProfile == null
|
||||
? Icons.add
|
||||
: Icons.edit),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child:
|
||||
AutoCompleteDropdownButton<BasalProfile>(
|
||||
controller: _basalProfileController,
|
||||
selectedItem: _basalProfile,
|
||||
label: 'Basal Profile',
|
||||
items: _basalProfiles,
|
||||
onChanged: updateBasalProfile,
|
||||
),
|
||||
),
|
||||
IconButton(
|
||||
onPressed: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => _basalProfile ==
|
||||
null
|
||||
? const BasalProfileDetailScreen()
|
||||
: BasalProfileDetailScreen(
|
||||
id: _basalProfile!.id),
|
||||
),
|
||||
).then((result) {
|
||||
updateBasalProfile(result?[1]);
|
||||
reload(message: result?[0]);
|
||||
});
|
||||
},
|
||||
icon: Icon(_basalProfile == null
|
||||
? Icons.add
|
||||
: Icons.edit),
|
||||
),
|
||||
],
|
||||
),
|
||||
]
|
||||
: []),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
maxLines: 5,
|
||||
),
|
||||
]),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
bottomNavigationBar: DetailBottomRow(
|
||||
onCancel: handleCancelAction,
|
||||
onAction: _isSaving ? null : handleSaveAction,
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
@ -1,123 +0,0 @@
|
||||
import 'package:diameter/models/log_event_type.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/log/log_event/log_event_type_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class LogEventTypeListScreen extends StatefulWidget {
|
||||
static const String routeName = '/log-event-types';
|
||||
const LogEventTypeListScreen({Key? key}) : super(key: key);
|
||||
|
||||
@override
|
||||
_LogEventTypeListScreenState createState() => _LogEventTypeListScreenState();
|
||||
}
|
||||
|
||||
class _LogEventTypeListScreenState extends State<LogEventTypeListScreen> {
|
||||
List<LogEventType> _logEventTypes = [];
|
||||
|
||||
final ScrollController _scrollController = ScrollController();
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
reload();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_scrollController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
void reload({String? message}) {
|
||||
setState(() {
|
||||
_logEventTypes = LogEventType.getAll();
|
||||
});
|
||||
setState(() {
|
||||
if (message != null) {
|
||||
var snackBar = SnackBar(
|
||||
content: Text(message),
|
||||
duration: const Duration(seconds: 2),
|
||||
);
|
||||
ScaffoldMessenger.of(context)
|
||||
..removeCurrentSnackBar()
|
||||
..showSnackBar(snackBar);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(title: const Text('Log Event Types'), actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
]),
|
||||
drawer:
|
||||
const Navigation(currentLocation: LogEventTypeListScreen.routeName),
|
||||
body: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: <Widget>[
|
||||
Expanded(
|
||||
child: _logEventTypes.isNotEmpty
|
||||
? Scrollbar(
|
||||
controller: _scrollController,
|
||||
child: ListView.builder(
|
||||
controller: _scrollController,
|
||||
padding: const EdgeInsets.all(10.0),
|
||||
itemCount: _logEventTypes.length,
|
||||
itemBuilder: (context, index) {
|
||||
final logEventType = _logEventTypes[index];
|
||||
return Card(
|
||||
child: ListTile(
|
||||
onTap: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => EventTypeDetailScreen(
|
||||
id: logEventType.id),
|
||||
),
|
||||
).then((result) => reload(message: result?[0]));
|
||||
},
|
||||
title: Text(
|
||||
logEventType.value.toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
subtitle: Text(logEventType.notes ?? ''),
|
||||
trailing: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
IconButton(
|
||||
onPressed: () async {
|
||||
LogEventType.remove(logEventType.id);
|
||||
reload(message: 'Log Event Type deleted');
|
||||
},
|
||||
icon: const Icon(Icons.delete,
|
||||
color: Colors.blue),
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
child:
|
||||
Text('You have not created any Log Event Types yet!'),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
floatingActionButton: FloatingActionButton(
|
||||
onPressed: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => const EventTypeDetailScreen(),
|
||||
),
|
||||
).then((result) => reload(message: result?[0]));
|
||||
},
|
||||
child: const Icon(Icons.add),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
230
lib/screens/log/log_filter_dialog.dart
Normal file
230
lib/screens/log/log_filter_dialog.dart
Normal file
@ -0,0 +1,230 @@
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/date_time_form_field.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/models/meal.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class LogFilterDialog extends StatefulWidget {
|
||||
static const String routeName = '/log/filter';
|
||||
|
||||
final DateTime? date;
|
||||
final void Function(
|
||||
{DateTime? startDate,
|
||||
DateTime? endDate,
|
||||
int? minMgPerDl,
|
||||
int? maxMgPerDl,
|
||||
double? minMmolPerL,
|
||||
double? maxMmolPerL,
|
||||
int? mealId,
|
||||
String? mealName,
|
||||
String? note}) onApplyFilter;
|
||||
|
||||
const LogFilterDialog({Key? key, this.date, required this.onApplyFilter})
|
||||
: super(key: key);
|
||||
|
||||
@override
|
||||
_LogFilterDialogState createState() => _LogFilterDialogState();
|
||||
}
|
||||
|
||||
class _LogFilterDialogState extends State<LogFilterDialog> {
|
||||
final ScrollController _scrollController = ScrollController();
|
||||
|
||||
late DateTime filterStartDate;
|
||||
late DateTime filterEndDate;
|
||||
|
||||
Meal? meal;
|
||||
List<Meal> _meals = [];
|
||||
|
||||
TextEditingController filterStartDateController =
|
||||
TextEditingController(text: '');
|
||||
TextEditingController filterEndDateController =
|
||||
TextEditingController(text: '');
|
||||
TextEditingController mealController = TextEditingController(text: '');
|
||||
TextEditingController minMgPerDlController = TextEditingController(text: '');
|
||||
TextEditingController maxMgPerDlController = TextEditingController(text: '');
|
||||
TextEditingController minMmolPerLController = TextEditingController(text: '');
|
||||
TextEditingController maxMmolPerLController = TextEditingController(text: '');
|
||||
TextEditingController mealNameController = TextEditingController(text: '');
|
||||
TextEditingController noteController = TextEditingController(text: '');
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_scrollController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
filterStartDate = widget.date ?? DateTimeUtils.today();
|
||||
filterEndDate = DateTimeUtils.today();
|
||||
|
||||
filterStartDateController.text = DateTimeUtils.displayDate(filterStartDate);
|
||||
filterEndDateController.text = DateTimeUtils.displayDate(filterEndDate);
|
||||
|
||||
_meals = Meal.getAll();
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return AlertDialog(
|
||||
content: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
crossAxisAlignment: CrossAxisAlignment.center,
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
Expanded(
|
||||
child: DateTimeFormField(
|
||||
date: filterStartDate,
|
||||
label: translate(LocalizationKeys.log_filter_startDate),
|
||||
controller: filterStartDateController,
|
||||
onChanged: (newDate) {
|
||||
if (newDate != null) {
|
||||
filterStartDate =
|
||||
DateTime(newDate.year, newDate.month, newDate.day);
|
||||
filterStartDateController.text =
|
||||
DateTimeUtils.displayDate(filterStartDate);
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 5.0),
|
||||
child: DateTimeFormField(
|
||||
date: filterEndDate,
|
||||
label: translate(LocalizationKeys.log_filter_endDate),
|
||||
controller: filterEndDateController,
|
||||
onChanged: (newDate) {
|
||||
if (newDate != null) {
|
||||
filterEndDate = DateTime(
|
||||
newDate.year, newDate.month, newDate.day);
|
||||
filterEndDateController.text =
|
||||
DateTimeUtils.displayDate(filterEndDate);
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
)
|
||||
],
|
||||
),
|
||||
Row(
|
||||
children: Settings.glucoseMeasurement ==
|
||||
GlucoseMeasurement.mgPerDl
|
||||
? [
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: translate(LocalizationKeys.log_filter_minGlucose, args: {
|
||||
"glucoseMeasurement": Settings.glucoseMeasurementSuffix
|
||||
}),
|
||||
controller: minMgPerDlController,
|
||||
onChanged: (value) {
|
||||
if (value != null) {
|
||||
minMgPerDlController.text = value.toString();
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 5.0),
|
||||
child: NumberFormField(
|
||||
label: translate(LocalizationKeys.log_filter_maxGlucose, args: {
|
||||
"glucoseMeasurement": Settings.glucoseMeasurementSuffix
|
||||
}),
|
||||
controller: maxMgPerDlController,
|
||||
onChanged: (value) {
|
||||
if (value != null) {
|
||||
maxMgPerDlController.text = value.toString();
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
)
|
||||
]
|
||||
: [
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: translate(LocalizationKeys.log_filter_minGlucose, args: {
|
||||
"glucoseMeasurement": Settings.glucoseMeasurementSuffix
|
||||
}),
|
||||
controller: minMmolPerLController,
|
||||
onChanged: (value) {
|
||||
if (value != null) {
|
||||
minMmolPerLController.text = value.toString();
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.only(left: 5.0),
|
||||
child: NumberFormField(
|
||||
label: translate(LocalizationKeys.log_filter_maxGlucose, args: {
|
||||
"glucoseMeasurement": Settings.glucoseMeasurementSuffix
|
||||
}),
|
||||
controller: maxMmolPerLController,
|
||||
onChanged: (value) {
|
||||
if (value != null) {
|
||||
maxMmolPerLController.text = value.toString();
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
)
|
||||
],
|
||||
),
|
||||
Expanded(
|
||||
child: AutoCompleteDropdownButton<Meal>(
|
||||
controller: mealController,
|
||||
selectedItem: meal,
|
||||
label: translate(LocalizationKeys.log_filter_meal),
|
||||
items: _meals,
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
meal = value;
|
||||
});
|
||||
},
|
||||
),
|
||||
),
|
||||
TextFormField(
|
||||
controller: mealNameController,
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_filter_mealNameContains),
|
||||
),
|
||||
),
|
||||
TextFormField(
|
||||
controller: noteController,
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.log_filter_noteContains),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
actions: <Widget>[
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context),
|
||||
child: Text(translate(LocalizationKeys.general_cancel).toUpperCase()),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: () => widget.onApplyFilter(
|
||||
startDate: filterStartDate,
|
||||
endDate: filterEndDate,
|
||||
minMgPerDl: int.tryParse(minMgPerDlController.text),
|
||||
maxMgPerDl: int.tryParse(maxMgPerDlController.text),
|
||||
minMmolPerL: double.tryParse(minMmolPerLController.text),
|
||||
maxMmolPerL: double.tryParse(maxMmolPerLController.text),
|
||||
mealId: meal?.id,
|
||||
mealName: mealNameController.text,
|
||||
note: noteController.text,
|
||||
),
|
||||
child: Text(translate(LocalizationKeys.general_apply).toUpperCase()),
|
||||
),
|
||||
]);
|
||||
}
|
||||
}
|
@ -1,3 +1,5 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/screens/log/log_filter_dialog.dart';
|
||||
import 'package:diameter/screens/reports/export.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/glucose_target.dart';
|
||||
@ -11,6 +13,9 @@ import 'package:diameter/utils/date_time_utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'dart:math' as math;
|
||||
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
|
||||
class LogScreen extends StatefulWidget {
|
||||
static const String routeName = '/log';
|
||||
const LogScreen({Key? key}) : super(key: key);
|
||||
@ -65,7 +70,7 @@ class _LogScreenState extends State<LogScreen> {
|
||||
|
||||
void onDelete(LogEntry logEntry) {
|
||||
LogEntry.remove(logEntry.id);
|
||||
reload(message: 'Log Entry deleted');
|
||||
reload(message: translate(LocalizationKeys.log_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(LogEntry logEntry) async {
|
||||
@ -73,7 +78,7 @@ class _LogScreenState extends State<LogScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(logEntry),
|
||||
message: 'Are you sure you want to delete this Log Entry?',
|
||||
message: translate(LocalizationKeys.log_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(logEntry);
|
||||
@ -90,15 +95,48 @@ class _LogScreenState extends State<LogScreen> {
|
||||
}
|
||||
}
|
||||
|
||||
void onChangeFilter(
|
||||
{DateTime? startDate,
|
||||
DateTime? endDate,
|
||||
int? minMgPerDl,
|
||||
int? maxMgPerDl,
|
||||
double? minMmolPerL,
|
||||
double? maxMmolPerL,
|
||||
int? mealId,
|
||||
String? mealName,
|
||||
String? note}) {
|
||||
setState(() {
|
||||
_logEntries = LogEntry.getAllByFilter(
|
||||
startDate: startDate,
|
||||
endDate: endDate,
|
||||
minMgPerDl: minMgPerDl,
|
||||
maxMgPerDl: maxMgPerDl,
|
||||
minMmolPerL: minMmolPerL,
|
||||
maxMmolPerL: maxMmolPerL,
|
||||
mealId: mealId,
|
||||
mealName: mealName,
|
||||
note: note,
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Log Entries'),
|
||||
title: Text(translate(LocalizationKeys.log_title)),
|
||||
actions: <Widget>[
|
||||
IconButton(
|
||||
onPressed: () => onChangeDate(DateTime.now()),
|
||||
icon: const Icon(Icons.today)),
|
||||
IconButton(
|
||||
onPressed: () => showDialog(
|
||||
context: context,
|
||||
builder: (context) => LogFilterDialog(
|
||||
date: _date,
|
||||
onApplyFilter: onChangeFilter,
|
||||
)),
|
||||
icon: const Icon(Icons.filter_list)),
|
||||
IconButton(
|
||||
onPressed: () => showDialog(
|
||||
context: context,
|
||||
@ -297,7 +335,7 @@ class _LogScreenState extends State<LogScreen> {
|
||||
? [
|
||||
Text(
|
||||
bolus.toStringAsPrecision(3)),
|
||||
const Text('U',
|
||||
Text(translate(LocalizationKeys.general_suffixes_units),
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
: [],
|
||||
@ -309,8 +347,12 @@ class _LogScreenState extends State<LogScreen> {
|
||||
? [
|
||||
Text(
|
||||
carbs.toStringAsPrecision(3)),
|
||||
Text(
|
||||
'${Settings.nutritionMeasurementSuffix} carbs',
|
||||
Text(translate(
|
||||
LocalizationKeys.general_suffixes_carbs,
|
||||
args: {
|
||||
"nutritionMeasurementSuffix": Settings.nutritionMeasurementSuffix,
|
||||
}
|
||||
),
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
: [],
|
||||
@ -334,9 +376,10 @@ class _LogScreenState extends State<LogScreen> {
|
||||
},
|
||||
),
|
||||
)
|
||||
: const Center(
|
||||
: Center(
|
||||
child: Text(
|
||||
'You have not created any Log Entries for this date yet!'),
|
||||
translate(LocalizationKeys.log_empty)
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,6 +1,7 @@
|
||||
import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
@ -17,6 +18,7 @@ import 'package:diameter/screens/category/meal_portion_type_detail.dart';
|
||||
import 'package:diameter/screens/category/meal_source_detail.dart';
|
||||
import 'package:diameter/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealDetailScreen extends StatefulWidget {
|
||||
static const String routeName = '/meal';
|
||||
@ -195,7 +197,9 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
meal.carbsRatioAccuracy.target = _carbsRatioAccuracy;
|
||||
|
||||
Meal.put(meal);
|
||||
Navigator.pop(context, ['${_isNew ? 'New' : ''} Meal Saved', meal]);
|
||||
Navigator.pop(context, [translate(LocalizationKeys.meal_saved, args: {
|
||||
"status": _isNew ? translate(LocalizationKeys.meal_new) : '',
|
||||
}), meal]);
|
||||
}
|
||||
setState(() {
|
||||
_isSaving = false;
|
||||
@ -340,7 +344,10 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: Text(_isNew ? 'New Meal' : _meal!.value),
|
||||
title: Text(translate(LocalizationKeys.meal_saved, args: {
|
||||
"status": _isNew ? translate(LocalizationKeys.meal_new) : '',
|
||||
"name": _meal?.value,
|
||||
})),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: MealDetailScreen.routeName),
|
||||
body: Scrollbar(
|
||||
@ -354,12 +361,12 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
fields: [
|
||||
TextFormField(
|
||||
controller: _valueController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Name',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.meal_fields_name),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty name';
|
||||
return translate(LocalizationKeys.meal_fields_validators_name);
|
||||
}
|
||||
return null;
|
||||
},
|
||||
@ -370,7 +377,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<MealSource>(
|
||||
controller: _mealSourceController,
|
||||
selectedItem: _mealSource,
|
||||
label: 'Meal Source',
|
||||
label: translate(LocalizationKeys.meal_fields_mealSource),
|
||||
items: _mealSources,
|
||||
onChanged: onSelectMealSource,
|
||||
),
|
||||
@ -400,7 +407,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
child: AutoCompleteDropdownButton<MealPortionType>(
|
||||
controller: _mealPortionTypeController,
|
||||
selectedItem: _mealPortionType,
|
||||
label: 'Meal Portion Type',
|
||||
label: translate(LocalizationKeys.meal_fields_mealPortionType),
|
||||
items: _mealPortionTypes,
|
||||
onChanged: updateMealPortionType,
|
||||
),
|
||||
@ -429,7 +436,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: 'Carbs ratio',
|
||||
label: translate(LocalizationKeys.meal_fields_carbsRatio),
|
||||
suffix: '%',
|
||||
controller: _carbsRatioController,
|
||||
showSteppers: false,
|
||||
@ -445,7 +452,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
const SizedBox(width: 10),
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: 'Portion size',
|
||||
label: translate(LocalizationKeys.meal_fields_portionSize),
|
||||
suffix: Settings.nutritionMeasurementSuffix,
|
||||
controller: _portionSizeController,
|
||||
showSteppers: false,
|
||||
@ -461,7 +468,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
const SizedBox(width: 10),
|
||||
Expanded(
|
||||
child: NumberFormField(
|
||||
label: 'Carbs per portion',
|
||||
label: translate(LocalizationKeys.meal_fields_carbsPerPortion),
|
||||
suffix: Settings.nutritionMeasurementSuffix,
|
||||
controller: _carbsPerPortionController,
|
||||
showSteppers: false,
|
||||
@ -479,7 +486,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
Expanded(
|
||||
child: BooleanFormField(
|
||||
value: _setManually,
|
||||
label: 'set carbs ratio manually',
|
||||
label: translate(LocalizationKeys.meal_fields_setManually),
|
||||
onChanged: (value) {
|
||||
setState(() {
|
||||
_setManually = value;
|
||||
@ -490,8 +497,8 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
),
|
||||
TextFormField(
|
||||
controller: _notesController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Notes',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.meal_fields_notes),
|
||||
),
|
||||
keyboardType: TextInputType.multiline,
|
||||
minLines: 2,
|
||||
@ -503,7 +510,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
child: Row(
|
||||
children: [
|
||||
Text(
|
||||
'BOLUS DELAY',
|
||||
translate(LocalizationKeys.meal_fields_delay_title).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
const Spacer(),
|
||||
@ -514,9 +521,9 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: TextFormField(
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Duration',
|
||||
suffixText: ' min',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.meal_fields_delay_duration),
|
||||
suffixText: translate(LocalizationKeys.general_suffixes_mins),
|
||||
),
|
||||
controller: _delayedBolusDurationController,
|
||||
onChanged: (value) => setState(() {}),
|
||||
@ -553,7 +560,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: Text(
|
||||
'ADDITIONAL FIELDS',
|
||||
translate(LocalizationKeys.meal_fields_additional_title).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
),
|
||||
@ -574,7 +581,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
AutoCompleteDropdownButton<MealCategory>(
|
||||
controller: _mealCategoryController,
|
||||
selectedItem: _mealCategory,
|
||||
label: 'Meal Category',
|
||||
label: translate(LocalizationKeys.meal_fields_additional_mealCategory),
|
||||
items: _mealCategories,
|
||||
onChanged: updateMealCategory,
|
||||
),
|
||||
@ -609,7 +616,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
AutoCompleteDropdownButton<Accuracy>(
|
||||
controller: _portionSizeAccuracyController,
|
||||
selectedItem: _portionSizeAccuracy,
|
||||
label: 'Portion Size Accuracy',
|
||||
label: translate(LocalizationKeys.meal_fields_additional_portionSizeAccuracy),
|
||||
items: _portionSizeAccuracies,
|
||||
onChanged: updatePortionSizeAccuracy,
|
||||
),
|
||||
@ -645,7 +652,7 @@ class _MealDetailScreenState extends State<MealDetailScreen> {
|
||||
AutoCompleteDropdownButton<Accuracy>(
|
||||
controller: _carbsRatioAccuracyController,
|
||||
selectedItem: _carbsRatioAccuracy,
|
||||
label: 'Carbs Ratio Accuracy',
|
||||
label: translate(LocalizationKeys.meal_fields_additional_carbsRatioAccuracy),
|
||||
items: _carbsRatioAccuracies,
|
||||
onChanged: updateCarbsRatioAccuracy,
|
||||
),
|
||||
|
@ -1,9 +1,11 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/meal.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/meal/meal_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class MealListScreen extends StatefulWidget {
|
||||
static const String routeName = '/meals';
|
||||
@ -50,7 +52,7 @@ class _MealListScreenState extends State<MealListScreen> {
|
||||
|
||||
void onDelete(Meal meal) {
|
||||
Meal.remove(meal.id);
|
||||
reload(message: 'Meal deleted');
|
||||
reload(message: translate(LocalizationKeys.meal_deleted));
|
||||
}
|
||||
|
||||
void handleDeleteAction(Meal meal) async {
|
||||
@ -58,7 +60,7 @@ class _MealListScreenState extends State<MealListScreen> {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: () => onDelete(meal),
|
||||
message: 'Are you sure you want to delete this Meal?',
|
||||
message: translate(LocalizationKeys.meal_confirmDelete),
|
||||
);
|
||||
} else {
|
||||
onDelete(meal);
|
||||
@ -68,7 +70,11 @@ class _MealListScreenState extends State<MealListScreen> {
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(title: const Text('Meals'), actions: <Widget>[
|
||||
appBar: AppBar(
|
||||
title: Text(
|
||||
translate(LocalizationKeys.meal_title)
|
||||
),
|
||||
actions: <Widget>[
|
||||
IconButton(onPressed: reload, icon: const Icon(Icons.refresh))
|
||||
]),
|
||||
drawer: const Navigation(currentLocation: MealListScreen.routeName),
|
||||
@ -84,7 +90,7 @@ class _MealListScreenState extends State<MealListScreen> {
|
||||
itemCount: _meals.length,
|
||||
itemBuilder: (context, index) {
|
||||
final meal = _meals[index];
|
||||
String portionType = meal.mealPortionType.hasValue ? ' per ${meal.mealPortionType.target!.value}' : '';
|
||||
String portionType = meal.mealPortionType.hasValue ? ' ${translate(LocalizationKeys.general_per)} ${meal.mealPortionType.target!.value}' : '';
|
||||
return Card(
|
||||
child: ListTile(
|
||||
isThreeLine: true,
|
||||
@ -118,7 +124,7 @@ class _MealListScreenState extends State<MealListScreen> {
|
||||
? [
|
||||
Text(meal.carbsPerPortion!.toStringAsPrecision(3)),
|
||||
Text(
|
||||
'${Settings.nutritionMeasurementSuffix} carbs',
|
||||
'${Settings.nutritionMeasurementSuffix} ${translate(LocalizationKeys.general_suffixes_carbs)}',
|
||||
textScaleFactor: 0.75),
|
||||
]
|
||||
: [],
|
||||
@ -166,8 +172,8 @@ class _MealListScreenState extends State<MealListScreen> {
|
||||
);
|
||||
},
|
||||
),
|
||||
): const Center(
|
||||
child: Text('You have not created any Meals yet!'),
|
||||
): Center(
|
||||
child: Text(translate(LocalizationKeys.meal_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -1,4 +1,5 @@
|
||||
import 'package:charts_flutter/flutter.dart' as charts;
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/models/basal.dart';
|
||||
import 'package:diameter/models/glucose_target.dart';
|
||||
import 'package:diameter/models/log_bolus.dart';
|
||||
@ -9,6 +10,7 @@ import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
charts.TimeSeriesChart generateChart(
|
||||
DateTime date,
|
||||
@ -48,7 +50,7 @@ charts.TimeSeriesChart generateChart(
|
||||
return charts.TimeSeriesChart(
|
||||
[
|
||||
charts.Series<LogEntry, DateTime>(
|
||||
id: 'Glucose',
|
||||
id: translate(LocalizationKeys.reports_ids_glucose),
|
||||
colorFn: (LogEntry entry, _) => charts.Color.black,
|
||||
domainFn: (LogEntry entry, _) => entry.time,
|
||||
measureFn: (LogEntry entry, _) =>
|
||||
@ -58,7 +60,7 @@ charts.TimeSeriesChart generateChart(
|
||||
data: logEntries,
|
||||
)..setAttribute(charts.rendererIdKey, 'glucoseRenderer'),
|
||||
charts.Series<LogEntry, DateTime>(
|
||||
id: 'Carbohydrates',
|
||||
id: translate(LocalizationKeys.reports_ids_carbs),
|
||||
colorFn: (LogEntry entry, _) =>
|
||||
charts.MaterialPalette.yellow.shadeDefault,
|
||||
domainFn: (LogEntry entry, _) => entry.time,
|
||||
@ -67,7 +69,7 @@ charts.TimeSeriesChart generateChart(
|
||||
data: showMeals ? logEntries : [],
|
||||
)..setAttribute(charts.rendererIdKey, 'carbsRenderer'),
|
||||
charts.Series<Basal, DateTime>(
|
||||
id: 'Basal',
|
||||
id: translate(LocalizationKeys.reports_ids_basal),
|
||||
colorFn: (Basal basal, _) => charts.Color.black.lighter,
|
||||
domainFn: (Basal basal, _) => basal.startTime,
|
||||
measureFn: (Basal basal, _) => basal.units,
|
||||
@ -77,7 +79,7 @@ charts.TimeSeriesChart generateChart(
|
||||
charts.measureAxisIdKey, charts.Axis.secondaryMeasureAxisId)
|
||||
..setAttribute(charts.rendererIdKey, 'basalRenderer'),
|
||||
charts.Series<LogEntry, DateTime>(
|
||||
id: 'Bolus',
|
||||
id: translate(LocalizationKeys.reports_ids_bolus),
|
||||
colorFn: (LogEntry entry, _) =>
|
||||
charts.MaterialPalette.blue.shadeDefault,
|
||||
domainFn: (LogEntry entry, _) => entry.time,
|
||||
@ -260,7 +262,7 @@ class _DailyChartState extends State<DailyChart> {
|
||||
value: showChart,
|
||||
onChanged: (_) =>
|
||||
setState(() => showChart = !showChart),
|
||||
title: const Text('show Chart'),
|
||||
title: Text(translate(LocalizationKeys.reports_dailyCharts_showChart)),
|
||||
controlAffinity: ListTileControlAffinity.leading,
|
||||
),
|
||||
Padding(
|
||||
@ -271,7 +273,7 @@ class _DailyChartState extends State<DailyChart> {
|
||||
? (_) =>
|
||||
setState(() => showBolus = !showBolus)
|
||||
: null,
|
||||
title: const Text('show Bolus'),
|
||||
title: Text(translate(LocalizationKeys.reports_dailyCharts_showBolus)),
|
||||
controlAffinity:
|
||||
ListTileControlAffinity.leading,
|
||||
),
|
||||
@ -284,7 +286,7 @@ class _DailyChartState extends State<DailyChart> {
|
||||
? (_) =>
|
||||
setState(() => showBasal = !showBasal)
|
||||
: null,
|
||||
title: const Text('show Basal'),
|
||||
title: Text(translate(LocalizationKeys.reports_dailyCharts_showBasal)),
|
||||
controlAffinity:
|
||||
ListTileControlAffinity.leading,
|
||||
),
|
||||
@ -297,7 +299,7 @@ class _DailyChartState extends State<DailyChart> {
|
||||
? (_) =>
|
||||
setState(() => showMeals = !showMeals)
|
||||
: null,
|
||||
title: const Text('show Meals'),
|
||||
title: Text(translate(LocalizationKeys.reports_dailyCharts_showMeals)),
|
||||
controlAffinity:
|
||||
ListTileControlAffinity.leading,
|
||||
),
|
||||
@ -307,7 +309,7 @@ class _DailyChartState extends State<DailyChart> {
|
||||
actions: <Widget>[
|
||||
ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context),
|
||||
child: const Text('CLOSE'),
|
||||
child: Text(translate(LocalizationKeys.general_close).toUpperCase()),
|
||||
),
|
||||
]),
|
||||
),
|
||||
@ -378,9 +380,8 @@ class _DailyChartState extends State<DailyChart> {
|
||||
child: logEntries.isNotEmpty
|
||||
? generateChart(date, logEntries, logEvents, targets, showMeals,
|
||||
showBasal, showBolus)
|
||||
: const Center(
|
||||
child: Text(
|
||||
'You have not created any Log Entries for this date yet!'),
|
||||
: Center(
|
||||
child: Text(translate(LocalizationKeys.reports_dailyCharts_empty)),
|
||||
),
|
||||
),
|
||||
],
|
||||
|
@ -2,6 +2,7 @@ import 'dart:io';
|
||||
import 'dart:math';
|
||||
import 'dart:typed_data';
|
||||
import 'package:diameter/components/forms/date_time_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/models/glucose_target.dart';
|
||||
import 'package:diameter/models/log_bolus.dart';
|
||||
import 'package:diameter/models/log_entry.dart';
|
||||
@ -12,6 +13,7 @@ import 'package:diameter/screens/reports/daily_chart.dart';
|
||||
import 'package:diameter/utils/date_time_utils.dart';
|
||||
import 'package:diameter/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
import 'package:open_file/open_file.dart';
|
||||
import 'package:path_provider/path_provider.dart';
|
||||
@ -121,7 +123,7 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: RadioListTile(
|
||||
title: const Text('single date'),
|
||||
title: Text(translate(LocalizationKeys.reports_export_singleDate)),
|
||||
groupValue: exportRange,
|
||||
value: false,
|
||||
onChanged: (_) {
|
||||
@ -132,7 +134,7 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
),
|
||||
Expanded(
|
||||
child: RadioListTile(
|
||||
title: const Text('range'),
|
||||
title: Text(translate(LocalizationKeys.reports_export_range)),
|
||||
groupValue: exportRange,
|
||||
value: true,
|
||||
onChanged: (value) {
|
||||
@ -148,7 +150,7 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
Expanded(
|
||||
child: DateTimeFormField(
|
||||
date: exportStartDate,
|
||||
label: 'Date',
|
||||
label: translate(LocalizationKeys.reports_export_date),
|
||||
controller: exportStartDateController,
|
||||
onChanged: (newDate) {
|
||||
if (newDate != null) {
|
||||
@ -166,7 +168,7 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
padding: const EdgeInsets.only(left: 5.0),
|
||||
child: DateTimeFormField(
|
||||
date: exportEndDate,
|
||||
label: 'End Date',
|
||||
label: translate(LocalizationKeys.reports_export_endDate),
|
||||
controller: exportEndDateController,
|
||||
onChanged: (newDate) {
|
||||
if (newDate != null) {
|
||||
@ -185,7 +187,7 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
CheckboxListTile(
|
||||
value: showChart,
|
||||
onChanged: (_) => setState(() => showChart = !showChart),
|
||||
title: const Text('show Chart'),
|
||||
title: Text(translate(LocalizationKeys.reports_export_showChart)),
|
||||
controlAffinity: ListTileControlAffinity.leading,
|
||||
),
|
||||
Padding(
|
||||
@ -195,7 +197,7 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
onChanged: showChart
|
||||
? (_) => setState(() => showBolus = !showBolus)
|
||||
: null,
|
||||
title: const Text('show Bolus'),
|
||||
title: Text(translate(LocalizationKeys.reports_export_showBolus)),
|
||||
controlAffinity: ListTileControlAffinity.leading,
|
||||
),
|
||||
),
|
||||
@ -206,7 +208,7 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
onChanged: showChart
|
||||
? (_) => setState(() => showBasal = !showBasal)
|
||||
: null,
|
||||
title: const Text('show Basal'),
|
||||
title: Text(translate(LocalizationKeys.reports_export_showBasal)),
|
||||
controlAffinity: ListTileControlAffinity.leading,
|
||||
),
|
||||
),
|
||||
@ -217,14 +219,14 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
onChanged: showChart
|
||||
? (_) => setState(() => showMeals = !showMeals)
|
||||
: null,
|
||||
title: const Text('show Meals'),
|
||||
title: Text(translate(LocalizationKeys.reports_export_showMeals)),
|
||||
controlAffinity: ListTileControlAffinity.leading,
|
||||
),
|
||||
),
|
||||
CheckboxListTile(
|
||||
value: showTable,
|
||||
onChanged: (_) => setState(() => showTable = !showTable),
|
||||
title: const Text('show Table'),
|
||||
title: Text(translate(LocalizationKeys.reports_export_showTable)),
|
||||
controlAffinity: ListTileControlAffinity.leading,
|
||||
),
|
||||
],
|
||||
@ -232,13 +234,13 @@ class _ExportDialogState extends State<ExportDialog> {
|
||||
actions: <Widget>[
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context),
|
||||
child: const Text('CANCEL'),
|
||||
child: Text(translate(LocalizationKeys.general_cancel).toUpperCase()),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: !_isSaving && (showChart || showTable)
|
||||
? () => onExport(context)
|
||||
: null,
|
||||
child: const Text('EXPORT'),
|
||||
child: Text(translate(LocalizationKeys.reports_export_export).toUpperCase()),
|
||||
),
|
||||
]);
|
||||
}
|
||||
@ -273,15 +275,14 @@ pw.Table generateTable(List<LogEntry> logEntries) {
|
||||
const small = pw.TextStyle(fontSize: 8.0);
|
||||
|
||||
final tableHeaders = [
|
||||
'Time',
|
||||
'Glucose',
|
||||
'Bolus',
|
||||
'Notes',
|
||||
'Meals',
|
||||
'Portion Size',
|
||||
'Carbohydrates',
|
||||
'Meal Bolus',
|
||||
'Meal Notes',
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_time),
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_glucose),
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_notes),
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_meals),
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_portionSize),
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_carbs),
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_mealBolus),
|
||||
translate(LocalizationKeys.reports_export_tableHeaders_mealNotes),
|
||||
];
|
||||
|
||||
final List<pw.TableRow> data = logEntries.map((logEntry) {
|
||||
@ -304,7 +305,9 @@ pw.Table generateTable(List<LogEntry> logEntries) {
|
||||
for (var bolus in boli
|
||||
.where((bolus) => bolus.meal.targetId == meal.id)
|
||||
.toList())
|
||||
'${bolus.units} U${bolus.delay != null ? ' (over ${bolus.delay} min)' : ''}'
|
||||
'${bolus.units} ${translate(LocalizationKeys.general_suffixes_units)} ${bolus.delay != null ? translate(LocalizationKeys.log_detail_tabs_bolus_delayedBy, args: {
|
||||
"delay": '${bolus.delay} ${translate(LocalizationKeys.general_suffixes_mins)}'
|
||||
}) : ''}'
|
||||
].join(' + '),
|
||||
notes: meal.notes ?? '',
|
||||
))
|
||||
@ -320,7 +323,9 @@ pw.Table generateTable(List<LogEntry> logEntries) {
|
||||
bolus.meal.targetId != 0 &&
|
||||
!meals.any((meal) => meal.id == bolus.meal.targetId))
|
||||
.map((bolus) =>
|
||||
'${bolus.units} U${bolus.delay != null ? ' (over ${bolus.delay} min)' : ''}${bolus.meal.target != null ? ' (for ${bolus.meal.target!.value})' : ''}')
|
||||
'${bolus.units} ${translate(LocalizationKeys.general_suffixes_units)} ${bolus.delay != null ? translate(LocalizationKeys.log_detail_tabs_bolus_delayedBy, args: {
|
||||
"delay": '${bolus.delay} ${translate(LocalizationKeys.general_suffixes_mins)}'
|
||||
}) : ''}')
|
||||
.toList();
|
||||
|
||||
return pw.TableRow(
|
||||
@ -358,7 +363,7 @@ pw.Table generateTable(List<LogEntry> logEntries) {
|
||||
),
|
||||
// Bolus
|
||||
pw.Text(
|
||||
[for (var bolus in glucoseBoli) '${bolus.units} U'].join(' + ')),
|
||||
[for (var bolus in glucoseBoli) '${bolus.units} ${translate(LocalizationKeys.general_suffixes_units)}'].join(' + ')),
|
||||
// Notes
|
||||
pw.Text(logEntry.notes ?? '', style: small),
|
||||
// Meals
|
||||
@ -449,7 +454,7 @@ Future<Uint8List> generateLogReport(
|
||||
return pw.Column(
|
||||
children: [
|
||||
pw.Text(
|
||||
'LOG REPORT',
|
||||
translate(LocalizationKeys.reports_export_title).toUpperCase(),
|
||||
style: pw.TextStyle(
|
||||
fontWeight: pw.FontWeight.bold,
|
||||
fontSize: 12,
|
||||
|
@ -1,6 +1,8 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/reports/export.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class ReportsOverviewScreen extends StatefulWidget {
|
||||
static const String routeName = '/reports';
|
||||
@ -23,7 +25,7 @@ class _ReportsOverviewScreenState extends State<ReportsOverviewScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Reports'),
|
||||
title: Text(translate(LocalizationKeys.reports_title)),
|
||||
),
|
||||
drawer:
|
||||
const Navigation(currentLocation: ReportsOverviewScreen.routeName),
|
||||
@ -47,7 +49,7 @@ class _ReportsOverviewScreenState extends State<ReportsOverviewScreen> {
|
||||
child: Icon(Icons.today, size: 50, color: Theme.of(context).textTheme.subtitle2?.color),
|
||||
),
|
||||
Text(
|
||||
'DAILY REPORT',
|
||||
translate(LocalizationKeys.reports_sections_dailyReport).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center
|
||||
),
|
||||
@ -72,7 +74,7 @@ class _ReportsOverviewScreenState extends State<ReportsOverviewScreen> {
|
||||
child: Icon(Icons.today, size: 50, color: Theme.of(context).textTheme.subtitle2?.color),
|
||||
),
|
||||
Text(
|
||||
'PDF REPORT',
|
||||
translate(LocalizationKeys.reports_sections_pdfReport).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
textAlign: TextAlign.center
|
||||
),
|
||||
|
@ -2,9 +2,9 @@ import 'package:diameter/components/detail.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/components/forms/form_wrapper.dart';
|
||||
import 'package:diameter/models/ingredient.dart';
|
||||
import 'package:diameter/models/x_ingredient.dart';
|
||||
import 'package:diameter/models/meal.dart';
|
||||
import 'package:diameter/models/recipe.dart';
|
||||
import 'package:diameter/models/x_recipe.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/meal/meal_detail.dart';
|
@ -1,9 +1,9 @@
|
||||
import 'package:diameter/screens/x_recipe/recipe_detail.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/models/ingredient.dart';
|
||||
import 'package:diameter/models/recipe.dart';
|
||||
import 'package:diameter/models/x_ingredient.dart';
|
||||
import 'package:diameter/models/x_recipe.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/screens/recipe/recipe_detail.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class RecipeListScreen extends StatefulWidget {
|
@ -1,11 +1,13 @@
|
||||
import 'package:diameter/components/forms/boolean_form_field.dart';
|
||||
import 'package:diameter/components/forms/number_form_field.dart';
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:diameter/utils/dialog_utils.dart';
|
||||
import 'package:diameter/components/forms/auto_complete_dropdown_button.dart';
|
||||
import 'package:diameter/models/settings.dart';
|
||||
import 'package:diameter/navigation.dart';
|
||||
import 'package:diameter/utils/utils.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
|
||||
class SettingsScreen extends StatefulWidget {
|
||||
@ -152,19 +154,19 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
showConfirmationDialogOnDelete: _showConfirmationDialogOnDelete,
|
||||
showConfirmationDialogOnStopEvent: _showConfirmationDialogOnStopEvent,
|
||||
));
|
||||
reload(message: 'Settings updated');
|
||||
reload(message: translate(LocalizationKeys.settings_updated));
|
||||
}
|
||||
|
||||
void onReset() {
|
||||
Settings.reset();
|
||||
reload(message: 'Settings have been reset to default');
|
||||
reload(message: translate(LocalizationKeys.settings_reset));
|
||||
}
|
||||
|
||||
void handleResetAction() async {
|
||||
DialogUtils.showConfirmationDialog(
|
||||
context: context,
|
||||
onConfirm: onReset,
|
||||
message: 'Are you sure you want to reset all settings?',
|
||||
message: translate(LocalizationKeys.settings_confirmReset),
|
||||
);
|
||||
}
|
||||
|
||||
@ -172,7 +174,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
appBar: AppBar(
|
||||
title: const Text('Application Settings'),
|
||||
title: Text(translate(LocalizationKeys.settings_title)),
|
||||
),
|
||||
drawer: const Navigation(currentLocation: SettingsScreen.routeName),
|
||||
body: SingleChildScrollView(
|
||||
@ -191,7 +193,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: Text(
|
||||
'MEASUREMENTS',
|
||||
translate(LocalizationKeys.settings_sections_measurements).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
),
|
||||
@ -208,7 +210,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
AutoCompleteDropdownButton<String>(
|
||||
controller: _nutritionMeasurementLabelController,
|
||||
selectedItem: _nutritionMeasurementLabelController.text,
|
||||
label: 'Preferred Nutrition Measurement',
|
||||
label: translate(LocalizationKeys.settings_fields_nutritionMeasurement),
|
||||
items: nutritionMeasurementLabels,
|
||||
onChanged: (value) {
|
||||
_nutritionMeasurementLabelController.text =
|
||||
@ -221,7 +223,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
child: AutoCompleteDropdownButton<String>(
|
||||
controller: _glucoseMeasurementLabelController,
|
||||
selectedItem: _glucoseMeasurementLabelController.text,
|
||||
label: 'Preferred Glucose Measurement',
|
||||
label: translate(LocalizationKeys.settings_fields_glucoseMeasurement),
|
||||
items: glucoseMeasurementLabels,
|
||||
onChanged: (value) {
|
||||
_glucoseMeasurementLabelController.text =
|
||||
@ -232,8 +234,8 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
),
|
||||
Settings.glucoseMeasurement == GlucoseMeasurement.mgPerDl
|
||||
? NumberFormField(
|
||||
label: 'Target glucose',
|
||||
suffix: 'mg/dl',
|
||||
label: translate(LocalizationKeys.settings_fields_targetGlucose),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
controller: _targetGlucoseMgPerDlController,
|
||||
showSteppers: false,
|
||||
onChanged: (_) async {
|
||||
@ -254,8 +256,8 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
},
|
||||
)
|
||||
: NumberFormField(
|
||||
label: 'Target glucose',
|
||||
suffix: 'mmol/l',
|
||||
label: translate(LocalizationKeys.settings_fields_targetGlucose),
|
||||
suffix: Settings.glucoseMeasurementSuffix,
|
||||
controller: _targetGlucoseMmolPerLController,
|
||||
showSteppers: false,
|
||||
onChanged: (_) async {
|
||||
@ -277,7 +279,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
child: NumberFormField(
|
||||
controller: _insulinIncrementsController,
|
||||
showSteppers: false,
|
||||
label: 'Insulin increment',
|
||||
label: translate(LocalizationKeys.settings_fields_insulinIncrement),
|
||||
onChanged: (value) {
|
||||
_insulinIncrementsController.text =
|
||||
(value ?? 0).toString();
|
||||
@ -287,7 +289,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
NumberFormField(
|
||||
controller: _nutritionIncrementsController,
|
||||
showSteppers: false,
|
||||
label: 'Nutrition increment',
|
||||
label: translate(LocalizationKeys.settings_fields_nutritionIncrement),
|
||||
onChanged: (value) {
|
||||
_nutritionIncrementsController.text =
|
||||
(value ?? 0).toString();
|
||||
@ -298,7 +300,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
child: NumberFormField(
|
||||
controller: _mmolPerLIncrementsController,
|
||||
showSteppers: false,
|
||||
label: 'Mmol/L increment',
|
||||
label: translate(LocalizationKeys.settings_fields_mmolLIncrement),
|
||||
onChanged: (value) {
|
||||
_mmolPerLIncrementsController.text =
|
||||
(value ?? 0).toString();
|
||||
@ -307,7 +309,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
),
|
||||
BooleanFormField(
|
||||
value: _onlyDisplayActiveGlucoseMeasurement,
|
||||
label: 'only display active glucose measurement',
|
||||
label: translate(LocalizationKeys.settings_fields_onlyDisplayActive),
|
||||
onChanged: (value) {
|
||||
_onlyDisplayActiveGlucoseMeasurement = value;
|
||||
saveSettings();
|
||||
@ -317,7 +319,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
value: _displayBothGlucoseMeasurementsInDetailView,
|
||||
enabled: !_onlyDisplayActiveGlucoseMeasurement,
|
||||
label:
|
||||
'display both glucose measurements in detail view',
|
||||
translate(LocalizationKeys.settings_fields_displayBothDetail),
|
||||
onChanged: (value) {
|
||||
_displayBothGlucoseMeasurementsInDetailView = value;
|
||||
saveSettings();
|
||||
@ -326,7 +328,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
BooleanFormField(
|
||||
value: _displayBothGlucoseMeasurementsInListView,
|
||||
enabled: !_onlyDisplayActiveGlucoseMeasurement,
|
||||
label: 'display both glucose measurements in list view',
|
||||
label: translate(LocalizationKeys.settings_fields_displayBothList),
|
||||
onChanged: (value) {
|
||||
_displayBothGlucoseMeasurementsInListView = value;
|
||||
saveSettings();
|
||||
@ -347,7 +349,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: Text(
|
||||
'CONFIRMATION PROMPTS',
|
||||
translate(LocalizationKeys.settings_sections_confirmation).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
),
|
||||
@ -364,7 +366,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
BooleanFormField(
|
||||
value: _showConfirmationDialogOnCancel,
|
||||
label:
|
||||
'on cancelling edit or creation of a record if changes have already been made',
|
||||
translate(LocalizationKeys.settings_fields_confirmOnCancel),
|
||||
onChanged: (value) {
|
||||
_showConfirmationDialogOnCancel = value;
|
||||
saveSettings();
|
||||
@ -372,7 +374,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
),
|
||||
BooleanFormField(
|
||||
value: _showConfirmationDialogOnDelete,
|
||||
label: 'on deleting a record',
|
||||
label: translate(LocalizationKeys.settings_fields_confirmOnDelete),
|
||||
onChanged: (value) {
|
||||
_showConfirmationDialogOnDelete = value;
|
||||
saveSettings();
|
||||
@ -380,7 +382,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
),
|
||||
BooleanFormField(
|
||||
value: _showConfirmationDialogOnStopEvent,
|
||||
label: 'on stopping (ending) an event',
|
||||
label: translate(LocalizationKeys.settings_fields_confirmOnEndEvent),
|
||||
onChanged: (value) {
|
||||
_showConfirmationDialogOnStopEvent = value;
|
||||
saveSettings();
|
||||
@ -401,7 +403,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
children: [
|
||||
Expanded(
|
||||
child: Text(
|
||||
'TIME & DATE FORMAT',
|
||||
translate(LocalizationKeys.settings_sections_dateTimeFormat).toUpperCase(),
|
||||
style: Theme.of(context).textTheme.subtitle2,
|
||||
),
|
||||
),
|
||||
@ -421,15 +423,9 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
Expanded(
|
||||
child: TextFormField(
|
||||
controller: _dateFormatController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Date Format',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.settings_fields_dateFormat),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty title';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
@ -439,7 +435,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
const Text('Example', textScaleFactor: 0.75),
|
||||
Text(translate(LocalizationKeys.general_example), textScaleFactor: 0.75),
|
||||
Text(
|
||||
DateFormat(_dateFormatController.text)
|
||||
.format(DateTime.now()),
|
||||
@ -459,15 +455,9 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
Expanded(
|
||||
child: TextFormField(
|
||||
controller: _longDateFormatController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Long Date Format',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.settings_fields_longDateFormat),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty title';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
@ -477,7 +467,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
const Text('Example',
|
||||
Text(translate(LocalizationKeys.general_example),
|
||||
textScaleFactor: 0.75),
|
||||
Text(
|
||||
DateFormat(_longDateFormatController.text)
|
||||
@ -497,15 +487,9 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
Expanded(
|
||||
child: TextFormField(
|
||||
controller: _timeFormatController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Time Format',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.settings_fields_timeFormat),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty title';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
@ -515,7 +499,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
const Text('Example', textScaleFactor: 0.75),
|
||||
Text(translate(LocalizationKeys.general_example), textScaleFactor: 0.75),
|
||||
Text(
|
||||
DateFormat(_timeFormatController.text)
|
||||
.format(DateTime.now()),
|
||||
@ -535,15 +519,9 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
Expanded(
|
||||
child: TextFormField(
|
||||
controller: _longTimeFormatController,
|
||||
decoration: const InputDecoration(
|
||||
labelText: 'Long Time Format',
|
||||
decoration: InputDecoration(
|
||||
labelText: translate(LocalizationKeys.settings_fields_longTimeFormat),
|
||||
),
|
||||
validator: (value) {
|
||||
if (value!.trim().isEmpty) {
|
||||
return 'Empty title';
|
||||
}
|
||||
return null;
|
||||
},
|
||||
),
|
||||
),
|
||||
Expanded(
|
||||
@ -553,7 +531,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
const Text('Example',
|
||||
Text(translate(LocalizationKeys.general_example),
|
||||
textScaleFactor: 0.75),
|
||||
Text(
|
||||
DateFormat(_longTimeFormatController.text)
|
||||
@ -584,7 +562,7 @@ class _SettingsScreenState extends State<SettingsScreen> {
|
||||
Icons.settings_backup_restore,
|
||||
size: 18.0,
|
||||
),
|
||||
label: const Text('RESET ALL'),
|
||||
label: Text(translate(LocalizationKeys.settings_resetAll).toUpperCase()),
|
||||
),
|
||||
const Spacer(),
|
||||
],
|
||||
|
@ -1,4 +1,6 @@
|
||||
import 'package:diameter/localization_keys.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_translate/flutter_translate.dart';
|
||||
|
||||
class DialogUtils {
|
||||
static void showCancelConfirmationDialog(
|
||||
@ -13,30 +15,29 @@ class DialogUtils {
|
||||
List<Widget> actions = [
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 'CANCEL'),
|
||||
child: const Text('CANCEL'),
|
||||
child: Text(translate(LocalizationKeys.general_cancel).toUpperCase()),
|
||||
),
|
||||
];
|
||||
|
||||
actions.add(isNew
|
||||
? ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 'DISCARD'),
|
||||
child: const Text('DISCARD'),
|
||||
child: Text(translate(LocalizationKeys.general_discard).toUpperCase()),
|
||||
)
|
||||
: TextButton(
|
||||
onPressed: () => Navigator.pop(context, 'DISCARD'),
|
||||
child: const Text('DISCARD'),
|
||||
child: Text(translate(LocalizationKeys.general_discard).toUpperCase()),
|
||||
));
|
||||
|
||||
if (!isNew) {
|
||||
actions.add(ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 'SAVE'),
|
||||
child: const Text('SAVE'),
|
||||
child: Text(translate(LocalizationKeys.general_save).toUpperCase()),
|
||||
));
|
||||
}
|
||||
|
||||
return AlertDialog(
|
||||
content: Text(message ??
|
||||
'You already made some changes. Discard your input?'),
|
||||
content: Text(message ?? translate(LocalizationKeys.general_confirmDiscard)),
|
||||
actions: actions,
|
||||
);
|
||||
}).then((value) {
|
||||
@ -52,21 +53,21 @@ class DialogUtils {
|
||||
static void showConfirmationDialog(
|
||||
{required BuildContext context,
|
||||
required void Function() onConfirm,
|
||||
String message = 'Are you sure you want to delete this record?',
|
||||
String confirmationLabel = 'DELETE'}) {
|
||||
String? message,
|
||||
String? confirmationLabel}) {
|
||||
showDialog(
|
||||
context: context,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
content: Text(message),
|
||||
content: Text(message ?? translate(LocalizationKeys.general_confirmDelete)),
|
||||
actions: <Widget>[
|
||||
TextButton(
|
||||
onPressed: () => Navigator.pop(context, 'CANCEL'),
|
||||
child: const Text('CANCEL'),
|
||||
child: Text(translate(LocalizationKeys.general_cancel).toUpperCase()),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: () => Navigator.pop(context, 'CONFIRM'),
|
||||
child: Text(confirmationLabel),
|
||||
child: Text(confirmationLabel ?? translate(LocalizationKeys.general_confirm).toUpperCase()),
|
||||
),
|
||||
],
|
||||
);
|
||||
|
Binary file not shown.
Binary file not shown.
77
pubspec.lock
77
pubspec.lock
@ -1,27 +1,34 @@
|
||||
# Generated by pub
|
||||
# See https://dart.dev/tools/pub/glossary#lockfile
|
||||
packages:
|
||||
_discoveryapis_commons:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: _discoveryapis_commons
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "1.0.2"
|
||||
_fe_analyzer_shared:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: _fe_analyzer_shared
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "36.0.0"
|
||||
version: "38.0.0"
|
||||
analyzer:
|
||||
dependency: "direct dev"
|
||||
description:
|
||||
name: analyzer
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "3.3.1"
|
||||
version: "3.4.1"
|
||||
archive:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: archive
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "3.2.2"
|
||||
version: "3.3.0"
|
||||
args:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -231,7 +238,7 @@ packages:
|
||||
name: flex_color_scheme
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "3.0.1"
|
||||
version: "4.2.0"
|
||||
flutter:
|
||||
dependency: "direct main"
|
||||
description: flutter
|
||||
@ -244,11 +251,23 @@ packages:
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "1.0.4"
|
||||
flutter_localizations:
|
||||
dependency: "direct main"
|
||||
description: flutter
|
||||
source: sdk
|
||||
version: "0.0.0"
|
||||
flutter_test:
|
||||
dependency: "direct dev"
|
||||
description: flutter
|
||||
source: sdk
|
||||
version: "0.0.0"
|
||||
flutter_translate:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: flutter_translate
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "3.0.1"
|
||||
flutter_web_plugins:
|
||||
dependency: transitive
|
||||
description: flutter
|
||||
@ -268,6 +287,34 @@ packages:
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "2.0.2"
|
||||
google_sign_in:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: google_sign_in
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "5.2.4"
|
||||
google_sign_in_platform_interface:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: google_sign_in_platform_interface
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "2.1.2"
|
||||
google_sign_in_web:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: google_sign_in_web
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "0.10.0+5"
|
||||
googleapis:
|
||||
dependency: "direct main"
|
||||
description:
|
||||
name: googleapis
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "7.0.0"
|
||||
graphs:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -519,7 +566,7 @@ packages:
|
||||
name: printing
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "5.7.4"
|
||||
version: "5.7.5"
|
||||
process:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -555,13 +602,20 @@ packages:
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "3.0.0"
|
||||
quiver:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: quiver
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "3.0.1+1"
|
||||
shelf:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: shelf
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "1.2.0"
|
||||
version: "1.3.0"
|
||||
shelf_web_socket:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -644,6 +698,13 @@ packages:
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "1.3.0"
|
||||
universal_io:
|
||||
dependency: transitive
|
||||
description:
|
||||
name: universal_io
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "2.0.4"
|
||||
vector_math:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -671,7 +732,7 @@ packages:
|
||||
name: win32
|
||||
url: "https://pub.dartlang.org"
|
||||
source: hosted
|
||||
version: "2.4.2"
|
||||
version: "2.5.1"
|
||||
xdg_directories:
|
||||
dependency: transitive
|
||||
description:
|
||||
@ -694,5 +755,5 @@ packages:
|
||||
source: hosted
|
||||
version: "3.1.0"
|
||||
sdks:
|
||||
dart: ">=2.15.0 <3.0.0"
|
||||
dart: ">=2.16.0 <3.0.0"
|
||||
flutter: ">=2.8.0"
|
||||
|
11
pubspec.yaml
11
pubspec.yaml
@ -11,16 +11,21 @@ environment:
|
||||
dependencies:
|
||||
flutter:
|
||||
sdk: flutter
|
||||
flutter_localizations:
|
||||
sdk: flutter
|
||||
path_provider: ^2.0.9
|
||||
cupertino_icons: ^1.0.2
|
||||
flex_color_scheme: ^3.0.1
|
||||
flex_color_scheme: ^4.2.0
|
||||
intl: ^0.17.0
|
||||
objectbox: ^1.2.0
|
||||
objectbox_sync_flutter_libs: any
|
||||
charts_flutter: ^0.12.0
|
||||
printing: ^5.7.2
|
||||
pdf: ^3.7.1
|
||||
open_file: ^3.2.1
|
||||
flutter_translate: ^3.0.1
|
||||
googleapis: ^7.0.0
|
||||
google_sign_in: ^5.2.1
|
||||
charts_flutter: ^0.12.0
|
||||
|
||||
dev_dependencies:
|
||||
flutter_test:
|
||||
@ -33,3 +38,5 @@ dev_dependencies:
|
||||
|
||||
flutter:
|
||||
uses-material-design: true
|
||||
assets:
|
||||
- assets/i18n/
|
||||
|
@ -8,7 +8,7 @@
|
||||
// import 'package:flutter/material.dart';
|
||||
// import 'package:flutter_test/flutter_test.dart';
|
||||
|
||||
// import 'package:tide/main.dart';
|
||||
// import 'package:diameter/main.dart';
|
||||
|
||||
void main() {
|
||||
// testWidgets('Counter increments smoke test', (WidgetTester tester) async {
|
||||
|
@ -14,20 +14,20 @@
|
||||
This is a placeholder for base href that will be replaced by the value of
|
||||
the `--base-href` argument provided to `flutter build`.
|
||||
-->
|
||||
<base href="$FLUTTER_BASE_HREF">
|
||||
<base href="$FLUTTER_BASE_HREF" />
|
||||
|
||||
<meta charset="UTF-8">
|
||||
<meta content="IE=Edge" http-equiv="X-UA-Compatible">
|
||||
<meta name="description" content="A new Flutter project.">
|
||||
<meta charset="UTF-8" />
|
||||
<meta content="IE=Edge" http-equiv="X-UA-Compatible" />
|
||||
<meta name="description" content="A new Flutter project." />
|
||||
|
||||
<!-- iOS meta tags & icons -->
|
||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||
<meta name="apple-mobile-web-app-status-bar-style" content="black">
|
||||
<meta name="apple-mobile-web-app-title" content="tide">
|
||||
<link rel="apple-touch-icon" href="icons/Icon-192.png">
|
||||
<meta name="apple-mobile-web-app-capable" content="yes" />
|
||||
<meta name="apple-mobile-web-app-status-bar-style" content="black" />
|
||||
<meta name="apple-mobile-web-app-title" content="diameter" />
|
||||
<link rel="apple-touch-icon" href="icons/Icon-192.png" />
|
||||
|
||||
<title>tide</title>
|
||||
<link rel="manifest" href="manifest.json">
|
||||
<title>diameter</title>
|
||||
<link rel="manifest" href="manifest.json" />
|
||||
</head>
|
||||
<body>
|
||||
<!-- This script installs service_worker.js to provide PWA functionality to
|
||||
@ -41,42 +41,60 @@
|
||||
return;
|
||||
}
|
||||
scriptLoaded = true;
|
||||
var scriptTag = document.createElement('script');
|
||||
scriptTag.src = 'main.dart.js';
|
||||
scriptTag.type = 'application/javascript';
|
||||
var scriptTag = document.createElement("script");
|
||||
scriptTag.src = "main.dart.js";
|
||||
scriptTag.type = "application/javascript";
|
||||
document.body.append(scriptTag);
|
||||
}
|
||||
|
||||
if ('serviceWorker' in navigator) {
|
||||
if ("serviceWorker" in navigator) {
|
||||
// Service workers are supported. Use them.
|
||||
window.addEventListener('load', function () {
|
||||
window.addEventListener("load", function () {
|
||||
// Wait for registration to finish before dropping the <script> tag.
|
||||
// Otherwise, the browser will load the script multiple times,
|
||||
// potentially different versions.
|
||||
var serviceWorkerUrl = 'flutter_service_worker.js?v=' + serviceWorkerVersion;
|
||||
navigator.serviceWorker.register(serviceWorkerUrl)
|
||||
var serviceWorkerUrl =
|
||||
"flutter_service_worker.js?v=" + serviceWorkerVersion;
|
||||
navigator.serviceWorker
|
||||
.register(serviceWorkerUrl)
|
||||
.then((reg) => {
|
||||
function waitForActivation(serviceWorker) {
|
||||
serviceWorker.addEventListener('statechange', () => {
|
||||
if (serviceWorker.state == 'activated') {
|
||||
console.log('Installed new service worker.');
|
||||
serviceWorker.addEventListener(
|
||||
"statechange",
|
||||
() => {
|
||||
if (
|
||||
serviceWorker.state == "activated"
|
||||
) {
|
||||
console.log(
|
||||
"Installed new service worker."
|
||||
);
|
||||
loadMainDartJs();
|
||||
}
|
||||
});
|
||||
}
|
||||
if (!reg.active && (reg.installing || reg.waiting)) {
|
||||
);
|
||||
}
|
||||
if (
|
||||
!reg.active &&
|
||||
(reg.installing || reg.waiting)
|
||||
) {
|
||||
// No active web worker and we have installed or are installing
|
||||
// one for the first time. Simply wait for it to activate.
|
||||
waitForActivation(reg.installing || reg.waiting);
|
||||
} else if (!reg.active.scriptURL.endsWith(serviceWorkerVersion)) {
|
||||
waitForActivation(
|
||||
reg.installing || reg.waiting
|
||||
);
|
||||
} else if (
|
||||
!reg.active.scriptURL.endsWith(
|
||||
serviceWorkerVersion
|
||||
)
|
||||
) {
|
||||
// When the app updates the serviceWorkerVersion changes, so we
|
||||
// need to ask the service worker to update.
|
||||
console.log('New service worker available.');
|
||||
console.log("New service worker available.");
|
||||
reg.update();
|
||||
waitForActivation(reg.installing);
|
||||
} else {
|
||||
// Existing service worker is still good.
|
||||
console.log('Loading app from service worker.');
|
||||
console.log("Loading app from service worker.");
|
||||
loadMainDartJs();
|
||||
}
|
||||
});
|
||||
@ -86,7 +104,7 @@
|
||||
setTimeout(() => {
|
||||
if (!scriptLoaded) {
|
||||
console.warn(
|
||||
'Failed to load app from service worker. Falling back to plain <script> tag.',
|
||||
"Failed to load app from service worker. Falling back to plain <script> tag."
|
||||
);
|
||||
loadMainDartJs();
|
||||
}
|
||||
|
@ -89,7 +89,7 @@ BEGIN
|
||||
BEGIN
|
||||
BLOCK "040904e4"
|
||||
BEGIN
|
||||
VALUE "CompanyName", "com.example" "\0"
|
||||
VALUE "CompanyName", "at.sarahziesel" "\0"
|
||||
VALUE "FileDescription", "diameter" "\0"
|
||||
VALUE "FileVersion", VERSION_AS_STRING "\0"
|
||||
VALUE "InternalName", "diameter" "\0"
|
||||
|
Loading…
Reference in New Issue
Block a user