Swift: Как изменить фон видов из других видов - PullRequest
1 голос
/ 28 марта 2019

Моя задача - создать приложение, в котором я могу изменить и сохранить фон.Я хотел сделать что-то более сложное, поэтому я решил создать цветное меню, которое позволяет мне изменять фоны всех видов.Теперь у меня есть меню, и я знаю, как установить и сохранить изображения / цвета в фоновом режиме.Но я не знаю, как сделать это от контроллера представления к другому.Прямо сейчас мой код меняет фон своего собственного представления.Это мой код (я знаю, что это много ...):

Прежде всего, у меня есть 2 расширения, которые позволяют мне 1) сохранить UIColor с пользовательскими настройками по умолчанию;2) использовать цвета RGB:

//MARK: - Extension for rgb
extension UIColor {
    convenience init(red: Int, green: Int, blue: Int) {
        assert(red >= 0 && red <= 255, "Invalid red component")
        assert(green >= 0 && green <= 255, "Invalid green component")
        assert(blue >= 0 && blue <= 255, "Invalid blue component")

        self.init(red: CGFloat(red) / 255.0, green: CGFloat(green) / 255.0, blue: CGFloat(blue) / 255.0, alpha: 1.0)
    }

    convenience init(rgb: Int) {
        self.init(
            red: (rgb >> 16) & 0xFF,
            green: (rgb >> 8) & 0xFF,
            blue: rgb & 0xFF
        )
    }
}


//MARK: - Extension to save colors
extension UserDefaults {
    func setColor(color: UIColor?, forKey key: String) {
        var colorData: NSData?
        if let color = color {
            do {
                colorData = try NSKeyedArchiver.archivedData(withRootObject: color, requiringSecureCoding: false) as NSData?
                set(colorData, forKey: key)
            } catch let error {
                print("error archiving color data", error)
            }
        }
    }

    func colorForKey(key: String) -> UIColor? {
        var color: UIColor?
        if let colorData = data(forKey: "Background") {
            do{
                color = try NSKeyedUnarchiver.unarchivedObject(ofClass: UIColor.self, from: colorData)
            } catch let error {
                print("error unarchivig color data", error)
            }
        }
        return color
    }

}

My viewDidLoad:

var defaults = UserDefaults.standard
private let backgroundKey = "backg"
 override func viewDidLoad() {
        super.viewDidLoad()


        //MARK: - Background Settings


        let color = defaults.colorForKey(key: backgroundKey)
        view.backgroundColor = color

        if let backgroundImage = self.getSavedBackgroundImage() {
            self.backgroundImageView.image = backgroundImage
        }
}

Код, позволяющий установить изображение в качестве фона

@IBOutlet weak var backgroundImageView: UIImageView!
//I KNOW THIS HAS TO BE PUT IN THE VC WHERE I WANT THE BACKGROUND TO BE CHANGED
 @IBAction func pictureButton(_ sender: Any) {
        let imagePicker = UIImagePickerController()
        imagePicker.delegate = self
        imagePicker.sourceType = .photoLibrary
        imagePicker.allowsEditing = false
        present(imagePicker, animated: true, completion: nil)
    }
    func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey : Any]) {
        if let image = info[UIImagePickerController.InfoKey.originalImage] as? UIImage {
            backgroundImageView.image = image
            let _ = self.saveBackgroundImage(image: image)

        }

        self.dismiss(animated: true, completion: nil)

    }
    func saveBackgroundImage(image: UIImage) -> Bool {
        guard let data = image.pngData() ?? backgroundImageView.image?.pngData() else {
            return false
        }
        guard let directory = FileManager.default.urls(for: .documentDirectory,
                                                       in: .userDomainMask).first else {
                                                        return false
        }
        do {
            try data.write(to: directory.appendingPathComponent("MainBackgroound.png"))
            return true
        } catch {
            print(error.localizedDescription)
            return false
        }
    }

    func getSavedBackgroundImage() -> UIImage? {
        if let dir = FileManager.default.urls(for: .documentDirectory,
                                              in: .userDomainMask).first {
            return UIImage(contentsOfFile: URL(fileURLWithPath: dir.absoluteString).appendingPathComponent("MainBackgroound.png").path)
        }
        return nil
    }

    func deleteImage() {
        let fileManager = FileManager.default
        let yourProjectImagesPath = (NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true)[0] as NSString).appendingPathComponent("MainBackgroound.png")
        if fileManager.fileExists(atPath: yourProjectImagesPath) {
            try! fileManager.removeItem(atPath: yourProjectImagesPath)
        }
        let yourProjectDirectoryPath = (NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true)[0] as NSString).appendingPathComponent("MainBackgroound.png")
        if fileManager.fileExists(atPath: yourProjectDirectoryPath) {
            try! fileManager.removeItem(atPath: yourProjectDirectoryPath)
        }
    }

Код, позволяющий изменить цвет фона

@IBAction func lightPinkButton(_ sender: Any) {
        self.deleteImage()
        let lightPink = UIColor(red: 255, green: 215, blue: 214).withAlphaComponent(1)
        defaults.setColor(color: lightPink, forKey: backgroundKey)
        view.backgroundColor = lightPink
    }

Ответы [ 2 ]

1 голос
/ 28 марта 2019

Как насчет того, чтобы создать базовый контроллер вида, что-то вроде ColorViewController, и там у вас есть ссылка на цвет фона, а затем просто наследовать его и обновлять его там, где вам нужно.

Как это

ColorViewController

class ColorViewController: UIViewController {

    var myBackgroundColor: UIColor = .white

    override func viewDidLoad() {
        super.viewDidLoad()
    }

}

ViewControllerOne

class ViewControllerOne: ColorViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)

        view.backgroundColor = myBackgroundColor
    }

    @IBAction func blueButtonTapped(_ sender: Any) {
        myBackgroundColor = .blue
        view.backgroundColor = myBackgroundColor
    }

}

ViewControllerTwo

class ViewControllerTwo: ColorViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)

        view.backgroundColor = myBackgroundColor
    }

    @IBAction func purpleButton(_ sender: Any) {
        myBackgroundColor = .purple
        view.backgroundColor = myBackgroundColor
    }

}
0 голосов
/ 28 марта 2019

Вы должны сделать Notification для того, когда цвет вашего фона изменяется и когда ваше фоновое изображение изменяется.Любой ViewController, который должен быть тематическим, должен прослушивать эти уведомления и обновляться соответствующим образом:

//Make your custom notification
extension Notification.Name {
    static let backgroundColorUpdated = Notification.Name("backgroundColorUpdated")
}

//Post the notification when you update backgroundColor
NotificationCenter.default.post(Notification(name: .backgroundColorUpdated, object: nil, userInfo: ["backgroundColor": UIColor.red]))

//Subscribe everywhere that needs to be interested in backgroundColor changes
var token: NSObjectProtocol? // retain token for life of subscription, ie as long as you vc exists
...
token = NotificationCenter.default.addObserver(forName: .backgroundColorUpdated, object: nil, queue: .main) { notification in
    guard let backgroundColor = notification.userInfo?["backgroundColor"] as? UIColor else {
        return
    }
    //Do stuff with backgroundColors here
}

...