设置页面,删除账户

This commit is contained in:
zeng 2025-04-29 15:47:30 +08:00
parent d5a6e44f49
commit 73078a1acf
26 changed files with 571 additions and 3 deletions

View File

@ -364,5 +364,21 @@ extension UIColor {
static func colorFF8E33(alpha: CGFloat = 1) -> UIColor {
return color(hex: 0xFF8E33, alpha: alpha)
}
static func colorB2B2B2(alpha: CGFloat = 1) -> UIColor {
return color(hex: 0xB2B2B2, alpha: alpha)
}
static func color8B8B8B(alpha: CGFloat = 1) -> UIColor {
return color(hex: 0x8B8B8B, alpha: alpha)
}
static func color272A30(alpha: CGFloat = 1) -> UIColor {
return color(hex: 0x272A30, alpha: alpha)
}
static func colorFF1F1F(alpha: CGFloat = 1) -> UIColor {
return color(hex: 0xFF1F1F, alpha: alpha)
}
}

View File

@ -9,7 +9,7 @@ import UIKit
class SPUserAPI: NSObject {
///
static func requestUserInfo(completer: ((_ userInfo: SPUserInfo?) -> Void)?) {
var param = SPNetworkParameters(path: "/customer/info")
@ -20,4 +20,19 @@ class SPUserAPI: NSObject {
}
}
///
static func requestLogoff(completer: ((_ isFinish: Bool) -> Void)?) {
var param = SPNetworkParameters(path: "/customer/logoff")
param.isLoding = true
SPNetwork.request(parameters: param) { (response: SPNetworkResponse<SPUserInfo>) in
if response.code == SPNetworkCodeSucceed {
completer?(true)
} else {
completer?(false)
}
}
}
}

View File

@ -0,0 +1,175 @@
//
// SPDeleteAccountViewController.swift
// MoviaBox
//
// Created by on 2025/4/29.
//
import UIKit
class SPDeleteAccountViewController: SPViewController {
private lazy var imageArr: [UIImage] = {
let arr = [
UIImage(named: "delete_account_image_01")!,
UIImage(named: "delete_account_image_02")!,
UIImage(named: "delete_account_image_03")!,
UIImage(named: "delete_account_image_04")!,
]
return arr
}()
private lazy var imageViewArr: [UIImageView] = []
//MARK: UI
private lazy var scrollView: SPScrollView = {
let scrollView = SPScrollView()
return scrollView
}()
private lazy var lineView: UIView = {
let view = UIView()
view.backgroundColor = .colorFFFFFF(alpha: 0.25)
return view
}()
private lazy var checkButton: UIButton = {
let button = UIButton(type: .custom)
button.setImage(UIImage(named: "check_icon_01"), for: .normal)
button.setImage(UIImage(named: "check_icon_01_selected"), for: .selected)
button.setImage(UIImage(named: "check_icon_01_selected"), for: [.selected, .highlighted])
button.addTarget(self, action: #selector(handleCheckButton), for: .touchUpInside)
return button
}()
private lazy var checkLabel: UILabel = {
let label = UILabel()
label.font = .fontMedium(ofSize: 14)
label.textColor = .colorFFFFFF()
label.numberOfLines = 0
label.text = "kDeleteAccountCheckText".localized
return label
}()
private lazy var deleteButton: UIButton = {
let button = UIButton(type: .custom)
button.setTitle("Delete Account".localized, for: .normal)
button.setTitleColor(.color8B8B8B(), for: .disabled)
button.setTitleColor(.colorFFFFFF(), for: .normal)
button.setBackgroundImage(UIImage(color: .color272A30()), for: .disabled)
button.setBackgroundImage(UIImage(color: .colorFF1F1F()), for: .normal)
button.titleLabel?.font = .fontMedium(ofSize: 14)
button.layer.cornerRadius = 18
button.layer.masksToBounds = true
button.isEnabled = false
button.addTarget(self, action: #selector(handleDeleteButton), for: .touchUpInside)
return button
}()
override func viewDidLoad() {
super.viewDidLoad()
self.title = "Account Deletion".localized
setBackgroundView(isShowGradient: false, bgImage: nil)
_setupUI()
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
self.navigationController?.setNavigationBarHidden(false, animated: true)
setNavigationNormalStyle()
}
}
extension SPDeleteAccountViewController {
@objc private func handleCheckButton() {
self.checkButton.isSelected = !self.checkButton.isSelected
self.deleteButton.isEnabled = self.checkButton.isSelected
}
@objc private func handleDeleteButton() {
SPUserAPI.requestLogoff { isFinish in
}
}
}
extension SPDeleteAccountViewController {
private func _setupUI() {
view.addSubview(scrollView)
scrollView.addSubview(lineView)
scrollView.addSubview(checkButton)
scrollView.addSubview(checkLabel)
scrollView.addSubview(deleteButton)
imageArr.forEach {
let imageSize = $0.size
let imageView = UIImageView(image: $0)
scrollView.addSubview(imageView)
if let lastImageView = self.imageViewArr.last {
let width = kSPScreenWidth - 32
let height = imageSize.height / imageSize.width * width
imageView.snp.makeConstraints { make in
make.top.equalTo(lastImageView.snp.bottom).offset(18)
make.width.equalTo(width)
make.height.equalTo(height)
make.centerX.equalToSuperview()
}
} else {
imageView.snp.makeConstraints { make in
make.top.equalToSuperview().offset(10)
make.width.equalTo(kSPMainW(imageSize.width))
make.height.equalTo(kSPMainW(imageSize.height))
make.centerX.equalToSuperview()
}
}
self.imageViewArr.append(imageView)
}
scrollView.snp.makeConstraints { make in
make.edges.equalToSuperview()
}
lineView.snp.makeConstraints { make in
make.centerX.equalToSuperview()
make.width.equalTo(kSPScreenWidth - 32)
make.top.equalTo(self.imageViewArr.last!.snp.bottom).offset(18)
make.height.equalTo(1)
}
checkButton.snp.makeConstraints { make in
make.left.equalToSuperview().offset(16)
make.top.equalTo(lineView.snp.bottom).offset(18)
}
checkLabel.snp.makeConstraints { make in
make.left.equalTo(checkButton.snp.right).offset(10)
// make.right.lessThanOrEqualToSuperview().offset(-16)
make.right.lessThanOrEqualTo(lineView)
make.top.equalTo(checkButton).offset(2)
}
deleteButton.snp.makeConstraints { make in
make.left.right.equalTo(lineView)
make.top.equalTo(checkLabel.snp.bottom).offset(18)
make.height.equalTo(36)
make.bottom.equalToSuperview().offset(-(kSPTabbarSafeBottomMargin + 15))
}
}
}

View File

@ -9,6 +9,14 @@ import UIKit
class SPSettingsViewController: SPViewController {
private lazy var dataArr: [SPMineItem] = createDataArr() {
didSet {
self.tableView.reloadData()
}
}
//MARK: UI
private lazy var tableView: SPTableView = {
let tableView = SPTableView(frame: .zero, style: .insetGrouped)
@ -25,6 +33,12 @@ class SPSettingsViewController: SPViewController {
super.viewDidLoad()
self.setBackgroundView(isShowGradient: false, bgImage: nil)
self.title = "Settings".localized
SPAppCacheManager.manager.getAllCache { [weak self] value in
guard let self = self else { return }
self.dataArr = self.createDataArr()
}
_setupUI()
}
@ -54,6 +68,7 @@ extension SPSettingsViewController: UITableViewDelegate, UITableViewDataSource {
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = SPSettingsCell.dequeueReusableCell(tableView: tableView, indexPath: indexPath)
cell.item = dataArr[indexPath.section]
return cell
}
@ -62,7 +77,7 @@ extension SPSettingsViewController: UITableViewDelegate, UITableViewDataSource {
}
func numberOfSections(in tableView: UITableView) -> Int {
return 5
return self.dataArr.count
}
func tableView(_ tableView: UITableView, heightForHeaderInSection section: Int) -> CGFloat {
@ -73,4 +88,48 @@ extension SPSettingsViewController: UITableViewDelegate, UITableViewDataSource {
return 0
}
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
let item = dataArr[indexPath.section]
switch item.type {
case .aboutUs:
let vc = SPAboutUsViewController()
self.navigationController?.pushViewController(vc, animated: true)
case .clearCache:
clearCache()
case .deleteAccount:
let vc = SPDeleteAccountViewController()
self.navigationController?.pushViewController(vc, animated: true)
default:
break
}
}
}
extension SPSettingsViewController {
private func clearCache() {
SPAppCacheManager.manager.cleanAllCache { [weak self] in
guard let self = self else { return }
self.dataArr = self.createDataArr()
}
}
private func createDataArr() -> [SPMineItem] {
let cache = SPAppCacheManager.manager.allCache
let arr = [
SPMineItem(type: .clearCache, title: "Clear Cache".localized, subtitle: SPAppCacheManager.cacheToString(cache: cache)),
SPMineItem(type: .aboutUs, title: "About Us".localized),
SPMineItem(type: .deleteAccount, title: "Delete Account".localized),
]
return arr
}
}

View File

@ -39,12 +39,17 @@ struct SPMineItem {
case purchaseRecords
///
case rewardCoins
///
case clearCache
///
case deleteAccount
}
var type: ItemType?
var iconImage: UIImage?
var title: String?
var subtitle: String?
}

View File

@ -9,15 +9,64 @@ import UIKit
class SPSettingsCell: SPTableViewCell {
var item: SPMineItem? {
didSet {
titleLabel.text = item?.title
if let subtitle = item?.subtitle {
subtitleLabel.text = subtitle
showIndicator = false
subtitleLabel.isHidden = false
} else {
showIndicator = true
subtitleLabel.isHidden = true
}
}
}
//MARK: UI
private lazy var titleLabel: UILabel = {
let label = UILabel()
label.font = .fontRegular(ofSize: 14)
label.textColor = .colorFFFFFF()
return label
}()
private lazy var subtitleLabel: UILabel = {
let label = UILabel()
label.font = .fontRegular(ofSize: 14)
label.textColor = .colorB2B2B2()
return label
}()
override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)
self.contentView.backgroundColor = .color202531()
self.showIndicator = true
_setupUI()
}
@MainActor required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}
extension SPSettingsCell {
private func _setupUI() {
contentView.addSubview(titleLabel)
contentView.addSubview(subtitleLabel)
titleLabel.snp.makeConstraints { make in
make.left.equalToSuperview().offset(16)
make.centerY.equalToSuperview()
}
subtitleLabel.snp.makeConstraints { make in
make.centerY.equalToSuperview()
make.right.equalToSuperview().offset(-16)
}
}
}

View File

@ -0,0 +1,113 @@
//
// SPAppCacheManager.swift
// MoviaBox
//
// Created by on 2025/4/29.
//
import UIKit
import Kingfisher
class SPAppCacheManager: NSObject {
static let manager = SPAppCacheManager()
private(set) var imageCache: Double = 0
private(set) var videoCache: Double = 0
var allCache: Double {
return imageCache + videoCache
}
private var group = DispatchGroup()
///
func getAllCache(completer: ((_ value: Double) -> Void)?) {
self.group.enter()
getImageCache { [weak self] value in
guard let self = self else { return }
self.group.leave()
}
self.group.enter()
getVideoCache { [weak self] value in
guard let self = self else { return }
self.group.leave()
}
self.group.notify(queue: DispatchQueue.main) { [weak self] in
guard let self = self else { return }
completer?(self.imageCache + self.videoCache)
}
}
///
func cleanAllCache(completer: (() -> Void)?) {
self.group.enter()
cleanImageCache { [weak self] in
self?.group.leave()
}
self.group.enter()
cleanVideoCache { [weak self] in
self?.group.leave()
}
self.group.notify(queue: DispatchQueue.main) { [weak self] in
guard let self = self else { return }
completer?()
}
}
///
func getImageCache(completer: ((_ value: Double) -> Void)?) {
ImageCache.default.calculateDiskStorageSize { [weak self] result in
guard let self = self else { return }
switch result {
case .success(let value):
self.imageCache = Double(value)
default:
self.imageCache = 0
break
}
completer?(self.imageCache)
}
}
///
func getVideoCache(completer: ((_ value: Double) -> Void)?) {
self.videoCache = Double(KTVHTTPCache.cacheTotalCacheLength())
completer?(self.videoCache)
}
///
func cleanImageCache(completer: (() -> Void)?) {
DispatchQueue.global().async {
ImageCache.default.clearDiskCache {
self.imageCache = 0
completer?()
}
}
}
///
func cleanVideoCache(completer: (() -> Void)?) {
KTVHTTPCache.cacheDeleteAllCaches()
self.videoCache = 0
completer?()
}
}
extension SPAppCacheManager {
static func cacheToString(cache: Double) -> String {
let size = cache / 1024 / 1024
if size > 100 {
return String(format: "%.0fM", size)
} else {
return String(format: "%.2fM", size)
}
}
}

View File

@ -0,0 +1,22 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x"
},
{
"filename" : "Ellipse 45@2x.png",
"idiom" : "universal",
"scale" : "2x"
},
{
"filename" : "Ellipse 45@3x.png",
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 580 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 897 B

View File

@ -0,0 +1,22 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x"
},
{
"filename" : "Vector@2x.png",
"idiom" : "universal",
"scale" : "2x"
},
{
"filename" : "Vector@3x.png",
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 833 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

View File

@ -0,0 +1,22 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x"
},
{
"filename" : "Group 481@2x.png",
"idiom" : "universal",
"scale" : "2x"
},
{
"filename" : "Group 481@3x.png",
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

View File

@ -0,0 +1,22 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x"
},
{
"filename" : "Group 477@2x.png",
"idiom" : "universal",
"scale" : "2x"
},
{
"filename" : "Group 477@3x.png",
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.8 KiB

View File

@ -0,0 +1,22 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x"
},
{
"filename" : "Group 476@2x.png",
"idiom" : "universal",
"scale" : "2x"
},
{
"filename" : "Group 476@3x.png",
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 44 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 68 KiB

View File

@ -0,0 +1,22 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x"
},
{
"filename" : "Group 478@2x.png",
"idiom" : "universal",
"scale" : "2x"
},
{
"filename" : "Group 478@3x.png",
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 15 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 24 KiB

View File

@ -69,10 +69,14 @@
"Membership" = "Membership";
"%@ Bonus" = "%@ Bonus";
"+ Extra %@" = "+ Extra %@";
"Clear Cache" = "Clear Cache";
"Delete Account" = "Delete Account";
"Account Deletion" = "Account Deletion";
"kLoginAgreementText" = "By continuing, you agree to the User Agreement and Privacy Policy";
"kBuyMemberTipText" = "Auto renew · Cancel anytime";
"kStoreTipTitle" = "Related terms and conditions:";
"kStoreTipText" = "1. Coins can only be used within this application.2. Payment: The purchase will be charged to your iTunes account. 3. Renewal: Your Apple iTunes account will be charged within 24 hours before the expiration and the subscription period will be extended for another subscription cycle upon successful deduction.4. Cancellation: To cancel the subscription renewal, please turn off the automatic renewal function in the iTunes/Apple ID settings at least 24 hours before the current subscription period expires. If canceled within the last 24 hours before expiration, a subscription fee will still be charged. 5. Payment successful but recharge not taking effect for an extended period? Click here to refresh or send an email to: cs.jiaer.developer@icloud.com. 6. Manage your subscriptions: You can view, change, or cancel your subscriptions. Terms of Service | Privacy Policy Renewal Agreement";
"kDeleteAccountCheckText" = "I accept the deletion risk and agree to delete my account";