GIMP Script-fu

Artifact [204b7dfe57]
Login

Artifact 204b7dfe57f8a2fb74947da44010f049c1192924:


; 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 2 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.

(define (script-fu-sg-mystic-rose image drawable number-of-points thickness random? color-mode palette gradient create-path?)
  (let* ((create-path? (not (zero? create-path?)))
         (color-mode (+ (* 2 color-mode) random?))
         (width (car (gimp-drawable-width drawable)))
         (height (car (gimp-drawable-height drawable)))
         (rad-x (floor (/ (- width thickness) 2)))
         (rad-y (floor (/ (- height thickness) 2)))
         (center-x (floor (/ width 2)))
         (center-y (floor (/ height 2)))
         (points-list 
           (let loop ((point number-of-points)
                      (points-list '()) )
             (if (zero? point)
               points-list
               (loop (- point 1)
                     (cons (cons (+ center-x (* rad-x (cos (/ (* 2 *pi* point) number-of-points))))
                                 (+ center-y (* rad-y (sin (/ (* 2 *pi* point) number-of-points)))) )
                           points-list )))))
         (path 0)
         (stroke 0)
         (brush "")
         (prog-count 0) 
         (prog-end (* 2 number-of-points (- number-of-points 1)))
         )
    (gimp-image-undo-group-start image)
    (gimp-context-push)
    (gimp-context-set-paint-method "gimp-paintbrush")
    (gimp-context-set-paint-mode NORMAL-MODE)
    (set! brush (car (gimp-brush-new "temporary")))
    (gimp-brush-set-shape brush BRUSH-GENERATED-CIRCLE)
    (gimp-brush-set-radius brush (/ thickness 2))
    (gimp-context-set-brush brush)
    (set! path (car (gimp-vectors-new image (string-append (number->string number-of-points)
                                                                           "pt mystic rose" ))))
    (gimp-image-add-vectors image path 0)
    (let next-start-point ((start points-list))
      (unless (null? start)
        (let next-end-point ((end (cdr points-list)))
          (unless (null? end)
            (set! stroke (car (gimp-vectors-bezier-stroke-new-moveto path (caar start) (cdar start))))
            (gimp-vectors-bezier-stroke-lineto path stroke (caar end) (cdar end))
            (gimp-progress-update (/ prog-count prog-end))
            (set! prog-count (+ prog-count 1))
            (next-end-point (cdr end)) ))
        (next-start-point (cdr start)) ))
    (let ((num-strokes (car (gimp-vectors-get-strokes path))))
      (let next-stroke ((strokes (vector->list (cadr (gimp-vectors-get-strokes path))))
                        (color-index 0) )
        (unless (null? strokes)
          (gimp-progress-update (/ prog-count prog-end))
          (set! prog-count (+ prog-count 1))
          (let ((points (caddr (gimp-vectors-stroke-get-points path (car strokes))))
                (num-palette-entries (car (gimp-palette-get-colors palette))) )
            (cond 
              ((= color-mode 1) ; random color 
                (gimp-context-set-foreground (list (random 256) (random 256) (random 256))) )
              ((= color-mode 2) ; palette - color-index is offset into palette
                (gimp-context-set-foreground 
                    (car (gimp-palette-entry-get-color palette (modulo color-index num-palette-entries))) ))
              ((= color-mode 3) ; palette random - choose random color from specified palette
                (gimp-context-set-foreground 
                    (car (gimp-palette-entry-get-color palette (random num-palette-entries))) ))
              ((= color-mode 4) ; gradient - color-index is offset into specified gradient
                (let* ((color-vector (cadr (gimp-gradient-get-custom-samples gradient 
                                                                             1 
                                                                             (vector (/ color-index num-strokes))
                                                                             FALSE )))
                       (color (map (lambda (x) (* x 255)) (butlast (vector->list color-vector))))
                       (opacity (* 100 (vector-ref color-vector 3)))
                       )
                   (gimp-context-set-foreground color)
                   (gimp-context-set-opacity opacity) ))
              ((= color-mode 5) ; gradient random - random color from specified gradient
                (let* ((color-vector (cadr (gimp-gradient-get-custom-samples gradient 
                                                                             1 
                                                                             (vector (/ (random num-strokes) num-strokes))
                                                                             FALSE )))
                       (color (map (lambda (x) (* x 255)) (butlast (vector->list color-vector))))
                       (opacity (* 100 (vector-ref color-vector 3)))
                       )
                   (gimp-context-set-foreground color)
                   (gimp-context-set-opacity opacity) )))
            (gimp-paintbrush drawable 
                             FALSE 
                             4 
                             (vector (vector-ref points 2)
                                     (vector-ref points 3)
                                     (vector-ref points 8)
                                     (vector-ref points 9) )
                             PAINT-CONSTANT
                             0 )
            (next-stroke (cdr strokes) (+ color-index 1)) ))))
    (gimp-brush-delete brush)            
    (unless create-path?
      (gimp-image-remove-vectors image path) )
    (gimp-context-pop)
    (gimp-image-undo-group-end image)
    (gimp-displays-flush)
    )
  )
             
(script-fu-register "script-fu-sg-mystic-rose"
  "Mystic rose..."
  "Create a mystic rose within the current layer"
  "Saul Goode"
  "Saul Goode"
  "July 2011"
  "RGB*"
  SF-IMAGE    "Image"    0
  SF-DRAWABLE "Layer" 0
  SF-ADJUSTMENT "Number of points" '(12 3 179 1 10 0 1)
  SF-ADJUSTMENT "Line thickness" '(1 1 20 1 3 0 1)
  SF-TOGGLE "Random?" FALSE
  SF-OPTION "Color method" '("Foreground" "Palette" "Gradient")
  SF-PALETTE "Palette" "Default"
  SF-GRADIENT "Gradient" "Full saturation spectrum CW"
  SF-TOGGLE "Create path?" FALSE
  )

(script-fu-menu-register "script-fu-sg-mystic-rose"
 "<Image>/Filters/Render/Pattern"
 )