Compare commits
25 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| cb33d93f8e | |||
| b4f711e4db | |||
| 1d5737d8a9 | |||
| edc6f350c5 | |||
| d30b7db3fb | |||
| 6e2e03f39e | |||
| cc62e24229 | |||
| cd24522b4d | |||
| 446638d6de | |||
| ae23ef2c58 | |||
| 38b9e8b31f | |||
| 84d657cabb | |||
| 4573b93921 | |||
| ab88a4c163 | |||
| baaf89c04e | |||
| d476f6c143 | |||
| b56e0818f3 | |||
| 1b5ffaf5d6 | |||
| f68cc2b49e | |||
| 27d430a465 | |||
| 28f52e2cea | |||
| 3f87f298fb | |||
| 585f32863d | |||
| 51e1c95e71 | |||
| 36c33c69b7 |
@@ -14,3 +14,4 @@ updates:
|
||||
- dependency-name: 'react'
|
||||
- dependency-name: 'react-native'
|
||||
- dependency-name: 'react-native-push-notifications'
|
||||
- dependency-name: '@babel/core'
|
||||
|
||||
@@ -201,7 +201,3 @@ More information about how the app calculates fertility status and bleeding pred
|
||||
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.
|
||||
|
||||
## 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 { deleteDbAndOpenNew, openDb } from '../db'
|
||||
import { passwordPrompt as labels, shared } from '../i18n/en/labels'
|
||||
import { Containers, Spacing } from '../styles'
|
||||
|
||||
const cancelButton = { text: shared.cancel, style: 'cancel' }
|
||||
import { useTranslation } from 'react-i18next'
|
||||
|
||||
const PasswordPrompt = ({ enableShowApp }) => {
|
||||
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 hash = new SHA512().hex(password)
|
||||
const connected = await openDb(hash)
|
||||
|
||||
if (!connected) {
|
||||
Alert.alert(shared.incorrectPassword, shared.incorrectPasswordMessage, [
|
||||
{
|
||||
text: shared.tryAgain,
|
||||
onPress: () => setPassword(null),
|
||||
},
|
||||
])
|
||||
Alert.alert(
|
||||
t('incorrectPasswordDialog.incorrectPassword'),
|
||||
t('incorrectPasswordDialog.incorrectPasswordMessage'),
|
||||
[
|
||||
{
|
||||
text: t('incorrectPasswordDialog.tryAgain'),
|
||||
onPress: () => setPassword(null),
|
||||
},
|
||||
]
|
||||
)
|
||||
return
|
||||
}
|
||||
enableShowApp()
|
||||
@@ -42,19 +50,22 @@ const PasswordPrompt = ({ enableShowApp }) => {
|
||||
}
|
||||
|
||||
const onDeleteData = () => {
|
||||
Alert.alert(labels.areYouSureTitle, labels.areYouSure, [
|
||||
Alert.alert(t('confirmationDialog.title'), t('confirmationDialog.text'), [
|
||||
cancelButton,
|
||||
{
|
||||
text: labels.reallyDeleteData,
|
||||
text: t('confirmationDialog.confirm'),
|
||||
onPress: onDeleteDataConfirmation,
|
||||
},
|
||||
])
|
||||
}
|
||||
|
||||
const onConfirmDeletion = async () => {
|
||||
Alert.alert(labels.deleteDatabaseTitle, labels.deleteDatabaseExplainer, [
|
||||
Alert.alert(t('forgotPassword'), t('forgotPasswordDialog.text'), [
|
||||
cancelButton,
|
||||
{ text: labels.deleteData, onPress: onDeleteData },
|
||||
{
|
||||
text: t('forgotPasswordDialog.confirm'),
|
||||
onPress: onDeleteData,
|
||||
},
|
||||
])
|
||||
}
|
||||
|
||||
@@ -65,17 +76,13 @@ const PasswordPrompt = ({ enableShowApp }) => {
|
||||
<KeyboardAvoidingView behavior="padding" keyboardVerticalOffset={150}>
|
||||
<AppTextInput
|
||||
onChangeText={setPassword}
|
||||
secureTextEntry={true}
|
||||
placeholder={labels.enterPassword}
|
||||
secureTextEntry
|
||||
placeholder={t('enterPassword')}
|
||||
/>
|
||||
<View style={styles.containerButtons}>
|
||||
<Button onPress={onConfirmDeletion}>{labels.forgotPassword}</Button>
|
||||
<Button
|
||||
disabled={!isPasswordEntered}
|
||||
isCTA={isPasswordEntered}
|
||||
onPress={unlockApp}
|
||||
>
|
||||
{labels.title}
|
||||
<Button onPress={onConfirmDeletion}>{t('forgotPassword')}</Button>
|
||||
<Button disabled={!password} isCTA={!!password} onPress={unlockApp}>
|
||||
{t('unlockApp')}
|
||||
</Button>
|
||||
</View>
|
||||
</KeyboardAvoidingView>
|
||||
@@ -7,7 +7,7 @@ import App from './app'
|
||||
import AppLoadingView from './common/app-loading'
|
||||
import AppStatusBar from './common/app-status-bar'
|
||||
import AcceptLicense from './AcceptLicense'
|
||||
import PasswordPrompt from './password-prompt'
|
||||
import PasswordPrompt from './PasswordPrompt'
|
||||
|
||||
export default function AppWrapper() {
|
||||
const [isLoading, setIsLoading] = useState(true)
|
||||
|
||||
@@ -6,16 +6,17 @@ import AppText from '../common/app-text'
|
||||
import CloseIcon from '../common/close-icon'
|
||||
|
||||
import { Containers, Spacing } from '../../styles'
|
||||
import { chart } from '../../i18n/en/labels'
|
||||
import { useTranslation } from 'react-i18next'
|
||||
|
||||
const image = require('../../assets/swipe.png')
|
||||
|
||||
const Tutorial = ({ onClose }) => {
|
||||
const { t } = useTranslation()
|
||||
return (
|
||||
<View style={styles.container}>
|
||||
<Image resizeMode="contain" source={image} style={styles.image} />
|
||||
<View style={styles.textContainer}>
|
||||
<AppText>{chart.tutorial}</AppText>
|
||||
<AppText>{t('chart.tutorial')}</AppText>
|
||||
</View>
|
||||
<CloseIcon onClose={onClose} />
|
||||
</View>
|
||||
@@ -9,7 +9,7 @@ import HorizontalGrid from './horizontal-grid'
|
||||
import MainGrid from './main-grid'
|
||||
import NoData from './no-data'
|
||||
import NoTemperature from './no-temperature'
|
||||
import Tutorial from './tutorial'
|
||||
import Tutorial from './Tutorial'
|
||||
import YAxis from './y-axis'
|
||||
|
||||
import { getCycleDaysSortedByDate } from '../../db'
|
||||
|
||||
@@ -6,20 +6,23 @@ import MenuItem from './menu-item'
|
||||
|
||||
import { Containers } from '../../styles'
|
||||
import { pages } from '../pages'
|
||||
import { useTranslation } from 'react-i18next'
|
||||
|
||||
const Menu = ({ currentPage, navigate }) => {
|
||||
const menuItems = pages.filter((page) => page.isInMenu)
|
||||
|
||||
const { t } = useTranslation(null, { keyPrefix: 'bottomMenu' })
|
||||
|
||||
return (
|
||||
<View style={styles.container}>
|
||||
{menuItems.map(({ icon, label, component }) => {
|
||||
{menuItems.map(({ icon, labelKey, component }) => {
|
||||
return (
|
||||
<MenuItem
|
||||
isActive={component === currentPage}
|
||||
onPress={() => navigate(component)}
|
||||
icon={icon}
|
||||
key={label}
|
||||
label={label}
|
||||
key={labelKey}
|
||||
label={t(labelKey)}
|
||||
/>
|
||||
)
|
||||
})}
|
||||
|
||||
+3
-15
@@ -1,75 +1,63 @@
|
||||
import settingsViews from './settings'
|
||||
|
||||
import settingsLabels from '../i18n/en/settings'
|
||||
const labels = settingsLabels.menuItems
|
||||
|
||||
export const pages = [
|
||||
{
|
||||
component: 'Home',
|
||||
icon: 'home',
|
||||
label: 'Home',
|
||||
},
|
||||
{
|
||||
component: 'Calendar',
|
||||
icon: 'calendar',
|
||||
isInMenu: true,
|
||||
label: 'Calendar',
|
||||
labelKey: 'calendar',
|
||||
parent: 'Home',
|
||||
},
|
||||
{
|
||||
component: 'Chart',
|
||||
icon: 'chart',
|
||||
isInMenu: true,
|
||||
label: 'Chart',
|
||||
labelKey: 'chart',
|
||||
parent: 'Home',
|
||||
},
|
||||
{
|
||||
component: 'Stats',
|
||||
icon: 'statistics',
|
||||
isInMenu: true,
|
||||
label: 'Stats',
|
||||
labelKey: 'stats',
|
||||
parent: 'Home',
|
||||
},
|
||||
{
|
||||
children: Object.keys(settingsViews),
|
||||
component: 'SettingsMenu',
|
||||
icon: 'settings',
|
||||
label: 'Settings',
|
||||
parent: 'Home',
|
||||
},
|
||||
{
|
||||
component: 'Reminders',
|
||||
label: labels.reminders.name,
|
||||
parent: 'SettingsMenu',
|
||||
},
|
||||
{
|
||||
component: 'NfpSettings',
|
||||
label: labels.nfpSettings.name,
|
||||
parent: 'SettingsMenu',
|
||||
},
|
||||
{
|
||||
component: 'DataManagement',
|
||||
label: labels.dataManagement.name,
|
||||
parent: 'SettingsMenu',
|
||||
},
|
||||
{
|
||||
component: 'Password',
|
||||
label: labels.password.name,
|
||||
parent: 'SettingsMenu',
|
||||
},
|
||||
{
|
||||
component: 'About',
|
||||
label: 'About',
|
||||
parent: 'SettingsMenu',
|
||||
},
|
||||
{
|
||||
component: 'License',
|
||||
label: 'License',
|
||||
parent: 'SettingsMenu',
|
||||
},
|
||||
{
|
||||
component: 'PrivacyPolicy',
|
||||
label: 'PrivacyPolicy',
|
||||
parent: 'SettingsMenu',
|
||||
},
|
||||
{
|
||||
|
||||
@@ -3,26 +3,34 @@ import React, { useState } from 'react'
|
||||
import AppLoadingView from '../../common/app-loading'
|
||||
import AppPage from '../../common/app-page'
|
||||
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 ExportData from './ExportData'
|
||||
|
||||
const DataManagement = () => {
|
||||
const [isLoading, setIsLoading] = useState(false)
|
||||
const [isDeletingData, setIsDeletingData] = useState(false)
|
||||
|
||||
const startExport = () => {
|
||||
setIsDeletingData(false)
|
||||
openShareDialogAndExport()
|
||||
}
|
||||
|
||||
if (isLoading) return <AppLoadingView />
|
||||
|
||||
return (
|
||||
<AppPage>
|
||||
<ExportData
|
||||
resetIsDeletingData={() => setIsDeletingData(false)}
|
||||
setIsLoading={setIsLoading}
|
||||
/>
|
||||
<Segment title={labels.export.button}>
|
||||
<AppText>{labels.export.segmentExplainer}</AppText>
|
||||
<Button isCTA onPress={startExport}>
|
||||
{labels.export.button}
|
||||
</Button>
|
||||
</Segment>
|
||||
<ImportData
|
||||
resetIsDeletingData={() => setIsDeletingData(false)}
|
||||
setIsLoading={setIsLoading}
|
||||
|
||||
@@ -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')
|
||||
return fileContent
|
||||
} 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) {
|
||||
const errorMessage = t('errors.noDataImported', { message })
|
||||
const errorMessage = t('error.noDataImported', { message })
|
||||
alertError(errorMessage)
|
||||
}
|
||||
|
||||
|
||||
@@ -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 +0,0 @@
|
||||
{}
|
||||
+30
-11
@@ -1,4 +1,12 @@
|
||||
{
|
||||
"bottomMenu": {
|
||||
"calendar": "Calendar",
|
||||
"chart": "Chart",
|
||||
"stats": "Stats"
|
||||
},
|
||||
"chart": {
|
||||
"tutorial": "You can swipe the chart to view more dates."
|
||||
},
|
||||
"cycleDay": {
|
||||
"symptomBox": {
|
||||
"bleeding": "Bleeding",
|
||||
@@ -81,16 +89,6 @@
|
||||
},
|
||||
"settings": {
|
||||
"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": {
|
||||
"button": "Import data",
|
||||
"dialog": {
|
||||
@@ -100,9 +98,10 @@
|
||||
"replace": "Import and replace",
|
||||
"title": "Keep existing data?"
|
||||
},
|
||||
"errors": {
|
||||
"error": {
|
||||
"couldNotOpenFile": "Could not open file",
|
||||
"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"
|
||||
},
|
||||
"segmentExplainer": "Import data in CSV format",
|
||||
@@ -133,6 +132,26 @@
|
||||
"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": {
|
||||
"noData": "At least one completed cycle is needed to display stats.",
|
||||
"intro": "Basic statistics about the length of your cycles.",
|
||||
|
||||
@@ -3,10 +3,6 @@ export const home = {
|
||||
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 = {
|
||||
cancel: 'Cancel',
|
||||
save: 'Save',
|
||||
@@ -30,17 +26,6 @@ export const shared = {
|
||||
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 = {
|
||||
predictionInFuture: (startDays, endDays) =>
|
||||
`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}.`,
|
||||
}
|
||||
|
||||
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 = {
|
||||
fertile: 'fertile',
|
||||
infertile: 'infertile',
|
||||
|
||||
+10
-16
@@ -1,23 +1,17 @@
|
||||
import links from './links'
|
||||
|
||||
export default {
|
||||
menuItems: {
|
||||
reminders: {
|
||||
name: 'Reminders',
|
||||
text: 'turn on/off reminders',
|
||||
},
|
||||
nfpSettings: {
|
||||
name: 'NFP settings',
|
||||
text: 'define how you want to use NFP',
|
||||
},
|
||||
dataManagement: {
|
||||
name: 'Data',
|
||||
text: 'import, export or delete your data',
|
||||
},
|
||||
password: {
|
||||
name: 'Password',
|
||||
text: '',
|
||||
export: {
|
||||
errors: {
|
||||
noData: 'There is no data to export',
|
||||
couldNotConvert: 'Could not convert data to CSV',
|
||||
problemSharing: 'There was a problem sharing the data export file',
|
||||
},
|
||||
title: 'My drip. data export',
|
||||
subject: 'My drip. data export',
|
||||
button: 'Export data',
|
||||
segmentExplainer:
|
||||
'Export data in CSV format for backup or so you can use it elsewhere',
|
||||
},
|
||||
deleteSegment: {
|
||||
title: 'Delete app data',
|
||||
|
||||
@@ -17,7 +17,6 @@ i18n
|
||||
compatibilityJSON: 'v3', // TODO: migrate json to v4 and afterwards remove it
|
||||
resources,
|
||||
fallbackLng: 'en',
|
||||
debug: true,
|
||||
|
||||
interpolation: {
|
||||
escapeValue: false, // not needed for react as it escapes by default
|
||||
|
||||
@@ -5,4 +5,8 @@ module.exports = {
|
||||
transformIgnorePatterns: [
|
||||
'node_modules/(?!((jest-)?react-native(-.*)?|@react-native(-community)?)/)',
|
||||
],
|
||||
watchPlugins: [
|
||||
'jest-watch-typeahead/filename',
|
||||
'jest-watch-typeahead/testname',
|
||||
],
|
||||
}
|
||||
|
||||
@@ -8,7 +8,7 @@ import {
|
||||
import getColumnNamesForCsv from './get-csv-column-names'
|
||||
import replaceWithNullIfAllPropertiesAreNull from './replace-with-null'
|
||||
import { LocalDate } from '@js-joda/core'
|
||||
import labels from '../../i18n/en/settings'
|
||||
import i18next from 'i18next'
|
||||
|
||||
export default async function importCsv(csv, deleteFirst) {
|
||||
const parseFuncs = {
|
||||
@@ -46,7 +46,10 @@ export default async function importCsv(csv, deleteFirst) {
|
||||
|
||||
const cycleDays = await csvParser(config)
|
||||
.fromString(csv)
|
||||
.on('header', validateHeaders)
|
||||
.on('header', (headers) => validateHeaders(headers))
|
||||
.on('error', (error) => {
|
||||
throw error
|
||||
})
|
||||
|
||||
//remove symptoms where all fields are null
|
||||
putNullForEmptySymptoms(cycleDays)
|
||||
@@ -67,8 +70,11 @@ function validateHeaders(headers) {
|
||||
return expectedHeaders.indexOf(header) > -1
|
||||
})
|
||||
) {
|
||||
const msg = `Expected CSV column titles to be ${expectedHeaders.join()}`
|
||||
throw new Error(msg)
|
||||
throw new Error(
|
||||
i18next.t('hamburgerMenu.settings.data.import.error.incorrectColumns', {
|
||||
incorrectColumns: expectedHeaders.join(),
|
||||
})
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -92,7 +98,9 @@ function throwIfFutureData(cycleDays) {
|
||||
day.date > today &&
|
||||
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
@@ -1,17 +1,16 @@
|
||||
export default function (feeling, texture) {
|
||||
|
||||
export default function getSensiplanMucus(feeling, texture) {
|
||||
if (typeof feeling != 'number' || typeof texture != 'number') return null
|
||||
|
||||
const feelingMapping = {
|
||||
0: 0,
|
||||
1: 1,
|
||||
2: 2,
|
||||
3: 4
|
||||
3: 4,
|
||||
}
|
||||
const textureMapping = {
|
||||
0: 0,
|
||||
1: 3,
|
||||
2: 4
|
||||
2: 4,
|
||||
}
|
||||
const nfpFeelingValue = feelingMapping[feeling]
|
||||
const nfpTextureValue = textureMapping[texture]
|
||||
|
||||
+8
-7
@@ -36,14 +36,14 @@
|
||||
"@react-native-community/datetimepicker": "^6.3.1",
|
||||
"@react-native-community/push-notification-ios": "^1.8.0",
|
||||
"csvtojson": "^2.0.8",
|
||||
"i18next": "^21.9.0",
|
||||
"i18next": "^22.0.2",
|
||||
"jshashes": "^1.0.8",
|
||||
"moment": "^2.29.4",
|
||||
"object-path": "^0.11.4",
|
||||
"obv": "0.0.1",
|
||||
"prop-types": "^15.8.1",
|
||||
"react": "17.0.2",
|
||||
"react-i18next": "^11.18.3",
|
||||
"react-i18next": "^12.0.0",
|
||||
"react-native": "0.67.4",
|
||||
"react-native-calendars": "^1.1287.0",
|
||||
"react-native-document-picker": "^8.1.1",
|
||||
@@ -58,17 +58,18 @@
|
||||
"sympto": "3.0.1"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@babel/core": "^7.12.9",
|
||||
"@babel/core": "^7.20.2",
|
||||
"@babel/eslint-parser": "^7.19.1",
|
||||
"@babel/preset-react": "^7.16.0",
|
||||
"@babel/preset-react": "^7.18.6",
|
||||
"@babel/runtime": "^7.12.5",
|
||||
"@testing-library/jest-native": "^4.0.12",
|
||||
"@testing-library/react-native": "^11.1.0",
|
||||
"basic-changelog": "gitlab:bloodyhealth/basic-changelog",
|
||||
"eslint": "7.14.0",
|
||||
"eslint-plugin-react": "^7.8.2",
|
||||
"eslint": "^7.32.0",
|
||||
"eslint-plugin-react": "^7.31.10",
|
||||
"husky": "^8.0.0",
|
||||
"jest": "^28.1.3",
|
||||
"jest": "^29.1.2",
|
||||
"jest-watch-typeahead": "^2.2.0",
|
||||
"jetifier": "^1.6.6",
|
||||
"metro-react-native-babel-preset": "^0.66.2",
|
||||
"prettier": "2.4.0",
|
||||
|
||||
@@ -1,27 +1,19 @@
|
||||
import React from 'react'
|
||||
import { render, screen, fireEvent } from '@testing-library/react-native'
|
||||
import AcceptLicense from '../components/AcceptLicense'
|
||||
|
||||
import { saveLicenseFlag } from '../local-storage'
|
||||
import { render, screen, fireEvent } from './test-utils'
|
||||
|
||||
jest.mock('../local-storage', () => ({
|
||||
saveLicenseFlag: jest.fn(() => Promise.resolve()),
|
||||
}))
|
||||
|
||||
jest.mock('react-i18next', () => ({
|
||||
useTranslation: () => ({
|
||||
t: (str, options) => {
|
||||
return str + (options ? JSON.stringify(options) : '')
|
||||
},
|
||||
}),
|
||||
}))
|
||||
|
||||
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()
|
||||
render(<AcceptLicense setLicense={mockedSetLicense} />)
|
||||
|
||||
const okButton = screen.getByText('ok', { exact: false })
|
||||
const okButton = screen.getByText('OK')
|
||||
|
||||
fireEvent(okButton, 'click')
|
||||
|
||||
@@ -29,9 +21,9 @@ describe('AcceptLicense', () => {
|
||||
expect(mockedSetLicense).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
test('There is a Cancel button', async () => {
|
||||
test('should render cancel button', async () => {
|
||||
render(<AcceptLicense setLicense={jest.fn()} />)
|
||||
|
||||
screen.getByText('cancel', { exact: false })
|
||||
screen.getByText('Cancel')
|
||||
})
|
||||
})
|
||||
|
||||
+3
-11
@@ -1,24 +1,16 @@
|
||||
import React from 'react'
|
||||
import { render, screen } from '@testing-library/react-native'
|
||||
import License from '../components/settings/License'
|
||||
|
||||
jest.mock('react-i18next', () => ({
|
||||
useTranslation: () => ({
|
||||
t: (str, options) => {
|
||||
return str + (options ? JSON.stringify(options) : '')
|
||||
},
|
||||
}),
|
||||
}))
|
||||
import { render, screen } from './test-utils'
|
||||
|
||||
describe('License screen', () => {
|
||||
test('It should have a correct year', async () => {
|
||||
test('should display license text with correct year', async () => {
|
||||
render(<License />)
|
||||
const year = new Date().getFullYear().toString()
|
||||
|
||||
screen.getByText(year, { exact: false })
|
||||
})
|
||||
|
||||
test('It should match the snapshot', async () => {
|
||||
test('should match the snapshot', async () => {
|
||||
const licenseScreen = render(<License />)
|
||||
|
||||
expect(licenseScreen).toMatchSnapshot()
|
||||
|
||||
@@ -1,9 +1,9 @@
|
||||
// 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
|
||||
style={
|
||||
Object {
|
||||
{
|
||||
"backgroundColor": "#E9F2ED",
|
||||
"flex": 1,
|
||||
}
|
||||
@@ -11,8 +11,8 @@ exports[`License screen It should match the snapshot 1`] = `
|
||||
>
|
||||
<RCTScrollView
|
||||
contentContainerStyle={
|
||||
Array [
|
||||
Object {
|
||||
[
|
||||
{
|
||||
"backgroundColor": "#E9F2ED",
|
||||
"flexGrow": 1,
|
||||
},
|
||||
@@ -23,13 +23,13 @@ exports[`License screen It should match the snapshot 1`] = `
|
||||
<View>
|
||||
<Text
|
||||
style={
|
||||
Array [
|
||||
Object {
|
||||
[
|
||||
{
|
||||
"color": "#555",
|
||||
"fontFamily": "Jost-Book",
|
||||
"fontSize": 34.285714285714285,
|
||||
},
|
||||
Object {
|
||||
{
|
||||
"alignSelf": "center",
|
||||
"color": "#3A2671",
|
||||
"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>
|
||||
<View
|
||||
style={
|
||||
Object {
|
||||
{
|
||||
"marginBottom": 34.285714285714285,
|
||||
"marginHorizontal": 34.285714285714285,
|
||||
}
|
||||
@@ -53,8 +53,8 @@ exports[`License screen It should match the snapshot 1`] = `
|
||||
>
|
||||
<Text
|
||||
style={
|
||||
Array [
|
||||
Object {
|
||||
[
|
||||
{
|
||||
"color": "#555",
|
||||
"fontFamily": "Jost-Book",
|
||||
"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
|
||||
onPress={[Function]}
|
||||
style={
|
||||
Object {
|
||||
{
|
||||
"color": "#3A2671",
|
||||
"fontFamily": "Jost-Book",
|
||||
"fontSize": 34.285714285714285,
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
exports[`Footnote component when children are present, renders them 1`] = `
|
||||
<View
|
||||
style={
|
||||
Object {
|
||||
{
|
||||
"alignContent": "flex-start",
|
||||
"flexDirection": "row",
|
||||
"marginBottom": 8.571428571428571,
|
||||
@@ -13,13 +13,13 @@ exports[`Footnote component when children are present, renders them 1`] = `
|
||||
>
|
||||
<Text
|
||||
style={
|
||||
Array [
|
||||
Object {
|
||||
[
|
||||
{
|
||||
"color": "#555",
|
||||
"fontFamily": "Jost-Book",
|
||||
"fontSize": 34.285714285714285,
|
||||
},
|
||||
Object {
|
||||
{
|
||||
"color": "#F38337",
|
||||
},
|
||||
]
|
||||
@@ -29,18 +29,18 @@ exports[`Footnote component when children are present, renders them 1`] = `
|
||||
</Text>
|
||||
<Text
|
||||
linkStyle={
|
||||
Object {
|
||||
{
|
||||
"color": "white",
|
||||
}
|
||||
}
|
||||
style={
|
||||
Array [
|
||||
Object {
|
||||
[
|
||||
{
|
||||
"color": "#555",
|
||||
"fontFamily": "Jost-Book",
|
||||
"fontSize": 34.285714285714285,
|
||||
},
|
||||
Object {
|
||||
{
|
||||
"color": "#555",
|
||||
"paddingLeft": 21.428571428571427,
|
||||
},
|
||||
|
||||
@@ -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 }
|
||||
Reference in New Issue
Block a user