mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-04-02 13:45:10 -05:00
Change: make saved views manage its own component (#8423)
This commit is contained in:
parent
ae4e8808b0
commit
ab548e36c7
@ -33,24 +33,3 @@ test('should apply appearance changes when set', async ({ page }) => {
|
|||||||
await page.getByLabel('Enable dark mode').click()
|
await page.getByLabel('Enable dark mode').click()
|
||||||
await expect(page.locator('html')).toHaveAttribute('data-bs-theme', /dark/)
|
await expect(page.locator('html')).toHaveAttribute('data-bs-theme', /dark/)
|
||||||
})
|
})
|
||||||
|
|
||||||
test('should toggle saved view options when set & saved', async ({ page }) => {
|
|
||||||
await page.routeFromHAR(REQUESTS_HAR, { notFound: 'fallback' })
|
|
||||||
await page.goto('/settings/savedviews')
|
|
||||||
await page.getByLabel('Show on dashboard').first().click()
|
|
||||||
await page.getByLabel('Show in sidebar').first().click()
|
|
||||||
const updatePromise = page.waitForRequest((request) => {
|
|
||||||
if (!request.url().includes('8')) return true // skip other saved views
|
|
||||||
const data = request.postDataJSON()
|
|
||||||
const isValid =
|
|
||||||
data['show_on_dashboard'] === true && data['show_in_sidebar'] === true
|
|
||||||
return (
|
|
||||||
isValid &&
|
|
||||||
request.method() === 'PATCH' &&
|
|
||||||
request.url().includes('/api/saved_views/')
|
|
||||||
)
|
|
||||||
})
|
|
||||||
await page.getByRole('button', { name: 'Save' }).scrollIntoViewIfNeeded()
|
|
||||||
await page.getByRole('button', { name: 'Save' }).click()
|
|
||||||
await updatePromise
|
|
||||||
})
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -27,6 +27,7 @@ import { UsersAndGroupsComponent } from './components/admin/users-groups/users-g
|
|||||||
import { CustomFieldsComponent } from './components/manage/custom-fields/custom-fields.component'
|
import { CustomFieldsComponent } from './components/manage/custom-fields/custom-fields.component'
|
||||||
import { ConfigComponent } from './components/admin/config/config.component'
|
import { ConfigComponent } from './components/admin/config/config.component'
|
||||||
import { TrashComponent } from './components/admin/trash/trash.component'
|
import { TrashComponent } from './components/admin/trash/trash.component'
|
||||||
|
import { SavedViewsComponent } from './components/manage/saved-views/saved-views.component'
|
||||||
|
|
||||||
export const routes: Routes = [
|
export const routes: Routes = [
|
||||||
{ path: '', redirectTo: 'dashboard', pathMatch: 'full' },
|
{ path: '', redirectTo: 'dashboard', pathMatch: 'full' },
|
||||||
@ -165,6 +166,10 @@ export const routes: Routes = [
|
|||||||
path: 'settings/usersgroups',
|
path: 'settings/usersgroups',
|
||||||
redirectTo: '/usersgroups',
|
redirectTo: '/usersgroups',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'settings/savedviews',
|
||||||
|
redirectTo: '/savedviews',
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: 'settings',
|
path: 'settings',
|
||||||
component: SettingsComponent,
|
component: SettingsComponent,
|
||||||
@ -255,6 +260,17 @@ export const routes: Routes = [
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: 'savedviews',
|
||||||
|
component: SavedViewsComponent,
|
||||||
|
canActivate: [PermissionsGuard],
|
||||||
|
data: {
|
||||||
|
requiredPermission: {
|
||||||
|
action: PermissionAction.View,
|
||||||
|
type: PermissionType.SavedView,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -165,7 +165,7 @@ export class AppComponent implements OnInit, OnDestroy {
|
|||||||
[
|
[
|
||||||
{
|
{
|
||||||
anchorId: 'tour.dashboard',
|
anchorId: 'tour.dashboard',
|
||||||
content: $localize`The dashboard can be used to show saved views, such as an 'Inbox'. Those settings are found under Settings > Saved Views once you have created some.`,
|
content: $localize`The dashboard can be used to show saved views, such as an 'Inbox'. Views are found under Manage > Saved Views once you have created some.`,
|
||||||
route: '/dashboard',
|
route: '/dashboard',
|
||||||
delayAfterNavigation: 500,
|
delayAfterNavigation: 500,
|
||||||
isOptional: false,
|
isOptional: false,
|
||||||
@ -227,7 +227,7 @@ export class AppComponent implements OnInit, OnDestroy {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
anchorId: 'tour.settings',
|
anchorId: 'tour.settings',
|
||||||
content: $localize`Check out the settings for various tweaks to the web app and toggle settings for saved views.`,
|
content: $localize`Check out the settings for various tweaks to the web app.`,
|
||||||
route: '/settings',
|
route: '/settings',
|
||||||
backdropConfig: {
|
backdropConfig: {
|
||||||
offset: 0,
|
offset: 0,
|
||||||
|
@ -132,6 +132,7 @@ import { HotkeyDialogComponent } from './components/common/hotkey-dialog/hotkey-
|
|||||||
import { DeletePagesConfirmDialogComponent } from './components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component'
|
import { DeletePagesConfirmDialogComponent } from './components/common/confirm-dialog/delete-pages-confirm-dialog/delete-pages-confirm-dialog.component'
|
||||||
import { TrashComponent } from './components/admin/trash/trash.component'
|
import { TrashComponent } from './components/admin/trash/trash.component'
|
||||||
import { EntriesComponent } from './components/common/input/entries/entries.component'
|
import { EntriesComponent } from './components/common/input/entries/entries.component'
|
||||||
|
import { SavedViewsComponent } from './components/manage/saved-views/saved-views.component'
|
||||||
import {
|
import {
|
||||||
airplane,
|
airplane,
|
||||||
archive,
|
archive,
|
||||||
@ -235,6 +236,7 @@ import {
|
|||||||
trash,
|
trash,
|
||||||
uiRadios,
|
uiRadios,
|
||||||
upcScan,
|
upcScan,
|
||||||
|
windowStack,
|
||||||
x,
|
x,
|
||||||
xCircle,
|
xCircle,
|
||||||
xLg,
|
xLg,
|
||||||
@ -343,6 +345,7 @@ const icons = {
|
|||||||
trash,
|
trash,
|
||||||
uiRadios,
|
uiRadios,
|
||||||
upcScan,
|
upcScan,
|
||||||
|
windowStack,
|
||||||
x,
|
x,
|
||||||
xCircle,
|
xCircle,
|
||||||
xLg,
|
xLg,
|
||||||
@ -524,6 +527,7 @@ function initializeApp(settings: SettingsService) {
|
|||||||
DeletePagesConfirmDialogComponent,
|
DeletePagesConfirmDialogComponent,
|
||||||
TrashComponent,
|
TrashComponent,
|
||||||
EntriesComponent,
|
EntriesComponent,
|
||||||
|
SavedViewsComponent,
|
||||||
],
|
],
|
||||||
bootstrap: [AppComponent],
|
bootstrap: [AppComponent],
|
||||||
imports: [
|
imports: [
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
<pngx-page-header
|
<pngx-page-header
|
||||||
title="Settings"
|
title="Settings"
|
||||||
i18n-title
|
i18n-title
|
||||||
info="Options to customize appearance, notifications, saved views and more. Settings apply to the <strong>current user only</strong>."
|
info="Options to customize appearance, notifications and more. Settings apply to the <strong>current user only</strong>."
|
||||||
i18n-info
|
i18n-info
|
||||||
>
|
>
|
||||||
<button class="btn btn-sm btn-outline-primary" (click)="tourService.start()">
|
<button class="btn btn-sm btn-outline-primary" (click)="tourService.start()">
|
||||||
@ -226,8 +226,14 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<h4 class="mt-4" i18n>Notes</h4>
|
<h4 class="mt-4" i18n>Saved Views</h4>
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="offset-md-3 col">
|
||||||
|
<pngx-input-check i18n-title title="Show warning when closing saved views with unsaved changes" formControlName="savedViewsWarnOnUnsavedChange"></pngx-input-check>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h4 class="mt-4" i18n>Notes</h4>
|
||||||
<div class="row mb-3">
|
<div class="row mb-3">
|
||||||
<div class="offset-md-3 col">
|
<div class="offset-md-3 col">
|
||||||
<pngx-input-check i18n-title title="Enable notes" formControlName="notesEnabled"></pngx-input-check>
|
<pngx-input-check i18n-title title="Enable notes" formControlName="notesEnabled"></pngx-input-check>
|
||||||
@ -336,87 +342,6 @@
|
|||||||
|
|
||||||
</ng-template>
|
</ng-template>
|
||||||
</li>
|
</li>
|
||||||
|
|
||||||
<li [ngbNavItem]="SettingsNavIDs.SavedViews" *pngxIfPermissions="{ action: PermissionAction.Change, type: PermissionType.SavedView }">
|
|
||||||
<a ngbNavLink i18n>Saved views</a>
|
|
||||||
<ng-template ngbNavContent>
|
|
||||||
|
|
||||||
<h4 i18n>Settings</h4>
|
|
||||||
<div class="row mb-3">
|
|
||||||
<div class="offset-md-3 col">
|
|
||||||
<pngx-input-check i18n-title title="Show warning when closing saved views with unsaved changes" formControlName="savedViewsWarnOnUnsavedChange"></pngx-input-check>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<h4 i18n>Views</h4>
|
|
||||||
<ul class="list-group" formGroupName="savedViews">
|
|
||||||
|
|
||||||
@for (view of savedViews; track view) {
|
|
||||||
<li class="list-group-item py-3">
|
|
||||||
<div [formGroupName]="view.id">
|
|
||||||
<div class="row">
|
|
||||||
<div class="col">
|
|
||||||
<pngx-input-text title="Name" formControlName="name"></pngx-input-text>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
<div class="form-check form-switch mt-3">
|
|
||||||
<input type="checkbox" class="form-check-input" id="show_on_dashboard_{{view.id}}" formControlName="show_on_dashboard">
|
|
||||||
<label class="form-check-label" for="show_on_dashboard_{{view.id}}" i18n>Show on dashboard</label>
|
|
||||||
</div>
|
|
||||||
<div class="form-check form-switch">
|
|
||||||
<input type="checkbox" class="form-check-input" id="show_in_sidebar_{{view.id}}" formControlName="show_in_sidebar">
|
|
||||||
<label class="form-check-label" for="show_in_sidebar_{{view.id}}" i18n>Show in sidebar</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="col-auto">
|
|
||||||
<label class="form-label" for="name_{{view.id}}" i18n>Actions</label>
|
|
||||||
<pngx-confirm-button
|
|
||||||
label="Delete"
|
|
||||||
i18n-label
|
|
||||||
(confirm)="deleteSavedView(view)"
|
|
||||||
*pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.SavedView }"
|
|
||||||
buttonClasses="btn-sm btn-outline-danger form-control"
|
|
||||||
iconName="trash">
|
|
||||||
</pngx-confirm-button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="row">
|
|
||||||
<div class="col">
|
|
||||||
<pngx-input-number i18n-title title="Documents page size" [showAdd]="false" formControlName="page_size"></pngx-input-number>
|
|
||||||
</div>
|
|
||||||
<div class="col">
|
|
||||||
<label class="form-label" for="display_mode_{{view.id}}" i18n>Display as</label>
|
|
||||||
<select class="form-select" formControlName="display_mode">
|
|
||||||
<option [ngValue]="DisplayMode.TABLE" i18n>Table</option>
|
|
||||||
<option [ngValue]="DisplayMode.SMALL_CARDS" i18n>Small Cards</option>
|
|
||||||
<option [ngValue]="DisplayMode.LARGE_CARDS" i18n>Large Cards</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
@if (displayFields) {
|
|
||||||
<pngx-input-drag-drop-select i18n-title title="Show" i18n-emptyText emptyText="Default" [items]="displayFields" formControlName="display_fields"></pngx-input-drag-drop-select>
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
|
|
||||||
@if (savedViews && savedViews.length === 0) {
|
|
||||||
<li class="list-group-item">
|
|
||||||
<div i18n>No saved views defined.</div>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
|
|
||||||
@if (!savedViews) {
|
|
||||||
<li class="list-group-item">
|
|
||||||
<div class="spinner-border spinner-border-sm fw-normal ms-2 me-auto" role="status"></div>
|
|
||||||
<div class="visually-hidden" i18n>Loading...</div>
|
|
||||||
</li>
|
|
||||||
}
|
|
||||||
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
</ng-template>
|
|
||||||
</li>
|
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
<div [ngbNavOutlet]="nav" class="border-start border-end border-bottom p-3 mb-3 shadow-sm"></div>
|
<div [ngbNavOutlet]="nav" class="border-start border-end border-bottom p-3 mb-3 shadow-sm"></div>
|
||||||
|
@ -15,7 +15,6 @@ import {
|
|||||||
import { NgSelectModule } from '@ng-select/ng-select'
|
import { NgSelectModule } from '@ng-select/ng-select'
|
||||||
import { of, throwError } from 'rxjs'
|
import { of, throwError } from 'rxjs'
|
||||||
import { routes } from 'src/app/app-routing.module'
|
import { routes } from 'src/app/app-routing.module'
|
||||||
import { SavedView } from 'src/app/data/saved-view'
|
|
||||||
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
||||||
@ -52,10 +51,6 @@ import { DragDropSelectComponent } from '../../common/input/drag-drop-select/dra
|
|||||||
import { DragDropModule } from '@angular/cdk/drag-drop'
|
import { DragDropModule } from '@angular/cdk/drag-drop'
|
||||||
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
|
||||||
const savedViews = [
|
|
||||||
{ id: 1, name: 'view1', show_in_sidebar: true, show_on_dashboard: true },
|
|
||||||
{ id: 2, name: 'view2', show_in_sidebar: false, show_on_dashboard: false },
|
|
||||||
]
|
|
||||||
const users = [
|
const users = [
|
||||||
{ id: 1, username: 'user1', is_superuser: false },
|
{ id: 1, username: 'user1', is_superuser: false },
|
||||||
{ id: 2, username: 'user2', is_superuser: false },
|
{ id: 2, username: 'user2', is_superuser: false },
|
||||||
@ -70,7 +65,6 @@ describe('SettingsComponent', () => {
|
|||||||
let fixture: ComponentFixture<SettingsComponent>
|
let fixture: ComponentFixture<SettingsComponent>
|
||||||
let router: Router
|
let router: Router
|
||||||
let settingsService: SettingsService
|
let settingsService: SettingsService
|
||||||
let savedViewService: SavedViewService
|
|
||||||
let activatedRoute: ActivatedRoute
|
let activatedRoute: ActivatedRoute
|
||||||
let viewportScroller: ViewportScroller
|
let viewportScroller: ViewportScroller
|
||||||
let toastService: ToastService
|
let toastService: ToastService
|
||||||
@ -139,7 +133,6 @@ describe('SettingsComponent', () => {
|
|||||||
.spyOn(permissionsService, 'currentUserOwnsObject')
|
.spyOn(permissionsService, 'currentUserOwnsObject')
|
||||||
.mockReturnValue(true)
|
.mockReturnValue(true)
|
||||||
groupService = TestBed.inject(GroupService)
|
groupService = TestBed.inject(GroupService)
|
||||||
savedViewService = TestBed.inject(SavedViewService)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
function completeSetup(excludeService = null) {
|
function completeSetup(excludeService = null) {
|
||||||
@ -161,15 +154,6 @@ describe('SettingsComponent', () => {
|
|||||||
})
|
})
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
if (excludeService !== savedViewService) {
|
|
||||||
jest.spyOn(savedViewService, 'listAll').mockReturnValue(
|
|
||||||
of({
|
|
||||||
all: savedViews.map((v) => v.id),
|
|
||||||
count: savedViews.length,
|
|
||||||
results: (savedViews as SavedView[]).concat([]),
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
fixture = TestBed.createComponent(SettingsComponent)
|
fixture = TestBed.createComponent(SettingsComponent)
|
||||||
component = fixture.componentInstance
|
component = fixture.componentInstance
|
||||||
@ -184,8 +168,6 @@ describe('SettingsComponent', () => {
|
|||||||
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'permissions'])
|
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'permissions'])
|
||||||
tabButtons[2].nativeElement.dispatchEvent(new MouseEvent('click'))
|
tabButtons[2].nativeElement.dispatchEvent(new MouseEvent('click'))
|
||||||
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'notifications'])
|
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'notifications'])
|
||||||
tabButtons[3].nativeElement.dispatchEvent(new MouseEvent('click'))
|
|
||||||
expect(navigateSpy).toHaveBeenCalledWith(['settings', 'savedviews'])
|
|
||||||
|
|
||||||
const initSpy = jest.spyOn(component, 'initialize')
|
const initSpy = jest.spyOn(component, 'initialize')
|
||||||
component.isDirty = true // mock dirty
|
component.isDirty = true // mock dirty
|
||||||
@ -213,90 +195,8 @@ describe('SettingsComponent', () => {
|
|||||||
expect(scrollSpy).toHaveBeenCalledWith('#notifications')
|
expect(scrollSpy).toHaveBeenCalledWith('#notifications')
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should enable organizing of sidebar saved views even on direct navigation', () => {
|
|
||||||
completeSetup()
|
|
||||||
jest
|
|
||||||
.spyOn(activatedRoute, 'paramMap', 'get')
|
|
||||||
.mockReturnValue(of(convertToParamMap({ section: 'savedviews' })))
|
|
||||||
activatedRoute.snapshot.fragment = '#savedviews'
|
|
||||||
component.ngOnInit()
|
|
||||||
expect(component.activeNavID).toEqual(4) // Saved Views
|
|
||||||
component.ngAfterViewInit()
|
|
||||||
expect(settingsService.organizingSidebarSavedViews).toBeTruthy()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should support save saved views, show error', () => {
|
|
||||||
completeSetup()
|
|
||||||
|
|
||||||
const tabButtons = fixture.debugElement.queryAll(By.directive(NgbNavLink))
|
|
||||||
tabButtons[3].nativeElement.dispatchEvent(new MouseEvent('click'))
|
|
||||||
fixture.detectChanges()
|
|
||||||
|
|
||||||
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
|
||||||
const toastSpy = jest.spyOn(toastService, 'show')
|
|
||||||
const savedViewPatchSpy = jest.spyOn(savedViewService, 'patchMany')
|
|
||||||
|
|
||||||
const toggle = fixture.debugElement.query(
|
|
||||||
By.css('.form-check.form-switch input')
|
|
||||||
)
|
|
||||||
toggle.nativeElement.checked = true
|
|
||||||
toggle.nativeElement.dispatchEvent(new Event('change'))
|
|
||||||
|
|
||||||
// saved views error first
|
|
||||||
savedViewPatchSpy.mockReturnValueOnce(
|
|
||||||
throwError(() => new Error('unable to save saved views'))
|
|
||||||
)
|
|
||||||
component.saveSettings()
|
|
||||||
expect(toastErrorSpy).toHaveBeenCalled()
|
|
||||||
expect(savedViewPatchSpy).toHaveBeenCalled()
|
|
||||||
toastSpy.mockClear()
|
|
||||||
toastErrorSpy.mockClear()
|
|
||||||
savedViewPatchSpy.mockClear()
|
|
||||||
|
|
||||||
// succeed saved views
|
|
||||||
savedViewPatchSpy.mockReturnValueOnce(of(savedViews as SavedView[]))
|
|
||||||
component.saveSettings()
|
|
||||||
expect(toastErrorSpy).not.toHaveBeenCalled()
|
|
||||||
expect(savedViewPatchSpy).toHaveBeenCalled()
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should update only patch saved views that have changed', () => {
|
|
||||||
completeSetup()
|
|
||||||
|
|
||||||
const tabButtons = fixture.debugElement.queryAll(By.directive(NgbNavLink))
|
|
||||||
tabButtons[3].nativeElement.dispatchEvent(new MouseEvent('click'))
|
|
||||||
fixture.detectChanges()
|
|
||||||
|
|
||||||
const patchSpy = jest.spyOn(savedViewService, 'patchMany')
|
|
||||||
component.saveSettings()
|
|
||||||
expect(patchSpy).not.toHaveBeenCalled()
|
|
||||||
|
|
||||||
const view = savedViews[0]
|
|
||||||
const toggle = fixture.debugElement.query(
|
|
||||||
By.css('.form-check.form-switch input')
|
|
||||||
)
|
|
||||||
toggle.nativeElement.checked = true
|
|
||||||
toggle.nativeElement.dispatchEvent(new Event('change'))
|
|
||||||
// register change
|
|
||||||
component.savedViewGroup.get(view.id.toString()).value[
|
|
||||||
'show_on_dashboard'
|
|
||||||
] = !view.show_on_dashboard
|
|
||||||
fixture.detectChanges()
|
|
||||||
|
|
||||||
component.saveSettings()
|
|
||||||
expect(patchSpy).toHaveBeenCalledWith([
|
|
||||||
{
|
|
||||||
id: view.id,
|
|
||||||
name: view.name,
|
|
||||||
show_in_sidebar: view.show_in_sidebar,
|
|
||||||
show_on_dashboard: !view.show_on_dashboard,
|
|
||||||
},
|
|
||||||
])
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should support save local settings updating appearance settings and calling API, show error', () => {
|
it('should support save local settings updating appearance settings and calling API, show error', () => {
|
||||||
completeSetup()
|
completeSetup()
|
||||||
jest.spyOn(savedViewService, 'patchMany').mockReturnValue(of([]))
|
|
||||||
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
||||||
const toastSpy = jest.spyOn(toastService, 'show')
|
const toastSpy = jest.spyOn(toastService, 'show')
|
||||||
const storeSpy = jest.spyOn(settingsService, 'storeSettings')
|
const storeSpy = jest.spyOn(settingsService, 'storeSettings')
|
||||||
@ -326,7 +226,6 @@ describe('SettingsComponent', () => {
|
|||||||
|
|
||||||
it('should offer reload if settings changes require', () => {
|
it('should offer reload if settings changes require', () => {
|
||||||
completeSetup()
|
completeSetup()
|
||||||
jest.spyOn(savedViewService, 'patchMany').mockReturnValue(of([]))
|
|
||||||
let toast: Toast
|
let toast: Toast
|
||||||
toastService.getToasts().subscribe((t) => (toast = t[0]))
|
toastService.getToasts().subscribe((t) => (toast = t[0]))
|
||||||
component.initialize(true) // reset
|
component.initialize(true) // reset
|
||||||
@ -361,18 +260,6 @@ describe('SettingsComponent', () => {
|
|||||||
component.clearThemeColor()
|
component.clearThemeColor()
|
||||||
})
|
})
|
||||||
|
|
||||||
it('should support delete saved view', () => {
|
|
||||||
completeSetup()
|
|
||||||
const toastSpy = jest.spyOn(toastService, 'showInfo')
|
|
||||||
const deleteSpy = jest.spyOn(savedViewService, 'delete')
|
|
||||||
deleteSpy.mockReturnValue(of(true))
|
|
||||||
component.deleteSavedView(savedViews[0] as SavedView)
|
|
||||||
expect(deleteSpy).toHaveBeenCalled()
|
|
||||||
expect(toastSpy).toHaveBeenCalledWith(
|
|
||||||
`Saved view "${savedViews[0].name}" deleted.`
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
it('should show errors on load if load users failure', () => {
|
it('should show errors on load if load users failure', () => {
|
||||||
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
||||||
jest
|
jest
|
||||||
|
@ -26,7 +26,6 @@ import {
|
|||||||
tap,
|
tap,
|
||||||
} from 'rxjs'
|
} from 'rxjs'
|
||||||
import { Group } from 'src/app/data/group'
|
import { Group } from 'src/app/data/group'
|
||||||
import { SavedView } from 'src/app/data/saved-view'
|
|
||||||
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
import { GlobalSearchType, SETTINGS_KEYS } from 'src/app/data/ui-settings'
|
||||||
import { User } from 'src/app/data/user'
|
import { User } from 'src/app/data/user'
|
||||||
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
import { DocumentListViewService } from 'src/app/services/document-list-view.service'
|
||||||
@ -36,7 +35,6 @@ import {
|
|||||||
PermissionType,
|
PermissionType,
|
||||||
} from 'src/app/services/permissions.service'
|
} from 'src/app/services/permissions.service'
|
||||||
import { GroupService } from 'src/app/services/rest/group.service'
|
import { GroupService } from 'src/app/services/rest/group.service'
|
||||||
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
|
||||||
import { UserService } from 'src/app/services/rest/user.service'
|
import { UserService } from 'src/app/services/rest/user.service'
|
||||||
import {
|
import {
|
||||||
SettingsService,
|
SettingsService,
|
||||||
@ -50,7 +48,6 @@ import {
|
|||||||
SystemStatusItemStatus,
|
SystemStatusItemStatus,
|
||||||
SystemStatus,
|
SystemStatus,
|
||||||
} from 'src/app/data/system-status'
|
} from 'src/app/data/system-status'
|
||||||
import { DisplayMode } from 'src/app/data/document'
|
|
||||||
|
|
||||||
enum SettingsNavIDs {
|
enum SettingsNavIDs {
|
||||||
General = 1,
|
General = 1,
|
||||||
@ -75,9 +72,6 @@ export class SettingsComponent
|
|||||||
implements OnInit, AfterViewInit, OnDestroy, DirtyComponent
|
implements OnInit, AfterViewInit, OnDestroy, DirtyComponent
|
||||||
{
|
{
|
||||||
activeNavID: number
|
activeNavID: number
|
||||||
DisplayMode = DisplayMode
|
|
||||||
|
|
||||||
savedViewGroup = new FormGroup({})
|
|
||||||
|
|
||||||
settingsForm = new FormGroup({
|
settingsForm = new FormGroup({
|
||||||
bulkEditConfirmationDialogs: new FormControl(null),
|
bulkEditConfirmationDialogs: new FormControl(null),
|
||||||
@ -110,14 +104,9 @@ export class SettingsComponent
|
|||||||
notificationsConsumerSuppressOnDashboard: new FormControl(null),
|
notificationsConsumerSuppressOnDashboard: new FormControl(null),
|
||||||
|
|
||||||
savedViewsWarnOnUnsavedChange: new FormControl(null),
|
savedViewsWarnOnUnsavedChange: new FormControl(null),
|
||||||
savedViews: this.savedViewGroup,
|
|
||||||
})
|
})
|
||||||
|
|
||||||
savedViews: SavedView[]
|
|
||||||
SettingsNavIDs = SettingsNavIDs
|
SettingsNavIDs = SettingsNavIDs
|
||||||
get displayFields() {
|
|
||||||
return this.settings.allDisplayFields
|
|
||||||
}
|
|
||||||
|
|
||||||
store: BehaviorSubject<any>
|
store: BehaviorSubject<any>
|
||||||
storeSub: Subscription
|
storeSub: Subscription
|
||||||
@ -152,7 +141,6 @@ export class SettingsComponent
|
|||||||
}
|
}
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
public savedViewService: SavedViewService,
|
|
||||||
private documentListViewService: DocumentListViewService,
|
private documentListViewService: DocumentListViewService,
|
||||||
private toastService: ToastService,
|
private toastService: ToastService,
|
||||||
private settings: SettingsService,
|
private settings: SettingsService,
|
||||||
@ -214,18 +202,6 @@ export class SettingsComponent
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
|
||||||
this.permissionsService.currentUserCan(
|
|
||||||
PermissionAction.View,
|
|
||||||
PermissionType.SavedView
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
this.savedViewService.listAll().subscribe((r) => {
|
|
||||||
this.savedViews = r.results
|
|
||||||
this.initialize(false)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
this.activatedRoute.paramMap.subscribe((paramMap) => {
|
this.activatedRoute.paramMap.subscribe((paramMap) => {
|
||||||
const section = paramMap.get('section')
|
const section = paramMap.get('section')
|
||||||
if (section) {
|
if (section) {
|
||||||
@ -235,9 +211,6 @@ export class SettingsComponent
|
|||||||
if (navIDKey) {
|
if (navIDKey) {
|
||||||
this.activeNavID = SettingsNavIDs[navIDKey]
|
this.activeNavID = SettingsNavIDs[navIDKey]
|
||||||
}
|
}
|
||||||
if (this.activeNavID === SettingsNavIDs.SavedViews) {
|
|
||||||
this.settings.organizingSidebarSavedViews = true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -314,7 +287,6 @@ export class SettingsComponent
|
|||||||
),
|
),
|
||||||
searchDbOnly: this.settings.get(SETTINGS_KEYS.SEARCH_DB_ONLY),
|
searchDbOnly: this.settings.get(SETTINGS_KEYS.SEARCH_DB_ONLY),
|
||||||
searchLink: this.settings.get(SETTINGS_KEYS.SEARCH_FULL_TYPE),
|
searchLink: this.settings.get(SETTINGS_KEYS.SEARCH_FULL_TYPE),
|
||||||
savedViews: {},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -327,15 +299,11 @@ export class SettingsComponent
|
|||||||
this.router
|
this.router
|
||||||
.navigate(['settings', foundNavIDkey.toLowerCase()])
|
.navigate(['settings', foundNavIDkey.toLowerCase()])
|
||||||
.then((navigated) => {
|
.then((navigated) => {
|
||||||
this.settings.organizingSidebarSavedViews = false
|
|
||||||
if (!navigated && this.isDirty) {
|
if (!navigated && this.isDirty) {
|
||||||
this.activeNavID = navChangeEvent.activeId
|
this.activeNavID = navChangeEvent.activeId
|
||||||
} else if (navigated && this.isDirty) {
|
} else if (navigated && this.isDirty) {
|
||||||
this.initialize()
|
this.initialize()
|
||||||
}
|
}
|
||||||
if (this.activeNavID === SettingsNavIDs.SavedViews) {
|
|
||||||
this.settings.organizingSidebarSavedViews = true
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -346,34 +314,6 @@ export class SettingsComponent
|
|||||||
|
|
||||||
let storeData = this.getCurrentSettings()
|
let storeData = this.getCurrentSettings()
|
||||||
|
|
||||||
if (this.savedViews) {
|
|
||||||
this.emptyGroup(this.savedViewGroup)
|
|
||||||
|
|
||||||
for (let view of this.savedViews) {
|
|
||||||
storeData.savedViews[view.id.toString()] = {
|
|
||||||
id: view.id,
|
|
||||||
name: view.name,
|
|
||||||
show_on_dashboard: view.show_on_dashboard,
|
|
||||||
show_in_sidebar: view.show_in_sidebar,
|
|
||||||
page_size: view.page_size,
|
|
||||||
display_mode: view.display_mode,
|
|
||||||
display_fields: view.display_fields,
|
|
||||||
}
|
|
||||||
this.savedViewGroup.addControl(
|
|
||||||
view.id.toString(),
|
|
||||||
new FormGroup({
|
|
||||||
id: new FormControl(null),
|
|
||||||
name: new FormControl(null),
|
|
||||||
show_on_dashboard: new FormControl(null),
|
|
||||||
show_in_sidebar: new FormControl(null),
|
|
||||||
page_size: new FormControl(null),
|
|
||||||
display_mode: new FormControl(null),
|
|
||||||
display_fields: new FormControl([]),
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this.store = new BehaviorSubject(storeData)
|
this.store = new BehaviorSubject(storeData)
|
||||||
|
|
||||||
this.storeSub = this.store.asObservable().subscribe((state) => {
|
this.storeSub = this.store.asObservable().subscribe((state) => {
|
||||||
@ -413,32 +353,12 @@ export class SettingsComponent
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private emptyGroup(group: FormGroup) {
|
|
||||||
Object.keys(group.controls).forEach((key) => group.removeControl(key))
|
|
||||||
}
|
|
||||||
|
|
||||||
ngOnDestroy() {
|
ngOnDestroy() {
|
||||||
if (this.isDirty) this.settings.updateAppearanceSettings() // in case user changed appearance but didn't save
|
if (this.isDirty) this.settings.updateAppearanceSettings() // in case user changed appearance but didn't save
|
||||||
this.storeSub && this.storeSub.unsubscribe()
|
this.storeSub && this.storeSub.unsubscribe()
|
||||||
this.settings.organizingSidebarSavedViews = false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteSavedView(savedView: SavedView) {
|
public saveSettings() {
|
||||||
this.savedViewService.delete(savedView).subscribe(() => {
|
|
||||||
this.savedViewGroup.removeControl(savedView.id.toString())
|
|
||||||
this.savedViews.splice(this.savedViews.indexOf(savedView), 1)
|
|
||||||
this.toastService.showInfo(
|
|
||||||
$localize`Saved view "${savedView.name}" deleted.`
|
|
||||||
)
|
|
||||||
this.savedViewService.clearCache()
|
|
||||||
this.savedViewService.listAll().subscribe((r) => {
|
|
||||||
this.savedViews = r.results
|
|
||||||
this.initialize(true)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
private saveLocalSettings() {
|
|
||||||
this.savePending = true
|
this.savePending = true
|
||||||
const reloadRequired =
|
const reloadRequired =
|
||||||
this.settingsForm.value.displayLanguage !=
|
this.settingsForm.value.displayLanguage !=
|
||||||
@ -600,31 +520,6 @@ export class SettingsComponent
|
|||||||
return new Date()
|
return new Date()
|
||||||
}
|
}
|
||||||
|
|
||||||
saveSettings() {
|
|
||||||
// only patch views that have actually changed
|
|
||||||
const changed: SavedView[] = []
|
|
||||||
Object.values(this.savedViewGroup.controls)
|
|
||||||
.filter((g: FormGroup) => !g.pristine)
|
|
||||||
.forEach((group: FormGroup) => {
|
|
||||||
changed.push(group.value)
|
|
||||||
})
|
|
||||||
if (changed.length > 0) {
|
|
||||||
this.savedViewService.patchMany(changed).subscribe({
|
|
||||||
next: () => {
|
|
||||||
this.saveLocalSettings()
|
|
||||||
},
|
|
||||||
error: (error) => {
|
|
||||||
this.toastService.showError(
|
|
||||||
$localize`Error while storing settings on server.`,
|
|
||||||
error
|
|
||||||
)
|
|
||||||
},
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
this.saveLocalSettings()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
reset() {
|
reset() {
|
||||||
this.settingsForm.patchValue(this.store.getValue())
|
this.settingsForm.patchValue(this.store.getValue())
|
||||||
}
|
}
|
||||||
|
@ -199,6 +199,13 @@
|
|||||||
<i-bs class="me-1" name="ui-radios"></i-bs><span> <ng-container i18n>Custom Fields</ng-container></span>
|
<i-bs class="me-1" name="ui-radios"></i-bs><span> <ng-container i18n>Custom Fields</ng-container></span>
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
|
<li class="nav-item app-link" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.SavedView }">
|
||||||
|
<a class="nav-link" routerLink="savedviews" routerLinkActive="active" (click)="closeMenu()"
|
||||||
|
ngbPopover="Custom Fields" i18n-ngbPopover [disablePopover]="!slimSidebarEnabled" placement="end"
|
||||||
|
container="body" triggers="mouseenter:mouseleave" popoverClass="popover-slim">
|
||||||
|
<i-bs class="me-1" name="window-stack"></i-bs><span> <ng-container i18n>Saved Views</ng-container></span>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
<li class="nav-item app-link"
|
<li class="nav-item app-link"
|
||||||
*pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Workflow }"
|
*pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.Workflow }"
|
||||||
tourAnchor="tour.workflows">
|
tourAnchor="tour.workflows">
|
||||||
|
@ -90,6 +90,7 @@
|
|||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
<button ngbDropdownItem (click)="saveViewConfigAs()" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.SavedView }" i18n>Save as...</button>
|
<button ngbDropdownItem (click)="saveViewConfigAs()" *pngxIfPermissions="{ action: PermissionAction.Add, type: PermissionType.SavedView }" i18n>Save as...</button>
|
||||||
|
<a ngbDropdownItem routerLink="/savedviews" *pngxIfPermissions="{ action: PermissionAction.View, type: PermissionType.SavedView }" i18n>All saved views</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -0,0 +1,75 @@
|
|||||||
|
<pngx-page-header
|
||||||
|
title="Saved Views"
|
||||||
|
i18n-title
|
||||||
|
info="Customize the views of your documents."
|
||||||
|
i18n-info
|
||||||
|
>
|
||||||
|
</pngx-page-header>
|
||||||
|
<form [formGroup]="savedViewsForm" (ngSubmit)="save()">
|
||||||
|
<ul class="list-group mb-3" formGroupName="savedViews">
|
||||||
|
@for (view of savedViews; track view) {
|
||||||
|
<li class="list-group-item py-3">
|
||||||
|
<div [formGroupName]="view.id">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-text title="Name" formControlName="name"></pngx-input-text>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
<div class="form-check form-switch mt-3">
|
||||||
|
<input type="checkbox" class="form-check-input" id="show_on_dashboard_{{view.id}}" formControlName="show_on_dashboard">
|
||||||
|
<label class="form-check-label" for="show_on_dashboard_{{view.id}}" i18n>Show on dashboard</label>
|
||||||
|
</div>
|
||||||
|
<div class="form-check form-switch">
|
||||||
|
<input type="checkbox" class="form-check-input" id="show_in_sidebar_{{view.id}}" formControlName="show_in_sidebar">
|
||||||
|
<label class="form-check-label" for="show_in_sidebar_{{view.id}}" i18n>Show in sidebar</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="col-auto">
|
||||||
|
<label class="form-label" for="name_{{view.id}}" i18n>Actions</label>
|
||||||
|
<pngx-confirm-button
|
||||||
|
label="Delete"
|
||||||
|
i18n-label
|
||||||
|
(confirm)="deleteSavedView(view)"
|
||||||
|
*pngxIfPermissions="{ action: PermissionAction.Delete, type: PermissionType.SavedView }"
|
||||||
|
buttonClasses="btn-sm btn-outline-danger form-control"
|
||||||
|
iconName="trash">
|
||||||
|
</pngx-confirm-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col">
|
||||||
|
<pngx-input-number i18n-title title="Documents page size" [showAdd]="false" formControlName="page_size"></pngx-input-number>
|
||||||
|
</div>
|
||||||
|
<div class="col">
|
||||||
|
<label class="form-label" for="display_mode_{{view.id}}" i18n>Display as</label>
|
||||||
|
<select class="form-select" formControlName="display_mode">
|
||||||
|
<option [ngValue]="DisplayMode.TABLE" i18n>Table</option>
|
||||||
|
<option [ngValue]="DisplayMode.SMALL_CARDS" i18n>Small Cards</option>
|
||||||
|
<option [ngValue]="DisplayMode.LARGE_CARDS" i18n>Large Cards</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
@if (displayFields) {
|
||||||
|
<pngx-input-drag-drop-select i18n-title title="Show" i18n-emptyText emptyText="Default" [items]="displayFields" formControlName="display_fields"></pngx-input-drag-drop-select>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (savedViews && savedViews.length === 0) {
|
||||||
|
<li class="list-group-item">
|
||||||
|
<div i18n>No saved views defined.</div>
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (!savedViews) {
|
||||||
|
<li class="list-group-item">
|
||||||
|
<div class="spinner-border spinner-border-sm fw-normal ms-2 me-auto" role="status"></div>
|
||||||
|
<div class="visually-hidden" i18n>Loading...</div>
|
||||||
|
</li>
|
||||||
|
}
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<button type="submit" class="btn btn-primary mb-2" [disabled]="(isDirty$ | async) === false" i18n>Save</button>
|
||||||
|
<button type="button" (click)="reset()" class="btn btn-secondary ms-2 mb-2" [disabled]="(isDirty$ | async) === false" i18n>Cancel</button>
|
||||||
|
</form>
|
@ -0,0 +1,3 @@
|
|||||||
|
:host {
|
||||||
|
display: block;
|
||||||
|
}
|
@ -0,0 +1,165 @@
|
|||||||
|
import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'
|
||||||
|
import { provideHttpClientTesting } from '@angular/common/http/testing'
|
||||||
|
import { ComponentFixture, TestBed } from '@angular/core/testing'
|
||||||
|
import { ReactiveFormsModule, FormsModule } from '@angular/forms'
|
||||||
|
import { By } from '@angular/platform-browser'
|
||||||
|
import { NgbModule } from '@ng-bootstrap/ng-bootstrap'
|
||||||
|
import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons'
|
||||||
|
import { of, throwError } from 'rxjs'
|
||||||
|
import { SavedView } from 'src/app/data/saved-view'
|
||||||
|
import { IfPermissionsDirective } from 'src/app/directives/if-permissions.directive'
|
||||||
|
import { PermissionsGuard } from 'src/app/guards/permissions.guard'
|
||||||
|
import { PermissionsService } from 'src/app/services/permissions.service'
|
||||||
|
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
import { ConfirmButtonComponent } from '../../common/confirm-button/confirm-button.component'
|
||||||
|
import { CheckComponent } from '../../common/input/check/check.component'
|
||||||
|
import { DragDropSelectComponent } from '../../common/input/drag-drop-select/drag-drop-select.component'
|
||||||
|
import { NumberComponent } from '../../common/input/number/number.component'
|
||||||
|
import { SelectComponent } from '../../common/input/select/select.component'
|
||||||
|
import { TextComponent } from '../../common/input/text/text.component'
|
||||||
|
import { PageHeaderComponent } from '../../common/page-header/page-header.component'
|
||||||
|
import { SavedViewsComponent } from './saved-views.component'
|
||||||
|
import { DragDropModule } from '@angular/cdk/drag-drop'
|
||||||
|
|
||||||
|
const savedViews = [
|
||||||
|
{ id: 1, name: 'view1', show_in_sidebar: true, show_on_dashboard: true },
|
||||||
|
{ id: 2, name: 'view2', show_in_sidebar: false, show_on_dashboard: false },
|
||||||
|
]
|
||||||
|
|
||||||
|
describe('SavedViewsComponent', () => {
|
||||||
|
let component: SavedViewsComponent
|
||||||
|
let fixture: ComponentFixture<SavedViewsComponent>
|
||||||
|
let savedViewService: SavedViewService
|
||||||
|
let toastService: ToastService
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
TestBed.configureTestingModule({
|
||||||
|
declarations: [
|
||||||
|
SavedViewsComponent,
|
||||||
|
PageHeaderComponent,
|
||||||
|
IfPermissionsDirective,
|
||||||
|
CheckComponent,
|
||||||
|
SelectComponent,
|
||||||
|
TextComponent,
|
||||||
|
NumberComponent,
|
||||||
|
ConfirmButtonComponent,
|
||||||
|
DragDropSelectComponent,
|
||||||
|
],
|
||||||
|
imports: [
|
||||||
|
NgbModule,
|
||||||
|
NgxBootstrapIconsModule.pick(allIcons),
|
||||||
|
ReactiveFormsModule,
|
||||||
|
FormsModule,
|
||||||
|
DragDropModule,
|
||||||
|
],
|
||||||
|
providers: [
|
||||||
|
{
|
||||||
|
provide: PermissionsService,
|
||||||
|
useValue: {
|
||||||
|
currentUserCan: () => true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
PermissionsGuard,
|
||||||
|
provideHttpClient(withInterceptorsFromDi()),
|
||||||
|
provideHttpClientTesting(),
|
||||||
|
],
|
||||||
|
}).compileComponents()
|
||||||
|
|
||||||
|
savedViewService = TestBed.inject(SavedViewService)
|
||||||
|
toastService = TestBed.inject(ToastService)
|
||||||
|
fixture = TestBed.createComponent(SavedViewsComponent)
|
||||||
|
component = fixture.componentInstance
|
||||||
|
|
||||||
|
jest.spyOn(savedViewService, 'listAll').mockReturnValue(
|
||||||
|
of({
|
||||||
|
all: savedViews.map((v) => v.id),
|
||||||
|
count: savedViews.length,
|
||||||
|
results: (savedViews as SavedView[]).concat([]),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
fixture.detectChanges()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should support save saved views, show error', () => {
|
||||||
|
const toastErrorSpy = jest.spyOn(toastService, 'showError')
|
||||||
|
const toastSpy = jest.spyOn(toastService, 'show')
|
||||||
|
const savedViewPatchSpy = jest.spyOn(savedViewService, 'patchMany')
|
||||||
|
|
||||||
|
const toggle = fixture.debugElement.query(
|
||||||
|
By.css('.form-check.form-switch input')
|
||||||
|
)
|
||||||
|
toggle.nativeElement.checked = true
|
||||||
|
toggle.nativeElement.dispatchEvent(new Event('change'))
|
||||||
|
|
||||||
|
// saved views error first
|
||||||
|
savedViewPatchSpy.mockReturnValueOnce(
|
||||||
|
throwError(() => new Error('unable to save saved views'))
|
||||||
|
)
|
||||||
|
component.save()
|
||||||
|
expect(toastErrorSpy).toHaveBeenCalled()
|
||||||
|
expect(savedViewPatchSpy).toHaveBeenCalled()
|
||||||
|
toastSpy.mockClear()
|
||||||
|
toastErrorSpy.mockClear()
|
||||||
|
savedViewPatchSpy.mockClear()
|
||||||
|
|
||||||
|
// succeed saved views
|
||||||
|
savedViewPatchSpy.mockReturnValueOnce(of(savedViews as SavedView[]))
|
||||||
|
component.save()
|
||||||
|
expect(toastErrorSpy).not.toHaveBeenCalled()
|
||||||
|
expect(savedViewPatchSpy).toHaveBeenCalled()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should update only patch saved views that have changed', () => {
|
||||||
|
const patchSpy = jest.spyOn(savedViewService, 'patchMany')
|
||||||
|
component.save()
|
||||||
|
expect(patchSpy).not.toHaveBeenCalled()
|
||||||
|
|
||||||
|
const view = savedViews[0]
|
||||||
|
const toggle = fixture.debugElement.query(
|
||||||
|
By.css('.form-check.form-switch input')
|
||||||
|
)
|
||||||
|
toggle.nativeElement.checked = true
|
||||||
|
toggle.nativeElement.dispatchEvent(new Event('change'))
|
||||||
|
// register change
|
||||||
|
component.savedViewsForm.get('savedViews').get(view.id.toString()).value[
|
||||||
|
'show_on_dashboard'
|
||||||
|
] = !view.show_on_dashboard
|
||||||
|
fixture.detectChanges()
|
||||||
|
|
||||||
|
component.save()
|
||||||
|
expect(patchSpy).toHaveBeenCalledWith([
|
||||||
|
{
|
||||||
|
id: view.id,
|
||||||
|
name: view.name,
|
||||||
|
show_in_sidebar: view.show_in_sidebar,
|
||||||
|
show_on_dashboard: !view.show_on_dashboard,
|
||||||
|
},
|
||||||
|
])
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should support delete saved view', () => {
|
||||||
|
const toastSpy = jest.spyOn(toastService, 'showInfo')
|
||||||
|
const deleteSpy = jest.spyOn(savedViewService, 'delete')
|
||||||
|
deleteSpy.mockReturnValue(of(true))
|
||||||
|
component.deleteSavedView(savedViews[0] as SavedView)
|
||||||
|
expect(deleteSpy).toHaveBeenCalled()
|
||||||
|
expect(toastSpy).toHaveBeenCalledWith(
|
||||||
|
`Saved view "${savedViews[0].name}" deleted.`
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should support reset', () => {
|
||||||
|
const view = savedViews[0]
|
||||||
|
component.savedViewsForm.get('savedViews').get(view.id.toString()).value[
|
||||||
|
'show_on_dashboard'
|
||||||
|
] = !view.show_on_dashboard
|
||||||
|
component.reset()
|
||||||
|
expect(
|
||||||
|
component.savedViewsForm.get('savedViews').get(view.id.toString()).value[
|
||||||
|
'show_on_dashboard'
|
||||||
|
]
|
||||||
|
).toEqual(view.show_on_dashboard)
|
||||||
|
})
|
||||||
|
})
|
@ -0,0 +1,150 @@
|
|||||||
|
import { Component, OnDestroy, OnInit } from '@angular/core'
|
||||||
|
import { FormControl, FormGroup } from '@angular/forms'
|
||||||
|
import { SavedView } from 'src/app/data/saved-view'
|
||||||
|
import { SavedViewService } from 'src/app/services/rest/saved-view.service'
|
||||||
|
import { SettingsService } from 'src/app/services/settings.service'
|
||||||
|
import { ComponentWithPermissions } from '../../with-permissions/with-permissions.component'
|
||||||
|
import { DisplayMode } from 'src/app/data/document'
|
||||||
|
import { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs'
|
||||||
|
import { dirtyCheck } from '@ngneat/dirty-check-forms'
|
||||||
|
import { ToastService } from 'src/app/services/toast.service'
|
||||||
|
|
||||||
|
@Component({
|
||||||
|
selector: 'pngx-saved-views',
|
||||||
|
templateUrl: './saved-views.component.html',
|
||||||
|
styleUrl: './saved-views.component.scss',
|
||||||
|
})
|
||||||
|
export class SavedViewsComponent
|
||||||
|
extends ComponentWithPermissions
|
||||||
|
implements OnInit, OnDestroy
|
||||||
|
{
|
||||||
|
DisplayMode = DisplayMode
|
||||||
|
|
||||||
|
public savedViews: SavedView[]
|
||||||
|
private savedViewsGroup = new FormGroup({})
|
||||||
|
public savedViewsForm: FormGroup = new FormGroup({
|
||||||
|
savedViews: this.savedViewsGroup,
|
||||||
|
})
|
||||||
|
|
||||||
|
private store: BehaviorSubject<any>
|
||||||
|
private storeSub: Subscription
|
||||||
|
public isDirty$: Observable<boolean>
|
||||||
|
private isDirty: boolean = false
|
||||||
|
private unsubscribeNotifier: Subject<any> = new Subject()
|
||||||
|
private savePending: boolean = false
|
||||||
|
|
||||||
|
get displayFields() {
|
||||||
|
return this.settings.allDisplayFields
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
private savedViewService: SavedViewService,
|
||||||
|
private settings: SettingsService,
|
||||||
|
private toastService: ToastService
|
||||||
|
) {
|
||||||
|
super()
|
||||||
|
}
|
||||||
|
|
||||||
|
ngOnInit(): void {
|
||||||
|
this.settings.organizingSidebarSavedViews = true
|
||||||
|
|
||||||
|
this.savedViewService.listAll().subscribe((r) => {
|
||||||
|
this.savedViews = r.results
|
||||||
|
this.initialize()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
ngOnDestroy(): void {
|
||||||
|
this.settings.organizingSidebarSavedViews = false
|
||||||
|
this.unsubscribeNotifier.next(this)
|
||||||
|
this.unsubscribeNotifier.complete()
|
||||||
|
this.storeSub.unsubscribe()
|
||||||
|
}
|
||||||
|
|
||||||
|
private initialize() {
|
||||||
|
this.emptyGroup(this.savedViewsGroup)
|
||||||
|
|
||||||
|
let storeData = {
|
||||||
|
savedViews: {},
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let view of this.savedViews) {
|
||||||
|
storeData.savedViews[view.id.toString()] = {
|
||||||
|
id: view.id,
|
||||||
|
name: view.name,
|
||||||
|
show_on_dashboard: view.show_on_dashboard,
|
||||||
|
show_in_sidebar: view.show_in_sidebar,
|
||||||
|
page_size: view.page_size,
|
||||||
|
display_mode: view.display_mode,
|
||||||
|
display_fields: view.display_fields,
|
||||||
|
}
|
||||||
|
this.savedViewsGroup.addControl(
|
||||||
|
view.id.toString(),
|
||||||
|
new FormGroup({
|
||||||
|
id: new FormControl(null),
|
||||||
|
name: new FormControl(null),
|
||||||
|
show_on_dashboard: new FormControl(null),
|
||||||
|
show_in_sidebar: new FormControl(null),
|
||||||
|
page_size: new FormControl(null),
|
||||||
|
display_mode: new FormControl(null),
|
||||||
|
display_fields: new FormControl([]),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
this.store = new BehaviorSubject(storeData)
|
||||||
|
this.storeSub = this.store.asObservable().subscribe((state) => {
|
||||||
|
this.savedViewsForm.patchValue(state, { emitEvent: false })
|
||||||
|
})
|
||||||
|
|
||||||
|
// Initialize dirtyCheck
|
||||||
|
this.isDirty$ = dirtyCheck(this.savedViewsForm, this.store.asObservable())
|
||||||
|
}
|
||||||
|
|
||||||
|
public reset() {
|
||||||
|
this.savedViewsForm.patchValue(this.store.getValue())
|
||||||
|
}
|
||||||
|
|
||||||
|
public deleteSavedView(savedView: SavedView) {
|
||||||
|
this.savedViewService.delete(savedView).subscribe(() => {
|
||||||
|
this.savedViewsGroup.removeControl(savedView.id.toString())
|
||||||
|
this.savedViews.splice(this.savedViews.indexOf(savedView), 1)
|
||||||
|
this.toastService.showInfo(
|
||||||
|
$localize`Saved view "${savedView.name}" deleted.`
|
||||||
|
)
|
||||||
|
this.savedViewService.clearCache()
|
||||||
|
this.savedViewService.listAll().subscribe((r) => {
|
||||||
|
this.savedViews = r.results
|
||||||
|
this.initialize()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
private emptyGroup(group: FormGroup) {
|
||||||
|
Object.keys(group.controls).forEach((key) => group.removeControl(key))
|
||||||
|
}
|
||||||
|
|
||||||
|
public save() {
|
||||||
|
// only patch views that have actually changed
|
||||||
|
const changed: SavedView[] = []
|
||||||
|
Object.values(this.savedViewsGroup.controls)
|
||||||
|
.filter((g: FormGroup) => !g.pristine)
|
||||||
|
.forEach((group: FormGroup) => {
|
||||||
|
changed.push(group.value)
|
||||||
|
})
|
||||||
|
if (changed.length) {
|
||||||
|
this.savedViewService.patchMany(changed).subscribe({
|
||||||
|
next: () => {
|
||||||
|
this.toastService.showInfo($localize`Views saved successfully.`)
|
||||||
|
this.store.next(this.savedViewsForm.value)
|
||||||
|
},
|
||||||
|
error: (error) => {
|
||||||
|
this.toastService.showError(
|
||||||
|
$localize`Error while saving views.`,
|
||||||
|
error
|
||||||
|
)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user