Add scaledFrame/scaledPadding APIs (#2079)

* Add scaled frame and padding modifiers

* Refactor AppLockScreen

* Refactor FormattingToolbar

* Refactor RoomAttachmentPicker

* Refactor ComposerToolbar

* Refactor VoiceMessageRecordingButton

* More refactors

* Refine ScaledPaddingModifier

* Cleanup

* Fix preview tests

* Fix preview test

* Use CompoundIcon for custom image

* Add .scaledFrame(size:) overload

* Set fallback style to .body
This commit is contained in:
Alfonso Grillo 2023-11-14 12:20:34 +01:00 committed by GitHub
parent 4257c20bc9
commit 752ad66397
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
43 changed files with 172 additions and 121 deletions

View File

@ -364,6 +364,7 @@
62A7FC3A0191BC7181AA432B /* AudioRecorder.swift in Sources */ = {isa = PBXBuildFile; fileRef = 907FA4DE17DEA1A3738EFB83 /* AudioRecorder.swift */; }; 62A7FC3A0191BC7181AA432B /* AudioRecorder.swift in Sources */ = {isa = PBXBuildFile; fileRef = 907FA4DE17DEA1A3738EFB83 /* AudioRecorder.swift */; };
63CDC201A5980F304F6D0A1C /* WaveformInteractionModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFEE91FB8ABB5F5884B6D940 /* WaveformInteractionModifier.swift */; }; 63CDC201A5980F304F6D0A1C /* WaveformInteractionModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFEE91FB8ABB5F5884B6D940 /* WaveformInteractionModifier.swift */; };
63E46D18B91D08E15FC04125 /* ExpiringTaskRunner.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B25F959A434BB9923A3223F /* ExpiringTaskRunner.swift */; }; 63E46D18B91D08E15FC04125 /* ExpiringTaskRunner.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7B25F959A434BB9923A3223F /* ExpiringTaskRunner.swift */; };
6409CE10CFF4DCB68C4C3872 /* ScaledPaddingModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = E26C69EC1157D71CC61ADAE4 /* ScaledPaddingModifier.swift */; };
642DF13C49ED4121C148230E /* TestablePreview.swift in Sources */ = {isa = PBXBuildFile; fileRef = B1E227F34BE43B08E098796E /* TestablePreview.swift */; }; 642DF13C49ED4121C148230E /* TestablePreview.swift in Sources */ = {isa = PBXBuildFile; fileRef = B1E227F34BE43B08E098796E /* TestablePreview.swift */; };
6448F8D1D3CA4CD27BB4CADD /* RoomMemberProxy.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2F36C5D9B37E50915ECBD3EE /* RoomMemberProxy.swift */; }; 6448F8D1D3CA4CD27BB4CADD /* RoomMemberProxy.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2F36C5D9B37E50915ECBD3EE /* RoomMemberProxy.swift */; };
644AA5001BCC58D7732EB772 /* MigrationScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 12EDAFB64FA5F6812D54F39A /* MigrationScreenViewModel.swift */; }; 644AA5001BCC58D7732EB772 /* MigrationScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 12EDAFB64FA5F6812D54F39A /* MigrationScreenViewModel.swift */; };
@ -937,6 +938,7 @@
F40B097470D3110DFDB1FAAA /* LegalInformationScreenModels.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47873756E45B46683D97DC32 /* LegalInformationScreenModels.swift */; }; F40B097470D3110DFDB1FAAA /* LegalInformationScreenModels.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47873756E45B46683D97DC32 /* LegalInformationScreenModels.swift */; };
F421FD5979EF53C8204BDC77 /* SecureBackupLogoutConfirmationScreenUITests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1CC09F30B0E1010951952BDC /* SecureBackupLogoutConfirmationScreenUITests.swift */; }; F421FD5979EF53C8204BDC77 /* SecureBackupLogoutConfirmationScreenUITests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 1CC09F30B0E1010951952BDC /* SecureBackupLogoutConfirmationScreenUITests.swift */; };
F4433EF57B4BB3C077F8B00E /* SessionVerificationScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ADD9E0FFA29EAACFF3AB9732 /* SessionVerificationScreenViewModel.swift */; }; F4433EF57B4BB3C077F8B00E /* SessionVerificationScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = ADD9E0FFA29EAACFF3AB9732 /* SessionVerificationScreenViewModel.swift */; };
F4971845B5C4F270F6BC5745 /* ScaledFrameModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5D82F234B3576BD6268C7950 /* ScaledFrameModifier.swift */; };
F508683B76EF7B23BB2CBD6D /* TimelineItemPlainStylerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94BCC8A9C73C1F838122C645 /* TimelineItemPlainStylerView.swift */; }; F508683B76EF7B23BB2CBD6D /* TimelineItemPlainStylerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 94BCC8A9C73C1F838122C645 /* TimelineItemPlainStylerView.swift */; };
F519DE17A3A0F760307B2E6D /* InviteUsersScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 02D155E09BF961BBA8F85263 /* InviteUsersScreenViewModel.swift */; }; F519DE17A3A0F760307B2E6D /* InviteUsersScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 02D155E09BF961BBA8F85263 /* InviteUsersScreenViewModel.swift */; };
F54E2D6CAD96E1AC15BC526F /* MessageForwardingScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = D8E60332509665C00179ACF6 /* MessageForwardingScreenViewModel.swift */; }; F54E2D6CAD96E1AC15BC526F /* MessageForwardingScreenViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = D8E60332509665C00179ACF6 /* MessageForwardingScreenViewModel.swift */; };
@ -1363,6 +1365,7 @@
5CD0FAE9EA761DA175D31CC7 /* MigrationScreenModels.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MigrationScreenModels.swift; sourceTree = "<group>"; }; 5CD0FAE9EA761DA175D31CC7 /* MigrationScreenModels.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MigrationScreenModels.swift; sourceTree = "<group>"; };
5D26A086A8278D39B5756D6F /* project.yml */ = {isa = PBXFileReference; lastKnownFileType = text.yaml; path = project.yml; sourceTree = "<group>"; }; 5D26A086A8278D39B5756D6F /* project.yml */ = {isa = PBXFileReference; lastKnownFileType = text.yaml; path = project.yml; sourceTree = "<group>"; };
5D2D0A6F1ABC99D29462FB84 /* AuthenticationCoordinatorUITests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AuthenticationCoordinatorUITests.swift; sourceTree = "<group>"; }; 5D2D0A6F1ABC99D29462FB84 /* AuthenticationCoordinatorUITests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AuthenticationCoordinatorUITests.swift; sourceTree = "<group>"; };
5D82F234B3576BD6268C7950 /* ScaledFrameModifier.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScaledFrameModifier.swift; sourceTree = "<group>"; };
5D99730313BEBF08CDE81EE3 /* EmojiDetection.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EmojiDetection.swift; sourceTree = "<group>"; }; 5D99730313BEBF08CDE81EE3 /* EmojiDetection.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = EmojiDetection.swift; sourceTree = "<group>"; };
5DE8D25D6A91030175D52A20 /* RoomTimelineItemProperties.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RoomTimelineItemProperties.swift; sourceTree = "<group>"; }; 5DE8D25D6A91030175D52A20 /* RoomTimelineItemProperties.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RoomTimelineItemProperties.swift; sourceTree = "<group>"; };
5EB2CAA266B921D128C35710 /* LegalInformationScreenCoordinator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LegalInformationScreenCoordinator.swift; sourceTree = "<group>"; }; 5EB2CAA266B921D128C35710 /* LegalInformationScreenCoordinator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LegalInformationScreenCoordinator.swift; sourceTree = "<group>"; };
@ -1821,6 +1824,7 @@
E1E0B4A34E69BD2132BEC521 /* MessageText.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MessageText.swift; sourceTree = "<group>"; }; E1E0B4A34E69BD2132BEC521 /* MessageText.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MessageText.swift; sourceTree = "<group>"; };
E24B88AD3D1599E8CB1376E0 /* AvatarSize.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AvatarSize.swift; sourceTree = "<group>"; }; E24B88AD3D1599E8CB1376E0 /* AvatarSize.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AvatarSize.swift; sourceTree = "<group>"; };
E26747B3154A5DBC3A7E24A5 /* Image.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Image.swift; sourceTree = "<group>"; }; E26747B3154A5DBC3A7E24A5 /* Image.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Image.swift; sourceTree = "<group>"; };
E26C69EC1157D71CC61ADAE4 /* ScaledPaddingModifier.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ScaledPaddingModifier.swift; sourceTree = "<group>"; };
E2B1CC9AA154F4D5435BF60A /* Comparable.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Comparable.swift; sourceTree = "<group>"; }; E2B1CC9AA154F4D5435BF60A /* Comparable.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Comparable.swift; sourceTree = "<group>"; };
E2DCA495ED42D2463DDAA94D /* TimelineBubbleLayout.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TimelineBubbleLayout.swift; sourceTree = "<group>"; }; E2DCA495ED42D2463DDAA94D /* TimelineBubbleLayout.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TimelineBubbleLayout.swift; sourceTree = "<group>"; };
E2F27BAB69EB568369F1F6B3 /* OnboardingScreenViewModelProtocol.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OnboardingScreenViewModelProtocol.swift; sourceTree = "<group>"; }; E2F27BAB69EB568369F1F6B3 /* OnboardingScreenViewModelProtocol.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OnboardingScreenViewModelProtocol.swift; sourceTree = "<group>"; };
@ -4309,6 +4313,8 @@
565F1B2B300597C616B37888 /* FullscreenDialog.swift */, 565F1B2B300597C616B37888 /* FullscreenDialog.swift */,
49ABAB186CF00B15C5521D04 /* MenuSheetLabelStyle.swift */, 49ABAB186CF00B15C5521D04 /* MenuSheetLabelStyle.swift */,
398817652FA8ABAE0A31AC6D /* ReadableFrameModifier.swift */, 398817652FA8ABAE0A31AC6D /* ReadableFrameModifier.swift */,
5D82F234B3576BD6268C7950 /* ScaledFrameModifier.swift */,
E26C69EC1157D71CC61ADAE4 /* ScaledPaddingModifier.swift */,
933B074F006F8E930DB98B4E /* TimelineMediaFrame.swift */, 933B074F006F8E930DB98B4E /* TimelineMediaFrame.swift */,
EFF7BF82A950B91BC5469E91 /* ViewFrameReader.swift */, EFF7BF82A950B91BC5469E91 /* ViewFrameReader.swift */,
); );
@ -5732,6 +5738,8 @@
D8385A51A3D0FA9283556281 /* RoundedLabelItem.swift in Sources */, D8385A51A3D0FA9283556281 /* RoundedLabelItem.swift in Sources */,
50C90117FE25390BFBD40173 /* RustTracing.swift in Sources */, 50C90117FE25390BFBD40173 /* RustTracing.swift in Sources */,
D43F0503EF2CBC55272538FE /* SDKGeneratedMocks.swift in Sources */, D43F0503EF2CBC55272538FE /* SDKGeneratedMocks.swift in Sources */,
F4971845B5C4F270F6BC5745 /* ScaledFrameModifier.swift in Sources */,
6409CE10CFF4DCB68C4C3872 /* ScaledPaddingModifier.swift in Sources */,
FB595EC9C00AB32F39034055 /* SceneDelegate.swift in Sources */, FB595EC9C00AB32F39034055 /* SceneDelegate.swift in Sources */,
0437765FF480249486893CC7 /* ScreenTrackerViewModifier.swift in Sources */, 0437765FF480249486893CC7 /* ScreenTrackerViewModifier.swift in Sources */,
0BFA67AFD757EE2BA569836A /* ScrollViewAdapter.swift in Sources */, 0BFA67AFD757EE2BA569836A /* ScrollViewAdapter.swift in Sources */,

View File

@ -263,7 +263,7 @@
{ {
"identity" : "swiftui-introspect", "identity" : "swiftui-introspect",
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/siteline/SwiftUI-Introspect", "location" : "https://github.com/siteline/SwiftUI-Introspect.git",
"state" : { "state" : {
"revision" : "b94da693e57eaf79d16464b8b7c90d09cba4e290", "revision" : "b94da693e57eaf79d16464b8b7c90d09cba4e290",
"version" : "0.9.2" "version" : "0.9.2"

View File

@ -125,14 +125,12 @@ struct FormButtonStyle: PrimitiveButtonStyle {
struct FormActionButtonStyle: ButtonStyle { struct FormActionButtonStyle: ButtonStyle {
let title: String let title: String
@ScaledMetric private var menuIconSize = 54.0
func makeBody(configuration: Configuration) -> some View { func makeBody(configuration: Configuration) -> some View {
VStack { VStack {
configuration.label configuration.label
.buttonStyle(.plain) .buttonStyle(.plain)
.foregroundColor(.compound.textPrimary) .foregroundColor(.compound.textPrimary)
.frame(width: menuIconSize, height: menuIconSize) .scaledFrame(size: 54)
.background { .background {
RoundedRectangle(cornerRadius: 16) RoundedRectangle(cornerRadius: 16)
.fill(configuration.isPressed ? Color.compound.bgSubtlePrimary : .compound.bgCanvasDefaultLevel1) .fill(configuration.isPressed ? Color.compound.bgSubtlePrimary : .compound.bgCanvasDefaultLevel1)

View File

@ -17,8 +17,6 @@
import SwiftUI import SwiftUI
struct FormRowLabelStyle: LabelStyle { struct FormRowLabelStyle: LabelStyle {
@ScaledMetric private var menuIconSize = 30.0
var alignment = VerticalAlignment.firstTextBaseline var alignment = VerticalAlignment.firstTextBaseline
var role: ButtonRole? var role: ButtonRole?
@ -51,7 +49,7 @@ struct FormRowLabelStyle: LabelStyle {
configuration.icon configuration.icon
.foregroundColor(iconForegroundColor) .foregroundColor(iconForegroundColor)
.padding(4) .padding(4)
.frame(width: menuIconSize, height: menuIconSize) .scaledFrame(size: 30)
.background(iconBackgroundColor) .background(iconBackgroundColor)
.clipShape(RoundedRectangle(cornerRadius: 8)) .clipShape(RoundedRectangle(cornerRadius: 8))
configuration.title configuration.title

View File

@ -0,0 +1,43 @@
//
// Copyright 2023 New Vector Ltd
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
import SwiftUI
extension View {
func scaledFrame(size: CGFloat, alignment: Alignment = .center, relativeTo textStyle: Font.TextStyle = .body) -> some View {
scaledFrame(width: size, height: size, alignment: alignment, relativeTo: textStyle)
}
func scaledFrame(width: CGFloat, height: CGFloat, alignment: Alignment = .center, relativeTo textStyle: Font.TextStyle = .body) -> some View {
modifier(ScaledFrameModifier(width: width, height: height, alignment: alignment, relativeTo: textStyle))
}
}
private struct ScaledFrameModifier: ViewModifier {
@ScaledMetric var width: CGFloat
@ScaledMetric var height: CGFloat
let alignment: Alignment
init(width: CGFloat, height: CGFloat, alignment: Alignment, relativeTo textStyle: Font.TextStyle) {
_width = ScaledMetric(wrappedValue: width, relativeTo: textStyle)
_height = ScaledMetric(wrappedValue: height, relativeTo: textStyle)
self.alignment = alignment
}
func body(content: Content) -> some View {
content.frame(width: width, height: height, alignment: alignment)
}
}

View File

@ -0,0 +1,41 @@
//
// Copyright 2023 New Vector Ltd
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
import SwiftUI
extension View {
func scaledPadding(_ length: CGFloat, relativeTo textStyle: Font.TextStyle = .body) -> some View {
scaledPadding(.all, length, relativeTo: textStyle)
}
func scaledPadding(_ edges: Edge.Set, _ length: CGFloat, relativeTo textStyle: Font.TextStyle = .body) -> some View {
modifier(ScaledPaddingModifier(edges: edges, length: length, textStyle: textStyle))
}
}
private struct ScaledPaddingModifier: ViewModifier {
let edges: Edge.Set
@ScaledMetric var length: CGFloat
init(edges: Edge.Set, length: CGFloat, textStyle: Font.TextStyle) {
self.edges = edges
_length = ScaledMetric(wrappedValue: length, relativeTo: textStyle)
}
func body(content: Content) -> some View {
content.padding(edges, length)
}
}

View File

@ -18,8 +18,6 @@ import SwiftUI
struct VoiceMessageButton: View { struct VoiceMessageButton: View {
@ScaledMetric private var buttonSize: CGFloat @ScaledMetric private var buttonSize: CGFloat
@ScaledMetric private var imageWidth: CGFloat = 12
@ScaledMetric private var imageHeight: CGFloat = 14
enum State { enum State {
case loading case loading
@ -71,7 +69,7 @@ struct VoiceMessageButton: View {
Image(asset: imageAsset) Image(asset: imageAsset)
.resizable() .resizable()
.scaledToFit() .scaledToFit()
.frame(width: imageWidth, height: imageHeight) .scaledFrame(width: 12, height: 14)
.offset(x: offset) .offset(x: offset)
} }
} }

View File

@ -20,8 +20,6 @@ import SwiftUI
struct AppLockScreen: View { struct AppLockScreen: View {
@ObservedObject var context: AppLockScreenViewModel.Context @ObservedObject var context: AppLockScreenViewModel.Context
/// The size of each dot within the PIN input field.
@ScaledMetric private var pinDotSize = 14
/// Used to animate the PIN input field on failure. /// Used to animate the PIN input field on failure.
@State private var pinInputFieldOffset = 0.0 @State private var pinInputFieldOffset = 0.0
@ -88,18 +86,20 @@ struct AppLockScreen: View {
/// The row of dots showing how many digits have been entered. /// The row of dots showing how many digits have been entered.
var pinInputField: some View { var pinInputField: some View {
HStack(spacing: 24) { HStack(spacing: 24) {
/// The size of each dot within the PIN input field.
let pinDotSize: CGFloat = 14
Circle() Circle()
.fill(context.viewState.numberOfDigitsEntered > 0 ? .compound.iconPrimary : .compound.bgSubtlePrimary) .fill(context.viewState.numberOfDigitsEntered > 0 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.frame(width: pinDotSize, height: pinDotSize) .scaledFrame(size: pinDotSize)
Circle() Circle()
.fill(context.viewState.numberOfDigitsEntered > 1 ? .compound.iconPrimary : .compound.bgSubtlePrimary) .fill(context.viewState.numberOfDigitsEntered > 1 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.frame(width: pinDotSize, height: pinDotSize) .scaledFrame(size: pinDotSize)
Circle() Circle()
.fill(context.viewState.numberOfDigitsEntered > 2 ? .compound.iconPrimary : .compound.bgSubtlePrimary) .fill(context.viewState.numberOfDigitsEntered > 2 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.frame(width: pinDotSize, height: pinDotSize) .scaledFrame(size: pinDotSize)
Circle() Circle()
.fill(context.viewState.numberOfDigitsEntered > 3 ? .compound.iconPrimary : .compound.bgSubtlePrimary) .fill(context.viewState.numberOfDigitsEntered > 3 ? .compound.iconPrimary : .compound.bgSubtlePrimary)
.frame(width: pinDotSize, height: pinDotSize) .scaledFrame(size: pinDotSize)
} }
} }

View File

@ -24,13 +24,6 @@ struct ComposerToolbar: View {
let keyCommandHandler: KeyCommandHandler let keyCommandHandler: KeyCommandHandler
@FocusState private var composerFocused: Bool @FocusState private var composerFocused: Bool
@ScaledMetric(relativeTo: .title) private var sendButtonIconSize = 16
@ScaledMetric(relativeTo: .title) private var sendButtonIconPadding = 10
@ScaledMetric(relativeTo: .title) private var sendButtonIconOffsetX = 1
@ScaledMetric(relativeTo: .title) private var spinnerSize = 44
@ScaledMetric(relativeTo: .title) private var closeRTEButtonSize = 30
@ScaledMetric(relativeTo: .title) private var deleteRecordingButtonSize = 30
@State private var frame: CGRect = .zero @State private var frame: CGRect = .zero
@Environment(\.verticalSizeClass) private var verticalSizeClass @Environment(\.verticalSizeClass) private var verticalSizeClass
@ -75,7 +68,7 @@ struct ComposerToolbar: View {
if !context.composerActionsEnabled { if !context.composerActionsEnabled {
if context.viewState.isUploading { if context.viewState.isUploading {
ProgressView() ProgressView()
.frame(width: spinnerSize, height: spinnerSize) .scaledFrame(size: 44, relativeTo: .title)
.padding(.leading, 3) .padding(.leading, 3)
} else if context.viewState.showSendButton { } else if context.viewState.showSendButton {
sendButton sendButton
@ -139,8 +132,8 @@ struct ComposerToolbar: View {
Image(Asset.Images.closeRte.name) Image(Asset.Images.closeRte.name)
.resizable() .resizable()
.scaledToFit() .scaledToFit()
.frame(width: closeRTEButtonSize, height: closeRTEButtonSize) .scaledFrame(size: 30, relativeTo: .title)
.padding(7) .scaledPadding(7, relativeTo: .title)
} }
.accessibilityLabel(L10n.actionClose) .accessibilityLabel(L10n.actionClose)
.accessibilityIdentifier(A11yIdentifiers.roomScreen.composerToolbar.closeFormattingOptions) .accessibilityIdentifier(A11yIdentifiers.roomScreen.composerToolbar.closeFormattingOptions)
@ -158,7 +151,7 @@ struct ComposerToolbar: View {
Circle() Circle()
.foregroundColor(context.viewState.sendButtonDisabled ? .clear : .compound.iconAccentTertiary) .foregroundColor(context.viewState.sendButtonDisabled ? .clear : .compound.iconAccentTertiary)
} }
.padding(4) .scaledPadding(4, relativeTo: .title)
} }
.disabled(context.viewState.sendButtonDisabled) .disabled(context.viewState.sendButtonDisabled)
.animation(.linear(duration: 0.1).disabledDuringTests(), value: context.viewState.sendButtonDisabled) .animation(.linear(duration: 0.1).disabledDuringTests(), value: context.viewState.sendButtonDisabled)
@ -217,11 +210,13 @@ struct ComposerToolbar: View {
.opacity(context.viewState.composerMode.isEdit ? 1 : 0) .opacity(context.viewState.composerMode.isEdit ? 1 : 0)
.accessibilityLabel(L10n.actionConfirm) .accessibilityLabel(L10n.actionConfirm)
.accessibilityHidden(!context.viewState.composerMode.isEdit) .accessibilityHidden(!context.viewState.composerMode.isEdit)
let sendImageOffset: CGFloat = 1
Image(asset: Asset.Images.sendMessage) Image(asset: Asset.Images.sendMessage)
.resizable() .resizable()
.offset(x: sendButtonIconOffsetX) .scaledToFit()
.frame(width: sendButtonIconSize, height: sendButtonIconSize) .scaledFrame(width: 16 + sendImageOffset, height: 16, alignment: .trailing, relativeTo: .title)
.padding(sendButtonIconPadding) .scaledPadding(10, relativeTo: .title)
.opacity(context.viewState.composerMode.isEdit ? 0 : 1) .opacity(context.viewState.composerMode.isEdit ? 0 : 1)
.accessibilityLabel(L10n.actionSend) .accessibilityLabel(L10n.actionSend)
.accessibilityHidden(context.viewState.composerMode.isEdit) .accessibilityHidden(context.viewState.composerMode.isEdit)
@ -270,8 +265,8 @@ struct ComposerToolbar: View {
} label: { } label: {
CompoundIcon(\.delete) CompoundIcon(\.delete)
.scaledToFit() .scaledToFit()
.frame(width: deleteRecordingButtonSize, height: deleteRecordingButtonSize) .scaledFrame(size: 30, relativeTo: .title)
.padding(7) .scaledPadding(7, relativeTo: .title)
} }
.buttonStyle(.compound(.plain)) .buttonStyle(.compound(.plain))
.accessibilityLabel(L10n.a11yDelete) .accessibilityLabel(L10n.a11yDelete)

View File

@ -22,8 +22,6 @@ struct FormattingToolbar: View {
/// The action when an item is selected /// The action when an item is selected
var formatAction: (FormatType) -> Void var formatAction: (FormatType) -> Void
@ScaledMetric private var toolbarButtonIconSize = 20
var body: some View { var body: some View {
ScrollView(.horizontal) { ScrollView(.horizontal) {
HStack(spacing: 4) { HStack(spacing: 4) {
@ -34,7 +32,7 @@ struct FormattingToolbar: View {
item.icon item.icon
.resizable() .resizable()
.scaledToFit() .scaledToFit()
.frame(width: toolbarButtonIconSize, height: toolbarButtonIconSize) .scaledFrame(size: 20)
.foregroundColor(item.foregroundColor) .foregroundColor(item.foregroundColor)
.padding(8) .padding(8)
.background(item.backgroundColor) .background(item.backgroundColor)

View File

@ -21,7 +21,6 @@ import WysiwygComposer
struct RoomAttachmentPicker: View { struct RoomAttachmentPicker: View {
@ObservedObject var context: ComposerToolbarViewModel.Context @ObservedObject var context: ComposerToolbarViewModel.Context
@Environment(\.isPresented) var isPresented @Environment(\.isPresented) var isPresented
@ScaledMetric(relativeTo: .title) private var attachmentButtonSize: CGFloat = 30
@State private var sheetContentHeight = CGFloat(0) @State private var sheetContentHeight = CGFloat(0)
@ -32,9 +31,9 @@ struct RoomAttachmentPicker: View {
Image(Asset.Images.composerAttachment.name) Image(Asset.Images.composerAttachment.name)
.resizable() .resizable()
.scaledToFit() .scaledToFit()
.frame(width: attachmentButtonSize, height: attachmentButtonSize) .scaledFrame(size: 30, relativeTo: .title)
.foregroundColor(.compound.textActionPrimary) .foregroundColor(.compound.textActionPrimary)
.padding(7) .scaledPadding(7, relativeTo: .title)
} }
.accessibilityLabel(L10n.actionAddToTimeline) .accessibilityLabel(L10n.actionAddToTimeline)
.accessibilityIdentifier(A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions) .accessibilityIdentifier(A11yIdentifiers.roomScreen.composerToolbar.openComposeOptions)

View File

@ -28,11 +28,6 @@ struct VoiceMessageRecordingButton: View {
var stopRecording: (() -> Void)? var stopRecording: (() -> Void)?
private let impactFeedbackGenerator = UIImpactFeedbackGenerator() private let impactFeedbackGenerator = UIImpactFeedbackGenerator()
@ScaledMetric(relativeTo: .title) private var idleImageSize = 16
@ScaledMetric(relativeTo: .title) private var idleImagePadding = 10
@ScaledMetric(relativeTo: .title) private var recordingImageSize = 24
@ScaledMetric(relativeTo: .title) private var recordingImagePadding = 6
var body: some View { var body: some View {
Button { Button {
@ -48,20 +43,16 @@ struct VoiceMessageRecordingButton: View {
case .idle: case .idle:
CompoundIcon(\.micOnOutline, size: .medium, relativeTo: .title) CompoundIcon(\.micOnOutline, size: .medium, relativeTo: .title)
.foregroundColor(.compound.iconSecondary) .foregroundColor(.compound.iconSecondary)
.frame(width: idleImageSize, height: idleImageSize) .scaledPadding(10, relativeTo: .title)
.padding(idleImagePadding)
.padding(4)
case .recording: case .recording:
Asset.Images.stopRecording.swiftUIImage CompoundIcon(customImage: Asset.Images.stopRecording.swiftUIImage, size: .medium, relativeTo: .title)
.resizable()
.foregroundColor(.compound.iconOnSolidPrimary) .foregroundColor(.compound.iconOnSolidPrimary)
.frame(width: recordingImageSize, height: recordingImageSize) .scaledPadding(6, relativeTo: .title)
.padding(recordingImagePadding)
.background( .background(
Circle() Circle()
.foregroundColor(.compound.bgActionPrimaryRest) .foregroundColor(.compound.bgActionPrimaryRest)
) )
.padding(4) .scaledPadding(4, relativeTo: .title)
} }
} }
.buttonStyle(VoiceMessageRecordingButtonStyle()) .buttonStyle(VoiceMessageRecordingButtonStyle())

View File

@ -45,7 +45,6 @@ struct CreateRoomScreen: View {
.alert(item: $context.alertInfo) .alert(item: $context.alertInfo)
} }
@ScaledMetric private var roomIconSize: CGFloat = 70
private var roomSection: some View { private var roomSection: some View {
Section { Section {
HStack(alignment: .center, spacing: 16) { HStack(alignment: .center, spacing: 16) {
@ -61,7 +60,7 @@ struct CreateRoomScreen: View {
} placeholder: { } placeholder: {
ProgressView() ProgressView()
} }
.frame(width: roomIconSize, height: roomIconSize) .scaledFrame(size: 70)
.clipShape(Circle()) .clipShape(Circle())
} else { } else {
cameraImage cameraImage
@ -106,7 +105,7 @@ struct CreateRoomScreen: View {
Image(systemName: "camera") Image(systemName: "camera")
.font(.system(size: 28, weight: .semibold)) .font(.system(size: 28, weight: .semibold))
.foregroundColor(.compound.iconSecondary) .foregroundColor(.compound.iconSecondary)
.frame(width: roomIconSize, height: roomIconSize) .scaledFrame(size: 70)
.background(Color.compound.bgSubtlePrimary) .background(Color.compound.bgSubtlePrimary)
.clipShape(Circle()) .clipShape(Circle())
} }

View File

@ -17,8 +17,6 @@
import SwiftUI import SwiftUI
struct HomeScreenInvitesButton: View { struct HomeScreenInvitesButton: View {
@ScaledMetric private var badgeSize = 12.0
let title: String let title: String
let hasBadge: Bool let hasBadge: Bool
let action: () -> Void let action: () -> Void
@ -44,7 +42,7 @@ struct HomeScreenInvitesButton: View {
private var badge: some View { private var badge: some View {
Circle() Circle()
.frame(width: badgeSize, height: badgeSize) .scaledFrame(size: 12)
.foregroundColor(.compound.iconAccentTertiary) .foregroundColor(.compound.iconAccentTertiary)
} }
} }

View File

@ -21,8 +21,6 @@ struct InviteUsersScreenSelectedItem: View {
let imageProvider: ImageProviderProtocol? let imageProvider: ImageProviderProtocol?
let dismissAction: () -> Void let dismissAction: () -> Void
@ScaledMetric private var buttonSize: CGFloat = 20
var body: some View { var body: some View {
VStack(spacing: 0) { VStack(spacing: 0) {
avatar avatar
@ -46,7 +44,7 @@ struct InviteUsersScreenSelectedItem: View {
Button(action: dismissAction) { Button(action: dismissAction) {
Image(systemName: "xmark.circle.fill") Image(systemName: "xmark.circle.fill")
.resizable() .resizable()
.frame(width: buttonSize, height: buttonSize) .scaledFrame(size: 20)
.symbolRenderingMode(.palette) .symbolRenderingMode(.palette)
.foregroundStyle(Color.compound.iconOnSolidPrimary, Color.compound.iconPrimary) .foregroundStyle(Color.compound.iconOnSolidPrimary, Color.compound.iconPrimary)
} }

View File

@ -23,8 +23,6 @@ struct InvitesScreenCell: View {
let acceptAction: () -> Void let acceptAction: () -> Void
let declineAction: () -> Void let declineAction: () -> Void
@ScaledMetric private var badgeSize = 12.0
var body: some View { var body: some View {
HStack(alignment: .top, spacing: 16) { HStack(alignment: .top, spacing: 16) {
LoadableAvatarImage(url: invite.roomDetails.avatarURL, LoadableAvatarImage(url: invite.roomDetails.avatarURL,
@ -150,7 +148,7 @@ struct InvitesScreenCell: View {
private var badge: some View { private var badge: some View {
Circle() Circle()
.frame(width: badgeSize, height: badgeSize) .scaledFrame(size: 12)
.foregroundColor(.compound.iconAccentTertiary) .foregroundColor(.compound.iconAccentTertiary)
} }
} }

View File

@ -104,7 +104,6 @@ struct StaticLocationScreen: View {
context.viewState.isLocationPickerMode ? .horizontal : [.horizontal, .bottom] context.viewState.isLocationPickerMode ? .horizontal : [.horizontal, .bottom]
} }
@ScaledMetric private var shareMarkerSize: CGFloat = 28
private var selectLocationButton: some View { private var selectLocationButton: some View {
Button { Button {
context.send(viewAction: .selectLocation) context.send(viewAction: .selectLocation)
@ -113,7 +112,7 @@ struct StaticLocationScreen: View {
Image(asset: Asset.Images.locationMarker) Image(asset: Asset.Images.locationMarker)
.resizable() .resizable()
.aspectRatio(contentMode: .fit) .aspectRatio(contentMode: .fit)
.frame(width: shareMarkerSize, height: shareMarkerSize) .scaledFrame(size: 28)
Text(context.viewState.isSharingUserLocation ? L10n.screenShareMyLocationAction : L10n.screenShareThisLocationAction) Text(context.viewState.isSharingUserLocation ? L10n.screenShareMyLocationAction : L10n.screenShareThisLocationAction)
} }
} }

View File

@ -17,8 +17,6 @@
import SwiftUI import SwiftUI
struct RoomMembersListScreenMemberCell: View { struct RoomMembersListScreenMemberCell: View {
@ScaledMetric private var avatarSize = AvatarSize.user(on: .roomDetails).value
let member: RoomMemberDetails let member: RoomMemberDetails
let context: RoomMembersListScreenViewModel.Context let context: RoomMembersListScreenViewModel.Context

View File

@ -25,8 +25,6 @@ struct TimelineItemBubbledStylerView<Content: View>: View {
let timelineItem: EventBasedTimelineItemProtocol let timelineItem: EventBasedTimelineItemProtocol
@ViewBuilder let content: () -> Content @ViewBuilder let content: () -> Content
@ScaledMetric private var senderNameVerticalPadding = 3
@State private var showItemActionMenu = false @State private var showItemActionMenu = false
@ -85,7 +83,7 @@ struct TimelineItemBubbledStylerView<Content: View>: View {
.font(.compound.bodySMSemibold) .font(.compound.bodySMSemibold)
.foregroundColor(.compound.avatarColor(for: timelineItem.sender.id).foreground) .foregroundColor(.compound.avatarColor(for: timelineItem.sender.id).foreground)
.lineLimit(1) .lineLimit(1)
.padding(.vertical, senderNameVerticalPadding) .scaledPadding(.vertical, 3)
} }
// sender info are read inside the `TimelineAccessibilityModifier` // sender info are read inside the `TimelineAccessibilityModifier`
.accessibilityHidden(true) .accessibilityHidden(true)

View File

@ -191,13 +191,11 @@ struct TimelineReactionButton: View {
} }
struct TimelineReactionAddMoreButtonLabel: View { struct TimelineReactionAddMoreButtonLabel: View {
@ScaledMetric private var addMoreButtonIconSize = 16
var body: some View { var body: some View {
TimelineReactionButtonLabel { TimelineReactionButtonLabel {
Image(asset: Asset.Images.addReaction) Image(asset: Asset.Images.addReaction)
.resizable() .resizable()
.frame(width: addMoreButtonIconSize, height: addMoreButtonIconSize) .scaledFrame(size: 16)
.padding(.vertical, 8) .padding(.vertical, 8)
.padding(.horizontal, 12) .padding(.horizontal, 12)
.foregroundColor(.compound.iconSecondary) .foregroundColor(.compound.iconSecondary)

View File

@ -20,8 +20,6 @@ struct PollRoomTimelineView: View {
let timelineItem: PollRoomTimelineItem let timelineItem: PollRoomTimelineItem
@Environment(\.timelineStyle) var timelineStyle @Environment(\.timelineStyle) var timelineStyle
@EnvironmentObject private var context: RoomScreenViewModel.Context @EnvironmentObject private var context: RoomScreenViewModel.Context
@ScaledMetric private var summaryPadding = 32
@ScaledMetric private var iconSize = 22
private let feedbackGenerator = UIImpactFeedbackGenerator(style: .heavy) private let feedbackGenerator = UIImpactFeedbackGenerator(style: .heavy)
@ -53,7 +51,7 @@ struct PollRoomTimelineView: View {
Image(asset.name) Image(asset.name)
.resizable() .resizable()
.frame(width: iconSize, height: iconSize) .scaledFrame(size: 22)
.accessibilityHidden(true) .accessibilityHidden(true)
Text(poll.question) Text(poll.question)
@ -83,7 +81,7 @@ struct PollRoomTimelineView: View {
if let summaryText = poll.summaryText { if let summaryText = poll.summaryText {
Text(summaryText) Text(summaryText)
.font(.compound.bodySM) .font(.compound.bodySM)
.padding(.leading, showVotes ? 0 : summaryPadding) .scaledPadding(.leading, showVotes ? 0 : 32)
.foregroundColor(.compound.textSecondary) .foregroundColor(.compound.textSecondary)
.frame(maxWidth: .infinity, alignment: showVotes ? .trailing : .leading) .frame(maxWidth: .infinity, alignment: showVotes ? .trailing : .leading)
} }

View File

@ -136,7 +136,6 @@ extension RoomTimelineItemProtocol {
public struct TimelineItemMenu: View { public struct TimelineItemMenu: View {
@EnvironmentObject private var context: RoomScreenViewModel.Context @EnvironmentObject private var context: RoomScreenViewModel.Context
@Environment(\.dismiss) private var dismiss @Environment(\.dismiss) private var dismiss
@ScaledMetric private var addMoreButtonIconSize = 24
let item: EventBasedTimelineItemProtocol let item: EventBasedTimelineItemProtocol
let actions: TimelineItemMenuActions let actions: TimelineItemMenuActions
@ -228,7 +227,7 @@ public struct TimelineItemMenu: View {
} label: { } label: {
Image(asset: Asset.Images.addReaction) Image(asset: Asset.Images.addReaction)
.resizable() .resizable()
.frame(width: addMoreButtonIconSize, height: addMoreButtonIconSize) .scaledFrame(size: 24)
.frame(maxHeight: .infinity, alignment: .center) .frame(maxHeight: .infinity, alignment: .center)
.foregroundColor(.compound.iconSecondary) .foregroundColor(.compound.iconSecondary)
.padding(10) .padding(10)

View File

@ -19,8 +19,7 @@ import SwiftUI
struct TimelineSenderAvatarView: View { struct TimelineSenderAvatarView: View {
@EnvironmentObject private var context: RoomScreenViewModel.Context @EnvironmentObject private var context: RoomScreenViewModel.Context
@ScaledMetric private var avatarSize = AvatarSize.user(on: .timeline).value
let timelineItem: EventBasedTimelineItemProtocol let timelineItem: EventBasedTimelineItemProtocol
var body: some View { var body: some View {

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.