/*
* Copyright (C) 2011 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "X11Windowing.h"

#include <stdio.h>
#include <stdlib.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>

#define DEBUG 0
#if DEBUG
#  define D(...) printf(__VA_ARGS__), printf("\n")
#else
#  define D(...) ((void)0)
#endif

/* Try to remember the window position between creates/destroys */
static int X11_wmXPos = 100;
static int X11_wmYPos = 100;

static int X11_wmXAdjust = 0;
static int X11_wmYAdjust = 0;

static void
get_window_pos( Display *disp, Window win, int *px, int *py )
{
    Window  child;

    XTranslateCoordinates( disp, win, DefaultRootWindow(disp),  0, 0, px, py, &child );
}


static void
set_window_pos(Display *disp, Window win, int x, int y)
{
    int   xNew, yNew;
    int   xAdjust = X11_wmXAdjust;
    int   yAdjust = X11_wmYAdjust;

    /* this code is tricky because some window managers, but not all,
     * will translate the final window position by a given offset
     * corresponding to the frame decoration.
     *
     * so we first try to move the window, get the position that the
     * window manager has set, and if they are different, re-position the
     * window again with an adjustment.
     *
     * this causes a slight flicker since the window 'jumps' very
     * quickly from one position to the other.
     */

    D("%s: move to [%d,%d] adjusted to [%d,%d]", __FUNCTION__,
      x, y, x+xAdjust, y+yAdjust);
    XMoveWindow(disp, win, x + xAdjust, y + yAdjust);
    XSync(disp, True);
    get_window_pos(disp, win, &xNew, &yNew);
    if (xNew != x || yNew != y) {
        X11_wmXAdjust = xAdjust = x - xNew;
        X11_wmYAdjust = yAdjust = y - yNew;
        D("%s: read pos [%d,%d], recomputing adjust=[%d,%d] moving to [%d,%d]\n",
          __FUNCTION__, xNew, yNew, xAdjust, yAdjust, x+xAdjust, y+yAdjust);
        XMoveWindow(disp, win, x + xAdjust, y + yAdjust );
    }
    XSync(disp, False);
}


NativeDisplayType X11Windowing::getNativeDisplay()
{
    Display *dpy = XOpenDisplay(NULL);
    return (NativeDisplayType)dpy;
}

NativeWindowType X11Windowing::createNativeWindow(NativeDisplayType _dpy, int width, int height)
{
    Display *dpy = (Display *) _dpy;

    long defaultScreen = DefaultScreen( dpy );
    Window rootWindow = RootWindow(dpy, defaultScreen);
    int depth = DefaultDepth(dpy, defaultScreen);
    XVisualInfo *visualInfo = new XVisualInfo;

    XMatchVisualInfo(dpy, defaultScreen, depth, TrueColor, visualInfo);
    if (visualInfo == NULL) {
        fprintf(stderr, "couldn't find matching visual\n");
        return NULL;
    }

    Colormap x11Colormap = XCreateColormap(dpy, rootWindow, visualInfo->visual, AllocNone);
    XSetWindowAttributes sWA;
    sWA.colormap = x11Colormap;
    sWA.event_mask = StructureNotifyMask | ExposureMask;
    sWA.background_pixel = 0;
    sWA.border_pixel = 0;
    unsigned int attributes_mask = CWBackPixel | CWBorderPixel | CWEventMask | CWColormap;

    Window win = XCreateWindow( dpy,
                                rootWindow,
                                X11_wmXPos, X11_wmYPos, width, height,
                                0, CopyFromParent, InputOutput,
                                CopyFromParent, attributes_mask, &sWA);

    XMapWindow(dpy, win);
    XFlush(dpy);
    set_window_pos(dpy, win, X11_wmXPos, X11_wmYPos);
    return NativeWindowType(win);
}

int X11Windowing::destroyNativeWindow(NativeDisplayType _dpy, NativeWindowType _win)
{
    Display *dpy = (Display *)_dpy;
    Window win = (Window)_win;
    get_window_pos(dpy, win, &X11_wmXPos, &X11_wmYPos);
    D("%s: Saved window position [%d, %d]\n", __FUNCTION__, X11_wmXPos, X11_wmYPos);
    XDestroyWindow(dpy, win);
    XFlush(dpy);
    return 0;
}