/* Copyright (C) 2009-2010 ProFUSION embedded systems Copyright (C) 2009-2010 Samsung Electronics This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ // Uncomment to view frame regions and debug messages // #define EWK_FRAME_DEBUG #include "config.h" #include "ewk_frame.h" #include "DocumentLoader.h" #include "DocumentMarkerController.h" #include "Editor.h" #include "EventHandler.h" #include "FocusController.h" #include "FrameLoadRequest.h" #include "FrameLoader.h" #include "FrameLoaderClientEfl.h" #include "FrameSelection.h" #include "FrameView.h" #include "HTMLCollection.h" #include "HTMLHeadElement.h" #include "HTMLImageElement.h" #include "HTMLNames.h" #include "HTMLPlugInElement.h" #include "HistoryItem.h" #include "HitTestRequest.h" #include "HitTestResult.h" #include "IntSize.h" #include "KURL.h" #include "PlatformEvent.h" #include "PlatformKeyboardEvent.h" #include "PlatformMessagePortChannel.h" #include "PlatformMouseEvent.h" #include "PlatformTouchEvent.h" #include "PlatformWheelEvent.h" #include "ProgressTracker.h" #include "ResourceRequest.h" #include "ScriptController.h" #include "ScriptValue.h" #include "SharedBuffer.h" #include "SubstituteData.h" #include "WindowsKeyboardCodes.h" #include "ewk_frame_private.h" #include "ewk_private.h" #include "ewk_security_origin_private.h" #include "ewk_touch_event_private.h" #include "ewk_view_private.h" #include #include #include #include #include #include #include #include #include static const char EWK_FRAME_TYPE_STR[] = "EWK_Frame"; struct Ewk_Frame_Smart_Data { Evas_Object_Smart_Clipped_Data base; Evas_Object* self; Evas_Object* view; #ifdef EWK_FRAME_DEBUG Evas_Object* region; #endif WebCore::Frame* frame; Ewk_Text_With_Direction title; const char* uri; const char* name; bool editable : 1; bool hasDisplayedMixedContent : 1; bool hasRunMixedContent : 1; }; struct Eina_Iterator_Ewk_Frame { Eina_Iterator base; Evas_Object* object; unsigned currentIndex; }; #ifndef EWK_TYPE_CHECK #define EWK_FRAME_TYPE_CHECK(ewkFrame, ...) do { } while (0) #else #define EWK_FRAME_TYPE_CHECK(ewkFrame, ...) \ do { \ const char* _tmp_otype = evas_object_type_get(ewkFrame); \ if (EINA_UNLIKELY(_tmp_otype != EWK_FRAME_TYPE_STR)) { \ EINA_LOG_CRIT \ ("%p (%s) is not of an ewk_frame!", ewkFrame, \ _tmp_otype ? _tmp_otype : "(null)"); \ return __VA_ARGS__; \ } \ } while (0) #endif #define EWK_FRAME_SD_GET(ewkFrame, pointer) \ Ewk_Frame_Smart_Data* pointer = static_cast(evas_object_smart_data_get(ewkFrame)) #define EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, pointer, ...) \ EWK_FRAME_TYPE_CHECK(ewkFrame, __VA_ARGS__); \ EWK_FRAME_SD_GET(ewkFrame, pointer); \ if (!pointer) { \ CRITICAL("no smart data for object %p (%s)", \ ewkFrame, evas_object_type_get(ewkFrame)); \ return __VA_ARGS__; \ } static Evas_Smart_Class _parent_sc = EVAS_SMART_CLASS_INIT_NULL; #ifdef EWK_FRAME_DEBUG static inline void _ewk_frame_debug(Evas_Object* ewkFrame) { Evas_Object* clip, * parent; Evas_Coord x, y, width, height, contentX, contentY, contentWidth, contentHeight; int red, green, blue, alpha, contentRed, contentGreen, contentBlue, contentAlpha; evas_object_color_get(ewkFrame, &red, &green, &blue, &alpha); evas_object_geometry_get(ewkFrame, &x, &y, &width, &height); clip = evas_object_clip_get(ewkFrame); evas_object_color_get(clip, &contentRed, &contentGreen, &contentBlue, &contentAlpha); evas_object_geometry_get(clip, &contentX, &contentY, &contentWidth, &contentHeight); EINA_LOG_DBG("%p: type=%s name=%s, visible=%d, color=%02x%02x%02x%02x, %d,%d+%dx%d, clipper=%p (%d, %02x%02x%02x%02x, %d,%d+%dx%d)\n", ewkFrame, evas_object_type_get(ewkFrame), evas_object_name_get(ewkFrame), evas_object_visible_get(ewkFrame), red, green, blue, alpha, x, y, width, height, clip, evas_object_visible_get(clip), contentRed, contentGreen, contentBlue, contentAlpha, contentX, contentY, contentWidth, contentHeight); parent = evas_object_smart_parent_get(ewkFrame); if (!parent) EINA_LOG_ERR("could not get parent object.\n"); else _ewk_frame_debug(parent); } #endif static WebCore::FrameLoaderClientEfl* _ewk_frame_loader_efl_get(const WebCore::Frame* frame) { return static_cast(frame->loader()->client()); } static Eina_Bool _ewk_frame_children_iterator_next(Eina_Iterator_Ewk_Frame* iterator, Evas_Object** data) { EWK_FRAME_SD_GET_OR_RETURN(iterator->object, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); WebCore::FrameTree* tree = smartData->frame->tree(); // check if it's still valid EINA_SAFETY_ON_NULL_RETURN_VAL(tree, false); if (iterator->currentIndex < tree->childCount()) { *data = EWKPrivate::kitFrame(tree->child(iterator->currentIndex++)); return true; } return false; } static Evas_Object* _ewk_frame_children_iterator_get_container(Eina_Iterator_Ewk_Frame* iterator) { return iterator->object; } static void _ewk_frame_smart_add(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET(ewkFrame, smartData); if (!smartData) { smartData = static_cast(calloc(1, sizeof(Ewk_Frame_Smart_Data))); if (!smartData) { CRITICAL("could not allocate Ewk_Frame_Smart_Data"); return; } evas_object_smart_data_set(ewkFrame, smartData); } smartData->self = ewkFrame; _parent_sc.add(ewkFrame); evas_object_static_clip_set(smartData->base.clipper, false); evas_object_move(smartData->base.clipper, 0, 0); evas_object_resize(smartData->base.clipper, 0, 0); #ifdef EWK_FRAME_DEBUG smartData->region = evas_object_rectangle_add(smartData->base.evas); static int i = 0; switch (i) { case 0: evas_object_color_set(smartData->region, 128, 0, 0, 128); break; case 1: evas_object_color_set(smartData->region, 0, 128, 0, 128); break; case 2: evas_object_color_set(smartData->region, 0, 0, 128, 128); break; case 3: evas_object_color_set(smartData->region, 128, 0, 0, 128); break; case 4: evas_object_color_set(smartData->region, 128, 128, 0, 128); break; case 5: evas_object_color_set(smartData->region, 128, 0, 128, 128); break; case 6: evas_object_color_set(smartData->region, 0, 128, 128, 128); break; default: break; } i++; if (i > 6) i = 0; evas_object_smart_member_add(smartData->region, ewkFrame); evas_object_hide(smartData->region); #endif } static void _ewk_frame_smart_del(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET(ewkFrame, smartData); if (smartData) { if (smartData->frame) { WebCore::FrameLoaderClientEfl* flc = _ewk_frame_loader_efl_get(smartData->frame); flc->setWebFrame(0); EWK_FRAME_SD_GET(ewk_view_frame_main_get(smartData->view), mainSmartData); if (mainSmartData->frame == smartData->frame) // applying only for main frame is enough (will traverse through frame tree) smartData->frame->loader()->detachFromParent(); smartData->frame = 0; } eina_stringshare_del(smartData->title.string); eina_stringshare_del(smartData->uri); eina_stringshare_del(smartData->name); } _parent_sc.del(ewkFrame); } static void _ewk_frame_smart_resize(Evas_Object* ewkFrame, Evas_Coord width, Evas_Coord height) { EWK_FRAME_SD_GET(ewkFrame, smartData); evas_object_resize(smartData->base.clipper, width, height); #ifdef EWK_FRAME_DEBUG evas_object_resize(smartData->region, width, height); Evas_Coord x, y; evas_object_geometry_get(smartData->region, &x, &y, &width, &height); INFO("region=%p, visible=%d, geo=%d,%d + %dx%d", smartData->region, evas_object_visible_get(smartData->region), x, y, width, height); _ewk_frame_debug(ewkFrame); #endif } static void _ewk_frame_smart_set(Evas_Smart_Class* api) { evas_object_smart_clipped_smart_set(api); api->add = _ewk_frame_smart_add; api->del = _ewk_frame_smart_del; api->resize = _ewk_frame_smart_resize; } static inline Evas_Smart* _ewk_frame_smart_class_new(void) { static Evas_Smart_Class smartClass = EVAS_SMART_CLASS_INIT_NAME_VERSION(EWK_FRAME_TYPE_STR); static Evas_Smart* smart = 0; if (EINA_UNLIKELY(!smart)) { evas_object_smart_clipped_smart_set(&_parent_sc); _ewk_frame_smart_set(&smartClass); smart = evas_smart_class_new(&smartClass); } return smart; } Evas_Object* ewk_frame_view_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); return smartData->view; } Ewk_Security_Origin* ewk_frame_security_origin_get(const Evas_Object *ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->document(), 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->document()->securityOrigin(), 0); return ewk_security_origin_new(smartData->frame->document()->securityOrigin()); } Eina_Iterator* ewk_frame_children_iterator_new(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); Eina_Iterator_Ewk_Frame* iterator = static_cast (calloc(1, sizeof(Eina_Iterator_Ewk_Frame))); if (!iterator) return 0; EINA_MAGIC_SET(&iterator->base, EINA_MAGIC_ITERATOR); iterator->base.next = FUNC_ITERATOR_NEXT(_ewk_frame_children_iterator_next); iterator->base.get_container = FUNC_ITERATOR_GET_CONTAINER(_ewk_frame_children_iterator_get_container); iterator->base.free = FUNC_ITERATOR_FREE(free); iterator->object = ewkFrame; iterator->currentIndex = 0; return &iterator->base; } Evas_Object* ewk_frame_child_find(Evas_Object* ewkFrame, const char* name) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(name, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); WTF::String frameName = WTF::String::fromUTF8(name); return EWKPrivate::kitFrame(smartData->frame->tree()->find(WTF::AtomicString(frameName))); } Eina_Bool ewk_frame_uri_set(Evas_Object* ewkFrame, const char* uri) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); WebCore::KURL kurl(WebCore::KURL(), WTF::String::fromUTF8(uri)); WebCore::ResourceRequest req(kurl); WebCore::FrameLoader* loader = smartData->frame->loader(); loader->load(WebCore::FrameLoadRequest(smartData->frame, req)); return true; } const char* ewk_frame_uri_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); return smartData->uri; } const Ewk_Text_With_Direction* ewk_frame_title_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); return &smartData->title; } const char* ewk_frame_name_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); if (!smartData->frame) { ERR("could not get name of uninitialized frame."); return 0; } const WTF::String frameName = smartData->frame->tree()->uniqueName(); if ((smartData->name) && (smartData->name == frameName)) return smartData->name; eina_stringshare_replace_length(&(smartData->name), frameName.utf8().data(), frameName.length()); return smartData->name; } Eina_Bool ewk_frame_contents_size_get(const Evas_Object* ewkFrame, Evas_Coord* width, Evas_Coord* height) { if (width) *width = 0; if (height) *height = 0; EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); if (!smartData->frame || !smartData->frame->view()) return false; if (width) *width = smartData->frame->view()->contentsWidth(); if (height) *height = smartData->frame->view()->contentsHeight(); return true; } static Eina_Bool _ewk_frame_contents_set_internal(Ewk_Frame_Smart_Data* smartData, const char* contents, size_t contentsSize, const char* mimeType, const char* encoding, const char* baseUri, const char* unreachableUri) { size_t length = strlen(contents); if (contentsSize < 1 || contentsSize > length) contentsSize = length; if (!mimeType) mimeType = "text/html"; if (!encoding) encoding = "UTF-8"; if (!baseUri) baseUri = "about:blank"; WebCore::KURL baseKURL(WebCore::KURL(), WTF::String::fromUTF8(baseUri)); WebCore::KURL unreachableKURL; if (unreachableUri) unreachableKURL = WebCore::KURL(WebCore::KURL(), WTF::String::fromUTF8(unreachableUri)); else unreachableKURL = WebCore::KURL(); WTF::RefPtr buffer = WebCore::SharedBuffer::create(contents, contentsSize); WebCore::SubstituteData substituteData (buffer.release(), WTF::String::fromUTF8(mimeType), WTF::String::fromUTF8(encoding), baseKURL, unreachableKURL); WebCore::ResourceRequest request(baseKURL); smartData->frame->loader()->load(WebCore::FrameLoadRequest(smartData->frame, request, substituteData)); return true; } Eina_Bool ewk_frame_contents_set(Evas_Object* ewkFrame, const char* contents, size_t contentsSize, const char* mimeType, const char* encoding, const char* baseUri) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(contents, false); return _ewk_frame_contents_set_internal (smartData, contents, contentsSize, mimeType, encoding, baseUri, 0); } Eina_Bool ewk_frame_contents_alternate_set(Evas_Object* ewkFrame, const char* contents, size_t contentsSize, const char* mimeType, const char* encoding, const char* baseUri, const char* unreachableUri) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(contents, false); EINA_SAFETY_ON_NULL_RETURN_VAL(unreachableUri, false); return _ewk_frame_contents_set_internal (smartData, contents, contentsSize, mimeType, encoding, baseUri, unreachableUri); } const char* ewk_frame_script_execute(Evas_Object* ewkFrame, const char* script) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(script, 0); WTF::String resultString; JSC::JSValue result = smartData->frame->script()->executeScript(WTF::String::fromUTF8(script), true).jsValue(); if (!smartData->frame) // In case the script removed our frame from the page. return 0; if (!result || (!result.isBoolean() && !result.isString() && !result.isNumber())) return 0; JSC::ExecState* exec = smartData->frame->script()->globalObject(WebCore::mainThreadNormalWorld())->globalExec(); JSC::JSLockHolder lock(exec); resultString = result.toString(exec)->value(exec); return eina_stringshare_add(resultString.utf8().data()); } Eina_Bool ewk_frame_editable_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); return smartData->editable; } Eina_Bool ewk_frame_editable_set(Evas_Object* ewkFrame, Eina_Bool editable) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); editable = !!editable; if (smartData->editable == editable) return true; smartData->editable = editable; if (editable) smartData->frame->editor().applyEditingStyleToBodyElement(); return true; } const char* ewk_frame_selection_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); WTF::CString selectedText = smartData->frame->editor().selectedText().utf8(); if (selectedText.isNull()) return 0; return eina_stringshare_add(selectedText.data()); } Eina_Bool ewk_frame_text_search(const Evas_Object* ewkFrame, const char* text, Eina_Bool caseSensitive, Eina_Bool forward, Eina_Bool wrap) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(text, false); return smartData->frame->editor().findString(WTF::String::fromUTF8(text), forward, caseSensitive, wrap, true); } unsigned int ewk_frame_text_matches_mark(Evas_Object* ewkFrame, const char* string, Eina_Bool caseSensitive, Eina_Bool highlight, unsigned int limit) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(string, 0); smartData->frame->editor().setMarkedTextMatchesAreHighlighted(highlight); return smartData->frame->editor().countMatchesForText(WTF::String::fromUTF8(string), 0, caseSensitive, limit, true, 0); } Eina_Bool ewk_frame_text_matches_unmark_all(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); smartData->frame->document()->markers()->removeMarkers(WebCore::DocumentMarker::TextMatch); return true; } Eina_Bool ewk_frame_text_matches_highlight_set(Evas_Object* ewkFrame, Eina_Bool highlight) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); smartData->frame->editor().setMarkedTextMatchesAreHighlighted(highlight); return true; } Eina_Bool ewk_frame_text_matches_highlight_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); return smartData->frame->editor().markedTextMatchesAreHighlighted(); } /** * Comparison function used by ewk_frame_text_matches_nth_pos_get */ static bool _ewk_frame_rect_cmp_less_than(const WebCore::IntRect& begin, const WebCore::IntRect& end) { return (begin.y() < end.y() || (begin.y() == end.y() && begin.x() < end.x())); } /** * Predicate used by ewk_frame_text_matches_nth_pos_get */ static bool _ewk_frame_rect_is_negative_value(const WebCore::IntRect& rect) { return (rect.x() < 0 || rect.y() < 0); } Eina_Bool ewk_frame_text_matches_nth_pos_get(const Evas_Object* ewkFrame, size_t number, int* x, int* y) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); Vector intRects = smartData->frame->document()->markers()->renderedRectsForMarkers(WebCore::DocumentMarker::TextMatch); /* remove useless values */ std::remove_if(intRects.begin(), intRects.end(), _ewk_frame_rect_is_negative_value); if (intRects.isEmpty() || number > intRects.size()) return false; std::sort(intRects.begin(), intRects.end(), _ewk_frame_rect_cmp_less_than); if (x) *x = intRects[number - 1].x(); if (y) *y = intRects[number - 1].y(); return true; } Eina_Bool ewk_frame_stop(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); smartData->frame->loader()->stopAllLoaders(); return true; } Eina_Bool ewk_frame_reload(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); smartData->frame->loader()->reload(); return true; } Eina_Bool ewk_frame_reload_full(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); smartData->frame->loader()->reload(true); return true; } Eina_Bool ewk_frame_back(Evas_Object* ewkFrame) { return ewk_frame_navigate(ewkFrame, -1); } Eina_Bool ewk_frame_forward(Evas_Object* ewkFrame) { return ewk_frame_navigate(ewkFrame, 1); } Eina_Bool ewk_frame_navigate(Evas_Object* ewkFrame, int steps) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); WebCore::Page* page = smartData->frame->page(); if (!page->canGoBackOrForward(steps)) return false; page->goBackOrForward(steps); return true; } Eina_Bool ewk_frame_back_possible(Evas_Object* ewkFrame) { return ewk_frame_navigate_possible(ewkFrame, -1); } Eina_Bool ewk_frame_forward_possible(Evas_Object* ewkFrame) { return ewk_frame_navigate_possible(ewkFrame, 1); } Eina_Bool ewk_frame_navigate_possible(Evas_Object* ewkFrame, int steps) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); WebCore::Page* page = smartData->frame->page(); return page->canGoBackOrForward(steps); } float ewk_frame_page_zoom_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, -1.0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, -1.0); return smartData->frame->pageZoomFactor(); } Eina_Bool ewk_frame_page_zoom_set(Evas_Object* ewkFrame, float pageZoomFactor) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); smartData->frame->setPageZoomFactor(pageZoomFactor); return true; } float ewk_frame_text_zoom_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, -1.0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, -1.0); return smartData->frame->textZoomFactor(); } Eina_Bool ewk_frame_text_zoom_set(Evas_Object* ewkFrame, float textZoomFactor) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); smartData->frame->setTextZoomFactor(textZoomFactor); return true; } void ewk_frame_hit_test_free(Ewk_Hit_Test* hitTest) { EINA_SAFETY_ON_NULL_RETURN(hitTest); eina_stringshare_del(hitTest->title.string); eina_stringshare_del(hitTest->alternate_text); eina_stringshare_del(hitTest->link.text); eina_stringshare_del(hitTest->link.url); eina_stringshare_del(hitTest->link.title); eina_stringshare_del(hitTest->image_uri); eina_stringshare_del(hitTest->media_uri); delete hitTest; } Ewk_Hit_Test* ewk_frame_hit_test_new(const Evas_Object* ewkFrame, int x, int y) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); WebCore::FrameView* view = smartData->frame->view(); EINA_SAFETY_ON_NULL_RETURN_VAL(view, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->contentRenderer(), 0); WebCore::HitTestResult result = smartData->frame->eventHandler()->hitTestResultAtPoint (view->windowToContents(WebCore::IntPoint(x, y)), WebCore::HitTestRequest::ReadOnly | WebCore::HitTestRequest::Active | WebCore::HitTestRequest::IgnoreClipping | WebCore::HitTestRequest::DisallowShadowContent); if (result.scrollbar()) return 0; if (!result.innerNode()) return 0; Ewk_Hit_Test* hitTest = new Ewk_Hit_Test; // FIXME: This should probably use pointInMainFrame, if it is to match the documentation of ewk_hit_test. hitTest->x = result.pointInInnerNodeFrame().x(); hitTest->y = result.pointInInnerNodeFrame().y(); #if 0 // FIXME hitTest->bounding_box.x = result.boundingBox().x(); hitTest->bounding_box.y = result.boundingBox().y(); hitTest->bounding_box.width = result.boundingBox().width(); hitTest->bounding_box.height = result.boundingBox().height(); #else hitTest->bounding_box.x = 0; hitTest->bounding_box.y = 0; hitTest->bounding_box.w = 0; hitTest->bounding_box.h = 0; #endif WebCore::TextDirection direction; hitTest->title.string = eina_stringshare_add(result.title(direction).utf8().data()); hitTest->title.direction = (direction == WebCore::LTR) ? EWK_TEXT_DIRECTION_LEFT_TO_RIGHT : EWK_TEXT_DIRECTION_RIGHT_TO_LEFT; hitTest->alternate_text = eina_stringshare_add(result.altDisplayString().utf8().data()); if (result.innerNonSharedNode() && result.innerNonSharedNode()->document() && result.innerNonSharedNode()->document()->frame()) hitTest->frame = EWKPrivate::kitFrame(result.innerNonSharedNode()->document()->frame()); hitTest->link.text = eina_stringshare_add(result.textContent().utf8().data()); hitTest->link.url = eina_stringshare_add(result.absoluteLinkURL().string().utf8().data()); hitTest->link.title = eina_stringshare_add(result.titleDisplayString().utf8().data()); hitTest->link.target_frame = EWKPrivate::kitFrame(result.targetFrame()); hitTest->image_uri = eina_stringshare_add(result.absoluteImageURL().string().utf8().data()); hitTest->media_uri = eina_stringshare_add(result.absoluteMediaURL().string().utf8().data()); int context = EWK_HIT_TEST_RESULT_CONTEXT_DOCUMENT; if (!result.absoluteLinkURL().isEmpty()) context |= EWK_HIT_TEST_RESULT_CONTEXT_LINK; if (!result.absoluteImageURL().isEmpty()) context |= EWK_HIT_TEST_RESULT_CONTEXT_IMAGE; if (!result.absoluteMediaURL().isEmpty()) context |= EWK_HIT_TEST_RESULT_CONTEXT_MEDIA; if (result.isSelected()) context |= EWK_HIT_TEST_RESULT_CONTEXT_SELECTION; if (result.isContentEditable()) context |= EWK_HIT_TEST_RESULT_CONTEXT_EDITABLE; hitTest->context = static_cast(context); return hitTest; } Eina_Bool ewk_frame_scroll_add(Evas_Object* ewkFrame, int deltaX, int deltaY) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->view(), false); smartData->frame->view()->scrollBy(WebCore::IntSize(deltaX, deltaY)); return true; } Eina_Bool ewk_frame_scroll_set(Evas_Object* ewkFrame, int x, int y) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->view(), false); smartData->frame->view()->setScrollPosition(WebCore::IntPoint(x, y)); return true; } Eina_Bool ewk_frame_scroll_size_get(const Evas_Object* ewkFrame, int* width, int* height) { if (width) *width = 0; if (height) *height = 0; EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->view(), false); WebCore::IntPoint point = smartData->frame->view()->maximumScrollPosition(); if (width) *width = point.x(); if (height) *height = point.y(); return true; } Eina_Bool ewk_frame_scroll_pos_get(const Evas_Object* ewkFrame, int* x, int* y) { if (x) *x = 0; if (y) *y = 0; EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->view(), false); WebCore::IntPoint pos = smartData->frame->view()->scrollPosition(); if (x) *x = pos.x(); if (y) *y = pos.y(); return true; } Eina_Bool ewk_frame_visible_content_geometry_get(const Evas_Object* ewkFrame, Eina_Bool includeScrollbars, int* x, int* y, int* width, int* height) { if (x) *x = 0; if (y) *y = 0; if (width) *width = 0; if (height) *height = 0; EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->view(), false); WebCore::IntRect rect = smartData->frame->view()->visibleContentRect(includeScrollbars ? WebCore::ScrollableArea::IncludeScrollbars : WebCore::ScrollableArea::ExcludeScrollbars); if (x) *x = rect.x(); if (y) *y = rect.y(); if (width) *width = rect.width(); if (height) *height = rect.height(); return true; } Eina_Bool ewk_frame_paint_full_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->view(), false); return smartData->frame->view()->paintsEntireContents(); } void ewk_frame_paint_full_set(Evas_Object* ewkFrame, Eina_Bool flag) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); EINA_SAFETY_ON_NULL_RETURN(smartData->frame); EINA_SAFETY_ON_NULL_RETURN(smartData->frame->view()); smartData->frame->view()->setPaintsEntireContents(flag); } Eina_Bool ewk_frame_feed_focus_in(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); WebCore::FocusController* focusController = smartData->frame->page()->focusController(); focusController->setFocusedFrame(smartData->frame); return true; } Eina_Bool ewk_frame_feed_focus_out(Evas_Object*) { // TODO: what to do on focus out? ERR("what to do?"); return false; } Eina_Bool ewk_frame_focused_element_geometry_get(const Evas_Object *ewkFrame, int *x, int *y, int *w, int *h) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); WebCore::Document* document = smartData->frame->document(); if (!document) return false; WebCore::Node* focusedNode = document->focusedElement(); if (!focusedNode) return false; WebCore::IntRect nodeRect = focusedNode->pixelSnappedBoundingBox(); if (x) *x = nodeRect.x(); if (y) *y = nodeRect.y(); if (w) *w = nodeRect.width(); if (h) *h = nodeRect.height(); return true; } Eina_Bool ewk_frame_feed_mouse_wheel(Evas_Object* ewkFrame, const Evas_Event_Mouse_Wheel* wheelEvent) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(wheelEvent, false); WebCore::FrameView* view = smartData->frame->view(); DBG("ewkFrame=%p, view=%p, direction=%d, z=%d, pos=%d,%d", ewkFrame, view, wheelEvent->direction, wheelEvent->z, wheelEvent->canvas.x, wheelEvent->canvas.y); EINA_SAFETY_ON_NULL_RETURN_VAL(view, false); WebCore::PlatformWheelEvent event(wheelEvent); return smartData->frame->eventHandler()->handleWheelEvent(event); } Eina_Bool ewk_frame_feed_mouse_down(Evas_Object* ewkFrame, const Evas_Event_Mouse_Down* downEvent) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(downEvent, false); WebCore::FrameView* view = smartData->frame->view(); DBG("ewkFrame=%p, view=%p, button=%d, pos=%d,%d", ewkFrame, view, downEvent->button, downEvent->canvas.x, downEvent->canvas.y); EINA_SAFETY_ON_NULL_RETURN_VAL(view, false); Evas_Coord x, y; evas_object_geometry_get(smartData->view, &x, &y, 0, 0); WebCore::PlatformMouseEvent event(downEvent, WebCore::IntPoint(x, y)); return smartData->frame->eventHandler()->handleMousePressEvent(event); } Eina_Bool ewk_frame_feed_mouse_up(Evas_Object* ewkFrame, const Evas_Event_Mouse_Up* upEvent) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(upEvent, false); WebCore::FrameView* view = smartData->frame->view(); DBG("ewkFrame=%p, view=%p, button=%d, pos=%d,%d", ewkFrame, view, upEvent->button, upEvent->canvas.x, upEvent->canvas.y); EINA_SAFETY_ON_NULL_RETURN_VAL(view, false); Evas_Coord x, y; evas_object_geometry_get(smartData->view, &x, &y, 0, 0); WebCore::PlatformMouseEvent event(upEvent, WebCore::IntPoint(x, y)); return smartData->frame->eventHandler()->handleMouseReleaseEvent(event); } Eina_Bool ewk_frame_feed_mouse_move(Evas_Object* ewkFrame, const Evas_Event_Mouse_Move* moveEvent) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(moveEvent, false); WebCore::FrameView* view = smartData->frame->view(); DBG("ewkFrame=%p, view=%p, pos: old=%d,%d, new=%d,%d, buttons=%d", ewkFrame, view, moveEvent->cur.canvas.x, moveEvent->cur.canvas.y, moveEvent->prev.canvas.x, moveEvent->prev.canvas.y, moveEvent->buttons); EINA_SAFETY_ON_NULL_RETURN_VAL(view, false); Evas_Coord x, y; evas_object_geometry_get(smartData->view, &x, &y, 0, 0); WebCore::PlatformMouseEvent event(moveEvent, WebCore::IntPoint(x, y)); return smartData->frame->eventHandler()->mouseMoved(event); } Eina_Bool ewk_frame_feed_touch_event(Evas_Object* ewkFrame, Ewk_Touch_Event_Type action, Eina_List* points, unsigned modifiers) { #if ENABLE(TOUCH_EVENTS) EINA_SAFETY_ON_NULL_RETURN_VAL(points, false); EWK_FRAME_SD_GET(ewkFrame, smartData); if (!smartData || !smartData->frame || !ewk_view_need_touch_events_get(smartData->view)) return false; Evas_Coord x, y; evas_object_geometry_get(smartData->view, &x, &y, 0, 0); return smartData->frame->eventHandler()->handleTouchEvent(EWKPrivate::platformTouchEvent(x, y, points, action, modifiers)); #else UNUSED_PARAM(ewkFrame); UNUSED_PARAM(action); UNUSED_PARAM(points); UNUSED_PARAM(modifiers); return false; #endif } static inline Eina_Bool _ewk_frame_handle_key_scrolling(WebCore::Frame* frame, const WebCore::PlatformKeyboardEvent& keyEvent) { WebCore::ScrollDirection direction; WebCore::ScrollGranularity granularity; int keyCode = keyEvent.windowsVirtualKeyCode(); switch (keyCode) { case VK_SPACE: granularity = WebCore::ScrollByPage; if (keyEvent.shiftKey()) direction = WebCore::ScrollUp; else direction = WebCore::ScrollDown; break; case VK_NEXT: granularity = WebCore::ScrollByPage; direction = WebCore::ScrollDown; break; case VK_PRIOR: granularity = WebCore::ScrollByPage; direction = WebCore::ScrollUp; break; case VK_HOME: granularity = WebCore::ScrollByDocument; direction = WebCore::ScrollUp; break; case VK_END: granularity = WebCore::ScrollByDocument; direction = WebCore::ScrollDown; break; case VK_LEFT: granularity = WebCore::ScrollByLine; direction = WebCore::ScrollLeft; break; case VK_RIGHT: granularity = WebCore::ScrollByLine; direction = WebCore::ScrollRight; break; case VK_UP: direction = WebCore::ScrollUp; if (keyEvent.ctrlKey()) granularity = WebCore::ScrollByDocument; else granularity = WebCore::ScrollByLine; break; case VK_DOWN: direction = WebCore::ScrollDown; if (keyEvent.ctrlKey()) granularity = WebCore::ScrollByDocument; else granularity = WebCore::ScrollByLine; break; default: return false; } if (frame->eventHandler()->scrollOverflow(direction, granularity)) return false; frame->view()->scroll(direction, granularity); return true; } Eina_Bool ewk_frame_feed_key_down(Evas_Object* ewkFrame, const Evas_Event_Key_Down* downEvent) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(downEvent, false); DBG("ewkFrame=%p keyname=%s (key=%s, string=%s)", ewkFrame, downEvent->keyname, downEvent->key ? downEvent->key : "", downEvent->string ? downEvent->string : ""); WebCore::PlatformKeyboardEvent event(downEvent); if (smartData->frame->eventHandler()->keyEvent(event)) return true; return _ewk_frame_handle_key_scrolling(smartData->frame, event); } Eina_Bool ewk_frame_feed_key_up(Evas_Object* ewkFrame, const Evas_Event_Key_Up* upEvent) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); EINA_SAFETY_ON_NULL_RETURN_VAL(upEvent, false); DBG("ewkFrame=%p keyname=%s (key=%s, string=%s)", ewkFrame, upEvent->keyname, upEvent->key ? upEvent->key : "", upEvent->string ? upEvent->string : ""); WebCore::PlatformKeyboardEvent event(upEvent); return smartData->frame->eventHandler()->keyEvent(event); } Ewk_Text_Selection_Type ewk_frame_text_selection_type_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, EWK_TEXT_SELECTION_NONE); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, EWK_TEXT_SELECTION_NONE); WebCore::FrameSelection* controller = smartData->frame->selection(); if (!controller) return EWK_TEXT_SELECTION_NONE; return static_cast(controller->selectionType()); } /* internal methods ****************************************************/ /** * @internal * * Creates a new EFL WebKit Frame object. * * Frames are low level entries contained in a page that is contained * by a view. Usually one operates on the view and not directly on the * frame. * * @param canvas canvas where to create the frame object * * @return a new frame object or @c 0 on failure */ Evas_Object* ewk_frame_add(Evas* canvas) { return evas_object_smart_add(canvas, _ewk_frame_smart_class_new()); } /** * @internal * * Initialize frame based on actual WebKit frame. * * This is internal and should never be called by external users. */ bool ewk_frame_init(Evas_Object* ewkFrame, Evas_Object* view, WebCore::Frame* frame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); if (!smartData->frame) { WebCore::FrameLoaderClientEfl* frameLoaderClient = _ewk_frame_loader_efl_get(frame); frameLoaderClient->setWebFrame(ewkFrame); smartData->frame = frame; smartData->view = view; frame->init(); return true; } ERR("frame %p already set for %p, ignored new %p", smartData->frame, ewkFrame, frame); return false; } /** * @internal * * Adds child to the frame. */ bool ewk_frame_child_add(Evas_Object* ewkFrame, WTF::PassRefPtr child, const WTF::String& name, const WebCore::KURL& url, const WTF::String& referrer) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); char buffer[256]; Evas_Object* frame; WebCore::Frame* coreFrame; frame = ewk_frame_add(smartData->base.evas); if (!frame) { ERR("Could not create ewk_frame object."); return false; } coreFrame = child.get(); if (coreFrame->tree()) coreFrame->tree()->setName(name); else ERR("no tree for child object"); smartData->frame->tree()->appendChild(child); if (!ewk_frame_init(frame, smartData->view, coreFrame)) { evas_object_del(frame); return false; } snprintf(buffer, sizeof(buffer), "EWK_Frame:child/%s", name.utf8().data()); evas_object_name_set(frame, buffer); evas_object_smart_member_add(frame, ewkFrame); evas_object_show(frame); // The creation of the frame may have run arbitrary JavaScript that removed it from the page already. if (!coreFrame->page()) { evas_object_del(frame); return true; } evas_object_smart_callback_call(smartData->view, "frame,created", frame); smartData->frame->loader()->loadURLIntoChildFrame(url, referrer, coreFrame); // The frame's onload handler may have removed it from the document. // See fast/dom/null-page-show-modal-dialog-crash.html for an example. if (!coreFrame->tree()->parent()) { evas_object_del(frame); return true; } return true; } /** * @internal * Change the ewk view this frame is associated with. * * @param ewkFrame The ewk frame to act upon. * @param newParent The new view that will be set as the parent of the frame. */ void ewk_frame_view_set(Evas_Object* ewkFrame, Evas_Object* newParent) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); evas_object_smart_member_del(ewkFrame); evas_object_smart_member_add(ewkFrame, newParent); smartData->view = newParent; } /** * @internal * Frame was destroyed by loader, remove internal reference. */ void ewk_frame_core_gone(Evas_Object* ewkFrame) { DBG("ewkFrame=%p", ewkFrame); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); smartData->frame = 0; } /** * @internal * Reports cancellation of a client redirect. * * @param ewkFrame Frame. * * Emits signal: "redirect,cancelled" */ void ewk_frame_redirect_cancelled(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "redirect,cancelled", 0); } /** * @internal * Reports receipt of server redirect for provisional load. * * @param ewkFrame Frame. * * Emits signal: "redirect,load,provisional" */ void ewk_frame_redirect_provisional_load(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "redirect,load,provisional", 0); } /** * @internal * Reports that a client redirect will be performed. * * @param ewkFrame Frame. * @param url Redirection URL. * * Emits signal: "redirect,requested" */ void ewk_frame_redirect_requested(Evas_Object* ewkFrame, const char* url) { evas_object_smart_callback_call(ewkFrame, "redirect,requested", (void*)url); } /** * @internal * Reports a resource will be requested. User may override behavior of webkit by * changing values in @param request. * * @param ewkFrame Frame. * @param messages Messages containing the request details that user may override and a * possible redirect reponse. Whenever values on this struct changes, it must be properly * malloc'd as it will be freed afterwards. * * Emits signal: "resource,request,willsend" */ void ewk_frame_request_will_send(Evas_Object* ewkFrame, Ewk_Frame_Resource_Messages* messages) { evas_object_smart_callback_call(ewkFrame, "resource,request,willsend", messages); } /** * @internal * Reports that there's a new resource. * * @param ewkFrame Frame. * @param request New request details. No changes are allowed to fields. * * Emits signal: "resource,request,new" */ void ewk_frame_request_assign_identifier(Evas_Object* ewkFrame, const Ewk_Frame_Resource_Request* request) { evas_object_smart_callback_call(ewkFrame, "resource,request,new", (void*)request); } /** * @internal * Reports that a response to a resource request was received. * * @param ewkFrame Frame. * @param request Response details. No changes are allowed to fields. * * Emits signal: "resource,response,received" */ void ewk_frame_response_received(Evas_Object* ewkFrame, Ewk_Frame_Resource_Response* response) { evas_object_smart_callback_call(ewkFrame, "resource,response,received", response); } /** * @internal * Reports that first navigation occurred * * @param ewkFrame Frame. * * Emits signal: "navigation,first" */ void ewk_frame_did_perform_first_navigation(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "navigation,first", 0); } /** * @internal * Reports frame will be saved to current state * * @param ewkFrame Frame. * @param item History item to save details to. * * Emits signal: "state,save" */ void ewk_frame_view_state_save(Evas_Object* ewkFrame, WebCore::HistoryItem*) { evas_object_smart_callback_call(ewkFrame, "state,save", 0); } /** * @internal * Reports the frame committed load. * * Emits signal: "load,committed" with no parameters. */ void ewk_frame_load_committed(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "load,committed", 0); } /** * @internal * Reports the frame started loading something. * * Emits signal: "load,started" with no parameters. */ void ewk_frame_load_started(Evas_Object* ewkFrame) { Evas_Object* mainFrame; DBG("ewkFrame=%p", ewkFrame); evas_object_smart_callback_call(ewkFrame, "load,started", 0); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); ewk_view_load_started(smartData->view, ewkFrame); mainFrame = ewk_view_frame_main_get(smartData->view); if (mainFrame == ewkFrame) ewk_view_frame_main_load_started(smartData->view); } /** * @internal * Reports the frame started provisional load. * * @param ewkFrame Frame. * * Emits signal: "load,provisional" with no parameters. */ void ewk_frame_load_provisional(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "load,provisional", 0); } /** * @internal * Reports the frame provisional load failed. * * @param ewkFrame Frame. * @param error Load error. * * Emits signal: "load,provisional,failed" with pointer to Ewk_Frame_Load_Error. */ void ewk_frame_load_provisional_failed(Evas_Object* ewkFrame, const Ewk_Frame_Load_Error* error) { evas_object_smart_callback_call(ewkFrame, "load,provisional,failed", const_cast(error)); } /** * @internal * Reports the frame finished first layout. * * @param ewkFrame Frame. * * Emits signal: "load,firstlayout,finished" with no parameters. */ void ewk_frame_load_firstlayout_finished(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "load,firstlayout,finished", 0); } /** * @internal * Reports the frame finished first non empty layout. * * @param ewkFrame Frame. * * Emits signal: "load,nonemptylayout,finished" with no parameters. */ void ewk_frame_load_firstlayout_nonempty_finished(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "load,nonemptylayout,finished", 0); } /** * @internal * Reports the loading of a document has finished on frame. * * @param ewkFrame Frame. * * Emits signal: "load,document,finished" with no parameters. */ void ewk_frame_load_document_finished(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "load,document,finished", 0); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); ewk_view_load_document_finished(smartData->view, ewkFrame); } /** * @internal * Reports load finished, optionally with error information. * * Emits signal: "load,finished" with pointer to Ewk_Frame_Load_Error * if any error, or @c NULL if successful load. * * @note there should notbe any error stuff here, but trying to be * compatible with previous WebKit. */ void ewk_frame_load_finished(Evas_Object* ewkFrame, const char* errorDomain, int errorCode, bool isCancellation, const char* errorDescription, const char* failingUrl) { Ewk_Frame_Load_Error buffer, *error; if (!errorDomain) { DBG("ewkFrame=%p, success.", ewkFrame); error = 0; } else { DBG("ewkFrame=%p, error=%s (%d, cancellation=%hhu) \"%s\", url=%s", ewkFrame, errorDomain, errorCode, isCancellation, errorDescription, failingUrl); buffer.domain = errorDomain; buffer.code = errorCode; buffer.is_cancellation = isCancellation; buffer.description = errorDescription; buffer.failing_url = failingUrl; buffer.resource_identifier = 0; buffer.frame = ewkFrame; error = &buffer; } evas_object_smart_callback_call(ewkFrame, "load,finished", error); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); ewk_view_load_finished(smartData->view, error); } /** * @internal * Reports resource load finished. * * Emits signal: "load,resource,finished" with the resource * request identifier. */ void ewk_frame_load_resource_finished(Evas_Object* ewkFrame, unsigned long identifier) { evas_object_smart_callback_call(ewkFrame, "load,resource,finished", &identifier); } /** * @internal * Reports resource load failure, with error information. * * Emits signal: "load,resource,failed" with the error information. */ void ewk_frame_load_resource_failed(Evas_Object* ewkFrame, Ewk_Frame_Load_Error* error) { evas_object_smart_callback_call(ewkFrame, "load,resource,failed", error); } /** * @internal * Reports load failed with error information. * * Emits signal: "load,error" with pointer to Ewk_Frame_Load_Error. */ void ewk_frame_load_error(Evas_Object* ewkFrame, const char* errorDomain, int errorCode, bool isCancellation, const char* errorDescription, const char* failingUrl) { Ewk_Frame_Load_Error error; DBG("ewkFrame=%p, error=%s (%d, cancellation=%hhu) \"%s\", url=%s", ewkFrame, errorDomain, errorCode, isCancellation, errorDescription, failingUrl); EINA_SAFETY_ON_NULL_RETURN(errorDomain); error.code = errorCode; error.is_cancellation = isCancellation; error.domain = errorDomain; error.description = errorDescription; error.failing_url = failingUrl; error.resource_identifier = 0; error.frame = ewkFrame; evas_object_smart_callback_call(ewkFrame, "load,error", &error); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); ewk_view_load_error(smartData->view, &error); } /** * @internal * Reports load progress changed. * * Emits signal: "load,progress" with pointer to a double from 0.0 to 1.0. */ void ewk_frame_load_progress_changed(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); EINA_SAFETY_ON_NULL_RETURN(smartData->frame); // TODO: this is per page, there should be a way to have per-frame. double progress = smartData->frame->page()->progress()->estimatedProgress(); DBG("ewkFrame=%p (p=%0.3f)", ewkFrame, progress); evas_object_smart_callback_call(ewkFrame, "load,progress", &progress); ewk_view_load_progress_changed(smartData->view); } /** * @internal * * Reports contents size changed. */ void ewk_frame_contents_size_changed(Evas_Object* ewkFrame, Evas_Coord width, Evas_Coord height) { Evas_Coord size[2] = {width, height}; evas_object_smart_callback_call(ewkFrame, "contents,size,changed", size); } /** * @internal * * Reports title changed. */ void ewk_frame_title_set(Evas_Object* ewkFrame, const Ewk_Text_With_Direction* title) { DBG("ewkFrame=%p, title=%s, direction=%s", ewkFrame, title->string ? title->string : "(null)", title->direction == EWK_TEXT_DIRECTION_LEFT_TO_RIGHT ? "ltr" : "rtl"); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); if (!eina_stringshare_replace(&smartData->title.string, title->string) && (smartData->title.direction == title->direction)) return; smartData->title.direction = title->direction; evas_object_smart_callback_call(ewkFrame, "title,changed", (void*)title); } /** * @internal * * Creates a view. */ void ewk_frame_view_create_for_view(Evas_Object* ewkFrame, Evas_Object* view) { DBG("ewkFrame=%p, view=%p", ewkFrame, view); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); EINA_SAFETY_ON_NULL_RETURN(smartData->frame); Evas_Coord width, height; evas_object_geometry_get(view, 0, 0, &width, &height); WebCore::IntSize size(width, height); int red, green, blue, alpha; WebCore::Color background; ewk_view_bg_color_get(view, &red, &green, &blue, &alpha); if (!alpha) background = WebCore::Color(0, 0, 0, 0); else if (alpha == 255) background = WebCore::Color(red, green, blue, alpha); else background = WebCore::Color(red * 255 / alpha, green * 255 / alpha, blue * 255 / alpha, alpha); smartData->frame->createView(size, background, !alpha); if (!smartData->frame->view()) return; smartData->frame->view()->setEvasObject(ewkFrame); ewk_frame_mixed_content_displayed_set(ewkFrame, false); ewk_frame_mixed_content_run_set(ewkFrame, false); } ssize_t ewk_frame_source_get(const Evas_Object* ewkFrame, char** frameSource) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, -1); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, -1); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->document(), -1); EINA_SAFETY_ON_NULL_RETURN_VAL(frameSource, -1); StringBuilder builder; *frameSource = 0; // Saves 0 to pointer until it's not allocated. if (!ewk_frame_uri_get(ewkFrame)) return -1; if (!smartData->frame->document()->isHTMLDocument()) { // FIXME: Support others documents. WARN("Only HTML documents are supported"); return -1; } // Look for tag. If it exists, the node contatins all document's source. WebCore::Node* documentNode = smartData->frame->document()->documentElement(); if (documentNode) for (WebCore::Node* node = documentNode->firstChild(); node; node = node->parentElement()) { if (node->hasTagName(WebCore::HTMLNames::htmlTag)) { WebCore::HTMLElement* element = static_cast(node); if (element) builder.append(element->outerHTML()); break; } } CString utf8String = builder.toString().utf8(); size_t sourceLength = utf8String.length(); *frameSource = static_cast(malloc(sourceLength + 1)); if (!*frameSource) { CRITICAL("Could not allocate memory."); return -1; } strncpy(*frameSource, utf8String.data(), sourceLength); (*frameSource)[sourceLength] = '\0'; return sourceLength; } Eina_List* ewk_frame_resources_location_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame->document(), 0); Eina_List* listOfImagesLocation = 0; // Get src attibute of images and saves them to the Eina_List. RefPtr images = smartData->frame->document()->images(); for (size_t index = 0; index < images->length(); ++index) { WebCore::HTMLImageElement* imageElement = static_cast(images->item(index)); if (!imageElement || imageElement->src().isNull() || imageElement->src().isEmpty()) continue; WTF::String imageLocation = WebCore::decodeURLEscapeSequences(imageElement->src().string()); // Look for duplicated location. Eina_List* listIterator = 0; void* data = 0; Eina_Bool found = false; EINA_LIST_FOREACH(listOfImagesLocation, listIterator, data) if ((found = !strcmp(static_cast(data), imageLocation.utf8().data()))) break; if (found) continue; const char* imageLocationCopy = eina_stringshare_add(imageLocation.utf8().data()); if (!imageLocationCopy) goto out_of_memory_handler; listOfImagesLocation = eina_list_append(listOfImagesLocation, imageLocationCopy); if (eina_error_get()) goto out_of_memory_handler; } // FIXME: Get URL others resources (plugins, css, media files). return listOfImagesLocation; out_of_memory_handler: CRITICAL("Could not allocate memory."); void* data; EINA_LIST_FREE(listOfImagesLocation, data) free(data); return 0; } const char* ewk_frame_plain_text_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, 0); if (!smartData->frame->document()) return 0; WebCore::Element* documentElement = smartData->frame->document()->documentElement(); if (!documentElement) return 0; return eina_stringshare_add(documentElement->innerText().utf8().data()); } Eina_Bool ewk_frame_mixed_content_displayed_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); return smartData->hasDisplayedMixedContent; } Eina_Bool ewk_frame_mixed_content_run_get(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); return smartData->hasRunMixedContent; } Ewk_Certificate_Status ewk_frame_certificate_status_get(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, EWK_CERTIFICATE_STATUS_NO_CERTIFICATE); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, EWK_CERTIFICATE_STATUS_NO_CERTIFICATE); const WebCore::FrameLoader* frameLoader = smartData->frame->loader(); const WebCore::DocumentLoader* documentLoader = frameLoader->documentLoader(); const WebCore::KURL documentURL = documentLoader->requestURL(); if (!documentURL.protocolIs("https")) return EWK_CERTIFICATE_STATUS_NO_CERTIFICATE; if (frameLoader->subframeIsLoading()) return EWK_CERTIFICATE_STATUS_NO_CERTIFICATE; SoupMessage* soupMessage = documentLoader->request().toSoupMessage(); if (soupMessage && (soup_message_get_flags(soupMessage) & SOUP_MESSAGE_CERTIFICATE_TRUSTED)) return EWK_CERTIFICATE_STATUS_TRUSTED; return EWK_CERTIFICATE_STATUS_UNTRUSTED; } /** * @internal * Reports frame favicon changed. * * @param ewkFrame Frame. * * Emits signal: "icon,changed" with no parameters. */ void ewk_frame_icon_changed(Evas_Object* ewkFrame) { DBG("ewkFrame=%p", ewkFrame); evas_object_smart_callback_call(ewkFrame, "icon,changed", 0); } /** * @internal * Reports uri changed and swap internal string reference. * * Emits signal: "uri,changed" with new uri as parameter. */ bool ewk_frame_uri_changed(Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, false); EINA_SAFETY_ON_NULL_RETURN_VAL(smartData->frame, false); WTF::CString uri(smartData->frame->document()->url().string().utf8()); INFO("uri=%s", uri.data()); if (!uri.data()) { ERR("no uri"); return false; } eina_stringshare_replace(&smartData->uri, uri.data()); evas_object_smart_callback_call(ewkFrame, "uri,changed", (void*)smartData->uri); return true; } /** * @internal * * Forces layout for frame. */ void ewk_frame_force_layout(Evas_Object* ewkFrame) { DBG("ewkFrame=%p", ewkFrame); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); EINA_SAFETY_ON_NULL_RETURN(smartData->frame); WebCore::FrameView* view = smartData->frame->view(); if (view) view->forceLayout(true); } /** * @internal * * Creates plugin. */ WTF::PassRefPtr ewk_frame_plugin_create(Evas_Object* ewkFrame, const WebCore::IntSize& pluginSize, WebCore::HTMLPlugInElement* element, const WebCore::KURL& url, const WTF::Vector& paramNames, const WTF::Vector& paramValues, const WTF::String& mimeType, bool loadManually) { #if ENABLE(NETSCAPE_PLUGIN_API) DBG("ewkFrame=%p, size=%dx%d, element=%p, url=%s, mimeType=%s", ewkFrame, pluginSize.width(), pluginSize.height(), element, url.string().utf8().data(), mimeType.utf8().data()); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, sd, 0); // TODO: emit signal and ask webkit users if something else should be done. // like creating another x window (see gtk, it allows users to create // GtkPluginWidget. WTF::RefPtr pluginView = WebCore::PluginView::create (sd->frame, pluginSize, element, url, paramNames, paramValues, mimeType, loadManually); if (pluginView->status() == WebCore::PluginStatusLoadedSuccessfully) return pluginView.release(); #else UNUSED_PARAM(ewkFrame); UNUSED_PARAM(pluginSize); UNUSED_PARAM(element); UNUSED_PARAM(url); UNUSED_PARAM(paramNames); UNUSED_PARAM(paramValues); UNUSED_PARAM(mimeType); UNUSED_PARAM(loadManually); #endif // #if ENABLE(NETSCAPE_PLUGIN_API) return 0; } /** * @internal * Reports that editor client selection was changed. * * @param ewkFrame Frame * * Emits signal: "editorclientselection,changed" with no parameters. */ void ewk_frame_editor_client_selection_changed(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "editorclient,selection,changed", 0); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); ewk_view_editor_client_selection_changed(smartData->view); } /** * @internal * Reports that editor client's contents were changed. * * @param o Frame * * Emits signal: "editorclient,contents,changed" with no parameters. */ void ewk_frame_editor_client_contents_changed(Evas_Object* ewkFrame) { evas_object_smart_callback_call(ewkFrame, "editorclient,contents,changed", 0); EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); ewk_view_editor_client_contents_changed(smartData->view); } /** * @internal * Defines whether the frame has displayed mixed content. * * When a frame has displayed mixed content, the currently loaded URI is secure (HTTPS) but it has * loaded and displayed a resource, such as an image, from an insecure (HTTP) source. * * @param hasDisplayed Do or do not clear the flag from the frame. If @c true, the container view * is also notified and it then emits the "mixedcontent,displayed" signal. * * Emits signal: "mixedcontent,displayed" with no parameters when @p hasDisplayed is @c true. */ void ewk_frame_mixed_content_displayed_set(Evas_Object* ewkFrame, bool hasDisplayed) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); smartData->hasDisplayedMixedContent = hasDisplayed; if (hasDisplayed) { ewk_view_mixed_content_displayed_set(smartData->view, true); evas_object_smart_callback_call(ewkFrame, "mixedcontent,displayed", 0); } } /** * @internal * Defines whether the frame has run mixed content. * * When a frame has run mixed content, the currently loaded URI is secure (HTTPS) but it has * loaded and run a resource, such as a script, from an insecure (HTTP) source. * * @param hasDisplayed Do or do not clear the flag from the frame. If @c true, the container view * is also notified and it then emits the "mixedcontent,run" signal. * * Emits signal: "mixedcontent,run" with no parameters when @p hasRun is @c true. */ void ewk_frame_mixed_content_run_set(Evas_Object* ewkFrame, bool hasRun) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData); smartData->hasRunMixedContent = hasRun; if (hasRun) { ewk_view_mixed_content_run_set(smartData->view, true); evas_object_smart_callback_call(ewkFrame, "mixedcontent,run", 0); } } /** * @internal * Reports that reflected XSS is encountered in the page and suppressed. * * @param xssInfo Information received from the XSSAuditor when XSS is * encountered in the page. * * Emits signal: "xss,detected" with pointer to Ewk_Frame_Xss_Notification. */ void ewk_frame_xss_detected(Evas_Object* ewkFrame, const Ewk_Frame_Xss_Notification* xssInfo) { evas_object_smart_callback_call(ewkFrame, "xss,detected", (void*)xssInfo); } namespace EWKPrivate { WebCore::Frame* coreFrame(const Evas_Object* ewkFrame) { EWK_FRAME_SD_GET_OR_RETURN(ewkFrame, smartData, 0); return smartData->frame; } Evas_Object* kitFrame(const WebCore::Frame* coreFrame) { if (!coreFrame) return 0; WebCore::FrameLoaderClientEfl* frameLoaderClient = _ewk_frame_loader_efl_get(coreFrame); if (!frameLoaderClient) return 0; return frameLoaderClient->webFrame(); } } // namespace EWKPrivate