FUBeautySkinView.m 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. //
  2. // FUBeautySkinView.m
  3. // FUBeautyComponent
  4. //
  5. // Created by 项林平 on 2022/6/21.
  6. //
  7. #import "FUBeautySkinView.h"
  8. #import "FUBeautyDefine.h"
  9. #import "FUCommonUIComponent.h"
  10. static NSString * const kFUBeautySkinCellIdentifier = @"FUBeautySkinCell";
  11. @interface FUBeautySkinView ()<UICollectionViewDataSource, UICollectionViewDelegate>
  12. @property (nonatomic, strong) UICollectionView *skinCollectionView;
  13. /// 程度调节
  14. @property (nonatomic, strong) FUSlider *slider;
  15. /// 恢复按钮
  16. @property (nonatomic, strong) FUSquareButton *recoverButton;
  17. @property (nonatomic, strong) FUBeautySkinViewModel *viewModel;
  18. @end
  19. @implementation FUBeautySkinView
  20. #pragma mark - Initializer
  21. - (instancetype)initWithFrame:(CGRect)frame {
  22. return [self initWithFrame:frame viewModel:[[FUBeautySkinViewModel alloc] init]];
  23. }
  24. - (instancetype)initWithFrame:(CGRect)frame viewModel:(FUBeautySkinViewModel *)viewModel {
  25. self = [super initWithFrame:frame];
  26. if (self) {
  27. self.viewModel = viewModel;
  28. [self configureUI];
  29. [self refreshSubviews];
  30. }
  31. return self;
  32. }
  33. #pragma mark - UI
  34. - (void)configureUI {
  35. UIBlurEffect *effect = [UIBlurEffect effectWithStyle:UIBlurEffectStyleDark];
  36. UIVisualEffectView *effectView = [[UIVisualEffectView alloc] initWithEffect:effect];
  37. effectView.frame = CGRectMake(0, 0, CGRectGetWidth(self.frame), CGRectGetHeight(self.frame));
  38. [self addSubview:effectView];
  39. [self addSubview:self.slider];
  40. [self addSubview:self.recoverButton];
  41. NSLayoutConstraint *recoverLeadingConstraint = [NSLayoutConstraint constraintWithItem:self.recoverButton attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeLeading multiplier:1 constant:17];
  42. NSLayoutConstraint *recoverBottomConstraint = [NSLayoutConstraint constraintWithItem:self.recoverButton attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeBottom multiplier:1 constant:-6];
  43. NSLayoutConstraint *recoverWidthConstraint = [NSLayoutConstraint constraintWithItem:self.recoverButton attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:44];
  44. NSLayoutConstraint *recoverHeightConstraint = [NSLayoutConstraint constraintWithItem:self.recoverButton attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:74];
  45. [self addConstraints:@[recoverLeadingConstraint, recoverBottomConstraint]];
  46. [self.recoverButton addConstraints:@[recoverWidthConstraint, recoverHeightConstraint]];
  47. // 分割线
  48. UIView *verticalLine = [[UIView alloc] init];
  49. verticalLine.backgroundColor = [UIColor colorWithRed:229/255.f green:229/255.f blue:229/255.f alpha:0.2];
  50. verticalLine.translatesAutoresizingMaskIntoConstraints = NO;
  51. [self addSubview:verticalLine];
  52. NSLayoutConstraint *lineLeadingConstraint = [NSLayoutConstraint constraintWithItem:verticalLine attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:self.recoverButton attribute:NSLayoutAttributeTrailing multiplier:1 constant:14];
  53. NSLayoutConstraint *lineCenterYConstraint = [NSLayoutConstraint constraintWithItem:verticalLine attribute:NSLayoutAttributeCenterY relatedBy:NSLayoutRelationEqual toItem:self.recoverButton attribute:NSLayoutAttributeCenterY multiplier:1 constant:-15];
  54. NSLayoutConstraint *lineWidthConstraint = [NSLayoutConstraint constraintWithItem:verticalLine attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:1];
  55. NSLayoutConstraint *lineHeightConstraint = [NSLayoutConstraint constraintWithItem:verticalLine attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:24];
  56. [self addConstraints:@[lineLeadingConstraint, lineCenterYConstraint]];
  57. [verticalLine addConstraints:@[lineWidthConstraint, lineHeightConstraint]];
  58. [self addSubview:self.skinCollectionView];
  59. NSLayoutConstraint *collectionLeadingConstraint = [NSLayoutConstraint constraintWithItem:self.skinCollectionView attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeLeading multiplier:1 constant:76];
  60. NSLayoutConstraint *collectionTrailingConstraint = [NSLayoutConstraint constraintWithItem:self.skinCollectionView attribute:NSLayoutAttributeTrailing relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeTrailing multiplier:1 constant:0];
  61. NSLayoutConstraint *collectionBottomConstraint = [NSLayoutConstraint constraintWithItem:self.skinCollectionView attribute:NSLayoutAttributeBottom relatedBy:NSLayoutRelationEqual toItem:self attribute:NSLayoutAttributeBottom multiplier:1 constant:0];
  62. NSLayoutConstraint *collectionHeightConstraint = [NSLayoutConstraint constraintWithItem:self.skinCollectionView attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:98];
  63. [self addConstraints:@[collectionLeadingConstraint, collectionTrailingConstraint, collectionBottomConstraint]];
  64. [self.skinCollectionView addConstraint:collectionHeightConstraint];
  65. }
  66. - (void)refreshSubviews {
  67. dispatch_async(dispatch_get_main_queue(), ^{
  68. if (self.viewModel.isDefaultValue) {
  69. self.recoverButton.alpha = 0.6;
  70. self.recoverButton.userInteractionEnabled = NO;
  71. } else {
  72. self.recoverButton.alpha = 1;
  73. self.recoverButton.userInteractionEnabled = YES;
  74. }
  75. if (!self.slider.hidden && self.viewModel.selectedIndex >= 0) {
  76. self.slider.bidirection = self.viewModel.beautySkins[self.viewModel.selectedIndex].defaultValueInMiddle;
  77. self.slider.value = self.viewModel.beautySkins[self.viewModel.selectedIndex].currentValue / self.viewModel.beautySkins[self.viewModel.selectedIndex].ratio;
  78. }
  79. [self.skinCollectionView reloadData];
  80. if (self.viewModel.selectedIndex >= 0) {
  81. [self.skinCollectionView selectItemAtIndexPath:[NSIndexPath indexPathForItem:self.viewModel.selectedIndex inSection:0] animated:NO scrollPosition:UICollectionViewScrollPositionNone];
  82. }
  83. });
  84. }
  85. #pragma mark - Event response
  86. - (void)recoverAction {
  87. [FUAlertManager showAlertWithTitle:nil message:FUBeautyStringWithKey(@"是否将所有参数恢复到默认值") cancel:FUBeautyStringWithKey(@"取消") confirm:FUBeautyStringWithKey(@"确定") inController:nil confirmHandler:^{
  88. [self.viewModel recoverAllSkinValuesToDefault];
  89. [self refreshSubviews];
  90. } cancelHandler:nil];
  91. }
  92. - (void)sliderValueChanged {
  93. [self.viewModel setSkinValue:self.slider.value];
  94. }
  95. - (void)sliderChangeEnded {
  96. [self refreshSubviews];
  97. }
  98. #pragma mark - Collection view data source
  99. - (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section {
  100. return self.viewModel.beautySkins.count;
  101. }
  102. - (__kindof UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath {
  103. FUBeautySkinCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:kFUBeautySkinCellIdentifier forIndexPath:indexPath];
  104. FUBeautySkinModel *skin = self.viewModel.beautySkins[indexPath.item];
  105. cell.textLabel.text = FUBeautyStringWithKey(skin.name);
  106. cell.imageName = skin.name;
  107. cell.defaultInMiddle = skin.defaultValueInMiddle;
  108. cell.defaultValue = skin.defaultValue;
  109. cell.currentValue = skin.currentValue;
  110. cell.selected = indexPath.item == self.viewModel.selectedIndex;
  111. return cell;
  112. }
  113. #pragma mark - Collection view delegate
  114. - (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath {
  115. if (indexPath.item == self.viewModel.selectedIndex) {
  116. return;
  117. }
  118. self.viewModel.selectedIndex = indexPath.item;
  119. FUBeautySkinModel *skin = self.viewModel.beautySkins[indexPath.item];
  120. if (self.slider.hidden) {
  121. self.slider.hidden = NO;
  122. }
  123. self.slider.bidirection = skin.defaultValueInMiddle;
  124. self.slider.value = skin.currentValue / skin.ratio;
  125. }
  126. #pragma mark - Getters
  127. - (UICollectionView *)skinCollectionView {
  128. if (!_skinCollectionView) {
  129. UICollectionViewFlowLayout *layout = [[UICollectionViewFlowLayout alloc] init];
  130. layout.scrollDirection = UICollectionViewScrollDirectionHorizontal;
  131. layout.itemSize = CGSizeMake(44, 74);
  132. layout.minimumLineSpacing = 22;
  133. layout.minimumInteritemSpacing = 22;
  134. layout.sectionInset = UIEdgeInsetsMake(16, 16, 6, 16);
  135. _skinCollectionView = [[UICollectionView alloc] initWithFrame:CGRectZero collectionViewLayout:layout];
  136. _skinCollectionView.translatesAutoresizingMaskIntoConstraints = NO;
  137. _skinCollectionView.backgroundColor = [UIColor clearColor];
  138. _skinCollectionView.showsVerticalScrollIndicator = NO;
  139. _skinCollectionView.showsHorizontalScrollIndicator = NO;
  140. _skinCollectionView.dataSource = self;
  141. _skinCollectionView.delegate = self;
  142. [_skinCollectionView registerClass:[FUBeautySkinCell class] forCellWithReuseIdentifier:kFUBeautySkinCellIdentifier];
  143. }
  144. return _skinCollectionView;
  145. }
  146. - (FUSquareButton *)recoverButton {
  147. if (!_recoverButton) {
  148. _recoverButton = [[FUSquareButton alloc] initWithFrame:CGRectMake(0, 0, 44, 74)];
  149. [_recoverButton setTitle:FUBeautyStringWithKey(@"恢复") forState:UIControlStateNormal];
  150. [_recoverButton setImage:[UIImage imageNamed:@"recover_item"] forState:UIControlStateNormal];
  151. _recoverButton.alpha = 0.6;
  152. _recoverButton.userInteractionEnabled = NO;
  153. [_recoverButton addTarget:self action:@selector(recoverAction) forControlEvents:UIControlEventTouchUpInside];
  154. _recoverButton.translatesAutoresizingMaskIntoConstraints = NO;
  155. }
  156. return _recoverButton;
  157. }
  158. -(FUSlider *)slider {
  159. if (!_slider) {
  160. _slider = [[FUSlider alloc] initWithFrame:CGRectMake(56, 16, CGRectGetWidth(self.frame) - 116, FUBeautyFunctionSliderHeight)];
  161. _slider.hidden = YES;
  162. [_slider addTarget:self action:@selector(sliderValueChanged) forControlEvents:UIControlEventValueChanged];
  163. [_slider addTarget:self action:@selector(sliderChangeEnded) forControlEvents:UIControlEventTouchUpInside | UIControlEventTouchUpOutside];
  164. }
  165. return _slider;
  166. }
  167. @end
  168. @interface FUBeautySkinCell ()
  169. @property (nonatomic, strong) UIImageView *imageView;
  170. @property (nonatomic, strong) UILabel *textLabel;
  171. @end
  172. @implementation FUBeautySkinCell
  173. - (instancetype)initWithFrame:(CGRect)frame {
  174. self = [super initWithFrame:frame];
  175. if (self) {
  176. [self.contentView addSubview:self.imageView];
  177. NSLayoutConstraint *imageTop = [NSLayoutConstraint constraintWithItem:self.imageView attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:self.contentView attribute:NSLayoutAttributeTop multiplier:1 constant:0];
  178. NSLayoutConstraint *imageLeading = [NSLayoutConstraint constraintWithItem:self.imageView attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:self.contentView attribute:NSLayoutAttributeLeading multiplier:1 constant:0];
  179. NSLayoutConstraint *imageTrailing = [NSLayoutConstraint constraintWithItem:self.imageView attribute:NSLayoutAttributeTrailing relatedBy:NSLayoutRelationEqual toItem:self.contentView attribute:NSLayoutAttributeTrailing multiplier:1 constant:0];
  180. NSLayoutConstraint *imageHeight = [NSLayoutConstraint constraintWithItem:self.imageView attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:self.imageView attribute:NSLayoutAttributeWidth multiplier:1 constant:0];
  181. [self.contentView addConstraints:@[imageTop, imageLeading, imageTrailing]];
  182. [self.imageView addConstraint:imageHeight];
  183. [self.contentView addSubview:self.textLabel];
  184. NSLayoutConstraint *textTop = [NSLayoutConstraint constraintWithItem:self.textLabel attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:self.imageView attribute:NSLayoutAttributeBottom multiplier:1 constant:7];
  185. NSLayoutConstraint *textLeading = [NSLayoutConstraint constraintWithItem:self.textLabel attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:self.contentView attribute:NSLayoutAttributeLeading multiplier:1 constant:0];
  186. NSLayoutConstraint *textTrailing = [NSLayoutConstraint constraintWithItem:self.textLabel attribute:NSLayoutAttributeTrailing relatedBy:NSLayoutRelationEqual toItem:self.contentView attribute:NSLayoutAttributeTrailing multiplier:1 constant:0];
  187. [self.contentView addConstraints:@[textTop, textLeading, textTrailing]];
  188. }
  189. return self;
  190. }
  191. - (void)setSelected:(BOOL)selected {
  192. [super setSelected:selected];
  193. BOOL changed = NO;
  194. if (self.defaultInMiddle) {
  195. changed = fabs(self.currentValue - 0.5) > 0.01;
  196. }else{
  197. changed = self.currentValue > 0.01;
  198. }
  199. if (selected) {
  200. self.imageView.image = changed ? [UIImage imageNamed:[NSString stringWithFormat:@"%@-3", self.imageName]] : [UIImage imageNamed:[NSString stringWithFormat:@"%@-2", self.imageName]];
  201. self.textLabel.textColor = [UIColor colorWithRed:94/255.f green:199/255.f blue:254/255.f alpha:1];
  202. } else {
  203. self.imageView.image = changed ? [UIImage imageNamed:[NSString stringWithFormat:@"%@-1", self.imageName]] : [UIImage imageNamed:[NSString stringWithFormat:@"%@-0", self.imageName]];
  204. self.textLabel.textColor = [UIColor whiteColor];
  205. }
  206. }
  207. - (UIImageView *)imageView {
  208. if (!_imageView) {
  209. _imageView = [[UIImageView alloc] init];
  210. _imageView.translatesAutoresizingMaskIntoConstraints = NO;
  211. }
  212. return _imageView;
  213. }
  214. - (UILabel *)textLabel {
  215. if (!_textLabel) {
  216. _textLabel = [[UILabel alloc] init];
  217. _textLabel.font = [UIFont systemFontOfSize:10];
  218. _textLabel.textColor = [UIColor whiteColor];
  219. _textLabel.textAlignment = NSTextAlignmentCenter;
  220. _textLabel.adjustsFontSizeToFitWidth = YES;
  221. _textLabel.translatesAutoresizingMaskIntoConstraints = NO;
  222. }
  223. return _textLabel;
  224. }
  225. @end