Получить значение RGB из пресетов UIColor
Я мое приложение передаю значение цвета RGB серверу. В моем приложении используются предопределенные значения UIColor, такие как [UIColor grayColor], [UIColor redColor]. Я знаю, что могу использовать следующий код:
const CGFloat *c = CGColorGetComponents(color.CGColor)
но только для цветов, находящихся в цветовом пространстве RBG, однако [UIColor grayColor] не является.
Есть ли способ получить значения RGB для цветов без RBG?
Спасибо!
Ответы
Ответ 1
UIColor имеет метод, который дает вам компоненты RGB (-getRed:green:blue:alpha:
), который отлично работает на iOS 7 или выше. На iOS 6 и ранее этот метод завершится с ошибкой и вернет NO
, если цвет не находится в цветовом пространстве RGB (как и для [UIColor grayColor]
.)
Для iOS 6 и более ранних версий единственным способом, который я знаю для этого, который работает во всех цветовых пространствах, является создание контекста растровой графики Core Graphics в цветовом пространстве RGB и втягивание в него вашего цвета. Затем вы можете считывать значения RGB из полученного растрового изображения. Обратите внимание, что это не будет работать для определенных цветов, таких как цвета шаблонов (например, [UIColor groupTableViewBackgroundColor]), которые не имеют разумных значений RGB.
- (void)getRGBComponents:(CGFloat [3])components forColor:(UIColor *)color {
CGColorSpaceRef rgbColorSpace = CGColorSpaceCreateDeviceRGB();
unsigned char resultingPixel[4];
CGContextRef context = CGBitmapContextCreate(&resultingPixel,
1,
1,
8,
4,
rgbColorSpace,
kCGImageAlphaNoneSkipLast);
CGContextSetFillColorWithColor(context, [color CGColor]);
CGContextFillRect(context, CGRectMake(0, 0, 1, 1));
CGContextRelease(context);
CGColorSpaceRelease(rgbColorSpace);
for (int component = 0; component < 3; component++) {
components[component] = resultingPixel[component] / 255.0f;
}
}
Вы можете использовать его примерно так:
CGFloat components[3];
[self getRGBComponents:components forColor:[UIColor grayColor]];
NSLog(@"%f %f %f", components[0], components[1], components[2]);
Ответ 2
Я знаю только один правильный способ сделать это - и это нужно вызвать CGGetColorComponents()
. Я попытался использовать UIColor
-getRed:green:blue:alpha
, но он не работает правильно в оттенках серого. Итак, вот что я делаю:
Во-первых, у меня есть простой struct
, который позволяет легко манипулировать цветами RGBA. Эта структура полезна в других местах для выполнения всех видов манипуляций с цветом, таких как коррекция гаммы и усреднение цвета.
typedef struct
{
CGFloat r; // Red component (0 <= r <= 1)
CGFloat g; // Green component (0 <= g <= 1)
CGFloat b; // Blue component (0 <= b <= 1)
CGFloat a; // Alpha/opacity component (0 <= a <= 1)
}
RGBA;
Тогда у меня есть функция, которая возвращает структуру RGBA
, заданную a UIColor
.
RGBA RGBAFromUIColor(UIColor *color)
{
return RGBAFromCGColor(color.CGColor);
}
Он передает UIColor
CGColor
функции нижнего уровня для тяжелого подъема.
RGBA RGBAFromCGColor(CGColorRef color)
{
RGBA rgba;
CGColorSpaceRef color_space = CGColorGetColorSpace(color);
CGColorSpaceModel color_space_model = CGColorSpaceGetModel(color_space);
const CGFloat *color_components = CGColorGetComponents(color);
int color_component_count = CGColorGetNumberOfComponents(color);
switch (color_space_model)
{
case kCGColorSpaceModelMonochrome:
{
assert(color_component_count == 2);
rgba = (RGBA)
{
.r = color_components[0],
.g = color_components[0],
.b = color_components[0],
.a = color_components[1]
};
break;
}
case kCGColorSpaceModelRGB:
{
assert(color_component_count == 4);
rgba = (RGBA)
{
.r = color_components[0],
.g = color_components[1],
.b = color_components[2],
.a = color_components[3]
};
break;
}
default:
{
NSLog(@"Unsupported color space model %i", color_space_model);
rgba = (RGBA) { 0, 0, 0, 0 };
break;
}
}
return rgba;
}
Таким образом, это отлично подходит для оттенков серого (например, UIColor.whiteColor
, UIColor.blackColor
, UIColor.grayColor
и т.д.), а также любого красного/зеленого/синего/альфа-цвета. Однако он не будет работать с цветами HSB.
Ответ 3
Добавляя к ответу Джесси, вот как сохранить альфа:
- (void)getRGBAComponents:(CGFloat [4])components forColor:(UIColor *)color {
CGColorSpaceRef rgbColorSpace = CGColorSpaceCreateDeviceRGB();
unsigned char resultingPixel[4] = {0};
CGContextRef context = CGBitmapContextCreate(&resultingPixel,
1,
1,
8,
4,
rgbColorSpace,
kCGImageAlphaPremultipliedLast);
CGContextSetFillColorWithColor(context, [color CGColor]);
CGContextFillRect(context, CGRectMake(0, 0, 1, 1));
CGContextRelease(context);
CGColorSpaceRelease(rgbColorSpace);
CGFloat a = resultingPixel[3] / 255.0;
CGFloat unpremultiply = (a != 0.0) ? 1.0 / a / 255.0 : 0.0;
for (int component = 0; component < 3; component++) {
components[component] = resultingPixel[component] * unpremultiply;
}
components[3] = a;
}
Ответ 4
Перекрестный указатель из мой ответ на "Как преобразовать цвета из одного цветового пространства в другое?" .
Чтобы определить разные цветовые пространства, вы можете получить CGColorSpaceModel
из цвета CGColorSpaceRef
:
UIColor* color = some color;
CGColorSpaceRef colorSpace = CGColorGetColorSpace([color CGColor]);
CGColorSpaceModel colorSpaceModel = CGColorSpaceGetModel(colorSpace);
Затем вы можете сравнить colorSpaceModel
с константами, определенными в CoreGraphics/CGColorSpace.h
. UIColor getRed:green:blue:alpha
работает для kCGColorSpaceModelRGB
, тогда как getWhite:alpha
работает для kCGColorSpaceModelMonochrome
.
Ответ 5
вы можете получить значение RGB, просто используя
UIColor *chooseColor = view.backgroundColor;
CGFloat red,green,blue,alpha;
[chooseColor getRed:&red green:&green blue:&blue alpha:&alpha];
спасибо:-)