diff --git a/package.cordovabuild.json b/package.cordovabuild.json index 7e1c05d56..f2ad5fc16 100644 --- a/package.cordovabuild.json +++ b/package.cordovabuild.json @@ -98,8 +98,7 @@ }, "cordova-plugin-bluetooth-classic-serial-port": {}, "cordova-custom-config": {}, - "com.unarin.cordova.beacon": {}, - "cordova-plugin-statusbar": {} + "com.unarin.cordova.beacon": {} } }, "dependencies": { @@ -137,7 +136,6 @@ "cordova-plugin-bluetooth-classic-serial-port": "git+https://github.com/e-mission/cordova-plugin-bluetooth-classic-serial-port.git", "cordova-custom-config": "^5.1.1", "com.unarin.cordova.beacon": "github:e-mission/cordova-plugin-ibeacon", - "cordova-plugin-statusbar": "^4.0.0", "core-js": "^2.5.7", "e-mission-common": "github:JGreenlee/e-mission-common#semver:0.6.1", "enketo-core": "^6.1.7", diff --git a/setup/setup_native.sh b/setup/setup_native.sh index a7c396ab2..05624a693 100644 --- a/setup/setup_native.sh +++ b/setup/setup_native.sh @@ -121,7 +121,7 @@ sed -i -e "s|/usr/bin/env node|/usr/bin/env node --unhandled-rejections=strict|" npx cordova prepare$PLATFORMS -EXPECTED_COUNT=26 +EXPECTED_COUNT=25 INSTALLED_COUNT=`npx cordova plugin list | wc -l` echo "Found $INSTALLED_COUNT plugins, expected $EXPECTED_COUNT" if [ $INSTALLED_COUNT -lt $EXPECTED_COUNT ]; diff --git a/www/__tests__/util.ts b/www/__tests__/datetimeUtil.test.ts similarity index 94% rename from www/__tests__/util.ts rename to www/__tests__/datetimeUtil.test.ts index 0a8678a34..6d3fab6f2 100644 --- a/www/__tests__/util.ts +++ b/www/__tests__/datetimeUtil.test.ts @@ -1,4 +1,4 @@ -import { formatForDisplay } from '../js/util'; +import { formatForDisplay } from '../js/datetimeUtil'; describe('util.ts', () => { describe('formatForDisplay', () => { diff --git a/www/index.js b/www/index.js index 4ea9e93ac..3b7c86a0b 100644 --- a/www/index.js +++ b/www/index.js @@ -24,7 +24,7 @@ window.skipLocalNotificationReady = true; deviceReady.then(() => { logDebug('deviceReady'); // On init, use 'default' status bar (black text) - window['StatusBar']?.styleDefault(); + // window['StatusBar']?.styleDefault(); cordova.plugin.http.setDataSerializer('json'); const rootEl = document.getElementById('appRoot'); const reactRoot = createRoot(rootEl); diff --git a/www/js/TimelineContext.ts b/www/js/TimelineContext.ts index 2a94d2b0b..9b53d2a27 100644 --- a/www/js/TimelineContext.ts +++ b/www/js/TimelineContext.ts @@ -22,7 +22,7 @@ import { getNotDeletedCandidates, mapInputsToTimelineEntries } from './survey/in import { EnketoUserInputEntry } from './survey/enketo/enketoHelper'; import { primarySectionForTrip } from './diary/diaryHelper'; import useAppStateChange from './useAppStateChange'; -import { isoDateRangeToTsRange, isoDateWithOffset } from './util'; +import { isoDateRangeToTsRange, isoDateWithOffset } from './datetimeUtil'; import { base_modes } from 'e-mission-common'; const TODAY_DATE = DateTime.now().toISODate(); diff --git a/www/js/components/Chart.tsx b/www/js/components/Chart.tsx index 8d7406b6c..97844f946 100644 --- a/www/js/components/Chart.tsx +++ b/www/js/components/Chart.tsx @@ -4,7 +4,8 @@ import { useTheme } from 'react-native-paper'; import { ChartData, Chart as ChartJS, ScriptableContext, registerables } from 'chart.js'; import { Chart as ChartJSChart } from 'react-chartjs-2'; import Annotation, { AnnotationOptions, LabelPosition } from 'chartjs-plugin-annotation'; -import { dedupColors, getChartHeight, darkenOrLighten } from './charting'; +import { getChartHeight } from './charting'; +import { base_modes } from 'e-mission-common'; import { logDebug } from '../plugin/logger'; ChartJS.register(...registerables, Annotation); @@ -64,7 +65,7 @@ const Chart = ({ let labelColorMap; // object mapping labels to colors if (getColorForLabel) { const colorEntries = chartDatasets.map((d) => [d.label, getColorForLabel(d.label)]); - labelColorMap = dedupColors(colorEntries); + labelColorMap = base_modes.dedupe_colors(colorEntries, [0.4, 1.6]); } return { datasets: chartDatasets.map((e, i) => ({ @@ -73,7 +74,7 @@ const Chart = ({ labelColorMap?.[e.label] || getColorForChartEl?.(chartRef.current, e, barCtx, 'background'), borderColor: (barCtx) => - darkenOrLighten(labelColorMap?.[e.label], -0.5) || + base_modes.scale_lightness(labelColorMap?.[e.label], 0.5) || getColorForChartEl?.(chartRef.current, e, barCtx, 'border'), borderWidth: borderWidth || 2, borderRadius: 3, diff --git a/www/js/components/charting.ts b/www/js/components/charting.ts index 657a3b8ab..91379f23b 100644 --- a/www/js/components/charting.ts +++ b/www/js/components/charting.ts @@ -59,44 +59,44 @@ function getBarHeight(stacks) { return totalHeight; } -//fill pattern creation -//https://stackoverflow.com/questions/28569667/fill-chart-js-bar-chart-with-diagonal-stripes-or-other-patterns -function createDiagonalPattern(color = 'black') { - let shape = document.createElement('canvas'); - shape.width = 10; - shape.height = 10; - let c = shape.getContext('2d') as CanvasRenderingContext2D; - c.strokeStyle = color; - c.lineWidth = 2; - c.beginPath(); - c.moveTo(2, 0); - c.lineTo(10, 8); - c.stroke(); - c.beginPath(); - c.moveTo(0, 8); - c.lineTo(2, 10); - c.stroke(); - return c.createPattern(shape, 'repeat'); -} +// //fill pattern creation +// //https://stackoverflow.com/questions/28569667/fill-chart-js-bar-chart-with-diagonal-stripes-or-other-patterns +// function createDiagonalPattern(color = 'black') { +// let shape = document.createElement('canvas'); +// shape.width = 10; +// shape.height = 10; +// let c = shape.getContext('2d') as CanvasRenderingContext2D; +// c.strokeStyle = color; +// c.lineWidth = 2; +// c.beginPath(); +// c.moveTo(2, 0); +// c.lineTo(10, 8); +// c.stroke(); +// c.beginPath(); +// c.moveTo(0, 8); +// c.lineTo(2, 10); +// c.stroke(); +// return c.createPattern(shape, 'repeat'); +// } -export function getMeteredBackgroundColor(meter, currDataset, barCtx, colors, darken = 0) { - if (!barCtx || !currDataset) return; - let bar_height = getBarHeight(barCtx.parsed._stacks); - logDebug(`bar height for ${barCtx.raw.y} is ${bar_height} which in chart is ${currDataset}`); - let meteredColor; - if (bar_height > meter.high) meteredColor = colors.danger; - else if (bar_height > meter.middle) meteredColor = colors.warn; - else meteredColor = colors.success; - if (darken) { - return color(meteredColor).darken(darken).hex(); - } - //if "unlabeled", etc -> stripes - if (currDataset.label == meter.uncertainty_prefix) { - return createDiagonalPattern(meteredColor); - } - //if :labeled", etc -> solid - return meteredColor; -} +// export function getMeteredBackgroundColor(meter, currDataset, barCtx, colors, darken = 0) { +// if (!barCtx || !currDataset) return; +// let bar_height = getBarHeight(barCtx.parsed._stacks); +// logDebug(`bar height for ${barCtx.raw.y} is ${bar_height} which in chart is ${currDataset}`); +// let meteredColor; +// if (bar_height > meter.high) meteredColor = colors.danger; +// else if (bar_height > meter.middle) meteredColor = colors.warn; +// else meteredColor = colors.success; +// if (darken) { +// return color(meteredColor).darken(darken).hex(); +// } +// //if "unlabeled", etc -> stripes +// if (currDataset.label == meter.uncertainty_prefix) { +// return createDiagonalPattern(meteredColor); +// } +// //if :labeled", etc -> solid +// return meteredColor; +// } const meterColors = { below: '#00cc95', // green oklch(75% 0.3 165) @@ -142,42 +142,3 @@ export function getGradient( } return gradient; } - -/** - * @param baseColor a color string - * @param change a number between -1 and 1, indicating the amount to darken or lighten the color - * @returns an adjusted color, either darkened or lightened, depending on the sign of change - */ -export function darkenOrLighten(baseColor: string, change: number) { - if (!baseColor) return baseColor; - let colorObj = color(baseColor); - if (change < 0) { - // darkening appears more drastic than lightening, so we will be less aggressive (scale change by .5) - return colorObj.darken(Math.abs(change * 0.5)).hex(); - } else { - return colorObj.lighten(Math.abs(change)).hex(); - } -} - -/** - * @param colors an array of colors, each of which is an array of [key, color string] - * @returns an object mapping keys to colors, with duplicates darkened/lightened to be distinguishable - */ -export function dedupColors(colors: string[][]) { - const dedupedColors = {}; - const maxAdjustment = 0.7; // more than this is too drastic and the colors approach black/white - for (const [key, clr] of colors) { - if (!clr) continue; // skip empty colors - const duplicates = colors.filter(([k, c]) => c == clr); - if (duplicates.length > 1) { - // there are duplicates; calculate an evenly-spaced adjustment for each one - duplicates.forEach(([k, c], i) => { - const change = -maxAdjustment + ((maxAdjustment * 2) / (duplicates.length - 1)) * i; - dedupedColors[k] = darkenOrLighten(clr, change); - }); - } else if (!dedupedColors[key]) { - dedupedColors[key] = clr; // not a dupe, & not already deduped, so use the color as-is - } - } - return dedupedColors; -} diff --git a/www/js/config/useImperialConfig.ts b/www/js/config/useImperialConfig.ts index 900c9854e..ba32ccec2 100644 --- a/www/js/config/useImperialConfig.ts +++ b/www/js/config/useImperialConfig.ts @@ -1,6 +1,6 @@ import React, { useMemo } from 'react'; import useAppConfig from '../useAppConfig'; -import { formatForDisplay } from '../util'; +import { formatForDisplay } from '../datetimeUtil'; export type ImperialConfig = { distanceSuffix: string; diff --git a/www/js/util.ts b/www/js/datetimeUtil.ts similarity index 100% rename from www/js/util.ts rename to www/js/datetimeUtil.ts diff --git a/www/js/diary/diaryHelper.ts b/www/js/diary/diaryHelper.ts index e182835c4..b6510f288 100644 --- a/www/js/diary/diaryHelper.ts +++ b/www/js/diary/diaryHelper.ts @@ -4,7 +4,7 @@ import { LabelOptions } from '../types/labelTypes'; import { LocalDt } from '../types/serverData'; import { ImperialConfig } from '../config/useImperialConfig'; import { base_modes } from 'e-mission-common'; -import { humanizeIsoRange } from '../util'; +import { humanizeIsoRange } from '../datetimeUtil'; export type BaseModeKey = string; // TODO figure out how to get keyof typeof base_modes.BASE_MODES diff --git a/www/js/diary/list/DateSelect.tsx b/www/js/diary/list/DateSelect.tsx index 2f629b3d1..52cebe63e 100644 --- a/www/js/diary/list/DateSelect.tsx +++ b/www/js/diary/list/DateSelect.tsx @@ -19,7 +19,7 @@ import { Text, useTheme } from 'react-native-paper'; import i18next from 'i18next'; import { useTranslation } from 'react-i18next'; import { NavBarButton } from '../../components/NavBar'; -import { formatIsoNoYear, isoDateRangeToTsRange } from '../../util'; +import { formatIsoNoYear, isoDateRangeToTsRange } from '../../datetimeUtil'; // formats as e.g. 'Aug 1' const MONTH_DAY_SHORT: Intl.DateTimeFormatOptions = { diff --git a/www/js/diary/list/TimelineScrollList.tsx b/www/js/diary/list/TimelineScrollList.tsx index 98f12d139..e640b2b11 100644 --- a/www/js/diary/list/TimelineScrollList.tsx +++ b/www/js/diary/list/TimelineScrollList.tsx @@ -7,7 +7,7 @@ import { ActivityIndicator, Banner, Button, Icon, Text } from 'react-native-pape import LoadMoreButton from './LoadMoreButton'; import { useTranslation } from 'react-i18next'; import TimelineContext from '../../TimelineContext'; -import { isoDateRangeToTsRange, isoDateWithOffset } from '../../util'; +import { isoDateRangeToTsRange, isoDateWithOffset } from '../../datetimeUtil'; import { DateTime } from 'luxon'; function renderCard({ item: listEntry, index }) { diff --git a/www/js/diary/useDerivedProperties.tsx b/www/js/diary/useDerivedProperties.tsx index 4071000f7..3a46f7cca 100644 --- a/www/js/diary/useDerivedProperties.tsx +++ b/www/js/diary/useDerivedProperties.tsx @@ -7,7 +7,7 @@ import { primarySectionForTrip, } from './diaryHelper'; import TimelineContext from '../TimelineContext'; -import { formatIsoNoYear, formatIsoWeekday, humanizeIsoRange, isoDatesDifference } from '../util'; +import { formatIsoNoYear, formatIsoWeekday, humanizeIsoRange } from '../datetimeUtil'; const useDerivedProperties = (tlEntry) => { const imperialConfig = useImperialConfig(); @@ -18,7 +18,8 @@ const useDerivedProperties = (tlEntry) => { const endFmt = tlEntry.end_fmt_time || tlEntry.exit_fmt_time; const beginDt = tlEntry.start_local_dt || tlEntry.enter_local_dt; const endDt = tlEntry.end_local_dt || tlEntry.exit_local_dt; - const tlEntryIsMultiDay = isoDatesDifference(beginFmt, endFmt); + // given YYYY-MM-DDTHH:MM:SSZ strings: if YYYY-MM-DD differs, is multi-day + const tlEntryIsMultiDay = beginFmt.substring(0, 10) != endFmt.substring(0, 10); return { confirmedMode: confirmedModeFor(tlEntry), diff --git a/www/js/metrics/MetricsTab.tsx b/www/js/metrics/MetricsTab.tsx index d4fdddc35..65a0bb847 100644 --- a/www/js/metrics/MetricsTab.tsx +++ b/www/js/metrics/MetricsTab.tsx @@ -14,7 +14,7 @@ import { metrics_summaries } from 'e-mission-common'; import MetricsScreen from './MetricsScreen'; import { LabelOptions } from '../types/labelTypes'; import { useAppTheme } from '../appTheme'; -import { isoDatesDifference } from '../util'; +import { isoDatesDifference } from '../datetimeUtil'; const N_DAYS_TO_LOAD = 14; // 2 weeks export const DEFAULT_METRIC_LIST: MetricList = { diff --git a/www/js/metrics/SumaryCard.tsx b/www/js/metrics/SummaryCard.tsx similarity index 89% rename from www/js/metrics/SumaryCard.tsx rename to www/js/metrics/SummaryCard.tsx index 7f3170c5d..577f2ee97 100644 --- a/www/js/metrics/SumaryCard.tsx +++ b/www/js/metrics/SummaryCard.tsx @@ -1,7 +1,7 @@ import React from 'react'; import { View, StyleSheet } from 'react-native'; import { Card, Text } from 'react-native-paper'; -import { formatForDisplay } from '../util'; +import { formatForDisplay } from '../datetimeUtil'; import { colors } from '../appTheme'; import { t } from 'i18next'; import { FootprintGoal } from '../types/appConfigTypes'; @@ -20,9 +20,8 @@ const SummaryCard = ({ title, unit, value, nDays, goals }: Props) => { const perDayValue = value.map((v) => v / nDays) as Value; const formatVal = (v: Value) => { - const opts = { maximumFractionDigits: 1 }; - if (valueIsRange) return `${formatForDisplay(v[0], opts)} - ${formatForDisplay(v[1], opts)}`; - return `${formatForDisplay(v[0], opts)}`; + if (valueIsRange) return `${formatForDisplay(v[0])} - ${formatForDisplay(v[1])}`; + return `${formatForDisplay(v[0])}`; }; const colorFn = (v: Value) => { diff --git a/www/js/metrics/footprint/FootprintComparisonCard.tsx b/www/js/metrics/footprint/FootprintComparisonCard.tsx index 344a94164..0dfd4bc42 100644 --- a/www/js/metrics/footprint/FootprintComparisonCard.tsx +++ b/www/js/metrics/footprint/FootprintComparisonCard.tsx @@ -5,7 +5,7 @@ import BarChart from '../../components/BarChart'; import { useTranslation } from 'react-i18next'; import { ChartRecord } from '../../components/Chart'; import TimelineContext from '../../TimelineContext'; -import { formatIsoNoYear } from '../../util'; +import { formatIsoNoYear } from '../../datetimeUtil'; const FootprintComparisonCard = ({ type, diff --git a/www/js/metrics/footprint/FootprintSection.tsx b/www/js/metrics/footprint/FootprintSection.tsx index c871f9b07..ce97c99e5 100644 --- a/www/js/metrics/footprint/FootprintSection.tsx +++ b/www/js/metrics/footprint/FootprintSection.tsx @@ -2,11 +2,11 @@ import React, { useContext, useMemo, useState } from 'react'; import { View } from 'react-native'; import { Text } from 'react-native-paper'; import color from 'color'; -import SummaryCard from '../SumaryCard'; +import SummaryCard from '../SummaryCard'; import { useTranslation } from 'react-i18next'; import { sumMetricEntries } from '../metricsHelper'; import TimelineContext from '../../TimelineContext'; -import { formatIso, isoDatesDifference } from '../../util'; +import { formatIso, isoDatesDifference } from '../../datetimeUtil'; import WeeklyFootprintCard from './WeeklyFootprintCard'; import useAppConfig from '../../useAppConfig'; import { getFootprintGoals } from './footprintHelper'; diff --git a/www/js/metrics/footprint/WeeklyFootprintCard.tsx b/www/js/metrics/footprint/WeeklyFootprintCard.tsx index 5f4968f1a..d3005362d 100644 --- a/www/js/metrics/footprint/WeeklyFootprintCard.tsx +++ b/www/js/metrics/footprint/WeeklyFootprintCard.tsx @@ -10,7 +10,7 @@ import { sumMetricEntries, trimGroupingPrefix, } from '../metricsHelper'; -import { formatIsoNoYear, isoDateWithOffset } from '../../util'; +import { formatIsoNoYear, isoDateWithOffset } from '../../datetimeUtil'; import { useTranslation } from 'react-i18next'; import BarChart from '../../components/BarChart'; import { ChartRecord } from '../../components/Chart'; diff --git a/www/js/metrics/metricsHelper.ts b/www/js/metrics/metricsHelper.ts index f0cc458c5..fc759f320 100644 --- a/www/js/metrics/metricsHelper.ts +++ b/www/js/metrics/metricsHelper.ts @@ -6,7 +6,12 @@ import { MetricName, groupingFields } from '../types/appConfigTypes'; import { ImperialConfig } from '../config/useImperialConfig'; import i18next from 'i18next'; import { base_modes, metrics_summaries } from 'e-mission-common'; -import { formatForDisplay, formatIsoNoYear, isoDatesDifference, isoDateWithOffset } from '../util'; +import { + formatForDisplay, + formatIsoNoYear, + isoDatesDifference, + isoDateWithOffset, +} from '../datetimeUtil'; import { LabelOptions, RichMode } from '../types/labelTypes'; import { labelOptions, textToLabelKey } from '../survey/multilabel/confirmHelper'; import { UNCERTAIN_OPACITY } from '../components/charting'; diff --git a/www/js/metrics/movement/WeeklyActiveMinutesCard.tsx b/www/js/metrics/movement/WeeklyActiveMinutesCard.tsx index 3fcdade74..bce6db844 100644 --- a/www/js/metrics/movement/WeeklyActiveMinutesCard.tsx +++ b/www/js/metrics/movement/WeeklyActiveMinutesCard.tsx @@ -13,7 +13,7 @@ import { useTranslation } from 'react-i18next'; import BarChart from '../../components/BarChart'; import { labelKeyToText } from '../../survey/multilabel/confirmHelper'; import TimelineContext from '../../TimelineContext'; -import { formatIsoNoYear, isoDateWithOffset } from '../../util'; +import { formatIsoNoYear, isoDateWithOffset } from '../../datetimeUtil'; type Props = { userMetrics?: MetricsData; activeModes: string[] }; const WeeklyActiveMinutesCard = ({ userMetrics, activeModes }: Props) => { diff --git a/www/js/onboarding/OnboardingStack.tsx b/www/js/onboarding/OnboardingStack.tsx index 38def9cc3..c84f484ac 100644 --- a/www/js/onboarding/OnboardingStack.tsx +++ b/www/js/onboarding/OnboardingStack.tsx @@ -14,11 +14,11 @@ const OnboardingStack = () => { if (onboardingState.route == OnboardingRoute.WELCOME) { // This page needs 'light content' status bar (white text) due to blue header at the top - window['StatusBar']?.styleLightContent(); + // window['StatusBar']?.styleLightContent(); return ; } // All other pages go back to 'default' (black text) - window['StatusBar']?.styleDefault(); + // window['StatusBar']?.styleDefault(); if (onboardingState.route == OnboardingRoute.SUMMARY) { return ; } else if (onboardingState.route == OnboardingRoute.PROTOCOL) { diff --git a/www/js/survey/enketo/AddedNotesList.tsx b/www/js/survey/enketo/AddedNotesList.tsx index caf53e89c..49e883cf5 100644 --- a/www/js/survey/enketo/AddedNotesList.tsx +++ b/www/js/survey/enketo/AddedNotesList.tsx @@ -11,7 +11,7 @@ import EnketoModal from './EnketoModal'; import { useTranslation } from 'react-i18next'; import { EnketoUserInputEntry } from './enketoHelper'; import { displayErrorMsg, logDebug } from '../../plugin/logger'; -import { formatIsoNoYear, isoDatesDifference } from '../../util'; +import { formatIsoNoYear, isoDatesDifference } from '../../datetimeUtil'; type Props = { timelineEntry: any;