"""
Support for Flux lights.

For more details about this platform, please refer to the documentation at
https://home-assistant.io/components/light.flux_led/
"""
import logging
import socket
import random

import voluptuous as vol

from homeassistant.const import CONF_DEVICES, CONF_NAME, CONF_PROTOCOL
from homeassistant.components.light import (
    ATTR_BRIGHTNESS, ATTR_RGB_COLOR, ATTR_EFFECT, ATTR_WHITE_VALUE,
    EFFECT_COLORLOOP, EFFECT_RANDOM, SUPPORT_BRIGHTNESS, SUPPORT_EFFECT,
    SUPPORT_RGB_COLOR, SUPPORT_WHITE_VALUE, Light,
    PLATFORM_SCHEMA)
import homeassistant.helpers.config_validation as cv

REQUIREMENTS = ['flux_led==0.19']

_LOGGER = logging.getLogger(__name__)

CONF_AUTOMATIC_ADD = 'automatic_add'
ATTR_MODE = 'mode'

DOMAIN = 'flux_led'

SUPPORT_FLUX_LED_RGB = (SUPPORT_BRIGHTNESS | SUPPORT_EFFECT |
                        SUPPORT_RGB_COLOR)
SUPPORT_FLUX_LED_RGBW = (SUPPORT_WHITE_VALUE | SUPPORT_EFFECT |
                         SUPPORT_RGB_COLOR)

MODE_RGB = 'rgb'
MODE_RGBW = 'rgbw'

# List of supported effects which aren't already declared in LIGHT
EFFECT_RED_FADE = 'red_fade'
EFFECT_GREEN_FADE = 'green_fade'
EFFECT_BLUE_FADE = 'blue_fade'
EFFECT_YELLOW_FADE = 'yellow_fade'
EFFECT_CYAN_FADE = 'cyan_fade'
EFFECT_PURPLE_FADE = 'purple_fade'
EFFECT_WHITE_FADE = 'white_fade'
EFFECT_RED_GREEN_CROSS_FADE = 'rg_cross_fade'
EFFECT_RED_BLUE_CROSS_FADE = 'rb_cross_fade'
EFFECT_GREEN_BLUE_CROSS_FADE = 'gb_cross_fade'
EFFECT_COLORSTROBE = 'colorstrobe'
EFFECT_RED_STROBE = 'red_strobe'
EFFECT_GREEN_STROBE = 'green_strobe'
EFFECT_BLUE_STOBE = 'blue_strobe'
EFFECT_YELLOW_STROBE = 'yellow_strobe'
EFFECT_CYAN_STROBE = 'cyan_strobe'
EFFECT_PURPLE_STROBE = 'purple_strobe'
EFFECT_WHITE_STROBE = 'white_strobe'
EFFECT_COLORJUMP = 'colorjump'

FLUX_EFFECT_LIST = [
    EFFECT_COLORLOOP,
    EFFECT_RANDOM,
    EFFECT_RED_FADE,
    EFFECT_GREEN_FADE,
    EFFECT_BLUE_FADE,
    EFFECT_YELLOW_FADE,
    EFFECT_CYAN_FADE,
    EFFECT_PURPLE_FADE,
    EFFECT_WHITE_FADE,
    EFFECT_RED_GREEN_CROSS_FADE,
    EFFECT_RED_BLUE_CROSS_FADE,
    EFFECT_GREEN_BLUE_CROSS_FADE,
    EFFECT_COLORSTROBE,
    EFFECT_RED_STROBE,
    EFFECT_GREEN_STROBE,
    EFFECT_BLUE_STOBE,
    EFFECT_YELLOW_STROBE,
    EFFECT_CYAN_STROBE,
    EFFECT_PURPLE_STROBE,
    EFFECT_WHITE_STROBE,
    EFFECT_COLORJUMP]

DEVICE_SCHEMA = vol.Schema({
    vol.Optional(CONF_NAME): cv.string,
    vol.Optional(ATTR_MODE, default=MODE_RGBW):
        vol.All(cv.string, vol.In([MODE_RGBW, MODE_RGB])),
    vol.Optional(CONF_PROTOCOL, default=None):
        vol.All(cv.string, vol.In(['ledenet'])),
})

PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
    vol.Optional(CONF_DEVICES, default={}): {cv.string: DEVICE_SCHEMA},
    vol.Optional(CONF_AUTOMATIC_ADD, default=False):  cv.boolean,
})


def setup_platform(hass, config, add_devices, discovery_info=None):
    """Set up the Flux lights."""
    import flux_led
    lights = []
    light_ips = []

    for ipaddr, device_config in config.get(CONF_DEVICES, {}).items():
        device = {}
        device['name'] = device_config[CONF_NAME]
        device['ipaddr'] = ipaddr
        device[CONF_PROTOCOL] = device_config[CONF_PROTOCOL]
        device[ATTR_MODE] = device_config[ATTR_MODE]
        light = FluxLight(device)
        if light.is_valid:
            lights.append(light)
            light_ips.append(ipaddr)

    if not config.get(CONF_AUTOMATIC_ADD, False):
        add_devices(lights)
        return

    # Find the bulbs on the LAN
    scanner = flux_led.BulbScanner()
    scanner.scan(timeout=10)
    for device in scanner.getBulbInfo():
        ipaddr = device['ipaddr']
        if ipaddr in light_ips:
            continue
        device['name'] = '{} {}'.format(device['id'], ipaddr)
        device[ATTR_MODE] = 'rgbw'
        device[CONF_PROTOCOL] = None
        light = FluxLight(device)
        if light.is_valid:
            lights.append(light)
            light_ips.append(ipaddr)

    add_devices(lights)


class FluxLight(Light):
    """Representation of a Flux light."""

    def __init__(self, device):
        """Initialize the light."""
        import flux_led

        self._name = device['name']
        self._ipaddr = device['ipaddr']
        self._protocol = device[CONF_PROTOCOL]
        self._mode = device[ATTR_MODE]
        self.is_valid = True
        self._bulb = None

        try:
            self._bulb = flux_led.WifiLedBulb(self._ipaddr)
            if self._protocol:
                self._bulb.setProtocol(self._protocol)

            # After bulb object is created the status is updated. We can
            # now set the correct mode if it was not explicitly defined.
            if not self._mode:
                if self._bulb.rgbwcapable:
                    self._mode = MODE_RGBW
                else:
                    self._mode = MODE_RGB

        except socket.error:
            self.is_valid = False
            _LOGGER.error(
                "Failed to connect to bulb %s, %s", self._ipaddr, self._name)

    @property
    def unique_id(self):
        """Return the ID of this light."""
        return '{}.{}'.format(self.__class__, self._ipaddr)

    @property
    def name(self):
        """Return the name of the device if any."""
        return self._name

    @property
    def is_on(self):
        """Return true if device is on."""
        return self._bulb.isOn()

    @property
    def brightness(self):
        """Return the brightness of this light between 0..255."""
        if self._mode == MODE_RGB:
            return self._bulb.brightness
        return None  # not used for RGBW

    @property
    def white_value(self):
        """Return the white value of this light between 0..255."""
        if self._mode == MODE_RGBW:
            return self._bulb.getRgbw()[3]
        return None  # not used for RGB

    @property
    def rgb_color(self):
        """Return the color property."""
        return self._bulb.getRgb()

    @property
    def supported_features(self):
        """Flag supported features."""
        if self._mode == MODE_RGBW:
            return SUPPORT_FLUX_LED_RGBW
        elif self._mode == MODE_RGB:
            return SUPPORT_FLUX_LED_RGB
        return 0

    @property
    def effect_list(self):
        """Return the list of supported effects."""
        return FLUX_EFFECT_LIST

    def turn_on(self, **kwargs):
        """Turn the specified or all lights on."""
        if not self.is_on:
            self._bulb.turnOn()

        rgb = kwargs.get(ATTR_RGB_COLOR)
        brightness = kwargs.get(ATTR_BRIGHTNESS)
        white_value = kwargs.get(ATTR_WHITE_VALUE)
        effect = kwargs.get(ATTR_EFFECT)

        if rgb is not None and brightness is not None:
            self._bulb.setRgb(*tuple(rgb), brightness=brightness)
        elif rgb is not None and white_value is not None:
            self._bulb.setRgbw(*tuple(rgb), w=white_value)
        elif rgb is not None:
            # self.white_value and self.brightness are appropriately
            # returning None for MODE_RGB and MODE_RGBW respectively
            self._bulb.setRgbw(*tuple(rgb),
                               w=self.white_value,
                               brightness=self.brightness)
        elif brightness is not None:
            self._bulb.setRgb(*self.rgb_color, brightness=brightness)
        elif white_value is not None:
            self._bulb.setRgbw(*self.rgb_color, w=white_value)
        elif effect == EFFECT_RANDOM:
            self._bulb.setRgb(random.randint(0, 255),
                              random.randint(0, 255),
                              random.randint(0, 255))
        elif effect == EFFECT_COLORLOOP:
            self._bulb.setPresetPattern(0x25, 50)
        elif effect == EFFECT_RED_FADE:
            self._bulb.setPresetPattern(0x26, 50)
        elif effect == EFFECT_GREEN_FADE:
            self._bulb.setPresetPattern(0x27, 50)
        elif effect == EFFECT_BLUE_FADE:
            self._bulb.setPresetPattern(0x28, 50)
        elif effect == EFFECT_YELLOW_FADE:
            self._bulb.setPresetPattern(0x29, 50)
        elif effect == EFFECT_CYAN_FADE:
            self._bulb.setPresetPattern(0x2a, 50)
        elif effect == EFFECT_PURPLE_FADE:
            self._bulb.setPresetPattern(0x2b, 50)
        elif effect == EFFECT_WHITE_FADE:
            self._bulb.setPresetPattern(0x2c, 50)
        elif effect == EFFECT_RED_GREEN_CROSS_FADE:
            self._bulb.setPresetPattern(0x2d, 50)
        elif effect == EFFECT_RED_BLUE_CROSS_FADE:
            self._bulb.setPresetPattern(0x2e, 50)
        elif effect == EFFECT_GREEN_BLUE_CROSS_FADE:
            self._bulb.setPresetPattern(0x2f, 50)
        elif effect == EFFECT_COLORSTROBE:
            self._bulb.setPresetPattern(0x30, 50)
        elif effect == EFFECT_RED_STROBE:
            self._bulb.setPresetPattern(0x31, 50)
        elif effect == EFFECT_GREEN_STROBE:
            self._bulb.setPresetPattern(0x32, 50)
        elif effect == EFFECT_BLUE_STOBE:
            self._bulb.setPresetPattern(0x33, 50)
        elif effect == EFFECT_YELLOW_STROBE:
            self._bulb.setPresetPattern(0x34, 50)
        elif effect == EFFECT_CYAN_STROBE:
            self._bulb.setPresetPattern(0x35, 50)
        elif effect == EFFECT_PURPLE_STROBE:
            self._bulb.setPresetPattern(0x36, 50)
        elif effect == EFFECT_WHITE_STROBE:
            self._bulb.setPresetPattern(0x37, 50)
        elif effect == EFFECT_COLORJUMP:
            self._bulb.setPresetPattern(0x38, 50)

    def turn_off(self, **kwargs):
        """Turn the specified or all lights off."""
        self._bulb.turnOff()

    def update(self):
        """Synchronize state with bulb."""
        self._bulb.refreshState()