/* X11Controller.m -- connect the IB ui, also the NSApp delegate
 *
 * Copyright (c) 2002-2012 Apple Inc. All rights reserved.
 *
 * Permission is hereby granted, free of charge, to any person
 * obtaining a copy of this software and associated documentation files
 * (the "Software"), to deal in the Software without restriction,
 * including without limitation the rights to use, copy, modify, merge,
 * publish, distribute, sublicense, and/or sell copies of the Software,
 * and to permit persons to whom the Software is furnished to do so,
 * subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be
 * included in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 * NONINFRINGEMENT.  IN NO EVENT SHALL THE ABOVE LISTED COPYRIGHT
 * HOLDER(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 * DEALINGS IN THE SOFTWARE.
 *
 * Except as contained in this notice, the name(s) of the above
 * copyright holders shall not be used in advertising or otherwise to
 * promote the sale, use or other dealings in this Software without
 * prior written authorization.
 */

#include "sanitizedCarbon.h"
#include <AvailabilityMacros.h>

#ifdef HAVE_DIX_CONFIG_H
#include <dix-config.h>
#endif

#include "quartzCommon.h"

#import "X11Controller.h"
#import "X11Application.h"

#include "opaque.h"
#include "darwin.h"
#include "darwinEvents.h"
#include "quartz.h"
#include "quartzKeyboard.h"
#include <X11/extensions/applewmconst.h>
#include "applewmExt.h"

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <asl.h>
#include <stdlib.h>

extern aslclient aslc;
extern char *bundle_id_prefix;

@implementation X11Controller

- (void) awakeFromNib
{
    X11Application *xapp = NSApp;
    NSArray *array;

    /* Point X11Application at ourself. */
    [xapp set_controller:self];

    array = [xapp prefs_get_array:@PREFS_APPSMENU];
    if (array != nil) {
        int count;

        /* convert from [TITLE1 COMMAND1 TITLE2 COMMAND2 ...]
           to [[TITLE1 COMMAND1] [TITLE2 COMMAND2] ...] format. */

        count = [array count];
        if (count > 0
            && ![[array objectAtIndex:0] isKindOfClass:[NSArray class]]) {
            int i;
            NSMutableArray *copy, *sub;

            copy = [NSMutableArray arrayWithCapacity:(count / 2)];

            for (i = 0; i < count / 2; i++) {
                sub = [[NSMutableArray alloc] initWithCapacity:3];
                [sub addObject:[array objectAtIndex:i * 2]];
                [sub addObject:[array objectAtIndex:i * 2 + 1]];
                [sub addObject:@""];
                [copy addObject:sub];
                [sub release];
            }

            array = copy;
        }

        [self set_apps_menu:array];
    }

    [[NSNotificationCenter defaultCenter]
     addObserver: self
        selector: @selector(apps_table_done:)
            name: NSWindowWillCloseNotification
          object: [apps_table window]];

    // Setup data about our Windows menu
    if (window_separator) {
        [[window_separator menu] removeItem:window_separator];
        window_separator = nil;
    }

    windows_menu_start = [[X11App windowsMenu] numberOfItems];
}

- (void) item_selected:sender
{
    [NSApp activateIgnoringOtherApps:YES];

    DarwinSendDDXEvent(kXquartzControllerNotify, 2,
                       AppleWMWindowMenuItem, [sender tag]);
}

- (void) remove_window_menu
{
    NSMenu *menu;
    int count, i;

    /* Work backwards so we don't mess up the indices */
    menu = [X11App windowsMenu];
    count = [menu numberOfItems];
    for (i = count - 1; i >= windows_menu_start; i--)
        [menu removeItemAtIndex:i];

    count = [dock_menu indexOfItem:dock_window_separator];
    for (i = 0; i < count; i++)
        [dock_menu removeItemAtIndex:0];
}

- (void) install_window_menu:(NSArray *)list
{
    NSMenu *menu;
    NSMenuItem *item;
    int first, count, i;

    menu = [X11App windowsMenu];
    first = windows_menu_start + 1;
    count = [list count];

    // Push a Separator
    if (count) {
        [menu addItem:[NSMenuItem separatorItem]];
    }

    for (i = 0; i < count; i++) {
        NSString *name, *shortcut;

        name = [[list objectAtIndex:i] objectAtIndex:0];
        shortcut = [[list objectAtIndex:i] objectAtIndex:1];

        if (windowItemModMask == 0 || windowItemModMask == -1)
            shortcut = @"";

        item =
            (NSMenuItem *)[menu addItemWithTitle:name action:
                           @selector
                           (item_selected:) keyEquivalent:shortcut];
        [item setKeyEquivalentModifierMask:(NSUInteger)windowItemModMask];
        [item setTarget:self];
        [item setTag:i];
        [item setEnabled:YES];

        item = (NSMenuItem *)[dock_menu  insertItemWithTitle:name
                                                      action:@selector
                              (item_selected:) keyEquivalent:shortcut
                                                     atIndex:i];
        [item setKeyEquivalentModifierMask:(NSUInteger)windowItemModMask];
        [item setTarget:self];
        [item setTag:i];
        [item setEnabled:YES];
    }

    if (checked_window_item >= 0 && checked_window_item < count) {
        item = (NSMenuItem *)[menu itemAtIndex:first + checked_window_item];
        [item setState:NSOnState];
        item = (NSMenuItem *)[dock_menu itemAtIndex:checked_window_item];
        [item setState:NSOnState];
    }
}

- (void) remove_apps_menu
{
    NSMenu *menu;
    NSMenuItem *item;
    int i;

    if (apps == nil || apps_separator == nil) return;

    menu = [apps_separator menu];

    if (menu != nil) {
        for (i = [menu numberOfItems] - 1; i >= 0; i--) {
            item = (NSMenuItem *)[menu itemAtIndex:i];
            if ([item tag] != 0)
                [menu removeItemAtIndex:i];
        }
    }

    if (dock_apps_menu != nil) {
        for (i = [dock_apps_menu numberOfItems] - 1; i >= 0; i--) {
            item = (NSMenuItem *)[dock_apps_menu itemAtIndex:i];
            if ([item tag] != 0)
                [dock_apps_menu removeItemAtIndex:i];
        }
    }

    [apps release];
    apps = nil;
}

- (void) prepend_apps_item:(NSArray *)list index:(int)i menu:(NSMenu *)menu
{
    NSString *title, *shortcut = @"";
    NSArray *group;
    NSMenuItem *item;

    group = [list objectAtIndex:i];
    title = [group objectAtIndex:0];
    if ([group count] >= 3)
        shortcut = [group objectAtIndex:2];

    if ([title length] != 0) {
        item = (NSMenuItem *)[menu insertItemWithTitle:title
                                                action:@selector (
                                  app_selected:)
                              keyEquivalent:shortcut atIndex:0];
        [item setTarget:self];
        [item setEnabled:YES];
    }
    else {
        item = (NSMenuItem *)[NSMenuItem separatorItem];
        [menu insertItem:item atIndex:0];
    }

    [item setTag:i + 1];                  /* can't be zero, so add one */
}

- (void) install_apps_menu:(NSArray *)list
{
    NSMenu *menu;
    int i, count;

    count = [list count];

    if (count == 0 || apps_separator == nil) return;

    menu = [apps_separator menu];

    for (i = count - 1; i >= 0; i--) {
        if (menu != nil)
            [self prepend_apps_item:list index:i menu:menu];
        if (dock_apps_menu != nil)
            [self prepend_apps_item:list index:i menu:dock_apps_menu];
    }

    apps = [list retain];
}

- (void) set_window_menu:(NSArray *)list
{
    [self remove_window_menu];
    [self install_window_menu:list];

    DarwinSendDDXEvent(kXquartzControllerNotify, 1,
                       AppleWMWindowMenuNotify);
}

- (void) set_window_menu_check:(NSNumber *)nn
{
    NSMenu *menu;
    NSMenuItem *item;
    int first, count;
    int n = [nn intValue];

    menu = [X11App windowsMenu];
    first = windows_menu_start + 1;
    count = [menu numberOfItems] - first;

    if (checked_window_item >= 0 && checked_window_item < count) {
        item = (NSMenuItem *)[menu itemAtIndex:first + checked_window_item];
        [item setState:NSOffState];
        item = (NSMenuItem *)[dock_menu itemAtIndex:checked_window_item];
        [item setState:NSOffState];
    }
    if (n >= 0 && n < count) {
        item = (NSMenuItem *)[menu itemAtIndex:first + n];
        [item setState:NSOnState];
        item = (NSMenuItem *)[dock_menu itemAtIndex:n];
        [item setState:NSOnState];
    }
    checked_window_item = n;
}

- (void) set_apps_menu:(NSArray *)list
{
    [self remove_apps_menu];
    [self install_apps_menu:list];
}

#ifdef XQUARTZ_SPARKLE
- (void) setup_sparkle
{
    if (check_for_updates_item)
        return;  // already did it...

    NSMenu *menu = [x11_about_item menu];

    check_for_updates_item =
        [menu insertItemWithTitle:NSLocalizedString(
             @"Check for X11 Updates...",
             @"Check for X11 Updates...")
         action:@selector (
             checkForUpdates:)
         keyEquivalent:@""
         atIndex:1];
    [check_for_updates_item setTarget:[SUUpdater sharedUpdater]];
    [check_for_updates_item setEnabled:YES];

    // Set X11Controller as the delegate for the updater.
    [[SUUpdater sharedUpdater] setDelegate:self];
}

// Sent immediately before installing the specified update.
- (void)updater:(SUUpdater *)updater willInstallUpdate:(SUAppcastItem *)
   update
{
    //[self set_can_quit:YES];
}

#endif

- (void) launch_client:(NSString *)filename
{
    int child1, child2 = 0;
    int status;
    const char *newargv[4];
    char buf[128];
    char *s;
#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1080
    int stdout_pipe[2];
    int stderr_pipe[2];
#endif

    newargv[0] = [X11App prefs_get_string:@PREFS_LOGIN_SHELL default:"/bin/sh"];
    newargv[1] = "-c";
    newargv[2] = [filename UTF8String];
    newargv[3] = NULL;

    s = getenv("DISPLAY");
    if (s == NULL || s[0] == 0) {
        snprintf(buf, sizeof(buf), ":%s", display);
        setenv("DISPLAY", buf, TRUE);
    }

#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1080
    if (asl_log_descriptor) {
        char *asl_sender;
        aslmsg amsg = asl_new(ASL_TYPE_MSG);
        assert(amsg);

        asprintf(&asl_sender, "%s.%s", bundle_id_prefix, newargv[2]);
        assert(asl_sender);
        for(s = asl_sender + strlen(bundle_id_prefix) + 1; *s; s++) {
            if(! ((*s >= 'a' && *s <= 'z') ||
                  (*s >= 'A' && *s <= 'Z') ||
                  (*s >= '0' && *s <= '9'))) {
                *s = '_';
            }
        }

        (void)asl_set(amsg, ASL_KEY_SENDER, asl_sender);
        free(asl_sender);

        assert(0 == pipe(stdout_pipe));
        fcntl(stdout_pipe[0], F_SETFD, FD_CLOEXEC);
        fcntl(stdout_pipe[0], F_SETFL, O_NONBLOCK);

        assert(0 == pipe(stderr_pipe));
        fcntl(stderr_pipe[0], F_SETFD, FD_CLOEXEC);
        fcntl(stderr_pipe[0], F_SETFL, O_NONBLOCK);

        asl_log_descriptor(aslc, amsg, ASL_LEVEL_INFO, stdout_pipe[0], ASL_LOG_DESCRIPTOR_READ);
        asl_log_descriptor(aslc, amsg, ASL_LEVEL_NOTICE, stderr_pipe[0], ASL_LOG_DESCRIPTOR_READ);

        asl_free(amsg);
    }
#endif

    /* Do the fork-twice trick to avoid having to reap zombies */
    child1 = fork();
    switch (child1) {
    case -1:                                    /* error */
        break;

    case 0:                                     /* child1 */
        child2 = fork();

        switch (child2) {
            int max_files, i;

        case -1:                                    /* error */
            _exit(1);

        case 0:                                     /* child2 */
#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1080
            if (asl_log_descriptor) {
                /* Replace our stdout/stderr */
                dup2(stdout_pipe[1], STDOUT_FILENO);
                dup2(stderr_pipe[1], STDERR_FILENO);
            }
#endif

            /* close all open files except for standard streams */
            max_files = sysconf(_SC_OPEN_MAX);
            for (i = 3; i < max_files; i++)
                close(i);

            /* ensure stdin is on /dev/null */
            close(0);
            open("/dev/null", O_RDONLY);

            execvp(newargv[0], (char * *const)newargv);
            _exit(2);

        default:                                    /* parent (child1) */
            _exit(0);
        }
        break;

    default:                                    /* parent */
        waitpid(child1, &status, 0);
    }

#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1080
    if (asl_log_descriptor) {
        /* Close the write ends of the pipe */
        close(stdout_pipe[1]);
        close(stderr_pipe[1]);
    }
#endif
}

- (void) app_selected:sender
{
    int tag;
    NSString *item;

    tag = [sender tag] - 1;
    if (apps == nil || tag < 0 || tag >= [apps count])
        return;

    item = [[apps objectAtIndex:tag] objectAtIndex:1];

    [self launch_client:item];
}

- (IBAction) apps_table_show:sender
{
    NSArray *columns;
    NSMutableArray *oldapps = nil;

    if (table_apps != nil)
        oldapps = table_apps;

    table_apps = [[NSMutableArray alloc] initWithCapacity:1];
    if (apps != nil)
        [table_apps addObjectsFromArray:apps];

    columns = [apps_table tableColumns];
    [[columns objectAtIndex:0] setIdentifier:@"0"];
    [[columns objectAtIndex:1] setIdentifier:@"1"];
    [[columns objectAtIndex:2] setIdentifier:@"2"];

    [apps_table setDataSource:self];
    [apps_table selectRowIndexes:[NSIndexSet indexSetWithIndex:0]
     byExtendingSelection:NO];

    [[apps_table window] makeKeyAndOrderFront:sender];
    [apps_table reloadData];
    if (oldapps != nil)
        [oldapps release];
}

- (IBAction) apps_table_done:sender
{
    [apps_table deselectAll:sender];    /* flush edits? */

    [self remove_apps_menu];
    [self install_apps_menu:table_apps];

    [NSApp prefs_set_array:@PREFS_APPSMENU value:table_apps];
    [NSApp prefs_synchronize];

    [[apps_table window] orderOut:sender];

    [table_apps release];
    table_apps = nil;
}

- (IBAction) apps_table_new:sender
{
    NSMutableArray *item;

    int row = [apps_table selectedRow], i;

    if (row < 0) row = 0;
    else row = row + 1;

    i = row;
    if (i > [table_apps count])
        return;                         /* avoid exceptions */

    [apps_table deselectAll:sender];

    item = [[NSMutableArray alloc] initWithCapacity:3];
    [item addObject:@""];
    [item addObject:@""];
    [item addObject:@""];

    [table_apps insertObject:item atIndex:i];
    [item release];

    [apps_table reloadData];
    [apps_table selectRowIndexes:[NSIndexSet indexSetWithIndex:row]
     byExtendingSelection:NO];
}

- (IBAction) apps_table_duplicate:sender
{
    int row = [apps_table selectedRow], i;
    NSObject *item;

    if (row < 0) {
        [self apps_table_new:sender];
        return;
    }

    i = row;
    if (i > [table_apps count] - 1) return;                             /* avoid exceptions */

    [apps_table deselectAll:sender];

    item = [[table_apps objectAtIndex:i] mutableCopy];
    [table_apps insertObject:item atIndex:i];
    [item release];

    [apps_table reloadData];
    [apps_table selectRowIndexes:[NSIndexSet indexSetWithIndex:row +
                                  1] byExtendingSelection:NO];
}

- (IBAction) apps_table_delete:sender
{
    int row = [apps_table selectedRow];

    if (row >= 0) {
        int i = row;

        if (i > [table_apps count] - 1) return;                 /* avoid exceptions */

        [apps_table deselectAll:sender];

        [table_apps removeObjectAtIndex:i];
    }

    [apps_table reloadData];

    row = MIN(row, [table_apps count] - 1);
    if (row >= 0)
        [apps_table selectRowIndexes:[NSIndexSet indexSetWithIndex:row]
         byExtendingSelection:NO];
}

- (NSInteger) numberOfRowsInTableView:(NSTableView *)tableView
{
    if (table_apps == nil) return 0;

    return [table_apps count];
}

- (id)             tableView:(NSTableView *)tableView
   objectValueForTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row
{
    NSArray *item;
    int col;

    if (table_apps == nil) return nil;

    col = [[tableColumn identifier] intValue];

    item = [table_apps objectAtIndex:row];
    if ([item count] > col)
        return [item objectAtIndex:col];
    else
        return @"";
}

- (void) tableView:(NSTableView *)tableView setObjectValue:(id)object
    forTableColumn:(NSTableColumn *)tableColumn row:(NSInteger)row
{
    NSMutableArray *item;
    int col;

    if (table_apps == nil) return;

    col = [[tableColumn identifier] intValue];

    item = [table_apps objectAtIndex:row];
    [item replaceObjectAtIndex:col withObject:object];
}

- (void) hide_window:sender
{
    if ([X11App x_active])
        DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMHideWindow);
    else
        NSBeep();                       /* FIXME: something here */
}

- (IBAction)bring_to_front:sender
{
    DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMBringAllToFront);
}

- (IBAction)close_window:sender
{
    if ([X11App x_active])
        DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMCloseWindow);
    else
        [[NSApp keyWindow] performClose:sender];
}

- (IBAction)minimize_window:sender
{
    if ([X11App x_active])
        DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMMinimizeWindow);
    else
        [[NSApp keyWindow] performMiniaturize:sender];
}

- (IBAction)zoom_window:sender
{
    if ([X11App x_active])
        DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMZoomWindow);
    else
        [[NSApp keyWindow] performZoom:sender];
}

- (IBAction) next_window:sender
{
    DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMNextWindow);
}

- (IBAction) previous_window:sender
{
    DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMPreviousWindow);
}

- (IBAction) enable_fullscreen_changed:sender
{
    XQuartzRootlessDefault = ![enable_fullscreen intValue];

    [enable_fullscreen_menu setEnabled:!XQuartzRootlessDefault];
    [enable_fullscreen_menu_text setTextColor:XQuartzRootlessDefault ?[
         NSColor disabledControlTextColor] : [NSColor controlTextColor]];

    DarwinSendDDXEvent(kXquartzSetRootless, 1, XQuartzRootlessDefault);

    [NSApp prefs_set_boolean:@PREFS_ROOTLESS value:XQuartzRootlessDefault];
    [NSApp prefs_synchronize];
}

- (IBAction) toggle_fullscreen:sender
{
    DarwinSendDDXEvent(kXquartzToggleFullscreen, 0);
}

- (void) set_can_quit:(OSX_BOOL)state
{
    can_quit = state;
}

- (IBAction)prefs_changed:sender
{
    if (!sender)
        return;

    if (sender == fake_buttons) {
        darwinFakeButtons = [fake_buttons intValue];
        [NSApp prefs_set_boolean:@PREFS_FAKEBUTTONS value:darwinFakeButtons];
    }
    else if (sender == enable_keyequivs) {
        XQuartzEnableKeyEquivalents = [enable_keyequivs intValue];
        [NSApp prefs_set_boolean:@PREFS_KEYEQUIVS value:
         XQuartzEnableKeyEquivalents];
    }
    else if (sender == sync_keymap) {
        darwinSyncKeymap = [sync_keymap intValue];
        [NSApp prefs_set_boolean:@PREFS_SYNC_KEYMAP value:darwinSyncKeymap];
    }
    else if (sender == enable_fullscreen_menu) {
        XQuartzFullscreenMenu = [enable_fullscreen_menu intValue];
        [NSApp prefs_set_boolean:@PREFS_FULLSCREEN_MENU value:
         XQuartzFullscreenMenu];
    }
    else if (sender == option_sends_alt) {
        BOOL prev_opt_sends_alt = XQuartzOptionSendsAlt;

        XQuartzOptionSendsAlt = [option_sends_alt intValue];
        [NSApp prefs_set_boolean:@PREFS_OPTION_SENDS_ALT value:
         XQuartzOptionSendsAlt];

        if (prev_opt_sends_alt != XQuartzOptionSendsAlt)
            QuartsResyncKeymap(TRUE);
    }
    else if (sender == click_through) {
        [NSApp prefs_set_boolean:@PREFS_CLICK_THROUGH value:[click_through
                                                             intValue]];
    }
    else if (sender == focus_follows_mouse) {
        [NSApp prefs_set_boolean:@PREFS_FFM value:[focus_follows_mouse
                                                   intValue]];
    }
    else if (sender == focus_on_new_window) {
        [NSApp prefs_set_boolean:@PREFS_FOCUS_ON_NEW_WINDOW value:[
             focus_on_new_window intValue]];
    }
    else if (sender == enable_auth) {
        [NSApp prefs_set_boolean:@PREFS_NO_AUTH value:![enable_auth intValue]
        ];
    }
    else if (sender == enable_tcp) {
        [NSApp prefs_set_boolean:@PREFS_NO_TCP value:![enable_tcp intValue]];
    }
    else if (sender == depth) {
        [NSApp prefs_set_integer:@PREFS_DEPTH value:[depth selectedTag]];
    }
    else if (sender == sync_pasteboard) {
        BOOL pbproxy_active = [sync_pasteboard intValue];
        [NSApp prefs_set_boolean:@PREFS_SYNC_PB value:pbproxy_active];

        [sync_pasteboard_to_clipboard setEnabled:pbproxy_active];
        [sync_pasteboard_to_primary setEnabled:pbproxy_active];
        [sync_clipboard_to_pasteboard setEnabled:pbproxy_active];
        [sync_primary_immediately setEnabled:pbproxy_active];

        // setEnabled doesn't do this...
        [sync_text1 setTextColor:pbproxy_active ?[NSColor controlTextColor] :
         [NSColor disabledControlTextColor]];
        [sync_text2 setTextColor:pbproxy_active ?[NSColor controlTextColor] :
         [NSColor disabledControlTextColor]];
    }
    else if (sender == sync_pasteboard_to_clipboard) {
        [NSApp prefs_set_boolean:@PREFS_SYNC_PB_TO_CLIPBOARD value:[
             sync_pasteboard_to_clipboard intValue]];
    }
    else if (sender == sync_pasteboard_to_primary) {
        [NSApp prefs_set_boolean:@PREFS_SYNC_PB_TO_PRIMARY value:[
             sync_pasteboard_to_primary intValue]];
    }
    else if (sender == sync_clipboard_to_pasteboard) {
        [NSApp prefs_set_boolean:@PREFS_SYNC_CLIPBOARD_TO_PB value:[
             sync_clipboard_to_pasteboard intValue]];
    }
    else if (sender == sync_primary_immediately) {
        [NSApp prefs_set_boolean:@PREFS_SYNC_PRIMARY_ON_SELECT value:[
             sync_primary_immediately intValue]];
    }
    else if (sender == scroll_in_device_direction) {
        XQuartzScrollInDeviceDirection =
            [scroll_in_device_direction intValue];
        [NSApp prefs_set_boolean:@PREFS_SCROLL_IN_DEV_DIRECTION value:
         XQuartzScrollInDeviceDirection];
    }

    [NSApp prefs_synchronize];

    DarwinSendDDXEvent(kXquartzReloadPreferences, 0);
}

- (IBAction) prefs_show:sender
{
    BOOL pbproxy_active =
        [NSApp prefs_get_boolean:@PREFS_SYNC_PB default:YES];

    // Remove preferences from the GUI which are not supported
    // TODO: Change 1117 to NSAppKitVersionNumber10_7 when it is defined
    if (scroll_in_device_direction && NSAppKitVersionNumber < 1117) {
        [scroll_in_device_direction removeFromSuperview];
        scroll_in_device_direction = nil;
    }
    else {
        [scroll_in_device_direction setIntValue:
         XQuartzScrollInDeviceDirection];
    }

    [fake_buttons setIntValue:darwinFakeButtons];
    [enable_keyequivs setIntValue:XQuartzEnableKeyEquivalents];
    [sync_keymap setIntValue:darwinSyncKeymap];
    [option_sends_alt setIntValue:XQuartzOptionSendsAlt];
    [click_through setIntValue:[NSApp prefs_get_boolean:@PREFS_CLICK_THROUGH
                                default:NO]];
    [focus_follows_mouse setIntValue:[NSApp prefs_get_boolean:@PREFS_FFM
                                      default:NO]];
    [focus_on_new_window setIntValue:[NSApp prefs_get_boolean:
                                      @PREFS_FOCUS_ON_NEW_WINDOW default:YES]
    ];

    [enable_auth setIntValue:![NSApp prefs_get_boolean:@PREFS_NO_AUTH default
                               :NO]];
    [enable_tcp setIntValue:![NSApp prefs_get_boolean:@PREFS_NO_TCP default:
                              NO]];

    [depth selectItemAtIndex:[depth indexOfItemWithTag:[NSApp
                                                        prefs_get_integer:
                                                        @PREFS_DEPTH default:
                                                        -1]]];

    [sync_pasteboard setIntValue:pbproxy_active];
    [sync_pasteboard_to_clipboard setIntValue:[NSApp prefs_get_boolean:
                                               @PREFS_SYNC_PB_TO_CLIPBOARD
                                               default:YES]];
    [sync_pasteboard_to_primary setIntValue:[NSApp prefs_get_boolean:
                                             @PREFS_SYNC_PB_TO_PRIMARY
                                             default:YES]];
    [sync_clipboard_to_pasteboard setIntValue:[NSApp prefs_get_boolean:
                                               @PREFS_SYNC_CLIPBOARD_TO_PB
                                               default:YES]];
    [sync_primary_immediately setIntValue:[NSApp prefs_get_boolean:
                                           @PREFS_SYNC_PRIMARY_ON_SELECT
                                           default:NO]];

    [sync_pasteboard_to_clipboard setEnabled:pbproxy_active];
    [sync_pasteboard_to_primary setEnabled:pbproxy_active];
    [sync_clipboard_to_pasteboard setEnabled:pbproxy_active];
    [sync_primary_immediately setEnabled:pbproxy_active];

    // setEnabled doesn't do this...
    [sync_text1 setTextColor:pbproxy_active ?[NSColor controlTextColor] : [
         NSColor disabledControlTextColor]];
    [sync_text2 setTextColor:pbproxy_active ?[NSColor controlTextColor] : [
         NSColor disabledControlTextColor]];

    [enable_fullscreen setIntValue:!XQuartzRootlessDefault];
    [enable_fullscreen_menu setIntValue:XQuartzFullscreenMenu];
    [enable_fullscreen_menu setEnabled:!XQuartzRootlessDefault];
    [enable_fullscreen_menu_text setTextColor:XQuartzRootlessDefault ?[
         NSColor disabledControlTextColor] : [NSColor controlTextColor]];

    [prefs_panel makeKeyAndOrderFront:sender];
}

- (IBAction) quit:sender
{
    DarwinSendDDXEvent(kXquartzQuit, 0);
}

- (IBAction) x11_help:sender
{
#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060
    AHLookupAnchor((CFStringRef)NSLocalizedString(@"Mac Help",
                                                  no comment),
                   CFSTR("mchlp2276"));
#else
    AHLookupAnchor(CFSTR("com.apple.machelp"), CFSTR("mchlp2276"));
#endif
}

- (OSX_BOOL) validateMenuItem:(NSMenuItem *)item
{
    NSMenu *menu = [item menu];

    if (item == toggle_fullscreen_item)
        return !XQuartzIsRootless;
    else if (menu == [X11App windowsMenu] || menu == dock_menu
             || (menu == [x11_about_item menu] && [item tag] == 42))
        return (AppleWMSelectedEvents() & AppleWMControllerNotifyMask) != 0;
    else
        return TRUE;
}

- (void) applicationDidHide:(NSNotification *)notify
{
    DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMHideAll);

    /* Toggle off fullscreen mode to leave our non-default video
     * mode and hide our guard window.
     */
    if (!XQuartzIsRootless && XQuartzFullscreenVisible) {
        DarwinSendDDXEvent(kXquartzToggleFullscreen, 0);
    }
}

- (void) applicationDidUnhide:(NSNotification *)notify
{
    DarwinSendDDXEvent(kXquartzControllerNotify, 1, AppleWMShowAll);
}

- (NSApplicationTerminateReply) applicationShouldTerminate:sender
{
    NSString *msg;
    NSString *title;

    if (can_quit ||
        [X11App prefs_get_boolean:@PREFS_NO_QUIT_ALERT default:NO])
        return NSTerminateNow;

    /* Make sure we're frontmost. */
    [NSApp activateIgnoringOtherApps:YES];

    title = NSLocalizedString(@"Do you really want to quit X11?",
                              @"Dialog title when quitting");
    msg = NSLocalizedString(
        @"Any open X11 applications will stop immediately, and you will lose any unsaved changes.",
        @"Dialog when quitting");

    /* FIXME: safe to run the alert in here? Or should we return Later
     *        and then run the alert on a timer? It seems to work here, so..
     */

    return (NSRunAlertPanel(title, msg, NSLocalizedString(@"Quit", @""),
                            NSLocalizedString(@"Cancel", @""), nil)
            == NSAlertDefaultReturn) ? NSTerminateNow : NSTerminateCancel;
}

- (void) applicationWillTerminate:(NSNotification *)aNotification
{
    int remain;
    [X11App prefs_synchronize];

    /* shutdown the X server, it will exit () for us. */
    DarwinSendDDXEvent(kXquartzQuit, 0);

    /* In case it doesn't, exit anyway after a while. */
    remain = 10000000;
    while ((remain = usleep(remain)) > 0) ;

    exit(1);
}

- (void) server_ready
{
    x_list *node;

    finished_launching = YES;

    for (node = pending_apps; node != NULL; node = node->next) {
        NSString *filename = node->data;
        [self launch_client:filename];
        [filename release];
    }

    x_list_free(pending_apps);
    pending_apps = NULL;
}

- (OSX_BOOL) application:(NSApplication *)app openFile:(NSString *)filename
{
    const char *name = [filename UTF8String];

    if (finished_launching)
        [self launch_client:filename];
    else if (name[0] != ':')            /* ignore display names */
        pending_apps = x_list_prepend(pending_apps, [filename retain]);

    /* FIXME: report failures. */
    return YES;
}

@end

void
X11ControllerMain(int argc, char **argv, char **envp)
{
    X11ApplicationMain(argc, argv, envp);
}