mirror of
https://github.com/grafana/grafana.git
synced 2025-02-20 11:48:34 -06:00
* user essentials mob! 🔱 lastFile:pkg/services/libraryelements/writers.go * user essentials mob! 🔱 lastFile:pkg/services/libraryelements/writers.go * user essentials mob! 🔱 lastFile:pkg/services/libraryelements/writers.go * user essentials mob! 🔱 lastFile:pkg/services/libraryelements/writers.go * user essentials mob! 🔱 lastFile:pkg/services/libraryelements/database.go * user essentials mob! 🔱 lastFile:pkg/services/libraryelements/writers.go * user essentials mob! 🔱 lastFile:pkg/services/libraryelements/writers.go * user essentials mob! 🔱 * support filterFolderUIDs in the frontend * move common logic to a variable * fixed FolderLibraryPanelsPage and improved unit test * fix backend lint error * fix formatting error Co-authored-by: Joao Silva <joao.silva@grafana.com> Co-authored-by: Ashley Harrison <ashley.harrison@grafana.com> Co-authored-by: eledobleefe <laura.fernandez@grafana.com> Co-authored-by: joshhunt <josh@trtr.co>
74 lines
2.4 KiB
TypeScript
74 lines
2.4 KiB
TypeScript
import { AnyAction } from '@reduxjs/toolkit';
|
|
import { Dispatch } from 'react';
|
|
import { from, merge, of, Subscription, timer } from 'rxjs';
|
|
import { catchError, finalize, mapTo, mergeMap, share, takeUntil } from 'rxjs/operators';
|
|
|
|
import { deleteLibraryPanel as apiDeleteLibraryPanel, getLibraryPanels } from '../../state/api';
|
|
|
|
import { initialLibraryPanelsViewState, initSearch, searchCompleted } from './reducer';
|
|
|
|
type DispatchResult = (dispatch: Dispatch<AnyAction>) => void;
|
|
interface SearchArgs {
|
|
perPage: number;
|
|
page: number;
|
|
searchString: string;
|
|
sortDirection?: string;
|
|
panelFilter?: string[];
|
|
folderFilterUIDs?: string[];
|
|
currentPanelId?: string;
|
|
}
|
|
|
|
export function searchForLibraryPanels(args: SearchArgs): DispatchResult {
|
|
return function (dispatch) {
|
|
const subscription = new Subscription();
|
|
const dataObservable = from(
|
|
getLibraryPanels({
|
|
searchString: args.searchString,
|
|
perPage: args.perPage,
|
|
page: args.page,
|
|
excludeUid: args.currentPanelId,
|
|
sortDirection: args.sortDirection,
|
|
typeFilter: args.panelFilter,
|
|
folderFilterUIDs: args.folderFilterUIDs,
|
|
})
|
|
).pipe(
|
|
mergeMap(({ perPage, elements: libraryPanels, page, totalCount }) =>
|
|
of(searchCompleted({ libraryPanels, page, perPage, totalCount }))
|
|
),
|
|
catchError((err) => {
|
|
console.error(err);
|
|
return of(searchCompleted({ ...initialLibraryPanelsViewState, page: args.page, perPage: args.perPage }));
|
|
}),
|
|
finalize(() => subscription.unsubscribe()), // make sure we unsubscribe
|
|
share()
|
|
);
|
|
|
|
subscription.add(
|
|
// If 50ms without a response dispatch a loading state
|
|
// mapTo will translate the timer event into a loading state
|
|
// takeUntil will cancel the timer emit when first response is received on the dataObservable
|
|
merge(timer(50).pipe(mapTo(initSearch()), takeUntil(dataObservable)), dataObservable).subscribe(dispatch)
|
|
);
|
|
};
|
|
}
|
|
|
|
export function deleteLibraryPanel(uid: string, args: SearchArgs): DispatchResult {
|
|
return async function (dispatch) {
|
|
try {
|
|
await apiDeleteLibraryPanel(uid);
|
|
searchForLibraryPanels(args)(dispatch);
|
|
} catch (e) {
|
|
console.error(e);
|
|
}
|
|
};
|
|
}
|
|
|
|
export function asyncDispatcher(dispatch: Dispatch<AnyAction>) {
|
|
return function (action: any) {
|
|
if (action instanceof Function) {
|
|
return action(dispatch);
|
|
}
|
|
return dispatch(action);
|
|
};
|
|
}
|