mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* mobx-6 migration -- part 1
Signed-off-by: Roman <ixrock@gmail.com>
* mobx-6 migration -- part 2 (npx mobx-undecorate --keepDecorators)
Signed-off-by: Roman <ixrock@gmail.com>
* mobx-6 migration -- part 3 (more fixes)
Signed-off-by: Roman <ixrock@gmail.com>
* unwrap possible observables from IPC-messaging
Signed-off-by: Roman <ixrock@gmail.com>
* mobx-6 migration -- remove @autobind as class-decorator
Signed-off-by: Roman <ixrock@gmail.com>
* mobx-6: replacing @autobind() as method-decorator to @boundMethod
Signed-off-by: Roman <ixrock@gmail.com>
* mobx-6: use toJS()-wrapper since monkey-patching require(mobx).toJS doesn't work
Signed-off-by: Roman <ixrock@gmail.com>
* removed `@observable static`
Signed-off-by: Roman <ixrock@gmail.com>
* use {useDefineForClassFields: true} in tsconfig.json
Signed-off-by: Roman <ixrock@gmail.com>
* remove ExtendedObservableMap
Signed-off-by: Jari Kolehmainen <jari.kolehmainen@gmail.com>
* fix: removed makeObservable(this) from "terminal-tab.tsx"
Signed-off-by: Roman <ixrock@gmail.com>
* storage-helper refactoring
Signed-off-by: Roman <ixrock@gmail.com>
* normalize usages of #observable-value.toJSON() / attempt to catch the wind
Signed-off-by: Roman <ixrock@gmail.com>
* refactoring, more possible branch fixes + lint
Signed-off-by: Roman <ixrock@gmail.com>
* debugging cluster-view error -- part 1
Signed-off-by: Roman <ixrock@gmail.com>
* fix: refreshing cluster-view on ready
Signed-off-by: Roman <ixrock@gmail.com>
* fix: various app-crashes related to KubeObject.spec.* access from "undefined"
fix: config-map-details crash
Signed-off-by: Roman <ixrock@gmail.com>
* fix: namespace-store refactoring / saving selected-namespaces to external json-file
Signed-off-by: Roman <ixrock@gmail.com>
* fix: don't cache mobx.when(() => this.someObservable) cause might not work as expected due later call of makeObservable(this) in constructor
Signed-off-by: Roman <ixrock@gmail.com>
* fix: app-crash on editing k8s resource
Signed-off-by: Roman <ixrock@gmail.com>
* fix: restore "all namespaces" on page reload
Signed-off-by: Roman <ixrock@gmail.com>
* - fix: persist table-sort params and cluster-view's sidebar state to lens-local-storage
- new-feature: auto-open main-window's devtools in development-mode (yes/no/ugly?)
Signed-off-by: Roman <ixrock@gmail.com>
* fix: crd definition details -> crashing with <AceEditor mode="json"> (added missing mode-file in ace-editor.tsx)
Signed-off-by: Roman <ixrock@gmail.com>
* fix: crd definitions -> groups selector couldn't deselect last selected option
Signed-off-by: Roman <ixrock@gmail.com>
* refactoring: extensions-api exports clarification for "@k8slens/extensions"
Signed-off-by: Roman <ixrock@gmail.com>
* fix: various app-crashes related to kube-events (events page, some details page, overview, etc.)
Signed-off-by: Roman <ixrock@gmail.com>
* Reverted "use {useDefineForClassFields: true} in tsconfig.json" (various app-crash fixes)
This flag seems to be not possible to use with class-inheritance in some cases.
Example / demo:
`KubeObject` class has initial type definitions for the fields like: "metadata", "kind", etc.
and constructor() has Object.assign(this, data);
Meanwhile child class, e.g. KubeEvent inherited from KubeObject and has it's own extra type definitions for underlying resource, e.g. "involvedObject", "source", etc.
So calling super(data) doesn't work as expected for child class as it's own type definitions overwrites data from parent's constructor with `undefined` at later point.
Signed-off-by: Roman <ixrock@gmail.com>
* master-merge lint-fixes
Signed-off-by: Roman <ixrock@gmail.com>
* catalog.tsx / catalog-entities.store.ts refactoring & fixes
Signed-off-by: Roman <ixrock@gmail.com>
* fix: Catalog -> Browse all tab
Signed-off-by: Roman <ixrock@gmail.com>
* fix: CommandPalette doesn't appear from global menu by click/hotkey
Signed-off-by: Roman <ixrock@gmail.com>
* - Merging interfaces & classses to avoid overwriting fields from parent's super(data)-call with Object.assign(this, data). Otherwise use "declare" keyword at class field definition.
- Revamping {useDefineForClassFields: true} to avoid issues with non-observable class fields in some cases (from previous commit):
```
@observer
export class CommandContainer extends React.Component<CommandContainerProps> {
// without some defined initial value "commandComponent" is non-observable for some reasons
// when tsconfig.ts has {useDefineForClassFields:false}
@observable.ref commandComponent: React.ReactNode = null;
constructor(props: CommandContainerProps) {
super(props);
makeObservable(this);
}
```
Signed-off-by: Roman <ixrock@gmail.com>
* update KubeObject class type definition
Signed-off-by: Roman <ixrock@gmail.com>
* clean up / responding to comments
Signed-off-by: Roman <ixrock@gmail.com>
* fix: app-crash when navigating to catalog from active cluster-view, refactoring `catalog-entity-store`
Signed-off-by: Roman <ixrock@gmail.com>
* catalog-pusher clean up, replaced .observe_() to external observe() helper from "mobx"
Signed-off-by: Roman <ixrock@gmail.com>
* fix: catalog's items stale/non-observable (after connection to the cluster status still "disconnected"), lint-fixes
Signed-off-by: Roman <ixrock@gmail.com>
* fix: Catalog is empty after closing main-window and re-opening app from Tray
Signed-off-by: Roman <ixrock@gmail.com>
* fix: HotBar's icon context menu items non-observable (no "disconnect cluster", etc.)
Signed-off-by: Roman <ixrock@gmail.com>
* lint-fix/license check
Signed-off-by: Roman <ixrock@gmail.com>
* fix: redirect to catalog when disconnecting active cluster
Signed-off-by: Roman <ixrock@gmail.com>
* fix: refresh visibility of active cluster-view on switching from hotbar/catalog
Signed-off-by: Roman <ixrock@gmail.com>
* updated package.json for built-in extensions to use "*" version for packages served from main app
Signed-off-by: Roman <ixrock@gmail.com>
* - added missing makeObservable(this) to metrics-settings.tsx
- updated package-lock.json for built-in extensions
- lint fixes
Signed-off-by: Roman <ixrock@gmail.com>
* master-merge clean up fix, updated package-lock.json for built-in extensions after `make clean-extensions && make build-extensions`
Signed-off-by: Roman <ixrock@gmail.com>
* fix unit-tests
Signed-off-by: Roman <ixrock@gmail.com>
* master-merge fixes
Signed-off-by: Roman <ixrock@gmail.com>
* make lint happy
Signed-off-by: Roman <ixrock@gmail.com>
* reverted some changes, removed auto-opening devtools in dev-mode
Signed-off-by: Roman <ixrock@gmail.com>
* merge fixes
Signed-off-by: Roman <ixrock@gmail.com>
* master-merge conflict fixes:
- proper handling and navigating into catalog's active category via URL-builder
Signed-off-by: Roman <ixrock@gmail.com>
* reverting splitted params for catalog's page route to "/catalog/:group?/:kind?"
Signed-off-by: Roman <ixrock@gmail.com>
* clean-up: remove app's injecting dependencies from `extensions/kube-object-event-status/package.json`
Signed-off-by: Roman <ixrock@gmail.com>
* master-merge fix: added missing makeObservable(this) for extensions.tsx
Signed-off-by: Roman <ixrock@gmail.com>
* fix: catalog entity context menu stale/unobservable
Signed-off-by: Roman <ixrock@gmail.com>
Co-authored-by: Jari Kolehmainen <jari.kolehmainen@gmail.com>
179 lines
5.4 KiB
TypeScript
179 lines
5.4 KiB
TypeScript
/**
|
|
* Copyright (c) 2021 OpenLens Authors
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
import { action, computed, observable, reaction, makeObservable } from "mobx";
|
|
import { dockStore } from "../renderer/components/dock/dock.store";
|
|
import { boundMethod } from "../renderer/utils";
|
|
|
|
export class SearchStore {
|
|
/**
|
|
* An utility methods escaping user string to safely pass it into new Regex(variable)
|
|
* @param value Unescaped string
|
|
*/
|
|
public static escapeRegex(value?: string): string {
|
|
return value ? value.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&") : "";
|
|
}
|
|
|
|
/**
|
|
* Text in the search input
|
|
*
|
|
* @observable
|
|
*/
|
|
@observable searchQuery = "";
|
|
|
|
/**
|
|
* Array with line numbers, eg [0, 0, 10, 21, 21, 40...]
|
|
*
|
|
* @observable
|
|
*/
|
|
@observable occurrences: number[] = [];
|
|
|
|
/**
|
|
* Index within the occurrences array. Showing where is activeOverlay currently located
|
|
*
|
|
* @observable
|
|
*/
|
|
@observable activeOverlayIndex = -1;
|
|
|
|
constructor() {
|
|
makeObservable(this);
|
|
reaction(() => dockStore.selectedTabId, () => {
|
|
searchStore.reset();
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Sets default activeOverlayIndex
|
|
* @param text An array of any textual data (logs, for example)
|
|
* @param query Search query from input
|
|
*/
|
|
@action
|
|
public onSearch(text?: string[] | null, query = this.searchQuery): void {
|
|
this.searchQuery = query;
|
|
|
|
if (!query) {
|
|
return this.reset();
|
|
}
|
|
|
|
this.occurrences = this.findOccurrences(text ?? [], query);
|
|
|
|
if (!this.occurrences.length) {
|
|
return;
|
|
}
|
|
|
|
// If new highlighted keyword in exact same place as previous one, then no changing in active overlay
|
|
if (this.occurrences[this.activeOverlayIndex] === undefined) {
|
|
this.activeOverlayIndex = this.getNextOverlay(true);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Does searching within text array, create a list of search keyword occurrences.
|
|
* Each keyword "occurrence" is saved as index of the line where keyword was found
|
|
* @param lines An array of any textual data (logs, for example)
|
|
* @param query Search query from input
|
|
* @returns Array of line indexes [0, 0, 14, 17, 17, 17, 20...]
|
|
*/
|
|
private findOccurrences(lines: string[], query?: string): number[] {
|
|
const regex = new RegExp(SearchStore.escapeRegex(query), "gi");
|
|
|
|
return lines
|
|
.flatMap((line, index) => Array.from(line.matchAll(regex), () => index));
|
|
}
|
|
|
|
/**
|
|
* Getting next overlay index within the occurrences array
|
|
* @param loopOver Allows to jump from last element to first
|
|
* @returns next overlay index
|
|
*/
|
|
private getNextOverlay(loopOver = false): number {
|
|
const next = this.activeOverlayIndex + 1;
|
|
|
|
if (next > this.occurrences.length - 1) {
|
|
return loopOver ? 0 : this.activeOverlayIndex;
|
|
}
|
|
|
|
return next;
|
|
}
|
|
|
|
/**
|
|
* Getting previous overlay index within the occurrences array of occurrences
|
|
* @param loopOver Allows to jump from first element to last one
|
|
* @returns previous overlay index
|
|
*/
|
|
private getPrevOverlay(loopOver = false): number {
|
|
const prev = this.activeOverlayIndex - 1;
|
|
|
|
if (prev < 0) {
|
|
return loopOver ? this.occurrences.length - 1 : this.activeOverlayIndex;
|
|
}
|
|
|
|
return prev;
|
|
}
|
|
|
|
@boundMethod
|
|
public setNextOverlayActive(): void {
|
|
this.activeOverlayIndex = this.getNextOverlay(true);
|
|
}
|
|
|
|
@boundMethod
|
|
public setPrevOverlayActive(): void {
|
|
this.activeOverlayIndex = this.getPrevOverlay(true);
|
|
}
|
|
|
|
/**
|
|
* Gets line index of where active overlay is located
|
|
* @returns A line index within the text/logs array
|
|
*/
|
|
@computed get activeOverlayLine(): number {
|
|
return this.occurrences[this.activeOverlayIndex];
|
|
}
|
|
|
|
@computed get activeFind(): number {
|
|
return this.activeOverlayIndex + 1;
|
|
}
|
|
|
|
@computed get totalFinds(): number {
|
|
return this.occurrences.length;
|
|
}
|
|
|
|
/**
|
|
* Checks if overlay is active (to highlight it with orange background usually)
|
|
* @param line Index of the line where overlay is located
|
|
* @param occurrence Number of the overlay within one line
|
|
*/
|
|
@boundMethod
|
|
public isActiveOverlay(line: number, occurrence: number): boolean {
|
|
const firstLineIndex = this.occurrences.findIndex(item => item === line);
|
|
|
|
return firstLineIndex + occurrence === this.activeOverlayIndex;
|
|
}
|
|
|
|
@action
|
|
private reset(): void {
|
|
this.searchQuery = "";
|
|
this.activeOverlayIndex = -1;
|
|
this.occurrences = [];
|
|
}
|
|
}
|
|
|
|
export const searchStore = new SearchStore;
|