我想在我的SpriteKit精灵中将一个指定的颜色更改为alpha,让所有其他颜色不受影响。我不认为有一个‘容易’的方式来做到这一点,并可能需要一个定制的着色器。
有人知道如何在Swift或Objective中这样做吗?
这是图像-我设计它的黑色背景,因为它更容易看到,但在我的程序,我希望背景是透明的。
如果我想编辑背景已经设置为alpha的图像,那么我使用的包(Pixaki)将显示为白色和灰色的检查,所以看起来如下:
发布于 2016-04-21 13:58:45
下面是示例代码,它可以实现您想要的结果。
它是基于这个answer(只是基本转换为Swift,经过一些修改)。学分就在那里。
// color - source color, which is must be replaced
// withColor - target color
// tolerance - value in range from 0 to 1
func replaceColor(color:SKColor, withColor:SKColor, image:UIImage, tolerance:CGFloat) -> UIImage{
// This function expects to get source color(color which is supposed to be replaced)
// and target color in RGBA color space, hence we expect to get 4 color components: r, g, b, a
assert(CGColorGetNumberOfComponents(color.CGColor) == 4 && CGColorGetNumberOfComponents(withColor.CGColor) == 4,
"Must be RGBA colorspace")
// Allocate bitmap in memory with the same width and size as source image
let imageRef = image.CGImage
let width = CGImageGetWidth(imageRef)
let height = CGImageGetHeight(imageRef)
let colorSpace = CGColorSpaceCreateWithName(kCGColorSpaceGenericRGB)!
let bytesPerPixel = 4
let bytesPerRow = bytesPerPixel * width;
let bitsPerComponent = 8
let bitmapByteCount = bytesPerRow * height
let rawData = UnsafeMutablePointer<UInt8>.alloc(bitmapByteCount)
let context = CGBitmapContextCreate(rawData, width, height, bitsPerComponent, bytesPerRow, colorSpace,
CGImageAlphaInfo.PremultipliedLast.rawValue | CGBitmapInfo.ByteOrder32Big.rawValue)
let rc = CGRect(x: 0, y: 0, width: width, height: height)
// Draw source image on created context
CGContextDrawImage(context, rc, imageRef)
// Get color components from replacement color
let withColorComponents = CGColorGetComponents(withColor.CGColor)
let r2 = UInt8(withColorComponents[0] * 255)
let g2 = UInt8(withColorComponents[1] * 255)
let b2 = UInt8(withColorComponents[2] * 255)
let a2 = UInt8(withColorComponents[3] * 255)
// Prepare to iterate over image pixels
var byteIndex = 0
while byteIndex < bitmapByteCount {
// Get color of current pixel
let red:CGFloat = CGFloat(rawData[byteIndex + 0])/255
let green:CGFloat = CGFloat(rawData[byteIndex + 1])/255
let blue:CGFloat = CGFloat(rawData[byteIndex + 2])/255
let alpha:CGFloat = CGFloat(rawData[byteIndex + 3])/255
let currentColor = SKColor(red: red, green: green, blue: blue, alpha: alpha);
// Compare two colors using given tolerance value
if compareColor(color, withColor: currentColor , withTolerance: tolerance) {
// If the're 'similar', then replace pixel color with given target color
rawData[byteIndex + 0] = r2
rawData[byteIndex + 1] = g2
rawData[byteIndex + 2] = b2
rawData[byteIndex + 3] = a2
}
byteIndex = byteIndex + 4;
}
// Retrieve image from memory context
let imgref = CGBitmapContextCreateImage(context)
let result = UIImage(CGImage: imgref!)
// Clean up a bit
rawData.destroy()
return result
}
func compareColor(color:SKColor, withColor:SKColor, withTolerance:CGFloat) -> Bool {
var r1: CGFloat = 0.0, g1: CGFloat = 0.0, b1: CGFloat = 0.0, a1: CGFloat = 0.0;
var r2: CGFloat = 0.0, g2: CGFloat = 0.0, b2: CGFloat = 0.0, a2: CGFloat = 0.0;
color.getRed(&r1, green: &g1, blue: &b1, alpha: &a1);
withColor.getRed(&r2, green: &g2, blue: &b2, alpha: &a2);
return fabs(r1 - r2) <= withTolerance &&
fabs(g1 - g2) <= withTolerance &&
fabs(b1 - b2) <= withTolerance &&
fabs(a1 - a2) <= withTolerance;
}
您可以调整它并创建命令行工具,或者在代码中直接使用它。
下面是示例项目:https://github.com/andrey-str/soa-replace-a-color-colour-in-a-skspritenode
免责声明:代码可能泄漏,我是Swift的新手。
90% -是一个公差值,结果与上面的代码相同。
祝好运!
发布于 2022-05-21 07:45:39
老问题,斯威夫特5回答:遮住UIImage
的颜色。
(由于这在UIImage.cgImage
上有效,您可能认为它也适用于SKTexture.cgImage
,但遗憾的是没有,因此我们需要首先转换为UIImage。
首先,将sprite纹理转换为UIImage,然后使用:
extension UIImage {
func remove(color: UIColor, tolerance: CGFloat = 4) -> UIImage {
let ciColor = CIColor(color: color)
let maskComponents: [CGFloat] = [ciColor.red, ciColor.green, ciColor.blue].flatMap { value in
[(value * 255) - tolerance, (value * 255) + tolerance]
}
guard let masked = cgImage?.copy(maskingColorComponents: maskComponents) else { return self }
return UIImage(cgImage: masked)
}
}
在macOS上,我们需要一个NSImage
来完成这个任务:
extension NSImage {
func remove(color: NSColor, tolerance: CGFloat = 4) -> NSImage {
if let ciColor = CIColor(color: color) {
let maskComponents: [CGFloat] = [ciColor.red, ciColor.green, ciColor.blue].flatMap { value in
[(value * 255) - tolerance, (value * 255) + tolerance]
}
guard let masked = cgImage(forProposedRect: nil, context: nil, hints: nil)?.copy(maskingColorComponents: maskComponents) else { return self }
return NSImage(cgImage: masked, size: size)
}
return self
}
}
https://stackoverflow.com/questions/36770611
复制相似问题