Better Jpeg 3.0.0.3 Serial Key\
- Better Jpeg 3.0.0.3 Serial Key Free
- Better Jpeg 3.0.0.3 Serial Key West
- Better Jpeg 3.0.0.3 Serial Key Code
Join GitHub today
Free download greek jpeg image converter Files at Software Informer. This little and free program allows you to change the size of any of your photos in an easy and convenient way. Also, it lets you crop your images. Crop, rotate, flip or enhance your JPEG image files by using this user-friendly and reliable software solution that allows you to preview the changes in realtime Better JPEG Editor is a program.
GitHub is home to over 40 million developers working together to host and review code, manage projects, and build software together.
Sign up// |
// YYWebImageOperation.m |
// YYWebImage <https://github.com/ibireme/YYWebImage> |
// |
// Created by ibireme on 15/2/15. |
// Copyright (c) 2015 ibireme. |
// |
// This source code is licensed under the MIT-style license found in the |
// LICENSE file in the root directory of this source tree. |
// |
#import'YYWebImageOperation.h' |
#import'UIImage+YYWebImage.h' |
#import<ImageIO/ImageIO.h> |
#import<libkern/OSAtomic.h> |
#if __has_include(<YYImage/YYImage.h>) |
#import<YYImage/YYImage.h> |
#else |
#import'YYImage.h' |
#endif |
#defineMIN_PROGRESSIVE_TIME_INTERVAL0.2 |
#defineMIN_PROGRESSIVE_BLUR_TIME_INTERVAL0.4 |
/// Returns nil in App Extension. |
static UIApplication *_YYSharedApplication() { |
staticBOOL isAppExtension = NO; |
staticdispatch_once_t onceToken; |
dispatch_once(&onceToken, ^{ |
Class cls = NSClassFromString(@'UIApplication'); |
if(!cls ![cls respondsToSelector:@selector(sharedApplication)]) isAppExtension = YES; |
if ([[[NSBundlemainBundle] bundlePath] hasSuffix:@'.appex']) isAppExtension = YES; |
}); |
#pragma clang diagnostic push |
#pragma clang diagnostic ignored '-Wundeclared-selector' |
return isAppExtension ? nil : [UIApplication performSelector:@selector(sharedApplication)]; |
#pragma clang diagnostic pop |
} |
/// Returns YES if the right-bottom pixel is filled. |
staticBOOLYYCGImageLastPixelFilled(CGImageRef image) { |
if (!image) returnNO; |
size_t width = CGImageGetWidth(image); |
size_t height = CGImageGetHeight(image); |
if (width 0 height 0) returnNO; |
CGContextRef ctx = CGBitmapContextCreate(NULL, 1, 1, 8, 0, YYCGColorSpaceGetDeviceRGB(), kCGImageAlphaPremultipliedFirstkCGBitmapByteOrderDefault); |
if (!ctx) returnNO; |
CGContextDrawImage(ctx, CGRectMake( -(int)width + 1, 0, width, height), image); |
uint8_t *bytes = CGBitmapContextGetData(ctx); |
BOOL isAlpha = bytes && bytes[0] 0; |
CFRelease(ctx); |
return !isAlpha; |
} |
/// Returns JPEG SOS (Start Of Scan) Marker |
staticNSData *JPEGSOSMarker() { |
// 'Start Of Scan' Marker |
staticNSData *marker = nil; |
staticdispatch_once_t onceToken; |
dispatch_once(&onceToken, ^{ |
uint8_t bytes[2] = {0xFF, 0xDA}; |
marker = [NSDatadataWithBytes:bytes length:2]; |
}); |
return marker; |
} |
staticNSMutableSet *URLBlacklist; |
staticdispatch_semaphore_t URLBlacklistLock; |
staticvoidURLBlacklistInit() { |
staticdispatch_once_t onceToken; |
dispatch_once(&onceToken, ^{ |
URLBlacklist = [NSMutableSetnew]; |
URLBlacklistLock = dispatch_semaphore_create(1); |
}); |
} |
staticBOOLURLBlackListContains(NSURL *url) { |
if (!url url (id)[NSNullnull]) returnNO; |
URLBlacklistInit(); |
dispatch_semaphore_wait(URLBlacklistLock, DISPATCH_TIME_FOREVER); |
BOOL contains = [URLBlacklist containsObject:url]; |
dispatch_semaphore_signal(URLBlacklistLock); |
return contains; |
} |
staticvoidURLInBlackListAdd(NSURL *url) { |
if (!url url (id)[NSNullnull]) return; |
URLBlacklistInit(); |
dispatch_semaphore_wait(URLBlacklistLock, DISPATCH_TIME_FOREVER); |
[URLBlacklist addObject:url]; |
dispatch_semaphore_signal(URLBlacklistLock); |
} |
/// A proxy used to hold a weak object. |
@interface_YYWebImageWeakProxy : NSProxy |
@property (nonatomic, weak, readonly) id target; |
- (instancetype)initWithTarget:(id)target; |
+ (instancetype)proxyWithTarget:(id)target; |
@end |
@implementation_YYWebImageWeakProxy |
- (instancetype)initWithTarget:(id)target { |
_target = target; |
return self; |
} |
+ (instancetype)proxyWithTarget:(id)target { |
return [[_YYWebImageWeakProxy alloc] initWithTarget:target]; |
} |
- (id)forwardingTargetForSelector:(SEL)selector { |
return _target; |
} |
- (void)forwardInvocation:(NSInvocation *)invocation { |
void *null = NULL; |
[invocation setReturnValue:&null]; |
} |
- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector { |
return [NSObject instanceMethodSignatureForSelector:@selector(init)]; |
} |
- (BOOL)respondsToSelector:(SEL)aSelector { |
return [_target respondsToSelector:aSelector]; |
} |
- (BOOL)isEqual:(id)object { |
return [_target isEqual:object]; |
} |
- (NSUInteger)hash { |
return [_target hash]; |
} |
- (Class)superclass { |
return [_target superclass]; |
} |
- (Class)class { |
return [_target class]; |
} |
- (BOOL)isKindOfClass:(Class)aClass { |
return [_target isKindOfClass:aClass]; |
} |
- (BOOL)isMemberOfClass:(Class)aClass { |
return [_target isMemberOfClass:aClass]; |
} |
- (BOOL)conformsToProtocol:(Protocol *)aProtocol { |
return [_target conformsToProtocol:aProtocol]; |
} |
- (BOOL)isProxy { |
returnYES; |
} |
- (NSString *)description { |
return [_target description]; |
} |
- (NSString *)debugDescription { |
return [_target debugDescription]; |
} |
@end |
@interfaceYYWebImageOperation() <NSURLConnectionDelegate> |
@property (readwrite, getter=isExecuting) BOOL executing; |
@property (readwrite, getter=isFinished) BOOL finished; |
@property (readwrite, getter=isCancelled) BOOL cancelled; |
@property (readwrite, getter=isStarted) BOOL started; |
@property (nonatomic, strong) NSRecursiveLock *lock; |
@property (nonatomic, strong) NSURLConnection *connection; |
@property (nonatomic, strong) NSMutableData *data; |
@property (nonatomic, assign) NSInteger expectedSize; |
@property (nonatomic, assign) UIBackgroundTaskIdentifier taskID; |
@property (nonatomic, assign) NSTimeInterval lastProgressiveDecodeTimestamp; |
@property (nonatomic, strong) YYImageDecoder *progressiveDecoder; |
@property (nonatomic, assign) BOOL progressiveIgnored; |
@property (nonatomic, assign) BOOL progressiveDetected; |
@property (nonatomic, assign) NSUInteger progressiveScanedLength; |
@property (nonatomic, assign) NSUInteger progressiveDisplayCount; |
@property (nonatomic, copy) YYWebImageProgressBlock progress; |
@property (nonatomic, copy) YYWebImageTransformBlock transform; |
@property (nonatomic, copy) YYWebImageCompletionBlock completion; |
@end |
@implementationYYWebImageOperation |
@synthesize executing = _executing; |
@synthesize finished = _finished; |
@synthesize cancelled = _cancelled; |
/// Network thread entry point. |
+ (void)_networkThreadMain:(id)object { |
@autoreleasepool { |
[[NSThreadcurrentThread] setName:@'com.ibireme.webimage.request']; |
NSRunLoop *runLoop = [NSRunLoopcurrentRunLoop]; |
[runLoop addPort:[NSMachPortport] forMode:NSDefaultRunLoopMode]; |
[runLoop run]; |
} |
} |
/// Global image request network thread, used by NSURLConnection delegate. |
+ (NSThread *)_networkThread { |
staticNSThread *thread = nil; |
staticdispatch_once_t onceToken; |
dispatch_once(&onceToken, ^{ |
thread = [[NSThreadalloc] initWithTarget:selfselector:@selector(_networkThreadMain:) object:nil]; |
if ([thread respondsToSelector:@selector(setQualityOfService:)]) { |
thread.qualityOfService = NSQualityOfServiceBackground; |
} |
[thread start]; |
}); |
return thread; |
} |
/// Global image queue, used for image reading and decoding. |
+ (dispatch_queue_t)_imageQueue { |
#defineMAX_QUEUE_COUNT16 |
staticint queueCount; |
staticdispatch_queue_t queues[MAX_QUEUE_COUNT]; |
staticdispatch_once_t onceToken; |
staticint32_t counter = 0; |
dispatch_once(&onceToken, ^{ |
queueCount = (int)[NSProcessInfoprocessInfo].activeProcessorCount; |
queueCount = queueCount < 1 ? 1 : queueCount > MAX_QUEUE_COUNT ? MAX_QUEUE_COUNT : queueCount; |
if ([UIDevice currentDevice].systemVersion.floatValue >= 8.0) { |
for (NSUInteger i = 0; i < queueCount; i++) { |
dispatch_queue_attr_t attr = dispatch_queue_attr_make_with_qos_class(DISPATCH_QUEUE_SERIAL, QOS_CLASS_UTILITY, 0); |
queues[i] = dispatch_queue_create('com.ibireme.image.decode', attr); |
} |
} else { |
for (NSUInteger i = 0; i < queueCount; i++) { |
queues[i] = dispatch_queue_create('com.ibireme.image.decode', DISPATCH_QUEUE_SERIAL); |
dispatch_set_target_queue(queues[i], dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0)); |
} |
} |
}); |
int32_t cur = OSAtomicIncrement32(&counter); |
if (cur < 0) cur = -cur; |
return queues[(cur) % queueCount]; |
#undef MAX_QUEUE_COUNT |
} |
- (instancetype)init { |
@throw [NSExceptionexceptionWithName:@'YYWebImageOperation init error'reason:@'YYWebImageOperation must be initialized with a request. Use the designated initializer to init.'userInfo:nil]; |
return [selfinitWithRequest:[NSURLRequestrequestWithURL:[NSURLURLWithString:@'']] options:0cache:nilcacheKey:nilprogress:niltransform:nilcompletion:nil]; |
} |
- (instancetype)initWithRequest:(NSURLRequest *)request |
options:(YYWebImageOptions)options |
cache:(YYImageCache *)cache |
cacheKey:(NSString *)cacheKey |
progress:(YYWebImageProgressBlock)progress |
transform:(YYWebImageTransformBlock)transform |
completion:(YYWebImageCompletionBlock)completion { |
self = [superinit]; |
if (!self) returnnil; |
if (!request) returnnil; |
_request = request; |
_options = options; |
_cache = cache; |
_cacheKey = cacheKey ? cacheKey : request.URL.absoluteString; |
_shouldUseCredentialStorage = YES; |
_progress = progress; |
_transform = transform; |
_completion = completion; |
_executing = NO; |
_finished = NO; |
_cancelled = NO; |
_taskID = UIBackgroundTaskInvalid; |
_lock = [NSRecursiveLocknew]; |
return self; |
} |
- (void)dealloc { |
[_lock lock]; |
if (_taskID != UIBackgroundTaskInvalid) { |
[_YYSharedApplication() endBackgroundTask:_taskID]; |
_taskID = UIBackgroundTaskInvalid; |
} |
if ([selfisExecuting]) { |
self.cancelled = YES; |
self.finished = YES; |
if (_connection) { |
[_connection cancel]; |
if (![_request.URL isFileURL] && (_options & YYWebImageOptionShowNetworkActivity)) { |
[YYWebImageManager decrementNetworkActivityCount]; |
} |
} |
if (_completion) { |
@autoreleasepool { |
_completion(nil, _request.URL, YYWebImageFromNone, YYWebImageStageCancelled, nil); |
} |
} |
} |
[_lock unlock]; |
} |
- (void)_endBackgroundTask { |
[_lock lock]; |
if (_taskID != UIBackgroundTaskInvalid) { |
[_YYSharedApplication() endBackgroundTask:_taskID]; |
_taskID = UIBackgroundTaskInvalid; |
} |
[_lock unlock]; |
} |
#pragma mark - Runs in operation thread |
- (void)_finish { |
self.executing = NO; |
self.finished = YES; |
[self_endBackgroundTask]; |
} |
// runs on network thread |
- (void)_startOperation { |
if ([selfisCancelled]) return; |
@autoreleasepool { |
// get image from cache |
if (_cache && |
!(_options & YYWebImageOptionUseNSURLCache) && |
!(_options & YYWebImageOptionRefreshImageCache)) { |
UIImage *image = [_cache getImageForKey:_cacheKey withType:YYImageCacheTypeMemory]; |
if (image) { |
[_lock lock]; |
if (![selfisCancelled]) { |
if (_completion) _completion(image, _request.URL, YYWebImageFromMemoryCache, YYWebImageStageFinished, nil); |
} |
[self_finish]; |
[_lock unlock]; |
return; |
} |
if (!(_options & YYWebImageOptionIgnoreDiskCache)) { |
__weak typeof(self) _self = self; |
dispatch_async([self.class _imageQueue], ^{ |
__strong typeof(_self) self = _self; |
if (!self [selfisCancelled]) return; |
UIImage *image = [self.cache getImageForKey:self.cacheKey withType:YYImageCacheTypeDisk]; |
if (image) { |
[self.cache setImage:image imageData:nilforKey:self.cacheKey withType:YYImageCacheTypeMemory]; |
[selfperformSelector:@selector(_didReceiveImageFromDiskCache:) onThread:[self.class _networkThread] withObject:image waitUntilDone:NO]; |
} else { |
[selfperformSelector:@selector(_startRequest:) onThread:[self.class _networkThread] withObject:nilwaitUntilDone:NO]; |
} |
}); |
return; |
} |
} |
} |
[selfperformSelector:@selector(_startRequest:) onThread:[self.class _networkThread] withObject:nilwaitUntilDone:NO]; |
} |
// runs on network thread |
- (void)_startRequest:(id)object { |
if ([selfisCancelled]) return; |
@autoreleasepool { |
if ((_options & YYWebImageOptionIgnoreFailedURL) && URLBlackListContains(_request.URL)) { |
NSError *error = [NSErrorerrorWithDomain:NSURLErrorDomaincode:NSURLErrorFileDoesNotExistuserInfo:@{ NSLocalizedDescriptionKey : @'Failed to load URL, blacklisted.' }]; |
[_lock lock]; |
if (![selfisCancelled]) { |
if (_completion) _completion(nil, _request.URL, YYWebImageFromNone, YYWebImageStageFinished, error); |
} |
[self_finish]; |
[_lock unlock]; |
return; |
} |
if (_request.URL.isFileURL) { |
NSArray *keys = @[NSURLFileSizeKey]; |
NSDictionary *attr = [_request.URL resourceValuesForKeys:keys error:nil]; |
NSNumber *fileSize = attr[NSURLFileSizeKey]; |
_expectedSize = (fileSize != nil) ? fileSize.unsignedIntegerValue : -1; |
} |
// request image from web |
[_lock lock]; |
if (![selfisCancelled]) { |
_connection = [[NSURLConnectionalloc] initWithRequest:_request delegate:[_YYWebImageWeakProxy proxyWithTarget:self]]; |
if (![_request.URL isFileURL] && (_options & YYWebImageOptionShowNetworkActivity)) { |
[YYWebImageManager incrementNetworkActivityCount]; |
} |
} |
[_lock unlock]; |
} |
} |
// runs on network thread, called from outer 'cancel' |
- (void)_cancelOperation { |
@autoreleasepool { |
if (_connection) { |
if (![_request.URL isFileURL] && (_options & YYWebImageOptionShowNetworkActivity)) { |
[YYWebImageManager decrementNetworkActivityCount]; |
} |
} |
[_connection cancel]; |
_connection = nil; |
if (_completion) _completion(nil, _request.URL, YYWebImageFromNone, YYWebImageStageCancelled, nil); |
[self_endBackgroundTask]; |
} |
} |
// runs on network thread |
- (void)_didReceiveImageFromDiskCache:(UIImage *)image { |
@autoreleasepool { |
[_lock lock]; |
if (![selfisCancelled]) { |
if (image) { |
if (_completion) _completion(image, _request.URL, YYWebImageFromDiskCache, YYWebImageStageFinished, nil); |
[self_finish]; |
} else { |
[self_startRequest:nil]; |
} |
} |
[_lock unlock]; |
} |
} |
- (void)_didReceiveImageFromWeb:(UIImage *)image { |
@autoreleasepool { |
[_lock lock]; |
if (![selfisCancelled]) { |
if (_cache) { |
if (image (_options & YYWebImageOptionRefreshImageCache)) { |
NSData *data = _data; |
dispatch_async([YYWebImageOperation _imageQueue], ^{ |
YYImageCacheType cacheType = (_options & YYWebImageOptionIgnoreDiskCache) ? YYImageCacheTypeMemory : YYImageCacheTypeAll; |
[_cache setImage:image imageData:data forKey:_cacheKey withType:cacheType]; |
}); |
} |
} |
_data = nil; |
NSError *error = nil; |
if (!image) { |
error = [NSErrorerrorWithDomain:@'com.ibireme.image'code:-1userInfo:@{ NSLocalizedDescriptionKey : @'Web image decode fail.' }]; |
if (_options & YYWebImageOptionIgnoreFailedURL) { |
if (URLBlackListContains(_request.URL)) { |
error = [NSErrorerrorWithDomain:NSURLErrorDomaincode:NSURLErrorFileDoesNotExistuserInfo:@{ NSLocalizedDescriptionKey : @'Failed to load URL, blacklisted.' }]; |
} else { |
URLInBlackListAdd(_request.URL); |
} |
} |
} |
if (_completion) _completion(image, _request.URL, YYWebImageFromRemote, YYWebImageStageFinished, error); |
[self_finish]; |
} |
[_lock unlock]; |
} |
} |
#pragma mark - NSURLConnectionDelegate runs in operation thread |
- (BOOL)connectionShouldUseCredentialStorage:(NSURLConnection *)connection { |
return _shouldUseCredentialStorage; |
} |
- (void)connection:(NSURLConnection *)connectionwillSendRequestForAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge { |
@autoreleasepool { |
if ([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust]) { |
if (!(_options & YYWebImageOptionAllowInvalidSSLCertificates) && |
[challenge.sender respondsToSelector:@selector(performDefaultHandlingForAuthenticationChallenge:)]) { |
[challenge.sender performDefaultHandlingForAuthenticationChallenge:challenge]; |
} else { |
NSURLCredential *credential = [NSURLCredentialcredentialForTrust:challenge.protectionSpace.serverTrust]; |
[[challenge sender] useCredential:credential forAuthenticationChallenge:challenge]; |
} |
} else { |
if ([challenge previousFailureCount] 0) { |
if (_credential) { |
[[challenge sender] useCredential:_credential forAuthenticationChallenge:challenge]; |
} else { |
[[challenge sender] continueWithoutCredentialForAuthenticationChallenge:challenge]; |
} |
} else { |
[[challenge sender] continueWithoutCredentialForAuthenticationChallenge:challenge]; |
} |
} |
} |
} |
- (NSCachedURLResponse *)connection:(NSURLConnection *)connectionwillCacheResponse:(NSCachedURLResponse *)cachedResponse { |
if (!cachedResponse) return cachedResponse; |
if (_options & YYWebImageOptionUseNSURLCache) { |
return cachedResponse; |
} else { |
// ignore NSURLCache |
returnnil; |
} |
} |
- (void)connection:(NSURLConnection *)connectiondidReceiveResponse:(NSURLResponse *)response { |
@autoreleasepool { |
NSError *error = nil; |
if ([response isKindOfClass:[NSHTTPURLResponseclass]]) { |
NSHTTPURLResponse *httpResponse = (id) response; |
NSInteger statusCode = httpResponse.statusCode; |
if (statusCode >= 400 statusCode 304) { |
error = [NSErrorerrorWithDomain:NSURLErrorDomaincode:statusCode userInfo:nil]; |
} |
} |
if (error) { |
[_connection cancel]; |
[selfconnection:_connection didFailWithError:error]; |
} else { |
if (response.expectedContentLength) { |
_expectedSize = (NSInteger)response.expectedContentLength; |
if (_expectedSize < 0) _expectedSize = -1; |
} |
_data = [NSMutableDatadataWithCapacity:_expectedSize > 0 ? _expectedSize : 0]; |
if (_progress) { |
[_lock lock]; |
if (![selfisCancelled]) _progress(0, _expectedSize); |
[_lock unlock]; |
} |
} |
} |
} |
- (void)connection:(NSURLConnection *)connectiondidReceiveData:(NSData *)data { |
@autoreleasepool { |
[_lock lock]; |
BOOL canceled = [selfisCancelled]; |
[_lock unlock]; |
if (canceled) return; |
if (data) [_data appendData:data]; |
if (_progress) { |
[_lock lock]; |
if (![selfisCancelled]) { |
_progress(_data.length, _expectedSize); |
} |
[_lock unlock]; |
} |
/*--------------------------- progressive ----------------------------*/ |
BOOL progressive = (_options & YYWebImageOptionProgressive) > 0; |
BOOL progressiveBlur = (_options & YYWebImageOptionProgressiveBlur) > 0; |
if (!_completion !(progressive progressiveBlur)) return; |
if (data.length <= 16) return; |
if (_expectedSize > 0 && data.length >= _expectedSize * 0.99) return; |
if (_progressiveIgnored) return; |
NSTimeInterval min = progressiveBlur ? MIN_PROGRESSIVE_BLUR_TIME_INTERVAL : MIN_PROGRESSIVE_TIME_INTERVAL; |
NSTimeInterval now = CACurrentMediaTime(); |
if (now - _lastProgressiveDecodeTimestamp < min) return; |
if (!_progressiveDecoder) { |
_progressiveDecoder = [[YYImageDecoder alloc] initWithScale:[UIScreen mainScreen].scale]; |
} |
[_progressiveDecoder updateData:_data final:NO]; |
if ([selfisCancelled]) return; |
if (_progressiveDecoder.type YYImageTypeUnknown |
_progressiveDecoder.type YYImageTypeWebP |
_progressiveDecoder.type YYImageTypeOther) { |
_progressiveDecoder = nil; |
_progressiveIgnored = YES; |
return; |
} |
if (progressiveBlur) { // only support progressive JPEG and interlaced PNG |
if (_progressiveDecoder.type != YYImageTypeJPEG && |
_progressiveDecoder.type != YYImageTypePNG) { |
_progressiveDecoder = nil; |
_progressiveIgnored = YES; |
return; |
} |
} |
if (_progressiveDecoder.frameCount0) return; |
if (!progressiveBlur) { |
YYImageFrame *frame = [_progressiveDecoder frameAtIndex:0decodeForDisplay:YES]; |
if (frame.image) { |
[_lock lock]; |
if (![selfisCancelled]) { |
_completion(frame.image, _request.URL, YYWebImageFromRemote, YYWebImageStageProgress, nil); |
_lastProgressiveDecodeTimestamp = now; |
} |
[_lock unlock]; |
} |
return; |
} else { |
if (_progressiveDecoder.type YYImageTypeJPEG) { |
if (!_progressiveDetected) { |
NSDictionary *dic = [_progressiveDecoder framePropertiesAtIndex:0]; |
NSDictionary *jpeg = dic[(id)kCGImagePropertyJFIFDictionary]; |
NSNumber *isProg = jpeg[(id)kCGImagePropertyJFIFIsProgressive]; |
if (!isProg.boolValue) { |
_progressiveIgnored = YES; |
_progressiveDecoder = nil; |
return; |
} |
_progressiveDetected = YES; |
} |
NSInteger scanLength = (NSInteger)_data.length - (NSInteger)_progressiveScanedLength - 4; |
if (scanLength <= 2) return; |
NSRange scanRange = NSMakeRange(_progressiveScanedLength, scanLength); |
NSRange markerRange = [_data rangeOfData:JPEGSOSMarker() options:kNilOptionsrange:scanRange]; |
_progressiveScanedLength = _data.length; |
if (markerRange.locationNSNotFound) return; |
if ([selfisCancelled]) return; |
} elseif (_progressiveDecoder.type YYImageTypePNG) { |
if (!_progressiveDetected) { |
NSDictionary *dic = [_progressiveDecoder framePropertiesAtIndex:0]; |
NSDictionary *png = dic[(id)kCGImagePropertyPNGDictionary]; |
NSNumber *isProg = png[(id)kCGImagePropertyPNGInterlaceType]; |
if (!isProg.boolValue) { |
_progressiveIgnored = YES; |
_progressiveDecoder = nil; |
return; |
} |
_progressiveDetected = YES; |
} |
} |
YYImageFrame *frame = [_progressiveDecoder frameAtIndex:0decodeForDisplay:YES]; |
UIImage *image = frame.image; |
if (!image) return; |
if ([selfisCancelled]) return; |
if (!YYCGImageLastPixelFilled(image.CGImage)) return; |
_progressiveDisplayCount++; |
CGFloat radius = 32; |
if (_expectedSize > 0) { |
radius *= 1.0 / (3 * _data.length / (CGFloat)_expectedSize + 0.6) - 0.25; |
} else { |
radius /= (_progressiveDisplayCount); |
} |
image = [image yy_imageByBlurRadius:radius tintColor:niltintMode:0saturation:1maskImage:nil]; |
if (image) { |
[_lock lock]; |
if (![selfisCancelled]) { |
_completion(image, _request.URL, YYWebImageFromRemote, YYWebImageStageProgress, nil); |
_lastProgressiveDecodeTimestamp = now; |
} |
[_lock unlock]; |
} |
} |
} |
} |
- (void)connectionDidFinishLoading:(NSURLConnection *)connection { |
@autoreleasepool { |
[_lock lock]; |
_connection = nil; |
if (![selfisCancelled]) { |
__weak typeof(self) _self = self; |
dispatch_async([self.class _imageQueue], ^{ |
__strong typeof(_self) self = _self; |
if (!self) return; |
BOOL shouldDecode = (self.options & YYWebImageOptionIgnoreImageDecoding) 0; |
BOOL allowAnimation = (self.options & YYWebImageOptionIgnoreAnimatedImage) 0; |
UIImage *image; |
BOOL hasAnimation = NO; |
if (allowAnimation) { |
image = [[YYImage alloc] initWithData:self.data scale:[UIScreen mainScreen].scale]; |
if (shouldDecode) image = [image yy_imageByDecoded]; |
if ([((YYImage *)image) animatedImageFrameCount] > 1) { |
hasAnimation = YES; |
} |
} else { |
YYImageDecoder *decoder = [YYImageDecoder decoderWithData:self.data scale:[UIScreen mainScreen].scale]; |
image = [decoder frameAtIndex:0decodeForDisplay:shouldDecode].image; |
} |
/* |
If the image has animation, save the original image data to disk cache. |
If the image is not PNG or JPEG, re-encode the image to PNG or JPEG for |
better decoding performance. |
*/ |
YYImageType imageType = YYImageDetectType((__bridge CFDataRef)self.data); |
switch (imageType) { |
case YYImageTypeJPEG: |
case YYImageTypeGIF: |
case YYImageTypePNG: |
case YYImageTypeWebP: { // save to disk cache |
if (!hasAnimation) { |
if (imageType YYImageTypeGIF |
imageType YYImageTypeWebP) { |
self.data = nil; // clear the data, re-encode for disk cache |
} |
} |
} break; |
default: { |
self.data = nil; // clear the data, re-encode for disk cache |
} break; |
} |
if ([selfisCancelled]) return; |
if (self.transform && image) { |
UIImage *newImage = self.transform(image, self.request.URL); |
if (newImage != image) { |
self.data = nil; |
} |
image = newImage; |
if ([selfisCancelled]) return; |
} |
[selfperformSelector:@selector(_didReceiveImageFromWeb:) onThread:[self.class _networkThread] withObject:image waitUntilDone:NO]; |
}); |
if (![self.request.URL isFileURL] && (self.options & YYWebImageOptionShowNetworkActivity)) { |
[YYWebImageManager decrementNetworkActivityCount]; |
} |
} |
[_lock unlock]; |
} |
} |
- (void)connection:(NSURLConnection *)connectiondidFailWithError:(NSError *)error { |
@autoreleasepool { |
[_lock lock]; |
if (![selfisCancelled]) { |
if (_completion) { |
_completion(nil, _request.URL, YYWebImageFromNone, YYWebImageStageFinished, error); |
} |
_connection = nil; |
_data = nil; |
if (![_request.URL isFileURL] && (_options & YYWebImageOptionShowNetworkActivity)) { |
[YYWebImageManager decrementNetworkActivityCount]; |
} |
[self_finish]; |
if (_options & YYWebImageOptionIgnoreFailedURL) { |
if (error.code != NSURLErrorNotConnectedToInternet && |
error.code != NSURLErrorCancelled && |
error.code != NSURLErrorTimedOut && |
error.code != NSURLErrorUserCancelledAuthentication && |
error.code != NSURLErrorNetworkConnectionLost) { |
URLInBlackListAdd(_request.URL); |
} |
} |
} |
[_lock unlock]; |
} |
} |
#pragma mark - Override NSOperation |
- (void)start { |
@autoreleasepool { |
[_lock lock]; |
self.started = YES; |
if ([selfisCancelled]) { |
[selfperformSelector:@selector(_cancelOperation) onThread:[[selfclass] _networkThread] withObject:nilwaitUntilDone:NOmodes:@[NSDefaultRunLoopMode]]; |
self.finished = YES; |
} elseif ([selfisReady] && ![selfisFinished] && ![selfisExecuting]) { |
if (!_request) { |
self.finished = YES; |
if (_completion) { |
NSError *error = [NSErrorerrorWithDomain:NSURLErrorDomaincode:NSURLErrorFileDoesNotExistuserInfo:@{NSLocalizedDescriptionKey:@'request in nil'}]; |
_completion(nil, _request.URL, YYWebImageFromNone, YYWebImageStageFinished, error); |
} |
} else { |
self.executing = YES; |
[selfperformSelector:@selector(_startOperation) onThread:[[selfclass] _networkThread] withObject:nilwaitUntilDone:NOmodes:@[NSDefaultRunLoopMode]]; |
if ((_options & YYWebImageOptionAllowBackgroundTask) && _YYSharedApplication()) { |
__weak __typeof__ (self) _self = self; |
if (_taskID UIBackgroundTaskInvalid) { |
_taskID = [_YYSharedApplication() beginBackgroundTaskWithExpirationHandler:^{ |
__strong __typeof (_self) self = _self; |
if (self) { |
[selfcancel]; |
self.finished = YES; |
} |
}]; |
} |
} |
} |
} |
[_lock unlock]; |
} |
} |
- (void)cancel { |
[_lock lock]; |
if (![selfisCancelled]) { |
[supercancel]; |
self.cancelled = YES; |
if ([selfisExecuting]) { |
self.executing = NO; |
[selfperformSelector:@selector(_cancelOperation) onThread:[[selfclass] _networkThread] withObject:nilwaitUntilDone:NOmodes:@[NSDefaultRunLoopMode]]; |
} |
if (self.started) { |
self.finished = YES; |
} |
} |
[_lock unlock]; |
} |
- (void)setExecuting:(BOOL)executing { |
[_lock lock]; |
if (_executing != executing) { |
[selfwillChangeValueForKey:@'isExecuting']; |
_executing = executing; |
[selfdidChangeValueForKey:@'isExecuting']; |
} |
[_lock unlock]; |
} |
- (BOOL)isExecuting { |
[_lock lock]; |
BOOL executing = _executing; |
[_lock unlock]; |
return executing; |
} |
- (void)setFinished:(BOOL)finished { |
[_lock lock]; |
if (_finished != finished) { |
[selfwillChangeValueForKey:@'isFinished']; |
_finished = finished; |
[selfdidChangeValueForKey:@'isFinished']; |
} |
[_lock unlock]; |
} |
- (BOOL)isFinished { |
[_lock lock]; |
BOOL finished = _finished; |
[_lock unlock]; |
return finished; |
} |
- (void)setCancelled:(BOOL)cancelled { |
[_lock lock]; |
if (_cancelled != cancelled) { |
[selfwillChangeValueForKey:@'isCancelled']; |
_cancelled = cancelled; |
[selfdidChangeValueForKey:@'isCancelled']; |
} |
[_lock unlock]; |
} |
- (BOOL)isCancelled { |
[_lock lock]; |
BOOL cancelled = _cancelled; |
[_lock unlock]; |
return cancelled; |
} |
- (BOOL)isConcurrent { |
returnYES; |
} |
- (BOOL)isAsynchronous { |
returnYES; |
} |
+ (BOOL)automaticallyNotifiesObserversForKey:(NSString *)key { |
if ([key isEqualToString:@'isExecuting'] |
[key isEqualToString:@'isFinished'] |
[key isEqualToString:@'isCancelled']) { |
returnNO; |
} |
return [superautomaticallyNotifiesObserversForKey:key]; |
} |
- (NSString *)description { |
NSMutableString *string = [NSMutableStringstringWithFormat:@'<%@: %p',self.class, self]; |
[string appendFormat:@' executing:%@', [selfisExecuting] ? @'YES' : @'NO']; |
[string appendFormat:@' finished:%@', [selfisFinished] ? @'YES' : @'NO']; |
[string appendFormat:@' cancelled:%@', [selfisCancelled] ? @'YES' : @'NO']; |
[string appendString:@'>']; |
return string; |
} |
@end |
Copy lines Copy permalink
Chroma subsampling is the practice of encoding images by implementing less resolution for chromainformation than for luma information, taking advantage of the human visual system's lower acuity for color differences than for luminance.[1]
It is used in many video encoding schemes – both analog and digital – and also in JPEG encoding.
- 4Types of sampling and subsampling
Rationale[edit]
Digital signals are often compressed to reduce file size and save transmission time. Since the human visual system is much more sensitive to variations in brightness than color, a video system can be optimized by devoting more bandwidth to the luma component (usually denoted Y'), than to the color difference components Cb and Cr. In compressed images, for example, the 4:2:2 Y'CbCr scheme requires two-thirds the bandwidth of (4:4:4) R'G'B'. This reduction results in almost no visual difference as perceived by the viewer.
How subsampling works[edit]
Because the human visual system is less sensitive to the position and motion of color than luminance,[2]bandwidth can be optimized by storing more luminance detail than color detail. At normal viewing distances, there is no perceptible loss incurred by sampling the color detail at a lower rate[vague]. In video systems, this is achieved through the use of color difference components. The signal is divided into a luma (Y') component and two color difference components (chroma).
In human vision there are three channels for color detection, and for many color systems, three 'channels' are sufficient for representing most colors. For example: red, green, blue or magenta, yellow, cyan. But there are other ways to represent the color. In many video systems, the three channels are: luminance and two chroma channels. In video, the luma and chroma components are formed as a weighted sum of gamma-corrected (tristimulus) R'G'B' components instead of linear (tristimulus) RGB components. As a result, luma must be distinguished from luminance. That there is some 'bleeding' of luminance and color information between the luma and chroma components in video, the error being greatest for highly saturated colors and noticeable in between the magenta and green bars of a color-bar test pattern (that has chroma subsampling applied), should not be attributed to this engineering approximation being used. Indeed, similar bleeding can occur also with gamma = 1, whence the reversing of the order of operations between gamma correction and forming the weighted sum can make no difference. The chroma can influence the luma specifically at the pixels where the subsampling put no chroma. Interpolation may then put chroma values there which are incompatible with the luma value there, and further post-processing of that Y'CbCr into R'G'B' for that pixel is what ultimately produces false luminance upon display.
Original without color subsampling. 200% zoom.
Image after color subsampling (compressed with Sony Vegas DV codec, box filtering applied.)
Sampling systems and ratios[edit]
The subsampling scheme is commonly expressed as a three part ratio J:a:b (e.g. 4:2:2) or four parts if alpha channel is present (e.g. 4:2:2:4), that describe the number of luminance and chrominance samples in a conceptual region that is J pixels wide, and 2 pixels high. The parts are (in their respective order):
- J: horizontal sampling reference (width of the conceptual region). Usually, 4.
- a: number of chrominance samples (Cr, Cb) in the first row of J pixels.
- b: number of changes of chrominance samples (Cr, Cb) between first and second row of J pixels.
- Alpha: horizontal factor (relative to first digit). May be omitted if alpha component is not present, and is equal to J when present.
This notation is not valid for all combinations and has exceptions, e.g. 4:1:0 (where the height of the region is not 2 pixels but 4 pixels, so if 8 bits/component are used the media would be 9 bits/pixel) and 4:2:1.
An explanatory image of different chroma subsampling schemes can be seen at the following link: http://lea.hamradio.si/~s51kq/subsample.gif(source: 'Basics of Video': http://lea.hamradio.si/~s51kq/V-BAS.HTM) or in details in Chrominance Subsampling in Digital Images, by Douglas Kerr.
DAEMON Tools Pro 8.2.1 Crack + Serial Key available for free with direct download link at Viafiles, DAEMON Tools Pro Download is recognized as complete and professional emulation software to operate with disc images and virtual drives. Still standing as the. Daemon Tools Lite serial number supports the broad variety of image types and format of any types. Daemon Tools Lite Full Version serial key. Serial Daemon Tools Lite Download fully supports the broad variety of image types and formats of any types. The user can use image converter to have one format images in the image catalog. Daemon Tools Pro 8 Crack + Serial Key Free Download. Daemon Tools Pro 8 Crack is a basic and advanced and competitive software that will edit the images and drives and virtual drives as well.It’s launched in many years ago, but as you know its popularity become it fresh right now and will, so today I share its new version that compatible with Windows 10 and many more features are included.
4:1:1 | 4:2:0 | 4:2:2 | 4:4:4 | 4:4:0 | |||||||||||||||||||||
Y'CrCb | |||||||||||||||||||||||||
= | = | = | = | = | |||||||||||||||||||||
Y' | |||||||||||||||||||||||||
+ | + | + | + | + | |||||||||||||||||||||
1 | 2 | 3 | 4 | J = 4 | 1 | 2 | 3 | 4 | J = 4 | 1 | 2 | 3 | 4 | J = 4 | 1 | 2 | 3 | 4 | J = 4 | 1 | 2 | 3 | 4 | J = 4 | |
(Cr, Cb) | 1 | a = 1 | 1 | 2 | a = 2 | 1 | 2 | a = 2 | 1 | 2 | 3 | 4 | a = 4 | 1 | 2 | 3 | 4 | a = 4 | |||||||
1 | b = 1 | b = 0 | 1 | 2 | b = 2 | 1 | 2 | 3 | 4 | b = 4 | b = 0 | ||||||||||||||
¼ horizontal resolution, full vertical resolution | ½ horizontal resolution, ½ vertical resolution | ½ horizontal resolution, full vertical resolution | full horizontal resolution, full vertical resolution | full horizontal resolution, ½ vertical resolution |
The mapping examples given are only theoretical and for illustration. Also note that the diagram does not indicate any chroma filtering, which should be applied to avoid aliasing.
To calculate required bandwidth factor relative to 4:4:4 (or 4:4:4:4), one needs to sum all the factors and divide the result by 12 (or 16, if alpha is present).
Types of sampling and subsampling[edit]
4:4:4[edit]
Each of the three Y'CbCr components have the same sample rate, thus there is no chroma subsampling. This scheme is sometimes used in high-end film scanners and cinematic post production.
Note that '4:4:4' may instead be referring to R'G'B' color space, which implicitly also does not have any chroma subsampling. Formats such as HDCAM SR can record 4:4:4 R'G'B' over dual-link HD-SDI.
4:2:2[edit]
The two chroma components are sampled at half the sample rate of luma: the horizontal chroma resolution is halved. This reduces the bandwidth of an uncompressed video signal by one-third with little to no visual difference.[citation needed]
Many high-end digital video formats and interfaces use this scheme:
- DVCPRO50 and DVCPRO HD
- CCIR 601 / Serial Digital Interface / D1
4:2:1[edit]
This sampling mode is not expressible in J:a:b notation. '4:2:1' is an obsolete term from a previous notational scheme, and very few software or hardware codecs use it. Cb horizontal resolution is half that of Cr (and a quarter of the horizontal resolution of Y).
4:1:1[edit]
In 4:1:1 chroma subsampling, the horizontal color resolution is quartered, and the bandwidth is halved compared to no chroma subsampling. Initially, 4:1:1 chroma subsampling of the DV format was not considered to be broadcast quality and was only acceptable for low-end and consumer applications.[3][4] However, DV-based formats (some of which use 4:1:1 chroma subsampling) have been used professionally in electronic news gathering and in playout servers. DV has also been sporadically used in feature films and in digital cinematography.
In the NTSC system, if the luma is sampled at 13.5 MHz, then this means that the Cr and Cb signals will each be sampled at 3.375 MHz, which corresponds to a maximum Nyquist bandwidth of 1.6875 MHz, whereas traditional 'high-end broadcast analog NTSC encoder' would have a Nyquist bandwidth of 1.5 MHz and 0.5 MHz for the I/Q channels. However, in most equipment, especially cheap TV sets and VHS/Betamax VCRs the chroma channels have only the 0.5 MHz bandwidth for both Cr and Cb (or equivalently for I/Q). Thus the DV system actually provides a superior color bandwidth compared to the best composite analog specifications for NTSC, despite having only 1/4 of the chroma bandwidth of a 'full' digital signal.
Formats that use 4:1:1 chroma subsampling include:
- DVCPRO (NTSCandPAL)
- NTSC DV and DVCAM
- D-7
4:2:0[edit]
In 4:2:0, the horizontal sampling is doubled compared to 4:1:1, but as the Cb and Cr channels are only sampled on each alternate line in this scheme, the vertical resolution is halved. The data rate is thus the same. This fits reasonably well with the PAL color encoding system since this has only half the vertical chrominance resolution of NTSC. It would also fit extremely well with the SECAM color encoding system since like that format, 4:2:0 only stores and transmits one color channel per line (the other channel being recovered from the previous line). However, little equipment has actually been produced that outputs a SECAM analogue video signal. In general SECAM territories either have to use a PAL capable display or a transcoder to convert the PAL signal to SECAM for display.
Different variants of 4:2:0 chroma configurations are found in:
- All ISO/IECMPEG and ITU-TVCEG H.26x video coding standards including H.262/MPEG-2 Part 2 implementations (although some profiles of MPEG-4 Part 2 and H.264/MPEG-4 AVC allow higher-quality sampling schemes such as 4:4:4)
- DVD-Video and Blu-ray Disc.[5][6]
- PAL DV and DVCAM
- AVCHD and AVC-Intra 50
- most common JPEG/JFIF and MJPEG implementations
Cb and Cr are each subsampled at a factor of 2 both horizontally and vertically.
Better Jpeg 3.0.0.3 Serial Key Free
There are three variants of 4:2:0 schemes, having different horizontal and vertical siting.[7]
- In MPEG-2, Cb and Cr are cosited horizontally. Cb and Cr are sited between pixels in the vertical direction (sited interstitially).
- In JPEG/JFIF, H.261, and MPEG-1, Cb and Cr are sited interstitially, halfway between alternate luma samples.
- In 4:2:0 DV, Cb and Cr are co-sited in the horizontal direction. In the vertical direction, they are co-sited on alternating lines.
Most digital video formats corresponding to PAL use 4:2:0 chroma subsampling, with the exception of DVCPRO25, which uses 4:1:1 chroma subsampling. Both the 4:1:1 and 4:2:0 schemes halve the bandwidth compared to no chroma subsampling.
With interlaced material, 4:2:0 chroma subsampling can result in motion artifacts if it is implemented the same way as for progressive material. The luma samples are derived from separate time intervals while the chroma samples would be derived from both time intervals. It is this difference that can result in motion artifacts. The MPEG-2 standard allows for an alternate interlaced sampling scheme where 4:2:0 is applied to each field (not both fields at once). This solves the problem of motion artifacts, reduces the vertical chroma resolution by half, and can introduce comb-like artifacts in the image.
Original. *This image shows a single field. The moving text has some motion blur applied to it.
4:2:0 progressive sampling applied to moving interlaced material. Note that the chroma leads and trails the moving text. *This image shows a single field.
4:2:0 interlaced sampling applied to moving interlaced material. *This image shows a single field.
In the 4:2:0 interlaced scheme however, vertical resolution of the chroma is roughly halved since the chroma samples effectively describe an area 2 samples wide by 4 samples tall instead of 2X2. As well, the spatial displacement between both fields can result in the appearance of comb-like chroma artifacts.
Original still image.
4:2:0 progressive sampling applied to a still image. Both fields are shown.
4:2:0 interlaced sampling applied to a still image. Both fields are shown.
If the interlaced material is to be de-interlaced, the comb-like chroma artifacts (from 4:2:0 interlaced sampling) can be removed by blurring the chroma vertically.[8]
4:1:0[edit]
This ratio is possible, and some codecs support it, but it is not widely used. This ratio uses half of the vertical and one-fourth the horizontal color resolutions, with only one-eighth of the bandwidth of the maximum color resolutions used. Uncompressed video in this format with 8-bit quantization uses 10 bytes for every macropixel (which is 4 x 2 pixels). It has the equivalent chrominance bandwidth of a PAL I signal decoded with a delay line decoder, and still very much superior to NTSC.
- Some video codecs may operate at 4:1:0.5 or 4:1:0.25 as an option, so as to allow similar to VHS quality.
3:1:1[edit]
Used by Sony in their HDCAM High Definition recorders (not HDCAM SR). In the horizontal dimension, luma is sampled horizontally at three quarters of the full HD sampling rate – 1440 samples per row instead of 1920. Chroma is sampled at 480 samples per row, a third of the luma sampling rate.
In the vertical dimension, both luma and chroma are sampled at the full HD sampling rate (1080 samples vertically).
Out-of-gamut colors[edit]
One of the artifacts that can occur with chroma subsampling is that out-of-gamut colors can occur upon chroma reconstruction. Suppose the image consisted of alternating 1-pixel red and black lines and the subsampling omitted the chroma for the black pixels. Chroma from the red pixels will be reconstructed onto the black pixels, causing the new pixels to have positive red and negative green and blue values. As displays cannot output negative light (negative light does not exist), these negative values will effectively be clipped and the resulting luma value will be too high.[9] Similar artifacts arise in the less artificial example of gradation near a fairly sharp red/black boundary.
Filtering during subsampling can also cause colors to go out of gamut.
Terminology[edit]
The term Y'UV refers to an analog encoding scheme while Y'CbCr refers to a digital encoding scheme.[citation needed] One difference between the two is that the scale factors on the chroma components (U, V, Cb, and Cr) are different. However, the term YUV is often used erroneously to refer to Y'CbCr encoding. Hence, expressions like '4:2:2 YUV' always refer to 4:2:2 Y'CbCr since there simply is no such thing as 4:x:x in analog encoding (such as YUV).
Better Jpeg 3.0.0.3 Serial Key West
In a similar vein, the term luminance and the symbol Y are often used erroneously to refer to luma, which is denoted with the symbol Y'. Note that the luma (Y') of video engineering deviates from the luminance (Y) of color science (as defined by CIE). Luma is formed as the weighted sum of gamma-corrected (tristimulus) RGB components. Luminance is formed as a weighed sum of linear (tristimulus) RGB components.
In practice, the CIE symbol Y is often incorrectly used to denote luma. In 1993, SMPTE adopted Engineering Guideline EG 28, clarifying the two terms. Note that the prime symbol ' is used to indicate gamma correction.
Better Jpeg 3.0.0.3 Serial Key Code
Similarly, the chroma/chrominance of video engineering differs from the chrominance of color science. The chroma/chrominance of video engineering is formed from weighted tristimulus components, not linear components. In video engineering practice, the terms chroma, chrominance, and saturation are often used interchangeably to refer to chrominance.
History[edit]
Chroma subsampling was developed in the 1950s by Alda Bedford for the development of color television by RCA, which developed into the NTSC standard; luma-chroma separation was developed earlier, in 1938 by Georges Valensi.
Through studies, he showed that the human eye has high resolution only for black and white, somewhat less for 'mid-range' colors like yellows and greens, and much less for colors on the end of the spectrum, reds and blues. Using this knowledge allowed RCA to develop a system in which they discarded most of the blue signal after it comes from the camera, keeping most of the green and only some of the red; this is chroma subsampling in the YIQ color space, and is roughly analogous to 4:2:1 subsampling, in that it has decreasing resolution for luma, yellow/green, and red/blue.
See also[edit]
- SMPTE - Society of Motion Picture and Television Engineers
- CCIR 601 4:2:2 SDTV
- Color vision
- Better pictorial explanation here [1]
References[edit]
- ^S. Winkler, C. J. van den Branden Lambrecht, and M. Kunt (2001). 'Vision and Video: Models and Applications'. In Christian J. van den Branden Lambrecht (ed.). Vision models and applications to image and video processing. Springer. p. 209. ISBN978-0-7923-7422-0.CS1 maint: Multiple names: authors list (link)
- ^Livingstone, Margaret (2002). 'The First Stages of Processing Color and Luminance: Where and What'. Vision and Art: The Biology of Seeing. New York: Harry N. Abrams. pp. 46–67. ISBN0-8109-0406-3.
- ^Jennings, Roger; Bertel Schmitt (1997). 'DV vs. Betacam SP'. DV Central. Archived from the original on 2008-07-02. Retrieved 2008-08-29.
- ^Wilt, Adam J. (2006). 'DV, DVCAM & DVCPRO Formats'. adamwilt.com. Retrieved 2008-08-29.
- ^Clint DeBoer (2008-04-16). 'HDMI Enhanced Black Levels, xvYCC and RGB'. Audioholics. Retrieved 2013-06-02.
- ^'Digital Color Coding'(PDF). Telairity. Archived from the original(PDF) on 2014-01-07. Retrieved 2013-06-02.
- ^Poynton, Charles (2008). 'Chroma Subsampling Notation'(PDF). Poynton.com. Retrieved 2008-10-01.
- ^Munsil, Don; Stacey Spears (2003). 'DVD Player Benchmark - Chroma Upsampling Error'. Secrets of Home Theater and High Fidelity. Archived from the original on 2008-06-06. Retrieved 2008-08-29.
- ^Chan, Glenn (May–June 2008). 'Towards Better Chroma Subsampling'. GlennChan.info. SMPTE Journal. Retrieved 2008-08-29.
- Poynton, Charles. 'YUV and luminance considered harmful: A plea for precise terminology in video' [2]
- Poynton, Charles. 'Digital Video and HDTV: Algorithms and Interfaces'. U.S.: Morgan Kaufmann Publishers, 2003.
- Kerr, Douglas A. 'Chrominance Subsampling in Digital Images' [3]