提交
This commit is contained in:
@@ -0,0 +1,24 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPopupController.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaPasterSelectControllerDelegate;
|
||||
|
||||
/**
|
||||
贴纸选择界面Controller
|
||||
*/
|
||||
@interface TUIMultimediaPasterSelectController : TUIMultimediaPopupController
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaPasterSelectControllerDelegate> delegate;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaPasterSelectControllerDelegate <NSObject>
|
||||
- (void)pasterSelectController:(TUIMultimediaPasterSelectController *)c onPasterSelected:(UIImage *)image;
|
||||
- (void)onPasterSelectControllerExit:(TUIMultimediaPasterSelectController *)c;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,92 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaPasterSelectController.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <ReactiveObjC/RACEXTScope.h>
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImagePicker.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterSelectView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPersistence.h"
|
||||
|
||||
@interface TUIMultimediaPasterSelectController () <TUIMultimediaPasterSelectViewDelegate> {
|
||||
TUIMultimediaPasterSelectView *_selectView;
|
||||
TUIMultimediaImagePicker *_picker;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaPasterSelectController
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)viewDidLoad {
|
||||
[super viewDidLoad];
|
||||
_selectView = [[TUIMultimediaPasterSelectView alloc] init];
|
||||
_selectView.config = [TUIMultimediaPasterConfig loadConfig];
|
||||
_selectView.delegate = self;
|
||||
self.mainView = _selectView;
|
||||
}
|
||||
|
||||
- (void)popupControllerDidCanceled {
|
||||
[_delegate onPasterSelectControllerExit:self];
|
||||
}
|
||||
|
||||
- (void)onPasterSelected:(UIImage *)image {
|
||||
[_delegate pasterSelectController:self onPasterSelected:image];
|
||||
}
|
||||
|
||||
- (void)pasterSelectView:(TUIMultimediaPasterSelectView *)v needAddCustomPaster:(TUIMultimediaPasterGroupConfig *)group completeCallback:(void (^)(void))callback {
|
||||
_picker = [[TUIMultimediaImagePicker alloc] init];
|
||||
@weakify(self);
|
||||
_picker.callback = ^(UIImage *img) {
|
||||
@strongify(self);
|
||||
if (img == nil || self == nil) {
|
||||
return;
|
||||
}
|
||||
NSURL *url = [TUIMultimediaPasterConfig saveCustomPaster:img];
|
||||
if (url == nil) {
|
||||
return;
|
||||
}
|
||||
TUIMultimediaPasterItemConfig *item = [[TUIMultimediaPasterItemConfig alloc] init];
|
||||
item.imageUrl = url;
|
||||
item.isUserAdded = YES;
|
||||
group.itemList = [group.itemList arrayByAddingObject:item];
|
||||
[TUIMultimediaPasterConfig saveConfig:v.config];
|
||||
callback();
|
||||
};
|
||||
[_picker presentOn:self];
|
||||
}
|
||||
|
||||
- (void)pasterSelectView:(TUIMultimediaPasterSelectView *)v
|
||||
needDeleteCustomPasterInGroup:(TUIMultimediaPasterGroupConfig *)group
|
||||
index:(NSInteger)index
|
||||
completeCallback:(void (^)(BOOL deleted))callback {
|
||||
UIAlertController *alert = [UIAlertController alertControllerWithTitle:[TUIMultimediaCommon localizedStringForKey:@"remove_paster"]
|
||||
message:nil
|
||||
preferredStyle:UIAlertControllerStyleActionSheet];
|
||||
|
||||
UIAlertAction *cancelAction = [UIAlertAction actionWithTitle:[TUIMultimediaCommon localizedStringForKey:@"cancel"]
|
||||
style:UIAlertActionStyleCancel
|
||||
handler:^(UIAlertAction *action) {
|
||||
callback(NO);
|
||||
}];
|
||||
UIAlertAction *deleteAction = [UIAlertAction actionWithTitle:[TUIMultimediaCommon localizedStringForKey:@"delete"]
|
||||
style:UIAlertActionStyleDestructive
|
||||
handler:^(UIAlertAction *action) {
|
||||
NSMutableArray<TUIMultimediaPasterItemConfig *> *list = [NSMutableArray arrayWithArray:group.itemList];
|
||||
TUIMultimediaPasterItemConfig *paster = list[index];
|
||||
[list removeObjectAtIndex:index];
|
||||
group.itemList = list;
|
||||
[TUIMultimediaPasterConfig removeCustomPaster:paster];
|
||||
[TUIMultimediaPasterConfig saveConfig:self->_selectView.config];
|
||||
callback(YES);
|
||||
}];
|
||||
[alert addAction:cancelAction];
|
||||
[alert addAction:deleteAction];
|
||||
[self presentViewController:alert animated:YES completion:nil];
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,22 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSubtitleInfo.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@class TUIMultimediaSubtitleEditController;
|
||||
|
||||
typedef void (^TUIMultimediaSubtitleEditControllerCallback)(TUIMultimediaSubtitleEditController *c, BOOL isOk);
|
||||
|
||||
/**
|
||||
字幕编辑界面Controller
|
||||
*/
|
||||
@interface TUIMultimediaSubtitleEditController : UIViewController
|
||||
@property(copy, nonatomic) TUIMultimediaSubtitleInfo *subtitleInfo;
|
||||
@property(nullable, nonatomic) TUIMultimediaSubtitleEditControllerCallback callback;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,59 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaSubtitleEditController.h"
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSubtitleEditView.h"
|
||||
|
||||
@interface TUIMultimediaSubtitleEditController () <TUIMultimediaSubtitleEditViewDelegate> {
|
||||
TUIMultimediaSubtitleEditView *_editView;
|
||||
BOOL _hasCallback;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaSubtitleEditController
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)viewDidLoad {
|
||||
_editView = [[TUIMultimediaSubtitleEditView alloc] initWithFrame:self.view.bounds];
|
||||
_editView.subtitleInfo = _subtitleInfo;
|
||||
_editView.delegate = self;
|
||||
[self.view addSubview:_editView];
|
||||
}
|
||||
|
||||
- (void)viewDidAppear:(BOOL)animated {
|
||||
[_editView activate];
|
||||
_hasCallback = false;
|
||||
}
|
||||
|
||||
- (void)viewDidLayoutSubviews {
|
||||
[super viewDidLayoutSubviews];
|
||||
_editView.frame = self.view.bounds;
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaSubtitleEditViewDelegate protocol
|
||||
- (void)subtitleEditViewOnOk:(TUIMultimediaSubtitleEditView *)view {
|
||||
if (_callback != nil && !_hasCallback) {
|
||||
_hasCallback = true;
|
||||
_callback(self, YES);
|
||||
}
|
||||
}
|
||||
- (void)subtitleEditViewOnCancel:(TUIMultimediaSubtitleEditView *)view {
|
||||
if (_callback != nil && !_hasCallback) {
|
||||
_hasCallback = true;
|
||||
_callback(self, NO);
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - Setters
|
||||
- (void)setSubtitleInfo:(TUIMultimediaSubtitleInfo *)subtitleInfo {
|
||||
_subtitleInfo = [subtitleInfo copy];
|
||||
_editView.subtitleInfo = _subtitleInfo;
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,17 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface TUIMultimediaColorCell : UICollectionViewCell
|
||||
@property(nullable, nonatomic) UIColor *color;
|
||||
@property(nonatomic) BOOL colorSelected;
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame;
|
||||
|
||||
+ (NSString *)reuseIdentifier;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,73 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaColorCell.h"
|
||||
|
||||
const CGFloat ContentMarginSelected = 0;
|
||||
const CGFloat ContentMarginDeselected = 5;
|
||||
|
||||
const CGFloat FrontMargin = 4;
|
||||
|
||||
@interface TUIMultimediaColorCell () {
|
||||
UIView *_front;
|
||||
UIView *_back;
|
||||
CGFloat _contentMargin;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaColorCell
|
||||
|
||||
+ (NSString *)reuseIdentifier {
|
||||
return NSStringFromClass([self class]);
|
||||
}
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
_contentMargin = ContentMarginDeselected;
|
||||
_color = UIColor.blackColor;
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
_back = [[UIView alloc] init];
|
||||
_back.backgroundColor = UIColor.whiteColor;
|
||||
[self.contentView addSubview:_back];
|
||||
|
||||
_front = [[UIView alloc] init];
|
||||
_front.backgroundColor = _color;
|
||||
[self.contentView addSubview:_front];
|
||||
|
||||
[self layoutSubviews];
|
||||
}
|
||||
|
||||
- (void)layoutSubviews {
|
||||
[super layoutSubviews];
|
||||
CGFloat len = MIN(self.contentView.bounds.size.width, self.contentView.bounds.size.height) - _contentMargin;
|
||||
CGPoint center = CGPointMake(self.contentView.bounds.size.width / 2, self.contentView.bounds.size.height / 2);
|
||||
|
||||
_back.bounds = CGRectMake(0, 0, len, len);
|
||||
_back.center = center;
|
||||
_back.layer.cornerRadius = len / 2;
|
||||
_back.clipsToBounds = YES;
|
||||
|
||||
_front.bounds = CGRectMake(0, 0, len - FrontMargin, len - FrontMargin);
|
||||
_front.center = center;
|
||||
_front.layer.cornerRadius = (len - FrontMargin) / 2;
|
||||
_front.clipsToBounds = YES;
|
||||
}
|
||||
|
||||
- (void)setColor:(UIColor *)color {
|
||||
_color = color;
|
||||
_front.backgroundColor = _color;
|
||||
}
|
||||
|
||||
- (void)setColorSelected:(BOOL)colorSelected {
|
||||
_colorSelected = colorSelected;
|
||||
_contentMargin = colorSelected ? ContentMarginSelected : ContentMarginDeselected;
|
||||
[self setNeedsLayout];
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,15 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface TUIMultimediaImageCell : UICollectionViewCell
|
||||
@property(nullable, nonatomic) UIImage *image;
|
||||
- (instancetype)initWithFrame:(CGRect)frame;
|
||||
|
||||
+ (NSString *)reuseIdentifier;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,44 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaImageCell.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
|
||||
@interface TUIMultimediaImageCell () {
|
||||
UIImageView *_imgView;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaImageCell
|
||||
|
||||
@dynamic image;
|
||||
|
||||
+ (NSString *)reuseIdentifier {
|
||||
return NSStringFromClass([self class]);
|
||||
}
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
_imgView = [[UIImageView alloc] init];
|
||||
_imgView.contentMode = UIViewContentModeScaleAspectFit;
|
||||
[self.contentView addSubview:_imgView];
|
||||
|
||||
[_imgView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
}
|
||||
- (UIImage *)image {
|
||||
return _imgView.image;
|
||||
}
|
||||
- (void)setImage:(UIImage *)image {
|
||||
_imgView.image = image;
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,13 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaStickerView.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface TUIMultimediaPasterView : TUIMultimediaStickerView
|
||||
@property(nullable, nonatomic) UIImage *paster;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,44 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaPasterView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaStickerView.h"
|
||||
|
||||
#define PASTER_INITIAL_WIDTH 150
|
||||
|
||||
@interface TUIMultimediaPasterView () {
|
||||
UIImageView *_imgView;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaPasterView
|
||||
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
_imgView = [[UIImageView alloc] init];
|
||||
_imgView.translatesAutoresizingMaskIntoConstraints = NO;
|
||||
self.editButtonHidden = YES;
|
||||
self.content = _imgView;
|
||||
}
|
||||
- (UIImage *)paster {
|
||||
return _imgView.image;
|
||||
}
|
||||
|
||||
- (void)setPaster:(UIImage *)paster {
|
||||
self.content = nil;
|
||||
_imgView.image = paster;
|
||||
int width = MIN(PASTER_INITIAL_WIDTH, paster.size.width);
|
||||
int height = width / paster.size.width * paster.size.height;
|
||||
_imgView.bounds = CGRectMake(0, 0, width, height);
|
||||
self.content = _imgView;
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,30 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaStickerViewDelegate;
|
||||
|
||||
@interface TUIMultimediaStickerView : UIView
|
||||
@property(nullable, nonatomic) UIView *content;
|
||||
@property(nonatomic) BOOL selected;
|
||||
@property(nonatomic) BOOL editButtonHidden;
|
||||
@property(nonatomic, readonly) CGFloat rotation;
|
||||
@property(nonatomic) CGFloat contentMargin;
|
||||
// 旋转夹角小于此值时,吸附到水平和竖直方向
|
||||
@property(nonatomic) CGFloat rotateAdsorptionLimitAngle;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaStickerViewDelegate> delegate;
|
||||
|
||||
- (void)scale:(CGFloat) scale;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaStickerViewDelegate <NSObject>
|
||||
- (void)onStickerViewSelected:(TUIMultimediaStickerView *)v;
|
||||
- (void)onStickerViewShouldDelete:(TUIMultimediaStickerView *)v;
|
||||
- (void)onStickerViewShouldEdit:(TUIMultimediaStickerView *)v;
|
||||
- (void)onStickerViewSizeChanged:(TUIMultimediaStickerView *)v;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,255 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaStickerView.h"
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "Masonry/Masonry.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGeometry.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImageUtil.h"
|
||||
|
||||
@interface TUIMultimediaStickerView () <UIGestureRecognizerDelegate> {
|
||||
UIButton *_btnEdit;
|
||||
UIButton *_btnTransform;
|
||||
UIButton *_btnDelete;
|
||||
UIView *_borderView;
|
||||
BOOL _hideEditButton;
|
||||
BOOL _active;
|
||||
|
||||
CGFloat _rawRotation;
|
||||
|
||||
UITapGestureRecognizer *_tapRec;
|
||||
UITapGestureRecognizer *_doubleTapRec;
|
||||
UIPanGestureRecognizer *_panRec;
|
||||
UIPinchGestureRecognizer *_pinchRec;
|
||||
UIRotationGestureRecognizer *_rotationRec;
|
||||
UIPanGestureRecognizer *_btnTransformPanRec;
|
||||
UIImpactFeedbackGenerator *_impactGen;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaStickerView
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
_contentMargin = 5;
|
||||
_rotateAdsorptionLimitAngle = M_PI / 180 * 5;
|
||||
_impactGen = [[UIImpactFeedbackGenerator alloc] init];
|
||||
[self frv_initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)frv_initUI {
|
||||
_borderView = [[UIView alloc] init];
|
||||
_borderView.translatesAutoresizingMaskIntoConstraints = NO;
|
||||
_borderView.layer.borderWidth = 1;
|
||||
_borderView.layer.borderColor = UIColor.whiteColor.CGColor;
|
||||
_borderView.backgroundColor = [UIColor clearColor];
|
||||
_borderView.hidden = YES;
|
||||
[self addSubview:_borderView];
|
||||
[_borderView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
|
||||
_btnEdit = [self newCornerButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"sticker_edit_img", @"round_edit") left:YES top:YES];
|
||||
_btnTransform = [self newCornerButtonWithImage:[TUIMultimediaImageUtil rotateImage:TUIMultimediaPluginBundleThemeImage(@"sticker_transform_img", @"round_transform") angle:90.0f] left:NO top:NO];
|
||||
_btnDelete = [self newCornerButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"sticker_delete_img", @"round_close") left:NO top:YES];
|
||||
|
||||
[_btnEdit addTarget:self action:@selector(onBtnEditClicked) forControlEvents:UIControlEventTouchUpInside];
|
||||
[_btnDelete addTarget:self action:@selector(onBtnDeleteClicked) forControlEvents:UIControlEventTouchUpInside];
|
||||
|
||||
_tapRec = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(onTap)];
|
||||
[self addGestureRecognizer:_tapRec];
|
||||
|
||||
_panRec = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(onPan:)];
|
||||
_panRec.delegate = self;
|
||||
[self addGestureRecognizer:_panRec];
|
||||
|
||||
_pinchRec = [[UIPinchGestureRecognizer alloc] initWithTarget:self action:@selector(onPinch:)];
|
||||
_pinchRec.delegate = self;
|
||||
[self addGestureRecognizer:_pinchRec];
|
||||
|
||||
_rotationRec = [[UIRotationGestureRecognizer alloc] initWithTarget:self action:@selector(onGestureRotation:)];
|
||||
_rotationRec.delegate = self;
|
||||
[self addGestureRecognizer:_rotationRec];
|
||||
|
||||
_btnTransformPanRec = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(onBtnTransformPan:)];
|
||||
[_btnTransform addGestureRecognizer:_btnTransformPanRec];
|
||||
|
||||
_doubleTapRec = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(onDoubleTap:)];
|
||||
_doubleTapRec.delaysTouchesEnded = NO;
|
||||
_doubleTapRec.numberOfTapsRequired = 2;
|
||||
[self addGestureRecognizer:_doubleTapRec];
|
||||
}
|
||||
|
||||
- (UIButton *)newCornerButtonWithImage:(UIImage *)image left:(BOOL)left top:(BOOL)top {
|
||||
UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
|
||||
btn.hidden = YES;
|
||||
[self addSubview:btn];
|
||||
btn.translatesAutoresizingMaskIntoConstraints = NO;
|
||||
[btn setImage:image forState:UIControlStateNormal];
|
||||
[btn mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.width.height.mas_equalTo(24);
|
||||
make.centerX.equalTo(left ? self.mas_left : self.mas_right);
|
||||
make.centerY.equalTo(top ? self.mas_top : self.mas_bottom);
|
||||
}];
|
||||
return btn;
|
||||
}
|
||||
|
||||
- (void)layoutSubviews {
|
||||
[super layoutSubviews];
|
||||
[_delegate onStickerViewSizeChanged:self];
|
||||
}
|
||||
|
||||
- (void)updateRotation {
|
||||
// 规范化到[0,2pi)
|
||||
_rawRotation -= floor(_rawRotation / (M_PI * 2)) * M_PI * 2;
|
||||
BOOL adsorped = NO;
|
||||
// 吸附到 0,90,180,270度
|
||||
for (int i = 0; i <= 3; i++) {
|
||||
CGFloat diff = fabs(_rawRotation - i * M_PI_2);
|
||||
CGFloat diff2 = fabs(_rawRotation - M_PI * 2 - i * M_PI_2);
|
||||
if (diff < _rotateAdsorptionLimitAngle || diff2 < _rotateAdsorptionLimitAngle) {
|
||||
adsorped = YES;
|
||||
if (_rotation != i * M_PI_2) {
|
||||
_rotation = i * M_PI_2;
|
||||
[_impactGen impactOccurred];
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!adsorped) {
|
||||
_rotation = _rawRotation;
|
||||
}
|
||||
self.transform = CGAffineTransformMakeRotation(_rotation);
|
||||
NSLog(@"updateRotation %@",NSStringFromCGRect(self.frame));
|
||||
}
|
||||
|
||||
#pragma mark - Actions
|
||||
- (void)onTap {
|
||||
if (!self.selected) {
|
||||
self.selected = YES;
|
||||
[_delegate onStickerViewSelected:self];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)onDoubleTap:(UITapGestureRecognizer *)rec {
|
||||
[_delegate onStickerViewShouldEdit:self];
|
||||
}
|
||||
|
||||
- (void)onPan:(UIPanGestureRecognizer *)rec {
|
||||
CGPoint offset = [rec translationInView:self.superview];
|
||||
[rec setTranslation:CGPointZero inView:self.superview];
|
||||
// 平移
|
||||
self.center = Vec2AddVector(self.center, offset);
|
||||
NSLog(@"onPan %@",NSStringFromCGRect(self.frame));
|
||||
}
|
||||
|
||||
|
||||
- (void)onPinch:(UIPinchGestureRecognizer *)rec {
|
||||
// pow用于加快放大/缩小速度,优化体验
|
||||
CGFloat scale = pow(rec.scale, 1.5);
|
||||
rec.scale = 1;
|
||||
CGSize raw = self.bounds.size;
|
||||
self.bounds = CGRectMake(0, 0, raw.width * scale, raw.height * scale);
|
||||
NSLog(@"onPinch %@",NSStringFromCGRect(self.frame));
|
||||
}
|
||||
|
||||
- (void)scale:(CGFloat) scale {
|
||||
CGSize raw = self.bounds.size;
|
||||
self.bounds = CGRectMake(0, 0, raw.width * scale, raw.height * scale);
|
||||
self.center = CGPointMake(self.center.x * scale, self.center.y * scale);
|
||||
}
|
||||
|
||||
- (void)onGestureRotation:(UIRotationGestureRecognizer *)rec {
|
||||
_rawRotation += rec.rotation;
|
||||
[self updateRotation];
|
||||
rec.rotation = 0;
|
||||
}
|
||||
|
||||
- (void)onBtnTransformPan:(UIPanGestureRecognizer *)rec {
|
||||
CGPoint offset = [rec translationInView:self.superview];
|
||||
[rec setTranslation:CGPointZero inView:self.superview];
|
||||
|
||||
CGPoint originSize = CGPointMake(self.bounds.size.width, self.bounds.size.height);
|
||||
CGPoint originV = Vec2Rotate(Vec2Mul(originSize, 0.5), _rawRotation);
|
||||
CGPoint newV = Vec2AddVector(originV, offset);
|
||||
// 缩放
|
||||
CGPoint newSize = Vec2Mul(originSize, Vec2Len(newV) / Vec2Len(originV));
|
||||
self.bounds = CGRectMake(0, 0, newSize.x, newSize.y);
|
||||
// 旋转
|
||||
_rawRotation += Vec2Degree(originV, newV);
|
||||
[self updateRotation];
|
||||
}
|
||||
|
||||
- (BOOL)pointInside:(CGPoint)point withEvent:(UIEvent *)event {
|
||||
if (CGRectContainsPoint(self.bounds, point)) {
|
||||
return YES;
|
||||
}
|
||||
for (UIView *v in self.subviews) {
|
||||
CGPoint localPoint = [v convertPoint:point fromView:self];
|
||||
if (CGRectContainsPoint(v.bounds, localPoint)) {
|
||||
return YES;
|
||||
}
|
||||
}
|
||||
return NO;
|
||||
}
|
||||
|
||||
- (void)onBtnEditClicked {
|
||||
[_delegate onStickerViewShouldEdit:self];
|
||||
}
|
||||
|
||||
- (void)onBtnDeleteClicked {
|
||||
[_delegate onStickerViewShouldDelete:self];
|
||||
}
|
||||
|
||||
#pragma mark - Properties
|
||||
- (BOOL)editButtonHidden {
|
||||
return _hideEditButton;
|
||||
}
|
||||
|
||||
- (void)setEditButtonHidden:(BOOL)hideEditButton {
|
||||
_hideEditButton = hideEditButton;
|
||||
}
|
||||
|
||||
- (void)setContent:(UIView *)content {
|
||||
[_content removeFromSuperview];
|
||||
_content = content;
|
||||
if (content == nil) {
|
||||
return;
|
||||
}
|
||||
self.bounds = CGRectMake(0, 0, content.bounds.size.width + _contentMargin * 2, content.bounds.size.height + _contentMargin * 2);
|
||||
|
||||
|
||||
|
||||
// [self mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
// make.center.equalTo(self.superview);
|
||||
// }];
|
||||
|
||||
|
||||
[self addSubview:content];
|
||||
[self bringSubviewToFront:_btnEdit];
|
||||
[self bringSubviewToFront:_btnTransform];
|
||||
[self bringSubviewToFront:_btnDelete];
|
||||
[content mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self).inset(_contentMargin);
|
||||
}];
|
||||
|
||||
|
||||
}
|
||||
|
||||
- (void)setSelected:(BOOL)selected {
|
||||
_selected = selected;
|
||||
_borderView.hidden = !selected;
|
||||
_btnEdit.hidden = !selected || _hideEditButton;
|
||||
_btnTransform.hidden = !selected;
|
||||
_btnDelete.hidden = !selected;
|
||||
}
|
||||
|
||||
#pragma mark - UIGestureRecognizerDelegate protocol
|
||||
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)g1 shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)g2 {
|
||||
NSArray<UIGestureRecognizer *> *recList = @[ _pinchRec, _rotationRec, _panRec ];
|
||||
return [recList containsObject:g1] && [recList containsObject:g2];
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,14 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaStickerView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSubtitleInfo.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface TUIMultimediaSubtitleView : TUIMultimediaStickerView
|
||||
@property(nullable, nonatomic) TUIMultimediaSubtitleInfo *subtitleInfo;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,123 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaSubtitleView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGeometry.h"
|
||||
|
||||
#define MIN_SUBTITLE_PASTER_WIDTH 100
|
||||
#define MAX_SUBTITLE_PASTER_HEIGH 400
|
||||
|
||||
@interface TUIMultimediaSubtitleView () <TUIMultimediaStickerViewDelegate> {
|
||||
id<TUIMultimediaStickerViewDelegate> _outerDelegate;
|
||||
UILabel *_label;
|
||||
BOOL _isSizeToFitingSubtitleInfo;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaSubtitleView
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
[self initUI];
|
||||
}
|
||||
_isSizeToFitingSubtitleInfo = NO;
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
[super setDelegate:self];
|
||||
_label = [[UILabel alloc] init];
|
||||
_label.font = [UIFont systemFontOfSize:20];
|
||||
_label.numberOfLines = 0;
|
||||
_label.lineBreakMode = NSLineBreakByClipping;
|
||||
_label.translatesAutoresizingMaskIntoConstraints = NO;
|
||||
}
|
||||
|
||||
#pragma mark - Getters & Setters
|
||||
- (void)setSubtitleInfo:(TUIMultimediaSubtitleInfo *)subtitleInfo {
|
||||
NSLog(@"TUIMultimediaSubtitleView setSubtitleInfo text = %@",subtitleInfo.wrappedText);
|
||||
_subtitleInfo = subtitleInfo;
|
||||
|
||||
self.content = nil;
|
||||
_label.text = subtitleInfo.wrappedText;
|
||||
_label.textColor = subtitleInfo.color;
|
||||
[self labeSizeToFit];
|
||||
self.content = _label;
|
||||
}
|
||||
|
||||
-(void)labeSizeToFit {
|
||||
_isSizeToFitingSubtitleInfo = YES;
|
||||
if (_subtitleInfo.wrappedText.length > 0) {
|
||||
_label.text = [NSString stringWithFormat:@"%C", [_subtitleInfo.wrappedText characterAtIndex:0]];
|
||||
} else {
|
||||
NSLog(@"_subtitleInfo.wrappedText is empty or nil.");
|
||||
_label.text = @"";
|
||||
}
|
||||
CGSize size = [_label sizeThatFits:CGSizeMake(MAXFLOAT, MAX_SUBTITLE_PASTER_HEIGH)];
|
||||
CGFloat singleLineTextHeight = size.height;
|
||||
|
||||
_label.text = _subtitleInfo.wrappedText;
|
||||
size = [_label sizeThatFits:CGSizeMake(MAXFLOAT, MAX_SUBTITLE_PASTER_HEIGH)];
|
||||
CGFloat adjustedWidth = size.width;
|
||||
if (size.height > singleLineTextHeight * 1.5) {
|
||||
// multi line text
|
||||
adjustedWidth = MAX(size.width, MIN_SUBTITLE_PASTER_WIDTH);
|
||||
}
|
||||
CGFloat adjustedHeight = MIN(size.height, MAX_SUBTITLE_PASTER_HEIGH);
|
||||
_label.frame = CGRectMake(_label.frame.origin.x, _label.frame.origin.y, adjustedWidth, adjustedHeight);
|
||||
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
|
||||
self->_isSizeToFitingSubtitleInfo = NO;
|
||||
});
|
||||
}
|
||||
|
||||
- (id<TUIMultimediaStickerViewDelegate>)delegate {
|
||||
return _outerDelegate;
|
||||
}
|
||||
|
||||
- (void)setDelegate:(id<TUIMultimediaStickerViewDelegate>)delegate {
|
||||
_outerDelegate = delegate;
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaFloatingResizableDelegate protocol
|
||||
- (void)onStickerViewShouldDelete:(TUIMultimediaStickerView *)v {
|
||||
[_outerDelegate onStickerViewShouldDelete:self];
|
||||
}
|
||||
|
||||
- (void)onStickerViewShouldEdit:(TUIMultimediaStickerView *)v {
|
||||
[_outerDelegate onStickerViewShouldEdit:self];
|
||||
}
|
||||
|
||||
- (void)onStickerViewSelected:(TUIMultimediaStickerView *)v {
|
||||
[_outerDelegate onStickerViewSelected:self];
|
||||
}
|
||||
|
||||
- (void)onStickerViewSizeChanged:(TUIMultimediaStickerView *)v {
|
||||
[_outerDelegate onStickerViewSizeChanged:self];
|
||||
if (_isSizeToFitingSubtitleInfo == NO) {
|
||||
[self adjustFontSize];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)adjustFontSize {
|
||||
NSString *text = _label.text;
|
||||
CGSize labelSize = _label.bounds.size;
|
||||
UIFont *font = _label.font;
|
||||
CGFloat minFontSize = 0;
|
||||
CGFloat maxFontSize = 1000;
|
||||
const CGFloat eps = 0.1;
|
||||
// 理论上需要14次二分
|
||||
while (maxFontSize - minFontSize > eps) {
|
||||
CGFloat mid = (maxFontSize + minFontSize) / 2;
|
||||
font = [font fontWithSize:mid];
|
||||
CGSize size = [text sizeWithAttributes:@{NSFontAttributeName : font}];
|
||||
if (size.width < labelSize.width && size.height < labelSize.height) {
|
||||
minFontSize = mid;
|
||||
} else {
|
||||
maxFontSize = mid;
|
||||
}
|
||||
}
|
||||
_label.font = [_label.font fontWithSize:minFontSize];
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,23 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaColorPanelDelegate;
|
||||
|
||||
/**
|
||||
条形颜色选择器
|
||||
*/
|
||||
@interface TUIMultimediaColorPanel : UIView
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaColorPanelDelegate> delegate;
|
||||
@property(nonatomic) NSArray<UIColor *> *colorList;
|
||||
@property(nonatomic) UIColor *selectedColor;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaColorPanelDelegate <NSObject>
|
||||
- (void)onColorPanel:(TUIMultimediaColorPanel *)panel selectColor:(UIColor *)color;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,120 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaColorPanel.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaColorCell.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGeometry.h"
|
||||
|
||||
const static CGFloat CellMargin = 4;
|
||||
|
||||
@interface TUIMultimediaColorPanel () <UICollectionViewDelegateFlowLayout, UICollectionViewDataSource> {
|
||||
UICollectionView *_collectionView;
|
||||
NSInteger _selectIndex;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaColorPanel
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
_colorList = @[
|
||||
UIColor.whiteColor,
|
||||
UIColor.blackColor,
|
||||
UIColor.grayColor,
|
||||
UIColor.redColor,
|
||||
UIColor.greenColor,
|
||||
UIColor.blueColor,
|
||||
UIColor.cyanColor,
|
||||
UIColor.yellowColor,
|
||||
UIColor.magentaColor,
|
||||
UIColor.orangeColor,
|
||||
UIColor.purpleColor,
|
||||
UIColor.brownColor,
|
||||
];
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
UICollectionViewFlowLayout *layout = [[UICollectionViewFlowLayout alloc] init];
|
||||
layout.scrollDirection = UICollectionViewScrollDirectionHorizontal;
|
||||
layout.minimumInteritemSpacing = 0;
|
||||
layout.minimumLineSpacing = 2;
|
||||
|
||||
_collectionView = [[UICollectionView alloc] initWithFrame:CGRectZero collectionViewLayout:layout];
|
||||
_collectionView.showsHorizontalScrollIndicator = NO;
|
||||
_collectionView.backgroundColor = UIColor.clearColor;
|
||||
_collectionView.delegate = self;
|
||||
_collectionView.dataSource = self;
|
||||
[_collectionView registerClass:TUIMultimediaColorCell.class forCellWithReuseIdentifier:TUIMultimediaColorCell.reuseIdentifier];
|
||||
[self addSubview:_collectionView];
|
||||
|
||||
[_collectionView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
}
|
||||
|
||||
#pragma mark - UICollectionViewDataSource protocol
|
||||
- (nonnull __kindof UICollectionViewCell *)collectionView:(nonnull UICollectionView *)collectionView cellForItemAtIndexPath:(nonnull NSIndexPath *)indexPath {
|
||||
TUIMultimediaColorCell *cell = (TUIMultimediaColorCell *)[collectionView dequeueReusableCellWithReuseIdentifier:TUIMultimediaColorCell.reuseIdentifier forIndexPath:indexPath];
|
||||
cell.color = _colorList[indexPath.item];
|
||||
cell.colorSelected = indexPath.item == _selectIndex;
|
||||
return cell;
|
||||
}
|
||||
|
||||
- (NSInteger)collectionView:(nonnull UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section {
|
||||
return _colorList.count;
|
||||
}
|
||||
|
||||
#pragma mark - UICollectionViewDelegateFlowLayout protocol
|
||||
- (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath {
|
||||
[collectionView deselectItemAtIndexPath:indexPath animated:NO];
|
||||
_selectIndex = indexPath.item;
|
||||
[_collectionView reloadData];
|
||||
|
||||
UIColor *c = _colorList[indexPath.item];
|
||||
[_delegate onColorPanel:self selectColor:c];
|
||||
}
|
||||
- (BOOL)collectionView:(UICollectionView *)collectionView shouldSelectItemAtIndexPath:(NSIndexPath *)indexPath {
|
||||
return YES;
|
||||
}
|
||||
- (CGSize)collectionView:(UICollectionView *)collectionView layout:(UICollectionViewLayout *)collectionViewLayout sizeForItemAtIndexPath:(NSIndexPath *)index {
|
||||
CGFloat len = collectionView.bounds.size.height - CellMargin;
|
||||
return CGSizeMake(len, len);
|
||||
}
|
||||
|
||||
#pragma mark - Setters
|
||||
- (void)setColorList:(NSArray<UIColor *> *)colorList {
|
||||
_colorList = colorList;
|
||||
_selectIndex = 0;
|
||||
if (_colorList.count > 0) {
|
||||
[_collectionView cellForItemAtIndexPath:[NSIndexPath indexPathWithIndex:0]].selected = YES;
|
||||
}
|
||||
[_collectionView reloadData];
|
||||
}
|
||||
|
||||
- (UIColor *)selectedColor {
|
||||
if (_selectIndex < 0 || _selectIndex >= _colorList.count) {
|
||||
return nil;
|
||||
}
|
||||
return _colorList[_selectIndex];
|
||||
}
|
||||
|
||||
- (void)setSelectedColor:(UIColor *)selectedColor {
|
||||
if (selectedColor == nil) {
|
||||
_selectIndex = -1;
|
||||
} else {
|
||||
for (int i = 0; i < _colorList.count; i++) {
|
||||
if ([_colorList[i] isEqual:selectedColor]) {
|
||||
_selectIndex = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
[_collectionView reloadData];
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,66 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#include "TUIMultimediaPlugin/TUIMultimediaPasterConfig.h"
|
||||
#include "TUIMultimediaPlugin/TUIMultimediaSticker.h"
|
||||
#include "TUIMultimediaPlugin/TUIMultimediaSubtitleInfo.h"
|
||||
#include "TUIMultimediaStickerView.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaCommonEditorControlViewDelegate;
|
||||
@class TUIMultimediaCommonEditorConfig;
|
||||
|
||||
/**
|
||||
编辑页面View
|
||||
*/
|
||||
@interface TUIMultimediaCommonEditorControlView : UIView
|
||||
@property(readonly, nonatomic) UIView *previewView;
|
||||
@property(nonatomic) BOOL isGenerating;
|
||||
@property(nonatomic) CGFloat progressBarProgress;
|
||||
@property(nonatomic) BOOL musicEdited;
|
||||
@property(nonatomic) CGSize previewSize;
|
||||
@property(nonatomic) BOOL modifyButtonsHidden;
|
||||
@property(nonatomic) BOOL isStartCrop;
|
||||
@property(nonatomic) int sourceType;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaCommonEditorControlViewDelegate> delegate;
|
||||
@property (nonatomic, strong) UIImage *mosaciOriginalImage;
|
||||
@property(nonatomic) CGRect previewLimitRect;
|
||||
|
||||
- (instancetype)initWithConfig:(TUIMultimediaCommonEditorConfig *)config;
|
||||
|
||||
- (void)addPaster:(UIImage *)paster;
|
||||
- (void)addSubtitle:(TUIMultimediaSubtitleInfo *)subtitle;
|
||||
|
||||
// crop相关
|
||||
- (void)previewScale:(CGFloat)scale center:(CGPoint)center;
|
||||
- (void)previewMove:(CGPoint)offset;
|
||||
- (void)previewRotation90:(CGPoint)center;
|
||||
- (void)previewRotationToZero;
|
||||
- (void)previewAdjustToLimitRect;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaCommonEditorControlViewDelegate <NSObject>
|
||||
- (void)onCommonEditorControlViewComplete:(TUIMultimediaCommonEditorControlView *)view stickers:(NSArray<TUIMultimediaSticker *> *)stickers;
|
||||
- (void)onCommonEditorControlViewCrop:(NSInteger)rotationAngle normalizedCropRect:(CGRect)normalizedCropRect
|
||||
stickers:(NSArray<TUIMultimediaSticker *> *)stickers;
|
||||
- (void)onCommonEditorControlViewCancel:(TUIMultimediaCommonEditorControlView *)view;
|
||||
- (void)onCommonEditorControlViewNeedAddPaster:(TUIMultimediaCommonEditorControlView *)view;
|
||||
- (void)onCommonEditorControlViewNeedModifySubtitle:(TUIMultimediaSubtitleInfo *)info callback:(void (^)(TUIMultimediaSubtitleInfo *newInfo, BOOL isOk))callback;
|
||||
- (void)onCommonEditorControlViewNeedEditMusic:(TUIMultimediaCommonEditorControlView *)view;
|
||||
- (void)onCommonEditorControlViewCancelGenerate:(TUIMultimediaCommonEditorControlView *)view;
|
||||
@end
|
||||
|
||||
@interface TUIMultimediaCommonEditorConfig : NSObject
|
||||
@property(nonatomic) BOOL pasterEnabled;
|
||||
@property(nonatomic) BOOL subtitleEnabled;
|
||||
@property(nonatomic) BOOL drawGraffitiEnabled;
|
||||
@property(nonatomic) BOOL drawMosaicEnabled;
|
||||
@property(nonatomic) BOOL musicEditEnabled;
|
||||
@property(nonatomic) BOOL cropEnabled;
|
||||
+ (instancetype)configForVideoEditor;
|
||||
+ (instancetype)configForPictureEditor;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,701 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaCommonEditorControlView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCircleProgressView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaDrawView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImageUtil.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterSelectView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSplitter.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaStickerView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSubtitleView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaTabPanel.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaConstant.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGeometry.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCropControlView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaDrawCtrlView.h"
|
||||
|
||||
#define FUNCTION_BUTTON_SIZE CGSizeMake(28, 28)
|
||||
#define BUTTON_SEND_SIZE CGSizeMake(60, 28)
|
||||
|
||||
@interface TUIMultimediaCommonEditorControlView () <TUIMultimediaStickerViewDelegate, UIGestureRecognizerDelegate, TUIMultimediaDrawCtrlViewDelegate, TUIMultimediaCropControlDelegate> {
|
||||
TUIMultimediaCommonEditorConfig *_config;
|
||||
UIStackView *_stkViewButtons;
|
||||
UIButton *_btnDrawGraffiti;
|
||||
UIButton *_btnDrawMosaic;
|
||||
UIButton *_btnMusic;
|
||||
UIButton *_btnSubtitle;
|
||||
UIButton *_btnPaster;
|
||||
UIButton *_btnCrop;
|
||||
UIButton *_btnSend;
|
||||
UIButton *_btnCancel;
|
||||
|
||||
NSMutableArray<TUIMultimediaStickerView *> *_stickerViewList;
|
||||
TUIMultimediaStickerView *_lastSelectedStickerView;
|
||||
UIView *_editContainerView;
|
||||
|
||||
TUIMultimediaDrawView *_drawView;
|
||||
TUIMultimediaDrawCtrlView *_drawCtrlView;
|
||||
|
||||
UIView *_generateView;
|
||||
TUIMultimediaCircleProgressView *_progressView;
|
||||
UIButton *_btnGenerateCancel;
|
||||
|
||||
BOOL _isStartCrop;
|
||||
NSInteger _previewRotationAngle;
|
||||
TUIMultimediaCropControlView* _cropControlView;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaCommonEditorControlView
|
||||
- (instancetype)initWithFrame:(CGRect)frame {
|
||||
return [self initWithConfig:[[TUIMultimediaCommonEditorConfig alloc] init]];
|
||||
}
|
||||
- (instancetype)initWithConfig:(TUIMultimediaCommonEditorConfig *)config {
|
||||
self = [super initWithFrame:CGRectZero];
|
||||
if (self != nil) {
|
||||
_config = config;
|
||||
_sourceType = SOURCE_TYPE_RECORD;
|
||||
_isStartCrop = NO;
|
||||
_stickerViewList = [NSMutableArray array];
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)addPaster:(UIImage *)paster {
|
||||
TUIMultimediaPasterView *vpaster = [[TUIMultimediaPasterView alloc] init];
|
||||
vpaster.delegate = self;
|
||||
[_stickerViewList addObject:vpaster];
|
||||
vpaster.paster = paster;
|
||||
vpaster.center = CGPointMake(_editContainerView.bounds.size.width / 2, _editContainerView.bounds.size.height / 2);
|
||||
[_editContainerView addSubview:vpaster];
|
||||
}
|
||||
|
||||
- (void)addSubtitle:(TUIMultimediaSubtitleInfo *)subtitle {
|
||||
TUIMultimediaSubtitleView *vsub = [[TUIMultimediaSubtitleView alloc] init];
|
||||
vsub.delegate = self;
|
||||
[_stickerViewList addObject:vsub];
|
||||
vsub.subtitleInfo = subtitle;
|
||||
vsub.center = CGPointMake(_editContainerView.bounds.size.width / 2, _editContainerView.bounds.size.height / 2);
|
||||
[_editContainerView addSubview:vsub];
|
||||
}
|
||||
|
||||
#pragma mark crop function
|
||||
- (void)previewScale:(CGFloat) scale center:(CGPoint)center {
|
||||
if (_isStartCrop) {
|
||||
return;
|
||||
}
|
||||
|
||||
CGFloat width = _previewView.frame.size.width * scale;
|
||||
CGFloat height = _previewView.frame.size.height * scale;
|
||||
|
||||
CGFloat left = center.x - (center.x - _previewView.frame.origin.x) * scale;
|
||||
CGFloat top = center.y - (center.y - _previewView.frame.origin.y) * scale;
|
||||
_previewView.frame = CGRectMake(left, top, width, height);
|
||||
|
||||
for (UIView *subview in _editContainerView.subviews) {
|
||||
if ([subview isKindOfClass:[TUIMultimediaStickerView class]]) {
|
||||
TUIMultimediaStickerView* stickView = (TUIMultimediaStickerView*)subview;
|
||||
[stickView scale:scale];
|
||||
}
|
||||
}
|
||||
|
||||
if (_cropControlView != nil) {
|
||||
[_cropControlView changeResetButtonStatus];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)previewMove:(CGPoint)offset {
|
||||
if (_isStartCrop) {
|
||||
return;
|
||||
}
|
||||
_previewView.frame = CGRectMake(_previewView.frame.origin.x + offset.x,
|
||||
_previewView.frame.origin.y + offset.y,
|
||||
_previewView.frame.size.width,
|
||||
_previewView.frame.size.height);
|
||||
if (_cropControlView != nil) {
|
||||
[_cropControlView changeResetButtonStatus];
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
- (void)previewRotation90:(CGPoint)center {
|
||||
_previewRotationAngle = (_previewRotationAngle + 90 + 360) % 360;
|
||||
float radians = _previewRotationAngle * M_PI / 180;
|
||||
int newTop = center.y - center.x + _previewView.frame.origin.x;
|
||||
int newLeft = center.x + center.y - _previewView.frame.origin.y - _previewView.frame.size.height;
|
||||
|
||||
_editContainerView.transform = CGAffineTransformMakeRotation(radians);
|
||||
_previewView.transform = CGAffineTransformMakeRotation(radians);
|
||||
_previewView.frame = CGRectMake(newLeft, newTop, _previewView.frame.size.width, _previewView.frame.size.height);
|
||||
|
||||
if (_cropControlView != nil) {
|
||||
[_cropControlView changeResetButtonStatus];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)previewRotationToZero {
|
||||
_previewRotationAngle = 0;
|
||||
_editContainerView.transform = CGAffineTransformMakeRotation(0);
|
||||
_previewView.transform = CGAffineTransformMakeRotation(0);
|
||||
}
|
||||
|
||||
- (void)previewAdjustToLimitRect {
|
||||
NSLog(@"preview adjust to limit rect");
|
||||
double scaleLimitX = _previewLimitRect.size.width * 1.0f / _previewView.frame.size.width;
|
||||
double scaleLimitY = _previewLimitRect.size.height * 1.0f / _previewView.frame.size.height;
|
||||
double scaleLimit = MAX(scaleLimitX, scaleLimitY);
|
||||
if (scaleLimit > 1.0f) {
|
||||
[self previewScale:scaleLimit center:CGPointMake(CGRectGetMidX(_previewView.frame), CGRectGetMidY(_previewView.frame))];
|
||||
}
|
||||
|
||||
|
||||
int maxLeft = _previewLimitRect.origin.x;
|
||||
int minLeft = _previewLimitRect.size.width + _previewLimitRect.origin.x - _previewView.frame.size.width;
|
||||
int left = MAX(MIN(_previewView.frame.origin.x , maxLeft), minLeft);
|
||||
int maxTop = _previewLimitRect.origin.y;
|
||||
int minTop = _previewLimitRect.size.height + _previewLimitRect.origin.y - _previewView.frame.size.height;
|
||||
int top = MAX(MIN(_previewView.frame.origin.y , maxTop), minTop);
|
||||
_previewView.frame = CGRectMake(left, top, _previewView.frame.size.width, _previewView.frame.size.height);
|
||||
}
|
||||
|
||||
#pragma mark UI init
|
||||
- (void)initUI {
|
||||
self.backgroundColor = UIColor.blackColor;
|
||||
|
||||
_previewView = [[UIView alloc] init];
|
||||
[self addSubview:_previewView];
|
||||
|
||||
[self initFuncitonBtnStackView];
|
||||
[self initGraffitiFunction];
|
||||
[self initMosaicFunction];
|
||||
[self initPasterFunciton];
|
||||
[self initSubtitleFunciton];
|
||||
[self initBGMFunction];
|
||||
[self initCropFunciton];
|
||||
[self initSendAndCancelBtn];
|
||||
[self initGenerateView];
|
||||
[self bringSubviewToFront:_stkViewButtons];
|
||||
}
|
||||
|
||||
- (void) initFuncitonBtnStackView {
|
||||
_stkViewButtons = [[UIStackView alloc] init];
|
||||
_stkViewButtons.translatesAutoresizingMaskIntoConstraints = NO;
|
||||
[self addSubview:_stkViewButtons];
|
||||
_stkViewButtons.axis = UILayoutConstraintAxisHorizontal;
|
||||
_stkViewButtons.alignment = UIStackViewAlignmentCenter;
|
||||
_stkViewButtons.distribution = UIStackViewDistributionEqualSpacing;
|
||||
[_stkViewButtons mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.left.right.equalTo(self).inset(10);
|
||||
make.bottom.equalTo(self.mas_safeAreaLayoutGuideBottom);
|
||||
make.height.mas_equalTo(FUNCTION_BUTTON_SIZE.height);
|
||||
}];
|
||||
}
|
||||
|
||||
- (void) initGraffitiFunction {
|
||||
if (!_config.drawGraffitiEnabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
_btnDrawGraffiti = [self addFunctionIconButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_scrawl_img", @"modify_scrawl")
|
||||
onTouchUpInside:@selector(onBtnDrawGraffitiClicked)];
|
||||
|
||||
[self initEditContainerView];
|
||||
[self initDrawCtrlView];
|
||||
}
|
||||
|
||||
- (void) initMosaicFunction {
|
||||
if (!_config.drawMosaicEnabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
_btnDrawMosaic = [self addFunctionIconButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"edit_mosaic_img", @"edit_mosaic")
|
||||
onTouchUpInside:@selector(onBtnDrawMosaicClicked)];
|
||||
|
||||
[self initEditContainerView];
|
||||
[self initDrawCtrlView];
|
||||
_drawCtrlView.drawView.mosaciOriginalImage = _mosaciOriginalImage;
|
||||
}
|
||||
|
||||
- (void) initPasterFunciton {
|
||||
if (!_config.pasterEnabled) {
|
||||
return;
|
||||
}
|
||||
_btnPaster = [self addFunctionIconButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_paster_img", @"modify_paster")
|
||||
onTouchUpInside:@selector(onBtnPasterClicked)];
|
||||
[self initEditContainerView];
|
||||
}
|
||||
|
||||
- (void) initSubtitleFunciton {
|
||||
if (!_config.subtitleEnabled) {
|
||||
return;
|
||||
}
|
||||
_btnSubtitle = [self addFunctionIconButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_subtitle_img", @"modify_subtitle")
|
||||
onTouchUpInside:@selector(onBtnSubtitleClicked)];
|
||||
[self initEditContainerView];
|
||||
}
|
||||
|
||||
- (void) initBGMFunction {
|
||||
if (!_config.musicEditEnabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
_btnMusic = [self addFunctionIconButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_music_img", @"modify_music")
|
||||
onTouchUpInside:@selector(onBtnMusicClicked)];
|
||||
}
|
||||
|
||||
- (void) initCropFunciton {
|
||||
if (!_config.cropEnabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
_btnCrop = [self addFunctionIconButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_crop_img", @"modify_crop")
|
||||
onTouchUpInside:@selector(onBtnCropClicked)];
|
||||
[self initCropCtrlView];
|
||||
}
|
||||
|
||||
- (void)initCropCtrlView {
|
||||
if (_cropControlView != nil) {
|
||||
return;
|
||||
}
|
||||
|
||||
_cropControlView = [[TUIMultimediaCropControlView alloc] initWithFrame:self.frame editorControl:self];
|
||||
[self addSubview:_cropControlView];
|
||||
_cropControlView.hidden = YES;
|
||||
[_cropControlView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
_cropControlView.delegate = self;
|
||||
}
|
||||
|
||||
- (void)initDrawCtrlView {
|
||||
if (_drawCtrlView != nil) {
|
||||
return;
|
||||
}
|
||||
|
||||
_drawCtrlView = [[TUIMultimediaDrawCtrlView alloc] init];
|
||||
[self addSubview:_drawCtrlView];
|
||||
[_drawCtrlView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.left.right.equalTo(self);
|
||||
make.top.equalTo(_stkViewButtons).offset(-50);
|
||||
make.bottom.equalTo(self);
|
||||
}];
|
||||
_drawCtrlView.drawEnable = NO;
|
||||
_drawCtrlView.delegate = self;
|
||||
|
||||
[_editContainerView addSubview:_drawCtrlView.drawView];
|
||||
[_drawCtrlView.drawView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(_previewView);
|
||||
}];
|
||||
}
|
||||
|
||||
- (void) initEditContainerView {
|
||||
if (_editContainerView != nil) {
|
||||
return;
|
||||
}
|
||||
|
||||
_editContainerView = [[UIView alloc] init];
|
||||
[self addSubview:_editContainerView];
|
||||
_editContainerView.userInteractionEnabled = YES;
|
||||
_editContainerView.clipsToBounds = YES;
|
||||
|
||||
UITapGestureRecognizer *containerTapRec = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(onTapContainerView:)];
|
||||
containerTapRec.cancelsTouchesInView = NO;
|
||||
[_editContainerView addGestureRecognizer:containerTapRec];
|
||||
[_editContainerView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(_previewView);
|
||||
}];
|
||||
}
|
||||
|
||||
- (void) initSendAndCancelBtn {
|
||||
_btnSend = [UIButton buttonWithType:UIButtonTypeSystem];
|
||||
|
||||
if (_stkViewButtons.arrangedSubviews.count == 0) {
|
||||
[_stkViewButtons removeFromSuperview];
|
||||
[self addSubview:_btnSend];
|
||||
[_btnSend mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.size.mas_equalTo(BUTTON_SEND_SIZE);
|
||||
make.right.equalTo(self).inset(20);
|
||||
make.bottom.equalTo(self.mas_safeAreaLayoutGuideBottom);
|
||||
}];
|
||||
} else {
|
||||
[_stkViewButtons addArrangedSubview:_btnSend];
|
||||
[_btnSend mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.size.mas_equalTo(BUTTON_SEND_SIZE);
|
||||
}];
|
||||
}
|
||||
|
||||
_btnSend.backgroundColor = [[TUIMultimediaConfig sharedInstance] getThemeColor];
|
||||
[_btnSend setTitleColor:UIColor.whiteColor forState:UIControlStateNormal];
|
||||
_btnSend.layer.cornerRadius = 5;
|
||||
NSString* titile = [TUIMultimediaCommon localizedStringForKey:@"send"];
|
||||
if (_sourceType == SOURCE_TYPE_ALBUM) {
|
||||
titile = [TUIMultimediaCommon localizedStringForKey:@"done"];
|
||||
}
|
||||
[_btnSend setTitle:titile forState:UIControlStateNormal];
|
||||
[_btnSend addTarget:self action:@selector(onBtnSendClicked) forControlEvents:UIControlEventTouchUpInside];
|
||||
|
||||
|
||||
_btnCancel = [UIButton buttonWithType:UIButtonTypeCustom];
|
||||
[self addSubview:_btnCancel];
|
||||
[_btnCancel setImage:TUIMultimediaPluginBundleThemeImage(@"editor_cancel_img", @"return_arrow") forState:UIControlStateNormal];
|
||||
[_btnCancel addTarget:self action:@selector(onBtnCancelClicked) forControlEvents:UIControlEventTouchUpInside];
|
||||
[_btnCancel mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.size.mas_equalTo(CGSizeMake(24, 24));
|
||||
make.top.equalTo(self.mas_safeAreaLayoutGuideTop).inset(10);
|
||||
if([[TUIGlobalization getPreferredLanguage] hasPrefix:@"ar"]) {
|
||||
make.right.equalTo(self.mas_safeAreaLayoutGuideRight).offset(-15);
|
||||
} else {
|
||||
make.left.equalTo(self.mas_safeAreaLayoutGuideLeft).offset(15);
|
||||
}
|
||||
}];
|
||||
}
|
||||
|
||||
- (void)setSourceType:(int)sourceType {
|
||||
NSLog(@"setSourceType sourceType = %d",sourceType);
|
||||
_sourceType = sourceType;
|
||||
NSString* titile = [TUIMultimediaCommon localizedStringForKey:@"send"];
|
||||
if (_sourceType == SOURCE_TYPE_ALBUM) {
|
||||
titile = [TUIMultimediaCommon localizedStringForKey:@"done"];
|
||||
}
|
||||
[_btnSend setTitle:titile forState:UIControlStateNormal];
|
||||
}
|
||||
|
||||
- (void)initGenerateView {
|
||||
_generateView = [[UIView alloc] init];
|
||||
_generateView.backgroundColor = TUIMultimediaPluginDynamicColor(@"editor_generate_view_bg_color", @"#0000007F");
|
||||
_generateView.hidden = YES;
|
||||
[self addSubview:_generateView];
|
||||
|
||||
_progressView = [[TUIMultimediaCircleProgressView alloc] init];
|
||||
_progressView.progressColor = [[TUIMultimediaConfig sharedInstance] getThemeColor];
|
||||
[_generateView addSubview:_progressView];
|
||||
|
||||
_btnGenerateCancel = [UIButton buttonWithType:UIButtonTypeSystem];
|
||||
[_btnGenerateCancel addTarget:self action:@selector(onBtnGenerateCancelClicked) forControlEvents:UIControlEventTouchUpInside];
|
||||
[_btnGenerateCancel setTitle:[TUIMultimediaCommon localizedStringForKey:@"cancel"] forState:UIControlStateNormal];
|
||||
[_btnGenerateCancel setTitleColor:UIColor.whiteColor forState:UIControlStateNormal];
|
||||
[_generateView addSubview:_btnGenerateCancel];
|
||||
|
||||
[_generateView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
[_progressView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.size.mas_equalTo(CGSizeMake(100, 100));
|
||||
make.center.equalTo(_generateView);
|
||||
}];
|
||||
[_btnGenerateCancel mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.top.equalTo(self.mas_safeAreaLayoutGuideTop).inset(10);
|
||||
make.left.equalTo(self).inset(10);
|
||||
make.size.mas_equalTo(CGSizeMake(54, 32));
|
||||
}];
|
||||
}
|
||||
|
||||
- (UIButton *)addFunctionIconButtonWithImage:(UIImage *)img onTouchUpInside:(SEL)sel {
|
||||
UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
|
||||
[_stkViewButtons addArrangedSubview:btn];
|
||||
UIImage *imgNormal = [TUIMultimediaImageUtil imageFromImage:img withTintColor:
|
||||
TUIMultimediaPluginDynamicColor(@"editor_func_btn_normal_color", @"#FFFFFF")];
|
||||
UIImage *imgDisabled = [TUIMultimediaImageUtil imageFromImage:img withTintColor:
|
||||
TUIMultimediaPluginDynamicColor(@"editor_func_btn_disabled_color", @"#6D6D6D")];
|
||||
UIImage *imgPressed = [TUIMultimediaImageUtil imageFromImage:img withTintColor:
|
||||
TUIMultimediaPluginDynamicColor(@"editor_func_btn_pressed_color", @"#7F7F7F")];
|
||||
UIImage *imgSelected = [TUIMultimediaImageUtil imageFromImage:img withTintColor:
|
||||
[[TUIMultimediaConfig sharedInstance] getThemeColor]];
|
||||
[btn setImage:imgNormal forState:UIControlStateNormal];
|
||||
[btn setImage:imgDisabled forState:UIControlStateDisabled];
|
||||
[btn setImage:imgPressed forState:UIControlStateHighlighted];
|
||||
[btn setImage:imgSelected forState:UIControlStateSelected];
|
||||
[btn setImageEdgeInsets:UIEdgeInsetsMake(0, 0, 0, 0)];
|
||||
[btn addTarget:self action:sel forControlEvents:UIControlEventTouchUpInside];
|
||||
|
||||
[btn mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.size.mas_equalTo(FUNCTION_BUTTON_SIZE);
|
||||
}];
|
||||
return btn;
|
||||
}
|
||||
|
||||
#pragma mark Internal Functions
|
||||
- (void)setDrawEnabled:(BOOL)drawGraffitiEnabled {
|
||||
if (!_config.drawGraffitiEnabled) {
|
||||
return;
|
||||
}
|
||||
_lastSelectedStickerView.selected = NO;
|
||||
_lastSelectedStickerView = nil;
|
||||
_btnDrawGraffiti.selected = drawGraffitiEnabled;
|
||||
if(drawGraffitiEnabled) {
|
||||
_btnDrawMosaic.selected = NO;
|
||||
}
|
||||
_drawCtrlView.drawMode = GRAFFITI;
|
||||
_drawCtrlView.drawEnable = drawGraffitiEnabled;
|
||||
for (TUIMultimediaStickerView *v in _stickerViewList) {
|
||||
v.userInteractionEnabled = !drawGraffitiEnabled;
|
||||
}
|
||||
}
|
||||
|
||||
- (void)setDrawMosaicEnabled:(BOOL)drawMosaicEnabled {
|
||||
if (!_config.drawMosaicEnabled) {
|
||||
return;
|
||||
}
|
||||
_lastSelectedStickerView.selected = NO;
|
||||
_lastSelectedStickerView = nil;
|
||||
_btnDrawMosaic.selected = drawMosaicEnabled;
|
||||
if(drawMosaicEnabled) {
|
||||
_btnDrawGraffiti.selected = NO;
|
||||
}
|
||||
_drawCtrlView.drawMode = MOSAIC;
|
||||
_drawCtrlView.drawEnable = drawMosaicEnabled;
|
||||
for (TUIMultimediaStickerView *v in _stickerViewList) {
|
||||
v.userInteractionEnabled = !drawMosaicEnabled;
|
||||
}
|
||||
}
|
||||
|
||||
- (NSArray<TUIMultimediaSticker *> *)getStickers {
|
||||
NSArray<TUIMultimediaSticker *> *stickers = [_stickerViewList tui_multimedia_map:^TUIMultimediaSticker *(TUIMultimediaStickerView *v) {
|
||||
[v resignFirstResponder];
|
||||
TUIMultimediaSticker *sticker = [[TUIMultimediaSticker alloc] init];
|
||||
sticker.image = [TUIMultimediaImageUtil imageFromView:v withRotate:v.rotation];
|
||||
sticker.frame = v.frame;
|
||||
return sticker;
|
||||
}];
|
||||
|
||||
TUIMultimediaSticker *drawSticker = _drawCtrlView != nil ? _drawCtrlView.drawSticker : nil;
|
||||
if (drawSticker != nil) {
|
||||
stickers = [NSArray tui_multimedia_arrayWithArray:stickers append:@[ drawSticker ]];
|
||||
}
|
||||
return stickers;
|
||||
}
|
||||
|
||||
- (void) clearAllStaticker {
|
||||
if (_drawCtrlView != nil) {
|
||||
[_drawCtrlView clearAllDraw];
|
||||
}
|
||||
|
||||
for (TUIMultimediaStickerView *v in _stickerViewList) {
|
||||
[v removeFromSuperview];
|
||||
}
|
||||
[_stickerViewList removeAllObjects];
|
||||
}
|
||||
|
||||
#pragma mark - UIGestureRecognizer actions
|
||||
- (void)onTapContainerView:(UITapGestureRecognizer *)rec {
|
||||
_lastSelectedStickerView.selected = NO;
|
||||
_lastSelectedStickerView = nil;
|
||||
}
|
||||
|
||||
#pragma mark - Actions
|
||||
- (void)onBtnSendClicked {
|
||||
[self setDrawEnabled:NO];
|
||||
[self setDrawMosaicEnabled:NO];
|
||||
[_delegate onCommonEditorControlViewComplete:self stickers:[self getStickers]];
|
||||
}
|
||||
|
||||
- (void)onBtnCancelClicked {
|
||||
[_delegate onCommonEditorControlViewCancel:self];
|
||||
}
|
||||
|
||||
- (void)onBtnSubtitleClicked {
|
||||
[_delegate onCommonEditorControlViewNeedModifySubtitle:[[TUIMultimediaSubtitleInfo alloc] init]
|
||||
callback:^(TUIMultimediaSubtitleInfo *subtitle, BOOL isOk) {
|
||||
if (isOk) {
|
||||
[self addSubtitle:subtitle];
|
||||
}
|
||||
}];
|
||||
[self setDrawEnabled:NO];
|
||||
[self setDrawMosaicEnabled:NO];
|
||||
_lastSelectedStickerView.selected = NO;
|
||||
_lastSelectedStickerView = nil;
|
||||
}
|
||||
- (void)onBtnPasterClicked {
|
||||
[_delegate onCommonEditorControlViewNeedAddPaster:self];
|
||||
[self setDrawEnabled:NO];
|
||||
[self setDrawMosaicEnabled:NO];
|
||||
_lastSelectedStickerView.selected = NO;
|
||||
_lastSelectedStickerView = nil;
|
||||
}
|
||||
|
||||
- (void)onBtnDrawGraffitiClicked {
|
||||
[self setDrawEnabled:!_btnDrawGraffiti.selected];
|
||||
_drawCtrlView.drawMode = GRAFFITI;
|
||||
}
|
||||
|
||||
- (void)onBtnDrawMosaicClicked {
|
||||
[self setDrawMosaicEnabled:!_btnDrawMosaic.selected];
|
||||
_drawCtrlView.drawMode = MOSAIC;
|
||||
}
|
||||
|
||||
- (void)onBtnMusicClicked {
|
||||
[self setDrawEnabled:NO];
|
||||
[self setDrawMosaicEnabled:NO];
|
||||
[_delegate onCommonEditorControlViewNeedEditMusic:self];
|
||||
_lastSelectedStickerView.selected = NO;
|
||||
_lastSelectedStickerView = nil;
|
||||
}
|
||||
|
||||
- (void)onBtnCropClicked {
|
||||
[self setDrawEnabled:NO];
|
||||
[self setDrawMosaicEnabled:NO];
|
||||
[_cropControlView show];
|
||||
_drawCtrlView.drawEnable = NO;
|
||||
_stkViewButtons.hidden = YES;
|
||||
_btnCancel.hidden = YES;
|
||||
}
|
||||
|
||||
- (void)onBtnGenerateCancelClicked {
|
||||
[_delegate onCommonEditorControlViewCancelGenerate:self];
|
||||
self.isGenerating = false;
|
||||
_progressView.progress = 0;
|
||||
}
|
||||
|
||||
#pragma mark - Properties
|
||||
- (BOOL)modifyButtonsHidden {
|
||||
return _stkViewButtons.hidden;
|
||||
}
|
||||
|
||||
- (void)setModifyButtonsHidden:(BOOL)modifyButtonsHidden {
|
||||
_stkViewButtons.hidden = modifyButtonsHidden;
|
||||
}
|
||||
|
||||
- (void)setPreviewSize:(CGSize)previewSize {
|
||||
_previewSize = previewSize;
|
||||
if (previewSize.width == 0 || previewSize.height == 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
int width = self.frame.size.width;
|
||||
int height = previewSize.height / previewSize.width * width;
|
||||
_previewView.frame = CGRectMake(0,0, width, height);
|
||||
_previewView.center = self.center;
|
||||
_previewLimitRect = _previewView.frame;
|
||||
}
|
||||
|
||||
- (BOOL)isGenerating {
|
||||
return !_generateView.hidden;
|
||||
}
|
||||
|
||||
- (void)setIsGenerating:(BOOL)isGenerating {
|
||||
_generateView.hidden = !isGenerating;
|
||||
_stkViewButtons.hidden = isGenerating;
|
||||
_btnCancel.hidden = isGenerating;
|
||||
}
|
||||
|
||||
- (CGFloat)progressBarProgress {
|
||||
return _progressView.progress;
|
||||
}
|
||||
|
||||
- (void)setProgressBarProgress:(CGFloat)progressBarProgress {
|
||||
[_progressView setProgress:progressBarProgress animated:YES];
|
||||
}
|
||||
|
||||
- (BOOL)musicEdited {
|
||||
return _btnMusic != nil ? _btnMusic.selected : FALSE;
|
||||
}
|
||||
|
||||
- (void)setMusicEdited:(BOOL)musicEdited {
|
||||
if (_btnMusic != nil) {
|
||||
_btnMusic.selected = musicEdited;
|
||||
}
|
||||
}
|
||||
|
||||
-(void)setMosaciOriginalImage:(UIImage *)mosaciOriginalImage {
|
||||
_mosaciOriginalImage = mosaciOriginalImage;
|
||||
if (_drawCtrlView) {
|
||||
_drawCtrlView.drawView.mosaciOriginalImage = mosaciOriginalImage;
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaCropControlDelegate protocol
|
||||
|
||||
- (void)onCancelCrop {
|
||||
[self previewRotationToZero];
|
||||
self.previewSize = _previewSize;
|
||||
_stkViewButtons.hidden = NO;
|
||||
_btnCancel.hidden = NO;
|
||||
}
|
||||
|
||||
- (void)onConfirmCrop:(CGRect)cropRect {
|
||||
CGFloat cropX = (cropRect.origin.x - _previewView.frame.origin.x) / _previewView.frame.size.width;
|
||||
CGFloat cropY = (cropRect.origin.y - _previewView.frame.origin.y) / _previewView.frame.size.height;
|
||||
CGFloat cropWidth = cropRect.size.width / _previewView.frame.size.width;
|
||||
CGFloat cropHeight = cropRect.size.height / _previewView.frame.size.height;
|
||||
CGRect normalizedCropRect = CGRectMake(cropX, cropY, cropWidth, cropHeight);
|
||||
|
||||
CGFloat rotationAngle = _previewRotationAngle;
|
||||
if ((_previewRotationAngle + 360) % 360 != 0) {
|
||||
_previewView.hidden = YES;
|
||||
}
|
||||
[self previewRotationToZero];
|
||||
|
||||
[_delegate onCommonEditorControlViewCrop:rotationAngle normalizedCropRect:normalizedCropRect stickers:[self getStickers]];
|
||||
_stkViewButtons.hidden = NO;
|
||||
_btnCancel.hidden = NO;
|
||||
[self clearAllStaticker];
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaStickerViewDelegate protocol
|
||||
- (void)onStickerViewSelected:(TUIMultimediaStickerView *)v {
|
||||
[_editContainerView bringSubviewToFront:v];
|
||||
_lastSelectedStickerView.selected = NO;
|
||||
_lastSelectedStickerView = v;
|
||||
}
|
||||
|
||||
- (void)onStickerViewShouldDelete:(TUIMultimediaStickerView *)v {
|
||||
[v removeFromSuperview];
|
||||
[_stickerViewList removeObject:v];
|
||||
}
|
||||
|
||||
- (void)onStickerViewShouldEdit:(TUIMultimediaStickerView *)v {
|
||||
if ([v isKindOfClass:TUIMultimediaSubtitleView.class]) {
|
||||
TUIMultimediaSubtitleView *vsub = (TUIMultimediaSubtitleView *)v;
|
||||
vsub.hidden = YES;
|
||||
TUIMultimediaSubtitleInfo *info = vsub.subtitleInfo;
|
||||
[_delegate onCommonEditorControlViewNeedModifySubtitle:info
|
||||
callback:^(TUIMultimediaSubtitleInfo *newInfo, BOOL isOk) {
|
||||
if (isOk) {
|
||||
vsub.subtitleInfo = newInfo;
|
||||
}
|
||||
vsub.hidden = NO;
|
||||
}];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)onStickerViewSizeChanged:(TUIMultimediaStickerView *)v {
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaDrawCtrlViewDelegate
|
||||
- (void)onIsDrawCtrlViewDrawing:(BOOL)Hidden {
|
||||
_stkViewButtons.hidden = Hidden;
|
||||
}
|
||||
@end
|
||||
|
||||
#pragma mark - TUIMultimediaCommonEditorConfig
|
||||
@implementation TUIMultimediaCommonEditorConfig
|
||||
+ (instancetype)configForVideoEditor {
|
||||
TUIMultimediaCommonEditorConfig *config = [[TUIMultimediaCommonEditorConfig alloc] init];
|
||||
config.pasterEnabled = [[TUIMultimediaConfig sharedInstance] isSupportVideoEditPaster];
|
||||
config.subtitleEnabled = [[TUIMultimediaConfig sharedInstance] isSupportVideoEditSubtitle];
|
||||
config.drawGraffitiEnabled = [[TUIMultimediaConfig sharedInstance] isSupportVideoEditGraffiti];
|
||||
config.musicEditEnabled = [[TUIMultimediaConfig sharedInstance] isSupportVideoEditBGM];
|
||||
config.cropEnabled = NO;
|
||||
config.drawMosaicEnabled = NO;
|
||||
return config;
|
||||
}
|
||||
+ (instancetype)configForPictureEditor {
|
||||
TUIMultimediaCommonEditorConfig *config = [[TUIMultimediaCommonEditorConfig alloc] init];
|
||||
config.pasterEnabled = [[TUIMultimediaConfig sharedInstance] isSupportPictureEditPaster];
|
||||
config.subtitleEnabled = [[TUIMultimediaConfig sharedInstance] isSupportPictureEditSubtitle];
|
||||
config.drawGraffitiEnabled = [[TUIMultimediaConfig sharedInstance] isSupportPictureEditGraffiti];
|
||||
config.drawMosaicEnabled = [[TUIMultimediaConfig sharedInstance] isSupportPictureEditMosaic];
|
||||
config.cropEnabled = [[TUIMultimediaConfig sharedInstance] isSupportPictureEditCrop];
|
||||
config.musicEditEnabled = NO;
|
||||
return config;
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,25 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaCropView.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@class TUIMultimediaCommonEditorControlView;
|
||||
@protocol TUIMultimediaCropControlDelegate;
|
||||
|
||||
@interface TUIMultimediaCropControlView : UIView
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaCropControlDelegate> delegate;
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame editorControl:(TUIMultimediaCommonEditorControlView*)editorControl;
|
||||
- (void)show;
|
||||
- (void)changeResetButtonStatus;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaCropControlDelegate <NSObject>
|
||||
- (void)onCancelCrop;
|
||||
- (void)onConfirmCrop:(CGRect)cropRect;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,166 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaCropControlView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImageUtil.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
|
||||
#import "TUIMultimediaCropView.h"
|
||||
#import "TUIMultimediaCommonEditorControlView.h"
|
||||
|
||||
@interface TUIMultimediaCropControlView()<TUIMultimediaCropDelegate> {
|
||||
TUIMultimediaCommonEditorControlView* _editorControl;
|
||||
TUIMultimediaCropView *_cropView;
|
||||
UIButton* _restoreButton;
|
||||
UIButton* _rotationButton;
|
||||
UIButton* _confirmButton;
|
||||
UIButton* _cancelButton;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaCropControlView
|
||||
- (instancetype)initWithFrame:(CGRect)frame editorControl:(TUIMultimediaCommonEditorControlView*)editorControl{
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
_editorControl = editorControl;
|
||||
[self initUI];
|
||||
self.backgroundColor = [UIColor clearColor];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
#pragma mark - UI init
|
||||
|
||||
- (void)initUI {
|
||||
_cropView = [[TUIMultimediaCropView alloc] initWithFrame:self.frame];
|
||||
[self addSubview:_cropView];
|
||||
[_cropView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.size.equalTo(self);
|
||||
}];
|
||||
_cropView.delegate = self;
|
||||
|
||||
_restoreButton = [self addFuncitonButtonWithText:[TUIMultimediaCommon localizedStringForKey:@"restore"]
|
||||
action:@selector(onResetBtnClicked) bottomOffset:-75 leftOffset:7.5 rightOffset:0 size:CGSizeMake(80, 22)];
|
||||
_cancelButton = [self addFunctionButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_crop_cancel", @"crop_cancel")
|
||||
action:@selector(onCancelBtnClicked) bottomOffset:-25 leftOffset:35 rightOffset:0 size:CGSizeMake(25, 25)];
|
||||
_confirmButton = [self addFunctionButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_crop_confirm", @"crop_ok")
|
||||
action:@selector(onConfirmBtnClicked) bottomOffset:-25 leftOffset:0 rightOffset:-32.5 size:CGSizeMake(30, 28)];
|
||||
_rotationButton = [self addFunctionButtonWithImage:TUIMultimediaPluginBundleThemeImage(@"editor_crop_rotation", @"crop_rotation")
|
||||
action:@selector(onRotationBtnClicked) bottomOffset:-75 leftOffset:0 rightOffset:-35 size:CGSizeMake(25, 25)];
|
||||
|
||||
_restoreButton.enabled = NO;
|
||||
}
|
||||
|
||||
-(void)show {
|
||||
self.hidden = NO;
|
||||
_cropView.preViewFrame = _editorControl.previewView.frame;
|
||||
[_cropView reset];
|
||||
_restoreButton.enabled = NO;
|
||||
}
|
||||
|
||||
- (void)changeResetButtonStatus {
|
||||
_restoreButton.enabled = ![self isApproximateSize:[_cropView getCropRect].size size2:_editorControl.previewView.frame.size];
|
||||
}
|
||||
|
||||
-(void)onResetBtnClicked {
|
||||
[_editorControl previewRotationToZero];
|
||||
_cropView.preViewFrame = _editorControl.previewView.frame;
|
||||
[_cropView reset];
|
||||
_restoreButton.enabled = NO;
|
||||
}
|
||||
|
||||
-(void)onRotationBtnClicked {
|
||||
CGRect cropRect = [_cropView getCropRect];
|
||||
CGPoint rotationCenter = CGPointMake(CGRectGetMidX(cropRect), CGRectGetMidY(cropRect));
|
||||
[_editorControl previewRotation90:rotationCenter];
|
||||
[_cropView rotation90];
|
||||
_restoreButton.enabled = YES;
|
||||
}
|
||||
|
||||
-(void)onCancelBtnClicked {
|
||||
self.hidden = YES;
|
||||
if (_delegate != nil) {
|
||||
[_delegate onCancelCrop];
|
||||
}
|
||||
}
|
||||
|
||||
-(void)onConfirmBtnClicked {
|
||||
self.hidden = YES;
|
||||
if (_delegate != nil) {
|
||||
if (_restoreButton.isEnabled) {
|
||||
[_delegate onConfirmCrop:[_cropView getCropRect]];
|
||||
} else {
|
||||
[_delegate onCancelCrop];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
-(UIButton*) addFuncitonButtonWithText:(NSString*)text action:(SEL)actionSel bottomOffset:(int)bottomOffset
|
||||
leftOffset:(int)leftOffset rightOffset:(int)rightOffset size:(CGSize)size {
|
||||
UIButton* button = [UIButton buttonWithType:UIButtonTypeSystem];
|
||||
[self addSubview:button];
|
||||
button.backgroundColor = [UIColor clearColor];
|
||||
[button mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
if (leftOffset > 0) {
|
||||
make.leading.equalTo(self).offset(leftOffset);
|
||||
}
|
||||
|
||||
if (rightOffset < 0) {
|
||||
make.trailing.equalTo(self).offset(rightOffset);
|
||||
}
|
||||
|
||||
make.bottom.equalTo(self).offset(bottomOffset);
|
||||
make.size.mas_equalTo(size);
|
||||
}];
|
||||
[button addTarget:self action:actionSel forControlEvents:UIControlEventTouchUpInside];
|
||||
[button setTitle:text forState:UIControlStateNormal];
|
||||
[button setTitleColor:[UIColor whiteColor] forState:UIControlStateNormal];
|
||||
[button setTitleColor:[UIColor grayColor] forState:UIControlStateDisabled];
|
||||
return button;
|
||||
}
|
||||
|
||||
- (UIButton *)addFunctionButtonWithImage:(UIImage *)image action:(SEL)actionSel bottomOffset:(int)bottomOffset
|
||||
leftOffset:(int)leftOffset rightOffset:(int)rightOffset size:(CGSize)size{
|
||||
UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
|
||||
[self addSubview:button];
|
||||
UIImage *imgNormal = [TUIMultimediaImageUtil imageFromImage:image withTintColor:
|
||||
TUIMultimediaPluginDynamicColor(@"editor_func_btn_normal_color", @"#FFFFFF")];
|
||||
[button setImage:imgNormal forState:UIControlStateNormal];
|
||||
[button setImageEdgeInsets:UIEdgeInsetsMake(0, 0, 0, 0)];
|
||||
[button addTarget:self action:actionSel forControlEvents:UIControlEventTouchUpInside];
|
||||
|
||||
[button mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
if (leftOffset > 0) {
|
||||
make.leading.equalTo(self).offset(leftOffset);
|
||||
}
|
||||
|
||||
if (rightOffset < 0) {
|
||||
make.trailing.equalTo(self).offset(rightOffset);
|
||||
}
|
||||
|
||||
make.bottom.equalTo(self).offset(bottomOffset);
|
||||
make.size.mas_equalTo(size);
|
||||
}];
|
||||
return button;
|
||||
}
|
||||
|
||||
|
||||
-(BOOL)isApproximateSize:(CGSize)size1 size2:(CGSize)size2 {
|
||||
return ABS(size1.width - size1.width ) < 1 && ABS(size1.height - size2.height) < 1;
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaCropViewDelegate
|
||||
- (void)onStartCrop {
|
||||
_editorControl.isStartCrop = YES;
|
||||
_restoreButton.enabled = YES;
|
||||
}
|
||||
|
||||
- (void)onCropComplete:(CGFloat)scale centerPoint:(CGPoint)centerPoint offset:(CGPoint)offset {
|
||||
_editorControl.isStartCrop = NO;
|
||||
_editorControl.previewLimitRect = [_cropView getCropRect];
|
||||
[_editorControl previewScale:scale center:centerPoint];
|
||||
[_editorControl previewMove:offset];
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,19 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
@protocol TUIMultimediaCropDelegate;
|
||||
|
||||
@interface TUIMultimediaCropView : UIView
|
||||
@property(nonatomic) CGRect preViewFrame;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaCropDelegate> delegate;
|
||||
-(void)reset;
|
||||
-(void)rotation90;
|
||||
-(CGRect)getCropRect;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaCropDelegate <NSObject>
|
||||
- (void)onStartCrop;
|
||||
- (void)onCropComplete:(CGFloat)scale centerPoint:(CGPoint)centerPoint offset:(CGPoint)offset;
|
||||
@end
|
||||
@@ -0,0 +1,334 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaCropView.h"
|
||||
|
||||
#import <ReactiveObjC/ReactiveObjC.h>
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
|
||||
#define TOUCH_MAX_DIS 40
|
||||
#define CORNER_LINE_WIDTH 4.0
|
||||
#define CORNER_LENGTH 25.0
|
||||
#define CROP_RECT_BORDER_LINE_WIDTH 2.0
|
||||
#define CROP_RECT_GRID_LINE_WIDTH 0.8
|
||||
|
||||
@interface TUIMultimediaCropView()<UIGestureRecognizerDelegate, UIGestureRecognizerDelegate>{
|
||||
UIColor * _borderColor;
|
||||
UIColor* _backgroundColor;
|
||||
|
||||
NSDate * _lastShowTransparentBackgroundTime;
|
||||
NSDate * _lastShowGridTime;
|
||||
BOOL _isShowTransparentBackground;
|
||||
BOOL _isShowGird;
|
||||
|
||||
CGRect _limitRect;
|
||||
CGRect _cropRect;
|
||||
|
||||
CGFloat _isMoveLeft;
|
||||
CGFloat _isMoveRight;
|
||||
CGFloat _isMoveTop;
|
||||
CGFloat _isMoveBottom;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaCropView
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self) {
|
||||
self.backgroundColor = [UIColor clearColor];
|
||||
|
||||
_borderColor = [UIColor whiteColor];
|
||||
_backgroundColor = [UIColor blackColor];
|
||||
_isShowTransparentBackground = NO;
|
||||
_isShowGird = NO;
|
||||
|
||||
UIPanGestureRecognizer *panGesture = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(onPan:)];
|
||||
panGesture.maximumNumberOfTouches = 1;
|
||||
panGesture.delegate = self;
|
||||
[self addGestureRecognizer:panGesture];
|
||||
|
||||
UITapGestureRecognizer *tapGesture = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(onTap:)];
|
||||
[self addGestureRecognizer:tapGesture];
|
||||
|
||||
UIPinchGestureRecognizer *pinchRec = [[UIPinchGestureRecognizer alloc] initWithTarget:self action:@selector(onPinch:)];
|
||||
[self addGestureRecognizer:pinchRec];
|
||||
pinchRec.delegate = self;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
-(CGRect)getCropRect {
|
||||
return _cropRect;
|
||||
}
|
||||
|
||||
-(void)reset {
|
||||
[self adjustCropRect:_preViewFrame];
|
||||
[self setNeedsDisplay];
|
||||
}
|
||||
|
||||
-(void)rotation90 {
|
||||
int centerX = CGRectGetMidX(_cropRect);
|
||||
int centerY = CGRectGetMidY(_cropRect);
|
||||
int width = _cropRect.size.width;
|
||||
int height = _cropRect.size.height;
|
||||
|
||||
_cropRect.origin.y = centerY - width / 2.0f;
|
||||
_cropRect.size.height = width;
|
||||
|
||||
_cropRect.origin.x = centerX - height / 2.0f;
|
||||
_cropRect.size.width = height;
|
||||
|
||||
[self adjustCropRect:_cropRect];
|
||||
[self setNeedsDisplay];
|
||||
}
|
||||
|
||||
- (void)layoutSubviews {
|
||||
[super layoutSubviews];
|
||||
_limitRect = CGRectMake(10, 50, self.frame.size.width - 20, self.frame.size.height - 160);
|
||||
_cropRect = _limitRect;
|
||||
}
|
||||
|
||||
- (void)onTap:(UITapGestureRecognizer *)gesture {
|
||||
[self showTransparentBackground];
|
||||
[self showGrid];
|
||||
}
|
||||
|
||||
- (void)onPinch:(UIPinchGestureRecognizer *)gestureRecognizer {
|
||||
[self showTransparentBackground];
|
||||
[self showGrid];
|
||||
}
|
||||
|
||||
- (void)onPan:(UIPanGestureRecognizer *)gesture {
|
||||
_backgroundColor = [UIColor clearColor];
|
||||
[self showTransparentBackground];
|
||||
[self showGrid];
|
||||
switch (gesture.state) {
|
||||
case UIGestureRecognizerStateBegan: {
|
||||
CGPoint p = [gesture locationInView:self];
|
||||
_isMoveLeft = fabs(p.x - _cropRect.origin.x) < TOUCH_MAX_DIS;
|
||||
_isMoveRight = fabs(p.x - CGRectGetMaxX(_cropRect)) < TOUCH_MAX_DIS;
|
||||
_isMoveTop = fabs(p.y - _cropRect.origin.y) < TOUCH_MAX_DIS;
|
||||
_isMoveBottom = fabs(p.y - CGRectGetMaxY(_cropRect)) < TOUCH_MAX_DIS;
|
||||
if (_isMoveTop || _isMoveLeft || _isMoveRight || _isMoveBottom) {
|
||||
[_delegate onStartCrop];
|
||||
}
|
||||
break;
|
||||
}
|
||||
case UIGestureRecognizerStateChanged: {
|
||||
CGPoint p = [gesture locationInView:self];
|
||||
if (_isMoveLeft && p.x > _limitRect.origin.x && p.x < CGRectGetMaxX(_cropRect)) {
|
||||
_cropRect.size.width += (_cropRect.origin.x - p.x);
|
||||
_cropRect.origin.x = p.x;
|
||||
}
|
||||
|
||||
if (_isMoveRight && p.x > _cropRect.origin.x && p.x < CGRectGetMaxX(_limitRect)) {
|
||||
_cropRect.size.width = p.x - _cropRect.origin.x;
|
||||
}
|
||||
|
||||
if (_isMoveTop && p.y > _limitRect.origin.y && p.y < CGRectGetMaxY(_cropRect)) {
|
||||
_cropRect.size.height += (_cropRect.origin.y - p.y);
|
||||
_cropRect.origin.y = p.y;
|
||||
}
|
||||
|
||||
if (_isMoveBottom && p.y > _cropRect.origin.y && p.y < CGRectGetMaxY(_limitRect)) {
|
||||
_cropRect.size.height = p.y - _cropRect.origin.y;
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
case UIGestureRecognizerStateEnded: {
|
||||
if (_isMoveTop || _isMoveLeft || _isMoveRight || _isMoveBottom) {
|
||||
[self adjustCropRect:_cropRect];
|
||||
}
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
[gesture setTranslation:CGPointZero inView:self.superview];
|
||||
[self setNeedsDisplay];
|
||||
}
|
||||
|
||||
|
||||
- (void) adjustCropRect:(CGRect) sourceCropRect {
|
||||
if (CGRectIsEmpty(sourceCropRect)) {
|
||||
return;
|
||||
}
|
||||
|
||||
float limitRectRation = _limitRect.size.width * 1.0f / _limitRect.size.height;
|
||||
float sourceRectRation = sourceCropRect.size.width * 1.0f / sourceCropRect.size.height;
|
||||
|
||||
if (sourceRectRation > limitRectRation) {
|
||||
_cropRect.origin.x = _limitRect.origin.x;
|
||||
_cropRect.size.width = _limitRect.size.width;
|
||||
_cropRect.size.height = _cropRect.size.width / sourceRectRation;
|
||||
_cropRect.origin.y = (_limitRect.size.height - _cropRect.size.height) / 2.0f + _limitRect.origin.y;
|
||||
} else {
|
||||
_cropRect.origin.y = _limitRect.origin.y;
|
||||
_cropRect.size.height = _limitRect.size.height;
|
||||
_cropRect.size.width = _cropRect.size.height * sourceRectRation;
|
||||
_cropRect.origin.x = (_limitRect.size.width - _cropRect.size.width) / 2.0f + _limitRect.origin.x;
|
||||
}
|
||||
|
||||
CGFloat scale = _cropRect.size.width * 1.0f / sourceCropRect.size.width;
|
||||
CGFloat moveX = _cropRect.origin.x - sourceCropRect.origin.x;
|
||||
CGFloat moveY = _cropRect.origin.y - sourceCropRect.origin.y;
|
||||
|
||||
[_delegate onCropComplete:scale centerPoint:sourceCropRect.origin
|
||||
offset:CGPointMake(moveX, moveY)];
|
||||
}
|
||||
|
||||
- (void)drawRect:(CGRect)rect {
|
||||
[super drawRect:rect];
|
||||
|
||||
if (_cropRect.size.width == 0 || _cropRect.size.height == 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
CGContextRef context = UIGraphicsGetCurrentContext();
|
||||
if (_isShowTransparentBackground) {
|
||||
[[UIColor clearColor] setFill];
|
||||
CGContextFillRect(context, rect);
|
||||
CGContextFillPath(context);
|
||||
} else {
|
||||
[[UIColor blackColor] setFill];
|
||||
CGContextFillRect(context, rect);
|
||||
CGContextAddRect(context, _cropRect);
|
||||
CGContextClip(context);
|
||||
CGContextSetBlendMode(context, kCGBlendModeClear);
|
||||
CGContextAddRect(context, _cropRect);
|
||||
CGContextFillPath(context);
|
||||
}
|
||||
|
||||
CGContextSetBlendMode(context, kCGBlendModeNormal);
|
||||
[self drawCropRect:context];
|
||||
[self drawCorner:context];
|
||||
if (_isShowGird) {
|
||||
[self drawGrid:context];
|
||||
}
|
||||
}
|
||||
|
||||
- (void) drawCropRect:(CGContextRef) context {
|
||||
CGContextSetStrokeColorWithColor(context, _borderColor.CGColor);
|
||||
CGContextSetLineWidth(context, CROP_RECT_BORDER_LINE_WIDTH );
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect) + CORNER_LENGTH, CGRectGetMinY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect) - CORNER_LENGTH, CGRectGetMinY(_cropRect));
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMinY(_cropRect) + CORNER_LENGTH);
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMaxY(_cropRect) - CORNER_LENGTH);
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMaxX(_cropRect) - CORNER_LENGTH, CGRectGetMaxY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect) + CORNER_LENGTH, CGRectGetMaxY(_cropRect));
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMaxY(_cropRect) - CORNER_LENGTH);
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMinY(_cropRect) + CORNER_LENGTH);
|
||||
|
||||
CGContextStrokePath(context);
|
||||
}
|
||||
|
||||
- (void) drawCorner:(CGContextRef) context {
|
||||
float corner_width = CORNER_LINE_WIDTH;
|
||||
if (!_isShowTransparentBackground) {
|
||||
corner_width = 1.5f * corner_width;
|
||||
}
|
||||
CGContextSetLineWidth(context, corner_width);
|
||||
float halfLineWidth = corner_width / 2.0f;
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect) - halfLineWidth, CGRectGetMinY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect) + CORNER_LENGTH, CGRectGetMinY(_cropRect));
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMinY(_cropRect) - halfLineWidth);
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMinY(_cropRect) + CORNER_LENGTH);
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMaxX(_cropRect) - CORNER_LENGTH, CGRectGetMinY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect) + halfLineWidth, CGRectGetMinY(_cropRect));
|
||||
CGContextMoveToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMinY(_cropRect) - halfLineWidth);
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMinY(_cropRect) + CORNER_LENGTH);
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMaxY(_cropRect) - CORNER_LENGTH);
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMaxY(_cropRect) + halfLineWidth);
|
||||
CGContextMoveToPoint(context, CGRectGetMaxX(_cropRect) - CORNER_LENGTH, CGRectGetMaxY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect) + halfLineWidth, CGRectGetMaxY(_cropRect));
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMaxY(_cropRect) - CORNER_LENGTH);
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMaxY(_cropRect) + halfLineWidth);
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect) - halfLineWidth, CGRectGetMaxY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect) + CORNER_LENGTH, CGRectGetMaxY(_cropRect));
|
||||
|
||||
CGContextStrokePath(context);
|
||||
}
|
||||
|
||||
- (void) drawGrid:(CGContextRef) context {
|
||||
CGContextSetLineWidth(context, CROP_RECT_GRID_LINE_WIDTH);
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect) + _cropRect.size.width / 3, CGRectGetMinY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect) + _cropRect.size.width / 3, CGRectGetMaxY(_cropRect));
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect) + _cropRect.size.width * 2 / 3, CGRectGetMinY(_cropRect));
|
||||
CGContextAddLineToPoint(context, CGRectGetMinX(_cropRect) + _cropRect.size.width * 2 / 3, CGRectGetMaxY(_cropRect));
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMinY(_cropRect) + _cropRect.size.height / 3);
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMinY(_cropRect) + _cropRect.size.height / 3);
|
||||
|
||||
CGContextMoveToPoint(context, CGRectGetMinX(_cropRect), CGRectGetMinY(_cropRect) + _cropRect.size.height * 2 / 3);
|
||||
CGContextAddLineToPoint(context, CGRectGetMaxX(_cropRect), CGRectGetMinY(_cropRect) + _cropRect.size.height * 2 / 3);
|
||||
|
||||
CGContextStrokePath(context);
|
||||
}
|
||||
|
||||
|
||||
- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer {
|
||||
return YES;
|
||||
}
|
||||
|
||||
- (void) showTransparentBackground {
|
||||
_lastShowTransparentBackgroundTime = [NSDate date];
|
||||
if (!_isShowTransparentBackground) {
|
||||
_isShowTransparentBackground = YES;
|
||||
[self setNeedsDisplay];
|
||||
[self delayCancelShowTransparentBackground];
|
||||
}
|
||||
}
|
||||
|
||||
- (void) delayCancelShowTransparentBackground {
|
||||
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0f * NSEC_PER_SEC));
|
||||
|
||||
@weakify(self)
|
||||
dispatch_after(popTime, dispatch_get_main_queue(), ^(void) {
|
||||
@strongify(self)
|
||||
if ([[NSDate date] timeIntervalSinceDate:self->_lastShowTransparentBackgroundTime] < 2.0f) {
|
||||
[self delayCancelShowTransparentBackground];
|
||||
return;
|
||||
}
|
||||
self->_isShowTransparentBackground = NO;
|
||||
[self setNeedsDisplay];
|
||||
});
|
||||
}
|
||||
|
||||
- (void) showGrid {
|
||||
_lastShowGridTime = [NSDate date];
|
||||
if (!_isShowGird) {
|
||||
_isShowGird = YES;
|
||||
[self setNeedsDisplay];
|
||||
[self delayCancelShowGrid];
|
||||
}
|
||||
}
|
||||
|
||||
- (void) delayCancelShowGrid {
|
||||
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0f * NSEC_PER_SEC));
|
||||
|
||||
@weakify(self)
|
||||
dispatch_after(popTime, dispatch_get_main_queue(), ^(void) {
|
||||
@strongify(self)
|
||||
if ([[NSDate date] timeIntervalSinceDate:self->_lastShowGridTime] < 4.0f) {
|
||||
[self delayCancelShowGrid];
|
||||
return;
|
||||
}
|
||||
self->_isShowGird = NO;
|
||||
[self setNeedsDisplay];
|
||||
});
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,27 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
@class TUIMultimediaDrawView;
|
||||
@class TUIMultimediaSticker;
|
||||
@protocol TUIMultimediaDrawCtrlViewDelegate;
|
||||
enum DrawMode:NSInteger;
|
||||
|
||||
@interface TUIMultimediaDrawCtrlView : UIView
|
||||
-(void)flushRedoUndoState;
|
||||
-(void)clearAllDraw;
|
||||
|
||||
@property (nonatomic, strong) TUIMultimediaDrawView* drawView;
|
||||
@property (nonatomic) enum DrawMode drawMode;
|
||||
@property (nonatomic) BOOL drawEnable;
|
||||
@property (nonatomic, strong, readonly) TUIMultimediaSticker * drawSticker;
|
||||
@property (weak, nullable, nonatomic) id<TUIMultimediaDrawCtrlViewDelegate> delegate;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaDrawCtrlViewDelegate <NSObject>
|
||||
- (void)onIsDrawCtrlViewDrawing:(BOOL)Hidden;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,201 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaDrawCtrlView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaColorPanel.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSplitter.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImageUtil.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaDrawView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSticker.h"
|
||||
|
||||
#define FunctionButtonColorNormal TUIMultimediaPluginDynamicColor(@"editor_func_btn_normal_color", @"#FFFFFF")
|
||||
#define FunctionButtonColorDisabled TUIMultimediaPluginDynamicColor(@"editor_func_btn_disabled_color", @"#6D6D6D")
|
||||
|
||||
@interface TUIMultimediaDrawCtrlView () <TUIMultimediaColorPanelDelegate, TUIMultimediaDrawViewDelegate> {
|
||||
TUIMultimediaColorPanel *_colorPanel;
|
||||
UIButton *_btnRedo;
|
||||
UIButton *_btnUndo;
|
||||
TUIMultimediaSplitter *_vertical_splitter;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaDrawCtrlView
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
[self initCtrlView];
|
||||
[self initDrawView];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initCtrlView {
|
||||
self.backgroundColor = TUIMultimediaPluginDynamicColor(@"editor_draw_panel_bg_color", @"#3333337F");
|
||||
|
||||
_colorPanel = [[TUIMultimediaColorPanel alloc] init];
|
||||
_colorPanel.delegate = self;
|
||||
[self addSubview:_colorPanel];
|
||||
|
||||
UIImage *imgUndo = TUIMultimediaPluginBundleThemeImage(@"editor_undo_img", @"undo");
|
||||
_btnUndo = [UIButton buttonWithType:UIButtonTypeCustom];
|
||||
[self addSubview:_btnUndo];
|
||||
_btnUndo.enabled = NO;
|
||||
[_btnUndo addTarget:self action:@selector(onBtnUndoClicked) forControlEvents:UIControlEventTouchUpInside];
|
||||
[_btnUndo setImage:[TUIMultimediaImageUtil imageFromImage:imgUndo withTintColor:FunctionButtonColorNormal] forState:UIControlStateNormal];
|
||||
[_btnUndo setImage:[TUIMultimediaImageUtil imageFromImage:imgUndo withTintColor:FunctionButtonColorDisabled] forState:UIControlStateDisabled];
|
||||
|
||||
UIImage *imgRedo = TUIMultimediaPluginBundleThemeImage(@"editor_redo_img", @"redo");
|
||||
_btnRedo = [UIButton buttonWithType:UIButtonTypeCustom];
|
||||
[self addSubview:_btnRedo];
|
||||
_btnRedo.enabled = NO;
|
||||
[_btnRedo addTarget:self action:@selector(onBtnRedoClicked) forControlEvents:UIControlEventTouchUpInside];
|
||||
[_btnRedo setImage:[TUIMultimediaImageUtil imageFromImage:imgRedo withTintColor:FunctionButtonColorNormal] forState:UIControlStateNormal];
|
||||
[_btnRedo setImage:[TUIMultimediaImageUtil imageFromImage:imgRedo withTintColor:FunctionButtonColorDisabled] forState:UIControlStateDisabled];
|
||||
|
||||
TUIMultimediaSplitter * horizontal_splitter = [[TUIMultimediaSplitter alloc] init];
|
||||
[self addSubview:horizontal_splitter];
|
||||
horizontal_splitter.axis = UILayoutConstraintAxisHorizontal;
|
||||
|
||||
_vertical_splitter = [[TUIMultimediaSplitter alloc] init];
|
||||
[self addSubview:_vertical_splitter];
|
||||
_vertical_splitter.axis = UILayoutConstraintAxisVertical;
|
||||
|
||||
[_btnRedo mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.right.equalTo(self).inset(5);
|
||||
make.centerY.equalTo(_colorPanel);
|
||||
make.size.mas_equalTo(CGSizeMake(30, 30));
|
||||
}];
|
||||
[_btnUndo mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.centerY.equalTo(_colorPanel);
|
||||
make.right.equalTo(_btnRedo.mas_left).inset(20);
|
||||
make.size.mas_equalTo(CGSizeMake(30, 30));
|
||||
}];
|
||||
[_colorPanel mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.top.left.equalTo(self).inset(5);
|
||||
make.right.equalTo(_btnUndo.mas_left).inset(10);
|
||||
make.height.mas_equalTo(32);
|
||||
}];
|
||||
[horizontal_splitter mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.left.right.equalTo(self).inset(3);
|
||||
make.top.equalTo(_colorPanel.mas_bottom).inset(5);
|
||||
make.height.mas_equalTo(5);
|
||||
}];
|
||||
[_vertical_splitter mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.top.equalTo(self).inset(3);
|
||||
make.bottom.equalTo(horizontal_splitter.mas_top).inset(3);
|
||||
make.left.equalTo(_colorPanel.mas_right).inset(3);
|
||||
make.width.mas_equalTo(5);
|
||||
}];
|
||||
}
|
||||
|
||||
- (void)initDrawView {
|
||||
_drawView = [[TUIMultimediaDrawView alloc] init];
|
||||
_drawView.userInteractionEnabled = NO;
|
||||
UITapGestureRecognizer *drawTapRec = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(onTapDrawView:)];
|
||||
[_drawView addGestureRecognizer:drawTapRec];
|
||||
_drawView.delegate = self;
|
||||
}
|
||||
|
||||
- (void) setDrawEnable:(BOOL)drawEnable {
|
||||
_drawView.userInteractionEnabled = drawEnable;
|
||||
self.hidden = !drawEnable;
|
||||
}
|
||||
|
||||
- (void) setDrawMode:(enum DrawMode)drawMode {
|
||||
_drawMode = drawMode;
|
||||
_drawView.drawMode = drawMode;
|
||||
[self updateUIAccordingToDrawMode];
|
||||
[self flushRedoUndoState];
|
||||
}
|
||||
|
||||
- (TUIMultimediaSticker *)drawSticker {
|
||||
if (_drawView != nil && _drawView.pathCount > 0) {
|
||||
TUIMultimediaSticker *drawSticker = [[TUIMultimediaSticker alloc] init];
|
||||
drawSticker.image = [TUIMultimediaImageUtil imageFromView:_drawView];
|
||||
drawSticker.frame = _drawView.frame;
|
||||
return drawSticker;
|
||||
}
|
||||
return nil;
|
||||
}
|
||||
|
||||
- (void)onTapDrawView:(UITapGestureRecognizer *)rec {
|
||||
self.hidden = !self.hidden;
|
||||
[self.delegate onIsDrawCtrlViewDrawing:self.hidden];
|
||||
}
|
||||
|
||||
- (void)flushRedoUndoState {
|
||||
_btnUndo.enabled = _drawView.canUndo;
|
||||
_btnRedo.enabled = _drawView.canRedo;
|
||||
}
|
||||
|
||||
-(void)clearAllDraw {
|
||||
[_drawView clear];
|
||||
}
|
||||
|
||||
-(void) updateUIAccordingToDrawMode{
|
||||
if (_drawMode == GRAFFITI) {
|
||||
_colorPanel.hidden = NO;
|
||||
_vertical_splitter.hidden = NO;
|
||||
|
||||
[_btnRedo mas_remakeConstraints:^(MASConstraintMaker *make) {
|
||||
make.right.equalTo(self).inset(5);
|
||||
make.centerY.equalTo(_colorPanel);
|
||||
make.size.mas_equalTo(CGSizeMake(30, 30));
|
||||
}];
|
||||
[_btnUndo mas_remakeConstraints:^(MASConstraintMaker *make) {
|
||||
make.centerY.equalTo(_colorPanel);
|
||||
make.right.equalTo(_btnRedo.mas_left).inset(20);
|
||||
make.size.mas_equalTo(CGSizeMake(30, 30));
|
||||
}];
|
||||
|
||||
} else {
|
||||
_colorPanel.hidden = YES;
|
||||
_vertical_splitter.hidden = YES;
|
||||
|
||||
[_btnUndo mas_remakeConstraints:^(MASConstraintMaker *make) {
|
||||
make.centerY.equalTo(_colorPanel);
|
||||
make.centerX.equalTo(self.mas_trailing).multipliedBy(0.25);
|
||||
make.size.mas_equalTo(CGSizeMake(30, 30));
|
||||
}];
|
||||
|
||||
[_btnRedo mas_remakeConstraints:^(MASConstraintMaker *make) {
|
||||
make.centerY.equalTo(_colorPanel);
|
||||
make.centerX.equalTo(self.mas_trailing).multipliedBy(0.75);
|
||||
make.size.mas_equalTo(CGSizeMake(30, 30));
|
||||
}];
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaColorPanelDelegate protocol
|
||||
- (void)onColorPanel:(TUIMultimediaColorPanel *)panel selectColor:(UIColor *)color {
|
||||
_drawView.color = color;
|
||||
}
|
||||
|
||||
- (void)onBtnUndoClicked {
|
||||
[_drawView undo];
|
||||
[self flushRedoUndoState];
|
||||
}
|
||||
|
||||
- (void)onBtnRedoClicked {
|
||||
[_drawView redo];
|
||||
[self flushRedoUndoState];
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaDrawViewDelegate
|
||||
- (void)drawViewPathListChanged:(TUIMultimediaDrawView *)v {
|
||||
[self flushRedoUndoState];
|
||||
}
|
||||
|
||||
- (void)drawViewDrawStarted:(TUIMultimediaDrawView *)v {
|
||||
[self.delegate onIsDrawCtrlViewDrawing:YES];
|
||||
self.hidden = YES;
|
||||
}
|
||||
|
||||
- (void)drawViewDrawEnded:(TUIMultimediaDrawView *)v {
|
||||
[self.delegate onIsDrawCtrlViewDrawing:NO];
|
||||
self.hidden = NO;
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,44 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@class TUIMultimediaPath;
|
||||
@protocol TUIMultimediaDrawViewDelegate;
|
||||
|
||||
typedef NS_ENUM(NSInteger, DrawMode){
|
||||
GRAFFITI,
|
||||
MOSAIC
|
||||
};
|
||||
|
||||
@interface TUIMultimediaDrawView : UIView
|
||||
@property(nonatomic) UIColor *color;
|
||||
@property(nonatomic) CGFloat lineWidth;
|
||||
@property(readonly, nonatomic) NSInteger pathCount;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaDrawViewDelegate> delegate;
|
||||
@property(readonly, nonatomic) BOOL canUndo;
|
||||
@property(readonly, nonatomic) BOOL canRedo;
|
||||
@property(nonatomic) DrawMode drawMode;
|
||||
@property (nonatomic, strong) UIImage *mosaciOriginalImage;
|
||||
|
||||
- (void)clear;
|
||||
- (void)undo;
|
||||
- (void)redo;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaDrawViewDelegate <NSObject>
|
||||
- (void)drawViewPathListChanged:(TUIMultimediaDrawView *)v;
|
||||
- (void)drawViewDrawStarted:(TUIMultimediaDrawView *)v;
|
||||
- (void)drawViewDrawEnded:(TUIMultimediaDrawView *)v;
|
||||
@end
|
||||
|
||||
|
||||
@interface TUIMultimediaPath : NSObject
|
||||
@property(nonatomic) CGSize originCanvasSize;
|
||||
@property(nonatomic, nonnull) NSMutableArray<NSValue *> *pathPoints;
|
||||
@property(nonatomic, nonnull) UIColor *color;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,112 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaDrawView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaColorPanel.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGeometry.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaMosaicDrawView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGraffitiDrawView.h"
|
||||
|
||||
@interface TUIMultimediaDrawView () {
|
||||
TUIMultimediaMosaicDrawView* _mosaciDrawView;
|
||||
TUIMultimediaGraffitiDrawView* _graffitiDrawView;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaDrawView
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
self.backgroundColor = UIColor.clearColor;
|
||||
UIPanGestureRecognizer *panRec = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(onPan:)];
|
||||
panRec.maximumNumberOfTouches = 1;
|
||||
[self addGestureRecognizer:panRec];
|
||||
|
||||
_mosaciDrawView = [[TUIMultimediaMosaicDrawView alloc] initWithFrame:self.bounds];
|
||||
[self addSubview:_mosaciDrawView];
|
||||
[_mosaciDrawView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
|
||||
_graffitiDrawView = [[TUIMultimediaGraffitiDrawView alloc] initWithFrame:self.bounds];
|
||||
[self addSubview:_graffitiDrawView];
|
||||
[_graffitiDrawView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (BOOL)canUndo {
|
||||
return _drawMode == GRAFFITI ? _graffitiDrawView.canUndo : _mosaciDrawView.canUndo;
|
||||
}
|
||||
|
||||
- (BOOL)canRedo {
|
||||
return _drawMode == GRAFFITI ? _graffitiDrawView.canRedo : _mosaciDrawView.canRedo;
|
||||
}
|
||||
|
||||
- (void)clear {
|
||||
[_graffitiDrawView clearGraffiti];
|
||||
[_mosaciDrawView clearMosaic];
|
||||
}
|
||||
|
||||
- (void)undo {
|
||||
if (_drawMode == GRAFFITI) {
|
||||
[_graffitiDrawView undo];
|
||||
} else {
|
||||
[_mosaciDrawView undo];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)redo {
|
||||
if (_drawMode == GRAFFITI) {
|
||||
[_graffitiDrawView redo];
|
||||
} else {
|
||||
[_mosaciDrawView redo];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)onPan:(UIPanGestureRecognizer *)rec {
|
||||
switch (rec.state) {
|
||||
case UIGestureRecognizerStateBegan:
|
||||
case UIGestureRecognizerStateChanged: {
|
||||
CGPoint p = [rec locationInView:self];
|
||||
if (_drawMode == MOSAIC) {
|
||||
[_mosaciDrawView addPathPoint:p];
|
||||
} else {
|
||||
[_graffitiDrawView addPathPoint:p];
|
||||
}
|
||||
[_delegate drawViewDrawStarted:self];
|
||||
break;
|
||||
}
|
||||
case UIGestureRecognizerStateEnded: {
|
||||
if (_drawMode == MOSAIC) {
|
||||
[_mosaciDrawView completeAddPoint];
|
||||
} else {
|
||||
[_graffitiDrawView completeAddPoint];
|
||||
}
|
||||
[_delegate drawViewDrawEnded:self];
|
||||
[_delegate drawViewPathListChanged:self];
|
||||
break;
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
- (NSInteger) pathCount {
|
||||
return _mosaciDrawView.pathCount + _graffitiDrawView.pathCount;
|
||||
}
|
||||
|
||||
-(void) setMosaciOriginalImage:(UIImage *)mosaciOriginalImage {
|
||||
_mosaciDrawView.originalImage = mosaciOriginalImage;
|
||||
}
|
||||
|
||||
-(void) setColor:(UIColor *)color {
|
||||
_graffitiDrawView.color = color;
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,21 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@interface TUIMultimediaGraffitiDrawView : UIView
|
||||
@property(nonatomic) UIColor *color;
|
||||
@property(readonly, nonatomic) BOOL canUndo;
|
||||
@property(readonly, nonatomic) BOOL canRedo;
|
||||
@property(readonly, nonatomic) NSInteger pathCount;
|
||||
|
||||
- (void)clearGraffiti;
|
||||
- (void)undo;
|
||||
- (void)redo;
|
||||
- (void)addPathPoint:(CGPoint) point;
|
||||
- (void)completeAddPoint;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,148 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaGraffitiDrawView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaColorPanel.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGeometry.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaDrawView.h"
|
||||
|
||||
#define GRAFFITI_LINE_WIDTH 4
|
||||
#define CGPointMultiply(point, scalar) CGPointMake(point.x * scalar, point.y * scalar)
|
||||
|
||||
@interface TUIMultimediaGraffitiDrawView () {
|
||||
NSMutableArray<NSValue *> *_currentPathPoints;
|
||||
UIBezierPath *_currentPath;
|
||||
NSMutableArray<TUIMultimediaPath *> *_pathList;
|
||||
NSMutableArray<TUIMultimediaPath *> *_redoPathList;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaGraffitiDrawView
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
self.backgroundColor = UIColor.clearColor;
|
||||
_pathList = [NSMutableArray array];
|
||||
_redoPathList = [NSMutableArray array];
|
||||
_currentPathPoints = [NSMutableArray array];
|
||||
_color = UIColor.whiteColor;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (NSInteger) pathCount {
|
||||
return _pathList.count;
|
||||
}
|
||||
|
||||
- (BOOL)canUndo {
|
||||
return _pathList.count != 0;
|
||||
}
|
||||
|
||||
- (BOOL)canRedo {
|
||||
return _redoPathList.count != 0;
|
||||
}
|
||||
|
||||
- (void)clearGraffiti {
|
||||
[_pathList removeAllObjects];
|
||||
[_redoPathList removeAllObjects];
|
||||
[_currentPath removeAllPoints];
|
||||
[_currentPathPoints removeAllObjects];
|
||||
[self setNeedsDisplay];
|
||||
}
|
||||
|
||||
- (void)undo {
|
||||
if (_pathList.count == 0) {
|
||||
return;
|
||||
}
|
||||
TUIMultimediaPath *path = [_pathList lastObject];
|
||||
[_pathList removeLastObject];
|
||||
[_redoPathList addObject:path];
|
||||
[self setNeedsDisplay];
|
||||
}
|
||||
|
||||
- (void)redo {
|
||||
if (_redoPathList.count == 0) {
|
||||
return;
|
||||
}
|
||||
TUIMultimediaPath *path = [_redoPathList lastObject];
|
||||
[_redoPathList removeLastObject];
|
||||
[_pathList addObject:path];
|
||||
[self setNeedsDisplay];
|
||||
}
|
||||
|
||||
- (void)drawRect:(CGRect)rect {
|
||||
[super drawRect:rect];
|
||||
|
||||
[[UIColor clearColor] setFill];
|
||||
UIRectFill(rect);
|
||||
|
||||
for (TUIMultimediaPath *path in _pathList) {
|
||||
[path.color set];
|
||||
CGFloat scale = self.frame.size.width / path.originCanvasSize.width;
|
||||
UIBezierPath * bezierPath = [TUIMultimediaGraffitiDrawView smoothBezierPathFromPoints:path.pathPoints scale:scale];
|
||||
[bezierPath stroke];
|
||||
}
|
||||
|
||||
if (_currentPath != nil) {
|
||||
[_color set];
|
||||
[_currentPath stroke];
|
||||
}
|
||||
}
|
||||
|
||||
- (NSArray<TUIMultimediaPath *> *)pathList {
|
||||
return _pathList;
|
||||
}
|
||||
|
||||
- (void)addPathPoint:(CGPoint) point {
|
||||
[_currentPathPoints addObject:@(point)];
|
||||
_currentPath = [TUIMultimediaGraffitiDrawView smoothBezierPathFromPoints:_currentPathPoints scale:1.0];
|
||||
[self setNeedsDisplay];
|
||||
}
|
||||
|
||||
- (void)completeAddPoint {
|
||||
TUIMultimediaPath *path = [[TUIMultimediaPath alloc] init];
|
||||
path.pathPoints = [[NSMutableArray alloc] initWithArray:_currentPathPoints copyItems:YES];
|
||||
path.color = _color;
|
||||
path.originCanvasSize = self.frame.size;
|
||||
[_pathList addObject:path];
|
||||
[_redoPathList removeAllObjects];
|
||||
[_currentPathPoints removeAllObjects];
|
||||
_currentPath = nil;
|
||||
}
|
||||
|
||||
+ (UIBezierPath *)smoothBezierPathFromPoints:(NSMutableArray<NSValue *> *) pathPoints scale:(CGFloat)scale{
|
||||
UIBezierPath *path = [UIBezierPath bezierPath];
|
||||
path.lineCapStyle = kCGLineCapRound;
|
||||
path.lineJoinStyle = kCGLineJoinRound;
|
||||
path.lineWidth = GRAFFITI_LINE_WIDTH * scale;
|
||||
|
||||
if (pathPoints.count <= 1) {
|
||||
return nil;
|
||||
}
|
||||
[path moveToPoint:CGPointMultiply(pathPoints.firstObject.CGPointValue, scale)];
|
||||
if (pathPoints.count == 2) {
|
||||
[path addLineToPoint:CGPointMultiply(pathPoints.lastObject.CGPointValue,scale)];
|
||||
return path;
|
||||
}
|
||||
|
||||
for (int i = 3; i < pathPoints.count; i++) {
|
||||
CGPoint p = CGPointMultiply(pathPoints[i].CGPointValue, scale);
|
||||
CGPoint prev = CGPointMultiply(pathPoints[i - 1].CGPointValue, scale);
|
||||
CGPoint midPoint = Vec2Mul(Vec2AddVector(p, prev), 0.5);
|
||||
[path addQuadCurveToPoint:midPoint controlPoint:prev];
|
||||
}
|
||||
[path addLineToPoint:CGPointMultiply(pathPoints.lastObject.CGPointValue, scale)];
|
||||
return path;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaPath
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
_color = UIColor.clearColor;
|
||||
return self;
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,17 @@
|
||||
// Copyright (c) 2025 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
@interface TUIMultimediaMosaicDrawView : UIImageView
|
||||
@property (nonatomic, strong) UIImage *originalImage;
|
||||
@property(readonly, nonatomic) BOOL canUndo;
|
||||
@property(readonly, nonatomic) BOOL canRedo;
|
||||
@property(readonly, nonatomic) NSInteger pathCount;
|
||||
|
||||
- (void)clearMosaic;
|
||||
- (void)undo;
|
||||
- (void)redo;
|
||||
- (void)addPathPoint:(CGPoint) point;
|
||||
- (void)completeAddPoint;
|
||||
@end
|
||||
@@ -0,0 +1,172 @@
|
||||
// Copyright (c) 2025 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaMosaicDrawView.h"
|
||||
#import <CoreImage/CoreImage.h>
|
||||
#import <Masonry/Masonry.h>
|
||||
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaDrawView.h"
|
||||
|
||||
#define MOSAIC_LINE_WIDTH 20
|
||||
#define MOSAIC_SCALE_VALUE 50
|
||||
|
||||
@interface TUIMultimediaMosaicDrawView () {
|
||||
UIBezierPath *_currentPath;
|
||||
|
||||
NSMutableArray<CAShapeLayer *> *_layerList;
|
||||
NSMutableArray<CAShapeLayer *> *_redoLayerList;
|
||||
|
||||
CALayer *_maskParentLayer;
|
||||
CAShapeLayer *_currentShapeLayer;
|
||||
BOOL _isAdddPathFirstPoint;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaMosaicDrawView
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame {
|
||||
if (self = [super initWithFrame:frame]) {
|
||||
[self setupView];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)setupView {
|
||||
NSLog(@"Mosaic setupView");
|
||||
_maskParentLayer = [CALayer layer];
|
||||
_maskParentLayer.frame = self.bounds;
|
||||
self.layer.mask = _maskParentLayer;
|
||||
|
||||
_layerList = [NSMutableArray array];
|
||||
_redoLayerList = [NSMutableArray array];
|
||||
_isAdddPathFirstPoint = YES;
|
||||
}
|
||||
|
||||
- (void)setOriginalImage:(UIImage *)originalImage {
|
||||
NSLog(@"Mosaic setOriginalImage");
|
||||
_originalImage = originalImage;
|
||||
self.image = [self generateMosaicImage:originalImage];
|
||||
}
|
||||
|
||||
-(void)addPathPoint:(CGPoint) point {
|
||||
if (_isAdddPathFirstPoint) {
|
||||
_isAdddPathFirstPoint = NO;
|
||||
_currentShapeLayer = [self createShapeLayer];
|
||||
[_maskParentLayer addSublayer:_currentShapeLayer];
|
||||
[_layerList addObject:_currentShapeLayer];
|
||||
_currentPath = [UIBezierPath bezierPath];
|
||||
[_currentPath moveToPoint:point];
|
||||
} else {
|
||||
[_currentPath addLineToPoint:point];
|
||||
_currentShapeLayer.path = _currentPath.CGPath;
|
||||
}
|
||||
}
|
||||
|
||||
-(CAShapeLayer*)createShapeLayer{
|
||||
CAShapeLayer* shapeLayer = [CAShapeLayer layer];
|
||||
shapeLayer.frame = self.bounds;
|
||||
shapeLayer.lineCap = kCALineCapRound;
|
||||
shapeLayer.lineJoin = kCALineJoinRound;
|
||||
shapeLayer.lineWidth = MOSAIC_LINE_WIDTH;
|
||||
shapeLayer.strokeColor = [UIColor blackColor].CGColor;
|
||||
shapeLayer.fillColor = nil;
|
||||
return shapeLayer;
|
||||
}
|
||||
|
||||
-(void)completeAddPoint {
|
||||
_isAdddPathFirstPoint = YES;
|
||||
for (CAShapeLayer * layer in _redoLayerList) {
|
||||
[self safeRemoveShapeLayer:layer];
|
||||
}
|
||||
[_redoLayerList removeAllObjects];
|
||||
}
|
||||
|
||||
- (NSInteger) pathCount {
|
||||
return _layerList.count;
|
||||
}
|
||||
|
||||
- (BOOL)canUndo {
|
||||
return _layerList.count != 0;
|
||||
}
|
||||
|
||||
- (BOOL)canRedo {
|
||||
return _redoLayerList.count != 0;
|
||||
}
|
||||
|
||||
- (void)clearMosaic {
|
||||
for (CAShapeLayer * layer in _layerList) {
|
||||
[self safeRemoveShapeLayer:layer];
|
||||
}
|
||||
|
||||
for (CAShapeLayer * layer in _redoLayerList) {
|
||||
[self safeRemoveShapeLayer:layer];
|
||||
}
|
||||
|
||||
[_currentPath removeAllPoints];
|
||||
[_layerList removeAllObjects];
|
||||
[_redoLayerList removeAllObjects];
|
||||
}
|
||||
|
||||
- (void)undo {
|
||||
if (_layerList.count == 0) {
|
||||
return;
|
||||
}
|
||||
CAShapeLayer *layer = [_layerList lastObject];
|
||||
[_layerList removeLastObject];
|
||||
layer.hidden = YES;
|
||||
[_redoLayerList addObject:layer];
|
||||
}
|
||||
|
||||
- (void)redo {
|
||||
if (_redoLayerList.count == 0) {
|
||||
return;
|
||||
}
|
||||
CAShapeLayer *layer = [_redoLayerList lastObject];
|
||||
[_redoLayerList removeLastObject];
|
||||
layer.hidden = NO;
|
||||
[_layerList addObject:layer];
|
||||
}
|
||||
|
||||
|
||||
- (void)layoutSubviews {
|
||||
[super layoutSubviews];
|
||||
_maskParentLayer.frame = self.bounds;
|
||||
|
||||
for (CAShapeLayer *layer in _maskParentLayer.sublayers) {
|
||||
if ([layer isKindOfClass:[CAShapeLayer class]]) {
|
||||
float scale = self.bounds.size.width / layer.frame.size.width;
|
||||
layer.frame = self.bounds;
|
||||
CGAffineTransform transform = CGAffineTransformMakeScale(scale, scale);
|
||||
CGPathRef scaledPath = CGPathCreateCopyByTransformingPath(layer.path, &transform);
|
||||
layer.lineWidth = layer.lineWidth * scale;
|
||||
layer.path = scaledPath;
|
||||
CGPathRelease(scaledPath);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
- (UIImage *)generateMosaicImage:(UIImage *)inputImage {
|
||||
CIImage *ciImage = [[CIImage alloc] initWithImage:inputImage];
|
||||
|
||||
CIFilter *filter = [CIFilter filterWithName:@"CIPixellate"];
|
||||
[filter setValue:ciImage forKey:kCIInputImageKey];
|
||||
[filter setValue:@(MOSAIC_SCALE_VALUE) forKey:kCIInputScaleKey];
|
||||
|
||||
CIImage *outputImage = filter.outputImage;
|
||||
CIContext *context = [CIContext contextWithOptions:nil];
|
||||
CGImageRef cgImage = [context createCGImage:outputImage fromRect:outputImage.extent];
|
||||
return [UIImage imageWithCGImage:cgImage];
|
||||
}
|
||||
|
||||
- (void)safeRemoveShapeLayer:(CAShapeLayer *)layer {
|
||||
if (!layer) return;
|
||||
|
||||
[layer removeAllAnimations];
|
||||
layer.delegate = nil;
|
||||
layer.path = NULL;
|
||||
layer.fillColor = NULL;
|
||||
layer.strokeColor = NULL;
|
||||
[layer removeFromSuperlayer];
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,27 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterConfig.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaPasterSelectViewDelegate;
|
||||
|
||||
@interface TUIMultimediaPasterSelectView : UIView
|
||||
@property(nonatomic) TUIMultimediaPasterConfig *config;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaPasterSelectViewDelegate> delegate;
|
||||
|
||||
- (instancetype)initWithFrame:(CGRect)frame;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaPasterSelectViewDelegate <NSObject>
|
||||
- (void)onPasterSelected:(UIImage *)image;
|
||||
- (void)pasterSelectView:(TUIMultimediaPasterSelectView *)v needAddCustomPaster:(TUIMultimediaPasterGroupConfig *)group completeCallback:(void (^)(void))callback;
|
||||
- (void)pasterSelectView:(TUIMultimediaPasterSelectView *)v
|
||||
needDeleteCustomPasterInGroup:(TUIMultimediaPasterGroupConfig *)group
|
||||
index:(NSInteger)index
|
||||
completeCallback:(void (^)(BOOL deleted))callback;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,168 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaPasterSelectView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImageCell.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaTabPanel.h"
|
||||
|
||||
static const CGFloat ItemInsect = 10;
|
||||
static const CGFloat ItemCountPerLine = 5;
|
||||
static const CGFloat ItemCountPerColumn = 3;
|
||||
|
||||
@interface TUIMultimediaPasterSelectView () <UICollectionViewDelegate, UICollectionViewDataSource, UICollectionViewDelegateFlowLayout, TUIMultimediaTabPanelDelegate> {
|
||||
TUIMultimediaTabPanel *_tabPanel;
|
||||
UILongPressGestureRecognizer *_longPressRec;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaPasterSelectView
|
||||
- (instancetype)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
_config = [[TUIMultimediaPasterConfig alloc] init];
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)reloadTabPanel {
|
||||
for (TUIMultimediaTabPanelTab *tab in _tabPanel.tabs) {
|
||||
if (![tab.view isKindOfClass:UICollectionView.class]) continue;
|
||||
UICollectionView *v = (UICollectionView *)tab.view;
|
||||
[v reloadData];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)layoutSubviews {
|
||||
[super layoutSubviews];
|
||||
for (TUIMultimediaTabPanelTab *tab in _tabPanel.tabs) {
|
||||
if (![tab.view isKindOfClass:UICollectionView.class]) continue;
|
||||
UICollectionView *v = (UICollectionView *)tab.view;
|
||||
CGFloat len = (self.bounds.size.width - ItemInsect * (ItemCountPerLine - 1)) / ItemCountPerLine;
|
||||
[v mas_updateConstraints:^(MASConstraintMaker *make) {
|
||||
make.width.equalTo(self);
|
||||
make.height.mas_equalTo(len * ItemCountPerColumn + ItemInsect * (ItemCountPerColumn - 1));
|
||||
}];
|
||||
[v reloadData];
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - UI init
|
||||
|
||||
- (void)initUI {
|
||||
_tabPanel = [[TUIMultimediaTabPanel alloc] init];
|
||||
_tabPanel.backgroundColor = TUIMultimediaPluginDynamicColor(@"editor_popup_view_bg_color", @"#000000BF");
|
||||
_tabPanel.delegate = self;
|
||||
[self addSubview:_tabPanel];
|
||||
|
||||
[_tabPanel mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self);
|
||||
}];
|
||||
|
||||
_longPressRec = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(onCollectionViewLongPress:)];
|
||||
}
|
||||
|
||||
#pragma mark - UICollectionViewDataSource protocol
|
||||
- (__kindof UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath {
|
||||
TUIMultimediaImageCell *cell = (TUIMultimediaImageCell *)[collectionView dequeueReusableCellWithReuseIdentifier:TUIMultimediaImageCell.reuseIdentifier forIndexPath:indexPath];
|
||||
TUIMultimediaPasterGroupConfig *config = _config.groups[collectionView.tag];
|
||||
cell.image = [config.itemList[indexPath.item] loadIcon];
|
||||
return cell;
|
||||
}
|
||||
|
||||
- (NSInteger)collectionView:(UICollectionView *)collectionView numberOfItemsInSection:(NSInteger)section {
|
||||
TUIMultimediaPasterGroupConfig *config = _config.groups[collectionView.tag];
|
||||
return config.itemList.count;
|
||||
}
|
||||
|
||||
#pragma mark - UICollectionViewDelegate protocol
|
||||
- (void)collectionView:(UICollectionView *)collectionView didSelectItemAtIndexPath:(NSIndexPath *)indexPath {
|
||||
TUIMultimediaPasterGroupConfig *config = _config.groups[collectionView.tag];
|
||||
TUIMultimediaPasterItemConfig *item = config.itemList[indexPath.item];
|
||||
if (item.isAddButton) {
|
||||
[_delegate pasterSelectView:self
|
||||
needAddCustomPaster:config
|
||||
completeCallback:^{
|
||||
[self reloadTabPanel];
|
||||
}];
|
||||
return;
|
||||
}
|
||||
[_delegate onPasterSelected:[item loadImage]];
|
||||
}
|
||||
|
||||
- (CGSize)collectionView:(UICollectionView *)collectionView
|
||||
layout:(UICollectionViewLayout *)collectionViewLayout
|
||||
sizeForItemAtIndexPath:(NSIndexPath *)indexPath {
|
||||
CGFloat len = (self.bounds.size.width - ItemInsect * (ItemCountPerLine + 1)) / ItemCountPerLine;
|
||||
return CGSizeMake(len, len);
|
||||
}
|
||||
|
||||
- (CGFloat)collectionView:(UICollectionView *)collectionView
|
||||
layout:(UICollectionViewLayout *)collectionViewLayout
|
||||
minimumLineSpacingForSectionAtIndex:(NSInteger)section {
|
||||
return ItemInsect;
|
||||
}
|
||||
- (CGFloat)collectionView:(UICollectionView *)collectionView
|
||||
layout:(UICollectionViewLayout *)collectionViewLayout
|
||||
minimumInteritemSpacingForSectionAtIndex:(NSInteger)section {
|
||||
return ItemInsect;
|
||||
}
|
||||
- (UIEdgeInsets)collectionView:(UICollectionView *)collectionView
|
||||
layout:(UICollectionViewLayout *)collectionViewLayout
|
||||
insetForSectionAtIndex:(NSInteger)section {
|
||||
return UIEdgeInsetsMake(ItemInsect, ItemInsect, 0, ItemInsect);
|
||||
}
|
||||
|
||||
#pragma mark - Actions
|
||||
- (void)onCollectionViewLongPress:(UILongPressGestureRecognizer *)rec {
|
||||
UICollectionView *collectionView = (UICollectionView *)rec.view;
|
||||
CGPoint p = [rec locationInView:collectionView];
|
||||
NSIndexPath *indexPath = [collectionView indexPathForItemAtPoint:p];
|
||||
if (indexPath == nil) {
|
||||
return;
|
||||
}
|
||||
TUIMultimediaPasterGroupConfig *config = _config.groups[_tabPanel.selectedIndex];
|
||||
TUIMultimediaPasterItemConfig *item = config.itemList[indexPath.item];
|
||||
if (!item.isUserAdded) {
|
||||
return;
|
||||
}
|
||||
[_delegate pasterSelectView:self
|
||||
needDeleteCustomPasterInGroup:_config.groups[_tabPanel.selectedIndex]
|
||||
index:indexPath.item
|
||||
completeCallback:^(BOOL deleted) {
|
||||
if (deleted) {
|
||||
[collectionView reloadData];
|
||||
}
|
||||
}];
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaTabPanelDelegate
|
||||
- (void)tabPanel:(TUIMultimediaTabPanel *)panel selectedIndexChanged:(NSInteger)selectedIndex {
|
||||
[panel.tabs[selectedIndex].view addGestureRecognizer:_longPressRec];
|
||||
}
|
||||
|
||||
#pragma mark - Properties
|
||||
- (void)setConfig:(TUIMultimediaPasterConfig *)config {
|
||||
_config = config;
|
||||
_tabPanel.tabs = [_config.groups tui_multimedia_mapWithIndex:^id(TUIMultimediaPasterGroupConfig *config, NSUInteger idx) {
|
||||
UICollectionViewFlowLayout *layout = [[UICollectionViewFlowLayout alloc] init];
|
||||
layout.scrollDirection = UICollectionViewScrollDirectionVertical;
|
||||
UICollectionView *cv = [[UICollectionView alloc] initWithFrame:CGRectZero collectionViewLayout:layout];
|
||||
cv.backgroundColor = UIColor.clearColor;
|
||||
cv.showsVerticalScrollIndicator = NO;
|
||||
cv.delegate = self;
|
||||
cv.dataSource = self;
|
||||
cv.tag = idx;
|
||||
[cv registerClass:TUIMultimediaImageCell.class forCellWithReuseIdentifier:TUIMultimediaImageCell.reuseIdentifier];
|
||||
NSString *localizedName = [TUIMultimediaCommon localizedStringForKey:config.name];
|
||||
UIImage *icon = [config loadIcon];
|
||||
return [[TUIMultimediaTabPanelTab alloc] initWithName:localizedName icon:icon view:cv];
|
||||
}];
|
||||
[_tabPanel.tabs.firstObject.view addGestureRecognizer:_longPressRec];
|
||||
}
|
||||
@end
|
||||
@@ -0,0 +1,22 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSubtitleInfo.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaSubtitleEditViewDelegate;
|
||||
|
||||
@interface TUIMultimediaSubtitleEditView : UIView
|
||||
@property(nonatomic) TUIMultimediaSubtitleInfo *subtitleInfo;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaSubtitleEditViewDelegate> delegate;
|
||||
- (void)activate;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaSubtitleEditViewDelegate <NSObject>
|
||||
- (void)subtitleEditViewOnOk:(TUIMultimediaSubtitleEditView *)view;
|
||||
- (void)subtitleEditViewOnCancel:(TUIMultimediaSubtitleEditView *)view;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,144 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaSubtitleEditView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaColorPanel.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaGeometry.h"
|
||||
|
||||
@interface TUIMultimediaSubtitleEditView () <TUIMultimediaColorPanelDelegate, UITextViewDelegate> {
|
||||
UIButton *_btnOk;
|
||||
UIButton *_btnCancel;
|
||||
UITextView *_textView;
|
||||
TUIMultimediaColorPanel *_colorPanel;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaSubtitleEditView
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
_subtitleInfo = [[TUIMultimediaSubtitleInfo alloc] init];
|
||||
[self initUI];
|
||||
[NSNotificationCenter.defaultCenter addObserver:self selector:@selector(keyboardWillShow:) name:UIKeyboardWillShowNotification object:nil];
|
||||
[NSNotificationCenter.defaultCenter addObserver:self selector:@selector(keyboardWillHide:) name:UIKeyboardWillHideNotification object:nil];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)activate {
|
||||
[_textView becomeFirstResponder];
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
self.backgroundColor = TUIMultimediaPluginDynamicColor(@"editor_popup_view_bg_color", @"#000000BF");
|
||||
|
||||
_btnOk = [UIButton buttonWithType:UIButtonTypeSystem];
|
||||
[_btnOk setTitle:[TUIMultimediaCommon localizedStringForKey:@"ok"] forState:UIControlStateNormal];
|
||||
[_btnOk setTitleColor:UIColor.whiteColor forState:UIControlStateNormal];
|
||||
_btnOk.titleLabel.font = [UIFont systemFontOfSize:20];
|
||||
[_btnOk addTarget:self action:@selector(onOk) forControlEvents:UIControlEventTouchUpInside];
|
||||
[self addSubview:_btnOk];
|
||||
|
||||
_btnCancel = [UIButton buttonWithType:UIButtonTypeSystem];
|
||||
[_btnCancel setTitle:[TUIMultimediaCommon localizedStringForKey:@"cancel"] forState:UIControlStateNormal];
|
||||
[_btnCancel setTitleColor:UIColor.whiteColor forState:UIControlStateNormal];
|
||||
_btnCancel.titleLabel.font = [UIFont systemFontOfSize:20];
|
||||
_btnCancel.titleLabel.textColor = UIColor.whiteColor;
|
||||
[_btnCancel addTarget:self action:@selector(onCancel) forControlEvents:UIControlEventTouchUpInside];
|
||||
[self addSubview:_btnCancel];
|
||||
|
||||
_textView = [[UITextView alloc] init];
|
||||
_textView.backgroundColor = UIColor.clearColor;
|
||||
_textView.font = [UIFont systemFontOfSize:20];
|
||||
_textView.text = _subtitleInfo.text;
|
||||
_textView.textColor = _subtitleInfo.color;
|
||||
_textView.delegate = self;
|
||||
[self addSubview:_textView];
|
||||
|
||||
_colorPanel = [[TUIMultimediaColorPanel alloc] init];
|
||||
_colorPanel.delegate = self;
|
||||
[self addSubview:_colorPanel];
|
||||
|
||||
[_btnOk mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.top.equalTo(self.mas_safeAreaLayoutGuideTop);
|
||||
make.right.equalTo(self).inset(30);
|
||||
make.width.mas_greaterThanOrEqualTo(100);
|
||||
make.height.mas_greaterThanOrEqualTo(50);
|
||||
}];
|
||||
[_btnCancel mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.top.equalTo(self.mas_safeAreaLayoutGuideTop);
|
||||
make.left.equalTo(self).inset(30);
|
||||
make.width.mas_greaterThanOrEqualTo(100);
|
||||
make.height.mas_greaterThanOrEqualTo(50);
|
||||
}];
|
||||
[_textView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.left.right.equalTo(self).inset(50);
|
||||
make.top.equalTo(_btnOk.mas_bottom).inset(50);
|
||||
make.bottom.equalTo(_colorPanel.mas_top).inset(10);
|
||||
}];
|
||||
[_colorPanel mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.left.right.equalTo(self).inset(10);
|
||||
make.height.mas_equalTo(32);
|
||||
make.bottom.equalTo(self);
|
||||
}];
|
||||
}
|
||||
|
||||
#pragma mark - Actions
|
||||
- (void)onOk {
|
||||
NSMutableString *wrappedText = [NSMutableString string];
|
||||
NSString *text = _textView.text;
|
||||
NSLayoutManager *layoutManager = _textView.layoutManager;
|
||||
NSUInteger numberOfLines, index;
|
||||
NSUInteger numberOfGlyphs = [layoutManager numberOfGlyphs];
|
||||
BOOL lastLineBreak = NO;
|
||||
for (numberOfLines = 0, index = 0; index < numberOfGlyphs; numberOfLines++) {
|
||||
NSRange lineRange;
|
||||
[layoutManager lineFragmentRectForGlyphAtIndex:index effectiveRange:&lineRange];
|
||||
// NSLog(@"TUIMultimedia Subtitle Line:%@", [text substringWithRange:lineRange]);
|
||||
index = NSMaxRange(lineRange);
|
||||
NSString *line = [text substringWithRange:lineRange];
|
||||
if (numberOfLines != 0 && !lastLineBreak) {
|
||||
[wrappedText appendString:@"\n"];
|
||||
}
|
||||
[wrappedText appendString:line];
|
||||
lastLineBreak = [line containsString:@"\n"];
|
||||
}
|
||||
_subtitleInfo.text = _textView.text;
|
||||
_subtitleInfo.wrappedText = wrappedText;
|
||||
[_delegate subtitleEditViewOnOk:self];
|
||||
}
|
||||
|
||||
- (void)onCancel {
|
||||
[_delegate subtitleEditViewOnCancel:self];
|
||||
}
|
||||
|
||||
- (void)keyboardWillShow:(NSNotification *)notification {
|
||||
CGRect rect = [notification.userInfo[UIKeyboardFrameEndUserInfoKey] CGRectValue];
|
||||
[_colorPanel mas_updateConstraints:^(MASConstraintMaker *make) {
|
||||
make.bottom.equalTo(self).offset(-CGRectGetHeight(rect));
|
||||
}];
|
||||
}
|
||||
|
||||
- (void)keyboardWillHide:(NSNotification *)notification {
|
||||
[self onCancel];
|
||||
}
|
||||
|
||||
#pragma mark - UITextViewDelegate protocol
|
||||
- (void)onColorPanel:(TUIMultimediaColorPanel *)panel selectColor:(UIColor *)color {
|
||||
_subtitleInfo.color = color;
|
||||
_textView.textColor = color;
|
||||
}
|
||||
|
||||
#pragma mark - Setters
|
||||
- (void)setSubtitleInfo:(TUIMultimediaSubtitleInfo *)subtitleInfo {
|
||||
_subtitleInfo = subtitleInfo;
|
||||
_textView.textColor = _subtitleInfo.color;
|
||||
_textView.text = subtitleInfo.text;
|
||||
_colorPanel.selectedColor = _subtitleInfo.color;
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,17 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <AVFoundation/AVFoundation.h>
|
||||
#import <UIKit/UIKit.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
typedef void (^TUIMultimediaPictureEditorControllerCallback)(UIImage * _Nullable outImage, int resultCode);
|
||||
|
||||
@interface TUIMultimediaPictureEditorController : UIViewController
|
||||
@property(nonatomic) UIImage *srcPicture;
|
||||
@property(nonatomic) TUIMultimediaPictureEditorControllerCallback completeCallback;
|
||||
@property(nonatomic)int sourceType;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,286 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaPictureEditorController.h"
|
||||
#import <ReactiveObjC/ReactiveObjC.h>
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TXLiteAVSDK_Professional/TXPictureEditer.h>
|
||||
#import <TXLiteAVSDK_Professional/TXVideoEditerTypeDef.h>
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommonEditorControlView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaConstant.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterSelectController.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSubtitleEditController.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSticker.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaAuthorizationPrompter.h"
|
||||
|
||||
@interface TUIMultimediaPictureEditorController () <TUIMultimediaCommonEditorControlViewDelegate, TUIMultimediaPasterSelectControllerDelegate> {
|
||||
TXPictureEditer *_editor;
|
||||
UIImageView *_imgView;
|
||||
|
||||
TUIMultimediaPasterSelectController *_pasterSelectController;
|
||||
TUIMultimediaSubtitleEditController *_subtitleEditController;
|
||||
TUIMultimediaCommonEditorControlView *_commonEditCtrlView;
|
||||
BOOL _originNavgationBarHidden;
|
||||
BOOL _hasCrop;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaPictureEditorController
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
_hasCrop = false;
|
||||
_sourceType = SOURCE_TYPE_RECORD;
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)viewDidLoad {
|
||||
_editor = [[TXPictureEditer alloc] init];
|
||||
|
||||
_pasterSelectController = [[TUIMultimediaPasterSelectController alloc] init];
|
||||
_pasterSelectController.delegate = self;
|
||||
_pasterSelectController.modalPresentationStyle = UIModalPresentationOverCurrentContext;
|
||||
|
||||
_subtitleEditController = [[TUIMultimediaSubtitleEditController alloc] init];
|
||||
_subtitleEditController.modalPresentationStyle = UIModalPresentationOverCurrentContext;
|
||||
|
||||
_commonEditCtrlView = [[TUIMultimediaCommonEditorControlView alloc] initWithConfig:TUIMultimediaCommonEditorConfig.configForPictureEditor];
|
||||
[self.view addSubview:_commonEditCtrlView];
|
||||
_commonEditCtrlView.delegate = self;
|
||||
_commonEditCtrlView.clipsToBounds = true;
|
||||
_commonEditCtrlView.sourceType = _sourceType;
|
||||
_commonEditCtrlView.frame = CGRectMake(0, 0, self.view.frame.size.width, self.view.frame.size.height);
|
||||
|
||||
_commonEditCtrlView.previewSize = _srcPicture.size;
|
||||
_imgView = [[UIImageView alloc] init];
|
||||
_imgView.image = _srcPicture;
|
||||
[_commonEditCtrlView.previewView addSubview:_imgView];
|
||||
[_imgView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(_commonEditCtrlView.previewView);
|
||||
}];
|
||||
_commonEditCtrlView.mosaciOriginalImage = _srcPicture;
|
||||
|
||||
UIPinchGestureRecognizer *pinchRec = [[UIPinchGestureRecognizer alloc] initWithTarget:self action:@selector(onPinch:)];
|
||||
[self.view addGestureRecognizer:pinchRec];
|
||||
|
||||
UIPanGestureRecognizer *panGesture = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(onPanGesture:)];
|
||||
[self.view addGestureRecognizer:panGesture];
|
||||
}
|
||||
|
||||
- (void)onPinch:(UIPinchGestureRecognizer *)gestureRecognizer {
|
||||
switch (gestureRecognizer.state) {
|
||||
case UIGestureRecognizerStateChanged: {
|
||||
if (gestureRecognizer.numberOfTouches < 2) {
|
||||
break;
|
||||
}
|
||||
CGPoint touchPoint1 = [gestureRecognizer locationOfTouch:0 inView:self.view];
|
||||
CGPoint touchPoint2 = [gestureRecognizer locationOfTouch:1 inView:self.view];
|
||||
CGPoint scaleCenter = CGPointMake((touchPoint1.x + touchPoint2.x) / 2, (touchPoint1.y + touchPoint2.y) / 2);
|
||||
[_commonEditCtrlView previewScale:gestureRecognizer.scale center:scaleCenter];
|
||||
gestureRecognizer.scale = 1.0f;
|
||||
break;
|
||||
}
|
||||
case UIGestureRecognizerStateEnded:
|
||||
[_commonEditCtrlView previewAdjustToLimitRect];
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
- (void)onPanGesture:(UIPanGestureRecognizer *)gestureRecognizer {
|
||||
switch (gestureRecognizer.state) {
|
||||
case UIGestureRecognizerStateChanged: {
|
||||
CGPoint offset = [gestureRecognizer translationInView:self.view];
|
||||
[gestureRecognizer setTranslation:CGPointZero inView:self.view];
|
||||
[_commonEditCtrlView previewMove:offset];
|
||||
break;
|
||||
}
|
||||
case UIGestureRecognizerStateEnded:
|
||||
[_commonEditCtrlView previewAdjustToLimitRect];
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
- (void)viewWillAppear:(BOOL)animated {
|
||||
if (self.navigationController != nil) {
|
||||
_originNavgationBarHidden = self.navigationController.navigationBarHidden;
|
||||
self.navigationController.navigationBarHidden = YES;
|
||||
}
|
||||
}
|
||||
|
||||
- (void)viewWillDisappear:(BOOL)animated {
|
||||
if (self.navigationController != nil) {
|
||||
self.navigationController.navigationBarHidden = _originNavgationBarHidden;
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - Properties
|
||||
- (void)setSrcPicture:(UIImage *)photo {
|
||||
_srcPicture = photo;
|
||||
_imgView.image = _srcPicture;
|
||||
_commonEditCtrlView.previewSize = _srcPicture.size;
|
||||
}
|
||||
|
||||
- (void)setSourceType:(int)sourceType {
|
||||
NSLog(@"setSourceType sourceType = %d",sourceType);
|
||||
_sourceType = sourceType;
|
||||
if (_commonEditCtrlView != nil) {
|
||||
_commonEditCtrlView.sourceType = sourceType;
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaCommonEditorControlViewDelegate protocol
|
||||
- (void)onCommonEditorControlViewCancel:(nonnull TUIMultimediaCommonEditorControlView *)view {
|
||||
if (_completeCallback == nil) {
|
||||
return;
|
||||
}
|
||||
_completeCallback(nil, PHOTO_EDIT_RESULT_CODE_CANCEL);
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewCancelGenerate:(nonnull TUIMultimediaCommonEditorControlView *)view {
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewComplete:(nonnull TUIMultimediaCommonEditorControlView *)view
|
||||
stickers:(nonnull NSArray<TUIMultimediaSticker *> *)stickers {
|
||||
if (![TUIMultimediaAuthorizationPrompter verifyPermissionGranted:self]) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (_completeCallback == nil) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (stickers.count == 0) {
|
||||
NSLog(@"Return directly without edit the picture");
|
||||
UIImage* outputImage = (_hasCrop || _sourceType == SOURCE_TYPE_RECORD) ? _srcPicture : nil;
|
||||
_completeCallback(outputImage, _hasCrop ? VIDEO_EDIT_RESULT_CODE_GENERATE_SUCCESS : VIDEO_EDIT_RESULT_CODE_NO_EDIT);
|
||||
return;
|
||||
}
|
||||
|
||||
[_editor setPicture:_srcPicture];
|
||||
[_editor setOutputRotation:0];
|
||||
[_editor setOutputSize:_srcPicture.size.width height:_srcPicture.size.height];
|
||||
[_editor setPasterList:[self stickercConvertToPaster:stickers rotationAngle:0]];
|
||||
@weakify(self)
|
||||
[_editor processPicture:^(UIImage *processedPicture) {
|
||||
@strongify(self)
|
||||
self.completeCallback(processedPicture, PHOTO_EDIT_RESULT_CODE_GENERATE_SUCCESS);
|
||||
}];
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewCrop:(NSInteger)rotationAngle normalizedCropRect:(CGRect)normalizedCropRect
|
||||
stickers:(NSArray<TUIMultimediaSticker *> *)stickers {
|
||||
[_editor setPicture:_srcPicture];
|
||||
[_editor setPasterList:[self stickercConvertToPaster:stickers rotationAngle:rotationAngle]];
|
||||
normalizedCropRect = [self adjustCropRectAccordingRotaionAngle:normalizedCropRect rotationAngle:rotationAngle];
|
||||
CGRect cropRect = CGRectMake(normalizedCropRect.origin.x * _srcPicture.size.width * _srcPicture.scale,
|
||||
normalizedCropRect.origin.y * _srcPicture.size.height * _srcPicture.scale,
|
||||
normalizedCropRect.size.width * _srcPicture.size.width * _srcPicture.scale,
|
||||
normalizedCropRect.size.height * _srcPicture.size.height * _srcPicture.scale);
|
||||
[_editor setCropRect:cropRect];
|
||||
|
||||
|
||||
CGSize outputSize = [self getOutputSize:cropRect.size rotationAngle:rotationAngle];
|
||||
[_editor setOutputSize:outputSize.width height:outputSize.height];
|
||||
[_editor setOutputRotation:(CGFloat)rotationAngle];
|
||||
[_editor processPicture:^(UIImage *processedPicture) {
|
||||
[self onProcessPictureForCrop:processedPicture];
|
||||
}];
|
||||
}
|
||||
|
||||
- (CGSize) getOutputSize:(CGSize) cropRectSize rotationAngle:(CGFloat) rotationAngle {
|
||||
int outputWidth = 1080;
|
||||
int outputHeight = 1920;
|
||||
if(((int)rotationAngle + 180) % 180 == 0) {
|
||||
outputHeight = cropRectSize.height / cropRectSize.width * outputWidth;
|
||||
} else {
|
||||
outputHeight = cropRectSize.width / cropRectSize.height * outputWidth;
|
||||
}
|
||||
return CGSizeMake(outputWidth, outputHeight);
|
||||
}
|
||||
|
||||
- (NSArray<TXPaster *> *) stickercConvertToPaster:(nonnull NSArray<TUIMultimediaSticker *> *)stickers rotationAngle:(NSInteger)rotationAngle {
|
||||
CGSize previewSize = _commonEditCtrlView.previewView.frame.size;
|
||||
return [stickers tui_multimedia_map:^TXPaster *(TUIMultimediaSticker *s) {
|
||||
TXPaster *p = [[TXPaster alloc] init];
|
||||
p.pasterImage = s.image;
|
||||
CGRect frame = CGRectMake(s.frame.origin.x / previewSize.width,
|
||||
s.frame.origin.y / previewSize.height,
|
||||
s.frame.size.width / previewSize.width,
|
||||
s.frame.size.height / previewSize.height);
|
||||
p.frame = frame;
|
||||
return p;
|
||||
}];
|
||||
}
|
||||
|
||||
- (CGRect) adjustCropRectAccordingRotaionAngle:(CGRect)normalizedCropRect rotationAngle:(CGFloat)rotationAngle {
|
||||
normalizedCropRect = CGRectMake(MIN(1.0f, MAX(normalizedCropRect.origin.x, 0)),
|
||||
MIN(1.0f, MAX(normalizedCropRect.origin.y, 0)),
|
||||
MIN(1.0f, MAX(normalizedCropRect.size.width, 0)),
|
||||
MIN(1.0f, MAX(normalizedCropRect.size.height, 0)));
|
||||
|
||||
if (rotationAngle == 90) {
|
||||
normalizedCropRect = CGRectMake(normalizedCropRect.origin.y,
|
||||
1 - normalizedCropRect.size.width - normalizedCropRect.origin.x,
|
||||
normalizedCropRect.size.height,
|
||||
normalizedCropRect.size.width);
|
||||
} else if (rotationAngle == 180) {
|
||||
normalizedCropRect = CGRectMake(1 - normalizedCropRect.size.width - normalizedCropRect.origin.x,
|
||||
1 - normalizedCropRect.size.height - normalizedCropRect.origin.y,
|
||||
normalizedCropRect.size.width,
|
||||
normalizedCropRect.size.height);
|
||||
} else if (rotationAngle == 270) {
|
||||
normalizedCropRect = CGRectMake(1 - normalizedCropRect.origin.y - normalizedCropRect.size.height,
|
||||
normalizedCropRect.origin.x,
|
||||
normalizedCropRect.size.height,
|
||||
normalizedCropRect.size.width);
|
||||
}
|
||||
return normalizedCropRect;
|
||||
}
|
||||
|
||||
- (void) onProcessPictureForCrop:(UIImage *) processedPicture {
|
||||
_hasCrop = true;
|
||||
_srcPicture = processedPicture;
|
||||
_commonEditCtrlView.previewView.hidden = NO;
|
||||
_commonEditCtrlView.previewSize = _srcPicture.size;
|
||||
_imgView.image = _srcPicture;
|
||||
_commonEditCtrlView.mosaciOriginalImage = _srcPicture;
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewNeedAddPaster:(TUIMultimediaCommonEditorControlView *)view {
|
||||
_commonEditCtrlView.modifyButtonsHidden = YES;
|
||||
[self presentViewController:_pasterSelectController animated:NO completion:nil];
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewNeedModifySubtitle:(TUIMultimediaSubtitleInfo *)info callback:(void (^)(TUIMultimediaSubtitleInfo *info, BOOL isOk))callback {
|
||||
_subtitleEditController.subtitleInfo = info;
|
||||
_subtitleEditController.callback = ^(TUIMultimediaSubtitleEditController *c, BOOL isOk) {
|
||||
[c.presentingViewController dismissViewControllerAnimated:NO completion:nil];
|
||||
if (callback != nil) {
|
||||
callback(c.subtitleInfo, isOk);
|
||||
}
|
||||
};
|
||||
[self presentViewController:_subtitleEditController animated:NO completion:nil];
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewNeedEditMusic:(nonnull TUIMultimediaCommonEditorControlView *)view {
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaPasterSelectControllerDelegate protocol
|
||||
- (void)pasterSelectController:(TUIMultimediaPasterSelectController *)c onPasterSelected:(UIImage *)image {
|
||||
[_commonEditCtrlView addPaster:image];
|
||||
_commonEditCtrlView.modifyButtonsHidden = NO;
|
||||
[c.presentingViewController dismissViewControllerAnimated:NO completion:nil];
|
||||
}
|
||||
- (void)onPasterSelectControllerExit:(TUIMultimediaPasterSelectController *)c {
|
||||
_commonEditCtrlView.modifyButtonsHidden = NO;
|
||||
[c.presentingViewController dismissViewControllerAnimated:NO completion:nil];
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,27 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <AVKit/AVKit.h>
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPopupController.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaVideoBgmEditInfo.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaBGMEditControllerDelegate;
|
||||
|
||||
/**
|
||||
背景音乐选择界面Controller
|
||||
*/
|
||||
@interface TUIMultimediaBGMEditController : TUIMultimediaPopupController
|
||||
@property(nonatomic) float clipDuration;
|
||||
@property(readonly, nonatomic) TUIMultimediaVideoBgmEditInfo *bgmEditInfo;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaBGMEditControllerDelegate> delegate;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaBGMEditControllerDelegate <NSObject>
|
||||
- (void)onBGMEditController:(TUIMultimediaBGMEditController *)c bgmInfoChanged:(TUIMultimediaVideoBgmEditInfo *)bgmInfo;
|
||||
- (void)onBGMEditControllerExit:(TUIMultimediaBGMEditController *)c;
|
||||
@end
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,80 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaBGMEditController.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <ReactiveObjC/RACEXTScope.h>
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaBGMEditView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImagePicker.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterSelectView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPersistence.h"
|
||||
|
||||
@interface TUIMultimediaBGMEditController () <TUIMultimediaBGMEditViewDelegate> {
|
||||
TUIMultimediaBGMEditView *_editView;
|
||||
NSArray<TUIMultimediaBGMGroup *> *_bgmConfig;
|
||||
dispatch_block_t _blockBGMNotify;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaBGMEditController
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
_bgmConfig = [TUIMultimediaBGMGroup loadBGMConfigs];
|
||||
_bgmEditInfo = [[TUIMultimediaVideoBgmEditInfo alloc] init];
|
||||
_bgmEditInfo.originAudio = YES;
|
||||
_bgmEditInfo.bgm = nil;
|
||||
|
||||
return self;
|
||||
}
|
||||
- (void)viewDidLoad {
|
||||
[super viewDidLoad];
|
||||
_editView = [[TUIMultimediaBGMEditView alloc] init];
|
||||
_editView.bgmConfig = _bgmConfig;
|
||||
_editView.originAudioEnabled = YES;
|
||||
_editView.clipDuration = _clipDuration;
|
||||
_editView.delegate = self;
|
||||
self.mainView = _editView;
|
||||
}
|
||||
- (void)viewDidLayoutSubviews {
|
||||
[super viewDidLayoutSubviews];
|
||||
}
|
||||
- (void)viewWillAppear:(BOOL)animated {
|
||||
[super viewWillAppear:animated];
|
||||
}
|
||||
|
||||
- (void)popupControllerDidCanceled {
|
||||
[_delegate onBGMEditControllerExit:self];
|
||||
}
|
||||
|
||||
#pragma mark - Properties
|
||||
- (void)setClipDuration:(float)videoDuration {
|
||||
_clipDuration = videoDuration;
|
||||
_editView.clipDuration = videoDuration;
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaBGMEditViewDelegate protocol
|
||||
- (void)bgmEditViewValueChanged:(TUIMultimediaBGMEditView *)v {
|
||||
TUIMultimediaVideoBgmEditInfo *newBgmInfo = [[TUIMultimediaVideoBgmEditInfo alloc] init];
|
||||
if (v.bgmEnabled) {
|
||||
newBgmInfo.bgm = [v.selectedBgm copy];
|
||||
} else {
|
||||
newBgmInfo.bgm = nil;
|
||||
}
|
||||
newBgmInfo.originAudio = v.originAudioEnabled;
|
||||
|
||||
// 延迟防抖, 防止拖动时产生怪音
|
||||
if (_blockBGMNotify != nil) {
|
||||
dispatch_block_cancel(_blockBGMNotify);
|
||||
_blockBGMNotify = nil;
|
||||
}
|
||||
_blockBGMNotify = dispatch_block_create(DISPATCH_BLOCK_INHERIT_QOS_CLASS, ^{
|
||||
self->_bgmEditInfo = newBgmInfo;
|
||||
[self->_delegate onBGMEditController:self bgmInfoChanged:self->_bgmEditInfo];
|
||||
});
|
||||
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.050 * NSEC_PER_SEC)), dispatch_get_main_queue(), _blockBGMNotify);
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,22 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <AVFoundation/AVFoundation.h>
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaEncodeConfig.h"
|
||||
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
/**
|
||||
视频编辑界面Controller
|
||||
*/
|
||||
@interface TUIMultimediaVideoEditorController : UIViewController
|
||||
@property(nullable, nonatomic) NSString *sourceVideoPath;
|
||||
@property(nullable, nonatomic) NSString *resultVideoPath;
|
||||
@property(nonatomic)int sourceType;
|
||||
@property(nonatomic, nullable) void (^completeCallback)(NSString *resultVideoPath, int resultCode);
|
||||
- (instancetype)init;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,299 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaVideoEditorController.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <ReactiveObjC/RACEXTScope.h>
|
||||
#import <TUICore/TUIDefine.h>
|
||||
#import <TXLiteAVSDK_Professional/TXVideoEditer.h>
|
||||
#import <TXLiteAVSDK_Professional/TXVideoEditerTypeDef.h>
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaBGMEditController.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommonEditorControlView.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaImagePicker.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPasterSelectController.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaPersistence.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaSubtitleEditController.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaConstant.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaAuthorizationPrompter.h"
|
||||
|
||||
@interface TUIMultimediaVideoEditorController () <TXVideoPreviewListener,
|
||||
TXVideoGenerateListener,
|
||||
TUIMultimediaPasterSelectControllerDelegate,
|
||||
TUIMultimediaCommonEditorControlViewDelegate,
|
||||
TUIMultimediaBGMEditControllerDelegate> {
|
||||
TUIMultimediaPasterSelectController *_pasterSelectController;
|
||||
TUIMultimediaCommonEditorControlView *_commonEditCtrlView;
|
||||
TXVideoEditer *_editor;
|
||||
NSString *_sourceVideoPath;
|
||||
TXVideoInfo *_videoInfo;
|
||||
TUIMultimediaSubtitleEditController *_subtitleEditController;
|
||||
TUIMultimediaBGMEditController *_musicController;
|
||||
TUIMultimediaEncodeConfig *_encodeConfig;
|
||||
BOOL _originNavgationBarHidden;
|
||||
float _lastGenerateProgress;
|
||||
BOOL _hasAudioEdited;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaVideoEditorController
|
||||
|
||||
@dynamic sourceVideoPath;
|
||||
|
||||
- (instancetype)init {
|
||||
self = [super init];
|
||||
_encodeConfig = [[TUIMultimediaEncodeConfig alloc] initWithVideoQuality:[[TUIMultimediaConfig sharedInstance] getVideoQuality]];
|
||||
_lastGenerateProgress = 0;
|
||||
_sourceType = SOURCE_TYPE_RECORD;
|
||||
_hasAudioEdited = NO;
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)viewDidLoad {
|
||||
[self initUI];
|
||||
_pasterSelectController = [[TUIMultimediaPasterSelectController alloc] init];
|
||||
_pasterSelectController.delegate = self;
|
||||
_pasterSelectController.modalPresentationStyle = UIModalPresentationOverCurrentContext;
|
||||
|
||||
_subtitleEditController = [[TUIMultimediaSubtitleEditController alloc] init];
|
||||
_subtitleEditController.modalPresentationStyle = UIModalPresentationOverCurrentContext;
|
||||
|
||||
_musicController = [[TUIMultimediaBGMEditController alloc] init];
|
||||
_musicController.modalPresentationStyle = UIModalPresentationOverCurrentContext;
|
||||
_musicController.delegate = self;
|
||||
_musicController.clipDuration = _videoInfo.duration;
|
||||
}
|
||||
|
||||
- (void)viewWillAppear:(BOOL)animated {
|
||||
if (self.navigationController != nil) {
|
||||
_originNavgationBarHidden = self.navigationController.navigationBarHidden;
|
||||
self.navigationController.navigationBarHidden = YES;
|
||||
}
|
||||
}
|
||||
|
||||
- (void)viewWillDisappear:(BOOL)animated {
|
||||
if (self.navigationController != nil) {
|
||||
self.navigationController.navigationBarHidden = _originNavgationBarHidden;
|
||||
}
|
||||
}
|
||||
|
||||
- (NSString *)sourceVideoPath {
|
||||
return _sourceVideoPath;
|
||||
}
|
||||
|
||||
- (void)setSourceVideoPath:(NSString *)sourceVideoPath {
|
||||
_sourceVideoPath = sourceVideoPath;
|
||||
NSString *currentSourceVideoPath = sourceVideoPath;
|
||||
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
|
||||
if (self->_sourceVideoPath != currentSourceVideoPath) return;
|
||||
self->_videoInfo = [TXVideoInfoReader getVideoInfo:self->_sourceVideoPath];
|
||||
NSLog(@"videoInfo angle = %d, width = %d, height = %d, duration = %f",
|
||||
self->_videoInfo.angle, self->_videoInfo.width, self->_videoInfo.height, self->_videoInfo.duration);
|
||||
if (self->_videoInfo.angle == 90 || self->_videoInfo.angle == 270) {
|
||||
int temp = self->_videoInfo.width;
|
||||
self->_videoInfo.width = self->_videoInfo.height;
|
||||
self->_videoInfo.height = temp;
|
||||
}
|
||||
dispatch_async(dispatch_get_main_queue(), ^{
|
||||
[self tryReloadVideoAsset];
|
||||
self->_musicController.clipDuration = self->_videoInfo.duration;
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
- (void)setSourceType:(int)sourceType {
|
||||
NSLog(@"setSourceType sourceType = %d",sourceType);
|
||||
_sourceType = sourceType;
|
||||
if (_commonEditCtrlView != nil) {
|
||||
_commonEditCtrlView.sourceType = sourceType;
|
||||
}
|
||||
}
|
||||
|
||||
- (void)tryReloadVideoAsset {
|
||||
if (_editor == nil || _sourceVideoPath == nil) return;
|
||||
int code = [_editor setVideoPath:_sourceVideoPath];
|
||||
[_editor startPlayFromTime:0 toTime:_videoInfo.duration];
|
||||
if (code != 0) {
|
||||
NSString *title = [TUIMultimediaCommon localizedStringForKey:@"modify_load_assert_failed"];
|
||||
[self showAlertWithTitle:title message:@"" action:@"OK"];
|
||||
}
|
||||
_commonEditCtrlView.previewSize = CGSizeMake(_videoInfo.width, _videoInfo.height);
|
||||
}
|
||||
|
||||
#pragma mark - UI Init
|
||||
|
||||
- (void)initUI {
|
||||
_commonEditCtrlView = [[TUIMultimediaCommonEditorControlView alloc] initWithConfig:TUIMultimediaCommonEditorConfig.configForVideoEditor];
|
||||
[self.view addSubview:_commonEditCtrlView];
|
||||
_commonEditCtrlView.backgroundColor = UIColor.blackColor;
|
||||
_commonEditCtrlView.previewSize = CGSizeMake(_videoInfo.width, _videoInfo.height);
|
||||
_commonEditCtrlView.sourceType = _sourceType;
|
||||
_commonEditCtrlView.delegate = self;
|
||||
|
||||
TXPreviewParam *param = [[TXPreviewParam alloc] init];
|
||||
param.videoView = _commonEditCtrlView.previewView;
|
||||
param.renderMode = PREVIEW_RENDER_MODE_FILL_EDGE;
|
||||
_editor = [[TXVideoEditer alloc] initWithPreview:param];
|
||||
[self tryReloadVideoAsset];
|
||||
_editor.previewDelegate = self;
|
||||
_editor.generateDelegate = self;
|
||||
|
||||
[_commonEditCtrlView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.edges.equalTo(self.view);
|
||||
}];
|
||||
}
|
||||
- (void)showAlertWithTitle:(NSString *)title message:(NSString *)message action:(NSString *)action {
|
||||
UIAlertController *alert = [UIAlertController alertControllerWithTitle:title message:message preferredStyle:UIAlertControllerStyleAlert];
|
||||
UIAlertAction *defaultAction = [UIAlertAction actionWithTitle:action style:UIAlertActionStyleDefault handler:nil];
|
||||
[alert addAction:defaultAction];
|
||||
[self presentViewController:alert animated:YES completion:nil];
|
||||
}
|
||||
|
||||
#pragma mark - TXVideoPreviewListener protocol
|
||||
- (void)onPreviewProgress:(CGFloat)time {
|
||||
}
|
||||
- (void)onPreviewFinished {
|
||||
[_editor startPlayFromTime:0 toTime:_videoInfo.duration];
|
||||
}
|
||||
|
||||
#pragma mark - TXVideoGenerateListener protocol
|
||||
- (void)onGenerateProgress:(float)progress {
|
||||
NSLog(@"TUIMultimediaVideoEditorController onGenerateProgress progress = %f",progress);
|
||||
if (progress - _lastGenerateProgress > 0.01f || progress == 1.0f) {
|
||||
_commonEditCtrlView.progressBarProgress = progress;
|
||||
_lastGenerateProgress = progress;
|
||||
}
|
||||
}
|
||||
|
||||
- (void)onGenerateComplete:(TXGenerateResult *)result {
|
||||
NSLog(@"TUIMultimediaVideoEditorController onGenerateComplete retCode = %ld",(long)result.retCode);
|
||||
int resultCode = (result.retCode != 0) ? VIDEO_EDIT_RESULT_CODE_GENERATE_FAIL : VIDEO_EDIT_RESULT_CODE_GENERATE_SUCCESS;
|
||||
_completeCallback(_resultVideoPath, resultCode);
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaCommonEditorControlViewDelegate protocol
|
||||
- (void)onCommonEditorControlViewComplete:(TUIMultimediaCommonEditorControlView *)view stickers:(NSArray<TUIMultimediaSticker *> *)stickers {
|
||||
if (![TUIMultimediaAuthorizationPrompter verifyPermissionGranted:self]) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (stickers.count == 0 && !_hasAudioEdited && _sourceType == SOURCE_TYPE_ALBUM) {
|
||||
NSLog(@"Return directly without encoding the video");
|
||||
_resultVideoPath = _sourceVideoPath;
|
||||
_completeCallback(_resultVideoPath, VIDEO_EDIT_RESULT_CODE_NO_EDIT);
|
||||
return;
|
||||
}
|
||||
|
||||
NSArray<TXPaster *> *pasterList = [stickers tui_multimedia_map:^TXPaster *(TUIMultimediaSticker *s) {
|
||||
TXPaster *p = [[TXPaster alloc] init];
|
||||
p.pasterImage = s.image;
|
||||
p.frame = s.frame;
|
||||
p.startTime = 0.0;
|
||||
p.endTime = self->_videoInfo.duration;
|
||||
return p;
|
||||
}];
|
||||
[_editor setPasterList:pasterList];
|
||||
|
||||
_commonEditCtrlView.isGenerating = YES;
|
||||
[self.view bringSubviewToFront:_commonEditCtrlView];
|
||||
if (_resultVideoPath == nil || _resultVideoPath.length == 0) {
|
||||
_resultVideoPath = [self getOutFilePath];
|
||||
}
|
||||
[_editor setVideoBitrate:_encodeConfig.bitrate];
|
||||
[_editor generateVideo:[_encodeConfig getVideoEditCompressed] videoOutputPath:_resultVideoPath];
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewCancel:(TUIMultimediaCommonEditorControlView *)view {
|
||||
[_editor stopPlay];
|
||||
_completeCallback(nil, VIDEO_EDIT_RESULT_CODE_CANCEL);
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewNeedAddPaster:(TUIMultimediaCommonEditorControlView *)view {
|
||||
NSLog(@"onCommonEditorControlViewNeedAddPaster");
|
||||
_commonEditCtrlView.modifyButtonsHidden = YES;
|
||||
[self presentViewController:_pasterSelectController animated:NO completion:nil];
|
||||
}
|
||||
|
||||
- (void)onCommonEditorControlViewNeedModifySubtitle:(TUIMultimediaSubtitleInfo *)info callback:(void (^)(TUIMultimediaSubtitleInfo *info, BOOL isOk))callback {
|
||||
_subtitleEditController.subtitleInfo = info;
|
||||
_subtitleEditController.callback = ^(TUIMultimediaSubtitleEditController *c, BOOL isOk) {
|
||||
[c.presentingViewController dismissViewControllerAnimated:NO completion:nil];
|
||||
if (callback != nil) {
|
||||
callback(c.subtitleInfo, isOk);
|
||||
}
|
||||
};
|
||||
[self presentViewController:_subtitleEditController animated:NO completion:nil];
|
||||
}
|
||||
- (void)onCommonEditorControlViewNeedEditMusic:(TUIMultimediaCommonEditorControlView *)view {
|
||||
_commonEditCtrlView.modifyButtonsHidden = YES;
|
||||
[self presentViewController:_musicController animated:NO completion:nil];
|
||||
}
|
||||
- (void)onCommonEditorControlViewCancelGenerate:(TUIMultimediaCommonEditorControlView *)view {
|
||||
[_editor cancelGenerate];
|
||||
TXPreviewParam *param = [[TXPreviewParam alloc] init];
|
||||
param.videoView = _commonEditCtrlView.previewView;
|
||||
param.renderMode = PREVIEW_RENDER_MODE_FILL_EDGE;
|
||||
_editor = [[TXVideoEditer alloc] initWithPreview:param];
|
||||
[self tryReloadVideoAsset];
|
||||
_editor.previewDelegate = self;
|
||||
_editor.generateDelegate = self;
|
||||
}
|
||||
|
||||
-(NSString*) getOutFilePath{
|
||||
NSDate* currentDate = [NSDate date];
|
||||
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
|
||||
[dateFormatter setDateFormat:@"yyyy-MM-dd HH:mm:ss.SSS"];
|
||||
NSString* currentDateString = [dateFormatter stringFromDate:currentDate];
|
||||
NSString* outFileName = [NSString stringWithFormat:@"%@-%u-temp.mp4", currentDateString, arc4random()];
|
||||
return [NSTemporaryDirectory() stringByAppendingPathComponent:outFileName];
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaPasterSelectControllerDelegate protocol
|
||||
- (void)pasterSelectController:(TUIMultimediaPasterSelectController *)c onPasterSelected:(UIImage *)image {
|
||||
[_commonEditCtrlView addPaster:image];
|
||||
_commonEditCtrlView.modifyButtonsHidden = NO;
|
||||
[c.presentingViewController dismissViewControllerAnimated:NO completion:nil];
|
||||
}
|
||||
- (void)onPasterSelectControllerExit:(TUIMultimediaPasterSelectController *)c {
|
||||
_commonEditCtrlView.modifyButtonsHidden = NO;
|
||||
[c.presentingViewController dismissViewControllerAnimated:NO completion:nil];
|
||||
}
|
||||
|
||||
#pragma mark - TUIMultimediaMusicControllerDelegate protocol
|
||||
- (void)onBGMEditController:(TUIMultimediaBGMEditController *)bgmController bgmInfoChanged:(TUIMultimediaVideoBgmEditInfo *)bgmInfo {
|
||||
if (![TUIMultimediaAuthorizationPrompter verifyPermissionGranted:bgmController]) {
|
||||
return;
|
||||
}
|
||||
|
||||
[_editor setBGMLoop:YES];
|
||||
@weakify(self);
|
||||
[_editor setBGMAsset:bgmInfo.bgm.asset
|
||||
result:^(int result) {
|
||||
@strongify(self);
|
||||
if (result != 0) {
|
||||
NSString *title = [TUIMultimediaCommon localizedStringForKey:@"modify_load_assert_failed"];
|
||||
[self showAlertWithTitle:title message:@"" action:@"OK"];
|
||||
}
|
||||
}];
|
||||
[_editor setBGMAtVideoTime:0];
|
||||
[_editor setBGMStartTime:bgmInfo.bgm.startTime endTime:bgmInfo.bgm.endTime];
|
||||
[_editor setBGMVolume:1];
|
||||
if (bgmInfo.originAudio) {
|
||||
[_editor setVideoVolume:1];
|
||||
} else {
|
||||
[_editor setVideoVolume:0];
|
||||
}
|
||||
_commonEditCtrlView.musicEdited = bgmInfo.bgm != nil;
|
||||
|
||||
_hasAudioEdited = (!bgmInfo.originAudio) || (bgmInfo.bgm != nil);
|
||||
}
|
||||
- (void)onBGMEditControllerExit:(nonnull TUIMultimediaBGMEditController *)c {
|
||||
_commonEditCtrlView.modifyButtonsHidden = NO;
|
||||
[c.presentingViewController dismissViewControllerAnimated:NO completion:nil];
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,31 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaBGM.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
typedef NS_ENUM(NSInteger, TUIMultimediaMusicCellState) {
|
||||
TUIMultimediaMusicCellStateNormal,
|
||||
TUIMultimediaMusicCellStateSelected,
|
||||
TUIMultimediaMusicCellStateEnabled,
|
||||
};
|
||||
|
||||
@protocol TUIMultimediaMusicCellDelegate;
|
||||
|
||||
@interface TUIMultimediaMusicCell : UITableViewCell
|
||||
@property(nonatomic) TUIMultimediaMusicCellState state;
|
||||
@property(nullable, nonatomic) TUIMultimediaBGM *music;
|
||||
@property(nonatomic) float selectDuration;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaMusicCellDelegate> delegate;
|
||||
|
||||
+ (NSString *)reuseIdentifier;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaMusicCellDelegate <NSObject>
|
||||
- (void)musicCell:(TUIMultimediaMusicCell *)cell onEditStateChanged:(BOOL)editState;
|
||||
- (void)musicCell:(TUIMultimediaMusicCell *)cell onBGMRangeChanged:(TUIMultimediaBGM *)bgm;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,128 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaMusicCell.h"
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "Masonry/Masonry.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaAutoScrollLabel.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaConfig.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaFakeAudioWaveView.h"
|
||||
|
||||
static const CGFloat TitleFontSize = 14;
|
||||
static const CGFloat SubtitleFontSize = 12;
|
||||
|
||||
@interface TUIMultimediaMusicCell () {
|
||||
TUIMultimediaAutoScrollLabel *_lbTitle;
|
||||
UILabel *_lbSubTitle;
|
||||
UILabel *_lbDuration;
|
||||
TUIMultimediaFakeAudioWaveView *_waveView;
|
||||
}
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaMusicCell
|
||||
|
||||
+ (NSString *)reuseIdentifier {
|
||||
return NSStringFromClass([self class]);
|
||||
}
|
||||
- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier {
|
||||
self = [super initWithStyle:style reuseIdentifier:reuseIdentifier];
|
||||
if (self != nil) {
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
self.contentView.backgroundColor = UIColor.clearColor;
|
||||
self.backgroundColor = UIColor.clearColor;
|
||||
self.selectionStyle = UITableViewCellSelectionStyleNone;
|
||||
|
||||
_lbTitle = [[TUIMultimediaAutoScrollLabel alloc] init];
|
||||
[self.contentView addSubview:_lbTitle];
|
||||
|
||||
_lbSubTitle = [[UILabel alloc] init];
|
||||
[self.contentView addSubview:_lbSubTitle];
|
||||
_lbSubTitle.font = [UIFont systemFontOfSize:SubtitleFontSize];
|
||||
_lbSubTitle.textColor = TUIMultimediaPluginDynamicColor(@"editor_bgm_text_color", @"#FFFFFF99");
|
||||
|
||||
_lbDuration = [[UILabel alloc] init];
|
||||
[self.contentView addSubview:_lbDuration];
|
||||
_lbDuration.font = [UIFont monospacedSystemFontOfSize:SubtitleFontSize weight:UIFontWeightMedium];
|
||||
_lbDuration.textColor = TUIMultimediaPluginDynamicColor(@"editor_bgm_text_color", @"#FFFFFF99");
|
||||
|
||||
_waveView = [[TUIMultimediaFakeAudioWaveView alloc] init];
|
||||
[self addSubview:_waveView];
|
||||
_waveView.hidden = YES;
|
||||
|
||||
[_lbTitle mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.left.top.equalTo(self.contentView).inset(10);
|
||||
make.width.mas_equalTo(120);
|
||||
}];
|
||||
[_lbSubTitle mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.top.equalTo(_lbTitle.mas_bottom).inset(5);
|
||||
make.left.equalTo(self.contentView).inset(14);
|
||||
make.bottom.equalTo(self.contentView).inset(10);
|
||||
}];
|
||||
[_lbDuration mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.centerY.equalTo(self.contentView);
|
||||
make.right.equalTo(self.contentView).inset(10);
|
||||
}];
|
||||
[_waveView mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.right.equalTo(_lbDuration.mas_left).inset(50);
|
||||
make.centerY.equalTo(self);
|
||||
make.height.mas_equalTo(12);
|
||||
make.width.mas_equalTo(48);
|
||||
}];
|
||||
}
|
||||
- (void)updateTitle {
|
||||
UIColor *color = UIColor.whiteColor;
|
||||
BOOL active = NO;
|
||||
if (_state == TUIMultimediaMusicCellStateEnabled) {
|
||||
color = [[TUIMultimediaConfig sharedInstance] getThemeColor];
|
||||
if (_music.lyric != nil && _music.lyric.length > 0) {
|
||||
active = YES;
|
||||
}
|
||||
}
|
||||
NSString *title = _music.lyric != nil && _music.lyric.length > 0 ? _music.lyric : _music.name;
|
||||
_lbTitle.text = [[NSAttributedString alloc] initWithString:title
|
||||
attributes:@{
|
||||
NSFontAttributeName : [UIFont systemFontOfSize:TitleFontSize],
|
||||
NSForegroundColorAttributeName : color,
|
||||
}];
|
||||
_lbTitle.active = active;
|
||||
}
|
||||
- (void)updateWave {
|
||||
switch (_state) {
|
||||
case TUIMultimediaMusicCellStateNormal:
|
||||
_waveView.hidden = YES;
|
||||
_waveView.enabled = NO;
|
||||
break;
|
||||
case TUIMultimediaMusicCellStateSelected:
|
||||
_waveView.color = TUIMultimediaPluginDynamicColor(@"editor_bgm_text_color", @"#FFFFFF99");
|
||||
_waveView.hidden = NO;
|
||||
_waveView.enabled = NO;
|
||||
break;
|
||||
case TUIMultimediaMusicCellStateEnabled:
|
||||
_waveView.color = [[TUIMultimediaConfig sharedInstance] getThemeColor];
|
||||
_waveView.hidden = NO;
|
||||
_waveView.enabled = YES;
|
||||
break;
|
||||
}
|
||||
}
|
||||
#pragma mark - Properties
|
||||
- (void)setState:(TUIMultimediaMusicCellState)state {
|
||||
_state = state;
|
||||
[self updateTitle];
|
||||
[self updateWave];
|
||||
}
|
||||
- (void)setMusic:(TUIMultimediaBGM *)music {
|
||||
_music = music;
|
||||
[self updateTitle];
|
||||
_lbSubTitle.text = music.source;
|
||||
float duration = music.asset == nil ? 0 : music.asset.duration.value / music.asset.duration.timescale;
|
||||
int min = (int)(duration / 60);
|
||||
_lbDuration.text = [NSString stringWithFormat:@"%02d:%02d", min, (int)(duration - min * 60)];
|
||||
}
|
||||
|
||||
@end
|
||||
@@ -0,0 +1,24 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import <UIKit/UIKit.h>
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaBGM.h"
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@protocol TUIMultimediaBGMEditViewDelegate;
|
||||
|
||||
@interface TUIMultimediaBGMEditView : UIView
|
||||
@property(nonatomic) NSArray<TUIMultimediaBGMGroup *> *bgmConfig;
|
||||
@property(nonatomic) float clipDuration;
|
||||
@property(nonatomic) TUIMultimediaBGM *selectedBgm;
|
||||
@property(nonatomic) BOOL originAudioEnabled;
|
||||
@property(nonatomic) BOOL bgmEnabled;
|
||||
@property(weak, nullable, nonatomic) id<TUIMultimediaBGMEditViewDelegate> delegate;
|
||||
@end
|
||||
|
||||
@protocol TUIMultimediaBGMEditViewDelegate <NSObject>
|
||||
- (void)bgmEditViewValueChanged:(TUIMultimediaBGMEditView *)v;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
||||
@@ -0,0 +1,164 @@
|
||||
// Copyright (c) 2024 Tencent. All rights reserved.
|
||||
// Author: eddardliu
|
||||
|
||||
#import "TUIMultimediaBGMEditView.h"
|
||||
#import <Masonry/Masonry.h>
|
||||
#import <TUICore/TUIThemeManager.h>
|
||||
#import "TUIMultimediaPlugin/NSArray+Functional.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCheckBox.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaCommon.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaMusicCell.h"
|
||||
#import "TUIMultimediaPlugin/TUIMultimediaTabPanel.h"
|
||||
|
||||
static const CGFloat ItemInset = 10;
|
||||
static const CGFloat ItemWidthFactor = 0.7;
|
||||
|
||||
static const CGFloat CollectionViewHeight = 100;
|
||||
|
||||
@interface TUIMultimediaBGMEditView () <UITableViewDataSource, UITableViewDelegate> {
|
||||
NSArray<UITableView *> *_tableViews;
|
||||
TUIMultimediaTabPanel *_tabPanel;
|
||||
NSInteger _selectedIndex;
|
||||
TUIMultimediaCheckBox *_switchOriginAudio;
|
||||
TUIMultimediaCheckBox *_switchBgm;
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
@implementation TUIMultimediaBGMEditView
|
||||
- (id)initWithFrame:(CGRect)frame {
|
||||
self = [super initWithFrame:frame];
|
||||
if (self != nil) {
|
||||
_bgmConfig = @[];
|
||||
_selectedIndex = -1;
|
||||
[self initUI];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)initUI {
|
||||
self.backgroundColor = TUIMultimediaPluginDynamicColor(@"editor_popup_view_bg_color", @"#000000BF");
|
||||
|
||||
_tabPanel = [[TUIMultimediaTabPanel alloc] initWithFrame:self.bounds];
|
||||
[self addSubview:_tabPanel];
|
||||
|
||||
_switchOriginAudio = [[TUIMultimediaCheckBox alloc] init];
|
||||
[self addSubview:_switchOriginAudio];
|
||||
_switchOriginAudio.text = [TUIMultimediaCommon localizedStringForKey:@"editor_origin_audio"];
|
||||
[_switchOriginAudio addTarget:self action:@selector(onSwitchOriginAudioChanged) forControlEvents:UIControlEventValueChanged];
|
||||
|
||||
_switchBgm = [[TUIMultimediaCheckBox alloc] init];
|
||||
[self addSubview:_switchBgm];
|
||||
_switchBgm.text = [TUIMultimediaCommon localizedStringForKey:@"editor_bgm"];
|
||||
[_switchBgm addTarget:self action:@selector(onSwitchBGMChanged) forControlEvents:UIControlEventValueChanged];
|
||||
|
||||
[_tabPanel mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.left.right.top.equalTo(self);
|
||||
make.bottom.equalTo(_switchBgm.mas_top).inset(10);
|
||||
make.height.mas_equalTo(400);
|
||||
}];
|
||||
[_switchBgm mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.bottom.equalTo(self.mas_safeAreaLayoutGuideBottom);
|
||||
make.height.mas_equalTo(24);
|
||||
make.left.equalTo(self).inset(20);
|
||||
}];
|
||||
[_switchOriginAudio mas_makeConstraints:^(MASConstraintMaker *make) {
|
||||
make.height.mas_equalTo(24);
|
||||
make.bottom.equalTo(self.mas_safeAreaLayoutGuideBottom);
|
||||
make.right.equalTo(self).inset(20);
|
||||
}];
|
||||
|
||||
[self reloadConfig];
|
||||
}
|
||||
|
||||
- (void)reloadConfig {
|
||||
_tableViews = [_bgmConfig tui_multimedia_mapWithIndex:^UITableView *(TUIMultimediaBGMGroup *group, NSUInteger idx) {
|
||||
UITableView *v = [[UITableView alloc] init];
|
||||
v.tag = idx;
|
||||
v.backgroundColor = UIColor.clearColor;
|
||||
v.dataSource = self;
|
||||
v.delegate = self;
|
||||
[v registerClass:TUIMultimediaMusicCell.class forCellReuseIdentifier:TUIMultimediaMusicCell.reuseIdentifier];
|
||||
return v;
|
||||
}];
|
||||
_tabPanel.tabs = [_tableViews tui_multimedia_map:^TUIMultimediaTabPanelTab *(UITableView *v) {
|
||||
return [[TUIMultimediaTabPanelTab alloc] initWithName:[TUIMultimediaCommon localizedStringForKey:self->_bgmConfig[v.tag].name] icon:nil view:v];
|
||||
}];
|
||||
}
|
||||
|
||||
- (void)layoutSubviews {
|
||||
[super layoutSubviews];
|
||||
}
|
||||
|
||||
- (NSArray<TUIMultimediaBGM *> *)getBgmListByTableView:(UITableView *)v {
|
||||
return _bgmConfig[v.tag].bgmList;
|
||||
}
|
||||
|
||||
#pragma mark - UITableViewDataSource
|
||||
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
|
||||
return [self getBgmListByTableView:tableView].count;
|
||||
}
|
||||
|
||||
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
|
||||
TUIMultimediaMusicCell *cell = [tableView dequeueReusableCellWithIdentifier:TUIMultimediaMusicCell.reuseIdentifier forIndexPath:indexPath];
|
||||
cell.music = [self getBgmListByTableView:tableView][indexPath.item];
|
||||
if (cell.music == _selectedBgm) {
|
||||
if (_switchBgm.on) {
|
||||
cell.state = TUIMultimediaMusicCellStateEnabled;
|
||||
} else {
|
||||
cell.state = TUIMultimediaMusicCellStateSelected;
|
||||
}
|
||||
} else {
|
||||
cell.state = TUIMultimediaMusicCellStateNormal;
|
||||
}
|
||||
return cell;
|
||||
}
|
||||
#pragma mark - UITableViewDelegate
|
||||
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
|
||||
[tableView deselectRowAtIndexPath:indexPath animated:NO];
|
||||
_selectedBgm = [self getBgmListByTableView:tableView][indexPath.item];
|
||||
if (!_switchBgm.on) {
|
||||
_switchBgm.on = YES;
|
||||
}
|
||||
[_delegate bgmEditViewValueChanged:self];
|
||||
for (UITableView *v in _tableViews) {
|
||||
[v reloadData];
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - Actions
|
||||
- (void)onSwitchOriginAudioChanged {
|
||||
[_delegate bgmEditViewValueChanged:self];
|
||||
}
|
||||
|
||||
- (void)onSwitchBGMChanged {
|
||||
[_delegate bgmEditViewValueChanged:self];
|
||||
for (UITableView *v in _tableViews) {
|
||||
[v reloadData];
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - Properties
|
||||
- (void)setBgmConfig:(NSArray<TUIMultimediaBGMGroup *> *)bgmConfig {
|
||||
_bgmConfig = bgmConfig;
|
||||
[self reloadConfig];
|
||||
}
|
||||
- (BOOL)originAudioEnabled {
|
||||
return _switchOriginAudio.on;
|
||||
}
|
||||
- (void)setOriginAudioEnabled:(BOOL)originAudioEnabled {
|
||||
_switchOriginAudio.on = originAudioEnabled;
|
||||
}
|
||||
|
||||
- (BOOL)bgmEnabled {
|
||||
return _switchBgm.on;
|
||||
}
|
||||
- (void)setBgmEnabled:(BOOL)bgmEnabled {
|
||||
_switchBgm.on = bgmEnabled;
|
||||
}
|
||||
|
||||
- (void)setClipDuration:(float)videoDuration {
|
||||
_clipDuration = videoDuration;
|
||||
}
|
||||
|
||||
@end
|
||||
Reference in New Issue
Block a user