从0到1实现小说阅读器(四、笔记功能)

上篇我们实现了单页文本排版和多页排版逻辑,但这仅仅是实现了基础的翻阅功能。除此之外,小说阅读器还一些设置功能,比如:切换背景、改变字体大小、切换章节、目录、笔记等等。这篇我们主要讲解笔记功能的实现,其他功能在我看来还是比较简单,直接看源码就好了。

首先让产品经理来描述一下笔记的过程:

小明正在看小说,对其中某句话感触颇深,于是长按该句话所在区域,这时小明长按的位置高亮显示,并标记默认选中两个字,所以小明拖动前后大头针刚好使得这句话高亮显示,然后点击笔记按钮,弹出输入框,小明输入笔记确定后,这句话显示一条下划线。再次点击这句话可以删除或修改笔记。

接着iOS开发来分析一下笔记的过程:

1.添加长按手势;
2.长按文本高亮显示、默认选中两个字;
3.拖动大头针,可以改变文本选中长度;
4.标记选中文本(添加下划线);
5.为标记的文本添加点击事件。

这里呢,先了解一下CTFrame,这是定位文本的基础逻辑。在CTFrame内部是由多个CTLine组成,每行CTLine又是由多个CTRun组成。每个CTRun代表一组风格一致的文本(CTlineCTRun的创建不需要我们管理),如图所示:

长按文本高亮显示、默认选中两个字

长按事件:

- (void)longPressAction:(UILongPressGestureRecognizer *)recognizer {
    // 获取长按手势所在视图的位置
    CGPoint point = [recognizer locationInView:self];
    NSUInteger state = recognizer.state;
    if (state == UIGestureRecognizerStateBegan || state == UIGestureRecognizerStateChanged) {
        // 传入point和frameRef,返回rect和selectRange
        CGRect rect = [HYReadParser parserRectWithPoint:point frameRef:_frameRef selectRange:&_selectRange];
        // 显示放大镜
        [self.magnifier show:point];
        // setNeedsDisplay异步执行的,它会自动调用drawRect方法
        if (!CGRectEqualToRect(rect, CGRectZero)) {
            _pathArray = @[NSStringFromCGRect(rect)];
            [self setNeedsDisplay];
        }
    }
    
    if (recognizer.state == UIGestureRecognizerStateEnded) {
        // 长按结束隐藏放大镜,显示菜单
        [self hideMagnifier];
        if (!CGRectEqualToRect(_menuRect, CGRectZero)) {
            [self showMenu];
        }
    }
}

传入pointframeRef,返回rectselectRange

+ (CGRect)parserRectWithPoint:(CGPoint)point frameRef:(CTFrameRef)frameRef selectRange:(NSRange *)selectRange {
    CFIndex index = -1;
    
    CGPathRef pathRef = CTFrameGetPath(frameRef);
    CGRect bounds = CGPathGetBoundingBox(pathRef);
    
    CGRect rect = CGRectZero;
    // 获得CTLineRef数组
    NSArray *lines = (NSArray *)CTFrameGetLines(frameRef);
    
    if (!lines) {
        return rect;
    }
    
    NSUInteger linesCount = lines.count;
    // 为每一行起点开辟内存空间,可以理解成声明一个存放每行起点的数组
    CGPoint *origins = malloc(linesCount *sizeof(CGPoint));
    if (linesCount) {
        // 获得每行起点的数据
        CTFrameGetLineOrigins(frameRef, CFRangeMake(0, 0), origins);
        // 遍历起点数组找到point所在行
        for (NSInteger i = 0; i < linesCount; i++) {
            // 获得第i行起点坐标
            CGPoint baselineOrigin = origins[i];
            // 获得第i行CTLineRef对象
            CTLineRef lineRef = (__bridge CTLineRef)[lines objectAtIndex:i];
            // 声明字体上行高度和下行高度
            CGFloat ascent, descent;
            // 传入行对象,拿到行宽、上行高度、下行高度
            CGFloat lineWidth = CTLineGetTypographicBounds(lineRef, &ascent, &descent, NULL);
            // 获取行对象frame(注意:因为CoreText绘制的坐标系和布局坐标系的起点方向分别是左下角和左上角,所以lineFrame的y值不是baselineOrigin.y,而是CGRectGetHeight(bounds) - baselineOrigin.y - ascent)
            CGRect lineFrame = CGRectMake(baselineOrigin.x,
                                          CGRectGetHeight(bounds) - baselineOrigin.y - ascent,
                                          lineWidth,
                                          ascent + descent);
            // 通过判断point是否在lineFrame中来确定所在行
            if (CGRectContainsPoint(lineFrame, point)) {
                // 获得所在行的range
                CFRange stringRange = CTLineGetStringRange(lineRef);
                // 获得所在行的index
                index = CTLineGetStringIndexForPosition(lineRef, point);
                
                // 获取选中文本的rect
                CGFloat xStart = CTLineGetOffsetForStringIndex(lineRef, index, NULL);
                CGFloat xEnd;
                if (index > stringRange.location + stringRange.length - 2) {
                    // 超出右边界处理
                    xEnd = xStart;
                    xStart = CTLineGetOffsetForStringIndex(lineRef, index - 2, NULL);
                    (*selectRange).location = index - 2;
                } else {
                    xEnd = CTLineGetOffsetForStringIndex(lineRef, index + 2, NULL);
                    (*selectRange).location = index - 1;
                }
                // 默认选中两个单位
                (*selectRange).length = 2;
                
                rect = CGRectMake(origins[i].x + xStart,
                                  baselineOrigin.y - descent,
                                  fabs(xStart - xEnd),
                                  ascent + descent);
                break;
            }
        }
    }
    free(origins);
    return rect;
}

异步绘制:

- (void)drawRect:(CGRect)rect {
    [super drawRect:rect];
    if (!_frameRef) {
        return;
    }
    CGContextRef context = UIGraphicsGetCurrentContext();
    // 翻转坐标系
    CGContextSetTextMatrix(context, CGAffineTransformIdentity);
    CGContextTranslateCTM(context, 0, self.bounds.size.height);
    CGContextScaleCTM(context, 1.0, -1.0);
    
    CGRect leftDot, rightDot = CGRectZero;
    _menuRect = CGRectZero;
    // 绘制高亮区域
    [self drawSelectedPath:_pathArray leftDot:&leftDot rightDot:&rightDot];
    // 绘制frameRef
    CTFrameDraw(_frameRef, context);
    [self drawDotWithLeft:leftDot right:rightDot];
}

文本高亮显示:

- (void)drawSelectedPath:(NSArray *)array leftDot:(CGRect *)leftDot rightDot:(CGRect *)rightDot {
    CGMutablePathRef _path = CGPathCreateMutable();
    // 高亮颜色
    [[UIColor cyanColor] setFill];
    
    for (int i = 0; i < array.count; i++) {
        CGRect rect = CGRectFromString([array objectAtIndex:i]);
        CGPathAddRect(_path, NULL, rect);
        if (i == 0) {
            *leftDot = rect;
            _menuRect = rect;
        }
        if (i == array.count - 1) {
            *rightDot = rect;
        }
    }
    CGContextRef ctx = UIGraphicsGetCurrentContext();
    CGContextAddPath(ctx, _path);
    CGContextFillPath(ctx);
    CGPathRelease(_path);
}

方法执行路径:
- (void)longPressAction:
+ (CGRect)parserRectWithPoint: frameRef: selectRange:
- (void)drawRect:
- (void)drawSelectedPath: leftDot: rightDot:

拖动大头针,可以改变文本选中长度

添加一个拖拽事件:

- (void)panAction:(UIPanGestureRecognizer *)recognizer {
    CGPoint point = [recognizer locationInView:self];
    // 显示放大镜
    [self.magnifier show:point];
    if (CGRectContainsPoint(_rightRect, point) || CGRectContainsPoint(_leftRect, point)) {
        _direction = !CGRectContainsPoint(_leftRect, point);
    }
    // 将rect添加到数组中
    _pathArray = [HYReadParser parserRectsWithPoint:point frameRef:_frameRef selectRange:&_selectRange paths:_pathArray direction:_direction];
    [self setNeedsDisplay];
    
    if (recognizer.state == UIGestureRecognizerStateEnded) {
        [self hideMagnifier];
        if (!CGRectEqualToRect(_menuRect, CGRectZero)) {
            [self showMenu];
        }
    }
}

传入directionpointframeRef,返回rect数组和selectRange

+ (NSArray *)parserRectsWithPoint:(CGPoint)point frameRef:(CTFrameRef)frameRef selectRange:(NSRange *)selectRange paths:(NSArray *)paths direction:(BOOL)direction {
    CFIndex index = -1;
    // 获取行数据
    NSArray *lines = (NSArray *)CTFrameGetLines(frameRef);
    NSMutableArray *muArr = [NSMutableArray array];
    NSInteger lineCount = lines.count;
    // 为每一行起点开辟内存空间
    CGPoint *origins = malloc(lineCount * sizeof(CGPoint));
    // 获取滑动后文本所在index
    index = [self parserIndexWithPoint:point frameRef:frameRef];
    if (index == -1) {
        return paths;
    }
    if (direction) { // 从右侧滑动
        if (!(index > (*selectRange).location)) {
            (*selectRange).length = (*selectRange).location - index + (*selectRange).length;
            (*selectRange).location = index;
        } else{
            (*selectRange).length = index - (*selectRange).location;
        }
    } else { // 从左侧滑动
        if (!(index > (*selectRange).location + (*selectRange).length)) {
            (*selectRange).length = (*selectRange).location - index + (*selectRange).length;
            (*selectRange).location = index;
        }
    }
    if (lineCount) {
        CTFrameGetLineOrigins(frameRef, CFRangeMake(0, 0), origins);
        for (int i = 0; i < lineCount; i++){
            CGPoint baselineOrigin = origins[i];
            CTLineRef line = (__bridge CTLineRef)[lines objectAtIndex:i];
            CGFloat ascent,descent;
            CTLineGetTypographicBounds(line, &ascent, &descent, NULL);
            CFRange stringRange = CTLineGetStringRange(line);
            CGFloat xStart;
            CGFloat xEnd;
            NSRange drawRange = [self selectRange:NSMakeRange((*selectRange).location, (*selectRange).length) lineRange:NSMakeRange(stringRange.location, stringRange.length)];
            if (drawRange.length) {
                xStart = CTLineGetOffsetForStringIndex(line, drawRange.location, NULL);
                xEnd = CTLineGetOffsetForStringIndex(line, drawRange.location+drawRange.length, NULL);
                CGRect rect = CGRectMake(xStart, baselineOrigin.y-descent, fabs(xStart-xEnd), ascent+descent);
                if (rect.size.width == 0 || rect.size.height == 0) {
                    continue;
                }
                [muArr addObject:NSStringFromCGRect(rect)];
            }
        }
    }
    return muArr;
}

获取当前滑动后文本所在index

+ (CFIndex)parserIndexWithPoint:(CGPoint)point frameRef:(CTFrameRef)frameRef {
    CFIndex index = -1;
    CGPathRef pathRef = CTFrameGetPath(frameRef);
    CGRect bounds = CGPathGetBoundingBox(pathRef);
    NSArray *lines = (__bridge NSArray *)CTFrameGetLines(frameRef);
    if (!lines) {
        return index;
    }
    NSInteger lineCount = [lines count];
    CGPoint *origins = malloc(lineCount * sizeof(CGPoint)); //给每行的起始点开辟内存
    if (lineCount) {
        CTFrameGetLineOrigins(frameRef, CFRangeMake(0, 0), origins);
        for (int i = 0; i<lineCount; i++) {
            CGPoint baselineOrigin = origins[i];
            CTLineRef line = (__bridge CTLineRef)[lines objectAtIndex:i];
            CGFloat ascent,descent,linegap; //声明字体的上行高度和下行高度和行距
            CGFloat lineWidth = CTLineGetTypographicBounds(line, &ascent, &descent, &linegap);
            CGRect lineFrame = CGRectMake(baselineOrigin.x, CGRectGetHeight(bounds)-baselineOrigin.y-ascent, lineWidth, ascent+descent+linegap+[HYReadConfig shareInstance].lineSpace);    //没有转换坐标系左下角为坐标原点 字体高度为上行高度加下行高度
            if (CGRectContainsPoint(lineFrame,point)){
                index = CTLineGetStringIndexForPosition(line, point);
                break;
            }
        }
    }
    free(origins);
    return index;
}

方法执行路径:
- (void)panAction:
+ (NSArray *)parserRectsWithPoint: frameRef: selectRange: paths: direction:
+ (CFIndex)parserIndexWithPoint: frameRef:
- (void)drawRect:
- (void)drawSelectedPath: leftDot: rightDot:

保存笔记,并添加下划线:

- (void)menuNote:(id)sender {
    [self hideMenu];
    UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"笔记" message:[_content substringWithRange:_selectRange]  preferredStyle:UIAlertControllerStyleAlert];
    [alertController addTextFieldWithConfigurationHandler:^(UITextField * _Nonnull textField) {
       textField.placeholder = @"输入内容";
    }];
    UIAlertAction *cancel = [UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:nil];
    UIAlertAction *confirm = [UIAlertAction actionWithTitle:@"确认" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
        HYNoteModel *model = [[HYNoteModel alloc] init];
        model.content = [self.content substringWithRange:self.selectRange];
        model.note = alertController.textFields.firstObject.text;
        model.date = [NSDate date];
        HYRecordModel *record = [HYReadManager sharedManager].readModel.record;
        NSValue *value = record.chapterModel.pageArray[record.page];
        NSRange pageLocation = value.rangeValue;
        // 笔记在章节中所在的位置 = 选中范围起点 + 当前页的范围起点
        model.locationInChapterContent = self.selectRange.location + pageLocation.location;
        [[NSNotificationCenter defaultCenter] postNotificationName:LSYNoteNotification object:model];
        [self cancelSelected];
    }];
    [alertController addAction:cancel];
    [alertController addAction:confirm];
    for (UIView* next = [self superview]; next; next = next.superview) {
        UIResponder* nextResponder = [next nextResponder];
        if ([nextResponder isKindOfClass:[UIViewController class]]) {
            [(UIViewController *)nextResponder presentViewController:alertController animated:YES completion:nil];
            break;
        }
    }
}
- (void)addNotes:(NSNotification *)no {
    HYNoteModel *model = no.object;
    model.chapter = _readModel.record.chapter;
    [_readModel addNote:model];
    
    HYReadViewController *vc = [self readViewWithChapter:_readModel.record.chapter page:_page];
    [_pageViewController setViewControllers:@[vc] direction:UIPageViewControllerNavigationDirectionForward animated:NO completion:nil];
    [self updateReadModelWithChapter:_readModel.record.chapter page:_page];

    NSLog(@"保存笔记成功!");
}

修改创建CTFrameRef的方法:

+ (CTFrameRef)parserContent:(NSString *)content config:(HYReadConfig *)config bouds:(CGRect)bounds notes:(NSArray *)notes pageRange:(NSRange)pageRange {
    NSMutableAttributedString *attributeStr = [[NSMutableAttributedString alloc] initWithString:content];
    NSDictionary *attribute = [self parserAttribute:config];
    [attributeStr addAttributes:attribute range:NSMakeRange(0, content.length)];
    
    if (notes) {
        for (HYNoteModel *noteModel in notes) {
            NSRange range = NSMakeRange(noteModel.locationInChapterContent, noteModel.content.length);
            NSMutableDictionary *attibutes = [NSMutableDictionary dictionary];
            [attibutes setObject:@(NSUnderlinePatternSolid|NSUnderlineStyleSingle) forKey:NSUnderlineStyleAttributeName];
            [attibutes setObject:[UIColor redColor] forKey:NSUnderlineColorAttributeName];
            [attibutes setObject:[noteModel getNoteURL] forKey:NSLinkAttributeName];
            // 返回range交集
            NSRange intersectionRange = NSIntersectionRange(range, pageRange);
            // 实际需要划线的range
            NSRange actualRange = NSMakeRange(NSNotFound, 0);
            // 存在交集
            if (intersectionRange.location != NSNotFound && intersectionRange.length != 0) {
                // 交集在中间位置
                if (range.location > pageRange.location) {
                    actualRange.location = range.location - pageRange.location;
                } else {
                    actualRange.location = 0;
                }
                actualRange.length = intersectionRange.length;
            }
            if (actualRange.location != NSNotFound) {
                [attributeStr addAttributes:attibutes range:actualRange];
            }
        }
    }
    
    CTFramesetterRef framesetter = CTFramesetterCreateWithAttributedString((__bridge CFAttributedStringRef)attributeStr);
    CGPathRef path = CGPathCreateWithRect(bounds, NULL);
    CTFrameRef frame = CTFramesetterCreateFrame(framesetter, CFRangeMake(0, 0), path, NULL);
    
    CFRelease(framesetter);
    CFRelease(path);
    
    return frame;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容