Как изменить цвет фона UIButton, когда он выделен?
В какой-то момент в моем приложении у меня есть выделенный UIButton
(например, когда пользователь нажимает на кнопку), и мне нужно изменить цвет фона, когда кнопка подсвечена (так что, пока палец пользователя все еще находится на кнопке).
Я попробовал следующее:
_button.backgroundColor = [UIColor redColor];
Но он не работает. Цвет остается тем же. Я попробовал ту же часть кода, когда кнопка не подсвечена, и она работает нормально. Я также попробовал позвонить -setNeedsDisplay
после изменения цвета, это не имело никакого эффекта.
Как заставить кнопку изменить цвет фона?
Ответы
Ответ 1
Вы можете переопределить метод UIButton
setHighlighted
.
Objective-C
- (void)setHighlighted:(BOOL)highlighted {
[super setHighlighted:highlighted];
if (highlighted) {
self.backgroundColor = UIColorFromRGB(0x387038);
} else {
self.backgroundColor = UIColorFromRGB(0x5bb75b);
}
}
Swift 3.0 и Swift 4.1
override open var isHighlighted: Bool {
didSet {
backgroundColor = isHighlighted ? UIColor.black : UIColor.white
}
}
Ответ 2
Не уверен, что этот вид решает, что вам нужно, или соответствует вашему общему ландшафту развития, но первое, что я хотел бы попробовать, это изменить цвет фона кнопки на событии touchDown.
Вариант 1:
Вам понадобится два события для захвата, например UIControlEventTouchDown, когда пользователь нажимает кнопку. UIControlEventTouchUpInside и UIControlEventTouchUpOutside будут, когда они отпустите кнопку, чтобы вернуть ее в нормальное состояние.
UIButton *myButton = [UIButton buttonWithType:UIButtonTypeCustom];
[myButton setFrame:CGRectMake(10.0f, 10.0f, 100.0f, 20.f)];
[myButton setBackgroundColor:[UIColor blueColor]];
[myButton setTitle:@"click me:" forState:UIControlStateNormal];
[myButton setTitle:@"changed" forState:UIControlStateHighlighted];
[myButton addTarget:self action:@selector(buttonHighlight:) forControlEvents:UIControlEventTouchDown];
[myButton addTarget:self action:@selector(buttonNormal:) forControlEvents:UIControlEventTouchUpInside];
Вариант 2:
Верните изображение, сделанное из выделенного цвета, который вы хотите. Это также может быть категория.
+ (UIImage *)imageWithColor:(UIColor *)color {
CGRect rect = CGRectMake(0.0f, 0.0f, 1.0f, 1.0f);
UIGraphicsBeginImageContext(rect.size);
CGContextRef context = UIGraphicsGetCurrentContext();
CGContextSetFillColorWithColor(context, [color CGColor]);
CGContextFillRect(context, rect);
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
return image;
}
а затем измените выделенное состояние кнопки:
[myButton setBackgroundImage:[self imageWithColor:[UIColor greenColor]] forState:UIControlStateHighlighted];
Ответ 3
Нет необходимости переопределять highlighted
как вычисляемое свойство. Вы можете использовать наблюдатель свойства, чтобы вызвать изменение цвета фона:
override var highlighted: Bool {
didSet {
backgroundColor = highlighted ? UIColor.lightGrayColor() : UIColor.whiteColor()
}
}
Swift 4
override open var isHighlighted: Bool {
didSet {
backgroundColor = isHighlighted ? UIColor.lightGray : UIColor.white
}
}
Ответ 4
В Swift вы можете переопределить метод доступа к выделенному (или выбранному) свойству, а не переопределять метод setHighlighted
override var highlighted: Bool {
get {
return super.highlighted
}
set {
if newValue {
backgroundColor = UIColor.blackColor()
}
else {
backgroundColor = UIColor.whiteColor()
}
super.highlighted = newValue
}
}
Ответ 5
Удобное общее расширение в Swift:
extension UIButton {
private func imageWithColor(color: UIColor) -> UIImage {
let rect = CGRectMake(0.0, 0.0, 1.0, 1.0)
UIGraphicsBeginImageContext(rect.size)
let context = UIGraphicsGetCurrentContext()
CGContextSetFillColorWithColor(context, color.CGColor)
CGContextFillRect(context, rect)
let image = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
return image
}
func setBackgroundColor(color: UIColor, forUIControlState state: UIControlState) {
self.setBackgroundImage(imageWithColor(color), forState: state)
}
}
Swift 3.0
extension UIButton {
private func imageWithColor(color: UIColor) -> UIImage? {
let rect = CGRect(x: 0.0, y: 0.0, width: 1.0, height: 1.0)
UIGraphicsBeginImageContext(rect.size)
let context = UIGraphicsGetCurrentContext()
context?.setFillColor(color.cgColor)
context?.fill(rect)
let image = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
return image
}
func setBackgroundColor(_ color: UIColor, for state: UIControlState) {
self.setBackgroundImage(imageWithColor(color: color), for: state)
}
}
Ответ 6
Переопределить выделенную переменную.
Добавление @IBInspectable
позволяет отредактировать выделенный backgroundColor в раскадровке, что тоже отлично.
class BackgroundHighlightedButton: UIButton {
@IBInspectable var highlightedBackgroundColor :UIColor?
@IBInspectable var nonHighlightedBackgroundColor :UIColor?
override var highlighted :Bool {
get {
return super.highlighted
}
set {
if newValue {
self.backgroundColor = highlightedBackgroundColor
}
else {
self.backgroundColor = nonHighlightedBackgroundColor
}
super.highlighted = newValue
}
}
}
Ответ 7
более компактное решение (на основе ответа @aleksejs-mjaliks):
Свифт 3/4+:
override var isHighlighted: Bool {
didSet {
backgroundColor = isHighlighted ? .lightGray : .white
}
}
Свифт 2:
override var highlighted: Bool {
didSet {
backgroundColor = highlighted ? UIColor.lightGrayColor() : UIColor.whiteColor()
}
}
Если вы не хотите переопределять, это обновленная версия ответа @timur-bernikowich (Swift 4.2):
extension UIButton {
func setBackgroundColor(_ color: UIColor, forState controlState: UIControl.State) {
let colorImage = UIGraphicsImageRenderer(size: CGSize(width: 1, height: 1)).image { _ in
color.setFill()
UIBezierPath(rect: CGRect(x: 0, y: 0, width: 1, height: 1)).fill()
}
setBackgroundImage(colorImage, for: controlState)
}
}
Ответ 8
Расширение UIButton с синтаксисом Swift 3+:
extension UIButton {
func setBackgroundColor(color: UIColor, forState: UIControlState) {
UIGraphicsBeginImageContext(CGSize(width: 1, height: 1))
UIGraphicsGetCurrentContext()!.setFillColor(color.cgColor)
UIGraphicsGetCurrentContext()!.fill(CGRect(x: 0, y: 0, width: 1, height: 1))
let colorImage = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
self.setBackgroundImage(colorImage, for: forState)
}}
Используйте это как:
YourButton.setBackgroundColor(color: UIColor.white, forState: .highlighted)
Оригинальный ответ: fooobar.com/questions/34842/...
Ответ 9
Здесь подход в Swift, используя расширение UIButton для добавления IBInspectable, называется highlightBackgroundColor. Подобно подклассу, не требуя подкласса.
private var HighlightedBackgroundColorKey = 0
private var NormalBackgroundColorKey = 0
extension UIButton {
@IBInspectable var highlightedBackgroundColor: UIColor? {
get {
return objc_getAssociatedObject(self, &HighlightedBackgroundColorKey) as? UIColor
}
set(newValue) {
objc_setAssociatedObject(self,
&HighlightedBackgroundColorKey, newValue, UInt(OBJC_ASSOCIATION_RETAIN))
}
}
private var normalBackgroundColor: UIColor? {
get {
return objc_getAssociatedObject(self, &NormalBackgroundColorKey) as? UIColor
}
set(newValue) {
objc_setAssociatedObject(self,
&NormalBackgroundColorKey, newValue, UInt(OBJC_ASSOCIATION_RETAIN))
}
}
override public var backgroundColor: UIColor? {
didSet {
if !highlighted {
normalBackgroundColor = backgroundColor
}
}
}
override public var highlighted: Bool {
didSet {
if let highlightedBackgroundColor = self.highlightedBackgroundColor {
if highlighted {
backgroundColor = highlightedBackgroundColor
} else {
backgroundColor = normalBackgroundColor
}
}
}
}
}
Надеюсь, это поможет.
Ответ 10
Вы можете использовать эту категорию, которая добавляет метод setBackgroundColor: forState:
https://github.com/damienromito/UIButton-setBackgroundColor-forState-
Ответ 11
Мое лучшее решение для Swift 3 + без подкласса.
extension UIButton {
func setBackgroundColor(_ color: UIColor, for state: UIControlState) {
let rect = CGRect(x: 0, y: 0, width: 1, height: 1)
UIGraphicsBeginImageContext(rect.size)
color.setFill()
UIRectFill(rect)
let colorImage = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
setBackgroundImage(colorImage, for: state)
}
}
С помощью этого расширения легко управлять цветами для разных состояний, и он автоматически изменяет ваш нормальный цвет, если выделенный цвет не предоставляется.
button.setBackgroundColor(.red, for: .normal)
Ответ 12
UPDATE:
Используйте UIButtonBackgroundColor библиотеку Swift.
OLD:
С помощью помощников ниже создайте образ 1 px x 1 px с оттенком серого:
UIImage *image = ACUTilingImageGray(248/255.0, 1);
или цвет заливки RGB:
UIImage *image = ACUTilingImageRGB(253/255.0, 123/255.0, 43/255.0, 1);
Затем используйте image
, чтобы установить фоновое изображение кнопки:
[button setBackgroundImage:image forState:UIControlStateNormal];
Помощники
#pragma mark - Helpers
UIImage *ACUTilingImageGray(CGFloat gray, CGFloat alpha)
{
return ACUTilingImage(alpha, ^(CGContextRef context) {
CGContextSetGrayFillColor(context, gray, alpha);
});
}
UIImage *ACUTilingImageRGB(CGFloat red, CGFloat green, CGFloat blue, CGFloat alpha)
{
return ACUTilingImage(alpha, ^(CGContextRef context) {
CGContextSetRGBFillColor(context, red, green, blue, alpha);
});
}
UIImage *ACUTilingImage(CGFloat alpha, void (^setFillColor)(CGContextRef context))
{
CGRect rect = CGRectMake(0, 0, 0.5, 0.5);
UIGraphicsBeginImageContextWithOptions(rect.size, alpha == 1, 0);
CGContextRef context = UIGraphicsGetCurrentContext();
setFillColor(context);
CGContextFillRect(context, rect);
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
UIGraphicsEndImageContext();
return image;
}
Примечание: ACU
- это префикс класса моей Cocoa Touch Static Library под названием Acani Utilities, где AC для Acani, а U - для служебных программ.
Ответ 13
Попробуйте это!!!!
Для параметра TouchedDown Event установлен один цвет, а для TouchUpInside - другой.
- (IBAction)touchedDown:(id)sender {
NSLog(@"Touched Down");
btn1.backgroundColor=[UIColor redColor];
}
- (IBAction)touchUpInside:(id)sender {
NSLog(@"TouchUpInside");
btn1.backgroundColor=[UIColor whiteColor];
}
Ответ 14
Подкласс UIButton и добавьте проверяемые свойства для удобного использования (написанные в Swift 3.0):
final class SelectableBackgroundButton: UIButton {
private struct Constants {
static let animationDuration: NSTimeInterval = 0.1
}
@IBInspectable
var animatedColorChange: Bool = true
@IBInspectable
var selectedBgColor: UIColor = UIColor.blackColor().colorWithAlphaComponent(0.2)
@IBInspectable
var normalBgColor: UIColor = UIColor.clearColor()
override var selected: Bool {
didSet {
if animatedColorChange {
UIView.animateWithDuration(Constants.animationDuration) {
self.backgroundColor = self.selected ? self.selectedBgColor : self.normalBgColor
}
} else {
self.backgroundColor = selected ? selectedBgColor : normalBgColor
}
}
}
override var highlighted: Bool {
didSet {
if animatedColorChange {
UIView.animateWithDuration(Constants.animationDuration) {
self.backgroundColor = self.highlighted ? self.selectedBgColor : self.normalBgColor
}
} else {
self.backgroundColor = highlighted ? selectedBgColor : normalBgColor
}
}
}
}
Ответ 15
На раскадровке...
Вы можете изменить настройки для отдельных состояний UIButton. Например, выберите "Выбрано", и затем вы можете изменить цвет текста для этого состояния.
Чтобы изменить фон при выделении кнопки UIB:
- Выберите нужную кнопку в раскадровке.
- Выберите желаемое состояние конфигурации. Для вашего случая выберите "Выделено".
![How to change the state configuration in Storyboard.]()
- Изменить цвет фона. Теперь вы можете перейти к другим состояниям с помощью шага 2 и указать цвет фона для этого состояния.
![How to change the background color in Storyboard.]()
Ответ 16
Попробуйте, если у вас есть изображение:
-(void)setBackgroundImage:(UIImage *)image forState:(UIControlState)state;
или посмотрите, достаточно ли для вас showsTouchWhenHighlighted
.
Ответ 17
Вы можете подклассифицировать UIButton и сделать приятным forState.
colourButton.h
#import <UIKit/UIKit.h>
@interface colourButton : UIButton
-(void)setBackgroundColor:(UIColor *)backgroundColor forState:(UIControlState)state;
@end
colourButton.m
#import "colourButton.h"
@implementation colourButton
{
NSMutableDictionary *colours;
}
-(id)initWithCoder:(NSCoder *)aDecoder
{
self = [super initWithCoder:aDecoder];
// If colours does not exist
if(!colours)
{
colours = [NSMutableDictionary new]; // The dictionary is used to store the colour, the key is a text version of the ENUM
colours[[NSString stringWithFormat:@"%lu", UIControlStateNormal]] = (UIColor*)self.backgroundColor; // Store the original background colour
}
return self;
}
-(void)setBackgroundColor:(UIColor *)backgroundColor forState:(UIControlState)state
{
// If it is normal then set the standard background here
if(state & UIControlStateNormal)
{
[super setBackgroundColor:backgroundColor];
}
// Store the background colour for that state
colours[[NSString stringWithFormat:@"%lu", state]]= backgroundColor;
}
-(void)setHighlighted:(BOOL)highlighted
{
// Do original Highlight
[super setHighlighted:highlighted];
// Highlight with new colour OR replace with orignial
if (highlighted && colours[[NSString stringWithFormat:@"%lu", UIControlStateHighlighted]])
{
self.backgroundColor = colours[[NSString stringWithFormat:@"%lu", UIControlStateHighlighted]];
}
else
{
self.backgroundColor = colours[[NSString stringWithFormat:@"%lu", UIControlStateNormal]];
}
}
-(void)setSelected:(BOOL)selected
{
// Do original Selected
[super setSelected:selected];
// Select with new colour OR replace with orignial
if (selected && colours[[NSString stringWithFormat:@"%lu", UIControlStateSelected]])
{
self.backgroundColor = colours[[NSString stringWithFormat:@"%lu", UIControlStateSelected]];
}
else
{
self.backgroundColor = colours[[NSString stringWithFormat:@"%lu", UIControlStateNormal]];
}
}
@end
Примечания (Это пример, я знаю, что есть проблемы, и вот некоторые)
Я использовал NSMutableDictionay для хранения UIColor для каждого штата, я должен сделать неприятное преобразование текста для ключа, поскольку UIControlState не является хорошим прямым Int. Если вы можете запустить массив с таким количеством объектов и использовать его как индекс.
Из-за этого у вас много трудностей, например. выбранная и отключенная кнопка, требуется еще одна логика.
Еще одна проблема заключается в том, что если вы одновременно пытаетесь установить несколько цветов, я не пробовал с помощью кнопки, но если вы можете это сделать, это может не работать.
[btn setBackgroundColor:colour forState:UIControlStateSelected & UIControlStateHighlighted];
Я предположил, что это StoryBoard, нет init, initWithFrame, поэтому добавьте их, если они вам понадобятся.
Ответ 18
У меня есть открытый подкласс UIButton, STAButton, чтобы заполнить это отверстие для зияющей функциональности. Доступно по лицензии MIT. Работает для iOS 7+ (я не тестировал старые версии iOS).
Ответ 19
Чтобы решить эту проблему, я создал категорию для обработки backgroundColor
состояний с помощью UIButtons
:
ButtonBackgroundColor-iOS
Вы можете установить категорию как pod.
Прост в использовании с Objective-C
@property (nonatomic, strong) UIButton *myButton;
...
[self.myButton bbc_backgroundColorNormal:[UIColor redColor]
backgroundColorSelected:[UIColor blueColor]];
Еще проще в использовании с Swift:
import ButtonBackgroundColor
...
let myButton:UIButton = UIButton(type:.Custom)
myButton.bbc_backgroundColorNormal(UIColor.redColor(), backgroundColorSelected: UIColor.blueColor())
Я рекомендую вам импортировать модуль с помощью:
platform :ios, '8.0'
use_frameworks!
pod 'ButtonBackgroundColor', '~> 1.0'
Использование use_frameworks! в вашем подфайле проще использовать ваши стручки со Swift и objective-C.
ВАЖНО
Я также написал сообщение в блоге с дополнительной информацией.
Ответ 20
class CustomButton: UIButton {
override var isHighlighted: Bool {
didSet {
if (isHighlighted) {
alpha = 0.5
}
else {
alpha = 1
}
}
}
}
Ответ 21
Используйте https://github.com/swordray/UIButtonSetBackgroundColorForState
Добавить в Podfile, используя CocoaPods
pod "UIButtonSetBackgroundColorForState"
стриж
button.setBackgroundColor(.red, forState: .highlighted)
Objective-C
[button setBackgroundColor:[UIColor redColor] forState:UIControlStateHighlighted];
Ответ 22
Попробуйте tintColor
:
_button.tintColor = [UIColor redColor];
Ответ 23
Вот код в Swift для выбора состояния кнопки:
func imageWithColor(color:UIColor) -> UIImage {
let rect:CGRect = CGRectMake(0.0, 0.0, 1.0, 1.0)
UIGraphicsBeginImageContext(rect.size)
let context:CGContextRef = UIGraphicsGetCurrentContext()!
CGContextSetFillColorWithColor(context, color.CGColor)
CGContextFillRect(context, rect)
let image:UIImage = UIGraphicsGetImageFromCurrentImageContext();
return image;
}
Пример:
self.button.setImage(self.imageWithColor(UIColor.blackColor()), forState: .Highlighted)
Ответ 24
Бросьте его, и вы хорошо пойдете:
* proerty может быть установлен в IB, и если выделенный фон не установлен, фон не будет изменяться при нажатии
private var highlightedBackgroundColors = [UIButton:UIColor]()
private var unhighlightedBackgroundColors = [UIButton:UIColor]()
extension UIButton {
@IBInspectable var highlightedBackgroundColor: UIColor? {
get {
return highlightedBackgroundColors[self]
}
set {
highlightedBackgroundColors[self] = newValue
}
}
override open var backgroundColor: UIColor? {
get {
return super.backgroundColor
}
set {
unhighlightedBackgroundColors[self] = newValue
super.backgroundColor = newValue
}
}
override open var isHighlighted: Bool {
get {
return super.isHighlighted
}
set {
if highlightedBackgroundColor != nil {
super.backgroundColor = newValue ? highlightedBackgroundColor : unhighlightedBackgroundColors[self]
}
super.isHighlighted = newValue
}
}
}
Ответ 25
Ниже расширение UIIImage
генерирует изображение объекта с указанным параметром цвета.
extension UIImage {
static func imageWithColor(tintColor: UIColor) -> UIImage {
let rect = CGRect(x: 0, y: 0, width: 1, height: 1)
UIGraphicsBeginImageContextWithOptions(rect.size, false, 0)
tintColor.setFill()
UIRectFill(rect)
let image: UIImage = UIGraphicsGetImageFromCurrentImageContext()!
UIGraphicsEndImageContext()
return image
}
}
Пример использования кнопки может быть применен для объекта кнопки как:
setupButton.setBackgroundImage(UIImage.imageWithColor(tintColor: UIColor(displayP3Red: 232/255, green: 130/255, blue: 121/255, alpha: 1.0)), for: UIControlState.highlighted)
setupButton.setBackgroundImage(UIImage.imageWithColor(tintColor: UIColor(displayP3Red: 255/255, green: 194/255, blue: 190/255, alpha: 1.0)), for: UIControlState.normal)
Ответ 26
Подробнее
- Xcode 11.1 (11A1027), Swift 5
Решение
import UIKit
extension UIColor {
func createOnePixelImage() -> UIImage? {
let size = CGSize(width: 1, height: 1)
UIGraphicsBeginImageContext(size)
defer { UIGraphicsEndImageContext() }
guard let context = UIGraphicsGetCurrentContext() else { return nil }
context.setFillColor(cgColor)
context.fill(CGRect(origin: .zero, size: size))
return UIGraphicsGetImageFromCurrentImageContext()
}
}
extension UIButton {
func setBackground(_ color: UIColor, for state: UIControl.State) {
setBackgroundImage(color.createOnePixelImage(), for: state)
}
}
Usage
Usage
button.setBackground(.green, for: .normal)
Ответ 27
если вы не будете переопределять
просто установите два действия
посадка
touchUpInside
Ответ 28
Swift 3:
extension UIButton {
private func imageWithColor(color: UIColor) -> UIImage {
let rect = CGRect(x:0.0,y:0.0,width: 1.0,height: 1.0)
UIGraphicsBeginImageContext(rect.size)
let context = UIGraphicsGetCurrentContext()
context!.setFillColor(color.cgColor)
context!.fill(rect)
let image = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
return image!
}
func setBackgroundColor(color: UIColor, forUIControlState state: UIControlState) {
self.setBackgroundImage(imageWithColor(color: color), for: state)
}
}
Ответ 29
Своя работая отлично с быстрым 3 для того чтобы изменить цвет кнопки и цвет названия кнопки на щелчке:
@IBAction func btnColorChange(_ sender: UIButton) {
sender.backgroundColor = .red
sender.setTitleColor(.white, for: .normal)
}
Ответ 30
extension UIButton {
func setBackgroundColor(color: UIColor, forState: UIControl.State) {
let size = CGSize(width: 1, height: 1)
UIGraphicsBeginImageContext(size)
let context = UIGraphicsGetCurrentContext()
context?.setFillColor(color.cgColor)
context?.fill(CGRect(origin: CGPoint.zero, size: size))
let colorImage = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()
setBackgroundImage(colorImage, for: forState)
}
}
Swift 5, спасибо @Maverick