2016-09-12 19:47:28 +02:00
|
|
|
import React from 'react'
|
2016-12-19 11:46:48 +01:00
|
|
|
import Color from 'color'
|
2016-12-17 22:19:03 +01:00
|
|
|
import ChromePicker from 'react-color/lib/components/chrome/Chrome'
|
2017-11-06 16:32:04 +01:00
|
|
|
import PropTypes from 'prop-types'
|
2016-12-17 21:25:00 +01:00
|
|
|
|
|
|
|
function formatColor(color) {
|
2016-12-17 21:52:27 +01:00
|
|
|
const rgb = color.rgb
|
|
|
|
return `rgba(${rgb.r}, ${rgb.g}, ${rgb.b}, ${rgb.a})`
|
2016-12-17 21:25:00 +01:00
|
|
|
}
|
2016-09-12 19:47:28 +02:00
|
|
|
|
|
|
|
/*** Number fields with support for min, max and units and documentation*/
|
|
|
|
class ColorField extends React.Component {
|
2016-12-17 22:28:24 +01:00
|
|
|
static propTypes = {
|
2017-11-06 16:32:04 +01:00
|
|
|
onChange: PropTypes.func.isRequired,
|
|
|
|
name: PropTypes.string.isRequired,
|
|
|
|
value: PropTypes.string,
|
|
|
|
doc: PropTypes.string,
|
|
|
|
style: PropTypes.object,
|
|
|
|
default: PropTypes.string,
|
2016-09-12 19:47:28 +02:00
|
|
|
}
|
|
|
|
|
2016-12-17 22:28:24 +01:00
|
|
|
constructor(props) {
|
|
|
|
super(props)
|
|
|
|
this.state = {
|
2016-12-21 12:06:33 +01:00
|
|
|
pickerOpened: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//TODO: I much rather would do this with absolute positioning
|
|
|
|
//but I am too stupid to get it to work together with fixed position
|
|
|
|
//and scrollbars so I have to fallback to JavaScript
|
|
|
|
calcPickerOffset() {
|
2017-11-08 09:47:36 +01:00
|
|
|
const elem = this.colorInput
|
2016-12-21 12:06:33 +01:00
|
|
|
if(elem) {
|
|
|
|
const pos = elem.getBoundingClientRect()
|
|
|
|
return {
|
|
|
|
top: pos.top,
|
2016-12-26 11:22:41 +01:00
|
|
|
left: pos.left + 196,
|
2016-12-21 12:06:33 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return {
|
|
|
|
top: 160,
|
2016-12-26 11:22:41 +01:00
|
|
|
left: 555,
|
2016-12-21 12:06:33 +01:00
|
|
|
}
|
2016-12-17 22:28:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
togglePicker() {
|
|
|
|
this.setState({ pickerOpened: !this.state.pickerOpened })
|
|
|
|
}
|
|
|
|
|
2016-12-20 14:50:38 +01:00
|
|
|
get color() {
|
2017-11-29 19:00:52 +01:00
|
|
|
// Catch invalid color.
|
|
|
|
try {
|
|
|
|
return Color(this.props.value).rgb()
|
|
|
|
}
|
|
|
|
catch(err) {
|
|
|
|
console.warn("Error parsing color: ", err);
|
2017-11-29 19:07:03 +01:00
|
|
|
return Color("rgb(255,255,255)");
|
2017-11-29 19:00:52 +01:00
|
|
|
}
|
2016-12-20 14:50:38 +01:00
|
|
|
}
|
|
|
|
|
2016-12-17 21:25:00 +01:00
|
|
|
render() {
|
2016-12-21 12:06:33 +01:00
|
|
|
const offset = this.calcPickerOffset()
|
2017-03-07 10:59:11 +01:00
|
|
|
var currentColor = this.color.object()
|
|
|
|
currentColor = {
|
|
|
|
r: currentColor.r,
|
|
|
|
g: currentColor.g,
|
|
|
|
b: currentColor.b,
|
|
|
|
// Rename alpha -> a for ChromePicker
|
|
|
|
a: currentColor.alpha
|
|
|
|
}
|
|
|
|
|
2016-12-21 12:06:33 +01:00
|
|
|
const picker = <div
|
2017-01-10 21:28:30 +01:00
|
|
|
className="maputnik-color-picker-offset"
|
2016-12-21 12:06:33 +01:00
|
|
|
style={{
|
2017-01-10 21:28:30 +01:00
|
|
|
position: 'fixed',
|
|
|
|
zIndex: 1,
|
2016-12-21 12:06:33 +01:00
|
|
|
left: offset.left,
|
|
|
|
top: offset.top,
|
2016-12-17 21:25:00 +01:00
|
|
|
}}>
|
2016-12-17 22:28:24 +01:00
|
|
|
<ChromePicker
|
2017-03-07 10:59:11 +01:00
|
|
|
color={currentColor}
|
2016-12-17 22:28:24 +01:00
|
|
|
onChange={c => this.props.onChange(formatColor(c))}
|
|
|
|
/>
|
|
|
|
<div
|
2017-01-10 21:28:30 +01:00
|
|
|
className="maputnik-color-picker-offset"
|
2016-12-17 22:28:24 +01:00
|
|
|
onClick={this.togglePicker.bind(this)}
|
|
|
|
style={{
|
|
|
|
zIndex: -1,
|
|
|
|
position: 'fixed',
|
|
|
|
top: '0px',
|
|
|
|
right: '0px',
|
|
|
|
bottom: '0px',
|
|
|
|
left: '0px',
|
2016-12-21 12:06:33 +01:00
|
|
|
}}
|
|
|
|
/>
|
2016-12-17 22:28:24 +01:00
|
|
|
</div>
|
2016-09-12 20:29:53 +02:00
|
|
|
|
2017-03-07 11:16:58 +01:00
|
|
|
var swatchStyle = {
|
2017-06-23 16:00:43 +02:00
|
|
|
backgroundColor: this.props.value
|
2017-03-07 11:16:58 +01:00
|
|
|
};
|
|
|
|
|
2017-01-10 21:28:30 +01:00
|
|
|
return <div className="maputnik-color-wrapper">
|
2016-12-17 22:28:24 +01:00
|
|
|
{this.state.pickerOpened && picker}
|
2017-03-07 11:16:58 +01:00
|
|
|
<div className="maputnik-color-swatch" style={swatchStyle}></div>
|
2016-12-17 21:25:00 +01:00
|
|
|
<input
|
2018-05-17 12:43:25 +02:00
|
|
|
spellCheck="false"
|
2017-01-10 21:28:30 +01:00
|
|
|
className="maputnik-color"
|
2017-11-08 09:47:36 +01:00
|
|
|
ref={(input) => this.colorInput = input}
|
2016-12-17 22:28:24 +01:00
|
|
|
onClick={this.togglePicker.bind(this)}
|
2017-01-10 21:28:30 +01:00
|
|
|
style={this.props.style}
|
2016-12-17 21:25:00 +01:00
|
|
|
name={this.props.name}
|
|
|
|
placeholder={this.props.default}
|
|
|
|
value={this.props.value ? this.props.value : ""}
|
|
|
|
onChange={(e) => this.props.onChange(e.target.value)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
}
|
2016-09-12 19:47:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export default ColorField
|