/**
 * @file llresizehandle.cpp
 * @brief LLResizeHandle base class
 *
 * $LicenseInfo:firstyear=2001&license=viewerlgpl$
 * Second Life Viewer Source Code
 * Copyright (C) 2010, Linden Research, Inc.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation;
 * version 2.1 of the License only.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 * $/LicenseInfo$
 */

#include "linden_common.h"

#include "llresizehandle.h"

#include "llfocusmgr.h"
#include "llmath.h"
#include "llui.h"
#include "llmenugl.h"
#include "llcontrol.h"
#include "llfloater.h"
#include "llwindow.h"

const S32 RESIZE_BORDER_WIDTH = 3;

LLResizeHandle::Params::Params()
:   corner("corner"),
    min_width("min_width"),
    min_height("min_height")
{
    name = "resize_handle";
}

LLResizeHandle::LLResizeHandle(const LLResizeHandle::Params& p)
:   LLView(p),
    mDragLastScreenX( 0 ),
    mDragLastScreenY( 0 ),
    mLastMouseScreenX( 0 ),
    mLastMouseScreenY( 0 ),
    mImage( NULL ),
    mMinWidth( p.min_width ),
    mMinHeight( p.min_height ),
    mCorner( p.corner )
{
    if( RIGHT_BOTTOM == mCorner)
    {
        mImage = LLUI::getUIImage("Resize_Corner");
    }
    switch( p.corner )
    {
        case LEFT_TOP:      setFollows( FOLLOWS_LEFT | FOLLOWS_TOP );       break;
        case LEFT_BOTTOM:   setFollows( FOLLOWS_LEFT | FOLLOWS_BOTTOM );    break;
        case RIGHT_TOP:     setFollows( FOLLOWS_RIGHT | FOLLOWS_TOP );      break;
        case RIGHT_BOTTOM:  setFollows( FOLLOWS_RIGHT | FOLLOWS_BOTTOM );   break;
    }
}

LLResizeHandle::~LLResizeHandle()
{
    gFocusMgr.removeKeyboardFocusWithoutCallback(this);
}


BOOL LLResizeHandle::handleMouseDown(S32 x, S32 y, MASK mask)
{
    BOOL handled = FALSE;
    if( pointInHandle(x, y) )
    {
        handled = TRUE;
        // Route future Mouse messages here preemptively.  (Release on mouse up.)
        // No handler needed for focus lost since this clas has no state that depends on it.
        gFocusMgr.setMouseCapture( this );

        localPointToScreen(x, y, &mDragLastScreenX, &mDragLastScreenY);
        mLastMouseScreenX = mDragLastScreenX;
        mLastMouseScreenY = mDragLastScreenY;
    }

    return handled;
}


BOOL LLResizeHandle::handleMouseUp(S32 x, S32 y, MASK mask)
{
    BOOL    handled = FALSE;

    if( hasMouseCapture() )
    {
        // Release the mouse
        gFocusMgr.setMouseCapture( NULL );
        handled = TRUE;
    }
    else if( pointInHandle(x, y) )
    {
        handled = TRUE;
    }

    return handled;
}


BOOL LLResizeHandle::handleHover(S32 x, S32 y, MASK mask)
{
    BOOL    handled = FALSE;

    // We only handle the click if the click both started and ended within us
    if( hasMouseCapture() )
    {
        // Make sure the mouse in still over the application.  We don't want to make the parent
        // so big that we can't see the resize handle any more.

        S32 screen_x;
        S32 screen_y;
        localPointToScreen(x, y, &screen_x, &screen_y);
        const LLRect valid_rect = getRootView()->getRect();
        screen_x = llclamp( screen_x, valid_rect.mLeft, valid_rect.mRight );
        screen_y = llclamp( screen_y, valid_rect.mBottom, valid_rect.mTop );

        LLView* resizing_view = getParent();
        if( resizing_view )
        {
            // undock floater when user resize it
            LLFloater* floater_parent = dynamic_cast<LLFloater*>(getParent());
            if (floater_parent && floater_parent->isDocked())
            {
                floater_parent->setDocked(false, false);
            }

            // Resize the parent
            LLRect orig_rect = resizing_view->getRect();
            LLRect scaled_rect = orig_rect;
            S32 delta_x = screen_x - mDragLastScreenX;
            S32 delta_y = screen_y - mDragLastScreenY;
            LLCoordGL mouse_dir;
            // use hysteresis on mouse motion to preserve user intent when mouse stops moving
            mouse_dir.mX = (screen_x == mLastMouseScreenX) ? mLastMouseDir.mX : screen_x - mLastMouseScreenX;
            mouse_dir.mY = (screen_y == mLastMouseScreenY) ? mLastMouseDir.mY : screen_y - mLastMouseScreenY;
            mLastMouseScreenX = screen_x;
            mLastMouseScreenY = screen_y;
            mLastMouseDir = mouse_dir;

            S32 x_multiple = 1;
            S32 y_multiple = 1;
            switch( mCorner )
            {
            case LEFT_TOP:
                x_multiple = -1;
                y_multiple =  1;
                break;
            case LEFT_BOTTOM:
                x_multiple = -1;
                y_multiple = -1;
                break;
            case RIGHT_TOP:
                x_multiple =  1;
                y_multiple =  1;
                break;
            case RIGHT_BOTTOM:
                x_multiple =  1;
                y_multiple = -1;
                break;
            }

            S32 new_width = orig_rect.getWidth() + x_multiple * delta_x;
            if( new_width < mMinWidth )
            {
                new_width = mMinWidth;
                delta_x = x_multiple * (mMinWidth - orig_rect.getWidth());
            }

            S32 new_height = orig_rect.getHeight() + y_multiple * delta_y;
            if( new_height < mMinHeight )
            {
                new_height = mMinHeight;
                delta_y = y_multiple * (mMinHeight - orig_rect.getHeight());
            }

            switch( mCorner )
            {
            case LEFT_TOP:
                scaled_rect.translate(delta_x, 0);
                break;
            case LEFT_BOTTOM:
                scaled_rect.translate(delta_x, delta_y);
                break;
            case RIGHT_TOP:
                break;
            case RIGHT_BOTTOM:
                scaled_rect.translate(0, delta_y);
                break;
            }

            // temporarily set new parent rect
            scaled_rect.mRight = scaled_rect.mLeft + new_width;
            scaled_rect.mTop = scaled_rect.mBottom + new_height;
            resizing_view->setRect(scaled_rect);

            LLView* snap_view = NULL;
            LLView* test_view = NULL;

            static LLUICachedControl<S32> snap_margin ("SnapMargin", 0);
            // now do snapping
            switch(mCorner)
            {
            case LEFT_TOP:
                snap_view = resizing_view->findSnapEdge(scaled_rect.mLeft, mouse_dir, SNAP_LEFT, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                test_view = resizing_view->findSnapEdge(scaled_rect.mTop, mouse_dir, SNAP_TOP, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                if (!snap_view)
                {
                    snap_view = test_view;
                }
                break;
            case LEFT_BOTTOM:
                snap_view = resizing_view->findSnapEdge(scaled_rect.mLeft, mouse_dir, SNAP_LEFT, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                test_view = resizing_view->findSnapEdge(scaled_rect.mBottom, mouse_dir, SNAP_BOTTOM, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                if (!snap_view)
                {
                    snap_view = test_view;
                }
                break;
            case RIGHT_TOP:
                snap_view = resizing_view->findSnapEdge(scaled_rect.mRight, mouse_dir, SNAP_RIGHT, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                test_view = resizing_view->findSnapEdge(scaled_rect.mTop, mouse_dir, SNAP_TOP, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                if (!snap_view)
                {
                    snap_view = test_view;
                }
                break;
            case RIGHT_BOTTOM:
                snap_view = resizing_view->findSnapEdge(scaled_rect.mRight, mouse_dir, SNAP_RIGHT, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                test_view = resizing_view->findSnapEdge(scaled_rect.mBottom, mouse_dir, SNAP_BOTTOM, SNAP_PARENT_AND_SIBLINGS, snap_margin);
                if (!snap_view)
                {
                    snap_view = test_view;
                }
                break;
            }

            // register "snap" behavior with snapped view
            resizing_view->setSnappedTo(snap_view);

            // reset parent rect
            resizing_view->setRect(orig_rect);

            // translate and scale to new shape
            resizing_view->setShape(scaled_rect, true);

            // update last valid mouse cursor position based on resized view's actual size
            LLRect new_rect = resizing_view->getRect();
            S32 actual_delta_x = 0;
            S32 actual_delta_y = 0;
            switch(mCorner)
            {
            case LEFT_TOP:
                actual_delta_x = new_rect.mLeft - orig_rect.mLeft;
                actual_delta_y = new_rect.mTop - orig_rect.mTop;
                if (actual_delta_x != delta_x
                    || actual_delta_y != delta_y)
                {
                    new_rect.mRight = orig_rect.mRight;
                    new_rect.mBottom = orig_rect.mBottom;
                    resizing_view->setShape(new_rect, true);
                }

                mDragLastScreenX += actual_delta_x;
                mDragLastScreenY += actual_delta_y;
                break;
            case LEFT_BOTTOM:
                actual_delta_x = new_rect.mLeft - orig_rect.mLeft;
                actual_delta_y = new_rect.mBottom - orig_rect.mBottom;
                if (actual_delta_x != delta_x
                    || actual_delta_y != delta_y)
                {
                    new_rect.mRight = orig_rect.mRight;
                    new_rect.mTop = orig_rect.mTop;
                    resizing_view->setShape(new_rect, true);
                }

                mDragLastScreenX += actual_delta_x;
                mDragLastScreenY += actual_delta_y;
                break;
            case RIGHT_TOP:
                actual_delta_x = new_rect.mRight - orig_rect.mRight;
                actual_delta_y = new_rect.mTop - orig_rect.mTop;
                if (actual_delta_x != delta_x
                    || actual_delta_y != delta_y)
                {
                    new_rect.mLeft = orig_rect.mLeft;
                    new_rect.mBottom = orig_rect.mBottom;
                    resizing_view->setShape(new_rect, true);
                }

                mDragLastScreenX += actual_delta_x;
                mDragLastScreenY += actual_delta_y;
                break;
            case RIGHT_BOTTOM:
                actual_delta_x = new_rect.mRight - orig_rect.mRight;
                actual_delta_y = new_rect.mBottom - orig_rect.mBottom;
                if (actual_delta_x != delta_x
                    || actual_delta_y != delta_y)
                {
                    new_rect.mLeft = orig_rect.mLeft;
                    new_rect.mTop = orig_rect.mTop;
                    resizing_view->setShape(new_rect, true);
                }

                mDragLastScreenX += actual_delta_x;
                mDragLastScreenY += actual_delta_y;
                break;
            default:
                break;
            }
        }

        handled = TRUE;
    }
    else // don't have mouse capture
    {
        if( pointInHandle( x, y ) )
        {
            handled = TRUE;
        }
    }

    if( handled )
    {
        switch( mCorner )
        {
        case RIGHT_BOTTOM:
        case LEFT_TOP:
            getWindow()->setCursor(UI_CURSOR_SIZENWSE);
            break;
        case LEFT_BOTTOM:
        case RIGHT_TOP:
            getWindow()->setCursor(UI_CURSOR_SIZENESW);
            break;
        }
    }

    return handled;
} // end handleHover


// assumes GL state is set for 2D
void LLResizeHandle::draw()
{
    if( mImage.notNull() && getVisible() && (RIGHT_BOTTOM == mCorner) )
    {
        mImage->draw(0, 0);
    }
}


BOOL LLResizeHandle::pointInHandle( S32 x, S32 y )
{
    if( pointInView(x, y) )
    {
        const S32 TOP_BORDER = (getRect().getHeight() - RESIZE_BORDER_WIDTH);
        const S32 RIGHT_BORDER = (getRect().getWidth() - RESIZE_BORDER_WIDTH);

        switch( mCorner )
        {
        case LEFT_TOP:      return (x <= RESIZE_BORDER_WIDTH) || (y >= TOP_BORDER);
        case LEFT_BOTTOM:   return (x <= RESIZE_BORDER_WIDTH) || (y <= RESIZE_BORDER_WIDTH);
        case RIGHT_TOP:     return (x >= RIGHT_BORDER) || (y >= TOP_BORDER);
        case RIGHT_BOTTOM:  return TRUE;
        }
    }
    return FALSE;
}