// WinClient.cc for Fluxbox - an X11 Window manager
// Copyright (c) 2003 Henrik Kinnunen (fluxgen(at)users.sourceforge.net)
//
// 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 AUTHORS OR COPYRIGHT HOLDERS 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.

// $Id: WinClient.cc,v 1.4 2003/04/25 11:21:17 fluxgen Exp $

#include "WinClient.hh"

#include "Window.hh"
#include "fluxbox.hh"
#include "Screen.hh"
#include "i18n.hh"
#include "FbAtoms.hh"
#include "EventManager.hh"

#include <iostream>
#include <algorithm>
#include <iterator>
using namespace std;

WinClient::WinClient(Window win, FluxboxWindow &fbwin):FbTk::FbWindow(win),
                     transient_for(0),
                     window_group(0),
                     x(0), y(0), old_bw(0),
                     min_width(1), min_height(1),
                     max_width(1), max_height(1),
                     width_inc(1), height_inc(1),
                     min_aspect_x(1), min_aspect_y(1),
                     max_aspect_x(1), max_aspect_y(1),
                     base_width(1), base_height(1),
                     win_gravity(0),
                     initial_state(0),
                     normal_hint_flags(0),
                     wm_hint_flags(0),
                     mwm_hint(0),
                     blackbox_hint(0),
                     m_win(&fbwin),
                     m_title(""), m_icon_title(""),
                     m_diesig(*this) { }

WinClient::~WinClient() {
#ifdef DEBUG
    cerr<<__FILE__<<"(~"<<__FUNCTION__<<")[this="<<this<<"]"<<endl;
#endif // DEBUG

    m_diesig.notify();

    Fluxbox *fluxbox = Fluxbox::instance();

    if (transient_for != 0) {
        if (transientFor() == m_win) {
            transient_for = 0;
        }

        fluxbox->setFocusedWindow(transient_for);

        if (transient_for != 0) {
            FluxboxWindow::ClientList::iterator client_it = 
                transientFor()->clientList().begin();
            FluxboxWindow::ClientList::iterator client_it_end = 
                transientFor()->clientList().end();
            for (; client_it != client_it_end; ++client_it) {
                (*client_it)->transientList().remove(m_win);
            }

            transient_for->setInputFocus();
            transient_for = 0;
        }
    }
	
    while (!transients.empty()) {
        FluxboxWindow::ClientList::iterator it = 
            transients.back()->clientList().begin();
        FluxboxWindow::ClientList::iterator it_end = 
            transients.back()->clientList().end();
        for (; it != it_end; ++it) {
            if ((*it)->transientFor() == m_win)
                (*it)->transient_for = 0;
        }

        transients.pop_back();
    }
	
    if (window_group != 0) {
        fluxbox->removeGroupSearch(window_group);
        window_group = 0;
    }

    if (mwm_hint != 0)
        XFree(mwm_hint);

    if (blackbox_hint != 0)
        XFree(blackbox_hint);

    if (window())
        fluxbox->removeWindowSearch(window());

    if (m_win != 0)
        m_win->removeClient(*this);
    FbTk::EventManager::instance()->remove(window());
    m_win = 0;

}

void WinClient::updateRect(int x, int y, 
                        unsigned int width, unsigned int height) {
    Display *disp = FbTk::App::instance()->display();
    XEvent event;
    event.type = ConfigureNotify;

    event.xconfigure.display = disp;
    event.xconfigure.event = window();
    event.xconfigure.window = window();
    event.xconfigure.x = x;
    event.xconfigure.y = y;
    event.xconfigure.width = width;
    event.xconfigure.height = height;
    //!! TODO
    event.xconfigure.border_width = 1;//client.old_bw;
    //!! TODO
    event.xconfigure.above = None; //m_frame.window().window();
    event.xconfigure.override_redirect = false;

    XSendEvent(disp, window(), False, StructureNotifyMask, &event);

}

void WinClient::sendFocus() {
    Display *disp = FbTk::App::instance()->display();
    // setup focus msg
    XEvent ce;
    ce.xclient.type = ClientMessage;
    ce.xclient.message_type = FbAtoms::instance()->getWMProtocolsAtom();
    ce.xclient.display = disp;
    ce.xclient.window = window();
    ce.xclient.format = 32;
    ce.xclient.data.l[0] = FbAtoms::instance()->getWMTakeFocusAtom();
    ce.xclient.data.l[1] = Fluxbox::instance()->getLastTime();
    ce.xclient.data.l[2] = 0l;
    ce.xclient.data.l[3] = 0l;
    ce.xclient.data.l[4] = 0l;
    // send focus msg
    XSendEvent(disp, window(), false, NoEventMask, &ce);
}

void WinClient::sendClose() {
    Display *disp = FbTk::App::instance()->display();
    // fill in XClientMessage structure for delete message
    XEvent ce;
    ce.xclient.type = ClientMessage;
    ce.xclient.message_type = FbAtoms::instance()->getWMProtocolsAtom();
    ce.xclient.display = disp;
    ce.xclient.window = window();
    ce.xclient.format = 32;
    ce.xclient.data.l[0] = FbAtoms::instance()->getWMDeleteAtom();
    ce.xclient.data.l[1] = CurrentTime;
    ce.xclient.data.l[2] = 0l;
    ce.xclient.data.l[3] = 0l;
    ce.xclient.data.l[4] = 0l;
    // send event delete message to client window
    XSendEvent(disp, window(), false, NoEventMask, &ce);
}

void WinClient::reparent(Window win, int x, int y) {
    XReparentWindow(FbTk::App::instance()->display(), window(), win, x, y);
}

bool WinClient::getAttrib(XWindowAttributes &attr) const {
    return XGetWindowAttributes(FbTk::App::instance()->display(), window(), &attr);
}

bool WinClient::getWMName(XTextProperty &textprop) const {
    return XGetWMName(FbTk::App::instance()->display(), window(), &textprop);
}

bool WinClient::getWMIconName(XTextProperty &textprop) const {
    return XGetWMName(FbTk::App::instance()->display(), window(), &textprop);
}

void WinClient::updateTransientInfo() {
    if (m_win == 0)
        return;
    // remove us from parent
    if (transientFor() != 0) {
        //!! TODO
        // since we don't know which client in transientFor()
        // that we're transient for then we just remove us 
        // from every client in transientFor() clientlist
        FluxboxWindow::ClientList::iterator client_it = 
            transientFor()->clientList().begin();
        FluxboxWindow::ClientList::iterator client_it_end = 
            transientFor()->clientList().end();
        for (; client_it != client_it_end; ++client_it) {
            (*client_it)->transientList().remove(m_win);
        }
    }
    
    transient_for = 0;
    Display *disp = FbTk::App::instance()->display();
    // determine if this is a transient window
    Window win;
    if (!XGetTransientForHint(disp, window(), &win))
        return;

    // we can't be transient to ourself
    if (win == window())
        return;
	
    if (win != 0 && m_win->getScreen().getRootWindow() == win) {
        m_win->modal = true;
        return;
    }

    transient_for = Fluxbox::instance()->searchWindow(win);
    if (transient_for != 0 &&
        window_group != None && win == window_group) {
        transient_for = Fluxbox::instance()->searchGroup(win, m_win);
    }
	
    // make sure we don't have deadlock loop in transient chain
    for (FluxboxWindow *w = m_win; w != 0; w = w->m_client->transient_for) {
        if (w == w->m_client->transient_for) {
            w->m_client->transient_for = 0;
            break;
        }
    }

    if (transientFor() != 0) {
        // we need to add ourself to the right client in
        // the transientFor() window so we search client
        WinClient *client = transientFor()->findClient(win);
        assert(client != 0);
        client->transientList().push_back(m_win);
        // make sure we only have on instance of this
        client->transientList().unique(); 
        if (transientFor()->isStuck())
            m_win->stick();       
    }
}


void WinClient::updateTitle() {
    XTextProperty text_prop;
    char **list = 0;
    int num = 0;
    I18n *i18n = I18n::instance();

    if (getWMName(text_prop)) {
        if (text_prop.value && text_prop.nitems > 0) {
            if (text_prop.encoding != XA_STRING) {
				
                text_prop.nitems = strlen((char *) text_prop.value);
				
                if (XmbTextPropertyToTextList(FbTk::App::instance()->display(), &text_prop,
                                              &list, &num) == Success &&
                    num > 0 && *list) {
                    m_title = static_cast<char *>(*list);
                    XFreeStringList(list);
                } else
                    m_title = (char *)text_prop.value;
					
            } else
                m_title = (char *)text_prop.value;
            XFree((char *) text_prop.value);
        } else { // ok, we don't have a name, set default name
            m_title = i18n->getMessage(
                                       FBNLS::WindowSet, FBNLS::WindowUnnamed,
                                       "Unnamed");
        }
    } else {
        m_title = i18n->getMessage(
                                   FBNLS::WindowSet, FBNLS::WindowUnnamed,
                                   "Unnamed");
    }

}

void WinClient::updateIconTitle() {
    XTextProperty text_prop;
    char **list = 0;
    int num = 0;
 
    if (getWMIconName(text_prop)) {
        if (text_prop.value && text_prop.nitems > 0) {
            if (text_prop.encoding != XA_STRING) {
                text_prop.nitems = strlen((char *) text_prop.value);

                if (XmbTextPropertyToTextList(FbTk::App::instance()->display(), &text_prop,
                                               &list, &num) == Success &&
                    num > 0 && *list) {
                    m_icon_title = (char *)*list;
                    XFreeStringList(list);
                } else
                    m_icon_title = (char *)text_prop.value;
            } else
                m_icon_title = (char *)text_prop.value;

            XFree((char *) text_prop.value);
        } else
            m_icon_title = title();
    } else
        m_icon_title = title();

}