43 lines
1.8 KiB
JavaScript
43 lines
1.8 KiB
JavaScript
import * as joda from 'js-joda'
|
|
|
|
const LocalDate = joda.LocalDate
|
|
|
|
export default function config(bleedingDaysSortedByDateView, opts) {
|
|
opts = opts || {
|
|
// at the very minimum, a cycle can be a bleeding day
|
|
// followed by a non-bleeding day, thus a length of 2
|
|
maxBreakInBleeding: 1
|
|
}
|
|
|
|
return function getCycleDayNumber(targetDateString) {
|
|
// sort the cycle days in descending order so we travel into
|
|
// the past as we iterate over the array
|
|
// also, to retrieve the number, we only need the cycle days before the target day
|
|
const targetDate = LocalDate.parse(targetDateString)
|
|
const withWrappedDates = bleedingDaysSortedByDateView
|
|
.filter(day => !day.bleeding.exclude)
|
|
.map(day => {
|
|
day.wrappedDate = LocalDate.parse(day.date)
|
|
return day
|
|
})
|
|
// TODO write test for what if there is no first day before?? aka no firstbleedingdaybeforeindex
|
|
const firstBleedingDayBeforeTargetDayIndex = withWrappedDates.findIndex(day => day.wrappedDate.isBefore(targetDate))
|
|
const cycleDays = withWrappedDates.slice(firstBleedingDayBeforeTargetDayIndex)
|
|
|
|
const lastPeriodStart = cycleDays.find((day, i) => {
|
|
return thereIsNoPreviousBleedingDayWithinTheThreshold(day, cycleDays.slice(i + 1), opts.maxBreakInBleeding)
|
|
})
|
|
|
|
if (!lastPeriodStart) return null
|
|
|
|
const diffInDays = lastPeriodStart.wrappedDate.until(targetDate, joda.ChronoUnit.DAYS)
|
|
|
|
// cycle starts at day 1
|
|
return diffInDays + 1
|
|
}
|
|
}
|
|
|
|
function thereIsNoPreviousBleedingDayWithinTheThreshold(bleedingDay, earlierCycleDays, allowedBleedingBreak) {
|
|
const periodThreshold = bleedingDay.wrappedDate.minusDays(allowedBleedingBreak + 1)
|
|
return !earlierCycleDays.some(({ wrappedDate }) => wrappedDate.equals(periodThreshold) || wrappedDate.isAfter(periodThreshold))
|
|
} |