/**
 * Copyright (c) Nicolas Gallagher.
 * Copyright (c) Facebook, Inc. and its affiliates.
 *
 * This source code is licensed under the MIT license found in the
 * LICENSE file in the root directory of this source tree.
 *
 * @flow
 */

import applyNativeMethods from '../../modules/applyNativeMethods';
import ColorPropType from '../ColorPropType';
import createElement from '../createElement';
import StyleSheet from '../StyleSheet';
import UIManager from '../UIManager';
import View from '../View';
import ViewPropTypes, { type ViewProps } from '../ViewPropTypes';
import React, { Component } from 'react';
import { bool, func } from 'prop-types';

type Props = ViewProps & {
  color?: ColorPropType,
  disabled?: boolean,
  onChange?: Function,
  onValueChange?: Function,
  value?: boolean
};

class CheckBox extends Component<Props> {
  _checkboxElement: HTMLInputElement;

  static displayName = 'CheckBox';

  static propTypes = {
    ...ViewPropTypes,
    color: ColorPropType,
    disabled: bool,
    onChange: func,
    onValueChange: func,
    value: bool
  };

  static defaultProps = {
    disabled: false,
    value: false
  };

  blur() {
    UIManager.blur(this._checkboxElement);
  }

  focus() {
    UIManager.focus(this._checkboxElement);
  }

  render() {
    const {
      color,
      disabled,
      /* eslint-disable */
      onChange,
      onValueChange,
      /* eslint-enable */
      style,
      value,
      ...other
    } = this.props;

    const fakeControl = (
      <View
        style={[
          styles.fakeControl,
          value && styles.fakeControlChecked,
          // custom color
          value && color && { backgroundColor: color, borderColor: color },
          disabled && styles.fakeControlDisabled,
          value && disabled && styles.fakeControlCheckedAndDisabled
        ]}
      />
    );

    const nativeControl = createElement('input', {
      checked: value,
      disabled: disabled,
      onChange: this._handleChange,
      ref: this._setCheckboxRef,
      style: [styles.nativeControl, styles.cursorInherit],
      type: 'checkbox'
    });

    return (
      <View {...other} style={[styles.root, style, disabled && styles.cursorDefault]}>
        {fakeControl}
        {nativeControl}
      </View>
    );
  }

  _handleChange = (event: Object) => {
    const { onChange, onValueChange } = this.props;
    const value = event.nativeEvent.target.checked;
    event.nativeEvent.value = value;
    onChange && onChange(event);
    onValueChange && onValueChange(value);
  };

  _setCheckboxRef = element => {
    this._checkboxElement = element;
  };
}

const styles = StyleSheet.create({
  root: {
    cursor: 'pointer',
    height: 16,
    userSelect: 'none',
    width: 16
  },
  cursorDefault: {
    cursor: 'default'
  },
  cursorInherit: {
    cursor: 'inherit'
  },
  fakeControl: {
    alignItems: 'center',
    backgroundColor: '#fff',
    borderColor: '#657786',
    borderRadius: 2,
    borderStyle: 'solid',
    borderWidth: 2,
    height: '100%',
    justifyContent: 'center',
    width: '100%'
  },
  fakeControlChecked: {
    backgroundColor: '#009688',
    backgroundImage:
      'url("")',
    backgroundRepeat: 'no-repeat',
    borderColor: '#009688'
  },
  fakeControlDisabled: {
    borderColor: '#CCD6DD'
  },
  fakeControlCheckedAndDisabled: {
    backgroundColor: '#AAB8C2',
    borderColor: '#AAB8C2'
  },
  nativeControl: {
    ...StyleSheet.absoluteFillObject,
    height: '100%',
    margin: 0,
    opacity: 0,
    padding: 0,
    width: '100%'
  }
});

export default applyNativeMethods(CheckBox);