2012-01-02 5 views
4

目標:イメージピクセルデータがイメージピクセルを「スキャン」する方法は?

のみ黒と透明ピクセルを含む画像の左側の最初の黒画素を見つけます。

私が持っているもの

私は(ここではそれを見つけた:https://stackoverflow.com/a/1262893/358480を):ピクセルデータを取得し、黒と透明ピクセルの配列を持ってする方法を知っている

+ (NSArray*)getRGBAsFromImage:(UIImage*)image atX:(int)xx andY:(int)yy count:(int)count 
{ 
NSMutableArray *result = [NSMutableArray arrayWithCapacity:count]; 

// First get the image into your data buffer 
CGImageRef imageRef = [image CGImage]; 
NSUInteger width = CGImageGetWidth(imageRef); 
NSUInteger height = CGImageGetHeight(imageRef); 
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB(); 
unsigned char *rawData = malloc(height * width * 4); 
NSUInteger bytesPerPixel = 4; 
NSUInteger bytesPerRow = bytesPerPixel * width; 
NSUInteger bitsPerComponent = 8; 
CGContextRef context = CGBitmapContextCreate(rawData, width, height, 
              bitsPerComponent, bytesPerRow, colorSpace, 
              kCGImageAlphaPremultipliedLast | kCGBitmapByteOrder32Big); 
CGColorSpaceRelease(colorSpace); 

CGContextDrawImage(context, CGRectMake(0, 0, width, height), imageRef); 
CGContextRelease(context); 

// Now your rawData contains the image data in the RGBA8888 pixel format. 
int byteIndex = (bytesPerRow * yy) + xx * bytesPerPixel; 
for (int ii = 0 ; ii < count ; ++ii) 
{ 
    NSUInteger alpha = (rawData[byteIndex + 3] * 1.0)/255.0; 
    byteIndex += 4; 
    [result addObject:[NSNumber numberWithInt:alpha]]; 
} 

free(rawData); 

return result; 
} 

は何ですか問題?

機能が画像を「スキャン」する順序を理解できません。

私が望むのは、画像の列だけを取得し、リスト1の非トランスペアレントピクセルを持つ最初の列を特定することです。このようにして、画像の左半透明面を切り取る方法を知りますか?

どのようにピクセルを列で取得できますか?

おかげ

シャニー

答えて

4

バイトは左から右、上から下に並べられます。だから、あなたがやりたい、私はあなたがこのようなrawDataをループにしたいと思う:

int x = 0; 
int y = 0; 
BOOL found = NO; 
for (x = 0; x < width; x++) { 
    for (y = 0; y < height; y++) { 
     unsigned char alphaByte = rawData[(y*bytesPerRow)+(x*bytesPerPixel)+3]; 
     if (alphaByte > 0) { 
      found = YES; 
      break; 
     } 
    } 
    if (found) break; 
} 

NSLog(@"First non-transparent pixel at %i, %i", x, y); 

そして、非透明画素が含まれているあなたの最初の列が列xになります。

0

通常、行の上から下へ、左上から列の上へ、各行内でイメージ配列を繰り返し処理します。この場合は、逆が必要です。左から始めて、各列を繰り返し、列内ですべての行を調べ、黒のピクセルが存在するかどうかを確認します。

これはあなたの一番左の黒画素与える:これは滑らかな印象あまりにもわずかに最適化され、mattjgallowayに等しく、そして、

size_t maxIndex = height * bytesPerRow; 
for (size_t x = 0; x < bytesPerRow; x += bytesPerPixel) 
{   
    for (size_t index = x; index < maxIndex; index += bytesPerRow) 
    { 
     if (rawData[index + 3] > 0) 
     { 
      goto exitLoop; 
     } 
    } 
} 
exitLoop: 

if (x < bytesPerRow) 
{ 
    x /= bytesPerPixel; 
    // left most column is `x` 
} 

まあの:

O gotoは通常に許可されているが、内側のループから2つのループを放棄すると、それはまだ醜いです。私が実際にそれらの気の利いたフロー制御ステートメントDがあるのを見逃してしまいます...

あなたがサンプルコードで提供した機能は何か異なっています。画像の特定の位置(xxyyで定義されます)から開始し、開始位置から右に移動して次の行に続くcountピクセルを超えます。それは私が疑ういくつかの配列にそれらのアルファ値を追加します。

xx = yy = 0が渡されると、一番左のピクセルではなく、一定の条件で一番上のピクセルが検出されます。この変換は上記のコードで与えられます。 2D画像はメモリ上の1次元配列で、左上から右に向かって次の行に進むことを思い出してください。簡単な数学演算を行うと、行または列を繰り返し処理できます。

関連する問題