Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/lixiang1994/AttributedString
基于Swift插值方式优雅的构建富文本, 支持点击长按事件, 支持不同类型过滤, 支持自定义视图等.
https://github.com/lixiang1994/AttributedString
app apple attributedstring attributes click cocoapods ios label macos nsattributedstring paragraph regex rich-text string swift swift5 text textview tvos watchos
Last synced: 3 months ago
JSON representation
基于Swift插值方式优雅的构建富文本, 支持点击长按事件, 支持不同类型过滤, 支持自定义视图等.
- Host: GitHub
- URL: https://github.com/lixiang1994/AttributedString
- Owner: lixiang1994
- License: mit
- Created: 2019-11-18T02:35:35.000Z (about 5 years ago)
- Default Branch: master
- Last Pushed: 2024-04-17T09:39:00.000Z (9 months ago)
- Last Synced: 2024-07-05T15:13:15.180Z (7 months ago)
- Topics: app, apple, attributedstring, attributes, click, cocoapods, ios, label, macos, nsattributedstring, paragraph, regex, rich-text, string, swift, swift5, text, textview, tvos, watchos
- Language: Swift
- Homepage:
- Size: 9.94 MB
- Stars: 827
- Watchers: 15
- Forks: 76
- Open Issues: 14
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- awesome-ios - AttributedString - 基于Swift插值方式优雅的构建富文本 <br> (UI Component)
README
![Logo](Resources/logo.png)
# AttributedString - 基于Swift插值方式优雅的构建富文本
[![License](https://img.shields.io/cocoapods/l/AttributedString.svg)](LICENSE)
![Swift](https://img.shields.io/badge/Swift-5.2-orange.svg)
![Platform](https://img.shields.io/cocoapods/p/AttributedString.svg?style=flat)
[![Swift Package Manager](https://img.shields.io/badge/Swift_Package_Manager-compatible-4BC51D.svg?style=flat")](https://swift.org/package-manager/)
[![Carthage](https://img.shields.io/badge/Carthage-compatible-4BC51D.svg?style=flat)](https://github.com/Carthage/Carthage)
[![Cocoapods](https://img.shields.io/cocoapods/v/AttributedString.svg)](https://cocoapods.org)## [🇨🇳天朝子民](README_CN.md)
## Features
- [x] Constructing rich text using interpolation, Smooth coding, Elegant and natural style.
- [x] More control extension support.
- [x] Support for multi-level rich text cascading and provide other style priority strategies.
- [x] Support for all `NSAttributedString.Key` functions.
- [x] Support iOS & macOS & watchOS & tvOS.
- [x] Support text and attachment click or press event callback, support highlight style.
- [x] Support async image attachment, you can load remote image to `UITextView`.
- [x] Support view attachment, you can add custom view to `UITextView`.
- [x] Continue to add more new features.## Screenshot
## Installation
#### CocoaPods - Podfile
```ruby
pod 'AttributedString'
```#### Carthage - Cartfile
```ruby
github "lixiang1994/AttributedString"
```#### [Swift Package Manager for Apple platforms](https://developer.apple.com/documentation/xcode/adding_package_dependencies_to_your_app)
Select Xcode menu `File > Swift Packages > Add Package Dependency` and enter repository URL with GUI.
```
Repository: https://github.com/lixiang1994/AttributedString
```#### [Swift Package Manager](https://swift.org/package-manager/)
Add the following to the dependencies of your `Package.swift`:
```swift
.package(url: "https://github.com/lixiang1994/AttributedString.git", from: "version")
```## Usage
First make sure to import the framework:
```swift
import AttributedString
```How to initialize:
```swift
// Normal
let a: ASAttributedString = .init("lee", .font(.systemFont(ofSize: 13)))
// Interpolation
let b: ASAttributedString = "\("lee", .font(.systemFont(ofSize: 13)))"
```Here are some usage examples. All devices are also available as simulators:
#### Font:
```swift
textView.attributed.text = """\("fontSize: 13", .font(.systemFont(ofSize: 13)))
\("fontSize: 20", .font(.systemFont(ofSize: 20)))
\("fontSize: 22 weight: semibold", .font(.systemFont(ofSize: 22, weight: .semibold)))
"""
```#### ForegroundColor:
```swift
textView.attributed.text = """\("foregroundColor", .foreground(.white))
\("foregroundColor", .foreground(.red))
"""
```#### Strikethrough:
```swift
textView.attributed.text = """\("strikethrough: single", .strikethrough(.single))
\("strikethrough: double color: .red", .strikethrough(.double, color: .red))
"""
```#### Attachment: (Does not include watchOS)
```swift
// ASAttributedString.AttachmenttextView.attributed.text = """
\(.data(xxxx, type: "zip"))
\(.file(try!.init(url: .init(fileURLWithPath: "xxxxx"), options: [])))
\(.attachment(NSTextAttachment()))
"""
```#### Attachment Image: (Does not include watchOS)
```swift
// ASAttributedString.ImageAttachmenttextView.attributed.text = """
\(.image(UIImage(named: "xxxx")))
\(.image(UIImage(named: "xxxx"), .custom(size: .init(width: 200, height: 200))))
\(.image(UIImage(named: "xxxx"), .proposed(.center))).
"""
```#### Attachment Async Image: (Only supports iOS: UITextView)
```swift
// ASAttributedString.AsyncImageAttachmenttextView.attributed.text = """
\(.image(url, placeholder: xxxxx))
"""
```Custom loader:
```swift
ASAttributedString.AsyncImageAttachment.Loader = AsyncImageAttachmentKingfisherLoader.self
```Please read the demo `AttachmentViewController.swift` file for details.
#### Attachment View: (Only supports iOS: UITextView)
```swift
// ASAttributedString.ViewAttachmenttextView.attributed.text = """
\(.view(xxxxView))
\(.view(xxxxView, .custom(size: .init(width: 200, height: 200))))
\(.view(xxxxView, .proposed(.center))).
"""
```#### Wrap:
```swift
let a: ASAttributedString = .init("123", .background(.blue))
let b: ASAttributedString = .init("456", .background(.red))
textView.attributed.text = "\(wrap: a) \(wrap: b, .paragraph(.alignment(.center)))"// Defalut embedding mode, Nested internal styles take precedence over external styles
textView.attributed.text = "\(wrap: a, .paragraph(.alignment(.center)))"
textView.attributed.text = "\(wrap: .embedding(a), .paragraph(.alignment(.center)))"
// Override mode, Nested outer style takes precedence over inner style
textView.attributed.text = "\(wrap: .override(a), .paragraph(.alignment(.center)))"
```#### Append:
```swift
let a: ASAttributedString = .init("123", .background(.blue))
let b: ASAttributedString = .init("456", .background(.red))
let c: ASAttributedString = .init("789", .background(.gray))
textView.attributed.text = a + b
textView.attributed.text += c
```#### Checking:
```swift
var string: ASAttributedString = .init("my phone number is +86 18611401994.", .background(.blue))
string.add(attributes: [.foreground(color)], checkings: [.phoneNumber])
textView.attributed.text = string
``````swift
var string: ASAttributedString = .init("open https://www.apple.com and https://github.com/lixiang1994/AttributedString", .background(.blue))
string.add(attributes: [.foreground(color)], checkings: [.link])
textView.attributed.text = string
``````swift
var string: ASAttributedString = .init("123456789", .background(.blue))
string.add(attributes: [.foreground(color)], checkings: [.regex("[0-6]")])
textView.attributed.text = string
```#### Action: (Only supports iOS: UILabel / UITextView & macOS: NSTextField)
For complex styles, it is recommended to use UITextView.
UITextview needs to set `isEditable` and `isSelectable` to `false`.
##### Click:
```swift
// Text
let a: ASAttributedString = .init("lee", .action({ }))
// Attachment (image)
let b: ASAttributedString = .init(.image(image), action: {
// code
})// It is recommended to use functions as parameters.
func clicked() {
// code
}
// Normal
let c: ASAttributedString = .init("lee", .action(clicked))
let d: ASAttributedString = .init(.image(image), action: clicked)
// Interpolation
let e: ASAttributedString = "\("lee", .action(clicked))"
let f: ASAttributedString = "\(.image(image), action: clicked)"// More information.
func clicked(_ result: ASAttributedString.Action.Result) {
switch result.content {
case .string(let value):
print("Currently clicked text: \(value) range: \(result.range)")
case .attachment(let value):
print("Currently clicked attachment: \(value) range: \(result.range)")
}
}label.attributed.text = "This is \("Label", .font(.systemFont(ofSize: 20)), .action(clicked))"
textView.attributed.text = "This is a picture \(.image(image, .custom(size: .init(width: 100, height: 100))), action: clicked) Displayed in custom size."
```##### Press:
```swift
func pressed(_ result: ASAttributedString.Action.Result) {
switch result.content {
case .string(let value):
print("Currently pressed text: \(value) range: \(result.range)")
case .attachment(let value):
print("Currently pressed attachment: \(value) range: \(result.range)")
}
}label.attributed.text = "This is \("Long Press", .font(.systemFont(ofSize: 20)), .action(.press, pressed))"
textView.attributed.text = "This is a picture \(.image(image, .custom(size: .init(width: 100, height: 100))), trigger: .press, action: pressed) Displayed in custom size."
```##### Highlight style:
```swift
func clicked(_ result: ASAttributedString.Action.Result) {
switch result.content {
case .string(let value):
print("Currently clicked text: \(value) range: \(result.range)")
case .attachment(let value):
print("Currently clicked attachment: \(value) range: \(result.range)")
}
}label.attributed.text = "This is \("Label", .font(.systemFont(ofSize: 20)), .action([.foreground(.blue)], clicked))"
```##### Custom:
```swift
let custom = ASAttributedString.Action(.press, highlights: [.background(.blue), .foreground(.white)]) { (result) in
switch result.content {
case .string(let value):
print("Currently pressed text: \(value) range: \(result.range)")
case .attachment(let value):
print("Currently pressed attachment: \(value) range: \(result.range)")
}
}label.attributed.text = "This is \("Custom", .font(.systemFont(ofSize: 20)), .action(custom))"
textView.attributed.text = "This is a picture \(.image(image, .original(.center)), action: custom) Displayed in original size."
```#### Observe: (Only supports iOS: UILabel / UITextView & macOS: NSTextField)
```swift
label.attributed.observe([.phoneNumber], highlights: [.foreground(.blue)]) { (result) in
print("Currently clicked \(result)")
}textView.attributed.observe([.link], highlights: [.foreground(.blue)]) { (result) in
print("Currently clicked \(result)")
}
```For more examples, see the sample application.
## Properties available via `Attribute` class
The following properties are available:
| PROPERTY | TYPE | DESCRIPTION |
| ----------------- | ------------------------------------ | ------------------------------------------------------------ |
| font | `UIFont` | font |
| color | `UIColor` | foreground color |
| background | `UIColor` | background color |
| paragraph | `ParagraphStyle` | paragraph attributes |
| ligature | `Bool` | Ligatures cause specific character combinations to be rendered using a single custom glyph that corresponds to those characters |
| kern | `CGFloat` | kerning |
| strikethrough | `NSUnderlineStyle` . `UIColor` | strikethrough style and color (if color is nil foreground is used) |
| underline | `NSUnderlineStyle` , `UIColor` | underline style and color (if color is nil foreground is used) |
| link | `String` / `URL` | URL |
| baselineOffset | `CGFloat` | character’s offset from the baseline, in point |
| shadow | `NSShadow` | shadow effect of the text |
| stroke | `CGFloat`, `UIColor` | stroke width and color |
| textEffect | `NSAttributedString.TextEffectStyle` | text effect |
| obliqueness | `CGFloat` | text obliqueness |
| expansion | `CGFloat` | expansion / shrink |
| writingDirection | `WritingDirection` / `[Int]` | initial writing direction used to determine the actual writing direction for text |
| verticalGlyphForm | `Bool` | vertical glyph (Currently on iOS, it's always horizontal.) |## Cases available via `Attribute.Checking` enumerated
| CASE | DESCRIPTION |
| ------------------------------------ | -------------------------------------------- |
| `range(NSRange)` | custom range |
| `regex(String)` | regular expression |
| `action` | action |
| `date` | date (Based on `NSDataDetector`) |
| `link` | link (Based on `NSDataDetector`) |
| `address` | address (Based on `NSDataDetector`) |
| `phoneNumber` | phone number (Based on `NSDataDetector`) |
| `transitInformation` | transit Information (Based on `NSDataDetector`) |## Contributing
If you have the need for a specific feature that you want implemented or if you experienced a bug, please open an issue.
If you extended the functionality of AttributedString yourself and want others to use it too, please submit a pull request.## License
AttributedString is under MIT license. See the [LICENSE](LICENSE) file for more info.
-----
> ## 欢迎入群交流
![QQ](https://github.com/lixiang1994/Resources/blob/master/QQClub/QQClub.JPG)