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>
312 lines
9.4 KiB
TypeScript
312 lines
9.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 "./resizing-anchor.scss";
|
|
import React from "react";
|
|
import { action, observable, makeObservable } from "mobx";
|
|
import _ from "lodash";
|
|
import { cssNames, noop } from "../../utils";
|
|
|
|
export enum ResizeDirection {
|
|
HORIZONTAL = "horizontal",
|
|
VERTICAL = "vertical",
|
|
}
|
|
|
|
/**
|
|
* ResizeSide is for customizing where the area should be rendered.
|
|
* That location is determined in conjunction with the `ResizeDirection` using the following table:
|
|
*
|
|
* +----------+------------+----------+
|
|
* | | HORIZONTAL | VERTICAL |
|
|
* +----------+------------+----------+
|
|
* | LEADING | left | top |
|
|
* +----------+------------+----------+
|
|
* | TRAILING | right | bottom |
|
|
* +----------+------------+----------+
|
|
*/
|
|
export enum ResizeSide {
|
|
LEADING = "leading",
|
|
TRAILING = "trailing",
|
|
}
|
|
|
|
/**
|
|
* ResizeGrowthDirection determines how the anchor interprets the drag.
|
|
*
|
|
* Because the origin of the screen is top left a drag from bottom to top
|
|
* results in a negative directional delta. However, if the component being
|
|
* dragged grows in the opposite direction, this needs to be compensated for.
|
|
*/
|
|
export enum ResizeGrowthDirection {
|
|
TOP_TO_BOTTOM = 1,
|
|
BOTTOM_TO_TOP = -1,
|
|
LEFT_TO_RIGHT = 1,
|
|
RIGHT_TO_LEFT = -1,
|
|
}
|
|
|
|
interface Props {
|
|
direction: ResizeDirection;
|
|
|
|
/**
|
|
* getCurrentExtent should return the current prominent dimension in the
|
|
* given resizing direction. Width for HORIZONTAL and height for VERTICAL
|
|
*/
|
|
getCurrentExtent: () => number;
|
|
|
|
disabled?: boolean;
|
|
placement?: ResizeSide;
|
|
growthDirection?: ResizeGrowthDirection;
|
|
|
|
// Ability to restrict which mouse buttons are allowed to resize this component
|
|
// Reference: https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons
|
|
onlyButtons?: number;
|
|
|
|
// onStart is called when the ResizeAnchor is first clicked (mouse down)
|
|
onStart?: () => void;
|
|
|
|
// onEnd is called when the ResizeAnchor is released (mouse up)
|
|
onEnd?: () => void;
|
|
|
|
/**
|
|
* onDrag is called whenever there is a mousemove event. All calls will be
|
|
* bounded by matching `onStart` and `onEnd` calls.
|
|
*/
|
|
onDrag?: (newExtent: number) => void;
|
|
|
|
// onDoubleClick is called when the the ResizeAnchor is double clicked
|
|
onDoubleClick?: () => void;
|
|
|
|
/**
|
|
* The following two extents represent the max and min values set to `onDrag`
|
|
*/
|
|
maxExtent?: number;
|
|
minExtent?: number;
|
|
|
|
/**
|
|
* The following events are triggerred with respect to the above values.
|
|
* - The "__Exceed" call will be made when the unbounded extent goes from
|
|
* < the above to >= the above
|
|
* - The "__Subceed" call is similar but is triggered when the unbounded
|
|
* extent goes from >= the above to < the above.
|
|
*/
|
|
onMaxExtentExceed?: () => void;
|
|
onMaxExtentSubceed?: () => void;
|
|
onMinExtentSubceed?: () => void;
|
|
onMinExtentExceed?: () => void;
|
|
}
|
|
|
|
interface Position {
|
|
readonly pageX: number;
|
|
readonly pageY: number;
|
|
}
|
|
|
|
/**
|
|
* Return the direction delta, but ignore drags leading up to a moved item
|
|
* 1. `->|` => return `false`
|
|
* 2. `<-|` => return `directed length (M, P2)` (negative)
|
|
* 3. `-|>` => return `directed length (M, P2)` (positive)
|
|
* 4. `<|-` => return `directed length (M, P2)` (negative)
|
|
* 5. `|->` => return `directed length (M, P2)` (positive)
|
|
* 6. `|<-` => return `false`
|
|
* @param P1 the starting position on the number line
|
|
* @param P2 the ending position on the number line
|
|
* @param M a third point that determines if the delta is meaningful
|
|
* @returns the directional difference between including appropriate sign.
|
|
*/
|
|
function directionDelta(P1: number, P2: number, M: number): number | false {
|
|
const delta = Math.abs(M - P2);
|
|
|
|
if (P1 < M) {
|
|
if (P2 >= M) {
|
|
// case 3
|
|
return delta;
|
|
}
|
|
|
|
if (P2 < P1) {
|
|
// case 2
|
|
return -delta;
|
|
}
|
|
|
|
// case 1
|
|
return false;
|
|
}
|
|
|
|
if (P2 < M) {
|
|
// case 4
|
|
return -delta;
|
|
}
|
|
|
|
if (P1 < P2) {
|
|
// case 5
|
|
return delta;
|
|
}
|
|
|
|
// case 6
|
|
return false;
|
|
}
|
|
|
|
export class ResizingAnchor extends React.PureComponent<Props> {
|
|
@observable lastMouseEvent?: MouseEvent;
|
|
@observable.ref ref?: React.RefObject<HTMLDivElement>;
|
|
|
|
static defaultProps = {
|
|
onStart: noop,
|
|
onDrag: noop,
|
|
onEnd: noop,
|
|
onMaxExtentExceed: noop,
|
|
onMinExtentExceed: noop,
|
|
onMinExtentSubceed: noop,
|
|
onMaxExtentSubceed: noop,
|
|
onDoubleClick: noop,
|
|
disabled: false,
|
|
growthDirection: ResizeGrowthDirection.BOTTOM_TO_TOP,
|
|
maxExtent: Number.POSITIVE_INFINITY,
|
|
minExtent: 0,
|
|
placement: ResizeSide.LEADING,
|
|
};
|
|
static IS_RESIZING = "resizing";
|
|
|
|
constructor(props: Props) {
|
|
super(props);
|
|
|
|
makeObservable(this);
|
|
|
|
if (props.maxExtent < props.minExtent) {
|
|
throw new Error("maxExtent must be >= minExtent");
|
|
}
|
|
|
|
this.ref = React.createRef<HTMLDivElement>();
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
document.removeEventListener("mousemove", this.onDrag);
|
|
document.removeEventListener("mouseup", this.onDragEnd);
|
|
}
|
|
|
|
@action
|
|
onDragInit = (event: React.MouseEvent) => {
|
|
const { onStart, onlyButtons } = this.props;
|
|
|
|
if (typeof onlyButtons === "number" && onlyButtons !== event.buttons) {
|
|
return;
|
|
}
|
|
|
|
document.addEventListener("mousemove", this.onDrag);
|
|
document.addEventListener("mouseup", this.onDragEnd);
|
|
document.body.classList.add(ResizingAnchor.IS_RESIZING);
|
|
|
|
this.lastMouseEvent = undefined;
|
|
onStart();
|
|
};
|
|
|
|
calculateDelta(from: Position, to: Position): number | false {
|
|
const node = this.ref.current;
|
|
|
|
if (!node) {
|
|
return false;
|
|
}
|
|
|
|
const boundingBox = node.getBoundingClientRect();
|
|
|
|
if (this.props.direction === ResizeDirection.HORIZONTAL) {
|
|
const barX = Math.round(boundingBox.x + (boundingBox.width / 2));
|
|
|
|
return directionDelta(from.pageX, to.pageX, barX);
|
|
} else { // direction === ResizeDirection.VERTICAL
|
|
const barY = Math.round(boundingBox.y + (boundingBox.height / 2));
|
|
|
|
return directionDelta(from.pageY, to.pageY, barY);
|
|
}
|
|
}
|
|
|
|
onDrag = _.throttle((event: MouseEvent) => {
|
|
/**
|
|
* Some notes to help understand the following:
|
|
* - A browser's origin point is in the top left of the screen
|
|
* - X increases going from left to right
|
|
* - Y increases going from top to bottom
|
|
* - Since the resize bar should always be a rectangle, use its centre
|
|
* line (in the resizing direction) as the line for determining if
|
|
* the bar has "jumped around"
|
|
*
|
|
* Desire:
|
|
* - Always ignore movement in the non-resizing direction
|
|
* - Figure out how much the user has "dragged" the resize bar
|
|
* - If the resize bar has jumped around, compensate by ignoring movement
|
|
* in the resizing direction if it is moving "towards" the resize bar's
|
|
* new location.
|
|
*/
|
|
|
|
if (!this.lastMouseEvent) {
|
|
this.lastMouseEvent = event;
|
|
|
|
return;
|
|
}
|
|
|
|
const { maxExtent, minExtent, getCurrentExtent, growthDirection } = this.props;
|
|
const { onDrag, onMaxExtentExceed, onMinExtentSubceed, onMaxExtentSubceed, onMinExtentExceed } = this.props;
|
|
const delta = this.calculateDelta(this.lastMouseEvent, event);
|
|
|
|
// always update the last mouse event
|
|
this.lastMouseEvent = event;
|
|
|
|
if (delta === false) {
|
|
return;
|
|
}
|
|
|
|
const previousExtent = getCurrentExtent();
|
|
const unboundedExtent = previousExtent + (delta * growthDirection);
|
|
const boundedExtent = Math.round(Math.max(minExtent, Math.min(maxExtent, unboundedExtent)));
|
|
|
|
onDrag(boundedExtent);
|
|
|
|
if (previousExtent <= minExtent && minExtent <= unboundedExtent) {
|
|
onMinExtentExceed();
|
|
} else if (previousExtent >= minExtent && minExtent >= unboundedExtent) {
|
|
onMinExtentSubceed();
|
|
}
|
|
|
|
if (previousExtent <= maxExtent && maxExtent <= unboundedExtent) {
|
|
onMaxExtentExceed();
|
|
} else if (previousExtent >= maxExtent && maxExtent >= unboundedExtent) {
|
|
onMaxExtentSubceed();
|
|
}
|
|
}, 100);
|
|
|
|
@action
|
|
onDragEnd = () => {
|
|
this.props.onEnd();
|
|
document.removeEventListener("mousemove", this.onDrag);
|
|
document.removeEventListener("mouseup", this.onDragEnd);
|
|
document.body.classList.remove(ResizingAnchor.IS_RESIZING);
|
|
};
|
|
|
|
render() {
|
|
const { disabled, direction, placement, onDoubleClick } = this.props;
|
|
|
|
return <div
|
|
ref={this.ref}
|
|
className={cssNames("ResizingAnchor", direction, placement, { disabled })}
|
|
onMouseDown={this.onDragInit}
|
|
onDoubleClick={onDoubleClick}
|
|
/>;
|
|
}
|
|
}
|