diff --git a/frontends/riscos/treeview.c b/frontends/riscos/treeview.c
index 17981466f..8dfd16b26 100644
--- a/frontends/riscos/treeview.c
+++ b/frontends/riscos/treeview.c
@@ -17,8 +17,9 @@
* along with this program. If not, see .
*/
-/** \file
- * Generic tree handling (implementation).
+/**
+ * \file
+ * Generic tree handling implementation.
*/
#include
@@ -86,121 +87,6 @@ struct ro_treeview
struct ro_treeview_callbacks *callbacks; /*< Callback handlers */
};
-static void ro_treeview_redraw_request(int x, int y, int width, int height,
- void *pw);
-static void ro_treeview_resized(struct tree *tree, int width, int height,
- void *pw);
-static void ro_treeview_scroll_visible(int y, int height, void *pw);
-static void ro_treeview_get_window_dimensions(int *width, int *height,
- void *pw);
-
-static void ro_treeview_redraw(wimp_draw *redraw);
-static void ro_treeview_scroll(wimp_scroll *scroll);
-static void ro_treeview_redraw_loop(wimp_draw *redraw, ro_treeview *tv,
- osbool more);
-static void ro_treeview_open(wimp_open *open);
-static bool ro_treeview_mouse_click(wimp_pointer *pointer);
-static void ro_treeview_pointer_entering(wimp_entering *entering);
-static void ro_treeview_drag_start(ro_treeview *tv, wimp_pointer *pointer,
- wimp_window_state *state);
-static void ro_treeview_drag_end(wimp_dragged *drag, void *data);
-static bool ro_treeview_keypress(wimp_key *key);
-
-static void ro_treeview_set_window_extent(ro_treeview *tv,
- int width, int height);
-
-static void ro_treeview_update_theme(void *data, bool ok);
-static void ro_treeview_update_toolbar(void *data);
-static void ro_treeview_button_update(void *data);
-static void ro_treeview_save_toolbar_buttons(void *data, char *config);
-static void ro_treeview_button_click(void *data,
- toolbar_action_type action_type, union toolbar_action action);
-
-static const struct treeview_table ro_tree_callbacks = {
- ro_treeview_redraw_request,
- ro_treeview_resized,
- ro_treeview_scroll_visible,
- ro_treeview_get_window_dimensions
-};
-
-static const struct toolbar_callbacks ro_treeview_toolbar_callbacks = {
- ro_treeview_update_theme,
- ro_treeview_update_toolbar,
- ro_treeview_button_update,
- ro_treeview_button_click,
- NULL, /* No toolbar keypress handler */
- ro_treeview_save_toolbar_buttons
-};
-
-
-/**
- * Create a RISC OS GUI implementation of a treeview tree.
- *
- * \param window The window to create the tree in.
- * \param *toolbar A toolbar to attach to the window.
- * \param *callbacks Callbacks to service the treeview.
- * \param flags The treeview flags.
- *
- * \return The RISC OS treeview pointer.
- */
-
-ro_treeview *ro_treeview_create(wimp_w window, struct toolbar *toolbar,
- struct ro_treeview_callbacks *callbacks, unsigned int flags)
-{
- ro_treeview *tv;
-
- /* Claim memory for the treeview block, and create a tree. */
-
- tv = malloc(sizeof(ro_treeview));
- if (tv == NULL)
- return NULL;
-
- tv->w = window;
- tv->tb = toolbar;
-
- /* Set the tree redraw origin at a default 0,0 RO units. */
-
- tv->origin.x = 0;
- tv->origin.y = 0;
-
- /* Set the tree size as 0,0 to indicate that we don't know. */
-
- tv->size.x = 0;
- tv->size.y = 0;
-
- /* Set the tree window extent to 0,0, to indicate that we
- * don't know. */
-
- tv->extent.x = 0;
- tv->extent.y = 0;
-
- /* Set that there is no drag opperation at the moment */
-
- tv->drag = TREE_NO_DRAG;
-
- tv->tree = tree_create(flags, &ro_tree_callbacks, tv);
- if (tv->tree == NULL) {
- free(tv);
- return NULL;
- }
-
- /* Record the callback info. */
-
- tv->callbacks = callbacks;
-
- /* Register wimp events to handle the supplied window. */
-
- ro_gui_wimp_event_register_redraw_window(tv->w, ro_treeview_redraw);
- ro_gui_wimp_event_register_scroll_window(tv->w, ro_treeview_scroll);
- ro_gui_wimp_event_register_pointer_entering_window(tv->w,
- ro_treeview_pointer_entering);
- ro_gui_wimp_event_register_open_window(tv->w, ro_treeview_open);
- ro_gui_wimp_event_register_mouse_click(tv->w, ro_treeview_mouse_click);
- ro_gui_wimp_event_register_keypress(tv->w, ro_treeview_keypress);
- ro_gui_wimp_event_set_user_data(tv->w, tv);
-
- return tv;
-}
/**
* Delete a RISC OS GUI implementation of a treeview tree. The window is
@@ -208,7 +94,6 @@ ro_treeview *ro_treeview_create(wimp_w window, struct toolbar *toolbar,
*
* \param tv The RISC OS treeview to delete.
*/
-
void ro_treeview_destroy(ro_treeview *tv)
{
ro_gui_wimp_event_finalise(tv->w);
@@ -218,46 +103,6 @@ void ro_treeview_destroy(ro_treeview *tv)
free(tv);
}
-/**
- * Return a pointer to a toolbar callbacks structure with the handlers to be
- * used by any treeview window toolbars.
- *
- * \return A pointer to the callback structure.
- */
-
-const struct toolbar_callbacks *ro_treeview_get_toolbar_callbacks(void)
-{
- return &ro_treeview_toolbar_callbacks;
-}
-
-/**
- * Change the redraw origin of a treeview tree in RISC OS graphics units.
- *
- * \param *tv The ro_treeview object to update.
- * \param x The X position, in terms of the RO window work area.
- * \param y The Y position, in terms of the RO window work area.
- *
- * \todo -- this probably needs a rework.
- */
-
-void ro_treeview_set_origin(ro_treeview *tv, int x, int y)
-{
- if (tv != NULL) {
- tv->origin.x = x;
- tv->origin.y = y;
-
- /* Assuming that we know how big the tree currently is, then
- * adjust the window work area extent to match. If we don't,
- * then presumably the tree isn't in an open window yet and
- * a subsequent Open Window Event should pick it up.
- */
-
- if (tv->size.x != 0 && tv->size.y != 0)
- ro_treeview_set_window_extent(tv,
- tv->origin.x + tv->size.x,
- tv->origin.y + tv->size.y);
- }
-}
/**
* Return details of the tree block associated with an ro_treeview object.
@@ -265,12 +110,12 @@ void ro_treeview_set_origin(ro_treeview *tv, int x, int y)
* \param *tv The ro_treeview object of interest.
* \return A pointer to the associated tree block.
*/
-
struct tree *ro_treeview_get_tree(ro_treeview *tv)
{
return (tv != NULL) ? (tv->tree) : (NULL);
}
+
/**
* Return details of the RISC OS window handle associated with an
* ro_treeview object.
@@ -278,85 +123,18 @@ struct tree *ro_treeview_get_tree(ro_treeview *tv)
* \param *tv The ro_treeview object of interest.
* \return The associated RISC OS window handle.
*/
-
wimp_w ro_treeview_get_window(ro_treeview *tv)
{
return (tv != NULL) ? (tv->w) : (NULL);
}
-/**
- * Callback to force a redraw of part of the treeview window.
- *
- * \param x Min X Coordinate of area to be redrawn.
- * \param y Min Y Coordinate of area to be redrawn.
- * \param width Width of area to be redrawn.
- * \param height Height of area to be redrawn.
- * \param pw The treeview object to be redrawn.
- */
-
-void ro_treeview_redraw_request(int x, int y, int width, int height,
- void *pw)
-{
- if (pw != NULL) {
- ro_treeview *tv = (ro_treeview *) pw;
- os_error *error;
- wimp_draw update;
- osbool more;
-
- update.w = tv->w;
- update.box.x0 = (2 * x) + tv->origin.x;
- update.box.y0 = (-2 * (y + height)) + tv->origin.y;
- update.box.x1 = (2 * (x + width)) + tv->origin.x;
- update.box.y1 = (-2 * y) + tv->origin.y;
-
- error = xwimp_update_window(&update, &more);
- if (error) {
- LOG("xwimp_update_window: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
- ro_treeview_redraw_loop(&update, tv, more);
- }
-}
-
-/**
- * Pass RISC OS redraw events on to the treeview widget.
- *
- * \param *redraw Pointer to Redraw Event block.
- */
-
-void ro_treeview_redraw(wimp_draw *redraw)
-{
- osbool more;
- os_error *error;
- ro_treeview *tv;
-
- tv = (ro_treeview *) ro_gui_wimp_event_get_user_data(redraw->w);
- if (tv == NULL) {
- LOG("NULL treeview block for window: 0x%x", (unsigned int)redraw->w);
- /* Don't return, as not servicing redraw events isn't a good
- * idea. The following code must handle (tv == NULL)
- * gracefully while clearing the redraw queue.
- */
- }
-
- error = xwimp_redraw_window(redraw, &more);
- if (error) {
- LOG("xwimp_redraw_window: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
-
- ro_treeview_redraw_loop(redraw, tv, more);
-}
/**
* Handle scroll events in treeview windows.
*
* \param *scroll Pointer to Scroll Event block.
*/
-
-void ro_treeview_scroll(wimp_scroll *scroll)
+static void ro_treeview_scroll(wimp_scroll *scroll)
{
os_error *error;
int x = scroll->visible.x1 - scroll->visible.x0 - 32;
@@ -420,8 +198,8 @@ void ro_treeview_scroll(wimp_scroll *scroll)
* /param *tv The treeview object being redrawn.
* /param more Flag to show if more actions are required.
*/
-
-void ro_treeview_redraw_loop(wimp_draw *redraw, ro_treeview *tv, osbool more)
+static void
+ro_treeview_redraw_loop(wimp_draw *redraw, ro_treeview *tv, osbool more)
{
struct redraw_context ctx = {
.interactive = true,
@@ -452,7 +230,7 @@ void ro_treeview_redraw_loop(wimp_draw *redraw, ro_treeview *tv, osbool more)
(redraw->clip.y1 - redraw->clip.y0)/2,
&ctx);
no_font_blending = false;
- }
+ }
error = xwimp_get_rectangle(redraw, &more);
if (error) {
@@ -463,6 +241,306 @@ void ro_treeview_redraw_loop(wimp_draw *redraw, ro_treeview *tv, osbool more)
}
}
+
+/**
+ * Pass RISC OS redraw events on to the treeview widget.
+ *
+ * \param *redraw Pointer to Redraw Event block.
+ */
+static void ro_treeview_redraw(wimp_draw *redraw)
+{
+ osbool more;
+ os_error *error;
+ ro_treeview *tv;
+
+ tv = (ro_treeview *) ro_gui_wimp_event_get_user_data(redraw->w);
+ if (tv == NULL) {
+ LOG("NULL treeview block for window: 0x%x", (unsigned int)redraw->w);
+ /* Don't return, as not servicing redraw events isn't a good
+ * idea. The following code must handle (tv == NULL)
+ * gracefully while clearing the redraw queue.
+ */
+ }
+
+ error = xwimp_redraw_window(redraw, &more);
+ if (error) {
+ LOG("xwimp_redraw_window: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+
+ ro_treeview_redraw_loop(redraw, tv, more);
+}
+
+
+/**
+ * Callback to force a redraw of part of the treeview window.
+ *
+ * \param x Min X Coordinate of area to be redrawn.
+ * \param y Min Y Coordinate of area to be redrawn.
+ * \param width Width of area to be redrawn.
+ * \param height Height of area to be redrawn.
+ * \param pw The treeview object to be redrawn.
+ */
+static void
+ro_treeview_redraw_request(int x, int y, int width, int height, void *pw)
+{
+ ro_treeview *tv = (ro_treeview *) pw;
+ os_error *error;
+ wimp_draw update;
+ osbool more;
+
+ if (pw == NULL) {
+ return;
+ }
+
+ update.w = tv->w;
+ update.box.x0 = (2 * x) + tv->origin.x;
+ update.box.y0 = (-2 * (y + height)) + tv->origin.y;
+ update.box.x1 = (2 * (x + width)) + tv->origin.x;
+ update.box.y1 = (-2 * y) + tv->origin.y;
+
+ error = xwimp_update_window(&update, &more);
+ if (error) {
+ LOG("xwimp_update_window: 0x%x: %s",
+ error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+ ro_treeview_redraw_loop(&update, tv, more);
+}
+
+
+/**
+ * Callback to request that a section of the tree is scrolled into view.
+ *
+ * \param y The Y coordinate of top of the area in NS units.
+ * \param height The height of the area in NS units.
+ * \param *pw The treeview object affected.
+ */
+static void ro_treeview_scroll_visible(int y, int height, void *pw)
+{
+ ro_treeview *tv = (ro_treeview *) pw;
+ os_error *error;
+ wimp_window_state state;
+ int visible_t, visible_b;
+ int request_t, request_b;
+
+ if (pw == NULL) {
+ return;
+ }
+
+ state.w = tv->w;
+ error = xwimp_get_window_state(&state);
+ if (error) {
+ LOG("xwimp_get_window_state: 0x%x: %s",
+ error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+
+ /* Work out top and bottom of both the currently visible and
+ * the required areas, in terms of the RO work area.
+ */
+
+ visible_t = state.yscroll;
+ visible_b = state.yscroll
+ - (state.visible.y1 - state.visible.y0);
+
+ request_t = -(2 * y);// - tv->origin.y;
+ request_b = -(2 * (y + height));// - tv->origin.y;
+
+ /* If the area is outside the visible window, then scroll it
+ * in to view.
+ */
+
+ if (request_t > visible_t || request_b < visible_b) {
+ if (request_t > visible_t) {
+ state.yscroll = request_t;
+ } else if (request_b < visible_b) {
+ state.yscroll = request_b + tv->origin.y
+ + (state.visible.y1 - state.visible.y0);
+
+ /* If the required area is bigger than the
+ * visible extent, then align to the top and
+ * let the bottom disappear out of view.
+ */
+
+ if (state.yscroll < request_t)
+ state.yscroll = request_t;
+ }
+
+ error = xwimp_open_window((wimp_open *) &state);
+ if (error) {
+ LOG("xwimp_open_window: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+ }
+
+}
+
+
+/**
+ * Callback to return the tree window dimensions to the treeview system.
+ *
+ * \param *width Return the window width.
+ * \param *height Return the window height.
+ * \param *pw The treeview object to use.
+ */
+static void ro_treeview_get_window_dimensions(int *width, int *height,
+ void *pw)
+{
+ if (pw != NULL && (width != NULL || height != NULL)) {
+ ro_treeview *tv = (ro_treeview *) pw;
+ os_error *error;
+ wimp_window_state state;
+
+ state.w = tv->w;
+ error = xwimp_get_window_state(&state);
+ if (error) {
+ LOG("xwimp_get_window_state: 0x%x: %s",
+ error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+
+ if (width != NULL)
+ *width = (state.visible.x1 - state.visible.x0) / 2;
+
+ if (height != NULL)
+ *height = (state.visible.y1 - state.visible.y0) / 2;
+ }
+}
+
+
+/**
+ * Resize the RISC OS window extent of a treeview.
+ *
+ * \param *tv The RISC OS treeview object to resize.
+ * \param width The new width of the work area, in RO units.
+ * \param height The new height of the work area, in RO units.
+ */
+static void
+ro_treeview_set_window_extent(ro_treeview *tv, int width, int height)
+{
+ os_error *error;
+ os_box extent;
+ wimp_window_state state;
+ int new_x, new_y;
+ int visible_x, visible_y;
+
+ if (tv == NULL) {
+ return;
+ }
+
+ /* Calculate the new window extents, in RISC OS units. */
+
+ new_x = width + tv->origin.x;
+ new_y = height + tv->origin.y;
+
+ /* Get details of the existing window, and start to sanity
+ * check the new extents.
+ */
+
+ state.w = tv->w;
+ error = xwimp_get_window_state(&state);
+ if (error) {
+ LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+
+ /* If the extent is smaller than the current visible area,
+ * then extend it so that it matches the visible area.
+ */
+
+ if (new_x < (state.visible.x1 - state.visible.x0))
+ new_x = state.visible.x1 - state.visible.x0;
+
+ if (new_y > (state.visible.y0 - state.visible.y1))
+ new_y = state.visible.y0 - state.visible.y1;
+
+ /* Calculate the maximum visible coordinates of the existing
+ * window.
+ */
+
+ visible_x = state.xscroll +
+ (state.visible.x1 - state.visible.x0);
+ visible_y = state.yscroll +
+ (state.visible.y0 - state.visible.y1);
+
+ /* If the window is currently open, and the exising visible
+ * area is bigger than the new extent, then we need to reopen
+ * the window in an appropriare position before setting the
+ * new extent.
+ */
+
+ if ((state.flags & wimp_WINDOW_OPEN) &&
+ (visible_x > new_x || visible_y < new_y)) {
+ int new_x_scroll = state.xscroll;
+ int new_y_scroll = state.yscroll;
+
+ if (visible_x > new_x)
+ new_x_scroll = new_x - (state.visible.x1
+ - state.visible.x0);
+
+ if (visible_y < new_y)
+ new_y_scroll = new_y - (state.visible.y0
+ - state.visible.y1);
+
+ if (new_x_scroll < 0) {
+ state.visible.x1 -= new_x_scroll;
+ state.xscroll = 0;
+ } else {
+ state.xscroll = new_x_scroll;
+ }
+
+ if (new_y_scroll > 0) {
+ state.visible.y0 += new_y_scroll;
+ state.yscroll = 0;
+ } else {
+ state.yscroll = new_y_scroll;
+ }
+
+ error = xwimp_open_window((wimp_open *) &state);
+ if (error) {
+ LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+
+ /* \todo -- Not sure if we need to reattach the
+ * toolbar here: the nested wimp seems to take care
+ * of it for us?
+ */
+ }
+
+ /* Now that the new extent fits into the visible window, we
+ * can resize the work area. If we succeed, the values are
+ * recorded to save having to ask the Wimp for them
+ * each time.
+ */
+
+ extent.x0 = 0;
+ extent.y0 = new_y;
+ extent.x1 = new_x;
+ extent.y1 = 0;
+
+ error = xwimp_set_extent(tv->w, &extent);
+ if (error) {
+ LOG("xwimp_set_extent: 0x%x: %s",
+ error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+
+ tv->extent.x = new_x;
+ tv->extent.y = new_y;
+}
+
+
+
/**
* Callback to notify us of a new overall tree size.
*
@@ -471,9 +549,8 @@ void ro_treeview_redraw_loop(wimp_draw *redraw, ro_treeview *tv, osbool more)
* \param height The new height of the window.
* \param *pw The treeview object to be resized.
*/
-
-void ro_treeview_resized(struct tree *tree, int width, int height,
- void *pw)
+static void
+ro_treeview_resized(struct tree *tree, int width, int height, void *pw)
{
if (pw != NULL) {
ro_treeview *tv = (ro_treeview *) pw;
@@ -489,232 +566,29 @@ void ro_treeview_resized(struct tree *tree, int width, int height,
}
}
-/**
- * Callback to request that a section of the tree is scrolled into view.
- *
- * \param y The Y coordinate of top of the area in NS units.
- * \param height The height of the area in NS units.
- * \param *pw The treeview object affected.
- */
-
-void ro_treeview_scroll_visible(int y, int height, void *pw)
-{
- if (pw != NULL) {
- ro_treeview *tv = (ro_treeview *) pw;
- os_error *error;
- wimp_window_state state;
- int visible_t, visible_b;
- int request_t, request_b;
-
- state.w = tv->w;
- error = xwimp_get_window_state(&state);
- if (error) {
- LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
-
- /* Work out top and bottom of both the currently visible and
- * the required areas, in terms of the RO work area.
- */
-
- visible_t = state.yscroll;
- visible_b = state.yscroll
- - (state.visible.y1 - state.visible.y0);
-
- request_t = -(2 * y);// - tv->origin.y;
- request_b = -(2 * (y + height));// - tv->origin.y;
-
- /* If the area is outside the visible window, then scroll it
- * in to view.
- */
-
- if (request_t > visible_t || request_b < visible_b) {
- if (request_t > visible_t) {
- state.yscroll = request_t;
- } else if (request_b < visible_b) {
- state.yscroll = request_b + tv->origin.y
- + (state.visible.y1 - state.visible.y0);
-
- /* If the required area is bigger than the
- * visible extent, then align to the top and
- * let the bottom disappear out of view.
- */
-
- if (state.yscroll < request_t)
- state.yscroll = request_t;
- }
-
- error = xwimp_open_window((wimp_open *) &state);
- if (error) {
- LOG("xwimp_open_window: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
- }
- }
-}
/**
- * Callback to return the tree window dimensions to the treeview system.
+ * Handle Pointer Entering Window events for treeview windows.
*
- * \param *width Return the window width.
- * \param *height Return the window height.
- * \param *pw The treeview object to use.
+ * \param *entering The Wimp_PointerEnteringWindow block.
*/
-
-void ro_treeview_get_window_dimensions(int *width, int *height,
- void *pw)
+static void ro_treeview_pointer_entering(wimp_entering *entering)
{
- if (pw != NULL && (width != NULL || height != NULL)) {
- ro_treeview *tv = (ro_treeview *) pw;
- os_error *error;
- wimp_window_state state;
+ ro_treeview *tv;
- state.w = tv->w;
- error = xwimp_get_window_state(&state);
- if (error) {
- LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
+ tv = (ro_treeview *) ro_gui_wimp_event_get_user_data(entering->w);
+ if (tv == NULL)
+ return;
- if (width != NULL)
- *width = (state.visible.x1 - state.visible.x0) / 2;
-
- if (height != NULL)
- *height = (state.visible.y1 - state.visible.y0) / 2;
- }
+ ro_mouse_track_start(NULL, ro_treeview_mouse_at, NULL);
}
-/**
- * Resize the RISC OS window extent of a treeview.
- *
- * \param *tv The RISC OS treeview object to resize.
- * \param width The new width of the work area, in RO units.
- * \param height The new height of the work area, in RO units.
- */
-
-void ro_treeview_set_window_extent(ro_treeview *tv, int width, int height)
-{
- if (tv != NULL) {
- os_error *error;
- os_box extent;
- wimp_window_state state;
- int new_x, new_y;
- int visible_x, visible_y;
-
- /* Calculate the new window extents, in RISC OS units. */
-
- new_x = width + tv->origin.x;
- new_y = height + tv->origin.y;
-
- /* Get details of the existing window, and start to sanity
- * check the new extents.
- */
-
- state.w = tv->w;
- error = xwimp_get_window_state(&state);
- if (error) {
- LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
-
- /* If the extent is smaller than the current visible area,
- * then extend it so that it matches the visible area.
- */
-
- if (new_x < (state.visible.x1 - state.visible.x0))
- new_x = state.visible.x1 - state.visible.x0;
-
- if (new_y > (state.visible.y0 - state.visible.y1))
- new_y = state.visible.y0 - state.visible.y1;
-
- /* Calculate the maximum visible coordinates of the existing
- * window.
- */
-
- visible_x = state.xscroll +
- (state.visible.x1 - state.visible.x0);
- visible_y = state.yscroll +
- (state.visible.y0 - state.visible.y1);
-
- /* If the window is currently open, and the exising visible
- * area is bigger than the new extent, then we need to reopen
- * the window in an appropriare position before setting the
- * new extent.
- */
-
- if ((state.flags & wimp_WINDOW_OPEN) &&
- (visible_x > new_x || visible_y < new_y)) {
- int new_x_scroll = state.xscroll;
- int new_y_scroll = state.yscroll;
-
- if (visible_x > new_x)
- new_x_scroll = new_x - (state.visible.x1
- - state.visible.x0);
-
- if (visible_y < new_y)
- new_y_scroll = new_y - (state.visible.y0
- - state.visible.y1);
-
- if (new_x_scroll < 0) {
- state.visible.x1 -= new_x_scroll;
- state.xscroll = 0;
- } else {
- state.xscroll = new_x_scroll;
- }
-
- if (new_y_scroll > 0) {
- state.visible.y0 += new_y_scroll;
- state.yscroll = 0;
- } else {
- state.yscroll = new_y_scroll;
- }
-
- error = xwimp_open_window((wimp_open *) &state);
- if (error) {
- LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
-
- /* \todo -- Not sure if we need to reattach the
- * toolbar here: the nested wimp seems to take care
- * of it for us?
- */
- }
-
- /* Now that the new extent fits into the visible window, we
- * can resize the work area. If we succeed, the values are
- * recorded to save having to ask the Wimp for them
- * each time.
- */
-
- extent.x0 = 0;
- extent.y0 = new_y;
- extent.x1 = new_x;
- extent.y1 = 0;
-
- error = xwimp_set_extent(tv->w, &extent);
- if (error) {
- LOG("xwimp_set_extent: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
-
- tv->extent.x = new_x;
- tv->extent.y = new_y;
- }
-}
/**
* Handle RISC OS Window Open events for a treeview window.
*
* \param *open Pointer to the Window Open Event block.
*/
-
static void ro_treeview_open(wimp_open *open)
{
ro_treeview *tv;
@@ -724,7 +598,8 @@ static void ro_treeview_open(wimp_open *open)
tv = (ro_treeview *) ro_gui_wimp_event_get_user_data(open->w);
if (tv == NULL) {
- LOG("NULL treeview block for window: ox%x", (unsigned int)open->w);
+ LOG("NULL treeview block for window: ox%x",
+ (unsigned int)open->w);
return;
}
@@ -765,7 +640,8 @@ static void ro_treeview_open(wimp_open *open)
error = xwimp_open_window(open);
if (error) {
- LOG("xwimp_open_window: 0x%x: %s", error->errnum, error->errmess);
+ LOG("xwimp_open_window: 0x%x: %s",
+ error->errnum, error->errmess);
ro_warn_user("WimpError", error->errmess);
}
@@ -774,13 +650,113 @@ static void ro_treeview_open(wimp_open *open)
}
+/**
+ * Process RISC OS User Drag Box events which relate to us: in effect, drags
+ * started by ro_treeview_drag_start().
+ *
+ * \param *drag Pointer to the User Drag Box Event block.
+ * \param *data NULL to allow use as a ro_mouse callback.
+ */
+static void ro_treeview_drag_end(wimp_dragged *drag, void *data)
+{
+ os_error *error;
+
+ error = xwimp_drag_box((wimp_drag *) -1);
+ if (error) {
+ LOG("xwimp_drag_box: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ }
+
+ error = xwimp_auto_scroll(0, NULL, NULL);
+ if (error) {
+ LOG("xwimp_auto_scroll: 0x%x: %s",
+ error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ }
+}
+
+
+/**
+ * Start a RISC OS drag event to reflect on screen what is happening
+ * during the core tree drag.
+ *
+ * \param *tv The RO treeview to which the drag is attached.
+ * \param *pointer The RO pointer event data block starting the drag.
+ * \param *state The RO window state block for the treeview window.
+ */
+static void ro_treeview_drag_start(ro_treeview *tv, wimp_pointer *pointer,
+ wimp_window_state *state)
+{
+ os_error *error;
+ wimp_drag drag;
+ wimp_auto_scroll_info auto_scroll;
+
+ drag.w = tv->w;
+ drag.bbox.x0 = state->visible.x0;
+ drag.bbox.y0 = state->visible.y0;
+ drag.bbox.x1 = state->visible.x1;
+ drag.bbox.y1 = state->visible.y1 - ro_toolbar_height(tv->tb) - 2;
+
+ switch (tv->drag) {
+ case TREE_SELECT_DRAG:
+ drag.type = wimp_DRAG_USER_RUBBER;
+
+ drag.initial.x0 = pointer->pos.x;
+ drag.initial.y0 = pointer->pos.y;
+ drag.initial.x1 = pointer->pos.x;
+ drag.initial.y1 = pointer->pos.y;
+ break;
+
+ case TREE_MOVE_DRAG:
+ drag.type = wimp_DRAG_USER_POINT;
+
+ drag.initial.x0 = pointer->pos.x - 4;
+ drag.initial.y0 = pointer->pos.y - 48;
+ drag.initial.x1 = pointer->pos.x + 48;
+ drag.initial.y1 = pointer->pos.y + 4;
+ break;
+
+ default:
+ /* No other drag types are supported. */
+ break;
+ }
+
+ LOG("Drag start...");
+
+ error = xwimp_drag_box_with_flags(&drag,
+ wimp_DRAG_BOX_KEEP_IN_LINE | wimp_DRAG_BOX_CLIP);
+ if (error) {
+ LOG("xwimp_drag_box: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ } else {
+ auto_scroll.w = tv->w;
+ auto_scroll.pause_zone_sizes.x0 = 80;
+ auto_scroll.pause_zone_sizes.y0 = 80;
+ auto_scroll.pause_zone_sizes.x1 = 80;
+ auto_scroll.pause_zone_sizes.y1 = 80 +
+ ro_toolbar_height(tv->tb);
+ auto_scroll.pause_duration = 0;
+ auto_scroll.state_change = (void *) 1;
+
+ error = xwimp_auto_scroll(wimp_AUTO_SCROLL_ENABLE_VERTICAL,
+ &auto_scroll, NULL);
+ if (error) {
+ LOG("xwimp_auto_scroll: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ }
+
+ ro_mouse_drag_start(ro_treeview_drag_end, ro_treeview_mouse_at,
+ NULL, NULL);
+ }
+}
+
+
/**
* Pass RISC OS Mouse Click events on to the treeview widget.
*
* \param *pointer Pointer to the Mouse Click Event block.
* \return Return true if click handled; else false.
*/
-
static bool ro_treeview_mouse_click(wimp_pointer *pointer)
{
os_error *error;
@@ -886,199 +862,12 @@ static bool ro_treeview_mouse_click(wimp_pointer *pointer)
}
-/**
- * Handle Pointer Entering Window events for treeview windows.
- *
- * \param *entering The Wimp_PointerEnteringWindow block.
- */
-
-void ro_treeview_pointer_entering(wimp_entering *entering)
-{
- ro_treeview *tv;
-
- tv = (ro_treeview *) ro_gui_wimp_event_get_user_data(entering->w);
- if (tv == NULL)
- return;
-
- ro_mouse_track_start(NULL, ro_treeview_mouse_at, NULL);
-}
-
-/**
- * Track the mouse under Null Polls from the wimp, to support dragging.
- *
- * \param *pointer Pointer to a Wimp Pointer block.
- * \param *data NULL to allow use as a ro_mouse callback.
- */
-
-void ro_treeview_mouse_at(wimp_pointer *pointer, void *data)
-{
- os_error *error;
- ro_treeview *tv;
- wimp_window_state state;
- int xpos, ypos;
- browser_mouse_state mouse;
-
- if (pointer->buttons & (wimp_CLICK_MENU))
- return;
-
- tv = (ro_treeview *) ro_gui_wimp_event_get_user_data(pointer->w);
- if (tv == NULL) {
- LOG("NULL treeview block for window: 0x%x", (unsigned int)pointer->w);
- return;
- }
-
- if (tv->drag == TREE_NO_DRAG)
- return;
-
- /* We know now that it's not a Menu click and the treeview thinks
- * that a drag is in progress.
- */
-
- state.w = tv->w;
- error = xwimp_get_window_state(&state);
- if (error) {
- LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- return;
- }
-
- /* Convert the returned mouse coordinates into NetSurf's internal
- * units.
- */
-
- xpos = ((pointer->pos.x - state.visible.x0) +
- state.xscroll - tv->origin.x) / 2;
- ypos = ((state.visible.y1 - pointer->pos.y) -
- state.yscroll + tv->origin.y) / 2;
-
- /* Start to process the mouse click. */
-
- mouse = ro_gui_mouse_drag_state(pointer->buttons,
- wimp_BUTTON_DOUBLE_CLICK_DRAG);
-
- tree_mouse_action(tv->tree, mouse, xpos, ypos);
-
- if (!(mouse & BROWSER_MOUSE_DRAG_ON)) {
- tree_drag_end(tv->tree, mouse, tv->drag_start.x,
- tv->drag_start.y, xpos, ypos);
- tv->drag = TREE_NO_DRAG;
- }
-
- if (tv->callbacks != NULL &&
- tv->callbacks->toolbar_button_update != NULL)
- tv->callbacks->toolbar_button_update();
-}
-
-
-/**
- * Start a RISC OS drag event to reflect on screen what is happening
- * during the core tree drag.
- *
- * \param *tv The RO treeview to which the drag is attached.
- * \param *pointer The RO pointer event data block starting the drag.
- * \param *state The RO window state block for the treeview window.
- */
-
-static void ro_treeview_drag_start(ro_treeview *tv, wimp_pointer *pointer,
- wimp_window_state *state)
-{
- os_error *error;
- wimp_drag drag;
- wimp_auto_scroll_info auto_scroll;
-
- drag.w = tv->w;
- drag.bbox.x0 = state->visible.x0;
- drag.bbox.y0 = state->visible.y0;
- drag.bbox.x1 = state->visible.x1;
- drag.bbox.y1 = state->visible.y1 - ro_toolbar_height(tv->tb) - 2;
-
- switch (tv->drag) {
- case TREE_SELECT_DRAG:
- drag.type = wimp_DRAG_USER_RUBBER;
-
- drag.initial.x0 = pointer->pos.x;
- drag.initial.y0 = pointer->pos.y;
- drag.initial.x1 = pointer->pos.x;
- drag.initial.y1 = pointer->pos.y;
- break;
-
- case TREE_MOVE_DRAG:
- drag.type = wimp_DRAG_USER_POINT;
-
- drag.initial.x0 = pointer->pos.x - 4;
- drag.initial.y0 = pointer->pos.y - 48;
- drag.initial.x1 = pointer->pos.x + 48;
- drag.initial.y1 = pointer->pos.y + 4;
- break;
-
- default:
- /* No other drag types are supported. */
- break;
- }
-
- LOG("Drag start...");
-
- error = xwimp_drag_box_with_flags(&drag,
- wimp_DRAG_BOX_KEEP_IN_LINE | wimp_DRAG_BOX_CLIP);
- if (error) {
- LOG("xwimp_drag_box: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- } else {
- auto_scroll.w = tv->w;
- auto_scroll.pause_zone_sizes.x0 = 80;
- auto_scroll.pause_zone_sizes.y0 = 80;
- auto_scroll.pause_zone_sizes.x1 = 80;
- auto_scroll.pause_zone_sizes.y1 = 80 +
- ro_toolbar_height(tv->tb);
- auto_scroll.pause_duration = 0;
- auto_scroll.state_change = (void *) 1;
-
- error = xwimp_auto_scroll(wimp_AUTO_SCROLL_ENABLE_VERTICAL,
- &auto_scroll, NULL);
- if (error) {
- LOG("xwimp_auto_scroll: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- }
-
- ro_mouse_drag_start(ro_treeview_drag_end, ro_treeview_mouse_at,
- NULL, NULL);
- }
-}
-
-
-/**
- * Process RISC OS User Drag Box events which relate to us: in effect, drags
- * started by ro_treeview_drag_start().
- *
- * \param *drag Pointer to the User Drag Box Event block.
- * \param *data NULL to allow use as a ro_mouse callback.
- */
-
-static void ro_treeview_drag_end(wimp_dragged *drag, void *data)
-{
- os_error *error;
-
- error = xwimp_drag_box((wimp_drag *) -1);
- if (error) {
- LOG("xwimp_drag_box: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- }
-
- error = xwimp_auto_scroll(0, NULL, NULL);
- if (error) {
- LOG("xwimp_auto_scroll: 0x%x: %s", error->errnum, error->errmess);
- ro_warn_user("WimpError", error->errmess);
- }
-}
-
-
/**
* Pass RISC OS keypress events on to the treeview widget.
*
* \param *key Pointer to the Key Pressed Event block.
* \return Return true if keypress handled; else false.
*/
-
static bool ro_treeview_keypress(wimp_key *key)
{
ro_treeview *tv;
@@ -1155,14 +944,203 @@ static bool ro_treeview_keypress(wimp_key *key)
}
+static const struct treeview_table ro_tree_callbacks = {
+ ro_treeview_redraw_request,
+ ro_treeview_resized,
+ ro_treeview_scroll_visible,
+ ro_treeview_get_window_dimensions
+};
+
+
+/**
+ * Create a RISC OS GUI implementation of a treeview tree.
+ *
+ * \param window The window to create the tree in.
+ * \param *toolbar A toolbar to attach to the window.
+ * \param *callbacks Callbacks to service the treeview.
+ * \param flags The treeview flags.
+ *
+ * \return The RISC OS treeview pointer.
+ */
+ro_treeview *ro_treeview_create(wimp_w window, struct toolbar *toolbar,
+ struct ro_treeview_callbacks *callbacks, unsigned int flags)
+{
+ ro_treeview *tv;
+
+ /* Claim memory for the treeview block, and create a tree. */
+
+ tv = malloc(sizeof(ro_treeview));
+ if (tv == NULL)
+ return NULL;
+
+ tv->w = window;
+ tv->tb = toolbar;
+
+ /* Set the tree redraw origin at a default 0,0 RO units. */
+
+ tv->origin.x = 0;
+ tv->origin.y = 0;
+
+ /* Set the tree size as 0,0 to indicate that we don't know. */
+
+ tv->size.x = 0;
+ tv->size.y = 0;
+
+ /* Set the tree window extent to 0,0, to indicate that we
+ * don't know. */
+
+ tv->extent.x = 0;
+ tv->extent.y = 0;
+
+ /* Set that there is no drag opperation at the moment */
+
+ tv->drag = TREE_NO_DRAG;
+
+ tv->tree = tree_create(flags, &ro_tree_callbacks, tv);
+ if (tv->tree == NULL) {
+ free(tv);
+ return NULL;
+ }
+
+ /* Record the callback info. */
+
+ tv->callbacks = callbacks;
+
+ /* Register wimp events to handle the supplied window. */
+
+ ro_gui_wimp_event_register_redraw_window(tv->w, ro_treeview_redraw);
+ ro_gui_wimp_event_register_scroll_window(tv->w, ro_treeview_scroll);
+ ro_gui_wimp_event_register_pointer_entering_window(tv->w,
+ ro_treeview_pointer_entering);
+ ro_gui_wimp_event_register_open_window(tv->w, ro_treeview_open);
+ ro_gui_wimp_event_register_mouse_click(tv->w, ro_treeview_mouse_click);
+ ro_gui_wimp_event_register_keypress(tv->w, ro_treeview_keypress);
+ ro_gui_wimp_event_set_user_data(tv->w, tv);
+
+ return tv;
+}
+
+
+/**
+ * Change the redraw origin of a treeview tree in RISC OS graphics units.
+ *
+ * \param *tv The ro_treeview object to update.
+ * \param x The X position, in terms of the RO window work area.
+ * \param y The Y position, in terms of the RO window work area.
+ *
+ * \todo -- this probably needs a rework.
+ */
+void ro_treeview_set_origin(ro_treeview *tv, int x, int y)
+{
+ if (tv != NULL) {
+ tv->origin.x = x;
+ tv->origin.y = y;
+
+ /* Assuming that we know how big the tree currently is, then
+ * adjust the window work area extent to match. If we don't,
+ * then presumably the tree isn't in an open window yet and
+ * a subsequent Open Window Event should pick it up.
+ */
+
+ if (tv->size.x != 0 && tv->size.y != 0)
+ ro_treeview_set_window_extent(tv,
+ tv->origin.x + tv->size.x,
+ tv->origin.y + tv->size.y);
+ }
+}
+
+
+/**
+ * Track the mouse under Null Polls from the wimp, to support dragging.
+ *
+ * \param *pointer Pointer to a Wimp Pointer block.
+ * \param *data NULL to allow use as a ro_mouse callback.
+ */
+void ro_treeview_mouse_at(wimp_pointer *pointer, void *data)
+{
+ os_error *error;
+ ro_treeview *tv;
+ wimp_window_state state;
+ int xpos, ypos;
+ browser_mouse_state mouse;
+
+ if (pointer->buttons & (wimp_CLICK_MENU))
+ return;
+
+ tv = (ro_treeview *) ro_gui_wimp_event_get_user_data(pointer->w);
+ if (tv == NULL) {
+ LOG("NULL treeview block for window: 0x%x", (unsigned int)pointer->w);
+ return;
+ }
+
+ if (tv->drag == TREE_NO_DRAG)
+ return;
+
+ /* We know now that it's not a Menu click and the treeview thinks
+ * that a drag is in progress.
+ */
+
+ state.w = tv->w;
+ error = xwimp_get_window_state(&state);
+ if (error) {
+ LOG("xwimp_get_window_state: 0x%x: %s", error->errnum, error->errmess);
+ ro_warn_user("WimpError", error->errmess);
+ return;
+ }
+
+ /* Convert the returned mouse coordinates into NetSurf's internal
+ * units.
+ */
+
+ xpos = ((pointer->pos.x - state.visible.x0) +
+ state.xscroll - tv->origin.x) / 2;
+ ypos = ((state.visible.y1 - pointer->pos.y) -
+ state.yscroll + tv->origin.y) / 2;
+
+ /* Start to process the mouse click. */
+
+ mouse = ro_gui_mouse_drag_state(pointer->buttons,
+ wimp_BUTTON_DOUBLE_CLICK_DRAG);
+
+ tree_mouse_action(tv->tree, mouse, xpos, ypos);
+
+ if (!(mouse & BROWSER_MOUSE_DRAG_ON)) {
+ tree_drag_end(tv->tree, mouse, tv->drag_start.x,
+ tv->drag_start.y, xpos, ypos);
+ tv->drag = TREE_NO_DRAG;
+ }
+
+ if (tv->callbacks != NULL &&
+ tv->callbacks->toolbar_button_update != NULL)
+ tv->callbacks->toolbar_button_update();
+}
+
+
+/**
+ * Change the size of a treeview's toolbar and redraw the window.
+ *
+ * \param *data The treeview to update.
+ */
+static void ro_treeview_update_toolbar(void *data)
+{
+ ro_treeview *tv = (ro_treeview *) data;
+
+ if (tv != NULL && tv->tb != NULL) {
+ ro_treeview_set_origin(tv, 0,
+ -(ro_toolbar_height(tv->tb)));
+
+ xwimp_force_redraw(tv->w, 0, tv->extent.y, tv->extent.x, 0);
+ }
+}
+
+
/**
* Update a treeview to use a new theme.
*
* \param *data Pointer to the treeview to update.
* \param ok true if the bar still exists; else false.
*/
-
-void ro_treeview_update_theme(void *data, bool ok)
+static void ro_treeview_update_theme(void *data, bool ok)
{
ro_treeview *tv = (ro_treeview *) data;
@@ -1176,33 +1154,13 @@ void ro_treeview_update_theme(void *data, bool ok)
}
-/**
- * Change the size of a treeview's toolbar and redraw the window.
- *
- * \param *data The treeview to update.
- */
-
-void ro_treeview_update_toolbar(void *data)
-{
- ro_treeview *tv = (ro_treeview *) data;
-
- if (tv != NULL && tv->tb != NULL) {
- ro_treeview_set_origin(tv, 0,
- -(ro_toolbar_height(tv->tb)));
-
- xwimp_force_redraw(tv->w, 0, tv->extent.y, tv->extent.x, 0);
- }
-}
-
-
/**
* Update the toolbar icons in a treeview window's toolbar. As we're just
* an intermediate widget, we pass the details on down the chain.
*
* \param *data The treeview owning the toolbar.
*/
-
-void ro_treeview_button_update(void *data)
+static void ro_treeview_button_update(void *data)
{
ro_treeview *tv = (ro_treeview *) data;
@@ -1221,8 +1179,7 @@ void ro_treeview_button_update(void *data)
* \param *data The treeview owning the toolbar.
* \param *config The new button config string.
*/
-
-void ro_treeview_save_toolbar_buttons(void *data, char *config)
+static void ro_treeview_save_toolbar_buttons(void *data, char *config)
{
ro_treeview *tv = (ro_treeview *) data;
@@ -1242,8 +1199,7 @@ void ro_treeview_save_toolbar_buttons(void *data, char *config)
* \param action_type The action type to be handled.
* \param action The action to handle.
*/
-
-void ro_treeview_button_click(void *data,
+static void ro_treeview_button_click(void *data,
toolbar_action_type action_type, union toolbar_action action)
{
ro_treeview *tv = (ro_treeview *) data;
@@ -1260,6 +1216,28 @@ void ro_treeview_button_click(void *data,
}
+static const struct toolbar_callbacks ro_treeview_toolbar_callbacks = {
+ ro_treeview_update_theme,
+ ro_treeview_update_toolbar,
+ ro_treeview_button_update,
+ ro_treeview_button_click,
+ NULL, /* No toolbar keypress handler */
+ ro_treeview_save_toolbar_buttons
+};
+
+
+/**
+ * Return a pointer to a toolbar callbacks structure with the handlers to be
+ * used by any treeview window toolbars.
+ *
+ * \return A pointer to the callback structure.
+ */
+const struct toolbar_callbacks *ro_treeview_get_toolbar_callbacks(void)
+{
+ return &ro_treeview_toolbar_callbacks;
+}
+
+
/**
* Return a token identifying the interactive help message for a given cursor
* position.
@@ -1269,9 +1247,7 @@ void ro_treeview_button_click(void *data,
* \param *message_data Pointer to the Wimp's help message block.
* \return Token value (-1 indicates no help available).
*/
-
int ro_treeview_get_help(help_full_message_request *message_data)
{
return -1;
}
-