私は第4版に取り組んでいます。Hillegass/Preble Cocoa の本を読んでいて、Core Animation で理解できない問題にぶつかりました。
具体的には、おそらく一度に 1 つずつ、レイヤー ホスティング ビューに画像のリストを表示する次のループがあります。期待される結果は、呼び出されたときに各画像が一度に 1 つずつ飛び出すことpresentImage
です。実際の結果は、ループが完了するまでビューが空のままであり、その後、画像が一度に飛び出すことです。遅延中に、 への複数の呼び出しを示す複数のログ メッセージが表示されpresentImage
ます。これらのログ メッセージが停止し、ループが完了したことを示すと、すべての画像が一度に飛び出します。
// loop to present each image in list of URLs
// called at end of applicationDidFinishLaunching
NSTimeInterval t0 = [NSDate timeIntervalSinceReferenceDate];
for(id url in urls) {
NSImage *img = [[NSImage alloc] initWithContentsOfURL:url];
if(!img)
continue;
NSImage *thumbImg = [self thumbImageFromImage:img];
[self presentImage:thumbImg];
NSString *dt = [NSString stringWithFormat:@"%0.1fs",
[NSDate timeIntervalSinceReferenceDate]-t0];
NSLog(@"Presented %@ at %@ sec",url,dt);
}
このthumbImageFromImage
メソッドは、あなたが思っていることを実行します (より小さな画像を作成します)。のコードpresentImage
は次のとおりです。
- (void)presentImage:(NSImage *)image
{
CGRect superlayerBounds = view.layer.bounds;
NSPoint center = NSMakePoint(CGRectGetMidX(superlayerBounds), CGRectGetMidY(superlayerBounds));
NSRect imageBounds = NSMakeRect(0, 0, image.size.width, image.size.height);
CGPoint randomPoint = CGPointMake(CGRectGetMaxX(superlayerBounds)*(double)random()/(double)RAND_MAX, CGRectGetMaxY(superlayerBounds)*(double)random()/(double)RAND_MAX);
CAMediaTimingFunction *tf = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseInEaseOut];
CABasicAnimation *posAnim = [CABasicAnimation animation];
posAnim.fromValue = [NSValue valueWithPoint:center];
posAnim.duration = 1.5;
posAnim.timingFunction = tf;
CABasicAnimation *bdsAnim = [CABasicAnimation animation];
bdsAnim.fromValue = [NSValue valueWithRect:NSZeroRect];
bdsAnim.duration = 1.5;
bdsAnim.timingFunction = tf;
CALayer *layer = [CALayer layer];
layer.contents = image;
layer.actions = [NSDictionary dictionaryWithObjectsAndKeys:posAnim,@"position", bdsAnim, @"bounds", nil];
[CATransaction begin];
[view.layer addSublayer:layer];
layer.position = randomPoint;
layer.bounds = NSRectToCGRect(imageBounds);
[CATransaction commit];
}
観測された結果は、insideの呼び出しではなく、 [CATransaction begin]
andの[CATransaction commit]
呼び出しがループを囲んでいるかのようです。実際、内側からandを削除し、代わりにループを括弧で囲むと、同じ動作が一度にすべて観察されます。実際、 and の呼び出しを完全に削除すると、同じ動作が一度にすべて観察されます。for
addSublayer
presentImage
[CATransaction begin]
[CATransaction commit]
presentImage
for
[CATransaction begin]
[CATransaction commit]