From 265b60e19233aee575d6aec1c37f81a7203cbd3b Mon Sep 17 00:00:00 2001 From: Balazs Bajorics <2226774+balazsbajorics@users.noreply.github.com> Date: Fri, 12 Jul 2024 15:56:43 +0200 Subject: [PATCH] skip calling getRemixValidPathsGenerationContext if not in remixScene (#6073) **Problem:** When resizing a PromiseCard, it takes 7-8 ms to run useGetStoryboardRoot / getValidElementPaths which seems to spend most of its time on Remix matchRoutes. **Diagnosis:** Turns out, we had this three lines of code running for _every single element visited_ during the recursive `getValidElementPathsFromElement` walk: ``` const isRemixScene = isRemixSceneElement(element, filePath, projectContents) const remixPathGenerationContext = getRemixValidPathsGenerationContext(path) if (remixPathGenerationContext.type === 'active' && isRemixScene) { ``` `getRemixValidPathsGenerationContext` is not trivially cheap to call, and it only returns a sensible result if `path` points at a remix scene. **Fix:** Only call getRemixValidPathsGenerationContext if `isRemixScene` is true. Now `useGetStoryboardRoot` only takes 0.7ms instead of 7ms. image **Note:** 0.7ms still feels excessive to run on every single frame, I'll continue the investigation and make a follow up PR **Manual Tests:** I hereby swear that: - [x] I opened a hydrogen project and it loaded - [x] I could navigate to various routes in Preview mode Fixes #6065 --- editor/src/components/canvas/canvas-utils.ts | 86 ++++++++++---------- 1 file changed, 44 insertions(+), 42 deletions(-) diff --git a/editor/src/components/canvas/canvas-utils.ts b/editor/src/components/canvas/canvas-utils.ts index 3350e9d918b1..5812e7bc39e8 100644 --- a/editor/src/components/canvas/canvas-utils.ts +++ b/editor/src/components/canvas/canvas-utils.ts @@ -1774,55 +1774,57 @@ function getValidElementPathsFromElement( let paths = includeElementInPath ? [path] : [] if (isJSXElementLike(element)) { const isRemixScene = isRemixSceneElement(element, filePath, projectContents) - const remixPathGenerationContext = getRemixValidPathsGenerationContext(path) - if (remixPathGenerationContext.type === 'active' && isRemixScene) { - function makeValidPathsFromModule(routeModulePath: string, parentPathInner: ElementPath) { - const file = getProjectFileByFilePath(projectContents, routeModulePath) - if (file == null || file.type !== 'TEXT_FILE') { - return - } + if (isRemixScene) { + const remixPathGenerationContext = getRemixValidPathsGenerationContext(path) + if (remixPathGenerationContext.type === 'active') { + function makeValidPathsFromModule(routeModulePath: string, parentPathInner: ElementPath) { + const file = getProjectFileByFilePath(projectContents, routeModulePath) + if (file == null || file.type !== 'TEXT_FILE') { + return + } - const topLevelElement = getDefaultExportedTopLevelElement(file) + const topLevelElement = getDefaultExportedTopLevelElement(file) - if (topLevelElement == null) { - return - } + if (topLevelElement == null) { + return + } - paths.push( - ...getValidElementPathsFromElement( - focusedElementPath, - topLevelElement, - parentPathInner, - projectContents, - autoFocusedPaths, - routeModulePath, - uiFilePath, - false, - true, - true, - resolve, - getRemixValidPathsGenerationContext, - ), - ) - } + paths.push( + ...getValidElementPathsFromElement( + focusedElementPath, + topLevelElement, + parentPathInner, + projectContents, + autoFocusedPaths, + routeModulePath, + uiFilePath, + false, + true, + true, + resolve, + getRemixValidPathsGenerationContext, + ), + ) + } - /** - * The null check is here to guard against the case when a route with children is missing an outlet - * that would render the children - */ - - for (const route of remixPathGenerationContext.currentlyRenderedRouteModules) { - const entry = remixPathGenerationContext.routeModulesToRelativePaths[route.id] - if (entry != null) { - const { relativePaths, filePath: filePathOfRouteModule } = entry - for (const relativePath of relativePaths) { - const basePath = EP.appendTwoPaths(path, relativePath) - makeValidPathsFromModule(filePathOfRouteModule, basePath) + /** + * The null check is here to guard against the case when a route with children is missing an outlet + * that would render the children + */ + + for (const route of remixPathGenerationContext.currentlyRenderedRouteModules) { + const entry = remixPathGenerationContext.routeModulesToRelativePaths[route.id] + if (entry != null) { + const { relativePaths, filePath: filePathOfRouteModule } = entry + for (const relativePath of relativePaths) { + const basePath = EP.appendTwoPaths(path, relativePath) + makeValidPathsFromModule(filePathOfRouteModule, basePath) + } } } - } - return paths + return paths + } } const isScene = isSceneElement(element, filePath, projectContents)