mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
125 lines
3.9 KiB
TypeScript
125 lines
3.9 KiB
TypeScript
import { action, computed, observable } from "mobx";
|
|
import { autobind } from "../../utils";
|
|
|
|
export class SearchStore {
|
|
@observable searchQuery = ""; // Text in the search input
|
|
@observable occurrences: number[] = []; // Array with line numbers, eg [0, 0, 10, 21, 21, 40...]
|
|
@observable activeOverlayIndex = -1; // Index withing the occurences array. Showing where is activeOverlay currently located
|
|
|
|
/**
|
|
* Sets default activeOverlayIndex
|
|
* @param text An array of any textual data (logs, for example)
|
|
* @param query Search query from input
|
|
*/
|
|
@action
|
|
onSearch(text: string[], query: string) {
|
|
this.searchQuery = query;
|
|
if (!query) {
|
|
this.reset();
|
|
return;
|
|
}
|
|
this.occurrences = this.findOccurencies(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) return;
|
|
this.activeOverlayIndex = this.getNextOverlay(true);
|
|
}
|
|
|
|
/**
|
|
* Does searching within text array, create a list of search keyword occurences.
|
|
* Each keyword "occurency" is saved as index of the the line where keyword founded
|
|
* @param text An array of any textual data (logs, for example)
|
|
* @param query Search query from input
|
|
* @returns {Array} Array of line indexes [0, 0, 14, 17, 17, 17, 20...]
|
|
*/
|
|
findOccurencies(text: string[], query: string) {
|
|
const occurences: number[] = [];
|
|
text.forEach((line, index) => {
|
|
const regex = new RegExp(this.escapeRegex(query), "g");
|
|
const matches = [...line.matchAll(regex)];
|
|
matches.forEach(() => occurences.push(index));
|
|
});
|
|
return occurences;
|
|
}
|
|
|
|
/**
|
|
* Getting next overlay index within the occurences array
|
|
* @param loopOver Allows to jump from last element to first
|
|
* @returns {number} next overlay index
|
|
*/
|
|
getNextOverlay(loopOver = false) {
|
|
const next = this.activeOverlayIndex + 1;
|
|
if (next > this.occurrences.length - 1) {
|
|
return loopOver ? 0 : this.activeOverlayIndex;
|
|
}
|
|
return next;
|
|
}
|
|
|
|
/**
|
|
* Getting previous overlay index within the occurences array of occurences
|
|
* @param loopOver Allows to jump from first element to last one
|
|
* @returns {number} prev overlay index
|
|
*/
|
|
getPrevOverlay(loopOver = false) {
|
|
const prev = this.activeOverlayIndex - 1;
|
|
if (prev < 0) {
|
|
return loopOver ? this.occurrences.length - 1 : this.activeOverlayIndex;
|
|
}
|
|
return prev;
|
|
}
|
|
|
|
@autobind()
|
|
setNextOverlayActive() {
|
|
this.activeOverlayIndex = this.getNextOverlay(true);
|
|
}
|
|
|
|
@autobind()
|
|
setPrevOverlayActive() {
|
|
this.activeOverlayIndex = this.getPrevOverlay(true);
|
|
}
|
|
|
|
/**
|
|
* Gets line index of where active overlay is located
|
|
* @returns {number} 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 occurence Number of the overlay within one line
|
|
*/
|
|
@autobind()
|
|
isActiveOverlay(line: number, occurence: number) {
|
|
const firstLineIndex = this.occurrences.findIndex(item => item === line);
|
|
return firstLineIndex + occurence === this.activeOverlayIndex;
|
|
}
|
|
|
|
/**
|
|
* An utility methods escaping user string to safely pass it into new Regex(variable)
|
|
* @param value Unescaped string
|
|
*/
|
|
escapeRegex(value: string) {
|
|
return value.replace( /[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&" );
|
|
}
|
|
|
|
@action
|
|
reset() {
|
|
this.searchQuery = "";
|
|
this.activeOverlayIndex = -1;
|
|
this.occurrences = [];
|
|
}
|
|
}
|
|
|
|
export const searchStore = new SearchStore; |