Skip to content

Commit 47c147a

Browse files
committed
Handle relative size using GeometryReader
1 parent 14b521e commit 47c147a

File tree

2 files changed

+38
-11
lines changed

2 files changed

+38
-11
lines changed

Sources/MarkdownUI/Utility/InlineNode+RawImageData.swift

+4-5
Original file line numberDiff line numberDiff line change
@@ -89,16 +89,15 @@ extension InlineNode {
8989
/// - `{width=50px height=100px}`
9090
/// - `{height=50px width=100px}`
9191
/// - `{width=50%}`
92-
/// - `{height=50%}`
93-
/// - `{width=50% height=100%}`
94-
/// - `{height=50% width=100%}`
92+
///
93+
/// - Note: Relative height is not supported
9594
struct MarkdownImageSize {
9695
let value: Value
9796

9897
enum Value {
99-
/// Represents a fixed value size:`.fixed(width, height)`
98+
/// Represents a fixed value size: `.fixed(width, height)`
10099
case fixed(CGFloat?, CGFloat?)
101-
/// Represents a relative value size: `.relative(proportionalWidth, proportionalHeight)`
100+
/// Represents a relative value size: `.relative(relativeWidth, relativeHeight)`
102101
case relative(CGFloat, CGFloat)
103102
}
104103
}

Sources/MarkdownUI/Views/Inlines/ImageView.swift

+34-6
Original file line numberDiff line numberDiff line change
@@ -105,22 +105,50 @@ extension View {
105105
private struct ImageViewFrameModifier: ViewModifier {
106106
let size: MarkdownImageSize?
107107

108+
@State private var currentSize: CGSize = .zero
109+
108110
func body(content: Content) -> some View {
109111
if let size {
110112
switch size.value {
111113
case .fixed(let width, let height):
112-
content.frame(width: width, height: height)
114+
content
115+
.frame(width: width, height: height)
113116
case .relative(let wRatio, _):
114-
if #available(iOS 17.0, *) {
115-
content
116-
// .containerRelativeFrame(.vertical) { height, _ in height * hRatio }
117-
.containerRelativeFrame(.horizontal) { width, _ in width * wRatio }
118-
} else {
117+
ZStack(alignment: .leading) {
118+
/// Track the full content width.
119+
GeometryReader { metrics in
120+
content
121+
.preference(key: BoundsPreferenceKey.self, value: metrics.frame(in: .global).size)
122+
}
123+
.opacity(0.0)
124+
125+
/// Draw the content applying relative width. Relative height is not handled.
119126
content
127+
.frame(
128+
width: currentSize.width * wRatio
129+
)
130+
}
131+
.onPreferenceChange(BoundsPreferenceKey.self) { newValue in
132+
/// Avoid recursive loop that could happens
133+
/// https://developer.apple.com/videos/play/wwdc2022/10056/?time=1107
134+
if Int(currentSize.width) == Int(newValue.width),
135+
Int(currentSize.height) == Int(newValue.height) {
136+
return
137+
}
138+
139+
self.currentSize = newValue
120140
}
121141
}
122142
} else {
123143
content
124144
}
125145
}
126146
}
147+
148+
private struct BoundsPreferenceKey: PreferenceKey {
149+
static var defaultValue: CGSize = .zero
150+
151+
static func reduce(value: inout Value, nextValue: () -> Value) {
152+
value = nextValue()
153+
}
154+
}

0 commit comments

Comments
 (0)