diff --git a/client/actions/assignments/ui.ts b/client/actions/assignments/ui.ts index aa4b841f6..d2a5ed2e6 100644 --- a/client/actions/assignments/ui.ts +++ b/client/actions/assignments/ui.ts @@ -24,6 +24,7 @@ const loadAssignments = ({ filterBy = 'Desk', searchQuery = null, orderByField = 'Scheduled', + dayField = null, filterByType = null, filterByPriority = null, selectedDeskId = null, @@ -34,6 +35,7 @@ const loadAssignments = ({ filterBy, searchQuery, orderByField, + dayField, filterByType, filterByPriority, selectedDeskId, @@ -304,6 +306,7 @@ const changeListSettings = ({ filterBy = 'Desk', searchQuery = null, orderByField = 'Scheduled', + dayField = null, filterByType = null, filterByPriority = null, selectedDeskId = null, @@ -314,6 +317,7 @@ const changeListSettings = ({ filterBy, searchQuery, orderByField, + dayField, filterByType, filterByPriority, selectedDeskId, @@ -805,6 +809,27 @@ const changeSortField = (field, savePreference = true) => ( } ); +/** + * Action dispatcher to set the day field filter for all lists + * @param {String} value - the value to set the field to + */ +const setDayField = (value) => ({ + type: ASSIGNMENTS.ACTIONS.SET_DAY_FIELD, + payload: value, +}); + +const changeDayField = (value, savePreference = true) => ( + (dispatch) => { + dispatch(self.setDayField(value)); + + // if (savePreference) { + // dispatch(actions.users.setAssignmentSortField()); + // } + + return dispatch(self.reloadAssignments(null, false)); + } +); + /** * Action dispatcher to load the current users' preferred sort field and list orders * (This assumes the users' preferences have already been loaded into redux) @@ -897,6 +922,8 @@ const self = { setListSortOrder, changeListSortOrder, setSortField, + setDayField, + changeDayField, loadDefaultListSort, changeSortField, validateStartWorkingOnScheduledUpdate, diff --git a/client/apps/Assignments/AssignmentsSubNav.tsx b/client/apps/Assignments/AssignmentsSubNav.tsx index 1ec20a61b..d2ba7b667 100644 --- a/client/apps/Assignments/AssignmentsSubNav.tsx +++ b/client/apps/Assignments/AssignmentsSubNav.tsx @@ -1,5 +1,4 @@ import React from 'react'; -import PropTypes from 'prop-types'; import {connect} from 'react-redux'; import * as selectors from '../../selectors'; @@ -9,7 +8,46 @@ import {WORKSPACE, ASSIGNMENTS} from '../../constants'; import {SubNavBar, FiltersBar} from '../../components/Assignments'; import {ArchiveItem} from '../../components/Archive'; -export class AssignmentsSubNavComponent extends React.Component { +interface IReduxStateProps { + filterBy?: string; + selectedDeskId?: string; + currentDeskId?: string; + myAssignmentsCount?: number; + orderByField?: string; + dayField?: string; + searchQuery?: string; + assignmentListSingleGroupView?: string; + filterByType?: string; + filterByPriority?: string; + assignmentsInTodoCount?: number; + assignmentsInInProgressCount?: number; + assignmentsInCompletedCount?: number; + assignmentCounts?: any; + userDesks?: Array; + workspace?: string; + listGroups?: Array; + privileges?: any; +} + +interface IReduxDispatchProps { + fetchMyAssignmentsCount?: () => any; + changeAssignmentListSingleGroupView?: (groupKey: string) => any; + loadAssignments: (filters: any) => any; + changeSortField?: (field: string, saveSortPreferences?: boolean) => any; + changeDayField?: (value: string) => any; +} + +interface IOwnProps { + archiveItem?: any; + withArchiveItem?: boolean; + showAllDeskOption?: boolean; + saveSortPreferences?: boolean; + ignoreScheduledUpdates?: boolean; +} + +type IProps = IOwnProps & IReduxStateProps & IReduxDispatchProps; + +export class AssignmentsSubNavComponent extends React.Component { constructor(props) { super(props); @@ -40,6 +78,7 @@ export class AssignmentsSubNavComponent extends React.Component { const { filterBy, orderByField, + dayField, loadAssignments, filterByType, filterByPriority, @@ -51,6 +90,7 @@ export class AssignmentsSubNavComponent extends React.Component { filterBy, searchQuery, orderByField, + dayField, filterByType, filterByPriority, selectedDeskId, @@ -68,12 +108,14 @@ export class AssignmentsSubNavComponent extends React.Component { filterByType, filterByPriority, ignoreScheduledUpdates, + dayField } = this.props; loadAssignments({ filterBy, searchQuery, orderByField, + dayField, filterByType, filterByPriority, selectedDeskId, @@ -82,7 +124,7 @@ export class AssignmentsSubNavComponent extends React.Component { } changeSortField(field) { - const {changeSortField, saveSortPreferences} = this.props; + const {changeSortField, saveSortPreferences = true} = this.props; changeSortField(field, saveSortPreferences); } @@ -102,6 +144,7 @@ export class AssignmentsSubNavComponent extends React.Component { filterBy, myAssignmentsCount, orderByField, + dayField, searchQuery, assignmentListSingleGroupView, changeAssignmentListSingleGroupView, @@ -111,8 +154,9 @@ export class AssignmentsSubNavComponent extends React.Component { workspace, userDesks, currentDeskId, - showAllDeskOption, + showAllDeskOption = false, privileges, + changeDayField, } = this.props; // Show the Desk selection if we're in Fulfil Assignment or Custom Workspace @@ -137,6 +181,8 @@ export class AssignmentsSubNavComponent extends React.Component { myAssignmentsCount={myAssignmentsCount} orderByField={orderByField} changeFilter={this.changeFilter} + dayField={dayField} + changeDayField={changeDayField} selectedDeskId={selectedDeskId} userDesks={showDeskSelection ? userDesks : []} selectAssignmentsFrom={this.selectAssignmentsFrom} @@ -150,46 +196,13 @@ export class AssignmentsSubNavComponent extends React.Component { } } -AssignmentsSubNavComponent.propTypes = { - filterBy: PropTypes.string, - selectedDeskId: PropTypes.string, - myAssignmentsCount: PropTypes.number, - orderByField: PropTypes.string, - fetchMyAssignmentsCount: PropTypes.func, - searchQuery: PropTypes.string, - assignmentListSingleGroupView: PropTypes.string, - changeAssignmentListSingleGroupView: PropTypes.func, - loadAssignments: PropTypes.func.isRequired, - filterByType: PropTypes.string, - filterByPriority: PropTypes.string, - assignmentsInTodoCount: PropTypes.number, - assignmentsInInProgressCount: PropTypes.number, - assignmentsInCompletedCount: PropTypes.number, - archiveItem: PropTypes.object, - withArchiveItem: PropTypes.bool, - userDesks: PropTypes.array, - workspace: PropTypes.string, - currentDeskId: PropTypes.string, - listGroups: PropTypes.array, - assignmentCounts: PropTypes.object, - showAllDeskOption: PropTypes.bool, - changeSortField: PropTypes.func, - saveSortPreferences: PropTypes.bool, - ignoreScheduledUpdates: PropTypes.bool, - privileges: PropTypes.object, -}; - -AssignmentsSubNavComponent.defaultProps = { - showAllDeskOption: false, - saveSortPreferences: true, -}; - const mapStateToProps = (state) => ({ filterBy: selectors.getFilterBy(state), selectedDeskId: selectors.getSelectedDeskId(state), currentDeskId: selectors.general.currentDeskId(state), myAssignmentsCount: selectors.getMyAssignmentsCount(state), orderByField: selectors.getOrderByField(state), + dayField: selectors.getDayField(state), searchQuery: selectors.getSearchQuery(state), assignmentListSingleGroupView: selectors.getAssignmentListSingleGroupView(state), @@ -222,12 +235,13 @@ const mapDispatchToProps = (dispatch) => ({ ) ), loadAssignments: (filters) => dispatch(actions.assignments.ui.loadAssignments(filters)), + changeDayField: (value) => dispatch(actions.assignments.ui.changeDayField(value)), changeSortField: (field, saveSortPreferences) => ( dispatch(actions.assignments.ui.changeSortField(field, saveSortPreferences)) ), }); -export const AssignmentsSubNav = connect( +export const AssignmentsSubNav = connect( mapStateToProps, mapDispatchToProps )(AssignmentsSubNavComponent); diff --git a/client/apps/Planning/PlanningList.tsx b/client/apps/Planning/PlanningList.tsx index 79b81fde0..3a2d7e932 100644 --- a/client/apps/Planning/PlanningList.tsx +++ b/client/apps/Planning/PlanningList.tsx @@ -28,7 +28,6 @@ import {ITEM_TYPE} from '../../constants'; import {ListPanel} from '../../components/Main'; import {PlanningListSubNav} from './PlanningListSubNav'; -import moment from 'moment'; interface IProps { groups: Array<{ @@ -186,13 +185,7 @@ export class PlanningListComponent extends React.PureComponent { { - const dateFilter = currentSearch.advancedSearch?.dates?.start ?? moment().date(); - - return groups.filter((group) => - moment(group.date).isSameOrAfter(dateFilter), - ); - })()} + groups={groups} onItemClick={openPreview} onDoubleClick={edit} agendas={agendas} diff --git a/client/components/Assignments/AssignmentGroupList.tsx b/client/components/Assignments/AssignmentGroupList.tsx index 3684fa8be..0ce6a7551 100644 --- a/client/components/Assignments/AssignmentGroupList.tsx +++ b/client/components/Assignments/AssignmentGroupList.tsx @@ -1,5 +1,4 @@ import React from 'react'; -import PropTypes from 'prop-types'; import {connect} from 'react-redux'; import {get, throttle} from 'lodash'; @@ -13,14 +12,58 @@ import {assignmentUtils} from '../../utils'; import {AssignmentItem} from './AssignmentItem'; import {Header, Group} from '../UI/List'; import {OrderDirectionIcon} from '../OrderBar'; -import {assignmentsViewRequiresArchiveItems} from './AssignmentItem/fields'; import {ListItemLoader} from 'superdesk-ui-framework/react/components/ListItemLoader'; +import moment from 'moment'; const focusElement = throttle((element: HTMLElement) => { element.focus(); }, 250, {leading: true}); -class AssignmentGroupListComponent extends React.Component { +interface IProps { + filterBy?: string; + orderByField?: string, + orderDirection?: string; + assignments: Array; + groupKey: string; + users?: Array; + session?: any; + loadMoreAssignments: (groupKey: string) => any; + lockedItems?: any; + currentAssignmentId?: string; + reassign?: () => any; + completeAssignment?: () => any; + editAssignmentPriority?: () => any; + hideItemActions?: boolean; + privileges?: any; + startWorking?: () => any; + totalCount?: number; + changeAssignmentListSingleGroupView?: (groupKey: string) => any; + assignmentListSingleGroupView?: string; + preview?: () => any; + priorities?: Array; + removeAssignment?: () => any; + openArchivePreview?: () => any; + revertAssignment?: () => any; + setMaxHeight?: boolean; + contentTypes?: Array; + desks?: Array; + groupLabel?: string; + groupStates?: Array; + groupEmptyMessage?: string; + showCount?: boolean; + changeListSortOrder?: (groupKey: string, order: any, savePreferences?: boolean) => any; + saveSortPreferences?: boolean; + contacts?: any; + isLoading?: boolean; + dayField?: string; +} + +interface IState { + isNextPageLoading: boolean; +} + +class AssignmentGroupListComponent extends React.Component { + dom: any; constructor(props) { super(props); this.state = {isNextPageLoading: false}; @@ -148,7 +191,7 @@ class AssignmentGroupListComponent extends React.Component { } changeListOrder(order) { - const {changeListSortOrder, groupKey, saveSortPreferences} = this.props; + const {changeListSortOrder, groupKey, saveSortPreferences = true} = this.props; changeListSortOrder(groupKey, order, saveSortPreferences); } @@ -205,7 +248,6 @@ class AssignmentGroupListComponent extends React.Component { contentTypes={contentTypes} assignedDesk={assignedDesk} contacts={contacts} - archiveItemForAssignment={this.props.archiveItemForAssignment} /> ); } @@ -214,18 +256,22 @@ class AssignmentGroupListComponent extends React.Component { const {gettext} = superdeskApi.localization; const { assignments, - totalCount, assignmentListSingleGroupView, - setMaxHeight, + setMaxHeight = true, groupLabel, groupEmptyMessage, - showCount, + showCount = true, changeAssignmentListSingleGroupView, orderDirection, isLoading, } = this.props; const listStyle = setMaxHeight ? {maxHeight: this.getListMaxHeight() + 'px'} : {}; const headingId = `heading--${this.props.groupKey}`; + const filteredAssignments = this.props.dayField == null + ? assignments + : assignments.filter((assignment) => + moment(assignment.planning.scheduled).isSameOrAfter(moment(this.props.dayField)), + ); return (
@@ -248,9 +294,9 @@ class AssignmentGroupListComponent extends React.Component {
{gettext( 'Number of Assignments: ', - {count: totalCount} + {count: (filteredAssignments?.length ?? 0)} )} - {totalCount} + {(filteredAssignments?.length ?? 0)}
)} @@ -283,8 +329,8 @@ class AssignmentGroupListComponent extends React.Component { )} {isLoading !== true && ( - get(assignments, 'length', 0) > 0 ? ( - assignments.map((assignment, index) => this.rowRenderer(index)) + (filteredAssignments?.length ?? 0) > 0 ? ( + filteredAssignments.map((_assignment, index) => this.rowRenderer(index)) ) : (
  • {groupEmptyMessage}
  • ) @@ -295,60 +341,15 @@ class AssignmentGroupListComponent extends React.Component { } } -AssignmentGroupListComponent.propTypes = { - filterBy: PropTypes.string, - orderByField: PropTypes.string, - orderDirection: PropTypes.string, - assignments: PropTypes.array.isRequired, - groupKey: PropTypes.string.isRequired, - users: PropTypes.array, - session: PropTypes.object, - loadMoreAssignments: PropTypes.func.isRequired, - lockedItems: PropTypes.object, - currentAssignmentId: PropTypes.string, - reassign: PropTypes.func, - completeAssignment: PropTypes.func, - editAssignmentPriority: PropTypes.func, - hideItemActions: PropTypes.bool, - privileges: PropTypes.object, - startWorking: PropTypes.func, - totalCount: PropTypes.number, - changeAssignmentListSingleGroupView: PropTypes.func, - assignmentListSingleGroupView: PropTypes.string, - preview: PropTypes.func, - priorities: PropTypes.array, - removeAssignment: PropTypes.func, - openArchivePreview: PropTypes.func, - revertAssignment: PropTypes.func, - setMaxHeight: PropTypes.bool, - contentTypes: PropTypes.array, - desks: PropTypes.array, - groupLabel: PropTypes.string, - groupStates: PropTypes.arrayOf(PropTypes.string), - groupEmptyMessage: PropTypes.string, - showCount: PropTypes.bool, - changeListSortOrder: PropTypes.func, - saveSortPreferences: PropTypes.bool, - contacts: PropTypes.object, - archiveItemForAssignment: PropTypes.object, - isLoading: PropTypes.bool, -}; - -AssignmentGroupListComponent.defaultProps = { - setMaxHeight: true, - showCount: true, - saveSortPreferences: true, -}; - const mapStateToProps = (state, ownProps) => { const assignmentDataSelector = selectors.getAssignmentGroupSelectors[ownProps.groupKey]; - const props = { + return { + dayField: selectors.getDayField(state), filterBy: selectors.getFilterBy(state), orderByField: selectors.getOrderByField(state), orderDirection: assignmentDataSelector.sortOrder(state), assignments: assignmentDataSelector.assignmentsSelector(state), - totalCount: assignmentDataSelector.countSelector(state), previewOpened: selectors.getPreviewAssignmentOpened(state), session: selectors.general.session(state), users: selectors.general.users(state), @@ -361,12 +362,6 @@ const mapStateToProps = (state, ownProps) => { contacts: selectors.general.contactsById(state), isLoading: assignmentDataSelector.isLoading(state), }; - - if (assignmentsViewRequiresArchiveItems()) { - props.archiveItemForAssignment = selectors.getStoredArchiveItems(state); - } - - return props; }; const mapDispatchToProps = (dispatch) => ({ diff --git a/client/components/Assignments/FiltersBar.tsx b/client/components/Assignments/FiltersBar.tsx index 09c37ece5..6d39f8304 100644 --- a/client/components/Assignments/FiltersBar.tsx +++ b/client/components/Assignments/FiltersBar.tsx @@ -1,28 +1,43 @@ import React, {Fragment} from 'react'; -import PropTypes from 'prop-types'; - import {superdeskApi} from '../../superdeskApi'; - -import {SubNav} from 'superdesk-ui-framework/react'; +import {DatePicker, SubNav, Tooltip} from 'superdesk-ui-framework/react'; import {StretchBar, Spacer} from '../UI/SubNav'; import {Checkbox} from '../UI/Form'; import {OrderFieldInput} from '../OrderBar'; import {DesksSubnavDropdown} from './DesksSubNavDropDown'; +import {appConfig} from 'superdesk-core/scripts/appConfig'; +interface IProps { + filterBy?: string; + myAssignmentsCount?: number; + orderByField?: string; + dayField?: string; + changeDayField: (value: string | null) => any; + changeFilter: (field: string, value: any, deskId: string) => void; + changeSortField: (...args: any) => any; + userDesks?: Array; + selectedDeskId?: string; + selectAssignmentsFrom?: (...args: any) => void; + showDeskSelection?: boolean; + showAllDeskOption?: boolean; + showDeskAssignmentView?: boolean; +} export const FiltersBar = ({ - filterBy, - orderByField, + filterBy = 'Desk', + orderByField = 'Updated', + dayField, changeSortField, changeFilter, - myAssignmentsCount, - userDesks, - selectedDeskId, + changeDayField, + myAssignmentsCount = 0, + userDesks = [], + selectedDeskId = '', selectAssignmentsFrom, - showDeskSelection, - showAllDeskOption, - showDeskAssignmentView, -}) => { + showDeskSelection = false, + showAllDeskOption = false, + showDeskAssignmentView = false, +}: IProps) => { const {gettext} = superdeskApi.localization; return ( @@ -51,9 +66,11 @@ export const FiltersBar = ({ labelPosition="inside" tabIndex={0} /> - - {myAssignmentsCount} - + + + {myAssignmentsCount} + +
    ) : ( @@ -66,9 +83,26 @@ export const FiltersBar = ({ /> )} - + { + if (val == null) { + changeDayField(null); + } else { + changeDayField(val.toString()); + } + }} + dateFormat={appConfig.view.dateformat} + data-test-id="date-input" + /> -
    ); }; - -FiltersBar.propTypes = { - filterBy: PropTypes.string, - myAssignmentsCount: PropTypes.number, - orderByField: PropTypes.string, - changeFilter: PropTypes.func.isRequired, - changeSortField: PropTypes.func.isRequired, - userDesks: PropTypes.array, - selectedDeskId: PropTypes.string, - selectAssignmentsFrom: PropTypes.func, - showDeskSelection: PropTypes.bool, - showAllDeskOption: PropTypes.bool, - showDeskAssignmentView: PropTypes.bool, -}; - -FiltersBar.defaultProps = { - filterBy: 'Desk', - myAssignmentsCount: 0, - orderByField: 'Updated', - userDesks: [], - selectedDeskId: '', - workspace: '', - showDeskSelection: false, - showAllDeskOption: false, - showDeskAssignmentView: false, -}; - diff --git a/client/components/Coverages/CoverageIcons.tsx b/client/components/Coverages/CoverageIcons.tsx index 08aa5ce11..a24bb7582 100644 --- a/client/components/Coverages/CoverageIcons.tsx +++ b/client/components/Coverages/CoverageIcons.tsx @@ -80,6 +80,7 @@ export function getAvatarForCoverage( displayName: user.display_name, icon: icon, customContent: getCustomAvatarContent(user), + statusDot: {color: planningUtils.getNewsCoverageStatusDotColor(coverage)}, }; return avatar; diff --git a/client/components/Main/ListPanel.tsx b/client/components/Main/ListPanel.tsx index d6c36a9b0..b53405661 100644 --- a/client/components/Main/ListPanel.tsx +++ b/client/components/Main/ListPanel.tsx @@ -422,18 +422,16 @@ export class ListPanel extends React.Component { /> ); })} - {!isAllListItemsLoaded && ( + {(!isAllListItemsLoaded && loadingIndicator) && (
    - - -
    - {gettext('loading more items...')} -
    -
    -
    + +
    + {gettext('loading more items...')} +
    +
    )} diff --git a/client/components/UI/List/Header.tsx b/client/components/UI/List/Header.tsx index 96d871ad0..e629c7983 100644 --- a/client/components/UI/List/Header.tsx +++ b/client/components/UI/List/Header.tsx @@ -1,13 +1,19 @@ import React from 'react'; -import PropTypes from 'prop-types'; import classNames from 'classnames'; +interface IProps { + title?: string; + id?: string; + marginTop?: boolean; + marginBottom?: boolean; + children?: React.ReactNode; +} /** * @ngdoc react * @name Header * @description Header Component of a list */ -export const Header = ({children, title, marginTop, marginBottom, id}) => ( +export const Header = ({children, title, marginTop, marginBottom, id}: IProps) => (
    ( {children}
    ); - -Header.propTypes = { - title: PropTypes.string, - id: PropTypes.string, - marginTop: PropTypes.bool, - marginBottom: PropTypes.bool, - children: PropTypes.node, -}; diff --git a/client/components/UI/SubNav/StretchBar.tsx b/client/components/UI/SubNav/StretchBar.tsx index 930125bd3..90d0b4cce 100644 --- a/client/components/UI/SubNav/StretchBar.tsx +++ b/client/components/UI/SubNav/StretchBar.tsx @@ -2,12 +2,17 @@ import React from 'react'; import PropTypes from 'prop-types'; import classNames from 'classnames'; +interface IProps { + children: JSX.Element | Array; + className?: string; + right?: boolean; +} /** * @ngdoc react * @name StretchBar * @description Stretch Bar of a Sub Nav bar */ -export const StretchBar = ({children, className, right}) => ( +export const StretchBar = ({children, className, right = false}: IProps) => (
    ( {children}
    ); - -StretchBar.propTypes = { - children: PropTypes.node, - className: PropTypes.string, - right: PropTypes.bool, -}; - -StretchBar.defaultProps = { - right: false, -}; diff --git a/client/constants/assignments.ts b/client/constants/assignments.ts index e12f0e19b..2876f61de 100644 --- a/client/constants/assignments.ts +++ b/client/constants/assignments.ts @@ -27,6 +27,7 @@ export const ASSIGNMENTS = { SET_GROUP_KEYS: 'SET_ASSIGNMENT_GROUP_KEYS', SET_GROUP_SORT_ORDER: 'SET_ASSIGNMENT_GROUP_SORT_ORDER', SET_SORT_FIELD: 'SET_ASSIGNMENT_SORT_FIELD', + SET_DAY_FIELD: 'SET_DAY_FIELD', SET_LOADING: 'SET_LOADING', }, WORKFLOW_STATE: { diff --git a/client/planning-extension/src/extension.ts b/client/planning-extension/src/extension.ts index 48e9632d4..141e04010 100644 --- a/client/planning-extension/src/extension.ts +++ b/client/planning-extension/src/extension.ts @@ -128,6 +128,7 @@ const extension: IExtension = { }, }, }, + notifications: {'email:notification:assignments': {type: 'email'}}, globalMenuHorizontal: displayTopbarWidget ? [AssignmentsList] : [], }, }; diff --git a/client/reducers/assignment.ts b/client/reducers/assignment.ts index b0c2f2ca6..d2c811abe 100644 --- a/client/reducers/assignment.ts +++ b/client/reducers/assignment.ts @@ -14,6 +14,7 @@ const initialState = { filterByType: null, myAssignmentsTotal: 0, orderByField: 'Scheduled', + dayField: null, previewOpened: false, readOnly: false, searchQuery: null, @@ -187,6 +188,11 @@ const assignmentReducer = createReducer(initialState, { orderByField: payload, }), + [ASSIGNMENTS.ACTIONS.SET_DAY_FIELD]: (state, payload) => ({ + ...state, + dayField: payload, + }), + [ASSIGNMENTS.ACTIONS.CHANGE_LIST_SETTINGS]: (state, payload) => ( { ...state, diff --git a/client/selectors/assignments.ts b/client/selectors/assignments.ts index a7af42e1f..fdd391721 100644 --- a/client/selectors/assignments.ts +++ b/client/selectors/assignments.ts @@ -122,6 +122,7 @@ export const getAssignmentGroupCounts = createSelector([ export const getFilterBy = (state) => get(state, 'assignment.filterBy', 'Desk'); export const getSearchQuery = (state) => get(state, 'assignment.searchQuery', null); export const getOrderByField = (state) => get(state, 'assignment.orderByField', 'Scheduled'); +export const getDayField = (state) => get(state, 'assignment.dayField', null); export const getOrderDirection = (state) => get(state, 'assignment.orderDirection', 'Asc'); export const getAssignmentFilterByState = (state) => get(state, 'assignment.filterByState', null); export const getAssignmentFilterByType = (state) => get(state, 'assignment.filterByType', null); @@ -134,6 +135,7 @@ export const getAssignmentListSettings = (state) => ({ filterBy: getFilterBy(state), searchQuery: getSearchQuery(state), orderByField: getOrderByField(state), + dayField: getDayField(state), orderDirection: getOrderDirection(state), filterByState: getAssignmentFilterByState(state), filterByType: getAssignmentFilterByType(state), @@ -262,4 +264,4 @@ export const getAssignmentGroupSelectors = { Object.keys(getAssignmentGroupSelectors).forEach((groupKey) => { getAssignmentGroupSelectors[groupKey].isLoading = (state) => getList(state, groupKey).isLoading; -}); \ No newline at end of file +}); diff --git a/client/selectors/events.ts b/client/selectors/events.ts index 0b848a8d5..7966a22de 100644 --- a/client/selectors/events.ts +++ b/client/selectors/events.ts @@ -2,7 +2,7 @@ import {createSelector} from 'reselect'; import {get, sortBy} from 'lodash'; import {appConfig} from 'appConfig'; -import {IEventItem, IEventState, IEventTemplate, IPlanningAppState, LIST_VIEW_TYPE} from '../interfaces'; +import {IEventItem, IEventState, IEventTemplate, IPlanningAppState, JUMP_INTERVAL, LIST_VIEW_TYPE} from '../interfaces'; import {currentPlanning, storedPlannings} from './planning'; import {agendas, userPreferences} from './general'; @@ -13,6 +13,11 @@ import {EVENTS, MAIN, SPIKED_STATE} from '../constants'; function getCurrentListViewType(state?: IPlanningAppState) { return state?.main?.listViewType ?? LIST_VIEW_TYPE.SCHEDULE; } + +function getCurrentViewInterval(state?: IPlanningAppState): JUMP_INTERVAL { + return state?.main?.search?.EVENTS?.jumpInterval ?? JUMP_INTERVAL.WEEK; +} + export const storedEvents = (state) => get(state, 'events.events', {}); export const eventIdsInList = (state) => get(state, 'events.eventsInList', []); export const eventHistory = (state) => get(state, 'events.eventHistoryItems'); @@ -36,8 +41,8 @@ export const eventsInList = createSelector( * the associated events. */ export const orderedEvents = createSelector( - [eventsInList, currentSearch, getCurrentListViewType], - (events, search, viewType) => { + [eventsInList, currentSearch, getCurrentListViewType, getCurrentViewInterval], + (events, search, viewType, viewInterval) => { if (!events?.length) { return []; } else if (viewType === LIST_VIEW_TYPE.LIST) { @@ -47,7 +52,7 @@ export const orderedEvents = createSelector( }]; } - const dateRange = getSearchDateRange(search, appConfig.start_of_week); + const dateRange = getSearchDateRange(search, appConfig.start_of_week, viewInterval); return eventUtils.getEventsByDate(events, dateRange.startDate, dateRange.endDate); } diff --git a/client/selectors/eventsplanning.ts b/client/selectors/eventsplanning.ts index 68e3d989b..04c834e09 100644 --- a/client/selectors/eventsplanning.ts +++ b/client/selectors/eventsplanning.ts @@ -2,7 +2,7 @@ import {createSelector} from 'reselect'; import {get, keyBy, sortBy, uniq} from 'lodash'; import {appConfig} from 'appConfig'; -import {IPlanningAppState, LIST_VIEW_TYPE} from '../interfaces'; +import {IPlanningAppState, JUMP_INTERVAL, LIST_VIEW_TYPE} from '../interfaces'; import {storedEvents} from './events'; import {storedPlannings} from './planning'; @@ -13,6 +13,11 @@ import {userPreferences} from './general'; function getCurrentListViewType(state?: IPlanningAppState) { return state?.main?.listViewType ?? LIST_VIEW_TYPE.SCHEDULE; } + +function getCurrentViewInterval(state?: IPlanningAppState): JUMP_INTERVAL { + return state?.main?.search?.COMBINED?.jumpInterval ?? JUMP_INTERVAL.WEEK; +} + export const getEventsPlanningList = (state) => get(state, 'eventsPlanning.eventsAndPlanningInList', []); export const getRelatedPlanningsList = (state) => get(state, 'eventsPlanning.relatedPlannings', {}); export const currentSearch = (state) => get(state, 'main.search.COMBINED.currentSearch'); @@ -44,8 +49,15 @@ export const eventsPlannignViewFilters = createSelector( * @type {Reselect.Selector} */ export const orderedEventsPlanning = createSelector( - [storedEvents, storedPlannings, getEventsPlanningList, currentSearch, getCurrentListViewType], - (events, plannings, eventPlanningList, search, viewType) => { + [ + storedEvents, + storedPlannings, + getEventsPlanningList, + currentSearch, + getCurrentListViewType, + getCurrentViewInterval, + ], + (events, plannings, eventPlanningList, search, viewType, viewInterval) => { if (!eventPlanningList?.length) { return []; } else if (viewType === LIST_VIEW_TYPE.LIST) { @@ -59,7 +71,7 @@ export const orderedEventsPlanning = createSelector( const eventsList = []; const planningList = []; - const dateRange = getSearchDateRange(search, appConfig.start_of_week); + const dateRange = getSearchDateRange(search, appConfig.start_of_week, viewInterval); eventPlanningList.forEach((_id) => { if (_id in events) { diff --git a/client/selectors/planning.ts b/client/selectors/planning.ts index ff7731ec3..96873a2b2 100644 --- a/client/selectors/planning.ts +++ b/client/selectors/planning.ts @@ -2,7 +2,7 @@ import {createSelector} from 'reselect'; import {cloneDeep, get} from 'lodash'; import {appConfig} from 'appConfig'; -import {IPlanningAppState, IPlanningItem, LIST_VIEW_TYPE} from '../interfaces'; +import {IPlanningAppState, IPlanningItem, JUMP_INTERVAL, LIST_VIEW_TYPE} from '../interfaces'; import {session, userPreferences} from './general'; import {getSearchDateRange, lockUtils, planningUtils} from '../utils'; @@ -11,6 +11,11 @@ import {AGENDA, SPIKED_STATE} from '../constants'; function getCurrentListViewType(state?: IPlanningAppState) { return state?.main?.listViewType ?? LIST_VIEW_TYPE.SCHEDULE; } + +function getCurrentViewInterval(state?: IPlanningAppState): JUMP_INTERVAL { + return state?.main?.search?.PLANNING?.jumpInterval ?? JUMP_INTERVAL.WEEK; +} + const storedEvents = (state) => get(state, 'events.events', {}); export const planningHistory = (state) => get(state, 'planning.planningHistoryItems'); @@ -71,8 +76,15 @@ export const plansInList = createSelector( ); export const orderedPlanningList = createSelector( - [currentAgenda, plansInList, storedEvents, currentSearch, getCurrentListViewType], - (currentAgenda, plansInList, events, search, viewType) => { + [ + currentAgenda, + plansInList, + storedEvents, + currentSearch, + getCurrentListViewType, + getCurrentViewInterval, + ], + (currentAgenda, plansInList, events, search, viewType, viewInterval) => { if (!plansInList?.length) { return []; } if (viewType === LIST_VIEW_TYPE.LIST) { @@ -82,7 +94,7 @@ export const orderedPlanningList = createSelector( }]; } - const dateRange = getSearchDateRange(search, appConfig.start_of_week); + const dateRange = getSearchDateRange(search, appConfig.start_of_week, viewInterval); return planningUtils.getPlanningByDate( plansInList, events, dateRange.startDate, dateRange.endDate diff --git a/client/selectors/tests/assignments_test.ts b/client/selectors/tests/assignments_test.ts index 8a5cf3d72..49c3b747a 100644 --- a/client/selectors/tests/assignments_test.ts +++ b/client/selectors/tests/assignments_test.ts @@ -138,6 +138,7 @@ describe('selectors', () => { filterBy: 'Desk', searchQuery: 'test', orderByField: 'Updated', + dayField: null, orderDirection: 'Desc', filterByState: null, filterByType: null, @@ -217,6 +218,7 @@ describe('selectors', () => { filterBy: 'Desk', searchQuery: null, orderByField: 'Scheduled', + dayField: null, orderDirection: 'Asc', filterByState: null, filterByType: null, diff --git a/client/selectors/tests/events_test.ts b/client/selectors/tests/events_test.ts index cfdd99af2..29a4352db 100644 --- a/client/selectors/tests/events_test.ts +++ b/client/selectors/tests/events_test.ts @@ -2,6 +2,7 @@ import * as selectors from '../index'; import moment from 'moment'; import {keyBy} from 'lodash'; +const ids = (day) => day.events.map((e) => e._id); describe('selectors', () => { const dateFormat = 'YYYY-MM-DDTHH:mm:ss'; @@ -32,13 +33,31 @@ describe('selectors', () => { end: moment('2017-01-17T14:00:00', dateFormat), }, }, + event4: { + _id: 'event4', + name: 'event4', + dates: { + start: moment('2024-07-16T13:00:00+0000'), + end: moment('2024-07-17T00:00:00+0000'), + no_end_time: true, + }, + }, + event5: { + _id: 'event5', + name: 'event5', + dates: { + start: moment('2024-07-15T00:00:00+0000'), + end: moment('2024-07-18T00:00:00+0000'), + all_day: true, + }, + } }, - eventsInList: ['event1', 'event2', 'event3'], + eventsInList: ['event1', 'event2', 'event3', 'event4', 'event5'], }, session: {identity: {_id: 'user1'}}, }); - const setAdvancedSearchDates = (state, startDate, endate) => { + const setAdvancedSearchDates = (state, startDate, endate?) => { state.main = { filter: 'EVENTS', search: { @@ -46,8 +65,8 @@ describe('selectors', () => { currentSearch: { advancedSearch: { dates: { - start: startDate ? moment(startDate, dateFormat) : null, - end: endate ? moment(endate, dateFormat) : null, + start: startDate ? moment(startDate) : null, + end: endate ? moment(endate) : null, }, }, }, @@ -60,7 +79,7 @@ describe('selectors', () => { it('all events', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-10T00:00:00'); + setAdvancedSearchDates(state, '2017-01-10T00:00:00+0000'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual([ @@ -87,7 +106,7 @@ describe('selectors', () => { state.events.events.event2.dates.start = moment('2017-01-17T08:00:00', dateFormat); state.events.events.event2.dates.end = moment('2017-01-17T14:00:00', dateFormat); - setAdvancedSearchDates(state, '2017-01-10T00:00:00'); + setAdvancedSearchDates(state, '2017-01-10T00:00:00+0000'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual([ @@ -112,7 +131,7 @@ describe('selectors', () => { it('for today 2017-01-15', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-15T00:00:00+0000', '2017-01-15T23:59:59+0000'); + setAdvancedSearchDates(state, '2017-01-15T00:00:00', '2017-01-15T23:59:59'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual([ @@ -126,7 +145,7 @@ describe('selectors', () => { it('from 2017-01-15 11:00 to 2017-01-16T14:00:00+0000', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-15T11:00:00+0000', '2017-01-16T14:00:00+0000'); + setAdvancedSearchDates(state, '2017-01-15T11:00:00', '2017-01-16T14:00:00'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual([ @@ -145,21 +164,22 @@ describe('selectors', () => { it('from 2017-01-16T06:00:00+0000 to 2017-01-16T07:00:00+0000', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-16T06:00:00+0000', '2017-01-16T07:00:00+0000'); + setAdvancedSearchDates(state, '2017-01-16T06:00:00', '2017-01-16T07:00:00'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual([ '2017-01-16', ]); - expect(events['2017-01-16'].events.length).toBe(1); + expect(events['2017-01-16'].events.length).toBe(2); expect(events['2017-01-16'].events[0]._id).toBe('event3'); + expect(events['2017-01-16'].events[1]._id).toBe('event2'); }); it('from 2017-01-16T06:00:00+0000 to 2017-01-16T10:00:00+0000', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-16T06:00:00+0000', '2017-01-16T10:00:00+0000'); + setAdvancedSearchDates(state, '2017-01-16T06:00:00', '2017-01-16T10:00:00'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual([ @@ -171,10 +191,10 @@ describe('selectors', () => { expect(events['2017-01-16'].events[1]._id).toBe('event2'); }); - it('from 2017-01-16T13:59:59+0000 to 2017-01-17T08:00:00+0000', () => { + it('from 2017-01-16T13:59:59 to 2017-01-17T08:00:00', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-16T13:59:59+0000', '2017-01-17T08:00:00+0000'); + setAdvancedSearchDates(state, '2017-01-16T13:59:59', '2017-01-17T08:00:00'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual(['2017-01-16', '2017-01-17']); @@ -187,7 +207,7 @@ describe('selectors', () => { it('from 2017-01-17T06:00:00+0000', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-17T06:00:00+0000'); + setAdvancedSearchDates(state, '2017-01-17T06:00:00'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events)).toEqual([ @@ -198,13 +218,79 @@ describe('selectors', () => { expect(events['2017-01-17'].events[0]._id).toBe('event3'); }); - it('from 2017-01-17T14:00:01+0000', () => { + it('from 2017-01-18T00:00:00+0000', () => { const state = getState(); - setAdvancedSearchDates(state, '2017-01-17T14:00:01+0000'); + setAdvancedSearchDates(state, '2017-01-18T00:00:00'); const events = keyBy(selectors.events.orderedEvents(state), 'date'); expect(Object.keys(events).length).toBe(0); }); + + it('from 2024-07-14', () => { + const state = getState(); + + setAdvancedSearchDates(state, '2024-07-14T12:00:00'); + const events = keyBy(selectors.events.orderedEvents(state), 'date'); + + expect(Object.keys(events)).toEqual([ + '2024-07-15', + '2024-07-16', + '2024-07-17', + '2024-07-18', + ]); + + expect(ids(events['2024-07-15'])).toEqual(['event5']); + expect(ids(events['2024-07-16'])).toEqual(['event5', 'event4']); + expect(ids(events['2024-07-17'])).toEqual(['event5', 'event4']); + expect(ids(events['2024-07-18'])).toEqual(['event5']); + }); + + it('from 2024-07-17', () => { + const state = getState(); + + setAdvancedSearchDates(state, '2024-07-17T12:00:00'); + const events = keyBy(selectors.events.orderedEvents(state), 'date'); + + expect(Object.keys(events)).toEqual([ + '2024-07-17', + '2024-07-18', + ]); + + expect(ids(events['2024-07-17'])).toEqual(['event5', 'event4']); + }); + + it('from 2024-07-14 Toronto timezone', () => { + moment.tz.setDefault('America/Toronto'); + const state = getState(); + + setAdvancedSearchDates(state, '2024-07-14T12:00:00'); + const events = keyBy(selectors.events.orderedEvents(state), 'date'); + + expect(Object.keys(events)).toEqual([ + '2024-07-15', + '2024-07-16', + '2024-07-17', + '2024-07-18', + ]); + + expect(ids(events['2024-07-15'])).toEqual(['event5']); + expect(ids(events['2024-07-16'])).toEqual(['event5', 'event4']); + expect(ids(events['2024-07-17'])).toEqual(['event5', 'event4']); + expect(ids(events['2024-07-18'])).toEqual(['event5']); + }); + + it('from 2024-07-17 Toronto timezone', () => { + moment.tz.setDefault('America/Toronto'); + const state = getState(); + + setAdvancedSearchDates(state, '2024-07-17T05:00:00'); + const events = keyBy(selectors.events.orderedEvents(state), 'date'); + + expect(Object.keys(events)).toEqual([ + '2024-07-17', + '2024-07-18', + ]); + }); }); }); diff --git a/client/services/tests/AssignmentsService_test.ts b/client/services/tests/AssignmentsService_test.ts index fc09d8a9f..0f3ee0413 100644 --- a/client/services/tests/AssignmentsService_test.ts +++ b/client/services/tests/AssignmentsService_test.ts @@ -140,6 +140,7 @@ describe('assignments service', () => { filterBy: 'Desk', searchQuery: 'planning.slugline.phrase:("test slugline")', orderByField: 'Scheduled', + dayField: null, filterByType: 'text', filterByPriority: null, selectedDeskId: ALL_DESKS, @@ -217,6 +218,7 @@ describe('assignments service', () => { searchQuery: 'planning.slugline.phrase:("test slugline")', orderByField: 'Scheduled', filterByType: 'text', + dayField: null, filterByPriority: null, selectedDeskId: ALL_DESKS, ignoreScheduledUpdates: true, diff --git a/client/tests.ts b/client/tests.ts index 0915d57ad..2fbe12cd5 100644 --- a/client/tests.ts +++ b/client/tests.ts @@ -20,9 +20,12 @@ Object.assign(appConfig, { planning_auto_assign_to_workflow: true, }); -moment.tz.setDefault('Australia/Sydney'); updateConfigAfterLoad(); +beforeEach(() => { + moment.tz.setDefault('Australia/Sydney'); +}); + var testsContext = require.context('.', true, /_test.[j|t]sx?$/); testsContext.keys().forEach(testsContext); diff --git a/client/utils/events.ts b/client/utils/events.ts index e49bf4aab..07c27119b 100644 --- a/client/utils/events.ts +++ b/client/utils/events.ts @@ -858,7 +858,6 @@ function getFlattenedEventsByDate(events: Array, startDate: moment.M return flatten(sortBy(eventsList, [(e) => (e.date)]).map((e) => e.events.map((k) => [e.date, k._id]))); } - const getStartDate = (event: IEventItem) => ( event.dates?.all_day ? moment.utc(event.dates.start) : moment(event.dates?.start) ); @@ -867,36 +866,6 @@ const getEndDate = (event: IEventItem) => ( (event.dates?.all_day || event.dates?.no_end_time) ? moment.utc(event.dates.end) : moment(event.dates?.end) ); -const isEventInRange = ( - event: IEventItem, - eventStart: moment.Moment, - eventEnd: moment.Moment, - start: moment.Moment, - end?: moment.Moment, -) => { - let localStart = eventStart; - let localEnd = eventEnd; - let startUnit : moment.unitOfTime.StartOf = 'second'; - let endUnit : moment.unitOfTime.StartOf = 'second'; - - if (event.dates?.all_day) { - // we have only dates in utc - localStart = moment(eventStart.format('YYYY-MM-DD')); - localEnd = moment(eventEnd.format('YYYY-MM-DD')); - startUnit = 'day'; - endUnit = 'day'; - } - - if (event.dates?.no_end_time) { - // we have time for start, but only date for end - localStart = moment(eventStart); - localEnd = moment(eventEnd.format('YYYY-MM-DD')); - endUnit = 'day'; - } - - return localEnd.isSameOrAfter(start, endUnit) && (end == null || localStart.isSameOrBefore(end, startUnit)); -}; - /* * Groups the events by date */ @@ -904,9 +873,9 @@ function getEventsByDate( events: Array, startDate: moment.Moment, endDate: moment.Moment -): Array { +) { if ((events?.length ?? 0) === 0) { - return []; + return {}; } // check if search exists @@ -920,23 +889,8 @@ function getEventsByDate( const days: {[date: string]: Array} = {}; - function addEventToDate(event: IEventItem, date?: moment.Moment) { - let eventDate = date || getStartDate(event); - let eventStart = getStartDate(event); - let eventEnd = getEndDate(event); - - if (!eventStart.isSame(eventEnd, 'day') && !event.dates.all_day && !event.dates.no_end_time) { - eventStart = eventDate; - eventEnd = eventEnd.isSame(eventDate, 'day') ? - eventEnd : - moment(eventDate.format('YYYY-MM-DD'), 'YYYY-MM-DD').add(86399, 'seconds'); - } - - if (!isEventInRange(event, eventDate, eventEnd, startDate, endDate)) { - return; - } - - let eventDateFormatted = eventDate.format('YYYY-MM-DD'); + function addEventToDate(event: IEventItem, date: moment.Moment, eventStart: moment.Moment) { + const eventDateFormatted = date.format('YYYY-MM-DD'); if (!days[eventDateFormatted]) { days[eventDateFormatted] = []; @@ -949,37 +903,36 @@ function getEventsByDate( } sortedEvents.forEach((event) => { - // compute the number of days of the event - const eventEndDate = event.actioned_date ? moment(event.actioned_date) : getEndDate(event); - const eventStartDate = getStartDate(event); - - if (!eventStartDate.isSame(eventEndDate, 'day')) { - let deltaDays = Math.max(Math.ceil(eventEndDate.diff(eventStartDate, 'days', true)), 1); - // if the event happens during more than one day, add it to every day - // add the event to the other days - - for (let i = 1; i < deltaDays; i++) { - // clone the date - const newDate = moment(eventStartDate.format('YYYY-MM-DD'), 'YYYY-MM-DD', true); - - newDate.add(i, 'days'); + const eventEndDate = event.actioned_date ? moment(event.actioned_date) : getLocalEndDate(event); + const eventStartDate = getLocalStartDate(event); - if (newDate.isSameOrBefore(eventEndDate, 'day')) { - addEventToDate(event, newDate); - } - } - } + const displayStartDate = eventStartDate.isSameOrAfter(startDate) ? eventStartDate : startDate; + const displayEndDate = eventEndDate.isSameOrBefore(endDate) ? eventEndDate : endDate; - // add event to its initial starting date - // add an event only if it's not actioned or actioned after this event's start date - if (!event.actioned_date || moment(event.actioned_date).isSameOrAfter(eventStartDate, 'date')) { - addEventToDate(event); + for (const day = displayStartDate.clone(); day.isSameOrBefore(displayEndDate, 'day'); day.add(1, 'days')) { + addEventToDate(event, day, eventStartDate); } }); return sortBasedOnTBC(days); } +function getLocalStartDate(event: IEventItem): moment.Moment { + if (event.dates.all_day) { + return moment(moment.utc(event.dates.start).format('YYYY-MM-DD')); + } + + return moment(event.dates.start); +} + +function getLocalEndDate(event: IEventItem): moment.Moment { + if (event.dates.all_day || event.dates.no_end_time) { + return moment(moment.utc(event.dates.end).format('YYYY-MM-DD')).endOf('day'); + } + + return moment(event.dates.end); +} + function modifyForClient(event: Partial): Partial { sanitizeItemFields(event); diff --git a/client/utils/index.ts b/client/utils/index.ts index ff17907e5..38d167fd9 100644 --- a/client/utils/index.ts +++ b/client/utils/index.ts @@ -11,6 +11,9 @@ import { IPlanningCoverageItem, IIngestProvider, IFeaturedPlanningItem, + ISearchParams, + ICommonSearchParams, + JUMP_INTERVAL, } from '../interfaces'; import {IUser} from 'superdesk-api'; import {superdeskApi} from '../superdeskApi'; @@ -688,43 +691,63 @@ export const isDateInRange = (inputDate, startDate, endDate) => { return true; }; -export const getSearchDateRange = (currentSearch, startOfWeek) => { - const dates = get(currentSearch, 'advancedSearch.dates', {}); - const dateRange = {startDate: null, endDate: null}; - - if (!get(dates, 'start') && !get(dates, 'end') && !get(dates, 'range')) { - dateRange.startDate = moment(moment().format('YYYY-MM-DD'), 'YYYY-MM-DD', true); - dateRange.endDate = moment().add(999, 'years'); - } else if (get(dates, 'range')) { - let range = get(dates, 'range'); +interface IDateRange { + startDate: moment.Moment; + endDate: moment.Moment; +} - if (range === MAIN.DATE_RANGE.TODAY) { - dateRange.startDate = moment(moment().format('YYYY-MM-DD'), 'YYYY-MM-DD', true); - dateRange.endDate = dateRange.startDate.clone().add('86399', 'seconds'); - } else if (range === MAIN.DATE_RANGE.TOMORROW) { - const tomorrow = moment().add(1, 'day'); +const INTERVAL_UNIT_MAPPING = { + [JUMP_INTERVAL.DAY]: 'day', + [JUMP_INTERVAL.WEEK]: 'week', + [JUMP_INTERVAL.MONTH]: 'month', +}; - dateRange.startDate = moment(tomorrow.format('YYYY-MM-DD'), 'YYYY-MM-DD', true); +export const getSearchDateRange = ( + currentSearch: ICommonSearchParams, + startOfWeek: number, + viewInterval?: JUMP_INTERVAL, +): IDateRange => { + const dates = currentSearch.advancedSearch.dates ?? {}; + const dateRange = {startDate: null, endDate: null}; + const jumpUnit = viewInterval ? INTERVAL_UNIT_MAPPING[viewInterval] : 'month'; + + if (dates.range) { + if (dates.range === MAIN.DATE_RANGE.TODAY) { + dateRange.startDate = moment().startOf('day'); + dateRange.endDate = dateRange.startDate.clone() + .add(1, 'day'); + } else if (dates.range === MAIN.DATE_RANGE.TOMORROW) { + const tomorrow = moment().add(1, 'day') + .startOf('day'); + + dateRange.startDate = tomorrow; dateRange.endDate = tomorrow.clone().add('86399', 'seconds'); - } else if (range === MAIN.DATE_RANGE.LAST_24) { + } else if (dates.range === MAIN.DATE_RANGE.LAST_24) { dateRange.endDate = moment(); dateRange.startDate = dateRange.endDate.clone().subtract('86400', 'seconds'); - } else if (range === MAIN.DATE_RANGE.THIS_WEEK) { + } else if (dates.range === MAIN.DATE_RANGE.THIS_WEEK) { dateRange.endDate = timeUtils.getStartOfNextWeek(null, startOfWeek); dateRange.startDate = dateRange.endDate.clone().subtract(7, 'days'); - } else if (range === MAIN.DATE_RANGE.NEXT_WEEK) { + } else if (dates.range === MAIN.DATE_RANGE.NEXT_WEEK) { dateRange.endDate = timeUtils.getStartOfNextWeek(null, startOfWeek).add(7, 'days'); dateRange.startDate = dateRange.endDate.clone().subtract(7, 'days'); } + } else if (dates.start && dates.end) { + dateRange.startDate = moment(dates.start); + dateRange.endDate = moment(dates.end); + } else if (dates.start) { + dateRange.startDate = moment(dates.start); + dateRange.endDate = dateRange.startDate.clone().add(1, jumpUnit) + .subtract(1, 'second'); // remove 1s not to display additional day + } else if (dates.end) { + dateRange.endDate = moment(dates.end); + dateRange.startDate = dateRange.endDate.clone().subtract(1, jumpUnit); } else { - if (get(dates, 'start')) { - dateRange.startDate = moment(get(dates, 'start')); - } - - if (get(dates, 'end')) { - dateRange.endDate = moment(get(dates, 'end')); - } + dateRange.startDate = moment().startOf('day'); + dateRange.endDate = dateRange.startDate.clone().add(1, jumpUnit) + .subtract(1, 'second'); // remove 1s not to display additional day } + return dateRange; }; diff --git a/client/utils/planning.ts b/client/utils/planning.ts index 081e76a34..3aa1464be 100644 --- a/client/utils/planning.ts +++ b/client/utils/planning.ts @@ -21,6 +21,7 @@ import { ICoverageScheduledUpdate, IDateTime, IItemAction, + IPlanningAssignedTo, } from '../interfaces'; const appConfig = config as IPlanningConfig; @@ -1223,15 +1224,28 @@ function getCoverageIcon( return coverageIcons[type]?.[iconType] ?? iconForUnknownType; } -function getCoverageIconColor(coverage: IPlanningCoverageItem): string { - if (coverage.workflow_status === COVERAGES.WORKFLOW_STATE.ACTIVE) { - return 'var(--sd-colour-success)'; - } else if (get(coverage, 'assigned_to.state') === ASSIGNMENTS.WORKFLOW_STATE.COMPLETED) { - return 'var(--sd-colour-success)'; - } else if (isCoverageDraft(coverage) || get(coverage, 'workflow_status') === COVERAGES.WORKFLOW_STATE.ACTIVE) { - return 'var(--sd-colour-highlight)'; +function getCoverageIconColor(item: IPlanningCoverageItem): string | undefined { + if (item.workflow_status === 'cancelled') { + return 'var(--sd-colour-state--canceled)'; + } + + if (item.assigned_to == null) { + return undefined; + } + + switch (getItemWorkflowState(item.assigned_to)) { + case ASSIGNMENTS.WORKFLOW_STATE.ASSIGNED: + return 'var(--sd-colour-state--in-workflow)'; + case ASSIGNMENTS.WORKFLOW_STATE.IN_PROGRESS: + return 'var(--sd-colour-state--in-progress)'; + case ASSIGNMENTS.WORKFLOW_STATE.COMPLETED: + return 'var(--sd-colour-state--completed)'; + } + + if (item.assigned_to.user != null || item.assigned_to.desk != null) { + return 'var(--sd-colour-state--assigned)'; } else { - return 'var(--color-text-lighter)'; + return 'var(--sd-colour-state--unassigned)'; } } @@ -1256,6 +1270,21 @@ function getCoverageWorkflowIcon(coverage: IPlanningCoverageItem): string | null } } +function getNewsCoverageStatusDotColor(coverage: DeepPartial): string | null { + if (coverage.news_coverage_status == null) { + return undefined; + } + + switch (coverage.news_coverage_status.qcode) { + case 'ncostat:notdec': + return 'var(--sd-colour-coverage-state--on-merit)'; + case 'ncostat:notint': + return 'var(--sd-colour-coverage-state--not-covering)'; + default: + return null; + } +} + function getCoverageContentType( coverage: IPlanningCoverageItem, contentTypes: Array = [] @@ -1672,6 +1701,7 @@ const self = { getCoverageIcon, getCoverageIconColor, getCoverageWorkflowIcon, + getNewsCoverageStatusDotColor, shouldLockPlanningForEdit, modifyForClient, modifyForServer, diff --git a/client/utils/testData.ts b/client/utils/testData.ts index dc1c42807..40efecba2 100644 --- a/client/utils/testData.ts +++ b/client/utils/testData.ts @@ -516,6 +516,7 @@ export const assignmentInitialState = { filterByPriority: null, filterByType: null, myAssignmentsTotal: 0, + dayField: null, orderByField: 'Scheduled', previewOpened: false, readOnly: false, diff --git a/e2e/cypress/e2e/events/edit_event.cy.ts b/e2e/cypress/e2e/events/edit_event.cy.ts index 0c78b9631..10737cb75 100644 --- a/e2e/cypress/e2e/events/edit_event.cy.ts +++ b/e2e/cypress/e2e/events/edit_event.cy.ts @@ -1,8 +1,9 @@ import {cloneDeep} from 'lodash'; -import {setup, login, waitForPageLoad, SubNavBar, Workqueue, Modal, addItems} from '../../support/common'; +import {setup, login, waitForPageLoad, SubNavBar, Workqueue, Modal, addItems, CLIENT_FORMAT} from '../../support/common'; import {EventEditor, PlanningList} from '../../support/planning'; -import {TEST_EVENTS} from '../../fixtures/events'; +import {createEventFor, TEST_EVENTS} from '../../fixtures/events'; +import moment from 'moment'; const list = new PlanningList(); const editor = new EventEditor(); @@ -16,7 +17,7 @@ describe('Planning.Events: edit metadata', () => { beforeEach(() => { event = { - 'dates.start.date': '12/12/2045', + 'dates.start.date': moment().format(CLIENT_FORMAT), slugline: 'slugline of the event', name: 'name of the event', definition_short: 'Desc.', @@ -34,7 +35,7 @@ describe('Planning.Events: edit metadata', () => { expectedEvent = { ...event, - 'dates.end.date': '12/12/2045', + 'dates.end.date': moment().format(CLIENT_FORMAT), }; setup({fixture_profile: 'planning_prepopulate_data'}, '/#/planning'); @@ -74,7 +75,7 @@ describe('Planning.Events: edit metadata', () => { event = { ...event, 'dates.recurring.enable': true, - 'dates.recurring.until': '13/12/2045', + 'dates.recurring.until': moment().add(1, 'day').format(CLIENT_FORMAT), 'dates.allDay': true, slugline: 'slugline of the recurring event', name: 'name of the recurring event', @@ -82,7 +83,7 @@ describe('Planning.Events: edit metadata', () => { expectedEvent = { ...expectedEvent, 'dates.recurring.enable': true, - 'dates.recurring.until': '13/12/2045', + 'dates.recurring.until': moment().add(1, 'day').format(CLIENT_FORMAT), 'dates.allDay': true, slugline: 'slugline of the recurring event', name: 'name of the recurring event', @@ -133,7 +134,7 @@ describe('Planning.Events: edit metadata', () => { // Enter minimum Event metadata editor.expectItemType(); editor.type({ - 'dates.start.date': '12/12/2045', + 'dates.start.date': moment().format(CLIENT_FORMAT), 'dates.allDay': true, slugline: 'slugline of the event', name: 'name of the event', @@ -181,20 +182,12 @@ describe('Planning.Events: edit metadata', () => { describe('Planing.Events: edit existing events', () => { beforeEach(() => { setup({fixture_profile: 'planning_prepopulate_data'}, '/#/planning'); - addItems('events', [{ + addItems('events', [createEventFor.tomorrow({ ...cloneDeep(TEST_EVENTS.date_01_02_2045), - dates: { - start: TEST_EVENTS.date_01_02_2045.dates.start, - end: TEST_EVENTS.date_01_02_2045.dates.end, - }, - }, { + }), createEventFor.tomorrow({ ...cloneDeep(TEST_EVENTS.date_02_02_2045), - dates: { - start: TEST_EVENTS.date_02_02_2045.dates.start, - end: TEST_EVENTS.date_02_02_2045.dates.end, - tz: null, - }, - }]); + }, null), + ]); login(); waitForPageLoad.planning(); diff --git a/e2e/cypress/e2e/events/event_action_cancel.cy.ts b/e2e/cypress/e2e/events/event_action_cancel.cy.ts index 762959d32..075d5adbc 100644 --- a/e2e/cypress/e2e/events/event_action_cancel.cy.ts +++ b/e2e/cypress/e2e/events/event_action_cancel.cy.ts @@ -1,7 +1,7 @@ import {setup, login, addItems, waitForPageLoad, Modal} from '../../support/common'; -import {TIME_STRINGS} from '../../support/utils/time'; import {PlanningList, EventEditor, PlanningPreview} from '../../support/planning'; import {getMenuItem} from '../../support/common/ui/actionMenu'; +import {createEventFor} from '../../fixtures/events'; describe('Planning.Events: event cancel action', () => { const editor = new EventEditor(); @@ -13,24 +13,19 @@ describe('Planning.Events: event cancel action', () => { beforeEach(() => { setup({fixture_profile: 'planning_prepopulate_data'}, '/#/planning'); - addItems('events', [{ + addItems('events', [createEventFor.today({ type: 'event', occur_status: { name: 'Planned, occurs certainly', label: 'Confirmed', qcode: 'eocstat:eos5', }, - dates: { - start: '2045-12-11' + TIME_STRINGS[0], - end: '2045-12-11' + TIME_STRINGS[1], - tz: 'Australia/Sydney', - }, calendars: [], state: 'draft', place: [], name: 'Test', slugline: 'Original', - }]); + })]); login(); diff --git a/e2e/cypress/e2e/events/event_action_create_planning.cy.ts b/e2e/cypress/e2e/events/event_action_create_planning.cy.ts index 312eaebdd..bf3652b76 100644 --- a/e2e/cypress/e2e/events/event_action_create_planning.cy.ts +++ b/e2e/cypress/e2e/events/event_action_create_planning.cy.ts @@ -1,6 +1,6 @@ import moment from 'moment-timezone'; -import {setup, login, addItems, waitForPageLoad} from '../../support/common'; +import {setup, login, addItems, waitForPageLoad, CLIENT_FORMAT} from '../../support/common'; import {TIMEZONE} from '../../support/utils/time'; import {PlanningList, PlanningPreview, EventEditor, PlanningEditor} from '../../support/planning'; @@ -14,10 +14,12 @@ describe('Planning.Events: create planning action', () => { const preview = new PlanningPreview(); let menu; + const start = moment(); + const expectedValues = { slugline: 'Original', - 'planning_date.date': '12/12/2025', - 'planning_date.time': '01:00', + 'planning_date.date': start.format(CLIENT_FORMAT), + 'planning_date.time': start.format('HH:mm'), description_text: 'Desc.', ednote: 'Ed. Note', anpa_category: ['Finance'], @@ -26,7 +28,6 @@ describe('Planning.Events: create planning action', () => { beforeEach(() => { setup({fixture_profile: 'planning_prepopulate_data'}, '/#/planning'); - const start = moment.tz("2025-12-12 01:00", TIMEZONE).utc(); addItems('events', [{ slugline: 'Original', definition_short: 'Desc.', @@ -36,8 +37,8 @@ describe('Planning.Events: create planning action', () => { qcode: 'eocstat:eos5', }, dates: { - start: start.format("YYYY-MM-DDTHH:mm:ss+0000"), - end: start.add(1, 'h').format('YYYY-MM-DDTHH:mm:ss+0000'), + start: start.utc().format("YYYY-MM-DDTHH:mm:ss+0000"), + end: start.clone().utc().add(1, 'h').format('YYYY-MM-DDTHH:mm:ss+0000'), tz: TIMEZONE, }, anpa_category: [{is_active: true, name: 'Finance', qcode: 'f', subject: '04000000'}], diff --git a/e2e/cypress/e2e/events/event_action_duplicate.cy.ts b/e2e/cypress/e2e/events/event_action_duplicate.cy.ts index 66210d4f2..c9d28bd6f 100644 --- a/e2e/cypress/e2e/events/event_action_duplicate.cy.ts +++ b/e2e/cypress/e2e/events/event_action_duplicate.cy.ts @@ -1,4 +1,5 @@ -import {setup, login, waitForPageLoad, Modal, SubNavBar} from '../../support/common'; +import moment from 'moment'; +import {setup, login, waitForPageLoad, Modal, SubNavBar, CLIENT_FORMAT} from '../../support/common'; import {PlanningList, PlanningPreview, EventEditor} from '../../support/planning'; describe('Planning.Events: duplicate event', () => { @@ -12,11 +13,13 @@ describe('Planning.Events: duplicate event', () => { const preview = new PlanningPreview(); beforeEach(() => { + const now = moment(); + event = { slugline: 'Original', name: 'Test', definition_short: 'Desc.', - 'dates.start.date': '12/12/2045', + 'dates.start.date': now.format(CLIENT_FORMAT), 'dates.start.time': '00:00', occur_status: 'Planned, occurs certainly', calendars: ['Sport'], @@ -29,7 +32,7 @@ describe('Planning.Events: duplicate event', () => { }; expectedValues = Object.assign({}, event, { - 'dates.end.date': '12/12/2045', + 'dates.end.date': now.format(CLIENT_FORMAT), 'dates.end.time': '01:00', }); diff --git a/e2e/cypress/e2e/events/event_embedded_coverage.cy.ts b/e2e/cypress/e2e/events/event_embedded_coverage.cy.ts index ac22deac5..9983d4452 100644 --- a/e2e/cypress/e2e/events/event_embedded_coverage.cy.ts +++ b/e2e/cypress/e2e/events/event_embedded_coverage.cy.ts @@ -1,7 +1,9 @@ -import {setup, login, waitForPageLoad, SubNavBar, addItems} from '../../support/common'; +import {setup, login, waitForPageLoad, SubNavBar, addItems, CLIENT_FORMAT} from '../../support/common'; import {TIME_STRINGS} from '../../support/utils/time'; import {EventEditor, PlanningList} from '../../support/planning'; import {EmbeddedCoverageEditor} from '../../support/planning/events/embeddedCoverageEditor'; +import moment from 'moment'; +import {createEventFor} from '../../fixtures/events'; describe('Planning.Events: embedded coverage', () => { const editor = new EventEditor(); @@ -21,7 +23,7 @@ describe('Planning.Events: embedded coverage', () => { // Enter required fields (so we can create the Event & Planning) editor.type({ - 'dates.start.date': '12/12/2045', + 'dates.start.date': moment().format(CLIENT_FORMAT), 'dates.allDay': true, slugline: 'slugline of the event', name: 'name of the event', @@ -119,24 +121,19 @@ describe('Planning.Events: embedded coverage', () => { }); it('can add a planning item to an existing event', () => { - addItems('events', [{ + addItems('events', [createEventFor.today({ type: 'event', occur_status: { name: 'Planned, occurs certainly', label: 'Confirmed', qcode: 'eocstat:eos5', }, - dates: { - start: '2045-12-11' + TIME_STRINGS[0], - end: '2045-12-11' + TIME_STRINGS[1], - tz: 'Australia/Sydney', - }, calendars: [], state: 'draft', place: [], name: 'Test', slugline: 'slugline of the event', - }]); + })]); list.item(0) .dblclick(); @@ -234,24 +231,19 @@ describe('Planning.Events: embedded coverage', () => { }); it('SDESK-6022: planning items should stay after post/unpost', () => { - addItems('events', [{ + addItems('events', [createEventFor.today({ type: 'event', occur_status: { name: 'Planned, occurs certainly', label: 'Confirmed', qcode: 'eocstat:eos5', }, - dates: { - start: '2045-12-11' + TIME_STRINGS[0], - end: '2045-12-11' + TIME_STRINGS[1], - tz: 'Australia/Sydney', - }, calendars: [], state: 'draft', place: [], name: 'Test', slugline: 'slugline of the event', - }]); + })]); list.item(0) .dblclick(); @@ -338,16 +330,18 @@ describe('Planning.Events: embedded coverage', () => { .should('exist') .should('be.enabled'); + const now = moment() + // Fill in the dates (which should also update the Planning/Coverage dates) editor.type({ - 'dates.start.date': '12/12/2045', + 'dates.start.date': now.format(CLIENT_FORMAT), 'dates.allDay': true, }); // Make sure the date has been updated for the Coverage embeddedCoverages.getRelatedCoverage(0, 0) .should('exist') - .should('contain.text', '12/12/2045 @ 00:00'); + .should('contain.text', `${now.format(CLIENT_FORMAT)} @ 00:00`); // Now create the Event & Planning item editor.waitForAutosave(); diff --git a/e2e/cypress/e2e/planning/edit_planning.cy.ts b/e2e/cypress/e2e/planning/edit_planning.cy.ts index a67861636..72f3c55ad 100644 --- a/e2e/cypress/e2e/planning/edit_planning.cy.ts +++ b/e2e/cypress/e2e/planning/edit_planning.cy.ts @@ -1,4 +1,5 @@ -import {setup, login, waitForPageLoad, SubNavBar, Workqueue} from '../../support/common'; +import moment from 'moment'; +import {setup, login, waitForPageLoad, SubNavBar, Workqueue, CLIENT_FORMAT} from '../../support/common'; import {PlanningList, PlanningEditor, AssignmentEditor} from '../../support/planning'; describe('Planning.Planning: edit metadata', () => { @@ -20,7 +21,7 @@ describe('Planning.Planning: edit metadata', () => { it('can create a Planning item', () => { const plan = { slugline: 'slugline of the planning', - 'planning_date.date': '12/12/2045', + 'planning_date.date': moment().format(CLIENT_FORMAT), 'planning_date.time': '12:13', description_text: 'Desc. Text', internal_note: 'Int. Note', @@ -38,7 +39,7 @@ describe('Planning.Planning: edit metadata', () => { ednote: 'something to write about', internal_note: 'internal to us', news_coverage_status: 'On merit', - 'scheduled.date': '12/12/2045', + 'scheduled.date': moment().format(CLIENT_FORMAT), 'scheduled.time': '13:15', }]; @@ -66,7 +67,7 @@ describe('Planning.Planning: edit metadata', () => { it('can add coverage to workflow', () => { editor.type({ slugline: 'Plan', - 'planning_date.date': '12/12/2045', + 'planning_date.date': moment().format(CLIENT_FORMAT), 'planning_date.time': '12:13', }); editor.addCoverage('Picture'); @@ -95,7 +96,7 @@ describe('Planning.Planning: edit metadata', () => { it('not_for_publication flag will not enable post button', () => { const plan = { slugline: 'slugline of the planning', - 'planning_date.date': '12/12/2045', + 'planning_date.date': moment().format(CLIENT_FORMAT), 'planning_date.time': '12:13', }; @@ -149,7 +150,7 @@ describe('Planning.Planning: edit metadata', () => { editor.expectItemType(); editor.type({ slugline: 'slugline of the planning', - 'planning_date.date': '12/12/2045', + 'planning_date.date': moment().format(CLIENT_FORMAT), 'planning_date.time': '12:13', }); diff --git a/e2e/cypress/e2e/planning/planning_action_cancel.cy.ts b/e2e/cypress/e2e/planning/planning_action_cancel.cy.ts index bb9eb710c..3dec21205 100644 --- a/e2e/cypress/e2e/planning/planning_action_cancel.cy.ts +++ b/e2e/cypress/e2e/planning/planning_action_cancel.cy.ts @@ -2,6 +2,7 @@ import {setup, addItems, login, waitForPageLoad, Modal} from '../../support/comm import {TIME_STRINGS} from '../../support/utils/time'; import {PlanningList, PlanningEditor, PlanningPreview} from '../../support/planning'; import {getMenuItem} from '../../support/common/ui/actionMenu'; +import moment from 'moment'; describe('Planning.Planning: cancel planning item', () => { const editor = new PlanningEditor(); @@ -14,7 +15,7 @@ describe('Planning.Planning: cancel planning item', () => { setup({fixture_profile: 'planning_prepopulate_data'}, '/#/planning'); addItems('planning', [{ slugline: 'Test Planning Item', - planning_date: '2045-12-11' + TIME_STRINGS[0], + planning_date: moment().format('yy-MM-DD') + TIME_STRINGS[0], }]); login(); @@ -60,7 +61,7 @@ describe('Planning.Planning: cancel planning item', () => { list.item(0) .click(); - getMenuItem(list.item(0), 'Cancel Planning').realClick(); + getMenuItem(list.item(0), 'Cancel Planning').click(); modal.waitTillOpen(30000); diff --git a/e2e/cypress/e2e/search/search_combined.cy.ts b/e2e/cypress/e2e/search/search_combined.cy.ts index 64aa06fbe..cd5468715 100644 --- a/e2e/cypress/e2e/search/search_combined.cy.ts +++ b/e2e/cypress/e2e/search/search_combined.cy.ts @@ -103,7 +103,7 @@ describe('Search.Combined: searching events and planning', () => { search.openAllToggleBoxes(); search.runSearchTests([{ params: {}, - expectedCount: 14, + expectedCount: 6, expectedText: [ 'Event Today', 'Plan Today', @@ -111,14 +111,6 @@ describe('Search.Combined: searching events and planning', () => { 'Plan Tomorrow', 'Event Next Week', 'Plan Next Week', - 'Event Feb 1', - 'Plan Feb 1', - 'Event Feb 2', - 'Plan Feb 2', - 'Event Feb 3', - 'Plan Feb 3', - 'Event Feb 4', - 'Plan Feb 4', ], }, { params: { diff --git a/e2e/cypress/e2e/search/search_events.cy.ts b/e2e/cypress/e2e/search/search_events.cy.ts index ae1c0cb35..08991d4dd 100644 --- a/e2e/cypress/e2e/search/search_events.cy.ts +++ b/e2e/cypress/e2e/search/search_events.cy.ts @@ -144,15 +144,11 @@ describe('Search.Events: searching events', () => { search.runSearchTests([{ params: {}, - expectedCount: 7, + expectedCount: 3, expectedText: [ 'Event Today', 'Event Tomorrow', 'Event Next Week', - 'Event Feb 1', - 'Event Feb 2', - 'Event Feb 3', - 'Event Feb 4', ] }, { params: { diff --git a/e2e/cypress/e2e/search/search_planning.cy.ts b/e2e/cypress/e2e/search/search_planning.cy.ts index e41e54445..8960385e9 100644 --- a/e2e/cypress/e2e/search/search_planning.cy.ts +++ b/e2e/cypress/e2e/search/search_planning.cy.ts @@ -115,15 +115,11 @@ describe('Search.Planning: searching planning items', () => { search.runSearchTests([{ params: {}, - expectedCount: 7, + expectedCount: 3, expectedText: [ 'Plan Today', 'Plan Tomorrow', 'Plan Next Week', - 'Plan Feb 1', - 'Plan Feb 2', - 'Plan Feb 3', - 'Plan Feb 4', ], }, { params: { diff --git a/e2e/cypress/e2e/workqueue.cy.ts b/e2e/cypress/e2e/workqueue.cy.ts index 9111c35e5..5def5122a 100644 --- a/e2e/cypress/e2e/workqueue.cy.ts +++ b/e2e/cypress/e2e/workqueue.cy.ts @@ -1,5 +1,5 @@ import {setup, login, waitForPageLoad, Workqueue, Modal, addItems} from '../support/common'; -import {EventEditor, PlanningList} from '../support/planning'; +import {AdvancedSearch, EventEditor, PlanningList} from '../support/planning'; import {TEST_EVENTS} from '../fixtures/events'; describe('Planning.Workqueue', () => { @@ -7,6 +7,7 @@ describe('Planning.Workqueue', () => { const list = new PlanningList(); const workqueue = new Workqueue(); const modal = new Modal(); + const search = new AdvancedSearch(); beforeEach(() => { setup({fixture_profile: 'planning_prepopulate_data'}, '/#/planning'); @@ -22,6 +23,8 @@ describe('Planning.Workqueue', () => { TEST_EVENTS.date_03_02_2045, ]); + search.setStartDate('01/02/2045'); + // Wait for the 3 items to appear in the list list.expectItemCount(3); workqueue.expectItemCount(0); diff --git a/e2e/cypress/fixtures/events.ts b/e2e/cypress/fixtures/events.ts index f68b297ef..353aa2c7f 100644 --- a/e2e/cypress/fixtures/events.ts +++ b/e2e/cypress/fixtures/events.ts @@ -42,13 +42,8 @@ const BASE_EVENT = { }; export const TEST_EVENTS = { - draft: { + draft: getEventForDate(getDateStringFor.today(), { ...BASE_EVENT, - dates: { - start: '2045-12-11' + TIME_STRINGS[0], - end: '2045-12-11' + TIME_STRINGS[1], - tz: TIMEZONE, - }, name: 'Test', slugline: 'Original', anpa_category: [ @@ -67,18 +62,13 @@ export const TEST_EVENTS = { name: LOCATIONS.sydney_opera_house.name, address: LOCATIONS.sydney_opera_house.address, }], - }, - spiked: { + }), + spiked: getEventForDate(getDateStringFor.today(), { ...BASE_EVENT, - dates: { - start: '2045-12-11' + TIME_STRINGS[0], - end: '2045-12-11' + TIME_STRINGS[1], - tz: TIMEZONE, - }, state: 'spiked', name: 'Spiker', slugline: 'Spiked', - }, + }), date_01_02_2045: { ...BASE_EVENT, dates: { @@ -121,21 +111,21 @@ export const TEST_EVENTS = { }, }; -function getEventForDate(dateString: string, metadata: {[key: string]: any} = {}) { +function getEventForDate(dateString: string, metadata: {[key: string]: any} = {}, timezone = TIMEZONE) { return { ...BASE_EVENT, + ...metadata, dates: { start: dateString + TIME_STRINGS[0], end: dateString + TIME_STRINGS[1], - tz: TIMEZONE, + tz: timezone, }, - ...metadata, }; } export const createEventFor = { - today: (metadata = {}) => getEventForDate(getDateStringFor.today(), metadata), - tomorrow: (metadata = {}) => getEventForDate(getDateStringFor.tomorrow(), metadata), - yesterday: (metadata = {}) => getEventForDate(getDateStringFor.yesterday(), metadata), - next_week: (metadata = {}) => getEventForDate(getDateStringFor.next_week(), metadata), + today: (metadata = {}, timezone = TIMEZONE) => getEventForDate(getDateStringFor.today(), metadata, timezone), + tomorrow: (metadata = {}, timezone = TIMEZONE) => getEventForDate(getDateStringFor.tomorrow(), metadata, timezone), + yesterday: (metadata = {}, timezone = TIMEZONE) => getEventForDate(getDateStringFor.yesterday(), metadata, timezone), + next_week: (metadata = {}, timezone = TIMEZONE) => getEventForDate(getDateStringFor.next_week(), metadata, timezone), }; diff --git a/e2e/cypress/fixtures/planning.ts b/e2e/cypress/fixtures/planning.ts index 707b257fc..fe317b55f 100644 --- a/e2e/cypress/fixtures/planning.ts +++ b/e2e/cypress/fixtures/planning.ts @@ -12,10 +12,9 @@ const BASE_PLANNING = { }; export const TEST_PLANNINGS = { - draft: { + draft: getPlanningForDate(getDateStringFor.today(), { ...BASE_PLANNING, slugline: 'Original', - planning_date: '2045-12-11T01:00:00+0000', anpa_category: [ {name: 'Overseas Sport', qcode: 's'}, {name: 'International News', qcode: 'i'}, @@ -24,19 +23,17 @@ export const TEST_PLANNINGS = { {qcode: '01001000', name: 'archaeology', parent: '01000000'}, {qcode: '01011000', name: 'music', parent: '01000000'}, ], - }, - spiked: { + }), + spiked: getPlanningForDate(getDateStringFor.today(), { ...BASE_PLANNING, slugline: 'Spiker', - planning_date: '2045-12-11T01:00:00+0000', state: 'spiked', - }, - featured: { + }), + featured: getPlanningForDate(getDateStringFor.today(), { ...BASE_PLANNING, slugline: 'Featured Planning', - planning_date: '2045-12-12T01:00:00+0000', featured: true, - }, + }), plan_date_01_02_2045: { ...BASE_PLANNING, slugline: 'Plan Feb 1', diff --git a/e2e/cypress/support/common/utils.ts b/e2e/cypress/support/common/utils.ts index ffa077b42..edd00d85f 100644 --- a/e2e/cypress/support/common/utils.ts +++ b/e2e/cypress/support/common/utils.ts @@ -7,3 +7,5 @@ export function constructUrl(base, uri) { return base.replace(/\/$/, '') + uri; } + +export const CLIENT_FORMAT = 'DD/MM/yy'; diff --git a/e2e/cypress/support/planning/advancedSearch.ts b/e2e/cypress/support/planning/advancedSearch.ts index 71f9d796e..fdd1c54a5 100644 --- a/e2e/cypress/support/planning/advancedSearch.ts +++ b/e2e/cypress/support/planning/advancedSearch.ts @@ -216,10 +216,9 @@ export class AdvancedSearch { } expectSearchResultCount(run: ISearchTest) { - cy.log('Search: ' + JSON.stringify(run.params)); + cy.log('Search: ' + JSON.stringify(run.params) + ', expected count: ' + run.expectedCount); if (Object.keys(run.params).length > 0) { - this.enterSearchParams(run.params); - this.clickSearch(); + this.searchFor(run.params); } if (run.expectedCount != null) { @@ -252,4 +251,16 @@ export class AdvancedSearch { .should('exist') .click(); } + + searchFor(params: ISearchTest['params']) { + this.enterSearchParams(params); + this.clickSearch(); + } + + setStartDate(date: string) { + this.toggleSearchPanel(); + this.openAllToggleBoxes(); + this.searchFor({'start_date.date': date}); + this.toggleSearchPanel(); + } } diff --git a/e2e/server/Dockerfile b/e2e/server/Dockerfile index dc7feea11..b26f1ce0c 100644 --- a/e2e/server/Dockerfile +++ b/e2e/server/Dockerfile @@ -6,6 +6,8 @@ RUN apt-get update && apt-get install -y --no-install-recommends \ python3 python3-dev python3-pip python3-venv git gcc curl \ # lxml libxml2-dev libxslt-dev \ +# xmlsec +pkg-config libxml2-dev libxmlsec1-dev libxmlsec1-openssl \ # PIL libjpeg-dev zlib1g-dev \ # magic diff --git a/package-lock.json b/package-lock.json index f01d7124f..a96fa855b 100644 --- a/package-lock.json +++ b/package-lock.json @@ -590,7 +590,7 @@ "ajv-keywords": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-2.1.1.tgz", - "integrity": "sha512-ZFztHzVRdGLAzJmpUT9LNFLe1YiVOEylcaNpEutM26PVTCtOD919IMfD01CgbRouB42Dd9atjx1HseC15DgOZA==", + "integrity": "sha1-YXmX/F9gV2iUxDX5QNgZ4TW4B2I=", "dev": true }, "align-text": { @@ -898,7 +898,7 @@ "arr-diff": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", - "integrity": "sha512-YVIQ82gZPGBebQV/a8dar4AitzCQs0jjXwMPZllpXMaGjXPYVUawSxQrRsjhjupyVxEvbHgUmIhKVlND+j02kA==" + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=" }, "arr-flatten": { "version": "1.1.0", @@ -908,7 +908,7 @@ "arr-union": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", - "integrity": "sha512-sKpyeERZ02v1FeCZT8lrfJq5u6goHCtpTAzPwJYe7c8SPFOboNjNg1vz2L4VTn9T4PQxEx13TbXLmYUcS6Ug7Q==" + "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=" }, "array-buffer-byte-length": { "version": "1.0.1", @@ -941,7 +941,7 @@ "array-from": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/array-from/-/array-from-2.1.1.tgz", - "integrity": "sha512-GQTc6Uupx1FCavi5mPzBvVT7nEOeWMmUA9P95wpfpW1XwMSKs+KaymD5C2Up7KAUKg/mYwbsUYzdZWcoajlNZg==", + "integrity": "sha1-z+nYwmYoudxa7MYqn12PHzUsEZU=", "dev": true }, "array-includes": { @@ -973,7 +973,7 @@ "array-unique": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", - "integrity": "sha512-SleRWjh9JUud2wH1hPs9rZBZ33H6T9HOiL0uwGnGx9FpE6wKGyfWugmbkEOIs6qWrZhg0LWeLziLrEwQJhs5mQ==" + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=" }, "array.prototype.filter": { "version": "1.0.4", @@ -1033,13 +1033,13 @@ "arrify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", - "integrity": "sha512-3CYzex9M9FGQjCGMGyi6/31c8GJbgb0qGyrx5HWxPd0aCwh4cB2YjMb2Xf9UuoogrMrlO9cTqnB5rI5GHZTcUA==", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", "dev": true }, "asap": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", - "integrity": "sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA==", + "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=", "dev": true }, "asn1": { @@ -1100,13 +1100,13 @@ "assert-plus": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", - "integrity": "sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw==", + "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", "dev": true }, "assign-symbols": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", - "integrity": "sha512-Q+JC7Whu8HhmTdBph/Tq59IoRtoy6KAm5zzPv00WdujX82lbAL8K7WVjne7vdCsAmbF4AYaDOPyO3k0kl8qIrw==" + "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=" }, "ast-types": { "version": "0.9.6", @@ -1144,7 +1144,7 @@ "asynckit": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", "dev": true }, "atob": { @@ -1178,7 +1178,7 @@ "aws-sign2": { "version": "0.7.0", "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", - "integrity": "sha512-08kcGqnYf/YmjoRhfxyu+CLxBjUtHLXLXX/vUfx9l2LYzG3c1m61nrpyFUZI6zeS+Li/wWMMidD9KgrqtGq3mA==", + "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", "dev": true }, "aws4": { @@ -1190,7 +1190,7 @@ "axios": { "version": "0.15.3", "resolved": "https://registry.npmjs.org/axios/-/axios-0.15.3.tgz", - "integrity": "sha512-w3/VNaraEcDri16lbemQWQGKfaFk9O0IZkzKlLeF5r6WWDv9TkcXkP+MWkRK8FbxwfozY/liI+qtvhV295t3HQ==", + "integrity": "sha1-LJ1jiy4ZGgjqHWzJiOrda6W9wFM=", "requires": { "follow-redirects": "1.0.0" } @@ -1198,7 +1198,7 @@ "babel-code-frame": { "version": "6.26.0", "resolved": "https://registry.npmjs.org/babel-code-frame/-/babel-code-frame-6.26.0.tgz", - "integrity": "sha512-XqYMR2dfdGMW+hd0IUZ2PwK+fGeFkOxZJ0wY+JaQAHzt1Zx8LcvpiZD2NiGkEG8qx0CfkAOr5xt76d1e8vG90g==", + "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", "dev": true, "requires": { "chalk": "^1.1.3", @@ -1209,19 +1209,19 @@ "ansi-regex": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", "dev": true }, "ansi-styles": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", - "integrity": "sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA==", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", "dev": true }, "chalk": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", - "integrity": "sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A==", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", "dev": true, "requires": { "ansi-styles": "^2.2.1", @@ -1234,13 +1234,13 @@ "js-tokens": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz", - "integrity": "sha512-RjTcuD4xjtthQkaWH7dFlH85L+QaVtSoOyGdZ3g6HFhS9dFNDfLyqgm2NFe2X6cQpeFmt0452FJjFG5UameExg==", + "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls=", "dev": true }, "strip-ansi": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", - "integrity": "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg==", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", "dev": true, "requires": { "ansi-regex": "^2.0.0" @@ -1249,7 +1249,7 @@ "supports-color": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", - "integrity": "sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g==", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", "dev": true } } @@ -1257,7 +1257,7 @@ "babel-runtime": { "version": "6.26.0", "resolved": "https://registry.npmjs.org/babel-runtime/-/babel-runtime-6.26.0.tgz", - "integrity": "sha512-ITKNuq2wKlW1fJg9sSW52eepoYgZBggvOAHC0u/CYu/qxQ9EVzThCgR69BnSXLHjy2f7SY5zaQ4yt7H9ZVxY2g==", + "integrity": "sha1-llxwWGaOgrVde/4E/yM3vItWR/4=", "requires": { "core-js": "^2.4.0", "regenerator-runtime": "^0.11.0" @@ -1286,7 +1286,7 @@ "define-property": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", "requires": { "is-descriptor": "^1.0.0" } @@ -1317,7 +1317,7 @@ "bcrypt-pbkdf": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", - "integrity": "sha512-qeFIXtP4MSoi6NLqO12WfqARWWuCKi2Rn/9hJLEmtB5yTNr9DqFWkJRCf2qShWzPeAMRnOgCrq0sg/KLv5ES9w==", + "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", "dev": true, "requires": { "tweetnacl": "^0.14.3" @@ -1433,7 +1433,7 @@ "boolbase": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", - "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==", + "integrity": "sha1-aN/1++YMUes3cl6p4+0xDcwed24=", "dev": true }, "bootstrap": { @@ -1472,7 +1472,7 @@ "extend-shallow": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "requires": { "is-extendable": "^0.1.0" } @@ -1625,7 +1625,7 @@ "builtin-modules": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", - "integrity": "sha512-wxXCdllwGhI2kCC0MnvTGYTMvnVZTvqgypkiTI8Pa5tcz2i6VqsqwYGgqwXji+4RgCzms6EajE4IxiUH6HH8nQ==", + "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", "dev": true }, "builtin-status-codes": { @@ -1672,7 +1672,7 @@ "caller-path": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-0.1.0.tgz", - "integrity": "sha512-UJiE1otjXPF5/x+T3zTnSFiTOEmJoGTD9HmBoxnCUwho61a2eSNn/VwtwuIBDAo2SEOv1AJ7ARI5gCmohFLu/g==", + "integrity": "sha1-lAhe9jWB7NPaqSREqP6U6CV3dR8=", "dev": true, "requires": { "callsites": "^0.2.0" @@ -1681,7 +1681,7 @@ "callsites": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-0.2.0.tgz", - "integrity": "sha512-Zv4Dns9IbXXmPkgRRUjAaJQgfN4xX5p6+RQFhWUqscdvvK2xK/ZL8b3IXIJsj+4sD+f24NwnWy2BY8AJ82JB0A==", + "integrity": "sha1-r6uWJikQp/M8GaV3WCXGnzTjUMo=", "dev": true } } @@ -1739,7 +1739,7 @@ "caseless": { "version": "0.12.0", "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", - "integrity": "sha512-4tYFyifaFfGacoiObjJegolkwSU4xQNGbVgUiNYVUxbQ2x2lUsFvY4hVgVzGiIe6WLOPqycWXA40l+PWsxthUw==", + "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", "dev": true }, "center-align": { @@ -1991,7 +1991,7 @@ "define-property": { "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", "requires": { "is-descriptor": "^0.1.0" } @@ -2102,7 +2102,7 @@ "co": { "version": "4.6.0", "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", - "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=", "dev": true }, "coa": { @@ -2129,7 +2129,7 @@ "collection-visit": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", - "integrity": "sha512-lNkKvzEeMBBjUGHZ+q6z9pSJla0KWAQPvtzhEV9+iGyQYG+pBpl7xKDhxoNSOZH2hhv0v5k0y2yAM4o4SjoSkw==", + "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", "requires": { "map-visit": "^1.0.0", "object-visit": "^1.0.0" @@ -2157,7 +2157,7 @@ "color-name": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==" + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" }, "color-string": { "version": "0.3.0", @@ -2232,7 +2232,7 @@ "concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", "dev": true }, "concat-stream": { @@ -2369,7 +2369,7 @@ "copy-descriptor": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", - "integrity": "sha512-XgZ0pFcakEUlbwQEVNg3+QAis1FyTL3Qel9FYy8pSkQqoG3PNoT0bOCQtOXcOkur21r2Eq2kI+IE+gsmAEVlYw==" + "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=" }, "core-js": { "version": "2.6.12", @@ -2646,7 +2646,7 @@ "custom-event": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/custom-event/-/custom-event-1.0.1.tgz", - "integrity": "sha512-GAj5FOq0Hd+RsCGVJxZuKaIDXDf3h6GQoNEjFgbLLI/trgtavwUbSnZ5pVfg27DVCaWjIohryS0JFwIJyT2cMg==", + "integrity": "sha1-XQKkaFCt8bSjF5RqOSj8y1v9BCU=", "dev": true }, "d": { @@ -2668,7 +2668,7 @@ "dashdash": { "version": "1.14.1", "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", - "integrity": "sha512-jRFi8UDGo6j+odZiEpjazZaWqEal3w/basFjQHQEwVtZJGDpxbH1MeYluwCS8Xq5wmLJooDlMgvVarmWfGM44g==", + "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", "dev": true, "requires": { "assert-plus": "^1.0.0" @@ -2748,7 +2748,7 @@ "deep-diff": { "version": "0.3.8", "resolved": "https://registry.npmjs.org/deep-diff/-/deep-diff-0.3.8.tgz", - "integrity": "sha512-yVn6RZmHiGnxRKR9sJb3iVV2XTF1Ghh2DiWRZ3dMnGc43yUdWWF/kX6lQyk3+P84iprfWKU/8zFTrlkvtFm1ug==", + "integrity": "sha1-wB3mPvsO7JeYgB1Ax+Da4ltYLIQ=", "dev": true }, "deep-equal": { @@ -2883,7 +2883,7 @@ "delayed-stream": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", "dev": true }, "delegates": { @@ -2923,7 +2923,7 @@ "di": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/di/-/di-0.0.1.tgz", - "integrity": "sha512-uJaamHkagcZtHPqCIHZxnFrXlunQXgBOsZSUOWwFw31QJCAbyTBoHMW75YOTur5ZNx8pIeAKgf6GWIgaqqiLhA==", + "integrity": "sha1-gGZJMmzqp8qjMG112YXqJ0i6kTw=", "dev": true }, "diff": { @@ -2969,7 +2969,7 @@ "discontinuous-range": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/discontinuous-range/-/discontinuous-range-1.0.0.tgz", - "integrity": "sha512-c68LpLbO+7kP/b1Hr1qs8/BJ09F5khZGTxqxZuhzxpmwJKOgRFHJWIb9/KmqnqHhLdO55aOxFH/EGBvUQbL/RQ==", + "integrity": "sha1-44Mx8IRLukm5qctxx3FYWqsbxlo=", "dev": true }, "dns-equal": { @@ -3028,7 +3028,7 @@ "dom-serialize": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/dom-serialize/-/dom-serialize-2.2.1.tgz", - "integrity": "sha512-Yra4DbvoW7/Z6LBN560ZwXMjoNOSAN2wRsKFGc4iBeso+mpIA6qj1vfdf9HpMaKAqG6wXTy+1SYEzmNpKXOSsQ==", + "integrity": "sha1-ViromZ9Evl6jB29UGdzVnrQ6yVs=", "dev": true, "requires": { "custom-event": "~1.0.0", @@ -3156,7 +3156,7 @@ "ecc-jsbn": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", - "integrity": "sha512-eh9O+hwRHNbG4BLTjEl3nw044CkGm5X6LoaCf7LPp7UU8Qrt47JYNi6nPX8xjW97TKGKm1ouctg0QSpZe9qrnw==", + "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", "dev": true, "requires": { "jsbn": "~0.1.0", @@ -3184,7 +3184,7 @@ "ee-first": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", - "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=", "dev": true }, "electron-to-chromium": { @@ -3230,7 +3230,7 @@ "encodeurl": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", - "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", "dev": true }, "encoding": { @@ -3294,7 +3294,7 @@ "enhanced-resolve": { "version": "3.4.1", "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-3.4.1.tgz", - "integrity": "sha512-ZaAux1rigq1e2nQrztHn4h2ugvpzZxs64qneNah+8Mh/K0CRqJFJc+UoXnUsq+1yX+DmQFPPdVqboKAJ89e0Iw==", + "integrity": "sha1-BCHjOf1xQZs9oT0Smzl5BAIwR24=", "requires": { "graceful-fs": "^4.1.2", "memory-fs": "^0.4.0", @@ -3305,7 +3305,7 @@ "ent": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/ent/-/ent-2.2.0.tgz", - "integrity": "sha512-GHrMyVZQWvTIdDtpiEXdHZnFQKzeO09apj8Cbl4pKWy4i0Oprcq17usfDt5aO63swf0JOeMWjWQE/LzgSRuWpA==", + "integrity": "sha1-6WQhkyWiHQX0RGai9obtbOX13R0=", "dev": true }, "entities": { @@ -3624,13 +3624,13 @@ "escape-html": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", - "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=", "dev": true }, "escape-string-regexp": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==" + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=" }, "escope": { "version": "3.6.0", @@ -3733,7 +3733,7 @@ "eslint-plugin-jasmine": { "version": "2.10.1", "resolved": "https://registry.npmjs.org/eslint-plugin-jasmine/-/eslint-plugin-jasmine-2.10.1.tgz", - "integrity": "sha512-dF2siVCguzZpEkqgRaJdR+dsBbXEQKog2tq7A0jYPHK+3qSD+E92f+Sb1jY5y4ua0j18FVIBzEm0yEBID/RdmQ==", + "integrity": "sha1-VzO3CedR9LxA4x4cFpib0s377Jc=", "dev": true }, "eslint-plugin-react": { @@ -3980,13 +3980,13 @@ "exit": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", - "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=", "dev": true }, "expand-brackets": { "version": "2.1.4", "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", - "integrity": "sha512-w/ozOKR9Obk3qoWeY/WDi6MFta9AoMR+zud60mdnbniMcBxRuFJyDt2LdX/14A1UABeqk+Uk+LDfUpvoGKppZA==", + "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", "requires": { "debug": "^2.3.3", "define-property": "^0.2.5", @@ -4000,7 +4000,7 @@ "define-property": { "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", "requires": { "is-descriptor": "^0.1.0" } @@ -4008,7 +4008,7 @@ "extend-shallow": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "requires": { "is-extendable": "^0.1.0" } @@ -4158,7 +4158,7 @@ "extend-shallow": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", - "integrity": "sha512-BwY5b5Ql4+qZoefgMj2NUmx+tehVTH/Kf4k1ZEtOHNFcm2wSxMRo992l6X3TIgni2eZVTZ85xMOjF31fwZAj6Q==", + "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", "requires": { "assign-symbols": "^1.0.0", "is-extendable": "^1.0.1" @@ -4214,7 +4214,7 @@ "define-property": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", "requires": { "is-descriptor": "^1.0.0" } @@ -4222,7 +4222,7 @@ "extend-shallow": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "requires": { "is-extendable": "^0.1.0" } @@ -4264,7 +4264,7 @@ "extsprintf": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha512-11Ndz7Nv+mvAC1j0ktTa7fAb0vLyGGX+rMHNBYQviQDGU0Hw7lhctJANqbPhu9nV9/izT/IntTgZ7Im/9LJs9g==", + "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", "dev": true }, "fast-deep-equal": { @@ -4340,7 +4340,7 @@ "fast-levenshtein": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", "dev": true }, "fastparse": { @@ -4385,7 +4385,7 @@ "core-js": { "version": "1.2.7", "resolved": "https://registry.npmjs.org/core-js/-/core-js-1.2.7.tgz", - "integrity": "sha512-ZiPp9pZlgxpWRu0M+YWbm6+aQ84XEfH1JRXvfOc/fILWI0VKhLC2LX13X1NYq4fULzLMq7Hfh43CSo2/aIaUPA==", + "integrity": "sha1-ZSKUwUZR2yj6k70tX/KYOk8IxjY=", "dev": true } } @@ -4464,7 +4464,7 @@ "fill-range": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", - "integrity": "sha512-VcpLTWqWDiTerugjj8e3+esbg+skS3M9e54UuR3iCeIDMXCLTsAH8hTSzDQU/X6/6t3eYkOKoZSef2PlU6U1XQ==", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", "requires": { "extend-shallow": "^2.0.1", "is-number": "^3.0.0", @@ -4475,7 +4475,7 @@ "extend-shallow": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "requires": { "is-extendable": "^0.1.0" } @@ -4580,7 +4580,7 @@ "follow-redirects": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.0.0.tgz", - "integrity": "sha512-7s+wBk4z5xTwVJuozRBAyRofWKjD3uG2CUjZfZTrw9f+f+z8ZSxOjAqfIDLtc0Hnz+wGK2Y8qd93nGGjXBYKsQ==", + "integrity": "sha1-jjQpjL0uF28lTv/sdaHHjMhJ/Tc=", "requires": { "debug": "^2.2.0" } @@ -4597,7 +4597,7 @@ "for-in": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", - "integrity": "sha512-7EwmXrOjyL+ChxMhmG5lnW9MPt1aIeZEwKhQzoBUdTV0N3zuwWDZYVJatDvZ2OyzPUvdIAZDsCetk3coyMfcnQ==" + "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=" }, "for-own": { "version": "1.0.0", @@ -4611,7 +4611,7 @@ "forever-agent": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", - "integrity": "sha512-j0KLYPhm6zeac4lz3oJ3o65qvgQCcPubiyotZrXqEaG4hNagNYO8qdlUrX5vwqv9ohqeT/Z3j6+yW067yWWdUw==", + "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", "dev": true }, "form-data": { @@ -4634,7 +4634,7 @@ "fragment-cache": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", - "integrity": "sha512-GMBAbW9antB8iZRHLoGw0b3HANt57diZYFO/HL1JGIC1MjKrdmhxvrJbupnVvpys0zsz7yBApXdQyfepKly2kA==", + "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", "requires": { "map-cache": "^0.2.2" } @@ -4648,7 +4648,7 @@ "fs-access": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/fs-access/-/fs-access-1.0.1.tgz", - "integrity": "sha512-05cXDIwNbFaoFWaz5gNHlUTbH5whiss/hr/ibzPd4MH3cR4w0ZKeIPiVdbyJurg3O5r/Bjpvn9KOb1/rPMf3nA==", + "integrity": "sha1-1qh/JiJxzv6+wwxVNAf7mV2od3o=", "dev": true, "requires": { "null-check": "^1.0.0" @@ -4668,7 +4668,7 @@ "fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", "dev": true }, "fsevents": { @@ -4710,7 +4710,7 @@ "functional-red-black-tree": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", - "integrity": "sha512-dsKNQNdj6xA3T+QlADDA7mOSlX0qiMINjn0cgr+eGHGsbSHzTabcIogz2+p/iqP1Xs6EP/sS2SbqH+brGTbq0g==", + "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", "dev": true }, "functions-have-names": { @@ -4826,7 +4826,7 @@ "get-value": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", - "integrity": "sha512-Ln0UQDlxH1BapMu3GPtf7CuYNwRZf2gwCuPqbyG6pB8WfmFpzqcy4xtAaAMUhnNqjMKTiCPZG2oMT3YSx8U2NA==" + "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=" }, "getobject": { "version": "0.1.0", @@ -4837,7 +4837,7 @@ "getpass": { "version": "0.1.7", "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", - "integrity": "sha512-0fzj9JxOLfJ+XGLhR8ze3unN0KZCgZwiSSDz168VERjK8Wl8kVSdcu2kspd4s4wtAa1y/qrVRiAA0WclVsu0ng==", + "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", "dev": true, "requires": { "assert-plus": "^1.0.0" @@ -5476,7 +5476,7 @@ "har-schema": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", - "integrity": "sha512-Oqluz6zhGX8cyRaTQlFMPw80bSJVG2x/cFb8ZPhUILGgHka9SsokCCOQgpveePerqidZOrT14ipqfJb7ILcW5Q==", + "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", "dev": true }, "har-validator": { @@ -5498,7 +5498,7 @@ "has-ansi": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", - "integrity": "sha512-C8vBJ8DwUCx19vhm7urhTuUsr4/IyP6l4VzNQDv+ryHQObW3TTTp9yB68WpYgRe2bbaGuZ/se74IqFeVnMnLZg==", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", "dev": true, "requires": { "ansi-regex": "^2.0.0" @@ -5507,7 +5507,7 @@ "ansi-regex": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA==", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", "dev": true } } @@ -5527,7 +5527,7 @@ "has-flag": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==" + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" }, "has-property-descriptors": { "version": "1.0.2", @@ -5568,7 +5568,7 @@ "has-value": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", - "integrity": "sha512-IBXk4GTsLYdQ7Rvt+GRBrFSVEkmuOUy4re0Xjd9kJSUQpnTrWR4/y9RpfexN9vkAPMFuQoeWKwqzPozRTlasGw==", + "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", "requires": { "get-value": "^2.0.6", "has-values": "^1.0.0", @@ -5578,7 +5578,7 @@ "has-values": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", - "integrity": "sha512-ODYZC64uqzmtfGMEAX/FvZiRyWLpAC3vYnNunURUnkGVTS+mI0smVsWaPydRBsE3g+ok7h960jChO8mFcWlHaQ==", + "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", "requires": { "is-number": "^3.0.0", "kind-of": "^4.0.0" @@ -5587,7 +5587,7 @@ "kind-of": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", - "integrity": "sha512-24XsCxmEbRwEDbz/qz3stgin8TTzZ1ESR56OMCN0ujYg+vRutNSiOj9bHH9u85DKgXguraugV5sFuvbD4FW/hw==", + "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", "requires": { "is-buffer": "^1.1.5" } @@ -6030,7 +6030,7 @@ "http-signature": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", - "integrity": "sha512-CAbnr6Rz4CYQkLYUtSNXxQPUH2gK8f3iWexVlsnMeD+GjlsQ0Xsy1cOX+mN3dtxYomRy21CiOzU8Uhw6OwncEQ==", + "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", "dev": true, "requires": { "assert-plus": "^1.0.0", @@ -6128,7 +6128,7 @@ "imurmurhash": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", "dev": true }, "in-publish": { @@ -6155,7 +6155,7 @@ "inflight": { "version": "1.0.6", "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", "dev": true, "requires": { "once": "^1.3.0", @@ -6456,12 +6456,12 @@ "is-extendable": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", - "integrity": "sha512-5BMULNob1vgFX6EjQw5izWDxrecWK9AM72rugNr0TFldMOi0fj6Jk+zeKIt0xGj4cEfQIJth4w3OKWOJ4f+AFw==" + "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=" }, "is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", "dev": true }, "is-finite": { @@ -6503,7 +6503,7 @@ "is-number": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", - "integrity": "sha512-4cboCqIpliH+mAvFNegjZQ4kgKc3ZUhQVr3HvWbSh5q3WH2v82ct+T2Y1hdU5Gdtorx/cLifQjqCbL7bpznLTg==", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", "requires": { "kind-of": "^3.0.2" }, @@ -6511,7 +6511,7 @@ "kind-of": { "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", "requires": { "is-buffer": "^1.1.5" } @@ -6605,7 +6605,7 @@ "is-stream": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", - "integrity": "sha512-uQPm8kcs47jx38atAcWTVxyltQYoPT68y9aWYdV6yWXSyW8mzSat0TL6CiWdZeCdF3KrAvpVtnHbTv4RN+rqdQ==", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", "dev": true }, "is-string": { @@ -6620,7 +6620,7 @@ "is-subset": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/is-subset/-/is-subset-0.1.1.tgz", - "integrity": "sha512-6Ybun0IkarhmEqxXCNw/C0bna6Zb/TkfUX9UbwJtK6ObwAVCxmAP308WWTHviM/zAqXk05cdhYsUsZeGQh99iw==", + "integrity": "sha1-ilkRfZMt4d4A8kX83TnOQ/HpOaY=", "dev": true }, "is-svg": { @@ -6653,7 +6653,7 @@ "is-typedarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", - "integrity": "sha512-cyA56iCMHAh5CdzjJIa4aohJyeO1YbwLi3Jc35MmRU6poroFjIGZzUzupGiRPOjgHg9TLu43xbpwXk523fMxKA==", + "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", "dev": true }, "is-upper-case": { @@ -6694,7 +6694,7 @@ "isarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==" + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" }, "isbinaryfile": { "version": "4.0.10", @@ -6705,18 +6705,18 @@ "isexe": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", "dev": true }, "isobject": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", - "integrity": "sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==" + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=" }, "isomorphic-fetch": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/isomorphic-fetch/-/isomorphic-fetch-2.2.1.tgz", - "integrity": "sha512-9c4TNAKYXM5PRyVcwUZrF3W09nQ+sO7+jydgs4ZGW9dhsLG2VOlISJABombdQqQRXCwuYG3sYV/puGf5rp0qmA==", + "integrity": "sha1-YRrhrPFPXoH3KVB0coGf6XM1WKk=", "dev": true, "requires": { "node-fetch": "^1.0.1", @@ -6726,13 +6726,13 @@ "isstream": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", - "integrity": "sha512-Yljz7ffyPbrLpLngrMtZ7NduUgVvi6wG9RJ9IUcyCd59YQ911PBJphODUcbOVbqYfxe1wuYf/LJ8PauMRwsM/g==", + "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", "dev": true }, "jasmine": { "version": "2.99.0", "resolved": "https://registry.npmjs.org/jasmine/-/jasmine-2.99.0.tgz", - "integrity": "sha512-kmuDC+6c9tA8BAZGd7wmucWKrM/aCCBSzCJEqRghvw9lKValw+pg88aN/BFIikmZwRTD57QmHamQ2wRpKb3FDQ==", + "integrity": "sha1-jKctEC5jm4Z8ZImFbg4YqceqQrc=", "dev": true, "requires": { "exit": "^0.1.2", @@ -6743,7 +6743,7 @@ "jasmine-core": { "version": "2.99.1", "resolved": "https://registry.npmjs.org/jasmine-core/-/jasmine-core-2.99.1.tgz", - "integrity": "sha512-ra97U4qu3OCcIxvN6eg3kyy8bLrID/TgxafSGMMICg3SFx5C/sUfDPpiOh7yoIsHdtjrOVdtT9rieYhqOsh9Ww==", + "integrity": "sha1-5kAN8ea1bhMLYcS80JPap/boyhU=", "dev": true }, "jasmine-reporters": { @@ -6817,7 +6817,7 @@ "jsbn": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", - "integrity": "sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg==", + "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", "dev": true }, "json-loader": { @@ -6850,13 +6850,13 @@ "json-stable-stringify-without-jsonify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", - "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", "dev": true }, "json-stringify-safe": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", - "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==", + "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", "dev": true }, "json3": { @@ -7072,13 +7072,13 @@ "karma-jasmine": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/karma-jasmine/-/karma-jasmine-1.1.2.tgz", - "integrity": "sha512-SENGE9DhlIIFTSZWiNq4eGeXL8G6z9cqHIOdkx9jh1qhhQqwEy3tAoLRyER0vOcHqdOlKmGpOuXk+HOipIy7sg==", + "integrity": "sha1-OU8rJf+0pkS5rabyLUQ+L9CIhsM=", "dev": true }, "karma-junit-reporter": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/karma-junit-reporter/-/karma-junit-reporter-1.2.0.tgz", - "integrity": "sha512-FeuLOKlXNtJhIQK3oQASbO5QOib762CEHV8+L9wwTQpiZJgp7xKg3sNno66rL5bQPV2soG6fJdAFWqqnMJuh2w==", + "integrity": "sha1-T5xAzt+xo5X4rvh2q/lhiZF8Y5Y=", "dev": true, "requires": { "path-is-absolute": "^1.0.0", @@ -7097,7 +7097,7 @@ "karma-verbose-reporter": { "version": "0.0.6", "resolved": "https://registry.npmjs.org/karma-verbose-reporter/-/karma-verbose-reporter-0.0.6.tgz", - "integrity": "sha512-xBkQTj7JUByofR1sounBEL0sVKTCBeoiPEMtNBU8Dr/Gtqt6W6XUq2hj36r+EKYHQPPJch7moINsYUWO8moWXQ==", + "integrity": "sha1-WQkFJFHGB/Aqx3x2N5Gi/hJRJgw=", "dev": true, "requires": { "colors": ">=1.0" @@ -7170,7 +7170,7 @@ "levn": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", - "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==", + "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", "dev": true, "requires": { "prelude-ls": "~1.1.2", @@ -7326,7 +7326,7 @@ "lodash.debounce": { "version": "4.0.8", "resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz", - "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==" + "integrity": "sha1-gteb/zCmfEAF/9XiUVMArZyk168=" }, "lodash.defaults": { "version": "4.2.0", @@ -7337,7 +7337,7 @@ "lodash.escape": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/lodash.escape/-/lodash.escape-4.0.1.tgz", - "integrity": "sha512-nXEOnb/jK9g0DYMr1/Xvq6l5xMD7GDG55+GSYIYmS0G4tBk/hURD4JR9WCavs04t33WmJx9kCyp9vJ+mr4BOUw==", + "integrity": "sha1-yQRGkMIeBClL6qUXcS/e0fqI3pg=", "dev": true }, "lodash.filter": { @@ -7355,7 +7355,7 @@ "lodash.flattendeep": { "version": "4.4.0", "resolved": "https://registry.npmjs.org/lodash.flattendeep/-/lodash.flattendeep-4.4.0.tgz", - "integrity": "sha512-uHaJFihxmJcEX3kT4I23ABqKKalJ/zDrDg0lsFtc1h+3uw49SIJ5beyhx5ExVRti3AvKoOJngIj7xz3oylPdWQ==", + "integrity": "sha1-+wMJF/hqMTTlvJvsDWngAT3f7bI=", "dev": true }, "lodash.foreach": { @@ -7367,13 +7367,13 @@ "lodash.get": { "version": "4.4.2", "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz", - "integrity": "sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ==", + "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=", "dev": true }, "lodash.isequal": { "version": "4.5.0", "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", - "integrity": "sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ==", + "integrity": "sha1-QVxEePK8wwEgwizhDtMib30+GOA=", "dev": true }, "lodash.map": { @@ -7427,7 +7427,7 @@ "lodash.unescape": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/lodash.unescape/-/lodash.unescape-4.0.1.tgz", - "integrity": "sha512-DhhGRshNS1aX6s5YdBE3njCCouPgnG29ebyHvImlZzXZf2SHgt+J08DHgytTPnpywNbO1Y8mNUFyQuIDBq2JZg==", + "integrity": "sha1-vyJJiGzlFM2hEvrpIYzcBlIR/Jw=", "dev": true }, "lodash.uniq": { @@ -7547,7 +7547,7 @@ "map-cache": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", - "integrity": "sha512-8y/eV9QQZCiyn1SprXSrCmqJN0yNRATe+PO8ztwqrvrbdRLA3eYJF0yaR0YayLWkMbsQSKWS9N2gPcGEc4UsZg==" + "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=" }, "map-obj": { "version": "1.0.1", @@ -7558,7 +7558,7 @@ "map-visit": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", - "integrity": "sha512-4y7uGv8bd2WdM9vpQsiQNo41Ln1NvhvDRuVt0k2JZQ+ezN2uaQes7lZeZ+QQUHOLQAtDaBJ+7wCbi+ab/KFs+w==", + "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", "requires": { "object-visit": "^1.0.0" } @@ -7589,7 +7589,7 @@ "media-typer": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", - "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=", "dev": true }, "mem": { @@ -7618,7 +7618,7 @@ "memory-fs": { "version": "0.4.1", "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", - "integrity": "sha512-cda4JKCxReDXFXRqOHPQscuIYg1PvxbE2S2GP45rnwfEK+vZaXC8C1OFvdHIbgw0DLzowXGVoxLaAmlgRy14GQ==", + "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", "requires": { "errno": "^0.1.3", "readable-stream": "^2.0.1" @@ -7825,7 +7825,7 @@ "ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" }, "multicast-dns": { "version": "6.2.3", @@ -7899,7 +7899,7 @@ "natural-compare": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", - "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", "dev": true }, "ncname": { @@ -8175,7 +8175,7 @@ "nominatim-browser": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/nominatim-browser/-/nominatim-browser-2.0.2.tgz", - "integrity": "sha512-UAlFWEC75GLTclpJIubspUkX9pzCmoz5axqH+6b1yNh+tZ7s9jtRPY45W6Mjm6ROj3/1pEyi+hHc1siZfANmOQ==", + "integrity": "sha1-obJ9nkmUBKRvgKz8l+NXAQPR+AI=", "requires": { "axios": "^0.15.3", "bluebird": "^3.3.5" @@ -8294,7 +8294,7 @@ "null-check": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/null-check/-/null-check-1.0.0.tgz", - "integrity": "sha512-j8ZNHg19TyIQOWCGeeQJBuu6xZYIEurf8M1Qsfd8mFrGEfIZytbw18YjKWg+LcO25NowXGZXZpKAx+Ui3TFfDw==", + "integrity": "sha1-l33/1xdgErnsMNKjnbXPcqBDnt0=", "dev": true }, "num2fraction": { @@ -8323,7 +8323,7 @@ "object-copy": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", - "integrity": "sha512-79LYn6VAb63zgtmAteVOWo9Vdj71ZVBy3Pbse+VqxDpEP83XuujMrGqHIwAXJ5I/aM0zU7dIyIAhifVTPrNItQ==", + "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", "requires": { "copy-descriptor": "^0.1.0", "define-property": "^0.2.5", @@ -8333,7 +8333,7 @@ "define-property": { "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", "requires": { "is-descriptor": "^0.1.0" } @@ -8341,7 +8341,7 @@ "kind-of": { "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", "requires": { "is-buffer": "^1.1.5" } @@ -8373,7 +8373,7 @@ "object-visit": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", - "integrity": "sha512-GBaMwwAVK9qbQN3Scdo0OyvgPW7l3lnaVMj84uTOZlswkX0KpF6fyDBJhtTthf7pymztoN36/KEr1DyhF96zEA==", + "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", "requires": { "isobject": "^3.0.0" } @@ -8437,7 +8437,7 @@ "object.pick": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", - "integrity": "sha512-tqa/UMy/CCoYmj+H5qc07qvSL9dqcs/WZENZ1JbtWBlATP+iVOe778gE6MSijnyCnORzDuX6hU+LA4SZ09YjFQ==", + "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", "requires": { "isobject": "^3.0.1" } @@ -8477,7 +8477,7 @@ "once": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", "dev": true, "requires": { "wrappy": "1" @@ -8548,7 +8548,7 @@ "os-tmpdir": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", - "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", "dev": true }, "osenv": { @@ -8734,7 +8734,7 @@ "pascalcase": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", - "integrity": "sha512-XHXfu/yOQRy9vYOtUDVMN60OEJjW013GoObG1o+xwQTpB9eYJX/BjXMsdW13ZDPruFhYYn0AG22w0xgQMwl3Nw==" + "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=" }, "patch-package": { "version": "6.2.2", @@ -8793,7 +8793,7 @@ "path-dirname": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", - "integrity": "sha512-ALzNPpyNq9AqXMBjeymIjFDAkAFH06mHJH/cSBHAgU0s4vfpBn6b2nf8tiRLvagKD8RbTpq2FKTBg7cl9l3c7Q==", + "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", "dev": true }, "path-exists": { @@ -8808,19 +8808,19 @@ "path-is-absolute": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", "dev": true }, "path-is-inside": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", - "integrity": "sha512-DUWJr3+ULp4zXmol/SZkFf3JGsS9/SIv+Y3Rt93/UjPpDpklB5f1er4O3POIbUuUJ3FXgqte2Q7SrU6zAqwk8w==", + "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", "dev": true }, "path-key": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", - "integrity": "sha512-fEHGKCSmUSDPv4uoj8AlD+joPlq3peND+HRYyxFz4KPw4z926S/b8rIuFs2FYJg3BwsxJf6A9/3eIdLaYC+9Dw==", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", "dev": true }, "path-parse": { @@ -8841,7 +8841,7 @@ "isarray": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", - "integrity": "sha512-D2S+3GLxWH+uhrNEcoh/fnmYeP8E8/zHl644d/jdA0g2uyXvy3sb0qxotE+ne0LtccHknQzWwZEzhak7oJ0COQ==", + "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=", "dev": true } } @@ -8868,7 +8868,7 @@ "performance-now": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", - "integrity": "sha512-7EAHlyLHI56VEIdK57uwHdHKIaAGbnXPiw0yWbarQZOKaKpvUIgW0jWRVLiatnM+XXlSwsanIBH/hzGMJulMow==", + "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", "dev": true }, "picocolors": { @@ -8892,13 +8892,13 @@ "pinkie": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", - "integrity": "sha512-MnUuEycAemtSaeFSjXKW/aroV7akBbY+Sv+RkyqFjgAe73F+MR0TBWKBRDkmfWq/HiFmdavfZ1G7h4SPZXaCSg==", + "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", "dev": true }, "pinkie-promise": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", - "integrity": "sha512-0Gni6D4UcLTbv9c57DfxDGdr41XfgUjqWZu492f0cIGr16zDU06BWP/RAEvOuo7CQ0CNjHaLlM59YJJFm3NWlw==", + "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", "dev": true, "requires": { "pinkie": "^2.0.0" @@ -8998,7 +8998,7 @@ "posix-character-classes": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", - "integrity": "sha512-xTgYBc3fuo7Yt7JbiuFxSYGToMoz8fLoE6TC9Wx1P/u+LfeThMOAqmuyECnlBaaJb+u1m9hHiXUEtwW4OzfUJg==" + "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=" }, "possible-typed-array-names": { "version": "1.0.0", @@ -9487,7 +9487,7 @@ "prelude-ls": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", - "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==", + "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", "dev": true }, "prepend-http": { @@ -9605,12 +9605,12 @@ "prr": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", - "integrity": "sha512-yPw4Sng1gWghHQWj0B3ZggWUm4qVbPwPFcRG8KyxiU7J2OHFSoEHKS+EZ3fv5l1t9CyCiop6l/ZYeWbrgoQejw==" + "integrity": "sha1-0/wRS6BplaRexok/SEzrHXj19HY=" }, "pseudomap": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", - "integrity": "sha512-b/YwNhb8lk1Zz2+bXXpS/LK9OisiZZ1SNsSLxN1x2OXVEhW2Ckr/7mWE5vrC1ZTiJlD9g19jWszTmJsB+oEpFQ==", + "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=", "dev": true }, "psl": { @@ -9711,7 +9711,7 @@ "railroad-diagrams": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/railroad-diagrams/-/railroad-diagrams-1.0.0.tgz", - "integrity": "sha512-cz93DjNeLY0idrCNOH6PviZGRN9GJhsdm9hpn1YCS879fj4W+x5IFJhhkRZcwVgMmFF7R82UA/7Oh+R8lLZg6A==", + "integrity": "sha1-635iZ1SN3t+4mcG5Dlc3RVnN234=", "dev": true }, "randexp": { @@ -9977,7 +9977,7 @@ "react-prop-types": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/react-prop-types/-/react-prop-types-0.4.0.tgz", - "integrity": "sha512-IyjsJhDX9JkoOV9wlmLaS7z+oxYoIWhfzDcFy7inwoAKTu+VcVNrVpPmLeioJ94y6GeDRsnwarG1py5qofFQMg==", + "integrity": "sha1-+ZsL+0AGkpya8gUefBQUpcdbk9A=", "requires": { "warning": "^3.0.0" } @@ -10192,7 +10192,7 @@ "redux-logger": { "version": "3.0.6", "resolved": "https://registry.npmjs.org/redux-logger/-/redux-logger-3.0.6.tgz", - "integrity": "sha512-JoCIok7bg/XpqA1JqCqXFypuqBbQzGQySrhFzewB7ThcnysTO30l4VCst86AuB9T9tuT03MAA56Jw2PNhRSNCg==", + "integrity": "sha1-91VZZvMJjzyIYExEnPC69XeCdL8=", "dev": true, "requires": { "deep-diff": "^0.3.5" @@ -10207,7 +10207,7 @@ "reflect.ownkeys": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/reflect.ownkeys/-/reflect.ownkeys-0.2.0.tgz", - "integrity": "sha512-qOLsBKHCpSOFKK1NUOCGC5VyeufB6lEsFe92AL2bhIJsacZS1qdoOZSbPk3MYKuT2cFlRDnulKXuuElIrMjGUg==", + "integrity": "sha1-dJrO7H8/34tj+SegSAnpDFwLNGA=", "dev": true }, "regenerator-runtime": { @@ -10260,7 +10260,7 @@ "remove-trailing-separator": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", - "integrity": "sha512-/hS+Y0u3aOfIETiaiirUFwDBDzmXPvO+jAfKTitUngIPzdKc6Z0LoFjM/CK5PL4C+eKwHohlHAb6H0VFfmmUsw==", + "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", "dev": true }, "repeat-element": { @@ -10271,7 +10271,7 @@ "repeat-string": { "version": "1.6.1", "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", - "integrity": "sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==" + "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=" }, "repeating": { "version": "2.0.1", @@ -10333,7 +10333,7 @@ "require-uncached": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/require-uncached/-/require-uncached-1.0.3.tgz", - "integrity": "sha512-Xct+41K3twrbBHdxAgMoOS+cNcoqIjfM2/VxBF4LL2hVph7YsF8VSKyQ3BDFZwEVbok9yeDl2le/qo0S77WG2w==", + "integrity": "sha1-Tg1W1slmL9MeQwEcS5WqSZVUIdM=", "dev": true, "requires": { "caller-path": "^0.1.0", @@ -10343,7 +10343,7 @@ "resolve-from": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-1.0.1.tgz", - "integrity": "sha512-kT10v4dhrlLNcnO084hEjvXCI1wUG9qZLoz2RogxqDQQYy7IxjI/iMUkOtQTNEh6rzHxvdQWHsJyel1pKOVCxg==", + "integrity": "sha1-Jsv+k10a7uq7Kbw/5a6wHpPUQiY=", "dev": true } } @@ -10360,13 +10360,13 @@ "requires-port": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", - "integrity": "sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ==", + "integrity": "sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8=", "dev": true }, "reselect": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/reselect/-/reselect-3.0.1.tgz", - "integrity": "sha512-b/6tFZCmRhtBMa4xGqiiRp9jh9Aqi2A687Lo265cN0/QohJQEBPiQ52f4QB6i0eF3yp3hmLL21LSGBcML2dlxA==" + "integrity": "sha1-79qpjqdFEyTQkrKyFjpqHXqaIUc=" }, "resolve": { "version": "1.22.8", @@ -10422,7 +10422,7 @@ "resolve-url": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", - "integrity": "sha512-ZuF55hVUQaaczgOIwqWzkEcEidmlD/xl44x1UZnhOXcYuFN2S6+rcxpG+C1N3So0wvNI3DmJICUFfu2SxhBmvg==" + "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=" }, "restore-cursor": { "version": "3.1.0", @@ -10487,7 +10487,7 @@ "rst-selector-parser": { "version": "2.2.3", "resolved": "https://registry.npmjs.org/rst-selector-parser/-/rst-selector-parser-2.2.3.tgz", - "integrity": "sha512-nDG1rZeP6oFTLN6yNDV/uiAvs1+FS/KlrEwh7+y7dpuApDBy6bI2HTBcc0/V8lv9OTqfyD34eF7au2pm8aBbhA==", + "integrity": "sha1-gbIw6i/MYGbInjRy3nlChdmwPZE=", "dev": true, "requires": { "lodash.flattendeep": "^4.4.0", @@ -10512,13 +10512,13 @@ "rx-lite": { "version": "4.0.8", "resolved": "https://registry.npmjs.org/rx-lite/-/rx-lite-4.0.8.tgz", - "integrity": "sha512-Cun9QucwK6MIrp3mry/Y7hqD1oFqTYLQ4pGxaHTjIdaFDWRGGLikqp6u8LcWJnzpoALg9hap+JGk8sFIUuEGNA==", + "integrity": "sha1-Cx4Rr4vESDbwSmQH6S2kJGe3lEQ=", "dev": true }, "rx-lite-aggregates": { "version": "4.0.8", "resolved": "https://registry.npmjs.org/rx-lite-aggregates/-/rx-lite-aggregates-4.0.8.tgz", - "integrity": "sha512-3xPNZGW93oCjiO7PtKxRK6iOVYBWBvtf9QHDfU23Oc+dLIQmAV//UnyXV/yihv81VS/UqoQPk4NegS8EFi55Hg==", + "integrity": "sha1-dTuHqJoRyVRnxKwWJsTvxOBcZ74=", "dev": true, "requires": { "rx-lite": "*" @@ -10561,7 +10561,7 @@ "safe-regex": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", - "integrity": "sha512-aJXcif4xnaNUzvUuC5gcb46oTS7zvg4jpMTnuqtrEPlR3vFr4pxtdTwaF1Qs3Enjn9HK+ZlwQui+a7z0SywIzg==", + "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", "requires": { "ret": "~0.1.10" } @@ -10977,7 +10977,7 @@ "extend-shallow": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "requires": { "is-extendable": "^0.1.0" } @@ -10987,7 +10987,7 @@ "setimmediate": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", - "integrity": "sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==", + "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", "dev": true }, "setprototypeof": { @@ -11038,7 +11038,7 @@ "shebang-command": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", - "integrity": "sha512-EV3L1+UQWGor21OmnvojK36mhg+TyIKDh3iFBKBohr5xeXIhNBcx8oWdgkTEEQ+BEFFYdLRuqMfd5L84N1V5Vg==", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", "dev": true, "requires": { "shebang-regex": "^1.0.0" @@ -11047,7 +11047,7 @@ "shebang-regex": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", - "integrity": "sha512-wpoSFAxys6b2a2wHZ1XpDSgD7N9iVjg29Ph9uV/uaP9Ex/KXlkTZTeddxDPSYQpgvzKLGJke2UU0AzoGCjNIvQ==", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", "dev": true }, "shelljs": { @@ -11088,7 +11088,7 @@ "simulant": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/simulant/-/simulant-0.2.2.tgz", - "integrity": "sha512-YV1S5PNZp2BBwdmv32bVMsvWy8zycEViiGs+To9AUa4ic6x5Sv/7UhvXq2lXWobIlC7zDoMoULO9IDHNdi+3PQ==", + "integrity": "sha1-8bzlJxK2p6DaON392n6DsgsdoB4=", "dev": true }, "sinon": { @@ -11126,7 +11126,7 @@ "is-fullwidth-code-point": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", "dev": true } } @@ -11158,7 +11158,7 @@ "define-property": { "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", "requires": { "is-descriptor": "^0.1.0" } @@ -11166,7 +11166,7 @@ "extend-shallow": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug==", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", "requires": { "is-extendable": "^0.1.0" } @@ -11186,7 +11186,7 @@ "define-property": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha512-cZTYKFWspt9jZsMscWo8sc/5lbPC9Q0N5nBLgb+Yd915iL3udB1uFgS3B8YCx66UVHq018DAVFoee7x+gxggeA==", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", "requires": { "is-descriptor": "^1.0.0" } @@ -11213,7 +11213,7 @@ "kind-of": { "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", "requires": { "is-buffer": "^1.1.5" } @@ -11318,7 +11318,7 @@ "source-map": { "version": "0.5.7", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==" + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" }, "source-map-resolve": { "version": "0.5.3", @@ -11427,7 +11427,7 @@ "sprintf-js": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", - "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", + "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", "dev": true }, "sshpk": { @@ -11450,7 +11450,7 @@ "static-extend": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", - "integrity": "sha512-72E9+uLc27Mt718pMHt9VMNiAL4LMsmDbBva8mxWUCkT07fSzEGMYUCk0XWY6lp0j6RBAG4cJ3mWuZv2OE3s0g==", + "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", "requires": { "define-property": "^0.2.5", "object-copy": "^0.1.0" @@ -11459,7 +11459,7 @@ "define-property": { "version": "0.2.5", "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha512-Rr7ADjQZenceVOAKop6ALkkRAmH1A4Gx9hV/7ZujPUN2rkATqFO0JZLZInbAjpZYoJ1gUx8MRMQVkYemcbMSTA==", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", "requires": { "is-descriptor": "^0.1.0" } @@ -11727,7 +11727,7 @@ "acorn-jsx": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-3.0.1.tgz", - "integrity": "sha512-AU7pnZkguthwBjKgCg6998ByQNIMjbuDQZ8bb78QAFZwPfmKia8AIzgY/gWgqCjnht8JLdXmB4YxA0KaV60ncQ==", + "integrity": "sha1-r9+UiPsezvyDSPb7IvRk4ypYs2s=", "dev": true, "requires": { "acorn": "^3.0.4" @@ -11736,7 +11736,7 @@ "acorn": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-3.3.0.tgz", - "integrity": "sha512-OLUyIIZ7mF5oaAUT1w0TFqQS81q3saT46x8t7ukpPjMNk+nbs4ZHhs7ToV8EWnLYLepjETXd4XaCE4uxkMeqUw==", + "integrity": "sha1-ReN/s56No/JbruP/U2niu18iAXo=", "dev": true } } @@ -11744,7 +11744,7 @@ "ajv": { "version": "5.5.2", "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz", - "integrity": "sha512-Ajr4IcMXq/2QmMkEmSvxqfLN5zGmJ92gHXAeOXq1OekoH2rfDNsgdDoL2f7QaRCy7G/E6TpxBVdRuNraMztGHw==", + "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=", "dev": true, "requires": { "co": "^4.6.0", @@ -11768,7 +11768,7 @@ "chardet": { "version": "0.4.2", "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.4.2.tgz", - "integrity": "sha512-j/Toj7f1z98Hh2cYo2BVr85EpIRWqUi7rtRSGxh/cqUjqrnJe9l9UE7IUGd2vQ2p+kSHLkSzObQPZPLUC6TQwg==", + "integrity": "sha1-tUc7M9yXxCTl2Y3IfVXU2KKci/I=", "dev": true }, "circular-json": { @@ -11780,7 +11780,7 @@ "cli-cursor": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", - "integrity": "sha512-8lgKz8LmCRYZZQDpRyT2m5rKJ08TnU4tR9FFFW2rxpxR1FzWi4PQ/NfyODchAatHaUgnSPVcx/R5w6NuTBzFiw==", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", "dev": true, "requires": { "restore-cursor": "^2.0.0" @@ -11795,7 +11795,7 @@ "cross-spawn": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-5.1.0.tgz", - "integrity": "sha512-pTgQJ5KC0d2hcY8eyL1IzlBPYjTkyH72XRZPnLyKus2mBfNjQs3klqbJU2VILqZryAZUt9JOb3h/mWMy23/f5A==", + "integrity": "sha1-6L0O/uWPz/b4+UUQoKVUu/ojVEk=", "dev": true, "requires": { "lru-cache": "^4.0.1", @@ -11907,13 +11907,13 @@ "fast-deep-equal": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.1.0.tgz", - "integrity": "sha512-fueX787WZKCV0Is4/T2cyAdM4+x1S3MXXOAhavE1ys/W42SHAPacLTQhucja22QBYrfGw50M2sRiXPtTGv9Ymw==", + "integrity": "sha1-wFNHeBfIa1HaqFPIHgWbcz0CNhQ=", "dev": true }, "figures": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", - "integrity": "sha512-Oa2M9atig69ZkfwiApY8F2Yy+tzMbazyvqv21R0NsSC8floSOC09BbT1ITWAdoMGQvJ/aZnR1KMwdx9tvHnTNA==", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", "dev": true, "requires": { "escape-string-regexp": "^1.0.5" @@ -11922,7 +11922,7 @@ "file-entry-cache": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-2.0.0.tgz", - "integrity": "sha512-uXP/zGzxxFvFfcZGgBIwotm+Tdc55ddPAzF7iHshP4YGaXMww7rSF9peD9D1sui5ebONg5UobsZv+FfgEpGv/w==", + "integrity": "sha1-w5KZDD5oR4PYOLjISkXYoEhFg2E=", "dev": true, "requires": { "flat-cache": "^1.2.1", @@ -11981,13 +11981,13 @@ "is-fullwidth-code-point": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", "dev": true }, "json-schema-traverse": { "version": "0.3.1", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", - "integrity": "sha512-4JD/Ivzg7PoW8NzdrBSr3UFwC9mHgvI7Z6z3QGBsSHgKaRTUDmyZAAKJo2UbG1kUVfS9WS8bi36N49U1xw43DA==", + "integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=", "dev": true }, "lru-cache": { @@ -12015,13 +12015,13 @@ "mute-stream": { "version": "0.0.7", "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", - "integrity": "sha512-r65nCZhrbXXb6dXOACihYApHw2Q6pV0M3V0PSxd74N0+D8nzAdEAITq2oAjA1jVnKI+tGvEBUpqiMh0+rW6zDQ==", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", "dev": true }, "onetime": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", - "integrity": "sha512-oyyPpiMaKARvvcgip+JV+7zci5L8D1W9RZIz2l1o08AM3pfspitVWnPt3mzHcBPp12oYMTy0pqrFs/C+m3EwsQ==", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", "dev": true, "requires": { "mimic-fn": "^1.0.0" @@ -12036,7 +12036,7 @@ "restore-cursor": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", - "integrity": "sha512-6IzJLuGi4+R14vwagDHX+JrXmPVtPpn4mffDJ1UdR7/Edm87fl6yi8mMBIVvFtJaNTUvjughmW4hwLhRG7gC1Q==", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", "dev": true, "requires": { "onetime": "^2.0.0", @@ -12065,7 +12065,7 @@ "strip-ansi": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha512-4XaJ2zQdCzROZDivEVIDPkcQn8LMFSa8kj8Gxb/Lnwzv9A8VctNZ+lfivC/sV3ivW8ElJTERXZoPBRrZKkNKow==", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { "ansi-regex": "^3.0.0" @@ -12074,7 +12074,7 @@ "strip-json-comments": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", - "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", "dev": true }, "table": { @@ -12094,7 +12094,7 @@ "write": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/write/-/write-0.2.1.tgz", - "integrity": "sha512-CJ17OoULEKXpA5pef3qLj5AxTJ6mSt7g84he2WIskKwqFO4T97d5V7Tadl0DYDk7qyUOQD5WlUlOMChaYrhxeA==", + "integrity": "sha1-X8A4KOJkzqP+kUVUdvejxWbLB1c=", "dev": true, "requires": { "mkdirp": "^0.5.1" @@ -12109,7 +12109,7 @@ } }, "superdesk-core": { - "version": "github:superdesk/superdesk-client-core#89d683b5687f00d08c2f3275cd5320fd8b6ae1eb", + "version": "github:superdesk/superdesk-client-core#61d411e5074fd6736693e67dcdf5b503d91198a8", "from": "github:superdesk/superdesk-client-core#develop", "dev": true, "requires": { @@ -12207,7 +12207,7 @@ "sass-loader": "6.0.6", "shortid": "2.2.8", "style-loader": "0.20.2", - "superdesk-ui-framework": "3.1.9", + "superdesk-ui-framework": "^3.1.15", "ts-loader": "3.5.0", "typescript": "4.9.5", "uuid": "8.3.1", @@ -12222,6 +12222,12 @@ "integrity": "sha512-xyf2m6tRbz8qQKcxYZa7PA4SllYcay+eh25DN3jmNYY6gSTL7Htc/bttVdkqj2wfJGbeWlQiX8pIyJpKU+tubw==", "dev": true }, + "@types/node": { + "version": "14.18.63", + "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.63.tgz", + "integrity": "sha512-fAtCfv4jJg+ExtXhvCkCqUKZ+4ok/JQk01qDKhL5BDDoS3AxKXhV5/MAVUZyQnSEd2GT92fkgZl0pz0Q0AzcIQ==", + "dev": true + }, "classnames": { "version": "2.2.5", "resolved": "https://registry.npmjs.org/classnames/-/classnames-2.2.5.tgz", @@ -12289,13 +12295,60 @@ "invariant": "^2.2.4", "prop-types": "^15.5.7" } + }, + "superdesk-ui-framework": { + "version": "3.1.16", + "resolved": "https://registry.npmjs.org/superdesk-ui-framework/-/superdesk-ui-framework-3.1.16.tgz", + "integrity": "sha512-6v2/DtCmENh+n4pfq0W2cVg9Hrrs/7wKNIZnzzKDFwblEbquklwCbC9Rju1e0fnkxxcNZxXOHCo6PqoUe4fDLA==", + "dev": true, + "requires": { + "@popperjs/core": "^2.4.0", + "@superdesk/common": "0.0.28", + "@superdesk/primereact": "^5.0.2-12", + "@superdesk/react-resizable-panels": "0.0.39", + "@types/enzyme-adapter-react-16": "^1.0.6", + "@types/node": "^14.10.2", + "chart.js": "^2.9.3", + "date-fns": "2.7.0", + "enzyme": "^3.11.0", + "enzyme-adapter-react-16": "^1.15.7", + "moment": "^2.29.3", + "popper-max-size-modifier": "^0.2.0", + "popper.js": "1.14.4", + "primeicons": "2.0.0", + "react-beautiful-dnd": "^13.0.0", + "react-id-generator": "^3.0.0", + "react-scrollspy": "^3.4.3" + }, + "dependencies": { + "@superdesk/common": { + "version": "0.0.28", + "resolved": "https://registry.npmjs.org/@superdesk/common/-/common-0.0.28.tgz", + "integrity": "sha512-EhsYMm340r3FVrakH00lLvQbxVYYTzL61J5GXI3BI2xLN2dPI3N0AJEaMGqjbt0xUpUFxE3T08OtYvIC5koZvg==", + "dev": true, + "requires": { + "date-fns": "2.7.0", + "lodash": "4.17.19", + "primereact": "^6.0.2", + "react": "16.9.0", + "react-dom": "16.9.0", + "react-sortable-hoc": "^1.11.0" + } + }, + "date-fns": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.7.0.tgz", + "integrity": "sha512-wxYp2PGoUDN5ZEACc61aOtYFvSsJUylIvCjpjDOqM1UDaKIIuMJ9fAnMYFHV3TQaDpfTVxhwNK/GiCaHKuemTA==", + "dev": true + } + } } } }, "superdesk-ui-framework": { - "version": "3.1.9", - "resolved": "https://registry.npmjs.org/superdesk-ui-framework/-/superdesk-ui-framework-3.1.9.tgz", - "integrity": "sha512-mgBkdsv/mvG02WUNt+7szbw+V0xDK0BtG5nuZag/GuvHjJmTGW0s0OhuC/Q+/SGkwg/iR+C7DLYqUY4l+7TQXQ==", + "version": "3.1.15", + "resolved": "https://registry.npmjs.org/superdesk-ui-framework/-/superdesk-ui-framework-3.1.15.tgz", + "integrity": "sha512-izv4Psj1NcGnOzQDAD1dSkR38oA1eTGVV5/zgnig8hI2Z9v9eKKKWkpFLCeqiRNLm2lopoibp4VI7VYyhmewmg==", "dev": true, "requires": { "@popperjs/core": "^2.4.0", @@ -12446,7 +12499,7 @@ "is-fullwidth-code-point": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", "dev": true }, "string-width": { @@ -12481,13 +12534,13 @@ "text-table": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", - "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", "dev": true }, "through": { "version": "2.3.8", "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", - "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", "dev": true }, "thunky": { @@ -12545,7 +12598,7 @@ "to-object-path": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", - "integrity": "sha512-9mWHdnGRuh3onocaHzukyvCZhzvr6tiflAy/JRFXcJX0TjgfWA9pk9t8CMbzmBE4Jfw58pXbkngtBtqYxzNEyg==", + "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", "requires": { "kind-of": "^3.0.2" }, @@ -12553,7 +12606,7 @@ "kind-of": { "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha512-NOW9QQXMoZGg/oqnVNoNTTIFEIid1627WCffUBJEdMxYApq7mNE7CpzucIPc+ZQg25Phej7IJSmX3hO+oblOtQ==", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", "requires": { "is-buffer": "^1.1.5" } @@ -12574,7 +12627,7 @@ "to-regex-range": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", - "integrity": "sha512-ZZWNfCjUokXXDGXFpZehJIkZqq91BcULFq/Pi7M5i4JnxXdhMKAK682z8bCW3o8Hj1wuuzoKcW3DfVzaP6VuNg==", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", "requires": { "is-number": "^3.0.0", "repeat-string": "^1.6.1" @@ -12648,7 +12701,7 @@ "tslint": { "version": "5.11.0", "resolved": "https://registry.npmjs.org/tslint/-/tslint-5.11.0.tgz", - "integrity": "sha512-yA5YmWkzQoMF/fOA6TjkJlJniZxpo7cneTvdQEJj7blUd7YsR23gcOC5caM2MP186hTrhpSf1VSuygnikerwmQ==", + "integrity": "sha1-mPMMAurjzecAYgHkwzywi0hYHu0=", "dev": true, "requires": { "babel-code-frame": "^6.22.0", @@ -12694,7 +12747,7 @@ "tunnel-agent": { "version": "0.6.0", "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", + "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", "dev": true, "requires": { "safe-buffer": "^5.0.1" @@ -12703,7 +12756,7 @@ "tweetnacl": { "version": "0.14.5", "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", - "integrity": "sha512-KXXFFdAbFXY4geFIwoyNK+f5Z1b7swfXABfL7HXCmoIWMKU3dmS26672A4EeQtDzLKy7SXmfBu51JolvEKwtGA==", + "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", "dev": true }, "type": { @@ -12715,7 +12768,7 @@ "type-check": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", - "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==", + "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", "dev": true, "requires": { "prelude-ls": "~1.1.2" @@ -12807,7 +12860,7 @@ "typedarray": { "version": "0.0.6", "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", - "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==", + "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", "dev": true }, "typescript": { @@ -12962,7 +13015,7 @@ "uncontrollable": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/uncontrollable/-/uncontrollable-4.1.0.tgz", - "integrity": "sha512-YN1vmvC+UkttgPcFaal2UaNVODu6Rf1FU2x1guyiQRHOzSKkfTJLb0dzhJAEfRsAtjog4PF9UyNWUM2crqDyvg==", + "integrity": "sha1-4DWCkSUuGGUiLZCTmxny9J+Bwak=", "requires": { "invariant": "^2.1.0" } @@ -13017,13 +13070,13 @@ "unpipe": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", - "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", "dev": true }, "unset-value": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", - "integrity": "sha512-PcA2tsuGSF9cnySLHTLSh2qrQiJ70mn+r+Glzxv2TWZblxsxCC52BDlZoPCsz7STd9pN7EZetkWZBAvk4cgZdQ==", + "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", "requires": { "has-value": "^0.3.1", "isobject": "^3.0.0" @@ -13032,7 +13085,7 @@ "has-value": { "version": "0.3.1", "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", - "integrity": "sha512-gpG936j8/MzaeID5Yif+577c17TxaDmhuyVgSwtnL/q8UUTySg8Mecb+8Cf1otgLoD7DDH75axp86ER7LFsf3Q==", + "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", "requires": { "get-value": "^2.0.3", "has-values": "^0.1.4", @@ -13042,7 +13095,7 @@ "isobject": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", - "integrity": "sha512-+OUdGJlgjOBZDfxnDjYYG6zp487z0JGNQq3cYQYg5f5hKR+syHMsaztzGeml/4kGG55CSpKSpWTY+jYGgsHLgA==", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", "requires": { "isarray": "1.0.0" } @@ -13052,7 +13105,7 @@ "has-values": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", - "integrity": "sha512-J8S0cEdWuQbqD9//tlZxiMuMNmxB8PlEwvYwuxsTmR1G5RXUePEX/SJn7aD0GMLieuZYSwNH0cQuJGwnYunXRQ==" + "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=" } } }, @@ -13089,7 +13142,7 @@ "urix": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", - "integrity": "sha512-Am1ousAhSLBeB9cG/7k7r2R0zj50uDRlZHPGbazid5s9rlF1F/QKYObEKSIunSjIOkJZqwRRLpvewjEkM7pSqg==" + "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=" }, "url": { "version": "0.11.3", @@ -13165,12 +13218,12 @@ "util-deprecate": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" }, "utils-merge": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", - "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=", "dev": true }, "uuid": { @@ -13210,7 +13263,7 @@ "verror": { "version": "1.10.0", "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", - "integrity": "sha512-ZZKSmDAEFOijERBLkmYfJ+vmk3w+7hOLYDNkRCuRuMJGEmqYNCNLyBBFwWKVMhfwaEF3WOd0Zlw86U/WC/+nYw==", + "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", "dev": true, "requires": { "assert-plus": "^1.0.0", @@ -13221,7 +13274,7 @@ "core-util-is": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ==", + "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", "dev": true } } @@ -13235,13 +13288,13 @@ "void-elements": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-2.0.1.tgz", - "integrity": "sha512-qZKX4RnBzH2ugr8Lxa7x+0V6XD9Sb/ouARtiasEQCHB1EVU4NXtmHsDDrx1dO4ne5fc3J6EW05BP1Dl0z0iung==", + "integrity": "sha1-wGavtYK7HLQSjWDqkjkulNXp2+w=", "dev": true }, "warning": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/warning/-/warning-3.0.0.tgz", - "integrity": "sha512-jMBt6pUrKn5I+OGgtQ4YZLdhIeJmObddh6CsibPxyQ5yPZm1XExSyzC1LCNX7BzhxWgiHmizBWJTHJIjMjTQYQ==", + "integrity": "sha1-MuU3fLVy3kqwR1O9+IIcAe1gW3w=", "requires": { "loose-envify": "^1.0.0" } @@ -14122,7 +14175,7 @@ "wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", "dev": true }, "write": { @@ -14149,7 +14202,7 @@ "xmlbuilder": { "version": "8.2.2", "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-8.2.2.tgz", - "integrity": "sha512-eKRAFz04jghooy8muekqzo8uCSVNeyRedbuJrp0fovbLIi7wlsYtdUn3vBAAPq2Y3/0xMz2WMEUQ8yhVVO9Stw==", + "integrity": "sha1-aSSGc0ELS6QuGmE2VR0pIjNap3M=", "dev": true }, "xtend": { @@ -14214,7 +14267,7 @@ "yn": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/yn/-/yn-2.0.0.tgz", - "integrity": "sha512-uTv8J/wiWTgUTg+9vLTi//leUl5vDQS6uii/emeTb2ssY7vl6QWf2fFbIIGjnhjvbdKlU0ed7QPgY1htTC86jQ==", + "integrity": "sha1-5a2ryKz0CPY4X8dklWhMiOavaJo=", "dev": true } } diff --git a/package.json b/package.json index 813ffe0b5..e9b07ddaf 100644 --- a/package.json +++ b/package.json @@ -68,7 +68,7 @@ "sinon": "^4.5.0", "superdesk-code-style": "1.5.0", "superdesk-core": "github:superdesk/superdesk-client-core#develop", - "superdesk-ui-framework": "^3.1.9", + "superdesk-ui-framework": "^3.1.15", "ts-node": "~7.0.1", "tslint": "5.11.0", "typescript-eslint-parser": "^18.0.0" diff --git a/server/features/assignments.feature b/server/features/assignments.feature index 664049bfa..67ad25d85 100644 --- a/server/features/assignments.feature +++ b/server/features/assignments.feature @@ -21,6 +21,10 @@ Feature: Assignments """ [{"name": "Sports", "content_expiry": 60, "members": [{"user": "#CONTEXT_USER_ID#"}]}] """ + And "users" + """ + [{"_id": "507f191e810c19729de87034", "name":"testfoo", "email":"foo@122d.com", "username":"johnfoo"}] + """ @auth Scenario: Empty planning list diff --git a/server/features/planning.feature b/server/features/planning.feature index d2df9c862..ee60b96c9 100644 --- a/server/features/planning.feature +++ b/server/features/planning.feature @@ -625,6 +625,10 @@ Feature: Planning """ [{"_id": "desk_123", "name": "Politic Desk"}] """ + Given "users" + """ + [{"_id": "507f191e810c19729de871eb", "name":"testfoo", "email":"foo@122d.com", "username":"johnfoo"}] + """ Given "vocabularies" """ [{ diff --git a/server/features/search_events.feature b/server/features/search_events.feature index 6faf89bcd..b01f6d2c0 100644 --- a/server/features/search_events.feature +++ b/server/features/search_events.feature @@ -368,3 +368,53 @@ Feature: Event Search ]} """ + @auth + Scenario: Filter by date using America/Toronto timezone + Given "events" + """ + [{ + "guid": "all_day_multi", + "name": "all day event multiday", + "dates": {"start": "2024-07-14T00:00:00+0000", "end": "2024-07-16T00:00:00+0000", "all_day": true} + }, { + "guid": "all_day_single", + "name": "all day single day", + "dates": {"start": "2024-07-15T00:00:00+0000", "end": "2024-07-15T00:00:00+0000", "all_day": true} + }, { + "guid": "no_end_time_multi", + "name": "no end time multiday", + "dates": {"start": "2024-07-13T10:00:00+0000", "end": "2024-07-15T00:00:00+0000", "no_end_time": true} + }, { + "guid": "no_end_time_single", + "name": "no end time single day", + "dates": {"start": "2024-07-15T10:00:00+0000", "end": "2024-07-15T10:00:00+0000", "no_end_time": true} + }, { + "guid": "matching", + "name": "regular", + "dates": {"start": "2024-07-15T10:00:00+0000", "end": "2024-07-16T00:00:00+0000"} + }, + { + "guid": "not matching", + "name": "not matching", + "dates": {"start": "2024-07-01T10:00:00+0000", "end": "2024-07-02T00:00:00+0000"} + } + ] + """ + When we get "/events_planning_search?repo=events&only_future=false&time_zone=America/Toronto&start_date=2024-07-15T04:00:00" + Then we get list with 5 items + """ + {"_items": [ + {"guid": "all_day_multi"}, + {"guid": "all_day_single"}, + {"guid": "no_end_time_multi"}, + {"guid": "no_end_time_single"}, + {"guid": "matching"} + ]} + """ + When we get "/events_planning_search?repo=events&only_future=false&time_zone=America/Toronto&start_date=2024-07-16T04:00:00" + Then we get list with 1 items + """ + {"_items": [ + {"guid": "all_day_multi"} + ]} + """ diff --git a/server/planning/assignments/__init__.py b/server/planning/assignments/__init__.py index 9d1606a40..3b489762c 100644 --- a/server/planning/assignments/__init__.py +++ b/server/planning/assignments/__init__.py @@ -137,3 +137,14 @@ def init_app(app): label=lazy_gettext("Default sort preferences for Assignment lists"), category=lazy_gettext("Assignments"), ) + + superdesk.register_default_user_preference( + "email:notification:assignments", + { + "type": "bool", + "enabled": True, + "default": True, + }, + label=lazy_gettext("Send Assignment notifications via email"), + category=lazy_gettext("notifications"), + ) diff --git a/server/planning/assignments/assignments.py b/server/planning/assignments/assignments.py index 71d4f0542..e3e730b89 100644 --- a/server/planning/assignments/assignments.py +++ b/server/planning/assignments/assignments.py @@ -345,6 +345,12 @@ def send_assignment_notification(self, updates, original=None, force=False): return assigned_to = updates.get("assigned_to", {}) + + # No assignment notification sent, if user is not enabled assignment notification + if assigned_to.get("user") and not superdesk.get_resource_service( + "preferences" + ).check_preference_email_notification_is_enabled("assignments", user_id=assigned_to.get("user")): + return assignment_id = updates.get("_id") or assigned_to.get("assignment_id", "Unknown") if not original: original = {} diff --git a/server/planning/assignments/assignments_link_tests.py b/server/planning/assignments/assignments_link_tests.py index 29cbdd503..a13fcd341 100644 --- a/server/planning/assignments/assignments_link_tests.py +++ b/server/planning/assignments/assignments_link_tests.py @@ -4,6 +4,7 @@ assignment_id = "5b20652a1d41c812e24aa49e" +USER_ID = ObjectId("5d385f31fe985ec67a0ca583") class AssignmentLinkTestCase(TestCase): @@ -81,12 +82,33 @@ def test_updates_creates_new_record(self): "coverage_item": "cov1", "assigned_to": { "state": "assigned", - "user": "test", + "user": USER_ID, "desk": "test", }, } ], ) + self.app.data.insert( + "users", + [ + { + "_id": USER_ID, + "username": "admin", + "password": "blabla", + "email": "admin@example.com", + "user_type": "administrator", + "is_active": True, + "needs_activation": False, + "is_author": True, + "is_enabled": True, + "display_name": "John Smith", + "sign_off": "ADM", + "first_name": "John", + "last_name": "Smith", + "role": ObjectId("5d542206c04280bc6d6157f9"), + } + ], + ) get_resource_service("assignments_link").post( [{"assignment_id": assignment_id, "item_id": "item1", "reassign": True}] diff --git a/server/planning/assignments/assignments_unlink_test.py b/server/planning/assignments/assignments_unlink_test.py index fdfe94f2a..c54d9beca 100644 --- a/server/planning/assignments/assignments_unlink_test.py +++ b/server/planning/assignments/assignments_unlink_test.py @@ -5,9 +5,34 @@ class AssignmentUnlinkTestCase(TestCase): + USER_ID = ObjectId("5d385f31fe985ec67a0ca583") + + def setUp(self): + super().setUp() + with self.app.app_context(): + users = [ + { + "_id": self.USER_ID, + "username": "admin", + "password": "blabla", + "email": "admin@example.com", + "user_type": "administrator", + "is_active": True, + "needs_activation": False, + "is_author": True, + "is_enabled": True, + "display_name": "John Smith", + "sign_off": "ADM", + "first_name": "John", + "last_name": "Smith", + "role": ObjectId("5d542206c04280bc6d6157f9"), + } + ] + self.app.data.insert("users", users) + def test_delivery_record(self): with self.app.app_context(): - flask.g.user = {"_id": ObjectId()} + flask.g.user = {"_id": self.USER_ID} self.app.data.insert( "vocabularies", [ @@ -105,8 +130,8 @@ def test_delivery_record(self): def test_unlinks_all_content_updates(self): with self.app.app_context(): self.app.config.update({"PLANNING_LINK_UPDATES_TO_COVERAGES": True}) - flask.g.user = {"_id": ObjectId()} - user_id = ObjectId() + flask.g.user = {"_id": self.USER_ID} + user_id = self.USER_ID desk_id = ObjectId() # Make sure users a members of the desks @@ -218,8 +243,8 @@ def test_unlinks_all_content_updates(self): def test_unlinks_properly_on_unlinking_any_update_in_chain(self): with self.app.app_context(): self.app.config.update({"PLANNING_LINK_UPDATES_TO_COVERAGES": True}) - flask.g.user = {"_id": ObjectId()} - user_id = ObjectId() + flask.g.user = {"_id": self.USER_ID} + user_id = self.USER_ID desk_id = ObjectId() # Make sure users a members of the desks @@ -336,8 +361,8 @@ def test_unlinks_properly_on_unlinking_any_update_in_chain(self): def test_unlinks_archived_content(self): with self.app.app_context(): self.app.config.update({"PLANNING_LINK_UPDATES_TO_COVERAGES": True}) - flask.g.user = {"_id": ObjectId()} - user_id = ObjectId() + flask.g.user = {"_id": self.USER_ID} + user_id = self.USER_ID desk_id = ObjectId() self.app.data.insert( "vocabularies", diff --git a/server/planning/events/events.py b/server/planning/events/events.py index 999fbd533..b6f41bbc3 100644 --- a/server/planning/events/events.py +++ b/server/planning/events/events.py @@ -103,14 +103,23 @@ def get_coverage_id(coverage: EmbeddedCoverageItem) -> str: ] +def get_subject_str(subject: Dict[str, str]) -> str: + return ":".join( + [ + subject.get("name", ""), + subject.get("qcode", ""), + subject.get("scheme", ""), + str(subject.get("translations", "")), + ] + ) + + def is_event_updated(new_item: Event, old_item: Event) -> bool: if new_item.get("name") != old_item.get("name"): return True - new_subject = set([subject.get("qcode") for subject in new_item.get("subject", [])]) - old_subject = set([subject.get("qcode") for subject in old_item.get("subject", [])]) - if new_subject != old_subject: - return True - return False + new_subject = set([get_subject_str(subject) for subject in new_item.get("subject", [])]) + old_subject = set([get_subject_str(subject) for subject in old_item.get("subject", [])]) + return new_subject != old_subject class EventsService(superdesk.Service): diff --git a/server/planning/planning/planning_tests.py b/server/planning/planning/planning_tests.py index 1ae04c00a..64664cf01 100644 --- a/server/planning/planning/planning_tests.py +++ b/server/planning/planning/planning_tests.py @@ -3,6 +3,9 @@ from planning.tests import TestCase from superdesk import get_resource_service from superdesk.errors import SuperdeskApiError +from bson import ObjectId + +USER_ID = ObjectId("5d385f31fe985ec67a0ca583") class DuplicateCoverageTestCase(TestCase): @@ -28,7 +31,7 @@ def setUp(self): }, "news_coverage_status": {"qcode": "ncostat:int"}, "assigned_to": { - "user": "59f7f0881d41c88cab3f2a99", + "user": USER_ID, "desk": "desk1", "state": "in_progress", }, @@ -37,6 +40,27 @@ def setUp(self): } ], ) + self.app.data.insert( + "users", + [ + { + "_id": USER_ID, + "username": "admin", + "password": "blabla", + "email": "admin@example.com", + "user_type": "administrator", + "is_active": True, + "needs_activation": False, + "is_author": True, + "is_enabled": True, + "display_name": "John Smith", + "sign_off": "ADM", + "first_name": "John", + "last_name": "Smith", + "role": ObjectId("5d542206c04280bc6d6157f9"), + } + ], + ) def test_duplicate(self): with self.app.app_context(): @@ -49,7 +73,7 @@ def test_duplicate(self): "scheduled": datetime(2029, 10, 13, 15, 00, tzinfo=pytz.UTC), }, "assigned_to": { - "user": "562435231d41c835d7b5fb55", + "user": USER_ID, "desk": "desk2", "state": "in_progress", }, @@ -63,7 +87,7 @@ def test_duplicate(self): self.assertEqual(new_coverage["planning"]["slugline"], "new slugline") self.assertEqual(new_coverage["planning"]["scheduled"], datetime(2029, 10, 13, 15, 00, tzinfo=pytz.UTC)) - self.assertEqual(new_coverage["assigned_to"]["user"], "562435231d41c835d7b5fb55") + self.assertEqual(new_coverage["assigned_to"]["user"], USER_ID) self.assertEqual(new_coverage["assigned_to"]["desk"], "desk2") self.assertEqual(new_coverage["assigned_to"]["state"], "in_progress") self.assertEqual(new_coverage["news_coverage_status"], {"qcode": "ncostat:onreq"}) diff --git a/server/planning/search/queries/elastic.py b/server/planning/search/queries/elastic.py index d72faa2c2..28b28240d 100644 --- a/server/planning/search/queries/elastic.py +++ b/server/planning/search/queries/elastic.py @@ -207,26 +207,66 @@ def field_range(query: ElasticRangeParams): local_params = params.copy() local_params.pop("time_zone", None) for key in ("gt", "gte", "lt", "lte"): - if local_params.get(key) and "T" in local_params[key] and query.time_zone: - tz = pytz.timezone(query.time_zone) + if local_params.get(key) and "T" in local_params[key] and params.get("time_zone"): + tz = pytz.timezone(params["time_zone"]) utc_value = datetime.fromisoformat(local_params[key].replace("+0000", "+00:00")) local_value = utc_value.astimezone(tz) local_params[key] = local_value.strftime("%Y-%m-%d") - return { - "bool": { - "should": [ - {"range": {query.field: params}}, - { - "bool": { - "must": [ - {"term": {"dates.all_day": True}}, - {"range": {query.field: local_params}}, - ], - } - }, - ], - }, - } + if query.field == "dates.start": + return { + "bool": { + "should": [ + { + "bool": { + "must_not": [ + {"term": {"dates.all_day": True}}, + ], + "must": [ + {"range": {query.field: params}}, + ], + }, + }, + { + "bool": { + "must": [ + {"term": {"dates.all_day": True}}, + {"range": {query.field: local_params}}, + ], + }, + }, + ], + }, + } + else: + return { + "bool": { + "should": [ + { + "bool": { + "must_not": [ + {"term": {"dates.all_day": True}}, + {"term": {"dates.no_end_time": True}}, + ], + "must": [ + {"range": {query.field: params}}, + ], + }, + }, + { + "bool": { + "should": [ + {"term": {"dates.all_day": True}}, + {"term": {"dates.no_end_time": True}}, + ], + "must": [ + {"range": {query.field: local_params}}, + ], + "minimum_should_match": 1, + }, + }, + ], + }, + } return {"range": {query.field: params}} diff --git a/server/planning/search/queries/events.py b/server/planning/search/queries/events.py index 5502a4d1b..6d89aca42 100644 --- a/server/planning/search/queries/events.py +++ b/server/planning/search/queries/events.py @@ -255,16 +255,7 @@ def search_date_start(params: Dict[str, Any], query: elastic.ElasticQuery): if not date_filter and start_date and not end_date: query.filter.append( - elastic.bool_or( - [ - elastic.date_range( - elastic.ElasticRangeParams(field="dates.start", gte=start_date, time_zone=time_zone) - ), - elastic.date_range( - elastic.ElasticRangeParams(field="dates.end", gte=start_date, time_zone=time_zone) - ), - ] - ) + elastic.date_range(elastic.ElasticRangeParams(field="dates.end", gte=start_date, time_zone=time_zone)), ) @@ -273,16 +264,7 @@ def search_date_end(params: Dict[str, Any], query: elastic.ElasticQuery): if not date_filter and not start_date and end_date: query.filter.append( - elastic.bool_or( - [ - elastic.date_range( - elastic.ElasticRangeParams(field="dates.start", lte=end_date, time_zone=time_zone) - ), - elastic.date_range( - elastic.ElasticRangeParams(field="dates.end", lte=end_date, time_zone=time_zone) - ), - ] - ) + elastic.date_range(elastic.ElasticRangeParams(field="dates.start", lte=end_date, time_zone=time_zone)), ) @@ -291,55 +273,17 @@ def search_date_range(params: Dict[str, Any], query: elastic.ElasticQuery): if not date_filter and start_date and end_date: query.filter.append( - elastic.bool_or( + elastic.bool_and( [ - elastic.bool_and( - [ - elastic.date_range( - elastic.ElasticRangeParams( - field="dates.start", - gte=start_date, - time_zone=time_zone, - ) - ), - elastic.date_range( - elastic.ElasticRangeParams(field="dates.end", lte=end_date, time_zone=time_zone) - ), - ] - ), - elastic.bool_and( - [ - elastic.date_range( - elastic.ElasticRangeParams( - field="dates.start", - lt=start_date, - time_zone=time_zone, - ) - ), - elastic.date_range( - elastic.ElasticRangeParams(field="dates.end", gt=end_date, time_zone=time_zone) - ), - ] + elastic.date_range( + elastic.ElasticRangeParams( + field="dates.start", + gte=start_date, + time_zone=time_zone, + ), ), - elastic.bool_or( - [ - elastic.date_range( - elastic.ElasticRangeParams( - field="dates.start", - gte=start_date, - lte=end_date, - time_zone=time_zone, - ) - ), - elastic.date_range( - elastic.ElasticRangeParams( - field="dates.end", - gte=start_date, - lte=end_date, - time_zone=time_zone, - ) - ), - ] + elastic.date_range( + elastic.ElasticRangeParams(field="dates.end", lte=end_date, time_zone=time_zone) ), ] ) diff --git a/server/planning/tests/events_service_test.py b/server/planning/tests/events_service_test.py index daa72b041..77cc09a1a 100644 --- a/server/planning/tests/events_service_test.py +++ b/server/planning/tests/events_service_test.py @@ -17,11 +17,36 @@ def test_is_new_version(): assert not service.is_new_version(new_event, old_event) + new_event["subject"] = [{"qcode": "foo"}, {"qcode": "bar"}] + old_event["subject"] = [{"qcode": "bar"}, {"qcode": "foo"}] + + assert not service.is_new_version(new_event, old_event) + new_event["subject"] = [{"qcode": "foo"}] old_event["subject"] = [{"qcode": "bar"}] assert service.is_new_version(new_event, old_event) + new_event["subject"] = [{"qcode": "foo", "name": "Foo"}] + old_event["subject"] = [{"qcode": "foo", "name": "foo"}] + + assert service.is_new_version(new_event, old_event) + + new_event["subject"] = [{}] + old_event["subject"] = [{"qcode": "foo", "name": "foo"}] + + assert service.is_new_version(new_event, old_event) + + new_event["subject"] = [{"qcode": "foo", "name": "foo", "translations": {"fr-CA": "Foo"}}] + old_event["subject"] = [{"qcode": "foo", "name": "foo", "translations": None}] + + assert service.is_new_version(new_event, old_event) + + new_event["subject"] = [{"qcode": "foo", "name": "foo", "translations": {"fr-CA": "Bar"}}] + old_event["subject"] = [{"qcode": "foo", "name": "foo", "translations": {"fr-CA": "Foo"}}] + + assert service.is_new_version(new_event, old_event) + def test_should_update(): service = EventsService()