Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[DO NOT MERGE] Ul indent styles #1410

Closed
wants to merge 12 commits into from
22 changes: 20 additions & 2 deletions Aztec/Classes/TextKit/LayoutManager.swift
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,9 @@ class LayoutManager: NSLayoutManager {
///
var blockquoteBorderWidth: CGFloat = 2

/// The list indent style
///
var listIndentStyle: TextList.IndentStyle = .default

/// Draws the background, associated to a given Text Range
///
Expand Down Expand Up @@ -213,15 +216,16 @@ private extension LayoutManager {
}

let characterRange = self.characterRange(forGlyphRange: glyphsToShow, actualGlyphRange: nil)
var firstLevelWidth: CGFloat?

textStorage.enumerateParagraphRanges(spanning: characterRange) { (range, enclosingRange) in

guard textStorage.string.isStartOfNewLine(atUTF16Offset: enclosingRange.location),
let paragraphStyle = textStorage.attribute(.paragraphStyle, at: enclosingRange.location, effectiveRange: nil) as? ParagraphStyle,
let list = paragraphStyle.lists.last
else {
return
}

let attributes = textStorage.attributes(at: enclosingRange.location, effectiveRange: nil)
let glyphRange = self.glyphRange(forCharacterRange: enclosingRange, actualCharacterRange: nil)
let markerRect = rectForItem(range: glyphRange, origin: origin, paragraphStyle: paragraphStyle)
Expand All @@ -233,8 +237,22 @@ private extension LayoutManager {
start = textStorage.numberOfItems(in: list, at: enclosingRange.location)
}
}

var indentLevel: Int?
// Determine indentation level, if needed. The indentation level is only determined for the standard list style
if listIndentStyle == .varied {
// only get the width of the first level once
if firstLevelWidth == nil {
firstLevelWidth = paragraphStyle.indentToFirst(TextList.self)
}

// calculate current indent level
let indentWidth = paragraphStyle.indentToLast(TextList.self)
indentLevel = Int(indentWidth / firstLevelWidth!)
}

markerNumber += start
let markerString = list.style.markerText(forItemNumber: markerNumber)
let markerString = list.style.markerText(forItemNumber: markerNumber, indentLevel: indentLevel)
drawItem(markerString, in: markerRect, styled: attributes, at: enclosingRange.location)
}
}
Expand Down
64 changes: 61 additions & 3 deletions Aztec/Classes/TextKit/ParagraphProperty/TextList.swift
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
import Foundation
import UIKit

fileprivate let DefaultUnorderedListMarkerText = "\u{2022}"
fileprivate let romanMarker = NSTextList(markerFormat: .lowercaseRoman, options: 0)

// MARK: - Text List
//
Expand All @@ -14,14 +16,51 @@ open class TextList: ParagraphProperty {
case ordered
case unordered

func markerText(forItemNumber number: Int) -> String {
func markerText(forItemNumber number: Int, indentLevel: Int? = nil) -> String {
switch self {
case .ordered: return "\(number)."
case .unordered: return "\u{2022}"
case .ordered:
if indentLevel == nil {
return "\(number)."
}

switch indentLevel {
case 1:
return "\(number)."
case 2:
let text = getLetter(for: number)
return "\(text)."
default:
// marker for all levels > 2
let text = romanMarker.marker(forItemNumber: number)
return "\(text)."
}
case .unordered:
if indentLevel == nil {
return DefaultUnorderedListMarkerText
}

switch indentLevel {
case 1:
return DefaultUnorderedListMarkerText
case 2:
return "\u{2E30}"
default:
// marker for all levels > 2
return "\u{2B29}"
}
}
}
}

/// List Indent Styles
///
public enum IndentStyle: Int {
/// A default single bullet style for each indentation level
case `default`
/// Use a varied (distinct) bullet style for each indentation level (i.e., WYSIWYG style)
case varied
}

public let reversed: Bool

public let start: Int?
Expand Down Expand Up @@ -90,3 +129,22 @@ open class TextList: ParagraphProperty {
return lhs.style == rhs.style && lhs.start == rhs.start && lhs.reversed == rhs.reversed
}
}

/// Returns the letters to use as the ordered list marker text
fileprivate func getLetter(for number: Int) -> String {
let listChars = "abcdefghijklmnopqrstuvwxyz"
let charCount = listChars.count

// for recursion
func convert(_ value: Int) -> String {
if value <= charCount {
return String(listChars[listChars.index(listChars.startIndex, offsetBy: value - 1)])
}

let quotient = (value - 1) / charCount
let remainder = (value - 1) % charCount
return convert(quotient) + String(listChars[listChars.index(listChars.startIndex, offsetBy: remainder)])
}

return convert(abs(number))
}
13 changes: 12 additions & 1 deletion Aztec/Classes/TextKit/TextView.swift
Original file line number Diff line number Diff line change
Expand Up @@ -231,7 +231,18 @@ open class TextView: UITextView {
// MARK: - Properties: Text Lists

var maximumListIndentationLevels = 7


/// The list indent style
/// Default is `default`, single style for each level.
public var listIndentStyle: TextList.IndentStyle {
get {
return layout.listIndentStyle
}
set {
layout.listIndentStyle = newValue
}
}

// MARK: - Properties: Blockquotes

/// The max levels of quote indentation allowed
Expand Down
2 changes: 1 addition & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ _None._

### New Features

_None._
* Added support for alternate bullet styles per level for ordered and unordered lists [#1409]

### Bug Fixes

Expand Down
3 changes: 2 additions & 1 deletion Example/Example/EditorDemoController.swift
Original file line number Diff line number Diff line change
Expand Up @@ -174,6 +174,7 @@ class EditorDemoController: UIViewController {
view.addSubview(separatorView)

editorView.richTextView.textContainer.lineFragmentPadding = 0
editorView.richTextView.listIndentStyle = .varied
// color setup
if #available(iOS 13.0, *) {
view.backgroundColor = UIColor.systemBackground
Expand Down Expand Up @@ -257,7 +258,7 @@ class EditorDemoController: UIViewController {
rightMargin -= view.safeAreaInsets.right

scrollInsets.right = -rightMargin
editorView.scrollIndicatorInsets = scrollInsets
editorView.horizontalScrollIndicatorInsets = scrollInsets
}

func updateTitleHeight() {
Expand Down