Skip to content
Snippets Groups Projects
Commit 3efc4efc authored by Sofiya Tepikin's avatar Sofiya Tepikin
Browse files

Merge branch 'rebased-redesign' into 'master'

Rebased redesign

See merge request bloodyhealth/drip!305
parents 0451b078 199aa198
No related branches found
No related tags found
No related merge requests found
Showing
with 555 additions and 473 deletions
...@@ -7,7 +7,7 @@ import { openDb } from '../db' ...@@ -7,7 +7,7 @@ import { openDb } from '../db'
import App from './app' import App from './app'
import PasswordPrompt from './password-prompt' import PasswordPrompt from './password-prompt'
import License from './license' import License from './license'
import AppLoadingView from './app-loading' import AppLoadingView from './common/app-loading'
import store from "../store" import store from "../store"
import { Provider } from 'react-redux' import { Provider } from 'react-redux'
......
import React, { Component } from 'react' import React, { Component } from 'react'
import { View, BackHandler } from 'react-native' import { BackHandler, StyleSheet, View } from 'react-native'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { connect } from 'react-redux' import { connect } from 'react-redux'
...@@ -10,7 +10,7 @@ import { getNavigation, navigate, goBack } from '../slices/navigation' ...@@ -10,7 +10,7 @@ import { getNavigation, navigate, goBack } from '../slices/navigation'
import Header from './header' import Header from './header'
import Menu from './menu' import Menu from './menu'
import { viewsList } from './views' import { viewsList } from './views'
import { isSymptomView, isSettingsView } from './pages' import { isSettingsView } from './pages'
import { headerTitles } from '../i18n/en/labels' import { headerTitles } from '../i18n/en/labels'
import setupNotifications from '../lib/notifications' import setupNotifications from '../lib/notifications'
...@@ -64,9 +64,8 @@ class App extends Component { ...@@ -64,9 +64,8 @@ class App extends Component {
const Page = viewsList[currentPage] const Page = viewsList[currentPage]
const title = headerTitles[currentPage] const title = headerTitles[currentPage]
const isSymptomEditView = isSymptomView(currentPage)
const isSettingsSubView = isSettingsView(currentPage) const isSettingsSubView = isSettingsView(currentPage)
const isCycleDayView = currentPage === 'CycleDay' const isTemperatureEditView = currentPage === 'TemperatureEditView'
const headerProps = { const headerProps = {
title, title,
...@@ -76,24 +75,25 @@ class App extends Component { ...@@ -76,24 +75,25 @@ class App extends Component {
const pageProps = { const pageProps = {
cycleDay: date && getCycleDay(date), cycleDay: date && getCycleDay(date),
date, date,
isTemperatureEditView,
} }
return ( return (
<View style={{ flex: 1 }}> <View style={styles.container}>
{ <Header { ...headerProps } />
!isSymptomEditView &&
!isCycleDayView &&
<Header { ...headerProps } />
}
<Page { ...pageProps } /> <Page { ...pageProps } />
<Menu />
{ !isSymptomEditView && <Menu /> }
</View> </View>
) )
} }
} }
const styles = StyleSheet.create({
container: {
flex: 1
}
})
const mapStateToProps = (state) => { const mapStateToProps = (state) => {
return({ return({
date: getDate(state), date: getDate(state),
......
import React from 'react'
import PropTypes from 'prop-types'
import { TouchableOpacity } from 'react-native'
import AppText from './app-text'
import styles from '../styles'
export default function Button({
backgroundColor,
children,
onPress,
style,
testID
}) {
return (
<TouchableOpacity
onPress={onPress}
style={[styles.button, style, { backgroundColor }]}
testID={testID}
>
<AppText style={styles.homeButtonText}>{children}</AppText>
</TouchableOpacity>
)
}
Button.propTypes = {
backgroundColor: PropTypes.string,
children: PropTypes.node,
onPress: PropTypes.func,
style: PropTypes.object,
testID: PropTypes.string
}
import React, { Component } from 'react' import React, { Component } from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { StyleSheet, View } from 'react-native'
import { CalendarList } from 'react-native-calendars' import { CalendarList } from 'react-native-calendars'
import { connect } from 'react-redux'
import { connect } from 'react-redux'
import { setDate } from '../slices/date' import { setDate } from '../slices/date'
import { navigate } from '../slices/navigation' import { navigate } from '../slices/navigation'
import { LocalDate } from 'js-joda'
import { getBleedingDaysSortedByDate } from '../db' import { getBleedingDaysSortedByDate } from '../db'
import cycleModule from '../lib/cycle' import cycleModule from '../lib/cycle'
import { shadesOfRed, calendarTheme } from '../styles/index'
import styles from '../styles/index'
import nothingChanged from '../db/db-unchanged' import nothingChanged from '../db/db-unchanged'
import {
calendarTheme,
predictionToCalFormat,
toCalFormat,
todayToCalFormat,
} from './helpers/calendar'
class CalendarView extends Component { class CalendarView extends Component {
static propTypes = { static propTypes = {
setDate: PropTypes.func.isRequired, setDate: PropTypes.func.isRequired,
navigate: PropTypes.func.isRequired navigate: PropTypes.func.isRequired,
} }
constructor(props) { constructor(props) {
...@@ -26,7 +30,7 @@ class CalendarView extends Component { ...@@ -26,7 +30,7 @@ class CalendarView extends Component {
this.state = { this.state = {
bleedingDaysInCalFormat: toCalFormat(this.bleedingDays), bleedingDaysInCalFormat: toCalFormat(this.bleedingDays),
predictedBleedingDaysInCalFormat: predictionToCalFormat(predictedMenses), predictedBleedingDaysInCalFormat: predictionToCalFormat(predictedMenses),
todayInCalFormat: todayToCalFormat() todayInCalFormat: todayToCalFormat(),
} }
this.bleedingDays.addListener(this.setStateWithCalFormattedDays) this.bleedingDays.addListener(this.setStateWithCalFormattedDays)
...@@ -38,7 +42,7 @@ class CalendarView extends Component { ...@@ -38,7 +42,7 @@ class CalendarView extends Component {
this.setState({ this.setState({
bleedingDaysInCalFormat: toCalFormat(this.bleedingDays), bleedingDaysInCalFormat: toCalFormat(this.bleedingDays),
predictedBleedingDaysInCalFormat: predictionToCalFormat(predictedMenses), predictedBleedingDaysInCalFormat: predictionToCalFormat(predictedMenses),
todayInCalFormat: todayToCalFormat() todayInCalFormat: todayToCalFormat(),
}) })
} }
...@@ -52,93 +56,44 @@ class CalendarView extends Component { ...@@ -52,93 +56,44 @@ class CalendarView extends Component {
} }
render() { render() {
const {
todayInCalFormat,
bleedingDaysInCalFormat,
predictedBleedingDaysInCalFormat,
} = this.state
const markedDates = Object.assign(
{},
todayInCalFormat,
bleedingDaysInCalFormat,
predictedBleedingDaysInCalFormat
)
return ( return (
<CalendarList <View style={styles.container}>
onDayPress={this.passDateToDayView.bind(this)} <CalendarList
markedDates={ // If firstDay=1 week starts from Monday. Note that dayNames and dayNamesShort should still start from Sunday.
Object.assign( firstDay={1}
{}, onDayPress={this.passDateToDayView.bind(this)}
this.state.todayInCalFormat, markedDates={markedDates}
this.state.bleedingDaysInCalFormat, markingType='custom'
this.state.predictedBleedingDaysInCalFormat theme={calendarTheme}
) // Max amount of months allowed to scroll to the past.
} pastScrollRange={120}
markingType={'custom'} />
// If firstDay=1 week starts from Monday. Note that dayNames and dayNamesShort should still start from Sunday. </View>
firstDay={1}
theme={calendarTheme}
/>
) )
} }
} }
const styles = StyleSheet.create({
container: { flex: 1 },
})
const mapDispatchToProps = (dispatch) => { const mapDispatchToProps = (dispatch) => {
return({ return {
setDate: (date) => dispatch(setDate(date)), setDate: (date) => dispatch(setDate(date)),
navigate: (page) => dispatch(navigate(page)), navigate: (page) => dispatch(navigate(page)),
}) }
}
export default connect(
null,
mapDispatchToProps,
)(CalendarView)
function toCalFormat(bleedingDaysSortedByDate) {
const todayDateString = LocalDate.now().toString()
return bleedingDaysSortedByDate.reduce((acc, day) => {
acc[day.date] = {
customStyles: {
container: {
backgroundColor: shadesOfRed[day.bleeding.value],
}
}
}
if (day.date === todayDateString) {
acc[day.date].customStyles.text = styles.calendarToday
}
return acc
}, {})
}
function predictionToCalFormat(predictedDays) {
if (!predictedDays.length) return {}
const todayDateString = LocalDate.now().toString()
const middleIndex = (predictedDays[0].length - 1) / 2
return predictedDays.reduce((acc, setOfDays) => {
setOfDays.reduce((accSet, day, i) => {
accSet[day] = {
customStyles: {
container: {
borderColor: (i === middleIndex) ? shadesOfRed[3] : shadesOfRed[2],
borderWidth: 3,
},
text: {
marginTop: 1,
}
}
}
if (day === todayDateString) {
accSet[day].customStyles.text = Object.assign(
{},
styles.calendarToday,
{marginTop: -2}
)
}
return accSet
}, acc)
return acc
}, {})
} }
function todayToCalFormat() { export default connect(null, mapDispatchToProps)(CalendarView)
const todayDateString = LocalDate.now().toString()
const todayFormated = {}
todayFormated[todayDateString] = {
customStyles: {
text: styles.calendarToday
}
}
return todayFormated
}
\ No newline at end of file
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View } from 'react-native' import { StyleSheet, View } from 'react-native'
import AppText from '../app-text' import AppText from '../common/app-text'
import DripHomeIcon from '../../assets/drip-home-icons'
import styles from './styles'
import { cycleDayColor } from '../../styles'
import { Typography } from '../../styles'
import { CHART_YAXIS_WIDTH } from '../../config'
import { shared as labels } from '../../i18n/en/labels' import { shared as labels } from '../../i18n/en/labels'
const ChartLegend = ({ xAxisHeight }) => { const ChartLegend = ({ height }) => {
return ( return (
<View style={[styles.yAxis, styles.chartLegend, {height: xAxisHeight}]}> <View style={[styles.container, { height }]}>
<DripHomeIcon <AppText style={styles.textBold}>#</AppText>
name="circle" <AppText style={styles.text}>{labels.date}</AppText>
size={styles.yAxis.width - 7}
color={cycleDayColor}
/>
<AppText style={styles.yAxisLabels.dateLabel}>
{labels.date.toLowerCase()}
</AppText>
</View> </View>
) )
} }
ChartLegend.propTypes = { ChartLegend.propTypes = {
xAxisHeight: PropTypes.number.isRequired height: PropTypes.number.isRequired
} }
const styles = StyleSheet.create({
container: {
alignItems: 'center',
justifyContent: 'flex-end',
width: CHART_YAXIS_WIDTH
},
text: {
...Typography.label,
},
textBold: {
...Typography.labelBold
}
})
export default ChartLegend export default ChartLegend
...@@ -3,20 +3,16 @@ import PropTypes from 'prop-types' ...@@ -3,20 +3,16 @@ import PropTypes from 'prop-types'
import { Shape } from 'react-native/Libraries/ART/ReactNativeART' import { Shape } from 'react-native/Libraries/ART/ReactNativeART'
import styles from './styles' import { Colors } from '../../styles'
import { CHART_STROKE_WIDTH, CHART_GRID_LINE_HORIZONTAL_WIDTH } from '../../config'
const ChartLine = ({ path, isNfpLine = false }) => { const ChartLine = ({ path, isNfpLine }) => {
const strokeStyle = const color = isNfpLine ? Colors.orange : Colors.grey
isNfpLine ? styles.nfpLine.stroke : styles.column.stroke.color const width = isNfpLine
const strokeWidth = ? CHART_STROKE_WIDTH : CHART_GRID_LINE_HORIZONTAL_WIDTH * 2.5
isNfpLine ? styles.nfpLine.strokeWidth : styles.column.stroke.width
return ( return (
<Shape <Shape d={path} stroke={color} strokeWidth={width} />
stroke={strokeStyle}
strokeWidth={strokeWidth}
d={path}
/>
) )
} }
...@@ -25,4 +21,8 @@ ChartLine.propTypes = { ...@@ -25,4 +21,8 @@ ChartLine.propTypes = {
isNfpLine: PropTypes.bool, isNfpLine: PropTypes.bool,
} }
ChartLine.defaultProps = {
isNfpLine: false
}
export default ChartLine export default ChartLine
import React, { Component } from 'react' import React, { Component } from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View, FlatList, ActivityIndicator } from 'react-native' import { ActivityIndicator, FlatList, Dimensions, StyleSheet, View } from 'react-native'
import AppLoadingView from '../common/app-loading'
import AppPage from '../common/app-page'
import AppText from '../common/app-text'
import NoData from './no-data'
import AppLoadingView from '../app-loading'
import YAxis from './y-axis'
import nfpLines from './nfp-lines'
import DayColumn from './day-column' import DayColumn from './day-column'
import HorizontalGrid from './horizontal-grid' import HorizontalGrid from './horizontal-grid'
import AppText from '../app-text' import NoData from './no-data'
import Tutorial from './tutorial'
import YAxis from './y-axis'
import { connect } from 'react-redux' import { connect } from 'react-redux'
import { navigate } from '../../slices/navigation' import { navigate } from '../../slices/navigation'
import { getCycleDaysSortedByDate } from '../../db' import { getCycleDaysSortedByDate } from '../../db'
import nothingChanged from '../../db/db-unchanged' import nothingChanged from '../../db/db-unchanged'
import { scaleObservable } from '../../local-storage' import { getChartFlag, scaleObservable, setChartFlag } from '../../local-storage'
import { makeColumnInfo } from '../helpers/chart' import { makeColumnInfo, nfpLines } from '../helpers/chart'
import config from '../../config' import {
CHART_COLUMN_WIDTH,
SYMPTOMS,
CHART_SYMPTOM_HEIGHT_RATIO,
CHART_XAXIS_HEIGHT_RATIO
} from '../../config'
import { shared } from '../../i18n/en/labels' import { shared } from '../../i18n/en/labels'
import styles from './styles' import { Colors, Spacing } from '../../styles'
class CycleChart extends Component { class CycleChart extends Component {
static propTypes = { static propTypes = {
...@@ -36,13 +43,37 @@ class CycleChart extends Component { ...@@ -36,13 +43,37 @@ class CycleChart extends Component {
this.getFhmAndLtlInfo = nfpLines() this.getFhmAndLtlInfo = nfpLines()
this.shouldShowTemperatureColumn = false this.shouldShowTemperatureColumn = false
this.checkShouldShowHint()
this.prepareSymptomData() this.prepareSymptomData()
} }
componentWillUnmount() {
this.cycleDaysSortedByDate.removeListener(this.handleDbChange)
this.removeObvListener()
}
checkShouldShowHint = async () => {
const flag = await getChartFlag()
const shouldShowHint = flag === 'true' ? true : false
this.setState({ shouldShowHint })
}
setShouldShowHint = async () => {
await setChartFlag()
this.setState({ shouldShowHint: false })
}
onLayout = () => {
if (this.state.chartHeight) return false
this.reCalculateChartInfo()
this.updateListeners(this.reCalculateChartInfo)
}
prepareSymptomData = () => { prepareSymptomData = () => {
this.symptomRowSymptoms = config.symptoms.filter((symptomName) => { this.symptomRowSymptoms = SYMPTOMS.filter((symptomName) => {
return this.cycleDaysSortedByDate.some(cycleDay => { return this.cycleDaysSortedByDate.some(cycleDay => {
return cycleDay[symptomName] return (symptomName !== 'temperature') && cycleDay[symptomName]
}) })
}) })
this.chartSymptoms = [...this.symptomRowSymptoms] this.chartSymptoms = [...this.symptomRowSymptoms]
...@@ -69,35 +100,23 @@ class CycleChart extends Component { ...@@ -69,35 +100,23 @@ class CycleChart extends Component {
) )
} }
reCalculateChartInfo = (nativeEvent) => { reCalculateChartInfo = () => {
const { height, width } = nativeEvent.layout const { width, height } = Dimensions.get('window')
const xAxisCoefficient = this.shouldShowTemperatureColumn ?
config.xAxisHeightPercentage : config.xAxisHeightPercentageLarge
const symptomCoefficient = this.shouldShowTemperatureColumn ?
config.symptomHeightPercentage : config.symptomHeightPercentageLarge
this.xAxisHeight = height * xAxisCoefficient this.xAxisHeight = height * 0.7 * CHART_XAXIS_HEIGHT_RATIO
const remainingHeight = height - this.xAxisHeight const remainingHeight = height * 0.7 - this.xAxisHeight
this.symptomHeight = remainingHeight * symptomCoefficient this.symptomHeight = remainingHeight * CHART_SYMPTOM_HEIGHT_RATIO
this.symptomRowHeight = this.symptomRowSymptoms.length * this.symptomRowHeight = this.symptomRowSymptoms.length *
this.symptomHeight this.symptomHeight
this.columnHeight = remainingHeight - this.symptomRowHeight this.columnHeight = remainingHeight - this.symptomRowHeight
const chartHeight = this.shouldShowTemperatureColumn ? const chartHeight = this.shouldShowTemperatureColumn ?
height : (this.symptomRowHeight + this.xAxisHeight) height * 0.7 : (this.symptomRowHeight + this.xAxisHeight)
const numberOfColumnsToRender = Math.round(width / CHART_COLUMN_WIDTH)
const numberOfColumnsToRender = Math.round(width / config.columnWidth)
const columns = makeColumnInfo() const columns = makeColumnInfo()
this.setState({ columns, chartHeight, numberOfColumnsToRender }) this.setState({ columns, chartHeight, numberOfColumnsToRender })
} }
onLayout = ({ nativeEvent }) => {
if (this.state.chartHeight) return
this.reCalculateChartInfo(nativeEvent)
this.updateListeners(this.reCalculateChartInfo)
}
updateListeners(dataUpdateHandler) { updateListeners(dataUpdateHandler) {
// remove existing listeners // remove existing listeners
if(this.handleDbChange) { if(this.handleDbChange) {
...@@ -114,38 +133,47 @@ class CycleChart extends Component { ...@@ -114,38 +133,47 @@ class CycleChart extends Component {
this.removeObvListener = scaleObservable(dataUpdateHandler, false) this.removeObvListener = scaleObservable(dataUpdateHandler, false)
} }
componentWillUnmount() {
this.cycleDaysSortedByDate.removeListener(this.handleDbChange)
this.removeObvListener()
}
render() { render() {
const { chartHeight, chartLoaded, numberOfColumnsToRender } = this.state const {
const shouldShowChart = this.chartSymptoms.length > 0 ? true : false chartHeight,
chartLoaded,
shouldShowHint,
numberOfColumnsToRender
} = this.state
const hasDataToDisplay = this.chartSymptoms.length > 0
return ( return (
<View onLayout={this.onLayout} style={styles.container}> <AppPage
{!shouldShowChart && <NoData navigate={this.props.navigate}/>} contentContainerStyle={styles.pageContainer}
{shouldShowChart && !chartHeight && !chartLoaded && <AppLoadingView />} onLayout={this.onLayout}
scrollViewStyle={styles.page}
>
{!hasDataToDisplay && <NoData />}
{hasDataToDisplay && !chartHeight && !chartLoaded && <AppLoadingView />}
<View style={styles.chartContainer}> <View style={styles.chartContainer}>
{shouldShowChart && ( {shouldShowHint && chartLoaded &&
<Tutorial onClose={this.setShouldShowHint} />
}
{hasDataToDisplay && chartLoaded &&
!this.shouldShowTemperatureColumn &&
<View style={styles.centerItem}>
<AppText style={styles.warning}>
{shared.noTemperatureWarning}
</AppText>
</View>
}
{hasDataToDisplay && (
<View style={styles.chartArea}> <View style={styles.chartArea}>
{chartHeight && chartLoaded && ( {chartHeight && chartLoaded && (
<React.Fragment> <YAxis
<YAxis height={this.columnHeight}
height={this.columnHeight} symptomsToDisplay={this.symptomRowSymptoms}
symptomsToDisplay={this.symptomRowSymptoms} symptomsSectionHeight={this.symptomRowHeight}
symptomsSectionHeight={this.symptomRowHeight} shouldShowTemperatureColumn=
shouldShowTemperatureColumn= {this.shouldShowTemperatureColumn}
{this.shouldShowTemperatureColumn} xAxisHeight={this.xAxisHeight}
xAxisHeight={this.xAxisHeight} />
/>
{this.shouldShowTemperatureColumn && (<HorizontalGrid
height={this.columnHeight}
startPosition={this.symptomRowHeight}
/>)}
</React.Fragment>
)} )}
{chartHeight && {chartHeight &&
...@@ -162,27 +190,28 @@ class CycleChart extends Component { ...@@ -162,27 +190,28 @@ class CycleChart extends Component {
onEndReached={() => this.setState({end: true})} onEndReached={() => this.setState({end: true})}
ListFooterComponent={<LoadingMoreView end={this.state.end}/>} ListFooterComponent={<LoadingMoreView end={this.state.end}/>}
updateCellsBatchingPeriod={800} updateCellsBatchingPeriod={800}
contentContainerStyle={{height: chartHeight}} contentContainerStyle={{ height: chartHeight }}
/> />
} }
{chartHeight && chartLoaded && (
<React.Fragment>
{this.shouldShowTemperatureColumn &&
<HorizontalGrid height={this.columnHeight} />
}
</React.Fragment>
)}
</View> </View>
)} )}
</View> </View>
{shouldShowChart && chartLoaded && !this.shouldShowTemperatureColumn </AppPage>
&& (
<View style={styles.centerItem}>
<AppText style={{textAlign: 'center'}}>{shared.noTemperatureWarning}</AppText>
</View>
)}
</View>
) )
} }
} }
function LoadingMoreView({ end }) { function LoadingMoreView({ end }) {
return ( return (
<View style={styles.loadingMore}> <View style={styles.loadingContainer}>
{!end && <ActivityIndicator size={'large'} color={'white'}/>} {!end && <ActivityIndicator size={'large'} color={Colors.orange}/>}
</View> </View>
) )
} }
...@@ -191,6 +220,29 @@ LoadingMoreView.propTypes = { ...@@ -191,6 +220,29 @@ LoadingMoreView.propTypes = {
end: PropTypes.bool end: PropTypes.bool
} }
const styles = StyleSheet.create({
chartArea: {
flexDirection: 'row'
},
chartContainer: {
flexDirection: 'column'
},
loadingContainer: {
height: '100%',
backgroundColor: Colors.turquoiseLight,
justifyContent: 'center'
},
page: {
marginVertical: Spacing.small
},
pageContainer: {
paddingHorizontal: Spacing.base
},
warning: {
padding: Spacing.large
}
})
const mapDispatchToProps = (dispatch) => { const mapDispatchToProps = (dispatch) => {
return({ return({
navigate: (page) => dispatch(navigate(page)), navigate: (page) => dispatch(navigate(page)),
......
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { StyleSheet, View } from 'react-native'
import { Text, View } from 'react-native'
import moment from 'moment' import moment from 'moment'
import { LocalDate } from 'js-joda'
import styles from './styles' import AppText from '../common/app-text'
import cycleModule from '../../lib/cycle' import cycleModule from '../../lib/cycle'
import { getOrdinalSuffix } from '../helpers/home'
import { Containers, Typography, Sizes } from '../../styles'
const CycleDayLabel = ({ height, date }) => { const CycleDayLabel = ({ height, date }) => {
const { label } = styles.column
const dayDate = LocalDate.parse(date)
const cycleDayNumber = cycleModule().getCycleDayNumber(date) const cycleDayNumber = cycleModule().getCycleDayNumber(date)
const cycleDayLabel = cycleDayNumber ? cycleDayNumber : ' '
const isFirstDayOfMonth = dayDate.dayOfMonth() === 1 const momentDate = moment(date)
const dateFormatting = isFirstDayOfMonth ? 'MMM' : 'Do' const dayOfMonth = momentDate.date()
const shortDate = moment(date, "YYYY-MM-DD").format(dateFormatting) const isFirstDayOfMonth = dayOfMonth === 1
const boldDateLabel = isFirstDayOfMonth ? {fontWeight: 'bold'} : {}
return ( return (
<View style={[styles.chartLegend, { height }]}> <View style={[styles.container, { height }]}>
<Text style={label.number}> <AppText style={styles.textBold}>{cycleDayLabel}</AppText>
{cycleDayNumber ? cycleDayNumber : ' '} <View style={styles.dateLabel}>
</Text> <AppText style={styles.text}>
<Text style={[label.date, boldDateLabel]}> {isFirstDayOfMonth ? momentDate.format('MMM') : dayOfMonth}
{shortDate} </AppText>
</Text> {!isFirstDayOfMonth &&
<AppText style={styles.textLight}>
{getOrdinalSuffix(dayOfMonth)}
</AppText>
}
</View>
</View> </View>
) )
} }
...@@ -36,4 +39,30 @@ CycleDayLabel.propTypes = { ...@@ -36,4 +39,30 @@ CycleDayLabel.propTypes = {
date: PropTypes.string, date: PropTypes.string,
} }
const styles = StyleSheet.create({
container: {
alignItems: 'flex-start',
justifyContent: 'flex-end',
left: 4,
},
containerRow: {
...Containers.rowContainer
},
text: {
...Typography.label,
fontSize: Sizes.small,
},
textBold: {
...Typography.labelBold
},
textLight: {
...Typography.labelLight,
},
dateLabel: {
flexDirection: 'row',
justifyContent: 'space-around',
alignItems: 'center',
}
})
export default CycleDayLabel export default CycleDayLabel
...@@ -92,10 +92,23 @@ class DayColumn extends Component { ...@@ -92,10 +92,23 @@ class DayColumn extends Component {
activeOpacity={1} activeOpacity={1}
> >
{ symptomRowSymptoms.map(symptom => { {shouldShowTemperatureColumn && <TemperatureColumn
horizontalLinePosition={this.fhmAndLtl.drawLtlAt}
isVerticalLine={this.fhmAndLtl.drawFhmLine}
data={this.data && this.data.temperature}
columnHeight={columnHeight}
/>}
<CycleDayLabel
height={xAxisHeight}
date={dateString}
/>
{ symptomRowSymptoms.map((symptom, i) => {
const hasSymptomData = this.data.hasOwnProperty(symptom) const hasSymptomData = this.data.hasOwnProperty(symptom)
return ( return (
<SymptomCell <SymptomCell
index={i}
key={symptom} key={symptom}
symptom={symptom} symptom={symptom}
symptomValue={hasSymptomData && this.data[symptom]} symptomValue={hasSymptomData && this.data[symptom]}
...@@ -107,18 +120,6 @@ class DayColumn extends Component { ...@@ -107,18 +120,6 @@ class DayColumn extends Component {
} }
)} )}
{shouldShowTemperatureColumn && <TemperatureColumn
horizontalLinePosition={this.fhmAndLtl.drawLtlAt}
isVerticalLine={this.fhmAndLtl.drawFhmLine}
data={this.data && this.data.temperature}
columnHeight={columnHeight}
/>}
<CycleDayLabel
height={xAxisHeight}
date={dateString}
/>
</TouchableOpacity> </TouchableOpacity>
) )
} }
......
...@@ -2,8 +2,14 @@ import React, { Component } from 'react' ...@@ -2,8 +2,14 @@ import React, { Component } from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { Path, Shape } from 'react-native/Libraries/ART/ReactNativeART' import { Path, Shape } from 'react-native/Libraries/ART/ReactNativeART'
import styles from './styles' import { Colors } from '../../styles'
import config from '../../config'
import {
CHART_COLUMN_WIDTH,
CHART_COLUMN_MIDDLE,
CHART_DOT_RADIUS,
CHART_STROKE_WIDTH
} from '../../config'
export default class DotAndLine extends Component { export default class DotAndLine extends Component {
static propTypes = { static propTypes = {
...@@ -20,48 +26,62 @@ export default class DotAndLine extends Component { ...@@ -20,48 +26,62 @@ export default class DotAndLine extends Component {
} }
render() { render() {
const y = this.props.y const {
const exclude = this.props.exclude exclude,
let lineToRight leftTemperatureExclude,
let lineToLeft leftY,
rightTemperatureExclude,
rightY,
y
} = this.props
let excludeLeftLine, excludeRightLine, lineLeft, lineRight
if (this.props.leftY) { if (leftY) {
const middleY = ((this.props.leftY - y) / 2) + y const middleY = ((leftY - y) / 2) + y
const excludedLine = this.props.leftTemperatureExclude || exclude excludeLeftLine = leftTemperatureExclude || exclude
lineToLeft = makeLine(y, middleY, 0, excludedLine) lineLeft = new Path()
.moveTo(CHART_COLUMN_MIDDLE, y)
.lineTo(0, middleY)
} }
if (this.props.rightY) { if (rightY) {
const middleY = ((y - this.props.rightY) / 2) + this.props.rightY const middleY = ((y - rightY) / 2) + rightY
const excludedLine = this.props.rightTemperatureExclude || exclude excludeRightLine = rightTemperatureExclude || exclude
lineToRight = makeLine(y, middleY, config.columnWidth, excludedLine) lineRight = new Path()
.moveTo(CHART_COLUMN_MIDDLE, y)
.lineTo(CHART_COLUMN_WIDTH, middleY)
} }
const dotStyle = exclude ? styles.curveDotsExcluded : styles.curveDots const dot = new Path().moveTo(CHART_COLUMN_MIDDLE , y - CHART_DOT_RADIUS)
const radius = dotStyle.r .arc(0, CHART_DOT_RADIUS * 2, CHART_DOT_RADIUS)
const dot = ( .arc(0, CHART_DOT_RADIUS * -2, CHART_DOT_RADIUS)
<Shape const dotColor = exclude ? Colors.turquoise : Colors.turquoiseDark
d={new Path() const lineColorLeft = excludeLeftLine ?
.moveTo(config.columnMiddle, y - radius) Colors.turquoise : Colors.turquoiseDark
.arc(0, radius * 2, radius) const lineColorRight = excludeRightLine ?
.arc(0, radius * -2, radius) Colors.turquoise : Colors.turquoiseDark
}
fill={dotStyle.fill} return(
key='dot' <React.Fragment>
/> <Shape
d={lineLeft}
stroke={lineColorLeft}
strokeWidth={CHART_STROKE_WIDTH}
key={y}
/>
<Shape
d={lineRight}
stroke={lineColorRight}
strokeWidth={CHART_STROKE_WIDTH}
key={y + CHART_DOT_RADIUS}
/>
<Shape
d={dot}
stroke={dotColor}
strokeWidth={CHART_STROKE_WIDTH}
fill="white"
key='dot'
/>
</React.Fragment>
) )
return [lineToLeft, lineToRight, dot]
} }
} }
function makeLine(currY, middleY, x, excludeLine) {
const lineStyle = excludeLine ? styles.curveExcluded : styles.curve
return <Shape
stroke={lineStyle.stroke}
d={new Path()
.moveTo(config.columnMiddle, currY)
.lineTo(x, middleY)
}
key={x.toString()}
/>
}
\ No newline at end of file
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View } from 'react-native' import { StyleSheet, View } from 'react-native'
import { getTickPositions } from '../helpers/chart' import { getTickPositions } from '../helpers/chart'
import styles from './styles' import { Colors } from '../../styles'
import { CHART_GRID_LINE_HORIZONTAL_WIDTH, CHART_YAXIS_WIDTH } from '../../config'
const HorizontalGrid = ({ height, startPosition }) => { const HorizontalGrid = ({ height }) => {
return getTickPositions(height).map(tick => { return getTickPositions(height).map(tick => {
return ( return (
<View <View key={tick} top={tick} {...styles.line}/>
top={startPosition + tick}
{...styles.horizontalGrid}
key={tick}
/>
) )
}) })
} }
HorizontalGrid.propTypes = { HorizontalGrid.propTypes = {
height: PropTypes.number, height: PropTypes.number
startPosition: PropTypes.number,
} }
const styles = StyleSheet.create({
line: {
borderStyle: 'solid',
borderBottomColor: Colors.grey,
borderBottomWidth: CHART_GRID_LINE_HORIZONTAL_WIDTH,
left: CHART_YAXIS_WIDTH,
position:'absolute',
right: 0
}
})
export default HorizontalGrid export default HorizontalGrid
import { getCycleStatusForDay } from '../../lib/sympto-adapter'
import { normalizeToScale } from '../helpers/chart'
export default function () {
const cycle = {
status: null
}
function updateCurrentCycle(dateString) {
// for the NFP lines, we don't care about potentially extending the
// preOvu phase, so we don't include all earlier cycles, as that is
// an expensive db operation at the moment
cycle.status = getCycleStatusForDay(
dateString, { excludeEarlierCycles: true }
)
if(!cycle.status) {
cycle.noMoreCycles = true
return
}
if (cycle.status.phases.preOvulatory) {
cycle.startDate = cycle.status.phases.preOvulatory.start.date
} else {
cycle.startDate = cycle.status.phases.periOvulatory.start.date
}
}
function dateIsInPeriOrPostPhase(dateString) {
return (
dateString >= cycle.status.phases.periOvulatory.start.date
)
}
function precededByAnotherTempValue(dateString) {
return (
// we are only interested in days that have a preceding
// temp
Object.keys(cycle.status.phases).some(phaseName => {
return cycle.status.phases[phaseName].cycleDays.some(day => {
return day.temperature && day.date < dateString
})
})
// and also a following temp, so we don't draw the line
// longer than necessary
&&
cycle.status.phases.postOvulatory.cycleDays.some(day => {
return day.temperature && day.date > dateString
})
)
}
function isInTempMeasuringPhase(temperature, dateString) {
return (
temperature || precededByAnotherTempValue(dateString)
)
}
return function(dateString, temperature, columnHeight) {
const ret = {
drawLtlAt: null,
drawFhmLine: false
}
if (!cycle.status && !cycle.noMoreCycles) updateCurrentCycle(dateString)
if (cycle.noMoreCycles) return ret
if (dateString < cycle.startDate) updateCurrentCycle(dateString)
if (cycle.noMoreCycles) return ret
const tempShift = cycle.status.temperatureShift
if (tempShift) {
if (tempShift.firstHighMeasurementDay.date === dateString) {
ret.drawFhmLine = true
}
if (
dateIsInPeriOrPostPhase(dateString) &&
isInTempMeasuringPhase(temperature, dateString)
) {
ret.drawLtlAt = normalizeToScale(tempShift.ltl, columnHeight)
}
}
return ret
}
}
\ No newline at end of file
import React from 'react' import React from 'react'
import { StyleSheet, View } from 'react-native'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View } from 'react-native'
import AppText from '../app-text' import AppText from '../common/app-text'
import SettingsButton from '../settings/shared/settings-button' import Button from '../common/button'
import { connect } from 'react-redux'
import { navigate } from '../../slices/navigation'
import { Containers } from '../../styles'
import { shared } from '../../i18n/en/labels' import { shared } from '../../i18n/en/labels'
import styles from './styles'
const NoData = ({ navigate }) => { const NoData = ({ navigate }) => {
return ( return (
<View flex={1}> <View style={styles.container}>
<View style={styles.centerItem}> <AppText>{shared.noDataWarning}</AppText>
<AppText>{shared.noDataWarning}</AppText> <Button isCTA onPress={() => {navigate('CycleDay')}}>
<SettingsButton {shared.noDataButtonText}
onPress={() => {navigate('CycleDay')}} </Button>
style={{marginHorizontal: 40}}
>
{shared.noDataButtonText}
</SettingsButton>
</View>
</View> </View>
) )
} }
NoData.propTypes = { NoData.propTypes = {
navigate: PropTypes.func, navigate: PropTypes.func.isRequired,
}
const styles = StyleSheet.create({
container: {
...Containers.centerItems
}
})
const mapDispatchToProps = (dispatch) => {
return({
navigate: (page) => dispatch(navigate(page)),
})
} }
export default NoData export default connect(
\ No newline at end of file null,
mapDispatchToProps,
)(NoData)
\ No newline at end of file
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View } from 'react-native' import { StyleSheet, View } from 'react-native'
import styles from './styles' import { Colors, Containers } from '../../styles'
import config from '../../config' import {
CHART_COLUMN_WIDTH,
CHART_DOT_RADIUS,
CHART_GRID_LINE_HORIZONTAL_WIDTH
} from '../../config'
const SymptomCell = ({ const SymptomCell = ({
height, height,
index,
symptom, symptom,
symptomValue, symptomValue,
isSymptomDataComplete isSymptomDataComplete
}) => { }) => {
const shouldDrawDot = symptomValue !== false const shouldDrawDot = symptomValue !== false
const styleParent = [styles.symptomRow, { height, width: config.columnWidth }] const styleCell = index !== 0
let styleChild ? [styles.cell, { height, width: CHART_COLUMN_WIDTH }]
: [styles.cell, { height, width: CHART_COLUMN_WIDTH }, styles.topBorder]
let styleDot
if (shouldDrawDot) { if (shouldDrawDot) {
const styleSymptom = styles.iconColors[symptom] const styleSymptom = Colors.iconColors[symptom]
const symptomColor = styleSymptom.shades[symptomValue] const symptomColor = styleSymptom.shades[symptomValue]
const isMucusOrCervix = (symptom === 'mucus') || (symptom === 'cervix') const isMucusOrCervix = (symptom === 'mucus') || (symptom === 'cervix')
const backgroundColor = (isMucusOrCervix && !isSymptomDataComplete) ? const backgroundColor = (isMucusOrCervix && !isSymptomDataComplete) ?
'white' : symptomColor 'white' : symptomColor
const borderWidth = (isMucusOrCervix && !isSymptomDataComplete) ? 2 : 0 const borderWidth = (isMucusOrCervix && !isSymptomDataComplete) ? 2 : 0
const borderColor = symptomColor const borderColor = symptomColor
styleChild = [styles.symptomDot, { styleDot = [styles.dot, { backgroundColor, borderColor, borderWidth }]
backgroundColor,
borderColor,
borderWidth
}]
} }
return ( return (
<View style={styleParent} key={symptom}> <View style={styleCell} key={symptom}>
{shouldDrawDot && <View style={styleChild} />} {shouldDrawDot && <View style={styleDot} />}
</View> </View>
) )
} }
SymptomCell.propTypes = { SymptomCell.propTypes = {
height: PropTypes.number, height: PropTypes.number,
index: PropTypes.number.isRequired,
symptom: PropTypes.string, symptom: PropTypes.string,
symptomValue: PropTypes.oneOfType([ symptomValue: PropTypes.oneOfType([
PropTypes.bool, PropTypes.bool,
...@@ -50,4 +52,23 @@ SymptomCell.propTypes = { ...@@ -50,4 +52,23 @@ SymptomCell.propTypes = {
isSymptomDataComplete: PropTypes.bool, isSymptomDataComplete: PropTypes.bool,
} }
const styles = StyleSheet.create({
cell: {
backgroundColor: 'white',
borderBottomColor: Colors.grey,
borderBottomWidth: CHART_GRID_LINE_HORIZONTAL_WIDTH,
borderLeftColor: Colors.grey,
borderLeftWidth: CHART_GRID_LINE_HORIZONTAL_WIDTH,
...Containers.centerItems
},
topBorder: {
borderTopColor: Colors.grey,
borderTopWidth: CHART_GRID_LINE_HORIZONTAL_WIDTH,
},
dot: {
width: CHART_DOT_RADIUS * 2,
height: CHART_DOT_RADIUS * 2,
borderRadius: 50
}
})
export default SymptomCell export default SymptomCell
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View } from 'react-native' import { StyleSheet , View } from 'react-native'
import DripIcon from '../../assets/drip-icons' import DripIcon from '../../assets/drip-icons'
import styles from './styles' import { Colors, Containers } from '../../styles'
import { CHART_YAXIS_WIDTH, CHART_ICON_SIZE } from '../../config'
const SymptomIcon = ({ symptom, height }) => { const SymptomIcon = ({ symptom, height }) => {
return ( return (
<View style={styles.symptomIcon} width={styles.yAxis.width} height={height}> <View style={styles.container} width={CHART_YAXIS_WIDTH} height={height}>
<DripIcon <DripIcon
size={16} size={CHART_ICON_SIZE}
name={`drip-icon-${symptom}`} name={`drip-icon-${symptom}`}
color={styles.iconColors[symptom].color} color={Colors.iconColors[symptom].color}
/> />
</View> </View>
) )
...@@ -23,4 +24,10 @@ SymptomIcon.propTypes = { ...@@ -23,4 +24,10 @@ SymptomIcon.propTypes = {
symptom: PropTypes.string, symptom: PropTypes.string,
} }
const styles = StyleSheet.create({
container: {
...Containers.centerItems
}
})
export default SymptomIcon export default SymptomIcon
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { StyleSheet } from 'react-native'
import { Surface , Path } from 'react-native/Libraries/ART/ReactNativeART' import { Surface , Path } from 'react-native/Libraries/ART/ReactNativeART'
import ChartLine from './chart-line' import ChartLine from './chart-line'
import DotAndLine from './dot-and-line' import DotAndLine from './dot-and-line'
import styles from './styles' import { CHART_COLUMN_WIDTH, CHART_STROKE_WIDTH } from '../../config'
import config from '../../config'
const TemperatureColumn = ({ const TemperatureColumn = ({
horizontalLinePosition, horizontalLinePosition,
...@@ -15,20 +15,21 @@ const TemperatureColumn = ({ ...@@ -15,20 +15,21 @@ const TemperatureColumn = ({
data, data,
columnHeight columnHeight
}) => { }) => {
const x = CHART_STROKE_WIDTH / 2
const x = styles.nfpLine.strokeWidth / 2
return ( return (
<Surface width={config.columnWidth} height={columnHeight}> <Surface
width={CHART_COLUMN_WIDTH}
height={columnHeight}
style={styles.container}
>
<ChartLine <ChartLine path={new Path().lineTo(0, columnHeight)}/>
path={new Path().lineTo(0, columnHeight)}
/>
{horizontalLinePosition && <ChartLine {horizontalLinePosition && <ChartLine
path={new Path() path={new Path()
.moveTo(0, horizontalLinePosition) .moveTo(0, horizontalLinePosition)
.lineTo(config.columnWidth, horizontalLinePosition) .lineTo(CHART_COLUMN_WIDTH, horizontalLinePosition)
} }
isNfpLine={true} isNfpLine={true}
key='ltl' key='ltl'
...@@ -40,7 +41,7 @@ const TemperatureColumn = ({ ...@@ -40,7 +41,7 @@ const TemperatureColumn = ({
key='fhm' key='fhm'
/>} />}
{data && data.y && <DotAndLine {data && typeof(data.y) !== 'undefined' && <DotAndLine
y={data.y} y={data.y}
exclude={data.temperatureExclude} exclude={data.temperatureExclude}
rightY={data.rightY} rightY={data.rightY}
...@@ -61,4 +62,10 @@ TemperatureColumn.propTypes = { ...@@ -61,4 +62,10 @@ TemperatureColumn.propTypes = {
columnHeight: PropTypes.number, columnHeight: PropTypes.number,
} }
const styles = StyleSheet.create({
container: {
backgroundColor: 'white'
}
})
export default TemperatureColumn export default TemperatureColumn
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View } from 'react-native' import { StyleSheet, View } from 'react-native'
import Tick from './tick' import Tick from './tick'
import { getTickList } from '../helpers/chart' import { getTickList } from '../helpers/chart'
import styles from './styles'
const TickList = ({ height }) => { const TickList = ({ height }) => {
return ( return (
<View style={[styles.yAxis, { height }]}>{ <View style={[styles.container, height]}>
getTickList(height) {
.map(({ label, position, isBold, shouldShowLabel}) => { getTickList(height)
return ( .map(({ isBold, label, position, shouldShowLabel, tickHeight}) => {
<Tick return (
key={label} <Tick
yPosition={position} height={tickHeight}
isBold={isBold} isBold={isBold}
shouldShowLabel={shouldShowLabel} key={label}
label={label} label={label}
/> shouldShowLabel={shouldShowLabel}
) yPosition={position}
}) />
}</View> )
})
}
</View>
) )
} }
...@@ -31,4 +33,10 @@ TickList.propTypes = { ...@@ -31,4 +33,10 @@ TickList.propTypes = {
height: PropTypes.number, height: PropTypes.number,
} }
const styles = StyleSheet.create({
container: {
flex: 1
}
})
export default TickList export default TickList
import React from 'react' import React from 'react'
import { StyleSheet, View } from 'react-native'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import AppText from '../app-text' import AppText from '../common/app-text'
import styles from './styles' import { Sizes } from '../../styles'
const Tick = ({ yPosition, isBold, shouldShowLabel, label }) => { const Tick = ({ yPosition, height, isBold, shouldShowLabel, label }) => {
// this eyeballing is sadly necessary because RN does not const top = yPosition - height / 2
// support percentage values for transforms, which we'd need const containerStyle = [ styles.container, { flexBasis: height, height, top }]
// to reliably place the label vertically centered to the grid const textStyle = isBold ? styles.textBold : styles.textNormal
const topPosition = yPosition - 8
const style = [
styles.yAxisLabels.tempScale,
{top: topPosition},
isBold && styles.boldTick
]
return <AppText style={style}>{shouldShowLabel && label}</AppText> return(
<View style={containerStyle}>
<AppText style={textStyle}>{shouldShowLabel && label}</AppText>
</View>
)
} }
Tick.propTypes = { Tick.propTypes = {
yPosition: PropTypes.number, yPosition: PropTypes.number,
height: PropTypes.number.isRequired,
isBold: PropTypes.bool, isBold: PropTypes.bool,
shouldShowLabel: PropTypes.bool, shouldShowLabel: PropTypes.bool,
label: PropTypes.string, label: PropTypes.string,
} }
const text = {
lineHeight: Sizes.base,
right: 4,
textAlign: 'right'
}
const styles = StyleSheet.create({
container: {
justifyContent: 'center',
position: 'absolute',
right: 0,
width: 40
},
textBold: {
fontSize: Sizes.base,
fontWeight: 'bold',
...text
},
textNormal: {
fontSize: Sizes.small,
...text
}
})
export default Tick export default Tick
import React from 'react'
import PropTypes from 'prop-types'
import { Image, StyleSheet, View } from 'react-native'
import AppText from '../common/app-text'
import CloseIcon from '../common/close-icon'
import { Containers, Spacing } from '../../styles'
import { chart } from '../../i18n/en/labels'
const image = require('../../assets/swipe.png')
const Tutorial = ({ onClose }) => {
return (
<View style={styles.container}>
<Image resizeMode='contain' source={image} style={styles.image} />
<View style={styles.textContainer}>
<AppText>{chart.tutorial}</AppText>
</View>
<CloseIcon onClose={onClose} />
</View>
)
}
Tutorial.propTypes = {
onClose: PropTypes.func.isRequired
}
const styles = StyleSheet.create({
container: {
...Containers.rowContainer,
padding: Spacing.large
},
image: {
height: 40
},
textContainer: {
width: '70%'
}
})
export default Tutorial
\ No newline at end of file
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { View } from 'react-native' import { StyleSheet, View } from 'react-native'
import SymptomIcon from './symptom-icon' import SymptomIcon from './symptom-icon'
import TickList from './tick-list' import TickList from './tick-list'
import ChartLegend from './chart-legend' import ChartLegend from './chart-legend'
import styles from './styles' import { CHART_YAXIS_WIDTH } from '../../config'
const YAxis = ({ const YAxis = ({
height, height,
...@@ -19,6 +19,8 @@ const YAxis = ({ ...@@ -19,6 +19,8 @@ const YAxis = ({
return ( return (
<View> <View>
{shouldShowTemperatureColumn && <TickList height={height} />}
<ChartLegend height={xAxisHeight} />
<View style={[styles.yAxis, {height: symptomsSectionHeight}]}> <View style={[styles.yAxis, {height: symptomsSectionHeight}]}>
{symptomsToDisplay.map(symptom => ( {symptomsToDisplay.map(symptom => (
<SymptomIcon <SymptomIcon
...@@ -29,8 +31,6 @@ const YAxis = ({ ...@@ -29,8 +31,6 @@ const YAxis = ({
) )
)} )}
</View> </View>
{shouldShowTemperatureColumn && <TickList height={height} />}
<ChartLegend xAxisHeight={xAxisHeight} />
</View> </View>
) )
} }
...@@ -43,4 +43,10 @@ YAxis.propTypes = { ...@@ -43,4 +43,10 @@ YAxis.propTypes = {
xAxisHeight: PropTypes.number.isRequired xAxisHeight: PropTypes.number.isRequired
} }
const styles = StyleSheet.create({
yAxis: {
width: CHART_YAXIS_WIDTH
}
})
export default YAxis export default YAxis
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment