Revert "ScriptFu: scripts: remove obsolete script unsharp-mask"

This reverts commit fbee943213.

We must not remove any PDB function because we promise API stability
(not only in libgimp, but also in the PDB). Even if this was not used
anywhere in our own scripts/plug-ins, it may be used by third-party
scripts. At best, we can deprecate them so that people are properly warn
that we plan on removing a function. Then we'll remove it for GIMP 4.

Also I believe that various people were not perfectly happy with GEGL's
unsharp-mask operation, compared to the old one. And that was a reason
for keeping the legacy Script-fu version around. See gegl#122 or other
reports which mention such concerns.
At the very least, this should be discussed first with more specialized
developers to determine whether we consider GEGL's replacement operation
to be adequately replacing the historical script by now, so that we can
plan the removal for GIMP 4.
This commit is contained in:
Jehan 2025-06-11 02:13:23 +02:00
parent 933a12335c
commit fca4fd1bcc
4 changed files with 129 additions and 22 deletions

View file

@ -41,6 +41,7 @@ scripts = [
'slide.scm', 'slide.scm',
'spinning-globe.scm', 'spinning-globe.scm',
'tileblur.scm', 'tileblur.scm',
'unsharp-mask.scm',
'waves-anim.scm', 'waves-anim.scm',
'weave.scm', 'weave.scm',
'xach-effect.scm', 'xach-effect.scm',

View file

@ -0,0 +1,105 @@
;;; unsharp-mask.scm
;;; Time-stamp: <1998/11/17 13:18:39 narazaki@gimp.org>
;;; Author: Narazaki Shuji <narazaki@gimp.org>
;;; Version 0.8
; This script-fu-unsharp-mask is not in the menus.
; There is an equivalent GEGL filter at Filters>Enhance>Sharpen (Unsharp).
; This might be kept for compatibility and used by third party scripts.
; Seems not used by any script in the repo.
; FUTURE move to gimp-data-extras or to scripts/test
; and maintain it with low priority.
; unsharp-mask is a filter AND renderer, creating a new, visible, dirty image
; from the given image.
(define (script-fu-unsharp-mask img drws mask-size mask-opacity)
(let* (
(drw (vector-ref drws 0))
(drawable-width (car (gimp-drawable-get-width drw)))
(drawable-height (car (gimp-drawable-get-height drw)))
(new-image (car (gimp-image-new drawable-width drawable-height RGB)))
(original-layer (car (gimp-layer-new new-image "Original"
drawable-width drawable-height
RGB-IMAGE
100 LAYER-MODE-NORMAL)))
(original-layer-for-darker 0)
(original-layer-for-lighter 0)
(blurred-layer-for-darker 0)
(blurred-layer-for-lighter 0)
(darker-layer 0)
(lighter-layer 0)
)
(gimp-selection-all img)
(gimp-edit-copy (vector drw))
(gimp-image-undo-disable new-image)
(gimp-image-insert-layer new-image original-layer 0 0)
(let* (
(pasted (car (gimp-edit-paste original-layer FALSE)))
(num-pasted (vector-length pasted))
(floating-sel (vector-ref pasted (- num-pasted 1)))
)
(gimp-floating-sel-anchor floating-sel)
)
(set! original-layer-for-darker (car (gimp-layer-copy original-layer)))
(gimp-layer-add-alpha original-layer-for-darker)
(set! original-layer-for-lighter (car (gimp-layer-copy original-layer)))
(gimp-layer-add-alpha original-layer-for-lighter)
(set! blurred-layer-for-darker (car (gimp-layer-copy original-layer)))
(gimp-layer-add-alpha blurred-layer-for-darker)
(gimp-item-set-visible original-layer FALSE)
(gimp-display-new new-image)
;; make darker mask
(gimp-image-insert-layer new-image blurred-layer-for-darker 0 -1)
(gimp-drawable-merge-new-filter blurred-layer-for-darker "gegl:gaussian-blur" 0 LAYER-MODE-REPLACE 1.0 "std-dev-x" (* 0.32 mask-size) "std-dev-y" (* 0.32 mask-size) "filter" "auto")
(set! blurred-layer-for-lighter
(car (gimp-layer-copy blurred-layer-for-darker)))
(gimp-layer-add-alpha blurred-layer-for-lighter)
(gimp-image-insert-layer new-image original-layer-for-darker 0 -1)
(gimp-layer-set-mode original-layer-for-darker LAYER-MODE-SUBTRACT)
(set! darker-layer
(car (gimp-image-merge-visible-layers new-image CLIP-TO-IMAGE)))
(gimp-item-set-name darker-layer "darker mask")
(gimp-item-set-visible darker-layer FALSE)
;; make lighter mask
(gimp-image-insert-layer new-image original-layer-for-lighter 0 -1)
(gimp-image-insert-layer new-image blurred-layer-for-lighter 0 -1)
(gimp-layer-set-mode blurred-layer-for-lighter LAYER-MODE-SUBTRACT)
(set! lighter-layer
(car (gimp-image-merge-visible-layers new-image CLIP-TO-IMAGE)))
(gimp-item-set-name lighter-layer "lighter mask")
;; combine them
(gimp-item-set-visible original-layer TRUE)
(gimp-layer-set-mode darker-layer LAYER-MODE-SUBTRACT)
(gimp-layer-set-opacity darker-layer mask-opacity)
(gimp-item-set-visible darker-layer TRUE)
(gimp-layer-set-mode lighter-layer LAYER-MODE-ADDITION)
(gimp-layer-set-opacity lighter-layer mask-opacity)
(gimp-item-set-visible lighter-layer TRUE)
(gimp-image-undo-enable new-image)
(gimp-displays-flush)
)
)
(script-fu-register-filter "script-fu-unsharp-mask"
"Unsharp Mask..."
"Make a new image from the current layer by applying the unsharp mask method"
"Shuji Narazaki <narazaki@gimp.org>"
"Shuji Narazaki"
"1997,1998"
"*"
SF-ONE-OR-MORE-DRAWABLE
SF-ADJUSTMENT _"Mask size" '(5 1 100 1 1 0 1)
SF-ADJUSTMENT _"Mask opacity" '(50 0 100 1 1 0 1)
)

View file

@ -1,7 +1,8 @@
; Test script calling plugin scripts non-interactive ; Test script calling plugin scripts non-interactive
; Some scripts must be non-interactive (no menu item.) ; Some are scripts that must be non-interactive (no menu item.)
; Such scripts can only be tested by another plugin or script. ; Most scripts can be tested using app menus.
; These scripts can only be tested by another plugin or script.
; This also tests and illustrates aspects ; This also tests and illustrates aspects
; of calling plugin scripts from other scripts: ; of calling plugin scripts from other scripts:
@ -13,10 +14,6 @@
; While calling a dialect v2 script interprocess ; While calling a dialect v2 script interprocess
; can be done from v3 binding state. ; can be done from v3 binding state.
; This also illustrates that some scripts are not well-behaved
; re non-interactive calls:
; they open a display on the new image.
; FUTURE: call all plugin scripts in repo ; FUTURE: call all plugin scripts in repo
@ -29,10 +26,10 @@
(define testImage (testing:load-test-image-basic-v3)) (define testImage (testing:load-test-image-basic-v3))
; get-layers returns just a vector #(layers) ; get-layers returns (length #(layers))
; Calling another script requires a vector of layers (ever since multi-layer feature.) ; Calling another script requires a vector of layers (ever since multi-layer feature.)
; cdr is always a list i.e. (#(layers)), so we need cadr here.
(define testDrawables (gimp-image-get-layers testImage)) (define testDrawables (cadr (gimp-image-get-layers testImage)))
(define testDrawable (vector-ref testDrawables 0)) (define testDrawable (vector-ref testDrawables 0))
;(newline) ;(newline)
@ -41,14 +38,15 @@
(test! "Calling a Scheme plugin script non-interactively") (test! "script-fu-unsharp-mask")
; The called plugin is a Scheme script that is already loaded in the interpreter. ; script-fu-unsharp-mask has been replaced by a GEGL filter.
; If it is removed from the repo, please keep a test of some script here,
; if only as an example of how the testing framework can test plugin scripts.
; The test is only: ensure it doesn't error or crash, ; The test is only: ensure it doesn't error or crash.
; and that it does not open a display on the new image
; The called plugin uses v2 binding ; unsharp-mask uses v2 binding
; If called without this, usually error is "must be pair" ; If called without this, usually error is "must be pair"
(script-fu-use-v2) (script-fu-use-v2)
@ -58,16 +56,17 @@
; only a call to a Scheme function, the run-func of the script. ; only a call to a Scheme function, the run-func of the script.
; Use the Scheme signature of the run-func: ; Use the Scheme signature of the run-func:
; no run_mode argument. ; no run_mode argument.
; ; no "num-drawables" argument of the C signature.
; The script is not a filter, but a renderer, (script-fu-unsharp-mask
; and does not use passed image and drawables.
(script-fu-gradient-example
; !!! Not pass run_mode ; !!! Not pass run_mode
40 40 ; dimensions of new image testImage
1 ; NO reverse gradient. Since v2 binding, this is a number, not #f ; !!! Not pass num_drawables, just a Scheme vector of int ID of drawables
testDrawables
128 ; mask-size, radius in pixels
50 ; opacity percent
) )
; Expect creates new image of current gradient in context ; Expect an image with extra layers: "lighter mask" and "darker mask"
; back to v3 binding so we don't need to unwrap calls to gimp-foo in this scope. ; back to v3 binding so we don't need to unwrap calls to gimp-foo in this scope.
(script-fu-use-v3) (script-fu-use-v3)
@ -120,6 +119,7 @@
(script-fu-test-sphere-v3 (script-fu-test-sphere-v3
RUN-NONINTERACTIVE ; run_mode see above interprocess RUN-NONINTERACTIVE ; run_mode see above interprocess
testImage ; unused Image testImage ; unused Image
1 ; num-drawables int size of GimpObjectArray
testDrawables ; unused GimpObjectArray, Scheme vector of int ID testDrawables ; unused GimpObjectArray, Scheme vector of int ID
; radius is actually dimension of the square rendered image ; radius is actually dimension of the square rendered image
300 ; radius int 300 ; radius int

View file

@ -47,6 +47,7 @@ plug-ins/script-fu/scripts/slide.scm
plug-ins/script-fu/scripts/spinning-globe.scm plug-ins/script-fu/scripts/spinning-globe.scm
plug-ins/script-fu/scripts/test-sphere-v3.scm plug-ins/script-fu/scripts/test-sphere-v3.scm
plug-ins/script-fu/scripts/tileblur.scm plug-ins/script-fu/scripts/tileblur.scm
plug-ins/script-fu/scripts/unsharp-mask.scm
plug-ins/script-fu/scripts/waves-anim.scm plug-ins/script-fu/scripts/waves-anim.scm
plug-ins/script-fu/scripts/weave.scm plug-ins/script-fu/scripts/weave.scm
plug-ins/script-fu/scripts/xach-effect.scm plug-ins/script-fu/scripts/xach-effect.scm