Compare commits

..

25 Commits

Author SHA1 Message Date
Lisa Hillebrand cb33d93f8e 623 Remove common translation file 2022-11-25 20:27:08 +01:00
Lisa Hillebrand b4f711e4db 623 Rename password component 2022-11-25 20:27:08 +01:00
Lisa Hillebrand 1d5737d8a9 623 Use translation library in password section 2022-11-25 20:27:08 +01:00
Lisa Hillebrand edc6f350c5 623 Add common.json for shared translation strings 2022-11-25 20:27:08 +01:00
Lisa Hillebrand d30b7db3fb 623 Remove unused stats labels 2022-11-25 20:27:08 +01:00
Sofiya Tepikin 6e2e03f39e Merge branch 'chore/update-babel-dependencies' into 'main'
Chore/update babel dependencies

See merge request bloodyhealth/drip!579
2022-11-13 20:00:23 +00:00
Sofiya Tepikin cc62e24229 Chore/update babel dependencies 2022-11-13 20:00:23 +00:00
Sofiya Tepikin cd24522b4d Merge branch '618/Refactor-import-section-to-use-translation-lib' into 'main'
618 Refactor import section to use translation lib

See merge request bloodyhealth/drip!555
2022-11-06 14:47:05 +00:00
Lisa 446638d6de 618 Refactor import section to use translation lib 2022-11-06 14:47:05 +00:00
Lisa ae23ef2c58 Merge branch '624/Use-translation-library-in-tutorial' into 'main'
624 Use translation library in tutorial

See merge request bloodyhealth/drip!573
2022-11-06 14:43:15 +00:00
Lisa 38b9e8b31f Merge branch 'dependabot-npm_and_yarn-react-i18next-12.0.0' into 'main'
Bump react-i18next from 11.18.3 to 12.0.0

See merge request bloodyhealth/drip!571
2022-10-23 10:11:06 +00:00
Lisa Hillebrand 84d657cabb 614 Uppercase tutorial component 2022-10-23 11:56:32 +02:00
Lisa Hillebrand 4573b93921 624 Use translation library for chart tutorial 2022-10-23 11:54:11 +02:00
Sofiya Tepikin ab88a4c163 Bump react-i18next from 11.18.3 to 12.0.0
Bumps [react-i18next](https://github.com/i18next/react-i18next) from 11.18.3 to 12.0.0.
- [Release notes](https://github.com/i18next/react-i18next/releases)
- [Changelog](https://github.com/i18next/react-i18next/blob/master/CHANGELOG.md)
- [Commits](https://github.com/i18next/react-i18next/compare/v11.18.3...v12.0.0)
2022-10-23 09:06:40 +00:00
Lisa baaf89c04e Merge branch 'dependabot-npm_and_yarn-i18next-22.0.2' into 'main'
Bump i18next from 21.9.0 to 22.0.2

See merge request bloodyhealth/drip!570
2022-10-23 08:13:38 +00:00
Sofiya Tepikin d476f6c143 Bump i18next from 21.9.0 to 22.0.2
Bumps [i18next](https://github.com/i18next/i18next) from 21.9.0 to 22.0.2.
- [Release notes](https://github.com/i18next/i18next/releases)
- [Changelog](https://github.com/i18next/i18next/blob/master/CHANGELOG.md)
- [Commits](https://github.com/i18next/i18next/compare/v21.9.0...v22.0.2)
2022-10-21 09:07:30 +00:00
Sofiya Tepikin b56e0818f3 Merge branch '619/Create-test-utils-for-react-testing-library' into 'main'
619 Create test utils for react testing library

See merge request bloodyhealth/drip!556
2022-10-19 13:17:19 +00:00
Lisa 1b5ffaf5d6 619 Create test utils for react testing library 2022-10-19 13:17:19 +00:00
Sofiya Tepikin f68cc2b49e Merge branch 'fix/exclude-babel-core-from-dependabot' into 'main'
Exclude @babel/core library from dependabot updates

See merge request bloodyhealth/drip!560
2022-10-05 11:31:38 +00:00
Sofiya Tepikin 27d430a465 Exclude @babel/core library from dependabot updates 2022-10-05 13:28:49 +02:00
Sofiya Tepikin 28f52e2cea Merge branch '617/Remove-weblate-reference-from-readme' into 'main'
617 Remove reference to weblate from readme

See merge request bloodyhealth/drip!554
2022-10-05 10:10:46 +00:00
Lisa 3f87f298fb Merge branch '615-Use-translation-library-for-pages' into 'main'
615 Use translation library for pages

Closes #615

See merge request bloodyhealth/drip!530
2022-10-01 08:13:24 +00:00
Lisa Hillebrand 585f32863d 617 Remove reference to weblate from readme 2022-09-30 11:58:11 +02:00
Lisa Hillebrand 51e1c95e71 615 Remove unused page labels 2022-09-30 11:39:16 +02:00
Lisa Hillebrand 36c33c69b7 615 Use translation library for bottom menu 2022-09-30 11:39:00 +02:00
28 changed files with 1152 additions and 997 deletions
+1
View File
@@ -14,3 +14,4 @@ updates:
- dependency-name: 'react' - dependency-name: 'react'
- dependency-name: 'react-native' - dependency-name: 'react-native'
- dependency-name: 'react-native-push-notifications' - dependency-name: 'react-native-push-notifications'
- dependency-name: '@babel/core'
-4
View File
@@ -201,7 +201,3 @@ More information about how the app calculates fertility status and bleeding pred
react-native link react-native link
5. You should be able to use the icon now within drip, e.g. in Cycle Day Overview and on the chart. 5. You should be able to use the icon now within drip, e.g. in Cycle Day Overview and on the chart.
## Translation
We are using [Weblate](https://weblate.org/) as translation software.
@@ -10,26 +10,34 @@ import Header from './header'
import { saveEncryptionFlag } from '../local-storage' import { saveEncryptionFlag } from '../local-storage'
import { deleteDbAndOpenNew, openDb } from '../db' import { deleteDbAndOpenNew, openDb } from '../db'
import { passwordPrompt as labels, shared } from '../i18n/en/labels'
import { Containers, Spacing } from '../styles' import { Containers, Spacing } from '../styles'
import { useTranslation } from 'react-i18next'
const cancelButton = { text: shared.cancel, style: 'cancel' }
const PasswordPrompt = ({ enableShowApp }) => { const PasswordPrompt = ({ enableShowApp }) => {
const [password, setPassword] = useState(null) const [password, setPassword] = useState(null)
const isPasswordEntered = Boolean(password)
const { t } = useTranslation(null, { keyPrefix: 'password' })
const cancelButton = {
text: t('forgotPasswordDialog.cancel'),
style: 'cancel',
}
const unlockApp = async () => { const unlockApp = async () => {
const hash = new SHA512().hex(password) const hash = new SHA512().hex(password)
const connected = await openDb(hash) const connected = await openDb(hash)
if (!connected) { if (!connected) {
Alert.alert(shared.incorrectPassword, shared.incorrectPasswordMessage, [ Alert.alert(
{ t('incorrectPasswordDialog.incorrectPassword'),
text: shared.tryAgain, t('incorrectPasswordDialog.incorrectPasswordMessage'),
onPress: () => setPassword(null), [
}, {
]) text: t('incorrectPasswordDialog.tryAgain'),
onPress: () => setPassword(null),
},
]
)
return return
} }
enableShowApp() enableShowApp()
@@ -42,19 +50,22 @@ const PasswordPrompt = ({ enableShowApp }) => {
} }
const onDeleteData = () => { const onDeleteData = () => {
Alert.alert(labels.areYouSureTitle, labels.areYouSure, [ Alert.alert(t('confirmationDialog.title'), t('confirmationDialog.text'), [
cancelButton, cancelButton,
{ {
text: labels.reallyDeleteData, text: t('confirmationDialog.confirm'),
onPress: onDeleteDataConfirmation, onPress: onDeleteDataConfirmation,
}, },
]) ])
} }
const onConfirmDeletion = async () => { const onConfirmDeletion = async () => {
Alert.alert(labels.deleteDatabaseTitle, labels.deleteDatabaseExplainer, [ Alert.alert(t('forgotPassword'), t('forgotPasswordDialog.text'), [
cancelButton, cancelButton,
{ text: labels.deleteData, onPress: onDeleteData }, {
text: t('forgotPasswordDialog.confirm'),
onPress: onDeleteData,
},
]) ])
} }
@@ -65,17 +76,13 @@ const PasswordPrompt = ({ enableShowApp }) => {
<KeyboardAvoidingView behavior="padding" keyboardVerticalOffset={150}> <KeyboardAvoidingView behavior="padding" keyboardVerticalOffset={150}>
<AppTextInput <AppTextInput
onChangeText={setPassword} onChangeText={setPassword}
secureTextEntry={true} secureTextEntry
placeholder={labels.enterPassword} placeholder={t('enterPassword')}
/> />
<View style={styles.containerButtons}> <View style={styles.containerButtons}>
<Button onPress={onConfirmDeletion}>{labels.forgotPassword}</Button> <Button onPress={onConfirmDeletion}>{t('forgotPassword')}</Button>
<Button <Button disabled={!password} isCTA={!!password} onPress={unlockApp}>
disabled={!isPasswordEntered} {t('unlockApp')}
isCTA={isPasswordEntered}
onPress={unlockApp}
>
{labels.title}
</Button> </Button>
</View> </View>
</KeyboardAvoidingView> </KeyboardAvoidingView>
+1 -1
View File
@@ -7,7 +7,7 @@ import App from './app'
import AppLoadingView from './common/app-loading' import AppLoadingView from './common/app-loading'
import AppStatusBar from './common/app-status-bar' import AppStatusBar from './common/app-status-bar'
import AcceptLicense from './AcceptLicense' import AcceptLicense from './AcceptLicense'
import PasswordPrompt from './password-prompt' import PasswordPrompt from './PasswordPrompt'
export default function AppWrapper() { export default function AppWrapper() {
const [isLoading, setIsLoading] = useState(true) const [isLoading, setIsLoading] = useState(true)
@@ -6,16 +6,17 @@ import AppText from '../common/app-text'
import CloseIcon from '../common/close-icon' import CloseIcon from '../common/close-icon'
import { Containers, Spacing } from '../../styles' import { Containers, Spacing } from '../../styles'
import { chart } from '../../i18n/en/labels' import { useTranslation } from 'react-i18next'
const image = require('../../assets/swipe.png') const image = require('../../assets/swipe.png')
const Tutorial = ({ onClose }) => { const Tutorial = ({ onClose }) => {
const { t } = useTranslation()
return ( return (
<View style={styles.container}> <View style={styles.container}>
<Image resizeMode="contain" source={image} style={styles.image} /> <Image resizeMode="contain" source={image} style={styles.image} />
<View style={styles.textContainer}> <View style={styles.textContainer}>
<AppText>{chart.tutorial}</AppText> <AppText>{t('chart.tutorial')}</AppText>
</View> </View>
<CloseIcon onClose={onClose} /> <CloseIcon onClose={onClose} />
</View> </View>
+1 -1
View File
@@ -9,7 +9,7 @@ import HorizontalGrid from './horizontal-grid'
import MainGrid from './main-grid' import MainGrid from './main-grid'
import NoData from './no-data' import NoData from './no-data'
import NoTemperature from './no-temperature' import NoTemperature from './no-temperature'
import Tutorial from './tutorial' import Tutorial from './Tutorial'
import YAxis from './y-axis' import YAxis from './y-axis'
import { getCycleDaysSortedByDate } from '../../db' import { getCycleDaysSortedByDate } from '../../db'
+6 -3
View File
@@ -6,20 +6,23 @@ import MenuItem from './menu-item'
import { Containers } from '../../styles' import { Containers } from '../../styles'
import { pages } from '../pages' import { pages } from '../pages'
import { useTranslation } from 'react-i18next'
const Menu = ({ currentPage, navigate }) => { const Menu = ({ currentPage, navigate }) => {
const menuItems = pages.filter((page) => page.isInMenu) const menuItems = pages.filter((page) => page.isInMenu)
const { t } = useTranslation(null, { keyPrefix: 'bottomMenu' })
return ( return (
<View style={styles.container}> <View style={styles.container}>
{menuItems.map(({ icon, label, component }) => { {menuItems.map(({ icon, labelKey, component }) => {
return ( return (
<MenuItem <MenuItem
isActive={component === currentPage} isActive={component === currentPage}
onPress={() => navigate(component)} onPress={() => navigate(component)}
icon={icon} icon={icon}
key={label} key={labelKey}
label={label} label={t(labelKey)}
/> />
) )
})} })}
+3 -15
View File
@@ -1,75 +1,63 @@
import settingsViews from './settings' import settingsViews from './settings'
import settingsLabels from '../i18n/en/settings'
const labels = settingsLabels.menuItems
export const pages = [ export const pages = [
{ {
component: 'Home', component: 'Home',
icon: 'home', icon: 'home',
label: 'Home',
}, },
{ {
component: 'Calendar', component: 'Calendar',
icon: 'calendar', icon: 'calendar',
isInMenu: true, isInMenu: true,
label: 'Calendar', labelKey: 'calendar',
parent: 'Home', parent: 'Home',
}, },
{ {
component: 'Chart', component: 'Chart',
icon: 'chart', icon: 'chart',
isInMenu: true, isInMenu: true,
label: 'Chart', labelKey: 'chart',
parent: 'Home', parent: 'Home',
}, },
{ {
component: 'Stats', component: 'Stats',
icon: 'statistics', icon: 'statistics',
isInMenu: true, isInMenu: true,
label: 'Stats', labelKey: 'stats',
parent: 'Home', parent: 'Home',
}, },
{ {
children: Object.keys(settingsViews), children: Object.keys(settingsViews),
component: 'SettingsMenu', component: 'SettingsMenu',
icon: 'settings', icon: 'settings',
label: 'Settings',
parent: 'Home', parent: 'Home',
}, },
{ {
component: 'Reminders', component: 'Reminders',
label: labels.reminders.name,
parent: 'SettingsMenu', parent: 'SettingsMenu',
}, },
{ {
component: 'NfpSettings', component: 'NfpSettings',
label: labels.nfpSettings.name,
parent: 'SettingsMenu', parent: 'SettingsMenu',
}, },
{ {
component: 'DataManagement', component: 'DataManagement',
label: labels.dataManagement.name,
parent: 'SettingsMenu', parent: 'SettingsMenu',
}, },
{ {
component: 'Password', component: 'Password',
label: labels.password.name,
parent: 'SettingsMenu', parent: 'SettingsMenu',
}, },
{ {
component: 'About', component: 'About',
label: 'About',
parent: 'SettingsMenu', parent: 'SettingsMenu',
}, },
{ {
component: 'License', component: 'License',
label: 'License',
parent: 'SettingsMenu', parent: 'SettingsMenu',
}, },
{ {
component: 'PrivacyPolicy', component: 'PrivacyPolicy',
label: 'PrivacyPolicy',
parent: 'SettingsMenu', parent: 'SettingsMenu',
}, },
{ {
@@ -2,30 +2,46 @@ import React, { useState } from 'react'
import AppLoadingView from '../../common/app-loading' import AppLoadingView from '../../common/app-loading'
import AppPage from '../../common/app-page' import AppPage from '../../common/app-page'
import DeleteData from './DeleteData' import AppText from '../../common/app-text'
import Button from '../../common/button'
import Segment from '../../common/segment'
import openShareDialogAndExport from './export-dialog'
import DeleteData from './delete-data'
import labels from '../../../i18n/en/settings'
import ImportData from './ImportData' import ImportData from './ImportData'
import ExportData from './ExportData'
const DataManagement = () => { const DataManagement = () => {
const [isLoading, setIsLoading] = useState(false) const [isLoading, setIsLoading] = useState(false)
const [isDeletingData, setIsDeletingData] = useState(false) const [isDeletingData, setIsDeletingData] = useState(false)
const startExport = () => {
setIsDeletingData(false)
openShareDialogAndExport()
}
if (isLoading) return <AppLoadingView /> if (isLoading) return <AppLoadingView />
return ( return (
<AppPage> <AppPage>
<ExportData <Segment title={labels.export.button}>
resetIsDeletingData={() => setIsDeletingData(false)} <AppText>{labels.export.segmentExplainer}</AppText>
setIsLoading={setIsLoading} <Button isCTA onPress={startExport}>
/> {labels.export.button}
</Button>
</Segment>
<ImportData <ImportData
resetIsDeletingData={() => setIsDeletingData(false)} resetIsDeletingData={() => setIsDeletingData(false)}
setIsLoading={setIsLoading} setIsLoading={setIsLoading}
/> />
<DeleteData <Segment title={labels.deleteSegment.title} last>
isDeletingData={isDeletingData} <AppText>{labels.deleteSegment.explainer}</AppText>
onStartDeletion={() => setIsDeletingData(true)} <DeleteData
/> isDeletingData={isDeletingData}
onStartDeletion={() => setIsDeletingData(true)}
/>
</Segment>
</AppPage> </AppPage>
) )
} }
@@ -1,77 +0,0 @@
import React from 'react'
import PropTypes from 'prop-types'
import { getCycleDaysSortedByDate, mapRealmObjToJsObj } from '../../../db'
import getDataAsCsvDataUri from '../../../lib/import-export/export-to-csv'
import alertError from '../common/alert-error'
import { EXPORT_FILE_NAME } from './constants'
import RNFS from 'react-native-fs'
import { useTranslation } from 'react-i18next'
import AppText from '../../common/app-text'
import Button from '../../common/button'
import Segment from '../../common/segment'
import Share from 'react-native-share'
export default function ExportData({ setIsLoading, resetIsDeletingData }) {
const { t } = useTranslation(null, {
keyPrefix: 'hamburgerMenu.settings.data.export',
})
async function startExport() {
resetIsDeletingData()
setIsLoading(true)
await exportData()
setIsLoading(false)
}
async function getData() {
const cycleDaysByDate = mapRealmObjToJsObj(getCycleDaysSortedByDate())
try {
return cycleDaysByDate.length
? getDataAsCsvDataUri(cycleDaysByDate)
: null
} catch (err) {
alertError(t('error.convert'))
return null
}
}
async function exportData() {
const data = await getData()
if (!data) {
alertError(t('error.data'))
return
}
try {
const path = `${RNFS.DocumentDirectoryPath}/${EXPORT_FILE_NAME}`
await RNFS.writeFile(path, data)
await Share.open({
title: t('title'),
url: `file://${path}`,
subject: t('title'),
type: 'text/csv',
showAppsToView: true,
failOnCancel: false,
})
} catch (err) {
return alertError(t('error.share'))
}
}
return (
<Segment title={t('button')}>
<AppText>{t('text')}</AppText>
<Button isCTA onPress={startExport}>
{t('button')}
</Button>
</Segment>
)
}
ExportData.propTypes = {
resetIsDeletingData: PropTypes.func.isRequired,
setIsLoading: PropTypes.func.isRequired,
}
@@ -41,7 +41,7 @@ export default function ImportData({ resetIsDeletingData, setIsLoading }) {
const fileContent = await rnfs.readFile(fileInfo.uri, 'utf8') const fileContent = await rnfs.readFile(fileInfo.uri, 'utf8')
return fileContent return fileContent
} catch (err) { } catch (err) {
return showImportErrorAlert(t('errors.couldNotOpenFile')) return showImportErrorAlert(t('error.couldNotOpenFile'))
} }
} }
@@ -77,7 +77,7 @@ export default function ImportData({ resetIsDeletingData, setIsLoading }) {
} }
function showImportErrorAlert(message) { function showImportErrorAlert(message) {
const errorMessage = t('errors.noDataImported', { message }) const errorMessage = t('error.noDataImported', { message })
alertError(errorMessage) alertError(errorMessage)
} }
@@ -11,10 +11,9 @@ import alertError from '../common/alert-error'
import { clearDb, isDbEmpty } from '../../../db' import { clearDb, isDbEmpty } from '../../../db'
import { showToast } from '../../helpers/general' import { showToast } from '../../helpers/general'
import { hasEncryptionObservable } from '../../../local-storage' import { hasEncryptionObservable } from '../../../local-storage'
import settings from '../../../i18n/en/settings'
import { shared as sharedLabels } from '../../../i18n/en/labels'
import { EXPORT_FILE_NAME } from './constants' import { EXPORT_FILE_NAME } from './constants'
import Segment from '../../common/segment'
import AppText from '../../common/app-text'
import { useTranslation } from 'react-i18next'
const exportedFilePath = `${RNFS.DocumentDirectoryPath}/${EXPORT_FILE_NAME}` const exportedFilePath = `${RNFS.DocumentDirectoryPath}/${EXPORT_FILE_NAME}`
@@ -23,10 +22,6 @@ const DeleteData = ({ onStartDeletion, isDeletingData }) => {
const [isConfirmingWithPassword, setIsConfirmingWithPassword] = const [isConfirmingWithPassword, setIsConfirmingWithPassword] =
useState(false) useState(false)
const { t } = useTranslation(null, {
keyPrefix: 'hamburgerMenu.settings.data.delete',
})
const onAlertConfirmation = () => { const onAlertConfirmation = () => {
onStartDeletion() onStartDeletion()
if (isPasswordSet) { if (isPasswordSet) {
@@ -37,16 +32,17 @@ const DeleteData = ({ onStartDeletion, isDeletingData }) => {
} }
const alertBeforeDeletion = async () => { const alertBeforeDeletion = async () => {
const { question, message, confirmation, errors } = settings.deleteSegment
if (isDbEmpty() && !(await RNFS.exists(exportedFilePath))) { if (isDbEmpty() && !(await RNFS.exists(exportedFilePath))) {
alertError(t('error.noData')) alertError(errors.noData)
} else { } else {
Alert.alert(t('dialog.title'), t('dialog.message'), [ Alert.alert(question, message, [
{ {
text: t('dialog.delete'), text: confirmation,
onPress: onAlertConfirmation, onPress: onAlertConfirmation,
}, },
{ {
text: t('dialog.cancel'), text: sharedLabels.cancel,
style: 'cancel', style: 'cancel',
onPress: cancelConfirmationWithPassword, onPress: cancelConfirmationWithPassword,
}, },
@@ -61,14 +57,16 @@ const DeleteData = ({ onStartDeletion, isDeletingData }) => {
} }
const deleteAppData = async () => { const deleteAppData = async () => {
const { errors, success } = settings.deleteSegment
try { try {
if (!isDbEmpty()) { if (!isDbEmpty()) {
clearDb() clearDb()
} }
await deleteExportedFile() await deleteExportedFile()
showToast(t('success.message')) showToast(success.message)
} catch (err) { } catch (err) {
alertError(t('error.delete')) alertError(errors.couldNotDeleteFile)
} }
cancelConfirmationWithPassword() cancelConfirmationWithPassword()
} }
@@ -87,12 +85,9 @@ const DeleteData = ({ onStartDeletion, isDeletingData }) => {
} }
return ( return (
<Segment title={t('title')} last> <Button isCTA onPress={alertBeforeDeletion}>
<AppText>{t('subTitle')}</AppText> {settings.deleteSegment.title}
<Button isCTA onPress={alertBeforeDeletion}> </Button>
{t('title')}
</Button>
</Segment>
) )
} }
@@ -0,0 +1,43 @@
import Share from 'react-native-share'
import { getCycleDaysSortedByDate, mapRealmObjToJsObj } from '../../../db'
import getDataAsCsvDataUri from '../../../lib/import-export/export-to-csv'
import alertError from '../common/alert-error'
import settings from '../../../i18n/en/settings'
import { EXPORT_FILE_NAME } from './constants'
import RNFS from 'react-native-fs'
export default async function exportData() {
let data
const labels = settings.export
const cycleDaysByDate = mapRealmObjToJsObj(getCycleDaysSortedByDate())
if (!cycleDaysByDate.length) return alertError(labels.errors.noData)
try {
data = getDataAsCsvDataUri(cycleDaysByDate)
if (!data) {
return alertError(labels.errors.noData)
}
} catch (err) {
console.error(err)
return alertError(labels.errors.couldNotConvert)
}
try {
const path = `${RNFS.DocumentDirectoryPath}/${EXPORT_FILE_NAME}`
await RNFS.writeFile(path, data)
await Share.open({
title: labels.title,
url: `file://${path}`,
subject: labels.subject,
type: 'text/csv',
showAppsToView: true,
failOnCancel: false,
})
} catch (err) {
console.error(err)
return alertError(labels.errors.problemSharing)
}
}
-1
View File
@@ -1 +0,0 @@
{}
+30 -28
View File
@@ -1,4 +1,12 @@
{ {
"bottomMenu": {
"calendar": "Calendar",
"chart": "Chart",
"stats": "Stats"
},
"chart": {
"tutorial": "You can swipe the chart to view more dates."
},
"cycleDay": { "cycleDay": {
"symptomBox": { "symptomBox": {
"bleeding": "Bleeding", "bleeding": "Bleeding",
@@ -81,33 +89,6 @@
}, },
"settings": { "settings": {
"data": { "data": {
"delete": {
"dialog": {
"cancel": "Cancel",
"delete": "Delete app data permanently",
"message": "Please note that deletion of the app data is permanent and irreversible. We recommend exporting existing data before deletion.",
"title": "Do you want to delete app data from this phone?"
},
"error": {
"delete": "Could not delete data",
"noData": "There is no data to delete"
},
"subTitle": "Delete app data from this phone",
"success": {
"message": "App data successfully deleted"
},
"title": "Delete app data"
},
"export": {
"button": "Export data",
"error": {
"convert": "Could not convert data to CSV",
"data": "There is no data to export",
"share": "There was a problem sharing the data export file"
},
"text": "Export data in CSV format for backup or so you can use it elsewhere",
"title": "My drip. data export"
},
"import": { "import": {
"button": "Import data", "button": "Import data",
"dialog": { "dialog": {
@@ -117,9 +98,10 @@
"replace": "Import and replace", "replace": "Import and replace",
"title": "Keep existing data?" "title": "Keep existing data?"
}, },
"errors": { "error": {
"couldNotOpenFile": "Could not open file", "couldNotOpenFile": "Could not open file",
"futureEdit": "Future dates may only contain a note, no other symptoms", "futureEdit": "Future dates may only contain a note, no other symptoms",
"incorrectColumns": "Expected CSV column titles to be {{incorrectColumns}}",
"noDataImported": "{{message}}\n\nNo data was imported or changed" "noDataImported": "{{message}}\n\nNo data was imported or changed"
}, },
"segmentExplainer": "Import data in CSV format", "segmentExplainer": "Import data in CSV format",
@@ -150,6 +132,26 @@
"title": "Settings" "title": "Settings"
} }
}, },
"password": {
"confirmationDialog": {
"confirm": "Yes, I am sure",
"text": "Are you absolutely sure you want to permanently delete all your data?",
"title": "Are you sure?"
},
"enterPassword": "Enter password here",
"forgotPassword": "Forgot your password?",
"forgotPasswordDialog": {
"cancel": "Cancel",
"confirm": "Yes, delete all my data",
"text": "If you've forgotten your password, unfortunately, there is nothing we can do to recover your data, because it is encrypted with the password only you know. You can, however, delete all your encrypted data and start fresh. Once all data has been erased, you can set a new password in the settings, if you like."
},
"incorrectPasswordDialog": {
"incorrectPassword": "Password incorrect",
"incorrectPasswordMessage": "That password is incorrect.",
"tryAgain": "Try again"
},
"unlockApp": "Unlock app"
},
"stats": { "stats": {
"noData": "At least one completed cycle is needed to display stats.", "noData": "At least one completed cycle is needed to display stats.",
"intro": "Basic statistics about the length of your cycles.", "intro": "Basic statistics about the length of your cycles.",
-29
View File
@@ -3,10 +3,6 @@ export const home = {
phase: (n) => `${['1st', '2nd', '3rd'][n - 1]} cycle phase`, phase: (n) => `${['1st', '2nd', '3rd'][n - 1]} cycle phase`,
} }
export const chart = {
tutorial: 'You can swipe the chart to view more dates.',
}
export const shared = { export const shared = {
cancel: 'Cancel', cancel: 'Cancel',
save: 'Save', save: 'Save',
@@ -30,17 +26,6 @@ export const shared = {
learnMore: 'Learn more', learnMore: 'Learn more',
} }
export const stats = {
cycleLengthExplainer: 'Basic statistics about the length of your cycles.',
emptyStats: 'At least one completed cycle is needed to display stats.',
daysLabel: 'days',
basisOfStatsEnd: 'completed\ncycles',
averageLabel: 'Average cycle',
minLabel: `Shortest`,
maxLabel: `Longest`,
stdLabel: `Standard\ndeviation`,
}
export const bleedingPrediction = { export const bleedingPrediction = {
predictionInFuture: (startDays, endDays) => predictionInFuture: (startDays, endDays) =>
`Your next period is likely to start in ${startDays} to ${endDays} days.`, `Your next period is likely to start in ${startDays} to ${endDays} days.`,
@@ -53,20 +38,6 @@ export const bleedingPrediction = {
`Based on your documented data, your period was likely to start between ${startDate} and ${endDate}.`, `Based on your documented data, your period was likely to start between ${startDate} and ${endDate}.`,
} }
export const passwordPrompt = {
title: 'Unlock app',
enterPassword: 'Enter password here',
deleteDatabaseExplainer:
"If you've forgotten your password, unfortunately, there is nothing we can do to recover your data, because it is encrypted with the password only you know. You can, however, delete all your encrypted data and start fresh. Once all data has been erased, you can set a new password in the settings, if you like.",
forgotPassword: 'Forgot your password?',
deleteDatabaseTitle: 'Forgot your password?',
deleteData: 'Yes, delete all my data',
areYouSureTitle: 'Are you sure?',
areYouSure:
'Are you absolutely sure you want to permanently delete all your data?',
reallyDeleteData: 'Yes, I am sure',
}
export const fertilityStatus = { export const fertilityStatus = {
fertile: 'fertile', fertile: 'fertile',
infertile: 'infertile', infertile: 'infertile',
+24 -14
View File
@@ -1,22 +1,32 @@
import links from './links' import links from './links'
export default { export default {
menuItems: { export: {
reminders: { errors: {
name: 'Reminders', noData: 'There is no data to export',
text: 'turn on/off reminders', couldNotConvert: 'Could not convert data to CSV',
problemSharing: 'There was a problem sharing the data export file',
}, },
nfpSettings: { title: 'My drip. data export',
name: 'NFP settings', subject: 'My drip. data export',
text: 'define how you want to use NFP', button: 'Export data',
segmentExplainer:
'Export data in CSV format for backup or so you can use it elsewhere',
},
deleteSegment: {
title: 'Delete app data',
explainer: 'Delete app data from this phone',
question: 'Do you want to delete app data from this phone?',
message:
'Please note that deletion of the app data is permanent and irreversible. We recommend exporting existing data before deletion.',
confirmation: 'Delete app data permanently',
errors: {
couldNotDeleteFile: 'Could not delete data',
postFix: 'No data was deleted or changed',
noData: 'There is no data to delete',
}, },
dataManagement: { success: {
name: 'Data', message: 'App data successfully deleted',
text: 'import, export or delete your data',
},
password: {
name: 'Password',
text: '',
}, },
}, },
tempScale: { tempScale: {
-1
View File
@@ -17,7 +17,6 @@ i18n
compatibilityJSON: 'v3', // TODO: migrate json to v4 and afterwards remove it compatibilityJSON: 'v3', // TODO: migrate json to v4 and afterwards remove it
resources, resources,
fallbackLng: 'en', fallbackLng: 'en',
debug: true,
interpolation: { interpolation: {
escapeValue: false, // not needed for react as it escapes by default escapeValue: false, // not needed for react as it escapes by default
+4
View File
@@ -5,4 +5,8 @@ module.exports = {
transformIgnorePatterns: [ transformIgnorePatterns: [
'node_modules/(?!((jest-)?react-native(-.*)?|@react-native(-community)?)/)', 'node_modules/(?!((jest-)?react-native(-.*)?|@react-native(-community)?)/)',
], ],
watchPlugins: [
'jest-watch-typeahead/filename',
'jest-watch-typeahead/testname',
],
} }
+13 -5
View File
@@ -8,7 +8,7 @@ import {
import getColumnNamesForCsv from './get-csv-column-names' import getColumnNamesForCsv from './get-csv-column-names'
import replaceWithNullIfAllPropertiesAreNull from './replace-with-null' import replaceWithNullIfAllPropertiesAreNull from './replace-with-null'
import { LocalDate } from '@js-joda/core' import { LocalDate } from '@js-joda/core'
import labels from '../../i18n/en/settings' import i18next from 'i18next'
export default async function importCsv(csv, deleteFirst) { export default async function importCsv(csv, deleteFirst) {
const parseFuncs = { const parseFuncs = {
@@ -46,7 +46,10 @@ export default async function importCsv(csv, deleteFirst) {
const cycleDays = await csvParser(config) const cycleDays = await csvParser(config)
.fromString(csv) .fromString(csv)
.on('header', validateHeaders) .on('header', (headers) => validateHeaders(headers))
.on('error', (error) => {
throw error
})
//remove symptoms where all fields are null //remove symptoms where all fields are null
putNullForEmptySymptoms(cycleDays) putNullForEmptySymptoms(cycleDays)
@@ -67,8 +70,11 @@ function validateHeaders(headers) {
return expectedHeaders.indexOf(header) > -1 return expectedHeaders.indexOf(header) > -1
}) })
) { ) {
const msg = `Expected CSV column titles to be ${expectedHeaders.join()}` throw new Error(
throw new Error(msg) i18next.t('hamburgerMenu.settings.data.import.error.incorrectColumns', {
incorrectColumns: expectedHeaders.join(),
})
)
} }
} }
@@ -92,7 +98,9 @@ function throwIfFutureData(cycleDays) {
day.date > today && day.date > today &&
Object.keys(day).some((symptom) => symptom != 'date' && symptom != 'note') Object.keys(day).some((symptom) => symptom != 'date' && symptom != 'note')
) { ) {
throw new Error(labels.import.errors.futureEdit) throw new Error(
i18next.t('hamburgerMenu.settings.data.import.error.futureEdit')
)
} }
} }
} }
+3 -4
View File
@@ -1,17 +1,16 @@
export default function (feeling, texture) { export default function getSensiplanMucus(feeling, texture) {
if (typeof feeling != 'number' || typeof texture != 'number') return null if (typeof feeling != 'number' || typeof texture != 'number') return null
const feelingMapping = { const feelingMapping = {
0: 0, 0: 0,
1: 1, 1: 1,
2: 2, 2: 2,
3: 4 3: 4,
} }
const textureMapping = { const textureMapping = {
0: 0, 0: 0,
1: 3, 1: 3,
2: 4 2: 4,
} }
const nfpFeelingValue = feelingMapping[feeling] const nfpFeelingValue = feelingMapping[feeling]
const nfpTextureValue = textureMapping[texture] const nfpTextureValue = textureMapping[texture]
+8 -7
View File
@@ -36,14 +36,14 @@
"@react-native-community/datetimepicker": "^6.3.1", "@react-native-community/datetimepicker": "^6.3.1",
"@react-native-community/push-notification-ios": "^1.8.0", "@react-native-community/push-notification-ios": "^1.8.0",
"csvtojson": "^2.0.8", "csvtojson": "^2.0.8",
"i18next": "^21.9.0", "i18next": "^22.0.2",
"jshashes": "^1.0.8", "jshashes": "^1.0.8",
"moment": "^2.29.4", "moment": "^2.29.4",
"object-path": "^0.11.4", "object-path": "^0.11.4",
"obv": "0.0.1", "obv": "0.0.1",
"prop-types": "^15.8.1", "prop-types": "^15.8.1",
"react": "17.0.2", "react": "17.0.2",
"react-i18next": "^11.18.3", "react-i18next": "^12.0.0",
"react-native": "0.67.4", "react-native": "0.67.4",
"react-native-calendars": "^1.1287.0", "react-native-calendars": "^1.1287.0",
"react-native-document-picker": "^8.1.1", "react-native-document-picker": "^8.1.1",
@@ -58,17 +58,18 @@
"sympto": "3.0.1" "sympto": "3.0.1"
}, },
"devDependencies": { "devDependencies": {
"@babel/core": "^7.12.9", "@babel/core": "^7.20.2",
"@babel/eslint-parser": "^7.19.1", "@babel/eslint-parser": "^7.19.1",
"@babel/preset-react": "^7.16.0", "@babel/preset-react": "^7.18.6",
"@babel/runtime": "^7.12.5", "@babel/runtime": "^7.12.5",
"@testing-library/jest-native": "^4.0.12", "@testing-library/jest-native": "^4.0.12",
"@testing-library/react-native": "^11.1.0", "@testing-library/react-native": "^11.1.0",
"basic-changelog": "gitlab:bloodyhealth/basic-changelog", "basic-changelog": "gitlab:bloodyhealth/basic-changelog",
"eslint": "7.14.0", "eslint": "^7.32.0",
"eslint-plugin-react": "^7.8.2", "eslint-plugin-react": "^7.31.10",
"husky": "^8.0.0", "husky": "^8.0.0",
"jest": "^28.1.3", "jest": "^29.1.2",
"jest-watch-typeahead": "^2.2.0",
"jetifier": "^1.6.6", "jetifier": "^1.6.6",
"metro-react-native-babel-preset": "^0.66.2", "metro-react-native-babel-preset": "^0.66.2",
"prettier": "2.4.0", "prettier": "2.4.0",
+5 -13
View File
@@ -1,27 +1,19 @@
import React from 'react' import React from 'react'
import { render, screen, fireEvent } from '@testing-library/react-native'
import AcceptLicense from '../components/AcceptLicense' import AcceptLicense from '../components/AcceptLicense'
import { saveLicenseFlag } from '../local-storage' import { saveLicenseFlag } from '../local-storage'
import { render, screen, fireEvent } from './test-utils'
jest.mock('../local-storage', () => ({ jest.mock('../local-storage', () => ({
saveLicenseFlag: jest.fn(() => Promise.resolve()), saveLicenseFlag: jest.fn(() => Promise.resolve()),
})) }))
jest.mock('react-i18next', () => ({
useTranslation: () => ({
t: (str, options) => {
return str + (options ? JSON.stringify(options) : '')
},
}),
}))
describe('AcceptLicense', () => { describe('AcceptLicense', () => {
test('On clicking OK button, the license is accepted', async () => { test('should accept license when clicking ok button', async () => {
const mockedSetLicense = jest.fn() const mockedSetLicense = jest.fn()
render(<AcceptLicense setLicense={mockedSetLicense} />) render(<AcceptLicense setLicense={mockedSetLicense} />)
const okButton = screen.getByText('ok', { exact: false }) const okButton = screen.getByText('OK')
fireEvent(okButton, 'click') fireEvent(okButton, 'click')
@@ -29,9 +21,9 @@ describe('AcceptLicense', () => {
expect(mockedSetLicense).toHaveBeenCalled() expect(mockedSetLicense).toHaveBeenCalled()
}) })
test('There is a Cancel button', async () => { test('should render cancel button', async () => {
render(<AcceptLicense setLicense={jest.fn()} />) render(<AcceptLicense setLicense={jest.fn()} />)
screen.getByText('cancel', { exact: false }) screen.getByText('Cancel')
}) })
}) })
+3 -11
View File
@@ -1,24 +1,16 @@
import React from 'react' import React from 'react'
import { render, screen } from '@testing-library/react-native'
import License from '../components/settings/License' import License from '../components/settings/License'
import { render, screen } from './test-utils'
jest.mock('react-i18next', () => ({
useTranslation: () => ({
t: (str, options) => {
return str + (options ? JSON.stringify(options) : '')
},
}),
}))
describe('License screen', () => { describe('License screen', () => {
test('It should have a correct year', async () => { test('should display license text with correct year', async () => {
render(<License />) render(<License />)
const year = new Date().getFullYear().toString() const year = new Date().getFullYear().toString()
screen.getByText(year, { exact: false }) screen.getByText(year, { exact: false })
}) })
test('It should match the snapshot', async () => { test('should match the snapshot', async () => {
const licenseScreen = render(<License />) const licenseScreen = render(<License />)
expect(licenseScreen).toMatchSnapshot() expect(licenseScreen).toMatchSnapshot()
+15 -13
View File
@@ -1,9 +1,9 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP // Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`License screen It should match the snapshot 1`] = ` exports[`License screen should match the snapshot 1`] = `
<View <View
style={ style={
Object { {
"backgroundColor": "#E9F2ED", "backgroundColor": "#E9F2ED",
"flex": 1, "flex": 1,
} }
@@ -11,8 +11,8 @@ exports[`License screen It should match the snapshot 1`] = `
> >
<RCTScrollView <RCTScrollView
contentContainerStyle={ contentContainerStyle={
Array [ [
Object { {
"backgroundColor": "#E9F2ED", "backgroundColor": "#E9F2ED",
"flexGrow": 1, "flexGrow": 1,
}, },
@@ -23,13 +23,13 @@ exports[`License screen It should match the snapshot 1`] = `
<View> <View>
<Text <Text
style={ style={
Array [ [
Object { {
"color": "#555", "color": "#555",
"fontFamily": "Jost-Book", "fontFamily": "Jost-Book",
"fontSize": 34.285714285714285, "fontSize": 34.285714285714285,
}, },
Object { {
"alignSelf": "center", "alignSelf": "center",
"color": "#3A2671", "color": "#3A2671",
"fontFamily": "Jost-Bold", "fontFamily": "Jost-Bold",
@@ -41,11 +41,11 @@ exports[`License screen It should match the snapshot 1`] = `
] ]
} }
> >
title drip. an open-source cycle tracking app
</Text> </Text>
<View <View
style={ style={
Object { {
"marginBottom": 34.285714285714285, "marginBottom": 34.285714285714285,
"marginHorizontal": 34.285714285714285, "marginHorizontal": 34.285714285714285,
} }
@@ -53,8 +53,8 @@ exports[`License screen It should match the snapshot 1`] = `
> >
<Text <Text
style={ style={
Array [ [
Object { {
"color": "#555", "color": "#555",
"fontFamily": "Jost-Book", "fontFamily": "Jost-Book",
"fontSize": 34.285714285714285, "fontSize": 34.285714285714285,
@@ -63,12 +63,14 @@ exports[`License screen It should match the snapshot 1`] = `
] ]
} }
> >
text{"currentYear":2022} Copyright (C) 2022 Heart of Code e.V.
This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details:
</Text> </Text>
<Text <Text
onPress={[Function]} onPress={[Function]}
style={ style={
Object { {
"color": "#3A2671", "color": "#3A2671",
"fontFamily": "Jost-Book", "fontFamily": "Jost-Book",
"fontSize": 34.285714285714285, "fontSize": 34.285714285714285,
@@ -3,7 +3,7 @@
exports[`Footnote component when children are present, renders them 1`] = ` exports[`Footnote component when children are present, renders them 1`] = `
<View <View
style={ style={
Object { {
"alignContent": "flex-start", "alignContent": "flex-start",
"flexDirection": "row", "flexDirection": "row",
"marginBottom": 8.571428571428571, "marginBottom": 8.571428571428571,
@@ -13,13 +13,13 @@ exports[`Footnote component when children are present, renders them 1`] = `
> >
<Text <Text
style={ style={
Array [ [
Object { {
"color": "#555", "color": "#555",
"fontFamily": "Jost-Book", "fontFamily": "Jost-Book",
"fontSize": 34.285714285714285, "fontSize": 34.285714285714285,
}, },
Object { {
"color": "#F38337", "color": "#F38337",
}, },
] ]
@@ -29,18 +29,18 @@ exports[`Footnote component when children are present, renders them 1`] = `
</Text> </Text>
<Text <Text
linkStyle={ linkStyle={
Object { {
"color": "white", "color": "white",
} }
} }
style={ style={
Array [ [
Object { {
"color": "#555", "color": "#555",
"fontFamily": "Jost-Book", "fontFamily": "Jost-Book",
"fontSize": 34.285714285714285, "fontSize": 34.285714285714285,
}, },
Object { {
"color": "#555", "color": "#555",
"paddingLeft": 21.428571428571427, "paddingLeft": 21.428571428571427,
}, },
+10
View File
@@ -0,0 +1,10 @@
import { render } from '@testing-library/react-native'
import '../i18n/i18n'
const customRender = (ui, options) => render(ui, { ...options })
// re-export everything
export * from '@testing-library/react-native'
// override render method
export { customRender as render }
+899 -706
View File
File diff suppressed because it is too large Load Diff