insomnia/app/ui/redux/selectors.js
2017-08-22 17:06:56 -07:00

233 lines
6.6 KiB
JavaScript

import {createSelector} from 'reselect';
import {fuzzyMatch} from '../../common/misc';
// ~~~~~~~~~ //
// Selectors //
// ~~~~~~~~~ //
export const selectEntitiesLists = createSelector(
state => state.entities,
entities => {
const entitiesLists = {};
for (const k of Object.keys(entities)) {
const entityMap = entities[k];
entitiesLists[k] = Object.keys(entityMap).map(id => entityMap[id]);
}
return entitiesLists;
}
);
export const selectActiveWorkspace = createSelector(
state => selectEntitiesLists(state).workspaces,
state => state.entities,
state => state.global.activeWorkspaceId,
(workspaces, entities, activeWorkspaceId) => {
return entities.workspaces[activeWorkspaceId] || workspaces[0];
}
);
export const selectActiveWorkspaceMeta = createSelector(
selectActiveWorkspace,
selectEntitiesLists,
(activeWorkspace, entities) => {
const id = activeWorkspace ? activeWorkspace._id : 'n/a';
return entities.workspaceMetas.find(m => m.parentId === id);
}
);
export const selectRequestsAndRequestGroups = createSelector(
selectEntitiesLists,
entities => [...entities.requests, ...entities.requestGroups]
);
export const selectCollapsedRequestGroups = createSelector(
selectEntitiesLists,
entities => {
const collapsed = {};
// Default all to collapsed
for (const requestGroup of entities.requestGroups) {
collapsed[requestGroup._id] = true;
}
// Update those that have metadata (not all do)
for (const meta of entities.requestGroupMetas) {
collapsed[meta.parentId] = meta.collapsed;
}
return collapsed;
}
);
export const selectSidebarChildren = createSelector(
selectCollapsedRequestGroups,
selectRequestsAndRequestGroups,
selectActiveWorkspace,
selectActiveWorkspaceMeta,
(collapsed, requestsAndRequestGroups, activeWorkspace, activeWorkspaceMeta) => {
const sidebarFilter = activeWorkspaceMeta ? activeWorkspaceMeta.sidebarFilter : '';
function next (parentId) {
const children = requestsAndRequestGroups
.filter(e => e.parentId === parentId)
.sort((a, b) => {
if (a.metaSortKey === b.metaSortKey) {
return a._id > b._id ? -1 : 1;
} else {
return a.metaSortKey < b.metaSortKey ? -1 : 1;
}
});
if (children.length > 0) {
return children.map(c => ({
doc: c,
children: next(c._id),
hidden: false,
collapsed: !!collapsed[c._id]
}));
} else {
return children;
}
}
function matchChildren (children, parentNames = []) {
// Bail early if no filter defined
if (!sidebarFilter) {
return children;
}
for (const child of children) {
// Gather all parents so we can match them too
matchChildren(child.children, [...parentNames, child.doc.name]);
const hasMatchedChildren = child.children.find(c => c.hidden === false);
// Build the monster string to match on
const method = child.doc.method || '';
const name = child.doc.name;
const toMatch = `${method}${name}${parentNames.join('❅')}`;
// Try to match name
let hasMatchedName = true;
for (const token of sidebarFilter.trim().split(' ')) {
// Filter failed. Don't render children
if (!fuzzyMatch(token, toMatch)) {
hasMatchedName = false;
break;
}
}
// Update hidden state depending on whether it matched
const matched = hasMatchedChildren || hasMatchedName;
child.hidden = !matched;
}
return children;
}
const childrenTree = next(activeWorkspace._id, false);
return matchChildren(childrenTree);
}
);
export const selectWorkspaceRequestsAndRequestGroups = createSelector(
selectActiveWorkspace,
selectEntitiesLists,
(activeWorkspace, entities) => {
function getChildren (doc) {
const requests = entities.requests.filter(r => r.parentId === doc._id);
const requestGroups = entities.requestGroups.filter(rg => rg.parentId === doc._id);
const requestGroupChildren = [];
for (const requestGroup of requestGroups) {
for (const requestGroupChild of getChildren(requestGroup)) {
requestGroupChildren.push(requestGroupChild);
}
}
return [...requests, ...requestGroups, ...requestGroupChildren];
}
return getChildren(activeWorkspace);
}
);
export const selectActiveRequest = createSelector(
state => state.entities,
selectActiveWorkspaceMeta,
(entities, workspaceMeta) => {
const id = workspaceMeta ? workspaceMeta.activeRequestId : 'n/a';
return entities.requests[id] || null;
}
);
export const selectActiveCookieJar = createSelector(
selectEntitiesLists,
selectActiveWorkspace,
(entities, workspace) => {
const cookieJar = entities.cookieJars.find(cj => cj.parentId === workspace._id);
return cookieJar || null;
}
);
export const selectActiveOAuth2Token = createSelector(
selectEntitiesLists,
selectActiveWorkspaceMeta,
(entities, workspaceMeta) => {
const id = workspaceMeta ? workspaceMeta.activeRequestId : 'n/a';
return entities.oAuth2Tokens.find(t => t.parentId === id);
}
);
export const selectUnseenWorkspaces = createSelector(
selectEntitiesLists,
entities => {
const {workspaces, workspaceMetas} = entities;
return workspaces.filter(workspace => {
const meta = workspaceMetas.find(m => m.parentId === workspace._id);
if (meta && !meta.hasSeen) {
return true;
} else {
return false;
}
});
}
);
export const selectActiveRequestMeta = createSelector(
selectActiveRequest,
selectEntitiesLists,
(activeRequest, entities) => {
const id = activeRequest ? activeRequest._id : 'n/a';
return entities.requestMetas.find(m => m.parentId === id);
}
);
export const selectActiveRequestResponses = createSelector(
selectActiveRequest,
selectEntitiesLists,
(activeRequest, entities) => {
const requestId = activeRequest ? activeRequest._id : 'n/a';
return entities.responses
.filter(response => requestId === response.parentId)
.sort((a, b) => a.created > b.created ? -1 : 1);
}
);
export const selectActiveResponse = createSelector(
selectActiveRequestMeta,
selectActiveRequestResponses,
(activeRequestMeta, responses) => {
const activeResponseId = activeRequestMeta ? activeRequestMeta.activeResponseId : 'n/a';
const activeResponse = responses.find(response => response._id === activeResponseId);
if (activeResponse) {
return activeResponse;
}
return responses[0] || null;
}
);