* ✨ Added main header tabs with current workflow execution count * ⚡ feat(editor): header tab navigation (no-changelog) (#4244) * ✨ Adding current workflow execution list to the Vuex store * ✨ Updating current workflow executions after running a workflow from the node view * ✨ Keeping the tab view content alive when switching tabs in main header * ✨ Updating main header controls to work with current workflow regardless of active tab * 🐛 Fixing a bug with previous WF executions still visible after creating a new WF * ⚡ Updating saved status when new WF is created * ✨ Implemented initial version of execution perview * ✨ Keeping the WF view alive when switching to executions tab in new navigation * ✨ Implemented executions landing page * ✨ Simplifying node view navigation * ✨ Updating executions view zoom and selection to work with the new layout * ✨ Using N8nRadioButtons component for main header tabs * 💄 Implementing executions page states. Minor refactoring. * ⚡ Merge conflict fixes and pieces of code that were left behind * ⚡ Fixing layout and scrolling changes introduced after sync with master branch * ⚡ Removing keep-alive from node view which broke template opening and some more leftover code * ✔️ Fixing linting errors * ✔️ One more lint error * ⚡ Implemented executions preview using iframes * ⚡ Fixing zoom menu positioning in iframe and adding different loading types to workflow preview * ⚡ Fixing navigation to and from WF templates and template loading * ⚡ Updating and fixing navigation to and from node view * 👌 Addressing previous PR comments * 🐛 Fixing infinite loading when saving a new workflow * 🐛 Handling opening already opened WF when not on Node view * ✨ Implemented empty states for executions view * ⚡ Adding execute button shake flag to the store so it doesn't mess up navigation by modifying route params * 💄 Started adding new styles to execution sidebar * 💄 Adding hover style for execution list * ⚡ Added ExecutionsCard component and added executions helper mixin * ✔️ Fixing leftover conflict * ✔️ One more conflict * ✨ Implemented retry execution menu and manual execution icon. Other minor updates * ✨ Implemented executions filtering * 💄 Updating running executions details in preview * ⚡ Added info accordion to executions sidebar * ✨ Implemented auto-refresh for executions sidebar * 💄 Adding running execution landing page, minor fixes * 💄 General refactoring * ✔️ Adding leftover conflict changes * ✔️ Updating `InfoTip` component test snapshots * ✔️ Fixing linting error * ✔️ Fixing lint errors in vuex store module * 👌 Started addressing review feedback * ⚡ Updating executions preview behaviour when filters are applied * 🐛 Fixing a bug where nodes and connections disappear if something is saved from executions view before loading WF in the main NodeView * 🐛 Fixing pasting in executions view and wrong workflow activator state * ⚡ Improved workflow switching and navigation, updated error message when trying to paste into execution * ⚡ Some more navigation updates * 💄 Fixing tab centering, execution filter button layout, added auto-refresh checkbox * 🐛 Fixing a bug when saving workflow using save button * 💄 Addressing design feedback, added delete execution button * ⚡ Moving main execution logic to the root executions view * ⚡ Implemented execution delete function * ⚡ Updating how switching tabs for new unsaved workflows work * ⚡ Remembering active execution when switching tabs * 💄 Addressing design feedback regarding info accordion * 💄 Updating execution card styling * ⚡ Resetting executions when creating new workflow * Fixing lint error * ⚡ Hiding executions preview is active execution is not in the results. Updated execution list spacing * ⚡ Fixing navigation to and from templates and executions * ⚡ Implemented execution lazy loading and added new background to execution preview * 💄 Disabling import when on executions tab * ⚡ Handling opening executions from different workflow * ⚡ Updating active execution on route change * ⚡ Updating execution tab detection * ⚡ Simplifying and updating navigation. Adding new route for new workflows * ⚡ Updating workflow saving logic to work with new routes * 🐛 Fixing a bug when returning to executions from different workflow * 💄 Updating executions info accordion and node details view modal in execution preview * 💄 Updating workflow activated modal to point to new executions view * ⚡ Implemented opening new executions view from execution modal * ⚡ Handling jsplumb init errors, updating unknown executions style * ⚡ Updating main sidebar after syncing branch * ⚡ Opening new trigger menu from executions view * 💄 Updating sidebar resize behaviour * ✔️ Fixing lint errors * ⚡ Loading executions when mounting executions view * ⚡ Resetting execution data when creating a new workflow * 💄 Minor wording updates * ⚡ Not reloading node view when new workflows are saved * Removing leftover console log * 🐛 Fixed a bug with save dialog not appearing when leaving executions tab * ⚡ Updating manual execution settings detection in info accordion * 💄 Addressing UI issues found during bug bash * Fixing workflow saving logic * ⚡ Preventing navigation if clicked tab is already opened * ⚡ Updating lazy loading behaviour * ⚡ Updating delete executions flow * ⚡ Added retry executions button to the execution preview * ⚡ Adding empty execution state, updating trigger detection logic, removing listeners when node view is not active * 💄 Cosmetic code improvements * ⚡ Trying the performance fix for nodeBase * ⚡ Removing the `NodeBase`fix * 🐛 Fixing a bug when saving the current workflow * 👌 Addressing code review feedback
197 lines
6.1 KiB
TypeScript
197 lines
6.1 KiB
TypeScript
import { CORE_NODES_CATEGORY, MAIN_HEADER_TABS, MAPPING_PARAMS, TEMPLATES_NODES_FILTER, VIEWS, NON_ACTIVATABLE_TRIGGER_NODE_TYPES } from '@/constants';
|
|
import { INodeUi, ITemplatesNode } from '@/Interface';
|
|
import { isResourceLocatorValue } from '@/typeGuards';
|
|
import dateformat from 'dateformat';
|
|
import {IDataObject, INodeProperties, INodeTypeDescription, NodeParameterValueType,INodeExecutionData, jsonParse} from 'n8n-workflow';
|
|
import { isJsonKeyObject } from "@/utils";
|
|
import { Route } from 'vue-router';
|
|
|
|
const CRED_KEYWORDS_TO_FILTER = ['API', 'OAuth1', 'OAuth2'];
|
|
const NODE_KEYWORDS_TO_FILTER = ['Trigger'];
|
|
const SI_SYMBOL = ['', 'k', 'M', 'G', 'T', 'P', 'E'];
|
|
|
|
const COMMUNITY_PACKAGE_NAME_REGEX = /(@\w+\/)?n8n-nodes-(?!base\b)\b\w+/g;
|
|
|
|
export function abbreviateNumber(num: number) {
|
|
const tier = (Math.log10(Math.abs(num)) / 3) | 0;
|
|
|
|
if (tier === 0) return num;
|
|
|
|
const suffix = SI_SYMBOL[tier];
|
|
const scale = Math.pow(10, tier * 3);
|
|
const scaled = num / scale;
|
|
|
|
return Number(scaled.toFixed(1)) + suffix;
|
|
}
|
|
|
|
export function convertToDisplayDate (epochTime: number) {
|
|
return dateformat(epochTime, 'yyyy-mm-dd HH:MM:ss');
|
|
}
|
|
|
|
export function convertToHumanReadableDate (epochTime: number) {
|
|
return dateformat(epochTime, 'd mmmm, yyyy @ HH:MM Z');
|
|
}
|
|
|
|
export function getAppNameFromCredType(name: string) {
|
|
return name.split(' ').filter((word) => !CRED_KEYWORDS_TO_FILTER.includes(word)).join(' ');
|
|
}
|
|
|
|
export function getAppNameFromNodeName(name: string) {
|
|
return name.split(' ').filter((word) => !NODE_KEYWORDS_TO_FILTER.includes(word)).join(' ');
|
|
}
|
|
|
|
export function getStyleTokenValue(name: string): string {
|
|
const style = getComputedStyle(document.body);
|
|
return style.getPropertyValue(name);
|
|
}
|
|
|
|
export function getTriggerNodeServiceName(nodeType: INodeTypeDescription): string {
|
|
return nodeType.displayName.replace(/ trigger/i, '');
|
|
}
|
|
|
|
export function getActivatableTriggerNodes(nodes: INodeUi[]) {
|
|
return nodes.filter((node: INodeUi) => !node.disabled && !NON_ACTIVATABLE_TRIGGER_NODE_TYPES.includes(node.type));
|
|
}
|
|
|
|
export function filterTemplateNodes(nodes: ITemplatesNode[]) {
|
|
const notCoreNodes = nodes.filter((node: ITemplatesNode) => {
|
|
return !(node.categories || []).some(
|
|
(category) => category.name === CORE_NODES_CATEGORY,
|
|
);
|
|
});
|
|
|
|
const results = notCoreNodes.length > 0 ? notCoreNodes : nodes;
|
|
return results.filter((elem) => !TEMPLATES_NODES_FILTER.includes(elem.name));
|
|
}
|
|
|
|
export function setPageTitle(title: string) {
|
|
window.document.title = title;
|
|
}
|
|
|
|
export function isString(value: unknown): value is string {
|
|
return typeof value === 'string';
|
|
}
|
|
|
|
export function isStringNumber(value: unknown): value is string {
|
|
return !isNaN(Number(value));
|
|
}
|
|
|
|
export function isNumber(value: unknown): value is number {
|
|
return typeof value === 'number';
|
|
}
|
|
|
|
export function stringSizeInBytes(input: string | IDataObject | IDataObject[] | undefined): number {
|
|
if (input === undefined) return 0;
|
|
|
|
return new Blob([typeof input === 'string' ? input : JSON.stringify(input)]).size;
|
|
}
|
|
|
|
export function isCommunityPackageName(packageName: string): boolean {
|
|
COMMUNITY_PACKAGE_NAME_REGEX.lastIndex = 0;
|
|
// Community packages names start with <@username/>n8n-nodes- not followed by word 'base'
|
|
const nameMatch = COMMUNITY_PACKAGE_NAME_REGEX.exec(packageName);
|
|
|
|
return !!nameMatch;
|
|
}
|
|
|
|
export function shorten(s: string, limit: number, keep: number) {
|
|
if (s.length <= limit) {
|
|
return s;
|
|
}
|
|
|
|
const first = s.slice(0, limit - keep);
|
|
const last = s.slice(s.length - keep, s.length);
|
|
|
|
return `${first}...${last}`;
|
|
}
|
|
|
|
export function hasExpressionMapping(value: unknown) {
|
|
return typeof value === 'string' && !!MAPPING_PARAMS.find((param) => value.includes(param));
|
|
}
|
|
|
|
export function isValueExpression (parameter: INodeProperties, paramValue: NodeParameterValueType): boolean {
|
|
if (parameter.noDataExpression === true) {
|
|
return false;
|
|
}
|
|
if (typeof paramValue === 'string' && paramValue.charAt(0) === '=') {
|
|
return true;
|
|
}
|
|
if (isResourceLocatorValue(paramValue) && paramValue.value && paramValue.value.toString().charAt(0) === '=') {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
export function convertRemToPixels(rem: string) {
|
|
return parseInt(rem, 10) * parseFloat(getComputedStyle(document.documentElement).fontSize);
|
|
}
|
|
|
|
export const executionDataToJson = (inputData: INodeExecutionData[]): IDataObject[] => inputData.reduce<IDataObject[]>(
|
|
(acc, item) => isJsonKeyObject(item) ? acc.concat(item.json) : acc,
|
|
[],
|
|
);
|
|
|
|
export const convertPath = (path: string): string => {
|
|
// TODO: That can for sure be done fancier but for now it works
|
|
const placeholder = '*___~#^#~___*';
|
|
let inBrackets = path.match(/\[(.*?)]/g);
|
|
|
|
if (inBrackets === null) {
|
|
inBrackets = [];
|
|
} else {
|
|
inBrackets = inBrackets.map(item => item.slice(1, -1)).map(item => {
|
|
if (item.startsWith('"') && item.endsWith('"')) {
|
|
return item.slice(1, -1);
|
|
}
|
|
return item;
|
|
});
|
|
}
|
|
const withoutBrackets = path.replace(/\[(.*?)]/g, placeholder);
|
|
const pathParts = withoutBrackets.split('.');
|
|
const allParts = [] as string[];
|
|
pathParts.forEach(part => {
|
|
let index = part.indexOf(placeholder);
|
|
while(index !== -1) {
|
|
if (index === 0) {
|
|
allParts.push(inBrackets!.shift() as string);
|
|
part = part.substr(placeholder.length);
|
|
} else {
|
|
allParts.push(part.substr(0, index));
|
|
part = part.substr(index);
|
|
}
|
|
index = part.indexOf(placeholder);
|
|
}
|
|
if (part !== '') {
|
|
allParts.push(part);
|
|
}
|
|
});
|
|
|
|
return '["' + allParts.join('"]["') + '"]';
|
|
};
|
|
|
|
export const clearJsonKey = (userInput: string | object) => {
|
|
const parsedUserInput = typeof userInput === 'string' ? jsonParse(userInput) : userInput;
|
|
|
|
if (!Array.isArray(parsedUserInput)) return parsedUserInput;
|
|
|
|
return parsedUserInput.map(item => isJsonKeyObject(item) ? item.json : item);
|
|
};
|
|
|
|
export const getNodeViewTab = (route: Route): string|null => {
|
|
const routeMeta = route.meta;
|
|
if (routeMeta && routeMeta.nodeView === true) {
|
|
return MAIN_HEADER_TABS.WORKFLOW;
|
|
} else {
|
|
const executionTabRoutes = [
|
|
VIEWS.EXECUTION.toString(),
|
|
VIEWS.EXECUTION_PREVIEW.toString(),
|
|
VIEWS.EXECUTION_HOME.toString(),
|
|
];
|
|
|
|
if (executionTabRoutes.includes(route.name || '')) {
|
|
return MAIN_HEADER_TABS.EXECUTIONS;
|
|
}
|
|
}
|
|
return null;
|
|
};
|