/* * rofi * * MIT/X11 License * Copyright © 2013-2017 Qball Cow * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * */ #include #include #include #include "widgets/scrollbar.h" #include "x11-helper.h" #include "theme.h" #define DEFAULT_SCROLLBAR_WIDTH 8 static void scrollbar_draw ( widget *, cairo_t * ); static void scrollbar_free ( widget * ); static gboolean scrollbar_motion_notify ( widget *wid, xcb_motion_notify_event_t *xme ); static int scrollbar_get_desired_height ( widget *wid ) { // Want height we are. return wid->h; } scrollbar *scrollbar_create ( const char *name ) { scrollbar *sb = g_malloc0 ( sizeof ( scrollbar ) ); widget_init ( WIDGET ( sb ), name ); sb->widget.x = 0; sb->widget.y = 0; sb->width = rofi_theme_get_distance ( WIDGET ( sb ), "handle-width", DEFAULT_SCROLLBAR_WIDTH ); int width = distance_get_pixel ( sb->width, ORIENTATION_HORIZONTAL ); sb->widget.w = widget_padding_get_padding_width ( WIDGET ( sb ) ) + width; sb->widget.h = widget_padding_get_padding_height ( WIDGET ( sb ) ); sb->widget.draw = scrollbar_draw; sb->widget.free = scrollbar_free; sb->widget.motion_notify = scrollbar_motion_notify; sb->widget.get_desired_height = scrollbar_get_desired_height; sb->length = 10; sb->pos = 0; sb->pos_length = 4; // Enabled by default sb->widget.enabled = rofi_theme_get_boolean ( WIDGET ( sb ), "enabled", TRUE ); return sb; } static void scrollbar_free ( widget *wid ) { scrollbar *sb = (scrollbar *) wid; g_free ( sb ); } void scrollbar_set_max_value ( scrollbar *sb, unsigned int max ) { if ( sb != NULL ) { sb->length = MAX ( 1u, max ); } } void scrollbar_set_handle ( scrollbar *sb, unsigned int pos ) { if ( sb != NULL ) { sb->pos = MIN ( sb->length, pos ); } } void scrollbar_set_handle_length ( scrollbar *sb, unsigned int pos_length ) { if ( sb != NULL ) { sb->pos_length = MIN ( sb->length, MAX ( 1u, pos_length ) ); } } /** * The range is the height - handle length. * r = h - handle; * handle is the element length of the handle* length of one element. * handle = r / ( num ) * hl * * r = h - r / ( num) *hl * r*num = num*h - r*hl * r*num+r*hl = num*h; * r ( num+hl ) = num*h * r = (num*h)/(num+hl) */ static void scrollbar_draw ( widget *wid, cairo_t *draw ) { scrollbar *sb = (scrollbar *) wid; unsigned int wh = widget_padding_get_remaining_height ( wid ); // Calculate position and size. unsigned int r = ( sb->length * wh ) / ( (double) ( sb->length + sb->pos_length ) ); unsigned int handle = wid->h - r; double sec = ( ( r ) / (double) ( sb->length - 1 ) ); unsigned int height = handle; unsigned int y = sb->pos * sec; // Set max pos. y = MIN ( y, wh - handle ); // Never go out of bar. height = MAX ( 2, height ); // Cap length; rofi_theme_get_color ( WIDGET ( sb ), "foreground", draw ); rofi_theme_get_color ( WIDGET ( sb ), "handle-color", draw ); cairo_rectangle ( draw, widget_padding_get_left ( wid ), widget_padding_get_top ( wid ) + y, widget_padding_get_remaining_width ( wid ), height ); cairo_fill ( draw ); } static gboolean scrollbar_motion_notify ( widget *wid, xcb_motion_notify_event_t *xme ) { xcb_button_press_event_t xle; xle.event_x = xme->event_x; xle.event_y = xme->event_y; return widget_clicked ( WIDGET ( wid ), &xle ); } // TODO // This should behave more like a real scrollbar. unsigned int scrollbar_clicked ( const scrollbar *sb, int y ) { if ( sb != NULL ) { if ( y >= sb->widget.y && y <= ( sb->widget.y + sb->widget.h ) ) { short r = ( sb->length * sb->widget.h ) / ( (double) ( sb->length + sb->pos_length ) ); short handle = sb->widget.h - r; double sec = ( ( r ) / (double) ( sb->length - 1 ) ); short half_handle = handle / 2; y -= sb->widget.y + half_handle; y = MIN ( MAX ( 0, y ), sb->widget.h - 2 * half_handle ); unsigned int sel = ( ( y ) / sec ); return MIN ( sel, sb->length - 1 ); } } return 0; }