mirror of
https://github.com/maputnik/editor.git
synced 2025-12-07 06:40:00 +00:00
## Launch Checklist This PR adds back the error panel which was under the map for some reason. It also highlights problematic layers in the layers list (which already worked). It also highlights the field that has an error related to it. It fixes the error types throughout the code. Before: <img width="1141" height="665" alt="image" src="https://github.com/user-attachments/assets/c0593d6c-8f14-41b3-8a51-bc359446656d" /> After: <img width="1141" height="665" alt="image" src="https://github.com/user-attachments/assets/1ffeebb7-31ea-4ed5-97f4-fc5f907a6aea" /> - [x] Briefly describe the changes in this PR. - [x] Include before/after visuals or gifs if this PR includes visual changes. - [x] Write tests for all new functionality. - [x] Add an entry to `CHANGELOG.md` under the `## main` section. --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
105 lines
2.9 KiB
TypeScript
105 lines
2.9 KiB
TypeScript
import React, {type CSSProperties, type PropsWithChildren, type SyntheticEvent} from "react";
|
|
import classnames from "classnames";
|
|
import FieldDocLabel from "./FieldDocLabel";
|
|
import Doc from "./Doc";
|
|
|
|
export type BlockProps = PropsWithChildren & {
|
|
"data-wd-key"?: string
|
|
label?: string
|
|
action?: React.ReactElement
|
|
style?: CSSProperties
|
|
onChange?(...args: unknown[]): unknown
|
|
fieldSpec?: object
|
|
wideMode?: boolean
|
|
error?: {message: string}
|
|
};
|
|
|
|
type BlockState = {
|
|
showDoc: boolean
|
|
};
|
|
|
|
/** Wrap a component with a label */
|
|
export default class Block extends React.Component<BlockProps, BlockState> {
|
|
_blockEl: HTMLDivElement | null = null;
|
|
|
|
constructor (props: BlockProps) {
|
|
super(props);
|
|
this.state = {
|
|
showDoc: false,
|
|
};
|
|
}
|
|
|
|
onChange(e: React.BaseSyntheticEvent<Event, HTMLInputElement, HTMLInputElement>) {
|
|
const value = e.target.value;
|
|
if (this.props.onChange) {
|
|
return this.props.onChange(value === "" ? undefined : value);
|
|
}
|
|
}
|
|
|
|
onToggleDoc = (val: boolean) => {
|
|
this.setState({
|
|
showDoc: val
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Some fields for example <InputColor/> bind click events inside the element
|
|
* to close the picker. This in turn propagates to the <label/> element
|
|
* causing the picker to reopen. This causes a scenario where the picker can
|
|
* never be closed once open.
|
|
*/
|
|
onLabelClick = (event: SyntheticEvent<any, any>) => {
|
|
const el = event.nativeEvent.target;
|
|
const contains = this._blockEl?.contains(el);
|
|
|
|
if (event.nativeEvent.target.nodeName !== "INPUT" && !contains) {
|
|
event.stopPropagation();
|
|
}
|
|
if (event.nativeEvent.target.nodeName !== "A") {
|
|
event.preventDefault();
|
|
}
|
|
};
|
|
|
|
render() {
|
|
return <label style={this.props.style}
|
|
data-wd-key={this.props["data-wd-key"]}
|
|
className={classnames({
|
|
"maputnik-input-block": true,
|
|
"maputnik-input-block--wide": this.props.wideMode,
|
|
"maputnik-action-block": this.props.action,
|
|
"maputnik-input-block--error": this.props.error
|
|
})}
|
|
onClick={this.onLabelClick}
|
|
>
|
|
{this.props.fieldSpec &&
|
|
<div className="maputnik-input-block-label">
|
|
<FieldDocLabel
|
|
label={this.props.label}
|
|
onToggleDoc={this.onToggleDoc}
|
|
fieldSpec={this.props.fieldSpec}
|
|
/>
|
|
</div>
|
|
}
|
|
{!this.props.fieldSpec &&
|
|
<div className="maputnik-input-block-label">
|
|
{this.props.label}
|
|
</div>
|
|
}
|
|
<div className="maputnik-input-block-action">
|
|
{this.props.action}
|
|
</div>
|
|
<div className="maputnik-input-block-content" ref={el => {this._blockEl = el;}}>
|
|
{this.props.children}
|
|
</div>
|
|
{this.props.fieldSpec &&
|
|
<div
|
|
className="maputnik-doc-inline"
|
|
style={{display: this.state.showDoc ? "" : "none"}}
|
|
>
|
|
<Doc fieldSpec={this.props.fieldSpec} />
|
|
</div>
|
|
}
|
|
</label>;
|
|
}
|
|
}
|