2019-12-19 17:32:49 +00:00
|
|
|
/*
|
2023-04-06 09:44:16 +00:00
|
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
2019-12-19 17:32:49 +00:00
|
|
|
*
|
|
|
|
|
* This source code is licensed under the MIT license found in the
|
|
|
|
|
* LICENSE file in the root directory of this source tree.
|
2016-09-07 22:37:59 +00:00
|
|
|
*/
|
|
|
|
|
|
2017-03-15 14:44:52 +00:00
|
|
|
import IGListKit
|
2018-01-22 20:01:53 +00:00
|
|
|
import UIKit
|
2016-09-07 22:37:59 +00:00
|
|
|
|
2017-03-12 20:12:23 +00:00
|
|
|
final class LabelCell: UICollectionViewCell {
|
2023-07-25 16:17:11 +00:00
|
|
|
enum Style {
|
|
|
|
|
case `default`
|
|
|
|
|
case grouped
|
|
|
|
|
}
|
2016-09-07 22:37:59 +00:00
|
|
|
|
2023-07-25 16:17:11 +00:00
|
|
|
fileprivate static let insets = UIEdgeInsets(top: 8, left: 15, bottom: 8, right: 15)
|
2023-07-20 05:08:21 +00:00
|
|
|
fileprivate static let font = UIFont.systemFont(ofSize: 18)
|
|
|
|
|
fileprivate static let symbolFont = UIFont.boldSystemFont(ofSize: 21)
|
2023-07-25 16:17:11 +00:00
|
|
|
fileprivate static let cornerRadius = 12.0
|
2016-09-07 22:37:59 +00:00
|
|
|
|
|
|
|
|
static var singleLineHeight: CGFloat {
|
|
|
|
|
return font.lineHeight + insets.top + insets.bottom
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static func textHeight(_ text: String, width: CGFloat) -> CGFloat {
|
|
|
|
|
let constrainedSize = CGSize(width: width - insets.left - insets.right, height: CGFloat.greatestFiniteMagnitude)
|
2019-12-19 05:50:02 +00:00
|
|
|
let attributes = [ NSAttributedString.Key.font: font ]
|
2016-09-07 22:37:59 +00:00
|
|
|
let options: NSStringDrawingOptions = [.usesFontLeading, .usesLineFragmentOrigin]
|
|
|
|
|
let bounds = (text as NSString).boundingRect(with: constrainedSize, options: options, attributes: attributes, context: nil)
|
|
|
|
|
return ceil(bounds.height) + insets.top + insets.bottom
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-11 20:20:56 +00:00
|
|
|
fileprivate let label: UILabel = {
|
2016-09-07 22:37:59 +00:00
|
|
|
let label = UILabel()
|
2016-10-24 18:06:17 +00:00
|
|
|
label.backgroundColor = .clear
|
2017-02-21 23:27:26 +00:00
|
|
|
label.numberOfLines = 0
|
2016-09-07 22:37:59 +00:00
|
|
|
label.font = LabelCell.font
|
|
|
|
|
return label
|
|
|
|
|
}()
|
|
|
|
|
|
2016-12-22 22:19:45 +00:00
|
|
|
let separator: CALayer = {
|
2016-09-07 22:37:59 +00:00
|
|
|
let layer = CALayer()
|
2020-10-03 08:37:52 +00:00
|
|
|
layer.backgroundColor = UIColor.defaultSeparator.cgColor
|
2016-09-07 22:37:59 +00:00
|
|
|
return layer
|
|
|
|
|
}()
|
|
|
|
|
|
2023-07-20 05:08:21 +00:00
|
|
|
let imageView: UIImageView = {
|
|
|
|
|
let imageView = UIImageView()
|
|
|
|
|
imageView.tintColor = .titleLabel
|
|
|
|
|
imageView.contentMode = .scaleAspectFit
|
|
|
|
|
return imageView
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
let disclosureImageView: UIImageView = {
|
|
|
|
|
let configuration = UIImage.SymbolConfiguration(pointSize: 17, weight: .bold, scale: .small)
|
|
|
|
|
let chevronImage = UIImage(systemName: "chevron.right", withConfiguration: configuration)
|
|
|
|
|
let imageView = UIImageView(image: chevronImage)
|
|
|
|
|
imageView.tintColor = .defaultSeparator
|
2023-07-25 16:17:11 +00:00
|
|
|
imageView.isHidden = true
|
2023-07-20 05:08:21 +00:00
|
|
|
return imageView
|
|
|
|
|
}()
|
|
|
|
|
|
2023-07-25 16:17:11 +00:00
|
|
|
let backdropView: UIView = {
|
|
|
|
|
let view = UIView()
|
|
|
|
|
view.layer.cornerCurve = .continuous
|
|
|
|
|
return view
|
|
|
|
|
}()
|
|
|
|
|
|
2017-04-11 20:20:56 +00:00
|
|
|
var text: String? {
|
|
|
|
|
get {
|
|
|
|
|
return label.text
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
label.text = newValue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-20 05:08:21 +00:00
|
|
|
var imageName: String? {
|
|
|
|
|
didSet {
|
|
|
|
|
guard let imageName else {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
let configuration = UIImage.SymbolConfiguration(font: LabelCell.symbolFont, scale: .default)
|
|
|
|
|
let image = UIImage(systemName: imageName, withConfiguration: configuration)
|
|
|
|
|
imageView.image = image
|
|
|
|
|
setNeedsLayout()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-25 16:17:11 +00:00
|
|
|
var style: Style = .default {
|
|
|
|
|
didSet {
|
|
|
|
|
backdropView.backgroundColor = backdropColor
|
|
|
|
|
setHighlighted(isSelected)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var isTopCell = false
|
|
|
|
|
var isBottomCell = false
|
|
|
|
|
|
|
|
|
|
var backdropColor: UIColor {
|
|
|
|
|
(style == .grouped) ? .secondaryGroupedBackground : .background
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-22 22:19:45 +00:00
|
|
|
override init(frame: CGRect) {
|
|
|
|
|
super.init(frame: frame)
|
2023-07-25 16:17:11 +00:00
|
|
|
contentView.backgroundColor = .clear
|
|
|
|
|
contentView.addSubview(backdropView)
|
2016-12-22 22:19:45 +00:00
|
|
|
contentView.addSubview(label)
|
|
|
|
|
contentView.layer.addSublayer(separator)
|
2023-07-20 05:08:21 +00:00
|
|
|
contentView.addSubview(disclosureImageView)
|
|
|
|
|
contentView.addSubview(imageView)
|
2016-12-22 22:19:45 +00:00
|
|
|
}
|
2017-05-16 14:30:08 +00:00
|
|
|
|
2016-12-22 22:19:45 +00:00
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-07 22:37:59 +00:00
|
|
|
override func layoutSubviews() {
|
|
|
|
|
super.layoutSubviews()
|
|
|
|
|
let bounds = contentView.bounds
|
2023-07-20 05:08:21 +00:00
|
|
|
|
2023-07-25 16:17:11 +00:00
|
|
|
backdropView.frame = contentView.bounds
|
|
|
|
|
|
2023-07-20 05:08:21 +00:00
|
|
|
let hasImage = imageName?.count ?? 0 > 0
|
|
|
|
|
let imageCenterX = (hasImage ? LabelCell.insets.left + 15 : 0) + safeAreaInsets.left
|
|
|
|
|
if hasImage {
|
|
|
|
|
imageView.sizeToFit()
|
|
|
|
|
imageView.frame.origin = CGPoint(x: imageCenterX - imageView.bounds.midX,
|
|
|
|
|
y: bounds.midY - imageView.bounds.midY)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
disclosureImageView.frame.origin = CGPoint(x: bounds.width - (LabelCell.insets.right + disclosureImageView.bounds.width + safeAreaInsets.right),
|
|
|
|
|
y: bounds.midY - disclosureImageView.bounds.midY)
|
|
|
|
|
|
|
|
|
|
let labelX = hasImage ? imageCenterX + 25 : (LabelCell.insets.left + safeAreaInsets.left)
|
|
|
|
|
label.frame = CGRect(x: labelX,
|
|
|
|
|
y: LabelCell.insets.top,
|
|
|
|
|
width: bounds.width - (labelX - disclosureImageView.frame.minX),
|
|
|
|
|
height: bounds.height - (LabelCell.insets.top + LabelCell.insets.bottom))
|
|
|
|
|
|
|
|
|
|
let separatorHeight: CGFloat = 1.0 / (window?.screen.nativeScale ?? 2.0)
|
|
|
|
|
let left = label.frame.minX
|
|
|
|
|
separator.frame = CGRect(x: left, y: bounds.height - separatorHeight, width: bounds.width - left, height: separatorHeight)
|
2023-07-25 16:17:11 +00:00
|
|
|
|
|
|
|
|
if style != .grouped {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if isBottomCell {
|
|
|
|
|
separator.isHidden = true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let layer = backdropView.layer
|
|
|
|
|
layer.cornerRadius = LabelCell.cornerRadius
|
|
|
|
|
|
|
|
|
|
if isTopCell {
|
|
|
|
|
layer.maskedCorners = [.layerMinXMinYCorner, .layerMaxXMinYCorner]
|
|
|
|
|
} else if isBottomCell {
|
|
|
|
|
layer.maskedCorners = [.layerMinXMaxYCorner, .layerMaxXMaxYCorner]
|
|
|
|
|
} else {
|
|
|
|
|
layer.maskedCorners = []
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-07 22:37:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
override var isHighlighted: Bool {
|
|
|
|
|
didSet {
|
2023-07-20 05:08:21 +00:00
|
|
|
setHighlighted(isHighlighted)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
override var isSelected: Bool {
|
|
|
|
|
didSet {
|
|
|
|
|
setHighlighted(isSelected)
|
2016-09-07 22:37:59 +00:00
|
|
|
}
|
|
|
|
|
}
|
2017-05-16 14:30:08 +00:00
|
|
|
|
2020-10-03 08:37:52 +00:00
|
|
|
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
|
|
|
|
super.traitCollectionDidChange(previousTraitCollection)
|
|
|
|
|
separator.backgroundColor = UIColor.defaultSeparator.cgColor
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-20 05:08:21 +00:00
|
|
|
private func setHighlighted(_ highlighted: Bool) {
|
2023-07-25 16:17:11 +00:00
|
|
|
let color = highlighted ? UIColor.gray.withAlphaComponent(0.3) : backdropColor
|
|
|
|
|
backdropView.backgroundColor = color
|
2023-07-20 05:08:21 +00:00
|
|
|
}
|
2016-09-07 22:37:59 +00:00
|
|
|
}
|
2017-03-15 14:44:52 +00:00
|
|
|
|
Swift name annotations
Summary:
This adds `NS_SWIFT_NAME` annotations to all public API's to provide cleaner integration into Swift:
- Removes the need to prefix classes in Swift code, instead rely on Swift module name spacing
- Adds more argument labels to C function API's like `IGListDiff([], [], .equality)` => `ListDiff(oldArray: [], newArray: [], option: .equality)`
While this is a large API change it should be as easy as:
- Find and replace `(IGList)([^K])` to `List$2` in Xcode with a scope set to Swift
- Build and follow compiler's auto fix corrections for C API's or any missed renames
I have not updated the documentation to reflect this yet, I am totally willing to do so but before I sink that amount of time into it I wanted to see if the Instagram team is even open to this change!
- [x] All tests pass. Demo project builds and runs.
- [x] I added tests, an experiment, or detailed why my change isn't tested.
- [x] I added an entry to the `CHANGELOG.md` for any breaking changes, enhancements, or bug fixes.
- [x] I have reviewed the [contributing guide](https://github.com/Instagram/IGListKit/blob/master/.github/CONTRIBUTING.md)
- [ ] I have updated the documentation
Closes https://github.com/Instagram/IGListKit/pull/593
Reviewed By: jessesquires
Differential Revision: D5028039
Pulled By: rnystrom
fbshipit-source-id: b473d874a1f9574e56b2ebaabd5b73d1b57d4bab
2017-05-09 21:29:52 +00:00
|
|
|
extension LabelCell: ListBindable {
|
2017-05-16 14:30:08 +00:00
|
|
|
|
2017-03-15 14:44:52 +00:00
|
|
|
func bindViewModel(_ viewModel: Any) {
|
|
|
|
|
guard let viewModel = viewModel as? String else { return }
|
|
|
|
|
label.text = viewModel
|
|
|
|
|
}
|
2017-05-16 14:30:08 +00:00
|
|
|
|
2017-03-15 14:44:52 +00:00
|
|
|
}
|