好的,所以我有一个可以产生可接受结果的工作解决方案 - 最终文件比我原来的手动过程略小,但在视觉上无法区分。
该解决方案不像我想要的那样跨平台(需要为 OptiPNG 查找/构建 Linux 二进制文件),但它仍然是一个足够好的解决方案。
正如 Leigh 在问题评论中所建议的那样,我使用的是量化器来减少颜色,然后使用 MapColorsFilter 来修复量化器破坏透明度的事实,最后使用 OptiPNG 将生成的文件压缩到合理的大小。
以下是相关代码:
<cfscript>
var Filename = './filename.png'
var MyImage = NewImage(Filename)
ImageFilter(MyImage,'quantize',{numColors:256,dither:false})
// ImageFilter(MyImage,'MapColors',{oldColor:'white',newColor:'ffffff00'})
var TransImage = ImageMapColors(MyImage,'white','ffffff00')
ImageWrite( TransImage , Filename )
</cfscript>
<cfexecute
name = "#Variables.OptiPngExecutable#"
arguments = "-o9 #Filename#"
timeout = 30
/>
Railo 的 MapColors 过滤器的 ImageFilter 目前存在一个错误,因此我必须直接访问过滤器,这是我用来解决此问题的代码:
<cffunction name="ImageMapColors" output=false >
<cfargument name="Image" rtype="Image" required />
<cfargument name="Old" type="String" required />
<cfargument name="New" type="String" required />
<cfset var ObjKey = 'ColorReplacer_#Arguments.Old#_#Arguments.New#' />
<cfif NOT StructKeyExists(Variables,ObjKey)>
<cfset var Old = createObject('java','railo.commons.color.ColorCaster').toColor(Arguments.Old) />
<cfset var New = createObject('java','railo.commons.color.ColorCaster').toColor(Arguments.New) />
<cfset Variables[ObjKey] = createObject("java","railo.runtime.img.filter.MapColorsFilter")
.init(Old.getRGB(),New.getRGB()) />
</cfif>
<cfreturn ImageNew( Variables[ObjKey].filter(ImageGetBufferedImage(Arguments.Image),{}) ) />
</cffunction>