/* GIMP - The GNU Image Manipulation Program * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * gimpviewrendererbrush.c * Copyright (C) 2003 Michael Natterer * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "config.h" #include #include "widgets-types.h" #include "base/temp-buf.h" #include "core/gimpbrushpipe.h" #include "core/gimpbrushgenerated.h" #include "gimpviewrendererbrush.h" static void gimp_view_renderer_brush_finalize (GObject *object); static void gimp_view_renderer_brush_render (GimpViewRenderer *renderer, GtkWidget *widget); static void gimp_view_renderer_brush_draw (GimpViewRenderer *renderer, GtkWidget *widget, cairo_t *cr, const GdkRectangle *area); static gboolean gimp_view_renderer_brush_render_timeout (gpointer data); G_DEFINE_TYPE (GimpViewRendererBrush, gimp_view_renderer_brush, GIMP_TYPE_VIEW_RENDERER) #define parent_class gimp_view_renderer_brush_parent_class static void gimp_view_renderer_brush_class_init (GimpViewRendererBrushClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GimpViewRendererClass *renderer_class = GIMP_VIEW_RENDERER_CLASS (klass); object_class->finalize = gimp_view_renderer_brush_finalize; renderer_class->render = gimp_view_renderer_brush_render; renderer_class->draw = gimp_view_renderer_brush_draw; } static void gimp_view_renderer_brush_init (GimpViewRendererBrush *renderer) { renderer->pipe_timeout_id = 0; renderer->pipe_animation_index = 0; } static void gimp_view_renderer_brush_finalize (GObject *object) { GimpViewRendererBrush *renderer = GIMP_VIEW_RENDERER_BRUSH (object); if (renderer->pipe_timeout_id) { g_source_remove (renderer->pipe_timeout_id); renderer->pipe_timeout_id = 0; } G_OBJECT_CLASS (parent_class)->finalize (object); } static void gimp_view_renderer_brush_render (GimpViewRenderer *renderer, GtkWidget *widget) { GimpViewRendererBrush *renderbrush = GIMP_VIEW_RENDERER_BRUSH (renderer); TempBuf *temp_buf; if (renderbrush->pipe_timeout_id) { g_source_remove (renderbrush->pipe_timeout_id); renderbrush->pipe_timeout_id = 0; } temp_buf = gimp_viewable_get_new_preview (renderer->viewable, renderer->context, renderer->width, renderer->height); if (temp_buf->width < renderer->width) temp_buf->x = (renderer->width - temp_buf->width) / 2; if (temp_buf->height < renderer->height) temp_buf->y = (renderer->height - temp_buf->height) / 2; if (renderer->is_popup) { gimp_view_renderer_render_surface (renderer, temp_buf, -1, GIMP_VIEW_BG_WHITE, GIMP_VIEW_BG_WHITE); temp_buf_free (temp_buf); if (GIMP_IS_BRUSH_PIPE (renderer->viewable)) { renderbrush->pipe_animation_index = 0; renderbrush->pipe_timeout_id = g_timeout_add (300, gimp_view_renderer_brush_render_timeout, renderbrush); } return; } gimp_view_renderer_render_surface (renderer, temp_buf, -1, GIMP_VIEW_BG_WHITE, GIMP_VIEW_BG_WHITE); temp_buf_free (temp_buf); } static gboolean gimp_view_renderer_brush_render_timeout (gpointer data) { GimpViewRendererBrush *renderbrush = GIMP_VIEW_RENDERER_BRUSH (data); GimpViewRenderer *renderer = GIMP_VIEW_RENDERER (data); GimpBrushPipe *brush_pipe; GimpBrush *brush; TempBuf *temp_buf; if (! renderer->viewable) { renderbrush->pipe_timeout_id = 0; renderbrush->pipe_animation_index = 0; return FALSE; } brush_pipe = GIMP_BRUSH_PIPE (renderer->viewable); renderbrush->pipe_animation_index++; if (renderbrush->pipe_animation_index >= brush_pipe->nbrushes) renderbrush->pipe_animation_index = 0; brush = GIMP_BRUSH (brush_pipe->brushes[renderbrush->pipe_animation_index]); temp_buf = gimp_viewable_get_new_preview (GIMP_VIEWABLE (brush), renderer->context, renderer->width, renderer->height); if (temp_buf->width < renderer->width) temp_buf->x = (renderer->width - temp_buf->width) / 2; if (temp_buf->height < renderer->height) temp_buf->y = (renderer->height - temp_buf->height) / 2; gimp_view_renderer_render_surface (renderer, temp_buf, -1, GIMP_VIEW_BG_WHITE, GIMP_VIEW_BG_WHITE); temp_buf_free (temp_buf); gimp_view_renderer_update (renderer); return TRUE; } static void gimp_view_renderer_brush_draw (GimpViewRenderer *renderer, GtkWidget *widget, cairo_t *cr, const GdkRectangle *area) { GIMP_VIEW_RENDERER_CLASS (parent_class)->draw (renderer, widget, cr, area); #define INDICATOR_WIDTH 7 #define INDICATOR_HEIGHT 7 if (renderer->width > 2 * INDICATOR_WIDTH && renderer->height > 2 * INDICATOR_HEIGHT) { gboolean pipe = GIMP_IS_BRUSH_PIPE (renderer->viewable); gboolean generated = GIMP_IS_BRUSH_GENERATED (renderer->viewable); gint brush_width; gint brush_height; if (generated || pipe) { cairo_move_to (cr, area->x + area->width, area->y + area->height); cairo_rel_line_to (cr, - INDICATOR_WIDTH, 0); cairo_rel_line_to (cr, INDICATOR_WIDTH, - INDICATOR_HEIGHT); cairo_rel_line_to (cr, 0, INDICATOR_HEIGHT); if (pipe) cairo_set_source_rgb (cr, 1.0, 0.5, 0.5); else cairo_set_source_rgb (cr, 0.5, 0.6, 1.0); cairo_fill (cr); } gimp_viewable_get_size (renderer->viewable, &brush_width, &brush_height); if (renderer->width < brush_width || renderer->height < brush_height) { cairo_move_to (cr, area->x + area->width - INDICATOR_WIDTH + 1, area->y + area->height - INDICATOR_HEIGHT / 2.0); cairo_rel_line_to (cr, INDICATOR_WIDTH - 2, 0); cairo_move_to (cr, area->x + area->width - INDICATOR_WIDTH / 2.0, area->y + area->height - INDICATOR_HEIGHT + 1); cairo_rel_line_to (cr, 0, INDICATOR_WIDTH - 2); cairo_set_source_rgb (cr, 0.0, 0.0, 0.0); cairo_set_line_width (cr, 1); cairo_stroke (cr); } } #undef INDICATOR_WIDTH #undef INDICATOR_HEIGHT }