#import "GBPaletteEditorController.h"
#import "GBHueSliderCell.h"
#import <Core/gb.h>

#define MAGIC 'SBPL'

typedef struct __attribute__ ((packed)) {
    uint32_t magic;
    bool manual:1;
    bool disabled_lcd_color:1;
    unsigned padding:6;
    struct GB_color_s colors[5];
    int32_t brightness_bias;
    uint32_t hue_bias;
    uint32_t hue_bias_strength;
} theme_t;

static double blend(double from, double to, double position)
{
    return from * (1 - position) + to * position;
}

@implementation NSColor (GBColor)

- (struct GB_color_s)gbColor
{
    NSColor *sRGB = [self colorUsingColorSpace:[NSColorSpace deviceRGBColorSpace]];
    return (struct GB_color_s){round(sRGB.redComponent * 255), round(sRGB.greenComponent * 255), round(sRGB.blueComponent * 255)};
}

- (uint32_t)intValue
{
    struct GB_color_s color = self.gbColor;
    return (color.r << 0) | (color.g << 8) | (color.b << 16) | 0xFF000000;
}

@end

@implementation GBPaletteEditorController

- (NSArray<NSColorWell *> *)colorWells
{
    return @[_colorWell0, _colorWell1, _colorWell2, _colorWell3, _colorWell4];
}

- (void)updateEnabledControls
{
    if (self.manualModeCheckbox.state) {
        _brightnessSlider.enabled = false;
        _hueSlider.enabled = false;
        _hueStrengthSlider.enabled = false;
        _colorWell1.enabled = true;
        _colorWell2.enabled = true;
        _colorWell3.enabled = true;
        if (!(_colorWell4.enabled = self.disableLCDColorCheckbox.state)) {
            _colorWell4.color = _colorWell3.color;
        }
    }
    else {
        _colorWell1.enabled = false;
        _colorWell2.enabled = false;
        _colorWell3.enabled = false;
        _colorWell4.enabled = true;
        _brightnessSlider.enabled = true;
        _hueSlider.enabled = true;
        _hueStrengthSlider.enabled = true;
        [self updateAutoColors];
    }
}

- (NSColor *)autoColorAtPositon:(double)position
{
    NSColor *first = [_colorWell0.color colorUsingColorSpace:[NSColorSpace deviceRGBColorSpace]];
    NSColor *second = [_colorWell4.color colorUsingColorSpace:[NSColorSpace deviceRGBColorSpace]];
    double brightness = 1 / pow(4, (_brightnessSlider.doubleValue - 128) / 128.0);
    position = pow(position, brightness);
    NSColor *hue = _hueSlider.colorValue;
    double bias = _hueStrengthSlider.doubleValue / 256.0;
    double red = 1 / pow(4, (hue.redComponent * 2 - 1) * bias);
    double green = 1 / pow(4, (hue.greenComponent * 2 - 1) * bias);
    double blue = 1 / pow(4, (hue.blueComponent * 2 - 1) * bias);
    NSColor *ret = [NSColor colorWithRed:blend(first.redComponent, second.redComponent, pow(position, red))
                                   green:blend(first.greenComponent, second.greenComponent, pow(position, green))
                                    blue:blend(first.blueComponent, second.blueComponent, pow(position, blue))
                                   alpha:1.0];
    return ret;
}

- (IBAction)updateAutoColors:(id)sender
{
    if (!self.manualModeCheckbox.state) {
        [self updateAutoColors];
    }
    else {
        [self savePalette:sender];
    }
}

- (void)updateAutoColors
{
    if (_disableLCDColorCheckbox.state) {
        _colorWell1.color = [self autoColorAtPositon:8 / 25.0];
        _colorWell2.color = [self autoColorAtPositon:16 / 25.0];
        _colorWell3.color = [self autoColorAtPositon:24 / 25.0];
    }
    else {
        _colorWell1.color = [self autoColorAtPositon:1 / 3.0];
        _colorWell2.color = [self autoColorAtPositon:2 / 3.0];
        _colorWell3.color = _colorWell4.color;
    }
    [self savePalette:nil];
}

- (IBAction)disabledLCDColorCheckboxChanged:(id)sender
{
    [self updateEnabledControls];
}

- (IBAction)manualModeChanged:(id)sender
{
    [self updateEnabledControls];
}

- (IBAction)updateColor4:(id)sender
{
    if (!self.disableLCDColorCheckbox.state) {
        self.colorWell4.color = self.colorWell3.color;
    }
    [self savePalette:self];
}

- (NSInteger)numberOfRowsInTableView:(NSTableView *)tableView
{
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSDictionary *themes = [defaults dictionaryForKey:@"GBThemes"];
    if (themes.count == 0) {
        [defaults setObject:@"Untitled Palette" forKey:@"GBCurrentTheme"];
        [self savePalette:nil];
        return 1;
    }
    return themes.count;
}

-(void)tableView:(NSTableView *)tableView setObjectValue:(id)object forTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row
{
    NSString *oldName = [self tableView:tableView objectValueForTableColumn:tableColumn row:row];
    if ([oldName isEqualToString:object]) {
        return;
    }
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSMutableDictionary *themes = [[defaults dictionaryForKey:@"GBThemes"] ?: @{} mutableCopy];
    NSString *newName = object;
    unsigned i = 2;
    if (!newName.length) {
        newName = @"Untitled Palette";
    }
    while (themes[newName]) {
        newName = [NSString stringWithFormat:@"%@ %d", object, i];
    }
    themes[newName] = themes[oldName];
    [themes removeObjectForKey:oldName];
    if ([oldName isEqualToString:[defaults stringForKey:@"GBCurrentTheme"]]) {
        [defaults setObject:newName forKey:@"GBCurrentTheme"];
    }
    [defaults setObject:themes forKey:@"GBThemes"];
    [tableView reloadData];
    [self awakeFromNib];
}

- (IBAction)deleteTheme:(id)sender
{
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSString *name = [defaults stringForKey:@"GBCurrentTheme"];
    NSMutableDictionary *themes = [[defaults dictionaryForKey:@"GBThemes"] ?: @{} mutableCopy];
    [themes removeObjectForKey:name];
    [defaults setObject:themes forKey:@"GBThemes"];
    [_themesList reloadData];
    [self awakeFromNib];
}

- (void)tableViewSelectionDidChange:(NSNotification *)notification
{
    NSString *name = [self tableView:nil objectValueForTableColumn:nil row:_themesList.selectedRow];
    [[NSUserDefaults standardUserDefaults] setObject:name forKey:@"GBCurrentTheme"];
    [self loadPalette];
    [[NSNotificationCenter defaultCenter] postNotificationName:@"GBColorPaletteChanged" object:nil];
}

- (void)tableViewSelectionIsChanging:(NSNotification *)notification
{
    [self tableViewSelectionDidChange:notification];
}

- (void)awakeFromNib
{
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSDictionary *themes = [defaults dictionaryForKey:@"GBThemes"];
    NSString *theme = [defaults stringForKey:@"GBCurrentTheme"];
    if (theme && themes[theme]) {
        unsigned index = [[themes.allKeys sortedArrayUsingSelector:@selector(localizedCaseInsensitiveCompare:)] indexOfObject:theme];
        [_themesList selectRowIndexes:[NSIndexSet indexSetWithIndex:index] byExtendingSelection:false];
    }
    else {
        [_themesList selectRowIndexes:[NSIndexSet indexSetWithIndex:0] byExtendingSelection:false];
    }
    [self tableViewSelectionDidChange:nil];
}

- (IBAction)addTheme:(id)sender
{
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSDictionary *themes = [defaults dictionaryForKey:@"GBThemes"];
    NSString *newName = @"Untitled Palette";
    unsigned i = 2;
    while (themes[newName]) {
        newName = [NSString stringWithFormat:@"Untitled Palette %d", i++];
    }
    [defaults setObject:newName forKey:@"GBCurrentTheme"];
    [self savePalette:sender];
    [_themesList reloadData];
    [self awakeFromNib];
}

- (id)tableView:(NSTableView *)tableView objectValueForTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row
{
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSDictionary *themes = [defaults dictionaryForKey:@"GBThemes"];
    return [themes.allKeys sortedArrayUsingSelector:@selector(localizedCaseInsensitiveCompare:)][row];
}

- (void)loadPalette
{
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSDictionary *theme = [defaults dictionaryForKey:@"GBThemes"][[defaults stringForKey:@"GBCurrentTheme"]];
    NSArray *colors = theme[@"Colors"];
    if (colors.count == 5) {
        unsigned i = 0;
        for (NSNumber *color in colors) {
            uint32_t c = [color unsignedIntValue];
            self.colorWells[i++].color = [NSColor colorWithRed:(c & 0xFF) / 255.0
                                                       green:((c >> 8) & 0xFF) / 255.0
                                                        blue:((c >> 16) & 0xFF) / 255.0
                                                       alpha:1.0];
        }
    }
    _disableLCDColorCheckbox.state = [theme[@"DisabledLCDColor"] boolValue];
    _manualModeCheckbox.state = [theme[@"Manual"] boolValue];
    _brightnessSlider.doubleValue = [theme[@"BrightnessBias"] doubleValue] * 128 + 128;
    _hueSlider.doubleValue = [theme[@"HueBias"] doubleValue] * 360;
    _hueStrengthSlider.doubleValue = [theme[@"HueBiasStrength"] doubleValue] * 256;
    [self updateEnabledControls];
}

- (IBAction)savePalette:(id)sender
{
    NSDictionary *theme = @{
        @"Colors":
            @[@(_colorWell0.color.intValue),
              @(_colorWell1.color.intValue),
              @(_colorWell2.color.intValue),
              @(_colorWell3.color.intValue),
              @(_colorWell4.color.intValue)],
        @"DisabledLCDColor": _disableLCDColorCheckbox.state? @YES : @NO,
        @"Manual": _manualModeCheckbox.state? @YES : @NO,
        @"BrightnessBias": @((_brightnessSlider.doubleValue - 128) / 128.0),
        @"HueBias": @(_hueSlider.doubleValue / 360.0),
        @"HueBiasStrength": @(_hueStrengthSlider.doubleValue / 256.0)
    };
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    NSMutableDictionary *themes = [[defaults dictionaryForKey:@"GBThemes"] ?: @{} mutableCopy];
    themes[[defaults stringForKey:@"GBCurrentTheme"]] = theme;
    [defaults setObject:themes forKey:@"GBThemes"];
    [[NSNotificationCenter defaultCenter] postNotificationName:@"GBColorPaletteChanged" object:nil];
}

+ (const GB_palette_t *)userPalette
{
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
    switch ([defaults integerForKey:@"GBColorPalette"]) {
        case 1: return &GB_PALETTE_DMG;
        case 2: return &GB_PALETTE_MGB;
        case 3: return &GB_PALETTE_GBL;
        default: return &GB_PALETTE_GREY;
        case -1: {
            static GB_palette_t customPalette;
            NSArray *colors = [defaults dictionaryForKey:@"GBThemes"][[defaults stringForKey:@"GBCurrentTheme"]][@"Colors"];
            if (colors.count == 5) {
                unsigned i = 0;
                for (NSNumber *color in colors) {
                    uint32_t c = [color unsignedIntValue];
                    customPalette.colors[i++] = (struct GB_color_s) {c, c >> 8, c >> 16};
                }
            }
            return &customPalette;
        }
    }
}

- (IBAction)export:(id)sender
{
    NSSavePanel *savePanel = [NSSavePanel savePanel];
    [savePanel setAllowedFileTypes:@[@"sbp"]];
    savePanel.nameFieldStringValue = [NSString stringWithFormat:@"%@.sbp", [[NSUserDefaults standardUserDefaults] stringForKey:@"GBCurrentTheme"]];
    if ([savePanel runModal] == NSModalResponseOK) {
        theme_t theme = {0,};
        theme.magic = MAGIC;
        theme.manual = _manualModeCheckbox.state;
        theme.disabled_lcd_color = _disableLCDColorCheckbox.state;
        unsigned i = 0;
        for (NSColorWell *well in self.colorWells) {
            theme.colors[i++] = well.color.gbColor;
        }
        theme.brightness_bias = (_brightnessSlider.doubleValue - 128) * (0x40000000 / 128);
        theme.hue_bias = round(_hueSlider.doubleValue * (0x80000000 / 360.0));
        theme.hue_bias_strength = (_hueStrengthSlider.doubleValue) * (0x80000000 / 256);
        size_t size = sizeof(theme);
        if (theme.manual) {
            size = theme.disabled_lcd_color? 5 + 5 * sizeof(theme.colors[0]) : 5 + 4 * sizeof(theme.colors[0]);
        }
        [[NSData dataWithBytes:&theme length:size] writeToURL:savePanel.URL atomically:false];
    }
}

- (IBAction)import:(id)sender
{
    NSOpenPanel *openPanel = [NSOpenPanel openPanel];
    [openPanel setAllowedFileTypes:@[@"sbp"]];
    if ([openPanel runModal] == NSModalResponseOK) {
        NSData *data = [NSData dataWithContentsOfURL:openPanel.URL];
        theme_t theme = {0,};
        memcpy(&theme, data.bytes, MIN(sizeof(theme), data.length));
        if (theme.magic != MAGIC) {
            NSBeep();
            return;
        }
        _manualModeCheckbox.state = theme.manual;
        _disableLCDColorCheckbox.state = theme.disabled_lcd_color;
        unsigned i = 0;
        for (NSColorWell *well in self.colorWells) {
            well.color = [NSColor colorWithRed:theme.colors[i].r / 255.0
                                         green:theme.colors[i].g / 255.0
                                          blue:theme.colors[i].b / 255.0
                                         alpha:1.0];
            i++;
        }
        if (!theme.disabled_lcd_color) {
            _colorWell4.color = _colorWell3.color;
        }
        _brightnessSlider.doubleValue = theme.brightness_bias / (0x40000000 / 128.0) + 128;
        _hueSlider.doubleValue = theme.hue_bias / (0x80000000 / 360.0);
        _hueStrengthSlider.doubleValue = theme.hue_bias_strength / (0x80000000 / 256.0);
        
        NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
        NSDictionary *themes = [defaults dictionaryForKey:@"GBThemes"];
        NSString *baseName = openPanel.URL.lastPathComponent.stringByDeletingPathExtension;
        NSString *newName = baseName;
        i = 2;
        while (themes[newName]) {
            newName = [NSString stringWithFormat:@"%@ %d", baseName, i++];
        }
        [defaults setObject:newName forKey:@"GBCurrentTheme"];
        [self savePalette:sender];
        [self awakeFromNib];
    }
}

- (IBAction)done:(NSButton *)sender
{
    [sender.window.sheetParent endSheet:sender.window];
}

- (instancetype)init
{
    static id singleton = nil;
    if (singleton) return singleton;
    return (singleton = [super init]);
}
@end