May I know is what are the solution to use in order to make the following code working in order.
- (CGFloat)getRowImageHeight
{
CGFloat defaultHeight = 300.f;
__block CGFloat height = defaultHeight;
[self configureImageTVCell:self.itemImageTVCell
block:^(UIImage *image, BOOL succeeded) {
if( succeeded )
height = image.size.height;
}];
return height;
}
I would expect the BLOCK to complete first before the return height, it is because I wanted to get the latest height from the completion of BLOCK.
I've tried a few solution by using dispatch_semaphore_t, but it causes deadlock where the simulator actually hang.
dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
[self configureImageTVCell:self.itemImageTVCell
block:^(UIImage *image, BOOL succeeded) {
if( succeeded )
height = image.size.height;
dispatch_semaphore_signal(semaphore);
}];
dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
Running out of idea for this already, need help indeed.
Things that I am trying to do is at view load, I preload the image before it appears on the view. Then get image block to return the image height and I wanted to return back the height to
But it seems that answers above told me that the approach is impossible. So I tried to use delegate and return the height to main view and update the cell height by using the code below after the image has been successfully fetch at cell level.
In viewDidLoad, I've set self.hasUpdatedImageCellHeight to NO and self.itemImageCellHeight to 300.f. self.hasUpdatedImageCellHeight is to control whether the delegate has already return me any value, if YES then I will update it to YES in order to prevent the delegate codes keep calling again & again.
This is my approach for it, but still it is best if there is an approach that can know the image height natively before the table view being completely loaded.
Basically, what you are trying to do is impossible. On the one hand, you have a method from which you need to return a value immediately:
On the other hand, in the middle of that code ("do some stuff"), you are performing an asynchronous operation:
That means, by definition, that the code in the block will run at some future, unknown time. Meanwhile, your outer method
getRowImageHeight
has finished and returned its value long ago.That is the nature of asynchronous code execution.
You need to rearchitect your entire approach so that it works in conjunction with your asynchronous code.
Of course, you have not revealed what you are really trying to do, so I can't tell you how to rearchitect it. But let us pretend for a moment that you are trying to populate a table view's cells with images to be downloaded from the Internet. Well, that is a well-established problem with answers all over the place, including many good explanations on Stack Overflow. So you would read those answers.
the good approach would be something like this: