/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #include #include #include #include #define LOK_USE_UNSTABLE_API #include #include #include #include #if !GLIB_CHECK_VERSION(2,32,0) #define G_SOURCE_REMOVE FALSE #define G_SOURCE_CONTINUE TRUE #endif #ifndef g_info #define g_info(...) g_log(G_LOG_DOMAIN, G_LOG_LEVEL_INFO, __VA_ARGS__) #endif // Cursor bitmaps from the Android app. #define CURSOR_HANDLE_DIR "android/experimental/LOAndroid3/res/drawable/" // Number of handles around a graphic selection. #define GRAPHIC_HANDLE_COUNT 8 // We know that VirtualDevices use a DPI of 96. static const int DPI = 96; /// Holds data used by LOKDocView only. struct LOKDocView_Impl { GtkWidget* m_pEventBox; GtkWidget* m_pTable; GtkWidget** m_pCanvas; float m_fZoom; LibreOfficeKit* m_pOffice; LibreOfficeKitDocument* m_pDocument; /// View or edit mode. bool m_bEdit; /// Position and size of the visible cursor. GdkRectangle m_aVisibleCursor; /// Cursor overlay is visible or hidden (for blinking). bool m_bCursorOverlayVisible; /// Cursor is visible or hidden (e.g. for graphic selection). bool m_bCursorVisible; /// Time of the last button press. guint32 m_nLastButtonPressTime; /// Time of the last button release. guint32 m_nLastButtonReleaseTime; /// Rectangles of the current text selection. GList* m_pTextSelectionRectangles; /// Position and size of the selection start (as if there would be a cursor caret there). GdkRectangle m_aTextSelectionStart; /// Position and size of the selection end. GdkRectangle m_aTextSelectionEnd; GdkRectangle m_aGraphicSelection; bool m_bInDragGraphicSelection; /// @name Start/middle/end handle. ///@{ /// Bitmap of the text selection start handle. cairo_surface_t* m_pHandleStart; /// Rectangle of the text selection start handle, to know if the user clicked on it or not GdkRectangle m_aHandleStartRect; /// If we are in the middle of a drag of the text selection end handle. bool m_bInDragStartHandle; /// Bitmap of the text selection middle handle. cairo_surface_t* m_pHandleMiddle; /// Rectangle of the text selection middle handle, to know if the user clicked on it or not GdkRectangle m_aHandleMiddleRect; /// If we are in the middle of a drag of the text selection middle handle. bool m_bInDragMiddleHandle; /// Bitmap of the text selection end handle. cairo_surface_t* m_pHandleEnd; /// Rectangle of the text selection end handle, to know if the user clicked on it or not GdkRectangle m_aHandleEndRect; /// If we are in the middle of a drag of the text selection end handle. bool m_bInDragEndHandle; ///@} /// @name Graphic handles. ///@{ /// Bitmap of a graphic selection handle. cairo_surface_t* m_pGraphicHandle; /// Rectangle of a graphic selection handle, to know if the user clicked on it or not. GdkRectangle m_aGraphicHandleRects[8]; /// If we are in the middle of a drag of a graphic selection handle. bool m_bInDragGraphicHandles[8]; ///@} LOKDocView_Impl(); ~LOKDocView_Impl(); /// Connected to the destroy signal of LOKDocView, deletes its LOKDocView_Impl. static void destroy(LOKDocView* pDocView, gpointer pData); /// Converts from screen pixels to document coordinates. float pixelToTwip(float fInput); /// Converts from document coordinates to screen pixels. float twipToPixel(float fInput); /// Receives a key press or release event. void signalKey(GdkEventKey* pEvent); }; LOKDocView_Impl::LOKDocView_Impl() : m_pEventBox(gtk_event_box_new()), m_pTable(0), m_pCanvas(0), m_fZoom(1), m_pOffice(0), m_pDocument(0), m_bEdit(false), m_aVisibleCursor({0, 0, 0, 0}), m_bCursorOverlayVisible(false), m_bCursorVisible(true), m_nLastButtonPressTime(0), m_nLastButtonReleaseTime(0), m_pTextSelectionRectangles(0), m_aTextSelectionStart({0, 0, 0, 0}), m_aTextSelectionEnd({0, 0, 0, 0}), m_aGraphicSelection({0, 0, 0, 0}), m_bInDragGraphicSelection(false), // Start/middle/end handle. m_pHandleStart(0), m_aHandleStartRect({0, 0, 0, 0}), m_bInDragStartHandle(false), m_pHandleMiddle(0), m_aHandleMiddleRect({0, 0, 0, 0}), m_bInDragMiddleHandle(false), m_pHandleEnd(0), m_aHandleEndRect({0, 0, 0, 0}), m_bInDragEndHandle(false), m_pGraphicHandle(0) { memset(&m_aGraphicHandleRects, 0, sizeof(m_aGraphicHandleRects)); memset(&m_bInDragGraphicHandles, 0, sizeof(m_bInDragGraphicHandles)); } LOKDocView_Impl::~LOKDocView_Impl() { if (m_pDocument) m_pDocument->pClass->destroy(m_pDocument); m_pDocument = 0; } void LOKDocView_Impl::destroy(LOKDocView* pDocView, gpointer /*pData*/) { // We specifically need to destroy the document when closing in order to ensure // that lock files etc. are cleaned up. delete pDocView->m_pImpl; } float LOKDocView_Impl::pixelToTwip(float fInput) { return (fInput / DPI / m_fZoom) * 1440.0f; } float LOKDocView_Impl::twipToPixel(float fInput) { return fInput / 1440.0f * DPI * m_fZoom; } void LOKDocView_Impl::signalKey(GdkEventKey* pEvent) { int nCharCode = 0; int nKeyCode = 0; if (!m_bEdit) { g_info("signalKey: not in edit mode, ignore"); return; } switch (pEvent->keyval) { case GDK_BackSpace: nKeyCode = com::sun::star::awt::Key::BACKSPACE; break; case GDK_Return: nKeyCode = com::sun::star::awt::Key::RETURN; break; case GDK_Escape: nKeyCode = com::sun::star::awt::Key::ESCAPE; break; case GDK_Tab: nKeyCode = com::sun::star::awt::Key::TAB; break; case GDK_Down: nKeyCode = com::sun::star::awt::Key::DOWN; break; case GDK_Up: nKeyCode = com::sun::star::awt::Key::UP; break; case GDK_Left: nKeyCode = com::sun::star::awt::Key::LEFT; break; case GDK_Right: nKeyCode = com::sun::star::awt::Key::RIGHT; break; default: if (pEvent->keyval >= GDK_F1 && pEvent->keyval <= GDK_F26) nKeyCode = com::sun::star::awt::Key::F1 + (pEvent->keyval - GDK_F1); else nCharCode = gdk_keyval_to_unicode(pEvent->keyval); } // rsc is not public API, but should be good enough for debugging purposes. // If this is needed for real, then probably a new param of type // css::awt::KeyModifier is needed in postKeyEvent(). if (pEvent->state & GDK_SHIFT_MASK) nKeyCode |= KEY_SHIFT; if (pEvent->type == GDK_KEY_RELEASE) m_pDocument->pClass->postKeyEvent(m_pDocument, LOK_KEYEVENT_KEYUP, nCharCode, nKeyCode); else m_pDocument->pClass->postKeyEvent(m_pDocument, LOK_KEYEVENT_KEYINPUT, nCharCode, nKeyCode); } static void lok_docview_class_init( gpointer ); static void lok_docview_init( GTypeInstance *, gpointer ); static gboolean renderOverlay(GtkWidget* pWidget, GdkEventExpose* pEvent, gpointer pData); /** * The user drags the handle, which is below the cursor, but wants to move the * cursor accordingly. * * @param pHandle the rectangle of the handle * @param pEvent the motion event * @param pPoint the computed point (output parameter) */ void lcl_getDragPoint(GdkRectangle* pHandle, GdkEventButton* pEvent, GdkPoint* pPoint) { GdkPoint aCursor, aHandle; // Center of the cursor rectangle: we know that it's above the handle. aCursor.x = pHandle->x + pHandle->width / 2; aCursor.y = pHandle->y - pHandle->height / 2; // Center of the handle rectangle. aHandle.x = pHandle->x + pHandle->width / 2; aHandle.y = pHandle->y + pHandle->height / 2; // Our target is the original cursor position + the dragged offset. pPoint->x = aCursor.x + (pEvent->x - aHandle.x); pPoint->y = aCursor.y + (pEvent->y - aHandle.y); } gboolean lcl_signalMotion(GtkWidget* /*pEventBox*/, GdkEventButton* pEvent, LOKDocView* pDocView) { GdkPoint aPoint; if (pDocView->m_pImpl->m_bInDragMiddleHandle) { g_info("lcl_signalMotion: dragging the middle handle"); lcl_getDragPoint(&pDocView->m_pImpl->m_aHandleMiddleRect, pEvent, &aPoint); pDocView->m_pImpl->m_pDocument->pClass->setTextSelection( pDocView->m_pImpl->m_pDocument, LOK_SETTEXTSELECTION_RESET, pDocView->m_pImpl->pixelToTwip(aPoint.x), pDocView->m_pImpl->pixelToTwip(aPoint.y)); return FALSE; } if (pDocView->m_pImpl->m_bInDragStartHandle) { g_info("lcl_signalMotion: dragging the start handle"); lcl_getDragPoint(&pDocView->m_pImpl->m_aHandleStartRect, pEvent, &aPoint); pDocView->m_pImpl->m_pDocument->pClass->setTextSelection( pDocView->m_pImpl->m_pDocument, LOK_SETTEXTSELECTION_START, pDocView->m_pImpl->pixelToTwip(aPoint.x), pDocView->m_pImpl->pixelToTwip(aPoint.y)); return FALSE; } if (pDocView->m_pImpl->m_bInDragEndHandle) { g_info("lcl_signalMotion: dragging the end handle"); lcl_getDragPoint(&pDocView->m_pImpl->m_aHandleEndRect, pEvent, &aPoint); pDocView->m_pImpl->m_pDocument->pClass->setTextSelection( pDocView->m_pImpl->m_pDocument, LOK_SETTEXTSELECTION_END, pDocView->m_pImpl->pixelToTwip(aPoint.x), pDocView->m_pImpl->pixelToTwip(aPoint.y)); return FALSE; } for (int i = 0; i < GRAPHIC_HANDLE_COUNT; ++i) { if (pDocView->m_pImpl->m_bInDragGraphicHandles[i]) { g_info("lcl_signalMotion: dragging the graphic handle #%d", i); return FALSE; } } if (pDocView->m_pImpl->m_bInDragGraphicSelection) { g_info("lcl_signalMotion: dragging the graphic selection"); return FALSE; } GdkRectangle aMotionInTwipsInTwips; aMotionInTwipsInTwips.x = pDocView->m_pImpl->pixelToTwip(pEvent->x); aMotionInTwipsInTwips.y = pDocView->m_pImpl->pixelToTwip(pEvent->y); aMotionInTwipsInTwips.width = 1; aMotionInTwipsInTwips.height = 1; if (gdk_rectangle_intersect(&aMotionInTwipsInTwips, &pDocView->m_pImpl->m_aGraphicSelection, 0)) { g_info("lcl_signalMotion: start of drag graphic selection"); pDocView->m_pImpl->m_bInDragGraphicSelection = true; pDocView->m_pImpl->m_pDocument->pClass->setGraphicSelection( pDocView->m_pImpl->m_pDocument, LOK_SETGRAPHICSELECTION_START, pDocView->m_pImpl->pixelToTwip(pEvent->x), pDocView->m_pImpl->pixelToTwip(pEvent->y)); return FALSE; } return FALSE; } /// Receives a button press event. gboolean lcl_signalButton(GtkWidget* /*pEventBox*/, GdkEventButton* pEvent, LOKDocView* pDocView) { g_info("lcl_signalButton: %d, %d (in twips: %d, %d)", (int)pEvent->x, (int)pEvent->y, (int)pDocView->m_pImpl->pixelToTwip(pEvent->x), (int)pDocView->m_pImpl->pixelToTwip(pEvent->y)); if (pEvent->type == GDK_BUTTON_RELEASE) { if (pDocView->m_pImpl->m_bInDragStartHandle) { g_info("lcl_signalButton: end of drag start handle"); pDocView->m_pImpl->m_bInDragStartHandle = false; return FALSE; } else if (pDocView->m_pImpl->m_bInDragMiddleHandle) { g_info("lcl_signalButton: end of drag middle handle"); pDocView->m_pImpl->m_bInDragMiddleHandle = false; return FALSE; } else if (pDocView->m_pImpl->m_bInDragEndHandle) { g_info("lcl_signalButton: end of drag end handle"); pDocView->m_pImpl->m_bInDragEndHandle = false; return FALSE; } for (int i = 0; i < GRAPHIC_HANDLE_COUNT; ++i) { if (pDocView->m_pImpl->m_bInDragGraphicHandles[i]) { g_info("lcl_signalButton: end of drag graphic handle #%d", i); pDocView->m_pImpl->m_bInDragGraphicHandles[i] = false; pDocView->m_pImpl->m_pDocument->pClass->setGraphicSelection( pDocView->m_pImpl->m_pDocument, LOK_SETGRAPHICSELECTION_END, pDocView->m_pImpl->pixelToTwip(pEvent->x), pDocView->m_pImpl->pixelToTwip(pEvent->y)); return FALSE; } } if (pDocView->m_pImpl->m_bInDragGraphicSelection) { g_info("lcl_signalButton: end of drag graphic selection"); pDocView->m_pImpl->m_bInDragGraphicSelection = false; pDocView->m_pImpl->m_pDocument->pClass->setGraphicSelection( pDocView->m_pImpl->m_pDocument, LOK_SETGRAPHICSELECTION_END, pDocView->m_pImpl->pixelToTwip(pEvent->x), pDocView->m_pImpl->pixelToTwip(pEvent->y)); return FALSE; } } if (pDocView->m_pImpl->m_bEdit) { GdkRectangle aClick; aClick.x = pEvent->x; aClick.y = pEvent->y; aClick.width = 1; aClick.height = 1; if (pEvent->type == GDK_BUTTON_PRESS) { if (gdk_rectangle_intersect(&aClick, &pDocView->m_pImpl->m_aHandleStartRect, NULL)) { g_info("lcl_signalButton: start of drag start handle"); pDocView->m_pImpl->m_bInDragStartHandle = true; return FALSE; } else if (gdk_rectangle_intersect(&aClick, &pDocView->m_pImpl->m_aHandleMiddleRect, NULL)) { g_info("lcl_signalButton: start of drag middle handle"); pDocView->m_pImpl->m_bInDragMiddleHandle = true; return FALSE; } else if (gdk_rectangle_intersect(&aClick, &pDocView->m_pImpl->m_aHandleEndRect, NULL)) { g_info("lcl_signalButton: start of drag end handle"); pDocView->m_pImpl->m_bInDragEndHandle = true; return FALSE; } for (int i = 0; i < GRAPHIC_HANDLE_COUNT; ++i) { if (gdk_rectangle_intersect(&aClick, &pDocView->m_pImpl->m_aGraphicHandleRects[i], NULL)) { g_info("lcl_signalButton: start of drag graphic handle #%d", i); pDocView->m_pImpl->m_bInDragGraphicHandles[i] = true; pDocView->m_pImpl->m_pDocument->pClass->setGraphicSelection( pDocView->m_pImpl->m_pDocument, LOK_SETGRAPHICSELECTION_START, pDocView->m_pImpl->pixelToTwip(pDocView->m_pImpl->m_aGraphicHandleRects[i].x + pDocView->m_pImpl->m_aGraphicHandleRects[i].width / 2), pDocView->m_pImpl->pixelToTwip(pDocView->m_pImpl->m_aGraphicHandleRects[i].y + pDocView->m_pImpl->m_aGraphicHandleRects[i].height / 2)); return FALSE; } } } } if (!pDocView->m_pImpl->m_bEdit) lok_docview_set_edit(pDocView, TRUE); switch (pEvent->type) { case GDK_BUTTON_PRESS: { int nCount = 1; if ((pEvent->time - pDocView->m_pImpl->m_nLastButtonPressTime) < 250) nCount++; pDocView->m_pImpl->m_nLastButtonPressTime = pEvent->time; pDocView->m_pImpl->m_pDocument->pClass->postMouseEvent( pDocView->m_pImpl->m_pDocument, LOK_MOUSEEVENT_MOUSEBUTTONDOWN, pDocView->m_pImpl->pixelToTwip(pEvent->x), pDocView->m_pImpl->pixelToTwip(pEvent->y), nCount); break; } case GDK_BUTTON_RELEASE: { int nCount = 1; if ((pEvent->time - pDocView->m_pImpl->m_nLastButtonReleaseTime) < 250) nCount++; pDocView->m_pImpl->m_nLastButtonReleaseTime = pEvent->time; pDocView->m_pImpl->m_pDocument->pClass->postMouseEvent( pDocView->m_pImpl->m_pDocument, LOK_MOUSEEVENT_MOUSEBUTTONUP, pDocView->m_pImpl->pixelToTwip(pEvent->x), pDocView->m_pImpl->pixelToTwip(pEvent->y), nCount); break; } default: break; } return FALSE; } SAL_DLLPUBLIC_EXPORT guint lok_docview_get_type() { static guint lok_docview_type = 0; if (!lok_docview_type) { char pName[] = "LokDocView"; GtkTypeInfo lok_docview_info = { pName, sizeof( LOKDocView ), sizeof( LOKDocViewClass ), lok_docview_class_init, lok_docview_init, NULL, NULL, (GtkClassInitFunc) NULL }; lok_docview_type = gtk_type_unique( gtk_scrolled_window_get_type(), &lok_docview_info ); } return lok_docview_type; } enum { EDIT_CHANGED, LAST_SIGNAL }; static guint docview_signals[LAST_SIGNAL] = { 0 }; static void lok_docview_class_init( gpointer ptr ) { LOKDocViewClass* pClass = static_cast(ptr); GObjectClass *gobject_class = G_OBJECT_CLASS(pClass); pClass->edit_changed = NULL; docview_signals[EDIT_CHANGED] = g_signal_new("edit-changed", G_TYPE_FROM_CLASS (gobject_class), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (LOKDocViewClass, edit_changed), NULL, NULL, g_cclosure_marshal_VOID__BOOLEAN, G_TYPE_NONE, 1, G_TYPE_BOOLEAN); } static void lok_docview_init( GTypeInstance* pInstance, gpointer ) { LOKDocView* pDocView = reinterpret_cast(pInstance); // Gtk ScrolledWindow is apparently not fully initialised yet, we specifically // have to set the [hv]adjustment to prevent GTK assertions from firing, see // https://bugzilla.gnome.org/show_bug.cgi?id=438114 for more info. gtk_scrolled_window_set_hadjustment( GTK_SCROLLED_WINDOW( pDocView ), NULL ); gtk_scrolled_window_set_vadjustment( GTK_SCROLLED_WINDOW( pDocView ), NULL ); pDocView->m_pImpl = new LOKDocView_Impl(); gtk_scrolled_window_add_with_viewport( GTK_SCROLLED_WINDOW(pDocView), pDocView->m_pImpl->m_pEventBox ); gtk_widget_set_events(pDocView->m_pImpl->m_pEventBox, GDK_BUTTON_PRESS_MASK); // So that drag doesn't try to move the whole window. gtk_signal_connect(GTK_OBJECT(pDocView->m_pImpl->m_pEventBox), "button-press-event", GTK_SIGNAL_FUNC(lcl_signalButton), pDocView); gtk_signal_connect(GTK_OBJECT(pDocView->m_pImpl->m_pEventBox), "button-release-event", GTK_SIGNAL_FUNC(lcl_signalButton), pDocView); gtk_signal_connect(GTK_OBJECT(pDocView->m_pImpl->m_pEventBox), "motion-notify-event", GTK_SIGNAL_FUNC(lcl_signalMotion), pDocView); gtk_widget_show( pDocView->m_pImpl->m_pEventBox ); gtk_signal_connect(GTK_OBJECT(pDocView), "destroy", GTK_SIGNAL_FUNC(LOKDocView_Impl::destroy), 0); g_signal_connect_after(pDocView->m_pImpl->m_pEventBox, "expose-event", G_CALLBACK(renderOverlay), pDocView); } SAL_DLLPUBLIC_EXPORT GtkWidget* lok_docview_new( LibreOfficeKit* pOffice ) { LOKDocView* pDocView = LOK_DOCVIEW(gtk_type_new(lok_docview_get_type())); pDocView->m_pImpl->m_pOffice = pOffice; return GTK_WIDGET( pDocView ); } static gboolean lcl_isEmptyRectangle(GdkRectangle* pRectangle) { return pRectangle->x == 0 && pRectangle->y == 0 && pRectangle->width == 0 && pRectangle->height == 0; } /// Takes care of the blinking cursor. static gboolean lcl_handleTimeout(gpointer pData) { LOKDocView* pDocView = LOK_DOCVIEW(pData); if (pDocView->m_pImpl->m_bEdit) { if (pDocView->m_pImpl->m_bCursorOverlayVisible) pDocView->m_pImpl->m_bCursorOverlayVisible = false; else pDocView->m_pImpl->m_bCursorOverlayVisible = true; gtk_widget_queue_draw(GTK_WIDGET(pDocView->m_pImpl->m_pEventBox)); } return G_SOURCE_CONTINUE; } /// Renders pHandle below a pCursor rectangle on pCairo. static void lcl_renderHandle(cairo_t* pCairo, GdkRectangle* pCursor, cairo_surface_t* pHandle, GdkRectangle* pRectangle, LOKDocView* pDocView) { GdkPoint aCursorBottom; int nHandleWidth, nHandleHeight; double fHandleScale; nHandleWidth = cairo_image_surface_get_width(pHandle); nHandleHeight = cairo_image_surface_get_height(pHandle); // We want to scale down the handle, so that its height is the same as the cursor caret. fHandleScale = pDocView->m_pImpl->twipToPixel(pCursor->height) / nHandleHeight; // We want the top center of the handle bitmap to be at the bottom center of the cursor rectangle. aCursorBottom.x = pDocView->m_pImpl->twipToPixel(pCursor->x) + pDocView->m_pImpl->twipToPixel(pCursor->width) / 2 - (nHandleWidth * fHandleScale) / 2; aCursorBottom.y = pDocView->m_pImpl->twipToPixel(pCursor->y) + pDocView->m_pImpl->twipToPixel(pCursor->height); cairo_save(pCairo); cairo_translate(pCairo, aCursorBottom.x, aCursorBottom.y); cairo_scale(pCairo, fHandleScale, fHandleScale); cairo_set_source_surface(pCairo, pHandle, 0, 0); cairo_paint(pCairo); cairo_restore(pCairo); if (pRectangle) { // Return the rectangle that contains the rendered handle. pRectangle->x = aCursorBottom.x; pRectangle->y = aCursorBottom.y; pRectangle->width = nHandleWidth * fHandleScale; pRectangle->height = nHandleHeight * fHandleScale; } } /// Renders pHandle around a pSelection rectangle on pCairo. static void lcl_renderGraphicHandle(cairo_t* pCairo, GdkRectangle* pSelection, cairo_surface_t* pHandle, GdkRectangle* pGraphicHandleRects, LOKDocView* pDocView) { int nHandleWidth, nHandleHeight; GdkRectangle aSelection; nHandleWidth = cairo_image_surface_get_width(pHandle); nHandleHeight = cairo_image_surface_get_height(pHandle); aSelection.x = pDocView->m_pImpl->twipToPixel(pSelection->x); aSelection.y = pDocView->m_pImpl->twipToPixel(pSelection->y); aSelection.width = pDocView->m_pImpl->twipToPixel(pSelection->width); aSelection.height = pDocView->m_pImpl->twipToPixel(pSelection->height); for (int i = 0; i < GRAPHIC_HANDLE_COUNT; ++i) { int x = aSelection.x, y = aSelection.y; cairo_save(pCairo); switch (i) { case 0: // top-left break; case 1: // top-middle x += aSelection.width / 2; break; case 2: // top-right x += aSelection.width; break; case 3: // middle-left y += aSelection.height / 2; break; case 4: // middle-right x += aSelection.width; y += aSelection.height / 2; break; case 5: // bottom-left y += aSelection.height; break; case 6: // bottom-middle x += aSelection.width / 2; y += aSelection.height; break; case 7: // bottom-right x += aSelection.width; y += aSelection.height; break; } // Center the handle. x -= nHandleWidth / 2; y -= nHandleHeight / 2; pGraphicHandleRects[i].x = x; pGraphicHandleRects[i].y = y; pGraphicHandleRects[i].width = nHandleWidth; pGraphicHandleRects[i].height = nHandleHeight; cairo_translate(pCairo, x, y); cairo_set_source_surface(pCairo, pHandle, 0, 0); cairo_paint(pCairo); cairo_restore(pCairo); } } static gboolean renderOverlay(GtkWidget* pWidget, GdkEventExpose* /*pEvent*/, gpointer pData) { #if GTK_CHECK_VERSION(2,14,0) // we need gtk_widget_get_window() LOKDocView* pDocView = LOK_DOCVIEW(pData); cairo_t* pCairo = gdk_cairo_create(gtk_widget_get_window(pWidget)); if (pDocView->m_pImpl->m_bEdit && pDocView->m_pImpl->m_bCursorVisible && pDocView->m_pImpl->m_bCursorOverlayVisible && !lcl_isEmptyRectangle(&pDocView->m_pImpl->m_aVisibleCursor)) { if (pDocView->m_pImpl->m_aVisibleCursor.width < 30) // Set a minimal width if it would be 0. pDocView->m_pImpl->m_aVisibleCursor.width = 30; cairo_set_source_rgb(pCairo, 0, 0, 0); cairo_rectangle(pCairo, pDocView->m_pImpl->twipToPixel(pDocView->m_pImpl->m_aVisibleCursor.x), pDocView->m_pImpl->twipToPixel(pDocView->m_pImpl->m_aVisibleCursor.y), pDocView->m_pImpl->twipToPixel(pDocView->m_pImpl->m_aVisibleCursor.width), pDocView->m_pImpl->twipToPixel(pDocView->m_pImpl->m_aVisibleCursor.height)); cairo_fill(pCairo); } if (pDocView->m_pImpl->m_bEdit && pDocView->m_pImpl->m_bCursorVisible && !lcl_isEmptyRectangle(&pDocView->m_pImpl->m_aVisibleCursor) && !pDocView->m_pImpl->m_pTextSelectionRectangles) { // Have a cursor, but no selection: we need the middle handle. if (!pDocView->m_pImpl->m_pHandleMiddle) pDocView->m_pImpl->m_pHandleMiddle = cairo_image_surface_create_from_png(CURSOR_HANDLE_DIR "handle_middle.png"); lcl_renderHandle(pCairo, &pDocView->m_pImpl->m_aVisibleCursor, pDocView->m_pImpl->m_pHandleMiddle, &pDocView->m_pImpl->m_aHandleMiddleRect, pDocView); } if (pDocView->m_pImpl->m_pTextSelectionRectangles) { for (GList* i = pDocView->m_pImpl->m_pTextSelectionRectangles; i != NULL; i = i->next) { GdkRectangle* pRectangle = static_cast(i->data); // Blue with 75% transparency. cairo_set_source_rgba(pCairo, ((double)0x43)/255, ((double)0xac)/255, ((double)0xe8)/255, 0.25); cairo_rectangle(pCairo, pDocView->m_pImpl->twipToPixel(pRectangle->x), pDocView->m_pImpl->twipToPixel(pRectangle->y), pDocView->m_pImpl->twipToPixel(pRectangle->width), pDocView->m_pImpl->twipToPixel(pRectangle->height)); cairo_fill(pCairo); } // Handles if (!lcl_isEmptyRectangle(&pDocView->m_pImpl->m_aTextSelectionStart)) { // Have a start position: we need a start handle. if (!pDocView->m_pImpl->m_pHandleStart) pDocView->m_pImpl->m_pHandleStart = cairo_image_surface_create_from_png(CURSOR_HANDLE_DIR "handle_start.png"); lcl_renderHandle(pCairo, &pDocView->m_pImpl->m_aTextSelectionStart, pDocView->m_pImpl->m_pHandleStart, &pDocView->m_pImpl->m_aHandleStartRect, pDocView); } if (!lcl_isEmptyRectangle(&pDocView->m_pImpl->m_aTextSelectionEnd)) { // Have a start position: we need an end handle. if (!pDocView->m_pImpl->m_pHandleEnd) pDocView->m_pImpl->m_pHandleEnd = cairo_image_surface_create_from_png(CURSOR_HANDLE_DIR "handle_end.png"); lcl_renderHandle(pCairo, &pDocView->m_pImpl->m_aTextSelectionEnd, pDocView->m_pImpl->m_pHandleEnd, &pDocView->m_pImpl->m_aHandleEndRect, pDocView); } } if (!lcl_isEmptyRectangle(&pDocView->m_pImpl->m_aGraphicSelection)) { if (!pDocView->m_pImpl->m_pGraphicHandle) pDocView->m_pImpl->m_pGraphicHandle = cairo_image_surface_create_from_png(CURSOR_HANDLE_DIR "handle_graphic.png"); lcl_renderGraphicHandle(pCairo, &pDocView->m_pImpl->m_aGraphicSelection, pDocView->m_pImpl->m_pGraphicHandle, pDocView->m_pImpl->m_aGraphicHandleRects, pDocView); } cairo_destroy(pCairo); #endif return FALSE; } void renderDocument(LOKDocView* pDocView, GdkRectangle* pPartial) { const int nTileSizePixels = 256; // Get document size and find out how many rows / columns we need. long nDocumentWidthTwips, nDocumentHeightTwips; pDocView->m_pImpl->m_pDocument->pClass->getDocumentSize(pDocView->m_pImpl->m_pDocument, &nDocumentWidthTwips, &nDocumentHeightTwips); long nDocumentWidthPixels = pDocView->m_pImpl->twipToPixel(nDocumentWidthTwips); long nDocumentHeightPixels = pDocView->m_pImpl->twipToPixel(nDocumentHeightTwips); // Total number of rows / columns in this document. guint nRows = ceil((double)nDocumentHeightPixels / nTileSizePixels); guint nColumns = ceil((double)nDocumentWidthPixels / nTileSizePixels); // Set up our table and the tile pointers. if (!pDocView->m_pImpl->m_pTable) pPartial = 0; if (pPartial) { // Same as nRows / nColumns, but from the previous renderDocument() call. guint nOldRows, nOldColumns; #if GTK_CHECK_VERSION(2,22,0) gtk_table_get_size(GTK_TABLE(pDocView->m_pImpl->m_pTable), &nOldRows, &nOldColumns); if (nOldRows != nRows || nOldColumns != nColumns) // Can't do partial rendering, document size changed. pPartial = 0; #else pPartial = 0; #endif } if (!pPartial) { if (pDocView->m_pImpl->m_pTable) gtk_container_remove(GTK_CONTAINER(pDocView->m_pImpl->m_pEventBox), pDocView->m_pImpl->m_pTable); pDocView->m_pImpl->m_pTable = gtk_table_new(nRows, nColumns, FALSE); gtk_container_add(GTK_CONTAINER(pDocView->m_pImpl->m_pEventBox), pDocView->m_pImpl->m_pTable); gtk_widget_show(pDocView->m_pImpl->m_pTable); if (pDocView->m_pImpl->m_pCanvas) g_free(pDocView->m_pImpl->m_pCanvas); pDocView->m_pImpl->m_pCanvas = static_cast(g_malloc0(sizeof(GtkWidget*) * nRows * nColumns)); } // Render the tiles. for (guint nRow = 0; nRow < nRows; ++nRow) { for (guint nColumn = 0; nColumn < nColumns; ++nColumn) { GdkRectangle aTileRectangleTwips, aTileRectanglePixels; bool bPaint = true; // Determine size of the tile: the rightmost/bottommost tiles may be smaller and we need the size to decide if we need to repaint. if (nColumn == nColumns - 1) aTileRectanglePixels.width = nDocumentWidthPixels - nColumn * nTileSizePixels; else aTileRectanglePixels.width = nTileSizePixels; if (nRow == nRows - 1) aTileRectanglePixels.height = nDocumentHeightPixels - nRow * nTileSizePixels; else aTileRectanglePixels.height = nTileSizePixels; // Determine size and position of the tile in document coordinates, so we can decide if we can skip painting for partial rendering. aTileRectangleTwips.x = pDocView->m_pImpl->pixelToTwip(nTileSizePixels) * nColumn; aTileRectangleTwips.y = pDocView->m_pImpl->pixelToTwip(nTileSizePixels) * nRow; aTileRectangleTwips.width = pDocView->m_pImpl->pixelToTwip(aTileRectanglePixels.width); aTileRectangleTwips.height = pDocView->m_pImpl->pixelToTwip(aTileRectanglePixels.height); if (pPartial && !gdk_rectangle_intersect(pPartial, &aTileRectangleTwips, 0)) bPaint = false; if (bPaint) { // Index of the current tile. guint nTile = nRow * nColumns + nColumn; GdkPixbuf* pPixBuf = gdk_pixbuf_new(GDK_COLORSPACE_RGB, TRUE, 8, aTileRectanglePixels.width, aTileRectanglePixels.height); unsigned char* pBuffer = gdk_pixbuf_get_pixels(pPixBuf); g_info("renderDocument: paintTile(%d, %d)", nRow, nColumn); pDocView->m_pImpl->m_pDocument->pClass->paintTile(pDocView->m_pImpl->m_pDocument, // Buffer and its size, depends on the position only. pBuffer, aTileRectanglePixels.width, aTileRectanglePixels.height, // Position of the tile. aTileRectangleTwips.x, aTileRectangleTwips.y, // Size of the tile, depends on the zoom factor and the tile position only. aTileRectangleTwips.width, aTileRectangleTwips.height); if (pDocView->m_pImpl->m_pCanvas[nTile]) gtk_widget_destroy(GTK_WIDGET(pDocView->m_pImpl->m_pCanvas[nTile])); pDocView->m_pImpl->m_pCanvas[nTile] = gtk_image_new(); gtk_image_set_from_pixbuf(GTK_IMAGE(pDocView->m_pImpl->m_pCanvas[nTile]), pPixBuf); g_object_unref(G_OBJECT(pPixBuf)); gtk_widget_show(pDocView->m_pImpl->m_pCanvas[nTile]); gtk_table_attach(GTK_TABLE(pDocView->m_pImpl->m_pTable), pDocView->m_pImpl->m_pCanvas[nTile], nColumn, nColumn + 1, nRow, nRow + 1, GTK_SHRINK, GTK_SHRINK, 0, 0); } } } } /// Callback data, allocated in lok_docview_callback_worker(), released in lok_docview_callback(). typedef struct { int m_nType; char* m_pPayload; LOKDocView* m_pDocView; } LOKDocViewCallbackData; /// Returns the GdkRectangle of a width,height,x,y string. static GdkRectangle lcl_payloadToRectangle(const char* pPayload) { GdkRectangle aRet; aRet.width = aRet.height = aRet.x = aRet.y = 0; gchar** ppCoordinates = g_strsplit(pPayload, ", ", 4); gchar** ppCoordinate = ppCoordinates; if (!*ppCoordinate) return aRet; aRet.x = atoi(*ppCoordinate); ++ppCoordinate; if (!*ppCoordinate) return aRet; aRet.y = atoi(*ppCoordinate); ++ppCoordinate; if (!*ppCoordinate) return aRet; aRet.width = atoi(*ppCoordinate); ++ppCoordinate; if (!*ppCoordinate) return aRet; aRet.height = atoi(*ppCoordinate); g_strfreev(ppCoordinates); return aRet; } /// Returns the GdkRectangle list of a w,h,x,y;w2,h2,x2,y2;... string. static GList* lcl_payloadToRectangles(const char* pPayload) { GList* pRet = NULL; gchar** ppRectangles = g_strsplit(pPayload, "; ", 0); for (gchar** ppRectangle = ppRectangles; *ppRectangle; ++ppRectangle) { GdkRectangle aRect = lcl_payloadToRectangle(*ppRectangle); GdkRectangle* pRect = g_new0(GdkRectangle, 1); *pRect = aRect; pRet = g_list_prepend(pRet, pRect); } g_strfreev(ppRectangles); return pRet; } static const gchar* lcl_LibreOfficeKitCallbackTypeToString(int nType) { switch (nType) { case LOK_CALLBACK_INVALIDATE_TILES: return "LOK_CALLBACK_INVALIDATE_TILES"; case LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR: return "LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR"; case LOK_CALLBACK_TEXT_SELECTION: return "LOK_CALLBACK_TEXT_SELECTION"; case LOK_CALLBACK_TEXT_SELECTION_START: return "LOK_CALLBACK_TEXT_SELECTION_START"; case LOK_CALLBACK_TEXT_SELECTION_END: return "LOK_CALLBACK_TEXT_SELECTION_END"; case LOK_CALLBACK_CURSOR_VISIBLE: return "LOK_CALLBACK_CURSOR_VISIBLE"; case LOK_CALLBACK_GRAPHIC_SELECTION: return "LOK_CALLBACK_GRAPHIC_SELECTION"; case LOK_CALLBACK_HYPERLINK_CLICKED: return "LOK_CALLBACK_HYPERLINK_CLICKED"; } return 0; } /// Invoked on the main thread if lok_docview_callback_worker() requests so. static gboolean lok_docview_callback(gpointer pData) { #if GLIB_CHECK_VERSION(2,28,0) // we need g_list_free_full() LOKDocViewCallbackData* pCallback = static_cast(pData); switch (pCallback->m_nType) { case LOK_CALLBACK_INVALIDATE_TILES: { if (strcmp(pCallback->m_pPayload, "EMPTY") != 0) { GdkRectangle aRectangle = lcl_payloadToRectangle(pCallback->m_pPayload); renderDocument(pCallback->m_pDocView, &aRectangle); } else renderDocument(pCallback->m_pDocView, NULL); } break; case LOK_CALLBACK_INVALIDATE_VISIBLE_CURSOR: { pCallback->m_pDocView->m_pImpl->m_aVisibleCursor = lcl_payloadToRectangle(pCallback->m_pPayload); pCallback->m_pDocView->m_pImpl->m_bCursorOverlayVisible = true; gtk_widget_queue_draw(GTK_WIDGET(pCallback->m_pDocView->m_pImpl->m_pEventBox)); } break; case LOK_CALLBACK_TEXT_SELECTION: { LOKDocView* pDocView = pCallback->m_pDocView; GList* pRectangles = lcl_payloadToRectangles(pCallback->m_pPayload); if (pDocView->m_pImpl->m_pTextSelectionRectangles) g_list_free_full(pDocView->m_pImpl->m_pTextSelectionRectangles, g_free); pDocView->m_pImpl->m_pTextSelectionRectangles = pRectangles; // In case the selection is empty, then we get no LOK_CALLBACK_TEXT_SELECTION_START/END events. if (pRectangles == NULL) { memset(&pDocView->m_pImpl->m_aTextSelectionStart, 0, sizeof(pDocView->m_pImpl->m_aTextSelectionStart)); memset(&pDocView->m_pImpl->m_aHandleStartRect, 0, sizeof(pDocView->m_pImpl->m_aHandleStartRect)); memset(&pDocView->m_pImpl->m_aTextSelectionEnd, 0, sizeof(pDocView->m_pImpl->m_aTextSelectionEnd)); memset(&pDocView->m_pImpl->m_aHandleEndRect, 0, sizeof(pDocView->m_pImpl->m_aHandleEndRect)); } else memset(&pDocView->m_pImpl->m_aHandleMiddleRect, 0, sizeof(pDocView->m_pImpl->m_aHandleMiddleRect)); gtk_widget_queue_draw(GTK_WIDGET(pDocView->m_pImpl->m_pEventBox)); } break; case LOK_CALLBACK_TEXT_SELECTION_START: { pCallback->m_pDocView->m_pImpl->m_aTextSelectionStart = lcl_payloadToRectangle(pCallback->m_pPayload); } break; case LOK_CALLBACK_TEXT_SELECTION_END: { pCallback->m_pDocView->m_pImpl->m_aTextSelectionEnd = lcl_payloadToRectangle(pCallback->m_pPayload); } break; case LOK_CALLBACK_CURSOR_VISIBLE: { pCallback->m_pDocView->m_pImpl->m_bCursorVisible = strcmp(pCallback->m_pPayload, "true") == 0; } break; case LOK_CALLBACK_GRAPHIC_SELECTION: { if (strcmp(pCallback->m_pPayload, "EMPTY") != 0) pCallback->m_pDocView->m_pImpl->m_aGraphicSelection = lcl_payloadToRectangle(pCallback->m_pPayload); else memset(&pCallback->m_pDocView->m_pImpl->m_aGraphicSelection, 0, sizeof(pCallback->m_pDocView->m_pImpl->m_aGraphicSelection)); gtk_widget_queue_draw(GTK_WIDGET(pCallback->m_pDocView->m_pImpl->m_pEventBox)); } break; case LOK_CALLBACK_HYPERLINK_CLICKED: { GError* pError = NULL; gtk_show_uri(NULL, pCallback->m_pPayload, GDK_CURRENT_TIME, &pError); } break; default: g_assert(false); break; } g_free(pCallback->m_pPayload); g_free(pCallback); #endif return G_SOURCE_REMOVE; } /// Our LOK callback, runs on the LO thread. static void lok_docview_callback_worker(int nType, const char* pPayload, void* pData) { LOKDocView* pDocView = static_cast(pData); LOKDocViewCallbackData* pCallback = g_new0(LOKDocViewCallbackData, 1); pCallback->m_nType = nType; pCallback->m_pPayload = g_strdup(pPayload); pCallback->m_pDocView = pDocView; g_info("lok_docview_callback_worker: %s, '%s'", lcl_LibreOfficeKitCallbackTypeToString(nType), pPayload); #if GTK_CHECK_VERSION(2,12,0) gdk_threads_add_idle(lok_docview_callback, pCallback); #else g_idle_add(lok_docview_callback, pDocView); #endif } SAL_DLLPUBLIC_EXPORT gboolean lok_docview_open_document( LOKDocView* pDocView, char* pPath ) { if ( pDocView->m_pImpl->m_pDocument ) { pDocView->m_pImpl->m_pDocument->pClass->destroy( pDocView->m_pImpl->m_pDocument ); pDocView->m_pImpl->m_pDocument = 0; } pDocView->m_pImpl->m_pDocument = pDocView->m_pImpl->m_pOffice->pClass->documentLoad( pDocView->m_pImpl->m_pOffice, pPath ); if ( !pDocView->m_pImpl->m_pDocument ) { // FIXME: should have a GError parameter and populate it. char *pError = pDocView->m_pImpl->m_pOffice->pClass->getError( pDocView->m_pImpl->m_pOffice ); fprintf( stderr, "Error opening document '%s'\n", pError ); return FALSE; } else { pDocView->m_pImpl->m_pDocument->pClass->initializeForRendering(pDocView->m_pImpl->m_pDocument); pDocView->m_pImpl->m_pDocument->pClass->registerCallback(pDocView->m_pImpl->m_pDocument, &lok_docview_callback_worker, pDocView); g_timeout_add(600, &lcl_handleTimeout, pDocView); renderDocument(pDocView, NULL); } return TRUE; } SAL_DLLPUBLIC_EXPORT LibreOfficeKitDocument* lok_docview_get_document(LOKDocView* pDocView) { return pDocView->m_pImpl->m_pDocument; } SAL_DLLPUBLIC_EXPORT void lok_docview_set_zoom ( LOKDocView* pDocView, float fZoom ) { pDocView->m_pImpl->m_fZoom = fZoom; if ( pDocView->m_pImpl->m_pDocument ) { renderDocument(pDocView, 0); } } SAL_DLLPUBLIC_EXPORT float lok_docview_get_zoom ( LOKDocView* pDocView ) { return pDocView->m_pImpl->m_fZoom; } SAL_DLLPUBLIC_EXPORT int lok_docview_get_parts( LOKDocView* pDocView ) { return pDocView->m_pImpl->m_pDocument->pClass->getParts( pDocView->m_pImpl->m_pDocument ); } SAL_DLLPUBLIC_EXPORT int lok_docview_get_part( LOKDocView* pDocView ) { return pDocView->m_pImpl->m_pDocument->pClass->getPart( pDocView->m_pImpl->m_pDocument ); } SAL_DLLPUBLIC_EXPORT void lok_docview_set_part( LOKDocView* pDocView, int nPart) { pDocView->m_pImpl->m_pDocument->pClass->setPart( pDocView->m_pImpl->m_pDocument, nPart ); renderDocument(pDocView, NULL); } SAL_DLLPUBLIC_EXPORT char* lok_docview_get_part_name( LOKDocView* pDocView, int nPart ) { return pDocView->m_pImpl->m_pDocument->pClass->getPartName( pDocView->m_pImpl->m_pDocument, nPart ); } SAL_DLLPUBLIC_EXPORT void lok_docview_set_partmode( LOKDocView* pDocView, int nPartMode ) { pDocView->m_pImpl->m_pDocument->pClass->setPartMode( pDocView->m_pImpl->m_pDocument, nPartMode ); renderDocument(pDocView, NULL); } SAL_DLLPUBLIC_EXPORT void lok_docview_set_edit( LOKDocView* pDocView, gboolean bEdit ) { gboolean bWasEdit = pDocView->m_pImpl->m_bEdit; if (!pDocView->m_pImpl->m_bEdit && bEdit) g_info("lok_docview_set_edit: entering edit mode"); else if (pDocView->m_pImpl->m_bEdit && !bEdit) { g_info("lok_docview_set_edit: leaving edit mode"); pDocView->m_pImpl->m_pDocument->pClass->resetSelection(pDocView->m_pImpl->m_pDocument); } pDocView->m_pImpl->m_bEdit = bEdit; g_signal_emit(pDocView, docview_signals[EDIT_CHANGED], 0, bWasEdit); gtk_widget_queue_draw(GTK_WIDGET(pDocView->m_pImpl->m_pEventBox)); } SAL_DLLPUBLIC_EXPORT gboolean lok_docview_get_edit(LOKDocView* pDocView) { return pDocView->m_pImpl->m_bEdit; } SAL_DLLPUBLIC_EXPORT void lok_docview_post_command(LOKDocView* pDocView, const char* pCommand) { pDocView->m_pImpl->m_pDocument->pClass->postUnoCommand(pDocView->m_pImpl->m_pDocument, pCommand); } SAL_DLLPUBLIC_EXPORT void lok_docview_post_key(GtkWidget* /*pWidget*/, GdkEventKey* pEvent, gpointer pData) { LOKDocView* pDocView = static_cast(pData); pDocView->m_pImpl->signalKey(pEvent); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */