diff --git a/src/components/Interpretations/InterpretationsUnit/InterpretationForm.js b/src/components/Interpretations/InterpretationsUnit/InterpretationForm.js
index 2b7e9d02c..c6467a453 100644
--- a/src/components/Interpretations/InterpretationsUnit/InterpretationForm.js
+++ b/src/components/Interpretations/InterpretationsUnit/InterpretationForm.js
@@ -3,11 +3,8 @@ import i18n from '@dhis2/d2-i18n'
import { Button, Input } from '@dhis2/ui'
import PropTypes from 'prop-types'
import React, { useRef, useState } from 'react'
-import {
- RichTextEditor,
- MessageEditorContainer,
- MessageButtonStrip,
-} from '../common/index.js'
+import { RichTextEditor } from '../../RichText/index.js'
+import { MessageEditorContainer, MessageButtonStrip } from '../common/index.js'
export const InterpretationForm = ({
type,
@@ -46,7 +43,7 @@ export const InterpretationForm = ({
dataTest="interpretation-form"
>
{showRichTextEditor ? (
-
+ <>
-
+ >
) : (
setShowRichTextEditor(true)}
diff --git a/src/components/Interpretations/common/Interpretation/InterpretationUpdateForm.js b/src/components/Interpretations/common/Interpretation/InterpretationUpdateForm.js
index cf900fdf1..9891d7052 100644
--- a/src/components/Interpretations/common/Interpretation/InterpretationUpdateForm.js
+++ b/src/components/Interpretations/common/Interpretation/InterpretationUpdateForm.js
@@ -3,9 +3,9 @@ import i18n from '@dhis2/d2-i18n'
import { Button, spacers, colors } from '@dhis2/ui'
import PropTypes from 'prop-types'
import React, { useState } from 'react'
+import { RichTextEditor } from '../../../RichText/index.js'
import {
MessageEditorContainer,
- RichTextEditor,
MessageButtonStrip,
InterpretationSharingLink,
} from '../index.js'
diff --git a/src/components/Interpretations/common/Message/Message.js b/src/components/Interpretations/common/Message/Message.js
index 016e8b9d0..94bd37b5a 100644
--- a/src/components/Interpretations/common/Message/Message.js
+++ b/src/components/Interpretations/common/Message/Message.js
@@ -1,9 +1,9 @@
import { useTimeZoneConversion } from '@dhis2/app-runtime'
-import { Parser as RichTextParser } from '@dhis2/d2-ui-rich-text'
import { UserAvatar, spacers, colors } from '@dhis2/ui'
import moment from 'moment'
import PropTypes from 'prop-types'
import React from 'react'
+import { RichTextParser } from '../../../RichText/index.js'
const Message = ({ children, text, created, username }) => {
const { fromServerDate } = useTimeZoneConversion()
diff --git a/src/components/Interpretations/common/Message/MessageEditorContainer.js b/src/components/Interpretations/common/Message/MessageEditorContainer.js
index b4c7ff338..4de5bec5d 100644
--- a/src/components/Interpretations/common/Message/MessageEditorContainer.js
+++ b/src/components/Interpretations/common/Message/MessageEditorContainer.js
@@ -19,6 +19,7 @@ const MessageEditorContainer = ({ children, currentUser, dataTest }) => (
}
.editor {
flex-grow: 1;
+ height: 100%;
}
`}
diff --git a/src/components/Interpretations/common/RichTextEditor/index.js b/src/components/Interpretations/common/RichTextEditor/index.js
deleted file mode 100644
index 31c0113ca..000000000
--- a/src/components/Interpretations/common/RichTextEditor/index.js
+++ /dev/null
@@ -1 +0,0 @@
-export { RichTextEditor } from './RichTextEditor.js'
diff --git a/src/components/Interpretations/common/index.js b/src/components/Interpretations/common/index.js
index 562614fb1..d3473298f 100644
--- a/src/components/Interpretations/common/index.js
+++ b/src/components/Interpretations/common/index.js
@@ -1,4 +1,3 @@
export * from './Interpretation/index.js'
export * from './Message/index.js'
-export * from './RichTextEditor/index.js'
export * from './getInterpretationAccess.js'
diff --git a/src/components/OrgUnitDimension/OrgUnitDimension.js b/src/components/OrgUnitDimension/OrgUnitDimension.js
index 3516daf46..5d2d7115a 100644
--- a/src/components/OrgUnitDimension/OrgUnitDimension.js
+++ b/src/components/OrgUnitDimension/OrgUnitDimension.js
@@ -40,6 +40,7 @@ const OrgUnitDimension = ({
hideLevelSelect,
hideUserOrgUnits,
warning,
+ displayNameProp,
}) => {
const [ouLevels, setOuLevels] = useState([])
const [ouGroups, setOuGroups] = useState([])
@@ -79,13 +80,16 @@ const OrgUnitDimension = ({
setOuLevels(result)
}
const doFetchOuGroups = async () => {
- const result = await apiFetchOrganisationUnitGroups(dataEngine)
+ const result = await apiFetchOrganisationUnitGroups(
+ dataEngine,
+ displayNameProp
+ )
setOuGroups(result)
}
!hideLevelSelect && doFetchOuLevels()
!hideGroupSelect && doFetchOuGroups()
- }, [dataEngine, hideLevelSelect, hideGroupSelect])
+ }, [dataEngine, hideLevelSelect, hideGroupSelect, displayNameProp])
const onLevelChange = (ids) => {
const items = ids.map((id) => ({
@@ -369,6 +373,7 @@ OrgUnitDimension.defaultProps = {
}
OrgUnitDimension.propTypes = {
+ displayNameProp: PropTypes.string,
hideGroupSelect: PropTypes.bool,
hideLevelSelect: PropTypes.bool,
hideUserOrgUnits: PropTypes.bool,
diff --git a/src/components/PeriodDimension/PeriodTransfer.js b/src/components/PeriodDimension/PeriodTransfer.js
index a660e6498..cd4095484 100644
--- a/src/components/PeriodDimension/PeriodTransfer.js
+++ b/src/components/PeriodDimension/PeriodTransfer.js
@@ -156,15 +156,18 @@ const PeriodTransfer = ({
const onSelectFixedPeriods = (filter) => {
setFixedFilter(filter)
- setAllPeriods(
- getFixedPeriodsOptionsById(
- filter.periodType,
- periodsSettings
- ).getPeriods(
- fixedPeriodConfig(Number(filter.year)),
- periodsSettings
+
+ if (filter.year.match(/[0-9]{4}/)) {
+ setAllPeriods(
+ getFixedPeriodsOptionsById(
+ filter.periodType,
+ periodsSettings
+ ).getPeriods(
+ fixedPeriodConfig(Number(filter.year)),
+ periodsSettings
+ )
)
- )
+ }
}
const renderEmptySelection = () => (
diff --git a/src/components/PeriodDimension/__tests__/__snapshots__/fixedPeriods.spec.js.snap b/src/components/PeriodDimension/__tests__/__snapshots__/fixedPeriods.spec.js.snap
index be670cf0c..69aaa2343 100644
--- a/src/components/PeriodDimension/__tests__/__snapshots__/fixedPeriods.spec.js.snap
+++ b/src/components/PeriodDimension/__tests__/__snapshots__/fixedPeriods.spec.js.snap
@@ -6,185 +6,237 @@ Object {
"name": "Bi-weekly",
"options": Array [
Object {
+ "displayName": "Bi-Week 1 - 2013-12-30 - 2014-01-12",
"endDate": "2014-01-12",
"id": "2014BiW1",
"iso": "2014BiW1",
"name": "Bi-Week 1 - 2013-12-30 - 2014-01-12",
+ "periodType": "BIWEEKLY",
"startDate": "2013-12-30",
},
Object {
+ "displayName": "Bi-Week 2 - 2014-01-13 - 2014-01-26",
"endDate": "2014-01-26",
"id": "2014BiW2",
"iso": "2014BiW2",
"name": "Bi-Week 2 - 2014-01-13 - 2014-01-26",
+ "periodType": "BIWEEKLY",
"startDate": "2014-01-13",
},
Object {
+ "displayName": "Bi-Week 3 - 2014-01-27 - 2014-02-09",
"endDate": "2014-02-09",
"id": "2014BiW3",
"iso": "2014BiW3",
"name": "Bi-Week 3 - 2014-01-27 - 2014-02-09",
+ "periodType": "BIWEEKLY",
"startDate": "2014-01-27",
},
Object {
+ "displayName": "Bi-Week 4 - 2014-02-10 - 2014-02-23",
"endDate": "2014-02-23",
"id": "2014BiW4",
"iso": "2014BiW4",
"name": "Bi-Week 4 - 2014-02-10 - 2014-02-23",
+ "periodType": "BIWEEKLY",
"startDate": "2014-02-10",
},
Object {
+ "displayName": "Bi-Week 5 - 2014-02-24 - 2014-03-09",
"endDate": "2014-03-09",
"id": "2014BiW5",
"iso": "2014BiW5",
"name": "Bi-Week 5 - 2014-02-24 - 2014-03-09",
+ "periodType": "BIWEEKLY",
"startDate": "2014-02-24",
},
Object {
+ "displayName": "Bi-Week 6 - 2014-03-10 - 2014-03-23",
"endDate": "2014-03-23",
"id": "2014BiW6",
"iso": "2014BiW6",
"name": "Bi-Week 6 - 2014-03-10 - 2014-03-23",
+ "periodType": "BIWEEKLY",
"startDate": "2014-03-10",
},
Object {
+ "displayName": "Bi-Week 7 - 2014-03-24 - 2014-04-06",
"endDate": "2014-04-06",
"id": "2014BiW7",
"iso": "2014BiW7",
"name": "Bi-Week 7 - 2014-03-24 - 2014-04-06",
+ "periodType": "BIWEEKLY",
"startDate": "2014-03-24",
},
Object {
+ "displayName": "Bi-Week 8 - 2014-04-07 - 2014-04-20",
"endDate": "2014-04-20",
"id": "2014BiW8",
"iso": "2014BiW8",
"name": "Bi-Week 8 - 2014-04-07 - 2014-04-20",
+ "periodType": "BIWEEKLY",
"startDate": "2014-04-07",
},
Object {
+ "displayName": "Bi-Week 9 - 2014-04-21 - 2014-05-04",
"endDate": "2014-05-04",
"id": "2014BiW9",
"iso": "2014BiW9",
"name": "Bi-Week 9 - 2014-04-21 - 2014-05-04",
+ "periodType": "BIWEEKLY",
"startDate": "2014-04-21",
},
Object {
+ "displayName": "Bi-Week 10 - 2014-05-05 - 2014-05-18",
"endDate": "2014-05-18",
"id": "2014BiW10",
"iso": "2014BiW10",
"name": "Bi-Week 10 - 2014-05-05 - 2014-05-18",
+ "periodType": "BIWEEKLY",
"startDate": "2014-05-05",
},
Object {
+ "displayName": "Bi-Week 11 - 2014-05-19 - 2014-06-01",
"endDate": "2014-06-01",
"id": "2014BiW11",
"iso": "2014BiW11",
"name": "Bi-Week 11 - 2014-05-19 - 2014-06-01",
+ "periodType": "BIWEEKLY",
"startDate": "2014-05-19",
},
Object {
+ "displayName": "Bi-Week 12 - 2014-06-02 - 2014-06-15",
"endDate": "2014-06-15",
"id": "2014BiW12",
"iso": "2014BiW12",
"name": "Bi-Week 12 - 2014-06-02 - 2014-06-15",
+ "periodType": "BIWEEKLY",
"startDate": "2014-06-02",
},
Object {
+ "displayName": "Bi-Week 13 - 2014-06-16 - 2014-06-29",
"endDate": "2014-06-29",
"id": "2014BiW13",
"iso": "2014BiW13",
"name": "Bi-Week 13 - 2014-06-16 - 2014-06-29",
+ "periodType": "BIWEEKLY",
"startDate": "2014-06-16",
},
Object {
+ "displayName": "Bi-Week 14 - 2014-06-30 - 2014-07-13",
"endDate": "2014-07-13",
"id": "2014BiW14",
"iso": "2014BiW14",
"name": "Bi-Week 14 - 2014-06-30 - 2014-07-13",
+ "periodType": "BIWEEKLY",
"startDate": "2014-06-30",
},
Object {
+ "displayName": "Bi-Week 15 - 2014-07-14 - 2014-07-27",
"endDate": "2014-07-27",
"id": "2014BiW15",
"iso": "2014BiW15",
"name": "Bi-Week 15 - 2014-07-14 - 2014-07-27",
+ "periodType": "BIWEEKLY",
"startDate": "2014-07-14",
},
Object {
+ "displayName": "Bi-Week 16 - 2014-07-28 - 2014-08-10",
"endDate": "2014-08-10",
"id": "2014BiW16",
"iso": "2014BiW16",
"name": "Bi-Week 16 - 2014-07-28 - 2014-08-10",
+ "periodType": "BIWEEKLY",
"startDate": "2014-07-28",
},
Object {
+ "displayName": "Bi-Week 17 - 2014-08-11 - 2014-08-24",
"endDate": "2014-08-24",
"id": "2014BiW17",
"iso": "2014BiW17",
"name": "Bi-Week 17 - 2014-08-11 - 2014-08-24",
+ "periodType": "BIWEEKLY",
"startDate": "2014-08-11",
},
Object {
+ "displayName": "Bi-Week 18 - 2014-08-25 - 2014-09-07",
"endDate": "2014-09-07",
"id": "2014BiW18",
"iso": "2014BiW18",
"name": "Bi-Week 18 - 2014-08-25 - 2014-09-07",
+ "periodType": "BIWEEKLY",
"startDate": "2014-08-25",
},
Object {
+ "displayName": "Bi-Week 19 - 2014-09-08 - 2014-09-21",
"endDate": "2014-09-21",
"id": "2014BiW19",
"iso": "2014BiW19",
"name": "Bi-Week 19 - 2014-09-08 - 2014-09-21",
+ "periodType": "BIWEEKLY",
"startDate": "2014-09-08",
},
Object {
+ "displayName": "Bi-Week 20 - 2014-09-22 - 2014-10-05",
"endDate": "2014-10-05",
"id": "2014BiW20",
"iso": "2014BiW20",
"name": "Bi-Week 20 - 2014-09-22 - 2014-10-05",
+ "periodType": "BIWEEKLY",
"startDate": "2014-09-22",
},
Object {
+ "displayName": "Bi-Week 21 - 2014-10-06 - 2014-10-19",
"endDate": "2014-10-19",
"id": "2014BiW21",
"iso": "2014BiW21",
"name": "Bi-Week 21 - 2014-10-06 - 2014-10-19",
+ "periodType": "BIWEEKLY",
"startDate": "2014-10-06",
},
Object {
+ "displayName": "Bi-Week 22 - 2014-10-20 - 2014-11-02",
"endDate": "2014-11-02",
"id": "2014BiW22",
"iso": "2014BiW22",
"name": "Bi-Week 22 - 2014-10-20 - 2014-11-02",
+ "periodType": "BIWEEKLY",
"startDate": "2014-10-20",
},
Object {
+ "displayName": "Bi-Week 23 - 2014-11-03 - 2014-11-16",
"endDate": "2014-11-16",
"id": "2014BiW23",
"iso": "2014BiW23",
"name": "Bi-Week 23 - 2014-11-03 - 2014-11-16",
+ "periodType": "BIWEEKLY",
"startDate": "2014-11-03",
},
Object {
+ "displayName": "Bi-Week 24 - 2014-11-17 - 2014-11-30",
"endDate": "2014-11-30",
"id": "2014BiW24",
"iso": "2014BiW24",
"name": "Bi-Week 24 - 2014-11-17 - 2014-11-30",
+ "periodType": "BIWEEKLY",
"startDate": "2014-11-17",
},
Object {
+ "displayName": "Bi-Week 25 - 2014-12-01 - 2014-12-14",
"endDate": "2014-12-14",
"id": "2014BiW25",
"iso": "2014BiW25",
"name": "Bi-Week 25 - 2014-12-01 - 2014-12-14",
+ "periodType": "BIWEEKLY",
"startDate": "2014-12-01",
},
Object {
+ "displayName": "Bi-Week 26 - 2014-12-15 - 2014-12-28",
"endDate": "2014-12-28",
"id": "2014BiW26",
"iso": "2014BiW26",
"name": "Bi-Week 26 - 2014-12-15 - 2014-12-28",
+ "periodType": "BIWEEKLY",
"startDate": "2014-12-15",
},
],
@@ -198,45 +250,57 @@ Object {
"name": "Bi-monthly",
"options": Array [
Object {
+ "displayName": "January - February 2014",
"endDate": "2014-02-28",
"id": "201401B",
"iso": "201401B",
"name": "January - February 2014",
+ "periodType": "BIMONTHLY",
"startDate": "2014-01-01",
},
Object {
+ "displayName": "March - April 2014",
"endDate": "2014-04-30",
"id": "201402B",
"iso": "201402B",
"name": "March - April 2014",
+ "periodType": "BIMONTHLY",
"startDate": "2014-03-01",
},
Object {
+ "displayName": "May - June 2014",
"endDate": "2014-06-30",
"id": "201403B",
"iso": "201403B",
"name": "May - June 2014",
+ "periodType": "BIMONTHLY",
"startDate": "2014-05-01",
},
Object {
+ "displayName": "July - August 2014",
"endDate": "2014-08-31",
"id": "201404B",
"iso": "201404B",
"name": "July - August 2014",
+ "periodType": "BIMONTHLY",
"startDate": "2014-07-01",
},
Object {
+ "displayName": "September - October 2014",
"endDate": "2014-10-31",
"id": "201405B",
"iso": "201405B",
"name": "September - October 2014",
+ "periodType": "BIMONTHLY",
"startDate": "2014-09-01",
},
Object {
+ "displayName": "November - December 2014",
"endDate": "2014-12-31",
"id": "201406B",
"iso": "201406B",
"name": "November - December 2014",
+ "periodType": "BIMONTHLY",
"startDate": "2014-11-01",
},
],
@@ -250,2558 +314,3288 @@ Object {
"name": "Daily",
"options": Array [
Object {
+ "displayName": "January 1, 2014",
"endDate": "2014-01-01",
"id": "20140101",
"iso": "20140101",
"name": "2014-01-01",
+ "periodType": "DAILY",
"startDate": "2014-01-01",
},
Object {
+ "displayName": "January 2, 2014",
"endDate": "2014-01-02",
"id": "20140102",
"iso": "20140102",
"name": "2014-01-02",
+ "periodType": "DAILY",
"startDate": "2014-01-02",
},
Object {
+ "displayName": "January 3, 2014",
"endDate": "2014-01-03",
"id": "20140103",
"iso": "20140103",
"name": "2014-01-03",
+ "periodType": "DAILY",
"startDate": "2014-01-03",
},
Object {
+ "displayName": "January 4, 2014",
"endDate": "2014-01-04",
"id": "20140104",
"iso": "20140104",
"name": "2014-01-04",
+ "periodType": "DAILY",
"startDate": "2014-01-04",
},
Object {
+ "displayName": "January 5, 2014",
"endDate": "2014-01-05",
"id": "20140105",
"iso": "20140105",
"name": "2014-01-05",
+ "periodType": "DAILY",
"startDate": "2014-01-05",
},
Object {
+ "displayName": "January 6, 2014",
"endDate": "2014-01-06",
"id": "20140106",
"iso": "20140106",
"name": "2014-01-06",
+ "periodType": "DAILY",
"startDate": "2014-01-06",
},
Object {
+ "displayName": "January 7, 2014",
"endDate": "2014-01-07",
"id": "20140107",
"iso": "20140107",
"name": "2014-01-07",
+ "periodType": "DAILY",
"startDate": "2014-01-07",
},
Object {
+ "displayName": "January 8, 2014",
"endDate": "2014-01-08",
"id": "20140108",
"iso": "20140108",
"name": "2014-01-08",
+ "periodType": "DAILY",
"startDate": "2014-01-08",
},
Object {
+ "displayName": "January 9, 2014",
"endDate": "2014-01-09",
"id": "20140109",
"iso": "20140109",
"name": "2014-01-09",
+ "periodType": "DAILY",
"startDate": "2014-01-09",
},
Object {
+ "displayName": "January 10, 2014",
"endDate": "2014-01-10",
"id": "20140110",
"iso": "20140110",
"name": "2014-01-10",
+ "periodType": "DAILY",
"startDate": "2014-01-10",
},
Object {
+ "displayName": "January 11, 2014",
"endDate": "2014-01-11",
"id": "20140111",
"iso": "20140111",
"name": "2014-01-11",
+ "periodType": "DAILY",
"startDate": "2014-01-11",
},
Object {
+ "displayName": "January 12, 2014",
"endDate": "2014-01-12",
"id": "20140112",
"iso": "20140112",
"name": "2014-01-12",
+ "periodType": "DAILY",
"startDate": "2014-01-12",
},
Object {
+ "displayName": "January 13, 2014",
"endDate": "2014-01-13",
"id": "20140113",
"iso": "20140113",
"name": "2014-01-13",
+ "periodType": "DAILY",
"startDate": "2014-01-13",
},
Object {
+ "displayName": "January 14, 2014",
"endDate": "2014-01-14",
"id": "20140114",
"iso": "20140114",
"name": "2014-01-14",
+ "periodType": "DAILY",
"startDate": "2014-01-14",
},
Object {
+ "displayName": "January 15, 2014",
"endDate": "2014-01-15",
"id": "20140115",
"iso": "20140115",
"name": "2014-01-15",
+ "periodType": "DAILY",
"startDate": "2014-01-15",
},
Object {
+ "displayName": "January 16, 2014",
"endDate": "2014-01-16",
"id": "20140116",
"iso": "20140116",
"name": "2014-01-16",
+ "periodType": "DAILY",
"startDate": "2014-01-16",
},
Object {
+ "displayName": "January 17, 2014",
"endDate": "2014-01-17",
"id": "20140117",
"iso": "20140117",
"name": "2014-01-17",
+ "periodType": "DAILY",
"startDate": "2014-01-17",
},
Object {
+ "displayName": "January 18, 2014",
"endDate": "2014-01-18",
"id": "20140118",
"iso": "20140118",
"name": "2014-01-18",
+ "periodType": "DAILY",
"startDate": "2014-01-18",
},
Object {
+ "displayName": "January 19, 2014",
"endDate": "2014-01-19",
"id": "20140119",
"iso": "20140119",
"name": "2014-01-19",
+ "periodType": "DAILY",
"startDate": "2014-01-19",
},
Object {
+ "displayName": "January 20, 2014",
"endDate": "2014-01-20",
"id": "20140120",
"iso": "20140120",
"name": "2014-01-20",
+ "periodType": "DAILY",
"startDate": "2014-01-20",
},
Object {
+ "displayName": "January 21, 2014",
"endDate": "2014-01-21",
"id": "20140121",
"iso": "20140121",
"name": "2014-01-21",
+ "periodType": "DAILY",
"startDate": "2014-01-21",
},
Object {
+ "displayName": "January 22, 2014",
"endDate": "2014-01-22",
"id": "20140122",
"iso": "20140122",
"name": "2014-01-22",
+ "periodType": "DAILY",
"startDate": "2014-01-22",
},
Object {
+ "displayName": "January 23, 2014",
"endDate": "2014-01-23",
"id": "20140123",
"iso": "20140123",
"name": "2014-01-23",
+ "periodType": "DAILY",
"startDate": "2014-01-23",
},
Object {
+ "displayName": "January 24, 2014",
"endDate": "2014-01-24",
"id": "20140124",
"iso": "20140124",
"name": "2014-01-24",
+ "periodType": "DAILY",
"startDate": "2014-01-24",
},
Object {
+ "displayName": "January 25, 2014",
"endDate": "2014-01-25",
"id": "20140125",
"iso": "20140125",
"name": "2014-01-25",
+ "periodType": "DAILY",
"startDate": "2014-01-25",
},
Object {
+ "displayName": "January 26, 2014",
"endDate": "2014-01-26",
"id": "20140126",
"iso": "20140126",
"name": "2014-01-26",
+ "periodType": "DAILY",
"startDate": "2014-01-26",
},
Object {
+ "displayName": "January 27, 2014",
"endDate": "2014-01-27",
"id": "20140127",
"iso": "20140127",
"name": "2014-01-27",
+ "periodType": "DAILY",
"startDate": "2014-01-27",
},
Object {
+ "displayName": "January 28, 2014",
"endDate": "2014-01-28",
"id": "20140128",
"iso": "20140128",
"name": "2014-01-28",
+ "periodType": "DAILY",
"startDate": "2014-01-28",
},
Object {
+ "displayName": "January 29, 2014",
"endDate": "2014-01-29",
"id": "20140129",
"iso": "20140129",
"name": "2014-01-29",
+ "periodType": "DAILY",
"startDate": "2014-01-29",
},
Object {
+ "displayName": "January 30, 2014",
"endDate": "2014-01-30",
"id": "20140130",
"iso": "20140130",
"name": "2014-01-30",
+ "periodType": "DAILY",
"startDate": "2014-01-30",
},
Object {
+ "displayName": "January 31, 2014",
"endDate": "2014-01-31",
"id": "20140131",
"iso": "20140131",
"name": "2014-01-31",
+ "periodType": "DAILY",
"startDate": "2014-01-31",
},
Object {
+ "displayName": "February 1, 2014",
"endDate": "2014-02-01",
"id": "20140201",
"iso": "20140201",
"name": "2014-02-01",
+ "periodType": "DAILY",
"startDate": "2014-02-01",
},
Object {
+ "displayName": "February 2, 2014",
"endDate": "2014-02-02",
"id": "20140202",
"iso": "20140202",
"name": "2014-02-02",
+ "periodType": "DAILY",
"startDate": "2014-02-02",
},
Object {
+ "displayName": "February 3, 2014",
"endDate": "2014-02-03",
"id": "20140203",
"iso": "20140203",
"name": "2014-02-03",
+ "periodType": "DAILY",
"startDate": "2014-02-03",
},
Object {
+ "displayName": "February 4, 2014",
"endDate": "2014-02-04",
"id": "20140204",
"iso": "20140204",
"name": "2014-02-04",
+ "periodType": "DAILY",
"startDate": "2014-02-04",
},
Object {
+ "displayName": "February 5, 2014",
"endDate": "2014-02-05",
"id": "20140205",
"iso": "20140205",
"name": "2014-02-05",
+ "periodType": "DAILY",
"startDate": "2014-02-05",
},
Object {
+ "displayName": "February 6, 2014",
"endDate": "2014-02-06",
"id": "20140206",
"iso": "20140206",
"name": "2014-02-06",
+ "periodType": "DAILY",
"startDate": "2014-02-06",
},
Object {
+ "displayName": "February 7, 2014",
"endDate": "2014-02-07",
"id": "20140207",
"iso": "20140207",
"name": "2014-02-07",
+ "periodType": "DAILY",
"startDate": "2014-02-07",
},
Object {
+ "displayName": "February 8, 2014",
"endDate": "2014-02-08",
"id": "20140208",
"iso": "20140208",
"name": "2014-02-08",
+ "periodType": "DAILY",
"startDate": "2014-02-08",
},
Object {
+ "displayName": "February 9, 2014",
"endDate": "2014-02-09",
"id": "20140209",
"iso": "20140209",
"name": "2014-02-09",
+ "periodType": "DAILY",
"startDate": "2014-02-09",
},
Object {
+ "displayName": "February 10, 2014",
"endDate": "2014-02-10",
"id": "20140210",
"iso": "20140210",
"name": "2014-02-10",
+ "periodType": "DAILY",
"startDate": "2014-02-10",
},
Object {
+ "displayName": "February 11, 2014",
"endDate": "2014-02-11",
"id": "20140211",
"iso": "20140211",
"name": "2014-02-11",
+ "periodType": "DAILY",
"startDate": "2014-02-11",
},
Object {
+ "displayName": "February 12, 2014",
"endDate": "2014-02-12",
"id": "20140212",
"iso": "20140212",
"name": "2014-02-12",
+ "periodType": "DAILY",
"startDate": "2014-02-12",
},
Object {
+ "displayName": "February 13, 2014",
"endDate": "2014-02-13",
"id": "20140213",
"iso": "20140213",
"name": "2014-02-13",
+ "periodType": "DAILY",
"startDate": "2014-02-13",
},
Object {
+ "displayName": "February 14, 2014",
"endDate": "2014-02-14",
"id": "20140214",
"iso": "20140214",
"name": "2014-02-14",
+ "periodType": "DAILY",
"startDate": "2014-02-14",
},
Object {
+ "displayName": "February 15, 2014",
"endDate": "2014-02-15",
"id": "20140215",
"iso": "20140215",
"name": "2014-02-15",
+ "periodType": "DAILY",
"startDate": "2014-02-15",
},
Object {
+ "displayName": "February 16, 2014",
"endDate": "2014-02-16",
"id": "20140216",
"iso": "20140216",
"name": "2014-02-16",
+ "periodType": "DAILY",
"startDate": "2014-02-16",
},
Object {
+ "displayName": "February 17, 2014",
"endDate": "2014-02-17",
"id": "20140217",
"iso": "20140217",
"name": "2014-02-17",
+ "periodType": "DAILY",
"startDate": "2014-02-17",
},
Object {
+ "displayName": "February 18, 2014",
"endDate": "2014-02-18",
"id": "20140218",
"iso": "20140218",
"name": "2014-02-18",
+ "periodType": "DAILY",
"startDate": "2014-02-18",
},
Object {
+ "displayName": "February 19, 2014",
"endDate": "2014-02-19",
"id": "20140219",
"iso": "20140219",
"name": "2014-02-19",
+ "periodType": "DAILY",
"startDate": "2014-02-19",
},
Object {
+ "displayName": "February 20, 2014",
"endDate": "2014-02-20",
"id": "20140220",
"iso": "20140220",
"name": "2014-02-20",
+ "periodType": "DAILY",
"startDate": "2014-02-20",
},
Object {
+ "displayName": "February 21, 2014",
"endDate": "2014-02-21",
"id": "20140221",
"iso": "20140221",
"name": "2014-02-21",
+ "periodType": "DAILY",
"startDate": "2014-02-21",
},
Object {
+ "displayName": "February 22, 2014",
"endDate": "2014-02-22",
"id": "20140222",
"iso": "20140222",
"name": "2014-02-22",
+ "periodType": "DAILY",
"startDate": "2014-02-22",
},
Object {
+ "displayName": "February 23, 2014",
"endDate": "2014-02-23",
"id": "20140223",
"iso": "20140223",
"name": "2014-02-23",
+ "periodType": "DAILY",
"startDate": "2014-02-23",
},
Object {
+ "displayName": "February 24, 2014",
"endDate": "2014-02-24",
"id": "20140224",
"iso": "20140224",
"name": "2014-02-24",
+ "periodType": "DAILY",
"startDate": "2014-02-24",
},
Object {
+ "displayName": "February 25, 2014",
"endDate": "2014-02-25",
"id": "20140225",
"iso": "20140225",
"name": "2014-02-25",
+ "periodType": "DAILY",
"startDate": "2014-02-25",
},
Object {
+ "displayName": "February 26, 2014",
"endDate": "2014-02-26",
"id": "20140226",
"iso": "20140226",
"name": "2014-02-26",
+ "periodType": "DAILY",
"startDate": "2014-02-26",
},
Object {
+ "displayName": "February 27, 2014",
"endDate": "2014-02-27",
"id": "20140227",
"iso": "20140227",
"name": "2014-02-27",
+ "periodType": "DAILY",
"startDate": "2014-02-27",
},
Object {
+ "displayName": "February 28, 2014",
"endDate": "2014-02-28",
"id": "20140228",
"iso": "20140228",
"name": "2014-02-28",
+ "periodType": "DAILY",
"startDate": "2014-02-28",
},
Object {
+ "displayName": "March 1, 2014",
"endDate": "2014-03-01",
"id": "20140301",
"iso": "20140301",
"name": "2014-03-01",
+ "periodType": "DAILY",
"startDate": "2014-03-01",
},
Object {
+ "displayName": "March 2, 2014",
"endDate": "2014-03-02",
"id": "20140302",
"iso": "20140302",
"name": "2014-03-02",
+ "periodType": "DAILY",
"startDate": "2014-03-02",
},
Object {
+ "displayName": "March 3, 2014",
"endDate": "2014-03-03",
"id": "20140303",
"iso": "20140303",
"name": "2014-03-03",
+ "periodType": "DAILY",
"startDate": "2014-03-03",
},
Object {
+ "displayName": "March 4, 2014",
"endDate": "2014-03-04",
"id": "20140304",
"iso": "20140304",
"name": "2014-03-04",
+ "periodType": "DAILY",
"startDate": "2014-03-04",
},
Object {
+ "displayName": "March 5, 2014",
"endDate": "2014-03-05",
"id": "20140305",
"iso": "20140305",
"name": "2014-03-05",
+ "periodType": "DAILY",
"startDate": "2014-03-05",
},
Object {
+ "displayName": "March 6, 2014",
"endDate": "2014-03-06",
"id": "20140306",
"iso": "20140306",
"name": "2014-03-06",
+ "periodType": "DAILY",
"startDate": "2014-03-06",
},
Object {
+ "displayName": "March 7, 2014",
"endDate": "2014-03-07",
"id": "20140307",
"iso": "20140307",
"name": "2014-03-07",
+ "periodType": "DAILY",
"startDate": "2014-03-07",
},
Object {
+ "displayName": "March 8, 2014",
"endDate": "2014-03-08",
"id": "20140308",
"iso": "20140308",
"name": "2014-03-08",
+ "periodType": "DAILY",
"startDate": "2014-03-08",
},
Object {
+ "displayName": "March 9, 2014",
"endDate": "2014-03-09",
"id": "20140309",
"iso": "20140309",
"name": "2014-03-09",
+ "periodType": "DAILY",
"startDate": "2014-03-09",
},
Object {
+ "displayName": "March 10, 2014",
"endDate": "2014-03-10",
"id": "20140310",
"iso": "20140310",
"name": "2014-03-10",
+ "periodType": "DAILY",
"startDate": "2014-03-10",
},
Object {
+ "displayName": "March 11, 2014",
"endDate": "2014-03-11",
"id": "20140311",
"iso": "20140311",
"name": "2014-03-11",
+ "periodType": "DAILY",
"startDate": "2014-03-11",
},
Object {
+ "displayName": "March 12, 2014",
"endDate": "2014-03-12",
"id": "20140312",
"iso": "20140312",
"name": "2014-03-12",
+ "periodType": "DAILY",
"startDate": "2014-03-12",
},
Object {
+ "displayName": "March 13, 2014",
"endDate": "2014-03-13",
"id": "20140313",
"iso": "20140313",
"name": "2014-03-13",
+ "periodType": "DAILY",
"startDate": "2014-03-13",
},
Object {
+ "displayName": "March 14, 2014",
"endDate": "2014-03-14",
"id": "20140314",
"iso": "20140314",
"name": "2014-03-14",
+ "periodType": "DAILY",
"startDate": "2014-03-14",
},
Object {
+ "displayName": "March 15, 2014",
"endDate": "2014-03-15",
"id": "20140315",
"iso": "20140315",
"name": "2014-03-15",
+ "periodType": "DAILY",
"startDate": "2014-03-15",
},
Object {
+ "displayName": "March 16, 2014",
"endDate": "2014-03-16",
"id": "20140316",
"iso": "20140316",
"name": "2014-03-16",
+ "periodType": "DAILY",
"startDate": "2014-03-16",
},
Object {
+ "displayName": "March 17, 2014",
"endDate": "2014-03-17",
"id": "20140317",
"iso": "20140317",
"name": "2014-03-17",
+ "periodType": "DAILY",
"startDate": "2014-03-17",
},
Object {
+ "displayName": "March 18, 2014",
"endDate": "2014-03-18",
"id": "20140318",
"iso": "20140318",
"name": "2014-03-18",
+ "periodType": "DAILY",
"startDate": "2014-03-18",
},
Object {
+ "displayName": "March 19, 2014",
"endDate": "2014-03-19",
"id": "20140319",
"iso": "20140319",
"name": "2014-03-19",
+ "periodType": "DAILY",
"startDate": "2014-03-19",
},
Object {
+ "displayName": "March 20, 2014",
"endDate": "2014-03-20",
"id": "20140320",
"iso": "20140320",
"name": "2014-03-20",
+ "periodType": "DAILY",
"startDate": "2014-03-20",
},
Object {
+ "displayName": "March 21, 2014",
"endDate": "2014-03-21",
"id": "20140321",
"iso": "20140321",
"name": "2014-03-21",
+ "periodType": "DAILY",
"startDate": "2014-03-21",
},
Object {
+ "displayName": "March 22, 2014",
"endDate": "2014-03-22",
"id": "20140322",
"iso": "20140322",
"name": "2014-03-22",
+ "periodType": "DAILY",
"startDate": "2014-03-22",
},
Object {
+ "displayName": "March 23, 2014",
"endDate": "2014-03-23",
"id": "20140323",
"iso": "20140323",
"name": "2014-03-23",
+ "periodType": "DAILY",
"startDate": "2014-03-23",
},
Object {
+ "displayName": "March 24, 2014",
"endDate": "2014-03-24",
"id": "20140324",
"iso": "20140324",
"name": "2014-03-24",
+ "periodType": "DAILY",
"startDate": "2014-03-24",
},
Object {
+ "displayName": "March 25, 2014",
"endDate": "2014-03-25",
"id": "20140325",
"iso": "20140325",
"name": "2014-03-25",
+ "periodType": "DAILY",
"startDate": "2014-03-25",
},
Object {
+ "displayName": "March 26, 2014",
"endDate": "2014-03-26",
"id": "20140326",
"iso": "20140326",
"name": "2014-03-26",
+ "periodType": "DAILY",
"startDate": "2014-03-26",
},
Object {
+ "displayName": "March 27, 2014",
"endDate": "2014-03-27",
"id": "20140327",
"iso": "20140327",
"name": "2014-03-27",
+ "periodType": "DAILY",
"startDate": "2014-03-27",
},
Object {
+ "displayName": "March 28, 2014",
"endDate": "2014-03-28",
"id": "20140328",
"iso": "20140328",
"name": "2014-03-28",
+ "periodType": "DAILY",
"startDate": "2014-03-28",
},
Object {
+ "displayName": "March 29, 2014",
"endDate": "2014-03-29",
"id": "20140329",
"iso": "20140329",
"name": "2014-03-29",
+ "periodType": "DAILY",
"startDate": "2014-03-29",
},
Object {
+ "displayName": "March 30, 2014",
"endDate": "2014-03-30",
"id": "20140330",
"iso": "20140330",
"name": "2014-03-30",
+ "periodType": "DAILY",
"startDate": "2014-03-30",
},
Object {
+ "displayName": "March 31, 2014",
"endDate": "2014-03-31",
"id": "20140331",
"iso": "20140331",
"name": "2014-03-31",
+ "periodType": "DAILY",
"startDate": "2014-03-31",
},
Object {
+ "displayName": "April 1, 2014",
"endDate": "2014-04-01",
"id": "20140401",
"iso": "20140401",
"name": "2014-04-01",
+ "periodType": "DAILY",
"startDate": "2014-04-01",
},
Object {
+ "displayName": "April 2, 2014",
"endDate": "2014-04-02",
"id": "20140402",
"iso": "20140402",
"name": "2014-04-02",
+ "periodType": "DAILY",
"startDate": "2014-04-02",
},
Object {
+ "displayName": "April 3, 2014",
"endDate": "2014-04-03",
"id": "20140403",
"iso": "20140403",
"name": "2014-04-03",
+ "periodType": "DAILY",
"startDate": "2014-04-03",
},
Object {
+ "displayName": "April 4, 2014",
"endDate": "2014-04-04",
"id": "20140404",
"iso": "20140404",
"name": "2014-04-04",
+ "periodType": "DAILY",
"startDate": "2014-04-04",
},
Object {
+ "displayName": "April 5, 2014",
"endDate": "2014-04-05",
"id": "20140405",
"iso": "20140405",
"name": "2014-04-05",
+ "periodType": "DAILY",
"startDate": "2014-04-05",
},
Object {
+ "displayName": "April 6, 2014",
"endDate": "2014-04-06",
"id": "20140406",
"iso": "20140406",
"name": "2014-04-06",
+ "periodType": "DAILY",
"startDate": "2014-04-06",
},
Object {
+ "displayName": "April 7, 2014",
"endDate": "2014-04-07",
"id": "20140407",
"iso": "20140407",
"name": "2014-04-07",
+ "periodType": "DAILY",
"startDate": "2014-04-07",
},
Object {
+ "displayName": "April 8, 2014",
"endDate": "2014-04-08",
"id": "20140408",
"iso": "20140408",
"name": "2014-04-08",
+ "periodType": "DAILY",
"startDate": "2014-04-08",
},
Object {
+ "displayName": "April 9, 2014",
"endDate": "2014-04-09",
"id": "20140409",
"iso": "20140409",
"name": "2014-04-09",
+ "periodType": "DAILY",
"startDate": "2014-04-09",
},
Object {
+ "displayName": "April 10, 2014",
"endDate": "2014-04-10",
"id": "20140410",
"iso": "20140410",
"name": "2014-04-10",
+ "periodType": "DAILY",
"startDate": "2014-04-10",
},
Object {
+ "displayName": "April 11, 2014",
"endDate": "2014-04-11",
"id": "20140411",
"iso": "20140411",
"name": "2014-04-11",
+ "periodType": "DAILY",
"startDate": "2014-04-11",
},
Object {
+ "displayName": "April 12, 2014",
"endDate": "2014-04-12",
"id": "20140412",
"iso": "20140412",
"name": "2014-04-12",
+ "periodType": "DAILY",
"startDate": "2014-04-12",
},
Object {
+ "displayName": "April 13, 2014",
"endDate": "2014-04-13",
"id": "20140413",
"iso": "20140413",
"name": "2014-04-13",
+ "periodType": "DAILY",
"startDate": "2014-04-13",
},
Object {
+ "displayName": "April 14, 2014",
"endDate": "2014-04-14",
"id": "20140414",
"iso": "20140414",
"name": "2014-04-14",
+ "periodType": "DAILY",
"startDate": "2014-04-14",
},
Object {
+ "displayName": "April 15, 2014",
"endDate": "2014-04-15",
"id": "20140415",
"iso": "20140415",
"name": "2014-04-15",
+ "periodType": "DAILY",
"startDate": "2014-04-15",
},
Object {
+ "displayName": "April 16, 2014",
"endDate": "2014-04-16",
"id": "20140416",
"iso": "20140416",
"name": "2014-04-16",
+ "periodType": "DAILY",
"startDate": "2014-04-16",
},
Object {
+ "displayName": "April 17, 2014",
"endDate": "2014-04-17",
"id": "20140417",
"iso": "20140417",
"name": "2014-04-17",
+ "periodType": "DAILY",
"startDate": "2014-04-17",
},
Object {
+ "displayName": "April 18, 2014",
"endDate": "2014-04-18",
"id": "20140418",
"iso": "20140418",
"name": "2014-04-18",
+ "periodType": "DAILY",
"startDate": "2014-04-18",
},
Object {
+ "displayName": "April 19, 2014",
"endDate": "2014-04-19",
"id": "20140419",
"iso": "20140419",
"name": "2014-04-19",
+ "periodType": "DAILY",
"startDate": "2014-04-19",
},
Object {
+ "displayName": "April 20, 2014",
"endDate": "2014-04-20",
"id": "20140420",
"iso": "20140420",
"name": "2014-04-20",
+ "periodType": "DAILY",
"startDate": "2014-04-20",
},
Object {
+ "displayName": "April 21, 2014",
"endDate": "2014-04-21",
"id": "20140421",
"iso": "20140421",
"name": "2014-04-21",
+ "periodType": "DAILY",
"startDate": "2014-04-21",
},
Object {
+ "displayName": "April 22, 2014",
"endDate": "2014-04-22",
"id": "20140422",
"iso": "20140422",
"name": "2014-04-22",
+ "periodType": "DAILY",
"startDate": "2014-04-22",
},
Object {
+ "displayName": "April 23, 2014",
"endDate": "2014-04-23",
"id": "20140423",
"iso": "20140423",
"name": "2014-04-23",
+ "periodType": "DAILY",
"startDate": "2014-04-23",
},
Object {
+ "displayName": "April 24, 2014",
"endDate": "2014-04-24",
"id": "20140424",
"iso": "20140424",
"name": "2014-04-24",
+ "periodType": "DAILY",
"startDate": "2014-04-24",
},
Object {
+ "displayName": "April 25, 2014",
"endDate": "2014-04-25",
"id": "20140425",
"iso": "20140425",
"name": "2014-04-25",
+ "periodType": "DAILY",
"startDate": "2014-04-25",
},
Object {
+ "displayName": "April 26, 2014",
"endDate": "2014-04-26",
"id": "20140426",
"iso": "20140426",
"name": "2014-04-26",
+ "periodType": "DAILY",
"startDate": "2014-04-26",
},
Object {
+ "displayName": "April 27, 2014",
"endDate": "2014-04-27",
"id": "20140427",
"iso": "20140427",
"name": "2014-04-27",
+ "periodType": "DAILY",
"startDate": "2014-04-27",
},
Object {
+ "displayName": "April 28, 2014",
"endDate": "2014-04-28",
"id": "20140428",
"iso": "20140428",
"name": "2014-04-28",
+ "periodType": "DAILY",
"startDate": "2014-04-28",
},
Object {
+ "displayName": "April 29, 2014",
"endDate": "2014-04-29",
"id": "20140429",
"iso": "20140429",
"name": "2014-04-29",
+ "periodType": "DAILY",
"startDate": "2014-04-29",
},
Object {
+ "displayName": "April 30, 2014",
"endDate": "2014-04-30",
"id": "20140430",
"iso": "20140430",
"name": "2014-04-30",
+ "periodType": "DAILY",
"startDate": "2014-04-30",
},
Object {
+ "displayName": "May 1, 2014",
"endDate": "2014-05-01",
"id": "20140501",
"iso": "20140501",
"name": "2014-05-01",
+ "periodType": "DAILY",
"startDate": "2014-05-01",
},
Object {
+ "displayName": "May 2, 2014",
"endDate": "2014-05-02",
"id": "20140502",
"iso": "20140502",
"name": "2014-05-02",
+ "periodType": "DAILY",
"startDate": "2014-05-02",
},
Object {
+ "displayName": "May 3, 2014",
"endDate": "2014-05-03",
"id": "20140503",
"iso": "20140503",
"name": "2014-05-03",
+ "periodType": "DAILY",
"startDate": "2014-05-03",
},
Object {
+ "displayName": "May 4, 2014",
"endDate": "2014-05-04",
"id": "20140504",
"iso": "20140504",
"name": "2014-05-04",
+ "periodType": "DAILY",
"startDate": "2014-05-04",
},
Object {
+ "displayName": "May 5, 2014",
"endDate": "2014-05-05",
"id": "20140505",
"iso": "20140505",
"name": "2014-05-05",
+ "periodType": "DAILY",
"startDate": "2014-05-05",
},
Object {
+ "displayName": "May 6, 2014",
"endDate": "2014-05-06",
"id": "20140506",
"iso": "20140506",
"name": "2014-05-06",
+ "periodType": "DAILY",
"startDate": "2014-05-06",
},
Object {
+ "displayName": "May 7, 2014",
"endDate": "2014-05-07",
"id": "20140507",
"iso": "20140507",
"name": "2014-05-07",
+ "periodType": "DAILY",
"startDate": "2014-05-07",
},
Object {
+ "displayName": "May 8, 2014",
"endDate": "2014-05-08",
"id": "20140508",
"iso": "20140508",
"name": "2014-05-08",
+ "periodType": "DAILY",
"startDate": "2014-05-08",
},
Object {
+ "displayName": "May 9, 2014",
"endDate": "2014-05-09",
"id": "20140509",
"iso": "20140509",
"name": "2014-05-09",
+ "periodType": "DAILY",
"startDate": "2014-05-09",
},
Object {
+ "displayName": "May 10, 2014",
"endDate": "2014-05-10",
"id": "20140510",
"iso": "20140510",
"name": "2014-05-10",
+ "periodType": "DAILY",
"startDate": "2014-05-10",
},
Object {
+ "displayName": "May 11, 2014",
"endDate": "2014-05-11",
"id": "20140511",
"iso": "20140511",
"name": "2014-05-11",
+ "periodType": "DAILY",
"startDate": "2014-05-11",
},
Object {
+ "displayName": "May 12, 2014",
"endDate": "2014-05-12",
"id": "20140512",
"iso": "20140512",
"name": "2014-05-12",
+ "periodType": "DAILY",
"startDate": "2014-05-12",
},
Object {
+ "displayName": "May 13, 2014",
"endDate": "2014-05-13",
"id": "20140513",
"iso": "20140513",
"name": "2014-05-13",
+ "periodType": "DAILY",
"startDate": "2014-05-13",
},
Object {
+ "displayName": "May 14, 2014",
"endDate": "2014-05-14",
"id": "20140514",
"iso": "20140514",
"name": "2014-05-14",
+ "periodType": "DAILY",
"startDate": "2014-05-14",
},
Object {
+ "displayName": "May 15, 2014",
"endDate": "2014-05-15",
"id": "20140515",
"iso": "20140515",
"name": "2014-05-15",
+ "periodType": "DAILY",
"startDate": "2014-05-15",
},
Object {
+ "displayName": "May 16, 2014",
"endDate": "2014-05-16",
"id": "20140516",
"iso": "20140516",
"name": "2014-05-16",
+ "periodType": "DAILY",
"startDate": "2014-05-16",
},
Object {
+ "displayName": "May 17, 2014",
"endDate": "2014-05-17",
"id": "20140517",
"iso": "20140517",
"name": "2014-05-17",
+ "periodType": "DAILY",
"startDate": "2014-05-17",
},
Object {
+ "displayName": "May 18, 2014",
"endDate": "2014-05-18",
"id": "20140518",
"iso": "20140518",
"name": "2014-05-18",
+ "periodType": "DAILY",
"startDate": "2014-05-18",
},
Object {
+ "displayName": "May 19, 2014",
"endDate": "2014-05-19",
"id": "20140519",
"iso": "20140519",
"name": "2014-05-19",
+ "periodType": "DAILY",
"startDate": "2014-05-19",
},
Object {
+ "displayName": "May 20, 2014",
"endDate": "2014-05-20",
"id": "20140520",
"iso": "20140520",
"name": "2014-05-20",
+ "periodType": "DAILY",
"startDate": "2014-05-20",
},
Object {
+ "displayName": "May 21, 2014",
"endDate": "2014-05-21",
"id": "20140521",
"iso": "20140521",
"name": "2014-05-21",
+ "periodType": "DAILY",
"startDate": "2014-05-21",
},
Object {
+ "displayName": "May 22, 2014",
"endDate": "2014-05-22",
"id": "20140522",
"iso": "20140522",
"name": "2014-05-22",
+ "periodType": "DAILY",
"startDate": "2014-05-22",
},
Object {
+ "displayName": "May 23, 2014",
"endDate": "2014-05-23",
"id": "20140523",
"iso": "20140523",
"name": "2014-05-23",
+ "periodType": "DAILY",
"startDate": "2014-05-23",
},
Object {
+ "displayName": "May 24, 2014",
"endDate": "2014-05-24",
"id": "20140524",
"iso": "20140524",
"name": "2014-05-24",
+ "periodType": "DAILY",
"startDate": "2014-05-24",
},
Object {
+ "displayName": "May 25, 2014",
"endDate": "2014-05-25",
"id": "20140525",
"iso": "20140525",
"name": "2014-05-25",
+ "periodType": "DAILY",
"startDate": "2014-05-25",
},
Object {
+ "displayName": "May 26, 2014",
"endDate": "2014-05-26",
"id": "20140526",
"iso": "20140526",
"name": "2014-05-26",
+ "periodType": "DAILY",
"startDate": "2014-05-26",
},
Object {
+ "displayName": "May 27, 2014",
"endDate": "2014-05-27",
"id": "20140527",
"iso": "20140527",
"name": "2014-05-27",
+ "periodType": "DAILY",
"startDate": "2014-05-27",
},
Object {
+ "displayName": "May 28, 2014",
"endDate": "2014-05-28",
"id": "20140528",
"iso": "20140528",
"name": "2014-05-28",
+ "periodType": "DAILY",
"startDate": "2014-05-28",
},
Object {
+ "displayName": "May 29, 2014",
"endDate": "2014-05-29",
"id": "20140529",
"iso": "20140529",
"name": "2014-05-29",
+ "periodType": "DAILY",
"startDate": "2014-05-29",
},
Object {
+ "displayName": "May 30, 2014",
"endDate": "2014-05-30",
"id": "20140530",
"iso": "20140530",
"name": "2014-05-30",
+ "periodType": "DAILY",
"startDate": "2014-05-30",
},
Object {
+ "displayName": "May 31, 2014",
"endDate": "2014-05-31",
"id": "20140531",
"iso": "20140531",
"name": "2014-05-31",
+ "periodType": "DAILY",
"startDate": "2014-05-31",
},
Object {
+ "displayName": "June 1, 2014",
"endDate": "2014-06-01",
"id": "20140601",
"iso": "20140601",
"name": "2014-06-01",
+ "periodType": "DAILY",
"startDate": "2014-06-01",
},
Object {
+ "displayName": "June 2, 2014",
"endDate": "2014-06-02",
"id": "20140602",
"iso": "20140602",
"name": "2014-06-02",
+ "periodType": "DAILY",
"startDate": "2014-06-02",
},
Object {
+ "displayName": "June 3, 2014",
"endDate": "2014-06-03",
"id": "20140603",
"iso": "20140603",
"name": "2014-06-03",
+ "periodType": "DAILY",
"startDate": "2014-06-03",
},
Object {
+ "displayName": "June 4, 2014",
"endDate": "2014-06-04",
"id": "20140604",
"iso": "20140604",
"name": "2014-06-04",
+ "periodType": "DAILY",
"startDate": "2014-06-04",
},
Object {
+ "displayName": "June 5, 2014",
"endDate": "2014-06-05",
"id": "20140605",
"iso": "20140605",
"name": "2014-06-05",
+ "periodType": "DAILY",
"startDate": "2014-06-05",
},
Object {
+ "displayName": "June 6, 2014",
"endDate": "2014-06-06",
"id": "20140606",
"iso": "20140606",
"name": "2014-06-06",
+ "periodType": "DAILY",
"startDate": "2014-06-06",
},
Object {
+ "displayName": "June 7, 2014",
"endDate": "2014-06-07",
"id": "20140607",
"iso": "20140607",
"name": "2014-06-07",
+ "periodType": "DAILY",
"startDate": "2014-06-07",
},
Object {
+ "displayName": "June 8, 2014",
"endDate": "2014-06-08",
"id": "20140608",
"iso": "20140608",
"name": "2014-06-08",
+ "periodType": "DAILY",
"startDate": "2014-06-08",
},
Object {
+ "displayName": "June 9, 2014",
"endDate": "2014-06-09",
"id": "20140609",
"iso": "20140609",
"name": "2014-06-09",
+ "periodType": "DAILY",
"startDate": "2014-06-09",
},
Object {
+ "displayName": "June 10, 2014",
"endDate": "2014-06-10",
"id": "20140610",
"iso": "20140610",
"name": "2014-06-10",
+ "periodType": "DAILY",
"startDate": "2014-06-10",
},
Object {
+ "displayName": "June 11, 2014",
"endDate": "2014-06-11",
"id": "20140611",
"iso": "20140611",
"name": "2014-06-11",
+ "periodType": "DAILY",
"startDate": "2014-06-11",
},
Object {
+ "displayName": "June 12, 2014",
"endDate": "2014-06-12",
"id": "20140612",
"iso": "20140612",
"name": "2014-06-12",
+ "periodType": "DAILY",
"startDate": "2014-06-12",
},
Object {
+ "displayName": "June 13, 2014",
"endDate": "2014-06-13",
"id": "20140613",
"iso": "20140613",
"name": "2014-06-13",
+ "periodType": "DAILY",
"startDate": "2014-06-13",
},
Object {
+ "displayName": "June 14, 2014",
"endDate": "2014-06-14",
"id": "20140614",
"iso": "20140614",
"name": "2014-06-14",
+ "periodType": "DAILY",
"startDate": "2014-06-14",
},
Object {
+ "displayName": "June 15, 2014",
"endDate": "2014-06-15",
"id": "20140615",
"iso": "20140615",
"name": "2014-06-15",
+ "periodType": "DAILY",
"startDate": "2014-06-15",
},
Object {
+ "displayName": "June 16, 2014",
"endDate": "2014-06-16",
"id": "20140616",
"iso": "20140616",
"name": "2014-06-16",
+ "periodType": "DAILY",
"startDate": "2014-06-16",
},
Object {
+ "displayName": "June 17, 2014",
"endDate": "2014-06-17",
"id": "20140617",
"iso": "20140617",
"name": "2014-06-17",
+ "periodType": "DAILY",
"startDate": "2014-06-17",
},
Object {
+ "displayName": "June 18, 2014",
"endDate": "2014-06-18",
"id": "20140618",
"iso": "20140618",
"name": "2014-06-18",
+ "periodType": "DAILY",
"startDate": "2014-06-18",
},
Object {
+ "displayName": "June 19, 2014",
"endDate": "2014-06-19",
"id": "20140619",
"iso": "20140619",
"name": "2014-06-19",
+ "periodType": "DAILY",
"startDate": "2014-06-19",
},
Object {
+ "displayName": "June 20, 2014",
"endDate": "2014-06-20",
"id": "20140620",
"iso": "20140620",
"name": "2014-06-20",
+ "periodType": "DAILY",
"startDate": "2014-06-20",
},
Object {
+ "displayName": "June 21, 2014",
"endDate": "2014-06-21",
"id": "20140621",
"iso": "20140621",
"name": "2014-06-21",
+ "periodType": "DAILY",
"startDate": "2014-06-21",
},
Object {
+ "displayName": "June 22, 2014",
"endDate": "2014-06-22",
"id": "20140622",
"iso": "20140622",
"name": "2014-06-22",
+ "periodType": "DAILY",
"startDate": "2014-06-22",
},
Object {
+ "displayName": "June 23, 2014",
"endDate": "2014-06-23",
"id": "20140623",
"iso": "20140623",
"name": "2014-06-23",
+ "periodType": "DAILY",
"startDate": "2014-06-23",
},
Object {
+ "displayName": "June 24, 2014",
"endDate": "2014-06-24",
"id": "20140624",
"iso": "20140624",
"name": "2014-06-24",
+ "periodType": "DAILY",
"startDate": "2014-06-24",
},
Object {
+ "displayName": "June 25, 2014",
"endDate": "2014-06-25",
"id": "20140625",
"iso": "20140625",
"name": "2014-06-25",
+ "periodType": "DAILY",
"startDate": "2014-06-25",
},
Object {
+ "displayName": "June 26, 2014",
"endDate": "2014-06-26",
"id": "20140626",
"iso": "20140626",
"name": "2014-06-26",
+ "periodType": "DAILY",
"startDate": "2014-06-26",
},
Object {
+ "displayName": "June 27, 2014",
"endDate": "2014-06-27",
"id": "20140627",
"iso": "20140627",
"name": "2014-06-27",
+ "periodType": "DAILY",
"startDate": "2014-06-27",
},
Object {
+ "displayName": "June 28, 2014",
"endDate": "2014-06-28",
"id": "20140628",
"iso": "20140628",
"name": "2014-06-28",
+ "periodType": "DAILY",
"startDate": "2014-06-28",
},
Object {
+ "displayName": "June 29, 2014",
"endDate": "2014-06-29",
"id": "20140629",
"iso": "20140629",
"name": "2014-06-29",
+ "periodType": "DAILY",
"startDate": "2014-06-29",
},
Object {
+ "displayName": "June 30, 2014",
"endDate": "2014-06-30",
"id": "20140630",
"iso": "20140630",
"name": "2014-06-30",
+ "periodType": "DAILY",
"startDate": "2014-06-30",
},
Object {
+ "displayName": "July 1, 2014",
"endDate": "2014-07-01",
"id": "20140701",
"iso": "20140701",
"name": "2014-07-01",
+ "periodType": "DAILY",
"startDate": "2014-07-01",
},
Object {
+ "displayName": "July 2, 2014",
"endDate": "2014-07-02",
"id": "20140702",
"iso": "20140702",
"name": "2014-07-02",
+ "periodType": "DAILY",
"startDate": "2014-07-02",
},
Object {
+ "displayName": "July 3, 2014",
"endDate": "2014-07-03",
"id": "20140703",
"iso": "20140703",
"name": "2014-07-03",
+ "periodType": "DAILY",
"startDate": "2014-07-03",
},
Object {
+ "displayName": "July 4, 2014",
"endDate": "2014-07-04",
"id": "20140704",
"iso": "20140704",
"name": "2014-07-04",
+ "periodType": "DAILY",
"startDate": "2014-07-04",
},
Object {
+ "displayName": "July 5, 2014",
"endDate": "2014-07-05",
"id": "20140705",
"iso": "20140705",
"name": "2014-07-05",
+ "periodType": "DAILY",
"startDate": "2014-07-05",
},
Object {
+ "displayName": "July 6, 2014",
"endDate": "2014-07-06",
"id": "20140706",
"iso": "20140706",
"name": "2014-07-06",
+ "periodType": "DAILY",
"startDate": "2014-07-06",
},
Object {
+ "displayName": "July 7, 2014",
"endDate": "2014-07-07",
"id": "20140707",
"iso": "20140707",
"name": "2014-07-07",
+ "periodType": "DAILY",
"startDate": "2014-07-07",
},
Object {
+ "displayName": "July 8, 2014",
"endDate": "2014-07-08",
"id": "20140708",
"iso": "20140708",
"name": "2014-07-08",
+ "periodType": "DAILY",
"startDate": "2014-07-08",
},
Object {
+ "displayName": "July 9, 2014",
"endDate": "2014-07-09",
"id": "20140709",
"iso": "20140709",
"name": "2014-07-09",
+ "periodType": "DAILY",
"startDate": "2014-07-09",
},
Object {
+ "displayName": "July 10, 2014",
"endDate": "2014-07-10",
"id": "20140710",
"iso": "20140710",
"name": "2014-07-10",
+ "periodType": "DAILY",
"startDate": "2014-07-10",
},
Object {
+ "displayName": "July 11, 2014",
"endDate": "2014-07-11",
"id": "20140711",
"iso": "20140711",
"name": "2014-07-11",
+ "periodType": "DAILY",
"startDate": "2014-07-11",
},
Object {
+ "displayName": "July 12, 2014",
"endDate": "2014-07-12",
"id": "20140712",
"iso": "20140712",
"name": "2014-07-12",
+ "periodType": "DAILY",
"startDate": "2014-07-12",
},
Object {
+ "displayName": "July 13, 2014",
"endDate": "2014-07-13",
"id": "20140713",
"iso": "20140713",
"name": "2014-07-13",
+ "periodType": "DAILY",
"startDate": "2014-07-13",
},
Object {
+ "displayName": "July 14, 2014",
"endDate": "2014-07-14",
"id": "20140714",
"iso": "20140714",
"name": "2014-07-14",
+ "periodType": "DAILY",
"startDate": "2014-07-14",
},
Object {
+ "displayName": "July 15, 2014",
"endDate": "2014-07-15",
"id": "20140715",
"iso": "20140715",
"name": "2014-07-15",
+ "periodType": "DAILY",
"startDate": "2014-07-15",
},
Object {
+ "displayName": "July 16, 2014",
"endDate": "2014-07-16",
"id": "20140716",
"iso": "20140716",
"name": "2014-07-16",
+ "periodType": "DAILY",
"startDate": "2014-07-16",
},
Object {
+ "displayName": "July 17, 2014",
"endDate": "2014-07-17",
"id": "20140717",
"iso": "20140717",
"name": "2014-07-17",
+ "periodType": "DAILY",
"startDate": "2014-07-17",
},
Object {
+ "displayName": "July 18, 2014",
"endDate": "2014-07-18",
"id": "20140718",
"iso": "20140718",
"name": "2014-07-18",
+ "periodType": "DAILY",
"startDate": "2014-07-18",
},
Object {
+ "displayName": "July 19, 2014",
"endDate": "2014-07-19",
"id": "20140719",
"iso": "20140719",
"name": "2014-07-19",
+ "periodType": "DAILY",
"startDate": "2014-07-19",
},
Object {
+ "displayName": "July 20, 2014",
"endDate": "2014-07-20",
"id": "20140720",
"iso": "20140720",
"name": "2014-07-20",
+ "periodType": "DAILY",
"startDate": "2014-07-20",
},
Object {
+ "displayName": "July 21, 2014",
"endDate": "2014-07-21",
"id": "20140721",
"iso": "20140721",
"name": "2014-07-21",
+ "periodType": "DAILY",
"startDate": "2014-07-21",
},
Object {
+ "displayName": "July 22, 2014",
"endDate": "2014-07-22",
"id": "20140722",
"iso": "20140722",
"name": "2014-07-22",
+ "periodType": "DAILY",
"startDate": "2014-07-22",
},
Object {
+ "displayName": "July 23, 2014",
"endDate": "2014-07-23",
"id": "20140723",
"iso": "20140723",
"name": "2014-07-23",
+ "periodType": "DAILY",
"startDate": "2014-07-23",
},
Object {
+ "displayName": "July 24, 2014",
"endDate": "2014-07-24",
"id": "20140724",
"iso": "20140724",
"name": "2014-07-24",
+ "periodType": "DAILY",
"startDate": "2014-07-24",
},
Object {
+ "displayName": "July 25, 2014",
"endDate": "2014-07-25",
"id": "20140725",
"iso": "20140725",
"name": "2014-07-25",
+ "periodType": "DAILY",
"startDate": "2014-07-25",
},
Object {
+ "displayName": "July 26, 2014",
"endDate": "2014-07-26",
"id": "20140726",
"iso": "20140726",
"name": "2014-07-26",
+ "periodType": "DAILY",
"startDate": "2014-07-26",
},
Object {
+ "displayName": "July 27, 2014",
"endDate": "2014-07-27",
"id": "20140727",
"iso": "20140727",
"name": "2014-07-27",
+ "periodType": "DAILY",
"startDate": "2014-07-27",
},
Object {
+ "displayName": "July 28, 2014",
"endDate": "2014-07-28",
"id": "20140728",
"iso": "20140728",
"name": "2014-07-28",
+ "periodType": "DAILY",
"startDate": "2014-07-28",
},
Object {
+ "displayName": "July 29, 2014",
"endDate": "2014-07-29",
"id": "20140729",
"iso": "20140729",
"name": "2014-07-29",
+ "periodType": "DAILY",
"startDate": "2014-07-29",
},
Object {
+ "displayName": "July 30, 2014",
"endDate": "2014-07-30",
"id": "20140730",
"iso": "20140730",
"name": "2014-07-30",
+ "periodType": "DAILY",
"startDate": "2014-07-30",
},
Object {
+ "displayName": "July 31, 2014",
"endDate": "2014-07-31",
"id": "20140731",
"iso": "20140731",
"name": "2014-07-31",
+ "periodType": "DAILY",
"startDate": "2014-07-31",
},
Object {
+ "displayName": "August 1, 2014",
"endDate": "2014-08-01",
"id": "20140801",
"iso": "20140801",
"name": "2014-08-01",
+ "periodType": "DAILY",
"startDate": "2014-08-01",
},
Object {
+ "displayName": "August 2, 2014",
"endDate": "2014-08-02",
"id": "20140802",
"iso": "20140802",
"name": "2014-08-02",
+ "periodType": "DAILY",
"startDate": "2014-08-02",
},
Object {
+ "displayName": "August 3, 2014",
"endDate": "2014-08-03",
"id": "20140803",
"iso": "20140803",
"name": "2014-08-03",
+ "periodType": "DAILY",
"startDate": "2014-08-03",
},
Object {
+ "displayName": "August 4, 2014",
"endDate": "2014-08-04",
"id": "20140804",
"iso": "20140804",
"name": "2014-08-04",
+ "periodType": "DAILY",
"startDate": "2014-08-04",
},
Object {
+ "displayName": "August 5, 2014",
"endDate": "2014-08-05",
"id": "20140805",
"iso": "20140805",
"name": "2014-08-05",
+ "periodType": "DAILY",
"startDate": "2014-08-05",
},
Object {
+ "displayName": "August 6, 2014",
"endDate": "2014-08-06",
"id": "20140806",
"iso": "20140806",
"name": "2014-08-06",
+ "periodType": "DAILY",
"startDate": "2014-08-06",
},
Object {
+ "displayName": "August 7, 2014",
"endDate": "2014-08-07",
"id": "20140807",
"iso": "20140807",
"name": "2014-08-07",
+ "periodType": "DAILY",
"startDate": "2014-08-07",
},
Object {
+ "displayName": "August 8, 2014",
"endDate": "2014-08-08",
"id": "20140808",
"iso": "20140808",
"name": "2014-08-08",
+ "periodType": "DAILY",
"startDate": "2014-08-08",
},
Object {
+ "displayName": "August 9, 2014",
"endDate": "2014-08-09",
"id": "20140809",
"iso": "20140809",
"name": "2014-08-09",
+ "periodType": "DAILY",
"startDate": "2014-08-09",
},
Object {
+ "displayName": "August 10, 2014",
"endDate": "2014-08-10",
"id": "20140810",
"iso": "20140810",
"name": "2014-08-10",
+ "periodType": "DAILY",
"startDate": "2014-08-10",
},
Object {
+ "displayName": "August 11, 2014",
"endDate": "2014-08-11",
"id": "20140811",
"iso": "20140811",
"name": "2014-08-11",
+ "periodType": "DAILY",
"startDate": "2014-08-11",
},
Object {
+ "displayName": "August 12, 2014",
"endDate": "2014-08-12",
"id": "20140812",
"iso": "20140812",
"name": "2014-08-12",
+ "periodType": "DAILY",
"startDate": "2014-08-12",
},
Object {
+ "displayName": "August 13, 2014",
"endDate": "2014-08-13",
"id": "20140813",
"iso": "20140813",
"name": "2014-08-13",
+ "periodType": "DAILY",
"startDate": "2014-08-13",
},
Object {
+ "displayName": "August 14, 2014",
"endDate": "2014-08-14",
"id": "20140814",
"iso": "20140814",
"name": "2014-08-14",
+ "periodType": "DAILY",
"startDate": "2014-08-14",
},
Object {
+ "displayName": "August 15, 2014",
"endDate": "2014-08-15",
"id": "20140815",
"iso": "20140815",
"name": "2014-08-15",
+ "periodType": "DAILY",
"startDate": "2014-08-15",
},
Object {
+ "displayName": "August 16, 2014",
"endDate": "2014-08-16",
"id": "20140816",
"iso": "20140816",
"name": "2014-08-16",
+ "periodType": "DAILY",
"startDate": "2014-08-16",
},
Object {
+ "displayName": "August 17, 2014",
"endDate": "2014-08-17",
"id": "20140817",
"iso": "20140817",
"name": "2014-08-17",
+ "periodType": "DAILY",
"startDate": "2014-08-17",
},
Object {
+ "displayName": "August 18, 2014",
"endDate": "2014-08-18",
"id": "20140818",
"iso": "20140818",
"name": "2014-08-18",
+ "periodType": "DAILY",
"startDate": "2014-08-18",
},
Object {
+ "displayName": "August 19, 2014",
"endDate": "2014-08-19",
"id": "20140819",
"iso": "20140819",
"name": "2014-08-19",
+ "periodType": "DAILY",
"startDate": "2014-08-19",
},
Object {
+ "displayName": "August 20, 2014",
"endDate": "2014-08-20",
"id": "20140820",
"iso": "20140820",
"name": "2014-08-20",
+ "periodType": "DAILY",
"startDate": "2014-08-20",
},
Object {
+ "displayName": "August 21, 2014",
"endDate": "2014-08-21",
"id": "20140821",
"iso": "20140821",
"name": "2014-08-21",
+ "periodType": "DAILY",
"startDate": "2014-08-21",
},
Object {
+ "displayName": "August 22, 2014",
"endDate": "2014-08-22",
"id": "20140822",
"iso": "20140822",
"name": "2014-08-22",
+ "periodType": "DAILY",
"startDate": "2014-08-22",
},
Object {
+ "displayName": "August 23, 2014",
"endDate": "2014-08-23",
"id": "20140823",
"iso": "20140823",
"name": "2014-08-23",
+ "periodType": "DAILY",
"startDate": "2014-08-23",
},
Object {
+ "displayName": "August 24, 2014",
"endDate": "2014-08-24",
"id": "20140824",
"iso": "20140824",
"name": "2014-08-24",
+ "periodType": "DAILY",
"startDate": "2014-08-24",
},
Object {
+ "displayName": "August 25, 2014",
"endDate": "2014-08-25",
"id": "20140825",
"iso": "20140825",
"name": "2014-08-25",
+ "periodType": "DAILY",
"startDate": "2014-08-25",
},
Object {
+ "displayName": "August 26, 2014",
"endDate": "2014-08-26",
"id": "20140826",
"iso": "20140826",
"name": "2014-08-26",
+ "periodType": "DAILY",
"startDate": "2014-08-26",
},
Object {
+ "displayName": "August 27, 2014",
"endDate": "2014-08-27",
"id": "20140827",
"iso": "20140827",
"name": "2014-08-27",
+ "periodType": "DAILY",
"startDate": "2014-08-27",
},
Object {
+ "displayName": "August 28, 2014",
"endDate": "2014-08-28",
"id": "20140828",
"iso": "20140828",
"name": "2014-08-28",
+ "periodType": "DAILY",
"startDate": "2014-08-28",
},
Object {
+ "displayName": "August 29, 2014",
"endDate": "2014-08-29",
"id": "20140829",
"iso": "20140829",
"name": "2014-08-29",
+ "periodType": "DAILY",
"startDate": "2014-08-29",
},
Object {
+ "displayName": "August 30, 2014",
"endDate": "2014-08-30",
"id": "20140830",
"iso": "20140830",
"name": "2014-08-30",
+ "periodType": "DAILY",
"startDate": "2014-08-30",
},
Object {
+ "displayName": "August 31, 2014",
"endDate": "2014-08-31",
"id": "20140831",
"iso": "20140831",
"name": "2014-08-31",
+ "periodType": "DAILY",
"startDate": "2014-08-31",
},
Object {
+ "displayName": "September 1, 2014",
"endDate": "2014-09-01",
"id": "20140901",
"iso": "20140901",
"name": "2014-09-01",
+ "periodType": "DAILY",
"startDate": "2014-09-01",
},
Object {
+ "displayName": "September 2, 2014",
"endDate": "2014-09-02",
"id": "20140902",
"iso": "20140902",
"name": "2014-09-02",
+ "periodType": "DAILY",
"startDate": "2014-09-02",
},
Object {
+ "displayName": "September 3, 2014",
"endDate": "2014-09-03",
"id": "20140903",
"iso": "20140903",
"name": "2014-09-03",
+ "periodType": "DAILY",
"startDate": "2014-09-03",
},
Object {
+ "displayName": "September 4, 2014",
"endDate": "2014-09-04",
"id": "20140904",
"iso": "20140904",
"name": "2014-09-04",
+ "periodType": "DAILY",
"startDate": "2014-09-04",
},
Object {
+ "displayName": "September 5, 2014",
"endDate": "2014-09-05",
"id": "20140905",
"iso": "20140905",
"name": "2014-09-05",
+ "periodType": "DAILY",
"startDate": "2014-09-05",
},
Object {
+ "displayName": "September 6, 2014",
"endDate": "2014-09-06",
"id": "20140906",
"iso": "20140906",
"name": "2014-09-06",
+ "periodType": "DAILY",
"startDate": "2014-09-06",
},
Object {
+ "displayName": "September 7, 2014",
"endDate": "2014-09-07",
"id": "20140907",
"iso": "20140907",
"name": "2014-09-07",
+ "periodType": "DAILY",
"startDate": "2014-09-07",
},
Object {
+ "displayName": "September 8, 2014",
"endDate": "2014-09-08",
"id": "20140908",
"iso": "20140908",
"name": "2014-09-08",
+ "periodType": "DAILY",
"startDate": "2014-09-08",
},
Object {
+ "displayName": "September 9, 2014",
"endDate": "2014-09-09",
"id": "20140909",
"iso": "20140909",
"name": "2014-09-09",
+ "periodType": "DAILY",
"startDate": "2014-09-09",
},
Object {
+ "displayName": "September 10, 2014",
"endDate": "2014-09-10",
"id": "20140910",
"iso": "20140910",
"name": "2014-09-10",
+ "periodType": "DAILY",
"startDate": "2014-09-10",
},
Object {
+ "displayName": "September 11, 2014",
"endDate": "2014-09-11",
"id": "20140911",
"iso": "20140911",
"name": "2014-09-11",
+ "periodType": "DAILY",
"startDate": "2014-09-11",
},
Object {
+ "displayName": "September 12, 2014",
"endDate": "2014-09-12",
"id": "20140912",
"iso": "20140912",
"name": "2014-09-12",
+ "periodType": "DAILY",
"startDate": "2014-09-12",
},
Object {
+ "displayName": "September 13, 2014",
"endDate": "2014-09-13",
"id": "20140913",
"iso": "20140913",
"name": "2014-09-13",
+ "periodType": "DAILY",
"startDate": "2014-09-13",
},
Object {
+ "displayName": "September 14, 2014",
"endDate": "2014-09-14",
"id": "20140914",
"iso": "20140914",
"name": "2014-09-14",
+ "periodType": "DAILY",
"startDate": "2014-09-14",
},
Object {
+ "displayName": "September 15, 2014",
"endDate": "2014-09-15",
"id": "20140915",
"iso": "20140915",
"name": "2014-09-15",
+ "periodType": "DAILY",
"startDate": "2014-09-15",
},
Object {
+ "displayName": "September 16, 2014",
"endDate": "2014-09-16",
"id": "20140916",
"iso": "20140916",
"name": "2014-09-16",
+ "periodType": "DAILY",
"startDate": "2014-09-16",
},
Object {
+ "displayName": "September 17, 2014",
"endDate": "2014-09-17",
"id": "20140917",
"iso": "20140917",
"name": "2014-09-17",
+ "periodType": "DAILY",
"startDate": "2014-09-17",
},
Object {
+ "displayName": "September 18, 2014",
"endDate": "2014-09-18",
"id": "20140918",
"iso": "20140918",
"name": "2014-09-18",
+ "periodType": "DAILY",
"startDate": "2014-09-18",
},
Object {
+ "displayName": "September 19, 2014",
"endDate": "2014-09-19",
"id": "20140919",
"iso": "20140919",
"name": "2014-09-19",
+ "periodType": "DAILY",
"startDate": "2014-09-19",
},
Object {
+ "displayName": "September 20, 2014",
"endDate": "2014-09-20",
"id": "20140920",
"iso": "20140920",
"name": "2014-09-20",
+ "periodType": "DAILY",
"startDate": "2014-09-20",
},
Object {
+ "displayName": "September 21, 2014",
"endDate": "2014-09-21",
"id": "20140921",
"iso": "20140921",
"name": "2014-09-21",
+ "periodType": "DAILY",
"startDate": "2014-09-21",
},
Object {
+ "displayName": "September 22, 2014",
"endDate": "2014-09-22",
"id": "20140922",
"iso": "20140922",
"name": "2014-09-22",
+ "periodType": "DAILY",
"startDate": "2014-09-22",
},
Object {
+ "displayName": "September 23, 2014",
"endDate": "2014-09-23",
"id": "20140923",
"iso": "20140923",
"name": "2014-09-23",
+ "periodType": "DAILY",
"startDate": "2014-09-23",
},
Object {
+ "displayName": "September 24, 2014",
"endDate": "2014-09-24",
"id": "20140924",
"iso": "20140924",
"name": "2014-09-24",
+ "periodType": "DAILY",
"startDate": "2014-09-24",
},
Object {
+ "displayName": "September 25, 2014",
"endDate": "2014-09-25",
"id": "20140925",
"iso": "20140925",
"name": "2014-09-25",
+ "periodType": "DAILY",
"startDate": "2014-09-25",
},
Object {
+ "displayName": "September 26, 2014",
"endDate": "2014-09-26",
"id": "20140926",
"iso": "20140926",
"name": "2014-09-26",
+ "periodType": "DAILY",
"startDate": "2014-09-26",
},
Object {
+ "displayName": "September 27, 2014",
"endDate": "2014-09-27",
"id": "20140927",
"iso": "20140927",
"name": "2014-09-27",
+ "periodType": "DAILY",
"startDate": "2014-09-27",
},
Object {
+ "displayName": "September 28, 2014",
"endDate": "2014-09-28",
"id": "20140928",
"iso": "20140928",
"name": "2014-09-28",
+ "periodType": "DAILY",
"startDate": "2014-09-28",
},
Object {
+ "displayName": "September 29, 2014",
"endDate": "2014-09-29",
"id": "20140929",
"iso": "20140929",
"name": "2014-09-29",
+ "periodType": "DAILY",
"startDate": "2014-09-29",
},
Object {
+ "displayName": "September 30, 2014",
"endDate": "2014-09-30",
"id": "20140930",
"iso": "20140930",
"name": "2014-09-30",
+ "periodType": "DAILY",
"startDate": "2014-09-30",
},
Object {
+ "displayName": "October 1, 2014",
"endDate": "2014-10-01",
"id": "20141001",
"iso": "20141001",
"name": "2014-10-01",
+ "periodType": "DAILY",
"startDate": "2014-10-01",
},
Object {
+ "displayName": "October 2, 2014",
"endDate": "2014-10-02",
"id": "20141002",
"iso": "20141002",
"name": "2014-10-02",
+ "periodType": "DAILY",
"startDate": "2014-10-02",
},
Object {
+ "displayName": "October 3, 2014",
"endDate": "2014-10-03",
"id": "20141003",
"iso": "20141003",
"name": "2014-10-03",
+ "periodType": "DAILY",
"startDate": "2014-10-03",
},
Object {
+ "displayName": "October 4, 2014",
"endDate": "2014-10-04",
"id": "20141004",
"iso": "20141004",
"name": "2014-10-04",
+ "periodType": "DAILY",
"startDate": "2014-10-04",
},
Object {
+ "displayName": "October 5, 2014",
"endDate": "2014-10-05",
"id": "20141005",
"iso": "20141005",
"name": "2014-10-05",
+ "periodType": "DAILY",
"startDate": "2014-10-05",
},
Object {
+ "displayName": "October 6, 2014",
"endDate": "2014-10-06",
"id": "20141006",
"iso": "20141006",
"name": "2014-10-06",
+ "periodType": "DAILY",
"startDate": "2014-10-06",
},
Object {
+ "displayName": "October 7, 2014",
"endDate": "2014-10-07",
"id": "20141007",
"iso": "20141007",
"name": "2014-10-07",
+ "periodType": "DAILY",
"startDate": "2014-10-07",
},
Object {
+ "displayName": "October 8, 2014",
"endDate": "2014-10-08",
"id": "20141008",
"iso": "20141008",
"name": "2014-10-08",
+ "periodType": "DAILY",
"startDate": "2014-10-08",
},
Object {
+ "displayName": "October 9, 2014",
"endDate": "2014-10-09",
"id": "20141009",
"iso": "20141009",
"name": "2014-10-09",
+ "periodType": "DAILY",
"startDate": "2014-10-09",
},
Object {
+ "displayName": "October 10, 2014",
"endDate": "2014-10-10",
"id": "20141010",
"iso": "20141010",
"name": "2014-10-10",
+ "periodType": "DAILY",
"startDate": "2014-10-10",
},
Object {
+ "displayName": "October 11, 2014",
"endDate": "2014-10-11",
"id": "20141011",
"iso": "20141011",
"name": "2014-10-11",
+ "periodType": "DAILY",
"startDate": "2014-10-11",
},
Object {
+ "displayName": "October 12, 2014",
"endDate": "2014-10-12",
"id": "20141012",
"iso": "20141012",
"name": "2014-10-12",
+ "periodType": "DAILY",
"startDate": "2014-10-12",
},
Object {
+ "displayName": "October 13, 2014",
"endDate": "2014-10-13",
"id": "20141013",
"iso": "20141013",
"name": "2014-10-13",
+ "periodType": "DAILY",
"startDate": "2014-10-13",
},
Object {
+ "displayName": "October 14, 2014",
"endDate": "2014-10-14",
"id": "20141014",
"iso": "20141014",
"name": "2014-10-14",
+ "periodType": "DAILY",
"startDate": "2014-10-14",
},
Object {
+ "displayName": "October 15, 2014",
"endDate": "2014-10-15",
"id": "20141015",
"iso": "20141015",
"name": "2014-10-15",
+ "periodType": "DAILY",
"startDate": "2014-10-15",
},
Object {
+ "displayName": "October 16, 2014",
"endDate": "2014-10-16",
"id": "20141016",
"iso": "20141016",
"name": "2014-10-16",
+ "periodType": "DAILY",
"startDate": "2014-10-16",
},
Object {
+ "displayName": "October 17, 2014",
"endDate": "2014-10-17",
"id": "20141017",
"iso": "20141017",
"name": "2014-10-17",
+ "periodType": "DAILY",
"startDate": "2014-10-17",
},
Object {
+ "displayName": "October 18, 2014",
"endDate": "2014-10-18",
"id": "20141018",
"iso": "20141018",
"name": "2014-10-18",
+ "periodType": "DAILY",
"startDate": "2014-10-18",
},
Object {
+ "displayName": "October 19, 2014",
"endDate": "2014-10-19",
"id": "20141019",
"iso": "20141019",
"name": "2014-10-19",
+ "periodType": "DAILY",
"startDate": "2014-10-19",
},
Object {
+ "displayName": "October 20, 2014",
"endDate": "2014-10-20",
"id": "20141020",
"iso": "20141020",
"name": "2014-10-20",
+ "periodType": "DAILY",
"startDate": "2014-10-20",
},
Object {
+ "displayName": "October 21, 2014",
"endDate": "2014-10-21",
"id": "20141021",
"iso": "20141021",
"name": "2014-10-21",
+ "periodType": "DAILY",
"startDate": "2014-10-21",
},
Object {
+ "displayName": "October 22, 2014",
"endDate": "2014-10-22",
"id": "20141022",
"iso": "20141022",
"name": "2014-10-22",
+ "periodType": "DAILY",
"startDate": "2014-10-22",
},
Object {
+ "displayName": "October 23, 2014",
"endDate": "2014-10-23",
"id": "20141023",
"iso": "20141023",
"name": "2014-10-23",
+ "periodType": "DAILY",
"startDate": "2014-10-23",
},
Object {
+ "displayName": "October 24, 2014",
"endDate": "2014-10-24",
"id": "20141024",
"iso": "20141024",
"name": "2014-10-24",
+ "periodType": "DAILY",
"startDate": "2014-10-24",
},
Object {
+ "displayName": "October 25, 2014",
"endDate": "2014-10-25",
"id": "20141025",
"iso": "20141025",
"name": "2014-10-25",
+ "periodType": "DAILY",
"startDate": "2014-10-25",
},
Object {
+ "displayName": "October 26, 2014",
"endDate": "2014-10-26",
"id": "20141026",
"iso": "20141026",
"name": "2014-10-26",
+ "periodType": "DAILY",
"startDate": "2014-10-26",
},
Object {
+ "displayName": "October 27, 2014",
"endDate": "2014-10-27",
"id": "20141027",
"iso": "20141027",
"name": "2014-10-27",
+ "periodType": "DAILY",
"startDate": "2014-10-27",
},
Object {
+ "displayName": "October 28, 2014",
"endDate": "2014-10-28",
"id": "20141028",
"iso": "20141028",
"name": "2014-10-28",
+ "periodType": "DAILY",
"startDate": "2014-10-28",
},
Object {
+ "displayName": "October 29, 2014",
"endDate": "2014-10-29",
"id": "20141029",
"iso": "20141029",
"name": "2014-10-29",
+ "periodType": "DAILY",
"startDate": "2014-10-29",
},
Object {
+ "displayName": "October 30, 2014",
"endDate": "2014-10-30",
"id": "20141030",
"iso": "20141030",
"name": "2014-10-30",
+ "periodType": "DAILY",
"startDate": "2014-10-30",
},
Object {
+ "displayName": "October 31, 2014",
"endDate": "2014-10-31",
"id": "20141031",
"iso": "20141031",
"name": "2014-10-31",
+ "periodType": "DAILY",
"startDate": "2014-10-31",
},
Object {
+ "displayName": "November 1, 2014",
"endDate": "2014-11-01",
"id": "20141101",
"iso": "20141101",
"name": "2014-11-01",
+ "periodType": "DAILY",
"startDate": "2014-11-01",
},
Object {
+ "displayName": "November 2, 2014",
"endDate": "2014-11-02",
"id": "20141102",
"iso": "20141102",
"name": "2014-11-02",
+ "periodType": "DAILY",
"startDate": "2014-11-02",
},
Object {
+ "displayName": "November 3, 2014",
"endDate": "2014-11-03",
"id": "20141103",
"iso": "20141103",
"name": "2014-11-03",
+ "periodType": "DAILY",
"startDate": "2014-11-03",
},
Object {
+ "displayName": "November 4, 2014",
"endDate": "2014-11-04",
"id": "20141104",
"iso": "20141104",
"name": "2014-11-04",
+ "periodType": "DAILY",
"startDate": "2014-11-04",
},
Object {
+ "displayName": "November 5, 2014",
"endDate": "2014-11-05",
"id": "20141105",
"iso": "20141105",
"name": "2014-11-05",
+ "periodType": "DAILY",
"startDate": "2014-11-05",
},
Object {
+ "displayName": "November 6, 2014",
"endDate": "2014-11-06",
"id": "20141106",
"iso": "20141106",
"name": "2014-11-06",
+ "periodType": "DAILY",
"startDate": "2014-11-06",
},
Object {
+ "displayName": "November 7, 2014",
"endDate": "2014-11-07",
"id": "20141107",
"iso": "20141107",
"name": "2014-11-07",
+ "periodType": "DAILY",
"startDate": "2014-11-07",
},
Object {
+ "displayName": "November 8, 2014",
"endDate": "2014-11-08",
"id": "20141108",
"iso": "20141108",
"name": "2014-11-08",
+ "periodType": "DAILY",
"startDate": "2014-11-08",
},
Object {
+ "displayName": "November 9, 2014",
"endDate": "2014-11-09",
"id": "20141109",
"iso": "20141109",
"name": "2014-11-09",
+ "periodType": "DAILY",
"startDate": "2014-11-09",
},
Object {
+ "displayName": "November 10, 2014",
"endDate": "2014-11-10",
"id": "20141110",
"iso": "20141110",
"name": "2014-11-10",
+ "periodType": "DAILY",
"startDate": "2014-11-10",
},
Object {
+ "displayName": "November 11, 2014",
"endDate": "2014-11-11",
"id": "20141111",
"iso": "20141111",
"name": "2014-11-11",
+ "periodType": "DAILY",
"startDate": "2014-11-11",
},
Object {
+ "displayName": "November 12, 2014",
"endDate": "2014-11-12",
"id": "20141112",
"iso": "20141112",
"name": "2014-11-12",
+ "periodType": "DAILY",
"startDate": "2014-11-12",
},
Object {
+ "displayName": "November 13, 2014",
"endDate": "2014-11-13",
"id": "20141113",
"iso": "20141113",
"name": "2014-11-13",
+ "periodType": "DAILY",
"startDate": "2014-11-13",
},
Object {
+ "displayName": "November 14, 2014",
"endDate": "2014-11-14",
"id": "20141114",
"iso": "20141114",
"name": "2014-11-14",
+ "periodType": "DAILY",
"startDate": "2014-11-14",
},
Object {
+ "displayName": "November 15, 2014",
"endDate": "2014-11-15",
"id": "20141115",
"iso": "20141115",
"name": "2014-11-15",
+ "periodType": "DAILY",
"startDate": "2014-11-15",
},
Object {
+ "displayName": "November 16, 2014",
"endDate": "2014-11-16",
"id": "20141116",
"iso": "20141116",
"name": "2014-11-16",
+ "periodType": "DAILY",
"startDate": "2014-11-16",
},
Object {
+ "displayName": "November 17, 2014",
"endDate": "2014-11-17",
"id": "20141117",
"iso": "20141117",
"name": "2014-11-17",
+ "periodType": "DAILY",
"startDate": "2014-11-17",
},
Object {
+ "displayName": "November 18, 2014",
"endDate": "2014-11-18",
"id": "20141118",
"iso": "20141118",
"name": "2014-11-18",
+ "periodType": "DAILY",
"startDate": "2014-11-18",
},
Object {
+ "displayName": "November 19, 2014",
"endDate": "2014-11-19",
"id": "20141119",
"iso": "20141119",
"name": "2014-11-19",
+ "periodType": "DAILY",
"startDate": "2014-11-19",
},
Object {
+ "displayName": "November 20, 2014",
"endDate": "2014-11-20",
"id": "20141120",
"iso": "20141120",
"name": "2014-11-20",
+ "periodType": "DAILY",
"startDate": "2014-11-20",
},
Object {
+ "displayName": "November 21, 2014",
"endDate": "2014-11-21",
"id": "20141121",
"iso": "20141121",
"name": "2014-11-21",
+ "periodType": "DAILY",
"startDate": "2014-11-21",
},
Object {
+ "displayName": "November 22, 2014",
"endDate": "2014-11-22",
"id": "20141122",
"iso": "20141122",
"name": "2014-11-22",
+ "periodType": "DAILY",
"startDate": "2014-11-22",
},
Object {
+ "displayName": "November 23, 2014",
"endDate": "2014-11-23",
"id": "20141123",
"iso": "20141123",
"name": "2014-11-23",
+ "periodType": "DAILY",
"startDate": "2014-11-23",
},
Object {
+ "displayName": "November 24, 2014",
"endDate": "2014-11-24",
"id": "20141124",
"iso": "20141124",
"name": "2014-11-24",
+ "periodType": "DAILY",
"startDate": "2014-11-24",
},
Object {
+ "displayName": "November 25, 2014",
"endDate": "2014-11-25",
"id": "20141125",
"iso": "20141125",
"name": "2014-11-25",
+ "periodType": "DAILY",
"startDate": "2014-11-25",
},
Object {
+ "displayName": "November 26, 2014",
"endDate": "2014-11-26",
"id": "20141126",
"iso": "20141126",
"name": "2014-11-26",
+ "periodType": "DAILY",
"startDate": "2014-11-26",
},
Object {
+ "displayName": "November 27, 2014",
"endDate": "2014-11-27",
"id": "20141127",
"iso": "20141127",
"name": "2014-11-27",
+ "periodType": "DAILY",
"startDate": "2014-11-27",
},
Object {
+ "displayName": "November 28, 2014",
"endDate": "2014-11-28",
"id": "20141128",
"iso": "20141128",
"name": "2014-11-28",
+ "periodType": "DAILY",
"startDate": "2014-11-28",
},
Object {
+ "displayName": "November 29, 2014",
"endDate": "2014-11-29",
"id": "20141129",
"iso": "20141129",
"name": "2014-11-29",
+ "periodType": "DAILY",
"startDate": "2014-11-29",
},
Object {
+ "displayName": "November 30, 2014",
"endDate": "2014-11-30",
"id": "20141130",
"iso": "20141130",
"name": "2014-11-30",
+ "periodType": "DAILY",
"startDate": "2014-11-30",
},
Object {
+ "displayName": "December 1, 2014",
"endDate": "2014-12-01",
"id": "20141201",
"iso": "20141201",
"name": "2014-12-01",
+ "periodType": "DAILY",
"startDate": "2014-12-01",
},
Object {
+ "displayName": "December 2, 2014",
"endDate": "2014-12-02",
"id": "20141202",
"iso": "20141202",
"name": "2014-12-02",
+ "periodType": "DAILY",
"startDate": "2014-12-02",
},
Object {
+ "displayName": "December 3, 2014",
"endDate": "2014-12-03",
"id": "20141203",
"iso": "20141203",
"name": "2014-12-03",
+ "periodType": "DAILY",
"startDate": "2014-12-03",
},
Object {
+ "displayName": "December 4, 2014",
"endDate": "2014-12-04",
"id": "20141204",
"iso": "20141204",
"name": "2014-12-04",
+ "periodType": "DAILY",
"startDate": "2014-12-04",
},
Object {
+ "displayName": "December 5, 2014",
"endDate": "2014-12-05",
"id": "20141205",
"iso": "20141205",
"name": "2014-12-05",
+ "periodType": "DAILY",
"startDate": "2014-12-05",
},
Object {
+ "displayName": "December 6, 2014",
"endDate": "2014-12-06",
"id": "20141206",
"iso": "20141206",
"name": "2014-12-06",
+ "periodType": "DAILY",
"startDate": "2014-12-06",
},
Object {
+ "displayName": "December 7, 2014",
"endDate": "2014-12-07",
"id": "20141207",
"iso": "20141207",
"name": "2014-12-07",
+ "periodType": "DAILY",
"startDate": "2014-12-07",
},
Object {
+ "displayName": "December 8, 2014",
"endDate": "2014-12-08",
"id": "20141208",
"iso": "20141208",
"name": "2014-12-08",
+ "periodType": "DAILY",
"startDate": "2014-12-08",
},
Object {
+ "displayName": "December 9, 2014",
"endDate": "2014-12-09",
"id": "20141209",
"iso": "20141209",
"name": "2014-12-09",
+ "periodType": "DAILY",
"startDate": "2014-12-09",
},
Object {
+ "displayName": "December 10, 2014",
"endDate": "2014-12-10",
"id": "20141210",
"iso": "20141210",
"name": "2014-12-10",
+ "periodType": "DAILY",
"startDate": "2014-12-10",
},
Object {
+ "displayName": "December 11, 2014",
"endDate": "2014-12-11",
"id": "20141211",
"iso": "20141211",
"name": "2014-12-11",
+ "periodType": "DAILY",
"startDate": "2014-12-11",
},
Object {
+ "displayName": "December 12, 2014",
"endDate": "2014-12-12",
"id": "20141212",
"iso": "20141212",
"name": "2014-12-12",
+ "periodType": "DAILY",
"startDate": "2014-12-12",
},
Object {
+ "displayName": "December 13, 2014",
"endDate": "2014-12-13",
"id": "20141213",
"iso": "20141213",
"name": "2014-12-13",
+ "periodType": "DAILY",
"startDate": "2014-12-13",
},
Object {
+ "displayName": "December 14, 2014",
"endDate": "2014-12-14",
"id": "20141214",
"iso": "20141214",
"name": "2014-12-14",
+ "periodType": "DAILY",
"startDate": "2014-12-14",
},
Object {
+ "displayName": "December 15, 2014",
"endDate": "2014-12-15",
"id": "20141215",
"iso": "20141215",
"name": "2014-12-15",
+ "periodType": "DAILY",
"startDate": "2014-12-15",
},
Object {
+ "displayName": "December 16, 2014",
"endDate": "2014-12-16",
"id": "20141216",
"iso": "20141216",
"name": "2014-12-16",
+ "periodType": "DAILY",
"startDate": "2014-12-16",
},
Object {
+ "displayName": "December 17, 2014",
"endDate": "2014-12-17",
"id": "20141217",
"iso": "20141217",
"name": "2014-12-17",
+ "periodType": "DAILY",
"startDate": "2014-12-17",
},
Object {
+ "displayName": "December 18, 2014",
"endDate": "2014-12-18",
"id": "20141218",
"iso": "20141218",
"name": "2014-12-18",
+ "periodType": "DAILY",
"startDate": "2014-12-18",
},
Object {
+ "displayName": "December 19, 2014",
"endDate": "2014-12-19",
"id": "20141219",
"iso": "20141219",
"name": "2014-12-19",
+ "periodType": "DAILY",
"startDate": "2014-12-19",
},
Object {
+ "displayName": "December 20, 2014",
"endDate": "2014-12-20",
"id": "20141220",
"iso": "20141220",
"name": "2014-12-20",
+ "periodType": "DAILY",
"startDate": "2014-12-20",
},
Object {
+ "displayName": "December 21, 2014",
"endDate": "2014-12-21",
"id": "20141221",
"iso": "20141221",
"name": "2014-12-21",
+ "periodType": "DAILY",
"startDate": "2014-12-21",
},
Object {
+ "displayName": "December 22, 2014",
"endDate": "2014-12-22",
"id": "20141222",
"iso": "20141222",
"name": "2014-12-22",
+ "periodType": "DAILY",
"startDate": "2014-12-22",
},
Object {
+ "displayName": "December 23, 2014",
"endDate": "2014-12-23",
"id": "20141223",
"iso": "20141223",
"name": "2014-12-23",
+ "periodType": "DAILY",
"startDate": "2014-12-23",
},
Object {
+ "displayName": "December 24, 2014",
"endDate": "2014-12-24",
"id": "20141224",
"iso": "20141224",
"name": "2014-12-24",
+ "periodType": "DAILY",
"startDate": "2014-12-24",
},
Object {
+ "displayName": "December 25, 2014",
"endDate": "2014-12-25",
"id": "20141225",
"iso": "20141225",
"name": "2014-12-25",
+ "periodType": "DAILY",
"startDate": "2014-12-25",
},
Object {
+ "displayName": "December 26, 2014",
"endDate": "2014-12-26",
"id": "20141226",
"iso": "20141226",
"name": "2014-12-26",
+ "periodType": "DAILY",
"startDate": "2014-12-26",
},
Object {
+ "displayName": "December 27, 2014",
"endDate": "2014-12-27",
"id": "20141227",
"iso": "20141227",
"name": "2014-12-27",
+ "periodType": "DAILY",
"startDate": "2014-12-27",
},
Object {
+ "displayName": "December 28, 2014",
"endDate": "2014-12-28",
"id": "20141228",
"iso": "20141228",
"name": "2014-12-28",
+ "periodType": "DAILY",
"startDate": "2014-12-28",
},
Object {
+ "displayName": "December 29, 2014",
"endDate": "2014-12-29",
"id": "20141229",
"iso": "20141229",
"name": "2014-12-29",
+ "periodType": "DAILY",
"startDate": "2014-12-29",
},
Object {
+ "displayName": "December 30, 2014",
"endDate": "2014-12-30",
"id": "20141230",
"iso": "20141230",
"name": "2014-12-30",
+ "periodType": "DAILY",
"startDate": "2014-12-30",
},
Object {
+ "displayName": "December 31, 2014",
"endDate": "2014-12-31",
"id": "20141231",
"iso": "20141231",
"name": "2014-12-31",
+ "periodType": "DAILY",
"startDate": "2014-12-31",
},
],
@@ -2815,63 +3609,93 @@ Object {
"name": "Financial year (Start April)",
"options": Array [
Object {
+ "displayName": "April 2005 - March 2006",
"endDate": "2006-03-31",
"id": "2005April",
+ "iso": "2005April",
"name": "April 2005 - March 2006",
+ "periodType": "FYAPR",
"startDate": "2005-04-01",
},
Object {
+ "displayName": "April 2006 - March 2007",
"endDate": "2007-03-31",
"id": "2006April",
+ "iso": "2006April",
"name": "April 2006 - March 2007",
+ "periodType": "FYAPR",
"startDate": "2006-04-01",
},
Object {
+ "displayName": "April 2007 - March 2008",
"endDate": "2008-03-31",
"id": "2007April",
+ "iso": "2007April",
"name": "April 2007 - March 2008",
+ "periodType": "FYAPR",
"startDate": "2007-04-01",
},
Object {
+ "displayName": "April 2008 - March 2009",
"endDate": "2009-03-31",
"id": "2008April",
+ "iso": "2008April",
"name": "April 2008 - March 2009",
+ "periodType": "FYAPR",
"startDate": "2008-04-01",
},
Object {
+ "displayName": "April 2009 - March 2010",
"endDate": "2010-03-31",
"id": "2009April",
+ "iso": "2009April",
"name": "April 2009 - March 2010",
+ "periodType": "FYAPR",
"startDate": "2009-04-01",
},
Object {
+ "displayName": "April 2010 - March 2011",
"endDate": "2011-03-31",
"id": "2010April",
+ "iso": "2010April",
"name": "April 2010 - March 2011",
+ "periodType": "FYAPR",
"startDate": "2010-04-01",
},
Object {
+ "displayName": "April 2011 - March 2012",
"endDate": "2012-03-31",
"id": "2011April",
+ "iso": "2011April",
"name": "April 2011 - March 2012",
+ "periodType": "FYAPR",
"startDate": "2011-04-01",
},
Object {
+ "displayName": "April 2012 - March 2013",
"endDate": "2013-03-31",
"id": "2012April",
+ "iso": "2012April",
"name": "April 2012 - March 2013",
+ "periodType": "FYAPR",
"startDate": "2012-04-01",
},
Object {
+ "displayName": "April 2013 - March 2014",
"endDate": "2014-03-31",
"id": "2013April",
+ "iso": "2013April",
"name": "April 2013 - March 2014",
+ "periodType": "FYAPR",
"startDate": "2013-04-01",
},
Object {
+ "displayName": "April 2014 - March 2015",
"endDate": "2015-03-31",
"id": "2014April",
+ "iso": "2014April",
"name": "April 2014 - March 2015",
+ "periodType": "FYAPR",
"startDate": "2014-04-01",
},
],
@@ -2885,63 +3709,93 @@ Object {
"name": "Financial year (Start July)",
"options": Array [
Object {
+ "displayName": "July 2005 - June 2006",
"endDate": "2006-06-30",
"id": "2005July",
+ "iso": "2005July",
"name": "July 2005 - June 2006",
+ "periodType": "FYJUL",
"startDate": "2005-07-01",
},
Object {
+ "displayName": "July 2006 - June 2007",
"endDate": "2007-06-30",
"id": "2006July",
+ "iso": "2006July",
"name": "July 2006 - June 2007",
+ "periodType": "FYJUL",
"startDate": "2006-07-01",
},
Object {
+ "displayName": "July 2007 - June 2008",
"endDate": "2008-06-30",
"id": "2007July",
+ "iso": "2007July",
"name": "July 2007 - June 2008",
+ "periodType": "FYJUL",
"startDate": "2007-07-01",
},
Object {
+ "displayName": "July 2008 - June 2009",
"endDate": "2009-06-30",
"id": "2008July",
+ "iso": "2008July",
"name": "July 2008 - June 2009",
+ "periodType": "FYJUL",
"startDate": "2008-07-01",
},
Object {
+ "displayName": "July 2009 - June 2010",
"endDate": "2010-06-30",
"id": "2009July",
+ "iso": "2009July",
"name": "July 2009 - June 2010",
+ "periodType": "FYJUL",
"startDate": "2009-07-01",
},
Object {
+ "displayName": "July 2010 - June 2011",
"endDate": "2011-06-30",
"id": "2010July",
+ "iso": "2010July",
"name": "July 2010 - June 2011",
+ "periodType": "FYJUL",
"startDate": "2010-07-01",
},
Object {
+ "displayName": "July 2011 - June 2012",
"endDate": "2012-06-30",
"id": "2011July",
+ "iso": "2011July",
"name": "July 2011 - June 2012",
+ "periodType": "FYJUL",
"startDate": "2011-07-01",
},
Object {
+ "displayName": "July 2012 - June 2013",
"endDate": "2013-06-30",
"id": "2012July",
+ "iso": "2012July",
"name": "July 2012 - June 2013",
+ "periodType": "FYJUL",
"startDate": "2012-07-01",
},
Object {
+ "displayName": "July 2013 - June 2014",
"endDate": "2014-06-30",
"id": "2013July",
+ "iso": "2013July",
"name": "July 2013 - June 2014",
+ "periodType": "FYJUL",
"startDate": "2013-07-01",
},
Object {
+ "displayName": "July 2014 - June 2015",
"endDate": "2015-06-30",
"id": "2014July",
+ "iso": "2014July",
"name": "July 2014 - June 2015",
+ "periodType": "FYJUL",
"startDate": "2014-07-01",
},
],
@@ -2955,63 +3809,93 @@ Object {
"name": "Financial year (Start November)",
"options": Array [
Object {
+ "displayName": "November 2005 - October 2006",
"endDate": "2006-10-31",
"id": "2005Nov",
+ "iso": "2005Nov",
"name": "November 2005 - October 2006",
+ "periodType": "FYNOV",
"startDate": "2005-11-01",
},
Object {
+ "displayName": "November 2006 - October 2007",
"endDate": "2007-10-31",
"id": "2006Nov",
+ "iso": "2006Nov",
"name": "November 2006 - October 2007",
+ "periodType": "FYNOV",
"startDate": "2006-11-01",
},
Object {
+ "displayName": "November 2007 - October 2008",
"endDate": "2008-10-31",
"id": "2007Nov",
+ "iso": "2007Nov",
"name": "November 2007 - October 2008",
+ "periodType": "FYNOV",
"startDate": "2007-11-01",
},
Object {
+ "displayName": "November 2008 - October 2009",
"endDate": "2009-10-31",
"id": "2008Nov",
+ "iso": "2008Nov",
"name": "November 2008 - October 2009",
+ "periodType": "FYNOV",
"startDate": "2008-11-01",
},
Object {
+ "displayName": "November 2009 - October 2010",
"endDate": "2010-10-31",
"id": "2009Nov",
+ "iso": "2009Nov",
"name": "November 2009 - October 2010",
+ "periodType": "FYNOV",
"startDate": "2009-11-01",
},
Object {
+ "displayName": "November 2010 - October 2011",
"endDate": "2011-10-31",
"id": "2010Nov",
+ "iso": "2010Nov",
"name": "November 2010 - October 2011",
+ "periodType": "FYNOV",
"startDate": "2010-11-01",
},
Object {
+ "displayName": "November 2011 - October 2012",
"endDate": "2012-10-31",
"id": "2011Nov",
+ "iso": "2011Nov",
"name": "November 2011 - October 2012",
+ "periodType": "FYNOV",
"startDate": "2011-11-01",
},
Object {
+ "displayName": "November 2012 - October 2013",
"endDate": "2013-10-31",
"id": "2012Nov",
+ "iso": "2012Nov",
"name": "November 2012 - October 2013",
+ "periodType": "FYNOV",
"startDate": "2012-11-01",
},
Object {
+ "displayName": "November 2013 - October 2014",
"endDate": "2014-10-31",
"id": "2013Nov",
+ "iso": "2013Nov",
"name": "November 2013 - October 2014",
+ "periodType": "FYNOV",
"startDate": "2013-11-01",
},
Object {
+ "displayName": "November 2014 - October 2015",
"endDate": "2015-10-31",
"id": "2014Nov",
+ "iso": "2014Nov",
"name": "November 2014 - October 2015",
+ "periodType": "FYNOV",
"startDate": "2014-11-01",
},
],
@@ -3025,63 +3909,93 @@ Object {
"name": "Financial year (Start October)",
"options": Array [
Object {
+ "displayName": "October 2005 - September 2006",
"endDate": "2006-09-30",
"id": "2005Oct",
+ "iso": "2005Oct",
"name": "October 2005 - September 2006",
+ "periodType": "FYOCT",
"startDate": "2005-10-01",
},
Object {
+ "displayName": "October 2006 - September 2007",
"endDate": "2007-09-30",
"id": "2006Oct",
+ "iso": "2006Oct",
"name": "October 2006 - September 2007",
+ "periodType": "FYOCT",
"startDate": "2006-10-01",
},
Object {
+ "displayName": "October 2007 - September 2008",
"endDate": "2008-09-30",
"id": "2007Oct",
+ "iso": "2007Oct",
"name": "October 2007 - September 2008",
+ "periodType": "FYOCT",
"startDate": "2007-10-01",
},
Object {
+ "displayName": "October 2008 - September 2009",
"endDate": "2009-09-30",
"id": "2008Oct",
+ "iso": "2008Oct",
"name": "October 2008 - September 2009",
+ "periodType": "FYOCT",
"startDate": "2008-10-01",
},
Object {
+ "displayName": "October 2009 - September 2010",
"endDate": "2010-09-30",
"id": "2009Oct",
+ "iso": "2009Oct",
"name": "October 2009 - September 2010",
+ "periodType": "FYOCT",
"startDate": "2009-10-01",
},
Object {
+ "displayName": "October 2010 - September 2011",
"endDate": "2011-09-30",
"id": "2010Oct",
+ "iso": "2010Oct",
"name": "October 2010 - September 2011",
+ "periodType": "FYOCT",
"startDate": "2010-10-01",
},
Object {
+ "displayName": "October 2011 - September 2012",
"endDate": "2012-09-30",
"id": "2011Oct",
+ "iso": "2011Oct",
"name": "October 2011 - September 2012",
+ "periodType": "FYOCT",
"startDate": "2011-10-01",
},
Object {
+ "displayName": "October 2012 - September 2013",
"endDate": "2013-09-30",
"id": "2012Oct",
+ "iso": "2012Oct",
"name": "October 2012 - September 2013",
+ "periodType": "FYOCT",
"startDate": "2012-10-01",
},
Object {
+ "displayName": "October 2013 - September 2014",
"endDate": "2014-09-30",
"id": "2013Oct",
+ "iso": "2013Oct",
"name": "October 2013 - September 2014",
+ "periodType": "FYOCT",
"startDate": "2013-10-01",
},
Object {
+ "displayName": "October 2014 - September 2015",
"endDate": "2015-09-30",
"id": "2014Oct",
+ "iso": "2014Oct",
"name": "October 2014 - September 2015",
+ "periodType": "FYOCT",
"startDate": "2014-10-01",
},
],
@@ -3095,87 +4009,111 @@ Object {
"name": "Monthly",
"options": Array [
Object {
+ "displayName": "January 2014",
"endDate": "2014-01-31",
"id": "201401",
"iso": "201401",
"name": "January 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-01-01",
},
Object {
+ "displayName": "February 2014",
"endDate": "2014-02-28",
"id": "201402",
"iso": "201402",
"name": "February 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-02-01",
},
Object {
+ "displayName": "March 2014",
"endDate": "2014-03-31",
"id": "201403",
"iso": "201403",
"name": "March 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-03-01",
},
Object {
+ "displayName": "April 2014",
"endDate": "2014-04-30",
"id": "201404",
"iso": "201404",
"name": "April 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-04-01",
},
Object {
+ "displayName": "May 2014",
"endDate": "2014-05-31",
"id": "201405",
"iso": "201405",
"name": "May 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-05-01",
},
Object {
+ "displayName": "June 2014",
"endDate": "2014-06-30",
"id": "201406",
"iso": "201406",
"name": "June 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-06-01",
},
Object {
+ "displayName": "July 2014",
"endDate": "2014-07-31",
"id": "201407",
"iso": "201407",
"name": "July 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-07-01",
},
Object {
+ "displayName": "August 2014",
"endDate": "2014-08-31",
"id": "201408",
"iso": "201408",
"name": "August 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-08-01",
},
Object {
+ "displayName": "September 2014",
"endDate": "2014-09-30",
"id": "201409",
"iso": "201409",
"name": "September 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-09-01",
},
Object {
+ "displayName": "October 2014",
"endDate": "2014-10-31",
"id": "201410",
"iso": "201410",
"name": "October 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-10-01",
},
Object {
+ "displayName": "November 2014",
"endDate": "2014-11-30",
"id": "201411",
"iso": "201411",
"name": "November 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-11-01",
},
Object {
+ "displayName": "December 2014",
"endDate": "2014-12-31",
"id": "201412",
"iso": "201412",
"name": "December 2014",
+ "periodType": "MONTHLY",
"startDate": "2014-12-01",
},
],
@@ -3189,31 +4127,39 @@ Object {
"name": "Quarterly",
"options": Array [
Object {
+ "displayName": "January - March 2014",
"endDate": "2014-03-31",
"id": "2014Q1",
"iso": "2014Q1",
"name": "January - March 2014",
+ "periodType": "QUARTERLY",
"startDate": "2014-01-01",
},
Object {
+ "displayName": "April - June 2014",
"endDate": "2014-06-30",
"id": "2014Q2",
"iso": "2014Q2",
"name": "April - June 2014",
+ "periodType": "QUARTERLY",
"startDate": "2014-04-01",
},
Object {
+ "displayName": "July - September 2014",
"endDate": "2014-09-30",
"id": "2014Q3",
"iso": "2014Q3",
"name": "July - September 2014",
+ "periodType": "QUARTERLY",
"startDate": "2014-07-01",
},
Object {
+ "displayName": "October - December 2014",
"endDate": "2014-12-31",
"id": "2014Q4",
"iso": "2014Q4",
"name": "October - December 2014",
+ "periodType": "QUARTERLY",
"startDate": "2014-10-01",
},
],
@@ -3227,17 +4173,21 @@ Object {
"name": "Six-monthly",
"options": Array [
Object {
+ "displayName": "January - June 2014",
"endDate": "2014-06-30",
"id": "2014S1",
"iso": "2014S1",
"name": "January - June 2014",
+ "periodType": "SIXMONTHLY",
"startDate": "2014-01-01",
},
Object {
+ "displayName": "July - December 2014",
"endDate": "2014-12-31",
"id": "2014S2",
"iso": "2014S2",
"name": "July - December 2014",
+ "periodType": "SIXMONTHLY",
"startDate": "2014-07-01",
},
],
@@ -3251,17 +4201,21 @@ Object {
"name": "Six-monthly April",
"options": Array [
Object {
+ "displayName": "April - September 2014",
"endDate": "2014-09-30",
"id": "2014AprilS1",
"iso": "2014AprilS1",
"name": "April - September 2014",
+ "periodType": "SIXMONTHLYAPR",
"startDate": "2014-04-01",
},
Object {
+ "displayName": "October 2014 - March 2015",
"endDate": "2015-03-31",
"id": "2014AprilS2",
"iso": "2014AprilS2",
"name": "October 2014 - March 2015",
+ "periodType": "SIXMONTHLYAPR",
"startDate": "2014-10-01",
},
],
@@ -3275,367 +4229,471 @@ Object {
"name": "Weekly",
"options": Array [
Object {
+ "displayName": "Week 1 - 2013-12-30 - 2014-01-05",
"endDate": "2014-01-05",
"id": "2014W1",
"iso": "2014W1",
"name": "Week 1 - 2013-12-30 - 2014-01-05",
+ "periodType": "WEEKLY",
"startDate": "2013-12-30",
},
Object {
+ "displayName": "Week 2 - 2014-01-06 - 2014-01-12",
"endDate": "2014-01-12",
"id": "2014W2",
"iso": "2014W2",
"name": "Week 2 - 2014-01-06 - 2014-01-12",
+ "periodType": "WEEKLY",
"startDate": "2014-01-06",
},
Object {
+ "displayName": "Week 3 - 2014-01-13 - 2014-01-19",
"endDate": "2014-01-19",
"id": "2014W3",
"iso": "2014W3",
"name": "Week 3 - 2014-01-13 - 2014-01-19",
+ "periodType": "WEEKLY",
"startDate": "2014-01-13",
},
Object {
+ "displayName": "Week 4 - 2014-01-20 - 2014-01-26",
"endDate": "2014-01-26",
"id": "2014W4",
"iso": "2014W4",
"name": "Week 4 - 2014-01-20 - 2014-01-26",
+ "periodType": "WEEKLY",
"startDate": "2014-01-20",
},
Object {
+ "displayName": "Week 5 - 2014-01-27 - 2014-02-02",
"endDate": "2014-02-02",
"id": "2014W5",
"iso": "2014W5",
"name": "Week 5 - 2014-01-27 - 2014-02-02",
+ "periodType": "WEEKLY",
"startDate": "2014-01-27",
},
Object {
+ "displayName": "Week 6 - 2014-02-03 - 2014-02-09",
"endDate": "2014-02-09",
"id": "2014W6",
"iso": "2014W6",
"name": "Week 6 - 2014-02-03 - 2014-02-09",
+ "periodType": "WEEKLY",
"startDate": "2014-02-03",
},
Object {
+ "displayName": "Week 7 - 2014-02-10 - 2014-02-16",
"endDate": "2014-02-16",
"id": "2014W7",
"iso": "2014W7",
"name": "Week 7 - 2014-02-10 - 2014-02-16",
+ "periodType": "WEEKLY",
"startDate": "2014-02-10",
},
Object {
+ "displayName": "Week 8 - 2014-02-17 - 2014-02-23",
"endDate": "2014-02-23",
"id": "2014W8",
"iso": "2014W8",
"name": "Week 8 - 2014-02-17 - 2014-02-23",
+ "periodType": "WEEKLY",
"startDate": "2014-02-17",
},
Object {
+ "displayName": "Week 9 - 2014-02-24 - 2014-03-02",
"endDate": "2014-03-02",
"id": "2014W9",
"iso": "2014W9",
"name": "Week 9 - 2014-02-24 - 2014-03-02",
+ "periodType": "WEEKLY",
"startDate": "2014-02-24",
},
Object {
+ "displayName": "Week 10 - 2014-03-03 - 2014-03-09",
"endDate": "2014-03-09",
"id": "2014W10",
"iso": "2014W10",
"name": "Week 10 - 2014-03-03 - 2014-03-09",
+ "periodType": "WEEKLY",
"startDate": "2014-03-03",
},
Object {
+ "displayName": "Week 11 - 2014-03-10 - 2014-03-16",
"endDate": "2014-03-16",
"id": "2014W11",
"iso": "2014W11",
"name": "Week 11 - 2014-03-10 - 2014-03-16",
+ "periodType": "WEEKLY",
"startDate": "2014-03-10",
},
Object {
+ "displayName": "Week 12 - 2014-03-17 - 2014-03-23",
"endDate": "2014-03-23",
"id": "2014W12",
"iso": "2014W12",
"name": "Week 12 - 2014-03-17 - 2014-03-23",
+ "periodType": "WEEKLY",
"startDate": "2014-03-17",
},
Object {
+ "displayName": "Week 13 - 2014-03-24 - 2014-03-30",
"endDate": "2014-03-30",
"id": "2014W13",
"iso": "2014W13",
"name": "Week 13 - 2014-03-24 - 2014-03-30",
+ "periodType": "WEEKLY",
"startDate": "2014-03-24",
},
Object {
+ "displayName": "Week 14 - 2014-03-31 - 2014-04-06",
"endDate": "2014-04-06",
"id": "2014W14",
"iso": "2014W14",
"name": "Week 14 - 2014-03-31 - 2014-04-06",
+ "periodType": "WEEKLY",
"startDate": "2014-03-31",
},
Object {
+ "displayName": "Week 15 - 2014-04-07 - 2014-04-13",
"endDate": "2014-04-13",
"id": "2014W15",
"iso": "2014W15",
"name": "Week 15 - 2014-04-07 - 2014-04-13",
+ "periodType": "WEEKLY",
"startDate": "2014-04-07",
},
Object {
+ "displayName": "Week 16 - 2014-04-14 - 2014-04-20",
"endDate": "2014-04-20",
"id": "2014W16",
"iso": "2014W16",
"name": "Week 16 - 2014-04-14 - 2014-04-20",
+ "periodType": "WEEKLY",
"startDate": "2014-04-14",
},
Object {
+ "displayName": "Week 17 - 2014-04-21 - 2014-04-27",
"endDate": "2014-04-27",
"id": "2014W17",
"iso": "2014W17",
"name": "Week 17 - 2014-04-21 - 2014-04-27",
+ "periodType": "WEEKLY",
"startDate": "2014-04-21",
},
Object {
+ "displayName": "Week 18 - 2014-04-28 - 2014-05-04",
"endDate": "2014-05-04",
"id": "2014W18",
"iso": "2014W18",
"name": "Week 18 - 2014-04-28 - 2014-05-04",
+ "periodType": "WEEKLY",
"startDate": "2014-04-28",
},
Object {
+ "displayName": "Week 19 - 2014-05-05 - 2014-05-11",
"endDate": "2014-05-11",
"id": "2014W19",
"iso": "2014W19",
"name": "Week 19 - 2014-05-05 - 2014-05-11",
+ "periodType": "WEEKLY",
"startDate": "2014-05-05",
},
Object {
+ "displayName": "Week 20 - 2014-05-12 - 2014-05-18",
"endDate": "2014-05-18",
"id": "2014W20",
"iso": "2014W20",
"name": "Week 20 - 2014-05-12 - 2014-05-18",
+ "periodType": "WEEKLY",
"startDate": "2014-05-12",
},
Object {
+ "displayName": "Week 21 - 2014-05-19 - 2014-05-25",
"endDate": "2014-05-25",
"id": "2014W21",
"iso": "2014W21",
"name": "Week 21 - 2014-05-19 - 2014-05-25",
+ "periodType": "WEEKLY",
"startDate": "2014-05-19",
},
Object {
+ "displayName": "Week 22 - 2014-05-26 - 2014-06-01",
"endDate": "2014-06-01",
"id": "2014W22",
"iso": "2014W22",
"name": "Week 22 - 2014-05-26 - 2014-06-01",
+ "periodType": "WEEKLY",
"startDate": "2014-05-26",
},
Object {
+ "displayName": "Week 23 - 2014-06-02 - 2014-06-08",
"endDate": "2014-06-08",
"id": "2014W23",
"iso": "2014W23",
"name": "Week 23 - 2014-06-02 - 2014-06-08",
+ "periodType": "WEEKLY",
"startDate": "2014-06-02",
},
Object {
+ "displayName": "Week 24 - 2014-06-09 - 2014-06-15",
"endDate": "2014-06-15",
"id": "2014W24",
"iso": "2014W24",
"name": "Week 24 - 2014-06-09 - 2014-06-15",
+ "periodType": "WEEKLY",
"startDate": "2014-06-09",
},
Object {
+ "displayName": "Week 25 - 2014-06-16 - 2014-06-22",
"endDate": "2014-06-22",
"id": "2014W25",
"iso": "2014W25",
"name": "Week 25 - 2014-06-16 - 2014-06-22",
+ "periodType": "WEEKLY",
"startDate": "2014-06-16",
},
Object {
+ "displayName": "Week 26 - 2014-06-23 - 2014-06-29",
"endDate": "2014-06-29",
"id": "2014W26",
"iso": "2014W26",
"name": "Week 26 - 2014-06-23 - 2014-06-29",
+ "periodType": "WEEKLY",
"startDate": "2014-06-23",
},
Object {
+ "displayName": "Week 27 - 2014-06-30 - 2014-07-06",
"endDate": "2014-07-06",
"id": "2014W27",
"iso": "2014W27",
"name": "Week 27 - 2014-06-30 - 2014-07-06",
+ "periodType": "WEEKLY",
"startDate": "2014-06-30",
},
Object {
+ "displayName": "Week 28 - 2014-07-07 - 2014-07-13",
"endDate": "2014-07-13",
"id": "2014W28",
"iso": "2014W28",
"name": "Week 28 - 2014-07-07 - 2014-07-13",
+ "periodType": "WEEKLY",
"startDate": "2014-07-07",
},
Object {
+ "displayName": "Week 29 - 2014-07-14 - 2014-07-20",
"endDate": "2014-07-20",
"id": "2014W29",
"iso": "2014W29",
"name": "Week 29 - 2014-07-14 - 2014-07-20",
+ "periodType": "WEEKLY",
"startDate": "2014-07-14",
},
Object {
+ "displayName": "Week 30 - 2014-07-21 - 2014-07-27",
"endDate": "2014-07-27",
"id": "2014W30",
"iso": "2014W30",
"name": "Week 30 - 2014-07-21 - 2014-07-27",
+ "periodType": "WEEKLY",
"startDate": "2014-07-21",
},
Object {
+ "displayName": "Week 31 - 2014-07-28 - 2014-08-03",
"endDate": "2014-08-03",
"id": "2014W31",
"iso": "2014W31",
"name": "Week 31 - 2014-07-28 - 2014-08-03",
+ "periodType": "WEEKLY",
"startDate": "2014-07-28",
},
Object {
+ "displayName": "Week 32 - 2014-08-04 - 2014-08-10",
"endDate": "2014-08-10",
"id": "2014W32",
"iso": "2014W32",
"name": "Week 32 - 2014-08-04 - 2014-08-10",
+ "periodType": "WEEKLY",
"startDate": "2014-08-04",
},
Object {
+ "displayName": "Week 33 - 2014-08-11 - 2014-08-17",
"endDate": "2014-08-17",
"id": "2014W33",
"iso": "2014W33",
"name": "Week 33 - 2014-08-11 - 2014-08-17",
+ "periodType": "WEEKLY",
"startDate": "2014-08-11",
},
Object {
+ "displayName": "Week 34 - 2014-08-18 - 2014-08-24",
"endDate": "2014-08-24",
"id": "2014W34",
"iso": "2014W34",
"name": "Week 34 - 2014-08-18 - 2014-08-24",
+ "periodType": "WEEKLY",
"startDate": "2014-08-18",
},
Object {
+ "displayName": "Week 35 - 2014-08-25 - 2014-08-31",
"endDate": "2014-08-31",
"id": "2014W35",
"iso": "2014W35",
"name": "Week 35 - 2014-08-25 - 2014-08-31",
+ "periodType": "WEEKLY",
"startDate": "2014-08-25",
},
Object {
+ "displayName": "Week 36 - 2014-09-01 - 2014-09-07",
"endDate": "2014-09-07",
"id": "2014W36",
"iso": "2014W36",
"name": "Week 36 - 2014-09-01 - 2014-09-07",
+ "periodType": "WEEKLY",
"startDate": "2014-09-01",
},
Object {
+ "displayName": "Week 37 - 2014-09-08 - 2014-09-14",
"endDate": "2014-09-14",
"id": "2014W37",
"iso": "2014W37",
"name": "Week 37 - 2014-09-08 - 2014-09-14",
+ "periodType": "WEEKLY",
"startDate": "2014-09-08",
},
Object {
+ "displayName": "Week 38 - 2014-09-15 - 2014-09-21",
"endDate": "2014-09-21",
"id": "2014W38",
"iso": "2014W38",
"name": "Week 38 - 2014-09-15 - 2014-09-21",
+ "periodType": "WEEKLY",
"startDate": "2014-09-15",
},
Object {
+ "displayName": "Week 39 - 2014-09-22 - 2014-09-28",
"endDate": "2014-09-28",
"id": "2014W39",
"iso": "2014W39",
"name": "Week 39 - 2014-09-22 - 2014-09-28",
+ "periodType": "WEEKLY",
"startDate": "2014-09-22",
},
Object {
+ "displayName": "Week 40 - 2014-09-29 - 2014-10-05",
"endDate": "2014-10-05",
"id": "2014W40",
"iso": "2014W40",
"name": "Week 40 - 2014-09-29 - 2014-10-05",
+ "periodType": "WEEKLY",
"startDate": "2014-09-29",
},
Object {
+ "displayName": "Week 41 - 2014-10-06 - 2014-10-12",
"endDate": "2014-10-12",
"id": "2014W41",
"iso": "2014W41",
"name": "Week 41 - 2014-10-06 - 2014-10-12",
+ "periodType": "WEEKLY",
"startDate": "2014-10-06",
},
Object {
+ "displayName": "Week 42 - 2014-10-13 - 2014-10-19",
"endDate": "2014-10-19",
"id": "2014W42",
"iso": "2014W42",
"name": "Week 42 - 2014-10-13 - 2014-10-19",
+ "periodType": "WEEKLY",
"startDate": "2014-10-13",
},
Object {
+ "displayName": "Week 43 - 2014-10-20 - 2014-10-26",
"endDate": "2014-10-26",
"id": "2014W43",
"iso": "2014W43",
"name": "Week 43 - 2014-10-20 - 2014-10-26",
+ "periodType": "WEEKLY",
"startDate": "2014-10-20",
},
Object {
+ "displayName": "Week 44 - 2014-10-27 - 2014-11-02",
"endDate": "2014-11-02",
"id": "2014W44",
"iso": "2014W44",
"name": "Week 44 - 2014-10-27 - 2014-11-02",
+ "periodType": "WEEKLY",
"startDate": "2014-10-27",
},
Object {
+ "displayName": "Week 45 - 2014-11-03 - 2014-11-09",
"endDate": "2014-11-09",
"id": "2014W45",
"iso": "2014W45",
"name": "Week 45 - 2014-11-03 - 2014-11-09",
+ "periodType": "WEEKLY",
"startDate": "2014-11-03",
},
Object {
+ "displayName": "Week 46 - 2014-11-10 - 2014-11-16",
"endDate": "2014-11-16",
"id": "2014W46",
"iso": "2014W46",
"name": "Week 46 - 2014-11-10 - 2014-11-16",
+ "periodType": "WEEKLY",
"startDate": "2014-11-10",
},
Object {
+ "displayName": "Week 47 - 2014-11-17 - 2014-11-23",
"endDate": "2014-11-23",
"id": "2014W47",
"iso": "2014W47",
"name": "Week 47 - 2014-11-17 - 2014-11-23",
+ "periodType": "WEEKLY",
"startDate": "2014-11-17",
},
Object {
+ "displayName": "Week 48 - 2014-11-24 - 2014-11-30",
"endDate": "2014-11-30",
"id": "2014W48",
"iso": "2014W48",
"name": "Week 48 - 2014-11-24 - 2014-11-30",
+ "periodType": "WEEKLY",
"startDate": "2014-11-24",
},
Object {
+ "displayName": "Week 49 - 2014-12-01 - 2014-12-07",
"endDate": "2014-12-07",
"id": "2014W49",
"iso": "2014W49",
"name": "Week 49 - 2014-12-01 - 2014-12-07",
+ "periodType": "WEEKLY",
"startDate": "2014-12-01",
},
Object {
+ "displayName": "Week 50 - 2014-12-08 - 2014-12-14",
"endDate": "2014-12-14",
"id": "2014W50",
"iso": "2014W50",
"name": "Week 50 - 2014-12-08 - 2014-12-14",
+ "periodType": "WEEKLY",
"startDate": "2014-12-08",
},
Object {
+ "displayName": "Week 51 - 2014-12-15 - 2014-12-21",
"endDate": "2014-12-21",
"id": "2014W51",
"iso": "2014W51",
"name": "Week 51 - 2014-12-15 - 2014-12-21",
+ "periodType": "WEEKLY",
"startDate": "2014-12-15",
},
Object {
+ "displayName": "Week 52 - 2014-12-22 - 2014-12-28",
"endDate": "2014-12-28",
"id": "2014W52",
"iso": "2014W52",
"name": "Week 52 - 2014-12-22 - 2014-12-28",
+ "periodType": "WEEKLY",
"startDate": "2014-12-22",
},
],
@@ -3649,367 +4707,471 @@ Object {
"name": "Weekly (Start Saturday)",
"options": Array [
Object {
+ "displayName": "Week 1 - 2014-01-04 - 2014-01-10",
"endDate": "2014-01-10",
"id": "2014SatW1",
"iso": "2014SatW1",
"name": "Week 1 - 2014-01-04 - 2014-01-10",
+ "periodType": "WEEKLY",
"startDate": "2014-01-04",
},
Object {
+ "displayName": "Week 2 - 2014-01-11 - 2014-01-17",
"endDate": "2014-01-17",
"id": "2014SatW2",
"iso": "2014SatW2",
"name": "Week 2 - 2014-01-11 - 2014-01-17",
+ "periodType": "WEEKLY",
"startDate": "2014-01-11",
},
Object {
+ "displayName": "Week 3 - 2014-01-18 - 2014-01-24",
"endDate": "2014-01-24",
"id": "2014SatW3",
"iso": "2014SatW3",
"name": "Week 3 - 2014-01-18 - 2014-01-24",
+ "periodType": "WEEKLY",
"startDate": "2014-01-18",
},
Object {
+ "displayName": "Week 4 - 2014-01-25 - 2014-01-31",
"endDate": "2014-01-31",
"id": "2014SatW4",
"iso": "2014SatW4",
"name": "Week 4 - 2014-01-25 - 2014-01-31",
+ "periodType": "WEEKLY",
"startDate": "2014-01-25",
},
Object {
+ "displayName": "Week 5 - 2014-02-01 - 2014-02-07",
"endDate": "2014-02-07",
"id": "2014SatW5",
"iso": "2014SatW5",
"name": "Week 5 - 2014-02-01 - 2014-02-07",
+ "periodType": "WEEKLY",
"startDate": "2014-02-01",
},
Object {
+ "displayName": "Week 6 - 2014-02-08 - 2014-02-14",
"endDate": "2014-02-14",
"id": "2014SatW6",
"iso": "2014SatW6",
"name": "Week 6 - 2014-02-08 - 2014-02-14",
+ "periodType": "WEEKLY",
"startDate": "2014-02-08",
},
Object {
+ "displayName": "Week 7 - 2014-02-15 - 2014-02-21",
"endDate": "2014-02-21",
"id": "2014SatW7",
"iso": "2014SatW7",
"name": "Week 7 - 2014-02-15 - 2014-02-21",
+ "periodType": "WEEKLY",
"startDate": "2014-02-15",
},
Object {
+ "displayName": "Week 8 - 2014-02-22 - 2014-02-28",
"endDate": "2014-02-28",
"id": "2014SatW8",
"iso": "2014SatW8",
"name": "Week 8 - 2014-02-22 - 2014-02-28",
+ "periodType": "WEEKLY",
"startDate": "2014-02-22",
},
Object {
+ "displayName": "Week 9 - 2014-03-01 - 2014-03-07",
"endDate": "2014-03-07",
"id": "2014SatW9",
"iso": "2014SatW9",
"name": "Week 9 - 2014-03-01 - 2014-03-07",
+ "periodType": "WEEKLY",
"startDate": "2014-03-01",
},
Object {
+ "displayName": "Week 10 - 2014-03-08 - 2014-03-14",
"endDate": "2014-03-14",
"id": "2014SatW10",
"iso": "2014SatW10",
"name": "Week 10 - 2014-03-08 - 2014-03-14",
+ "periodType": "WEEKLY",
"startDate": "2014-03-08",
},
Object {
+ "displayName": "Week 11 - 2014-03-15 - 2014-03-21",
"endDate": "2014-03-21",
"id": "2014SatW11",
"iso": "2014SatW11",
"name": "Week 11 - 2014-03-15 - 2014-03-21",
+ "periodType": "WEEKLY",
"startDate": "2014-03-15",
},
Object {
+ "displayName": "Week 12 - 2014-03-22 - 2014-03-28",
"endDate": "2014-03-28",
"id": "2014SatW12",
"iso": "2014SatW12",
"name": "Week 12 - 2014-03-22 - 2014-03-28",
+ "periodType": "WEEKLY",
"startDate": "2014-03-22",
},
Object {
+ "displayName": "Week 13 - 2014-03-29 - 2014-04-04",
"endDate": "2014-04-04",
"id": "2014SatW13",
"iso": "2014SatW13",
"name": "Week 13 - 2014-03-29 - 2014-04-04",
+ "periodType": "WEEKLY",
"startDate": "2014-03-29",
},
Object {
+ "displayName": "Week 14 - 2014-04-05 - 2014-04-11",
"endDate": "2014-04-11",
"id": "2014SatW14",
"iso": "2014SatW14",
"name": "Week 14 - 2014-04-05 - 2014-04-11",
+ "periodType": "WEEKLY",
"startDate": "2014-04-05",
},
Object {
+ "displayName": "Week 15 - 2014-04-12 - 2014-04-18",
"endDate": "2014-04-18",
"id": "2014SatW15",
"iso": "2014SatW15",
"name": "Week 15 - 2014-04-12 - 2014-04-18",
+ "periodType": "WEEKLY",
"startDate": "2014-04-12",
},
Object {
+ "displayName": "Week 16 - 2014-04-19 - 2014-04-25",
"endDate": "2014-04-25",
"id": "2014SatW16",
"iso": "2014SatW16",
"name": "Week 16 - 2014-04-19 - 2014-04-25",
+ "periodType": "WEEKLY",
"startDate": "2014-04-19",
},
Object {
+ "displayName": "Week 17 - 2014-04-26 - 2014-05-02",
"endDate": "2014-05-02",
"id": "2014SatW17",
"iso": "2014SatW17",
"name": "Week 17 - 2014-04-26 - 2014-05-02",
+ "periodType": "WEEKLY",
"startDate": "2014-04-26",
},
Object {
+ "displayName": "Week 18 - 2014-05-03 - 2014-05-09",
"endDate": "2014-05-09",
"id": "2014SatW18",
"iso": "2014SatW18",
"name": "Week 18 - 2014-05-03 - 2014-05-09",
+ "periodType": "WEEKLY",
"startDate": "2014-05-03",
},
Object {
+ "displayName": "Week 19 - 2014-05-10 - 2014-05-16",
"endDate": "2014-05-16",
"id": "2014SatW19",
"iso": "2014SatW19",
"name": "Week 19 - 2014-05-10 - 2014-05-16",
+ "periodType": "WEEKLY",
"startDate": "2014-05-10",
},
Object {
+ "displayName": "Week 20 - 2014-05-17 - 2014-05-23",
"endDate": "2014-05-23",
"id": "2014SatW20",
"iso": "2014SatW20",
"name": "Week 20 - 2014-05-17 - 2014-05-23",
+ "periodType": "WEEKLY",
"startDate": "2014-05-17",
},
Object {
+ "displayName": "Week 21 - 2014-05-24 - 2014-05-30",
"endDate": "2014-05-30",
"id": "2014SatW21",
"iso": "2014SatW21",
"name": "Week 21 - 2014-05-24 - 2014-05-30",
+ "periodType": "WEEKLY",
"startDate": "2014-05-24",
},
Object {
+ "displayName": "Week 22 - 2014-05-31 - 2014-06-06",
"endDate": "2014-06-06",
"id": "2014SatW22",
"iso": "2014SatW22",
"name": "Week 22 - 2014-05-31 - 2014-06-06",
+ "periodType": "WEEKLY",
"startDate": "2014-05-31",
},
Object {
+ "displayName": "Week 23 - 2014-06-07 - 2014-06-13",
"endDate": "2014-06-13",
"id": "2014SatW23",
"iso": "2014SatW23",
"name": "Week 23 - 2014-06-07 - 2014-06-13",
+ "periodType": "WEEKLY",
"startDate": "2014-06-07",
},
Object {
+ "displayName": "Week 24 - 2014-06-14 - 2014-06-20",
"endDate": "2014-06-20",
"id": "2014SatW24",
"iso": "2014SatW24",
"name": "Week 24 - 2014-06-14 - 2014-06-20",
+ "periodType": "WEEKLY",
"startDate": "2014-06-14",
},
Object {
+ "displayName": "Week 25 - 2014-06-21 - 2014-06-27",
"endDate": "2014-06-27",
"id": "2014SatW25",
"iso": "2014SatW25",
"name": "Week 25 - 2014-06-21 - 2014-06-27",
+ "periodType": "WEEKLY",
"startDate": "2014-06-21",
},
Object {
+ "displayName": "Week 26 - 2014-06-28 - 2014-07-04",
"endDate": "2014-07-04",
"id": "2014SatW26",
"iso": "2014SatW26",
"name": "Week 26 - 2014-06-28 - 2014-07-04",
+ "periodType": "WEEKLY",
"startDate": "2014-06-28",
},
Object {
+ "displayName": "Week 27 - 2014-07-05 - 2014-07-11",
"endDate": "2014-07-11",
"id": "2014SatW27",
"iso": "2014SatW27",
"name": "Week 27 - 2014-07-05 - 2014-07-11",
+ "periodType": "WEEKLY",
"startDate": "2014-07-05",
},
Object {
+ "displayName": "Week 28 - 2014-07-12 - 2014-07-18",
"endDate": "2014-07-18",
"id": "2014SatW28",
"iso": "2014SatW28",
"name": "Week 28 - 2014-07-12 - 2014-07-18",
+ "periodType": "WEEKLY",
"startDate": "2014-07-12",
},
Object {
+ "displayName": "Week 29 - 2014-07-19 - 2014-07-25",
"endDate": "2014-07-25",
"id": "2014SatW29",
"iso": "2014SatW29",
"name": "Week 29 - 2014-07-19 - 2014-07-25",
+ "periodType": "WEEKLY",
"startDate": "2014-07-19",
},
Object {
+ "displayName": "Week 30 - 2014-07-26 - 2014-08-01",
"endDate": "2014-08-01",
"id": "2014SatW30",
"iso": "2014SatW30",
"name": "Week 30 - 2014-07-26 - 2014-08-01",
+ "periodType": "WEEKLY",
"startDate": "2014-07-26",
},
Object {
+ "displayName": "Week 31 - 2014-08-02 - 2014-08-08",
"endDate": "2014-08-08",
"id": "2014SatW31",
"iso": "2014SatW31",
"name": "Week 31 - 2014-08-02 - 2014-08-08",
+ "periodType": "WEEKLY",
"startDate": "2014-08-02",
},
Object {
+ "displayName": "Week 32 - 2014-08-09 - 2014-08-15",
"endDate": "2014-08-15",
"id": "2014SatW32",
"iso": "2014SatW32",
"name": "Week 32 - 2014-08-09 - 2014-08-15",
+ "periodType": "WEEKLY",
"startDate": "2014-08-09",
},
Object {
+ "displayName": "Week 33 - 2014-08-16 - 2014-08-22",
"endDate": "2014-08-22",
"id": "2014SatW33",
"iso": "2014SatW33",
"name": "Week 33 - 2014-08-16 - 2014-08-22",
+ "periodType": "WEEKLY",
"startDate": "2014-08-16",
},
Object {
+ "displayName": "Week 34 - 2014-08-23 - 2014-08-29",
"endDate": "2014-08-29",
"id": "2014SatW34",
"iso": "2014SatW34",
"name": "Week 34 - 2014-08-23 - 2014-08-29",
+ "periodType": "WEEKLY",
"startDate": "2014-08-23",
},
Object {
+ "displayName": "Week 35 - 2014-08-30 - 2014-09-05",
"endDate": "2014-09-05",
"id": "2014SatW35",
"iso": "2014SatW35",
"name": "Week 35 - 2014-08-30 - 2014-09-05",
+ "periodType": "WEEKLY",
"startDate": "2014-08-30",
},
Object {
+ "displayName": "Week 36 - 2014-09-06 - 2014-09-12",
"endDate": "2014-09-12",
"id": "2014SatW36",
"iso": "2014SatW36",
"name": "Week 36 - 2014-09-06 - 2014-09-12",
+ "periodType": "WEEKLY",
"startDate": "2014-09-06",
},
Object {
+ "displayName": "Week 37 - 2014-09-13 - 2014-09-19",
"endDate": "2014-09-19",
"id": "2014SatW37",
"iso": "2014SatW37",
"name": "Week 37 - 2014-09-13 - 2014-09-19",
+ "periodType": "WEEKLY",
"startDate": "2014-09-13",
},
Object {
+ "displayName": "Week 38 - 2014-09-20 - 2014-09-26",
"endDate": "2014-09-26",
"id": "2014SatW38",
"iso": "2014SatW38",
"name": "Week 38 - 2014-09-20 - 2014-09-26",
+ "periodType": "WEEKLY",
"startDate": "2014-09-20",
},
Object {
+ "displayName": "Week 39 - 2014-09-27 - 2014-10-03",
"endDate": "2014-10-03",
"id": "2014SatW39",
"iso": "2014SatW39",
"name": "Week 39 - 2014-09-27 - 2014-10-03",
+ "periodType": "WEEKLY",
"startDate": "2014-09-27",
},
Object {
+ "displayName": "Week 40 - 2014-10-04 - 2014-10-10",
"endDate": "2014-10-10",
"id": "2014SatW40",
"iso": "2014SatW40",
"name": "Week 40 - 2014-10-04 - 2014-10-10",
+ "periodType": "WEEKLY",
"startDate": "2014-10-04",
},
Object {
+ "displayName": "Week 41 - 2014-10-11 - 2014-10-17",
"endDate": "2014-10-17",
"id": "2014SatW41",
"iso": "2014SatW41",
"name": "Week 41 - 2014-10-11 - 2014-10-17",
+ "periodType": "WEEKLY",
"startDate": "2014-10-11",
},
Object {
+ "displayName": "Week 42 - 2014-10-18 - 2014-10-24",
"endDate": "2014-10-24",
"id": "2014SatW42",
"iso": "2014SatW42",
"name": "Week 42 - 2014-10-18 - 2014-10-24",
+ "periodType": "WEEKLY",
"startDate": "2014-10-18",
},
Object {
+ "displayName": "Week 43 - 2014-10-25 - 2014-10-31",
"endDate": "2014-10-31",
"id": "2014SatW43",
"iso": "2014SatW43",
"name": "Week 43 - 2014-10-25 - 2014-10-31",
+ "periodType": "WEEKLY",
"startDate": "2014-10-25",
},
Object {
+ "displayName": "Week 44 - 2014-11-01 - 2014-11-07",
"endDate": "2014-11-07",
"id": "2014SatW44",
"iso": "2014SatW44",
"name": "Week 44 - 2014-11-01 - 2014-11-07",
+ "periodType": "WEEKLY",
"startDate": "2014-11-01",
},
Object {
+ "displayName": "Week 45 - 2014-11-08 - 2014-11-14",
"endDate": "2014-11-14",
"id": "2014SatW45",
"iso": "2014SatW45",
"name": "Week 45 - 2014-11-08 - 2014-11-14",
+ "periodType": "WEEKLY",
"startDate": "2014-11-08",
},
Object {
+ "displayName": "Week 46 - 2014-11-15 - 2014-11-21",
"endDate": "2014-11-21",
"id": "2014SatW46",
"iso": "2014SatW46",
"name": "Week 46 - 2014-11-15 - 2014-11-21",
+ "periodType": "WEEKLY",
"startDate": "2014-11-15",
},
Object {
+ "displayName": "Week 47 - 2014-11-22 - 2014-11-28",
"endDate": "2014-11-28",
"id": "2014SatW47",
"iso": "2014SatW47",
"name": "Week 47 - 2014-11-22 - 2014-11-28",
+ "periodType": "WEEKLY",
"startDate": "2014-11-22",
},
Object {
+ "displayName": "Week 48 - 2014-11-29 - 2014-12-05",
"endDate": "2014-12-05",
"id": "2014SatW48",
"iso": "2014SatW48",
"name": "Week 48 - 2014-11-29 - 2014-12-05",
+ "periodType": "WEEKLY",
"startDate": "2014-11-29",
},
Object {
+ "displayName": "Week 49 - 2014-12-06 - 2014-12-12",
"endDate": "2014-12-12",
"id": "2014SatW49",
"iso": "2014SatW49",
"name": "Week 49 - 2014-12-06 - 2014-12-12",
+ "periodType": "WEEKLY",
"startDate": "2014-12-06",
},
Object {
+ "displayName": "Week 50 - 2014-12-13 - 2014-12-19",
"endDate": "2014-12-19",
"id": "2014SatW50",
"iso": "2014SatW50",
"name": "Week 50 - 2014-12-13 - 2014-12-19",
+ "periodType": "WEEKLY",
"startDate": "2014-12-13",
},
Object {
+ "displayName": "Week 51 - 2014-12-20 - 2014-12-26",
"endDate": "2014-12-26",
"id": "2014SatW51",
"iso": "2014SatW51",
"name": "Week 51 - 2014-12-20 - 2014-12-26",
+ "periodType": "WEEKLY",
"startDate": "2014-12-20",
},
Object {
+ "displayName": "Week 52 - 2014-12-27 - 2015-01-02",
"endDate": "2015-01-02",
"id": "2014SatW52",
"iso": "2014SatW52",
"name": "Week 52 - 2014-12-27 - 2015-01-02",
+ "periodType": "WEEKLY",
"startDate": "2014-12-27",
},
],
@@ -4023,374 +5185,480 @@ Object {
"name": "Weekly (Start Sunday)",
"options": Array [
Object {
+ "displayName": "Week 1 - 2013-12-29 - 2014-01-04",
"endDate": "2014-01-04",
"id": "2014SunW1",
"iso": "2014SunW1",
"name": "Week 1 - 2013-12-29 - 2014-01-04",
+ "periodType": "WEEKLY",
"startDate": "2013-12-29",
},
Object {
+ "displayName": "Week 2 - 2014-01-05 - 2014-01-11",
"endDate": "2014-01-11",
"id": "2014SunW2",
"iso": "2014SunW2",
"name": "Week 2 - 2014-01-05 - 2014-01-11",
+ "periodType": "WEEKLY",
"startDate": "2014-01-05",
},
Object {
+ "displayName": "Week 3 - 2014-01-12 - 2014-01-18",
"endDate": "2014-01-18",
"id": "2014SunW3",
"iso": "2014SunW3",
"name": "Week 3 - 2014-01-12 - 2014-01-18",
+ "periodType": "WEEKLY",
"startDate": "2014-01-12",
},
Object {
+ "displayName": "Week 4 - 2014-01-19 - 2014-01-25",
"endDate": "2014-01-25",
"id": "2014SunW4",
"iso": "2014SunW4",
"name": "Week 4 - 2014-01-19 - 2014-01-25",
+ "periodType": "WEEKLY",
"startDate": "2014-01-19",
},
Object {
+ "displayName": "Week 5 - 2014-01-26 - 2014-02-01",
"endDate": "2014-02-01",
"id": "2014SunW5",
"iso": "2014SunW5",
"name": "Week 5 - 2014-01-26 - 2014-02-01",
+ "periodType": "WEEKLY",
"startDate": "2014-01-26",
},
Object {
+ "displayName": "Week 6 - 2014-02-02 - 2014-02-08",
"endDate": "2014-02-08",
"id": "2014SunW6",
"iso": "2014SunW6",
"name": "Week 6 - 2014-02-02 - 2014-02-08",
+ "periodType": "WEEKLY",
"startDate": "2014-02-02",
},
Object {
+ "displayName": "Week 7 - 2014-02-09 - 2014-02-15",
"endDate": "2014-02-15",
"id": "2014SunW7",
"iso": "2014SunW7",
"name": "Week 7 - 2014-02-09 - 2014-02-15",
+ "periodType": "WEEKLY",
"startDate": "2014-02-09",
},
Object {
+ "displayName": "Week 8 - 2014-02-16 - 2014-02-22",
"endDate": "2014-02-22",
"id": "2014SunW8",
"iso": "2014SunW8",
"name": "Week 8 - 2014-02-16 - 2014-02-22",
+ "periodType": "WEEKLY",
"startDate": "2014-02-16",
},
Object {
+ "displayName": "Week 9 - 2014-02-23 - 2014-03-01",
"endDate": "2014-03-01",
"id": "2014SunW9",
"iso": "2014SunW9",
"name": "Week 9 - 2014-02-23 - 2014-03-01",
+ "periodType": "WEEKLY",
"startDate": "2014-02-23",
},
Object {
+ "displayName": "Week 10 - 2014-03-02 - 2014-03-08",
"endDate": "2014-03-08",
"id": "2014SunW10",
"iso": "2014SunW10",
"name": "Week 10 - 2014-03-02 - 2014-03-08",
+ "periodType": "WEEKLY",
"startDate": "2014-03-02",
},
Object {
+ "displayName": "Week 11 - 2014-03-09 - 2014-03-15",
"endDate": "2014-03-15",
"id": "2014SunW11",
"iso": "2014SunW11",
"name": "Week 11 - 2014-03-09 - 2014-03-15",
+ "periodType": "WEEKLY",
"startDate": "2014-03-09",
},
Object {
+ "displayName": "Week 12 - 2014-03-16 - 2014-03-22",
"endDate": "2014-03-22",
"id": "2014SunW12",
"iso": "2014SunW12",
"name": "Week 12 - 2014-03-16 - 2014-03-22",
+ "periodType": "WEEKLY",
"startDate": "2014-03-16",
},
Object {
+ "displayName": "Week 13 - 2014-03-23 - 2014-03-29",
"endDate": "2014-03-29",
"id": "2014SunW13",
"iso": "2014SunW13",
"name": "Week 13 - 2014-03-23 - 2014-03-29",
+ "periodType": "WEEKLY",
"startDate": "2014-03-23",
},
Object {
+ "displayName": "Week 14 - 2014-03-30 - 2014-04-05",
"endDate": "2014-04-05",
"id": "2014SunW14",
"iso": "2014SunW14",
"name": "Week 14 - 2014-03-30 - 2014-04-05",
+ "periodType": "WEEKLY",
"startDate": "2014-03-30",
},
Object {
+ "displayName": "Week 15 - 2014-04-06 - 2014-04-12",
"endDate": "2014-04-12",
"id": "2014SunW15",
"iso": "2014SunW15",
"name": "Week 15 - 2014-04-06 - 2014-04-12",
+ "periodType": "WEEKLY",
"startDate": "2014-04-06",
},
Object {
+ "displayName": "Week 16 - 2014-04-13 - 2014-04-19",
"endDate": "2014-04-19",
"id": "2014SunW16",
"iso": "2014SunW16",
"name": "Week 16 - 2014-04-13 - 2014-04-19",
+ "periodType": "WEEKLY",
"startDate": "2014-04-13",
},
Object {
+ "displayName": "Week 17 - 2014-04-20 - 2014-04-26",
"endDate": "2014-04-26",
"id": "2014SunW17",
"iso": "2014SunW17",
"name": "Week 17 - 2014-04-20 - 2014-04-26",
+ "periodType": "WEEKLY",
"startDate": "2014-04-20",
},
Object {
+ "displayName": "Week 18 - 2014-04-27 - 2014-05-03",
"endDate": "2014-05-03",
"id": "2014SunW18",
"iso": "2014SunW18",
"name": "Week 18 - 2014-04-27 - 2014-05-03",
+ "periodType": "WEEKLY",
"startDate": "2014-04-27",
},
Object {
+ "displayName": "Week 19 - 2014-05-04 - 2014-05-10",
"endDate": "2014-05-10",
"id": "2014SunW19",
"iso": "2014SunW19",
"name": "Week 19 - 2014-05-04 - 2014-05-10",
+ "periodType": "WEEKLY",
"startDate": "2014-05-04",
},
Object {
+ "displayName": "Week 20 - 2014-05-11 - 2014-05-17",
"endDate": "2014-05-17",
"id": "2014SunW20",
"iso": "2014SunW20",
"name": "Week 20 - 2014-05-11 - 2014-05-17",
+ "periodType": "WEEKLY",
"startDate": "2014-05-11",
},
Object {
+ "displayName": "Week 21 - 2014-05-18 - 2014-05-24",
"endDate": "2014-05-24",
"id": "2014SunW21",
"iso": "2014SunW21",
"name": "Week 21 - 2014-05-18 - 2014-05-24",
+ "periodType": "WEEKLY",
"startDate": "2014-05-18",
},
Object {
+ "displayName": "Week 22 - 2014-05-25 - 2014-05-31",
"endDate": "2014-05-31",
"id": "2014SunW22",
"iso": "2014SunW22",
"name": "Week 22 - 2014-05-25 - 2014-05-31",
+ "periodType": "WEEKLY",
"startDate": "2014-05-25",
},
Object {
+ "displayName": "Week 23 - 2014-06-01 - 2014-06-07",
"endDate": "2014-06-07",
"id": "2014SunW23",
"iso": "2014SunW23",
"name": "Week 23 - 2014-06-01 - 2014-06-07",
+ "periodType": "WEEKLY",
"startDate": "2014-06-01",
},
Object {
+ "displayName": "Week 24 - 2014-06-08 - 2014-06-14",
"endDate": "2014-06-14",
"id": "2014SunW24",
"iso": "2014SunW24",
"name": "Week 24 - 2014-06-08 - 2014-06-14",
+ "periodType": "WEEKLY",
"startDate": "2014-06-08",
},
Object {
+ "displayName": "Week 25 - 2014-06-15 - 2014-06-21",
"endDate": "2014-06-21",
"id": "2014SunW25",
"iso": "2014SunW25",
"name": "Week 25 - 2014-06-15 - 2014-06-21",
+ "periodType": "WEEKLY",
"startDate": "2014-06-15",
},
Object {
+ "displayName": "Week 26 - 2014-06-22 - 2014-06-28",
"endDate": "2014-06-28",
"id": "2014SunW26",
"iso": "2014SunW26",
"name": "Week 26 - 2014-06-22 - 2014-06-28",
+ "periodType": "WEEKLY",
"startDate": "2014-06-22",
},
Object {
+ "displayName": "Week 27 - 2014-06-29 - 2014-07-05",
"endDate": "2014-07-05",
"id": "2014SunW27",
"iso": "2014SunW27",
"name": "Week 27 - 2014-06-29 - 2014-07-05",
+ "periodType": "WEEKLY",
"startDate": "2014-06-29",
},
Object {
+ "displayName": "Week 28 - 2014-07-06 - 2014-07-12",
"endDate": "2014-07-12",
"id": "2014SunW28",
"iso": "2014SunW28",
"name": "Week 28 - 2014-07-06 - 2014-07-12",
+ "periodType": "WEEKLY",
"startDate": "2014-07-06",
},
Object {
+ "displayName": "Week 29 - 2014-07-13 - 2014-07-19",
"endDate": "2014-07-19",
"id": "2014SunW29",
"iso": "2014SunW29",
"name": "Week 29 - 2014-07-13 - 2014-07-19",
+ "periodType": "WEEKLY",
"startDate": "2014-07-13",
},
Object {
+ "displayName": "Week 30 - 2014-07-20 - 2014-07-26",
"endDate": "2014-07-26",
"id": "2014SunW30",
"iso": "2014SunW30",
"name": "Week 30 - 2014-07-20 - 2014-07-26",
+ "periodType": "WEEKLY",
"startDate": "2014-07-20",
},
Object {
+ "displayName": "Week 31 - 2014-07-27 - 2014-08-02",
"endDate": "2014-08-02",
"id": "2014SunW31",
"iso": "2014SunW31",
"name": "Week 31 - 2014-07-27 - 2014-08-02",
+ "periodType": "WEEKLY",
"startDate": "2014-07-27",
},
Object {
+ "displayName": "Week 32 - 2014-08-03 - 2014-08-09",
"endDate": "2014-08-09",
"id": "2014SunW32",
"iso": "2014SunW32",
"name": "Week 32 - 2014-08-03 - 2014-08-09",
+ "periodType": "WEEKLY",
"startDate": "2014-08-03",
},
Object {
+ "displayName": "Week 33 - 2014-08-10 - 2014-08-16",
"endDate": "2014-08-16",
"id": "2014SunW33",
"iso": "2014SunW33",
"name": "Week 33 - 2014-08-10 - 2014-08-16",
+ "periodType": "WEEKLY",
"startDate": "2014-08-10",
},
Object {
+ "displayName": "Week 34 - 2014-08-17 - 2014-08-23",
"endDate": "2014-08-23",
"id": "2014SunW34",
"iso": "2014SunW34",
"name": "Week 34 - 2014-08-17 - 2014-08-23",
+ "periodType": "WEEKLY",
"startDate": "2014-08-17",
},
Object {
+ "displayName": "Week 35 - 2014-08-24 - 2014-08-30",
"endDate": "2014-08-30",
"id": "2014SunW35",
"iso": "2014SunW35",
"name": "Week 35 - 2014-08-24 - 2014-08-30",
+ "periodType": "WEEKLY",
"startDate": "2014-08-24",
},
Object {
+ "displayName": "Week 36 - 2014-08-31 - 2014-09-06",
"endDate": "2014-09-06",
"id": "2014SunW36",
"iso": "2014SunW36",
"name": "Week 36 - 2014-08-31 - 2014-09-06",
+ "periodType": "WEEKLY",
"startDate": "2014-08-31",
},
Object {
+ "displayName": "Week 37 - 2014-09-07 - 2014-09-13",
"endDate": "2014-09-13",
"id": "2014SunW37",
"iso": "2014SunW37",
"name": "Week 37 - 2014-09-07 - 2014-09-13",
+ "periodType": "WEEKLY",
"startDate": "2014-09-07",
},
Object {
+ "displayName": "Week 38 - 2014-09-14 - 2014-09-20",
"endDate": "2014-09-20",
"id": "2014SunW38",
"iso": "2014SunW38",
"name": "Week 38 - 2014-09-14 - 2014-09-20",
+ "periodType": "WEEKLY",
"startDate": "2014-09-14",
},
Object {
+ "displayName": "Week 39 - 2014-09-21 - 2014-09-27",
"endDate": "2014-09-27",
"id": "2014SunW39",
"iso": "2014SunW39",
"name": "Week 39 - 2014-09-21 - 2014-09-27",
+ "periodType": "WEEKLY",
"startDate": "2014-09-21",
},
Object {
+ "displayName": "Week 40 - 2014-09-28 - 2014-10-04",
"endDate": "2014-10-04",
"id": "2014SunW40",
"iso": "2014SunW40",
"name": "Week 40 - 2014-09-28 - 2014-10-04",
+ "periodType": "WEEKLY",
"startDate": "2014-09-28",
},
Object {
+ "displayName": "Week 41 - 2014-10-05 - 2014-10-11",
"endDate": "2014-10-11",
"id": "2014SunW41",
"iso": "2014SunW41",
"name": "Week 41 - 2014-10-05 - 2014-10-11",
+ "periodType": "WEEKLY",
"startDate": "2014-10-05",
},
Object {
+ "displayName": "Week 42 - 2014-10-12 - 2014-10-18",
"endDate": "2014-10-18",
"id": "2014SunW42",
"iso": "2014SunW42",
"name": "Week 42 - 2014-10-12 - 2014-10-18",
+ "periodType": "WEEKLY",
"startDate": "2014-10-12",
},
Object {
+ "displayName": "Week 43 - 2014-10-19 - 2014-10-25",
"endDate": "2014-10-25",
"id": "2014SunW43",
"iso": "2014SunW43",
"name": "Week 43 - 2014-10-19 - 2014-10-25",
+ "periodType": "WEEKLY",
"startDate": "2014-10-19",
},
Object {
+ "displayName": "Week 44 - 2014-10-26 - 2014-11-01",
"endDate": "2014-11-01",
"id": "2014SunW44",
"iso": "2014SunW44",
"name": "Week 44 - 2014-10-26 - 2014-11-01",
+ "periodType": "WEEKLY",
"startDate": "2014-10-26",
},
Object {
+ "displayName": "Week 45 - 2014-11-02 - 2014-11-08",
"endDate": "2014-11-08",
"id": "2014SunW45",
"iso": "2014SunW45",
"name": "Week 45 - 2014-11-02 - 2014-11-08",
+ "periodType": "WEEKLY",
"startDate": "2014-11-02",
},
Object {
+ "displayName": "Week 46 - 2014-11-09 - 2014-11-15",
"endDate": "2014-11-15",
"id": "2014SunW46",
"iso": "2014SunW46",
"name": "Week 46 - 2014-11-09 - 2014-11-15",
+ "periodType": "WEEKLY",
"startDate": "2014-11-09",
},
Object {
+ "displayName": "Week 47 - 2014-11-16 - 2014-11-22",
"endDate": "2014-11-22",
"id": "2014SunW47",
"iso": "2014SunW47",
"name": "Week 47 - 2014-11-16 - 2014-11-22",
+ "periodType": "WEEKLY",
"startDate": "2014-11-16",
},
Object {
+ "displayName": "Week 48 - 2014-11-23 - 2014-11-29",
"endDate": "2014-11-29",
"id": "2014SunW48",
"iso": "2014SunW48",
"name": "Week 48 - 2014-11-23 - 2014-11-29",
+ "periodType": "WEEKLY",
"startDate": "2014-11-23",
},
Object {
+ "displayName": "Week 49 - 2014-11-30 - 2014-12-06",
"endDate": "2014-12-06",
"id": "2014SunW49",
"iso": "2014SunW49",
"name": "Week 49 - 2014-11-30 - 2014-12-06",
+ "periodType": "WEEKLY",
"startDate": "2014-11-30",
},
Object {
+ "displayName": "Week 50 - 2014-12-07 - 2014-12-13",
"endDate": "2014-12-13",
"id": "2014SunW50",
"iso": "2014SunW50",
"name": "Week 50 - 2014-12-07 - 2014-12-13",
+ "periodType": "WEEKLY",
"startDate": "2014-12-07",
},
Object {
+ "displayName": "Week 51 - 2014-12-14 - 2014-12-20",
"endDate": "2014-12-20",
"id": "2014SunW51",
"iso": "2014SunW51",
"name": "Week 51 - 2014-12-14 - 2014-12-20",
+ "periodType": "WEEKLY",
"startDate": "2014-12-14",
},
Object {
+ "displayName": "Week 52 - 2014-12-21 - 2014-12-27",
"endDate": "2014-12-27",
"id": "2014SunW52",
"iso": "2014SunW52",
"name": "Week 52 - 2014-12-21 - 2014-12-27",
+ "periodType": "WEEKLY",
"startDate": "2014-12-21",
},
Object {
+ "displayName": "Week 53 - 2014-12-28 - 2015-01-03",
"endDate": "2015-01-03",
"id": "2014SunW53",
"iso": "2014SunW53",
"name": "Week 53 - 2014-12-28 - 2015-01-03",
+ "periodType": "WEEKLY",
"startDate": "2014-12-28",
},
],
@@ -4404,374 +5672,480 @@ Object {
"name": "Weekly (Start Sunday)",
"options": Array [
Object {
+ "displayName": "Week 1 - 2013-12-29 - 2014-01-04",
"endDate": "2014-01-04",
"id": "2014SunW1",
"iso": "2014SunW1",
"name": "Week 1 - 2013-12-29 - 2014-01-04",
+ "periodType": "WEEKLY",
"startDate": "2013-12-29",
},
Object {
+ "displayName": "Week 2 - 2014-01-05 - 2014-01-11",
"endDate": "2014-01-11",
"id": "2014SunW2",
"iso": "2014SunW2",
"name": "Week 2 - 2014-01-05 - 2014-01-11",
+ "periodType": "WEEKLY",
"startDate": "2014-01-05",
},
Object {
+ "displayName": "Week 3 - 2014-01-12 - 2014-01-18",
"endDate": "2014-01-18",
"id": "2014SunW3",
"iso": "2014SunW3",
"name": "Week 3 - 2014-01-12 - 2014-01-18",
+ "periodType": "WEEKLY",
"startDate": "2014-01-12",
},
Object {
+ "displayName": "Week 4 - 2014-01-19 - 2014-01-25",
"endDate": "2014-01-25",
"id": "2014SunW4",
"iso": "2014SunW4",
"name": "Week 4 - 2014-01-19 - 2014-01-25",
+ "periodType": "WEEKLY",
"startDate": "2014-01-19",
},
Object {
+ "displayName": "Week 5 - 2014-01-26 - 2014-02-01",
"endDate": "2014-02-01",
"id": "2014SunW5",
"iso": "2014SunW5",
"name": "Week 5 - 2014-01-26 - 2014-02-01",
+ "periodType": "WEEKLY",
"startDate": "2014-01-26",
},
Object {
+ "displayName": "Week 6 - 2014-02-02 - 2014-02-08",
"endDate": "2014-02-08",
"id": "2014SunW6",
"iso": "2014SunW6",
"name": "Week 6 - 2014-02-02 - 2014-02-08",
+ "periodType": "WEEKLY",
"startDate": "2014-02-02",
},
Object {
+ "displayName": "Week 7 - 2014-02-09 - 2014-02-15",
"endDate": "2014-02-15",
"id": "2014SunW7",
"iso": "2014SunW7",
"name": "Week 7 - 2014-02-09 - 2014-02-15",
+ "periodType": "WEEKLY",
"startDate": "2014-02-09",
},
Object {
+ "displayName": "Week 8 - 2014-02-16 - 2014-02-22",
"endDate": "2014-02-22",
"id": "2014SunW8",
"iso": "2014SunW8",
"name": "Week 8 - 2014-02-16 - 2014-02-22",
+ "periodType": "WEEKLY",
"startDate": "2014-02-16",
},
Object {
+ "displayName": "Week 9 - 2014-02-23 - 2014-03-01",
"endDate": "2014-03-01",
"id": "2014SunW9",
"iso": "2014SunW9",
"name": "Week 9 - 2014-02-23 - 2014-03-01",
+ "periodType": "WEEKLY",
"startDate": "2014-02-23",
},
Object {
+ "displayName": "Week 10 - 2014-03-02 - 2014-03-08",
"endDate": "2014-03-08",
"id": "2014SunW10",
"iso": "2014SunW10",
"name": "Week 10 - 2014-03-02 - 2014-03-08",
+ "periodType": "WEEKLY",
"startDate": "2014-03-02",
},
Object {
+ "displayName": "Week 11 - 2014-03-09 - 2014-03-15",
"endDate": "2014-03-15",
"id": "2014SunW11",
"iso": "2014SunW11",
"name": "Week 11 - 2014-03-09 - 2014-03-15",
+ "periodType": "WEEKLY",
"startDate": "2014-03-09",
},
Object {
+ "displayName": "Week 12 - 2014-03-16 - 2014-03-22",
"endDate": "2014-03-22",
"id": "2014SunW12",
"iso": "2014SunW12",
"name": "Week 12 - 2014-03-16 - 2014-03-22",
+ "periodType": "WEEKLY",
"startDate": "2014-03-16",
},
Object {
+ "displayName": "Week 13 - 2014-03-23 - 2014-03-29",
"endDate": "2014-03-29",
"id": "2014SunW13",
"iso": "2014SunW13",
"name": "Week 13 - 2014-03-23 - 2014-03-29",
+ "periodType": "WEEKLY",
"startDate": "2014-03-23",
},
Object {
+ "displayName": "Week 14 - 2014-03-30 - 2014-04-05",
"endDate": "2014-04-05",
"id": "2014SunW14",
"iso": "2014SunW14",
"name": "Week 14 - 2014-03-30 - 2014-04-05",
+ "periodType": "WEEKLY",
"startDate": "2014-03-30",
},
Object {
+ "displayName": "Week 15 - 2014-04-06 - 2014-04-12",
"endDate": "2014-04-12",
"id": "2014SunW15",
"iso": "2014SunW15",
"name": "Week 15 - 2014-04-06 - 2014-04-12",
+ "periodType": "WEEKLY",
"startDate": "2014-04-06",
},
Object {
+ "displayName": "Week 16 - 2014-04-13 - 2014-04-19",
"endDate": "2014-04-19",
"id": "2014SunW16",
"iso": "2014SunW16",
"name": "Week 16 - 2014-04-13 - 2014-04-19",
+ "periodType": "WEEKLY",
"startDate": "2014-04-13",
},
Object {
+ "displayName": "Week 17 - 2014-04-20 - 2014-04-26",
"endDate": "2014-04-26",
"id": "2014SunW17",
"iso": "2014SunW17",
"name": "Week 17 - 2014-04-20 - 2014-04-26",
+ "periodType": "WEEKLY",
"startDate": "2014-04-20",
},
Object {
+ "displayName": "Week 18 - 2014-04-27 - 2014-05-03",
"endDate": "2014-05-03",
"id": "2014SunW18",
"iso": "2014SunW18",
"name": "Week 18 - 2014-04-27 - 2014-05-03",
+ "periodType": "WEEKLY",
"startDate": "2014-04-27",
},
Object {
+ "displayName": "Week 19 - 2014-05-04 - 2014-05-10",
"endDate": "2014-05-10",
"id": "2014SunW19",
"iso": "2014SunW19",
"name": "Week 19 - 2014-05-04 - 2014-05-10",
+ "periodType": "WEEKLY",
"startDate": "2014-05-04",
},
Object {
+ "displayName": "Week 20 - 2014-05-11 - 2014-05-17",
"endDate": "2014-05-17",
"id": "2014SunW20",
"iso": "2014SunW20",
"name": "Week 20 - 2014-05-11 - 2014-05-17",
+ "periodType": "WEEKLY",
"startDate": "2014-05-11",
},
Object {
+ "displayName": "Week 21 - 2014-05-18 - 2014-05-24",
"endDate": "2014-05-24",
"id": "2014SunW21",
"iso": "2014SunW21",
"name": "Week 21 - 2014-05-18 - 2014-05-24",
+ "periodType": "WEEKLY",
"startDate": "2014-05-18",
},
Object {
+ "displayName": "Week 22 - 2014-05-25 - 2014-05-31",
"endDate": "2014-05-31",
"id": "2014SunW22",
"iso": "2014SunW22",
"name": "Week 22 - 2014-05-25 - 2014-05-31",
+ "periodType": "WEEKLY",
"startDate": "2014-05-25",
},
Object {
+ "displayName": "Week 23 - 2014-06-01 - 2014-06-07",
"endDate": "2014-06-07",
"id": "2014SunW23",
"iso": "2014SunW23",
"name": "Week 23 - 2014-06-01 - 2014-06-07",
+ "periodType": "WEEKLY",
"startDate": "2014-06-01",
},
Object {
+ "displayName": "Week 24 - 2014-06-08 - 2014-06-14",
"endDate": "2014-06-14",
"id": "2014SunW24",
"iso": "2014SunW24",
"name": "Week 24 - 2014-06-08 - 2014-06-14",
+ "periodType": "WEEKLY",
"startDate": "2014-06-08",
},
Object {
+ "displayName": "Week 25 - 2014-06-15 - 2014-06-21",
"endDate": "2014-06-21",
"id": "2014SunW25",
"iso": "2014SunW25",
"name": "Week 25 - 2014-06-15 - 2014-06-21",
+ "periodType": "WEEKLY",
"startDate": "2014-06-15",
},
Object {
+ "displayName": "Week 26 - 2014-06-22 - 2014-06-28",
"endDate": "2014-06-28",
"id": "2014SunW26",
"iso": "2014SunW26",
"name": "Week 26 - 2014-06-22 - 2014-06-28",
+ "periodType": "WEEKLY",
"startDate": "2014-06-22",
},
Object {
+ "displayName": "Week 27 - 2014-06-29 - 2014-07-05",
"endDate": "2014-07-05",
"id": "2014SunW27",
"iso": "2014SunW27",
"name": "Week 27 - 2014-06-29 - 2014-07-05",
+ "periodType": "WEEKLY",
"startDate": "2014-06-29",
},
Object {
+ "displayName": "Week 28 - 2014-07-06 - 2014-07-12",
"endDate": "2014-07-12",
"id": "2014SunW28",
"iso": "2014SunW28",
"name": "Week 28 - 2014-07-06 - 2014-07-12",
+ "periodType": "WEEKLY",
"startDate": "2014-07-06",
},
Object {
+ "displayName": "Week 29 - 2014-07-13 - 2014-07-19",
"endDate": "2014-07-19",
"id": "2014SunW29",
"iso": "2014SunW29",
"name": "Week 29 - 2014-07-13 - 2014-07-19",
+ "periodType": "WEEKLY",
"startDate": "2014-07-13",
},
Object {
+ "displayName": "Week 30 - 2014-07-20 - 2014-07-26",
"endDate": "2014-07-26",
"id": "2014SunW30",
"iso": "2014SunW30",
"name": "Week 30 - 2014-07-20 - 2014-07-26",
+ "periodType": "WEEKLY",
"startDate": "2014-07-20",
},
Object {
+ "displayName": "Week 31 - 2014-07-27 - 2014-08-02",
"endDate": "2014-08-02",
"id": "2014SunW31",
"iso": "2014SunW31",
"name": "Week 31 - 2014-07-27 - 2014-08-02",
+ "periodType": "WEEKLY",
"startDate": "2014-07-27",
},
Object {
+ "displayName": "Week 32 - 2014-08-03 - 2014-08-09",
"endDate": "2014-08-09",
"id": "2014SunW32",
"iso": "2014SunW32",
"name": "Week 32 - 2014-08-03 - 2014-08-09",
+ "periodType": "WEEKLY",
"startDate": "2014-08-03",
},
Object {
+ "displayName": "Week 33 - 2014-08-10 - 2014-08-16",
"endDate": "2014-08-16",
"id": "2014SunW33",
"iso": "2014SunW33",
"name": "Week 33 - 2014-08-10 - 2014-08-16",
+ "periodType": "WEEKLY",
"startDate": "2014-08-10",
},
Object {
+ "displayName": "Week 34 - 2014-08-17 - 2014-08-23",
"endDate": "2014-08-23",
"id": "2014SunW34",
"iso": "2014SunW34",
"name": "Week 34 - 2014-08-17 - 2014-08-23",
+ "periodType": "WEEKLY",
"startDate": "2014-08-17",
},
Object {
+ "displayName": "Week 35 - 2014-08-24 - 2014-08-30",
"endDate": "2014-08-30",
"id": "2014SunW35",
"iso": "2014SunW35",
"name": "Week 35 - 2014-08-24 - 2014-08-30",
+ "periodType": "WEEKLY",
"startDate": "2014-08-24",
},
Object {
+ "displayName": "Week 36 - 2014-08-31 - 2014-09-06",
"endDate": "2014-09-06",
"id": "2014SunW36",
"iso": "2014SunW36",
"name": "Week 36 - 2014-08-31 - 2014-09-06",
+ "periodType": "WEEKLY",
"startDate": "2014-08-31",
},
Object {
+ "displayName": "Week 37 - 2014-09-07 - 2014-09-13",
"endDate": "2014-09-13",
"id": "2014SunW37",
"iso": "2014SunW37",
"name": "Week 37 - 2014-09-07 - 2014-09-13",
+ "periodType": "WEEKLY",
"startDate": "2014-09-07",
},
Object {
+ "displayName": "Week 38 - 2014-09-14 - 2014-09-20",
"endDate": "2014-09-20",
"id": "2014SunW38",
"iso": "2014SunW38",
"name": "Week 38 - 2014-09-14 - 2014-09-20",
+ "periodType": "WEEKLY",
"startDate": "2014-09-14",
},
Object {
+ "displayName": "Week 39 - 2014-09-21 - 2014-09-27",
"endDate": "2014-09-27",
"id": "2014SunW39",
"iso": "2014SunW39",
"name": "Week 39 - 2014-09-21 - 2014-09-27",
+ "periodType": "WEEKLY",
"startDate": "2014-09-21",
},
Object {
+ "displayName": "Week 40 - 2014-09-28 - 2014-10-04",
"endDate": "2014-10-04",
"id": "2014SunW40",
"iso": "2014SunW40",
"name": "Week 40 - 2014-09-28 - 2014-10-04",
+ "periodType": "WEEKLY",
"startDate": "2014-09-28",
},
Object {
+ "displayName": "Week 41 - 2014-10-05 - 2014-10-11",
"endDate": "2014-10-11",
"id": "2014SunW41",
"iso": "2014SunW41",
"name": "Week 41 - 2014-10-05 - 2014-10-11",
+ "periodType": "WEEKLY",
"startDate": "2014-10-05",
},
Object {
+ "displayName": "Week 42 - 2014-10-12 - 2014-10-18",
"endDate": "2014-10-18",
"id": "2014SunW42",
"iso": "2014SunW42",
"name": "Week 42 - 2014-10-12 - 2014-10-18",
+ "periodType": "WEEKLY",
"startDate": "2014-10-12",
},
Object {
+ "displayName": "Week 43 - 2014-10-19 - 2014-10-25",
"endDate": "2014-10-25",
"id": "2014SunW43",
"iso": "2014SunW43",
"name": "Week 43 - 2014-10-19 - 2014-10-25",
+ "periodType": "WEEKLY",
"startDate": "2014-10-19",
},
Object {
+ "displayName": "Week 44 - 2014-10-26 - 2014-11-01",
"endDate": "2014-11-01",
"id": "2014SunW44",
"iso": "2014SunW44",
"name": "Week 44 - 2014-10-26 - 2014-11-01",
+ "periodType": "WEEKLY",
"startDate": "2014-10-26",
},
Object {
+ "displayName": "Week 45 - 2014-11-02 - 2014-11-08",
"endDate": "2014-11-08",
"id": "2014SunW45",
"iso": "2014SunW45",
"name": "Week 45 - 2014-11-02 - 2014-11-08",
+ "periodType": "WEEKLY",
"startDate": "2014-11-02",
},
Object {
+ "displayName": "Week 46 - 2014-11-09 - 2014-11-15",
"endDate": "2014-11-15",
"id": "2014SunW46",
"iso": "2014SunW46",
"name": "Week 46 - 2014-11-09 - 2014-11-15",
+ "periodType": "WEEKLY",
"startDate": "2014-11-09",
},
Object {
+ "displayName": "Week 47 - 2014-11-16 - 2014-11-22",
"endDate": "2014-11-22",
"id": "2014SunW47",
"iso": "2014SunW47",
"name": "Week 47 - 2014-11-16 - 2014-11-22",
+ "periodType": "WEEKLY",
"startDate": "2014-11-16",
},
Object {
+ "displayName": "Week 48 - 2014-11-23 - 2014-11-29",
"endDate": "2014-11-29",
"id": "2014SunW48",
"iso": "2014SunW48",
"name": "Week 48 - 2014-11-23 - 2014-11-29",
+ "periodType": "WEEKLY",
"startDate": "2014-11-23",
},
Object {
+ "displayName": "Week 49 - 2014-11-30 - 2014-12-06",
"endDate": "2014-12-06",
"id": "2014SunW49",
"iso": "2014SunW49",
"name": "Week 49 - 2014-11-30 - 2014-12-06",
+ "periodType": "WEEKLY",
"startDate": "2014-11-30",
},
Object {
+ "displayName": "Week 50 - 2014-12-07 - 2014-12-13",
"endDate": "2014-12-13",
"id": "2014SunW50",
"iso": "2014SunW50",
"name": "Week 50 - 2014-12-07 - 2014-12-13",
+ "periodType": "WEEKLY",
"startDate": "2014-12-07",
},
Object {
+ "displayName": "Week 51 - 2014-12-14 - 2014-12-20",
"endDate": "2014-12-20",
"id": "2014SunW51",
"iso": "2014SunW51",
"name": "Week 51 - 2014-12-14 - 2014-12-20",
+ "periodType": "WEEKLY",
"startDate": "2014-12-14",
},
Object {
+ "displayName": "Week 52 - 2014-12-21 - 2014-12-27",
"endDate": "2014-12-27",
"id": "2014SunW52",
"iso": "2014SunW52",
"name": "Week 52 - 2014-12-21 - 2014-12-27",
+ "periodType": "WEEKLY",
"startDate": "2014-12-21",
},
Object {
+ "displayName": "Week 53 - 2014-12-28 - 2015-01-03",
"endDate": "2015-01-03",
"id": "2014SunW53",
"iso": "2014SunW53",
"name": "Week 53 - 2014-12-28 - 2015-01-03",
+ "periodType": "WEEKLY",
"startDate": "2014-12-28",
},
],
@@ -4785,367 +6159,471 @@ Object {
"name": "Weekly (Start Thursday)",
"options": Array [
Object {
+ "displayName": "Week 1 - 2014-01-02 - 2014-01-08",
"endDate": "2014-01-08",
"id": "2014ThuW1",
"iso": "2014ThuW1",
"name": "Week 1 - 2014-01-02 - 2014-01-08",
+ "periodType": "WEEKLY",
"startDate": "2014-01-02",
},
Object {
+ "displayName": "Week 2 - 2014-01-09 - 2014-01-15",
"endDate": "2014-01-15",
"id": "2014ThuW2",
"iso": "2014ThuW2",
"name": "Week 2 - 2014-01-09 - 2014-01-15",
+ "periodType": "WEEKLY",
"startDate": "2014-01-09",
},
Object {
+ "displayName": "Week 3 - 2014-01-16 - 2014-01-22",
"endDate": "2014-01-22",
"id": "2014ThuW3",
"iso": "2014ThuW3",
"name": "Week 3 - 2014-01-16 - 2014-01-22",
+ "periodType": "WEEKLY",
"startDate": "2014-01-16",
},
Object {
+ "displayName": "Week 4 - 2014-01-23 - 2014-01-29",
"endDate": "2014-01-29",
"id": "2014ThuW4",
"iso": "2014ThuW4",
"name": "Week 4 - 2014-01-23 - 2014-01-29",
+ "periodType": "WEEKLY",
"startDate": "2014-01-23",
},
Object {
+ "displayName": "Week 5 - 2014-01-30 - 2014-02-05",
"endDate": "2014-02-05",
"id": "2014ThuW5",
"iso": "2014ThuW5",
"name": "Week 5 - 2014-01-30 - 2014-02-05",
+ "periodType": "WEEKLY",
"startDate": "2014-01-30",
},
Object {
+ "displayName": "Week 6 - 2014-02-06 - 2014-02-12",
"endDate": "2014-02-12",
"id": "2014ThuW6",
"iso": "2014ThuW6",
"name": "Week 6 - 2014-02-06 - 2014-02-12",
+ "periodType": "WEEKLY",
"startDate": "2014-02-06",
},
Object {
+ "displayName": "Week 7 - 2014-02-13 - 2014-02-19",
"endDate": "2014-02-19",
"id": "2014ThuW7",
"iso": "2014ThuW7",
"name": "Week 7 - 2014-02-13 - 2014-02-19",
+ "periodType": "WEEKLY",
"startDate": "2014-02-13",
},
Object {
+ "displayName": "Week 8 - 2014-02-20 - 2014-02-26",
"endDate": "2014-02-26",
"id": "2014ThuW8",
"iso": "2014ThuW8",
"name": "Week 8 - 2014-02-20 - 2014-02-26",
+ "periodType": "WEEKLY",
"startDate": "2014-02-20",
},
Object {
+ "displayName": "Week 9 - 2014-02-27 - 2014-03-05",
"endDate": "2014-03-05",
"id": "2014ThuW9",
"iso": "2014ThuW9",
"name": "Week 9 - 2014-02-27 - 2014-03-05",
+ "periodType": "WEEKLY",
"startDate": "2014-02-27",
},
Object {
+ "displayName": "Week 10 - 2014-03-06 - 2014-03-12",
"endDate": "2014-03-12",
"id": "2014ThuW10",
"iso": "2014ThuW10",
"name": "Week 10 - 2014-03-06 - 2014-03-12",
+ "periodType": "WEEKLY",
"startDate": "2014-03-06",
},
Object {
+ "displayName": "Week 11 - 2014-03-13 - 2014-03-19",
"endDate": "2014-03-19",
"id": "2014ThuW11",
"iso": "2014ThuW11",
"name": "Week 11 - 2014-03-13 - 2014-03-19",
+ "periodType": "WEEKLY",
"startDate": "2014-03-13",
},
Object {
+ "displayName": "Week 12 - 2014-03-20 - 2014-03-26",
"endDate": "2014-03-26",
"id": "2014ThuW12",
"iso": "2014ThuW12",
"name": "Week 12 - 2014-03-20 - 2014-03-26",
+ "periodType": "WEEKLY",
"startDate": "2014-03-20",
},
Object {
+ "displayName": "Week 13 - 2014-03-27 - 2014-04-02",
"endDate": "2014-04-02",
"id": "2014ThuW13",
"iso": "2014ThuW13",
"name": "Week 13 - 2014-03-27 - 2014-04-02",
+ "periodType": "WEEKLY",
"startDate": "2014-03-27",
},
Object {
+ "displayName": "Week 14 - 2014-04-03 - 2014-04-09",
"endDate": "2014-04-09",
"id": "2014ThuW14",
"iso": "2014ThuW14",
"name": "Week 14 - 2014-04-03 - 2014-04-09",
+ "periodType": "WEEKLY",
"startDate": "2014-04-03",
},
Object {
+ "displayName": "Week 15 - 2014-04-10 - 2014-04-16",
"endDate": "2014-04-16",
"id": "2014ThuW15",
"iso": "2014ThuW15",
"name": "Week 15 - 2014-04-10 - 2014-04-16",
+ "periodType": "WEEKLY",
"startDate": "2014-04-10",
},
Object {
+ "displayName": "Week 16 - 2014-04-17 - 2014-04-23",
"endDate": "2014-04-23",
"id": "2014ThuW16",
"iso": "2014ThuW16",
"name": "Week 16 - 2014-04-17 - 2014-04-23",
+ "periodType": "WEEKLY",
"startDate": "2014-04-17",
},
Object {
+ "displayName": "Week 17 - 2014-04-24 - 2014-04-30",
"endDate": "2014-04-30",
"id": "2014ThuW17",
"iso": "2014ThuW17",
"name": "Week 17 - 2014-04-24 - 2014-04-30",
+ "periodType": "WEEKLY",
"startDate": "2014-04-24",
},
Object {
+ "displayName": "Week 18 - 2014-05-01 - 2014-05-07",
"endDate": "2014-05-07",
"id": "2014ThuW18",
"iso": "2014ThuW18",
"name": "Week 18 - 2014-05-01 - 2014-05-07",
+ "periodType": "WEEKLY",
"startDate": "2014-05-01",
},
Object {
+ "displayName": "Week 19 - 2014-05-08 - 2014-05-14",
"endDate": "2014-05-14",
"id": "2014ThuW19",
"iso": "2014ThuW19",
"name": "Week 19 - 2014-05-08 - 2014-05-14",
+ "periodType": "WEEKLY",
"startDate": "2014-05-08",
},
Object {
+ "displayName": "Week 20 - 2014-05-15 - 2014-05-21",
"endDate": "2014-05-21",
"id": "2014ThuW20",
"iso": "2014ThuW20",
"name": "Week 20 - 2014-05-15 - 2014-05-21",
+ "periodType": "WEEKLY",
"startDate": "2014-05-15",
},
Object {
+ "displayName": "Week 21 - 2014-05-22 - 2014-05-28",
"endDate": "2014-05-28",
"id": "2014ThuW21",
"iso": "2014ThuW21",
"name": "Week 21 - 2014-05-22 - 2014-05-28",
+ "periodType": "WEEKLY",
"startDate": "2014-05-22",
},
Object {
+ "displayName": "Week 22 - 2014-05-29 - 2014-06-04",
"endDate": "2014-06-04",
"id": "2014ThuW22",
"iso": "2014ThuW22",
"name": "Week 22 - 2014-05-29 - 2014-06-04",
+ "periodType": "WEEKLY",
"startDate": "2014-05-29",
},
Object {
+ "displayName": "Week 23 - 2014-06-05 - 2014-06-11",
"endDate": "2014-06-11",
"id": "2014ThuW23",
"iso": "2014ThuW23",
"name": "Week 23 - 2014-06-05 - 2014-06-11",
+ "periodType": "WEEKLY",
"startDate": "2014-06-05",
},
Object {
+ "displayName": "Week 24 - 2014-06-12 - 2014-06-18",
"endDate": "2014-06-18",
"id": "2014ThuW24",
"iso": "2014ThuW24",
"name": "Week 24 - 2014-06-12 - 2014-06-18",
+ "periodType": "WEEKLY",
"startDate": "2014-06-12",
},
Object {
+ "displayName": "Week 25 - 2014-06-19 - 2014-06-25",
"endDate": "2014-06-25",
"id": "2014ThuW25",
"iso": "2014ThuW25",
"name": "Week 25 - 2014-06-19 - 2014-06-25",
+ "periodType": "WEEKLY",
"startDate": "2014-06-19",
},
Object {
+ "displayName": "Week 26 - 2014-06-26 - 2014-07-02",
"endDate": "2014-07-02",
"id": "2014ThuW26",
"iso": "2014ThuW26",
"name": "Week 26 - 2014-06-26 - 2014-07-02",
+ "periodType": "WEEKLY",
"startDate": "2014-06-26",
},
Object {
+ "displayName": "Week 27 - 2014-07-03 - 2014-07-09",
"endDate": "2014-07-09",
"id": "2014ThuW27",
"iso": "2014ThuW27",
"name": "Week 27 - 2014-07-03 - 2014-07-09",
+ "periodType": "WEEKLY",
"startDate": "2014-07-03",
},
Object {
+ "displayName": "Week 28 - 2014-07-10 - 2014-07-16",
"endDate": "2014-07-16",
"id": "2014ThuW28",
"iso": "2014ThuW28",
"name": "Week 28 - 2014-07-10 - 2014-07-16",
+ "periodType": "WEEKLY",
"startDate": "2014-07-10",
},
Object {
+ "displayName": "Week 29 - 2014-07-17 - 2014-07-23",
"endDate": "2014-07-23",
"id": "2014ThuW29",
"iso": "2014ThuW29",
"name": "Week 29 - 2014-07-17 - 2014-07-23",
+ "periodType": "WEEKLY",
"startDate": "2014-07-17",
},
Object {
+ "displayName": "Week 30 - 2014-07-24 - 2014-07-30",
"endDate": "2014-07-30",
"id": "2014ThuW30",
"iso": "2014ThuW30",
"name": "Week 30 - 2014-07-24 - 2014-07-30",
+ "periodType": "WEEKLY",
"startDate": "2014-07-24",
},
Object {
+ "displayName": "Week 31 - 2014-07-31 - 2014-08-06",
"endDate": "2014-08-06",
"id": "2014ThuW31",
"iso": "2014ThuW31",
"name": "Week 31 - 2014-07-31 - 2014-08-06",
+ "periodType": "WEEKLY",
"startDate": "2014-07-31",
},
Object {
+ "displayName": "Week 32 - 2014-08-07 - 2014-08-13",
"endDate": "2014-08-13",
"id": "2014ThuW32",
"iso": "2014ThuW32",
"name": "Week 32 - 2014-08-07 - 2014-08-13",
+ "periodType": "WEEKLY",
"startDate": "2014-08-07",
},
Object {
+ "displayName": "Week 33 - 2014-08-14 - 2014-08-20",
"endDate": "2014-08-20",
"id": "2014ThuW33",
"iso": "2014ThuW33",
"name": "Week 33 - 2014-08-14 - 2014-08-20",
+ "periodType": "WEEKLY",
"startDate": "2014-08-14",
},
Object {
+ "displayName": "Week 34 - 2014-08-21 - 2014-08-27",
"endDate": "2014-08-27",
"id": "2014ThuW34",
"iso": "2014ThuW34",
"name": "Week 34 - 2014-08-21 - 2014-08-27",
+ "periodType": "WEEKLY",
"startDate": "2014-08-21",
},
Object {
+ "displayName": "Week 35 - 2014-08-28 - 2014-09-03",
"endDate": "2014-09-03",
"id": "2014ThuW35",
"iso": "2014ThuW35",
"name": "Week 35 - 2014-08-28 - 2014-09-03",
+ "periodType": "WEEKLY",
"startDate": "2014-08-28",
},
Object {
+ "displayName": "Week 36 - 2014-09-04 - 2014-09-10",
"endDate": "2014-09-10",
"id": "2014ThuW36",
"iso": "2014ThuW36",
"name": "Week 36 - 2014-09-04 - 2014-09-10",
+ "periodType": "WEEKLY",
"startDate": "2014-09-04",
},
Object {
+ "displayName": "Week 37 - 2014-09-11 - 2014-09-17",
"endDate": "2014-09-17",
"id": "2014ThuW37",
"iso": "2014ThuW37",
"name": "Week 37 - 2014-09-11 - 2014-09-17",
+ "periodType": "WEEKLY",
"startDate": "2014-09-11",
},
Object {
+ "displayName": "Week 38 - 2014-09-18 - 2014-09-24",
"endDate": "2014-09-24",
"id": "2014ThuW38",
"iso": "2014ThuW38",
"name": "Week 38 - 2014-09-18 - 2014-09-24",
+ "periodType": "WEEKLY",
"startDate": "2014-09-18",
},
Object {
+ "displayName": "Week 39 - 2014-09-25 - 2014-10-01",
"endDate": "2014-10-01",
"id": "2014ThuW39",
"iso": "2014ThuW39",
"name": "Week 39 - 2014-09-25 - 2014-10-01",
+ "periodType": "WEEKLY",
"startDate": "2014-09-25",
},
Object {
+ "displayName": "Week 40 - 2014-10-02 - 2014-10-08",
"endDate": "2014-10-08",
"id": "2014ThuW40",
"iso": "2014ThuW40",
"name": "Week 40 - 2014-10-02 - 2014-10-08",
+ "periodType": "WEEKLY",
"startDate": "2014-10-02",
},
Object {
+ "displayName": "Week 41 - 2014-10-09 - 2014-10-15",
"endDate": "2014-10-15",
"id": "2014ThuW41",
"iso": "2014ThuW41",
"name": "Week 41 - 2014-10-09 - 2014-10-15",
+ "periodType": "WEEKLY",
"startDate": "2014-10-09",
},
Object {
+ "displayName": "Week 42 - 2014-10-16 - 2014-10-22",
"endDate": "2014-10-22",
"id": "2014ThuW42",
"iso": "2014ThuW42",
"name": "Week 42 - 2014-10-16 - 2014-10-22",
+ "periodType": "WEEKLY",
"startDate": "2014-10-16",
},
Object {
+ "displayName": "Week 43 - 2014-10-23 - 2014-10-29",
"endDate": "2014-10-29",
"id": "2014ThuW43",
"iso": "2014ThuW43",
"name": "Week 43 - 2014-10-23 - 2014-10-29",
+ "periodType": "WEEKLY",
"startDate": "2014-10-23",
},
Object {
+ "displayName": "Week 44 - 2014-10-30 - 2014-11-05",
"endDate": "2014-11-05",
"id": "2014ThuW44",
"iso": "2014ThuW44",
"name": "Week 44 - 2014-10-30 - 2014-11-05",
+ "periodType": "WEEKLY",
"startDate": "2014-10-30",
},
Object {
+ "displayName": "Week 45 - 2014-11-06 - 2014-11-12",
"endDate": "2014-11-12",
"id": "2014ThuW45",
"iso": "2014ThuW45",
"name": "Week 45 - 2014-11-06 - 2014-11-12",
+ "periodType": "WEEKLY",
"startDate": "2014-11-06",
},
Object {
+ "displayName": "Week 46 - 2014-11-13 - 2014-11-19",
"endDate": "2014-11-19",
"id": "2014ThuW46",
"iso": "2014ThuW46",
"name": "Week 46 - 2014-11-13 - 2014-11-19",
+ "periodType": "WEEKLY",
"startDate": "2014-11-13",
},
Object {
+ "displayName": "Week 47 - 2014-11-20 - 2014-11-26",
"endDate": "2014-11-26",
"id": "2014ThuW47",
"iso": "2014ThuW47",
"name": "Week 47 - 2014-11-20 - 2014-11-26",
+ "periodType": "WEEKLY",
"startDate": "2014-11-20",
},
Object {
+ "displayName": "Week 48 - 2014-11-27 - 2014-12-03",
"endDate": "2014-12-03",
"id": "2014ThuW48",
"iso": "2014ThuW48",
"name": "Week 48 - 2014-11-27 - 2014-12-03",
+ "periodType": "WEEKLY",
"startDate": "2014-11-27",
},
Object {
+ "displayName": "Week 49 - 2014-12-04 - 2014-12-10",
"endDate": "2014-12-10",
"id": "2014ThuW49",
"iso": "2014ThuW49",
"name": "Week 49 - 2014-12-04 - 2014-12-10",
+ "periodType": "WEEKLY",
"startDate": "2014-12-04",
},
Object {
+ "displayName": "Week 50 - 2014-12-11 - 2014-12-17",
"endDate": "2014-12-17",
"id": "2014ThuW50",
"iso": "2014ThuW50",
"name": "Week 50 - 2014-12-11 - 2014-12-17",
+ "periodType": "WEEKLY",
"startDate": "2014-12-11",
},
Object {
+ "displayName": "Week 51 - 2014-12-18 - 2014-12-24",
"endDate": "2014-12-24",
"id": "2014ThuW51",
"iso": "2014ThuW51",
"name": "Week 51 - 2014-12-18 - 2014-12-24",
+ "periodType": "WEEKLY",
"startDate": "2014-12-18",
},
Object {
+ "displayName": "Week 52 - 2014-12-25 - 2014-12-31",
"endDate": "2014-12-31",
"id": "2014ThuW52",
"iso": "2014ThuW52",
"name": "Week 52 - 2014-12-25 - 2014-12-31",
+ "periodType": "WEEKLY",
"startDate": "2014-12-25",
},
],
@@ -5159,367 +6637,471 @@ Object {
"name": "Weekly (Start Wednesday)",
"options": Array [
Object {
+ "displayName": "Week 1 - 2014-01-01 - 2014-01-07",
"endDate": "2014-01-07",
"id": "2014WedW1",
"iso": "2014WedW1",
"name": "Week 1 - 2014-01-01 - 2014-01-07",
+ "periodType": "WEEKLY",
"startDate": "2014-01-01",
},
Object {
+ "displayName": "Week 2 - 2014-01-08 - 2014-01-14",
"endDate": "2014-01-14",
"id": "2014WedW2",
"iso": "2014WedW2",
"name": "Week 2 - 2014-01-08 - 2014-01-14",
+ "periodType": "WEEKLY",
"startDate": "2014-01-08",
},
Object {
+ "displayName": "Week 3 - 2014-01-15 - 2014-01-21",
"endDate": "2014-01-21",
"id": "2014WedW3",
"iso": "2014WedW3",
"name": "Week 3 - 2014-01-15 - 2014-01-21",
+ "periodType": "WEEKLY",
"startDate": "2014-01-15",
},
Object {
+ "displayName": "Week 4 - 2014-01-22 - 2014-01-28",
"endDate": "2014-01-28",
"id": "2014WedW4",
"iso": "2014WedW4",
"name": "Week 4 - 2014-01-22 - 2014-01-28",
+ "periodType": "WEEKLY",
"startDate": "2014-01-22",
},
Object {
+ "displayName": "Week 5 - 2014-01-29 - 2014-02-04",
"endDate": "2014-02-04",
"id": "2014WedW5",
"iso": "2014WedW5",
"name": "Week 5 - 2014-01-29 - 2014-02-04",
+ "periodType": "WEEKLY",
"startDate": "2014-01-29",
},
Object {
+ "displayName": "Week 6 - 2014-02-05 - 2014-02-11",
"endDate": "2014-02-11",
"id": "2014WedW6",
"iso": "2014WedW6",
"name": "Week 6 - 2014-02-05 - 2014-02-11",
+ "periodType": "WEEKLY",
"startDate": "2014-02-05",
},
Object {
+ "displayName": "Week 7 - 2014-02-12 - 2014-02-18",
"endDate": "2014-02-18",
"id": "2014WedW7",
"iso": "2014WedW7",
"name": "Week 7 - 2014-02-12 - 2014-02-18",
+ "periodType": "WEEKLY",
"startDate": "2014-02-12",
},
Object {
+ "displayName": "Week 8 - 2014-02-19 - 2014-02-25",
"endDate": "2014-02-25",
"id": "2014WedW8",
"iso": "2014WedW8",
"name": "Week 8 - 2014-02-19 - 2014-02-25",
+ "periodType": "WEEKLY",
"startDate": "2014-02-19",
},
Object {
+ "displayName": "Week 9 - 2014-02-26 - 2014-03-04",
"endDate": "2014-03-04",
"id": "2014WedW9",
"iso": "2014WedW9",
"name": "Week 9 - 2014-02-26 - 2014-03-04",
+ "periodType": "WEEKLY",
"startDate": "2014-02-26",
},
Object {
+ "displayName": "Week 10 - 2014-03-05 - 2014-03-11",
"endDate": "2014-03-11",
"id": "2014WedW10",
"iso": "2014WedW10",
"name": "Week 10 - 2014-03-05 - 2014-03-11",
+ "periodType": "WEEKLY",
"startDate": "2014-03-05",
},
Object {
+ "displayName": "Week 11 - 2014-03-12 - 2014-03-18",
"endDate": "2014-03-18",
"id": "2014WedW11",
"iso": "2014WedW11",
"name": "Week 11 - 2014-03-12 - 2014-03-18",
+ "periodType": "WEEKLY",
"startDate": "2014-03-12",
},
Object {
+ "displayName": "Week 12 - 2014-03-19 - 2014-03-25",
"endDate": "2014-03-25",
"id": "2014WedW12",
"iso": "2014WedW12",
"name": "Week 12 - 2014-03-19 - 2014-03-25",
+ "periodType": "WEEKLY",
"startDate": "2014-03-19",
},
Object {
+ "displayName": "Week 13 - 2014-03-26 - 2014-04-01",
"endDate": "2014-04-01",
"id": "2014WedW13",
"iso": "2014WedW13",
"name": "Week 13 - 2014-03-26 - 2014-04-01",
+ "periodType": "WEEKLY",
"startDate": "2014-03-26",
},
Object {
+ "displayName": "Week 14 - 2014-04-02 - 2014-04-08",
"endDate": "2014-04-08",
"id": "2014WedW14",
"iso": "2014WedW14",
"name": "Week 14 - 2014-04-02 - 2014-04-08",
+ "periodType": "WEEKLY",
"startDate": "2014-04-02",
},
Object {
+ "displayName": "Week 15 - 2014-04-09 - 2014-04-15",
"endDate": "2014-04-15",
"id": "2014WedW15",
"iso": "2014WedW15",
"name": "Week 15 - 2014-04-09 - 2014-04-15",
+ "periodType": "WEEKLY",
"startDate": "2014-04-09",
},
Object {
+ "displayName": "Week 16 - 2014-04-16 - 2014-04-22",
"endDate": "2014-04-22",
"id": "2014WedW16",
"iso": "2014WedW16",
"name": "Week 16 - 2014-04-16 - 2014-04-22",
+ "periodType": "WEEKLY",
"startDate": "2014-04-16",
},
Object {
+ "displayName": "Week 17 - 2014-04-23 - 2014-04-29",
"endDate": "2014-04-29",
"id": "2014WedW17",
"iso": "2014WedW17",
"name": "Week 17 - 2014-04-23 - 2014-04-29",
+ "periodType": "WEEKLY",
"startDate": "2014-04-23",
},
Object {
+ "displayName": "Week 18 - 2014-04-30 - 2014-05-06",
"endDate": "2014-05-06",
"id": "2014WedW18",
"iso": "2014WedW18",
"name": "Week 18 - 2014-04-30 - 2014-05-06",
+ "periodType": "WEEKLY",
"startDate": "2014-04-30",
},
Object {
+ "displayName": "Week 19 - 2014-05-07 - 2014-05-13",
"endDate": "2014-05-13",
"id": "2014WedW19",
"iso": "2014WedW19",
"name": "Week 19 - 2014-05-07 - 2014-05-13",
+ "periodType": "WEEKLY",
"startDate": "2014-05-07",
},
Object {
+ "displayName": "Week 20 - 2014-05-14 - 2014-05-20",
"endDate": "2014-05-20",
"id": "2014WedW20",
"iso": "2014WedW20",
"name": "Week 20 - 2014-05-14 - 2014-05-20",
+ "periodType": "WEEKLY",
"startDate": "2014-05-14",
},
Object {
+ "displayName": "Week 21 - 2014-05-21 - 2014-05-27",
"endDate": "2014-05-27",
"id": "2014WedW21",
"iso": "2014WedW21",
"name": "Week 21 - 2014-05-21 - 2014-05-27",
+ "periodType": "WEEKLY",
"startDate": "2014-05-21",
},
Object {
+ "displayName": "Week 22 - 2014-05-28 - 2014-06-03",
"endDate": "2014-06-03",
"id": "2014WedW22",
"iso": "2014WedW22",
"name": "Week 22 - 2014-05-28 - 2014-06-03",
+ "periodType": "WEEKLY",
"startDate": "2014-05-28",
},
Object {
+ "displayName": "Week 23 - 2014-06-04 - 2014-06-10",
"endDate": "2014-06-10",
"id": "2014WedW23",
"iso": "2014WedW23",
"name": "Week 23 - 2014-06-04 - 2014-06-10",
+ "periodType": "WEEKLY",
"startDate": "2014-06-04",
},
Object {
+ "displayName": "Week 24 - 2014-06-11 - 2014-06-17",
"endDate": "2014-06-17",
"id": "2014WedW24",
"iso": "2014WedW24",
"name": "Week 24 - 2014-06-11 - 2014-06-17",
+ "periodType": "WEEKLY",
"startDate": "2014-06-11",
},
Object {
+ "displayName": "Week 25 - 2014-06-18 - 2014-06-24",
"endDate": "2014-06-24",
"id": "2014WedW25",
"iso": "2014WedW25",
"name": "Week 25 - 2014-06-18 - 2014-06-24",
+ "periodType": "WEEKLY",
"startDate": "2014-06-18",
},
Object {
+ "displayName": "Week 26 - 2014-06-25 - 2014-07-01",
"endDate": "2014-07-01",
"id": "2014WedW26",
"iso": "2014WedW26",
"name": "Week 26 - 2014-06-25 - 2014-07-01",
+ "periodType": "WEEKLY",
"startDate": "2014-06-25",
},
Object {
+ "displayName": "Week 27 - 2014-07-02 - 2014-07-08",
"endDate": "2014-07-08",
"id": "2014WedW27",
"iso": "2014WedW27",
"name": "Week 27 - 2014-07-02 - 2014-07-08",
+ "periodType": "WEEKLY",
"startDate": "2014-07-02",
},
Object {
+ "displayName": "Week 28 - 2014-07-09 - 2014-07-15",
"endDate": "2014-07-15",
"id": "2014WedW28",
"iso": "2014WedW28",
"name": "Week 28 - 2014-07-09 - 2014-07-15",
+ "periodType": "WEEKLY",
"startDate": "2014-07-09",
},
Object {
+ "displayName": "Week 29 - 2014-07-16 - 2014-07-22",
"endDate": "2014-07-22",
"id": "2014WedW29",
"iso": "2014WedW29",
"name": "Week 29 - 2014-07-16 - 2014-07-22",
+ "periodType": "WEEKLY",
"startDate": "2014-07-16",
},
Object {
+ "displayName": "Week 30 - 2014-07-23 - 2014-07-29",
"endDate": "2014-07-29",
"id": "2014WedW30",
"iso": "2014WedW30",
"name": "Week 30 - 2014-07-23 - 2014-07-29",
+ "periodType": "WEEKLY",
"startDate": "2014-07-23",
},
Object {
+ "displayName": "Week 31 - 2014-07-30 - 2014-08-05",
"endDate": "2014-08-05",
"id": "2014WedW31",
"iso": "2014WedW31",
"name": "Week 31 - 2014-07-30 - 2014-08-05",
+ "periodType": "WEEKLY",
"startDate": "2014-07-30",
},
Object {
+ "displayName": "Week 32 - 2014-08-06 - 2014-08-12",
"endDate": "2014-08-12",
"id": "2014WedW32",
"iso": "2014WedW32",
"name": "Week 32 - 2014-08-06 - 2014-08-12",
+ "periodType": "WEEKLY",
"startDate": "2014-08-06",
},
Object {
+ "displayName": "Week 33 - 2014-08-13 - 2014-08-19",
"endDate": "2014-08-19",
"id": "2014WedW33",
"iso": "2014WedW33",
"name": "Week 33 - 2014-08-13 - 2014-08-19",
+ "periodType": "WEEKLY",
"startDate": "2014-08-13",
},
Object {
+ "displayName": "Week 34 - 2014-08-20 - 2014-08-26",
"endDate": "2014-08-26",
"id": "2014WedW34",
"iso": "2014WedW34",
"name": "Week 34 - 2014-08-20 - 2014-08-26",
+ "periodType": "WEEKLY",
"startDate": "2014-08-20",
},
Object {
+ "displayName": "Week 35 - 2014-08-27 - 2014-09-02",
"endDate": "2014-09-02",
"id": "2014WedW35",
"iso": "2014WedW35",
"name": "Week 35 - 2014-08-27 - 2014-09-02",
+ "periodType": "WEEKLY",
"startDate": "2014-08-27",
},
Object {
+ "displayName": "Week 36 - 2014-09-03 - 2014-09-09",
"endDate": "2014-09-09",
"id": "2014WedW36",
"iso": "2014WedW36",
"name": "Week 36 - 2014-09-03 - 2014-09-09",
+ "periodType": "WEEKLY",
"startDate": "2014-09-03",
},
Object {
+ "displayName": "Week 37 - 2014-09-10 - 2014-09-16",
"endDate": "2014-09-16",
"id": "2014WedW37",
"iso": "2014WedW37",
"name": "Week 37 - 2014-09-10 - 2014-09-16",
+ "periodType": "WEEKLY",
"startDate": "2014-09-10",
},
Object {
+ "displayName": "Week 38 - 2014-09-17 - 2014-09-23",
"endDate": "2014-09-23",
"id": "2014WedW38",
"iso": "2014WedW38",
"name": "Week 38 - 2014-09-17 - 2014-09-23",
+ "periodType": "WEEKLY",
"startDate": "2014-09-17",
},
Object {
+ "displayName": "Week 39 - 2014-09-24 - 2014-09-30",
"endDate": "2014-09-30",
"id": "2014WedW39",
"iso": "2014WedW39",
"name": "Week 39 - 2014-09-24 - 2014-09-30",
+ "periodType": "WEEKLY",
"startDate": "2014-09-24",
},
Object {
+ "displayName": "Week 40 - 2014-10-01 - 2014-10-07",
"endDate": "2014-10-07",
"id": "2014WedW40",
"iso": "2014WedW40",
"name": "Week 40 - 2014-10-01 - 2014-10-07",
+ "periodType": "WEEKLY",
"startDate": "2014-10-01",
},
Object {
+ "displayName": "Week 41 - 2014-10-08 - 2014-10-14",
"endDate": "2014-10-14",
"id": "2014WedW41",
"iso": "2014WedW41",
"name": "Week 41 - 2014-10-08 - 2014-10-14",
+ "periodType": "WEEKLY",
"startDate": "2014-10-08",
},
Object {
+ "displayName": "Week 42 - 2014-10-15 - 2014-10-21",
"endDate": "2014-10-21",
"id": "2014WedW42",
"iso": "2014WedW42",
"name": "Week 42 - 2014-10-15 - 2014-10-21",
+ "periodType": "WEEKLY",
"startDate": "2014-10-15",
},
Object {
+ "displayName": "Week 43 - 2014-10-22 - 2014-10-28",
"endDate": "2014-10-28",
"id": "2014WedW43",
"iso": "2014WedW43",
"name": "Week 43 - 2014-10-22 - 2014-10-28",
+ "periodType": "WEEKLY",
"startDate": "2014-10-22",
},
Object {
+ "displayName": "Week 44 - 2014-10-29 - 2014-11-04",
"endDate": "2014-11-04",
"id": "2014WedW44",
"iso": "2014WedW44",
"name": "Week 44 - 2014-10-29 - 2014-11-04",
+ "periodType": "WEEKLY",
"startDate": "2014-10-29",
},
Object {
+ "displayName": "Week 45 - 2014-11-05 - 2014-11-11",
"endDate": "2014-11-11",
"id": "2014WedW45",
"iso": "2014WedW45",
"name": "Week 45 - 2014-11-05 - 2014-11-11",
+ "periodType": "WEEKLY",
"startDate": "2014-11-05",
},
Object {
+ "displayName": "Week 46 - 2014-11-12 - 2014-11-18",
"endDate": "2014-11-18",
"id": "2014WedW46",
"iso": "2014WedW46",
"name": "Week 46 - 2014-11-12 - 2014-11-18",
+ "periodType": "WEEKLY",
"startDate": "2014-11-12",
},
Object {
+ "displayName": "Week 47 - 2014-11-19 - 2014-11-25",
"endDate": "2014-11-25",
"id": "2014WedW47",
"iso": "2014WedW47",
"name": "Week 47 - 2014-11-19 - 2014-11-25",
+ "periodType": "WEEKLY",
"startDate": "2014-11-19",
},
Object {
+ "displayName": "Week 48 - 2014-11-26 - 2014-12-02",
"endDate": "2014-12-02",
"id": "2014WedW48",
"iso": "2014WedW48",
"name": "Week 48 - 2014-11-26 - 2014-12-02",
+ "periodType": "WEEKLY",
"startDate": "2014-11-26",
},
Object {
+ "displayName": "Week 49 - 2014-12-03 - 2014-12-09",
"endDate": "2014-12-09",
"id": "2014WedW49",
"iso": "2014WedW49",
"name": "Week 49 - 2014-12-03 - 2014-12-09",
+ "periodType": "WEEKLY",
"startDate": "2014-12-03",
},
Object {
+ "displayName": "Week 50 - 2014-12-10 - 2014-12-16",
"endDate": "2014-12-16",
"id": "2014WedW50",
"iso": "2014WedW50",
"name": "Week 50 - 2014-12-10 - 2014-12-16",
+ "periodType": "WEEKLY",
"startDate": "2014-12-10",
},
Object {
+ "displayName": "Week 51 - 2014-12-17 - 2014-12-23",
"endDate": "2014-12-23",
"id": "2014WedW51",
"iso": "2014WedW51",
"name": "Week 51 - 2014-12-17 - 2014-12-23",
+ "periodType": "WEEKLY",
"startDate": "2014-12-17",
},
Object {
+ "displayName": "Week 52 - 2014-12-24 - 2014-12-30",
"endDate": "2014-12-30",
"id": "2014WedW52",
"iso": "2014WedW52",
"name": "Week 52 - 2014-12-24 - 2014-12-30",
+ "periodType": "WEEKLY",
"startDate": "2014-12-24",
},
],
@@ -5533,73 +7115,93 @@ Object {
"name": "Yearly",
"options": Array [
Object {
+ "displayName": "2005",
"endDate": "2005-12-31",
"id": "2005",
"iso": "2005",
"name": "2005",
+ "periodType": "YEARLY",
"startDate": "2005-01-01",
},
Object {
+ "displayName": "2006",
"endDate": "2006-12-31",
"id": "2006",
"iso": "2006",
"name": "2006",
+ "periodType": "YEARLY",
"startDate": "2006-01-01",
},
Object {
+ "displayName": "2007",
"endDate": "2007-12-31",
"id": "2007",
"iso": "2007",
"name": "2007",
+ "periodType": "YEARLY",
"startDate": "2007-01-01",
},
Object {
+ "displayName": "2008",
"endDate": "2008-12-31",
"id": "2008",
"iso": "2008",
"name": "2008",
+ "periodType": "YEARLY",
"startDate": "2008-01-01",
},
Object {
+ "displayName": "2009",
"endDate": "2009-12-31",
"id": "2009",
"iso": "2009",
"name": "2009",
+ "periodType": "YEARLY",
"startDate": "2009-01-01",
},
Object {
+ "displayName": "2010",
"endDate": "2010-12-31",
"id": "2010",
"iso": "2010",
"name": "2010",
+ "periodType": "YEARLY",
"startDate": "2010-01-01",
},
Object {
+ "displayName": "2011",
"endDate": "2011-12-31",
"id": "2011",
"iso": "2011",
"name": "2011",
+ "periodType": "YEARLY",
"startDate": "2011-01-01",
},
Object {
+ "displayName": "2012",
"endDate": "2012-12-31",
"id": "2012",
"iso": "2012",
"name": "2012",
+ "periodType": "YEARLY",
"startDate": "2012-01-01",
},
Object {
+ "displayName": "2013",
"endDate": "2013-12-31",
"id": "2013",
"iso": "2013",
"name": "2013",
+ "periodType": "YEARLY",
"startDate": "2013-01-01",
},
Object {
+ "displayName": "2014",
"endDate": "2014-12-31",
"id": "2014",
"iso": "2014",
"name": "2014",
+ "periodType": "YEARLY",
"startDate": "2014-01-01",
},
],
diff --git a/src/components/PivotTable/PivotTableValueCell.js b/src/components/PivotTable/PivotTableValueCell.js
index 9013a0a83..f20fe554d 100644
--- a/src/components/PivotTable/PivotTableValueCell.js
+++ b/src/components/PivotTable/PivotTableValueCell.js
@@ -1,8 +1,12 @@
+import i18n from '@dhis2/d2-i18n'
import PropTypes from 'prop-types'
import React, { useRef } from 'react'
import { applyLegendSet } from '../../modules/pivotTable/applyLegendSet.js'
import { CELL_TYPE_VALUE } from '../../modules/pivotTable/pivotTableConstants.js'
-import { VALUE_TYPE_NUMBER } from '../../modules/valueTypes.js'
+import {
+ isNumericValueType,
+ isBooleanValueType,
+} from '../../modules/valueTypes.js'
import { PivotTableCell } from './PivotTableCell.js'
import { PivotTableEmptyCell } from './PivotTableEmptyCell.js'
import { usePivotTableEngine } from './PivotTableEngineContext.js'
@@ -43,10 +47,10 @@ export const PivotTableValueCell = ({
)
}
- // TODO: Add support for 'INTEGER' type (requires server changes)
const legendStyle =
cellContent.cellType === CELL_TYPE_VALUE &&
- cellContent.valueType === VALUE_TYPE_NUMBER
+ (isNumericValueType(cellContent.valueType) ||
+ isBooleanValueType(cellContent.valueType))
? applyLegendSet(
cellContent.rawValue,
cellContent.dxDimension,
@@ -71,7 +75,13 @@ export const PivotTableValueCell = ({
(
- onInsertMarkdown(EMOJI_SMILEY_FACE)}>
- {emojis[EMOJI_SMILEY_FACE]}
+ {emojis[EMOJI_SMILEY_FACE]}
- onInsertMarkdown(EMOJI_SAD_FACE)}>
- {emojis[EMOJI_SAD_FACE]}
+ {emojis[EMOJI_SAD_FACE]}
- onInsertMarkdown(EMOJI_THUMBS_UP)}>
- {emojis[EMOJI_THUMBS_UP]}
+ {emojis[EMOJI_THUMBS_UP]}
- onInsertMarkdown(EMOJI_THUMBS_DOWN)}>
- {emojis[EMOJI_THUMBS_DOWN]}
+ {emojis[EMOJI_THUMBS_DOWN]}
@@ -190,29 +191,59 @@ Toolbar.propTypes = {
disabled: PropTypes.bool,
}
-export const RichTextEditor = forwardRef(
+export const Editor = forwardRef(
(
- { value, disabled, inputPlaceholder, onChange, errorText, helpText },
+ {
+ value,
+ disabled,
+ inputPlaceholder,
+ onChange,
+ errorText,
+ helpText,
+ initialFocus,
+ resizable,
+ },
externalRef
) => {
const [previewMode, setPreviewMode] = useState(false)
const internalRef = useRef()
const textareaRef = externalRef || internalRef
+ const caretPosRef = useRef(undefined)
- useEffect(() => textareaRef.current?.focus(), [textareaRef])
+ const insertMarkdownCallback = (text, caretPos) => {
+ caretPosRef.current = caretPos
+ onChange(text)
+ textareaRef.current.focus()
+ }
+
+ useEffect(() => {
+ if (initialFocus) {
+ textareaRef.current?.focus()
+ }
+ }, [initialFocus, textareaRef])
+
+ useEffect(() => {
+ if (caretPosRef.current) {
+ textareaRef.current?.setSelectionRange(
+ caretPosRef.current,
+ caretPosRef.current
+ )
+
+ caretPosRef.current = undefined
+ }
+ }, [value, textareaRef])
return (
-
+
{
insertMarkdown(
markdown,
textareaRef.current,
- (text, caretPos) => {
- onChange(text)
- textareaRef.current.focus()
- textareaRef.current.selectionEnd = caretPos
- }
+ insertMarkdownCallback
)
if (markdown === MENTION) {
@@ -231,20 +262,18 @@ export const RichTextEditor = forwardRef(
/>
{previewMode ? (
) : (
-
+
-
+ {errorText && (
+ {errorText}
+ )}
+ {helpText && {helpText}}
+
)}
@@ -265,13 +301,20 @@ export const RichTextEditor = forwardRef(
}
)
-RichTextEditor.displayName = 'RichTextEditor'
+Editor.displayName = 'Editor'
+
+Editor.defaultProps = {
+ initialFocus: true,
+ resizable: true,
+}
-RichTextEditor.propTypes = {
+Editor.propTypes = {
value: PropTypes.string.isRequired,
onChange: PropTypes.func.isRequired,
disabled: PropTypes.bool,
errorText: PropTypes.string,
helpText: PropTypes.string,
+ initialFocus: PropTypes.bool,
inputPlaceholder: PropTypes.string,
+ resizable: PropTypes.bool,
}
diff --git a/src/components/RichText/Editor/__tests__/Editor.spec.js b/src/components/RichText/Editor/__tests__/Editor.spec.js
new file mode 100644
index 000000000..97eaf217b
--- /dev/null
+++ b/src/components/RichText/Editor/__tests__/Editor.spec.js
@@ -0,0 +1,47 @@
+import '@testing-library/jest-dom'
+import { render, screen, fireEvent } from '@testing-library/react'
+import React from 'react'
+import { Editor } from '../Editor.js'
+
+const mockConvertCtrlKey = jest.fn()
+jest.mock('../markdownHandler.js', () => ({
+ convertCtrlKey: () => mockConvertCtrlKey(),
+}))
+
+jest.mock('../../../UserMention/UserMentionWrapper.js', () => ({
+ UserMentionWrapper: jest.fn((props) => <>{props.children}>),
+}))
+
+describe('RichText: Editor component', () => {
+ const componentProps = {
+ value: '',
+ onChange: jest.fn(),
+ }
+
+ beforeEach(() => {
+ mockConvertCtrlKey.mockClear()
+ })
+
+ const renderComponent = (props) => {
+ return render(
)
+ }
+
+ it('renders a result', () => {
+ renderComponent(componentProps)
+
+ expect(
+ screen.getByTestId('@dhis2-analytics-richtexteditor')
+ ).toBeVisible()
+ })
+
+ it('calls convertCtrlKey on keydown', () => {
+ renderComponent(componentProps)
+
+ fireEvent.keyDown(screen.getByRole('textbox'), {
+ key: 'A',
+ code: 'keyA',
+ })
+
+ expect(mockConvertCtrlKey).toHaveBeenCalled()
+ })
+})
diff --git a/src/components/RichText/Editor/__tests__/convertCtrlKey.spec.js b/src/components/RichText/Editor/__tests__/convertCtrlKey.spec.js
new file mode 100644
index 000000000..5ebc93a2b
--- /dev/null
+++ b/src/components/RichText/Editor/__tests__/convertCtrlKey.spec.js
@@ -0,0 +1,230 @@
+import { convertCtrlKey } from '../markdownHandler.js'
+
+describe('convertCtrlKey', () => {
+ it('does not trigger callback if no ctrl key', () => {
+ const cb = jest.fn()
+ const e = { key: 'j', preventDefault: () => {} }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).not.toHaveBeenCalled()
+ })
+
+ describe('when ctrl key + "b" pressed', () => {
+ it('triggers callback with open/close markers and caret pos in between', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ ctrlKey: true,
+ target: {
+ selectionStart: 0,
+ selectionEnd: 0,
+ value: 'rainbow dash',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('** rainbow dash', 1)
+ })
+
+ it('triggers callback with open/close markers and caret pos in between (end of text)', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ ctrlKey: true,
+ target: {
+ selectionStart: 22,
+ selectionEnd: 22,
+ value: 'rainbow dash is purple',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('rainbow dash is purple **', 24)
+ })
+
+ it('triggers callback with open/close markers mid-text with surrounding spaces (1)', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 4, // caret located just before "quick"
+ selectionEnd: 4,
+ value: 'the quick brown fox',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('the ** quick brown fox', 5)
+ })
+
+ it('triggers callback with open/close markers mid-text with surrounding spaces (2)', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 3, // caret located just after "the"
+ selectionEnd: 3,
+ value: 'the quick brown fox',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('the ** quick brown fox', 5)
+ })
+
+ it('triggers callback with correct double markers and padding', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 9, // between the underscores
+ selectionEnd: 9,
+ value: 'rainbow __',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('rainbow _**_', 10)
+ })
+
+ describe('selected text', () => {
+ it('triggers callback with open/close markers around text and caret pos after closing marker', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 5, // "ow da" is selected
+ selectionEnd: 10,
+ value: 'rainbow dash is purple',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith(
+ 'rainb *ow da* sh is purple',
+ 13
+ )
+ })
+
+ it('triggers callback with open/close markers around text when starting at beginning of line', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 0, // "rainbow" is selected
+ selectionEnd: 7,
+ value: 'rainbow dash is purple',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('*rainbow* dash is purple', 9)
+ })
+
+ it('triggers callback with open/close markers around text when ending at end of line', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 16, // "purple" is selected
+ selectionEnd: 22,
+ value: 'rainbow dash is purple',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('rainbow dash is *purple*', 24)
+ })
+
+ it('triggers callback with open/close markers around word', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 8, // "dash" is selected
+ selectionEnd: 12,
+ value: 'rainbow dash is purple',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('rainbow *dash* is purple', 14)
+ })
+
+ it('triggers callback with leading/trailing spaces trimmed from selection', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'b',
+ metaKey: true,
+ target: {
+ selectionStart: 8, // " dash " is selected (note leading and trailing space)
+ selectionEnd: 13,
+ value: 'rainbow dash is purple',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('rainbow *dash* is purple', 14)
+ })
+ })
+ })
+
+ describe('when ctrl key + "i" pressed', () => {
+ it('triggers callback with open/close italics markers and caret pos in between', () => {
+ const cb = jest.fn()
+ const e = {
+ key: 'i',
+ ctrlKey: true,
+ target: {
+ selectionStart: 0,
+ selectionEnd: 0,
+ value: '',
+ },
+ preventDefault: () => {},
+ }
+
+ convertCtrlKey(e, cb)
+
+ expect(cb).toHaveBeenCalled()
+ expect(cb).toHaveBeenCalledWith('__', 1)
+ })
+ })
+})
diff --git a/src/components/Interpretations/common/RichTextEditor/markdownHandler.js b/src/components/RichText/Editor/markdownHandler.js
similarity index 86%
rename from src/components/Interpretations/common/RichTextEditor/markdownHandler.js
rename to src/components/RichText/Editor/markdownHandler.js
index e32a771a5..f9b01895f 100644
--- a/src/components/Interpretations/common/RichTextEditor/markdownHandler.js
+++ b/src/components/RichText/Editor/markdownHandler.js
@@ -89,14 +89,20 @@ export const insertMarkdown = (markdown, target, cb) => {
if (start === end) {
//no text
const valueArr = value.split('')
- let markdown = marker.prefix
+ let markdownString = marker.prefix
if (marker.postfix) {
- markdown += marker.postfix
+ markdownString += marker.postfix
}
- valueArr.splice(start, 0, padMarkers(markdown))
+ valueArr.splice(start, 0, padMarkers(markdownString))
newValue = valueArr.join('')
+
+ // for smileys, put the caret after a space
+ if (Object.keys(emojis).includes(markdown)) {
+ newValue += ' '
+ caretPos = caretPos + newValue.length - 1
+ }
} else {
const text = value.slice(start, end)
const trimmedText = trim(text) // TODO really needed?
@@ -104,15 +110,15 @@ export const insertMarkdown = (markdown, target, cb) => {
// adjust caretPos based on trimmed text selection
caretPos = caretPos - (text.length - trimmedText.length) + 1
- let markdown = `${marker.prefix}${trimmedText}`
+ let markdownString = `${marker.prefix}${trimmedText}`
if (marker.postfix) {
- markdown += marker.postfix
+ markdownString += marker.postfix
}
newValue = [
value.slice(0, start),
- padMarkers(markdown),
+ padMarkers(markdownString),
value.slice(end),
].join('')
}
diff --git a/src/components/Interpretations/common/RichTextEditor/styles/RichTextEditor.style.js b/src/components/RichText/Editor/styles/Editor.style.js
similarity index 81%
rename from src/components/Interpretations/common/RichTextEditor/styles/RichTextEditor.style.js
rename to src/components/RichText/Editor/styles/Editor.style.js
index 53b9a5457..607be4c8d 100644
--- a/src/components/Interpretations/common/RichTextEditor/styles/RichTextEditor.style.js
+++ b/src/components/RichText/Editor/styles/Editor.style.js
@@ -6,18 +6,29 @@ export const mainClasses = css`
display: flex;
flex-direction: column;
width: 100%;
+ height: 100%;
}
.preview {
+ padding: ${spacers.dp8} ${spacers.dp12};
font-size: 14px;
- line-height: 19px;
+ line-height: ${spacers.dp16};
color: ${colors.grey900};
+ overflow-y: auto;
+ scroll-behavior: smooth;
+ }
+
+ .edit {
+ width: 100%;
+ height: 100%;
+ scroll-behavior: smooth;
}
.textarea {
width: 100%;
+ height: 100%;
box-sizing: border-box;
- padding: ${spacers.dp8} ${spacers.dp12};
+ padding: ${spacers.dp8} 15px;
color: ${colors.grey900};
background-color: ${colors.white};
@@ -31,12 +42,20 @@ export const mainClasses = css`
font-size: 14px;
line-height: ${spacers.dp16};
user-select: text;
+ resize: none;
+ }
+
+ .textarea.resizable {
+ resize: vertical;
}
.textarea:focus {
outline: none;
box-shadow: 0 0 0 3px ${theme.focus};
- width: calc(100% - 3px);
+ width: calc(100% - 6px);
+ height: calc(100% - 3px);
+ padding: ${spacers.dp8} ${spacers.dp12};
+ margin-left: 3px;
}
.textarea:disabled {
diff --git a/src/components/RichText/Parser/MdParser.js b/src/components/RichText/Parser/MdParser.js
new file mode 100644
index 000000000..0ec97a5f6
--- /dev/null
+++ b/src/components/RichText/Parser/MdParser.js
@@ -0,0 +1,125 @@
+import MarkdownIt from 'markdown-it'
+
+const emojiDb = {
+ ':-)': '\u{1F642}',
+ ':)': '\u{1F642}',
+ ':-(': '\u{1F641}',
+ ':(': '\u{1F641}',
+ ':+1': '\u{1F44D}',
+ ':-1': '\u{1F44E}',
+}
+
+const codes = {
+ bold: {
+ name: 'bold',
+ char: '*',
+ domEl: 'strong',
+ encodedChar: 0x2a,
+ // see https://regex101.com/r/evswdV/8 for explanation of regexp
+ regexString: '\\B\\*((?!\\s)[^*]+(?:\\b|[^*\\s]))\\*\\B',
+ contentFn: (val) => val,
+ },
+ italic: {
+ name: 'italic',
+ char: '_',
+ domEl: 'em',
+ encodedChar: 0x5f,
+ // see https://regex101.com/r/p6LpjK/6 for explanation of regexp
+ regexString: '\\b_((?!\\s)[^_]+(?:\\B|[^_\\s]))_\\b',
+ contentFn: (val) => val,
+ },
+ emoji: {
+ name: 'emoji',
+ char: ':',
+ domEl: 'span',
+ encodedChar: 0x3a,
+ regexString: '^(:-\\)|:\\)|:\\(|:-\\(|:\\+1|:-1)',
+ contentFn: (val) => emojiDb[val],
+ },
+}
+
+let linksInText
+
+const markerIsInLinkText = (pos) =>
+ linksInText.some((link) => pos >= link.index && pos <= link.lastIndex)
+
+const parse = (code) => (state, silent) => {
+ if (silent) {
+ return false
+ }
+
+ const start = state.pos
+
+ // skip parsing emphasis if marker is within a link
+ if (markerIsInLinkText(start)) {
+ return false
+ }
+
+ const marker = state.src.charCodeAt(start)
+
+ // marker character: "_", "*", ":"
+ if (marker !== codes[code].encodedChar) {
+ return false
+ }
+
+ const MARKER_REGEX = new RegExp(codes[code].regexString)
+ const token = state.src.slice(start)
+
+ if (MARKER_REGEX.test(token)) {
+ const markerMatch = token.match(MARKER_REGEX)
+
+ // skip parsing sections where the marker is not at the start of the token
+ if (markerMatch.index !== 0) {
+ return false
+ }
+
+ const text = markerMatch[1]
+
+ state.push(`${codes[code].domEl}_open`, codes[code].domEl, 1)
+
+ const t = state.push('text', '', 0)
+ t.content = codes[code].contentFn(text)
+
+ state.push(`${codes.bold.domEl}_close`, codes[code].domEl, -1)
+ state.pos += markerMatch[0].length
+
+ return true
+ }
+
+ return false
+}
+
+export class MdParser {
+ constructor() {
+ // disable all rules, enable autolink for URLs and email addresses
+ const md = new MarkdownIt('zero', { linkify: true, breaks: true })
+
+ // *bold* ->
bold
+ md.inline.ruler.push('strong', parse(codes.bold.name))
+
+ // _italic_ ->
italic
+ md.inline.ruler.push('italic', parse(codes.italic.name))
+
+ // :-) :) :-( :( :+1 :-1 ->
[unicode]
+ md.inline.ruler.push('emoji', parse(codes.emoji.name))
+
+ md.enable([
+ 'heading',
+ 'link',
+ 'linkify',
+ 'list',
+ 'newline',
+ 'strong',
+ 'italic',
+ 'emoji',
+ ])
+
+ this.md = md
+ }
+
+ render(text) {
+ linksInText = this.md.linkify.match(text) || []
+
+ return this.md.render(text)
+ }
+}
diff --git a/src/components/RichText/Parser/Parser.js b/src/components/RichText/Parser/Parser.js
new file mode 100644
index 000000000..172049bc1
--- /dev/null
+++ b/src/components/RichText/Parser/Parser.js
@@ -0,0 +1,28 @@
+import PropTypes from 'prop-types'
+import React, { useMemo } from 'react'
+import { MdParser } from './MdParser.js'
+
+export const Parser = ({ children, style }) => {
+ const MdParserInstance = useMemo(() => new MdParser(), [])
+
+ return children ? (
+
+ ) : null
+}
+
+Parser.defaultProps = {
+ style: null,
+}
+
+Parser.propTypes = {
+ children: PropTypes.oneOfType([
+ PropTypes.arrayOf(PropTypes.node),
+ PropTypes.node,
+ ]),
+ style: PropTypes.object,
+}
diff --git a/src/components/RichText/Parser/__tests__/MdParser.spec.js b/src/components/RichText/Parser/__tests__/MdParser.spec.js
new file mode 100644
index 000000000..397b27e76
--- /dev/null
+++ b/src/components/RichText/Parser/__tests__/MdParser.spec.js
@@ -0,0 +1,166 @@
+import { MdParser } from '../MdParser.js'
+
+const Parser = new MdParser()
+
+describe('MdParser class', () => {
+ it('converts text into HTML', () => {
+ const inlineTests = [
+ ['_italic_', '
italic'],
+ ['*bold*', '
bold'],
+ [
+ '_ not italic because there is a space _',
+ '_ not italic because there is a space _',
+ ],
+ [':-)', '
\u{1F642}'],
+ [':)', '
\u{1F642}'],
+ [':-(', '
\u{1F641}'],
+ [':(', '
\u{1F641}'],
+ [':+1', '
\u{1F44D}'],
+ [':-1', '
\u{1F44E}'],
+ [
+ 'mixed _italic_ *bold* and :+1',
+ 'mixed
italic bold and
\u{1F44D}',
+ ],
+ ['_italic with * inside_', '
italic with * inside'],
+ ['*bold with _ inside*', '
bold with _ inside'],
+
+ // italic marker followed by : should work
+ ['_italic_:', '
italic:'],
+ [
+ '_italic_: some text, *bold*: some other text',
+ '
italic: some text,
bold: some other text',
+ ],
+ // bold marker followed by : should work
+ ['*bold*:', '
bold:'],
+ [
+ '*bold*: some text, _italic_: some other text',
+ '
bold: some text,
italic: some other text',
+ ],
+
+ // italic marker inside an italic string not allowed
+ ['_italic with _ inside_', '_italic with _ inside_'],
+ // bold marker inside a bold string not allowed
+ ['*bold with * inside*', '*bold with * inside*'],
+ [
+ '_multiple_ italic in the _same line_',
+ '
multiple italic in the
same line',
+ ],
+ // nested italic/bold combinations not allowed
+ [
+ '_italic with *bold* inside_',
+ '
italic with *bold* inside',
+ ],
+ [
+ '*bold with _italic_ inside*',
+ '
bold with _italic_ inside',
+ ],
+ ['text with : and :)', 'text with : and
\u{1F642}'],
+ [
+ '(parenthesis and :))',
+ '(parenthesis and
\u{1F642})',
+ ],
+ [
+ ':((parenthesis:))',
+ '
\u{1F641}(parenthesis
\u{1F642})',
+ ],
+ [':+1+1', '
\u{1F44D}+1'],
+ ['-1:-1', '-1
\u{1F44E}'],
+
+ // links
+ [
+ 'example.com/path',
+ '
example.com/path',
+ ],
+
+ // not recognized links with italic marker inside not converted
+ [
+ 'example_with_underscore.com/path',
+ 'example_with_underscore.com/path',
+ ],
+ [
+ 'example_with_underscore.com/path_with_underscore',
+ 'example_with_underscore.com/path_with_underscore',
+ ],
+
+ // markers around non-recognized links
+ [
+ 'link example_with_underscore.com/path should _not_ be converted',
+ 'link example_with_underscore.com/path should
not be converted',
+ ],
+ [
+ 'link example_with_underscore.com/path should *not* be converted',
+ 'link example_with_underscore.com/path should
not be converted',
+ ],
+
+ // italic marker inside links not converted
+ [
+ 'example.com/path_with_underscore',
+ '
example.com/path_with_underscore',
+ ],
+ [
+ '_italic_ and *bold* with a example.com/link_with_underscore',
+ '
italic and
bold with a
example.com/link_with_underscore',
+ ],
+ [
+ 'example.com/path with *bold* after :)',
+ '
example.com/path with
bold after
\u{1F642}',
+ ],
+ [
+ '_before_ example.com/path_with_underscore *after* :)',
+ '
before example.com/path_with_underscore after \u{1F642}',
+ ],
+
+ // italic/bold markers right after non-word characters
+ [
+ '_If % of ART retention rate after 12 months >90(%)_: Sustain the efforts.',
+ '
If % of ART retention rate after 12 months >90(%): Sustain the efforts.',
+ ],
+ [
+ '*If % of ART retention rate after 12 months >90(%)*: Sustain the efforts.',
+ '
If % of ART retention rate after 12 months >90(%): Sustain the efforts.',
+ ],
+ ]
+
+ inlineTests.forEach((test) => {
+ const renderedText = Parser.render(test[0])
+
+ expect(renderedText).toEqual(`
${test[1]}
\n`)
+ })
+
+ const blockTests = [
+ // heading
+ ['# Heading 1', '
Heading 1
'],
+ ['## Heading 2', '
Heading 2
'],
+ ['### Heading 3', '
Heading 3
'],
+ ['#### Heading 4', '
Heading 4
'],
+ ['##### Heading 5', '
Heading 5
'],
+ ['###### Heading 6', '
Heading 6
'],
+ ['# *Bold head*', '
Bold head
'],
+ ['## _Italic title_', '
Italic title
'],
+ [
+ '### *Bold* and _italic_ title',
+ '
Bold and italic title
',
+ ],
+
+ // lists
+ [
+ '* first\n* second\n* third',
+ '
',
+ ],
+ [
+ '1. one\n1. two\n1. three\n',
+ '
\n- one
\n- two
\n- three
\n
',
+ ],
+ [
+ '* *first*\n* second\n* _third_',
+ '
',
+ ],
+ ]
+
+ blockTests.forEach((test) => {
+ const renderedText = Parser.render(test[0])
+
+ expect(renderedText).toEqual(`${test[1]}\n`)
+ })
+ })
+})
diff --git a/src/components/RichText/Parser/__tests__/Parser.spec.js b/src/components/RichText/Parser/__tests__/Parser.spec.js
new file mode 100644
index 000000000..26011d492
--- /dev/null
+++ b/src/components/RichText/Parser/__tests__/Parser.spec.js
@@ -0,0 +1,43 @@
+import { shallow } from 'enzyme'
+import React from 'react'
+import { Parser } from '../Parser.js'
+
+jest.mock('../MdParser.js', () => ({
+ MdParser: jest.fn().mockImplementation(() => {
+ return { render: () => 'converted text' }
+ }),
+}))
+
+describe('RichText: Parser component', () => {
+ let richTextParser
+ const defaultProps = {
+ style: { color: 'blue', whiteSpace: 'pre-line' },
+ }
+
+ const renderComponent = (props, text) => {
+ return shallow(
{text})
+ }
+
+ it('should have rendered a result', () => {
+ richTextParser = renderComponent({}, 'test')
+
+ expect(richTextParser).toHaveLength(1)
+ })
+
+ it('should have rendered a result with the style prop', () => {
+ richTextParser = renderComponent(defaultProps, 'test prop')
+
+ expect(richTextParser.props().style).toEqual(defaultProps.style)
+ })
+
+ it('should have rendered content', () => {
+ richTextParser = renderComponent({}, 'plain text')
+
+ expect(richTextParser.html()).toEqual('
converted text
')
+ })
+
+ it('should return null if no children is passed', () => {
+ richTextParser = renderComponent({}, undefined)
+ expect(richTextParser.html()).toBe(null)
+ })
+})
diff --git a/src/components/RichText/index.js b/src/components/RichText/index.js
new file mode 100644
index 000000000..6d9ff0e75
--- /dev/null
+++ b/src/components/RichText/index.js
@@ -0,0 +1,3 @@
+export { Editor as RichTextEditor } from './Editor/Editor.js'
+export { Parser as RichTextParser } from './Parser/Parser.js'
+export { MdParser as RichTextMdParser } from './Parser/MdParser.js'
diff --git a/src/components/Interpretations/common/UserMention/UserList.js b/src/components/UserMention/UserList.js
similarity index 100%
rename from src/components/Interpretations/common/UserMention/UserList.js
rename to src/components/UserMention/UserList.js
diff --git a/src/components/Interpretations/common/UserMention/UserMentionWrapper.js b/src/components/UserMention/UserMentionWrapper.js
similarity index 88%
rename from src/components/Interpretations/common/UserMention/UserMentionWrapper.js
rename to src/components/UserMention/UserMentionWrapper.js
index 394cc39bc..4550d5513 100644
--- a/src/components/Interpretations/common/UserMention/UserMentionWrapper.js
+++ b/src/components/UserMention/UserMentionWrapper.js
@@ -2,12 +2,12 @@ import i18n from '@dhis2/d2-i18n'
import {
CenteredContent,
CircularLoader,
+ Layer,
Menu,
MenuSectionHeader,
MenuItem,
Popper,
Card,
- Portal,
} from '@dhis2/ui'
import PropTypes from 'prop-types'
import React, { useState, useRef } from 'react'
@@ -43,6 +43,7 @@ export const UserMentionWrapper = ({
inputReference,
onUserSelect,
}) => {
+ const [listIsOpen, setListIsOpen] = useState(false)
const [captureText, setCaptureText] = useState(false)
const [capturedText, setCapturedText] = useState('')
const [cloneText, setCloneText] = useState('')
@@ -54,6 +55,7 @@ export const UserMentionWrapper = ({
})
const reset = () => {
+ setListIsOpen(false)
setCaptureText(false)
setCapturedText('')
setCloneText('')
@@ -63,6 +65,12 @@ export const UserMentionWrapper = ({
clear()
}
+ // focus the input/textarea when the user list is closed by clicking above the input/textarea
+ const onClick = () => inputReference.current.focus()
+
+ // close the user list when clicking in the input/textarea or outside of it (input/textarea blur)
+ const onUserListClose = () => reset()
+
// event bubbles up from the input/textarea
const onInput = ({ target }) => {
const { selectionEnd, value } = target
@@ -72,10 +80,12 @@ export const UserMentionWrapper = ({
const spacePosition = value.indexOf(' ', captureStartPosition - 1)
- const filterValue = value.substring(
- captureStartPosition,
- spacePosition > 0 ? spacePosition : selectionEnd + 1
- )
+ const filterValue = value
+ .substring(
+ captureStartPosition,
+ spacePosition > 0 ? spacePosition : selectionEnd + 1
+ )
+ .replace(/\n+/, '')
if (filterValue !== capturedText) {
setCapturedText(filterValue)
@@ -91,6 +101,7 @@ export const UserMentionWrapper = ({
const { selectionStart } = target
if (!captureText && key === '@') {
+ setListIsOpen(true)
setCaptureText(true)
setCaptureStartPosition(selectionStart + 1)
setCloneText(target.value.substring(0, selectionStart) + '@')
@@ -159,16 +170,21 @@ export const UserMentionWrapper = ({
)
}
- const onClick = (user) => () => onSelect(user)
+ const onUserClick = (user) => () => onSelect(user)
return (
-
+
{children}
-
{cloneText}
+
{cloneText}
- {captureText && (
-
+ {listIsOpen && (
+
)}
@@ -228,7 +244,7 @@ export const UserMentionWrapper = ({
-
+
)}
{resolvedHeaderStyle.styles}
@@ -245,5 +261,3 @@ UserMentionWrapper.propTypes = {
onUserSelect: PropTypes.func.isRequired,
children: PropTypes.node,
}
-
-export default UserMentionWrapper
diff --git a/src/components/Interpretations/common/UserMention/styles/UserMentionWrapper.style.js b/src/components/UserMention/styles/UserMentionWrapper.style.js
similarity index 88%
rename from src/components/Interpretations/common/UserMention/styles/UserMentionWrapper.style.js
rename to src/components/UserMention/styles/UserMentionWrapper.style.js
index e98517db1..a075ff11d 100644
--- a/src/components/Interpretations/common/UserMention/styles/UserMentionWrapper.style.js
+++ b/src/components/UserMention/styles/UserMentionWrapper.style.js
@@ -8,6 +8,8 @@ import css from 'styled-jsx/css'
*/
export const userMentionWrapperClasses = css`
.wrapper {
+ width: 100%;
+ height: 100%;
position: relative;
}
.clone {
@@ -15,19 +17,20 @@ export const userMentionWrapperClasses = css`
visibility: hidden;
inset: 0;
box-sizing: border-box;
- padding: ${spacers.dp8} ${spacers.dp12};
+ padding: ${spacers.dp8} 15px;
border: 1px solid ${colors.grey500};
font-size: 14px;
line-height: ${spacers.dp16};
z-index: 1;
pointer-events: none;
}
- .clone > pre {
+ .clone > p {
display: inline;
word-wrap: break-word;
overflow-wrap: break-word;
font: inherit;
margin: 0;
+ white-space: break-spaces;
}
.container {
background-color: ${colors.white};
diff --git a/src/components/Interpretations/common/UserMention/useUserSearchResults.js b/src/components/UserMention/useUserSearchResults.js
similarity index 94%
rename from src/components/Interpretations/common/UserMention/useUserSearchResults.js
rename to src/components/UserMention/useUserSearchResults.js
index b9d46b46d..9adfc4613 100644
--- a/src/components/Interpretations/common/UserMention/useUserSearchResults.js
+++ b/src/components/UserMention/useUserSearchResults.js
@@ -32,10 +32,10 @@ export const useUserSearchResults = ({ searchText }) => {
}, [searchText, debouncedRefetch])
useEffect(() => {
- if (data) {
+ if (fetching === false && data) {
setData(data.users)
}
- }, [data])
+ }, [data, fetching])
return {
users,
diff --git a/src/index.js b/src/index.js
index 0fb8fe59b..1202981b5 100644
--- a/src/index.js
+++ b/src/index.js
@@ -47,6 +47,8 @@ export {
useCachedDataQuery,
} from './components/CachedDataQueryProvider.js'
+export * from './components/RichText/index.js'
+
// Api
export { default as Analytics } from './api/analytics/Analytics.js'
diff --git a/src/modules/__tests__/renderValue.spec.js b/src/modules/__tests__/renderValue.spec.js
index 845629bbe..eaf3d2a7c 100644
--- a/src/modules/__tests__/renderValue.spec.js
+++ b/src/modules/__tests__/renderValue.spec.js
@@ -25,21 +25,21 @@ const tests = [
// Numbers
{
value: 1000.5,
- expected: '1 000.5',
+ expected: '1 000.50',
valueType: VALUE_TYPE_NUMBER,
round: true,
dgs: DGS_SPACE,
},
{
- value: 33777889.55,
- expected: '33,777,889.5',
+ value: 33777889.555,
+ expected: '33,777,889.55',
valueType: VALUE_TYPE_NUMBER,
round: true,
dgs: DGS_COMMA,
},
{
value: 33777889.556,
- expected: '33 777 889.6',
+ expected: '33 777 889.56',
valueType: VALUE_TYPE_NUMBER,
round: true,
dgs: DGS_SPACE,
@@ -53,7 +53,7 @@ const tests = [
},
{
value: 33777889.56,
- expected: '33777889.6',
+ expected: '33777889.56',
valueType: VALUE_TYPE_NUMBER,
round: true,
dgs: DGS_NONE,
@@ -74,7 +74,7 @@ const tests = [
},
{
value: 1.101,
- expected: '1.1',
+ expected: '1.10',
valueType: VALUE_TYPE_NUMBER,
round: true,
dgs: DGS_SPACE,
@@ -135,16 +135,16 @@ const tests = [
dgs: DGS_SPACE,
},
{
- value: -0.0234,
- expected: '-2.3%',
+ value: -0.02345,
+ expected: '-2.34%',
valueType: VALUE_TYPE_NUMBER,
numberType: NUMBER_TYPE_ROW_PERCENTAGE,
round: true,
dgs: DGS_SPACE,
},
{
- value: -0.0234,
- expected: '-2.34%',
+ value: -0.02345,
+ expected: '-2.345%',
valueType: VALUE_TYPE_NUMBER,
numberType: NUMBER_TYPE_ROW_PERCENTAGE,
round: false,
diff --git a/src/modules/pivotTable/PivotTableEngine.js b/src/modules/pivotTable/PivotTableEngine.js
index b5788fee1..ad798e686 100644
--- a/src/modules/pivotTable/PivotTableEngine.js
+++ b/src/modules/pivotTable/PivotTableEngine.js
@@ -8,7 +8,13 @@ import {
} from '../dataTypes.js'
import { DIMENSION_ID_ORGUNIT } from '../predefinedDimensions.js'
import { renderValue } from '../renderValue.js'
-import { VALUE_TYPE_NUMBER, VALUE_TYPE_TEXT } from '../valueTypes.js'
+import {
+ VALUE_TYPE_NUMBER,
+ VALUE_TYPE_TEXT,
+ isBooleanValueType,
+ isCumulativeValueType,
+ isNumericValueType,
+} from '../valueTypes.js'
import { AdaptiveClippingController } from './AdaptiveClippingController.js'
import { addToTotalIfNumber } from './addToTotalIfNumber.js'
import { parseValue } from './parseValue.js'
@@ -36,6 +42,8 @@ import {
NUMBER_TYPE_COLUMN_PERCENTAGE,
NUMBER_TYPE_ROW_PERCENTAGE,
NUMBER_TYPE_VALUE,
+ VALUE_TYPE_NA,
+ VALUE_NA,
} from './pivotTableConstants.js'
const dataFields = [
@@ -240,7 +248,7 @@ const applyTotalAggregationType = (
) => {
switch (overrideTotalAggregationType || totalAggregationType) {
case AGGREGATE_TYPE_NA:
- return 'N/A'
+ return VALUE_NA
case AGGREGATE_TYPE_AVERAGE:
return (
((numerator || value) * multiplier) /
@@ -396,19 +404,46 @@ export class PivotTableEngine {
rawCell.renderedValue = renderedValue
}
+ if (
+ [CELL_TYPE_TOTAL, CELL_TYPE_SUBTOTAL].includes(rawCell.cellType) &&
+ rawCell.rawValue === AGGREGATE_TYPE_NA
+ ) {
+ rawCell.titleValue = i18n.t('Not applicable')
+ }
+
if (this.options.cumulativeValues) {
+ let titleValue
+
+ if (this.data[row] && this.data[row][column]) {
+ const dataRow = this.data[row][column]
+
+ const rawValue =
+ cellType === CELL_TYPE_VALUE
+ ? dataRow[this.dimensionLookup.dataHeaders.value]
+ : dataRow.value
+
+ titleValue = i18n.t('Value: {{value}}', {
+ value: renderValue(rawValue, valueType, this.visualization),
+ nsSeparator: '^^',
+ })
+ }
+
const cumulativeValue = this.getCumulative({
row,
column,
})
if (cumulativeValue !== undefined && cumulativeValue !== null) {
- // force to NUMBER for accumulated values
+ // force to TEXT for N/A (accumulated) values
+ // force to NUMBER for accumulated values if no valueType present
rawCell.valueType =
- valueType === undefined || valueType === null
+ cumulativeValue === VALUE_NA
+ ? VALUE_TYPE_NA
+ : valueType === undefined || valueType === null
? VALUE_TYPE_NUMBER
: valueType
rawCell.empty = false
+ rawCell.titleValue = titleValue
rawCell.rawValue = cumulativeValue
rawCell.renderedValue = renderValue(
cumulativeValue,
@@ -515,8 +550,14 @@ export class PivotTableEngine {
if (!this.data[row]) {
return undefined
}
+
const cellValue = this.data[row][column]
+ // empty cell
+ if (!cellValue) {
+ return undefined
+ }
+
if (cellValue && !Array.isArray(cellValue)) {
// This is a total cell
return {
@@ -532,6 +573,7 @@ export class PivotTableEngine {
const dxRowIndex = this.dimensionLookup.rows.findIndex(
(dim) => dim.isDxDimension
)
+
if (rowHeaders.length && dxRowIndex !== -1) {
return {
valueType: rowHeaders[dxRowIndex].valueType,
@@ -553,11 +595,6 @@ export class PivotTableEngine {
}
}
- // Empty cell
- // The cell still needs to get the valueType to render correctly 0 and cumulative values
- //
- // OR
- //
// Data is in Filter
// TODO : This assumes the server ignores text types, we should confirm this is the case
return {
@@ -730,15 +767,30 @@ export class PivotTableEngine {
totalCell.totalAggregationType = currentAggType
}
- const currentValueType = dxDimension?.valueType
+ // Force value type of total cells to NUMBER for value cells with numeric or boolean types.
+ // This is to simplify the code below where we compare the previous value type.
+ // All numeric/boolean value types use the same style for rendering the total cell (right aligned content)
+ // and using NUMBER for the total cell is enough for that.
+ // (see DHIS2-9155)
+ const currentValueType =
+ isNumericValueType(dxDimension?.valueType) ||
+ isBooleanValueType(dxDimension?.valueType)
+ ? VALUE_TYPE_NUMBER
+ : dxDimension?.valueType
+
const previousValueType = totalCell.valueType
if (previousValueType && currentValueType !== previousValueType) {
- totalCell.valueType = AGGREGATE_TYPE_NA
+ totalCell.valueType = VALUE_TYPE_NA
} else {
totalCell.valueType = currentValueType
}
- if (dxDimension?.valueType === VALUE_TYPE_NUMBER) {
+ // Compute totals for all numeric and boolean value types only.
+ // In practice valueType here is NUMBER (see the comment above).
+ // When is not, it means there is some value cell with a valueType other than numeric/boolean,
+ // the total should not be computed then.
+ // (see DHIS2-9155)
+ if (isNumericValueType(totalCell.valueType)) {
dataFields.forEach((field) => {
const headerIndex = this.dimensionLookup.dataHeaders[field]
const value = parseValue(dataRow[headerIndex])
@@ -863,6 +915,28 @@ export class PivotTableEngine {
}
}
}
+
+ computeOverrideTotalAggregationType(totalCell, visualization) {
+ // Avoid undefined on total cells with valueTypes that cannot be totalized.
+ // This happens for example when a column/row has all value cells of type TEXT.
+ if (
+ !(
+ isNumericValueType(totalCell.valueType) ||
+ isBooleanValueType(totalCell.valueType)
+ )
+ ) {
+ return AGGREGATE_TYPE_NA
+ }
+
+ // DHIS2-15698: do not override total aggregation type when numberType option is not present
+ // (numberType option default is VALUE)
+ return (
+ visualization.numberType &&
+ visualization.numberType !== NUMBER_TYPE_VALUE &&
+ AGGREGATE_TYPE_SUM
+ )
+ }
+
finalizeTotal({ row, column }) {
if (!this.data[row]) {
return
@@ -871,12 +945,17 @@ export class PivotTableEngine {
if (totalCell && totalCell.count) {
totalCell.value = applyTotalAggregationType(
totalCell,
- // DHIS2-15698: do not override total aggregation type when numberType option is not present
- // (numberType option default is VALUE)
- this.visualization.numberType &&
- this.visualization.numberType !== NUMBER_TYPE_VALUE &&
- AGGREGATE_TYPE_SUM
+ this.computeOverrideTotalAggregationType(
+ totalCell,
+ this.visualization
+ )
)
+
+ // override valueType for styling cells with N/A value
+ if (totalCell.value === AGGREGATE_TYPE_NA) {
+ totalCell.valueType = VALUE_TYPE_NA
+ }
+
this.adaptiveClippingController.add(
{ row, column },
renderValue(
@@ -1009,10 +1088,19 @@ export class PivotTableEngine {
column,
})
const valueType = dxDimension?.valueType || VALUE_TYPE_TEXT
+ const totalAggregationType =
+ dxDimension?.totalAggregationType
+
+ // only accumulate numeric (except for PERCENTAGE and UNIT_INTERVAL) and boolean values
+ // accumulating other value types like text values does not make sense
+ if (
+ isCumulativeValueType(valueType) &&
+ totalAggregationType === AGGREGATE_TYPE_SUM
+ ) {
+ // initialise to 0 for cumulative types
+ // (||= is not transformed correctly in Babel with the current setup)
+ acc || (acc = 0)
- // only accumulate numeric values
- // accumulating text values does not make sense
- if (valueType === VALUE_TYPE_NUMBER) {
if (this.data[row] && this.data[row][column]) {
const dataRow = this.data[row][column]
@@ -1030,7 +1118,7 @@ export class PivotTableEngine {
}
return acc
- }, 0)
+ }, '')
})
} else {
this.accumulators = { rows: {} }
diff --git a/src/modules/pivotTable/pivotTableConstants.js b/src/modules/pivotTable/pivotTableConstants.js
index 1221972c9..1ab1b290d 100644
--- a/src/modules/pivotTable/pivotTableConstants.js
+++ b/src/modules/pivotTable/pivotTableConstants.js
@@ -9,6 +9,8 @@ export const AGGREGATE_TYPE_SUM = 'SUM'
export const AGGREGATE_TYPE_AVERAGE = 'AVERAGE'
export const AGGREGATE_TYPE_NA = 'N/A'
+export const VALUE_TYPE_NA = 'N_A' // this ends up as CSS class and / is problematic
+
export const NUMBER_TYPE_VALUE = 'VALUE'
export const NUMBER_TYPE_ROW_PERCENTAGE = 'ROW_PERCENTAGE'
export const NUMBER_TYPE_COLUMN_PERCENTAGE = 'COLUMN_PERCENTAGE'
@@ -35,3 +37,5 @@ export const WRAPPED_TEXT_JUSTIFY_BUFFER = 25
export const WRAPPED_TEXT_LINE_HEIGHT = 1.0
export const CLIPPED_AXIS_PARTITION_SIZE_PX = 1000
+
+export const VALUE_NA = 'N/A'
diff --git a/src/modules/renderValue.js b/src/modules/renderValue.js
index 17e288043..9c2f1c763 100644
--- a/src/modules/renderValue.js
+++ b/src/modules/renderValue.js
@@ -2,13 +2,11 @@ import {
NUMBER_TYPE_ROW_PERCENTAGE,
NUMBER_TYPE_COLUMN_PERCENTAGE,
} from './pivotTable/pivotTableConstants.js'
-import { isNumericValueType } from './valueTypes.js'
+import { isNumericValueType, isBooleanValueType } from './valueTypes.js'
const trimTrailingZeros = (stringValue) => stringValue.replace(/\.?0+$/, '')
-const decimalSeparator = '.'
-
-const separateDigitGroups = (stringValue, decimalSeparator) => {
+export const separateDigitGroups = (stringValue, decimalSeparator = '.') => {
const isNegative = stringValue[0] === '-'
const [integer, remainder] = stringValue.replace(/^-/, '').split('.')
@@ -49,13 +47,16 @@ const toFixedPrecisionString = (value, skipRounding) => {
return value
}
- const precision = skipRounding ? 10 : value > -1 && value < 1 ? 2 : 1
+ const precision = skipRounding ? 10 : 2
return value.toFixed(precision)
}
export const renderValue = (value, valueType, visualization) => {
- if (!isNumericValueType(valueType) || value === undefined) {
+ if (
+ !(isNumericValueType(valueType) || isBooleanValueType(valueType)) ||
+ value === undefined
+ ) {
return String(value).replace(/[^\S\n]+/, ' ')
}
@@ -68,9 +69,8 @@ export const renderValue = (value, valueType, visualization) => {
)
return (
- separateDigitGroups(stringValue, decimalSeparator).join(
- getSeparator(visualization)
- ) + '%'
+ separateDigitGroups(stringValue).join(getSeparator(visualization)) +
+ '%'
)
} else {
const stringValue = toFixedPrecisionString(
@@ -78,7 +78,7 @@ export const renderValue = (value, valueType, visualization) => {
visualization.skipRounding
)
- return separateDigitGroups(stringValue, decimalSeparator).join(
+ return separateDigitGroups(stringValue).join(
getSeparator(visualization)
)
}
diff --git a/src/modules/valueTypes.js b/src/modules/valueTypes.js
index cf5dc01f0..1097ac84f 100644
--- a/src/modules/valueTypes.js
+++ b/src/modules/valueTypes.js
@@ -34,4 +34,18 @@ const NUMERIC_VALUE_TYPES = [
VALUE_TYPE_INTEGER_ZERO_OR_POSITIVE,
]
+const BOOLEAN_VALUE_TYPES = [VALUE_TYPE_BOOLEAN, VALUE_TYPE_TRUE_ONLY]
+
+const CUMULATIVE_VALUE_TYPES = [
+ VALUE_TYPE_NUMBER,
+ VALUE_TYPE_INTEGER,
+ VALUE_TYPE_INTEGER_POSITIVE,
+ VALUE_TYPE_INTEGER_NEGATIVE,
+ VALUE_TYPE_INTEGER_ZERO_OR_POSITIVE,
+ ...BOOLEAN_VALUE_TYPES,
+]
+
+export const isCumulativeValueType = (type) =>
+ CUMULATIVE_VALUE_TYPES.includes(type)
export const isNumericValueType = (type) => NUMERIC_VALUE_TYPES.includes(type)
+export const isBooleanValueType = (type) => BOOLEAN_VALUE_TYPES.includes(type)
diff --git a/src/modules/visTypes.js b/src/modules/visTypes.js
index 2040dad5e..bd7a9b37a 100644
--- a/src/modules/visTypes.js
+++ b/src/modules/visTypes.js
@@ -141,7 +141,7 @@ const legendSetTypes = [
VIS_TYPE_STACKED_BAR,
]
-export const defaultVisType = VIS_TYPE_COLUMN
+export const defaultVisType = VIS_TYPE_PIVOT_TABLE
export const isStacked = (type) => stackedTypes.includes(type)
export const isYearOverYear = (type) => yearOverYearTypes.includes(type)
export const isDualAxisType = (type) => dualAxisTypes.includes(type)
diff --git a/src/visualizations/config/adapters/dhis_dhis/index.js b/src/visualizations/config/adapters/dhis_dhis/index.js
deleted file mode 100644
index 06a5256bf..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/index.js
+++ /dev/null
@@ -1,38 +0,0 @@
-import getSubtitle from './subtitle/index.js'
-import getTitle from './title/index.js'
-import getValue from './value/index.js'
-
-export const INDICATOR_FACTOR_100 = 100
-
-export default function ({ store, layout, extraOptions }) {
- const data = store.generateData({
- type: layout.type,
- seriesId:
- layout.columns && layout.columns.length
- ? layout.columns[0].dimension
- : null,
- categoryId:
- layout.rows && layout.rows.length ? layout.rows[0].dimension : null,
- })
- const metaData = store.data[0].metaData
-
- const config = {
- value: data[0],
- formattedValue:
- data[0] === undefined
- ? extraOptions.noData.text
- : getValue(data[0], layout, metaData),
- title: getTitle(layout, metaData, extraOptions.dashboard),
- subtitle: getSubtitle(layout, metaData, extraOptions.dashboard),
- }
-
- const indicatorType =
- metaData.items[metaData.dimensions.dx[0]].indicatorType
-
- // Use % symbol for factor 100 and the full string for others
- if (indicatorType?.factor !== INDICATOR_FACTOR_100) {
- config.subText = indicatorType?.displayName
- }
-
- return config
-}
diff --git a/src/visualizations/config/adapters/dhis_dhis/subtitle/__tests__/index.spec.js b/src/visualizations/config/adapters/dhis_dhis/subtitle/__tests__/index.spec.js
deleted file mode 100644
index 486333c8c..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/subtitle/__tests__/index.spec.js
+++ /dev/null
@@ -1,53 +0,0 @@
-import { VIS_TYPE_SINGLE_VALUE } from '../../../../../../modules/visTypes.js'
-import getSubtitle from '../index.js'
-
-jest.mock('../singleValue', () => () => 'The sv filter title')
-jest.mock(
- '../../../../../util/getFilterText',
- () => () => 'The default filter text'
-)
-
-describe('getSubtitle', () => {
- it('returns empty subtitle when flag hideSubtitle exists', () => {
- expect(getSubtitle({ hideSubtitle: true })).toEqual('')
- })
-
- it('returns the subtitle provided in the layout', () => {
- const subtitle = 'The subtitle was already set'
- expect(getSubtitle({ subtitle })).toEqual(subtitle)
- })
-
- it('returns subtitle for single value vis', () => {
- expect(getSubtitle({ type: VIS_TYPE_SINGLE_VALUE })).toEqual(
- 'The sv filter title'
- )
- })
-
- describe('not dashboard', () => {
- describe('layout does not include title', () => {
- it('returns empty subtitle', () => {
- expect(getSubtitle({ filters: {} }, {}, false)).toEqual('')
- })
- })
-
- describe('layout includes title', () => {
- it('returns filter title as subtitle', () => {
- expect(
- getSubtitle(
- { filters: {}, title: 'Chart title' },
- {},
- false
- )
- ).toEqual('The default filter text')
- })
- })
- })
-
- describe('dashboard', () => {
- it('returns filter title as subtitle', () => {
- expect(getSubtitle({ filters: {} }, {}, true)).toEqual(
- 'The default filter text'
- )
- })
- })
-})
diff --git a/src/visualizations/config/adapters/dhis_dhis/subtitle/__tests__/singleValue.spec.js b/src/visualizations/config/adapters/dhis_dhis/subtitle/__tests__/singleValue.spec.js
deleted file mode 100644
index 39b497f64..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/subtitle/__tests__/singleValue.spec.js
+++ /dev/null
@@ -1,15 +0,0 @@
-import getSingleValueSubtitle from '../singleValue.js'
-
-jest.mock('../../../../../util/getFilterText', () => () => 'The filter text')
-
-describe('getSingleValueSubtitle', () => {
- it('returns null when layout does not have filters', () => {
- expect(getSingleValueSubtitle({})).toEqual('')
- })
-
- it('returns the filter text', () => {
- expect(getSingleValueSubtitle({ filters: [] })).toEqual(
- 'The filter text'
- )
- })
-})
diff --git a/src/visualizations/config/adapters/dhis_dhis/subtitle/index.js b/src/visualizations/config/adapters/dhis_dhis/subtitle/index.js
deleted file mode 100644
index 1be507be4..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/subtitle/index.js
+++ /dev/null
@@ -1,33 +0,0 @@
-import { VIS_TYPE_SINGLE_VALUE } from '../../../../../modules/visTypes.js'
-import getFilterText from '../../../../util/getFilterText.js'
-import getSingleValueTitle from './singleValue.js'
-
-function getDefault(layout, dashboard, metaData) {
- if (dashboard || typeof layout.title === 'string') {
- return getFilterText(layout.filters, metaData)
- }
-
- return ''
-}
-
-export default function (layout, metaData, dashboard) {
- if (layout.hideSubtitle) {
- return ''
- }
-
- if (typeof layout.subtitle === 'string' && layout.subtitle.length) {
- return layout.subtitle
- } else {
- let subtitle
- switch (layout.type) {
- case VIS_TYPE_SINGLE_VALUE:
- subtitle = getSingleValueTitle(layout, metaData)
-
- break
- default:
- subtitle = getDefault(layout, dashboard, metaData)
- }
-
- return subtitle
- }
-}
diff --git a/src/visualizations/config/adapters/dhis_dhis/subtitle/singleValue.js b/src/visualizations/config/adapters/dhis_dhis/subtitle/singleValue.js
deleted file mode 100644
index de246ba2f..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/subtitle/singleValue.js
+++ /dev/null
@@ -1,5 +0,0 @@
-import getFilterText from '../../../../util/getFilterText.js'
-
-export default function (layout, metaData) {
- return layout.filters ? getFilterText(layout.filters, metaData) : ''
-}
diff --git a/src/visualizations/config/adapters/dhis_dhis/title/__tests__/index.spec.js b/src/visualizations/config/adapters/dhis_dhis/title/__tests__/index.spec.js
deleted file mode 100644
index 15a4b8a56..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/title/__tests__/index.spec.js
+++ /dev/null
@@ -1,36 +0,0 @@
-import { VIS_TYPE_SINGLE_VALUE } from '../../../../../../modules/visTypes.js'
-import getTitle from '../index.js'
-
-jest.mock('../singleValue', () => () => 'The sv filter title')
-jest.mock('../../../../../util/getFilterText', () => () => 'The filter text')
-
-describe('getTitle', () => {
- it('returns empty title when flag hideTitle exists', () => {
- expect(getTitle({ hideTitle: true })).toEqual('')
- })
-
- it('returns the title provided in the layout', () => {
- const title = 'The title was already set'
- expect(getTitle({ title })).toEqual(title)
- })
-
- it('returns title for single value vis', () => {
- expect(getTitle({ type: VIS_TYPE_SINGLE_VALUE })).toEqual(
- 'The sv filter title'
- )
- })
-
- describe('not dashboard', () => {
- it('returns filter text as title', () => {
- expect(getTitle({ filters: {} }, {}, false)).toEqual(
- 'The filter text'
- )
- })
- })
-
- describe('dashboard', () => {
- it('returns empty string', () => {
- expect(getTitle({ filters: {} }, {}, true)).toEqual('')
- })
- })
-})
diff --git a/src/visualizations/config/adapters/dhis_dhis/title/__tests__/singleValue.spec.js b/src/visualizations/config/adapters/dhis_dhis/title/__tests__/singleValue.spec.js
deleted file mode 100644
index 304be7bdb..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/title/__tests__/singleValue.spec.js
+++ /dev/null
@@ -1,21 +0,0 @@
-import getSingleValueTitle from '../singleValue.js'
-
-jest.mock('../../../../../util/getFilterText', () => () => 'The filter text')
-
-describe('getSingleValueTitle', () => {
- it('returns null when layout does not have columns', () => {
- expect(getSingleValueTitle({})).toEqual('')
- })
-
- it('returns the filter text based on column items', () => {
- expect(
- getSingleValueTitle({
- columns: [
- {
- items: [{}],
- },
- ],
- })
- ).toEqual('The filter text')
- })
-})
diff --git a/src/visualizations/config/adapters/dhis_dhis/title/index.js b/src/visualizations/config/adapters/dhis_dhis/title/index.js
deleted file mode 100644
index fb4c6b040..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/title/index.js
+++ /dev/null
@@ -1,30 +0,0 @@
-import { VIS_TYPE_SINGLE_VALUE } from '../../../../../modules/visTypes.js'
-import getFilterText from '../../../../util/getFilterText.js'
-import getSingleValueTitle from './singleValue.js'
-
-function getDefault(layout, metaData, dashboard) {
- return layout.filters && !dashboard
- ? getFilterText(layout.filters, metaData)
- : ''
-}
-
-export default function (layout, metaData, dashboard) {
- if (layout.hideTitle) {
- return ''
- }
-
- if (typeof layout.title === 'string' && layout.title.length) {
- return layout.title
- } else {
- let title
- switch (layout.type) {
- case VIS_TYPE_SINGLE_VALUE:
- title = getSingleValueTitle(layout, metaData)
-
- break
- default:
- title = getDefault(layout, metaData, dashboard)
- }
- return title
- }
-}
diff --git a/src/visualizations/config/adapters/dhis_dhis/type.js b/src/visualizations/config/adapters/dhis_dhis/type.js
deleted file mode 100644
index 412124e58..000000000
--- a/src/visualizations/config/adapters/dhis_dhis/type.js
+++ /dev/null
@@ -1,10 +0,0 @@
-import { VIS_TYPE_SINGLE_VALUE } from '../../../../modules/visTypes.js'
-
-export default function (type) {
- switch (type) {
- case VIS_TYPE_SINGLE_VALUE:
- return { type: VIS_TYPE_SINGLE_VALUE }
- default:
- return { type: VIS_TYPE_SINGLE_VALUE }
- }
-}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/chart/default.js b/src/visualizations/config/adapters/dhis_highcharts/chart/default.js
new file mode 100644
index 000000000..9d4af9829
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/chart/default.js
@@ -0,0 +1,27 @@
+import { getEvents } from '../events/index.js'
+import getType from '../type.js'
+
+const DEFAULT_CHART = {
+ spacingTop: 20,
+ style: {
+ fontFamily: 'Roboto,Helvetica Neue,Helvetica,Arial,sans-serif',
+ },
+}
+
+const DASHBOARD_CHART = {
+ spacingTop: 0,
+ spacingRight: 5,
+ spacingBottom: 2,
+ spacingLeft: 5,
+}
+
+export default function getDefaultChart(layout, el, extraOptions) {
+ return Object.assign(
+ {},
+ getType(layout.type),
+ { renderTo: el || layout.el },
+ DEFAULT_CHART,
+ extraOptions.dashboard ? DASHBOARD_CHART : undefined,
+ getEvents(layout.type)
+ )
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/chart/index.js b/src/visualizations/config/adapters/dhis_highcharts/chart/index.js
new file mode 100644
index 000000000..c6010e016
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/chart/index.js
@@ -0,0 +1,12 @@
+import { VIS_TYPE_SINGLE_VALUE } from '../../../../../modules/visTypes.js'
+import getDefaultChart from './default.js'
+import getSingleValueChart from './singleValue.js'
+
+export default function getChart(layout, el, extraOptions, series) {
+ switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ return getSingleValueChart(layout, el, extraOptions, series)
+ default:
+ return getDefaultChart(layout, el, extraOptions)
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/chart/singleValue.js b/src/visualizations/config/adapters/dhis_highcharts/chart/singleValue.js
new file mode 100644
index 000000000..43a6f66a2
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/chart/singleValue.js
@@ -0,0 +1,19 @@
+import { getSingleValueBackgroundColor } from '../customSVGOptions/singleValue/getSingleValueBackgroundColor.js'
+import getDefaultChart from './default.js'
+
+export default function getSingleValueChart(layout, el, extraOptions, series) {
+ const chart = {
+ ...getDefaultChart(layout, el, extraOptions),
+ backgroundColor: getSingleValueBackgroundColor(
+ layout.legend,
+ extraOptions.legendSets,
+ series[0]
+ ),
+ }
+
+ if (extraOptions.dashboard) {
+ chart.spacingTop = 7
+ }
+
+ return chart
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/index.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/index.js
new file mode 100644
index 000000000..ef5b18509
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/index.js
@@ -0,0 +1,29 @@
+import { VIS_TYPE_SINGLE_VALUE } from '../../../../../modules/visTypes.js'
+import getSingleValueCustomSVGOptions from './singleValue/index.js'
+
+export default function getCustomSVGOptions({
+ extraConfig,
+ layout,
+ extraOptions,
+ metaData,
+ series,
+}) {
+ const baseOptions = {
+ visualizationType: layout.type,
+ }
+ switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ return {
+ ...baseOptions,
+ ...getSingleValueCustomSVGOptions({
+ extraConfig,
+ layout,
+ extraOptions,
+ metaData,
+ series,
+ }),
+ }
+ default:
+ break
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueBackgroundColor.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueBackgroundColor.js
new file mode 100644
index 000000000..650c895a5
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueBackgroundColor.js
@@ -0,0 +1,17 @@
+import { LEGEND_DISPLAY_STYLE_FILL } from '../../../../../../modules/legends.js'
+import { getSingleValueLegendColor } from './getSingleValueLegendColor.js'
+
+export function getSingleValueBackgroundColor(
+ legendOptions,
+ legendSets,
+ value
+) {
+ const legendColor = getSingleValueLegendColor(
+ legendOptions,
+ legendSets,
+ value
+ )
+ return legendColor && legendOptions.style === LEGEND_DISPLAY_STYLE_FILL
+ ? legendColor
+ : 'transparent'
+}
diff --git a/src/visualizations/config/adapters/dhis_dhis/value/index.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueFormattedValue.js
similarity index 69%
rename from src/visualizations/config/adapters/dhis_dhis/value/index.js
rename to src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueFormattedValue.js
index 508f1c9a4..f0b91dee3 100644
--- a/src/visualizations/config/adapters/dhis_dhis/value/index.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueFormattedValue.js
@@ -1,8 +1,9 @@
-import { renderValue } from '../../../../../modules/renderValue.js'
-import { VALUE_TYPE_TEXT } from '../../../../../modules/valueTypes.js'
-import { INDICATOR_FACTOR_100 } from '../index.js'
+import { renderValue } from '../../../../../../modules/renderValue.js'
+import { VALUE_TYPE_TEXT } from '../../../../../../modules/valueTypes.js'
-export default function (value, layout, metaData) {
+export const INDICATOR_FACTOR_100 = 100
+
+export function getSingleValueFormattedValue(value, layout, metaData) {
const valueType = metaData.items[metaData.dimensions.dx[0]].valueType
const indicatorType =
metaData.items[metaData.dimensions.dx[0]].indicatorType
diff --git a/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueLegendColor.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueLegendColor.js
new file mode 100644
index 000000000..9f042fc4d
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueLegendColor.js
@@ -0,0 +1,8 @@
+import { getColorByValueFromLegendSet } from '../../../../../../modules/legends.js'
+
+export function getSingleValueLegendColor(legendOptions, legendSets, value) {
+ const legendSet = legendOptions && legendSets[0]
+ return legendSet
+ ? getColorByValueFromLegendSet(legendSet, value)
+ : undefined
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueSubtext.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueSubtext.js
new file mode 100644
index 000000000..b14a3f263
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueSubtext.js
@@ -0,0 +1,11 @@
+import { INDICATOR_FACTOR_100 } from './getSingleValueFormattedValue.js'
+
+export function getSingleValueSubtext(metaData) {
+ const indicatorType =
+ metaData.items[metaData.dimensions.dx[0]].indicatorType
+
+ return indicatorType?.displayName &&
+ indicatorType?.factor !== INDICATOR_FACTOR_100
+ ? indicatorType?.displayName
+ : undefined
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueTextColor.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueTextColor.js
new file mode 100644
index 000000000..2f3eb0da0
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueTextColor.js
@@ -0,0 +1,27 @@
+import { colors } from '@dhis2/ui'
+import { LEGEND_DISPLAY_STYLE_TEXT } from '../../../../../../modules/legends.js'
+import { shouldUseContrastColor } from '../../../../../util/shouldUseContrastColor.js'
+import { getSingleValueLegendColor } from './getSingleValueLegendColor.js'
+
+export function getSingleValueTextColor(
+ baseColor,
+ value,
+ legendOptions,
+ legendSets
+) {
+ const legendColor = getSingleValueLegendColor(
+ legendOptions,
+ legendSets,
+ value
+ )
+
+ if (!legendColor) {
+ return baseColor
+ }
+
+ if (legendOptions.style === LEGEND_DISPLAY_STYLE_TEXT) {
+ return legendColor
+ }
+
+ return shouldUseContrastColor(legendColor) ? colors.white : baseColor
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueTitleColor.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueTitleColor.js
new file mode 100644
index 000000000..bf4f0672b
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/getSingleValueTitleColor.js
@@ -0,0 +1,34 @@
+import { colors } from '@dhis2/ui'
+import { LEGEND_DISPLAY_STYLE_FILL } from '../../../../../../modules/legends.js'
+import { shouldUseContrastColor } from '../../../../../util/shouldUseContrastColor.js'
+import { getSingleValueLegendColor } from './getSingleValueLegendColor.js'
+
+export function getSingleValueTitleColor(
+ customColor,
+ defaultColor,
+ value,
+ legendOptions,
+ legendSets
+) {
+ // Never override custom color
+ if (customColor) {
+ return customColor
+ }
+
+ const isUsingLegendBackground =
+ legendOptions?.style === LEGEND_DISPLAY_STYLE_FILL
+
+ // If not using legend background, always return default color
+ if (!isUsingLegendBackground) {
+ return defaultColor
+ }
+
+ const legendColor = getSingleValueLegendColor(
+ legendOptions,
+ legendSets,
+ value
+ )
+
+ // Return default color or contrasting color when using legend background and default color
+ return shouldUseContrastColor(legendColor) ? colors.white : defaultColor
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/index.js b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/index.js
new file mode 100644
index 000000000..bb0ff56f1
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/customSVGOptions/singleValue/index.js
@@ -0,0 +1,27 @@
+import { colors } from '@dhis2/ui'
+import { getSingleValueFormattedValue } from './getSingleValueFormattedValue.js'
+import { getSingleValueSubtext } from './getSingleValueSubtext.js'
+import { getSingleValueTextColor } from './getSingleValueTextColor.js'
+
+export default function getSingleValueCustomSVGOptions({
+ layout,
+ extraOptions,
+ metaData,
+ series,
+}) {
+ const { dashboard, icon } = extraOptions
+ const value = series[0]
+ return {
+ value,
+ fontColor: getSingleValueTextColor(
+ colors.grey900,
+ value,
+ layout.legend,
+ extraOptions.legendSets
+ ),
+ formattedValue: getSingleValueFormattedValue(value, layout, metaData),
+ icon,
+ dashboard,
+ subText: getSingleValueSubtext(metaData),
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/chart.js b/src/visualizations/config/adapters/dhis_highcharts/events/index.js
similarity index 51%
rename from src/visualizations/config/adapters/dhis_highcharts/chart.js
rename to src/visualizations/config/adapters/dhis_highcharts/events/index.js
index e50a52ca9..4f8bf0904 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/chart.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/index.js
@@ -1,20 +1,6 @@
-import getType from './type.js'
+import loadCustomSVG from './loadCustomSVG/index.js'
-const DEFAULT_CHART = {
- spacingTop: 20,
- style: {
- fontFamily: 'Roboto,Helvetica Neue,Helvetica,Arial,sans-serif',
- },
-}
-
-const DASHBOARD_CHART = {
- spacingTop: 0,
- spacingRight: 5,
- spacingBottom: 2,
- spacingLeft: 5,
-}
-
-const getEvents = () => ({
+export const getEvents = (visType) => ({
events: {
load: function () {
// Align legend icon with legend text
@@ -31,17 +17,7 @@ const getEvents = () => ({
})
}
})
+ loadCustomSVG.call(this, visType)
},
},
})
-
-export default function (layout, el, dashboard) {
- return Object.assign(
- {},
- getType(layout.type),
- { renderTo: el || layout.el },
- DEFAULT_CHART,
- dashboard ? DASHBOARD_CHART : undefined,
- getEvents()
- )
-}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/index.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/index.js
new file mode 100644
index 000000000..6e01df566
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/index.js
@@ -0,0 +1,12 @@
+import { VIS_TYPE_SINGLE_VALUE } from '../../../../../../modules/visTypes.js'
+import loadSingleValueSVG from './singleValue/index.js'
+
+export default function loadCustomSVG(visType) {
+ switch (visType) {
+ case VIS_TYPE_SINGLE_VALUE:
+ loadSingleValueSVG.call(this)
+ break
+ default:
+ break
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/addIconElement.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/addIconElement.js
new file mode 100644
index 000000000..dfa2c0c57
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/addIconElement.js
@@ -0,0 +1,32 @@
+const parser = new DOMParser()
+
+export function addIconElement(svgString, color) {
+ const svgIconDocument = parser.parseFromString(svgString, 'image/svg+xml')
+ const iconElHeight = svgIconDocument.documentElement.getAttribute('height')
+ const iconElWidth = svgIconDocument.documentElement.getAttribute('width')
+ const iconGroup = this.renderer
+ .g('icon')
+ .attr({ color, 'data-test': 'visualization-icon' })
+ .css({
+ visibility: 'hidden',
+ })
+
+ /* Force the group element to have the same dimensions as the original
+ * SVG image by adding this rect. This ensures the icon has the intended
+ * whitespace around it and makes scaling and translating easier. */
+ this.renderer.rect(0, 0, iconElWidth, iconElHeight).add(iconGroup)
+
+ Array.from(svgIconDocument.documentElement.children).forEach((pathNode) => {
+ /* It is also possible to use the SVGRenderer to draw the icon but that
+ * approach is more error prone, so during review it was decided to just
+ * append the SVG children to the iconGroup using native the native DOM
+ * API. For reference see this commit, for an implementation using the
+ * SVVGRenderer:
+ * https://github.com/dhis2/analytics/pull/1698/commits/f95bee838e07f4cdfc3cab6e92f28f49a386a0ad */
+ iconGroup.element.appendChild(pathNode)
+ })
+
+ iconGroup.add()
+
+ return iconGroup
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/checkIfFitsWithinContainer.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/checkIfFitsWithinContainer.js
new file mode 100644
index 000000000..182611977
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/checkIfFitsWithinContainer.js
@@ -0,0 +1,29 @@
+import { ACTUAL_NUMBER_HEIGHT_FACTOR } from './constants.js'
+
+export function checkIfFitsWithinContainer(
+ availableSpace,
+ valueElement,
+ subTextElement,
+ icon,
+ subText,
+ spacing
+) {
+ const valueRect = valueElement.getBBox(true)
+ const subTextRect = subText
+ ? subTextElement.getBBox(true)
+ : { width: 0, height: 0 }
+ const requiredValueWidth = icon
+ ? valueRect.width + spacing.iconGap + spacing.iconSize
+ : valueRect.width
+ const requiredHeight = subText
+ ? valueRect.height * ACTUAL_NUMBER_HEIGHT_FACTOR +
+ spacing.subTextTop +
+ subTextRect.height
+ : valueRect.height * ACTUAL_NUMBER_HEIGHT_FACTOR
+ const fitsHorizontally =
+ availableSpace.width > requiredValueWidth &&
+ availableSpace.width > subTextRect.width
+ const fitsVertically = availableSpace.height > requiredHeight
+
+ return fitsHorizontally && fitsVertically
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/computeLayoutRect.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/computeLayoutRect.js
new file mode 100644
index 000000000..a5d2705c9
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/computeLayoutRect.js
@@ -0,0 +1,43 @@
+import { computeSpacingTop } from './computeSpacingTop.js'
+import { ACTUAL_NUMBER_HEIGHT_FACTOR } from './constants.js'
+
+export function computeLayoutRect(
+ valueElement,
+ subTextElement,
+ iconElement,
+ spacing
+) {
+ const valueRect = valueElement.getBBox()
+ const containerCenterY = this.chartHeight / 2
+ const containerCenterX = this.chartWidth / 2
+ const minY = computeSpacingTop.call(this, spacing.valueTop)
+
+ let width = valueRect.width
+ let height = valueRect.height * ACTUAL_NUMBER_HEIGHT_FACTOR
+ let sideMarginTop = 0
+ let sideMarginBottom = 0
+
+ if (iconElement) {
+ width += spacing.iconGap + spacing.iconSize
+ }
+
+ if (subTextElement) {
+ const subTextRect = subTextElement.getBBox()
+ if (subTextRect.width > width) {
+ sideMarginTop = (subTextRect.width - width) / 2
+ width = subTextRect.width
+ } else {
+ sideMarginBottom = (width - subTextRect.width) / 2
+ }
+ height += spacing.subTextTop + subTextRect.height
+ }
+
+ return {
+ x: containerCenterX - width / 2,
+ y: Math.max(containerCenterY - height / 2, minY),
+ width,
+ height,
+ sideMarginTop,
+ sideMarginBottom,
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/computeSpacingTop.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/computeSpacingTop.js
new file mode 100644
index 000000000..1de00c836
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/computeSpacingTop.js
@@ -0,0 +1,15 @@
+export function computeSpacingTop(valueSpacingTop) {
+ if (this.subtitle.textStr) {
+ /* If a subtitle is present this will be below the title so base
+ * the value X position on this */
+ const subTitleRect = this.subtitle.element.getBBox()
+ return subTitleRect.y + subTitleRect.height + valueSpacingTop
+ } else if (this.title.textStr) {
+ // Otherwise base on title
+ const titleRect = this.title.element.getBBox()
+ return titleRect.y + titleRect.height + valueSpacingTop
+ } else {
+ // If neither are present only adjust for valueSpacingTop
+ return valueSpacingTop
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/constants.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/constants.js
new file mode 100644
index 000000000..b76e26a44
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/constants.js
@@ -0,0 +1,4 @@
+// multiply value text size with this factor
+// to get very close to the actual number height
+// as numbers don't go below the baseline like e.g. "j" and "g"
+export const ACTUAL_NUMBER_HEIGHT_FACTOR = 2 / 3
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/getAvailableSpace.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/getAvailableSpace.js
new file mode 100644
index 000000000..c9f567f4c
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/getAvailableSpace.js
@@ -0,0 +1,10 @@
+import { computeSpacingTop } from './computeSpacingTop.js'
+import { MIN_SIDE_WHITESPACE } from './styles.js'
+
+export function getAvailableSpace(valueSpacingTop) {
+ return {
+ height:
+ this.chartHeight - computeSpacingTop.call(this, valueSpacingTop),
+ width: this.chartWidth - MIN_SIDE_WHITESPACE * 2,
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/index.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/index.js
new file mode 100644
index 000000000..84cc83e7d
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/index.js
@@ -0,0 +1,55 @@
+import { addIconElement } from './addIconElement.js'
+import { checkIfFitsWithinContainer } from './checkIfFitsWithinContainer.js'
+import { getAvailableSpace } from './getAvailableSpace.js'
+import { positionElements } from './positionElements.js'
+import { DynamicStyles } from './styles.js'
+
+export default function loadSingleValueSVG() {
+ const { formattedValue, icon, subText, fontColor } =
+ this.userOptions.customSVGOptions
+ const dynamicStyles = new DynamicStyles(this.userOptions?.isPdfExport)
+ const valueElement = this.renderer
+ .text(formattedValue)
+ .attr('data-test', 'visualization-primary-value')
+ .css({ color: fontColor, visibility: 'hidden' })
+ .add()
+ const subTextElement = subText
+ ? this.renderer
+ .text(subText)
+ .attr('data-test', 'visualization-subtext')
+ .css({ color: fontColor, visibility: 'hidden' })
+ .add()
+ : null
+ const iconElement = icon ? addIconElement.call(this, icon, fontColor) : null
+
+ let fitsWithinContainer = false
+ let styles = {}
+
+ while (!fitsWithinContainer && dynamicStyles.hasNext()) {
+ styles = dynamicStyles.next()
+
+ valueElement.css(styles.value)
+ subTextElement?.css(styles.subText)
+
+ fitsWithinContainer = checkIfFitsWithinContainer(
+ getAvailableSpace.call(this, styles.spacing.valueTop),
+ valueElement,
+ subTextElement,
+ icon,
+ subText,
+ styles.spacing
+ )
+ }
+
+ positionElements.call(
+ this,
+ valueElement,
+ subTextElement,
+ iconElement,
+ styles.spacing
+ )
+
+ valueElement.css({ visibility: 'visible' })
+ iconElement?.css({ visibility: 'visible' })
+ subTextElement?.css({ visibility: 'visible' })
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/positionElements.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/positionElements.js
new file mode 100644
index 000000000..052c86b5b
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/positionElements.js
@@ -0,0 +1,62 @@
+import { computeLayoutRect } from './computeLayoutRect.js'
+import { ACTUAL_NUMBER_HEIGHT_FACTOR } from './constants.js'
+
+export function positionElements(
+ valueElement,
+ subTextElement,
+ iconElement,
+ spacing
+) {
+ const valueElementBox = valueElement.getBBox()
+ /* Layout here refers to a virtual rect that wraps
+ * all indiviual parts of the single value visualization
+ * (value, subtext and icon) */
+ const layoutRect = computeLayoutRect.call(
+ this,
+ valueElement,
+ subTextElement,
+ iconElement,
+ spacing
+ )
+
+ valueElement.align(
+ {
+ align: 'right',
+ verticalAlign: 'top',
+ alignByTranslate: false,
+ x: (valueElementBox.width + layoutRect.sideMarginTop) * -1,
+ y: valueElementBox.height * ACTUAL_NUMBER_HEIGHT_FACTOR,
+ },
+ false,
+ layoutRect
+ )
+
+ if (iconElement) {
+ const { height } = iconElement.getBBox()
+ const scale = spacing.iconSize / height
+ const translateX = layoutRect.x + layoutRect.sideMarginTop
+ const iconHeight = height * scale
+ const valueElementHeight =
+ valueElementBox.height * ACTUAL_NUMBER_HEIGHT_FACTOR
+ const translateY = layoutRect.y + (valueElementHeight - iconHeight) / 2
+
+ /* The icon is a
with elements that contain coordinates.
+ * These path-coordinates only scale correctly when using CSS translate */
+ iconElement.css({
+ transform: `translate(${translateX}px, ${translateY}px) scale(${scale})`,
+ })
+ }
+
+ if (subTextElement) {
+ subTextElement.align(
+ {
+ align: 'left',
+ verticalAlign: 'bottom',
+ alignByTranslate: false,
+ x: layoutRect.sideMarginBottom,
+ },
+ false,
+ layoutRect
+ )
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/styles.js b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/styles.js
new file mode 100644
index 000000000..f1b944ee2
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/events/loadCustomSVG/singleValue/styles.js
@@ -0,0 +1,62 @@
+const valueStyles = [
+ { 'font-size': '164px', 'letter-spacing': '-5px' },
+ { 'font-size': '128px', 'letter-spacing': '-4px' },
+ { 'font-size': '96px', 'letter-spacing': '-3px' },
+ { 'font-size': '64px', 'letter-spacing': '-2.5px' },
+ { 'font-size': '40px', 'letter-spacing': '-1.5px' },
+ { 'font-size': '20px', 'letter-spacing': '-1px' },
+]
+
+const subTextStyles = [
+ { 'font-size': '36px', 'letter-spacing': '-1.4px' },
+ { 'font-size': '32px', 'letter-spacing': '-1.2px' },
+ { 'font-size': '26px', 'letter-spacing': '-0.8px' },
+ { 'font-size': '20px', 'letter-spacing': '-0.6px' },
+ { 'font-size': '14px', 'letter-spacing': '0.2px' },
+ { 'font-size': '9px', 'letter-spacing': '0px' },
+]
+
+const spacings = [
+ { valueTop: 8, subTextTop: 12, iconGap: 8, iconSize: 164 },
+ { valueTop: 8, subTextTop: 12, iconGap: 6, iconSize: 128 },
+ { valueTop: 8, subTextTop: 8, iconGap: 4, iconSize: 96 },
+ { valueTop: 8, subTextTop: 8, iconGap: 4, iconSize: 64 },
+ { valueTop: 8, subTextTop: 8, iconGap: 4, iconSize: 40 },
+ { valueTop: 8, subTextTop: 4, iconGap: 2, iconSize: 20 },
+]
+
+export const MIN_SIDE_WHITESPACE = 4
+
+export class DynamicStyles {
+ constructor(isPdfExport) {
+ this.currentIndex = 0
+ this.isPdfExport = isPdfExport
+ }
+ getStyle() {
+ return {
+ value: {
+ ...valueStyles[this.currentIndex],
+ 'font-weight': this.isPdfExport ? 'normal' : '300',
+ },
+ subText: subTextStyles[this.currentIndex],
+ spacing: spacings[this.currentIndex],
+ }
+ }
+ next() {
+ if (this.currentIndex === valueStyles.length - 1) {
+ throw new Error('No next available, already on the smallest style')
+ } else {
+ ++this.currentIndex
+ }
+
+ return this.getStyle()
+ }
+ first() {
+ this.currentIndex = 0
+
+ return this.getStyle()
+ }
+ hasNext() {
+ return this.currentIndex < valueStyles.length - 1
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/exporting.js b/src/visualizations/config/adapters/dhis_highcharts/exporting.js
new file mode 100644
index 000000000..032a9c689
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/exporting.js
@@ -0,0 +1,25 @@
+import { VIS_TYPE_SINGLE_VALUE } from '../../../../modules/visTypes.js'
+import loadSingleValueSVG from './events/loadCustomSVG/singleValue/index.js'
+
+export default function getExporting(visType) {
+ const exporting = {
+ // disable exporting context menu
+ enabled: false,
+ }
+ switch (visType) {
+ case VIS_TYPE_SINGLE_VALUE:
+ return {
+ ...exporting,
+ chartOptions: {
+ chart: {
+ events: {
+ load: loadSingleValueSVG,
+ },
+ },
+ },
+ }
+
+ default:
+ return exporting
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/index.js b/src/visualizations/config/adapters/dhis_highcharts/index.js
index 29ecf41c0..0f3ddb271 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/index.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/index.js
@@ -14,10 +14,13 @@ import {
} from '../../../../modules/visTypes.js'
import { defaultMultiAxisTheme1 } from '../../../util/colors/themes.js'
import addTrendLines, { isRegressionIneligible } from './addTrendLines.js'
-import getChart from './chart.js'
+import getChart from './chart/index.js'
+import getCustomSVGOptions from './customSVGOptions/index.js'
+import getExporting from './exporting.js'
import getScatterData from './getScatterData.js'
import getSortedConfig from './getSortedConfig.js'
import getTrimmedConfig from './getTrimmedConfig.js'
+import getLang from './lang.js'
import getLegend from './legend.js'
import { applyLegendSet, getLegendSetTooltip } from './legendSet.js'
import getNoData from './noData.js'
@@ -77,21 +80,17 @@ export default function ({ store, layout, el, extraConfig, extraOptions }) {
let config = {
// type etc
- chart: getChart(_layout, el, _extraOptions.dashboard),
+ chart: getChart(_layout, el, _extraOptions, series),
// title
- title: getTitle(
- _layout,
- store.data[0].metaData,
- _extraOptions.dashboard
- ),
+ title: getTitle(_layout, store.data[0].metaData, _extraOptions, series),
// subtitle
subtitle: getSubtitle(
series,
_layout,
store.data[0].metaData,
- _extraOptions.dashboard
+ _extraOptions
),
// x-axis
@@ -123,11 +122,8 @@ export default function ({ store, layout, el, extraConfig, extraOptions }) {
pane: getPane(_layout.type),
// no data + zoom
- lang: {
- noData: _extraOptions.noData.text,
- resetZoom: _extraOptions.resetZoom.text,
- },
- noData: getNoData(),
+ lang: getLang(_layout.type, _extraOptions),
+ noData: getNoData(_layout.type),
// credits
credits: {
@@ -135,10 +131,20 @@ export default function ({ store, layout, el, extraConfig, extraOptions }) {
},
// exporting
- exporting: {
- // disable exporting context menu
- enabled: false,
- },
+ exporting: getExporting(_layout.type),
+
+ /* The config object passed to the Highcharts Chart constructor
+ * can contain arbitrary properties, which are made accessible
+ * under the Chart instance's `userOptions` member. This means
+ * that in event callback functions the custom SVG options are
+ * accessible as `this.userOptions.customSVGOptions` */
+ customSVGOptions: getCustomSVGOptions({
+ extraConfig,
+ layout: _layout,
+ extraOptions: _extraOptions,
+ metaData: store.data[0].metaData,
+ series,
+ }),
}
// get plot options for scatter
@@ -234,5 +240,7 @@ export default function ({ store, layout, el, extraConfig, extraOptions }) {
// force apply extra config
Object.assign(config, extraConfig)
+ console.log(objectClean(config))
+
return objectClean(config)
}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/lang.js b/src/visualizations/config/adapters/dhis_highcharts/lang.js
new file mode 100644
index 000000000..80299fe41
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/lang.js
@@ -0,0 +1,15 @@
+import { VIS_TYPE_SINGLE_VALUE } from '../../../../modules/visTypes.js'
+
+export default function getLang(visType, extraOptions) {
+ return {
+ /* The SingleValue visualization consists of some custom SVG elements
+ * rendered on an empty chart. Since the chart is empty, there is never
+ * any data and Highcharts will show the noData text. To avoid this we
+ * clear the text here. */
+ noData:
+ visType === VIS_TYPE_SINGLE_VALUE
+ ? undefined
+ : extraOptions.noData.text,
+ resetZoom: extraOptions.resetZoom.text,
+ }
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/plotOptions.js b/src/visualizations/config/adapters/dhis_highcharts/plotOptions.js
index 928019506..e9e775096 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/plotOptions.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/plotOptions.js
@@ -79,6 +79,6 @@ export default ({
}
: {}
default:
- return {}
+ return null
}
}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/series/__tests__/pie.spec.js b/src/visualizations/config/adapters/dhis_highcharts/series/__tests__/pie.spec.js
new file mode 100644
index 000000000..eb9abac19
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/series/__tests__/pie.spec.js
@@ -0,0 +1,66 @@
+import { formatDataLabel } from '../pie.js'
+
+describe('formatDataLabel', () => {
+ it('should format data label correctly with integers', () => {
+ const result = formatDataLabel('Test', 1000, 50)
+ expect(result).toEqual(
+ 'Test
1 000 (50%)'
+ )
+ })
+
+ it('should format data label correctly with decimals', () => {
+ const result = formatDataLabel('Test', 1000.123456789, 50.1234)
+ expect(result).toEqual(
+ 'Test
1 000.123456789 (50.1%)'
+ )
+ })
+
+ it('should handle large numbers correctly', () => {
+ const result = formatDataLabel('Test', 1000000, 75.5678)
+ expect(result).toEqual(
+ 'Test
1 000 000 (75.6%)'
+ )
+ })
+
+ it('should handle small percentages correctly', () => {
+ const result = formatDataLabel('Test', 1000.000001, 0.09)
+ expect(result).toEqual(
+ 'Test
1 000.000001 (0.1%)'
+ )
+ })
+
+ it('should handle zero correctly', () => {
+ const result = formatDataLabel('Test', 0, 0)
+ expect(result).toEqual(
+ 'Test
0 (0%)'
+ )
+ })
+
+ it('should handle negative numbers correctly', () => {
+ const result = formatDataLabel('Test', -1000, -50)
+ expect(result).toEqual(
+ 'Test
-1 000 (-50%)'
+ )
+ })
+
+ it('should handle empty string as name correctly', () => {
+ const result = formatDataLabel('', 1000, 50)
+ expect(result).toEqual(
+ '
1 000 (50%)'
+ )
+ })
+
+ it('should handle undefined as name correctly', () => {
+ const result = formatDataLabel(undefined, 1000, 50)
+ expect(result).toEqual(
+ '
1 000 (50%)'
+ )
+ })
+
+ it('should handle special characters in name correctly', () => {
+ const result = formatDataLabel('Test&Test', 1000, 50)
+ expect(result).toEqual(
+ 'Test&Test
1 000 (50%)'
+ )
+ })
+})
diff --git a/src/visualizations/config/adapters/dhis_highcharts/series/index.js b/src/visualizations/config/adapters/dhis_highcharts/series/index.js
index e4d4eae67..e4ec840f0 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/series/index.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/series/index.js
@@ -9,6 +9,7 @@ import {
isYearOverYear,
VIS_TYPE_LINE,
VIS_TYPE_SCATTER,
+ VIS_TYPE_SINGLE_VALUE,
} from '../../../../../modules/visTypes.js'
import { getAxisStringFromId } from '../../../../util/axisId.js'
import {
@@ -225,6 +226,9 @@ export default function ({
displayStrategy,
}) {
switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ series = []
+ break
case VIS_TYPE_PIE:
series = getPie(
series,
@@ -249,7 +253,7 @@ export default function ({
})
}
- series.forEach((seriesObj) => {
+ series?.forEach((seriesObj) => {
// animation
seriesObj.animation = {
duration: getAnimation(
diff --git a/src/visualizations/config/adapters/dhis_highcharts/series/pie.js b/src/visualizations/config/adapters/dhis_highcharts/series/pie.js
index 67a4e772d..b5afb2c49 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/series/pie.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/series/pie.js
@@ -1,3 +1,18 @@
+import { separateDigitGroups } from '../../../../../modules/renderValue.js'
+
+export const formatDataLabel = (name = '', y, percentage) => {
+ const value = separateDigitGroups(y.toString()).join(' ')
+ return (
+ '' +
+ name +
+ '
' +
+ value +
+ ' (' +
+ parseFloat(percentage.toFixed(1)) +
+ '%)'
+ )
+}
+
export default function (series, colors) {
return [
{
@@ -9,14 +24,10 @@ export default function (series, colors) {
dataLabels: {
enabled: true,
formatter: function () {
- return (
- '' +
- this.point.name +
- '
' +
- this.y +
- ' (' +
- this.percentage.toFixed(1) +
- ' %)'
+ return formatDataLabel(
+ this.point.name,
+ this.y,
+ this.percentage
)
},
},
diff --git a/src/visualizations/config/adapters/dhis_highcharts/subtitle/__tests__/singleValue.spec.js b/src/visualizations/config/adapters/dhis_highcharts/subtitle/__tests__/singleValue.spec.js
new file mode 100644
index 000000000..c7baa2ad6
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/subtitle/__tests__/singleValue.spec.js
@@ -0,0 +1,64 @@
+import getSingleValueSubtitle from '../singleValue.js'
+
+jest.mock(
+ '../../../../../util/getFilterText',
+ () => () => 'The default filter text'
+)
+
+describe('getSingleValueSubtitle', () => {
+ it('returns empty subtitle when flag hideSubtitle exists', () => {
+ expect(getSingleValueSubtitle({ hideSubtitle: true })).toEqual('')
+ })
+
+ it('returns the subtitle provided in the layout', () => {
+ const subtitle = 'The subtitle was already set'
+ expect(getSingleValueSubtitle({ subtitle })).toEqual(subtitle)
+ })
+
+ it('returns an empty string when layout does not have filters', () => {
+ expect(getSingleValueSubtitle({})).toEqual('')
+ })
+
+ it('returns the filter text', () => {
+ expect(getSingleValueSubtitle({ filters: [] })).toEqual(
+ 'The default filter text'
+ )
+ })
+
+ describe('not dashboard', () => {
+ describe('layout does not include title', () => {
+ it('returns empty subtitle', () => {
+ expect(
+ getSingleValueSubtitle({ filters: undefined }, {}, false)
+ ).toEqual('')
+ })
+ })
+
+ /* All these tests have been moved and adjusted from here:
+ * src/visualizations/config/adapters/dhis_dhis/title/__tests__`
+ * The test below asserted the default subtitle behaviour, for
+ * visualization types other than SingleValue. It expected that
+ * the title was being used as subtitle. It fails now, and I
+ * believe that this behaviour does not make sense. So instead
+ * of fixing it, I disabled it. */
+ // describe('layout includes title', () => {
+ // it('returns filter title as subtitle', () => {
+ // expect(
+ // getSingleValueSubtitle(
+ // { filters: undefined, title: 'Chart title' },
+ // {},
+ // false
+ // )
+ // ).toEqual('The default filter text')
+ // })
+ // })
+ })
+
+ describe('dashboard', () => {
+ it('returns filter title as subtitle', () => {
+ expect(getSingleValueSubtitle({ filters: {} }, {}, true)).toEqual(
+ 'The default filter text'
+ )
+ })
+ })
+})
diff --git a/src/visualizations/config/adapters/dhis_highcharts/subtitle/index.js b/src/visualizations/config/adapters/dhis_highcharts/subtitle/index.js
index 9d2dc1bc7..6509c3e5a 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/subtitle/index.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/subtitle/index.js
@@ -7,16 +7,21 @@ import {
FONT_STYLE_OPTION_TEXT_ALIGN,
FONT_STYLE_VISUALIZATION_SUBTITLE,
mergeFontStyleWithDefault,
+ defaultFontStyle,
} from '../../../../../modules/fontStyle.js'
import {
VIS_TYPE_YEAR_OVER_YEAR_LINE,
VIS_TYPE_YEAR_OVER_YEAR_COLUMN,
isVerticalType,
VIS_TYPE_SCATTER,
+ VIS_TYPE_SINGLE_VALUE,
} from '../../../../../modules/visTypes.js'
import getFilterText from '../../../../util/getFilterText.js'
import { getTextAlignOption } from '../getTextAlignOption.js'
import getYearOverYearTitle from '../title/yearOverYear.js'
+import getSingleValueSubtitle, {
+ getSingleValueSubtitleColor,
+} from './singleValue.js'
const DASHBOARD_SUBTITLE = {
style: {
@@ -31,23 +36,48 @@ const DASHBOARD_SUBTITLE = {
}
function getDefault(layout, dashboard, filterTitle) {
- return {
- text: dashboard || isString(layout.title) ? filterTitle : undefined,
- }
+ return dashboard || isString(layout.title) ? filterTitle : undefined
}
-export default function (series, layout, metaData, dashboard) {
+export default function (series, layout, metaData, extraOptions) {
+ if (layout.hideSubtitle) {
+ return null
+ }
+
+ const { dashboard, legendSets } = extraOptions
+ const legendOptions = layout.legend
const fontStyle = mergeFontStyleWithDefault(
layout.fontStyle && layout.fontStyle[FONT_STYLE_VISUALIZATION_SUBTITLE],
FONT_STYLE_VISUALIZATION_SUBTITLE
)
- let subtitle = {
- text: undefined,
- }
-
- if (layout.hideSubtitle) {
- return null
- }
+ const subtitle = Object.assign(
+ {
+ text: undefined,
+ },
+ dashboard
+ ? DASHBOARD_SUBTITLE
+ : {
+ align: getTextAlignOption(
+ fontStyle[FONT_STYLE_OPTION_TEXT_ALIGN],
+ FONT_STYLE_VISUALIZATION_SUBTITLE,
+ isVerticalType(layout.type)
+ ),
+ style: {
+ // DHIS2-578: dynamically truncate subtitle when it's taking more than 1 line
+ color: undefined,
+ fontSize: `${fontStyle[FONT_STYLE_OPTION_FONT_SIZE]}px`,
+ fontWeight: fontStyle[FONT_STYLE_OPTION_BOLD]
+ ? FONT_STYLE_OPTION_BOLD
+ : 'normal',
+ fontStyle: fontStyle[FONT_STYLE_OPTION_ITALIC]
+ ? FONT_STYLE_OPTION_ITALIC
+ : 'normal',
+ whiteSpace: 'nowrap',
+ overflow: 'hidden',
+ textOverflow: 'ellipsis',
+ },
+ }
+ )
// DHIS2-578: allow for optional custom subtitle
const customSubtitle =
@@ -59,6 +89,9 @@ export default function (series, layout, metaData, dashboard) {
const filterTitle = getFilterText(layout.filters, metaData)
switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ subtitle.text = getSingleValueSubtitle(layout, metaData)
+ break
case VIS_TYPE_YEAR_OVER_YEAR_LINE:
case VIS_TYPE_YEAR_OVER_YEAR_COLUMN:
subtitle.text = getYearOverYearTitle(
@@ -71,37 +104,46 @@ export default function (series, layout, metaData, dashboard) {
subtitle.text = filterTitle
break
default:
- subtitle = getDefault(layout, dashboard, filterTitle)
+ subtitle.text = getDefault(layout, dashboard, filterTitle)
}
}
+ switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ {
+ const defaultColor =
+ defaultFontStyle?.[FONT_STYLE_VISUALIZATION_SUBTITLE]?.[
+ FONT_STYLE_OPTION_TEXT_COLOR
+ ]
+ const customColor =
+ layout?.fontStyle?.[FONT_STYLE_VISUALIZATION_SUBTITLE]?.[
+ FONT_STYLE_OPTION_TEXT_COLOR
+ ]
+ subtitle.style.color = getSingleValueSubtitleColor(
+ customColor,
+ defaultColor,
+ series[0],
+ legendOptions,
+ legendSets
+ )
+ if (dashboard) {
+ // Single value subtitle text should be multiline
+ /* TODO: The default color of the subtitle now is #4a5768 but the
+ * original implementation used #666, which is a lighter grey.
+ * If we want to keep this color, changes are needed here. */
+ Object.assign(subtitle.style, {
+ wordWrap: 'normal',
+ whiteSpace: 'normal',
+ overflow: 'visible',
+ textOverflow: 'initial',
+ })
+ }
+ }
+ break
+ default:
+ subtitle.style.color = fontStyle[FONT_STYLE_OPTION_TEXT_COLOR]
+ break
+ }
+
return subtitle
- ? Object.assign(
- {},
- dashboard
- ? DASHBOARD_SUBTITLE
- : {
- align: getTextAlignOption(
- fontStyle[FONT_STYLE_OPTION_TEXT_ALIGN],
- FONT_STYLE_VISUALIZATION_SUBTITLE,
- isVerticalType(layout.type)
- ),
- style: {
- // DHIS2-578: dynamically truncate subtitle when it's taking more than 1 line
- color: fontStyle[FONT_STYLE_OPTION_TEXT_COLOR],
- fontSize: `${fontStyle[FONT_STYLE_OPTION_FONT_SIZE]}px`,
- fontWeight: fontStyle[FONT_STYLE_OPTION_BOLD]
- ? FONT_STYLE_OPTION_BOLD
- : 'normal',
- fontStyle: fontStyle[FONT_STYLE_OPTION_ITALIC]
- ? FONT_STYLE_OPTION_ITALIC
- : 'normal',
- whiteSpace: 'nowrap',
- overflow: 'hidden',
- textOverflow: 'ellipsis',
- },
- },
- subtitle
- )
- : subtitle
}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/subtitle/singleValue.js b/src/visualizations/config/adapters/dhis_highcharts/subtitle/singleValue.js
new file mode 100644
index 000000000..922f142cf
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/subtitle/singleValue.js
@@ -0,0 +1,18 @@
+import getFilterText from '../../../../util/getFilterText.js'
+export { getSingleValueTitleColor as getSingleValueSubtitleColor } from '../customSVGOptions/singleValue/getSingleValueTitleColor.js'
+
+export default function getSingleValueSubtitle(layout, metaData) {
+ if (layout.hideSubtitle || 1 === 0) {
+ return ''
+ }
+
+ if (typeof layout.subtitle === 'string' && layout.subtitle.length) {
+ return layout.subtitle
+ }
+
+ if (layout.filters) {
+ return getFilterText(layout.filters, metaData)
+ }
+
+ return ''
+}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/title/__tests__/singleValue.spec.js b/src/visualizations/config/adapters/dhis_highcharts/title/__tests__/singleValue.spec.js
new file mode 100644
index 000000000..bc8022f81
--- /dev/null
+++ b/src/visualizations/config/adapters/dhis_highcharts/title/__tests__/singleValue.spec.js
@@ -0,0 +1,57 @@
+import { getSingleValueTitleText } from '../singleValue.js'
+
+jest.mock('../../../../../util/getFilterText', () => () => 'The filter text')
+
+describe('getSingleValueTitle', () => {
+ it('returns empty title when flag hideTitle exists', () => {
+ expect(getSingleValueTitleText({ hideTitle: true })).toEqual('')
+ })
+
+ it('returns the title provided in the layout', () => {
+ const title = 'The title was already set'
+ expect(getSingleValueTitleText({ title })).toEqual(title)
+ })
+
+ it('returns null when layout does not have columns', () => {
+ expect(getSingleValueTitleText({})).toEqual('')
+ })
+
+ it('returns the filter text based on column items', () => {
+ expect(
+ getSingleValueTitleText({
+ columns: [
+ {
+ items: [{}],
+ },
+ ],
+ })
+ ).toEqual('The filter text')
+ })
+
+ describe('not dashboard', () => {
+ it('returns filter text as title', () => {
+ expect(
+ getSingleValueTitleText(
+ {
+ columns: [
+ {
+ items: [{}],
+ },
+ ],
+ filters: [],
+ },
+ {},
+ false
+ )
+ ).toEqual('The filter text')
+ })
+ })
+
+ describe('dashboard', () => {
+ it('returns empty string', () => {
+ expect(getSingleValueTitleText({ filters: {} }, {}, true)).toEqual(
+ ''
+ )
+ })
+ })
+})
diff --git a/src/visualizations/config/adapters/dhis_highcharts/title/index.js b/src/visualizations/config/adapters/dhis_highcharts/title/index.js
index e4e4f1a4a..7a86ec47f 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/title/index.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/title/index.js
@@ -7,6 +7,7 @@ import {
FONT_STYLE_OPTION_TEXT_ALIGN,
FONT_STYLE_VISUALIZATION_TITLE,
mergeFontStyleWithDefault,
+ defaultFontStyle,
} from '../../../../../modules/fontStyle.js'
import {
VIS_TYPE_YEAR_OVER_YEAR_LINE,
@@ -14,10 +15,15 @@ import {
VIS_TYPE_GAUGE,
isVerticalType,
VIS_TYPE_SCATTER,
+ VIS_TYPE_SINGLE_VALUE,
} from '../../../../../modules/visTypes.js'
import getFilterText from '../../../../util/getFilterText.js'
import { getTextAlignOption } from '../getTextAlignOption.js'
import getScatterTitle from './scatter.js'
+import {
+ getSingleValueTitleColor,
+ getSingleValueTitleText,
+} from './singleValue.js'
import getYearOverYearTitle from './yearOverYear.js'
const DASHBOARD_TITLE_STYLE = {
@@ -41,42 +47,22 @@ function getDefault(layout, metaData, dashboard) {
return null
}
-export default function (layout, metaData, dashboard) {
+export default function (layout, metaData, extraOptions, series) {
+ if (layout.hideTitle) {
+ return {
+ text: undefined,
+ }
+ }
+ const { dashboard, legendSets } = extraOptions
+ const legendOptions = layout.legend
const fontStyle = mergeFontStyleWithDefault(
layout.fontStyle && layout.fontStyle[FONT_STYLE_VISUALIZATION_TITLE],
FONT_STYLE_VISUALIZATION_TITLE
)
-
- const title = {
- text: undefined,
- }
-
- if (layout.hideTitle) {
- return title
- }
-
- const customTitle = (layout.title && layout.displayTitle) || layout.title
-
- if (isString(customTitle) && customTitle.length) {
- title.text = customTitle
- } else {
- switch (layout.type) {
- case VIS_TYPE_GAUGE:
- case VIS_TYPE_YEAR_OVER_YEAR_LINE:
- case VIS_TYPE_YEAR_OVER_YEAR_COLUMN:
- title.text = getYearOverYearTitle(layout, metaData, dashboard)
- break
- case VIS_TYPE_SCATTER:
- title.text = getScatterTitle(layout, metaData, dashboard)
- break
- default:
- title.text = getDefault(layout, metaData, dashboard)
- break
- }
- }
-
- return Object.assign(
- {},
+ const title = Object.assign(
+ {
+ text: undefined,
+ },
dashboard
? DASHBOARD_TITLE_STYLE
: {
@@ -87,7 +73,7 @@ export default function (layout, metaData, dashboard) {
isVerticalType(layout.type)
),
style: {
- color: fontStyle[FONT_STYLE_OPTION_TEXT_COLOR],
+ color: undefined,
fontSize: `${fontStyle[FONT_STYLE_OPTION_FONT_SIZE]}px`,
fontWeight: fontStyle[FONT_STYLE_OPTION_BOLD]
? FONT_STYLE_OPTION_BOLD
@@ -99,7 +85,65 @@ export default function (layout, metaData, dashboard) {
overflow: 'hidden',
textOverflow: 'ellipsis',
},
- },
- title
+ }
)
+
+ const customTitleText =
+ (layout.title && layout.displayTitle) || layout.title
+
+ if (isString(customTitleText) && customTitleText.length) {
+ title.text = customTitleText
+ } else {
+ switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ title.text = getSingleValueTitleText(
+ layout,
+ metaData,
+ dashboard
+ )
+ break
+ case VIS_TYPE_GAUGE:
+ case VIS_TYPE_YEAR_OVER_YEAR_LINE:
+ case VIS_TYPE_YEAR_OVER_YEAR_COLUMN:
+ title.text = getYearOverYearTitle(layout, metaData, dashboard)
+ break
+ case VIS_TYPE_SCATTER:
+ title.text = getScatterTitle(layout, metaData, dashboard)
+ break
+ default:
+ title.text = getDefault(layout, metaData, dashboard)
+ break
+ }
+ }
+
+ switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ {
+ const defaultColor =
+ defaultFontStyle?.[FONT_STYLE_VISUALIZATION_TITLE]?.[
+ FONT_STYLE_OPTION_TEXT_COLOR
+ ]
+ const customColor =
+ layout?.fontStyle?.[FONT_STYLE_VISUALIZATION_TITLE]?.[
+ FONT_STYLE_OPTION_TEXT_COLOR
+ ]
+ title.style.color = getSingleValueTitleColor(
+ customColor,
+ defaultColor,
+ series[0],
+ legendOptions,
+ legendSets
+ )
+ if (dashboard) {
+ // TODO: is this always what we want?
+ title.style.fontWeight = 'normal'
+ }
+ }
+ break
+ default:
+ title.style.color = fontStyle[FONT_STYLE_OPTION_TEXT_COLOR]
+ break
+ }
+
+ return title
}
diff --git a/src/visualizations/config/adapters/dhis_dhis/title/singleValue.js b/src/visualizations/config/adapters/dhis_highcharts/title/singleValue.js
similarity index 50%
rename from src/visualizations/config/adapters/dhis_dhis/title/singleValue.js
rename to src/visualizations/config/adapters/dhis_highcharts/title/singleValue.js
index 802c866c0..fdf5d891a 100644
--- a/src/visualizations/config/adapters/dhis_dhis/title/singleValue.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/title/singleValue.js
@@ -1,6 +1,15 @@
import getFilterText from '../../../../util/getFilterText.js'
+export { getSingleValueTitleColor } from '../customSVGOptions/singleValue/getSingleValueTitleColor.js'
+
+export function getSingleValueTitleText(layout, metaData) {
+ if (layout.hideTitle) {
+ return ''
+ }
+
+ if (typeof layout.title === 'string' && layout.title.length) {
+ return layout.title
+ }
-export default function (layout, metaData) {
if (layout.columns) {
const firstItem = layout.columns[0].items[0]
@@ -10,6 +19,5 @@ export default function (layout, metaData) {
return getFilterText([column], metaData)
}
-
return ''
}
diff --git a/src/visualizations/config/adapters/dhis_highcharts/type.js b/src/visualizations/config/adapters/dhis_highcharts/type.js
index bc56c6d98..08cb62a49 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/type.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/type.js
@@ -12,6 +12,7 @@ import {
VIS_TYPE_STACKED_COLUMN,
VIS_TYPE_YEAR_OVER_YEAR_COLUMN,
VIS_TYPE_SCATTER,
+ VIS_TYPE_SINGLE_VALUE,
} from '../../../../modules/visTypes.js'
export default function (type) {
@@ -33,6 +34,8 @@ export default function (type) {
return { type: 'solidgauge' }
case VIS_TYPE_SCATTER:
return { type: 'scatter', zoomType: 'xy' }
+ case VIS_TYPE_SINGLE_VALUE:
+ return {}
case VIS_TYPE_COLUMN:
case VIS_TYPE_STACKED_COLUMN:
case VIS_TYPE_YEAR_OVER_YEAR_COLUMN:
diff --git a/src/visualizations/config/adapters/dhis_highcharts/xAxis/index.js b/src/visualizations/config/adapters/dhis_highcharts/xAxis/index.js
index c3af4b20b..1439fc201 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/xAxis/index.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/xAxis/index.js
@@ -16,6 +16,7 @@ import {
VIS_TYPE_RADAR,
VIS_TYPE_SCATTER,
isTwoCategoryChartType,
+ VIS_TYPE_SINGLE_VALUE,
} from '../../../../../modules/visTypes.js'
import { getAxis } from '../../../../util/axes.js'
import getAxisTitle from '../getAxisTitle.js'
@@ -82,6 +83,7 @@ export default function (store, layout, extraOptions, series) {
switch (layout.type) {
case VIS_TYPE_PIE:
case VIS_TYPE_GAUGE:
+ case VIS_TYPE_SINGLE_VALUE:
xAxis = noAxis()
break
case VIS_TYPE_YEAR_OVER_YEAR_LINE:
diff --git a/src/visualizations/config/adapters/dhis_highcharts/yAxis/index.js b/src/visualizations/config/adapters/dhis_highcharts/yAxis/index.js
index 1e9aab2a9..d253acdff 100644
--- a/src/visualizations/config/adapters/dhis_highcharts/yAxis/index.js
+++ b/src/visualizations/config/adapters/dhis_highcharts/yAxis/index.js
@@ -11,6 +11,7 @@ import {
isStacked,
VIS_TYPE_GAUGE,
VIS_TYPE_SCATTER,
+ VIS_TYPE_SINGLE_VALUE,
} from '../../../../../modules/visTypes.js'
import { getAxis } from '../../../../util/axes.js'
import { getAxisStringFromId } from '../../../../util/axisId.js'
@@ -148,14 +149,12 @@ function getDefault(layout, series, extraOptions) {
}
export default function (layout, series, extraOptions) {
- let yAxis
switch (layout.type) {
+ case VIS_TYPE_SINGLE_VALUE:
+ return null
case VIS_TYPE_GAUGE:
- yAxis = getGauge(layout, series, extraOptions.legendSets[0])
- break
+ return getGauge(layout, series, extraOptions.legendSets[0])
default:
- yAxis = getDefault(layout, series, extraOptions)
+ return getDefault(layout, series, extraOptions)
}
-
- return yAxis
}
diff --git a/src/visualizations/config/adapters/index.js b/src/visualizations/config/adapters/index.js
index 7b49438ee..4db1838e0 100644
--- a/src/visualizations/config/adapters/index.js
+++ b/src/visualizations/config/adapters/index.js
@@ -1,7 +1,5 @@
-import dhis_dhis from './dhis_dhis/index.js'
import dhis_highcharts from './dhis_highcharts/index.js'
export default {
dhis_highcharts,
- dhis_dhis,
}
diff --git a/src/visualizations/config/generators/dhis/index.js b/src/visualizations/config/generators/dhis/index.js
deleted file mode 100644
index b5a6c3958..000000000
--- a/src/visualizations/config/generators/dhis/index.js
+++ /dev/null
@@ -1,36 +0,0 @@
-import { VIS_TYPE_SINGLE_VALUE } from '../../../../modules/visTypes.js'
-import getSingleValueGenerator from './singleValue.js'
-
-export default function (config, parentEl, extraOptions) {
- if (config) {
- const node =
- typeof parentEl === 'object'
- ? parentEl
- : typeof parentEl === 'string'
- ? document.querySelector(parentEl)
- : null
-
- if (node) {
- if (node.lastChild) {
- node.removeChild(node.lastChild)
- }
-
- let content
-
- switch (config.type) {
- case VIS_TYPE_SINGLE_VALUE:
- default:
- content = getSingleValueGenerator(
- config,
- node,
- extraOptions
- )
- break
- }
-
- node.appendChild(content)
-
- return node.innerHTML
- }
- }
-}
diff --git a/src/visualizations/config/generators/dhis/singleValue.js b/src/visualizations/config/generators/dhis/singleValue.js
deleted file mode 100644
index 25ec5bab9..000000000
--- a/src/visualizations/config/generators/dhis/singleValue.js
+++ /dev/null
@@ -1,531 +0,0 @@
-import { colors } from '@dhis2/ui'
-import {
- FONT_STYLE_VISUALIZATION_TITLE,
- FONT_STYLE_VISUALIZATION_SUBTITLE,
- FONT_STYLE_OPTION_FONT_SIZE,
- FONT_STYLE_OPTION_TEXT_COLOR,
- FONT_STYLE_OPTION_TEXT_ALIGN,
- FONT_STYLE_OPTION_ITALIC,
- FONT_STYLE_OPTION_BOLD,
- TEXT_ALIGN_LEFT,
- TEXT_ALIGN_RIGHT,
- TEXT_ALIGN_CENTER,
- mergeFontStyleWithDefault,
- defaultFontStyle,
-} from '../../../../modules/fontStyle.js'
-import {
- getColorByValueFromLegendSet,
- LEGEND_DISPLAY_STYLE_FILL,
-} from '../../../../modules/legends.js'
-
-const svgNS = 'http://www.w3.org/2000/svg'
-
-// multiply text width with this factor
-// to get very close to actual text width
-// nb: dependent on viewbox etc
-const ACTUAL_TEXT_WIDTH_FACTOR = 0.9
-
-// multiply value text size with this factor
-// to get very close to the actual number height
-// as numbers don't go below the baseline like e.g. "j" and "g"
-const ACTUAL_NUMBER_HEIGHT_FACTOR = 0.67
-
-// do not allow text width to exceed this threshold
-// a threshold >1 does not really make sense but text width vs viewbox is complicated
-const TEXT_WIDTH_CONTAINER_WIDTH_FACTOR = 1.3
-
-// do not allow text size to exceed this
-const TEXT_SIZE_CONTAINER_HEIGHT_FACTOR = 0.6
-const TEXT_SIZE_MAX_THRESHOLD = 400
-
-// multiply text size with this factor
-// to get an appropriate letter spacing
-const LETTER_SPACING_TEXT_SIZE_FACTOR = (1 / 35) * -1
-const LETTER_SPACING_MIN_THRESHOLD = -6
-const LETTER_SPACING_MAX_THRESHOLD = -1
-
-// fixed top margin above title/subtitle
-const TOP_MARGIN_FIXED = 16
-
-// multiply text size with this factor
-// to get an appropriate sub text size
-const SUB_TEXT_SIZE_FACTOR = 0.5
-const SUB_TEXT_SIZE_MIN_THRESHOLD = 26
-const SUB_TEXT_SIZE_MAX_THRESHOLD = 40
-
-// multiply text size with this factor
-// to get an appropriate icon padding
-const ICON_PADDING_FACTOR = 0.3
-
-// Compute text width before rendering
-// Not exactly precise but close enough
-const getTextWidth = (text, font) => {
- const canvas = document.createElement('canvas')
- const context = canvas.getContext('2d')
- context.font = font
- return Math.round(
- context.measureText(text).width * ACTUAL_TEXT_WIDTH_FACTOR
- )
-}
-
-const getTextHeightForNumbers = (textSize) =>
- textSize * ACTUAL_NUMBER_HEIGHT_FACTOR
-
-const getIconPadding = (textSize) => Math.round(textSize * ICON_PADDING_FACTOR)
-
-const getTextSize = (
- formattedValue,
- containerWidth,
- containerHeight,
- showIcon
-) => {
- let size = Math.min(
- Math.round(containerHeight * TEXT_SIZE_CONTAINER_HEIGHT_FACTOR),
- TEXT_SIZE_MAX_THRESHOLD
- )
-
- const widthThreshold = Math.round(
- containerWidth * TEXT_WIDTH_CONTAINER_WIDTH_FACTOR
- )
-
- const textWidth =
- getTextWidth(formattedValue, `${size}px Roboto`) +
- (showIcon ? getIconPadding(size) : 0)
-
- if (textWidth > widthThreshold) {
- size = Math.round(size * (widthThreshold / textWidth))
- }
-
- return size
-}
-
-const generateValueSVG = ({
- formattedValue,
- subText,
- valueColor,
- textColor,
- icon,
- noData,
- containerWidth,
- containerHeight,
- topMargin = 0,
-}) => {
- const showIcon = icon && formattedValue !== noData.text
-
- const textSize = getTextSize(
- formattedValue,
- containerWidth,
- containerHeight,
- showIcon
- )
-
- const textWidth = getTextWidth(formattedValue, `${textSize}px Roboto`)
-
- const iconSize = textSize
-
- const subTextSize =
- textSize * SUB_TEXT_SIZE_FACTOR > SUB_TEXT_SIZE_MAX_THRESHOLD
- ? SUB_TEXT_SIZE_MAX_THRESHOLD
- : textSize * SUB_TEXT_SIZE_FACTOR < SUB_TEXT_SIZE_MIN_THRESHOLD
- ? SUB_TEXT_SIZE_MIN_THRESHOLD
- : textSize * SUB_TEXT_SIZE_FACTOR
-
- const svgValue = document.createElementNS(svgNS, 'svg')
- svgValue.setAttribute('viewBox', `0 0 ${containerWidth} ${containerHeight}`)
- svgValue.setAttribute('width', '50%')
- svgValue.setAttribute('height', '50%')
- svgValue.setAttribute('x', '50%')
- svgValue.setAttribute('y', '50%')
- svgValue.setAttribute('style', 'overflow: visible')
-
- let fillColor = colors.grey900
-
- if (valueColor) {
- fillColor = valueColor
- } else if (formattedValue === noData.text) {
- fillColor = colors.grey600
- }
-
- // show icon if configured in maintenance app
- if (showIcon) {
- // embed icon to allow changing color
- // (elements with fill need to use "currentColor" for this to work)
- const iconSvgNode = document.createElementNS(svgNS, 'svg')
- iconSvgNode.setAttribute('viewBox', '0 0 48 48')
- iconSvgNode.setAttribute('width', iconSize)
- iconSvgNode.setAttribute('height', iconSize)
- iconSvgNode.setAttribute('y', (iconSize / 2 - topMargin / 2) * -1)
- iconSvgNode.setAttribute(
- 'x',
- `-${(iconSize + getIconPadding(textSize) + textWidth) / 2}`
- )
- iconSvgNode.setAttribute('style', `color: ${fillColor}`)
- iconSvgNode.setAttribute('data-test', 'visualization-icon')
-
- const parser = new DOMParser()
- const svgIconDocument = parser.parseFromString(icon, 'image/svg+xml')
-
- Array.from(svgIconDocument.documentElement.children).forEach((node) =>
- iconSvgNode.appendChild(node)
- )
-
- svgValue.appendChild(iconSvgNode)
- }
-
- const letterSpacing = Math.round(textSize * LETTER_SPACING_TEXT_SIZE_FACTOR)
-
- const textNode = document.createElementNS(svgNS, 'text')
- textNode.setAttribute('font-size', textSize)
- textNode.setAttribute('font-weight', '300')
- textNode.setAttribute(
- 'letter-spacing',
- letterSpacing < LETTER_SPACING_MIN_THRESHOLD
- ? LETTER_SPACING_MIN_THRESHOLD
- : letterSpacing > LETTER_SPACING_MAX_THRESHOLD
- ? LETTER_SPACING_MAX_THRESHOLD
- : letterSpacing
- )
- textNode.setAttribute('text-anchor', 'middle')
- textNode.setAttribute(
- 'x',
- showIcon ? `${(iconSize + getIconPadding(textSize)) / 2}` : 0
- )
- textNode.setAttribute(
- 'y',
- topMargin / 2 + getTextHeightForNumbers(textSize) / 2
- )
- textNode.setAttribute('fill', fillColor)
- textNode.setAttribute('data-test', 'visualization-primary-value')
-
- textNode.appendChild(document.createTextNode(formattedValue))
-
- svgValue.appendChild(textNode)
-
- if (subText) {
- const subTextNode = document.createElementNS(svgNS, 'text')
- subTextNode.setAttribute('text-anchor', 'middle')
- subTextNode.setAttribute('font-size', subTextSize)
- subTextNode.setAttribute('y', iconSize / 2 + topMargin / 2)
- subTextNode.setAttribute('dy', subTextSize * 1.7)
- subTextNode.setAttribute('fill', textColor)
- subTextNode.appendChild(document.createTextNode(subText))
-
- svgValue.appendChild(subTextNode)
- }
-
- return svgValue
-}
-
-const generateDashboardItem = (
- config,
- {
- svgContainer,
- width,
- height,
- valueColor,
- titleColor,
- backgroundColor,
- noData,
- icon,
- }
-) => {
- svgContainer.appendChild(
- generateValueSVG({
- formattedValue: config.formattedValue,
- subText: config.subText,
- valueColor,
- textColor: titleColor,
- noData,
- icon,
- containerWidth: width,
- containerHeight: height,
- })
- )
-
- const container = document.createElement('div')
- container.setAttribute(
- 'style',
- `display: flex; flex-direction: column; align-items: center; justify-content: center; width: 100%; height: 100%; padding-top: 8px; ${
- backgroundColor ? `background-color:${backgroundColor};` : ''
- }`
- )
-
- const titleStyle = `padding: 0 8px; text-align: center; font-size: 12px; color: ${
- titleColor || '#666'
- };`
-
- const title = document.createElement('span')
- title.setAttribute('style', titleStyle)
- if (config.title) {
- title.appendChild(document.createTextNode(config.title))
-
- container.appendChild(title)
- }
-
- if (config.subtitle) {
- const subtitle = document.createElement('span')
- subtitle.setAttribute('style', titleStyle + ' margin-top: 4px;')
-
- subtitle.appendChild(document.createTextNode(config.subtitle))
-
- container.appendChild(subtitle)
- }
-
- container.appendChild(svgContainer)
-
- return container
-}
-
-const getTextAnchorFromTextAlign = (textAlign) => {
- switch (textAlign) {
- default:
- case TEXT_ALIGN_LEFT:
- return 'start'
- case TEXT_ALIGN_CENTER:
- return 'middle'
- case TEXT_ALIGN_RIGHT:
- return 'end'
- }
-}
-
-const getXFromTextAlign = (textAlign) => {
- switch (textAlign) {
- default:
- case TEXT_ALIGN_LEFT:
- return '1%'
- case TEXT_ALIGN_CENTER:
- return '50%'
- case TEXT_ALIGN_RIGHT:
- return '99%'
- }
-}
-
-const generateDVItem = (
- config,
- {
- svgContainer,
- width,
- height,
- valueColor,
- noData,
- backgroundColor,
- titleColor,
- fontStyle,
- icon,
- }
-) => {
- if (backgroundColor) {
- svgContainer.setAttribute(
- 'style',
- `background-color: ${backgroundColor};`
- )
-
- const background = document.createElementNS(svgNS, 'rect')
- background.setAttribute('width', '100%')
- background.setAttribute('height', '100%')
- background.setAttribute('fill', backgroundColor)
- svgContainer.appendChild(background)
- }
-
- const svgWrapper = document.createElementNS(svgNS, 'svg')
-
- // title
- const title = document.createElementNS(svgNS, 'text')
-
- const titleFontStyle = mergeFontStyleWithDefault(
- fontStyle && fontStyle[FONT_STYLE_VISUALIZATION_TITLE],
- FONT_STYLE_VISUALIZATION_TITLE
- )
-
- const titleYPosition =
- TOP_MARGIN_FIXED +
- parseInt(titleFontStyle[FONT_STYLE_OPTION_FONT_SIZE]) +
- 'px'
-
- const titleAttributes = {
- x: getXFromTextAlign(titleFontStyle[FONT_STYLE_OPTION_TEXT_ALIGN]),
- y: titleYPosition,
- 'text-anchor': getTextAnchorFromTextAlign(
- titleFontStyle[FONT_STYLE_OPTION_TEXT_ALIGN]
- ),
- 'font-size': `${titleFontStyle[FONT_STYLE_OPTION_FONT_SIZE]}px`,
- 'font-weight': titleFontStyle[FONT_STYLE_OPTION_BOLD]
- ? FONT_STYLE_OPTION_BOLD
- : 'normal',
- 'font-style': titleFontStyle[FONT_STYLE_OPTION_ITALIC]
- ? FONT_STYLE_OPTION_ITALIC
- : 'normal',
- 'data-test': 'visualization-title',
- fill:
- titleColor &&
- titleFontStyle[FONT_STYLE_OPTION_TEXT_COLOR] ===
- defaultFontStyle[FONT_STYLE_VISUALIZATION_TITLE][
- FONT_STYLE_OPTION_TEXT_COLOR
- ]
- ? titleColor
- : titleFontStyle[FONT_STYLE_OPTION_TEXT_COLOR],
- }
-
- Object.entries(titleAttributes).forEach(([key, value]) =>
- title.setAttribute(key, value)
- )
-
- if (config.title) {
- title.appendChild(document.createTextNode(config.title))
- svgWrapper.appendChild(title)
- }
-
- // subtitle
- const subtitle = document.createElementNS(svgNS, 'text')
-
- const subtitleFontStyle = mergeFontStyleWithDefault(
- fontStyle && fontStyle[FONT_STYLE_VISUALIZATION_SUBTITLE],
- FONT_STYLE_VISUALIZATION_SUBTITLE
- )
-
- const subtitleAttributes = {
- x: getXFromTextAlign(subtitleFontStyle[FONT_STYLE_OPTION_TEXT_ALIGN]),
- y: titleYPosition,
- dy: `${subtitleFontStyle[FONT_STYLE_OPTION_FONT_SIZE] + 10}`,
- 'text-anchor': getTextAnchorFromTextAlign(
- subtitleFontStyle[FONT_STYLE_OPTION_TEXT_ALIGN]
- ),
- 'font-size': `${subtitleFontStyle[FONT_STYLE_OPTION_FONT_SIZE]}px`,
- 'font-weight': subtitleFontStyle[FONT_STYLE_OPTION_BOLD]
- ? FONT_STYLE_OPTION_BOLD
- : 'normal',
- 'font-style': subtitleFontStyle[FONT_STYLE_OPTION_ITALIC]
- ? FONT_STYLE_OPTION_ITALIC
- : 'normal',
- fill:
- titleColor &&
- subtitleFontStyle[FONT_STYLE_OPTION_TEXT_COLOR] ===
- defaultFontStyle[FONT_STYLE_VISUALIZATION_SUBTITLE][
- FONT_STYLE_OPTION_TEXT_COLOR
- ]
- ? titleColor
- : subtitleFontStyle[FONT_STYLE_OPTION_TEXT_COLOR],
- 'data-test': 'visualization-subtitle',
- }
-
- Object.entries(subtitleAttributes).forEach(([key, value]) =>
- subtitle.setAttribute(key, value)
- )
-
- if (config.subtitle) {
- subtitle.appendChild(document.createTextNode(config.subtitle))
- svgWrapper.appendChild(subtitle)
- }
-
- svgContainer.appendChild(svgWrapper)
-
- svgContainer.appendChild(
- generateValueSVG({
- formattedValue: config.formattedValue,
- subText: config.subText,
- valueColor,
- textColor: titleColor,
- noData,
- icon,
- containerWidth: width,
- containerHeight: height,
- topMargin:
- TOP_MARGIN_FIXED +
- ((config.title
- ? parseInt(title.getAttribute('font-size'))
- : 0) +
- (config.subtitle
- ? parseInt(subtitle.getAttribute('font-size'))
- : 0)) *
- 2.5,
- })
- )
-
- return svgContainer
-}
-
-const shouldUseContrastColor = (inputColor = '') => {
- // based on https://stackoverflow.com/questions/3942878/how-to-decide-font-color-in-white-or-black-depending-on-background-color
- var color =
- inputColor.charAt(0) === '#' ? inputColor.substring(1, 7) : inputColor
- var r = parseInt(color.substring(0, 2), 16) // hexToR
- var g = parseInt(color.substring(2, 4), 16) // hexToG
- var b = parseInt(color.substring(4, 6), 16) // hexToB
- var uicolors = [r / 255, g / 255, b / 255]
- var c = uicolors.map((col) => {
- if (col <= 0.03928) {
- return col / 12.92
- }
- return Math.pow((col + 0.055) / 1.055, 2.4)
- })
- var L = 0.2126 * c[0] + 0.7152 * c[1] + 0.0722 * c[2]
- return L <= 0.179
-}
-
-export default function (
- config,
- parentEl,
- { dashboard, legendSets, fontStyle, noData, legendOptions, icon }
-) {
- const legendSet = legendOptions && legendSets[0]
- const legendColor =
- legendSet && getColorByValueFromLegendSet(legendSet, config.value)
- let valueColor, titleColor, backgroundColor
- if (legendColor) {
- if (legendOptions.style === LEGEND_DISPLAY_STYLE_FILL) {
- backgroundColor = legendColor
- valueColor = titleColor =
- shouldUseContrastColor(legendColor) && colors.white
- } else {
- valueColor = legendColor
- }
- }
-
- parentEl.style.overflow = 'hidden'
- parentEl.style.display = 'flex'
- parentEl.style.justifyContent = 'center'
-
- const parentElBBox = parentEl.getBoundingClientRect()
- const width = parentElBBox.width
- const height = parentElBBox.height
-
- const svgContainer = document.createElementNS(svgNS, 'svg')
- svgContainer.setAttribute('xmlns', svgNS)
- svgContainer.setAttribute('viewBox', `0 0 ${width} ${height}`)
- svgContainer.setAttribute('width', dashboard ? '100%' : width)
- svgContainer.setAttribute('height', dashboard ? '100%' : height)
- svgContainer.setAttribute('data-test', 'visualization-container')
-
- if (dashboard) {
- parentEl.style.borderRadius = '3px'
-
- return generateDashboardItem(config, {
- svgContainer,
- width,
- height,
- valueColor,
- backgroundColor,
- noData,
- icon,
- ...(legendOptions.style === LEGEND_DISPLAY_STYLE_FILL &&
- legendColor &&
- shouldUseContrastColor(legendColor)
- ? { titleColor: colors.white }
- : {}),
- })
- } else {
- parentEl.style.height = `100%`
-
- return generateDVItem(config, {
- svgContainer,
- width,
- height,
- valueColor,
- backgroundColor,
- titleColor,
- noData,
- icon,
- fontStyle,
- })
- }
-}
diff --git a/src/visualizations/config/generators/highcharts/index.js b/src/visualizations/config/generators/highcharts/index.js
index 92a775910..3620e81f5 100644
--- a/src/visualizations/config/generators/highcharts/index.js
+++ b/src/visualizations/config/generators/highcharts/index.js
@@ -3,16 +3,24 @@ import HM from 'highcharts/highcharts-more'
import HB from 'highcharts/modules/boost'
import HE from 'highcharts/modules/exporting'
import HNDTD from 'highcharts/modules/no-data-to-display'
+import HOE from 'highcharts/modules/offline-exporting'
import HPF from 'highcharts/modules/pattern-fill'
import HSG from 'highcharts/modules/solid-gauge'
+import PEBFP from './pdfExportBugFixPlugin/index.js'
// apply
HM(H)
HSG(H)
HNDTD(H)
HE(H)
+HOE(H)
HPF(H)
HB(H)
+PEBFP(H)
+
+/* Whitelist some additional SVG attributes here. Without this,
+ * the PDF export for the SingleValue visualization breaks. */
+H.AST.allowedAttributes.push('fill-rule', 'clip-rule')
function drawLegendSymbolWrap() {
const pick = H.pick
@@ -75,7 +83,6 @@ export default function (config, el) {
// silence warning about accessibility
config.accessibility = { enabled: false }
-
if (config.lang) {
H.setOptions({
lang: config.lang,
diff --git a/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/index.js b/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/index.js
new file mode 100644
index 000000000..7b4899cde
--- /dev/null
+++ b/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/index.js
@@ -0,0 +1,7 @@
+import nonASCIIFontBugfix from './nonASCIIFont.js'
+import textShadowBugFix from './textShadow.js'
+
+export default function (H) {
+ textShadowBugFix(H)
+ nonASCIIFontBugfix(H)
+}
diff --git a/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/nonASCIIFont.js b/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/nonASCIIFont.js
new file mode 100644
index 000000000..d2c8d9835
--- /dev/null
+++ b/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/nonASCIIFont.js
@@ -0,0 +1,9 @@
+/* This is a workaround for https://github.com/highcharts/highcharts/issues/22008
+ * We add some transparent text in a non-ASCII script to the chart to prevent
+ * the chart from being exported in a serif font */
+
+export default function (H) {
+ H.addEvent(H.Chart, 'load', function () {
+ this.renderer.text('모', 20, 20).attr({ opacity: 0 }).add()
+ })
+}
diff --git a/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/textShadow.js b/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/textShadow.js
new file mode 100644
index 000000000..21a96e1a5
--- /dev/null
+++ b/src/visualizations/config/generators/highcharts/pdfExportBugFixPlugin/textShadow.js
@@ -0,0 +1,308 @@
+/* This plugin was provided by HighCharts support and resolves an issue with label
+ * text that has a white outline, such as the one we use for stacked bar charts.
+ * For example: "ANC: 1-4 visits by districts this year (stacked)"
+ * This issue has actually been resolved in HighCharts v11, so once we have upgraded
+ * to that version, this plugin can be removed. */
+
+export default function (H) {
+ const { AST, defaultOptions, downloadURL } = H,
+ { ajax } = H.HttpUtilities,
+ doc = document,
+ win = window,
+ OfflineExporting =
+ H._modules['Extensions/OfflineExporting/OfflineExporting.js'],
+ { getScript, svgToPdf, imageToDataUrl, svgToDataUrl } = OfflineExporting
+
+ H.wrap(
+ OfflineExporting,
+ 'downloadSVGLocal',
+ function (proceed, svg, options, failCallback, successCallback) {
+ var dummySVGContainer = doc.createElement('div'),
+ imageType = options.type || 'image/png',
+ filename =
+ (options.filename || 'chart') +
+ '.' +
+ (imageType === 'image/svg+xml'
+ ? 'svg'
+ : imageType.split('/')[1]),
+ scale = options.scale || 1
+ var svgurl,
+ blob,
+ finallyHandler,
+ libURL = options.libURL || defaultOptions.exporting.libURL,
+ objectURLRevoke = true,
+ pdfFont = options.pdfFont
+ // Allow libURL to end with or without fordward slash
+ libURL = libURL.slice(-1) !== '/' ? libURL + '/' : libURL
+ /*
+ * Detect if we need to load TTF fonts for the PDF, then load them and
+ * proceed.
+ *
+ * @private
+ */
+ var loadPdfFonts = function (svgElement, callback) {
+ var hasNonASCII = function (s) {
+ return (
+ // eslint-disable-next-line no-control-regex
+ /[^\u0000-\u007F\u200B]+/.test(s)
+ )
+ }
+ // Register an event in order to add the font once jsPDF is
+ // initialized
+ var addFont = function (variant, base64) {
+ win.jspdf.jsPDF.API.events.push([
+ 'initialized',
+ function () {
+ this.addFileToVFS(variant, base64)
+ this.addFont(variant, 'HighchartsFont', variant)
+ if (!this.getFontList().HighchartsFont) {
+ this.setFont('HighchartsFont')
+ }
+ },
+ ])
+ }
+ // If there are no non-ASCII characters in the SVG, do not use
+ // bother downloading the font files
+ if (pdfFont && !hasNonASCII(svgElement.textContent || '')) {
+ pdfFont = void 0
+ }
+ // Add new font if the URL is declared, #6417.
+ var variants = ['normal', 'italic', 'bold', 'bolditalic']
+ // Shift the first element off the variants and add as a font.
+ // Then asynchronously trigger the next variant until calling the
+ // callback when the variants are empty.
+ var normalBase64
+ var shiftAndLoadVariant = function () {
+ var variant = variants.shift()
+ // All variants shifted and possibly loaded, proceed
+ if (!variant) {
+ return callback()
+ }
+ var url = pdfFont && pdfFont[variant]
+ if (url) {
+ ajax({
+ url: url,
+ responseType: 'blob',
+ success: function (data, xhr) {
+ var reader = new FileReader()
+ reader.onloadend = function () {
+ if (typeof this.result === 'string') {
+ var base64 = this.result.split(',')[1]
+ addFont(variant, base64)
+ if (variant === 'normal') {
+ normalBase64 = base64
+ }
+ }
+ shiftAndLoadVariant()
+ }
+ reader.readAsDataURL(xhr.response)
+ },
+ error: shiftAndLoadVariant,
+ })
+ } else {
+ // For other variants, fall back to normal text weight/style
+ if (normalBase64) {
+ addFont(variant, normalBase64)
+ }
+ shiftAndLoadVariant()
+ }
+ }
+ shiftAndLoadVariant()
+ }
+ /*
+ * @private
+ */
+ var downloadPDF = function () {
+ AST.setElementHTML(dummySVGContainer, svg)
+ var textElements =
+ dummySVGContainer.getElementsByTagName('text'),
+ // Copy style property to element from parents if it's not
+ // there. Searches up hierarchy until it finds prop, or hits the
+ // chart container.
+ setStylePropertyFromParents = function (el, propName) {
+ var curParent = el
+ while (curParent && curParent !== dummySVGContainer) {
+ if (curParent.style[propName]) {
+ el.style[propName] = curParent.style[propName]
+ break
+ }
+ curParent = curParent.parentNode
+ }
+ }
+ var titleElements,
+ outlineElements
+ // Workaround for the text styling. Making sure it does pick up
+ // settings for parent elements.
+ ;[].forEach.call(textElements, function (el) {
+ // Workaround for the text styling. making sure it does pick up
+ // the root element
+ ;['font-family', 'font-size'].forEach(function (property) {
+ setStylePropertyFromParents(el, property)
+ })
+ el.style.fontFamily =
+ pdfFont && pdfFont.normal
+ ? // Custom PDF font
+ 'HighchartsFont'
+ : // Generic font (serif, sans-serif etc)
+ String(
+ el.style.fontFamily &&
+ el.style.fontFamily.split(' ').splice(-1)
+ )
+ // Workaround for plotband with width, removing title from text
+ // nodes
+ titleElements = el.getElementsByTagName('title')
+ ;[].forEach.call(titleElements, function (titleElement) {
+ el.removeChild(titleElement)
+ })
+
+ // Remove all .highcharts-text-outline elements, #17170
+ outlineElements = el.getElementsByClassName(
+ 'highcharts-text-outline'
+ )
+ while (outlineElements.length > 0) {
+ const outline = outlineElements[0]
+ if (outline.parentNode) {
+ outline.parentNode.removeChild(outline)
+ }
+ }
+ })
+ var svgNode = dummySVGContainer.querySelector('svg')
+ if (svgNode) {
+ loadPdfFonts(svgNode, function () {
+ svgToPdf(svgNode, 0, function (pdfData) {
+ try {
+ downloadURL(pdfData, filename)
+ if (successCallback) {
+ successCallback()
+ }
+ } catch (e) {
+ failCallback(e)
+ }
+ })
+ })
+ }
+ }
+ // Initiate download depending on file type
+ if (imageType === 'image/svg+xml') {
+ // SVG download. In this case, we want to use Microsoft specific
+ // Blob if available
+ try {
+ if (typeof win.navigator.msSaveOrOpenBlob !== 'undefined') {
+ // eslint-disable-next-line no-undef
+ blob = new MSBlobBuilder()
+ blob.append(svg)
+ svgurl = blob.getBlob('image/svg+xml')
+ } else {
+ svgurl = svgToDataUrl(svg)
+ }
+ downloadURL(svgurl, filename)
+ if (successCallback) {
+ successCallback()
+ }
+ } catch (e) {
+ failCallback(e)
+ }
+ } else if (imageType === 'application/pdf') {
+ if (win.jspdf && win.jspdf.jsPDF) {
+ downloadPDF()
+ } else {
+ // Must load pdf libraries first. // Don't destroy the object
+ // URL yet since we are doing things asynchronously. A cleaner
+ // solution would be nice, but this will do for now.
+ objectURLRevoke = true
+ getScript(libURL + 'jspdf.js', function () {
+ getScript(libURL + 'svg2pdf.js', downloadPDF)
+ })
+ }
+ } else {
+ // PNG/JPEG download - create bitmap from SVG
+ svgurl = svgToDataUrl(svg)
+ finallyHandler = function () {
+ try {
+ OfflineExporting.domurl.revokeObjectURL(svgurl)
+ } catch (e) {
+ // Ignore
+ }
+ }
+ // First, try to get PNG by rendering on canvas
+ imageToDataUrl(
+ svgurl,
+ imageType,
+ {},
+ scale,
+ function (imageURL) {
+ // Success
+ try {
+ downloadURL(imageURL, filename)
+ if (successCallback) {
+ successCallback()
+ }
+ } catch (e) {
+ failCallback(e)
+ }
+ },
+ function () {
+ // Failed due to tainted canvas
+ // Create new and untainted canvas
+ var canvas = doc.createElement('canvas'),
+ ctx = canvas.getContext('2d'),
+ imageWidth =
+ svg.match(
+ // eslint-disable-next-line no-useless-escape
+ /^