1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00
lens/src/renderer/components/input/drop-file-input.tsx
Sebastian Malton ade0775061 Remove all usages of legacy global logger
Signed-off-by: Sebastian Malton <sebastian@malton.name>
2023-01-05 13:27:53 -05:00

116 lines
3.3 KiB
TypeScript

/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import "./drop-file-input.scss";
import React from "react";
import type { IClassName } from "../../utils";
import { autoBind, cssNames } from "../../utils";
import { observable, makeObservable } from "mobx";
import { observer } from "mobx-react";
import type { Logger } from "../../../common/logger";
import { withInjectables } from "@ogre-tools/injectable-react";
import loggerInjectable from "../../../common/logger.injectable";
export interface DropFileInputProps<T extends HTMLElement> extends React.DOMAttributes<T> {
className?: IClassName;
disabled?: boolean;
onDropFiles(files: File[], meta: DropFileMeta<T>): void;
}
export interface DropFileMeta<T extends HTMLElement> {
evt: React.DragEvent<T>;
}
interface Dependencies {
logger: Logger;
}
@observer
class NonInjectedDropFileInput<T extends HTMLElement> extends React.Component<DropFileInputProps<T> & Dependencies> {
@observable dropAreaActive = false;
dragCounter = 0; // Counter preventing firing onDragLeave() too early (https://stackoverflow.com/questions/7110353/html5-dragleave-fired-when-hovering-a-child-element)
constructor(props: DropFileInputProps<T> & Dependencies) {
super(props);
makeObservable(this);
autoBind(this);
}
onDragEnter() {
this.dragCounter++;
this.dropAreaActive = true;
}
onDragLeave() {
this.dragCounter--;
if (this.dragCounter == 0) {
this.dropAreaActive = false;
}
}
onDragOver(evt: React.DragEvent<T>) {
if (this.props.onDragOver) {
this.props.onDragOver(evt);
}
evt.preventDefault(); // enable onDrop()-callback
evt.dataTransfer.dropEffect = "move";
}
onDrop(evt: React.DragEvent<T>) {
if (this.props.onDrop) {
this.props.onDrop(evt);
}
this.dropAreaActive = false;
const files = Array.from(evt.dataTransfer.files);
if (files.length > 0) {
this.props.onDropFiles(files, { evt });
}
}
render() {
const { onDragEnter, onDragLeave, onDragOver, onDrop } = this;
const { disabled, className } = this.props;
try {
const contentElem = React.Children.only(this.props.children) as React.ReactElement<React.HTMLProps<HTMLElement>>;
if (disabled) {
return contentElem;
}
if (React.isValidElement(contentElem)) {
const contentElemProps: React.HTMLProps<HTMLElement> = {
className: cssNames("DropFileInput", contentElem.props.className, className, {
droppable: this.dropAreaActive,
}),
onDragEnter,
onDragLeave,
onDragOver,
onDrop,
};
return React.cloneElement(contentElem, contentElemProps);
}
return null;
} catch (err) {
this.props.logger.error(`Error: <DropFileInput/> must contain only single child element`);
return this.props.children;
}
}
}
const InjectedDropFileInput = withInjectables<Dependencies, DropFileInputProps<HTMLElement>>(NonInjectedDropFileInput, {
getProps: (di, props) => ({
...props,
logger: di.inject(loggerInjectable),
}),
});
export const DropFileInput = <T extends HTMLElement>(props: DropFileInputProps<T>) => <InjectedDropFileInput {...props} />;