diff --git a/spiffworkflow-backend/bin/recreate_db b/spiffworkflow-backend/bin/recreate_db index 5af0209cb..6a9d286ab 100755 --- a/spiffworkflow-backend/bin/recreate_db +++ b/spiffworkflow-backend/bin/recreate_db @@ -28,8 +28,8 @@ if [[ -n "${SPIFFWORKFLOW_BACKEND_DATABASE_URI:-}" ]]; then database_host=$(grep -oP "^[^:]+://.*@\K(.+?)[:/]" <<<"$SPIFFWORKFLOW_BACKEND_DATABASE_URI" | sed -E 's/[:\/]$//') fi -# this will fix branching conflicts -# poetry run flask db merge heads -m "merging two heads" +# uncomment this line to fix branching conflicts +# poetry run flask db merge heads -m "merging heads" tasks="" if [[ "${1:-}" == "clean" ]]; then diff --git a/spiffworkflow-backend/migrations/versions/844cee572018_.py b/spiffworkflow-backend/migrations/versions/844cee572018_.py new file mode 100644 index 000000000..498d4202a --- /dev/null +++ b/spiffworkflow-backend/migrations/versions/844cee572018_.py @@ -0,0 +1,44 @@ +"""empty message + +Revision ID: 844cee572018 +Revises: 57df21dc569d +Create Date: 2023-09-07 14:18:12.357989 + +""" +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision = '844cee572018' +down_revision = '57df21dc569d' +branch_labels = None +depends_on = None + + +def upgrade(): + # ### commands auto generated by Alembic - please adjust! ### + op.create_table('user_property', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('user_id', sa.Integer(), nullable=False), + sa.Column('key', sa.String(length=255), nullable=False), + sa.Column('value', sa.String(length=255), nullable=True), + sa.ForeignKeyConstraint(['user_id'], ['user.id'], ), + sa.PrimaryKeyConstraint('id'), + sa.UniqueConstraint('user_id', 'key', name='user_id_key_uniq') + ) + with op.batch_alter_table('user_property', schema=None) as batch_op: + batch_op.create_index(batch_op.f('ix_user_property_key'), ['key'], unique=False) + batch_op.create_index(batch_op.f('ix_user_property_user_id'), ['user_id'], unique=False) + + # ### end Alembic commands ### + + +def downgrade(): + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table('user_property', schema=None) as batch_op: + batch_op.drop_index(batch_op.f('ix_user_property_user_id')) + batch_op.drop_index(batch_op.f('ix_user_property_key')) + + op.drop_table('user_property') + # ### end Alembic commands ### diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/load_database_models.py b/spiffworkflow-backend/src/spiffworkflow_backend/load_database_models.py index fbf3bbcb3..01fb95485 100644 --- a/spiffworkflow-backend/src/spiffworkflow_backend/load_database_models.py +++ b/spiffworkflow-backend/src/spiffworkflow_backend/load_database_models.py @@ -88,5 +88,8 @@ from spiffworkflow_backend.models.task_draft_data import ( from spiffworkflow_backend.models.configuration import ( ConfigurationModel, ) # noqa: F401 +from spiffworkflow_backend.models.user_property import ( + UserPropertyModel, +) # noqa: F401 add_listeners() diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/models/user_property.py b/spiffworkflow-backend/src/spiffworkflow_backend/models/user_property.py new file mode 100644 index 000000000..74313f4ed --- /dev/null +++ b/spiffworkflow-backend/src/spiffworkflow_backend/models/user_property.py @@ -0,0 +1,20 @@ +from __future__ import annotations + +from dataclasses import dataclass + +from sqlalchemy import ForeignKey + +from spiffworkflow_backend.models.db import SpiffworkflowBaseDBModel +from spiffworkflow_backend.models.db import db +from spiffworkflow_backend.models.user import UserModel + + +@dataclass +class UserPropertyModel(SpiffworkflowBaseDBModel): + __tablename__ = "user_property" + __table_args__ = (db.UniqueConstraint("user_id", "key", name="user_id_key_uniq"),) + + id: int = db.Column(db.Integer, primary_key=True) + user_id: int = db.Column(ForeignKey(UserModel.id), nullable=False, index=True) # type: ignore + key: str = db.Column(db.String(255), nullable=False, index=True) + value: str | None = db.Column(db.String(255)) diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/routes/extensions_controller.py b/spiffworkflow-backend/src/spiffworkflow_backend/routes/extensions_controller.py index 076dad070..749d59a9a 100644 --- a/spiffworkflow-backend/src/spiffworkflow_backend/routes/extensions_controller.py +++ b/spiffworkflow-backend/src/spiffworkflow_backend/routes/extensions_controller.py @@ -21,6 +21,7 @@ from spiffworkflow_backend.services.process_instance_processor import CustomBpmn from spiffworkflow_backend.services.process_instance_processor import ProcessInstanceProcessor from spiffworkflow_backend.services.process_instance_queue_service import ProcessInstanceIsAlreadyLockedError from spiffworkflow_backend.services.process_instance_queue_service import ProcessInstanceIsNotEnqueuedError +from spiffworkflow_backend.services.process_instance_service import ProcessInstanceService from spiffworkflow_backend.services.process_model_service import ProcessModelService from spiffworkflow_backend.services.spec_file_service import SpecFileService from spiffworkflow_backend.services.workflow_execution_service import WorkflowExecutionServiceError @@ -121,17 +122,25 @@ def _run_extension( status_code=400, ) - ui_schema_page_definition = None - if body and "ui_schema_page_definition" in body: - ui_schema_page_definition = body["ui_schema_page_definition"] + ui_schema_action = None + persistence_level = "none" + if body and "ui_schema_action" in body: + ui_schema_action = body["ui_schema_action"] + persistence_level = ui_schema_action.get("persistence_level", "none") - process_instance = ProcessInstanceModel( - status=ProcessInstanceStatus.not_started.value, - process_initiator_id=g.user.id, - process_model_identifier=process_model.id, - process_model_display_name=process_model.display_name, - persistence_level="none", - ) + process_instance = None + if persistence_level == "none": + process_instance = ProcessInstanceModel( + status=ProcessInstanceStatus.not_started.value, + process_initiator_id=g.user.id, + process_model_identifier=process_model.id, + process_model_display_name=process_model.display_name, + persistence_level=persistence_level, + ) + else: + process_instance = ProcessInstanceService.create_process_instance_from_process_model_identifier( + process_model_identifier, g.user + ) processor = None try: @@ -170,10 +179,10 @@ def _run_extension( task_data = processor.get_data() result: dict[str, Any] = {"task_data": task_data} - if ui_schema_page_definition: - if "results_markdown_filename" in ui_schema_page_definition: + if ui_schema_action: + if "results_markdown_filename" in ui_schema_action: file_contents = SpecFileService.get_data( - process_model, ui_schema_page_definition["results_markdown_filename"] + process_model, ui_schema_action["results_markdown_filename"] ).decode("utf-8") form_contents = JinjaService.render_jinja_template(file_contents, task_data=task_data) result["rendered_results_markdown"] = form_contents diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/scripts/get_user_properties.py b/spiffworkflow-backend/src/spiffworkflow_backend/scripts/get_user_properties.py new file mode 100644 index 000000000..46266703b --- /dev/null +++ b/spiffworkflow-backend/src/spiffworkflow_backend/scripts/get_user_properties.py @@ -0,0 +1,23 @@ +from typing import Any + +from flask import g +from spiffworkflow_backend.models.script_attributes_context import ScriptAttributesContext +from spiffworkflow_backend.models.user_property import UserPropertyModel +from spiffworkflow_backend.scripts.script import Script + + +class GetUserProperties(Script): + @staticmethod + def requires_privileged_permissions() -> bool: + """We have deemed this function safe to run without elevated permissions.""" + return False + + def get_description(self) -> str: + return """Gets the user properties for current user.""" + + def run(self, script_attributes_context: ScriptAttributesContext, *_args: Any, **kwargs: Any) -> Any: + user_properties = UserPropertyModel.query.filter_by(user_id=g.user.id).all() + dict_to_return = {} + for up in user_properties: + dict_to_return[up.key] = up.value + return dict_to_return diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/scripts/script.py b/spiffworkflow-backend/src/spiffworkflow_backend/scripts/script.py index f33b8e4e2..777f3ef07 100644 --- a/spiffworkflow-backend/src/spiffworkflow_backend/scripts/script.py +++ b/spiffworkflow-backend/src/spiffworkflow_backend/scripts/script.py @@ -31,6 +31,10 @@ class ProcessModelIdentifierMissingError(Exception): pass +class InvalidArgsGivenToScriptError(Exception): + pass + + class Script: """Provides an abstract class that defines how scripts should work, this must be extended in all Script Tasks.""" diff --git a/spiffworkflow-backend/src/spiffworkflow_backend/scripts/set_user_properties.py b/spiffworkflow-backend/src/spiffworkflow_backend/scripts/set_user_properties.py new file mode 100644 index 000000000..b6ed4bba6 --- /dev/null +++ b/spiffworkflow-backend/src/spiffworkflow_backend/scripts/set_user_properties.py @@ -0,0 +1,36 @@ +from typing import Any + +from flask import g +from spiffworkflow_backend.models.db import db +from spiffworkflow_backend.models.script_attributes_context import ScriptAttributesContext +from spiffworkflow_backend.models.user_property import UserPropertyModel +from spiffworkflow_backend.scripts.script import InvalidArgsGivenToScriptError +from spiffworkflow_backend.scripts.script import Script + + +class SetUserProperties(Script): + @staticmethod + def requires_privileged_permissions() -> bool: + """We have deemed this function safe to run without elevated permissions.""" + return False + + def get_description(self) -> str: + return """Sets given user properties on current user.""" + + def run(self, script_attributes_context: ScriptAttributesContext, *args: Any, **kwargs: Any) -> Any: + properties = args[0] + if not isinstance(properties, dict): + raise InvalidArgsGivenToScriptError( + f"Args to set_user_properties must be a dict. '{properties}' is invalid." + ) + # consider using engine-specific insert or update metaphor in future: https://stackoverflow.com/a/68431412/6090676 + for property_key, property_value in properties.items(): + user_property = UserPropertyModel.query.filter_by(user_id=g.user.id, key=property_key).first() + if user_property is None: + user_property = UserPropertyModel( + user_id=g.user.id, + key=property_key, + ) + user_property.value = property_value + db.session.add(user_property) + db.session.commit() diff --git a/spiffworkflow-frontend/src/App.tsx b/spiffworkflow-frontend/src/App.tsx index 4fee5ec53..92f140fe2 100644 --- a/spiffworkflow-frontend/src/App.tsx +++ b/spiffworkflow-frontend/src/App.tsx @@ -48,11 +48,7 @@ export default function App() { } /> } /> } - /> - } /> diff --git a/spiffworkflow-frontend/src/components/NavigationBar.tsx b/spiffworkflow-frontend/src/components/NavigationBar.tsx index 23d7a3e83..4fc02da91 100644 --- a/spiffworkflow-frontend/src/components/NavigationBar.tsx +++ b/spiffworkflow-frontend/src/components/NavigationBar.tsx @@ -24,13 +24,11 @@ import { Can } from '@casl/react'; import logo from '../logo.svg'; import UserService from '../services/UserService'; import { useUriListForPermissions } from '../hooks/UriListForPermissions'; +import { PermissionsToCheck, ProcessModel, ProcessFile } from '../interfaces'; import { - PermissionsToCheck, - ProcessModel, - ProcessFile, ExtensionUiSchema, - UiSchemaNavItem, -} from '../interfaces'; + UiSchemaUxElement, +} from '../extension_ui_schema_interfaces'; import { usePermissionFetcher } from '../hooks/PermissionService'; import HttpService, { UnauthenticatedError } from '../services/HttpService'; import { DOCUMENTATION_URL, SPIFF_ENVIRONMENT } from '../config'; @@ -49,7 +47,7 @@ export default function NavigationBar() { const location = useLocation(); const [activeKey, setActiveKey] = useState(''); const [extensionNavigationItems, setExtensionNavigationItems] = useState< - UiSchemaNavItem[] | null + UiSchemaUxElement[] | null >(null); const { targetUris } = useUriListForPermissions(); @@ -109,7 +107,7 @@ export default function NavigationBar() { } const processExtensionResult = (processModels: ProcessModel[]) => { - const eni: UiSchemaNavItem[] = processModels + const eni: UiSchemaUxElement[] = processModels .map((processModel: ProcessModel) => { const extensionUiSchemaFile = processModel.files.find( (file: ProcessFile) => file.name === 'extension_uischema.json' @@ -119,8 +117,8 @@ export default function NavigationBar() { const extensionUiSchema: ExtensionUiSchema = JSON.parse( extensionUiSchemaFile.file_contents ); - if (extensionUiSchema.navigation_items) { - return extensionUiSchema.navigation_items; + if (extensionUiSchema.ux_elements) { + return extensionUiSchema.ux_elements; } } catch (jsonParseError: any) { console.error( @@ -128,7 +126,7 @@ export default function NavigationBar() { ); } } - return [] as UiSchemaNavItem[]; + return [] as UiSchemaUxElement[]; }) .flat(); if (eni) { @@ -157,6 +155,27 @@ export default function NavigationBar() { const userEmail = UserService.getUserEmail(); const username = UserService.getPreferredUsername(); + const extensionNavigationElementsForDisplayLocation = ( + displayLocation: string, + elementCallback: Function + ) => { + if (!extensionNavigationItems) { + return null; + } + + return extensionNavigationItems.map((uxElement: UiSchemaUxElement) => { + if (uxElement.display_location === displayLocation) { + return elementCallback(uxElement); + } + return null; + }); + }; + + const extensionUserProfileElement = (uxElement: UiSchemaUxElement) => { + const navItemPage = `/extensions${uxElement.page}`; + return {uxElement.label}; + }; + const profileToggletip = (
@@ -177,6 +196,10 @@ export default function NavigationBar() { Documentation + {extensionNavigationElementsForDisplayLocation( + 'user_profile_item', + extensionUserProfileElement + )} {!UserService.authenticationDisabled() ? ( <>
@@ -258,27 +281,21 @@ export default function NavigationBar() { ); }; - const extensionNavigationElements = () => { - if (!extensionNavigationItems) { - return null; + const extensionHeaderMenuItemElement = (uxElement: UiSchemaUxElement) => { + const navItemPage = `/extensions${uxElement.page}`; + const regexp = new RegExp(`^${navItemPage}$`); + if (regexp.test(location.pathname)) { + setActiveKey(navItemPage); } - - return extensionNavigationItems.map((navItem: UiSchemaNavItem) => { - const navItemRoute = `/extensions${navItem.route}`; - const regexp = new RegExp(`^${navItemRoute}`); - if (regexp.test(location.pathname)) { - setActiveKey(navItemRoute); - } - return ( - - {navItem.label} - - ); - }); + return ( + + {uxElement.label} + + ); }; const headerMenuItems = () => { @@ -328,7 +345,10 @@ export default function NavigationBar() { {configurationElement()} - {extensionNavigationElements()} + {extensionNavigationElementsForDisplayLocation( + 'header_menu_item', + extensionHeaderMenuItemElement + )} ); }; diff --git a/spiffworkflow-frontend/src/extension_ui_schema_interfaces.ts b/spiffworkflow-frontend/src/extension_ui_schema_interfaces.ts new file mode 100644 index 000000000..e68cb45cb --- /dev/null +++ b/spiffworkflow-frontend/src/extension_ui_schema_interfaces.ts @@ -0,0 +1,49 @@ +export enum UiSchemaDisplayLocation { + header_menu_item = 'header_menu_item', + user_profile_item = 'user_profile_item', +} + +export enum UiSchemaPersistenceLevel { + full = 'full', + none = 'none', +} + +export interface UiSchemaUxElement { + label: string; + page: string; + display_location: UiSchemaDisplayLocation; +} + +export interface UiSchemaAction { + api_path: string; + + persistence_level?: UiSchemaPersistenceLevel; + navigate_to_on_form_submit?: string; + results_markdown_filename?: string; +} + +export interface UiSchemaPageDefinition { + header: string; + api: string; + + on_load?: UiSchemaAction; + on_form_submit?: UiSchemaAction; + form_schema_filename?: any; + form_ui_schema_filename?: any; + markdown_instruction_filename?: string; + navigate_to_on_form_submit?: string; +} + +export interface UiSchemaPage { + [key: string]: UiSchemaPageDefinition; +} + +export interface ExtensionUiSchema { + ux_elements?: UiSchemaUxElement[]; + pages: UiSchemaPage; +} + +export interface ExtensionPostBody { + extension_input: any; + ui_schema_action?: UiSchemaAction; +} diff --git a/spiffworkflow-frontend/src/interfaces.ts b/spiffworkflow-frontend/src/interfaces.ts index 30c6bfd38..b4ae0a140 100644 --- a/spiffworkflow-frontend/src/interfaces.ts +++ b/spiffworkflow-frontend/src/interfaces.ts @@ -435,29 +435,3 @@ export interface DataStore { name: string; type: string; } - -export interface UiSchemaNavItem { - label: string; - route: string; -} -export interface UiSchemaPageDefinition { - header: string; - api: string; - - form_schema_filename?: any; - form_ui_schema_filename?: any; - markdown_instruction_filename?: string; - navigate_to_on_form_submit?: string; -} -export interface UiSchemaRoute { - [key: string]: UiSchemaPageDefinition; -} -export interface ExtensionUiSchema { - navigation_items?: UiSchemaNavItem[]; - routes: UiSchemaRoute; -} - -export interface ExtensionPostBody { - extension_input: any; - ui_schema_page_definition?: UiSchemaPageDefinition; -} diff --git a/spiffworkflow-frontend/src/routes/Extension.tsx b/spiffworkflow-frontend/src/routes/Extension.tsx index bf7795607..be9dca519 100644 --- a/spiffworkflow-frontend/src/routes/Extension.tsx +++ b/spiffworkflow-frontend/src/routes/Extension.tsx @@ -1,20 +1,19 @@ -import { useEffect, useState } from 'react'; +import { useCallback, useEffect, useState } from 'react'; import MDEditor from '@uiw/react-md-editor'; import { useParams } from 'react-router-dom'; import { Editor } from '@monaco-editor/react'; import { useUriListForPermissions } from '../hooks/UriListForPermissions'; -import { - ExtensionPostBody, - ExtensionUiSchema, - ProcessFile, - ProcessModel, - UiSchemaPageDefinition, -} from '../interfaces'; +import { ProcessFile, ProcessModel } from '../interfaces'; import HttpService from '../services/HttpService'; import useAPIError from '../hooks/UseApiError'; import { recursivelyChangeNullAndUndefined } from '../helpers'; import CustomForm from '../components/CustomForm'; import { BACKEND_BASE_URL } from '../config'; +import { + ExtensionPostBody, + ExtensionUiSchema, + UiSchemaPageDefinition, +} from '../extension_ui_schema_interfaces'; import ErrorDisplay from '../components/ErrorDisplay'; // eslint-disable-next-line sonarjs/cognitive-complexity @@ -26,7 +25,12 @@ export default function Extension() { const [formData, setFormData] = useState(null); const [formButtonsDisabled, setFormButtonsDisabled] = useState(false); const [processedTaskData, setProcessedTaskData] = useState(null); - const [markdownToRender, setMarkdownToRender] = useState(null); + const [markdownToRenderOnSubmit, setMarkdownToRenderOnSubmit] = useState< + string | null + >(null); + const [markdownToRenderOnLoad, setMarkdownToRenderOnLoad] = useState< + string | null + >(null); const [filesByName] = useState<{ [key: string]: ProcessFile; }>({}); @@ -35,18 +39,15 @@ export default function Extension() { const { addError, removeError } = useAPIError(); - useEffect(() => { - const processExtensionResult = (pm: ProcessModel) => { - setProcessModel(pm); - let extensionUiSchemaFile: ProcessFile | null = null; - pm.files.forEach((file: ProcessFile) => { - filesByName[file.name] = file; - if (file.name === 'extension_uischema.json') { - extensionUiSchemaFile = file; + const setConfigsIfDesiredSchemaFile = useCallback( + (extensionUiSchemaFile: ProcessFile | null, pm: ProcessModel) => { + const processLoadResult = (result: any) => { + setFormData(result.task_data); + if (result.rendered_results_markdown) { + setMarkdownToRenderOnLoad(result.rendered_results_markdown); } - }); + }; - // typescript is really confused by extensionUiSchemaFile so force it since we are properly checking if ( extensionUiSchemaFile && (extensionUiSchemaFile as ProcessFile).file_contents @@ -55,24 +56,61 @@ export default function Extension() { (extensionUiSchemaFile as any).file_contents ); - let routeIdentifier = `/${params.process_model}`; - if (params.extension_route) { - routeIdentifier = `${routeIdentifier}/${params.extension_route}`; + const pageIdentifier = `/${params.page_identifier}`; + if ( + extensionUiSchema.pages && + Object.keys(extensionUiSchema.pages).includes(pageIdentifier) + ) { + const pageDefinition = extensionUiSchema.pages[pageIdentifier]; + setUiSchemaPageDefinition(pageDefinition); + setProcessModel(pm); + + const postBody: ExtensionPostBody = { extension_input: {} }; + postBody.ui_schema_action = pageDefinition.on_load; + if (pageDefinition.on_load) { + HttpService.makeCallToBackend({ + path: `${targetUris.extensionListPath}/${pageDefinition.on_load.api_path}`, + successCallback: processLoadResult, + httpMethod: 'POST', + postBody, + }); + } } - setUiSchemaPageDefinition(extensionUiSchema.routes[routeIdentifier]); } + }, + [targetUris.extensionListPath, params] + ); + + useEffect(() => { + const processExtensionResult = (processModels: ProcessModel[]) => { + processModels.forEach((pm: ProcessModel) => { + let extensionUiSchemaFile: ProcessFile | null = null; + pm.files.forEach((file: ProcessFile) => { + filesByName[file.name] = file; + if (file.name === 'extension_uischema.json') { + extensionUiSchemaFile = file; + } + }); + setConfigsIfDesiredSchemaFile(extensionUiSchemaFile, pm); + }); }; HttpService.makeCallToBackend({ - path: targetUris.extensionPath, + path: targetUris.extensionListPath, successCallback: processExtensionResult, }); - }, [targetUris.extensionPath, params, filesByName]); + }, [ + filesByName, + params, + setConfigsIfDesiredSchemaFile, + targetUris.extensionListPath, + targetUris.extensionPath, + ]); const processSubmitResult = (result: any) => { setProcessedTaskData(result.task_data); if (result.rendered_results_markdown) { - setMarkdownToRender(result.rendered_results_markdown); + setMarkdownToRenderOnSubmit(result.rendered_results_markdown); } setFormButtonsDisabled(false); }; @@ -111,15 +149,15 @@ export default function Extension() { if (!isValid) { return; } - const url = `${BACKEND_BASE_URL}/extensions-get-data/${params.process_model}/${optionString}`; + const url = `${BACKEND_BASE_URL}/extensions-get-data/${params.page_identifier}/${optionString}`; window.location.href = url; setFormButtonsDisabled(false); } else { const postBody: ExtensionPostBody = { extension_input: dataToSubmit }; let apiPath = targetUris.extensionPath; - if (uiSchemaPageDefinition && uiSchemaPageDefinition.api) { - apiPath = `${targetUris.extensionListPath}/${uiSchemaPageDefinition.api}`; - postBody.ui_schema_page_definition = uiSchemaPageDefinition; + if (uiSchemaPageDefinition && uiSchemaPageDefinition.on_form_submit) { + apiPath = `${targetUris.extensionListPath}/${uiSchemaPageDefinition.on_form_submit.api_path}`; + postBody.ui_schema_action = uiSchemaPageDefinition.on_form_submit; } // NOTE: rjsf sets blanks values to undefined and JSON.stringify removes keys with undefined values @@ -141,22 +179,30 @@ export default function Extension() { if (uiSchemaPageDefinition) { const componentsToDisplay = [

{uiSchemaPageDefinition.header}

]; + const markdownContentsToRender = []; if (uiSchemaPageDefinition.markdown_instruction_filename) { const markdownFile = filesByName[uiSchemaPageDefinition.markdown_instruction_filename]; if (markdownFile.file_contents) { - componentsToDisplay.push( -
- -
- ); + markdownContentsToRender.push(markdownFile.file_contents); } } + if (markdownToRenderOnLoad) { + markdownContentsToRender.push(markdownToRenderOnLoad); + } + + if (markdownContentsToRender.length > 0) { + componentsToDisplay.push( +
+ +
+ ); + } if (uiSchemaPageDefinition.form_schema_filename) { const formSchemaFile = @@ -180,13 +226,13 @@ export default function Extension() { } } if (processedTaskData) { - if (markdownToRender) { + if (markdownToRenderOnSubmit) { componentsToDisplay.push(
);