blob: 2050340cf51faae85f38735d6e65ea9448b7b325 [file] [log] [blame]
/*
* Copyright (C) 2007 Holger Hans Peter Freyther zecke@selfish.org
* (C) 2009 Kenneth Rohde Christiansen
* (C) 2009 INdT, Instituto Nokia de Technologia
* (C) 2009-2010 ProFUSION embedded systems
* (C) 2009-2010 Samsung Electronics
*
* 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; either
* version 2 of the License, or (at your option) any later version.
*
* 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
*/
#include "config.h"
#include "ScrollbarEfl.h"
#include "Frame.h"
#include "FrameView.h"
#include "GraphicsContext.h"
#include "HostWindow.h"
#include "IntRect.h"
#include "Page.h"
#include "RenderThemeEfl.h"
#include "ScrollbarTheme.h"
#include "Settings.h"
#include <Ecore.h>
#include <Edje.h>
#include <Evas.h>
#include <new>
#include <string>
#include <wtf/text/CString.h>
using namespace std;
using namespace WebCore;
PassRefPtr<Scrollbar> Scrollbar::createNativeScrollbar(ScrollableArea* scrollableArea, ScrollbarOrientation orientation, ScrollbarControlSize size)
{
if (Settings::mockScrollbarsEnabled())
return adoptRef(new Scrollbar(scrollableArea, orientation, size));
return adoptRef(new ScrollbarEfl(scrollableArea, orientation, size));
}
ScrollbarEfl::ScrollbarEfl(ScrollableArea* scrollableArea, ScrollbarOrientation orientation, ScrollbarControlSize controlSize)
: Scrollbar(scrollableArea, orientation, controlSize)
, m_lastPos(0)
, m_lastTotalSize(0)
, m_lastVisibleSize(0)
{
}
ScrollbarEfl::~ScrollbarEfl()
{
if (!evasObject())
return;
evas_object_del(evasObject());
setEvasObject(0);
}
static void scrollbarEflEdjeMessage(void* data, Evas_Object*, Edje_Message_Type messageType, int id, void* message)
{
ScrollbarEfl* that = static_cast<ScrollbarEfl*>(data);
Edje_Message_Float* messageFloat;
int value;
if (!id) {
EINA_LOG_ERR("Unknown message id '%d' from scroll bar theme.", id);
return;
}
if (messageType != EDJE_MESSAGE_FLOAT) {
EINA_LOG_ERR("Message id '%d' of incorrect type from scroll bar theme. "
"Expected '%d', got '%d'.",
id, EDJE_MESSAGE_FLOAT, messageType);
return;
}
messageFloat = static_cast<Edje_Message_Float*>(message);
value = messageFloat->val * (that->totalSize() - that->visibleSize());
that->scrollableArea()->scrollToOffsetWithoutAnimation(that->orientation(), value);
}
void ScrollbarEfl::setParent(ScrollView* view)
{
Evas_Object* object = evasObject();
Widget::setParent(view);
if (!object) {
if (!view || !view->evas() || !view->evasObject())
return;
object = edje_object_add(view->evas());
if (!object) {
EINA_LOG_ERR("Could not create edje object for view=%p (evas=%p)",
view, view->evas());
return;
}
edje_object_message_handler_set(object, scrollbarEflEdjeMessage, this);
setEvasObject(object);
} else if (!view || !view->evas() || !view->evasObject()) {
evas_object_hide(object);
return;
}
Frame* frame = static_cast<FrameView*>(view)->frame();
if (!frame)
return;
Page* page = frame->page();
if (!page)
return;
const char* group = (orientation() == HorizontalScrollbar)
? "scrollbar.horizontal" : "scrollbar.vertical";
String theme = static_cast<RenderThemeEfl*>(page->theme())->themePath();
if (theme.isEmpty()) {
EINA_LOG_ERR("Could not load theme '%s': no theme path set.", group);
evas_object_hide(object);
return;
}
if (!edje_object_file_set(object, theme.utf8().data(), group)) {
Edje_Load_Error err = edje_object_load_error_get(object);
const char* errmessage = edje_load_error_str(err);
EINA_LOG_ERR("Could not load theme '%s' from file '%s': #%d '%s'",
group, theme.utf8().data(), err, errmessage);
return;
}
setPlatformWidget(object);
evas_object_smart_member_add(object, view->evasObject());
evas_object_show(object);
}
void ScrollbarEfl::updateThumbPosition()
{
updateThumbPositionAndProportion();
}
void ScrollbarEfl::updateThumbProportion()
{
updateThumbPositionAndProportion();
}
void ScrollbarEfl::updateThumbPositionAndProportion()
{
if (!platformWidget())
return;
int pos = currentPos();
int tSize = totalSize();
int vSize = visibleSize();
if (m_lastPos == pos
&& m_lastTotalSize == tSize
&& m_lastVisibleSize == vSize)
return;
m_lastPos = pos;
m_lastTotalSize = tSize;
m_lastVisibleSize = vSize;
OwnArrayPtr<char> buffer = adoptArrayPtr(new char[sizeof(Edje_Message_Float_Set) + sizeof(double)]);
Edje_Message_Float_Set* message = new(buffer.get()) Edje_Message_Float_Set;
message->count = 2;
if (tSize - vSize > 0)
message->val[0] = pos / static_cast<float>(tSize - vSize);
else
message->val[0] = 0.0;
if (tSize > 0)
message->val[1] = vSize / static_cast<float>(tSize);
else
message->val[1] = 0.0;
edje_object_message_send(platformWidget(), EDJE_MESSAGE_FLOAT_SET, 0, message);
}
void ScrollbarEfl::setFrameRect(const IntRect& rect)
{
Widget::setFrameRect(rect);
frameRectsChanged();
}
void ScrollbarEfl::frameRectsChanged()
{
Evas_Object* object = platformWidget();
Evas_Coord x, y;
if (!parent() || !object)
return;
IntRect rect = frameRect();
if (parent()->isScrollViewScrollbar(this))
rect.setLocation(parent()->convertToContainingWindow(rect.location()));
else
rect.setLocation(parent()->contentsToWindow(rect.location()));
evas_object_geometry_get(root()->evasObject(), &x, &y, 0, 0);
evas_object_move(object, x + rect.x(), y + rect.y());
evas_object_resize(object, rect.width(), rect.height());
}