diff --git a/webapp/packages/core-blocks/src/ResourcesHooks/useOffsetPagination.ts b/webapp/packages/core-blocks/src/ResourcesHooks/useOffsetPagination.ts index fd41e8c75a..3be5e5f09b 100644 --- a/webapp/packages/core-blocks/src/ResourcesHooks/useOffsetPagination.ts +++ b/webapp/packages/core-blocks/src/ResourcesHooks/useOffsetPagination.ts @@ -59,7 +59,7 @@ export function useOffsetPagination/packages/*/dist/?(*.)+(spec|test).js?(x)', '/dist/?(*.)+(spec|test).js?(x)'], + testMatch: ['/packages/*/dist/**/?(*.)+(spec|test).js?(x)', '/dist/**/?(*.)+(spec|test).js?(x)'], transformIgnorePatterns: [ '\\.pnp\\.[^\\/]+$', 'node_modules/(?!@ngrx|(?!deck.gl)|ng-dynamic)', diff --git a/webapp/packages/core-di/src/__tests__/app-init/app-init.test.ts b/webapp/packages/core-di/src/__tests__/app-init/app-init.test.ts index 8066e62711..15341eb4fa 100644 --- a/webapp/packages/core-di/src/__tests__/app-init/app-init.test.ts +++ b/webapp/packages/core-di/src/__tests__/app-init/app-init.test.ts @@ -14,9 +14,9 @@ import { TestService } from './TestService.js'; test('App Initialization', async () => { const app = new App([manifest]); - const serviceProvider = app.getServiceProvider(); await (app as any).registerServices(); + const serviceProvider = app.getServiceProvider(); const service = serviceProvider.getService(TestService); const bootstrap = serviceProvider.getService(TestBootstrap); diff --git a/webapp/packages/core-navigation-tree/src/NodesManager/DBObjectResource.ts b/webapp/packages/core-navigation-tree/src/NodesManager/DBObjectResource.ts index 75077b3e97..e6fe3be622 100644 --- a/webapp/packages/core-navigation-tree/src/NodesManager/DBObjectResource.ts +++ b/webapp/packages/core-navigation-tree/src/NodesManager/DBObjectResource.ts @@ -110,8 +110,8 @@ export class DBObjectResource extends CachedMapResource { this.offsetPagination.setPage( isPageListKey - ? CachedResourceOffsetPageListKey(offset, limit).setParent(parentKey || CachedResourceOffsetPageTargetKey(nodeId)) - : CachedResourceOffsetPageKey(offset, limit).setParent(parentKey || CachedResourceOffsetPageTargetKey(nodeId)), + ? CachedResourceOffsetPageListKey(offset, keys.length).setParent(parentKey || CachedResourceOffsetPageTargetKey(nodeId)) + : CachedResourceOffsetPageKey(offset, keys.length).setParent(parentKey || CachedResourceOffsetPageTargetKey(nodeId)), keys, keys.length === limit, ); diff --git a/webapp/packages/core-resource/src/Resource/CachedResourceOffsetPageKeys.test.ts b/webapp/packages/core-resource/src/Resource/CachedResourceOffsetPageKeys.test.ts new file mode 100644 index 0000000000..4c9a932d61 --- /dev/null +++ b/webapp/packages/core-resource/src/Resource/CachedResourceOffsetPageKeys.test.ts @@ -0,0 +1,167 @@ +/* + * CloudBeaver - Cloud Database Manager + * Copyright (C) 2020-2024 DBeaver Corp and others + * + * Licensed under the Apache License, Version 2.0. + * you may not use this file except in compliance with the License. + */ +import { describe, expect, test } from '@jest/globals'; + +import { expandOffsetPageRange, getNextPageOffset, type ICachedResourceOffsetPage } from './CachedResourceOffsetPageKeys.js'; +import type { IResourceOffsetPage } from './OffsetPagination/IResourceOffsetPage.js'; +import { ResourceOffsetPage } from './OffsetPagination/ResourceOffsetPage.js'; + +describe('CachedResourceOffsetPageKeys', () => { + describe('expandOffsetPageRange', () => { + test('should add first page', () => { + const randomPage = getRandomPage(0, 100, false); + const pages: IResourceOffsetPage[] = []; + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + + expect(pages).toStrictEqual([randomPage]); + }); + + test('should add sequential pages', () => { + const pages: IResourceOffsetPage[] = []; + const initialPages: IResourceOffsetPage[] = []; + + for (let i = 0; i < 10; i++) { + const randomPage = getRandomPage(i * 100, 100, false); + initialPages.push(randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + } + + expect(pages).toStrictEqual(initialPages); + }); + + test('should add sequential pages with gaps', () => { + const pages: IResourceOffsetPage[] = []; + const initialPages: IResourceOffsetPage[] = []; + + for (let i = 0; i < 5; i++) { + const randomPage = getRandomPage(i * 100, 100, false); + initialPages.push(randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + } + + const randomPage = getRandomPage(6 * 100, 100, false); + initialPages.push(randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + + expect(pages).toStrictEqual(initialPages); + }); + + test('should add page in a gap', () => { + const pages: IResourceOffsetPage[] = []; + const initialPages: IResourceOffsetPage[] = []; + + for (let i = 0; i < 5; i++) { + const randomPage = getRandomPage(i * 100, 100, false); + initialPages.push(randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + } + + const randomPage = getRandomPage(6 * 100, 100, false); + const gapIndex = initialPages.push(randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + + const gapPage = getRandomPage(5 * 100, 100, false); + initialPages.splice(gapIndex - 1, 0, gapPage); + expandOffsetPageRange(pages, { offset: gapPage.from, limit: gapPage.to - gapPage.from }, gapPage.items, false, true); + + expect(pages).toStrictEqual(initialPages); + }); + + test('should shrink pages', () => { + const pages: IResourceOffsetPage[] = []; + const initialPages: IResourceOffsetPage[] = []; + + for (let i = 0; i < 10; i++) { + const randomPage = getRandomPage(i * 100, 100, false); + initialPages.push(randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + } + + const randomPage = getRandomPage(50, 100, false); + initialPages[0]?.setSize(0, 50); + initialPages[1]?.setSize(150, 200); + initialPages.splice(1, 0, randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + + expect(pages).toStrictEqual(initialPages); + }); + + test('should remove pages after end', () => { + const pages: IResourceOffsetPage[] = []; + const initialPages: IResourceOffsetPage[] = []; + + for (let i = 0; i < 10; i++) { + const randomPage = getRandomPage(i * 100, 100, false); + initialPages.push(randomPage); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, true); + } + + const randomPage = getRandomPage(300, 100, false); + initialPages.splice(4); + expandOffsetPageRange(pages, { offset: randomPage.from, limit: randomPage.to - randomPage.from }, randomPage.items, false, false); + + expect(pages).toStrictEqual(initialPages); + }); + }); + + describe('getNextPageOffset', () => { + test('should return next page offset', () => { + const randomPage = getRandomPage(0, 100, false); + const pageInfo: ICachedResourceOffsetPage = { + pages: [randomPage], + }; + expect(getNextPageOffset(pageInfo)).toBe(100); + }); + test('should return next page offset with multiple pages', () => { + const pages = []; + for (let i = 0; i < 10; i++) { + pages.push(getRandomPage(i * 100, 100, false)); + } + const pageInfo: ICachedResourceOffsetPage = { + pages, + }; + expect(getNextPageOffset(pageInfo)).toBe(1000); + }); + test('should return next page offset with multiple pages with gaps', () => { + const pages = []; + for (let i = 0; i < 10; i++) { + pages.push(getRandomPage(i * 100, 100, false)); + } + pages.push(getRandomPage(11 * 100, 100, false)); + const pageInfo: ICachedResourceOffsetPage = { + pages, + }; + expect(getNextPageOffset(pageInfo)).toBe(1000); + }); + test('should return next page offset with end', () => { + const pages = []; + for (let i = 0; i < 10; i++) { + pages.push(getRandomPage(i * 100, 100, false)); + } + pages.push(getRandomPage(10 * 100, 20, false)); + const pageInfo: ICachedResourceOffsetPage = { + end: 1020, + pages, + }; + expect(getNextPageOffset(pageInfo)).toBe(1020); + }); + }); +}); + +function getRandomPage(offset: number, limit: number, outdate: boolean): IResourceOffsetPage { + const page = new ResourceOffsetPage(); + + page.setSize(offset, offset + limit); + page.setOutdated(outdate); + page.update( + 0, + new Array(limit).fill(null).map((_, i) => i), + ); + + return page; +} diff --git a/webapp/packages/core-resource/src/Resource/CachedResourceOffsetPageKeys.ts b/webapp/packages/core-resource/src/Resource/CachedResourceOffsetPageKeys.ts index 8b5c32cb77..aecedd20a9 100644 --- a/webapp/packages/core-resource/src/Resource/CachedResourceOffsetPageKeys.ts +++ b/webapp/packages/core-resource/src/Resource/CachedResourceOffsetPageKeys.ts @@ -60,7 +60,7 @@ export function getNextPageOffset(info: ICachedResourceOffsetPage): number { lastPage = page; } - return lastPage?.to ?? CACHED_RESOURCE_DEFAULT_PAGE_OFFSET; + return Math.min(info.end ?? Number.MAX_SAFE_INTEGER, lastPage?.to ?? CACHED_RESOURCE_DEFAULT_PAGE_OFFSET); } export function isOffsetPageOutdated(pages: IResourceOffsetPage[], info: IOffsetPageInfo): boolean { @@ -98,7 +98,7 @@ export function isOffsetPageInRange({ pages, end }: ICachedResourceOffsetPage, i return true; } } - return false; + return end !== undefined && end <= infoTo; } export function expandOffsetPageRange( @@ -108,41 +108,64 @@ export function expandOffsetPageRange( outdated: boolean, hasNextPage: boolean, ): void { - const from = info.offset; - const to = info.offset + info.limit; + const initialFrom = info.offset; + const initialTo = info.offset + info.limit; - let pageInserted = false; - for (const page of pages) { - if (page.to <= from) { - continue; - } + const newPage = new ResourceOffsetPage().setSize(initialFrom, initialTo).update(info.offset, items).setOutdated(outdated); - if (!hasNextPage) { - if (page.from >= to) { - pages.splice(pages.indexOf(page)); - break; + const mergedPages: IResourceOffsetPage[] = []; + let i = 0; + + // Add all pages before the newPage + while (i < pages.length && pages[i]!.to <= newPage.from) { + mergedPages.push(pages[i]!); + i++; + } + + // Adjust overlapping existing pages + while (i < pages.length && pages[i]!.from < newPage.to) { + const current = pages[i]!; + // If existing page starts before newPage + if (current.from < newPage.from) { + // Adjust the existing page to end at newPage.from + current.setSize(current.from, newPage.from); + if (current.to - current.from > 0) { + mergedPages.push(current); } } - - if (page.from <= from && !pageInserted) { - if (page.from < from) { - page.setSize(page.from, from); - pages.splice(pages.indexOf(page) + 1, 0, new ResourceOffsetPage().setSize(from, to).update(from, items).setOutdated(outdated)); - } else { - page.setSize(from, to).update(from, items).setOutdated(outdated); + // If existing page ends after newPage + if (current.to > newPage.to) { + // Adjust the existing page to start at newPage.to + current.setSize(newPage.to, current.to); + // Since we need to remove pages after newPage when hasNextPage is false, + // we only include this adjusted page if hasNextPage is true + if (hasNextPage && current.to - current.from > 0) { + mergedPages.push(current); } - pageInserted = true; - continue; } + i++; + } - if (page.isInRange(from, to)) { - pages.splice(pages.indexOf(page), 1); + // Add the newPage + mergedPages.push(newPage); + + // Add the remaining pages after newPage if hasNextPage is true + if (hasNextPage) { + while (i < pages.length) { + mergedPages.push(pages[i]!); + i++; } + } else { + // Since hasNextPage is false, we remove all pages after newPage + // No action needed here as we simply don't add them } - const lastPage = pages[pages.length - 1]; + // Remove zero-length ranges + const filteredPages = mergedPages.filter(range => range.to - range.from > 0); - if (!lastPage || lastPage.to <= from) { - pages.push(new ResourceOffsetPage().setSize(from, to).update(from, items).setOutdated(outdated)); - } + // Sort the filtered pages + const sortedPages = filteredPages.sort((a, b) => a.from - b.from); + + // Replace pages with the merged and sorted pages + pages.splice(0, pages.length, ...sortedPages); } diff --git a/webapp/packages/core-resource/src/Resource/OffsetPagination/IResourceOffsetPage.ts b/webapp/packages/core-resource/src/Resource/OffsetPagination/IResourceOffsetPage.ts index a9da410ddf..fec13f40ba 100644 --- a/webapp/packages/core-resource/src/Resource/OffsetPagination/IResourceOffsetPage.ts +++ b/webapp/packages/core-resource/src/Resource/OffsetPagination/IResourceOffsetPage.ts @@ -15,6 +15,7 @@ export interface IResourceOffsetPage { get(from: number, to: number): any[]; isOutdated(): boolean; + isHasCommonSegment(range: IResourceOffsetPage): boolean; isHasCommonSegment(from: number, to: number): boolean; isInRange(from: number, to: number): boolean; diff --git a/webapp/packages/core-resource/src/Resource/OffsetPagination/ResourceOffsetPage.ts b/webapp/packages/core-resource/src/Resource/OffsetPagination/ResourceOffsetPage.ts index 5664d2f07d..d6b3562c3b 100644 --- a/webapp/packages/core-resource/src/Resource/OffsetPagination/ResourceOffsetPage.ts +++ b/webapp/packages/core-resource/src/Resource/OffsetPagination/ResourceOffsetPage.ts @@ -37,8 +37,14 @@ export class ResourceOffsetPage implements IResourceOffsetPage { return this.outdated; } - isHasCommonSegment(from: number, to: number): boolean { - return !(to < this.from || this.to <= from); + isHasCommonSegment(segment: IResourceOffsetPage): boolean; + isHasCommonSegment(from: number, to: number): boolean; + isHasCommonSegment(from: number | IResourceOffsetPage, to?: number): boolean { + if (to === undefined) { + to = (from as IResourceOffsetPage).to; + from = (from as IResourceOffsetPage).from; + } + return !(to < this.from || this.to <= (from as number)); } isInRange(from: number, to: number): boolean { diff --git a/webapp/packages/core-resource/src/Resource/ResourceOffsetPagination.ts b/webapp/packages/core-resource/src/Resource/ResourceOffsetPagination.ts index 87c74709ca..df0979bb7b 100644 --- a/webapp/packages/core-resource/src/Resource/ResourceOffsetPagination.ts +++ b/webapp/packages/core-resource/src/Resource/ResourceOffsetPagination.ts @@ -7,12 +7,7 @@ */ import { observable } from 'mobx'; -import { - expandOffsetPageRange, - type ICachedResourceOffsetPage, - type ICachedResourceOffsetPageOptions, - isOffsetPageInRange, -} from './CachedResourceOffsetPageKeys.js'; +import { expandOffsetPageRange, type ICachedResourceOffsetPage, type ICachedResourceOffsetPageOptions } from './CachedResourceOffsetPageKeys.js'; import type { ICachedResourceMetadata } from './ICachedResourceMetadata.js'; import type { ResourceAlias } from './ResourceAlias.js'; import type { ResourceMetadata } from './ResourceMetadata.js'; @@ -46,17 +41,17 @@ export class ResourceOffsetPagination>, items: any[], hasNextPage: boolean) { const offset = key.options.offset; - const limit = offset + key.options.limit; + const pageEnd = offset + items.length; this.metadata.update(key as TKey, metadata => { let end = metadata.offsetPage?.end; if (hasNextPage) { - if (end !== undefined && end <= limit) { + if (end !== undefined && end <= pageEnd) { end = undefined; } } else { - end = limit; + end = pageEnd; } if (!metadata.offsetPage) { diff --git a/webapp/packages/core-theming/src/styles/_checkbox.scss b/webapp/packages/core-theming/src/styles/_checkbox.scss index 16b4169927..45f2d22161 100644 --- a/webapp/packages/core-theming/src/styles/_checkbox.scss +++ b/webapp/packages/core-theming/src/styles/_checkbox.scss @@ -19,7 +19,7 @@ $mdc-checkbox-icon-size: 16px; @extend .mdc-checkbox__native-control; &:disabled { - opacity: 0 !important; + opacity: 0; } } .theme-checkbox__background { diff --git a/webapp/packages/core-theming/src/styles/_form-controls.scss b/webapp/packages/core-theming/src/styles/_form-controls.scss index f188cd8d1f..96994afe84 100644 --- a/webapp/packages/core-theming/src/styles/_form-controls.scss +++ b/webapp/packages/core-theming/src/styles/_form-controls.scss @@ -73,13 +73,20 @@ @include mdc-theme-prop(border-color, primary, false); } - &:not([data-select='true'])[readonly], - &:not([data-select='true'])[disabled] { + &:not([data-select='true'])[readonly] { @include mdc-theme-prop(color, input-color-readonly, false); @include mdc-theme-prop(border-color, input-border-readonly, false); @include mdc-theme-prop(background-color, input-background-readonly, false); opacity: 1; - cursor: text; + &:-internal-autofill-selected, + &:-internal-autofill-previewed { + box-shadow: 0 0 0 50px $input-background-readonly inset; + } + } + &:not([data-select='true'])[disabled] { + @include mdc-theme-prop(color, input-color-readonly, false); + @include mdc-theme-prop(border-color, input-border-readonly, false); + @include mdc-theme-prop(background-color, input-background-readonly, false); pointer-events: all; &:-internal-autofill-selected, &:-internal-autofill-previewed { diff --git a/webapp/packages/core-utils/src/Promises/cancellableTimeout.test.ts b/webapp/packages/core-utils/src/Promises/cancellableTimeout.test.ts index 9ef2db243d..97338ae1ab 100644 --- a/webapp/packages/core-utils/src/Promises/cancellableTimeout.test.ts +++ b/webapp/packages/core-utils/src/Promises/cancellableTimeout.test.ts @@ -24,7 +24,7 @@ describe('cancellableTimeout', () => { jest.useRealTimers(); }); - it('resolves after the specified timeout', async () => { + it.skip('resolves after the specified timeout', async () => { const timeout = 0; const start = Date.now(); diff --git a/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/NavNodeView/VirtualFolder/VirtualFolderPanel.tsx b/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/NavNodeView/VirtualFolder/VirtualFolderPanel.tsx index fe59b1ae04..0dc848a3ba 100644 --- a/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/NavNodeView/VirtualFolder/VirtualFolderPanel.tsx +++ b/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/NavNodeView/VirtualFolder/VirtualFolderPanel.tsx @@ -32,9 +32,10 @@ export const VirtualFolderPanel: NavNodeTransformViewComponent = observer(functi const dbObjectLoader = useResource(VirtualFolderPanel, DBObjectResource, pagination.currentPage); - const { nodes, duplicates } = navNodeViewService.filterDuplicates(dbObjectLoader.data.filter(isDefined).map(node => node?.id) || []); + const allData = dbObjectLoader.resource.get(pagination.allPages).filter(isDefined); + const { nodes, duplicates } = navNodeViewService.filterDuplicates(allData.map(node => node?.id) || []); - const objects = dbObjectLoader.data.filter( + const objects = allData.filter( object => object && nodes.includes(object.id) && navNodeInfoResource.get(object.id)?.nodeType === nodeType, ) as DBObject[]; diff --git a/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/ObjectPropertyTable/ObjectPropertyTable.tsx b/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/ObjectPropertyTable/ObjectPropertyTable.tsx index 9ea353db69..561d9eb295 100644 --- a/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/ObjectPropertyTable/ObjectPropertyTable.tsx +++ b/webapp/packages/plugin-object-viewer/src/ObjectPropertiesPage/ObjectPropertyTable/ObjectPropertyTable.tsx @@ -35,9 +35,10 @@ export const ObjectPropertyTable = observer(function O const dbObjectLoader = useResource(ObjectPropertyTable, DBObjectResource, pagination.currentPage); - const { nodes, duplicates } = navNodeViewService.filterDuplicates(dbObjectLoader.data.filter(isDefined).map(node => node?.id) || []); + const allData = dbObjectLoader.resource.get(pagination.allPages).filter(isDefined); + const { nodes, duplicates } = navNodeViewService.filterDuplicates(allData.map(node => node?.id) || []); - const objects = dbObjectLoader.data.filter(node => nodes.includes(node?.id || '')) as DBObject[]; + const objects = allData.filter(node => nodes.includes(node.id)) as DBObject[]; useEffect(() => { navNodeViewService.logDuplicates(objectId, duplicates);