I\'m failing at translating the following PHP/ImageMagick code into Ruby RMagick (to make it more manageable for future users and to understand what it\'s really doing):
Posting this for anyone who is still looking for a solution to this, there is a way of doing this in ruby using the rmagick
gem described here https://stackoverflow.com/a/41282162/1975112
I've never coded a single line of Ruby, but here is a partial remake of the PHP script:
require 'find'
pngFiles = Dir.glob("*.png")
Find.find('./') do |f|
if system 'convert ' + f + ' -resize 1x1 -alpha on -channel o -format "%[fx:u.a]" info' do
print "PNG file contains some alpha transparency and will not be modified"
I think that the only thing you need to check for is just an alpha channel, not all of that complicated stuff. Look at this page for more info: http://www.imagemagick.org/script/escape.php. It has a %A
operator, and that outputs something about the alpha channel.
The code checks to see if a particular image contains transparency.
-format '%[fx:u.a]' info:
This instructs image magick to check the first image u
, the alpha channel of that a
and output info on it, it will return 0 if the top-left pixel is transparent and non-zero if not I think. That is why the image is being resized to 1x1, so that only a single pixel needs to be consulted. The -channel o
is the opacity channel.
So the code in English would read, cycle through all PNG files, look at the alpha channel (opacity) only, resize to a single pixel and see if it's transparent. Hence the echo message.
Unfortunately I do not know Ruby, or RMagick, but a quick look at the API seems to suggest using image.channel(AlphaChannel)
to get the alpha channel (AlphaChannel
is a ChannelType
value, not sure if you have to specify ChannelType.AlphaChannel
), then follow with .resize(1,1)
to get the size down, and finish with either .pixel_color(0,0)
or .get_pixels(0,0,1,1)
to get the Pixel
object back (get_pixels()
returns an array), which I believe has an opacity
attribute. However, the channel()
command changes the RGB values to the value of the channel selected, and I'm not sure it preserves the opacity channel so you might just need to look at red forinstance, or omit the .channel()
call entirely - though I do not know if that would disrupt the result offhand.
Perhaps if Ruby supports decent functional programming approaches.
image.channel(AlphaChannel).resize(1,1).pixel_color(0,0).red
or this if pixel_color()
does not return the opacity for some reason
image.channel(AlphaChannel).resize(1,1).get_pixels(0,0,1,1)[0].red
Without the channel()
calls it would be:
image.resize(1,1).pixel_color(0,0).opacity
or
image.resize(1,1).get_pixels(0,0,1,1)[0].opacity
Again, my Ruby is non-existent, so you may have to rearrange those a lot, but the primitives are there.
References
The php code you show (which really just sends the work to convert in a shell) does not check to see if the images have alpha channels, it just takes whatever file is given and turns it on. If it already had one there would be no file change, but convert is not being asked to make any decision based on the status, just go ahead and add the channel.
why did you not copy just the command and call it with system ?
no gems required ... no questions, code should look almost the same.
EDIT: RMagic only wraps imagemagic so why to bother with rmagic if you already have imagmagic command string.