jsdw_ios/QuickLocation/Section/Schedule/ScheduleDetail/ScheduleDetailView.swift

576 lines
17 KiB
Swift

//
// ScheduleDetailView.swift
// QuickLocation
//
// Created by on 2026/6/25.
//
import UIKit
import RxSwift
import RxCocoa
class ScheduleDetailView: UIView {
var disposeBag = DisposeBag()
private func setupRx() {
backBtn.rx.tap.subscribe(onNext: { _ in
AppRouter.shared.popOrDismiss()
}).disposed(by: disposeBag)
}
private func setupUI() {
addSubview(navBgView)
addSubview(navBarView)
navBarView.addSubview(navTitleLabel)
navBarView.addSubview(backBtn)
addSubview(headerView)
addSubview(travelRouteView)
addSubview(tableView)
addSubview(bottomView)
navBgView.layoutChain
.edges(excludingEdge: .bottom)
.heightToWidth(160/375)
navBarView.layoutChain
.edges(excludingEdge: .bottom)
.height(kNaviHeight)
navTitleLabel.layoutChain
.top(kStatusBarHeight + 12)
.centerY(backBtn)
.centerX()
backBtn.layoutChain
.centerY(navTitleLabel)
.left(15)
.width(24)
.height(24)
headerView.layoutChain
.topToBottomOfView(navBarView)
.edgesHorzontal()
travelRouteView.layoutChain
.topToBottomOfView(headerView)
.edgesHorzontal()
bottomView.layoutChain
.edgesHorzontal()
.height(kSafeBottomMargin + 80)
.bottom()
tableView.layoutChain
.topToBottomOfView(travelRouteView, offset: 15)
.edgesHorzontal()
.bottomToTopOfView(bottomView, offset: 0)
}
lazy var navBgView: UIImageView = {
let iv = UIImageView()
iv.image = UIImage(named: "Common/navBar_bg_2")
iv.contentMode = .scaleAspectFill
return iv
}()
lazy var navBarView: UIView = {
let view = UIView()
view.backgroundColor = .clear
return view
}()
lazy var navTitleLabel: UILabel = {
let label = UILabel()
label.text = "行程详情"
label.font = .systemFont(ofSize: 18, weight: .medium)
label.textColor = ThemeManager.shared.color.titleAuxColor
label.textAlignment = .center
return label
}()
lazy var backBtn: UIButton = {
let btn = UIButton(type: .custom)
btn.setImage(UIImage(named: "Common/back"), for: .normal)
btn.extendEdgeInsets = UIEdgeInsets(top: 54, left: 15, bottom: 100, right: 100)
return btn
}()
///
lazy var headerView: UIView = {
let view = UIView()
view.backgroundColor = .clear
let titleLab = UILabel()
titleLab.text = "谁在关注"
titleLab.font = .systemFont(ofSize: 16, weight: .medium)
titleLab.textColor = ThemeManager.shared.color.titleAuxColor
view.addSubview(titleLab)
titleLab.layoutChain
.top(15)
let dotView = UIView()
dotView.backgroundColor = UIColor(hexStr: "#16B3FF")
dotView.cornerRadius = 2
view.addSubview(dotView)
dotView.layoutChain
.left(15)
.centerY(titleLab)
.width(4)
.height(11)
titleLab.layoutChain.leftToRightOfView(dotView, offset: 5)
view.addSubview(collectionView)
collectionView.layoutChain
.topToBottomOfView(titleLab, offset: 15)
.edgesHorzontal()
.height(80)
.bottom(15)
view.addSubview(noDataLab)
noDataLab.layoutChain
.centerX().centerY()
return view
}()
lazy var collectionView: UICollectionView = {
let layout = UICollectionViewFlowLayout()
layout.scrollDirection = .horizontal
layout.itemSize = CGSize(width: 80, height: 80)
layout.minimumLineSpacing = 15
layout.sectionInset = UIEdgeInsets(top: 0, left: 15, bottom: 0, right: 15)
let cv = UICollectionView(frame: .zero, collectionViewLayout: layout)
cv.backgroundColor = .clear
cv.showsHorizontalScrollIndicator = false
cv.register(ViewedCell.self)
return cv
}()
lazy var noDataLab: UILabel = {
let label = UILabel()
label.text = " 暂无关注"
label.textColor = UIColor(hexStr: "#999999")
label.font = .systemFont(ofSize: 14, weight: .regular)
label.isHidden = true
return label
}()
///
lazy var travelRouteView: UIView = {
let view = UIView()
view.backgroundColor = .clear
let titleLab = UILabel()
titleLab.text = "行程日期"
titleLab.font = .systemFont(ofSize: 16, weight: .medium)
titleLab.textColor = ThemeManager.shared.color.titleAuxColor
view.addSubview(titleLab)
titleLab.layoutChain
.top(5)
let dotView = UIView()
dotView.backgroundColor = UIColor(hexStr: "#16B3FF")
dotView.cornerRadius = 2
view.addSubview(dotView)
dotView.layoutChain
.left(15)
.centerY(titleLab)
.width(4)
.height(11)
titleLab.layoutChain.leftToRightOfView(dotView, offset: 5)
view.addSubview(dateLab)
dateLab.layoutChain
.centerY(titleLab)
.leftToRightOfView(titleLab, offset: 10)
view.addSubview(creatorIcon)
creatorIcon.layoutChain
.right(15)
.width(30)
.height(30)
.centerY(titleLab)
let creatorTitleLab = UILabel()
creatorTitleLab.text = "创建人"
creatorTitleLab.font = .systemFont(ofSize: 12, weight: .medium)
creatorTitleLab.textColor = ThemeManager.shared.color.titleAuxColor
view.addSubview(creatorTitleLab)
creatorTitleLab.layoutChain
.rightToLeftOfView(creatorIcon, offset: -5)
.centerY(titleLab)
view.addSubview(vipTipsLab)
vipTipsLab.layoutChain
.topToBottomOfView(titleLab, offset: 5)
.leftToView(titleLab)
.bottom()
return view
}()
lazy var dateLab: UILabel = {
let label = UILabel()
label.textColor = UIColor(hexStr: "#16B3FF")
label.font = .systemFont(ofSize: 14, weight: .medium)
return label
}()
lazy var creatorIcon: UIImageView = {
let view = UIImageView()
view.cornerRadius = 15
return view
}()
lazy var vipTipsLab: UILabel = {
let label = UILabel()
label.textColor = UIColor(hexStr: "#FF7D52")
label.font = .systemFont(ofSize: 10, weight: .regular)
return label
}()
lazy var tableView: UITableView = {
let tv = UITableView(frame: .zero, style: .plain)
tv.backgroundColor = .clear
tv.separatorStyle = .none
tv.estimatedRowHeight = 77
tv.showsVerticalScrollIndicator = false
tv.bounces = false
tv.register(SchedulePointDetailCell.self)
tv.register(SchedulePointEventCell.self)
tv.contentInset = UIEdgeInsets(top: 0, left: 0, bottom: 15, right: 0)
return tv
}()
lazy var bottomView: UIView = {
let v = UIView()
v.backgroundColor = .white
v.layer.cornerRadius = 16
v.layer.maskedCorners = [.layerMinXMinYCorner, .layerMaxXMinYCorner]
v.layer.shadowColor = UIColor.black.withAlphaComponent(0.1).cgColor
v.layer.shadowOffset = CGSize(width: 0, height: -2)
v.layer.shadowRadius = 10
v.layer.shadowOpacity = 1
forwardBtn.isHidden = true
v.addSubview(forwardBtn)
v.addSubview(operateBtn)
v.addSubview(routeBtn)
forwardBtn.layoutChain
.centerY()
.left(15)
.width(80)
routeBtn.layoutChain
.centerY()
.right(15)
.width(80)
operateBtn.layoutChain
.centerY()
.leftToRightOfView(forwardBtn, offset: 20)
.rightToLeftOfView(routeBtn, offset: -20)
.height(50)
return v
}()
lazy var forwardBtn: UIButton = makeVerticalButton(image: UIImage(named: "Schedule/forward"), title: "发送到圈子")
lazy var routeBtn: UIButton = makeVerticalButton(image: UIImage(named: "Schedule/route"), title: "查看路线")
///
private func makeVerticalButton(image: UIImage?, title: String) -> UIButton {
let btn = UIButton(type: .custom)
btn.setImage(image, for: .normal)
btn.setTitle(title, for: .normal)
btn.setTitleColor(UIColor(hexStr: "#333333"), for: .normal)
btn.titleLabel?.font = .systemFont(ofSize: 12, weight: .medium)
btn.titleLabel?.textAlignment = .center
btn.imageView?.contentMode = .scaleAspectFit
//
let width = btn.titleLabel?.intrinsicContentSize.width ?? 0
btn.imageEdgeInsets = UIEdgeInsets(top: -20, left: 0, bottom: 0, right: -width)
btn.titleEdgeInsets = UIEdgeInsets(top: 0, left: -24, bottom: -24, right: 0)
btn.contentEdgeInsets = UIEdgeInsets(top: 24, left: 0, bottom: 24, right: 0)
return btn
}
lazy var operateBtn: UIButton = {
let btn = UIButton()
btn.setTitle("取消关注", for: .selected)
btn.setTitleColor(.white, for: .normal)
btn.titleLabel?.font = .systemFont(ofSize: 16, weight: .medium)
btn.setBackgroundImage(UIImage(named: "Common/button_bg_2"), for: .normal)
btn.cornerRadius = 25
return btn
}()
override init(frame: CGRect) {
super.init(frame: .zero)
backgroundColor = .white
setupUI()
setupRx()
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
// MARK: - SchedulePointDetailCell
class SchedulePointDetailCell: UITableViewCell {
var disposeBag = DisposeBag()
func configure(model: SchedulePointModel, index: Int, total: Int) {
indexLabel.text = "\(index + 1)"
locationLabel.text = model.street
remarkLab.text = model.remark.isEmpty ? "备注:无备注" : model.remark
timeLabel.text = getDateInterval2String(date: "\(model.expected_timestamp / 1000)", dateFormat: "HH:mm")
var indexName = ""
if index == 0 {
indexName = "起点:"
}
else if index == total - 1 {
indexName = "终点:"
}
else {
indexName = "途经点:"
}
indexNameLab.text = indexName
bottomDashView.isHidden = index == total - 1
}
// MARK: - Init
override init(style: CellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)
selectionStyle = .none
backgroundColor = .clear
setupViews()
}
required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") }
override func prepareForReuse() {
super.prepareForReuse()
disposeBag = DisposeBag()
}
private func setupViews() {
contentView.addSubview(pointIcon)
contentView.addSubview(bottomDashView)
contentView.addSubview(indexLabel)
contentView.addSubview(indexNameLab)
contentView.addSubview(locationLabel)
contentView.addSubview(timeLabel)
contentView.addSubview(remarkLab)
pointIcon.layoutChain
.top()
.left(15)
.width(18).height(18)
indexLabel.layoutChain
.top().leftToRightOfView(pointIcon, offset: 10)
.width(22).height(20)
indexNameLab.layoutChain
.leftToRightOfView(indexLabel, offset: 5)
.centerY(indexLabel)
locationLabel.layoutChain
.centerY(indexLabel)
.leftToRightOfView(indexNameLab, offset: 2)
.compressionHorizontal(.defaultLow)
timeLabel.layoutChain
.centerY(indexLabel)
.leftToRightOfView(locationLabel, offset: 20)
.right(10, relation: .greaterThanOrEqual)
.compressionHorizontal(.required)
remarkLab.layoutChain
.topToBottomOfView(indexLabel, offset: 5)
.leftToView(indexLabel)
.right(15)
.bottom(10)
.compressionVertical(.required)
bottomDashView.layoutChain
.topToBottomOfView(pointIcon, offset: 5)
.centerX(pointIcon)
.width(0.5)
.bottom(5)
}
// MARK: - Views
private let pointIcon: UIImageView = {
let iv = UIImageView(image: UIImage(named: "Schedule/point"))
iv.contentMode = .scaleAspectFit
return iv
}()
private let bottomDashView: DashLineView = {
let v = DashLineView()
v.backgroundColor = .clear
return v
}()
private let indexLabel: UILabel = {
let l = UILabel()
l.backgroundColor = UIColor(hexStr: "#DCF4FF")
l.textColor = UIColor(hexStr: "#176F9B")
l.font = .systemFont(ofSize: 12, weight: .medium)
l.textAlignment = .center
return l
}()
lazy var indexNameLab: UILabel = {
let l = UILabel()
l.font = .systemFont(ofSize: 14, weight: .medium)
l.textColor = UIColor(hexStr: "#16B3FF")
return l
}()
private let locationLabel: UILabel = {
let l = UILabel()
l.font = .systemFont(ofSize: 14, weight: .medium)
l.textColor = UIColor(hexStr: "#333333")
// l.numberOfLines = 0
return l
}()
private let timeLabel: UILabel = {
let l = UILabel()
l.font = .systemFont(ofSize: 12, weight: .medium)
l.textColor = UIColor(hexStr: "#333333")
l.isUserInteractionEnabled = true
l.textAlignment = .right
return l
}()
lazy var remarkLab: UILabel = {
let l = UILabel()
l.font = .systemFont(ofSize: 12, weight: .regular)
l.textColor = UIColor(hexStr: "#999999")
l.numberOfLines = 0
return l
}()
override func layoutSubviews() {
super.layoutSubviews()
contentView.layoutIfNeeded()
indexLabel.setCornerRadius(corners: [.bottomRight, .topLeft], withCornerRadii: CGSize(width: 10, height: 10))
}
}
// MARK: - SchedulePointEventCell
class SchedulePointEventCell: UITableViewCell {
var disposeBag = DisposeBag()
func configure(model: SchedulePointEventModel, index: Int, total: Int) {
var suffix = ""
if index == 0 {
suffix = " 起点"
} else if index == total - 1 {
suffix = " 终点"
} else {
suffix = " 途经点"
}
infoLab.text = model.text + "到达" + suffix
}
// MARK: - Init
override init(style: CellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)
selectionStyle = .none
backgroundColor = .clear
setupViews()
}
required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") }
override func prepareForReuse() {
super.prepareForReuse()
disposeBag = DisposeBag()
}
private func setupViews() {
contentView.addSubview(verticalDashLineView)
contentView.addSubview(horizontalDashLineView)
contentView.addSubview(dotView)
contentView.addSubview(infoView)
infoView.addSubview(infoLab)
verticalDashLineView.layoutChain
.left(24)
.edgesVertical()
.width(1)
dotView.layoutChain
.centerX(verticalDashLineView)
.centerY()
.width(8)
.height(8)
horizontalDashLineView.layoutChain
.leftToRightOfView(dotView)
.centerY(dotView)
.height(1)
.width(30)
infoView.layoutChain
.leftToRightOfView(horizontalDashLineView, offset: 5)
.edgesVertical(10)
.right(15)
infoLab.layoutChain
.edgesHorzontal(15)
.edgesVertical(15)
}
// MARK: - Views
lazy var verticalDashLineView: DashLineView = {
let v = DashLineView()
v.backgroundColor = .clear
return v
}()
lazy var horizontalDashLineView: HorizontalDashLineView = {
let v = HorizontalDashLineView()
v.backgroundColor = .clear
return v
}()
lazy var dotView: UIView = {
let view = UIView()
view.backgroundColor = UIColor(hexStr: "#16B3FF")
view.cornerRadius = 4
return view
}()
lazy var infoView: UIView = {
let v = UIView()
v.backgroundColor = .white
v.layer.cornerRadius = 10
v.layer.shadowColor = UIColor.black.withAlphaComponent(0.1).cgColor
v.layer.shadowOffset = CGSize(width: 0, height: -2)
v.layer.shadowRadius = 10
v.layer.shadowOpacity = 1
return v
}()
lazy var infoLab: UILabel = {
let l = UILabel()
l.font = .systemFont(ofSize: 12, weight: .medium)
l.textColor = UIColor(hexStr: "#333333")
return l
}()
}