我想填充基于Paint的应用程序的白色区域的颜色 所以请给我建议如何做这项工作..
答案 0 :(得分:19)
我找到了具有Flood填充算法的解决方案
private void FloodFill(Bitmap bmp, Point pt, int targetColor, int replacementColor){
Queue<Point> q = new LinkedList<Point>();
q.add(pt);
while (q.size() > 0) {
Point n = q.poll();
if (bmp.getPixel(n.x, n.y) != targetColor)
continue;
Point w = n, e = new Point(n.x + 1, n.y);
while ((w.x > 0) && (bmp.getPixel(w.x, w.y) == targetColor)) {
bmp.setPixel(w.x, w.y, replacementColor);
if ((w.y > 0) && (bmp.getPixel(w.x, w.y - 1) == targetColor))
q.add(new Point(w.x, w.y - 1));
if ((w.y < bmp.getHeight() - 1)
&& (bmp.getPixel(w.x, w.y + 1) == targetColor))
q.add(new Point(w.x, w.y + 1));
w.x--;
}
while ((e.x < bmp.getWidth() - 1)
&& (bmp.getPixel(e.x, e.y) == targetColor)) {
bmp.setPixel(e.x, e.y, replacementColor);
if ((e.y > 0) && (bmp.getPixel(e.x, e.y - 1) == targetColor))
q.add(new Point(e.x, e.y - 1));
if ((e.y < bmp.getHeight() - 1)
&& (bmp.getPixel(e.x, e.y + 1) == targetColor))
q.add(new Point(e.x, e.y + 1));
e.x++;
}
}}
洪水填写android:
请参阅此FloodFill
答案 1 :(得分:4)
这是一个使用Python和OpenCV的快速应用程序(如果你足够努力,应该可以在Android上使用):
"""Flood fills with random color on click. Press `q' to exit."""
import cv
import sys
import random
TOL = 10
TOL_BGR = (TOL, TOL, TOL, 0)
def click(event,x,y,flags,im):
if event == cv.CV_EVENT_LBUTTONDOWN:
b,g,r = [ random.random() * 255 for i in range(3) ]
cv.FloodFill(im, (x,y), (b,g,r,0), TOL_BGR, TOL_BGR)
im = cv.LoadImage(sys.argv[1], cv.CV_LOAD_IMAGE_COLOR)
cv.NamedWindow(__file__, 1)
cv.SetMouseCallback(__file__, click, im)
while True:
cv.ShowImage(__file__, im)
key = cv.WaitKey(33)
if chr(key & 0xff) == 'q':
break
cv.SaveImage('floodfill.png', im)
每次用户点击图片时,应用程序都会使用点击位置作为种子进行填充。颜色随机挑选。您可以通过修改TOL(或TOL_BGR)的值来更改公差。这是几次点击后的结果:
无论您使用何种技术,一般算法都是相同的。
答案 2 :(得分:0)
1)将每个颜色部分的分割图像分别与实际图像相同,其他部分是透明的。 2)在可绘制的文件夹中,每个部分都有不同颜色的完整图像 - 这只是参考图像。
3)在帧布局中添加所有分割图像,并将所有分割初始设置为不可见,并将其设置为仅对实际图像可见
4)热门代码为每个从参考图像(步骤2)拆分的颜色为Eg handSplitImageColor =绿色;
4)设置框架布局的监听器找出x和y位置,并将x和y位置传递给你的参考图像(步骤2),找出该特定位置的颜色,并将颜色与步骤4和填充该图像中的特定内容,并为该分割图像设置可见性true。因此,只有那部分会被颜色填充,因为其他部分是透明的。
这些是我用于同类问题的步骤。
但我不认为这是更好的解决方案,但对我来说效果很好。
答案 3 :(得分:0)
经过大量搜索,我想到了以下内容。 快速高效
private void myFloodFillMethod(Point node,int targetColor, int fillColor) {
if (targetColor != fillColor) {
Queue<Point> queue = new LinkedList<>();
do {
int x = node.x;
int y = node.y;
while (x > 0 && getPixelColorFromArr(x - 1, y) == targetColor) {
x--; }
boolean spanUp = false;
boolean spanDown = false;
while (x < width && checkEquality(getPixelColorFromArr(x, y),targetColor)) {
setPixelColorFromArr(x, y, fillColor);
if (!spanUp && y > 0
&& checkEquality(getPixelColorFromArr(x, y - 1),targetColor)) {
queue.add(new Point(x, y - 1));
spanUp = true;
} else if (spanUp && y > 0
&& !checkEquality(getPixelColorFromArr(x, y - 1),targetColor)) {
spanUp = false;
}
if (!spanDown && y < height - 1
&& checkEquality(getPixelColorFromArr(x, y + 1), targetColor)) {
queue.add(new Point(x, y + 1));
spanDown = true;
} else if (spanDown && y < height - 1
&& !checkEquality(getPixelColorFromArr(x, y + 1),targetColor)) {
spanDown = false;
}
x++;
}
} while ((node = queue.poll()) != null);
}
}
private boolean checkEquality(int pixelColorFromArr, int targetColor) {
return pixelColorFromArr == targetColor;
}
void fillColorInImage(Bitmap image, Point node, int targetColor, int replacementColor){
if(targetColor == replacementColor) return;
width = image.getWidth();
height = image.getHeight();
pixelArr = new int[width*height];
image.getPixels(pixelArr,0,width,0,0,width,height);
myFloodFillMethod(node,targetColor,replacementColor);
image.setPixels(pixelArr,0,width,0,0,width,height);
}
呼叫方式:
fillColorInImage(bmp, point, targetColor, replacementColor);
其中
bmp
是图像的位图(从图像视图中提取)
point
是用户点击的Point(x,y):从event.getX()
和event.getY()
targetColor
是点的颜色:从bitmap.getPixel()
replacementColor
是您要替换为的颜色