/*
 * upclick(params)
 *
 *  parameters:
 *      element:        DOM object
 *      action:         Server script who receive file
 *      action_params:  Server script parameters. Array: key=value
 *      dataname:       File data name. Default: Filedata
 *      maxsize:        Maximum file size (in Bytes). 0 - unlimited
 *      target:         Response target name: '_new', '_blank',... Default: <Hidden frame name>
 *      zindex:         z-index listener
 *      onstart:        Callback function
 *                        onstart(filename).
 *                        Emited when file started upload
 *      oncomplete:      Callback function
 *                        oncomplete(server_response_data).
 *                        Emited when file successfully onloaded
 */
function upclick(params)
{
    // Parse params
    var defaults =
        {
            element: null,
            action: 'about:blank',
            action_params: {},
            auto_submit: true,
            maxsize: 0,
            onstart: null,
            oncomplete: null,
            dataname: 'Filedata',
            target: null,
            zindex: 'auto'
        };

    for(var key in defaults)
    {
        params[key] = params[key] ? params[key] : defaults[key];
    }

    var element = params['element'];
    if (typeof element == 'string')
        element = document.getElementById(element);

    var doc = element.ownerDocument;
    var input;

    // div
    var container = doc.createElement("div");

    // frame -> div
    var frame_name = 'frame' + new Date().getTime().toString().substr(8);

    // IE require such creation method
    container.innerHTML =
        '<iframe name="'+frame_name+'" src="about:blank" onload="this.onload_callback()"></iframe>';
    var frame = container.childNodes[0];

    // Callback for 'onload' event. Fired when form submited and data retrived
    frame.onload_callback =
        function () {
            // Phase 1. First 'onload' when element created
            // form -> div
            var form = doc.createElement('form');
            container.appendChild(form);
            form.method = 'post';
            form.enctype = 'multipart/form-data';
            form.encoding = 'multipart/form-data';
            if (params['target']) {
                form.target = params['target'];
                form.setAttribute('target', params['target']);
            }
            else {
                form.target = frame_name;
                form.setAttribute('target', frame_name);
            }
            form.action = params['action'];
            form.setAttribute('action', params['action']);
            form.style.margin = 0;
            form.style.padding = 0;
            form.style.height = '80px';
            form.style.width = '40px';
            form.runat = 'server';

            // append params in form
            var action_params = params['action_params'];
            for (var key in action_params) {
                var hidden = doc.createElement("input");
                hidden.type = "hidden";
                hidden.name = key;
                hidden.value = String(action_params[key]);
                form.appendChild(hidden);
            }

            // MAX_FILESIZE. Maximum file size
            if (params['maxsize']) {
                var input_ms = doc.createElement('input');
                input_ms.type = 'hidden';
                input_ms.name = 'MAX_FILE_SIZE';
                input_ms.value = String(params['maxsize']);
                form.appendChild(input_ms);
            }

            // input -> form
            input = doc.createElement("input");
            input.name = params['dataname'];
            input.type = 'file';
            input.multiple = true;
            input.size = '1';
            input.runat = 'server';
            form.appendChild(input);

            // input style
            input.style.position = 'absolute';
            input.style.display = 'block';
            input.style.top = 0;
            input.style.left = 0;
            input.style.height = form.style.height;
            input.style.width = '80px';
            input.style.opacity = 0;
            input.style.filter = 'alpha(opacity=0)';
            input.style.fontSize = 8;
            input.style.zIndex = 1;
            input.style.visiblity = 'hidden';
            input.style.marginLeft = '-40px'; // hide IE text field
            //input.style.cursor = 'pointer';

            // input click handler (enable container event listener)

            // 'change' event handler. Submit form
            var onchange_callback =
                function (e) {
                    // empty filename check
                    if (!input.value)
                        return;
                    var auto_submit = params['auto_submit'];
                    // Run onstart callback. When upload started
                    var onstart = params['onstart'];
                    if (onstart)
                        auto_submit = onstart(input);

                    if (auto_submit) {
                        form.submit();
                    }
                };

            // bind 'change' callback
            // DOM2: FF, Opera, Chrome
            if (input.addEventListener)
                input.addEventListener('change', onchange_callback, false);

            // IE 5+
            else if (input.attachEvent) {
                input.attachEvent(
                    'onpropertychange',
                    function (e) {
                        // Get event details for IE
                        if (!e)
                            e = window.event;

                        if (e.propertyName == 'value')
                            onchange_callback();
                    }
                    );
            }
            // IE 4
            else
                input.onpropertychange = onchange_callback;


            // Phase 2. Next 'onload' when data received from server
            frame.onload_callback =
            function () {
                var ifwin = null;

                // Get frame window
                // IE5.5+, Mozilla, NN7
               if (frame.contentWindow)
                    ifwin = frame.contentWindow;
                // NN6, Konqueror
                else if (frame.contentDocument)
                      ifwin = frame.contentDocument.defaultView;
               /* if (frame.contentDocument)
                    ifwin = frame.contentDocument.defaultView;
                else if (frame.contentWindow)
                    ifwin = frame.contentWindow;*/

                // Run 'oncomplete' callback                
                //var data = ifwin.document.body.innerHTML;
                var oncomplete = params['oncomplete'];
                if (oncomplete)
                    oncomplete(ifwin.document);

                // Clear filename
                form.reset();
            }
        };

    // frame style
    frame.style.display = 'none';
    frame.width = 0;
    frame.height = 0;
    frame.marginHeight = 0;
    frame.marginWidth = 0;

    // container -> DOM
    doc.body.insertBefore(container, doc.body.firstChild);

    // container style
    container.style.position = 'absolute';
    container.style.overflow = 'hidden';
    container.style.padding = 0;
    container.style.margin = 0;
    container.style.visiblity = 'hidden';
    container.style.width = '0px';
    container.style.height = '0px';
    //container.style.cursor = 'pointer'; // FIXME flashed on IE

    // zindex detection
    if (params['zindex'] == 'auto')
    {
        var zi=0, ziparsed;
        var obj = element;
        var comp;

        while (obj.tagName != 'BODY')
        {
            comp = obj.currentStyle ? obj.currentStyle : getComputedStyle(obj, null);
            ziparsed = parseInt(comp.zIndex);
            ziparsed = isNaN(ziparsed) ? 0 : ziparsed;
            zi += ziparsed + 1;
            obj = obj.parentNode
        }

        container.style.zIndex = zi;
    }
    else
    {
        container.style.zIndex = params['zindex'];
    }

    // If cursor out of element => shitch off listener
  /*  var onmouseout_callback =
    function(e)
    {
        if (!e)
            e = window.event;

        container.style.width = '0px';
        container.style.height = '0px';
        var receiver = doc.elementFromPoint(e.clientX, e.clientY);

        if (receiver === element)
        {
            container.style.width = '40px';
            container.style.height = '80px';
        }
    }
    // DOM2: FF, Chrome, Opera
    if (container.addEventListener)
        container.addEventListener('mousemove', onmouseout_callback, false);
     // IE 5+
    else if (container.attachEvent)
        container.attachEvent("onmousemove", onmouseout_callback);*/
// this uses jquery but it is much smoother!
    $(container).mouseout(function(e) {
        if (!e)
            e = window.event;

        container.style.width = '0px';
        container.style.height = '0px';
        var receiver = doc.elementFromPoint(e.clientX, e.clientY);

        if (receiver === element) {
            container.style.width = '40px';
            container.style.height = '80px';
        } 
    });


    // Move the input with the mouse to make sure it get clicked!
    var onmousemove_callback =
        function(e)
        {
            // Get event details for IE
            if (!e)
                e = window.event;

            // find element position,
            var x = y = 0;

            if (e.pageX)
                x = e.pageX;
            else if (e.clientX)
               x = e.clientX +
                (doc.documentElement.scrollLeft ?
                    doc.documentElement.scrollLeft :
                    doc.body.scrollLeft);

            if (e.pageY)
                y = e.pageY;
            else if (e.clientY)
               y = e.clientY +
                    (doc.documentElement.scrollTop ?
                        doc.documentElement.scrollTop :
                        doc.body.scrollTop);

            // move listener
            container.style.left = x - 20 + 'px';
            container.style.top = y - 40 + 'px';
            container.style.width = '40px';
            container.style.height = '80px';
        };

    // bind mousemove callback (for place button under cursor)
    // DOM2: FF, Chrome, Opera
    if (element.addEventListener)
        element.addEventListener('mousemove', onmousemove_callback, false);
     // IE 5+
    else if (element.attachEvent)
        element.attachEvent("onmousemove", onmousemove_callback);
}

