mirror of
				https://github.com/reduxjs/redux-devtools.git
				synced 2025-10-25 21:21:11 +03:00 
			
		
		
		
	[feat][monitor] Add rkt-query-inspector-monitor - feat/rtk query monitor (#750)
* chore: copy rtk-query example from toolkit
* feat(rtk-query): complete initial setup of rtk-query
* feat: complete inspector layout and add initial JSONTree setup
* fix: unintentional removal of tsconfig
* feat(search): add search logic and refactor monitor state shape
* fix: inverted monitor theme inside  devtoop-app
Othetr changes:
* simplify monitor integration
* fix: rtk monitor reducer not working in  app
* refactor(rtk-query): simplify theme config
* feat(rtk-query-monitor): add query preview tabs
* fix: wip
* refactor(examples): add rtk-query-polling to workspace
Other changes:
* docs(rtk-query-polling): add README.md
* chore(rtk-query-inspector): add demo to monorepo
Other changes:
chore: increase isWideScreen polling interval to 300
refactor: add subscription as root node in QueryPreviewSubscriptions
* feat(rtk-query): add multiple filter options
* chore(rtk-queery): rename demo build script and add SKIP_PREFLIGHT_CHECK=true
* feat(rtk-query): display status flags in QueryPreviewInfo
* chore(rtk-query): update typescript versions in rkt-inspector-monitor & its demo
* docs(rtk-query): add proper README
Other changes:
* fix examples/rtk-query-poilling
* docs(rtk-query): improve rtk-query-inspector-monitor demo gif
* docs(rtk-query): clean up demo
* fix(rtk-query): clear button not updating redux state
* docs(rtk-query): add link to rtk-query-inspector-monitor demo site
* chore(rtk-query): run prettier after prettier upgrade (55e2284)
* docs(rtk.query): clean up readme add features, todo and development section
* docs(rtk-query): fix link href
* chore(rtk-query): clean up rtk-query-imspector-monitor-demo and add post example
* feat(rtk-query): add counters on tags & subs tab buttons
* fix(rtk-query): layering issue between queryPreview tabList and select
Other changes:
* clean up demo styles
* run prettier
* fix: revert accidental changes packages/redux-devtools-serialize/tsconfig.json
* chore: change QueryComparators.fulfilledTimeStamp label
* feat(rtk-query): display api stats
* refactor: remove rtk-query-polling example from monorepo
* chore: regenerate lock file and add @types/react as monorepo devDep
* chore: display apiState
Other changes:
* fix close button overflow
* minor responsive style tweaks
* display reducerPath in query tab
* fix(rtk-query): resolve CI errors
- fix(prettier:check): unformatted file
- fix(lint:all): fix accidentallly removed .eslintignore
* chore(rtk-query): rename package to '@redux-devtools/rtk-query-monitor'
* fix(rtk-query): lint:all error
https://github.com/reduxjs/redux-devtools/runs/2869729048?check_suite_focus=true
* feat(rtk-query): add fallback message if there are no rtk-query apis
* refactor(rtk-query): move Inspector & Monitor to containers clean up typings
Other changes:
* chore: improved type coverage
* chore: do not lint packages/redux-devtools-rtk-query-monitor/demo folder
* refactor: put sort order buttons inside a component
* chore: hopefully resolve mockServiceWorker formatting issues
* fix(rtk-query): incorrect link color
Other changes:
* fix: add missing anchor property value noopener
* refactor(rtk-query): simplify sort order control
* feat(rtk-query): add timings to apiStats sections
* feat(rtk-query): add slowest and fastest in timings section
* feat(rtk-query): improve formatting of timings and display average loading time
* fix(rtk-query): rtk-query imports
* refactor(rtk-query): reduce selector computations
Other changes:
* simplify TreeView props
* feat(rtk-query): add actions tab
* refactor(rtk-query): add custom logic for TreeView shouldExpandNode
Other changes:
* feat: added duration in QueryPreviewInfo tab
* refactor: TreeView component
* chore(rtk-query): improve demo visibility on small devices
* feat(rtk-query): do not display tree node preview
Other changes:
* improve visibility of demo devTools on small devices
* tweak QueryPreviewInfo labels
* chore(rtk-query): improve responsiveness
* refactor(rtk-query): move preview to containers remove unnecessary computation
* feat(rtk-query): display median of timings
Other changes:
* improved shouldExpandNode logic of QueryPreviewActions
* tweaked mean logic
* refactor(rtk-query-monitor): conform demo setup to repo standards
* chore(rtk-query-monitor): add option to select active devtools
* chore(rtk-query-monitor): remove redux-devtools/examples/rtk-query-polling
* refactor(rtk-query): improve UI of api tab
* feat(rtk-query): add regex search
* feat(rtk-query): display mutations in queryList
* refactor(rtk-query): track all fulfilled requests using actions
Other changes:
* refactor(rtk-query): rename tally properties
* chore(rtk-query): update @redux-devtools/rtk-query-monitor dependencies
* fix(rtk-query): demo build failing caused by a typing error
			
			
This commit is contained in:
		
							parent
							
								
									6cdc18f2fa
								
							
						
					
					
						commit
						7d92a5e186
					
				|  | @ -9,3 +9,5 @@ node_modules | |||
| __snapshots__ | ||||
| dev | ||||
| .yarn/* | ||||
| **/.yarn/* | ||||
| **/demo/public/** | ||||
|  | @ -50,6 +50,9 @@ | |||
|     "webpack-cli": "^3.3.12", | ||||
|     "webpack-dev-server": "^3.11.2" | ||||
|   }, | ||||
|   "resolutions": { | ||||
|     "@types/react": "16.14.8" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "lerna": "lerna", | ||||
|     "build:all": "lerna run build", | ||||
|  |  | |||
|  | @ -43,7 +43,9 @@ | |||
|     "@redux-devtools/inspector-monitor-test-tab": "^0.7.2", | ||||
|     "@redux-devtools/inspector-monitor-trace-tab": "^0.2.2", | ||||
|     "@redux-devtools/log-monitor": "^2.3.0", | ||||
|     "@redux-devtools/rtk-query-monitor": "^1.0.0", | ||||
|     "@redux-devtools/slider-monitor": "^2.0.0-8", | ||||
|     "@reduxjs/toolkit": "^1.6.0", | ||||
|     "d3-state-visualizer": "^1.4.0", | ||||
|     "devui": "^1.0.0-9", | ||||
|     "javascript-stringify": "^2.1.0", | ||||
|  |  | |||
|  | @ -2,11 +2,13 @@ import React from 'react'; | |||
| import LogMonitor from '@redux-devtools/log-monitor'; | ||||
| import ChartMonitorWrapper from '../containers/monitors/ChartMonitorWrapper'; | ||||
| import InspectorWrapper from '../containers/monitors/InspectorWrapper'; | ||||
| import RtkQueryMonitor from '@redux-devtools/rtk-query-monitor'; | ||||
| 
 | ||||
| export const monitors = [ | ||||
|   { value: 'InspectorMonitor', name: 'Inspector' }, | ||||
|   { value: 'LogMonitor', name: 'Log monitor' }, | ||||
|   { value: 'ChartMonitor', name: 'Chart' }, | ||||
|   { value: 'RtkQueryMonitor', name: 'RTK Query' }, | ||||
| ]; | ||||
| 
 | ||||
| export default function getMonitor({ monitor }: { monitor: string }) { | ||||
|  | @ -17,6 +19,8 @@ export default function getMonitor({ monitor }: { monitor: string }) { | |||
|       ); | ||||
|     case 'ChartMonitor': | ||||
|       return <ChartMonitorWrapper />; | ||||
|     case 'RtkQueryMonitor': | ||||
|       return <RtkQueryMonitor />; | ||||
|     default: | ||||
|       return <InspectorWrapper />; | ||||
|   } | ||||
|  |  | |||
							
								
								
									
										0
									
								
								packages/redux-devtools-cli/bin/redux-devtools.js
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										0
									
								
								packages/redux-devtools-cli/bin/redux-devtools.js
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
								
								
									
										8
									
								
								packages/redux-devtools-rtk-query-monitor/.babelrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								packages/redux-devtools-rtk-query-monitor/.babelrc
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,8 @@ | |||
| { | ||||
|   "presets": [ | ||||
|     "@babel/preset-env", | ||||
|     "@babel/preset-react", | ||||
|     "@babel/preset-typescript" | ||||
|   ], | ||||
|   "plugins": ["@babel/plugin-proposal-class-properties"] | ||||
| } | ||||
							
								
								
									
										2
									
								
								packages/redux-devtools-rtk-query-monitor/.eslintignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								packages/redux-devtools-rtk-query-monitor/.eslintignore
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | |||
| lib | ||||
| demo/ | ||||
							
								
								
									
										13
									
								
								packages/redux-devtools-rtk-query-monitor/.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								packages/redux-devtools-rtk-query-monitor/.eslintrc.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,13 @@ | |||
| module.exports = { | ||||
|   extends: '../../.eslintrc', | ||||
|   overrides: [ | ||||
|     { | ||||
|       files: ['*.ts', '*.tsx'], | ||||
|       extends: '../../eslintrc.ts.react.base.json', | ||||
|       parserOptions: { | ||||
|         tsconfigRootDir: __dirname, | ||||
|         project: ['./tsconfig.json'], | ||||
|       }, | ||||
|     }, | ||||
|   ], | ||||
| }; | ||||
							
								
								
									
										21
									
								
								packages/redux-devtools-rtk-query-monitor/LICENSE.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								packages/redux-devtools-rtk-query-monitor/LICENSE.md
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,21 @@ | |||
| The MIT License (MIT) | ||||
| 
 | ||||
| Copyright (c) 2021 Fabrizio Vitale | ||||
| 
 | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
| 
 | ||||
| The above copyright notice and this permission notice shall be included in all | ||||
| copies or substantial portions of the Software. | ||||
| 
 | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||||
| SOFTWARE. | ||||
							
								
								
									
										108
									
								
								packages/redux-devtools-rtk-query-monitor/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								packages/redux-devtools-rtk-query-monitor/README.md
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,108 @@ | |||
| # Redux DevTools RTK Query inspector monitor | ||||
| 
 | ||||
| A monitor that displays [RTK query](https://redux-toolkit.js.org/rtk-query/overview) queries and mutations for [Redux DevTools](https://github.com/gaearon/redux-devtools). | ||||
| 
 | ||||
| Created by [FaberVitale](https://github.com/FaberVitale), inspired by [react-query devtools](https://github.com/tannerlinsley/react-query/tree/master/devtools). | ||||
| 
 | ||||
| ## Demo | ||||
| 
 | ||||
| - [link](https://rtk-query-monitor-demo.netlify.app/) | ||||
| - [demo source](https://github.com/FaberVitale/redux-devtools/tree/feat/rtk-query-monitor/packages/redux-devtools-rtk-query-monitor/demo) | ||||
| 
 | ||||
| ## Preview | ||||
| 
 | ||||
|  | ||||
| 
 | ||||
| ## Installation | ||||
| 
 | ||||
| ### npm | ||||
| 
 | ||||
| ```bash | ||||
| npm i @redux-devtools/rtk-query-monitor --save | ||||
| ``` | ||||
| 
 | ||||
| ### yarn | ||||
| 
 | ||||
| ```bash | ||||
| yarn add @redux-devtools/rtk-query-monitor | ||||
| ``` | ||||
| 
 | ||||
| ## Usage | ||||
| 
 | ||||
| You can use `RtkQueryMonitor` as the only monitor in your app: | ||||
| 
 | ||||
| ##### `containers/DevTools.js` | ||||
| 
 | ||||
| ```ts | ||||
| import React from 'react'; | ||||
| import { createDevTools } from '@redux-devtools/core'; | ||||
| import RtkQueryrMonitor from '@redux-devtools/rtk-query-monitor'; | ||||
| 
 | ||||
| export default createDevTools(<RtkQueryrMonitor />); | ||||
| ``` | ||||
| 
 | ||||
| Then you can render `<DevTools>` to any place inside app or even into a separate popup window. | ||||
| 
 | ||||
| Alternatively, you can use it together with [`DockMonitor`](https://github.com/reduxjs/redux-devtools/tree/master/packages/redux-devtools-dock-monitor) to make it dockable. | ||||
| 
 | ||||
| See also | ||||
| 
 | ||||
| - [`DockMonitor` README](https://github.com/reduxjs/redux-devtools/tree/master/packages/redux-devtools-dock-monitor) | ||||
| 
 | ||||
| - [Read how to start using Redux DevTools.](https://github.com/reduxjs/redux-devtools) | ||||
| 
 | ||||
| - [Redux Devtools walkthrough](https://github.com/reduxjs/redux-devtools/tree/master/docs/Walkthrough.md) | ||||
| 
 | ||||
| ## Features | ||||
| 
 | ||||
| - sorts queries in ascending or descending order by: | ||||
|   - fulfilledTimeStamp | ||||
|   - query key | ||||
|   - query status | ||||
|   - endpoint | ||||
|   - api reducerPath | ||||
| - filters queries by: | ||||
|   - fulfilledTimeStamp | ||||
|   - query key | ||||
|   - query status | ||||
|   - endpoint | ||||
|   - api reducerPath | ||||
| - displays | ||||
|   - status flags | ||||
|   - query state | ||||
|   - tags | ||||
|   - subscriptions | ||||
|   - api state | ||||
|   - api stats | ||||
| 
 | ||||
| ## TODO | ||||
| 
 | ||||
| - [ ] display mutations | ||||
| - [ ] filter by tags types | ||||
| - [ ] download query.data | ||||
| - [ ] upload query.data(?) | ||||
| - [ ] refetch query button(?) | ||||
| - ...suggestions are welcome | ||||
| 
 | ||||
| ## Redux DevTools props | ||||
| 
 | ||||
| | Name          | Description                                                                                                                                                                                                                                                                                                                         | | ||||
| | ------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | ||||
| | `theme`       | Either a string referring to one of the themes provided by [redux-devtools-themes](https://github.com/gaearon/redux-devtools-themes) (feel free to contribute!) or a custom object of the same format. Optional. By default, set to [`'nicinabox'`](https://github.com/gaearon/redux-devtools-themes/blob/master/src/nicinabox.js). | | ||||
| | `invertTheme` | Boolean value that will invert the colors of the selected theme. Optional. By default, set to `false`                                                                                                                                                                                                                               | | ||||
| 
 | ||||
| <br/> | ||||
| 
 | ||||
| ### Development | ||||
| 
 | ||||
| #### Start Demo | ||||
| 
 | ||||
| ```bash | ||||
| yarn lerna run start --stream --scope @redux-devtools/rtk-query-monitor | ||||
| ``` | ||||
| 
 | ||||
| <br/> | ||||
| 
 | ||||
| ## License | ||||
| 
 | ||||
| [MIT](./LICENSE.md) | ||||
							
								
								
									
										11
									
								
								packages/redux-devtools-rtk-query-monitor/demo/.babelrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/redux-devtools-rtk-query-monitor/demo/.babelrc
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | |||
| { | ||||
|   "presets": [ | ||||
|     "@babel/preset-env", | ||||
|     "@babel/preset-react", | ||||
|     "@babel/preset-typescript" | ||||
|   ], | ||||
|   "plugins": [ | ||||
|     "@babel/plugin-proposal-class-properties", | ||||
|     "@babel/plugin-transform-runtime" | ||||
|   ] | ||||
| } | ||||
|  | @ -0,0 +1,3 @@ | |||
| module.exports = { | ||||
|   extends: ['react-app'], | ||||
| }; | ||||
							
								
								
									
										4
									
								
								packages/redux-devtools-rtk-query-monitor/demo/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								packages/redux-devtools-rtk-query-monitor/demo/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,4 @@ | |||
| .snowpack | ||||
| build | ||||
| node_modules | ||||
| .yarn/* | ||||
|  | @ -0,0 +1,7 @@ | |||
| { | ||||
|   "extends": "../../../../tsconfig.base.json", | ||||
|   "compilerOptions": { | ||||
|     "resolveJsonModule": true | ||||
|   }, | ||||
|   "include": ["webpack.config.ts"] | ||||
| } | ||||
|  | @ -0,0 +1,88 @@ | |||
| import * as path from 'path'; | ||||
| import * as webpack from 'webpack'; | ||||
| import HtmlWebpackPlugin from 'html-webpack-plugin'; | ||||
| import { CleanWebpackPlugin } from 'clean-webpack-plugin'; | ||||
| import ForkTsCheckerWebpackPlugin from 'fork-ts-checker-webpack-plugin'; | ||||
| import * as pkg from '../../package.json'; | ||||
| const CopyWebpackPlugin = require('copy-webpack-plugin'); | ||||
| const isProduction = process.env.NODE_ENV === 'production'; | ||||
| 
 | ||||
| const demoSrc = path.join(__dirname, '../src'); | ||||
| const libSrc = path.join(__dirname, '../../src'); | ||||
| const publicDir = path.join(__dirname, '../public'); | ||||
| 
 | ||||
| module.exports = { | ||||
|   mode: process.env.NODE_ENV || 'development', | ||||
|   entry: isProduction | ||||
|     ? ['./demo/src/index'] | ||||
|     : [ | ||||
|         'webpack-dev-server/client?http://localhost:3000', | ||||
|         'webpack/hot/only-dev-server', | ||||
|         './demo/src/index', | ||||
|       ], | ||||
|   output: { | ||||
|     path: path.join(__dirname, '../dist'), | ||||
|     filename: isProduction ? '[name].[contenthash:8].js' : 'js/bundle.js', | ||||
|   }, | ||||
|   module: { | ||||
|     rules: [ | ||||
|       { | ||||
|         test: /\.(js|ts)x?$/, | ||||
|         loader: 'babel-loader', | ||||
|         exclude: /node_modules/, | ||||
|         include: [demoSrc, libSrc], | ||||
|       }, | ||||
|       { | ||||
|         test: /\.css?$/, | ||||
|         loaders: ['style-loader', 'css-loader'], | ||||
|         include: demoSrc, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   resolve: { | ||||
|     modules: ['node_modules', demoSrc], | ||||
|     extensions: ['.js', '.jsx', '.ts', '.tsx'], | ||||
|   }, | ||||
|   optimization: { | ||||
|     minimize: isProduction, | ||||
|   }, | ||||
|   plugins: [ | ||||
|     new CleanWebpackPlugin(), | ||||
|     new HtmlWebpackPlugin({ | ||||
|       inject: true, | ||||
|       template: 'demo/public/index.html', | ||||
|       package: pkg, | ||||
|     }), | ||||
|     new CopyWebpackPlugin({ | ||||
|       patterns: [ | ||||
|         { | ||||
|           from: 'demo/public/assets/*.js', | ||||
|           to: ({ absoluteFilename }: any) => { | ||||
|             return `./${path.basename(absoluteFilename)}`; | ||||
|           }, | ||||
|           globOptions: { | ||||
|             ignore: ['*.DS_Store'], | ||||
|           }, | ||||
|         }, | ||||
|       ], | ||||
|     }), | ||||
|     new ForkTsCheckerWebpackPlugin({ | ||||
|       typescript: { | ||||
|         configFile: 'demo/tsconfig.json', | ||||
|       }, | ||||
|     }), | ||||
|   ].concat(isProduction ? [] : [new webpack.HotModuleReplacementPlugin()]), | ||||
|   devServer: isProduction | ||||
|     ? {} | ||||
|     : { | ||||
|         quiet: false, | ||||
|         port: 3000, | ||||
|         hot: true, | ||||
|         stats: { | ||||
|           chunkModules: false, | ||||
|           colors: true, | ||||
|         }, | ||||
|         historyApiFallback: true, | ||||
|       }, | ||||
|   devtool: isProduction ? 'source-map' : 'cheap-module-source-map', | ||||
| }; | ||||
|  | @ -0,0 +1,322 @@ | |||
| /** | ||||
|  * Mock Service Worker. | ||||
|  * @see https://github.com/mswjs/msw
 | ||||
|  * - Please do NOT modify this file. | ||||
|  * - Please do NOT serve this file on production. | ||||
|  */ | ||||
| /* eslint-disable */ | ||||
| /* tslint:disable */ | ||||
| 
 | ||||
| const INTEGRITY_CHECKSUM = '82ef9b96d8393b6da34527d1d6e19187' | ||||
| const bypassHeaderName = 'x-msw-bypass' | ||||
| const activeClientIds = new Set() | ||||
| 
 | ||||
| self.addEventListener('install', function () { | ||||
|   return self.skipWaiting() | ||||
| }) | ||||
| 
 | ||||
| self.addEventListener('activate', async function (event) { | ||||
|   return self.clients.claim() | ||||
| }) | ||||
| 
 | ||||
| self.addEventListener('message', async function (event) { | ||||
|   const clientId = event.source.id | ||||
| 
 | ||||
|   if (!clientId || !self.clients) { | ||||
|     return | ||||
|   } | ||||
| 
 | ||||
|   const client = await self.clients.get(clientId) | ||||
| 
 | ||||
|   if (!client) { | ||||
|     return | ||||
|   } | ||||
| 
 | ||||
|   const allClients = await self.clients.matchAll() | ||||
| 
 | ||||
|   switch (event.data) { | ||||
|     case 'KEEPALIVE_REQUEST': { | ||||
|       sendToClient(client, { | ||||
|         type: 'KEEPALIVE_RESPONSE', | ||||
|       }) | ||||
|       break | ||||
|     } | ||||
| 
 | ||||
|     case 'INTEGRITY_CHECK_REQUEST': { | ||||
|       sendToClient(client, { | ||||
|         type: 'INTEGRITY_CHECK_RESPONSE', | ||||
|         payload: INTEGRITY_CHECKSUM, | ||||
|       }) | ||||
|       break | ||||
|     } | ||||
| 
 | ||||
|     case 'MOCK_ACTIVATE': { | ||||
|       activeClientIds.add(clientId) | ||||
| 
 | ||||
|       sendToClient(client, { | ||||
|         type: 'MOCKING_ENABLED', | ||||
|         payload: true, | ||||
|       }) | ||||
|       break | ||||
|     } | ||||
| 
 | ||||
|     case 'MOCK_DEACTIVATE': { | ||||
|       activeClientIds.delete(clientId) | ||||
|       break | ||||
|     } | ||||
| 
 | ||||
|     case 'CLIENT_CLOSED': { | ||||
|       activeClientIds.delete(clientId) | ||||
| 
 | ||||
|       const remainingClients = allClients.filter((client) => { | ||||
|         return client.id !== clientId | ||||
|       }) | ||||
| 
 | ||||
|       // Unregister itself when there are no more clients
 | ||||
|       if (remainingClients.length === 0) { | ||||
|         self.registration.unregister() | ||||
|       } | ||||
| 
 | ||||
|       break | ||||
|     } | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| // Resolve the "master" client for the given event.
 | ||||
| // Client that issues a request doesn't necessarily equal the client
 | ||||
| // that registered the worker. It's with the latter the worker should
 | ||||
| // communicate with during the response resolving phase.
 | ||||
| async function resolveMasterClient(event) { | ||||
|   const client = await self.clients.get(event.clientId) | ||||
| 
 | ||||
|   if (client.frameType === 'top-level') { | ||||
|     return client | ||||
|   } | ||||
| 
 | ||||
|   const allClients = await self.clients.matchAll() | ||||
| 
 | ||||
|   return allClients | ||||
|     .filter((client) => { | ||||
|       // Get only those clients that are currently visible.
 | ||||
|       return client.visibilityState === 'visible' | ||||
|     }) | ||||
|     .find((client) => { | ||||
|       // Find the client ID that's recorded in the
 | ||||
|       // set of clients that have registered the worker.
 | ||||
|       return activeClientIds.has(client.id) | ||||
|     }) | ||||
| } | ||||
| 
 | ||||
| async function handleRequest(event, requestId) { | ||||
|   const client = await resolveMasterClient(event) | ||||
|   const response = await getResponse(event, client, requestId) | ||||
| 
 | ||||
|   // Send back the response clone for the "response:*" life-cycle events.
 | ||||
|   // Ensure MSW is active and ready to handle the message, otherwise
 | ||||
|   // this message will pend indefinitely.
 | ||||
|   if (client && activeClientIds.has(client.id)) { | ||||
|     ;(async function () { | ||||
|       const clonedResponse = response.clone() | ||||
|       sendToClient(client, { | ||||
|         type: 'RESPONSE', | ||||
|         payload: { | ||||
|           requestId, | ||||
|           type: clonedResponse.type, | ||||
|           ok: clonedResponse.ok, | ||||
|           status: clonedResponse.status, | ||||
|           statusText: clonedResponse.statusText, | ||||
|           body: | ||||
|             clonedResponse.body === null ? null : await clonedResponse.text(), | ||||
|           headers: serializeHeaders(clonedResponse.headers), | ||||
|           redirected: clonedResponse.redirected, | ||||
|         }, | ||||
|       }) | ||||
|     })() | ||||
|   } | ||||
| 
 | ||||
|   return response | ||||
| } | ||||
| 
 | ||||
| async function getResponse(event, client, requestId) { | ||||
|   const { request } = event | ||||
|   const requestClone = request.clone() | ||||
|   const getOriginalResponse = () => fetch(requestClone) | ||||
| 
 | ||||
|   // Bypass mocking when the request client is not active.
 | ||||
|   if (!client) { | ||||
|     return getOriginalResponse() | ||||
|   } | ||||
| 
 | ||||
|   // Bypass initial page load requests (i.e. static assets).
 | ||||
|   // The absence of the immediate/parent client in the map of the active clients
 | ||||
|   // means that MSW hasn't dispatched the "MOCK_ACTIVATE" event yet
 | ||||
|   // and is not ready to handle requests.
 | ||||
|   if (!activeClientIds.has(client.id)) { | ||||
|     return await getOriginalResponse() | ||||
|   } | ||||
| 
 | ||||
|   // Bypass requests with the explicit bypass header
 | ||||
|   if (requestClone.headers.get(bypassHeaderName) === 'true') { | ||||
|     const cleanRequestHeaders = serializeHeaders(requestClone.headers) | ||||
| 
 | ||||
|     // Remove the bypass header to comply with the CORS preflight check.
 | ||||
|     delete cleanRequestHeaders[bypassHeaderName] | ||||
| 
 | ||||
|     const originalRequest = new Request(requestClone, { | ||||
|       headers: new Headers(cleanRequestHeaders), | ||||
|     }) | ||||
| 
 | ||||
|     return fetch(originalRequest) | ||||
|   } | ||||
| 
 | ||||
|   // Send the request to the client-side MSW.
 | ||||
|   const reqHeaders = serializeHeaders(request.headers) | ||||
|   const body = await request.text() | ||||
| 
 | ||||
|   const clientMessage = await sendToClient(client, { | ||||
|     type: 'REQUEST', | ||||
|     payload: { | ||||
|       id: requestId, | ||||
|       url: request.url, | ||||
|       method: request.method, | ||||
|       headers: reqHeaders, | ||||
|       cache: request.cache, | ||||
|       mode: request.mode, | ||||
|       credentials: request.credentials, | ||||
|       destination: request.destination, | ||||
|       integrity: request.integrity, | ||||
|       redirect: request.redirect, | ||||
|       referrer: request.referrer, | ||||
|       referrerPolicy: request.referrerPolicy, | ||||
|       body, | ||||
|       bodyUsed: request.bodyUsed, | ||||
|       keepalive: request.keepalive, | ||||
|     }, | ||||
|   }) | ||||
| 
 | ||||
|   switch (clientMessage.type) { | ||||
|     case 'MOCK_SUCCESS': { | ||||
|       return delayPromise( | ||||
|         () => respondWithMock(clientMessage), | ||||
|         clientMessage.payload.delay, | ||||
|       ) | ||||
|     } | ||||
| 
 | ||||
|     case 'MOCK_NOT_FOUND': { | ||||
|       return getOriginalResponse() | ||||
|     } | ||||
| 
 | ||||
|     case 'NETWORK_ERROR': { | ||||
|       const { name, message } = clientMessage.payload | ||||
|       const networkError = new Error(message) | ||||
|       networkError.name = name | ||||
| 
 | ||||
|       // Rejecting a request Promise emulates a network error.
 | ||||
|       throw networkError | ||||
|     } | ||||
| 
 | ||||
|     case 'INTERNAL_ERROR': { | ||||
|       const parsedBody = JSON.parse(clientMessage.payload.body) | ||||
| 
 | ||||
|       console.error( | ||||
|         `\
 | ||||
| [MSW] Request handler function for "%s %s" has thrown the following exception: | ||||
| 
 | ||||
| ${parsedBody.errorType}: ${parsedBody.message} | ||||
| (see more detailed error stack trace in the mocked response body) | ||||
| 
 | ||||
| This exception has been gracefully handled as a 500 response, however, it's strongly recommended to resolve this error. | ||||
| If you wish to mock an error response, please refer to this guide: https://mswjs.io/docs/recipes/mocking-error-responses\
 | ||||
| `,
 | ||||
|         request.method, | ||||
|         request.url, | ||||
|       ) | ||||
| 
 | ||||
|       return respondWithMock(clientMessage) | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return getOriginalResponse() | ||||
| } | ||||
| 
 | ||||
| self.addEventListener('fetch', function (event) { | ||||
|   const { request } = event | ||||
| 
 | ||||
|   // Bypass navigation requests.
 | ||||
|   if (request.mode === 'navigate') { | ||||
|     return | ||||
|   } | ||||
| 
 | ||||
|   // Opening the DevTools triggers the "only-if-cached" request
 | ||||
|   // that cannot be handled by the worker. Bypass such requests.
 | ||||
|   if (request.cache === 'only-if-cached' && request.mode !== 'same-origin') { | ||||
|     return | ||||
|   } | ||||
| 
 | ||||
|   // Bypass all requests when there are no active clients.
 | ||||
|   // Prevents the self-unregistered worked from handling requests
 | ||||
|   // after it's been deleted (still remains active until the next reload).
 | ||||
|   if (activeClientIds.size === 0) { | ||||
|     return | ||||
|   } | ||||
| 
 | ||||
|   const requestId = uuidv4() | ||||
| 
 | ||||
|   return event.respondWith( | ||||
|     handleRequest(event, requestId).catch((error) => { | ||||
|       console.error( | ||||
|         '[MSW] Failed to mock a "%s" request to "%s": %s', | ||||
|         request.method, | ||||
|         request.url, | ||||
|         error, | ||||
|       ) | ||||
|     }), | ||||
|   ) | ||||
| }) | ||||
| 
 | ||||
| function serializeHeaders(headers) { | ||||
|   const reqHeaders = {} | ||||
|   headers.forEach((value, name) => { | ||||
|     reqHeaders[name] = reqHeaders[name] | ||||
|       ? [].concat(reqHeaders[name]).concat(value) | ||||
|       : value | ||||
|   }) | ||||
|   return reqHeaders | ||||
| } | ||||
| 
 | ||||
| function sendToClient(client, message) { | ||||
|   return new Promise((resolve, reject) => { | ||||
|     const channel = new MessageChannel() | ||||
| 
 | ||||
|     channel.port1.onmessage = (event) => { | ||||
|       if (event.data && event.data.error) { | ||||
|         return reject(event.data.error) | ||||
|       } | ||||
| 
 | ||||
|       resolve(event.data) | ||||
|     } | ||||
| 
 | ||||
|     client.postMessage(JSON.stringify(message), [channel.port2]) | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| function delayPromise(cb, duration) { | ||||
|   return new Promise((resolve) => { | ||||
|     setTimeout(() => resolve(cb()), duration) | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| function respondWithMock(clientMessage) { | ||||
|   return new Response(clientMessage.payload.body, { | ||||
|     ...clientMessage.payload, | ||||
|     headers: clientMessage.payload.headers, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| function uuidv4() { | ||||
|   return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) { | ||||
|     const r = (Math.random() * 16) | 0 | ||||
|     const v = c == 'x' ? r : (r & 0x3) | 0x8 | ||||
|     return v.toString(16) | ||||
|   }) | ||||
| } | ||||
|  | @ -0,0 +1,27 @@ | |||
| <!DOCTYPE html> | ||||
| <html lang="en"> | ||||
|   <head> | ||||
|     <meta charset="utf-8" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1" /> | ||||
|     <meta | ||||
|       name="description" | ||||
|       content="@redux-devtools/rtk-query-monitor demo site" | ||||
|     /> | ||||
|     <title>RTK Query monitor demo</title> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="root"></div> | ||||
|     <noscript>You need to enable JavaScript to run this app.</noscript> | ||||
|     <script type="module" src="/dist/index.js"></script> | ||||
|     <!-- | ||||
|       This HTML file is a template. | ||||
|       If you open it directly in the browser, you will see an empty page. | ||||
| 
 | ||||
|       You can add webfonts, meta tags, or analytics to this file. | ||||
|       The build step will place the bundled scripts into the <body> tag. | ||||
| 
 | ||||
|       To begin the development, run `npm start` or `yarn start`. | ||||
|       To create a production bundle, use `npm run build` or `yarn build`. | ||||
|     --> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										71
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/App.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/App.tsx
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,71 @@ | |||
| import PokemonView from 'features/pokemon/PokemonView'; | ||||
| import PostsView from 'features/posts/PostsView'; | ||||
| import { Box, Flex, Heading } from '@chakra-ui/react'; | ||||
| import { Link, UnorderedList, ListItem } from '@chakra-ui/react'; | ||||
| import { Code } from '@chakra-ui/react'; | ||||
| import * as React from 'react'; | ||||
| import { DevToolsSelector } from 'features/DevTools/DevToolsSelector'; | ||||
| 
 | ||||
| export function App() { | ||||
|   return ( | ||||
|     <main className="rtk-query-demo-app"> | ||||
|       <Heading as="h1" p="2"> | ||||
|         RTK Query inspector monitor demo | ||||
|       </Heading> | ||||
|       <PokemonView /> | ||||
|       <PostsView /> | ||||
|       <DevToolsSelector /> | ||||
|       <Flex p="2" as="section" flexWrap="nowrap" flexDirection="column"> | ||||
|         <Heading as="h2">Dock controls</Heading> | ||||
|         <Box as="pre" p="2" paddingX="4"> | ||||
|           <Code> | ||||
|             {`toggleVisibilityKey="ctrl-h"\nchangePositionKey="ctrl-q"`} | ||||
|           </Code> | ||||
|         </Box> | ||||
|         <Box as="p" p="2" paddingX="4"> | ||||
|           Drag its border to resize | ||||
|         </Box> | ||||
|       </Flex> | ||||
|       <Flex p="2" as="footer"> | ||||
|         <UnorderedList p="2"> | ||||
|           <ListItem> | ||||
|             <Link | ||||
|               className="link" | ||||
|               isExternal | ||||
|               href="https://github.com/FaberVitale/redux-devtools/tree/feat/rtk-query-monitor/packages/redux-devtools-rtk-query-monitor/demo" | ||||
|             > | ||||
|               demo source | ||||
|             </Link> | ||||
|           </ListItem> | ||||
|           <ListItem> | ||||
|             <Link | ||||
|               className="link" | ||||
|               isExternal | ||||
|               href="https://github.com/FaberVitale/redux-devtools/tree/feat/rtk-query-monitor/packages/redux-devtools-rtk-query-monitor" | ||||
|             > | ||||
|               @redux-devtools/rtk-query-monitor source | ||||
|             </Link> | ||||
|           </ListItem> | ||||
|           <ListItem> | ||||
|             <Link | ||||
|               className="link" | ||||
|               isExternal | ||||
|               href="https://github.com/reduxjs/redux-toolkit/tree/master/examples/query/react/polling" | ||||
|             > | ||||
|               polling example | ||||
|             </Link> | ||||
|           </ListItem> | ||||
|           <ListItem> | ||||
|             <Link | ||||
|               className="link" | ||||
|               isExternal | ||||
|               href="https://github.com/reduxjs/redux-toolkit/tree/master/examples/query/react/mutations" | ||||
|             > | ||||
|               mutations example | ||||
|             </Link> | ||||
|           </ListItem> | ||||
|         </UnorderedList> | ||||
|       </Flex> | ||||
|     </main> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,18 @@ | |||
| import * as React from 'react'; | ||||
| import { createDevTools } from '@redux-devtools/core'; | ||||
| import DockMonitor from '@redux-devtools/dock-monitor'; | ||||
| import RtkQueryMonitor from '../../../../src'; | ||||
| 
 | ||||
| const largeScreenQuery = window.matchMedia('(min-width: 1024px)'); | ||||
| 
 | ||||
| export default createDevTools( | ||||
|   <DockMonitor | ||||
|     toggleVisibilityKey="ctrl-h" | ||||
|     changePositionKey="ctrl-q" | ||||
|     changeMonitorKey="ctrl-m" | ||||
|     fluid | ||||
|     defaultSize={largeScreenQuery.matches ? 0.44 : 0.55} | ||||
|   > | ||||
|     <RtkQueryMonitor /> | ||||
|   </DockMonitor> | ||||
| ); | ||||
|  | @ -0,0 +1,40 @@ | |||
| import * as React from 'react'; | ||||
| import { ButtonGroup, Button } from '@chakra-ui/react'; | ||||
| import { isExtensionEnabled, setIsExtensionEnabled } from './helpers'; | ||||
| import { Box, Heading } from '@chakra-ui/react'; | ||||
| 
 | ||||
| export function DevToolsSelector() { | ||||
|   const handleClick = (evt: React.MouseEvent<HTMLButtonElement>) => { | ||||
|     setIsExtensionEnabled(evt.currentTarget.dataset.extension === '1'); | ||||
|     window.location.reload(); | ||||
|   }; | ||||
| 
 | ||||
|   const extensionEnabled = isExtensionEnabled(); | ||||
| 
 | ||||
|   return ( | ||||
|     <Box as="section" p="2"> | ||||
|       <Heading as="h2">Set active devTools</Heading> | ||||
|       <ButtonGroup variant="outline" spacing="4" p="4"> | ||||
|         <Button | ||||
|           aria-selected={!extensionEnabled} | ||||
|           colorScheme="blue" | ||||
|           selected={!extensionEnabled} | ||||
|           data-extension="0" | ||||
|           variant={!extensionEnabled ? 'solid' : 'outline'} | ||||
|           onClick={handleClick} | ||||
|         > | ||||
|           Dock | ||||
|         </Button> | ||||
|         <Button | ||||
|           aria-selected={extensionEnabled} | ||||
|           data-extension="1" | ||||
|           colorScheme="blue" | ||||
|           variant={extensionEnabled ? 'solid' : 'outline'} | ||||
|           onClick={handleClick} | ||||
|         > | ||||
|           Extension | ||||
|         </Button> | ||||
|       </ButtonGroup> | ||||
|     </Box> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1 @@ | |||
| export const isExtensionEnabledKey = 'prefer-extension'; | ||||
|  | @ -0,0 +1,22 @@ | |||
| import { isExtensionEnabledKey } from './config'; | ||||
| 
 | ||||
| export function isExtensionEnabled(): boolean { | ||||
|   let extensionEnabled = false; | ||||
| 
 | ||||
|   try { | ||||
|     extensionEnabled = | ||||
|       window.sessionStorage.getItem(isExtensionEnabledKey) === '1'; | ||||
|   } catch (err) { | ||||
|     console.error(err); | ||||
|   } | ||||
| 
 | ||||
|   return extensionEnabled; | ||||
| } | ||||
| 
 | ||||
| export function setIsExtensionEnabled(active: boolean): void { | ||||
|   try { | ||||
|     window.sessionStorage.setItem(isExtensionEnabledKey, active ? '1' : '0'); | ||||
|   } catch (err) { | ||||
|     console.error(err); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,74 @@ | |||
| import React, { useState } from 'react'; | ||||
| import { Button, Select } from '@chakra-ui/react'; | ||||
| import { useGetPokemonByNameQuery } from '../../services/pokemon'; | ||||
| import type { PokemonName } from '../../pokemon.data'; | ||||
| 
 | ||||
| const intervalOptions = [ | ||||
|   { label: 'Off', value: 0 }, | ||||
|   { label: '3s', value: 3000 }, | ||||
|   { label: '5s', value: 5000 }, | ||||
|   { label: '10s', value: 10000 }, | ||||
|   { label: '1m', value: 60000 }, | ||||
| ]; | ||||
| 
 | ||||
| export function Pokemon({ name }: { name: PokemonName }) { | ||||
|   const [pollingInterval, setPollingInterval] = useState(60000); | ||||
| 
 | ||||
|   const { data, error, isLoading, isFetching, refetch } = | ||||
|     useGetPokemonByNameQuery(name, { | ||||
|       pollingInterval, | ||||
|     }); | ||||
| 
 | ||||
|   return ( | ||||
|     <div | ||||
|       className="pokemon" | ||||
|       style={{ | ||||
|         ...(isFetching ? { background: '#e6ffe8' } : {}), | ||||
|       }} | ||||
|     > | ||||
|       {error ? ( | ||||
|         <>Oh no, there was an error loading {name}</> | ||||
|       ) : isLoading ? ( | ||||
|         <>Loading...</> | ||||
|       ) : data ? ( | ||||
|         <> | ||||
|           <h3>{data.species.name}</h3> | ||||
|           <div style={{ minWidth: 96, minHeight: 96 }}> | ||||
|             <img | ||||
|               src={data.sprites.front_shiny} | ||||
|               alt={data.species.name} | ||||
|               style={{ ...(isFetching ? { opacity: 0.3 } : {}) }} | ||||
|             /> | ||||
|           </div> | ||||
|           <div> | ||||
|             <label style={{ display: 'block' }}>Polling interval</label> | ||||
|             <Select | ||||
|               value={pollingInterval} | ||||
|               onChange={({ target: { value } }) => | ||||
|                 setPollingInterval(Number(value)) | ||||
|               } | ||||
|             > | ||||
|               {intervalOptions.map(({ label, value }) => ( | ||||
|                 <option key={value} value={value}> | ||||
|                   {label} | ||||
|                 </option> | ||||
|               ))} | ||||
|             </Select> | ||||
|           </div> | ||||
|           <div> | ||||
|             <Button | ||||
|               colorScheme="blue" | ||||
|               variant="outline" | ||||
|               onClick={refetch} | ||||
|               disabled={isFetching} | ||||
|             > | ||||
|               {isFetching ? 'Loading' : 'Manually refetch'} | ||||
|             </Button> | ||||
|           </div> | ||||
|         </> | ||||
|       ) : ( | ||||
|         'No Data' | ||||
|       )} | ||||
|     </div> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,41 @@ | |||
| import * as React from 'react'; | ||||
| import { Pokemon } from './Pokemon'; | ||||
| import { PokemonName, POKEMON_NAMES } from '../../pokemon.data'; | ||||
| import { Flex, Heading, Button } from '@chakra-ui/react'; | ||||
| 
 | ||||
| const getRandomPokemonName = () => | ||||
|   POKEMON_NAMES[Math.floor(Math.random() * POKEMON_NAMES.length)]; | ||||
| 
 | ||||
| export default function PokemonView() { | ||||
|   const [pokemon, setPokemon] = React.useState<PokemonName[]>(['bulbasaur']); | ||||
| 
 | ||||
|   return ( | ||||
|     <Flex p="2" as="section" flexWrap="nowrap" flexDirection="column"> | ||||
|       <Heading as="h2">Pokemon polling demo</Heading> | ||||
|       <Flex p="2" gridGap="0.5em" flexDirection="row" flexWrap="wrap"> | ||||
|         <Button | ||||
|           colorScheme="blue" | ||||
|           variant="outline" | ||||
|           onClick={() => | ||||
|             setPokemon((prev) => [...prev, getRandomPokemonName()]) | ||||
|           } | ||||
|         > | ||||
|           Add random pokemon | ||||
|         </Button> | ||||
|         <Button | ||||
|           colorScheme="blue" | ||||
|           variant="outline" | ||||
|           onClick={() => setPokemon((prev) => [...prev, 'bulbasaur'])} | ||||
|         > | ||||
|           Add bulbasaur | ||||
|         </Button> | ||||
|       </Flex> | ||||
| 
 | ||||
|       <div className="pokemon-list"> | ||||
|         {pokemon.map((name, index) => ( | ||||
|           <Pokemon key={index} name={name} /> | ||||
|         ))} | ||||
|       </div> | ||||
|     </Flex> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,155 @@ | |||
| import React, { useState } from 'react'; | ||||
| import { useHistory, useParams } from 'react-router-dom'; | ||||
| import { | ||||
|   useDeletePostMutation, | ||||
|   useGetPostQuery, | ||||
|   useUpdatePostMutation, | ||||
| } from 'services/posts'; | ||||
| import { | ||||
|   Box, | ||||
|   Button, | ||||
|   Center, | ||||
|   CloseButton, | ||||
|   Flex, | ||||
|   Heading, | ||||
|   Input, | ||||
|   Spacer, | ||||
|   Stack, | ||||
|   useToast, | ||||
| } from '@chakra-ui/react'; | ||||
| 
 | ||||
| const EditablePostName = ({ | ||||
|   name: initialName, | ||||
|   onUpdate, | ||||
|   onCancel, | ||||
|   isLoading = false, | ||||
| }: { | ||||
|   name: string; | ||||
|   onUpdate: (name: string) => void; | ||||
|   onCancel: () => void; | ||||
|   isLoading?: boolean; | ||||
| }) => { | ||||
|   const [name, setName] = useState(initialName); | ||||
| 
 | ||||
|   const handleChange = ({ | ||||
|     target: { value }, | ||||
|   }: React.ChangeEvent<HTMLInputElement>) => setName(value); | ||||
| 
 | ||||
|   const handleUpdate = () => onUpdate(name); | ||||
|   const handleCancel = () => onCancel(); | ||||
| 
 | ||||
|   return ( | ||||
|     <Flex> | ||||
|       <Box flex={10}> | ||||
|         <Input | ||||
|           type="text" | ||||
|           onChange={handleChange} | ||||
|           value={name} | ||||
|           disabled={isLoading} | ||||
|         /> | ||||
|       </Box> | ||||
|       <Spacer /> | ||||
|       <Box> | ||||
|         <Stack spacing={4} direction="row" align="center"> | ||||
|           <Button onClick={handleUpdate} isLoading={isLoading}> | ||||
|             Update | ||||
|           </Button> | ||||
|           <CloseButton bg="red" onClick={handleCancel} disabled={isLoading} /> | ||||
|         </Stack> | ||||
|       </Box> | ||||
|     </Flex> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| const PostJsonDetail = ({ id }: { id: string }) => { | ||||
|   const { data: post } = useGetPostQuery(id); | ||||
| 
 | ||||
|   return ( | ||||
|     <Box mt={5} bg="#eee"> | ||||
|       <pre>{JSON.stringify(post, null, 2)}</pre> | ||||
|     </Box> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export const PostDetail = () => { | ||||
|   const { id } = useParams<{ id: any }>(); | ||||
|   const history = useHistory(); | ||||
| 
 | ||||
|   const toast = useToast(); | ||||
| 
 | ||||
|   const [isEditing, setIsEditing] = useState(false); | ||||
| 
 | ||||
|   const { data: post, isLoading } = useGetPostQuery(id); | ||||
| 
 | ||||
|   const [updatePost, { isLoading: isUpdating }] = useUpdatePostMutation(); | ||||
| 
 | ||||
|   const [deletePost, { isLoading: isDeleting }] = useDeletePostMutation(); | ||||
| 
 | ||||
|   if (isLoading) { | ||||
|     return <div>Loading...</div>; | ||||
|   } | ||||
| 
 | ||||
|   if (!post) { | ||||
|     return ( | ||||
|       <Center h="200px"> | ||||
|         <Heading size="md"> | ||||
|           Post {id} is missing! Try reloading or selecting another post... | ||||
|         </Heading> | ||||
|       </Center> | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <Box p={4}> | ||||
|       {isEditing ? ( | ||||
|         <EditablePostName | ||||
|           name={post.name} | ||||
|           onUpdate={async (name) => { | ||||
|             try { | ||||
|               await updatePost({ id, name }).unwrap(); | ||||
|             } catch { | ||||
|               toast({ | ||||
|                 title: 'An error occurred', | ||||
|                 description: "We couldn't save your changes, try again!", | ||||
|                 status: 'error', | ||||
|                 duration: 2000, | ||||
|                 isClosable: true, | ||||
|               }); | ||||
|             } finally { | ||||
|               setIsEditing(false); | ||||
|             } | ||||
|           }} | ||||
|           onCancel={() => setIsEditing(false)} | ||||
|           isLoading={isUpdating} | ||||
|         /> | ||||
|       ) : ( | ||||
|         <Flex> | ||||
|           <Box> | ||||
|             <Heading size="md">{post.name}</Heading> | ||||
|           </Box> | ||||
|           <Spacer /> | ||||
|           <Box> | ||||
|             <Stack spacing={4} direction="row" align="center"> | ||||
|               <Button | ||||
|                 onClick={() => setIsEditing(true)} | ||||
|                 disabled={isDeleting || isUpdating} | ||||
|               > | ||||
|                 {isUpdating ? 'Updating...' : 'Edit'} | ||||
|               </Button> | ||||
|               <Button | ||||
|                 onClick={() => | ||||
|                   deletePost(id).then(() => history.push('/posts')) | ||||
|                 } | ||||
|                 disabled={isDeleting} | ||||
|                 colorScheme="red" | ||||
|               > | ||||
|                 {isDeleting ? 'Deleting...' : 'Delete'} | ||||
|               </Button> | ||||
|             </Stack> | ||||
|           </Box> | ||||
|         </Flex> | ||||
|       )} | ||||
|       <PostJsonDetail id={post.id} /> | ||||
|     </Box> | ||||
|   ); | ||||
| }; | ||||
|  | @ -0,0 +1,160 @@ | |||
| import { | ||||
|   Box, | ||||
|   Button, | ||||
|   Center, | ||||
|   Divider, | ||||
|   Flex, | ||||
|   FormControl, | ||||
|   FormLabel, | ||||
|   Heading, | ||||
|   Input, | ||||
|   List, | ||||
|   ListIcon, | ||||
|   ListItem, | ||||
|   Spacer, | ||||
|   Stat, | ||||
|   StatLabel, | ||||
|   StatNumber, | ||||
|   useToast, | ||||
| } from '@chakra-ui/react'; | ||||
| import { Route, Switch, useHistory } from 'react-router-dom'; | ||||
| import { MdBook } from 'react-icons/md'; | ||||
| import React, { useState } from 'react'; | ||||
| import { Post, useAddPostMutation, useGetPostsQuery } from 'services/posts'; | ||||
| import { PostDetail } from './PostDetail'; | ||||
| 
 | ||||
| const AddPost = () => { | ||||
|   const initialValue = { name: '' }; | ||||
|   const [post, setPost] = useState<Pick<Post, 'name'>>(initialValue); | ||||
|   const [addPost, { isLoading }] = useAddPostMutation(); | ||||
|   const toast = useToast(); | ||||
| 
 | ||||
|   const handleChange = ({ target }: React.ChangeEvent<HTMLInputElement>) => { | ||||
|     setPost((prev) => ({ | ||||
|       ...prev, | ||||
|       [target.name]: target.value, | ||||
|     })); | ||||
|   }; | ||||
| 
 | ||||
|   const handleAddPost = async () => { | ||||
|     try { | ||||
|       await addPost(post).unwrap(); | ||||
|       setPost(initialValue); | ||||
|     } catch { | ||||
|       toast({ | ||||
|         title: 'An error occurred', | ||||
|         description: "We couldn't save your post, try again!", | ||||
|         status: 'error', | ||||
|         duration: 2000, | ||||
|         isClosable: true, | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   return ( | ||||
|     <Flex p={'5px 0'} flexDirection="row" flexWrap="wrap" maxWidth={'85%'}> | ||||
|       <Box flex={'5 0 auto'} padding="0 5px 0 0"> | ||||
|         <FormControl | ||||
|           flexDirection="column" | ||||
|           isInvalid={Boolean(post.name.length < 3 && post.name)} | ||||
|         > | ||||
|           <FormLabel htmlFor="name">Post name</FormLabel> | ||||
|           <Input | ||||
|             id="name" | ||||
|             name="name" | ||||
|             placeholder="Enter post name" | ||||
|             value={post.name} | ||||
|             onChange={handleChange} | ||||
|           /> | ||||
|         </FormControl> | ||||
|       </Box> | ||||
|       <Box> | ||||
|         <Button | ||||
|           mt={8} | ||||
|           colorScheme="purple" | ||||
|           isLoading={isLoading} | ||||
|           onClick={handleAddPost} | ||||
|         > | ||||
|           Add Post | ||||
|         </Button> | ||||
|       </Box> | ||||
|     </Flex> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| const PostList = () => { | ||||
|   const { data: posts, isLoading } = useGetPostsQuery(); | ||||
|   const history = useHistory(); | ||||
| 
 | ||||
|   if (isLoading) { | ||||
|     return <div>Loading</div>; | ||||
|   } | ||||
| 
 | ||||
|   if (!posts) { | ||||
|     return <div>No posts :(</div>; | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <List spacing={3}> | ||||
|       {posts.map(({ id, name }) => ( | ||||
|         <ListItem key={id} onClick={() => history.push(`/posts/${id}`)}> | ||||
|           <ListIcon as={MdBook} color="green.500" /> {name} | ||||
|         </ListItem> | ||||
|       ))} | ||||
|     </List> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export const PostsCountStat = () => { | ||||
|   const { data: posts } = useGetPostsQuery(); | ||||
| 
 | ||||
|   if (!posts) return null; | ||||
| 
 | ||||
|   return ( | ||||
|     <Stat> | ||||
|       <StatLabel>Active Posts</StatLabel> | ||||
|       <StatNumber>{posts?.length}</StatNumber> | ||||
|     </Stat> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export const PostsManager = () => { | ||||
|   return ( | ||||
|     <Box> | ||||
|       <Flex bg="#011627" p={4} color="white"> | ||||
|         <Box> | ||||
|           <Heading size="xl">Manage Posts</Heading> | ||||
|         </Box> | ||||
|         <Spacer /> | ||||
|         <Box> | ||||
|           <PostsCountStat /> | ||||
|         </Box> | ||||
|       </Flex> | ||||
|       <Divider /> | ||||
|       <AddPost /> | ||||
|       <Divider /> | ||||
|       <Flex wrap="wrap"> | ||||
|         <Box flex={1} borderRight="1px solid #eee"> | ||||
|           <Box p={4} borderBottom="1px solid #eee"> | ||||
|             <Heading size="sm">Posts</Heading> | ||||
|           </Box> | ||||
|           <Box p={4}> | ||||
|             <PostList /> | ||||
|           </Box> | ||||
|         </Box> | ||||
|         <Box flex={2}> | ||||
|           <Switch> | ||||
|             <Route path="/posts/:id" component={PostDetail} /> | ||||
|             <Route> | ||||
|               <Center h="200px"> | ||||
|                 <Heading size="md">Select a post to edit!</Heading> | ||||
|               </Center> | ||||
|             </Route> | ||||
|           </Switch> | ||||
|         </Box> | ||||
|       </Flex> | ||||
|     </Box> | ||||
|   ); | ||||
| }; | ||||
| 
 | ||||
| export default PostsManager; | ||||
|  | @ -0,0 +1,17 @@ | |||
| import * as React from 'react'; | ||||
| import { Switch, Route } from 'react-router-dom'; | ||||
| import { PostsManager } from 'features/posts/PostsManager'; | ||||
| import { Box, Heading } from '@chakra-ui/react'; | ||||
| 
 | ||||
| function PostsView() { | ||||
|   return ( | ||||
|     <Box as="section" p="2"> | ||||
|       <Heading as="h2">Posts Demo</Heading> | ||||
|       <Switch> | ||||
|         <Route path="/" component={PostsManager} /> | ||||
|       </Switch> | ||||
|     </Box> | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export default PostsView; | ||||
							
								
								
									
										41
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/index.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/index.css
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,41 @@ | |||
| body { | ||||
|   margin: 0; | ||||
|   font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen', | ||||
|     'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', | ||||
|     sans-serif; | ||||
|   -webkit-font-smoothing: antialiased; | ||||
|   -moz-osx-font-smoothing: grayscale; | ||||
| } | ||||
| 
 | ||||
| code { | ||||
|   font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', | ||||
|     monospace; | ||||
| } | ||||
| 
 | ||||
| section { | ||||
|   display: block; | ||||
|   max-width: 65%; | ||||
| } | ||||
| 
 | ||||
| .pokemon-list { | ||||
|   display: flex; | ||||
|   flex-flow: row wrap; | ||||
|   overflow-x: hidden; | ||||
| } | ||||
| 
 | ||||
| .pokemon { | ||||
|   padding: 0.2em; | ||||
|   text-align: center; | ||||
| } | ||||
| 
 | ||||
| pre code { | ||||
|   white-space: pre-wrap; | ||||
| } | ||||
| 
 | ||||
| article { | ||||
|   padding: 0 0 0.5em; | ||||
| } | ||||
| 
 | ||||
| .link.link { | ||||
|   color: #805ad5; | ||||
| } | ||||
							
								
								
									
										28
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/index.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/index.tsx
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | |||
| import React from 'react'; | ||||
| import ReactDOM from 'react-dom'; | ||||
| import { Provider } from 'react-redux'; | ||||
| import { ChakraProvider } from '@chakra-ui/react'; | ||||
| import './index.css'; | ||||
| import { store } from './store'; | ||||
| import DevTools from './features/DevTools/DevTools'; | ||||
| import { BrowserRouter } from 'react-router-dom'; | ||||
| import { App } from 'App'; | ||||
| import { worker } from './mocks/browser'; | ||||
| 
 | ||||
| function renderApp() { | ||||
|   const rootElement = document.getElementById('root'); | ||||
| 
 | ||||
|   ReactDOM.render( | ||||
|     <Provider store={store}> | ||||
|       <ChakraProvider> | ||||
|         <BrowserRouter> | ||||
|           <App /> | ||||
|           <DevTools /> | ||||
|         </BrowserRouter> | ||||
|       </ChakraProvider> | ||||
|     </Provider>, | ||||
|     rootElement | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| worker.start({ quiet: true }).then(renderApp, renderApp); | ||||
|  | @ -0,0 +1,4 @@ | |||
| import { setupWorker } from 'msw'; | ||||
| import { handlers } from './db'; | ||||
| 
 | ||||
| export const worker = setupWorker(...handlers); | ||||
|  | @ -0,0 +1,59 @@ | |||
| import { factory, primaryKey } from '@mswjs/data'; | ||||
| import { nanoid } from '@reduxjs/toolkit'; | ||||
| import { rest } from 'msw'; | ||||
| import { Post } from '../services/posts'; | ||||
| 
 | ||||
| const db = factory({ | ||||
|   post: { | ||||
|     id: primaryKey(String), | ||||
|     name: String, | ||||
|   }, | ||||
| }); | ||||
| 
 | ||||
| [ | ||||
|   'A sample post', | ||||
|   'A post about RTK Query', | ||||
|   'How to randomly throw errors, a novella', | ||||
| ].forEach((name) => { | ||||
|   db.post.create({ id: nanoid(), name }); | ||||
| }); | ||||
| 
 | ||||
| export const handlers = [ | ||||
|   rest.post('/posts', async (req, res, ctx) => { | ||||
|     const { name } = req.body as Partial<Post>; | ||||
| 
 | ||||
|     if (Math.random() < 0.3) { | ||||
|       return res( | ||||
|         ctx.json({ error: 'Oh no, there was an error, try again.' }), | ||||
|         ctx.status(500), | ||||
|         ctx.delay(300) | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     const post = db.post.create({ | ||||
|       id: nanoid(), | ||||
|       name, | ||||
|     }); | ||||
| 
 | ||||
|     return res(ctx.json(post), ctx.delay(300)); | ||||
|   }), | ||||
|   rest.put('/posts/:id', (req, res, ctx) => { | ||||
|     const { name } = req.body as Partial<Post>; | ||||
| 
 | ||||
|     if (Math.random() < 0.3) { | ||||
|       return res( | ||||
|         ctx.json({ error: 'Oh no, there was an error, try again.' }), | ||||
|         ctx.status(500), | ||||
|         ctx.delay(300) | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     const post = db.post.update({ | ||||
|       where: { id: { equals: req.params.id } }, | ||||
|       data: { name }, | ||||
|     }); | ||||
| 
 | ||||
|     return res(ctx.json(post), ctx.delay(300)); | ||||
|   }), | ||||
|   ...db.post.toHandlers('rest'), | ||||
| ] as const; | ||||
|  | @ -0,0 +1,155 @@ | |||
| export const POKEMON_NAMES = [ | ||||
|   'bulbasaur', | ||||
|   'ivysaur', | ||||
|   'venusaur', | ||||
|   'charmander', | ||||
|   'charmeleon', | ||||
|   'charizard', | ||||
|   'squirtle', | ||||
|   'wartortle', | ||||
|   'blastoise', | ||||
|   'caterpie', | ||||
|   'metapod', | ||||
|   'butterfree', | ||||
|   'weedle', | ||||
|   'kakuna', | ||||
|   'beedrill', | ||||
|   'pidgey', | ||||
|   'pidgeotto', | ||||
|   'pidgeot', | ||||
|   'rattata', | ||||
|   'raticate', | ||||
|   'spearow', | ||||
|   'fearow', | ||||
|   'ekans', | ||||
|   'arbok', | ||||
|   'pikachu', | ||||
|   'raichu', | ||||
|   'sandshrew', | ||||
|   'sandslash', | ||||
|   'nidoran', | ||||
|   'nidorina', | ||||
|   'nidoqueen', | ||||
|   'nidoran', | ||||
|   'nidorino', | ||||
|   'nidoking', | ||||
|   'clefairy', | ||||
|   'clefable', | ||||
|   'vulpix', | ||||
|   'ninetales', | ||||
|   'jigglypuff', | ||||
|   'wigglytuff', | ||||
|   'zubat', | ||||
|   'golbat', | ||||
|   'oddish', | ||||
|   'gloom', | ||||
|   'vileplume', | ||||
|   'paras', | ||||
|   'parasect', | ||||
|   'venonat', | ||||
|   'venomoth', | ||||
|   'diglett', | ||||
|   'dugtrio', | ||||
|   'meowth', | ||||
|   'persian', | ||||
|   'psyduck', | ||||
|   'golduck', | ||||
|   'mankey', | ||||
|   'primeape', | ||||
|   'growlithe', | ||||
|   'arcanine', | ||||
|   'poliwag', | ||||
|   'poliwhirl', | ||||
|   'poliwrath', | ||||
|   'abra', | ||||
|   'kadabra', | ||||
|   'alakazam', | ||||
|   'machop', | ||||
|   'machoke', | ||||
|   'machamp', | ||||
|   'bellsprout', | ||||
|   'weepinbell', | ||||
|   'victreebel', | ||||
|   'tentacool', | ||||
|   'tentacruel', | ||||
|   'geodude', | ||||
|   'graveler', | ||||
|   'golem', | ||||
|   'ponyta', | ||||
|   'rapidash', | ||||
|   'slowpoke', | ||||
|   'slowbro', | ||||
|   'magnemite', | ||||
|   'magneton', | ||||
|   "farfetch'd", | ||||
|   'doduo', | ||||
|   'dodrio', | ||||
|   'seel', | ||||
|   'dewgong', | ||||
|   'grimer', | ||||
|   'muk', | ||||
|   'shellder', | ||||
|   'cloyster', | ||||
|   'gastly', | ||||
|   'haunter', | ||||
|   'gengar', | ||||
|   'onix', | ||||
|   'drowzee', | ||||
|   'hypno', | ||||
|   'krabby', | ||||
|   'kingler', | ||||
|   'voltorb', | ||||
|   'electrode', | ||||
|   'exeggcute', | ||||
|   'exeggutor', | ||||
|   'cubone', | ||||
|   'marowak', | ||||
|   'hitmonlee', | ||||
|   'hitmonchan', | ||||
|   'lickitung', | ||||
|   'koffing', | ||||
|   'weezing', | ||||
|   'rhyhorn', | ||||
|   'rhydon', | ||||
|   'chansey', | ||||
|   'tangela', | ||||
|   'kangaskhan', | ||||
|   'horsea', | ||||
|   'seadra', | ||||
|   'goldeen', | ||||
|   'seaking', | ||||
|   'staryu', | ||||
|   'starmie', | ||||
|   'mr. mime', | ||||
|   'scyther', | ||||
|   'jynx', | ||||
|   'electabuzz', | ||||
|   'magmar', | ||||
|   'pinsir', | ||||
|   'tauros', | ||||
|   'magikarp', | ||||
|   'gyarados', | ||||
|   'lapras', | ||||
|   'ditto', | ||||
|   'eevee', | ||||
|   'vaporeon', | ||||
|   'jolteon', | ||||
|   'flareon', | ||||
|   'porygon', | ||||
|   'omanyte', | ||||
|   'omastar', | ||||
|   'kabuto', | ||||
|   'kabutops', | ||||
|   'aerodactyl', | ||||
|   'snorlax', | ||||
|   'articuno', | ||||
|   'zapdos', | ||||
|   'moltres', | ||||
|   'dratini', | ||||
|   'dragonair', | ||||
|   'dragonite', | ||||
|   'mewtwo', | ||||
|   'mew', | ||||
| ] as const; | ||||
| 
 | ||||
| export type PokemonName = typeof POKEMON_NAMES[number]; | ||||
							
								
								
									
										5
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/react-app-env.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/react-app-env.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| /// <reference types="react-scripts" />
 | ||||
| 
 | ||||
| declare module '@redux-devtools/app'; | ||||
| 
 | ||||
| declare module 'remote-redux-devtools'; | ||||
|  | @ -0,0 +1,20 @@ | |||
| import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; | ||||
| import type { PokemonName } from '../pokemon.data'; | ||||
| 
 | ||||
| export const pokemonApi = createApi({ | ||||
|   reducerPath: 'pokemonApi', | ||||
|   baseQuery: fetchBaseQuery({ baseUrl: 'https://pokeapi.co/api/v2/' }), | ||||
|   tagTypes: ['pokemon'], | ||||
|   endpoints: (builder) => ({ | ||||
|     getPokemonByName: builder.query({ | ||||
|       query: (name: PokemonName) => `pokemon/${name}`, | ||||
|       providesTags: (result, error, name: PokemonName) => [ | ||||
|         { type: 'pokemon' }, | ||||
|         { type: 'pokemon', id: name }, | ||||
|       ], | ||||
|     }), | ||||
|   }), | ||||
| }); | ||||
| 
 | ||||
| // Export hooks for usage in functional components
 | ||||
| export const { useGetPokemonByNameQuery } = pokemonApi; | ||||
|  | @ -0,0 +1,63 @@ | |||
| import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'; | ||||
| 
 | ||||
| export interface Post { | ||||
|   id: string; | ||||
|   name: string; | ||||
| } | ||||
| 
 | ||||
| type PostsResponse = Post[]; | ||||
| 
 | ||||
| export const postsApi = createApi({ | ||||
|   reducerPath: 'postsApi', | ||||
|   baseQuery: fetchBaseQuery({ baseUrl: '/' }), | ||||
|   tagTypes: ['Post'], | ||||
|   endpoints: (build) => ({ | ||||
|     getPosts: build.query<PostsResponse, void>({ | ||||
|       query: () => 'posts', | ||||
|       providesTags: (result) => | ||||
|         result | ||||
|           ? [ | ||||
|               ...result.map(({ id }) => ({ type: 'Post' as const, id })), | ||||
|               { type: 'Post', id: 'LIST' }, | ||||
|             ] | ||||
|           : [{ type: 'Post', id: 'LIST' }], | ||||
|     }), | ||||
|     addPost: build.mutation<Post, Partial<Post>>({ | ||||
|       query: (body) => ({ | ||||
|         url: `posts`, | ||||
|         method: 'POST', | ||||
|         body, | ||||
|       }), | ||||
|       invalidatesTags: [{ type: 'Post', id: 'LIST' }], | ||||
|     }), | ||||
|     getPost: build.query<Post, string>({ | ||||
|       query: (id) => `posts/${id}`, | ||||
|       providesTags: (result, error, id) => [{ type: 'Post', id }], | ||||
|     }), | ||||
|     updatePost: build.mutation<void, Pick<Post, 'id'> & Partial<Post>>({ | ||||
|       query: ({ id, ...patch }) => ({ | ||||
|         url: `posts/${id}`, | ||||
|         method: 'PUT', | ||||
|         body: patch, | ||||
|       }), | ||||
|       invalidatesTags: (result, error, { id }) => [{ type: 'Post', id }], | ||||
|     }), | ||||
|     deletePost: build.mutation<{ success: boolean; id: number }, number>({ | ||||
|       query(id) { | ||||
|         return { | ||||
|           url: `posts/${id}`, | ||||
|           method: 'DELETE', | ||||
|         }; | ||||
|       }, | ||||
|       invalidatesTags: (result, error, id) => [{ type: 'Post', id }], | ||||
|     }), | ||||
|   }), | ||||
| }); | ||||
| 
 | ||||
| export const { | ||||
|   useGetPostQuery, | ||||
|   useGetPostsQuery, | ||||
|   useAddPostMutation, | ||||
|   useUpdatePostMutation, | ||||
|   useDeletePostMutation, | ||||
| } = postsApi; | ||||
							
								
								
									
										29
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/store.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								packages/redux-devtools-rtk-query-monitor/demo/src/store.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,29 @@ | |||
| import { | ||||
|   configureStore, | ||||
|   Middleware, | ||||
|   combineReducers, | ||||
|   EnhancedStore, | ||||
| } from '@reduxjs/toolkit'; | ||||
| import { pokemonApi } from './services/pokemon'; | ||||
| import { postsApi } from 'services/posts'; | ||||
| import DevTools from './features/DevTools/DevTools'; | ||||
| import { isExtensionEnabled } from 'features/DevTools/helpers'; | ||||
| 
 | ||||
| const devTools = isExtensionEnabled(); | ||||
| 
 | ||||
| const reducer = combineReducers({ | ||||
|   [pokemonApi.reducerPath]: pokemonApi.reducer, | ||||
|   [postsApi.reducerPath]: postsApi.reducer, | ||||
| }); | ||||
| 
 | ||||
| export const store: EnhancedStore<ReturnType<typeof reducer>> = configureStore({ | ||||
|   reducer, | ||||
|   devTools, | ||||
|   // adding the api middleware enables caching, invalidation, polling and other features of `rtk-query`
 | ||||
|   middleware: (getDefaultMiddleware) => | ||||
|     getDefaultMiddleware().concat([ | ||||
|       pokemonApi.middleware, | ||||
|       postsApi.middleware, | ||||
|     ]) as Middleware[], | ||||
|   enhancers: (devTools ? [] : [DevTools.instrument()]) as any, | ||||
| }); | ||||
							
								
								
									
										11
									
								
								packages/redux-devtools-rtk-query-monitor/demo/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								packages/redux-devtools-rtk-query-monitor/demo/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,11 @@ | |||
| { | ||||
|   "extends": "../../../tsconfig.react.base.json", | ||||
|   "compilerOptions": { | ||||
|     "resolveJsonModule": true, | ||||
|     "baseUrl": "./src", | ||||
|     "target": "esNext", | ||||
|     "module": "es6", | ||||
|     "moduleResolution": "node" | ||||
|   }, | ||||
|   "include": ["../src", "src"] | ||||
| } | ||||
							
								
								
									
										17887
									
								
								packages/redux-devtools-rtk-query-monitor/demo/yarn.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17887
									
								
								packages/redux-devtools-rtk-query-monitor/demo/yarn.lock
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								packages/redux-devtools-rtk-query-monitor/monitor-demo.gif
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								packages/redux-devtools-rtk-query-monitor/monitor-demo.gif
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 4.4 MiB | 
							
								
								
									
										88
									
								
								packages/redux-devtools-rtk-query-monitor/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								packages/redux-devtools-rtk-query-monitor/package.json
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,88 @@ | |||
| { | ||||
|   "name": "@redux-devtools/rtk-query-monitor", | ||||
|   "version": "1.0.0", | ||||
|   "description": "rtk-query monitor for Redux DevTools", | ||||
|   "keywords": [ | ||||
|     "redux", | ||||
|     "devtools", | ||||
|     "flux", | ||||
|     "react", | ||||
|     "redux-toolkit", | ||||
|     "rtk-query" | ||||
|   ], | ||||
|   "homepage": "https://github.com/FaberVitale/redux-devtools/tree/feat/rtk-query-monitor/packages/redux-devtools-rtk-query-monitor", | ||||
|   "bugs": { | ||||
|     "url": "https://github.com/reduxjs/redux-devtools/issues" | ||||
|   }, | ||||
|   "license": "MIT", | ||||
|   "author": { | ||||
|     "name": "FaberVitale", | ||||
|     "url": "https://github.com/FaberVitale" | ||||
|   }, | ||||
|   "files": [ | ||||
|     "lib", | ||||
|     "src" | ||||
|   ], | ||||
|   "main": "lib/index.js", | ||||
|   "types": "lib/index.d.ts", | ||||
|   "repository": { | ||||
|     "type": "git", | ||||
|     "url": "https://github.com/reduxjs/redux-devtools.git" | ||||
|   }, | ||||
|   "scripts": { | ||||
|     "start": "webpack-dev-server --config demo/config/webpack.config.ts", | ||||
|     "build": "npm run build:types && npm run build:js", | ||||
|     "stats": "webpack --profile --json > stats.json", | ||||
|     "build:demo": "cross-env NODE_ENV=production webpack -p --config demo/config/webpack.config.ts", | ||||
|     "build:types": "tsc --emitDeclarationOnly", | ||||
|     "build:js": "babel src --out-dir lib --extensions \".ts,.tsx\" --source-maps inline", | ||||
|     "clean": "rimraf lib", | ||||
|     "lint": "eslint . --ext .ts,.tsx", | ||||
|     "lint:fix": "eslint . --ext .ts,.tsx --fix", | ||||
|     "type-check": "tsc --noEmit", | ||||
|     "type-check:watch": "npm run type-check -- --watch", | ||||
|     "preversion": "npm run type-check && npm run lint", | ||||
|     "prepublishOnly": "npm run clean && npm run build" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@redux-devtools/dock-monitor": "^1.4.0", | ||||
|     "@types/prop-types": "^15.7.4", | ||||
|     "@types/redux-devtools-themes": "^1.0.0", | ||||
|     "devui": "^1.0.0-9", | ||||
|     "lodash.debounce": "^4.0.8", | ||||
|     "prop-types": "^15.7.2", | ||||
|     "react-json-tree": "^0.15.0", | ||||
|     "redux-devtools-themes": "^1.0.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@chakra-ui/react": "^1.6.5", | ||||
|     "@emotion/react": "^11", | ||||
|     "@emotion/styled": "^11", | ||||
|     "@mswjs/data": "^0.3.0", | ||||
|     "@redux-devtools/core": "^3.9.0", | ||||
|     "@redux-devtools/dock-monitor": "^1.4.0", | ||||
|     "@reduxjs/toolkit": "^1.6.0", | ||||
|     "@types/react": "^17.0.2", | ||||
|     "@types/react-dom": "17.0.0", | ||||
|     "@types/react-redux": "7.1.9", | ||||
|     "@types/react-router-dom": "5.1.6", | ||||
|     "cross-env": "^7.0.3", | ||||
|     "framer-motion": "^4", | ||||
|     "msw": "0.28.2", | ||||
|     "react": "^16.14.0", | ||||
|     "react-dom": "^16.14.0", | ||||
|     "react-redux": "^7.2.1", | ||||
|     "react-router-dom": "^5.2.0", | ||||
|     "redux": "^4.0.5" | ||||
|   }, | ||||
|   "peerDependencies": { | ||||
|     "@redux-devtools/core": "^3.7.0", | ||||
|     "@reduxjs/toolkit": "^1.6.0", | ||||
|     "@types/react": "^16.3.0 || ^17.0.0", | ||||
|     "react": "^16.3.0 || ^17.0.0", | ||||
|     "redux": "^3.4.0 || ^4.0.0" | ||||
|   }, | ||||
|   "msw": { | ||||
|     "workerDirectory": "demo/public" | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,27 @@ | |||
| import React, { HTMLAttributes } from 'react'; | ||||
| 
 | ||||
| export type ArrowUpIconProps = Omit< | ||||
|   HTMLAttributes<SVGElement>, | ||||
|   'xmlns' | 'children' | 'viewBox' | ||||
| >; | ||||
| 
 | ||||
| /* eslint-disable max-len */ | ||||
| /** | ||||
|  * @see https://icons.getbootstrap.com/icons/arrow-up/
 | ||||
|  */ | ||||
| export function ArrowUpIcon(props: ArrowUpIconProps): JSX.Element { | ||||
|   return ( | ||||
|     <svg | ||||
|       fill="currentColor" | ||||
|       {...props} | ||||
|       xmlns="http://www.w3.org/2000/svg" | ||||
|       viewBox="0 0 16 16" | ||||
|     > | ||||
|       <path | ||||
|         fillRule="evenodd" | ||||
|         d="M8 15a.5.5 0 0 0 .5-.5V2.707l3.146 3.147a.5.5 0 0 0 .708-.708l-4-4a.5.5 0 0 0-.708 0l-4 4a.5.5 0 1 0 .708.708L7.5 2.707V14.5a.5.5 0 0 0 .5.5z" | ||||
|       /> | ||||
|     </svg> | ||||
|   ); | ||||
| } | ||||
| /* eslint-enable max-len */ | ||||
|  | @ -0,0 +1,24 @@ | |||
| import React from 'react'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| 
 | ||||
| export function NoRtkQueryApi(): JSX.Element { | ||||
|   return ( | ||||
|     <StyleUtilsContext.Consumer> | ||||
|       {({ styling }) => ( | ||||
|         <div {...styling('noApiFound')}> | ||||
|           No rtk-query api found. | ||||
|           <br /> | ||||
|           Make sure to follow{' '} | ||||
|           <a | ||||
|             href="https://redux-toolkit.js.org/rtk-query/overview#basic-usage" | ||||
|             target="_blank" | ||||
|             rel="noreferrer noopener" | ||||
|           > | ||||
|             the instructions | ||||
|           </a> | ||||
|           . | ||||
|         </div> | ||||
|       )} | ||||
|     </StyleUtilsContext.Consumer> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,205 @@ | |||
| import React, { ReactNode, FormEvent, MouseEvent, ChangeEvent } from 'react'; | ||||
| import { QueryFormValues } from '../types'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| import { Select } from 'devui'; | ||||
| import { SelectOption } from '../types'; | ||||
| import debounce from 'lodash.debounce'; | ||||
| import { sortQueryOptions, QueryComparators } from '../utils/comparators'; | ||||
| import { QueryFilters, filterQueryOptions } from '../utils/filters'; | ||||
| import { SortOrderButton } from './SortOrderButton'; | ||||
| import { RegexIcon } from './RegexIcon'; | ||||
| 
 | ||||
| export interface QueryFormProps { | ||||
|   values: QueryFormValues; | ||||
|   searchQueryRegex: RegExp | null; | ||||
|   onFormValuesChange: (values: Partial<QueryFormValues>) => void; | ||||
| } | ||||
| 
 | ||||
| interface QueryFormState { | ||||
|   searchValue: string; | ||||
| } | ||||
| 
 | ||||
| const selectId = 'rtk-query-comp-select'; | ||||
| const searchId = 'rtk-query-search-query'; | ||||
| const filterSelectId = 'rtk-query-search-query-select'; | ||||
| const searchPlaceholder = 'filter query by...'; | ||||
| 
 | ||||
| const labels = { | ||||
|   regexToggle: { | ||||
|     info: 'Use regular expression search', | ||||
|     error: 'Invalid regular expression provided', | ||||
|   }, | ||||
| }; | ||||
| 
 | ||||
| export class QueryForm extends React.PureComponent< | ||||
|   QueryFormProps, | ||||
|   QueryFormState | ||||
| > { | ||||
|   constructor(props: QueryFormProps) { | ||||
|     super(props); | ||||
| 
 | ||||
|     this.state = { | ||||
|       searchValue: props.values.searchValue, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   inputSearchRef = React.createRef<HTMLInputElement>(); | ||||
| 
 | ||||
|   handleSubmit = (evt: FormEvent<HTMLFormElement>): void => { | ||||
|     evt.preventDefault(); | ||||
|   }; | ||||
| 
 | ||||
|   handleButtonGroupClick = (isAsc: boolean): void => { | ||||
|     this.props.onFormValuesChange({ isAscendingQueryComparatorOrder: isAsc }); | ||||
|   }; | ||||
| 
 | ||||
|   handleSelectComparatorChange = ( | ||||
|     option: SelectOption<QueryComparators> | undefined | null | ||||
|   ): void => { | ||||
|     if (typeof option?.value === 'string') { | ||||
|       this.props.onFormValuesChange({ queryComparator: option.value }); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   handleSelectFilterChange = ( | ||||
|     option: SelectOption<QueryFilters> | undefined | null | ||||
|   ): void => { | ||||
|     if (typeof option?.value === 'string') { | ||||
|       this.props.onFormValuesChange({ queryFilter: option.value }); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   handleRegexSearchClick = (): void => { | ||||
|     this.props.onFormValuesChange({ | ||||
|       isRegexSearch: !this.props.values.isRegexSearch, | ||||
|     }); | ||||
|   }; | ||||
| 
 | ||||
|   restoreCaretPosition = (start: number | null, end: number | null): void => { | ||||
|     window.requestAnimationFrame(() => { | ||||
|       if (this.inputSearchRef.current) { | ||||
|         this.inputSearchRef.current.selectionStart = start; | ||||
|         this.inputSearchRef.current.selectionEnd = end; | ||||
|       } | ||||
|     }); | ||||
|   }; | ||||
| 
 | ||||
|   invalidateSearchValueFromProps = debounce(() => { | ||||
|     this.props.onFormValuesChange({ | ||||
|       searchValue: this.state.searchValue, | ||||
|     }); | ||||
|   }, 150); | ||||
| 
 | ||||
|   handleSearchChange = (evt: ChangeEvent<HTMLInputElement>): void => { | ||||
|     const searchValue = evt.target.value.trim(); | ||||
|     this.setState({ searchValue }); | ||||
|     this.invalidateSearchValueFromProps(); | ||||
|   }; | ||||
| 
 | ||||
|   handleClearSearchClick = (evt: MouseEvent<HTMLButtonElement>): void => { | ||||
|     evt.preventDefault(); | ||||
| 
 | ||||
|     if (this.state.searchValue) { | ||||
|       this.setState({ searchValue: '' }); | ||||
|       this.invalidateSearchValueFromProps(); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { | ||||
|       searchQueryRegex, | ||||
|       values: { | ||||
|         isAscendingQueryComparatorOrder: isAsc, | ||||
|         queryComparator, | ||||
|         searchValue, | ||||
|         queryFilter, | ||||
|         isRegexSearch, | ||||
|       }, | ||||
|     } = this.props; | ||||
| 
 | ||||
|     const isRegexInvalid = | ||||
|       isRegexSearch && searchValue.length > 0 && searchQueryRegex == null; | ||||
|     const regexToggleType = isRegexInvalid ? 'error' : 'info'; | ||||
|     const regexToggleLabel = labels.regexToggle[regexToggleType]; | ||||
| 
 | ||||
|     return ( | ||||
|       <StyleUtilsContext.Consumer> | ||||
|         {({ styling, base16Theme }) => { | ||||
|           return ( | ||||
|             <form | ||||
|               action="#" | ||||
|               onSubmit={this.handleSubmit} | ||||
|               {...styling('queryForm')} | ||||
|             > | ||||
|               <div {...styling('queryListHeader')}> | ||||
|                 <label htmlFor={searchId} {...styling('srOnly')}> | ||||
|                   filter query | ||||
|                 </label> | ||||
|                 <div {...styling('querySearch')}> | ||||
|                   <input | ||||
|                     ref={this.inputSearchRef} | ||||
|                     type="search" | ||||
|                     value={this.state.searchValue} | ||||
|                     onChange={this.handleSearchChange} | ||||
|                     placeholder={searchPlaceholder} | ||||
|                   /> | ||||
|                   <button | ||||
|                     type="reset" | ||||
|                     aria-label="clear search" | ||||
|                     data-invisible={ | ||||
|                       +(this.state.searchValue.length === 0) || undefined | ||||
|                     } | ||||
|                     onClick={this.handleClearSearchClick} | ||||
|                     {...styling('closeButton')} | ||||
|                   /> | ||||
|                   <button | ||||
|                     type="button" | ||||
|                     aria-label={regexToggleLabel} | ||||
|                     title={regexToggleLabel} | ||||
|                     data-type={regexToggleType} | ||||
|                     aria-pressed={isRegexSearch} | ||||
|                     onClick={this.handleRegexSearchClick} | ||||
|                     {...styling('toggleButton')} | ||||
|                   > | ||||
|                     <RegexIcon /> | ||||
|                   </button> | ||||
|                 </div> | ||||
|                 <label htmlFor={selectId} {...styling('srOnly')}> | ||||
|                   filter by | ||||
|                 </label> | ||||
|                 <Select<SelectOption<QueryFilters>> | ||||
|                   id={filterSelectId} | ||||
|                   isSearchable={false} | ||||
|                   options={filterQueryOptions} | ||||
|                   theme={base16Theme as any} | ||||
|                   value={filterQueryOptions.find( | ||||
|                     (opt) => opt?.value === queryFilter | ||||
|                   )} | ||||
|                   onChange={this.handleSelectFilterChange} | ||||
|                 /> | ||||
|               </div> | ||||
|               <div {...styling('sortBySection')}> | ||||
|                 <label htmlFor={selectId}>Sort by</label> | ||||
|                 <Select<SelectOption<QueryComparators>> | ||||
|                   id={selectId} | ||||
|                   isSearchable={false} | ||||
|                   theme={base16Theme as any} | ||||
|                   value={sortQueryOptions.find( | ||||
|                     (opt) => opt?.value === queryComparator | ||||
|                   )} | ||||
|                   options={sortQueryOptions} | ||||
|                   onChange={this.handleSelectComparatorChange} | ||||
|                 /> | ||||
|                 <SortOrderButton | ||||
|                   id={'rtk-query-sort-order-button'} | ||||
|                   isAsc={isAsc} | ||||
|                   onChange={this.handleButtonGroupClick} | ||||
|                 /> | ||||
|               </div> | ||||
|             </form> | ||||
|           ); | ||||
|         }} | ||||
|       </StyleUtilsContext.Consumer> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,69 @@ | |||
| import React, { PureComponent, ReactNode } from 'react'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| import { RtkResourceInfo, RtkQueryMonitorState } from '../types'; | ||||
| import { isQuerySelected } from '../utils/rtk-query'; | ||||
| 
 | ||||
| export interface QueryListProps { | ||||
|   resInfos: RtkResourceInfo[]; | ||||
|   selectedQueryKey: RtkQueryMonitorState['selectedQueryKey']; | ||||
|   onSelectQuery: (query: RtkResourceInfo) => void; | ||||
| } | ||||
| 
 | ||||
| export class QueryList extends PureComponent<QueryListProps> { | ||||
|   static isItemSelected( | ||||
|     selectedQueryKey: QueryListProps['selectedQueryKey'], | ||||
|     queryInfo: RtkResourceInfo | ||||
|   ): boolean { | ||||
|     return ( | ||||
|       !!selectedQueryKey && | ||||
|       selectedQueryKey.queryKey === queryInfo.queryKey && | ||||
|       selectedQueryKey.reducerPath === queryInfo.reducerPath | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   static formatQuery(resInfo: RtkResourceInfo): string { | ||||
|     const key = | ||||
|       resInfo.type === 'query' | ||||
|         ? resInfo.queryKey | ||||
|         : `${resInfo.state.endpointName ?? ''} ${resInfo.queryKey}`; | ||||
| 
 | ||||
|     return key; | ||||
|   } | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { resInfos, selectedQueryKey, onSelectQuery } = this.props; | ||||
| 
 | ||||
|     return ( | ||||
|       <StyleUtilsContext.Consumer> | ||||
|         {({ styling }) => ( | ||||
|           <ul {...styling('queryList')}> | ||||
|             {resInfos.map((resInfo) => { | ||||
|               const isSelected = isQuerySelected(selectedQueryKey, resInfo); | ||||
| 
 | ||||
|               return ( | ||||
|                 <li | ||||
|                   key={resInfo.queryKey} | ||||
|                   onClick={() => onSelectQuery(resInfo)} | ||||
|                   {...styling( | ||||
|                     ['queryListItem', isSelected && 'queryListItemSelected'], | ||||
|                     isSelected | ||||
|                   )} | ||||
|                 > | ||||
|                   <p {...styling('queryListItemKey')}> | ||||
|                     {QueryList.formatQuery(resInfo)} | ||||
|                   </p> | ||||
|                   <div {...styling('queryStatusWrapper')}> | ||||
|                     <strong {...styling(['queryStatus', 'queryType'])}> | ||||
|                       {resInfo.type === 'query' ? 'Q' : 'M'} | ||||
|                     </strong> | ||||
|                     <p {...styling('queryStatus')}>{resInfo.state.status}</p> | ||||
|                   </div> | ||||
|                 </li> | ||||
|               ); | ||||
|             })} | ||||
|           </ul> | ||||
|         )} | ||||
|       </StyleUtilsContext.Consumer> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,80 @@ | |||
| import { createSelector } from '@reduxjs/toolkit'; | ||||
| import React, { ReactNode, PureComponent } from 'react'; | ||||
| import { Action, AnyAction } from 'redux'; | ||||
| import { emptyRecord, identity } from '../utils/object'; | ||||
| import { TreeView } from './TreeView'; | ||||
| 
 | ||||
| export interface QueryPreviewActionsProps { | ||||
|   isWideLayout: boolean; | ||||
|   actionsOfQuery: AnyAction[]; | ||||
| } | ||||
| 
 | ||||
| const keySep = ' - '; | ||||
| 
 | ||||
| export class QueryPreviewActions extends PureComponent<QueryPreviewActionsProps> { | ||||
|   selectFormattedActions = createSelector< | ||||
|     AnyAction[], | ||||
|     AnyAction[], | ||||
|     Record<string, AnyAction> | ||||
|   >(identity, (actions) => { | ||||
|     const output: Record<string, AnyAction> = {}; | ||||
| 
 | ||||
|     if (actions.length === 0) { | ||||
|       return emptyRecord; | ||||
|     } | ||||
| 
 | ||||
|     for (let i = 0, len = actions.length; i < len; i++) { | ||||
|       const action = actions[i]; | ||||
|       const key = `${i}${keySep}${(action as Action<string>)?.type ?? ''}`; | ||||
|       output[key] = action; | ||||
|     } | ||||
| 
 | ||||
|     return output; | ||||
|   }); | ||||
| 
 | ||||
|   isLastActionNode = (keyPath: (string | number)[], layer: number): boolean => { | ||||
|     if (layer >= 1) { | ||||
|       const len = this.props.actionsOfQuery.length; | ||||
|       const actionKey = keyPath[keyPath.length - 1]; | ||||
| 
 | ||||
|       if (typeof actionKey === 'string') { | ||||
|         const index = Number(actionKey.split(keySep)[0]); | ||||
| 
 | ||||
|         return len > 0 && len - index < 2; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     return false; | ||||
|   }; | ||||
| 
 | ||||
|   shouldExpandNode = ( | ||||
|     keyPath: (string | number)[], | ||||
|     value: unknown, | ||||
|     layer: number | ||||
|   ): boolean => { | ||||
|     if (layer === 1) { | ||||
|       return this.isLastActionNode(keyPath, layer); | ||||
|     } | ||||
| 
 | ||||
|     if (layer === 2) { | ||||
|       return ( | ||||
|         this.isLastActionNode(keyPath, layer) && | ||||
|         (keyPath[0] === 'meta' || keyPath[0] === 'error') | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return layer <= 1; | ||||
|   }; | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { isWideLayout, actionsOfQuery } = this.props; | ||||
| 
 | ||||
|     return ( | ||||
|       <TreeView | ||||
|         data={this.selectFormattedActions(actionsOfQuery)} | ||||
|         isWideLayout={isWideLayout} | ||||
|         shouldExpandNode={this.shouldExpandNode} | ||||
|       /> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,72 @@ | |||
| import React, { ReactNode, PureComponent } from 'react'; | ||||
| import { ApiStats, RtkQueryApiState } from '../types'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| import { TreeView } from './TreeView'; | ||||
| 
 | ||||
| export interface QueryPreviewApiProps { | ||||
|   apiStats: ApiStats | null; | ||||
|   apiState: RtkQueryApiState | null; | ||||
|   isWideLayout: boolean; | ||||
| } | ||||
| 
 | ||||
| export class QueryPreviewApi extends PureComponent<QueryPreviewApiProps> { | ||||
|   shouldExpandApiStateNode = ( | ||||
|     keyPath: (string | number)[], | ||||
|     value: unknown, | ||||
|     layer: number | ||||
|   ): boolean => { | ||||
|     const lastKey = keyPath[keyPath.length - 1]; | ||||
| 
 | ||||
|     return layer <= 1 && lastKey !== 'config'; | ||||
|   }; | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { apiStats, isWideLayout, apiState } = this.props; | ||||
| 
 | ||||
|     if (!apiState) { | ||||
|       return null; | ||||
|     } | ||||
| 
 | ||||
|     const hasMutations = Object.keys(apiState.mutations).length > 0; | ||||
|     const hasQueries = Object.keys(apiState.queries).length > 0; | ||||
| 
 | ||||
|     return ( | ||||
|       <StyleUtilsContext.Consumer> | ||||
|         {({ styling }) => ( | ||||
|           <article {...styling('tabContent')}> | ||||
|             <h2>{apiState.config.reducerPath}</h2> | ||||
|             <TreeView | ||||
|               before={<h3>State</h3>} | ||||
|               data={apiState} | ||||
|               shouldExpandNode={this.shouldExpandApiStateNode} | ||||
|               isWideLayout={isWideLayout} | ||||
|             /> | ||||
|             {apiStats && ( | ||||
|               <> | ||||
|                 <TreeView | ||||
|                   before={<h3>Tally</h3>} | ||||
|                   data={apiStats.tally} | ||||
|                   isWideLayout={isWideLayout} | ||||
|                 /> | ||||
|                 {hasQueries && ( | ||||
|                   <TreeView | ||||
|                     before={<h3>Queries Timings</h3>} | ||||
|                     data={apiStats.timings.queries} | ||||
|                     isWideLayout={isWideLayout} | ||||
|                   /> | ||||
|                 )} | ||||
|                 {hasMutations && ( | ||||
|                   <TreeView | ||||
|                     before={<h3>Mutations Timings</h3>} | ||||
|                     data={apiStats.timings.mutations} | ||||
|                     isWideLayout={isWideLayout} | ||||
|                   /> | ||||
|                 )} | ||||
|               </> | ||||
|             )} | ||||
|           </article> | ||||
|         )} | ||||
|       </StyleUtilsContext.Consumer> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,57 @@ | |||
| import React, { ReactNode } from 'react'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| import { QueryPreviewTabs, TabOption } from '../types'; | ||||
| import { emptyArray } from '../utils/object'; | ||||
| 
 | ||||
| export interface QueryPreviewHeaderProps { | ||||
|   tabs: ReadonlyArray< | ||||
|     TabOption<QueryPreviewTabs, unknown, 'query' | 'mutation'> | ||||
|   >; | ||||
|   onTabChange: (tab: QueryPreviewTabs) => void; | ||||
|   selectedTab: QueryPreviewTabs; | ||||
|   renderTabLabel?: (tab: QueryPreviewHeaderProps['tabs'][number]) => ReactNode; | ||||
| } | ||||
| 
 | ||||
| export class QueryPreviewHeader extends React.Component<QueryPreviewHeaderProps> { | ||||
|   handleTabClick = (tab: QueryPreviewHeaderProps['tabs'][number]): void => { | ||||
|     if (this.props.selectedTab !== tab.value) { | ||||
|       this.props.onTabChange(tab.value); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { tabs, selectedTab, renderTabLabel } = this.props; | ||||
| 
 | ||||
|     return ( | ||||
|       <StyleUtilsContext.Consumer> | ||||
|         {({ styling }) => ( | ||||
|           <div {...styling('previewHeader')}> | ||||
|             <div {...styling('tabSelector')}> | ||||
|               {tabs.map((tab) => ( | ||||
|                 <div | ||||
|                   onClick={() => this.handleTabClick(tab)} | ||||
|                   key={tab.value} | ||||
|                   {...styling( | ||||
|                     [ | ||||
|                       'selectorButton', | ||||
|                       tab.value === selectedTab && 'selectorButtonSelected', | ||||
|                     ], | ||||
|                     tab.value === selectedTab | ||||
|                   )} | ||||
|                 > | ||||
|                   <span> | ||||
|                     {renderTabLabel ? renderTabLabel(tab) : tab.label} | ||||
|                   </span> | ||||
|                 </div> | ||||
|               ))} | ||||
|             </div> | ||||
|           </div> | ||||
|         )} | ||||
|       </StyleUtilsContext.Consumer> | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   static defaultProps = { | ||||
|     tabs: emptyArray, | ||||
|   }; | ||||
| } | ||||
|  | @ -0,0 +1,103 @@ | |||
| import { createSelector, Selector } from '@reduxjs/toolkit'; | ||||
| import { QueryStatus } from '@reduxjs/toolkit/dist/query'; | ||||
| import React, { ReactNode, PureComponent } from 'react'; | ||||
| import { RtkResourceInfo, RTKStatusFlags } from '../types'; | ||||
| import { formatMs } from '../utils/formatters'; | ||||
| import { identity } from '../utils/object'; | ||||
| import { getQueryStatusFlags } from '../utils/rtk-query'; | ||||
| import { TreeView } from './TreeView'; | ||||
| 
 | ||||
| type QueryTimings = { | ||||
|   startedAt: string; | ||||
|   loadedAt: string; | ||||
|   duration: string; | ||||
| }; | ||||
| 
 | ||||
| type FormattedQuery = { | ||||
|   key: string; | ||||
|   reducerPath: string; | ||||
|   timings: QueryTimings; | ||||
|   statusFlags: RTKStatusFlags; | ||||
| } & ( | ||||
|   | { mutation: RtkResourceInfo['state'] } | ||||
|   | { query: RtkResourceInfo['state'] } | ||||
| ); | ||||
| 
 | ||||
| export interface QueryPreviewInfoProps { | ||||
|   resInfo: RtkResourceInfo; | ||||
|   isWideLayout: boolean; | ||||
| } | ||||
| export class QueryPreviewInfo extends PureComponent<QueryPreviewInfoProps> { | ||||
|   shouldExpandNode = ( | ||||
|     keyPath: (string | number)[], | ||||
|     value: unknown, | ||||
|     layer: number | ||||
|   ): boolean => { | ||||
|     const lastKey = keyPath[keyPath.length - 1]; | ||||
| 
 | ||||
|     return layer <= 1 && lastKey !== 'query' && lastKey !== 'mutation'; | ||||
|   }; | ||||
| 
 | ||||
|   selectFormattedQuery: Selector<RtkResourceInfo, FormattedQuery> = | ||||
|     createSelector(identity, (resInfo: RtkResourceInfo): FormattedQuery => { | ||||
|       const { state, queryKey, reducerPath } = resInfo; | ||||
| 
 | ||||
|       const startedAt = state.startedTimeStamp | ||||
|         ? new Date(state.startedTimeStamp).toISOString() | ||||
|         : '-'; | ||||
| 
 | ||||
|       const loadedAt = state.fulfilledTimeStamp | ||||
|         ? new Date(state.fulfilledTimeStamp).toISOString() | ||||
|         : '-'; | ||||
| 
 | ||||
|       const statusFlags = getQueryStatusFlags(state); | ||||
| 
 | ||||
|       const timings = { | ||||
|         startedAt, | ||||
|         loadedAt, | ||||
|         duration: '-', | ||||
|       }; | ||||
| 
 | ||||
|       if ( | ||||
|         state.fulfilledTimeStamp && | ||||
|         state.startedTimeStamp && | ||||
|         state.status !== QueryStatus.pending && | ||||
|         state.startedTimeStamp <= state.fulfilledTimeStamp | ||||
|       ) { | ||||
|         timings.duration = formatMs( | ||||
|           state.fulfilledTimeStamp - state.startedTimeStamp | ||||
|         ); | ||||
|       } | ||||
| 
 | ||||
|       if (resInfo.type === 'query') { | ||||
|         return { | ||||
|           key: queryKey, | ||||
|           reducerPath, | ||||
|           query: resInfo.state, | ||||
|           statusFlags, | ||||
|           timings, | ||||
|         }; | ||||
|       } | ||||
| 
 | ||||
|       return { | ||||
|         key: queryKey, | ||||
|         reducerPath, | ||||
|         mutation: resInfo.state, | ||||
|         statusFlags, | ||||
|         timings, | ||||
|       }; | ||||
|     }); | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { resInfo, isWideLayout } = this.props; | ||||
|     const formattedQuery = this.selectFormattedQuery(resInfo); | ||||
| 
 | ||||
|     return ( | ||||
|       <TreeView | ||||
|         data={formattedQuery} | ||||
|         isWideLayout={isWideLayout} | ||||
|         shouldExpandNode={this.shouldExpandNode} | ||||
|       /> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,18 @@ | |||
| import React, { ReactNode, PureComponent } from 'react'; | ||||
| import { RtkQueryApiState } from '../types'; | ||||
| import { TreeView } from './TreeView'; | ||||
| 
 | ||||
| export interface QueryPreviewSubscriptionsProps { | ||||
|   subscriptions: RtkQueryApiState['subscriptions'][keyof RtkQueryApiState['subscriptions']]; | ||||
|   isWideLayout: boolean; | ||||
| } | ||||
| 
 | ||||
| export class QueryPreviewSubscriptions extends PureComponent<QueryPreviewSubscriptionsProps> { | ||||
|   render(): ReactNode { | ||||
|     const { subscriptions } = this.props; | ||||
| 
 | ||||
|     return ( | ||||
|       <TreeView data={subscriptions} isWideLayout={this.props.isWideLayout} /> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,31 @@ | |||
| import React, { ReactNode, PureComponent } from 'react'; | ||||
| import { RtkQueryTag } from '../types'; | ||||
| import { TreeView } from './TreeView'; | ||||
| 
 | ||||
| interface QueryPreviewTagsState { | ||||
|   data: { tags: RtkQueryTag[] }; | ||||
| } | ||||
| 
 | ||||
| export interface QueryPreviewTagsProps { | ||||
|   tags: RtkQueryTag[]; | ||||
|   isWideLayout: boolean; | ||||
| } | ||||
| 
 | ||||
| export class QueryPreviewTags extends PureComponent< | ||||
|   QueryPreviewTagsProps, | ||||
|   QueryPreviewTagsState | ||||
| > { | ||||
|   constructor(props: QueryPreviewTagsProps) { | ||||
|     super(props); | ||||
| 
 | ||||
|     this.state = { | ||||
|       data: { tags: props.tags }, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { isWideLayout, tags } = this.props; | ||||
| 
 | ||||
|     return <TreeView data={tags} isWideLayout={isWideLayout} />; | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,23 @@ | |||
| import * as React from 'react'; | ||||
| 
 | ||||
| export type RegexIconProps = Omit< | ||||
|   React.HTMLAttributes<SVGElement>, | ||||
|   'viewBox' | 'children' | ||||
| >; | ||||
| 
 | ||||
| // `OOjs_UI_icon_regular-expression.svg` (MIT License)
 | ||||
| // from https://commons.wikimedia.org/wiki/File:OOjs_UI_icon_regular-expression.svg
 | ||||
| export function RegexIcon( | ||||
|   props: React.HTMLAttributes<SVGElement> | ||||
| ): JSX.Element { | ||||
|   return ( | ||||
|     <svg fill="currentColor" {...props} viewBox="0 0 24 24"> | ||||
|       <g> | ||||
|         <path d="M3 12.045c0-.99.15-1.915.45-2.777A6.886 6.886 0 0 1 4.764 7H6.23a7.923 7.923 0 0 0-1.25 2.374 8.563 8.563 0 0 0 .007 5.314c.29.85.7 1.622 1.23 2.312h-1.45a6.53 6.53 0 0 1-1.314-2.223 8.126 8.126 0 0 1-.45-2.732" /> | ||||
|         <path id="dot" d="M10 16a1 1 0 1 1-2 0 1 1 0 0 1 2 0z" /> | ||||
|         <path d="M14.25 7.013l-.24 2.156 2.187-.61.193 1.47-1.992.14 1.307 1.74-1.33.71-.914-1.833-.8 1.822-1.38-.698 1.296-1.74-1.98-.152.23-1.464 2.14.61-.24-2.158h1.534" /> | ||||
|         <path d="M21 12.045c0 .982-.152 1.896-.457 2.744A6.51 6.51 0 0 1 19.236 17h-1.453a8.017 8.017 0 0 0 1.225-2.31c.29-.855.434-1.74.434-2.66 0-.91-.14-1.797-.422-2.66a7.913 7.913 0 0 0-1.248-2.374h1.465a6.764 6.764 0 0 1 1.313 2.28c.3.86.45 1.782.45 2.764" /> | ||||
|       </g> | ||||
|     </svg> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,46 @@ | |||
| import React, { CSSProperties } from 'react'; | ||||
| import { ArrowUpIcon } from './ArrowUpIcon'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| 
 | ||||
| export interface SortOrderButtonProps { | ||||
|   readonly isAsc?: boolean; | ||||
|   readonly onChange: (isAsc: boolean) => void; | ||||
|   id?: string; | ||||
| } | ||||
| 
 | ||||
| export function SortOrderButton({ | ||||
|   isAsc, | ||||
|   onChange, | ||||
|   id, | ||||
| }: SortOrderButtonProps): JSX.Element { | ||||
|   const handleButtonClick = (): void => { | ||||
|     if (!isAsc) { | ||||
|       onChange(true); | ||||
|     } else onChange(false); | ||||
|   }; | ||||
| 
 | ||||
|   const buttonLabel = isAsc ? 'asc' : 'desc'; | ||||
| 
 | ||||
|   const arrowStyles: CSSProperties = { | ||||
|     width: '1em', | ||||
|     height: '1em', | ||||
|     transform: !isAsc ? 'scaleY(-1)' : undefined, | ||||
|   }; | ||||
| 
 | ||||
|   return ( | ||||
|     <StyleUtilsContext.Consumer> | ||||
|       {({ styling }) => ( | ||||
|         <button | ||||
|           type="button" | ||||
|           id={id} | ||||
|           onClick={handleButtonClick} | ||||
|           aria-pressed={isAsc} | ||||
|           {...styling(['sortButton'])} | ||||
|         > | ||||
|           <ArrowUpIcon style={arrowStyles} /> | ||||
|           {buttonLabel} | ||||
|         </button> | ||||
|       )} | ||||
|     </StyleUtilsContext.Consumer> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,100 @@ | |||
| import { createSelector } from '@reduxjs/toolkit'; | ||||
| import React, { ComponentProps, ReactNode } from 'react'; | ||||
| import JSONTree from 'react-json-tree'; | ||||
| import { Base16Theme, StylingFunction } from 'react-base16-styling'; | ||||
| import { DATA_TYPE_KEY } from '../monitor-config'; | ||||
| import { | ||||
|   getJsonTreeTheme, | ||||
|   StyleUtilsContext, | ||||
| } from '../styles/createStylingFromTheme'; | ||||
| import { createTreeItemLabelRenderer, getItemString } from '../styles/tree'; | ||||
| import { identity } from '../utils/object'; | ||||
| 
 | ||||
| export interface TreeViewProps | ||||
|   extends Partial< | ||||
|     Pick< | ||||
|       ComponentProps<typeof JSONTree>, | ||||
|       'keyPath' | 'shouldExpandNode' | 'hideRoot' | ||||
|     > | ||||
|   > { | ||||
|   data: unknown; | ||||
|   isWideLayout: boolean; | ||||
|   before?: ReactNode; | ||||
|   after?: ReactNode; | ||||
|   children?: ReactNode; | ||||
| } | ||||
| 
 | ||||
| export class TreeView extends React.PureComponent<TreeViewProps> { | ||||
|   static defaultProps = { | ||||
|     hideRoot: true, | ||||
|     shouldExpandNode: ( | ||||
|       keyPath: (string | number)[], | ||||
|       value: unknown, | ||||
|       layer: number | ||||
|     ): boolean => { | ||||
|       return layer < 2; | ||||
|     }, | ||||
|   }; | ||||
| 
 | ||||
|   readonly selectLabelRenderer = createSelector< | ||||
|     StylingFunction, | ||||
|     StylingFunction, | ||||
|     ReturnType<typeof createTreeItemLabelRenderer> | ||||
|   >(identity, createTreeItemLabelRenderer); | ||||
| 
 | ||||
|   readonly selectGetItemString = createSelector< | ||||
|     StylingFunction, | ||||
|     StylingFunction, | ||||
|     (type: string, data: unknown) => ReactNode | ||||
|   >( | ||||
|     identity, | ||||
|     (styling) => (type, data) => | ||||
|       getItemString(styling, type, data, DATA_TYPE_KEY, false) | ||||
|   ); | ||||
| 
 | ||||
|   readonly selectTheme = createSelector< | ||||
|     Base16Theme, | ||||
|     Base16Theme, | ||||
|     ReturnType<typeof getJsonTreeTheme> | ||||
|   >(identity, getJsonTreeTheme); | ||||
| 
 | ||||
|   constructor(props: TreeViewProps) { | ||||
|     super(props); | ||||
|   } | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { | ||||
|       data, | ||||
|       before, | ||||
|       after, | ||||
|       children, | ||||
|       keyPath, | ||||
|       shouldExpandNode, | ||||
|       hideRoot, | ||||
|     } = this.props; | ||||
| 
 | ||||
|     return ( | ||||
|       <StyleUtilsContext.Consumer> | ||||
|         {({ styling, invertTheme, base16Theme }) => { | ||||
|           return ( | ||||
|             <div {...styling('treeWrapper')}> | ||||
|               {before} | ||||
|               <JSONTree | ||||
|                 keyPath={keyPath} | ||||
|                 shouldExpandNode={shouldExpandNode} | ||||
|                 data={data} | ||||
|                 labelRenderer={this.selectLabelRenderer(styling)} | ||||
|                 theme={this.selectTheme(base16Theme)} | ||||
|                 invertTheme={invertTheme} | ||||
|                 getItemString={this.selectGetItemString(styling)} | ||||
|                 hideRoot={hideRoot} | ||||
|               /> | ||||
|               {after} | ||||
|               {children} | ||||
|             </div> | ||||
|           ); | ||||
|         }} | ||||
|       </StyleUtilsContext.Consumer> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,12 @@ | |||
| import * as React from 'react'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| 
 | ||||
| export type UListProps = React.HTMLAttributes<HTMLUListElement>; | ||||
| 
 | ||||
| export function UList(props: UListProps): JSX.Element { | ||||
|   return ( | ||||
|     <StyleUtilsContext.Consumer> | ||||
|       {({ styling }) => <ul {...props} {...styling('uList')} />} | ||||
|     </StyleUtilsContext.Consumer> | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,246 @@ | |||
| import React, { ReactNode } from 'react'; | ||||
| import { StyleUtilsContext } from '../styles/createStylingFromTheme'; | ||||
| import { createTreeItemLabelRenderer } from '../styles/tree'; | ||||
| import { | ||||
|   QueryPreviewTabs, | ||||
|   RtkResourceInfo, | ||||
|   SelectorsSource, | ||||
|   TabOption, | ||||
| } from '../types'; | ||||
| import { QueryPreviewHeader } from '../components/QueryPreviewHeader'; | ||||
| import { | ||||
|   QueryPreviewInfo, | ||||
|   QueryPreviewInfoProps, | ||||
| } from '../components/QueryPreviewInfo'; | ||||
| import { | ||||
|   QueryPreviewApi, | ||||
|   QueryPreviewApiProps, | ||||
| } from '../components/QueryPreviewApi'; | ||||
| import { | ||||
|   QueryPreviewSubscriptions, | ||||
|   QueryPreviewSubscriptionsProps, | ||||
| } from '../components/QueryPreviewSubscriptions'; | ||||
| import { | ||||
|   QueryPreviewTags, | ||||
|   QueryPreviewTagsProps, | ||||
| } from '../components/QueryPreviewTags'; | ||||
| import { NoRtkQueryApi } from '../components/NoRtkQueryApi'; | ||||
| import { InspectorSelectors } from '../selectors'; | ||||
| import { StylingFunction } from 'react-base16-styling'; | ||||
| import { mapProps } from './mapProps'; | ||||
| import { | ||||
|   QueryPreviewActions, | ||||
|   QueryPreviewActionsProps, | ||||
| } from '../components/QueryPreviewActions'; | ||||
| import { isTabVisible } from '../utils/tabs'; | ||||
| 
 | ||||
| export interface QueryPreviewProps<S = unknown> { | ||||
|   readonly selectedTab: QueryPreviewTabs; | ||||
|   readonly hasNoApis: boolean; | ||||
|   readonly onTabChange: (tab: QueryPreviewTabs) => void; | ||||
|   readonly resInfo: RtkResourceInfo | null; | ||||
|   readonly styling: StylingFunction; | ||||
|   readonly isWideLayout: boolean; | ||||
|   readonly selectorsSource: SelectorsSource<S>; | ||||
|   readonly selectors: InspectorSelectors<S>; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Tab content is not rendered if there's no selected query. | ||||
|  */ | ||||
| type QueryPreviewTabProps = Omit<QueryPreviewProps<unknown>, 'resInfo'> & { | ||||
|   resInfo: RtkResourceInfo; | ||||
| }; | ||||
| 
 | ||||
| const MappedQueryPreviewTags = mapProps< | ||||
|   QueryPreviewTabProps, | ||||
|   QueryPreviewTagsProps | ||||
| >(({ selectors, selectorsSource, isWideLayout, resInfo }) => ({ | ||||
|   resInfo, | ||||
|   tags: selectors.selectCurrentQueryTags(selectorsSource), | ||||
|   isWideLayout, | ||||
| }))(QueryPreviewTags); | ||||
| 
 | ||||
| const MappedQueryPreviewInfo = mapProps< | ||||
|   QueryPreviewTabProps, | ||||
|   QueryPreviewInfoProps | ||||
| >(({ resInfo, isWideLayout }) => ({ resInfo, isWideLayout }))(QueryPreviewInfo); | ||||
| 
 | ||||
| const MappedQuerySubscriptipns = mapProps< | ||||
|   QueryPreviewTabProps, | ||||
|   QueryPreviewSubscriptionsProps | ||||
| >(({ selectors, selectorsSource, isWideLayout }) => ({ | ||||
|   isWideLayout, | ||||
|   subscriptions: selectors.selectSubscriptionsOfCurrentQuery(selectorsSource), | ||||
| }))(QueryPreviewSubscriptions); | ||||
| 
 | ||||
| const MappedApiPreview = mapProps<QueryPreviewTabProps, QueryPreviewApiProps>( | ||||
|   ({ isWideLayout, selectors, selectorsSource }) => ({ | ||||
|     isWideLayout, | ||||
|     apiState: selectors.selectApiOfCurrentQuery(selectorsSource), | ||||
|     apiStats: selectors.selectApiStatsOfCurrentQuery(selectorsSource), | ||||
|   }) | ||||
| )(QueryPreviewApi); | ||||
| 
 | ||||
| const MappedQueryPreviewActions = mapProps< | ||||
|   QueryPreviewTabProps, | ||||
|   QueryPreviewActionsProps | ||||
| >(({ isWideLayout, selectorsSource, selectors }) => ({ | ||||
|   isWideLayout, | ||||
|   actionsOfQuery: selectors.selectActionsOfCurrentQuery(selectorsSource), | ||||
| }))(QueryPreviewActions); | ||||
| 
 | ||||
| const tabs: ReadonlyArray< | ||||
|   TabOption<QueryPreviewTabs, QueryPreviewTabProps, RtkResourceInfo['type']> | ||||
| > = [ | ||||
|   { | ||||
|     label: 'query', | ||||
|     value: QueryPreviewTabs.queryinfo, | ||||
|     component: MappedQueryPreviewInfo, | ||||
|     visible: { | ||||
|       query: true, | ||||
|       mutation: true, | ||||
|       default: true, | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     label: 'actions', | ||||
|     value: QueryPreviewTabs.actions, | ||||
|     component: MappedQueryPreviewActions, | ||||
|     visible: { | ||||
|       query: true, | ||||
|       mutation: true, | ||||
|       default: true, | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     label: 'tags', | ||||
|     value: QueryPreviewTabs.queryTags, | ||||
|     component: MappedQueryPreviewTags, | ||||
|     visible: { | ||||
|       query: true, | ||||
|       mutation: false, | ||||
|       default: true, | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     label: 'subs', | ||||
|     value: QueryPreviewTabs.querySubscriptions, | ||||
|     component: MappedQuerySubscriptipns, | ||||
|     visible: { | ||||
|       query: true, | ||||
|       mutation: false, | ||||
|       default: true, | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     label: 'api', | ||||
|     value: QueryPreviewTabs.apiConfig, | ||||
|     component: MappedApiPreview, | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export class QueryPreview<S> extends React.PureComponent<QueryPreviewProps<S>> { | ||||
|   readonly labelRenderer: ReturnType<typeof createTreeItemLabelRenderer>; | ||||
| 
 | ||||
|   constructor(props: QueryPreviewProps<S>) { | ||||
|     super(props); | ||||
| 
 | ||||
|     this.labelRenderer = createTreeItemLabelRenderer(this.props.styling); | ||||
|   } | ||||
| 
 | ||||
|   renderLabelWithCounter = ( | ||||
|     label: React.ReactText, | ||||
|     counter: number | ||||
|   ): string => { | ||||
|     let counterAsString = counter.toFixed(0); | ||||
| 
 | ||||
|     if (counterAsString.length > 3) { | ||||
|       counterAsString = counterAsString.slice(0, 2) + '...'; | ||||
|     } | ||||
| 
 | ||||
|     return `${label}(${counterAsString})`; | ||||
|   }; | ||||
| 
 | ||||
|   renderTabLabel = ( | ||||
|     tab: TabOption<QueryPreviewTabs, unknown, 'query' | 'mutation'> | ||||
|   ): ReactNode => { | ||||
|     const { selectors, selectorsSource, resInfo } = this.props; | ||||
|     const tabCount = selectors.selectTabCounters(selectorsSource)[tab.value]; | ||||
| 
 | ||||
|     let tabLabel = tab.label; | ||||
| 
 | ||||
|     if (tabLabel === 'query' && resInfo?.type === 'mutation') { | ||||
|       tabLabel = resInfo.type; | ||||
|     } | ||||
| 
 | ||||
|     if (tabCount > 0) { | ||||
|       return this.renderLabelWithCounter(tabLabel, tabCount); | ||||
|     } | ||||
| 
 | ||||
|     return tabLabel; | ||||
|   }; | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { resInfo, selectedTab, onTabChange, hasNoApis } = this.props; | ||||
| 
 | ||||
|     const { component: TabComponent } = | ||||
|       tabs.find((tab) => tab.value === selectedTab) || tabs[0]; | ||||
| 
 | ||||
|     if (!resInfo) { | ||||
|       return ( | ||||
|         <StyleUtilsContext.Consumer> | ||||
|           {({ styling }) => ( | ||||
|             <div {...styling('queryPreview')}> | ||||
|               <QueryPreviewHeader | ||||
|                 selectedTab={selectedTab} | ||||
|                 onTabChange={onTabChange} | ||||
|                 tabs={ | ||||
|                   tabs.filter((tab) => | ||||
|                     isTabVisible(tab, 'default') | ||||
|                   ) as ReadonlyArray< | ||||
|                     TabOption< | ||||
|                       QueryPreviewTabs, | ||||
|                       unknown, | ||||
|                       RtkResourceInfo['type'] | ||||
|                     > | ||||
|                   > | ||||
|                 } | ||||
|                 renderTabLabel={this.renderTabLabel} | ||||
|               /> | ||||
|               {hasNoApis && <NoRtkQueryApi />} | ||||
|             </div> | ||||
|           )} | ||||
|         </StyleUtilsContext.Consumer> | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return ( | ||||
|       <StyleUtilsContext.Consumer> | ||||
|         {({ styling }) => { | ||||
|           return ( | ||||
|             <div {...styling('queryPreview')}> | ||||
|               <QueryPreviewHeader | ||||
|                 selectedTab={selectedTab} | ||||
|                 onTabChange={onTabChange} | ||||
|                 tabs={ | ||||
|                   tabs.filter((tab) => | ||||
|                     isTabVisible(tab, resInfo.type) | ||||
|                   ) as ReadonlyArray< | ||||
|                     TabOption< | ||||
|                       QueryPreviewTabs, | ||||
|                       unknown, | ||||
|                       RtkResourceInfo['type'] | ||||
|                     > | ||||
|                   > | ||||
|                 } | ||||
|                 renderTabLabel={this.renderTabLabel} | ||||
|               /> | ||||
|               <TabComponent {...(this.props as QueryPreviewTabProps)} /> | ||||
|             </div> | ||||
|           ); | ||||
|         }} | ||||
|       </StyleUtilsContext.Consumer> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -0,0 +1,166 @@ | |||
| import React, { PureComponent, createRef, ReactNode } from 'react'; | ||||
| import type { AnyAction, Dispatch, Action } from '@reduxjs/toolkit'; | ||||
| import type { LiftedAction, LiftedState } from '@redux-devtools/core'; | ||||
| import { | ||||
|   QueryFormValues, | ||||
|   QueryPreviewTabs, | ||||
|   RtkQueryMonitorState, | ||||
|   StyleUtils, | ||||
|   SelectorsSource, | ||||
|   RtkResourceInfo, | ||||
| } from '../types'; | ||||
| import { createInspectorSelectors, computeSelectorSource } from '../selectors'; | ||||
| import { | ||||
|   changeQueryFormValues, | ||||
|   selectedPreviewTab, | ||||
|   selectQueryKey, | ||||
| } from '../reducers'; | ||||
| import { QueryList } from '../components/QueryList'; | ||||
| import { QueryForm } from '../components/QueryForm'; | ||||
| import { QueryPreview } from './QueryPreview'; | ||||
| 
 | ||||
| type ForwardedMonitorProps<S, A extends Action<unknown>> = Pick< | ||||
|   LiftedState<S, A, RtkQueryMonitorState>, | ||||
|   'monitorState' | 'currentStateIndex' | 'computedStates' | 'actionsById' | ||||
| >; | ||||
| 
 | ||||
| export interface RtkQueryInspectorProps<S, A extends Action<unknown>> | ||||
|   extends ForwardedMonitorProps<S, A> { | ||||
|   dispatch: Dispatch<LiftedAction<S, A, RtkQueryMonitorState>>; | ||||
|   styleUtils: StyleUtils; | ||||
| } | ||||
| 
 | ||||
| type RtkQueryInspectorState<S> = { | ||||
|   selectorsSource: SelectorsSource<S>; | ||||
|   isWideLayout: boolean; | ||||
| }; | ||||
| 
 | ||||
| class RtkQueryInspector<S, A extends Action<unknown>> extends PureComponent< | ||||
|   RtkQueryInspectorProps<S, A>, | ||||
|   RtkQueryInspectorState<S> | ||||
| > { | ||||
|   inspectorRef = createRef<HTMLDivElement>(); | ||||
| 
 | ||||
|   isWideIntervalRef: ReturnType<typeof setInterval> | null = null; | ||||
| 
 | ||||
|   constructor(props: RtkQueryInspectorProps<S, A>) { | ||||
|     super(props); | ||||
| 
 | ||||
|     this.state = { | ||||
|       isWideLayout: true, | ||||
|       selectorsSource: computeSelectorSource(props, null), | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   static wideLayout = 600; | ||||
| 
 | ||||
|   static getDerivedStateFromProps( | ||||
|     props: RtkQueryInspectorProps<unknown, Action<unknown>>, | ||||
|     state: RtkQueryInspectorState<unknown> | ||||
|   ): null | Partial<RtkQueryInspectorState<unknown>> { | ||||
|     const selectorsSource = computeSelectorSource<unknown, Action<unknown>>( | ||||
|       props, | ||||
|       state.selectorsSource | ||||
|     ); | ||||
| 
 | ||||
|     if (selectorsSource !== state.selectorsSource) { | ||||
|       return { | ||||
|         selectorsSource, | ||||
|       }; | ||||
|     } | ||||
| 
 | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   selectors = createInspectorSelectors<S>(); | ||||
| 
 | ||||
|   updateSizeMode = (): void => { | ||||
|     if (this.inspectorRef.current) { | ||||
|       const isWideLayout = | ||||
|         this.inspectorRef.current.offsetWidth >= RtkQueryInspector.wideLayout; | ||||
| 
 | ||||
|       if (isWideLayout !== this.state.isWideLayout) { | ||||
|         this.setState({ isWideLayout }); | ||||
|       } | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   componentDidMount(): void { | ||||
|     this.updateSizeMode(); | ||||
| 
 | ||||
|     this.isWideIntervalRef = setInterval(this.updateSizeMode, 300); | ||||
|   } | ||||
| 
 | ||||
|   componentWillUnmount(): void { | ||||
|     if (this.isWideIntervalRef) { | ||||
|       clearTimeout(this.isWideIntervalRef); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   handleQueryFormValuesChange = (values: Partial<QueryFormValues>): void => { | ||||
|     this.props.dispatch(changeQueryFormValues(values) as AnyAction); | ||||
|   }; | ||||
| 
 | ||||
|   handleSelectQuery = (queryInfo: RtkResourceInfo): void => { | ||||
|     this.props.dispatch(selectQueryKey(queryInfo) as AnyAction); | ||||
|   }; | ||||
| 
 | ||||
|   handleTabChange = (tab: QueryPreviewTabs): void => { | ||||
|     this.props.dispatch(selectedPreviewTab(tab) as AnyAction); | ||||
|   }; | ||||
| 
 | ||||
|   render(): ReactNode { | ||||
|     const { selectorsSource, isWideLayout } = this.state; | ||||
|     const { | ||||
|       styleUtils: { styling }, | ||||
|     } = this.props; | ||||
|     const allVisibleRtkResourceInfos = | ||||
|       this.selectors.selectAllVisbileQueries(selectorsSource); | ||||
| 
 | ||||
|     const currentResInfo = | ||||
|       this.selectors.selectCurrentQueryInfo(selectorsSource); | ||||
| 
 | ||||
|     const apiStates = this.selectors.selectApiStates(selectorsSource); | ||||
| 
 | ||||
|     const hasNoApi = apiStates == null; | ||||
| 
 | ||||
|     const searchQueryRegex = | ||||
|       this.selectors.selectSearchQueryRegex(selectorsSource); | ||||
| 
 | ||||
|     return ( | ||||
|       <div | ||||
|         ref={this.inspectorRef} | ||||
|         data-wide-layout={+this.state.isWideLayout} | ||||
|         {...styling('inspector')} | ||||
|       > | ||||
|         <div | ||||
|           {...styling('querySectionWrapper')} | ||||
|           data-wide-layout={+this.state.isWideLayout} | ||||
|         > | ||||
|           <QueryForm | ||||
|             searchQueryRegex={searchQueryRegex} | ||||
|             values={selectorsSource.monitorState.queryForm.values} | ||||
|             onFormValuesChange={this.handleQueryFormValuesChange} | ||||
|           /> | ||||
|           <QueryList | ||||
|             onSelectQuery={this.handleSelectQuery} | ||||
|             resInfos={allVisibleRtkResourceInfos} | ||||
|             selectedQueryKey={selectorsSource.monitorState.selectedQueryKey} | ||||
|           /> | ||||
|         </div> | ||||
|         <QueryPreview<S> | ||||
|           selectorsSource={this.state.selectorsSource} | ||||
|           selectors={this.selectors} | ||||
|           resInfo={currentResInfo} | ||||
|           selectedTab={selectorsSource.monitorState.selectedPreviewTab} | ||||
|           onTabChange={this.handleTabChange} | ||||
|           styling={styling} | ||||
|           isWideLayout={isWideLayout} | ||||
|           hasNoApis={hasNoApi} | ||||
|         /> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export default RtkQueryInspector; | ||||
|  | @ -0,0 +1,90 @@ | |||
| import React, { Component } from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import { Action, AnyAction } from 'redux'; | ||||
| import RtkQueryInspector from './RtkQueryInspector'; | ||||
| import { reducer } from '../reducers'; | ||||
| import { | ||||
|   ExternalProps, | ||||
|   RtkQueryMonitorProps, | ||||
|   RtkQueryMonitorState, | ||||
|   StyleUtils, | ||||
| } from '../types'; | ||||
| import { | ||||
|   createThemeState, | ||||
|   StyleUtilsContext, | ||||
| } from '../styles/createStylingFromTheme'; | ||||
| 
 | ||||
| interface DefaultProps { | ||||
|   theme: string; | ||||
|   invertTheme: boolean; | ||||
| } | ||||
| 
 | ||||
| export interface RtkQueryComponentState { | ||||
|   readonly styleUtils: StyleUtils; | ||||
| } | ||||
| 
 | ||||
| class RtkQueryMonitor<S, A extends Action<unknown>> extends Component< | ||||
|   RtkQueryMonitorProps<S, A>, | ||||
|   RtkQueryComponentState | ||||
| > { | ||||
|   static update = reducer; | ||||
| 
 | ||||
|   static propTypes = { | ||||
|     dispatch: PropTypes.func, | ||||
|     computedStates: PropTypes.array, | ||||
|     currentStateIndex: PropTypes.number, | ||||
|     actionsById: PropTypes.object, | ||||
|     stagedActionIds: PropTypes.array, | ||||
|     skippedActionIds: PropTypes.array, | ||||
|     monitorState: PropTypes.object, | ||||
|     theme: PropTypes.oneOfType([PropTypes.object, PropTypes.string]), | ||||
|     invertTheme: PropTypes.bool, | ||||
|   }; | ||||
| 
 | ||||
|   static defaultProps: DefaultProps = { | ||||
|     theme: 'nicinabox', | ||||
|     invertTheme: false, | ||||
|   }; | ||||
| 
 | ||||
|   constructor(props: RtkQueryMonitorProps<S, A>) { | ||||
|     super(props); | ||||
| 
 | ||||
|     this.state = { | ||||
|       styleUtils: createThemeState<S, A>(props), | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   render() { | ||||
|     const { | ||||
|       currentStateIndex, | ||||
|       computedStates, | ||||
|       monitorState, | ||||
|       dispatch, | ||||
|       actionsById, | ||||
|     } = this.props; | ||||
| 
 | ||||
|     return ( | ||||
|       <StyleUtilsContext.Provider value={this.state.styleUtils}> | ||||
|         <RtkQueryInspector<S, AnyAction> | ||||
|           computedStates={computedStates} | ||||
|           currentStateIndex={currentStateIndex} | ||||
|           monitorState={monitorState} | ||||
|           dispatch={dispatch} | ||||
|           styleUtils={this.state.styleUtils} | ||||
|           actionsById={actionsById} | ||||
|         /> | ||||
|       </StyleUtilsContext.Provider> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export default RtkQueryMonitor as unknown as React.ComponentType< | ||||
|   ExternalProps<unknown, Action<unknown>> | ||||
| > & { | ||||
|   update( | ||||
|     monitorProps: ExternalProps<unknown, Action<unknown>>, | ||||
|     state: RtkQueryMonitorState | undefined, | ||||
|     action: Action | ||||
|   ): RtkQueryMonitorState; | ||||
|   defaultProps: DefaultProps; | ||||
| }; | ||||
|  | @ -0,0 +1,29 @@ | |||
| import React, { ComponentType, ReactNode, Component } from 'react'; | ||||
| 
 | ||||
| interface Mapper<In, Out> { | ||||
|   (inProps: In): Out; | ||||
| } | ||||
| 
 | ||||
| interface MapPropsOutput<In, Out> { | ||||
|   (comp: ComponentType<Out>): ComponentType<In>; | ||||
| } | ||||
| 
 | ||||
| export function mapProps<In, Out>( | ||||
|   mapper: Mapper<In, Out> | ||||
| ): MapPropsOutput<In, Out> { | ||||
|   return function mapPropsHoc(Comp) { | ||||
|     class MapPropsHoc extends Component<In> { | ||||
|       render(): ReactNode { | ||||
|         const mappedProps = mapper(this.props); | ||||
| 
 | ||||
|         return <Comp {...mappedProps} />; | ||||
|       } | ||||
| 
 | ||||
|       static displayName = `mapProps(${ | ||||
|         Comp.displayName || Comp.name || 'Component' | ||||
|       })`;
 | ||||
|     } | ||||
| 
 | ||||
|     return MapPropsHoc; | ||||
|   }; | ||||
| } | ||||
							
								
								
									
										2
									
								
								packages/redux-devtools-rtk-query-monitor/src/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								packages/redux-devtools-rtk-query-monitor/src/index.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,2 @@ | |||
| export { default } from './containers/RtkQueryMonitor'; | ||||
| export type { ExternalProps } from './types'; | ||||
|  | @ -0,0 +1,6 @@ | |||
| export const DATA_TYPE_KEY = Symbol.for('__serializedType__'); | ||||
| 
 | ||||
| /** | ||||
|  * @see https://github.com/reduxjs/redux-toolkit/blob/b718e01d323d3ab4b913e5d88c9b90aa790bb975/src/query/core/buildSlice.ts#L259
 | ||||
|  */ | ||||
| export const missingTagId = '__internal_without_id'; | ||||
							
								
								
									
										65
									
								
								packages/redux-devtools-rtk-query-monitor/src/reducers.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								packages/redux-devtools-rtk-query-monitor/src/reducers.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,65 @@ | |||
| import { Action, AnyAction } from 'redux'; | ||||
| import { createSlice, PayloadAction } from '@reduxjs/toolkit'; | ||||
| import { | ||||
|   QueryInfo, | ||||
|   RtkQueryMonitorState, | ||||
|   QueryFormValues, | ||||
|   RtkQueryMonitorProps, | ||||
|   QueryPreviewTabs, | ||||
| } from './types'; | ||||
| import { QueryComparators } from './utils/comparators'; | ||||
| import { QueryFilters } from './utils/filters'; | ||||
| 
 | ||||
| const initialState: RtkQueryMonitorState = { | ||||
|   queryForm: { | ||||
|     values: { | ||||
|       queryComparator: QueryComparators.fulfilledTimeStamp, | ||||
|       isAscendingQueryComparatorOrder: false, | ||||
|       searchValue: '', | ||||
|       isRegexSearch: false, | ||||
|       queryFilter: QueryFilters.queryKey, | ||||
|     }, | ||||
|   }, | ||||
|   selectedPreviewTab: QueryPreviewTabs.queryinfo, | ||||
|   selectedQueryKey: null, | ||||
| }; | ||||
| 
 | ||||
| const monitorSlice = createSlice({ | ||||
|   /** | ||||
|    * `@@` prefix is mandatory. | ||||
|    * @see lifedAction @ `packages/redux-devtools-app/src/actions/index.ts` | ||||
|    */ | ||||
|   name: '@@rtk-query-monitor', | ||||
|   initialState, | ||||
|   reducers: { | ||||
|     changeQueryFormValues( | ||||
|       state, | ||||
|       action: PayloadAction<Partial<QueryFormValues>> | ||||
|     ) { | ||||
|       state.queryForm.values = { ...state.queryForm.values, ...action.payload }; | ||||
|     }, | ||||
|     selectQueryKey( | ||||
|       state, | ||||
|       action: PayloadAction<Pick<QueryInfo, 'reducerPath' | 'queryKey'>> | ||||
|     ) { | ||||
|       state.selectedQueryKey = { | ||||
|         queryKey: action.payload.queryKey, | ||||
|         reducerPath: action.payload.reducerPath, | ||||
|       }; | ||||
|     }, | ||||
|     selectedPreviewTab(state, action: PayloadAction<QueryPreviewTabs>) { | ||||
|       state.selectedPreviewTab = action.payload; | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
| 
 | ||||
| export function reducer<S, A extends Action<unknown>>( | ||||
|   props: RtkQueryMonitorProps<S, A>, | ||||
|   state: RtkQueryMonitorState | undefined, | ||||
|   action: AnyAction | ||||
| ): RtkQueryMonitorState { | ||||
|   return monitorSlice.reducer(state, action); | ||||
| } | ||||
| 
 | ||||
| export const { selectQueryKey, changeQueryFormValues, selectedPreviewTab } = | ||||
|   monitorSlice.actions; | ||||
							
								
								
									
										283
									
								
								packages/redux-devtools-rtk-query-monitor/src/selectors.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										283
									
								
								packages/redux-devtools-rtk-query-monitor/src/selectors.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,283 @@ | |||
| import { Action, createSelector, Selector } from '@reduxjs/toolkit'; | ||||
| import { RtkQueryInspectorProps } from './containers/RtkQueryInspector'; | ||||
| import { | ||||
|   ApiStats, | ||||
|   QueryInfo, | ||||
|   RtkQueryApiState, | ||||
|   RtkQueryTag, | ||||
|   SelectorsSource, | ||||
|   RtkQueryProvided, | ||||
|   QueryPreviewTabs, | ||||
|   RtkResourceInfo, | ||||
| } from './types'; | ||||
| import { Comparator, queryComparators } from './utils/comparators'; | ||||
| import { FilterList, queryListFilters } from './utils/filters'; | ||||
| import { emptyRecord } from './utils/object'; | ||||
| import { escapeRegExpSpecialCharacter } from './utils/regexp'; | ||||
| import { | ||||
|   getApiStatesOf, | ||||
|   extractAllApiQueries, | ||||
|   flipComparator, | ||||
|   getQueryTagsOf, | ||||
|   generateApiStatsOfCurrentQuery, | ||||
|   getActionsOfCurrentQuery, | ||||
|   extractAllApiMutations, | ||||
| } from './utils/rtk-query'; | ||||
| 
 | ||||
| type InspectorSelector<S, Output> = Selector<SelectorsSource<S>, Output>; | ||||
| 
 | ||||
| export function computeSelectorSource<S, A extends Action<unknown>>( | ||||
|   props: RtkQueryInspectorProps<S, A>, | ||||
|   previous: SelectorsSource<S> | null = null | ||||
| ): SelectorsSource<S> { | ||||
|   const { computedStates, currentStateIndex, monitorState, actionsById } = | ||||
|     props; | ||||
| 
 | ||||
|   const userState = | ||||
|     computedStates.length > 0 ? computedStates[currentStateIndex].state : null; | ||||
| 
 | ||||
|   if ( | ||||
|     !previous || | ||||
|     previous.userState !== userState || | ||||
|     previous.monitorState !== monitorState || | ||||
|     previous.actionsById !== actionsById | ||||
|   ) { | ||||
|     return { | ||||
|       userState, | ||||
|       monitorState, | ||||
|       currentStateIndex, | ||||
|       actionsById, | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   return previous; | ||||
| } | ||||
| 
 | ||||
| export interface InspectorSelectors<S> { | ||||
|   readonly selectQueryComparator: InspectorSelector<S, Comparator<QueryInfo>>; | ||||
|   readonly selectApiStates: InspectorSelector< | ||||
|     S, | ||||
|     ReturnType<typeof getApiStatesOf> | ||||
|   >; | ||||
|   readonly selectAllQueries: InspectorSelector< | ||||
|     S, | ||||
|     ReturnType<typeof extractAllApiQueries> | ||||
|   >; | ||||
|   readonly selectAllVisbileQueries: InspectorSelector<S, RtkResourceInfo[]>; | ||||
|   readonly selectCurrentQueryInfo: InspectorSelector<S, RtkResourceInfo | null>; | ||||
|   readonly selectSearchQueryRegex: InspectorSelector<S, RegExp | null>; | ||||
|   readonly selectCurrentQueryTags: InspectorSelector<S, RtkQueryTag[]>; | ||||
|   readonly selectApiStatsOfCurrentQuery: InspectorSelector<S, ApiStats | null>; | ||||
|   readonly selectApiOfCurrentQuery: InspectorSelector< | ||||
|     S, | ||||
|     RtkQueryApiState | null | ||||
|   >; | ||||
|   readonly selectTabCounters: InspectorSelector< | ||||
|     S, | ||||
|     Record<QueryPreviewTabs, number> | ||||
|   >; | ||||
|   readonly selectSubscriptionsOfCurrentQuery: InspectorSelector< | ||||
|     S, | ||||
|     RtkQueryApiState['subscriptions'][string] | ||||
|   >; | ||||
|   readonly selectActionsOfCurrentQuery: InspectorSelector< | ||||
|     S, | ||||
|     ReturnType<typeof getActionsOfCurrentQuery> | ||||
|   >; | ||||
| } | ||||
| 
 | ||||
| export function createInspectorSelectors<S>(): InspectorSelectors<S> { | ||||
|   const selectQueryComparator = ({ | ||||
|     monitorState, | ||||
|   }: SelectorsSource<S>): Comparator<RtkResourceInfo> => { | ||||
|     return queryComparators[monitorState.queryForm.values.queryComparator]; | ||||
|   }; | ||||
| 
 | ||||
|   const selectQueryListFilter = ({ | ||||
|     monitorState, | ||||
|   }: SelectorsSource<S>): FilterList<RtkResourceInfo> => { | ||||
|     return queryListFilters[monitorState.queryForm.values.queryFilter]; | ||||
|   }; | ||||
| 
 | ||||
|   const selectActionsById = ({ actionsById }: SelectorsSource<S>) => | ||||
|     actionsById; | ||||
| 
 | ||||
|   const selectApiStates = createSelector( | ||||
|     ({ userState }: SelectorsSource<S>) => userState, | ||||
|     getApiStatesOf | ||||
|   ); | ||||
|   const selectAllQueries = createSelector( | ||||
|     selectApiStates, | ||||
|     extractAllApiQueries | ||||
|   ); | ||||
| 
 | ||||
|   const selectAllMutations = createSelector( | ||||
|     selectApiStates, | ||||
|     extractAllApiMutations | ||||
|   ); | ||||
| 
 | ||||
|   const selectSearchQueryRegex = createSelector( | ||||
|     ({ monitorState }: SelectorsSource<S>) => | ||||
|       monitorState.queryForm.values.searchValue, | ||||
|     ({ monitorState }: SelectorsSource<S>) => | ||||
|       monitorState.queryForm.values.isRegexSearch, | ||||
|     (searchValue, isRegexSearch) => { | ||||
|       if (searchValue) { | ||||
|         try { | ||||
|           const regexPattern = isRegexSearch | ||||
|             ? searchValue | ||||
|             : escapeRegExpSpecialCharacter(searchValue); | ||||
| 
 | ||||
|           return new RegExp(regexPattern, 'i'); | ||||
|         } catch (err) { | ||||
|           // We notify that the search regex provided is not valid
 | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       return null; | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
|   const selectComparatorOrder = ({ monitorState }: SelectorsSource<S>) => | ||||
|     monitorState.queryForm.values.isAscendingQueryComparatorOrder; | ||||
| 
 | ||||
|   const selectAllVisbileQueries = createSelector( | ||||
|     [ | ||||
|       selectQueryComparator, | ||||
|       selectQueryListFilter, | ||||
|       selectAllQueries, | ||||
|       selectAllMutations, | ||||
|       selectComparatorOrder, | ||||
|       selectSearchQueryRegex, | ||||
|     ], | ||||
|     ( | ||||
|       comparator, | ||||
|       queryListFilter, | ||||
|       queryList, | ||||
|       mutationsList, | ||||
|       isAscending, | ||||
|       searchRegex | ||||
|     ) => { | ||||
|       const filteredList = queryListFilter( | ||||
|         searchRegex, | ||||
|         (queryList as RtkResourceInfo[]).concat(mutationsList) | ||||
|       ); | ||||
| 
 | ||||
|       const computedComparator = isAscending | ||||
|         ? comparator | ||||
|         : flipComparator(comparator); | ||||
| 
 | ||||
|       return filteredList.slice().sort(computedComparator); | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
|   const selectCurrentQueryInfo = createSelector( | ||||
|     selectAllQueries, | ||||
|     selectAllMutations, | ||||
|     ({ monitorState }: SelectorsSource<S>) => monitorState.selectedQueryKey, | ||||
|     (allQueries, allMutations, selectedQueryKey) => { | ||||
|       if (!selectedQueryKey) { | ||||
|         return null; | ||||
|       } | ||||
| 
 | ||||
|       let currentQueryInfo: null | RtkResourceInfo = | ||||
|         allQueries.find( | ||||
|           (query) => | ||||
|             query.queryKey === selectedQueryKey.queryKey && | ||||
|             selectedQueryKey.reducerPath === query.reducerPath | ||||
|         ) || null; | ||||
| 
 | ||||
|       if (!currentQueryInfo) { | ||||
|         currentQueryInfo = | ||||
|           allMutations.find( | ||||
|             (mutation) => | ||||
|               mutation.queryKey === selectedQueryKey.queryKey && | ||||
|               selectedQueryKey.reducerPath === mutation.reducerPath | ||||
|           ) || null; | ||||
|       } | ||||
| 
 | ||||
|       return currentQueryInfo; | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
|   const selectApiOfCurrentQuery: InspectorSelector<S, null | RtkQueryApiState> = | ||||
|     (selectorsSource: SelectorsSource<S>) => { | ||||
|       const apiStates = selectApiStates(selectorsSource); | ||||
|       const currentQueryInfo = selectCurrentQueryInfo(selectorsSource); | ||||
| 
 | ||||
|       if (!apiStates || !currentQueryInfo) { | ||||
|         return null; | ||||
|       } | ||||
| 
 | ||||
|       return apiStates[currentQueryInfo.reducerPath] ?? null; | ||||
|     }; | ||||
| 
 | ||||
|   const selectProvidedOfCurrentQuery: InspectorSelector< | ||||
|     S, | ||||
|     null | RtkQueryProvided | ||||
|   > = (selectorsSource: SelectorsSource<S>) => { | ||||
|     return selectApiOfCurrentQuery(selectorsSource)?.provided ?? null; | ||||
|   }; | ||||
| 
 | ||||
|   const selectSubscriptionsOfCurrentQuery = createSelector( | ||||
|     [selectApiOfCurrentQuery, selectCurrentQueryInfo], | ||||
|     (apiState, queryInfo) => { | ||||
|       if (!queryInfo || !apiState) { | ||||
|         return emptyRecord; | ||||
|       } | ||||
| 
 | ||||
|       return apiState.subscriptions[queryInfo.queryKey]; | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
|   const selectCurrentQueryTags = createSelector( | ||||
|     [selectCurrentQueryInfo, selectProvidedOfCurrentQuery], | ||||
|     getQueryTagsOf | ||||
|   ); | ||||
| 
 | ||||
|   const selectApiStatsOfCurrentQuery = createSelector( | ||||
|     selectApiOfCurrentQuery, | ||||
|     (selectorsSource) => selectorsSource.actionsById, | ||||
|     (selectorsSource) => selectorsSource.currentStateIndex, | ||||
|     generateApiStatsOfCurrentQuery | ||||
|   ); | ||||
| 
 | ||||
|   const selectActionsOfCurrentQuery = createSelector( | ||||
|     selectCurrentQueryInfo, | ||||
|     selectActionsById, | ||||
|     getActionsOfCurrentQuery | ||||
|   ); | ||||
| 
 | ||||
|   const selectTabCounters = createSelector( | ||||
|     [ | ||||
|       selectSubscriptionsOfCurrentQuery, | ||||
|       selectActionsOfCurrentQuery, | ||||
|       selectCurrentQueryTags, | ||||
|     ], | ||||
|     (subscriptions, actions, tags) => { | ||||
|       return { | ||||
|         [QueryPreviewTabs.queryTags]: tags.length, | ||||
|         [QueryPreviewTabs.querySubscriptions]: Object.keys(subscriptions ?? {}) | ||||
|           .length, | ||||
|         [QueryPreviewTabs.apiConfig]: 0, | ||||
|         [QueryPreviewTabs.queryinfo]: 0, | ||||
|         [QueryPreviewTabs.actions]: actions.length, | ||||
|       }; | ||||
|     } | ||||
|   ); | ||||
| 
 | ||||
|   return { | ||||
|     selectQueryComparator, | ||||
|     selectApiStates, | ||||
|     selectAllQueries, | ||||
|     selectAllVisbileQueries, | ||||
|     selectSearchQueryRegex, | ||||
|     selectCurrentQueryInfo, | ||||
|     selectCurrentQueryTags, | ||||
|     selectApiStatsOfCurrentQuery, | ||||
|     selectSubscriptionsOfCurrentQuery, | ||||
|     selectApiOfCurrentQuery, | ||||
|     selectTabCounters, | ||||
|     selectActionsOfCurrentQuery, | ||||
|   }; | ||||
| } | ||||
|  | @ -0,0 +1,559 @@ | |||
| import jss, { StyleSheet } from 'jss'; | ||||
| import preset from 'jss-preset-default'; | ||||
| import { | ||||
|   createStyling, | ||||
|   getBase16Theme, | ||||
|   invertTheme, | ||||
|   StylingConfig, | ||||
| } from 'react-base16-styling'; | ||||
| import rgba from 'hex-rgba'; | ||||
| import * as reduxThemes from 'redux-devtools-themes'; | ||||
| import { Action } from 'redux'; | ||||
| import { RtkQueryMonitorProps, StyleUtils } from '../types'; | ||||
| import { createContext } from 'react'; | ||||
| 
 | ||||
| jss.setup(preset()); | ||||
| 
 | ||||
| export const colorMap = (theme: reduxThemes.Base16Theme) => | ||||
|   ({ | ||||
|     TEXT_COLOR: theme.base06, | ||||
|     TEXT_PLACEHOLDER_COLOR: rgba(theme.base06, 60), | ||||
|     BACKGROUND_COLOR: theme.base00, | ||||
|     SELECTED_BACKGROUND_COLOR: rgba(theme.base03, 20), | ||||
|     SKIPPED_BACKGROUND_COLOR: rgba(theme.base03, 10), | ||||
|     HEADER_BACKGROUND_COLOR: rgba(theme.base03, 30), | ||||
|     HEADER_BORDER_COLOR: rgba(theme.base03, 20), | ||||
|     BORDER_COLOR: rgba(theme.base03, 50), | ||||
|     LIST_BORDER_COLOR: rgba(theme.base03, 50), | ||||
|     ACTION_TIME_BACK_COLOR: rgba(theme.base03, 20), | ||||
|     ACTION_TIME_COLOR: theme.base04, | ||||
|     PIN_COLOR: theme.base04, | ||||
|     ITEM_HINT_COLOR: rgba(theme.base0F, 90), | ||||
|     TAB_BACK_SELECTED_COLOR: rgba(theme.base03, 20), | ||||
|     TAB_BACK_COLOR: rgba(theme.base00, 70), | ||||
|     TAB_BACK_HOVER_COLOR: rgba(theme.base03, 40), | ||||
|     TAB_BORDER_COLOR: rgba(theme.base03, 50), | ||||
|     DIFF_ADD_COLOR: rgba(theme.base0B, 40), | ||||
|     DIFF_REMOVE_COLOR: rgba(theme.base08, 40), | ||||
|     DIFF_ARROW_COLOR: theme.base0E, | ||||
|     LINK_COLOR: rgba(theme.base0E, 90), | ||||
|     LINK_HOVER_COLOR: theme.base0E, | ||||
|     ERROR_COLOR: theme.base08, | ||||
|     ULIST_DISC_COLOR: theme.base0D, | ||||
|     ULIST_COLOR: rgba(theme.base06, 60), | ||||
|     ULIST_STRONG_COLOR: theme.base0B, | ||||
|     TAB_CONTENT_COLOR: rgba(theme.base06, 60), | ||||
|     TOGGLE_BUTTON_BACKGROUND: rgba(theme.base00, 70), | ||||
|     TOGGLE_BUTTON_SELECTED_BACKGROUND: theme.base04, | ||||
|     TOGGLE_BUTTON_ERROR: rgba(theme.base08, 40), | ||||
|   } as const); | ||||
| 
 | ||||
| type Color = keyof ReturnType<typeof colorMap>; | ||||
| type ColorMap = { | ||||
|   [color in Color]: string; | ||||
| }; | ||||
| 
 | ||||
| const getSheetFromColorMap = (map: ColorMap) => { | ||||
|   const appearanceNone = { | ||||
|     '-webkit-appearance': 'none', | ||||
|   }; | ||||
| 
 | ||||
|   return { | ||||
|     inspector: { | ||||
|       display: 'flex', | ||||
|       flexFlow: 'column nowrap', | ||||
|       overflow: 'hidden', | ||||
|       width: '100%', | ||||
|       height: '100%', | ||||
|       'font-family': 'monaco, Consolas, "Lucida Console", monospace', | ||||
|       'font-size': '12px', | ||||
|       'font-smoothing': 'antialiased', | ||||
|       'line-height': '1.5em', | ||||
| 
 | ||||
|       'background-color': map.BACKGROUND_COLOR, | ||||
|       color: map.TEXT_COLOR, | ||||
| 
 | ||||
|       '&[data-wide-layout="1"]': { | ||||
|         flexFlow: 'row nowrap', | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     querySectionWrapper: { | ||||
|       display: 'flex', | ||||
|       flex: '0 0 auto', | ||||
|       height: '50%', | ||||
|       width: '100%', | ||||
|       borderColor: map.TAB_BORDER_COLOR, | ||||
| 
 | ||||
|       '&[data-wide-layout="0"]': { | ||||
|         borderBottomWidth: 1, | ||||
|         borderStyle: 'solid', | ||||
|       }, | ||||
| 
 | ||||
|       '&[data-wide-layout="1"]': { | ||||
|         height: '100%', | ||||
|         width: '44%', | ||||
|         borderRightWidth: 1, | ||||
|         borderStyle: 'solid', | ||||
|       }, | ||||
|       flexFlow: 'column nowrap', | ||||
|       '& > :first-child': { | ||||
|         flex: '0 0 auto', | ||||
|         'border-bottom-width': '1px', | ||||
|         'border-bottom-style': 'solid', | ||||
|         'border-color': map.LIST_BORDER_COLOR, | ||||
|       }, | ||||
|       '& > :nth-child(n + 2)': { | ||||
|         flex: '1 1 auto', | ||||
|         overflowX: 'hidden', | ||||
|         overflowY: 'auto', | ||||
|         maxHeight: 'calc(100% - 70px)', | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     queryList: { | ||||
|       listStyle: 'none', | ||||
|       margin: '0', | ||||
|       padding: '0', | ||||
|     }, | ||||
| 
 | ||||
|     queryListItem: { | ||||
|       'border-bottom-width': '1px', | ||||
|       'border-bottom-style': 'solid', | ||||
|       display: 'flex', | ||||
|       'justify-content': 'space-between', | ||||
|       padding: '5px 10px', | ||||
|       cursor: 'pointer', | ||||
|       'user-select': 'none', | ||||
|       '&:last-child': { | ||||
|         'border-bottom-width': 0, | ||||
|       }, | ||||
|       overflow: 'hidden', | ||||
|       maxHeight: 47, | ||||
|       'border-bottom-color': map.BORDER_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     queryListItemKey: { | ||||
|       display: '-webkit-box', | ||||
|       boxOrient: 'vertical', | ||||
|       '-webkit-line-clamp': 2, | ||||
|       whiteSpace: 'normal', | ||||
|       overflow: 'hidden', | ||||
|       width: '100%', | ||||
|       maxWidth: 'calc(100% - 70px)', | ||||
|       wordBreak: 'break-all', | ||||
|       margin: 0, | ||||
|     }, | ||||
| 
 | ||||
|     queryListHeader: { | ||||
|       display: 'flex', | ||||
|       padding: 4, | ||||
|       flex: '0 0 auto', | ||||
|       'align-items': 'center', | ||||
|       'border-bottom-width': '1px', | ||||
|       'border-bottom-style': 'solid', | ||||
| 
 | ||||
|       'border-color': map.LIST_BORDER_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     queryStatusWrapper: { | ||||
|       display: 'flex', | ||||
|       width: 'auto', | ||||
|       justifyContent: 'center', | ||||
|       alignItems: 'center', | ||||
|       margin: 0, | ||||
|       flex: '0 0 auto', | ||||
|       overflow: 'hidden', | ||||
|     }, | ||||
| 
 | ||||
|     queryType: { | ||||
|       marginRight: 4, | ||||
|     }, | ||||
| 
 | ||||
|     queryStatus: { | ||||
|       display: 'inline-flex', | ||||
|       alignItems: 'center', | ||||
|       justifyContent: 'center', | ||||
|       height: 22, | ||||
|       padding: '0 6px', | ||||
|       'border-radius': '3px', | ||||
|       'font-size': '0.7em', | ||||
|       'line-height': '1em', | ||||
|       'flex-shrink': 0, | ||||
|       fontWeight: 700, | ||||
|       'background-color': map.ACTION_TIME_BACK_COLOR, | ||||
|       color: map.ACTION_TIME_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     queryListItemSelected: { | ||||
|       'background-color': map.SELECTED_BACKGROUND_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     tabSelector: { | ||||
|       display: 'flex', | ||||
|       width: '100%', | ||||
|       justifyContent: 'flex-end', | ||||
|       overflow: 'hidden', | ||||
|       '& > *': { | ||||
|         flex: '0 1 auto', | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     srOnly: { | ||||
|       position: 'absolute', | ||||
|       width: 1, | ||||
|       height: 1, | ||||
|       padding: 0, | ||||
|       margin: '-1px', | ||||
|       overflow: 'hidden', | ||||
|       clip: 'rect(0,0,0,0)', | ||||
|       border: 0, | ||||
|     }, | ||||
| 
 | ||||
|     selectorButton: { | ||||
|       cursor: 'pointer', | ||||
|       position: 'relative', | ||||
|       height: '33px', | ||||
|       padding: '0 8px', | ||||
|       display: 'inline-flex', | ||||
|       alignItems: 'center', | ||||
|       color: map.TEXT_COLOR, | ||||
|       'border-style': 'solid', | ||||
|       'border-width': '1px', | ||||
|       'border-left-width': 0, | ||||
| 
 | ||||
|       '&:first-child': { | ||||
|         'border-left-width': '1px', | ||||
|         'border-top-left-radius': '3px', | ||||
|         'border-bottom-left-radius': '3px', | ||||
|       }, | ||||
| 
 | ||||
|       '&:last-child': { | ||||
|         'border-top-right-radius': '3px', | ||||
|         'border-bottom-right-radius': '3px', | ||||
|       }, | ||||
| 
 | ||||
|       'background-color': map.TAB_BACK_COLOR, | ||||
| 
 | ||||
|       '&:hover': { | ||||
|         'background-color': map.TAB_BACK_HOVER_COLOR, | ||||
|       }, | ||||
| 
 | ||||
|       'border-color': map.TAB_BORDER_COLOR, | ||||
| 
 | ||||
|       '& > *': { | ||||
|         display: '-webkit-box', | ||||
|         boxOrient: 'vertical', | ||||
|         '-webkit-line-clamp': 1, | ||||
|         overflow: 'hidden', | ||||
|         wordBreak: 'break-all', | ||||
|         '-webkit-box-pack': 'end', | ||||
|         paddingBottom: 0, | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     selectorButtonSmall: { | ||||
|       padding: '0px 8px', | ||||
|       'font-size': '0.8em', | ||||
|     }, | ||||
| 
 | ||||
|     selectorButtonSelected: { | ||||
|       'background-color': map.TAB_BACK_SELECTED_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     sortButton: { | ||||
|       display: 'flex', | ||||
|       justifyContent: 'space-between', | ||||
|       alignItems: 'center', | ||||
|       flexFlow: 'row nowrap', | ||||
|       cursor: 'pointer', | ||||
|       position: 'relative', | ||||
|       padding: '0 8px', | ||||
|       color: map.TEXT_COLOR, | ||||
|       borderStyle: 'solid', | ||||
|       borderWidth: '1px', | ||||
|       borderRadius: '3px', | ||||
|       backgroundColor: map.TAB_BACK_COLOR, | ||||
|       borderColor: map.TAB_BORDER_COLOR, | ||||
|       height: 30, | ||||
|       fontSize: 12, | ||||
|       width: 64, | ||||
| 
 | ||||
|       '&:active': { | ||||
|         backgroundColor: map.TAB_BACK_SELECTED_COLOR, | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     toggleButton: { | ||||
|       width: '24px', | ||||
|       height: '24px', | ||||
|       display: 'inline-block', | ||||
|       flex: '0 0 auto', | ||||
|       color: map.TEXT_PLACEHOLDER_COLOR, | ||||
|       cursor: 'pointer', | ||||
|       padding: 0, | ||||
|       fontSize: '0.7em', | ||||
|       letterSpacing: '-0.7px', | ||||
|       outline: 'none', | ||||
|       boxShadow: 'none', | ||||
|       fontWeight: '700', | ||||
|       border: 'none', | ||||
| 
 | ||||
|       '&:hover': { | ||||
|         color: map.TEXT_COLOR, | ||||
|       }, | ||||
| 
 | ||||
|       backgroundColor: 'transparent', | ||||
|       '&[aria-pressed="true"]': { | ||||
|         color: map.BACKGROUND_COLOR, | ||||
|         backgroundColor: map.TEXT_COLOR, | ||||
|       }, | ||||
| 
 | ||||
|       '&[data-type="error"]': { | ||||
|         color: map.TEXT_COLOR, | ||||
|         backgroundColor: map.TOGGLE_BUTTON_ERROR, | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     queryForm: { | ||||
|       display: 'flex', | ||||
|       flexFlow: 'column nowrap', | ||||
|     }, | ||||
|     sortBySection: { | ||||
|       display: 'flex', | ||||
|       padding: '0.4em', | ||||
|       '& label': { | ||||
|         display: 'flex', | ||||
|         flex: '0 0 auto', | ||||
|         whiteSpace: 'noWrap', | ||||
|         alignItems: 'center', | ||||
|         paddingRight: '0.4em', | ||||
|       }, | ||||
| 
 | ||||
|       '& > :last-child': { | ||||
|         flex: '0 0 auto', | ||||
|         marginLeft: '0.4em', | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     querySearch: { | ||||
|       maxWidth: '65%', | ||||
|       'background-color': map.BACKGROUND_COLOR, | ||||
|       display: 'flex', | ||||
|       alignItems: 'center', | ||||
|       flexFlow: 'row nowrap', | ||||
|       flex: '1 1 auto', | ||||
|       paddingRight: 6, | ||||
|       '& input': { | ||||
|         outline: 'none', | ||||
|         border: 'none', | ||||
|         width: '100%', | ||||
|         flex: '1 1 auto', | ||||
|         padding: '5px 10px', | ||||
|         'font-size': '1em', | ||||
|         position: 'relative', | ||||
|         fontFamily: 'monaco, Consolas, "Lucida Console", monospace', | ||||
| 
 | ||||
|         'background-color': map.BACKGROUND_COLOR, | ||||
|         color: map.TEXT_COLOR, | ||||
| 
 | ||||
|         '&::-webkit-input-placeholder': { | ||||
|           color: map.TEXT_PLACEHOLDER_COLOR, | ||||
|         }, | ||||
| 
 | ||||
|         '&::-moz-placeholder': { | ||||
|           color: map.TEXT_PLACEHOLDER_COLOR, | ||||
|         }, | ||||
|         '&::-webkit-search-cancel-button': appearanceNone, | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     closeButton: { | ||||
|       ...appearanceNone, | ||||
|       border: 'none', | ||||
|       outline: 'none', | ||||
|       boxShadow: 'none', | ||||
|       display: 'block', | ||||
|       flex: '0 0 auto', | ||||
|       cursor: 'pointer', | ||||
|       background: 'transparent', | ||||
|       position: 'relative', | ||||
|       fontSize: 'inherit', | ||||
|       '&[data-invisible="1"]': { | ||||
|         visibility: 'hidden !important', | ||||
|       }, | ||||
|       '&::after': { | ||||
|         content: '"\u00d7"', | ||||
|         display: 'block', | ||||
|         padding: 4, | ||||
|         fontSize: '1.2em', | ||||
|         color: map.TEXT_PLACEHOLDER_COLOR, | ||||
|         background: 'transparent', | ||||
|       }, | ||||
|       '&:hover::after': { | ||||
|         color: map.TEXT_COLOR, | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     noApiFound: { | ||||
|       width: '100%', | ||||
|       textAlign: 'center', | ||||
|       color: map.TEXT_COLOR, | ||||
|       padding: '1.4em', | ||||
|       '& a': { | ||||
|         fontSize: 'inherit', | ||||
|         color: map.TEXT_COLOR, | ||||
|         textDecoration: 'underline', | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     searchSelectLabel: { | ||||
|       display: 'inline-block', | ||||
|       padding: 4, | ||||
|       borderLeft: '1px solid currentColor', | ||||
|     }, | ||||
| 
 | ||||
|     queryPreview: { | ||||
|       flex: '1 1 50%', | ||||
|       overflowX: 'hidden', | ||||
|       oveflowY: 'auto', | ||||
|       display: 'flex', | ||||
|       'flex-direction': 'column', | ||||
|       'overflow-y': 'hidden', | ||||
|       '& pre': { | ||||
|         border: 'inherit', | ||||
|         'border-radius': '3px', | ||||
|         'line-height': 'inherit', | ||||
|         color: 'inherit', | ||||
|       }, | ||||
| 
 | ||||
|       'background-color': map.BACKGROUND_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     previewHeader: { | ||||
|       flex: '0 0 30px', | ||||
|       padding: '5px 4px', | ||||
|       'align-items': 'center', | ||||
|       'border-bottom-width': '1px', | ||||
|       'border-bottom-style': 'solid', | ||||
| 
 | ||||
|       'background-color': map.HEADER_BACKGROUND_COLOR, | ||||
|       'border-bottom-color': map.HEADER_BORDER_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     treeItemPin: { | ||||
|       'font-size': '0.7em', | ||||
|       'padding-left': '5px', | ||||
|       cursor: 'pointer', | ||||
|       '&:hover': { | ||||
|         'text-decoration': 'underline', | ||||
|       }, | ||||
| 
 | ||||
|       color: map.PIN_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     treeItemKey: { | ||||
|       color: map.TEXT_PLACEHOLDER_COLOR, | ||||
|     }, | ||||
| 
 | ||||
|     treeWrapper: { | ||||
|       overflowX: 'auto', | ||||
|       overflowY: 'auto', | ||||
|       padding: '0.5em 1em', | ||||
|     }, | ||||
| 
 | ||||
|     tabContent: { | ||||
|       display: 'block', | ||||
|       overflowY: 'auto', | ||||
|       padding: '0.5em 0', | ||||
|       color: map.TAB_CONTENT_COLOR, | ||||
|       '& h2': { | ||||
|         color: map.ULIST_STRONG_COLOR, | ||||
|         padding: '0.5em 1em', | ||||
|         fontWeight: 700, | ||||
|       }, | ||||
|       '& h3': { | ||||
|         color: map.ULIST_STRONG_COLOR, | ||||
|       }, | ||||
|     }, | ||||
| 
 | ||||
|     uList: { | ||||
|       listStyle: 'none', | ||||
|       padding: '0 0 0 1em', | ||||
|       color: map.ULIST_COLOR, | ||||
|       '& > li': { | ||||
|         listStyle: 'none', | ||||
|       }, | ||||
|       '& > li::before': { | ||||
|         content: '"\\2022"', | ||||
|         display: 'inline-block', | ||||
|         paddingRight: '0.5em', | ||||
|         color: map.ULIST_DISC_COLOR, | ||||
|         fontSize: '0.8em', | ||||
|       }, | ||||
| 
 | ||||
|       '& strong': { | ||||
|         color: map.ULIST_STRONG_COLOR, | ||||
|       }, | ||||
|     }, | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| let themeSheet: StyleSheet; | ||||
| 
 | ||||
| const getDefaultThemeStyling = (theme: reduxThemes.Base16Theme) => { | ||||
|   if (themeSheet) { | ||||
|     themeSheet.detach(); | ||||
|   } | ||||
| 
 | ||||
|   themeSheet = jss | ||||
|     .createStyleSheet(getSheetFromColorMap(colorMap(theme))) | ||||
|     .attach(); | ||||
| 
 | ||||
|   return themeSheet.classes; | ||||
| }; | ||||
| 
 | ||||
| export const createStylingFromTheme = createStyling(getDefaultThemeStyling, { | ||||
|   defaultBase16: reduxThemes.nicinabox, | ||||
|   base16Themes: { ...reduxThemes }, | ||||
| }); | ||||
| 
 | ||||
| export function createThemeState<S, A extends Action<unknown>>( | ||||
|   props: RtkQueryMonitorProps<S, A> | ||||
| ): StyleUtils { | ||||
|   const base16Theme = | ||||
|     getBase16Theme(props.theme, { ...reduxThemes }) ?? reduxThemes.nicinabox; | ||||
| 
 | ||||
|   const theme = props.invertTheme ? invertTheme(props.theme) : props.theme; | ||||
|   const styling = createStylingFromTheme(theme); | ||||
| 
 | ||||
|   return { base16Theme, styling, invertTheme: !!props.invertTheme }; | ||||
| } | ||||
| 
 | ||||
| const mockStyling = () => ({ className: '', style: {} }); | ||||
| 
 | ||||
| export const StyleUtilsContext = createContext<StyleUtils>({ | ||||
|   base16Theme: reduxThemes.nicinabox, | ||||
|   invertTheme: false, | ||||
|   styling: mockStyling, | ||||
| }); | ||||
| 
 | ||||
| export function getJsonTreeTheme( | ||||
|   base16Theme: reduxThemes.Base16Theme | ||||
| ): StylingConfig { | ||||
|   return { | ||||
|     extend: base16Theme, | ||||
|     nestedNode: ({ style }, keyPath, nodeType, expanded) => ({ | ||||
|       style: { | ||||
|         ...style, | ||||
|         whiteSpace: expanded ? 'inherit' : 'nowrap', | ||||
|       }, | ||||
|     }), | ||||
|     nestedNodeItemString: ({ style }, keyPath, nodeType, expanded) => ({ | ||||
|       style: { | ||||
|         ...style, | ||||
|         display: expanded ? 'none' : 'inline', | ||||
|       }, | ||||
|     }), | ||||
|   }; | ||||
| } | ||||
							
								
								
									
										106
									
								
								packages/redux-devtools-rtk-query-monitor/src/styles/tree.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								packages/redux-devtools-rtk-query-monitor/src/styles/tree.tsx
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,106 @@ | |||
| import React, { ReactNode } from 'react'; | ||||
| import { StylingFunction } from 'react-base16-styling'; | ||||
| import { isCollection, isIndexed, isKeyed } from 'immutable'; | ||||
| import isIterable from '../utils/isIterable'; | ||||
| 
 | ||||
| const IS_IMMUTABLE_KEY = '@@__IS_IMMUTABLE__@@'; | ||||
| 
 | ||||
| function isImmutable(value: unknown) { | ||||
|   return isKeyed(value) || isIndexed(value) || isCollection(value); | ||||
| } | ||||
| 
 | ||||
| function getShortTypeString(val: unknown, diff: boolean | undefined) { | ||||
|   if (diff && Array.isArray(val)) { | ||||
|     val = val[val.length === 2 ? 1 : 0]; | ||||
|   } | ||||
| 
 | ||||
|   if (isIterable(val) && !isImmutable(val)) { | ||||
|     return '(…)'; | ||||
|   } else if (Array.isArray(val)) { | ||||
|     return val.length > 0 ? '[…]' : '[]'; | ||||
|   } else if (val === null) { | ||||
|     return 'null'; | ||||
|   } else if (val === undefined) { | ||||
|     return 'undef'; | ||||
|   } else if (typeof val === 'object') { | ||||
|     return Object.keys(val as Record<string, unknown>).length > 0 | ||||
|       ? '{…}' | ||||
|       : '{}'; | ||||
|   } else if (typeof val === 'function') { | ||||
|     return 'fn'; | ||||
|   } else if (typeof val === 'string') { | ||||
|     return `"${val.substr(0, 10) + (val.length > 10 ? '…' : '')}"`; | ||||
|   } else if (typeof val === 'symbol') { | ||||
|     return 'symbol'; | ||||
|   } else { | ||||
|     return val; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| function getText( | ||||
|   type: string, | ||||
|   data: any, | ||||
|   previewContent: boolean, | ||||
|   isDiff: boolean | undefined | ||||
| ) { | ||||
|   if (type === 'Object') { | ||||
|     const keys = Object.keys(data); | ||||
|     if (!previewContent) return keys.length ? '{…}' : '{}'; | ||||
| 
 | ||||
|     const str = keys | ||||
|       .slice(0, 3) | ||||
|       .map( | ||||
|         (key) => `${key}: ${getShortTypeString(data[key], isDiff) as string}` | ||||
|       ) | ||||
|       .concat(keys.length > 3 ? ['…'] : []) | ||||
|       .join(', '); | ||||
| 
 | ||||
|     return `{ ${str} }`; | ||||
|   } else if (type === 'Array') { | ||||
|     if (!previewContent) return data.length ? '[…]' : '[]'; | ||||
| 
 | ||||
|     const str = data | ||||
|       .slice(0, 4) | ||||
|       .map((val: any) => getShortTypeString(val, isDiff)) | ||||
|       .concat(data.length > 4 ? ['…'] : []) | ||||
|       .join(', '); | ||||
| 
 | ||||
|     return `[${str as string}]`; | ||||
|   } else { | ||||
|     return type; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export function getItemString( | ||||
|   styling: StylingFunction, | ||||
|   type: string, | ||||
|   data: any, | ||||
|   dataTypeKey: string | symbol | undefined, | ||||
|   previewContent: boolean, | ||||
|   isDiff?: boolean | ||||
| ): ReactNode { | ||||
|   return ( | ||||
|     <span {...styling('treeItemHint')}> | ||||
|       {data[IS_IMMUTABLE_KEY] ? 'Immutable' : ''} | ||||
|       {dataTypeKey && data[dataTypeKey] | ||||
|         ? `${data[dataTypeKey] as string} ` | ||||
|         : ''} | ||||
|       {getText(type, data, previewContent, isDiff)} | ||||
|     </span> | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export function createTreeItemLabelRenderer(styling: StylingFunction) { | ||||
|   return function labelRenderer( | ||||
|     [key]: (string | number)[], | ||||
|     nodeType: string, | ||||
|     expanded: boolean | ||||
|   ): ReactNode { | ||||
|     return ( | ||||
|       <span> | ||||
|         <span {...styling('treeItemKey')}>{key}</span> | ||||
|         {!expanded && ': '} | ||||
|       </span> | ||||
|     ); | ||||
|   }; | ||||
| } | ||||
							
								
								
									
										179
									
								
								packages/redux-devtools-rtk-query-monitor/src/types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										179
									
								
								packages/redux-devtools-rtk-query-monitor/src/types.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,179 @@ | |||
| import type { LiftedAction, LiftedState } from '@redux-devtools/instrument'; | ||||
| import type { createApi, QueryStatus } from '@reduxjs/toolkit/query'; | ||||
| import type { Action, AnyAction, Dispatch } from '@reduxjs/toolkit'; | ||||
| import type { ComponentType } from 'react'; | ||||
| import type { Base16Theme, StylingFunction } from 'react-base16-styling'; | ||||
| import type * as themes from 'redux-devtools-themes'; | ||||
| import type { QueryComparators } from './utils/comparators'; | ||||
| import type { QueryFilters } from './utils/filters'; | ||||
| 
 | ||||
| export enum QueryPreviewTabs { | ||||
|   queryinfo, | ||||
|   apiConfig, | ||||
|   querySubscriptions, | ||||
|   queryTags, | ||||
|   actions, | ||||
| } | ||||
| 
 | ||||
| export interface QueryFormValues { | ||||
|   queryComparator: QueryComparators; | ||||
|   isAscendingQueryComparatorOrder: boolean; | ||||
|   searchValue: string; | ||||
|   isRegexSearch: boolean; | ||||
|   queryFilter: QueryFilters; | ||||
| } | ||||
| export interface RtkQueryMonitorState { | ||||
|   readonly queryForm: { | ||||
|     values: QueryFormValues; | ||||
|   }; | ||||
|   readonly selectedQueryKey: Pick<QueryInfo, 'reducerPath' | 'queryKey'> | null; | ||||
|   readonly selectedPreviewTab: QueryPreviewTabs; | ||||
| } | ||||
| 
 | ||||
| export interface RtkQueryMonitorProps<S, A extends Action<unknown>> | ||||
|   extends LiftedState<S, A, RtkQueryMonitorState> { | ||||
|   dispatch: Dispatch<Action | LiftedAction<S, A, RtkQueryMonitorState>>; | ||||
|   theme: keyof typeof themes | Base16Theme; | ||||
|   invertTheme?: boolean; | ||||
| } | ||||
| 
 | ||||
| export type RtkQueryApiState = ReturnType< | ||||
|   ReturnType<typeof createApi>['reducer'] | ||||
| >; | ||||
| 
 | ||||
| export type RtkQueryState = NonNullable< | ||||
|   RtkQueryApiState['queries'][keyof RtkQueryApiState] | ||||
| >; | ||||
| 
 | ||||
| export type RtkMutationState = NonNullable< | ||||
|   RtkQueryApiState['mutations'][keyof RtkQueryApiState] | ||||
| >; | ||||
| 
 | ||||
| export type RtkQueryApiConfig = RtkQueryApiState['config']; | ||||
| 
 | ||||
| export type RtkQueryProvided = RtkQueryApiState['provided']; | ||||
| 
 | ||||
| export interface ExternalProps<S, A extends Action<unknown>> { | ||||
|   dispatch: Dispatch<Action | LiftedAction<S, A, RtkQueryMonitorState>>; | ||||
|   theme: keyof typeof themes | Base16Theme; | ||||
|   hideMainButtons?: boolean; | ||||
|   invertTheme: boolean; | ||||
| } | ||||
| 
 | ||||
| export interface QueryInfo { | ||||
|   type: 'query'; | ||||
|   state: RtkQueryState; | ||||
|   queryKey: string; | ||||
|   reducerPath: string; | ||||
| } | ||||
| 
 | ||||
| export interface MutationInfo { | ||||
|   type: 'mutation'; | ||||
|   state: RtkMutationState; | ||||
|   queryKey: string; | ||||
|   reducerPath: string; | ||||
| } | ||||
| 
 | ||||
| export type RtkResourceInfo = QueryInfo | MutationInfo; | ||||
| 
 | ||||
| export interface ApiInfo { | ||||
|   reducerPath: string; | ||||
|   apiState: RtkQueryApiState; | ||||
| } | ||||
| 
 | ||||
| export interface SelectOption< | ||||
|   T = string, | ||||
|   VisConfig extends string = 'default' | ||||
| > { | ||||
|   label: string; | ||||
|   value: T; | ||||
|   visible?: Record<VisConfig | 'default', boolean> | boolean; | ||||
| } | ||||
| 
 | ||||
| export interface SelectorsSource<S> { | ||||
|   userState: S | null; | ||||
|   monitorState: RtkQueryMonitorState; | ||||
|   currentStateIndex: number; | ||||
|   actionsById: LiftedState<unknown, AnyAction, unknown>['actionsById']; | ||||
| } | ||||
| 
 | ||||
| export interface StyleUtils { | ||||
|   readonly base16Theme: Base16Theme; | ||||
|   readonly styling: StylingFunction; | ||||
|   readonly invertTheme: boolean; | ||||
| } | ||||
| 
 | ||||
| export type RTKQuerySubscribers = NonNullable< | ||||
|   RtkQueryApiState['subscriptions'][keyof RtkQueryApiState['subscriptions']] | ||||
| >; | ||||
| 
 | ||||
| export interface RtkQueryTag { | ||||
|   type: string; | ||||
|   id?: number | string; | ||||
| } | ||||
| 
 | ||||
| interface Tally { | ||||
|   count: number; | ||||
| } | ||||
| 
 | ||||
| export type QueryTally = { | ||||
|   [key in QueryStatus]?: number; | ||||
| } & | ||||
|   Tally; | ||||
| 
 | ||||
| export interface RtkRequestTiming { | ||||
|   requestId: string; | ||||
|   queryKey: string; | ||||
|   endpointName: string; | ||||
|   startedAt: string; | ||||
|   completedAt: string; | ||||
|   duration: string; | ||||
| } | ||||
| 
 | ||||
| export interface QueryTimings { | ||||
|   readonly oldest: RtkRequestTiming | null; | ||||
|   readonly latest: RtkRequestTiming | null; | ||||
|   readonly slowest: RtkRequestTiming | null; | ||||
|   readonly fastest: RtkRequestTiming | null; | ||||
|   readonly average: string; | ||||
|   readonly median: string; | ||||
| } | ||||
| 
 | ||||
| export interface ApiTimings { | ||||
|   readonly queries: QueryTimings; | ||||
|   readonly mutations: QueryTimings; | ||||
| } | ||||
| 
 | ||||
| export interface ApiStats { | ||||
|   readonly timings: ApiTimings; | ||||
|   readonly tally: Readonly<{ | ||||
|     subscriptions: number; | ||||
|     cachedQueries: QueryTally; | ||||
|     tagTypes: number; | ||||
|     cachedMutations: QueryTally; | ||||
|   }>; | ||||
| } | ||||
| 
 | ||||
| export interface TabOption<S, P, V extends string = 'default'> | ||||
|   extends SelectOption<S, V> { | ||||
|   component: ComponentType<P>; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * It is Omit<RequestStatusFlags, 'status'> & { isFetching: boolean; } | ||||
|  */ | ||||
| export interface RTKStatusFlags { | ||||
|   readonly isUninitialized: boolean; | ||||
|   readonly isFetching: boolean; | ||||
|   readonly isSuccess: boolean; | ||||
|   readonly isError: boolean; | ||||
| } | ||||
| 
 | ||||
| export type RtkRequest = { | ||||
|   status: QueryStatus; | ||||
|   queryKey: string; | ||||
|   requestId: string; | ||||
|   endpointName: string; | ||||
|   startedTimeStamp?: number; | ||||
|   fulfilledTimeStamp?: number; | ||||
| }; | ||||
|  | @ -0,0 +1,96 @@ | |||
| import { QueryStatus } from '@reduxjs/toolkit/query'; | ||||
| import { RtkResourceInfo, SelectOption } from '../types'; | ||||
| 
 | ||||
| export interface Comparator<T> { | ||||
|   (a: T, b: T): number; | ||||
| } | ||||
| 
 | ||||
| export enum QueryComparators { | ||||
|   fulfilledTimeStamp = 'timestamp', | ||||
|   queryKey = 'key', | ||||
|   status = 'status', | ||||
|   endpointName = 'endpointName', | ||||
|   apiReducerPath = 'apiReducerPath', | ||||
| } | ||||
| 
 | ||||
| export const sortQueryOptions: SelectOption<QueryComparators>[] = [ | ||||
|   { label: 'last updated', value: QueryComparators.fulfilledTimeStamp }, | ||||
|   { label: 'query key', value: QueryComparators.queryKey }, | ||||
|   { label: 'status', value: QueryComparators.status }, | ||||
|   { label: 'endpoint', value: QueryComparators.endpointName }, | ||||
|   { label: 'reducerPath', value: QueryComparators.apiReducerPath }, | ||||
| ]; | ||||
| 
 | ||||
| function sortQueryByFulfilled( | ||||
|   thisQueryInfo: RtkResourceInfo, | ||||
|   thatQueryInfo: RtkResourceInfo | ||||
| ): number { | ||||
|   const thisFulfilled = thisQueryInfo.state.fulfilledTimeStamp ?? -1; | ||||
|   const thatFulfilled = thatQueryInfo.state.fulfilledTimeStamp ?? -1; | ||||
| 
 | ||||
|   return thisFulfilled - thatFulfilled; | ||||
| } | ||||
| 
 | ||||
| const mapStatusToFactor = { | ||||
|   [QueryStatus.uninitialized]: 1, | ||||
|   [QueryStatus.pending]: 2, | ||||
|   [QueryStatus.rejected]: 3, | ||||
|   [QueryStatus.fulfilled]: 4, | ||||
| }; | ||||
| 
 | ||||
| function sortQueryByStatus( | ||||
|   thisQueryInfo: RtkResourceInfo, | ||||
|   thatQueryInfo: RtkResourceInfo | ||||
| ): number { | ||||
|   const thisTerm = mapStatusToFactor[thisQueryInfo.state.status] || -1; | ||||
|   const thatTerm = mapStatusToFactor[thatQueryInfo.state.status] || -1; | ||||
| 
 | ||||
|   return thisTerm - thatTerm; | ||||
| } | ||||
| 
 | ||||
| export function compareJSONPrimitive< | ||||
|   T extends string | number | boolean | null | ||||
| >(a: T, b: T): number { | ||||
|   if (a === b) { | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   return a > b ? 1 : -1; | ||||
| } | ||||
| 
 | ||||
| function sortByQueryKey( | ||||
|   thisQueryInfo: RtkResourceInfo, | ||||
|   thatQueryInfo: RtkResourceInfo | ||||
| ): number { | ||||
|   return compareJSONPrimitive(thisQueryInfo.queryKey, thatQueryInfo.queryKey); | ||||
| } | ||||
| 
 | ||||
| function sortQueryByEndpointName( | ||||
|   thisQueryInfo: RtkResourceInfo, | ||||
|   thatQueryInfo: RtkResourceInfo | ||||
| ): number { | ||||
|   const thisEndpointName = thisQueryInfo.state.endpointName ?? ''; | ||||
|   const thatEndpointName = thatQueryInfo.state.endpointName ?? ''; | ||||
| 
 | ||||
|   return compareJSONPrimitive(thisEndpointName, thatEndpointName); | ||||
| } | ||||
| 
 | ||||
| function sortByApiReducerPath( | ||||
|   thisQueryInfo: RtkResourceInfo, | ||||
|   thatQueryInfo: RtkResourceInfo | ||||
| ): number { | ||||
|   return compareJSONPrimitive( | ||||
|     thisQueryInfo.reducerPath, | ||||
|     thatQueryInfo.reducerPath | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export const queryComparators: Readonly< | ||||
|   Record<QueryComparators, Comparator<RtkResourceInfo>> | ||||
| > = { | ||||
|   [QueryComparators.fulfilledTimeStamp]: sortQueryByFulfilled, | ||||
|   [QueryComparators.status]: sortQueryByStatus, | ||||
|   [QueryComparators.endpointName]: sortQueryByEndpointName, | ||||
|   [QueryComparators.queryKey]: sortByQueryKey, | ||||
|   [QueryComparators.apiReducerPath]: sortByApiReducerPath, | ||||
| }; | ||||
|  | @ -0,0 +1,78 @@ | |||
| import { RtkResourceInfo, SelectOption } from '../types'; | ||||
| 
 | ||||
| export interface FilterList<T> { | ||||
|   (regex: RegExp | null, list: T[]): T[]; | ||||
| } | ||||
| 
 | ||||
| export enum QueryFilters { | ||||
|   queryKey = 'query key', | ||||
|   reducerPath = 'reducerPath', | ||||
|   endpointName = 'endpoint', | ||||
|   status = 'status', | ||||
| } | ||||
| 
 | ||||
| function filterByQueryKey( | ||||
|   regex: RegExp | null, | ||||
|   list: RtkResourceInfo[] | ||||
| ): RtkResourceInfo[] { | ||||
|   if (!regex) { | ||||
|     return list; | ||||
|   } | ||||
| 
 | ||||
|   return list.filter((RtkResourceInfo) => regex.test(RtkResourceInfo.queryKey)); | ||||
| } | ||||
| 
 | ||||
| function filterByReducerPath( | ||||
|   regex: RegExp | null, | ||||
|   list: RtkResourceInfo[] | ||||
| ): RtkResourceInfo[] { | ||||
|   if (!regex) { | ||||
|     return list; | ||||
|   } | ||||
| 
 | ||||
|   return list.filter((RtkResourceInfo) => | ||||
|     regex.test(RtkResourceInfo.reducerPath) | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| function filterByEndpointName( | ||||
|   regex: RegExp | null, | ||||
|   list: RtkResourceInfo[] | ||||
| ): RtkResourceInfo[] { | ||||
|   if (!regex) { | ||||
|     return list; | ||||
|   } | ||||
| 
 | ||||
|   return list.filter((RtkResourceInfo) => | ||||
|     regex.test(RtkResourceInfo.state.endpointName ?? 'undefined') | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| function filterByStatus( | ||||
|   regex: RegExp | null, | ||||
|   list: RtkResourceInfo[] | ||||
| ): RtkResourceInfo[] { | ||||
|   if (!regex) { | ||||
|     return list; | ||||
|   } | ||||
| 
 | ||||
|   return list.filter((RtkResourceInfo) => | ||||
|     regex.test(RtkResourceInfo.state.status) | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export const filterQueryOptions: SelectOption<QueryFilters>[] = [ | ||||
|   { label: 'query key', value: QueryFilters.queryKey }, | ||||
|   { label: 'reducerPath', value: QueryFilters.reducerPath }, | ||||
|   { label: 'status', value: QueryFilters.status }, | ||||
|   { label: 'endpoint', value: QueryFilters.endpointName }, | ||||
| ]; | ||||
| 
 | ||||
| export const queryListFilters: Readonly< | ||||
|   Record<QueryFilters, FilterList<RtkResourceInfo>> | ||||
| > = { | ||||
|   [QueryFilters.queryKey]: filterByQueryKey, | ||||
|   [QueryFilters.endpointName]: filterByEndpointName, | ||||
|   [QueryFilters.reducerPath]: filterByReducerPath, | ||||
|   [QueryFilters.status]: filterByStatus, | ||||
| }; | ||||
|  | @ -0,0 +1,31 @@ | |||
| export function formatMs(milliseconds: number): string { | ||||
|   if (!Number.isFinite(milliseconds)) { | ||||
|     return 'NaN'; | ||||
|   } | ||||
| 
 | ||||
|   const absInput = Math.abs(Math.round(milliseconds)); | ||||
|   let millis = (absInput % 1000).toString(); | ||||
| 
 | ||||
|   if (millis.length < 3) { | ||||
|     if (millis.length === 2) { | ||||
|       millis = '0' + millis; | ||||
|     } else { | ||||
|       millis = '00' + millis; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const seconds = Math.floor(absInput / 1_000) % 60; | ||||
|   const minutes = Math.floor(absInput / 60_000); | ||||
| 
 | ||||
|   let output = `${seconds}.${millis}s`; | ||||
| 
 | ||||
|   if (minutes > 0) { | ||||
|     output = `${minutes}m${output}`; | ||||
|   } | ||||
| 
 | ||||
|   if (milliseconds < 0) { | ||||
|     output = `-${output}`; | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
|  | @ -0,0 +1,10 @@ | |||
| export default function isIterable(obj: unknown): boolean { | ||||
|   return ( | ||||
|     obj !== null && | ||||
|     typeof obj === 'object' && | ||||
|     !Array.isArray(obj) && | ||||
|     typeof (obj as Record<string | typeof Symbol.iterator, unknown>)[ | ||||
|       window.Symbol.iterator | ||||
|     ] === 'function' | ||||
|   ); | ||||
| } | ||||
|  | @ -0,0 +1,9 @@ | |||
| import { freeze } from '@reduxjs/toolkit'; | ||||
| 
 | ||||
| export const emptyArray = freeze([]); | ||||
| 
 | ||||
| export const emptyRecord = freeze({}); | ||||
| 
 | ||||
| export function identity<T>(val: T): T { | ||||
|   return val; | ||||
| } | ||||
|  | @ -0,0 +1,4 @@ | |||
| // https://stackoverflow.com/a/9310752
 | ||||
| export function escapeRegExpSpecialCharacter(text: string): string { | ||||
|   return text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, '\\$&'); | ||||
| } | ||||
							
								
								
									
										666
									
								
								packages/redux-devtools-rtk-query-monitor/src/utils/rtk-query.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										666
									
								
								packages/redux-devtools-rtk-query-monitor/src/utils/rtk-query.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,666 @@ | |||
| import { Action, AnyAction, isAllOf, isPlainObject } from '@reduxjs/toolkit'; | ||||
| import { QueryStatus } from '@reduxjs/toolkit/query'; | ||||
| import { | ||||
|   QueryInfo, | ||||
|   RtkQueryMonitorState, | ||||
|   RtkQueryApiState, | ||||
|   RTKQuerySubscribers, | ||||
|   RtkQueryTag, | ||||
|   RTKStatusFlags, | ||||
|   RtkQueryState, | ||||
|   MutationInfo, | ||||
|   ApiStats, | ||||
|   QueryTally, | ||||
|   RtkQueryProvided, | ||||
|   ApiTimings, | ||||
|   QueryTimings, | ||||
|   SelectorsSource, | ||||
|   RtkMutationState, | ||||
|   RtkResourceInfo, | ||||
|   RtkRequest, | ||||
|   RtkRequestTiming, | ||||
| } from '../types'; | ||||
| import { missingTagId } from '../monitor-config'; | ||||
| import { Comparator, compareJSONPrimitive } from './comparators'; | ||||
| import { emptyArray } from './object'; | ||||
| import { formatMs } from './formatters'; | ||||
| import * as statistics from './statistics'; | ||||
| 
 | ||||
| const rtkqueryApiStateKeys: ReadonlyArray<keyof RtkQueryApiState> = [ | ||||
|   'queries', | ||||
|   'mutations', | ||||
|   'config', | ||||
|   'provided', | ||||
|   'subscriptions', | ||||
| ]; | ||||
| 
 | ||||
| /** | ||||
|  * Type guard used to select apis from the user store state. | ||||
|  * @param val | ||||
|  * @returns {boolean} | ||||
|  */ | ||||
| export function isApiSlice(val: unknown): val is RtkQueryApiState { | ||||
|   if (!isPlainObject(val)) { | ||||
|     return false; | ||||
|   } | ||||
| 
 | ||||
|   for (let i = 0, len = rtkqueryApiStateKeys.length; i < len; i++) { | ||||
|     if ( | ||||
|       !isPlainObject((val as Record<string, unknown>)[rtkqueryApiStateKeys[i]]) | ||||
|     ) { | ||||
|       return false; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return true; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Indexes api states by their `reducerPath`. | ||||
|  * | ||||
|  * Returns `null` if there are no api slice or `reduxStoreState` | ||||
|  * is not an object. | ||||
|  * | ||||
|  * @param reduxStoreState | ||||
|  * @returns | ||||
|  */ | ||||
| export function getApiStatesOf( | ||||
|   reduxStoreState: unknown | ||||
| ): null | Readonly<Record<string, RtkQueryApiState>> { | ||||
|   if (!isPlainObject(reduxStoreState)) { | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   const output: null | Record<string, RtkQueryApiState> = {}; | ||||
|   const keys = Object.keys(reduxStoreState); | ||||
| 
 | ||||
|   for (let i = 0, len = keys.length; i < len; i++) { | ||||
|     const key = keys[i]; | ||||
|     const value = (reduxStoreState as Record<string, unknown>)[key]; | ||||
| 
 | ||||
|     if (isApiSlice(value)) { | ||||
|       output[key] = value; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   if (Object.keys(output).length === 0) { | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| export function extractAllApiQueries( | ||||
|   apiStatesByReducerPath: null | Readonly<Record<string, RtkQueryApiState>> | ||||
| ): ReadonlyArray<QueryInfo> { | ||||
|   if (!apiStatesByReducerPath) { | ||||
|     return emptyArray; | ||||
|   } | ||||
| 
 | ||||
|   const reducerPaths = Object.keys(apiStatesByReducerPath); | ||||
| 
 | ||||
|   const output: QueryInfo[] = []; | ||||
| 
 | ||||
|   for (let i = 0, len = reducerPaths.length; i < len; i++) { | ||||
|     const reducerPath = reducerPaths[i]; | ||||
|     const api = apiStatesByReducerPath[reducerPath]; | ||||
|     const queryKeys = Object.keys(api.queries); | ||||
| 
 | ||||
|     for (let j = 0, qKeysLen = queryKeys.length; j < qKeysLen; j++) { | ||||
|       const queryKey = queryKeys[j]; | ||||
|       const state = api.queries[queryKey]; | ||||
| 
 | ||||
|       if (state) { | ||||
|         output.push({ | ||||
|           type: 'query', | ||||
|           reducerPath, | ||||
|           queryKey, | ||||
|           state, | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| export function extractAllApiMutations( | ||||
|   apiStatesByReducerPath: null | Readonly<Record<string, RtkQueryApiState>> | ||||
| ): ReadonlyArray<MutationInfo> { | ||||
|   if (!apiStatesByReducerPath) { | ||||
|     return emptyArray; | ||||
|   } | ||||
| 
 | ||||
|   const reducerPaths = Object.keys(apiStatesByReducerPath); | ||||
|   const output: MutationInfo[] = []; | ||||
| 
 | ||||
|   for (let i = 0, len = reducerPaths.length; i < len; i++) { | ||||
|     const reducerPath = reducerPaths[i]; | ||||
|     const api = apiStatesByReducerPath[reducerPath]; | ||||
|     const mutationKeys = Object.keys(api.mutations); | ||||
| 
 | ||||
|     for (let j = 0, mKeysLen = mutationKeys.length; j < mKeysLen; j++) { | ||||
|       const queryKey = mutationKeys[j]; | ||||
|       const state = api.mutations[queryKey]; | ||||
| 
 | ||||
|       if (state) { | ||||
|         output.push({ | ||||
|           type: 'mutation', | ||||
|           reducerPath, | ||||
|           queryKey, | ||||
|           state, | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| function computeQueryTallyOf( | ||||
|   queryState: RtkQueryApiState['queries'] | RtkQueryApiState['mutations'] | ||||
| ): QueryTally { | ||||
|   const queries = Object.values(queryState); | ||||
| 
 | ||||
|   const output: QueryTally = { | ||||
|     count: 0, | ||||
|   }; | ||||
| 
 | ||||
|   for (let i = 0, len = queries.length; i < len; i++) { | ||||
|     const query = queries[i]; | ||||
| 
 | ||||
|     if (query) { | ||||
|       output.count++; | ||||
| 
 | ||||
|       if (!output[query.status]) { | ||||
|         output[query.status] = 1; | ||||
|       } else { | ||||
|         (output[query.status] as number)++; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| function tallySubscriptions( | ||||
|   subsState: RtkQueryApiState['subscriptions'] | ||||
| ): number { | ||||
|   const subsOfQueries = Object.values(subsState); | ||||
| 
 | ||||
|   let output = 0; | ||||
| 
 | ||||
|   for (let i = 0, len = subsOfQueries.length; i < len; i++) { | ||||
|     const subsOfQuery = subsOfQueries[i]; | ||||
| 
 | ||||
|     if (subsOfQuery) { | ||||
|       output += Object.keys(subsOfQuery).length; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| function computeRtkQueryRequests( | ||||
|   type: 'queries' | 'mutations', | ||||
|   api: RtkQueryApiState, | ||||
|   sortedActions: AnyAction[], | ||||
|   currentStateIndex: SelectorsSource<unknown>['currentStateIndex'] | ||||
| ): Readonly<Record<string, RtkRequest>> { | ||||
|   const requestById: Record<string, RtkRequest> = {}; | ||||
| 
 | ||||
|   const matcher = | ||||
|     type === 'queries' | ||||
|       ? matchesExecuteQuery(api.config.reducerPath) | ||||
|       : matchesExecuteMutation(api.config.reducerPath); | ||||
| 
 | ||||
|   for ( | ||||
|     let i = 0, len = sortedActions.length; | ||||
|     i < len && i <= currentStateIndex; | ||||
|     i++ | ||||
|   ) { | ||||
|     const action = sortedActions[i]; | ||||
| 
 | ||||
|     if (matcher(action)) { | ||||
|       let requestRecord: RtkRequest | undefined = | ||||
|         requestById[action.meta.requestId]; | ||||
| 
 | ||||
|       if (!requestRecord) { | ||||
|         const queryCacheKey: string | undefined = ( | ||||
|           action.meta as Record<string, any> | ||||
|         )?.arg?.queryCacheKey; | ||||
| 
 | ||||
|         const queryKey = | ||||
|           typeof queryCacheKey === 'string' | ||||
|             ? queryCacheKey | ||||
|             : action.meta.requestId; | ||||
| 
 | ||||
|         const endpointName: string = | ||||
|           (action.meta as any)?.arg?.endpointName ?? '-'; | ||||
| 
 | ||||
|         requestById[action.meta.requestId] = requestRecord = { | ||||
|           queryKey, | ||||
|           requestId: action.meta.requestId, | ||||
|           endpointName, | ||||
|           status: action.meta.requestStatus, | ||||
|         }; | ||||
|       } | ||||
| 
 | ||||
|       requestRecord.status = action.meta.requestStatus; | ||||
| 
 | ||||
|       if ( | ||||
|         action.meta.requestStatus === QueryStatus.pending && | ||||
|         typeof (action.meta as any).startedTimeStamp === 'number' | ||||
|       ) { | ||||
|         requestRecord.startedTimeStamp = (action.meta as any).startedTimeStamp; | ||||
|       } | ||||
| 
 | ||||
|       if ( | ||||
|         action.meta.requestStatus === QueryStatus.fulfilled && | ||||
|         typeof (action.meta as any).fulfilledTimeStamp === 'number' | ||||
|       ) { | ||||
|         requestRecord.fulfilledTimeStamp = ( | ||||
|           action.meta as any | ||||
|         ).fulfilledTimeStamp; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const requestIds = Object.keys(requestById); | ||||
| 
 | ||||
|   // Patch queries that have pending actions that are committed
 | ||||
|   for (let i = 0, len = requestIds.length; i < len; i++) { | ||||
|     const requestId = requestIds[i]; | ||||
|     const request = requestById[requestId]; | ||||
| 
 | ||||
|     if ( | ||||
|       typeof request.startedTimeStamp === 'undefined' && | ||||
|       typeof request.fulfilledTimeStamp === 'number' | ||||
|     ) { | ||||
|       const startedTimeStampFromCache = | ||||
|         api[type][request.queryKey]?.startedTimeStamp; | ||||
| 
 | ||||
|       if (typeof startedTimeStampFromCache === 'number') { | ||||
|         request.startedTimeStamp = startedTimeStampFromCache; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   // Add queries that have pending and fulfilled actions committed
 | ||||
|   const queryCacheEntries = Object.entries(api[type] ?? {}); | ||||
| 
 | ||||
|   for (let i = 0, len = queryCacheEntries.length; i < len; i++) { | ||||
|     const [queryCacheKey, queryCache] = queryCacheEntries[i]; | ||||
|     const requestId: string = | ||||
|       type === 'queries' | ||||
|         ? (queryCache as typeof api['queries'][string])?.requestId ?? '' | ||||
|         : queryCacheKey; | ||||
|     if ( | ||||
|       queryCache && | ||||
|       !Object.prototype.hasOwnProperty.call(requestById, requestId) | ||||
|     ) { | ||||
|       const startedTimeStamp = queryCache?.startedTimeStamp; | ||||
|       const fulfilledTimeStamp = queryCache?.fulfilledTimeStamp; | ||||
| 
 | ||||
|       if ( | ||||
|         typeof startedTimeStamp === 'number' && | ||||
|         typeof fulfilledTimeStamp === 'number' | ||||
|       ) { | ||||
|         requestById[requestId] = { | ||||
|           queryKey: queryCacheKey, | ||||
|           requestId, | ||||
|           endpointName: queryCache.endpointName ?? '', | ||||
|           startedTimeStamp, | ||||
|           fulfilledTimeStamp, | ||||
|           status: queryCache.status, | ||||
|         }; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return requestById; | ||||
| } | ||||
| 
 | ||||
| function formatRtkRequest( | ||||
|   rtkRequest: RtkRequest | null | ||||
| ): RtkRequestTiming | null { | ||||
|   if (!rtkRequest) { | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   const fulfilledTimeStamp = rtkRequest.fulfilledTimeStamp; | ||||
|   const startedTimeStamp = rtkRequest.startedTimeStamp; | ||||
| 
 | ||||
|   const output: RtkRequestTiming = { | ||||
|     queryKey: rtkRequest.queryKey, | ||||
|     requestId: rtkRequest.requestId, | ||||
|     endpointName: rtkRequest.endpointName, | ||||
|     startedAt: '-', | ||||
|     completedAt: '-', | ||||
|     duration: '-', | ||||
|   }; | ||||
| 
 | ||||
|   if ( | ||||
|     typeof fulfilledTimeStamp === 'number' && | ||||
|     typeof startedTimeStamp === 'number' | ||||
|   ) { | ||||
|     output.startedAt = new Date(startedTimeStamp).toISOString(); | ||||
|     output.completedAt = new Date(fulfilledTimeStamp).toISOString(); | ||||
|     output.duration = formatMs(fulfilledTimeStamp - startedTimeStamp); | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| function computeQueryApiTimings( | ||||
|   requestById: Readonly<Record<string, RtkRequest>> | ||||
| ): QueryTimings { | ||||
|   const requests = Object.values(requestById); | ||||
| 
 | ||||
|   let latestRequest: RtkRequest | null = null; | ||||
|   let oldestRequest: null | RtkRequest = null; | ||||
|   let slowestRequest: RtkRequest | null = null; | ||||
|   let fastestRequest: RtkRequest | null = null; | ||||
|   let slowestDuration = 0; | ||||
|   let fastestDuration = Number.MAX_SAFE_INTEGER; | ||||
| 
 | ||||
|   const pendingDurations: number[] = []; | ||||
| 
 | ||||
|   for (let i = 0, len = requests.length; i < len; i++) { | ||||
|     const request = requests[i]; | ||||
|     const { fulfilledTimeStamp, startedTimeStamp } = request; | ||||
| 
 | ||||
|     if (typeof fulfilledTimeStamp === 'number') { | ||||
|       const latestFulfilledTimeStamp = latestRequest?.fulfilledTimeStamp || 0; | ||||
|       const oldestFulfilledTimeStamp = | ||||
|         oldestRequest?.fulfilledTimeStamp || Number.MAX_SAFE_INTEGER; | ||||
| 
 | ||||
|       if (fulfilledTimeStamp > latestFulfilledTimeStamp) { | ||||
|         latestRequest = request; | ||||
|       } | ||||
| 
 | ||||
|       if (fulfilledTimeStamp < oldestFulfilledTimeStamp) { | ||||
|         oldestRequest = request; | ||||
|       } | ||||
| 
 | ||||
|       if ( | ||||
|         typeof startedTimeStamp === 'number' && | ||||
|         startedTimeStamp <= fulfilledTimeStamp | ||||
|       ) { | ||||
|         const pendingDuration = fulfilledTimeStamp - startedTimeStamp; | ||||
|         pendingDurations.push(pendingDuration); | ||||
| 
 | ||||
|         if (pendingDuration > slowestDuration) { | ||||
|           slowestDuration = pendingDuration; | ||||
|           slowestRequest = request; | ||||
|         } | ||||
| 
 | ||||
|         if (pendingDuration < fastestDuration) { | ||||
|           fastestDuration = pendingDuration; | ||||
|           fastestRequest = request; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const average = | ||||
|     pendingDurations.length > 0 | ||||
|       ? formatMs(statistics.mean(pendingDurations)) | ||||
|       : '-'; | ||||
| 
 | ||||
|   const median = | ||||
|     pendingDurations.length > 0 | ||||
|       ? formatMs(statistics.median(pendingDurations)) | ||||
|       : '-'; | ||||
| 
 | ||||
|   return { | ||||
|     latest: formatRtkRequest(latestRequest), | ||||
|     oldest: formatRtkRequest(oldestRequest), | ||||
|     slowest: formatRtkRequest(slowestRequest), | ||||
|     fastest: formatRtkRequest(fastestRequest), | ||||
|     average, | ||||
|     median, | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| function computeApiTimings( | ||||
|   api: RtkQueryApiState, | ||||
|   actionsById: SelectorsSource<unknown>['actionsById'], | ||||
|   currentStateIndex: SelectorsSource<unknown>['currentStateIndex'] | ||||
| ): ApiTimings { | ||||
|   const sortedActions = Object.entries(actionsById) | ||||
|     .sort((thisAction, thatAction) => | ||||
|       compareJSONPrimitive(Number(thisAction[0]), Number(thatAction[0])) | ||||
|     ) | ||||
|     .map((entry) => entry[1].action); | ||||
| 
 | ||||
|   const queryRequestsById = computeRtkQueryRequests( | ||||
|     'queries', | ||||
|     api, | ||||
|     sortedActions, | ||||
|     currentStateIndex | ||||
|   ); | ||||
| 
 | ||||
|   const mutationRequestsById = computeRtkQueryRequests( | ||||
|     'mutations', | ||||
|     api, | ||||
|     sortedActions, | ||||
|     currentStateIndex | ||||
|   ); | ||||
| 
 | ||||
|   return { | ||||
|     queries: computeQueryApiTimings(queryRequestsById), | ||||
|     mutations: computeQueryApiTimings(mutationRequestsById), | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| export function generateApiStatsOfCurrentQuery( | ||||
|   api: RtkQueryApiState | null, | ||||
|   actionsById: SelectorsSource<unknown>['actionsById'], | ||||
|   currentStateIndex: SelectorsSource<unknown>['currentStateIndex'] | ||||
| ): ApiStats | null { | ||||
|   if (!api) { | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   return { | ||||
|     timings: computeApiTimings(api, actionsById, currentStateIndex), | ||||
|     tally: { | ||||
|       cachedQueries: computeQueryTallyOf(api.queries), | ||||
|       cachedMutations: computeQueryTallyOf(api.mutations), | ||||
|       tagTypes: Object.keys(api.provided).length, | ||||
|       subscriptions: tallySubscriptions(api.subscriptions), | ||||
|     }, | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| export function flipComparator<T>(comparator: Comparator<T>): Comparator<T> { | ||||
|   return function flipped(a: T, b: T) { | ||||
|     return comparator(b, a); | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| export function isQuerySelected( | ||||
|   selectedQueryKey: RtkQueryMonitorState['selectedQueryKey'], | ||||
|   queryInfo: RtkResourceInfo | ||||
| ): boolean { | ||||
|   return ( | ||||
|     !!selectedQueryKey && | ||||
|     selectedQueryKey.queryKey === queryInfo.queryKey && | ||||
|     selectedQueryKey.reducerPath === queryInfo.reducerPath | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export function getApiStateOf( | ||||
|   queryInfo: RtkResourceInfo | null, | ||||
|   apiStates: ReturnType<typeof getApiStatesOf> | ||||
| ): RtkQueryApiState | null { | ||||
|   if (!apiStates || !queryInfo) { | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   return apiStates[queryInfo.reducerPath] ?? null; | ||||
| } | ||||
| 
 | ||||
| export function getQuerySubscriptionsOf( | ||||
|   queryInfo: QueryInfo | null, | ||||
|   apiStates: ReturnType<typeof getApiStatesOf> | ||||
| ): RTKQuerySubscribers | null { | ||||
|   if (!apiStates || !queryInfo) { | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     apiStates[queryInfo.reducerPath]?.subscriptions?.[queryInfo.queryKey] ?? | ||||
|     null | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export function getProvidedOf( | ||||
|   queryInfo: QueryInfo | null, | ||||
|   apiStates: ReturnType<typeof getApiStatesOf> | ||||
| ): RtkQueryApiState['provided'] | null { | ||||
|   if (!apiStates || !queryInfo) { | ||||
|     return null; | ||||
|   } | ||||
| 
 | ||||
|   return apiStates[queryInfo.reducerPath]?.provided ?? null; | ||||
| } | ||||
| 
 | ||||
| export function getQueryTagsOf( | ||||
|   resInfo: RtkResourceInfo | null, | ||||
|   provided: RtkQueryProvided | null | ||||
| ): RtkQueryTag[] { | ||||
|   if (!resInfo || resInfo.type === 'mutation' || !provided) { | ||||
|     return emptyArray; | ||||
|   } | ||||
| 
 | ||||
|   const tagTypes = Object.keys(provided); | ||||
| 
 | ||||
|   if (tagTypes.length < 1) { | ||||
|     return emptyArray; | ||||
|   } | ||||
| 
 | ||||
|   const output: RtkQueryTag[] = []; | ||||
| 
 | ||||
|   for (const [type, tagIds] of Object.entries(provided)) { | ||||
|     if (tagIds) { | ||||
|       for (const [id, queryKeys] of Object.entries(tagIds)) { | ||||
|         if ((queryKeys as unknown[]).includes(resInfo.queryKey)) { | ||||
|           const tag: RtkQueryTag = { type }; | ||||
| 
 | ||||
|           if (id !== missingTagId) { | ||||
|             tag.id = id; | ||||
|           } | ||||
| 
 | ||||
|           output.push(tag); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Computes query status flags. | ||||
|  * @param status | ||||
|  * @see https://redux-toolkit.js.org/rtk-query/usage/queries#frequently-used-query-hook-return-values
 | ||||
|  * @see https://github.com/reduxjs/redux-toolkit/blob/b718e01d323d3ab4b913e5d88c9b90aa790bb975/src/query/core/apiState.ts#L63
 | ||||
|  */ | ||||
| export function getQueryStatusFlags({ | ||||
|   status, | ||||
|   data, | ||||
| }: RtkQueryState | RtkMutationState): RTKStatusFlags { | ||||
|   return { | ||||
|     isUninitialized: status === QueryStatus.uninitialized, | ||||
|     isFetching: status === QueryStatus.pending, | ||||
|     isSuccess: status === QueryStatus.fulfilled && !!data, | ||||
|     isError: status === QueryStatus.rejected, | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * endpoint matcher | ||||
|  * @param endpointName | ||||
|  * @see https://github.com/reduxjs/redux-toolkit/blob/b718e01d323d3ab4b913e5d88c9b90aa790bb975/src/query/core/buildThunks.ts#L415
 | ||||
|  */ | ||||
| export function matchesEndpoint(endpointName: unknown) { | ||||
|   return (action: any): action is Action => | ||||
|     endpointName != null && action?.meta?.arg?.endpointName === endpointName; | ||||
| } | ||||
| 
 | ||||
| function matchesQueryKey(queryKey: string) { | ||||
|   return (action: any): action is Action => | ||||
|     action?.meta?.arg?.queryCacheKey === queryKey; | ||||
| } | ||||
| 
 | ||||
| function macthesRequestId(requestId: string) { | ||||
|   return (action: any): action is Action => | ||||
|     action?.meta?.requestId === requestId; | ||||
| } | ||||
| 
 | ||||
| function matchesReducerPath(reducerPath: string) { | ||||
|   return (action: any): action is Action<string> => | ||||
|     typeof action?.type === 'string' && action.type.startsWith(reducerPath); | ||||
| } | ||||
| 
 | ||||
| function matchesExecuteQuery(reducerPath: string) { | ||||
|   return ( | ||||
|     action: any | ||||
|   ): action is Action<string> & { | ||||
|     meta: { requestId: string; requestStatus: QueryStatus }; | ||||
|   } => { | ||||
|     return ( | ||||
|       typeof action?.type === 'string' && | ||||
|       action.type.startsWith(`${reducerPath}/executeQuery`) && | ||||
|       typeof action.meta?.requestId === 'string' && | ||||
|       typeof action.meta?.requestStatus === 'string' | ||||
|     ); | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| function matchesExecuteMutation(reducerPath: string) { | ||||
|   return ( | ||||
|     action: any | ||||
|   ): action is Action<string> & { | ||||
|     meta: { requestId: string; requestStatus: QueryStatus }; | ||||
|   } => | ||||
|     typeof action?.type === 'string' && | ||||
|     action.type.startsWith(`${reducerPath}/executeMutation`) && | ||||
|     typeof action.meta?.requestId === 'string' && | ||||
|     typeof action.meta?.requestStatus === 'string'; | ||||
| } | ||||
| 
 | ||||
| export function getActionsOfCurrentQuery( | ||||
|   currentQuery: RtkResourceInfo | null, | ||||
|   actionById: SelectorsSource<unknown>['actionsById'] | ||||
| ): Action[] { | ||||
|   if (!currentQuery) { | ||||
|     return emptyArray; | ||||
|   } | ||||
| 
 | ||||
|   let matcher: ReturnType<typeof macthesRequestId>; | ||||
| 
 | ||||
|   if (currentQuery.type === 'mutation') { | ||||
|     matcher = isAllOf( | ||||
|       matchesReducerPath(currentQuery.reducerPath), | ||||
|       macthesRequestId(currentQuery.queryKey) | ||||
|     ); | ||||
|   } else { | ||||
|     matcher = isAllOf( | ||||
|       matchesReducerPath(currentQuery.reducerPath), | ||||
|       matchesQueryKey(currentQuery.queryKey) | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   const output: AnyAction[] = []; | ||||
| 
 | ||||
|   for (const [, liftedAction] of Object.entries(actionById)) { | ||||
|     if (matcher(liftedAction?.action)) { | ||||
|       output.push(liftedAction.action); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   return output.length === 0 ? emptyArray : output; | ||||
| } | ||||
|  | @ -0,0 +1,68 @@ | |||
| /** | ||||
|  * An implementation of `Kahan-Babuska algorithm` | ||||
|  * that reduces numerical floating point errors. | ||||
|  * @param {number[]} nums | ||||
|  * @returns {number} | ||||
|  * @see https://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.582.288&rep=rep1&type=pdf
 | ||||
|  */ | ||||
| function sum(nums: number[]): number { | ||||
|   if (nums.length === 0) { | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   let t; | ||||
|   let correction = 0; | ||||
|   let output = nums[0]; | ||||
| 
 | ||||
|   for (let i = 1, len = nums.length; i < len; i++) { | ||||
|     t = output + nums[i]; | ||||
| 
 | ||||
|     if (Math.abs(output) >= Math.abs(nums[i])) { | ||||
|       correction += output - t + nums[i]; | ||||
|     } else { | ||||
|       correction += nums[i] - t + output; | ||||
|     } | ||||
| 
 | ||||
|     output = t; | ||||
|   } | ||||
| 
 | ||||
|   return output + correction; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Returns mean, also known as average, of numerical sequences. | ||||
|  * @param nums | ||||
|  * @returns | ||||
|  */ | ||||
| export function mean(nums: number[]): number { | ||||
|   if (nums.length === 0) { | ||||
|     return NaN; | ||||
|   } | ||||
| 
 | ||||
|   return +(sum(nums) / nums.length).toFixed(6); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Returns median value of a numeric sequence. | ||||
|  * @param nums | ||||
|  * @returns | ||||
|  */ | ||||
| export function median(nums: number[]): number { | ||||
|   const len = nums.length; | ||||
| 
 | ||||
|   if (len === 0) { | ||||
|     return NaN; | ||||
|   } | ||||
| 
 | ||||
|   if (len === 1) { | ||||
|     return nums[0]; | ||||
|   } | ||||
| 
 | ||||
|   const sorted = nums.slice().sort(); | ||||
| 
 | ||||
|   if (len % 2 === 1) { | ||||
|     return sorted[(len + 1) / 2 - 1]; | ||||
|   } | ||||
| 
 | ||||
|   return +(0.5 * (sorted[len / 2 - 1] + sorted[len / 2])).toFixed(6); | ||||
| } | ||||
							
								
								
									
										16
									
								
								packages/redux-devtools-rtk-query-monitor/src/utils/tabs.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								packages/redux-devtools-rtk-query-monitor/src/utils/tabs.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,16 @@ | |||
| import { TabOption } from '../types'; | ||||
| 
 | ||||
| export function isTabVisible<St, Props, Vis extends string>( | ||||
|   tab: TabOption<St, Props, Vis>, | ||||
|   visKey: Vis | 'default' | ||||
| ): boolean { | ||||
|   if (typeof tab.visible === 'boolean') { | ||||
|     return tab.visible; | ||||
|   } | ||||
| 
 | ||||
|   if (typeof tab.visible === 'object' && tab.visible) { | ||||
|     return !!tab.visible[visKey]; | ||||
|   } | ||||
| 
 | ||||
|   return true; | ||||
| } | ||||
|  | @ -0,0 +1,9 @@ | |||
| { | ||||
|   "extends": "../../tsconfig.react.base.json", | ||||
|   "compilerOptions": { | ||||
|     "outDir": "./demo/src/build", | ||||
|     "module": "ES2015", | ||||
|     "strict": false | ||||
|   }, | ||||
|   "include": ["src"] | ||||
| } | ||||
							
								
								
									
										8
									
								
								packages/redux-devtools-rtk-query-monitor/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								packages/redux-devtools-rtk-query-monitor/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,8 @@ | |||
| { | ||||
|   "extends": "../../tsconfig.react.base.json", | ||||
|   "compilerOptions": { | ||||
|     "outDir": "lib", | ||||
|     "resolveJsonModule": true | ||||
|   }, | ||||
|   "include": ["src"] | ||||
| } | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user