Compare commits

..

1 Commits

Author SHA1 Message Date
tyiu e456ac864d Add Apple translation popovers for notes for iOS 17.4+ and macOS 14.4+
Changelog-Added: Add Apple translation popovers for notes for iOS 17.4+ and macOS 14.4+
2024-09-22 00:38:15 -04:00
132 changed files with 2111 additions and 10520 deletions
-36
View File
@@ -1,36 +0,0 @@
## Summary
_[Please provide a summary of the changes in this PR.]_
## Checklist
- [ ] I have read (or I am familiar with) the [Contribution Guidelines](../docs/CONTRIBUTING.md)
- [ ] I have tested the changes in this PR
- [ ] My PR is either small, or I have split it into smaller logical commits that are easier to review
- [ ] I have added the signoff line to all my commits. See [Signing off your work](../docs/CONTRIBUTING.md#sign-your-work---the-developers-certificate-of-origin)
- [ ] I have added appropriate changelog entries for the changes in this PR. See [Adding changelog entries](../docs/CONTRIBUTING.md#add-changelog-changed-changelog-fixed-etc)
- [ ] I do not need to add a changelog entry. Reason: _[Please provide a reason]_
- [ ] I have added appropriate `Closes:` or `Fixes:` tags in the commit messages wherever applicable, or made sure those are not needed. See [Submitting patches](https://github.com/damus-io/damus/blob/master/docs/CONTRIBUTING.md#submitting-patches)
## Test report
_Please provide a test report for the changes in this PR. You can use the template below, but feel free to modify it as needed._
**Device:** _[Please specify the device you used for testing]_
**iOS:** _[Please specify the iOS version you used for testing]_
**Damus:** _[Please specify the Damus version or commit hash you used for testing]_
**Setup:** _[Please provide a brief description of the setup you used for testing, if applicable]_
**Steps:** _[Please provide a list of steps you took to test the changes in this PR]_
**Results:**
- [ ] PASS
- [ ] Partial PASS
- Details: _[Please provide details of the partial pass]_
## Other notes
_[Please provide any other information that you think is relevant to this PR.]_
-104
View File
@@ -1,107 +1,3 @@
## [v1.11(10)](https://github.com/damus-io/damus/releases/tag/v1.11-10) - 2024-11-18
### Added
- Add Damus Share Feature (Swift)
- Added new easy to use video controls for full screen video (Daniel DAquino)
- Add Edit, Share, and Tap-gesture in Profile pic image viewer (Swift Coder)
- Disappearing header, tabbar, and post button on scroll (ericholguin)
- Add Apple translation popovers for notes for iOS 17.4+ and macOS 14.4+ (Terry Yiu)
- Added NDB search functionality to the universe view (ericholguin)
- Added mute button to ProfileActionSheet (chungwwei)
- Added mute action to selected text menu (ericholguin)
- Added support for pasting images from the clipboard to the post composer (Swift Coder)
### Changed
- Improved image carousel image fill behavior (Daniel DAquino)
- Improved video syncing and bandwidth usage when switching between timeline video and full screen mode (Daniel DAquino)
- Swipe to dismiss on full screen carousel now shows an opacity effect for improved UX (Daniel DAquino)
- Removed event contents from full screen media carousel for cleaner view (Daniel DAquino)
- Add share button for images on full screen image carousel view (Swift)
- Changed boldness of font in side menu labels. (ericholguin)
- Changed search notes button with searched keyword (ericholguin)
- Changed opacity of tabbar and post button (ericholguin)
- Allow multiple images to be uploaded at the same time (swiftcoder) (William Casarin)
- Changed side menu design (ericholguin)
- Truncate fulltext search results (William Casarin)
- Expanded profile search results to 128 (William Casarin)
- Expand nostrdb text search results to 128 items (William Casarin)
- Use LazyVStack in text search results (William Casarin)
### Fixed
- Fixed missing tab bar on navigation (Swift Coder)
- Fixed some issues where QR code would not work, and improved UX (Daniel DAquino)
- Fixed iOS 18 gesture issues that would take user to the thread view when clicking on a video or unmuting it (Daniel DAquino)
- Fixed several issues that would cause video to automatically play or pause incorrectly (Daniel DAquino)
- Fixed issue where full screen video would disappear when going to landscape mode (Daniel DAquino)
- Fixed portrait video size on full screen carousel (Daniel DAquino)
- Fix avatar image on qrcode view (Swift Coder)
- Fix banner image upload (Swift Coder)
- Fix dismiss button visibility (Swift Coder)
- Fix quote repost counting (William Casarin)
- Fixed overlapping text in Universe View (ericholguin)
- Fixed localization issues and exported strings (Terry Yiu)
- Fix sensitive long-press gesture on event chat bubble in iOS 18 (Daniel DAquino)
- Fixed bottom padding for tabbar (ericholguin)
- Fixed localization build failures (Terry Yiu)
- Fixed back nav button placement in profile edit view (ericholguin)
- Friend profiles will now more likely show up in profile search (William Casarin)
- Fix broken QR code scanner and fix landscape mode (Terry Yiu)
[1.11(10)](https://github.com/damus-io/damus/releases/tag/v1.11-10): https://github.com/damus-io/damus/releases/tag/v1.11-10
## [1.10.1] - 2024-09-22
### Added
- Push notification support (Daniel DAquino)
- Added profile edit safe guards (Eric Holguin)
- Tor relay icon (ericholguin)
- Add highlighter for web pages (Daniel DAquino)
- Add support for adding comments when creating a highlight (Daniel DAquino)
- Add support for rendering highlights with comments (Daniel DAquino)
- Ability to create highlights (ericholguin)
- Highlights (NIP-84) (ericholguin)
- Revamp emoji picker to be less error-prone and add search, frequently used, and multiple skin tone support capabilities (Terry Yiu)
### Changed
- Improve notification view filtering UX (Daniel DAquino)
- Improve visibility of friends filter button (Daniel DAquino)
- Changed the default banner from ostriches to damoose (Eric Holguin)
- Changed image and banner url text fields to new sheet view (Eric Holguin)
- Onboarding design (ericholguin)
### Fixed
- Fix items that became unclickable on iOS 18 (Daniel DAquino)
- Fix many reconnection issues (William Casarin)
- Fixed issue where theme would be changed to black and can't be switched back on iOS 18 (cr0bar)
- Fixed some scenarios where the contact list would never be saved locally and cause issues when switching relays. (Daniel DAquino)
- Fix albyhub zaps not appearing (William Casarin)
- Fix inadvertent escape from mention suggestion menu when typing a space character (Daniel DAquino)
- Fix profile view toolbar alignment bug in iOS 18 (Terry Yiu)
- Create Account model now uses correct metadata (ericholguin)
- Restore localization for custom tabs (William Casarin)
- Fix iOS 18 reflection runtime error for custom picker (William Casarin)
[1.10.1]: https://github.com/damus-io/damus/releases/tag/v1.10.1
## [1.9.1 (4)] - 2024-08-13
### Fixed
- Fix crash when viewing notes with invalid image dimension metadata (Daniel DAquino)
[1.9.1 (4)]: https://github.com/damus-io/damus/releases/tag/v1.9.1-4
## [1.9 (14)] - 2024-07-14 ## [1.9 (14)] - 2024-07-14
### Added ### Added
File diff suppressed because it is too large Load Diff
@@ -1,14 +1,6 @@
{ {
"originHash" : "534c8e58993919d5ead25ceb4788c8e492c86bc2cf5833dc651ae60a0f30169c", "originHash" : "babaf4d5748afecf49bbb702530d8e9576460692f478b0a50ee43195dd4440e2",
"pins" : [ "pins" : [
{
"identity" : "codescanner",
"kind" : "remoteSourceControl",
"location" : "https://github.com/twostraws/CodeScanner.git",
"state" : {
"revision" : "9fa582f4b36c69c2a55bff5fb3377eb170ae273c"
}
},
{ {
"identity" : "emojikit", "identity" : "emojikit",
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
@@ -100,9 +92,10 @@
{ {
"identity" : "swipeactions", "identity" : "swipeactions",
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/damus-io/SwipeActions.git", "location" : "https://github.com/aheze/SwipeActions",
"state" : { "state" : {
"revision" : "33d99756c3112e1a07c1732e3cddc5ad5bd0c5f4" "revision" : "41e6f6dce02d8cfa164f8c5461a41340850ca3ab",
"version" : "1.1.0"
} }
} }
], ],
@@ -40,7 +40,7 @@
</BuildableReference> </BuildableReference>
</TestableReference> </TestableReference>
<TestableReference <TestableReference
skipped = "NO"> skipped = "YES">
<BuildableReference <BuildableReference
BuildableIdentifier = "primary" BuildableIdentifier = "primary"
BlueprintIdentifier = "4CE6DEFC27F7A08200C66700" BlueprintIdentifier = "4CE6DEFC27F7A08200C66700"
+1
View File
@@ -46,6 +46,7 @@ struct CustomPicker<SelectionValue: Hashable>: View {
.accentColor(tag == selection ? textColor() : .gray) .accentColor(tag == selection ? textColor() : .gray)
} }
} }
.background(Color(UIColor.systemBackground))
} }
func textColor() -> Color { func textColor() -> Color {
@@ -20,7 +20,6 @@ struct DamusBackground: View {
.resizable() .resizable()
.frame(maxWidth: .infinity, maxHeight: maxHeight, alignment: .center) .frame(maxWidth: .infinity, maxHeight: maxHeight, alignment: .center)
.ignoresSafeArea() .ignoresSafeArea()
.accessibilityHidden(true)
} }
} }
+110 -196
View File
@@ -7,7 +7,6 @@
import SwiftUI import SwiftUI
import Kingfisher import Kingfisher
import Combine
// TODO: all this ShareSheet complexity can be replaced with ShareLink once we update to iOS 16 // TODO: all this ShareSheet complexity can be replaced with ShareLink once we update to iOS 16
struct ShareSheet: UIViewControllerRepresentable { struct ShareSheet: UIViewControllerRepresentable {
@@ -96,203 +95,64 @@ enum ImageShape {
} }
} }
/// The `CarouselModel` helps `ImageCarousel` with some state management logic, keeping track of media sizes, and the ideal display size
///
/// This model is necessary because the state management logic required to keep track of media sizes for each one of the carousel items,
/// and the ideal display size at each moment is not a trivial task.
///
/// The rules for the media fill are as follows:
/// 1. The media item should generally have a width that completely fills the width of its parent view
/// 2. The height of the carousel should be adjusted accordingly
/// 3. The only exception to rules 1 and 2 is when the total height would be 20% larger than the height of the device
/// 4. If none of the above can be computed (e.g. due to missing information), default to a reasonable height, where the media item will fit into.
///
/// ## Usage notes
///
/// The view is has the following state management responsibilities:
/// 1. Watching the size of the images (via the `.observe_image_size` modifier)
/// 2. Notifying this class of geometry reader changes, by setting `geo_size`
///
/// ## Implementation notes
///
/// This class is organized in a way to reduce stateful behavior and the transiency bugs it can cause.
///
/// This is accomplished through the following pattern:
/// 1. The `current_item_fill` is a published property so that any updates instantly re-render the view
/// 2. However, `current_item_fill` has a mathematical dependency on other members of this class
/// 3. Therefore, the members on which the fill property depends on all have `didSet` observers that will cause the `current_item_fill` to be recalculated and published.
///
/// This pattern helps ensure that the state is always consistent and that the view is always up-to-date.
///
/// This class is marked as `@MainActor` since most of its properties are published and should be accessed from the main thread to avoid inconsistent SwiftUI state during renders
@MainActor
class CarouselModel: ObservableObject { class CarouselModel: ObservableObject {
// MARK: Immutable object attributes var current_url: URL?
// These are some attributes that are not expected to change throughout the lifecycle of this object var fillHeight: CGFloat
// These should not be modified after initialization to avoid state inconsistency var maxHeight: CGFloat
var firstImageHeight: CGFloat?
/// The state of the app
let damus_state: DamusState
/// All urls in the carousel
let urls: [MediaUrl]
/// The default fill height for the carousel, if we cannot calculate a more appropriate height
/// **Usage note:** Default to this when `current_item_fill` is nil
let default_fill_height: CGFloat
/// The maximum height for any carousel item
let max_height: CGFloat
// MARK: Miscellaneous
/// Holds items that allows us to cancel video size observers during de-initialization
private var all_cancellables: [AnyCancellable] = []
// MARK: State management properties
/// Properties relevant to state management.
/// These should be made into computed/functional properties when possible to avoid stateful behavior
/// When that is not possible (e.g. when dealing with an observed published property), establish its mathematical dependencies,
/// and use `didSet` observers to ensure that the state is always re-computed when necessary.
/// Stores information about the size of each media item in `urls`. @Published var open_sheet: Bool
/// **Usage note:** The view is responsible for setting the size of image urls @Published var selectedIndex: Int
var media_size_information: [URL: CGSize] { @Published var video_size: CGSize?
didSet { @Published var image_fill: ImageFill?
guard let current_url else { return }
// Upon updating information, update the carousel fill size if the size for the current url has changed
if oldValue[current_url] != media_size_information[current_url] {
self.refresh_current_item_fill()
}
}
}
/// Stores information about the geometry reader
/// **Usage note:** The view is responsible for setting this value
var geo_size: CGSize? {
didSet { self.refresh_current_item_fill() }
}
/// The index of the currently selected item
/// **Usage note:** The view is responsible for setting this value
@Published var selectedIndex: Int {
didSet { self.refresh_current_item_fill() }
}
/// The current fill for the media item.
/// **Usage note:** This property is read-only and should not be set directly. Update `selectedIndex` to update the current item being viewed.
var current_url: URL? {
return urls[safe: selectedIndex]?.url
}
/// Holds the ideal fill dimensions for the current item.
/// **Usage note:** This property is automatically updated when other properties are set, and should not be set directly
/// **Implementation note:** This property is mathematically dependent on geo_size, media_size_information, and `selectedIndex`,
/// and is automatically updated upon changes to these properties.
@Published private(set) var current_item_fill: ImageFill?
// MARK: Initialization and de-initialization
/// Initializes the `CarouselModel` with the given `DamusState` and `MediaUrl` array init(image_fill: ImageFill?) {
init(damus_state: DamusState, urls: [MediaUrl]) { self.current_url = nil
// Immutable object attributes self.fillHeight = 350
self.damus_state = damus_state self.maxHeight = UIScreen.main.bounds.height * 1.2 // 1.2
self.urls = urls self.firstImageHeight = nil
self.default_fill_height = 350 self.open_sheet = false
self.max_height = UIScreen.main.bounds.height * 1.2 // 1.2
// State management properties
self.selectedIndex = 0 self.selectedIndex = 0
self.current_item_fill = nil self.video_size = nil
self.geo_size = nil self.image_fill = image_fill
self.media_size_information = [:]
// Setup the rest of the state management logic
self.observe_video_sizes()
Task {
self.refresh_current_item_fill()
}
}
/// This private function observes the video sizes for all videos
private func observe_video_sizes() {
for media_url in urls {
switch media_url {
case .video(let url):
let video_player = damus_state.video.get_player(for: url)
if let video_size = video_player.video_size {
self.media_size_information[url] = video_size // Set the initial size if available
}
let observer_cancellable = video_player.$video_size.sink(receiveValue: { new_size in
self.media_size_information[url] = new_size // Update the size when it changes
})
all_cancellables.append(observer_cancellable) // Store the cancellable to cancel it later
case .image(_):
break; // Observing an image size needs to be done on the view directly, through the `.observe_image_size` modifier
}
}
}
deinit {
for cancellable_item in all_cancellables {
cancellable_item.cancel()
}
}
// MARK: State management and logic
/// This function refreshes the current item fill based on the current state of the model
/// **Usage note:** This is private, do not call this directly from outside the class.
/// **Implementation note:** This should be called using `didSet` observers on properties that affect the fill
private func refresh_current_item_fill() {
if let current_url,
let item_size = self.media_size_information[current_url],
let geo_size {
self.current_item_fill = ImageFill.calculate_image_fill(
geo_size: geo_size,
img_size: item_size,
maxHeight: self.max_height,
fillHeight: self.default_fill_height
)
}
else {
self.current_item_fill = nil // Not enough information to compute the proper fill. Default to nil
}
} }
} }
// MARK: - Image Carousel // MARK: - Image Carousel
/// A carousel that displays images and videos
///
/// ## Implementation notes
///
/// - State management logic is mostly handled by `CarouselModel`, as it is complex, and becomes difficult to manage in a view
///
@MainActor @MainActor
struct ImageCarousel<Content: View>: View { struct ImageCarousel<Content: View>: View {
/// The event id of the note that this carousel is displaying var urls: [MediaUrl]
let evid: NoteId let evid: NoteId
/// The model that holds information and state of this carousel
/// This is observed to update the view when the model changes let state: DamusState
@ObservedObject var model: CarouselModel @ObservedObject var model: CarouselModel
let content: ((_ dismiss: @escaping (() -> Void)) -> Content)? let content: ((_ dismiss: @escaping (() -> Void)) -> Content)?
init(state: DamusState, evid: NoteId, urls: [MediaUrl]) { init(state: DamusState, evid: NoteId, urls: [MediaUrl]) {
self.urls = urls
self.evid = evid self.evid = evid
self._model = ObservedObject(initialValue: CarouselModel(damus_state: state, urls: urls)) self.state = state
let media_model = state.events.get_cache_data(evid).media_metadata_model
self._model = ObservedObject(initialValue: CarouselModel(image_fill: media_model.fill))
self.content = nil self.content = nil
} }
init(state: DamusState, evid: NoteId, urls: [MediaUrl], @ViewBuilder content: @escaping (_ dismiss: @escaping (() -> Void)) -> Content) { init(state: DamusState, evid: NoteId, urls: [MediaUrl], @ViewBuilder content: @escaping (_ dismiss: @escaping (() -> Void)) -> Content) {
self.urls = urls
self.evid = evid self.evid = evid
self._model = ObservedObject(initialValue: CarouselModel(damus_state: state, urls: urls)) self.state = state
let media_model = state.events.get_cache_data(evid).media_metadata_model
self._model = ObservedObject(initialValue: CarouselModel(image_fill: media_model.fill))
self.content = content self.content = content
} }
var filling: Bool { var filling: Bool {
model.current_item_fill?.filling == true model.image_fill?.filling == true
} }
var height: CGFloat { var height: CGFloat {
// Use the calculated fill height if available, otherwise use the default fill height model.firstImageHeight ?? model.image_fill?.height ?? model.fillHeight
model.current_item_fill?.height ?? model.default_fill_height
} }
func Placeholder(url: URL, geo_size: CGSize, num_urls: Int) -> some View { func Placeholder(url: URL, geo_size: CGSize, num_urls: Int) -> some View {
@@ -300,7 +160,7 @@ struct ImageCarousel<Content: View>: View {
if num_urls > 1 { if num_urls > 1 {
// jb55: quick hack since carousel with multiple images looks horrible with blurhash background // jb55: quick hack since carousel with multiple images looks horrible with blurhash background
Color.clear Color.clear
} else if let meta = model.damus_state.events.lookup_img_metadata(url: url), } else if let meta = state.events.lookup_img_metadata(url: url),
case .processed(let blurhash) = meta.state { case .processed(let blurhash) = meta.state {
Image(uiImage: blurhash) Image(uiImage: blurhash)
.resizable() .resizable()
@@ -309,6 +169,12 @@ struct ImageCarousel<Content: View>: View {
Color.clear Color.clear
} }
} }
.onAppear {
if self.model.image_fill == nil, let size = state.video.size_for_url(url) {
let fill = ImageFill.calculate_image_fill(geo_size: geo_size, img_size: size, maxHeight: model.maxHeight, fillHeight: model.fillHeight)
self.model.image_fill = fill
}
}
} }
func Media(geo: GeometryProxy, url: MediaUrl, index: Int) -> some View { func Media(geo: GeometryProxy, url: MediaUrl, index: Int) -> some View {
@@ -317,17 +183,24 @@ struct ImageCarousel<Content: View>: View {
case .image(let url): case .image(let url):
Img(geo: geo, url: url, index: index) Img(geo: geo, url: url, index: index)
.onTapGesture { .onTapGesture {
present(full_screen_item: .full_screen_carousel(urls: model.urls, selectedIndex: $model.selectedIndex)) model.open_sheet = true
} }
case .video(let url): case .video(let url):
let video_model = model.damus_state.video.get_player(for: url) DamusVideoPlayer(url: url, video_size: $model.video_size, controller: state.video, style: .preview(on_tap: { model.open_sheet = true }))
DamusVideoPlayerView( .onChange(of: model.video_size) { size in
model: video_model, guard let size else { return }
coordinator: model.damus_state.video,
style: .preview(on_tap: { let fill = ImageFill.calculate_image_fill(geo_size: geo.size, img_size: size, maxHeight: model.maxHeight, fillHeight: model.fillHeight)
present(full_screen_item: .full_screen_carousel(urls: model.urls, selectedIndex: $model.selectedIndex))
}) print("video_size changed \(size)")
) if self.model.image_fill == nil {
print("video_size firstImageHeight \(fill.height)")
self.model.firstImageHeight = fill.height
state.events.get_cache_data(evid).media_metadata_model.fill = fill
}
self.model.image_fill = fill
}
} }
} }
} }
@@ -336,18 +209,31 @@ struct ImageCarousel<Content: View>: View {
KFAnimatedImage(url) KFAnimatedImage(url)
.callbackQueue(.dispatch(.global(qos:.background))) .callbackQueue(.dispatch(.global(qos:.background)))
.backgroundDecode(true) .backgroundDecode(true)
.imageContext(.note, disable_animation: model.damus_state.settings.disable_animation) .imageContext(.note, disable_animation: state.settings.disable_animation)
.image_fade(duration: 0.25) .image_fade(duration: 0.25)
.cancelOnDisappear(true) .cancelOnDisappear(true)
.configure { view in .configure { view in
view.framePreloadCount = 3 view.framePreloadCount = 3
} }
.observe_image_size(size_changed: { size in .imageFill(for: geo.size, max: model.maxHeight, fill: model.fillHeight) { fill in
// Observe the image size to update the model when the size changes, so we can calculate the fill state.events.get_cache_data(evid).media_metadata_model.fill = fill
model.media_size_information[url] = size // blur hash can be discarded when we have the url
}) // NOTE: this is the wrong place for this... we need to remove
// it when the image is loaded in memory. This may happen
// earlier than this (by the preloader, etc)
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) {
state.events.lookup_img_metadata(url: url)?.state = .not_needed
}
self.model.image_fill = fill
if index == 0 {
self.model.firstImageHeight = fill.height
//maxHeight = firstImageHeight ?? maxHeight
} else {
//maxHeight = firstImageHeight ?? fill.height
}
}
.background { .background {
Placeholder(url: url, geo_size: geo.size, num_urls: model.urls.count) Placeholder(url: url, geo_size: geo.size, num_urls: urls.count)
} }
.aspectRatio(contentMode: filling ? .fill : .fit) .aspectRatio(contentMode: filling ? .fill : .fit)
.kfClickable() .kfClickable()
@@ -362,19 +248,25 @@ struct ImageCarousel<Content: View>: View {
var Medias: some View { var Medias: some View {
TabView(selection: $model.selectedIndex) { TabView(selection: $model.selectedIndex) {
ForEach(model.urls.indices, id: \.self) { index in ForEach(urls.indices, id: \.self) { index in
GeometryReader { geo in GeometryReader { geo in
Media(geo: geo, url: model.urls[index], index: index) Media(geo: geo, url: urls[index], index: index)
.onChange(of: geo.size, perform: { new_size in
model.geo_size = new_size
})
.onAppear {
model.geo_size = geo.size
}
} }
} }
} }
.tabViewStyle(PageTabViewStyle(indexDisplayMode: .never)) .tabViewStyle(PageTabViewStyle(indexDisplayMode: .never))
.fullScreenCover(isPresented: $model.open_sheet) {
if let content {
FullScreenCarouselView<Content>(video_controller: state.video, urls: urls, settings: state.settings, selectedIndex: $model.selectedIndex) {
content({ // Dismiss closure
model.open_sheet = false
})
}
}
else {
FullScreenCarouselView<AnyView>(video_controller: state.video, urls: urls, settings: state.settings, selectedIndex: $model.selectedIndex)
}
}
.frame(height: height) .frame(height: height)
.onChange(of: model.selectedIndex) { value in .onChange(of: model.selectedIndex) { value in
model.selectedIndex = value model.selectedIndex = value
@@ -392,8 +284,8 @@ struct ImageCarousel<Content: View>: View {
} }
if model.urls.count > 1 { if urls.count > 1 {
PageControlView(currentPage: $model.selectedIndex, numberOfPages: model.urls.count) PageControlView(currentPage: $model.selectedIndex, numberOfPages: urls.count)
.frame(maxWidth: 0, maxHeight: 0) .frame(maxWidth: 0, maxHeight: 0)
.padding(.top, 5) .padding(.top, 5)
} }
@@ -401,6 +293,27 @@ struct ImageCarousel<Content: View>: View {
} }
} }
// MARK: - Image Modifier
extension KFOptionSetter {
/// Sets a block to get image size
///
/// - Parameter block: The block which is used to read the image object.
/// - Returns: `Self` value after read size
public func imageFill(for size: CGSize, max: CGFloat, fill: CGFloat, block: @escaping (ImageFill) throws -> Void) -> Self {
let modifier = AnyImageModifier { image -> KFCrossPlatformImage in
let img_size = image.size
let geo_size = size
let fill = ImageFill.calculate_image_fill(geo_size: geo_size, img_size: img_size, maxHeight: max, fillHeight: fill)
DispatchQueue.main.async { [block, fill] in
try? block(fill)
}
return image
}
options.imageModifier = modifier
return self
}
}
public struct ImageFill { public struct ImageFill {
let filling: Bool? let filling: Bool?
@@ -437,3 +350,4 @@ struct ImageCarousel_Previews: PreviewProvider {
.environmentObject(OrientationTracker()) .environmentObject(OrientationTracker())
} }
} }
+33 -95
View File
@@ -57,47 +57,10 @@ enum Sheets: Identifiable {
} }
} }
/// An item to be presented full screen in a mechanism that is more robust for timeline views.
///
/// ## Implementation notes
///
/// This is part of the `present(full_screen_item: FullScreenItem)` interface that allows views in a timeline to show something full-screen without the lazy stack issues
/// Full screen cover modifiers are not suitable in those cases because device orientation changes or programmatic scroll commands will cause the view to be unloaded along with the cover,
/// causing the user to lose the full screen view randomly.
///
/// The `ContentView` is responsible for handling these objects
///
/// New items can be added as needed.
///
enum FullScreenItem: Identifiable, Equatable {
/// A full screen media carousel for images and videos.
case full_screen_carousel(urls: [MediaUrl], selectedIndex: Binding<Int>)
var id: String {
switch self {
case .full_screen_carousel(let urls, _): return "full_screen_carousel:\(urls.map(\.url))"
}
}
static func == (lhs: FullScreenItem, rhs: FullScreenItem) -> Bool {
return lhs.id == rhs.id
}
/// The view to display the item
func view(damus_state: DamusState) -> some View {
switch self {
case .full_screen_carousel(let urls, let selectedIndex):
return FullScreenCarouselView<AnyView>(video_coordinator: damus_state.video, urls: urls, settings: damus_state.settings, selectedIndex: selectedIndex)
}
}
}
func present_sheet(_ sheet: Sheets) { func present_sheet(_ sheet: Sheets) {
notify(.present_sheet(sheet)) notify(.present_sheet(sheet))
} }
var tabHeight: CGFloat = 0.0
struct ContentView: View { struct ContentView: View {
let keypair: Keypair let keypair: Keypair
let appDelegate: AppDelegate? let appDelegate: AppDelegate?
@@ -113,7 +76,6 @@ struct ContentView: View {
@Environment(\.scenePhase) var scenePhase @Environment(\.scenePhase) var scenePhase
@State var active_sheet: Sheets? = nil @State var active_sheet: Sheets? = nil
@State var active_full_screen_item: FullScreenItem? = nil
@State var damus_state: DamusState! @State var damus_state: DamusState!
@State var menu_subtitle: String? = nil @State var menu_subtitle: String? = nil
@SceneStorage("ContentView.selected_timeline") var selected_timeline: Timeline = .home { @SceneStorage("ContentView.selected_timeline") var selected_timeline: Timeline = .home {
@@ -127,7 +89,6 @@ struct ContentView: View {
@State var user_muted_confirm: Bool = false @State var user_muted_confirm: Bool = false
@State var confirm_overwrite_mutelist: Bool = false @State var confirm_overwrite_mutelist: Bool = false
@State private var isSideBarOpened = false @State private var isSideBarOpened = false
@State var headerOffset: CGFloat = 0.0
var home: HomeModel = HomeModel() var home: HomeModel = HomeModel()
@StateObject var navigationCoordinator: NavigationCoordinator = NavigationCoordinator() @StateObject var navigationCoordinator: NavigationCoordinator = NavigationCoordinator()
@AppStorage("has_seen_suggested_users") private var hasSeenOnboardingSuggestions = false @AppStorage("has_seen_suggested_users") private var hasSeenOnboardingSuggestions = false
@@ -170,7 +131,7 @@ struct ContentView: View {
} }
case .home: case .home:
PostingTimelineView(damus_state: damus_state!, home: home, isSideBarOpened: $isSideBarOpened, active_sheet: $active_sheet, headerOffset: $headerOffset) PostingTimelineView(damus_state: damus_state!, home: home, active_sheet: $active_sheet)
case .notifications: case .notifications:
NotificationsView(state: damus, notifications: home.notifications, subtitle: $menu_subtitle) NotificationsView(state: damus, notifications: home.notifications, subtitle: $menu_subtitle)
@@ -179,16 +140,25 @@ struct ContentView: View {
DirectMessagesView(damus_state: damus_state!, model: damus_state!.dms, settings: damus_state!.settings) DirectMessagesView(damus_state: damus_state!, model: damus_state!.dms, settings: damus_state!.settings)
} }
} }
.background(DamusColors.adaptableWhite)
.edgesIgnoringSafeArea(selected_timeline != .home ? [] : [.top, .bottom])
.navigationBarTitle(timeline_name(selected_timeline), displayMode: .inline) .navigationBarTitle(timeline_name(selected_timeline), displayMode: .inline)
.toolbar(selected_timeline != .home ? .visible : .hidden)
.toolbar { .toolbar {
ToolbarItem(placement: .principal) { ToolbarItem(placement: .principal) {
VStack { VStack {
timelineNavItem if selected_timeline == .home {
.opacity(isSideBarOpened ? 0 : 1) Image("damus-home")
.animation(isSideBarOpened ? .none : .default, value: isSideBarOpened) .resizable()
.frame(width:30,height:30)
.shadow(color: DamusColors.purple, radius: 2)
.opacity(isSideBarOpened ? 0 : 1)
.animation(isSideBarOpened ? .none : .default, value: isSideBarOpened)
.onTapGesture {
isSideBarOpened.toggle()
}
} else {
timelineNavItem
.opacity(isSideBarOpened ? 0 : 1)
.animation(isSideBarOpened ? .none : .default, value: isSideBarOpened)
}
} }
} }
} }
@@ -239,7 +209,14 @@ struct ContentView: View {
MainContent(damus: damus) MainContent(damus: damus)
.toolbar() { .toolbar() {
ToolbarItem(placement: .navigationBarLeading) { ToolbarItem(placement: .navigationBarLeading) {
TopbarSideMenuButton(damus_state: damus, isSideBarOpened: $isSideBarOpened) Button {
isSideBarOpened.toggle()
} label: {
ProfilePicView(pubkey: damus_state!.pubkey, size: 32, highlight: .none, profiles: damus_state!.profiles, disable_animation: damus_state!.settings.disable_animation)
.opacity(isSideBarOpened ? 0 : 1)
.animation(isSideBarOpened ? .none : .default, value: isSideBarOpened)
}
.disabled(isSideBarOpened)
} }
ToolbarItem(placement: .navigationBarTrailing) { ToolbarItem(placement: .navigationBarTrailing) {
@@ -260,11 +237,9 @@ struct ContentView: View {
} }
} }
} }
.background(DamusColors.adaptableWhite)
.edgesIgnoringSafeArea(selected_timeline != .home ? [] : [.top, .bottom])
.tabViewStyle(.page(indexDisplayMode: .never)) .tabViewStyle(.page(indexDisplayMode: .never))
.overlay( .overlay(
SideMenuView(damus_state: damus_state!, isSidebarVisible: $isSideBarOpened.animation(), selected: $selected_timeline) SideMenuView(damus_state: damus_state!, isSidebarVisible: $isSideBarOpened.animation())
) )
.navigationDestination(for: Route.self) { route in .navigationDestination(for: Route.self) { route in
route.view(navigationCoordinator: navigationCoordinator, damusState: damus_state!) route.view(navigationCoordinator: navigationCoordinator, damusState: damus_state!)
@@ -274,28 +249,13 @@ struct ContentView: View {
} }
} }
.navigationViewStyle(.stack) .navigationViewStyle(.stack)
.damus_full_screen_cover($active_full_screen_item, damus_state: damus, content: { item in
return item.view(damus_state: damus) if !hide_bar {
}) TabBar(nstatus: home.notification_status, selected: $selected_timeline, settings: damus.settings, action: switch_timeline)
.overlay(alignment: .bottom) { .padding([.bottom], 8)
if !hide_bar { .background(Color(uiColor: .systemBackground).ignoresSafeArea())
if !isSideBarOpened { } else {
TabBar(nstatus: home.notification_status, navIsAtRoot: navIsAtRoot(), selected: $selected_timeline, headerOffset: $headerOffset, settings: damus.settings, action: switch_timeline) Text("")
.padding([.bottom], 8)
.background(selected_timeline != .home || (selected_timeline == .home && !self.navIsAtRoot()) ? DamusColors.adaptableWhite : DamusColors.adaptableWhite.opacity(abs(1.25 - (abs(headerOffset/100.0)))))
.anchorPreference(key: HeaderBoundsKey.self, value: .bounds){$0}
.overlayPreferenceValue(HeaderBoundsKey.self) { value in
GeometryReader{ proxy in
if let anchor = value{
Color.clear
.onAppear {
tabHeight = proxy[anchor].height
}
}
}
}
}
}
} }
} }
} }
@@ -453,9 +413,6 @@ struct ContentView: View {
.onReceive(handle_notify(.present_sheet)) { sheet in .onReceive(handle_notify(.present_sheet)) { sheet in
self.active_sheet = sheet self.active_sheet = sheet
} }
.onReceive(handle_notify(.present_full_screen_item)) { item in
self.active_full_screen_item = item
}
.onReceive(handle_notify(.zapping)) { zap_ev in .onReceive(handle_notify(.zapping)) { zap_ev in
guard !zap_ev.is_custom else { guard !zap_ev.is_custom else {
return return
@@ -721,7 +678,7 @@ struct ContentView: View {
wallet: WalletModel(settings: settings), wallet: WalletModel(settings: settings),
nav: self.navigationCoordinator, nav: self.navigationCoordinator,
music: MusicController(onChange: music_changed), music: MusicController(onChange: music_changed),
video: DamusVideoCoordinator(), video: VideoController(),
ndb: ndb, ndb: ndb,
quote_reposts: .init(our_pubkey: pubkey), quote_reposts: .init(our_pubkey: pubkey),
emoji_provider: DefaultEmojiProvider(showAllVariations: true) emoji_provider: DefaultEmojiProvider(showAllVariations: true)
@@ -785,25 +742,6 @@ struct ContentView: View {
} }
} }
struct TopbarSideMenuButton: View {
let damus_state: DamusState
@Binding var isSideBarOpened: Bool
var body: some View {
Button {
isSideBarOpened.toggle()
} label: {
ProfilePicView(pubkey: damus_state.pubkey, size: 32, highlight: .none, profiles: damus_state.profiles, disable_animation: damus_state.settings.disable_animation)
.opacity(isSideBarOpened ? 0 : 1)
.animation(isSideBarOpened ? .none : .default, value: isSideBarOpened)
.accessibilityHidden(true) // Knowing there is a profile picture here leads to no actionable outcome to VoiceOver users, so it is best not to show it
}
.accessibilityIdentifier(AppAccessibilityIdentifiers.main_side_menu_button.rawValue)
.accessibilityLabel(NSLocalizedString("Side menu", comment: "Accessibility label for the side menu button at the topbar"))
.disabled(isSideBarOpened)
}
}
struct ContentView_Previews: PreviewProvider { struct ContentView_Previews: PreviewProvider {
static var previews: some View { static var previews: some View {
ContentView(keypair: Keypair(pubkey: test_pubkey, privkey: nil), appDelegate: nil) ContentView(keypair: Keypair(pubkey: test_pubkey, privkey: nil), appDelegate: nil)
+4 -4
View File
@@ -34,13 +34,13 @@ class DamusState: HeadlessDamusState {
let wallet: WalletModel let wallet: WalletModel
let nav: NavigationCoordinator let nav: NavigationCoordinator
let music: MusicController? let music: MusicController?
let video: DamusVideoCoordinator let video: VideoController
let ndb: Ndb let ndb: Ndb
var purple: DamusPurple var purple: DamusPurple
var push_notification_client: PushNotificationClient var push_notification_client: PushNotificationClient
let emoji_provider: EmojiProvider let emoji_provider: EmojiProvider
init(pool: RelayPool, keypair: Keypair, likes: EventCounter, boosts: EventCounter, contacts: Contacts, mutelist_manager: MutelistManager, profiles: Profiles, dms: DirectMessagesModel, previews: PreviewCache, zaps: Zaps, lnurls: LNUrls, settings: UserSettingsStore, relay_filters: RelayFilters, relay_model_cache: RelayModelCache, drafts: Drafts, events: EventCache, bookmarks: BookmarksManager, postbox: PostBox, bootstrap_relays: [RelayURL], replies: ReplyCounter, wallet: WalletModel, nav: NavigationCoordinator, music: MusicController?, video: DamusVideoCoordinator, ndb: Ndb, purple: DamusPurple? = nil, quote_reposts: EventCounter, emoji_provider: EmojiProvider) { init(pool: RelayPool, keypair: Keypair, likes: EventCounter, boosts: EventCounter, contacts: Contacts, mutelist_manager: MutelistManager, profiles: Profiles, dms: DirectMessagesModel, previews: PreviewCache, zaps: Zaps, lnurls: LNUrls, settings: UserSettingsStore, relay_filters: RelayFilters, relay_model_cache: RelayModelCache, drafts: Drafts, events: EventCache, bookmarks: BookmarksManager, postbox: PostBox, bootstrap_relays: [RelayURL], replies: ReplyCounter, wallet: WalletModel, nav: NavigationCoordinator, music: MusicController?, video: VideoController, ndb: Ndb, purple: DamusPurple? = nil, quote_reposts: EventCounter, emoji_provider: EmojiProvider) {
self.pool = pool self.pool = pool
self.keypair = keypair self.keypair = keypair
self.likes = likes self.likes = likes
@@ -141,7 +141,7 @@ class DamusState: HeadlessDamusState {
wallet: WalletModel(settings: settings), wallet: WalletModel(settings: settings),
nav: navigationCoordinator, nav: navigationCoordinator,
music: MusicController(onChange: { _ in }), music: MusicController(onChange: { _ in }),
video: DamusVideoCoordinator(), video: VideoController(),
ndb: ndb, ndb: ndb,
quote_reposts: .init(our_pubkey: pubkey), quote_reposts: .init(our_pubkey: pubkey),
emoji_provider: DefaultEmojiProvider(showAllVariations: true) emoji_provider: DefaultEmojiProvider(showAllVariations: true)
@@ -209,7 +209,7 @@ class DamusState: HeadlessDamusState {
wallet: WalletModel(settings: UserSettingsStore()), wallet: WalletModel(settings: UserSettingsStore()),
nav: NavigationCoordinator(), nav: NavigationCoordinator(),
music: nil, music: nil,
video: DamusVideoCoordinator(), video: VideoController(),
ndb: .empty, ndb: .empty,
quote_reposts: .init(our_pubkey: empty_pub), quote_reposts: .init(our_pubkey: empty_pub),
emoji_provider: DefaultEmojiProvider(showAllVariations: true) emoji_provider: DefaultEmojiProvider(showAllVariations: true)
-24
View File
@@ -213,27 +213,3 @@ enum HighlightSource: Hashable {
} }
} }
} }
struct ShareContent {
let title: String
let content: ContentType
enum ContentType {
case link(URL)
case media([PreUploadedMedia])
}
func getLinkURL() -> URL? {
if case let .link(url) = content {
return url
}
return nil
}
func getMediaArray() -> [PreUploadedMedia] {
if case let .media(mediaArray) = content {
return mediaArray
}
return []
}
}
-4
View File
@@ -59,14 +59,10 @@ enum TranslationService: String, CaseIterable, Identifiable, StringCodable {
} }
static var isAppleTranslationPopoverSupported: Bool { static var isAppleTranslationPopoverSupported: Bool {
#if targetEnvironment(macCatalyst)
return false
#else
if #available(iOS 17.4, macOS 14.4, *) { if #available(iOS 17.4, macOS 14.4, *) {
return true return true
} else { } else {
return false return false
} }
#endif
} }
} }
+1 -5
View File
@@ -12,15 +12,11 @@ struct SwipeToDismissModifier: ViewModifier {
var onDismiss: () -> Void var onDismiss: () -> Void
@State private var offset: CGSize = .zero @State private var offset: CGSize = .zero
@GestureState private var viewOffset: CGSize = .zero @GestureState private var viewOffset: CGSize = .zero
let threshold_offset: CGFloat = 100.0
let minimum_opacity: CGFloat = 0.1
func body(content: Content) -> some View { func body(content: Content) -> some View {
content content
.offset(y: viewOffset.height) .offset(y: viewOffset.height)
.animation(.interactiveSpring(), value: viewOffset) .animation(.interactiveSpring(), value: viewOffset)
.opacity(max(min(1.0 - (abs(offset.height) / threshold_offset), 1.0), minimum_opacity))
.simultaneousGesture( .simultaneousGesture(
DragGesture(minimumDistance: minDistance ?? 10) DragGesture(minimumDistance: minDistance ?? 10)
.updating($viewOffset, body: { value, gestureState, transaction in .updating($viewOffset, body: { value, gestureState, transaction in
@@ -32,7 +28,7 @@ struct SwipeToDismissModifier: ViewModifier {
} }
} }
.onEnded { _ in .onEnded { _ in
if abs(offset.height) > threshold_offset { if abs(offset.height) > 100 {
onDismiss() onDismiss()
} else { } else {
offset = .zero offset = .zero
@@ -1,42 +0,0 @@
//
// PresentFullScreenItemNotify.swift
// damus
//
// Created by Daniel DAquino on 2024-11-01.
//
struct PresentFullScreenItemNotify: Notify {
typealias Payload = FullScreenItem
var payload: Payload
}
extension NotifyHandler {
static var present_full_screen_item: NotifyHandler<PresentFullScreenItemNotify> {
.init()
}
}
extension Notifications {
static func present_full_screen_item(_ item: FullScreenItem) -> Notifications<PresentFullScreenItemNotify> {
.init(.init(payload: item))
}
}
/// Tell the app to present an item in full screen. Use this when presenting items coming from a timeline or any lazy stack.
///
/// ## Usage notes
///
/// Use this instead of `.damus_full_screen_cover` when the source view is on a lazy stack or timeline.
///
/// The reason is that when using a full screen modifier in those scenarios, the full screen view may abruptly disappear.
/// One example is when showing videos from the timeline in full screen, where changing the orientation of the device (landscape/portrait)
/// can cause the source view to be unloaded by the lazy stack, making your full screen overlay to simply disappear, causing a feeling of flakiness to the app
///
/// ## Implementation notes
///
/// The requests from this function will be received and handled at the top level app view (`ContentView`), which contains a `.damus_full_screen_cover`.
///
func present(full_screen_item: FullScreenItem) {
notify(.present_full_screen_item(full_screen_item))
}
-3
View File
@@ -31,7 +31,4 @@ class Constants {
static let DAMUS_WEBSITE_LOCAL_TEST_URL: URL = URL(string: "http://localhost:3000")! static let DAMUS_WEBSITE_LOCAL_TEST_URL: URL = URL(string: "http://localhost:3000")!
static let DAMUS_WEBSITE_STAGING_URL: URL = URL(string: "https://staging.damus.io")! static let DAMUS_WEBSITE_STAGING_URL: URL = URL(string: "https://staging.damus.io")!
static let DAMUS_WEBSITE_PRODUCTION_URL: URL = URL(string: "https://damus.io")! static let DAMUS_WEBSITE_PRODUCTION_URL: URL = URL(string: "https://damus.io")!
// MARK: General constants
static let GIF_IMAGE_TYPE: String = "com.compuserve.gif"
} }
@@ -58,22 +58,6 @@ extension KFOptionSetter {
return self return self
} }
/// This allows you to observe the size of the image, and get a callback when the size changes
/// This is useful for when you need to layout views based on the size of the image
/// - Parameter size_changed: A callback that will be called when the size of the image changes
/// - Returns: The same KFOptionSetter instance
func observe_image_size(size_changed: @escaping (CGSize) -> Void) -> Self {
let modifier = AnyImageModifier { image -> KFCrossPlatformImage in
let image_size = image.size
DispatchQueue.main.async { [size_changed, image_size] in
size_changed(image_size)
}
return image
}
options.imageModifier = modifier
return self
}
} }
let MAX_FILE_SIZE = 20_971_520 // 20MiB let MAX_FILE_SIZE = 20_971_520 // 20MiB
@@ -1,78 +0,0 @@
//
// OffsetExtension.swift
// damus
//
// Created by eric on 9/6/24.
//
import SwiftUI
enum SwipeDirection {
case up
case down
case none
}
extension View {
@ViewBuilder
func offsetY(completion: @escaping (CGFloat, CGFloat)->())->some View {
self
.modifier(OffsetHelper(onChange: completion))
}
func safeArea() -> UIEdgeInsets {
guard let scene = this_app.connectedScenes.first as? UIWindowScene else{return .zero}
guard let safeArea = scene.windows.first?.safeAreaInsets else{return .zero}
return safeArea
}
}
struct OffsetHelper: ViewModifier{
var onChange: (CGFloat,CGFloat)->()
@State var currentOffset: CGFloat = 0
@State var previousOffset: CGFloat = 0
func body(content: Content) -> some View {
content
.overlay {
GeometryReader{proxy in
let minY = proxy.frame(in: .named("scroll")).minY
Color.clear
.preference(key: OffsetKey.self, value: minY)
.onPreferenceChange(OffsetKey.self) { value in
previousOffset = currentOffset
currentOffset = value
onChange(previousOffset,currentOffset)
}
}
}
}
}
struct OffsetKey: PreferenceKey {
static var defaultValue: CGFloat = 0
static func reduce(value: inout CGFloat, nextValue: () -> CGFloat) {
value = nextValue()
}
}
struct HeaderBoundsKey: PreferenceKey{
static var defaultValue: Anchor<CGRect>?
static func reduce(value: inout Anchor<CGRect>?, nextValue: () -> Anchor<CGRect>?) {
value = nextValue()
}
}
func getSafeAreaTop()->CGFloat{
guard let scene = this_app.connectedScenes.first as? UIWindowScene else{return .zero}
guard let topSafeArea = scene.windows.first?.safeAreaInsets.top else{return .zero}
return topSafeArea
}
func getSafeAreaBottom()->CGFloat{
guard let scene = this_app.connectedScenes.first as? UIWindowScene else{return .zero}
guard let bottomSafeArea = scene.windows.first?.safeAreaInsets.bottom else{return .zero}
return bottomSafeArea
}
+7 -3
View File
@@ -7,12 +7,16 @@
import Foundation import Foundation
func bundleForLocale(locale: Locale) -> Bundle { func bundleForLocale(locale: Locale?) -> Bundle {
let path = Bundle.main.path(forResource: locale.identifier, ofType: "lproj") if locale == nil {
return Bundle.main
}
let path = Bundle.main.path(forResource: locale!.identifier, ofType: "lproj")
return path != nil ? (Bundle(path: path!) ?? Bundle.main) : Bundle.main return path != nil ? (Bundle(path: path!) ?? Bundle.main) : Bundle.main
} }
func localizedStringFormat(key: String, locale: Locale) -> String { func localizedStringFormat(key: String, locale: Locale?) -> String {
let bundle = bundleForLocale(locale: locale) let bundle = bundleForLocale(locale: locale)
let fallback = bundleForLocale(locale: Locale(identifier: "en-US")).localizedString(forKey: key, value: nil, table: nil) let fallback = bundleForLocale(locale: Locale(identifier: "en-US")).localizedString(forKey: key, value: nil, table: nil)
return bundle.localizedString(forKey: key, value: fallback, table: nil) return bundle.localizedString(forKey: key, value: fallback, table: nil)
-1
View File
@@ -17,7 +17,6 @@ enum LogCategory: String {
case push_notifications case push_notifications
case damus_purple case damus_purple
case image_uploading case image_uploading
case video_coordination
} }
/// Damus structured logger /// Damus structured logger
-5
View File
@@ -37,7 +37,6 @@ enum Route: Hashable {
case Reactions(reactions: EventsModel) case Reactions(reactions: EventsModel)
case Zaps(target: ZapTarget) case Zaps(target: ZapTarget)
case Search(search: SearchModel) case Search(search: SearchModel)
case NDBSearch(results: Binding<[NostrEvent]>)
case EULA case EULA
case Login case Login
case CreateAccount case CreateAccount
@@ -106,8 +105,6 @@ enum Route: Hashable {
ZapsView(state: damusState, target: target) ZapsView(state: damusState, target: target)
case .Search(let search): case .Search(let search):
SearchView(appstate: damusState, search: search) SearchView(appstate: damusState, search: search)
case .NDBSearch(let results):
NDBSearchView(damus_state: damusState, results: results)
case .EULA: case .EULA:
EULAView(nav: navigationCoordinator) EULAView(nav: navigationCoordinator)
case .Login: case .Login:
@@ -203,8 +200,6 @@ enum Route: Hashable {
case .Search(let search): case .Search(let search):
hasher.combine("search") hasher.combine("search")
hasher.combine(search.search) hasher.combine(search.search)
case .NDBSearch(let results):
hasher.combine("results")
case .EULA: case .EULA:
hasher.combine("eula") hasher.combine("eula")
case .Login: case .Login:
@@ -1,66 +0,0 @@
//
// AppAccessibilityIdentifiers.swift
// damus
//
// Created by Daniel DAquino on 2024-11-18.
//
import Foundation
/// A collection of app-wide identifier constants used to facilitate UI tests to find the element they are looking for.
///
/// ## Implementation notes
///
/// - This is not an exhaustive list. Add more identifiers as needed.
/// - Organize this by separating each category with `MARK` comment markers and a unique prefix, each category separated by 2 empty lines
enum AppAccessibilityIdentifiers: String {
// MARK: Login
// Prefix: `sign_in`
/// Sign in button at the very start of the app
case sign_in_option_button
/// A secure text entry field where the user can put their private key when logging in
case sign_in_nsec_key_entry_field
/// Button to sign in after entering private key
case sign_in_confirm_button
// MARK: Onboarding
// Prefix: `onboarding`
/// The skip button on the onboarding sheet
case onboarding_sheet_skip_button
// MARK: Post composer
// Prefix: `post_composer`
/// The cancel post button
case post_composer_cancel_button
// MARK: Main interface layout
// Prefix: `main`
/// Profile picture item on the top toolbar, used to open the side menu
case main_side_menu_button
// MARK: Side menu
// Prefix: `side_menu`
/// The profile option in the side menu
case side_menu_profile_button
// MARK: Items specific to the user's own profile
// Prefix: `own_profile`
/// The edit profile button
case own_profile_edit_button
/// The button to edit the banner image on the profile
case own_profile_banner_image_edit_button
/// The button to pick the new banner image from URL
case own_profile_banner_image_edit_from_url
}
+1 -16
View File
@@ -14,7 +14,6 @@ struct EditBannerImageView: View {
@ObservedObject var viewModel: ImageUploadingObserver @ObservedObject var viewModel: ImageUploadingObserver
let callback: (URL?) -> Void let callback: (URL?) -> Void
let defaultImage = UIImage(named: "damoose") ?? UIImage() let defaultImage = UIImage(named: "damoose") ?? UIImage()
let safeAreaInsets: EdgeInsets
@State var banner_image: URL? = nil @State var banner_image: URL? = nil
@@ -32,21 +31,7 @@ struct EditBannerImageView: View {
.onFailureImage(defaultImage) .onFailureImage(defaultImage)
.kfClickable() .kfClickable()
EditPictureControl(uploader: damus_state.settings.default_media_uploader, keypair: damus_state.keypair, pubkey: damus_state.pubkey, image_url: $banner_image, uploadObserver: viewModel, callback: callback) EditPictureControl(uploader: damus_state.settings.default_media_uploader, pubkey: damus_state.pubkey, image_url: $banner_image, uploadObserver: viewModel, callback: callback)
.padding(10)
.backwardsCompatibleSafeAreaPadding(self.safeAreaInsets)
.accessibilityLabel(NSLocalizedString("Edit banner image", comment: "Accessibility label for edit banner image button"))
.accessibilityIdentifier(AppAccessibilityIdentifiers.own_profile_banner_image_edit_button.rawValue)
}
}
}
extension View {
fileprivate func backwardsCompatibleSafeAreaPadding(_ insets: EdgeInsets) -> some View {
if #available(iOS 17.0, *) {
return self.safeAreaPadding(insets)
} else {
return self.padding(.top, insets.top)
} }
} }
} }
-1
View File
@@ -39,7 +39,6 @@ struct BookmarksView: View {
ScrollView { ScrollView {
InnerTimelineView(events: EventHolder(events: bookmarks, incoming: []), damus: state, filter: noneFilter) InnerTimelineView(events: EventHolder(events: bookmarks, incoming: []), damus: state, filter: noneFilter)
} }
.padding(.bottom, 10 + tabHeight + getSafeAreaBottom())
} }
} }
.onReceive(handle_notify(.switched_timeline)) { _ in .onReceive(handle_notify(.switched_timeline)) { _ in
@@ -29,18 +29,13 @@ struct GradientFollowButton: View {
.fontWeight(.medium) .fontWeight(.medium)
.padding([.top, .bottom], 10) .padding([.top, .bottom], 10)
.padding([.leading, .trailing], 12) .padding([.leading, .trailing], 12)
.background(follow_state == .unfollows ? PinkGradient : GrayGradient)
.cornerRadius(12)
.overlay( .overlay(
RoundedRectangle(cornerRadius: 12) RoundedRectangle(cornerRadius: 12)
.stroke(grayBorder, lineWidth: follow_state == .unfollows ? 0 : 1) .stroke(grayBorder, lineWidth: follow_state == .unfollows ? 0 : 1)
.frame(width: 100)
) )
.frame(width: 100)
.minimumScaleFactor(0.5)
.lineLimit(1)
} }
.background(follow_state == .unfollows ? PinkGradient : GrayGradient)
.cornerRadius(12)
.frame(width: 100)
.onReceive(handle_notify(.followed)) { ref in .onReceive(handle_notify(.followed)) { ref in
guard target.follow_ref == ref else { return } guard target.follow_ref == ref else { return }
self.follow_state = .follows self.follow_state = .follows
+19 -9
View File
@@ -27,6 +27,8 @@ struct ChatEventView: View {
// MARK: long-press reaction control objects // MARK: long-press reaction control objects
/// Whether the user is actively pressing the view /// Whether the user is actively pressing the view
@State var is_pressing = false @State var is_pressing = false
/// The dispatched work item scheduled by a timer to bounce the event bubble and show the emoji selector
@State var long_press_bounce_work_item: DispatchWorkItem?
@State var popover_state: PopoverState = .closed { @State var popover_state: PopoverState = .closed {
didSet { didSet {
let generator = UIImpactFeedbackGenerator(style: popover_state.some_sheet_open() ? .heavy : .light) let generator = UIImpactFeedbackGenerator(style: popover_state.some_sheet_open() ? .heavy : .light)
@@ -37,7 +39,6 @@ struct ChatEventView: View {
@State private var isOnTopHalfOfScreen: Bool = false @State private var isOnTopHalfOfScreen: Bool = false
@ObservedObject var bar: ActionBarModel @ObservedObject var bar: ActionBarModel
@Environment(\.swipeViewGroupSelection) var swipeViewGroupSelection
enum PopoverState: String { enum PopoverState: String {
case closed case closed
@@ -205,18 +206,28 @@ struct ChatEventView: View {
.scaleEffect(self.popover_state.some_sheet_open() ? 1.08 : is_pressing ? 1.02 : 1) .scaleEffect(self.popover_state.some_sheet_open() ? 1.08 : is_pressing ? 1.02 : 1)
.shadow(color: (is_pressing || self.popover_state.some_sheet_open()) ? .black.opacity(0.1) : .black.opacity(0.3), radius: (is_pressing || self.popover_state.some_sheet_open()) ? 8 : 0, y: (is_pressing || self.popover_state.some_sheet_open()) ? 15 : 0) .shadow(color: (is_pressing || self.popover_state.some_sheet_open()) ? .black.opacity(0.1) : .black.opacity(0.3), radius: (is_pressing || self.popover_state.some_sheet_open()) ? 8 : 0, y: (is_pressing || self.popover_state.some_sheet_open()) ? 15 : 0)
.onLongPressGesture(minimumDuration: 0.5, maximumDistance: 10, perform: { .onLongPressGesture(minimumDuration: 0.5, maximumDistance: 10, perform: {
withAnimation(.bouncy(duration: 0.2, extraBounce: 0.35)) { long_press_bounce_work_item?.cancel()
let should_show_zap_sheet = !damus_state.settings.nozaps && damus_state.settings.onlyzaps_mode
popover_state = should_show_zap_sheet ? .open_zap_sheet : .open_emoji_selector
}
}, onPressingChanged: { is_pressing in }, onPressingChanged: { is_pressing in
withAnimation(is_pressing ? .easeIn(duration: 0.5) : .easeOut(duration: 0.1)) { withAnimation(is_pressing ? .easeIn(duration: 0.5) : .easeOut(duration: 0.1)) {
self.is_pressing = is_pressing self.is_pressing = is_pressing
if popover_state != .closed {
return
}
if self.is_pressing {
let item = DispatchWorkItem {
// Ensure the action is performed only if the condition is still valid
if self.is_pressing {
withAnimation(.bouncy(duration: 0.2, extraBounce: 0.35)) {
let should_show_zap_sheet = !damus_state.settings.nozaps && damus_state.settings.onlyzaps_mode
popover_state = should_show_zap_sheet ? .open_zap_sheet : .open_emoji_selector
}
}
}
long_press_bounce_work_item = item
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5, execute: item)
}
} }
}) })
.onChange(of: swipeViewGroupSelection.wrappedValue) { newValue in
self.is_pressing = false
}
.background( .background(
GeometryReader { geometry in GeometryReader { geometry in
EmptyView() EmptyView()
@@ -299,7 +310,6 @@ struct ChatEventView: View {
.swipeSpacing(-20) .swipeSpacing(-20)
.swipeActionsStyle(.mask) .swipeActionsStyle(.mask)
.swipeMinimumDistance(20) .swipeMinimumDistance(20)
.swipeDragGesturePriority(.normal)
} }
} }
@@ -135,9 +135,6 @@ struct ChatroomThreadView: View {
} }
.padding(.top) .padding(.top)
EndBlock() EndBlock()
HStack {}
.frame(height: tabHeight + getSafeAreaBottom())
} }
.onReceive(handle_notify(.post), perform: { notify in .onReceive(handle_notify(.post), perform: { notify in
switch notify { switch notify {
+114
View File
@@ -0,0 +1,114 @@
//
// CodeScanner.swift
// https://github.com/twostraws/CodeScanner
//
// Created by Paul Hudson on 14/12/2021.
// Copyright © 2021 Paul Hudson. All rights reserved.
//
import AVFoundation
import SwiftUI
/// An enum describing the ways CodeScannerView can hit scanning problems.
public enum ScanError: Error {
/// The camera could not be accessed.
case badInput
/// The camera was not capable of scanning the requested codes.
case badOutput
/// Initialization failed.
case initError(_ error: Error)
}
/// The result from a successful scan: the string that was scanned, and also the type of data that was found.
/// The type is useful for times when you've asked to scan several different code types at the same time, because
/// it will report the exact code type that was found.
public struct ScanResult {
/// The contents of the code.
public let string: String
/// The type of code that was matched.
public let type: AVMetadataObject.ObjectType
}
/// The operating mode for CodeScannerView.
public enum ScanMode {
/// Scan exactly one code, then stop.
case once
/// Scan each code no more than once.
case oncePerCode
/// Keep scanning all codes until dismissed.
case continuous
}
/// A SwiftUI view that is able to scan barcodes, QR codes, and more, and send back what was found.
/// To use, set `codeTypes` to be an array of things to scan for, e.g. `[.qr]`, and set `completion` to
/// a closure that will be called when scanning has finished. This will be sent the string that was detected or a `ScanError`.
/// For testing inside the simulator, set the `simulatedData` property to some test data you want to send back.
public struct CodeScannerView: UIViewControllerRepresentable {
public let codeTypes: [AVMetadataObject.ObjectType]
public let scanMode: ScanMode
public let scanInterval: Double
public let showViewfinder: Bool
public var simulatedData = ""
public var shouldVibrateOnSuccess: Bool
public var isTorchOn: Bool
public var isGalleryPresented: Binding<Bool>
public var videoCaptureDevice: AVCaptureDevice?
public var completion: (Result<ScanResult, ScanError>) -> Void
public init(
codeTypes: [AVMetadataObject.ObjectType],
scanMode: ScanMode = .once,
scanInterval: Double = 2.0,
showViewfinder: Bool = false,
simulatedData: String = "",
shouldVibrateOnSuccess: Bool = true,
isTorchOn: Bool = false,
isGalleryPresented: Binding<Bool> = .constant(false),
videoCaptureDevice: AVCaptureDevice? = AVCaptureDevice.default(for: .video),
completion: @escaping (Result<ScanResult, ScanError>) -> Void
) {
self.codeTypes = codeTypes
self.scanMode = scanMode
self.showViewfinder = showViewfinder
self.scanInterval = scanInterval
self.simulatedData = simulatedData
self.shouldVibrateOnSuccess = shouldVibrateOnSuccess
self.isTorchOn = isTorchOn
self.isGalleryPresented = isGalleryPresented
self.videoCaptureDevice = videoCaptureDevice
self.completion = completion
}
public func makeCoordinator() -> ScannerCoordinator {
ScannerCoordinator(parent: self)
}
public func makeUIViewController(context: Context) -> ScannerViewController {
let viewController = ScannerViewController(showViewfinder: showViewfinder)
viewController.delegate = context.coordinator
return viewController
}
public func updateUIViewController(_ uiViewController: ScannerViewController, context: Context) {
uiViewController.updateViewController(
isTorchOn: isTorchOn,
isGalleryPresented: isGalleryPresented.wrappedValue
)
}
}
@available(macCatalyst 14.0, *)
struct CodeScannerView_Previews: PreviewProvider {
static var previews: some View {
CodeScannerView(codeTypes: [.qr]) { result in
// do nothing
}
}
}
@@ -0,0 +1,75 @@
//
// CodeScanner.swift
// https://github.com/twostraws/CodeScanner
//
// Created by Paul Hudson on 14/12/2021.
// Copyright © 2021 Paul Hudson. All rights reserved.
//
import AVFoundation
import SwiftUI
extension CodeScannerView {
@available(macCatalyst 14.0, *)
public class ScannerCoordinator: NSObject, AVCaptureMetadataOutputObjectsDelegate {
var parent: CodeScannerView
var codesFound = Set<String>()
var didFinishScanning = false
var lastTime = Date(timeIntervalSince1970: 0)
init(parent: CodeScannerView) {
self.parent = parent
}
public func reset() {
codesFound.removeAll()
didFinishScanning = false
lastTime = Date(timeIntervalSince1970: 0)
}
public func metadataOutput(_ output: AVCaptureMetadataOutput, didOutput metadataObjects: [AVMetadataObject], from connection: AVCaptureConnection) {
if let metadataObject = metadataObjects.first {
guard let readableObject = metadataObject as? AVMetadataMachineReadableCodeObject else { return }
guard let stringValue = readableObject.stringValue else { return }
guard didFinishScanning == false else { return }
let result = ScanResult(string: stringValue, type: readableObject.type)
switch parent.scanMode {
case .once:
found(result)
// make sure we only trigger scan once per use
didFinishScanning = true
case .oncePerCode:
if !codesFound.contains(stringValue) {
codesFound.insert(stringValue)
found(result)
}
case .continuous:
if isPastScanInterval() {
found(result)
}
}
}
}
func isPastScanInterval() -> Bool {
Date().timeIntervalSince(lastTime) >= parent.scanInterval
}
func found(_ result: ScanResult) {
lastTime = Date()
if parent.shouldVibrateOnSuccess {
AudioServicesPlaySystemSound(SystemSoundID(kSystemSoundID_Vibrate))
}
parent.completion(.success(result))
}
func didFail(reason: ScanError) {
parent.completion(.failure(reason))
}
}
}
@@ -0,0 +1,296 @@
//
// CodeScanner.swift
// https://github.com/twostraws/CodeScanner
//
// Created by Paul Hudson on 14/12/2021.
// Copyright © 2021 Paul Hudson. All rights reserved.
//
import AVFoundation
import UIKit
extension CodeScannerView {
@available(macCatalyst 14.0, *)
public class ScannerViewController: UIViewController, UIImagePickerControllerDelegate, UINavigationControllerDelegate {
var delegate: ScannerCoordinator?
private let showViewfinder: Bool
private var isGalleryShowing: Bool = false {
didSet {
// Update binding
if delegate?.parent.isGalleryPresented.wrappedValue != isGalleryShowing {
delegate?.parent.isGalleryPresented.wrappedValue = isGalleryShowing
}
}
}
public init(showViewfinder: Bool = false) {
self.showViewfinder = showViewfinder
super.init(nibName: nil, bundle: nil)
}
required init?(coder: NSCoder) {
self.showViewfinder = false
super.init(coder: coder)
}
func openGallery() {
isGalleryShowing = true
let imagePicker = UIImagePickerController()
imagePicker.delegate = self
present(imagePicker, animated: true, completion: nil)
}
@objc func openGalleryFromButton(_ sender: UIButton) {
openGallery()
}
public func imagePickerController(_ picker: UIImagePickerController, didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey: Any]) {
isGalleryShowing = false
if let qrcodeImg = info[.originalImage] as? UIImage {
let detector = CIDetector(ofType: CIDetectorTypeQRCode, context: nil, options: [CIDetectorAccuracy: CIDetectorAccuracyHigh])!
let ciImage = CIImage(image:qrcodeImg)!
var qrCodeLink = ""
let features = detector.features(in: ciImage)
for feature in features as! [CIQRCodeFeature] {
qrCodeLink += feature.messageString!
}
if qrCodeLink == "" {
delegate?.didFail(reason: .badOutput)
} else {
let result = ScanResult(string: qrCodeLink, type: .qr)
delegate?.found(result)
}
} else {
print("Something went wrong")
}
dismiss(animated: true, completion: nil)
}
public func imagePickerControllerDidCancel(_ picker: UIImagePickerController) {
isGalleryShowing = false
}
#if targetEnvironment(simulator)
override public func loadView() {
view = UIView()
view.isUserInteractionEnabled = true
let label = UILabel()
label.translatesAutoresizingMaskIntoConstraints = false
label.numberOfLines = 0
label.text = "You're running in the simulator, which means the camera isn't available. Tap anywhere to send back some simulated data."
label.textAlignment = .center
let button = UIButton()
button.translatesAutoresizingMaskIntoConstraints = false
button.setTitle("Select a custom image", for: .normal)
button.setTitleColor(UIColor.systemBlue, for: .normal)
button.setTitleColor(UIColor.gray, for: .highlighted)
button.addTarget(self, action: #selector(openGalleryFromButton), for: .touchUpInside)
let stackView = UIStackView()
stackView.translatesAutoresizingMaskIntoConstraints = false
stackView.axis = .vertical
stackView.spacing = 50
stackView.addArrangedSubview(label)
stackView.addArrangedSubview(button)
view.addSubview(stackView)
NSLayoutConstraint.activate([
button.heightAnchor.constraint(equalToConstant: 50),
stackView.leadingAnchor.constraint(equalTo: view.layoutMarginsGuide.leadingAnchor),
stackView.trailingAnchor.constraint(equalTo: view.layoutMarginsGuide.trailingAnchor),
stackView.centerYAnchor.constraint(equalTo: view.centerYAnchor)
])
}
override public func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
guard let simulatedData = delegate?.parent.simulatedData else {
print("Simulated Data Not Provided!")
return
}
// Send back their simulated data, as if it was one of the types they were scanning for
let result = ScanResult(string: simulatedData, type: delegate?.parent.codeTypes.first ?? .qr)
delegate?.found(result)
}
#else
var captureSession: AVCaptureSession!
var previewLayer: AVCaptureVideoPreviewLayer!
let fallbackVideoCaptureDevice = AVCaptureDevice.default(for: .video)
private lazy var viewFinder: UIImageView? = {
guard let image = UIImage(named: "viewfinder", in: .main, with: nil) else {
return nil
}
let imageView = UIImageView(image: image)
imageView.translatesAutoresizingMaskIntoConstraints = false
return imageView
}()
override public func viewDidLoad() {
super.viewDidLoad()
NotificationCenter.default.addObserver(self,
selector: #selector(updateOrientation),
name: Notification.Name("UIDeviceOrientationDidChangeNotification"),
object: nil)
view.backgroundColor = UIColor.black
captureSession = AVCaptureSession()
guard let videoCaptureDevice = delegate?.parent.videoCaptureDevice ?? fallbackVideoCaptureDevice else {
return
}
let videoInput: AVCaptureDeviceInput
do {
videoInput = try AVCaptureDeviceInput(device: videoCaptureDevice)
} catch {
delegate?.didFail(reason: .initError(error))
return
}
if (captureSession.canAddInput(videoInput)) {
captureSession.addInput(videoInput)
} else {
delegate?.didFail(reason: .badInput)
return
}
let metadataOutput = AVCaptureMetadataOutput()
if (captureSession.canAddOutput(metadataOutput)) {
captureSession.addOutput(metadataOutput)
metadataOutput.setMetadataObjectsDelegate(delegate, queue: DispatchQueue.main)
metadataOutput.metadataObjectTypes = delegate?.parent.codeTypes
} else {
delegate?.didFail(reason: .badOutput)
return
}
if previewLayer == nil {
previewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
}
previewLayer.frame = view.layer.bounds
previewLayer.videoGravity = .resizeAspectFill
view.layer.addSublayer(previewLayer)
addviewfinder()
delegate?.reset()
if (captureSession?.isRunning == false) {
DispatchQueue.global(qos: .userInitiated).async {
self.captureSession.startRunning()
}
}
}
override public func viewWillLayoutSubviews() {
previewLayer?.frame = view.layer.bounds
}
@objc func updateOrientation() {
guard let orientation = view.window?.windowScene?.interfaceOrientation else { return }
guard let connection = captureSession.connections.last, connection.isVideoOrientationSupported else { return }
connection.videoOrientation = AVCaptureVideoOrientation(rawValue: orientation.rawValue) ?? .portrait
}
override public func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
updateOrientation()
}
private func addviewfinder() {
guard showViewfinder, let imageView = viewFinder else { return }
view.addSubview(imageView)
NSLayoutConstraint.activate([
imageView.centerYAnchor.constraint(equalTo: view.centerYAnchor),
imageView.centerXAnchor.constraint(equalTo: view.centerXAnchor),
imageView.widthAnchor.constraint(equalToConstant: 200),
imageView.heightAnchor.constraint(equalToConstant: 200),
])
}
override public func viewDidDisappear(_ animated: Bool) {
super.viewDidDisappear(animated)
if (captureSession?.isRunning == true) {
DispatchQueue.global(qos: .userInitiated).async {
self.captureSession.stopRunning()
}
}
NotificationCenter.default.removeObserver(self)
}
override public var prefersStatusBarHidden: Bool {
true
}
override public var supportedInterfaceOrientations: UIInterfaceOrientationMask {
.all
}
/** Touch the screen for autofocus */
public override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
guard touches.first?.view == view,
let touchPoint = touches.first,
let device = delegate?.parent.videoCaptureDevice ?? fallbackVideoCaptureDevice
else { return }
let videoView = view
let screenSize = videoView!.bounds.size
let xPoint = touchPoint.location(in: videoView).y / screenSize.height
let yPoint = 1.0 - touchPoint.location(in: videoView).x / screenSize.width
let focusPoint = CGPoint(x: xPoint, y: yPoint)
do {
try device.lockForConfiguration()
} catch {
return
}
// Focus to the correct point, make continiuous focus and exposure so the point stays sharp when moving the device closer
device.focusPointOfInterest = focusPoint
device.focusMode = .continuousAutoFocus
device.exposurePointOfInterest = focusPoint
device.exposureMode = AVCaptureDevice.ExposureMode.continuousAutoExposure
device.unlockForConfiguration()
}
#endif
func updateViewController(isTorchOn: Bool, isGalleryPresented: Bool) {
if let backCamera = AVCaptureDevice.default(for: AVMediaType.video),
backCamera.hasTorch
{
try? backCamera.lockForConfiguration()
backCamera.torchMode = isTorchOn ? .on : .off
backCamera.unlockForConfiguration()
}
if isGalleryPresented && !isGalleryShowing {
openGallery()
}
}
}
}
+1 -4
View File
@@ -99,10 +99,7 @@ struct ConfigView: View {
} }
} }
Section( Section(NSLocalizedString("Version", comment: "Section title for displaying the version number of the Damus app.")) {
header: Text(NSLocalizedString("Version", comment: "Section title for displaying the version number of the Damus app.")),
footer: Text("").padding(.bottom, tabHeight + getSafeAreaBottom())
) {
Text(verbatim: VersionInfo.version) Text(verbatim: VersionInfo.version)
.contextMenu { .contextMenu {
Button { Button {
+1 -1
View File
@@ -28,7 +28,7 @@ struct CreateAccountView: View {
Spacer() Spacer()
VStack(alignment: .center) { VStack(alignment: .center) {
EditPictureControl(uploader: .nostrBuild, keypair: account.keypair, pubkey: account.pubkey, size: 75, setup: true, image_url: $account.profile_image , uploadObserver: profileUploadObserver, callback: uploadedProfilePicture) EditPictureControl(uploader: .nostrBuild, pubkey: account.pubkey, size: 75, setup: true, image_url: $account.profile_image , uploadObserver: profileUploadObserver, callback: uploadedProfilePicture)
.shadow(radius: 2) .shadow(radius: 2)
.padding(.top, 100) .padding(.top, 100)
-3
View File
@@ -10,7 +10,6 @@ import Combine
struct DMChatView: View, KeyboardReadable { struct DMChatView: View, KeyboardReadable {
let damus_state: DamusState let damus_state: DamusState
@FocusState private var isTextFieldFocused: Bool
@ObservedObject var dms: DirectMessageModel @ObservedObject var dms: DirectMessageModel
var pubkey: Pubkey { var pubkey: Pubkey {
@@ -47,7 +46,6 @@ struct DMChatView: View, KeyboardReadable {
} }
} }
} }
.padding(.bottom, isTextFieldFocused ? 0 : tabHeight)
} }
func scroll_to_end(_ scroller: ScrollViewProxy, animated: Bool = false) { func scroll_to_end(_ scroller: ScrollViewProxy, animated: Bool = false) {
@@ -76,7 +74,6 @@ struct DMChatView: View, KeyboardReadable {
.textEditorBackground { .textEditorBackground {
InputBackground() InputBackground()
} }
.focused($isTextFieldFocused)
.cornerRadius(8) .cornerRadius(8)
.background( .background(
RoundedRectangle(cornerRadius: 8) RoundedRectangle(cornerRadius: 8)
-1
View File
@@ -35,7 +35,6 @@ struct DirectMessagesView: View {
} }
.padding(.horizontal) .padding(.horizontal)
} }
.padding(.bottom, tabHeight)
} }
func filter_dms(dms: [DirectMessageModel]) -> [DirectMessageModel] { func filter_dms(dms: [DirectMessageModel]) -> [DirectMessageModel] {
@@ -59,7 +59,7 @@ struct HighlightEventRef: View {
} }
VStack(alignment: .leading, spacing: 5) { VStack(alignment: .leading, spacing: 5) {
Text(longform_event.title ?? NSLocalizedString("Untitled", comment: "Title of longform event if it is untitled.")) Text(longform_event.title ?? "Untitled")
.font(.system(size: 14, weight: .bold)) .font(.system(size: 14, weight: .bold))
.lineLimit(1) .lineLimit(1)
@@ -130,7 +130,7 @@ struct LongformPreviewBody: View {
} }
} }
Text(event.title ?? NSLocalizedString("Untitled", comment: "Title of longform event if it is untitled.")) Text(event.title ?? "Untitled")
.font(header ? .title : .headline) .font(header ? .title : .headline)
.padding(.horizontal, 10) .padding(.horizontal, 10)
.padding(.top, 5) .padding(.top, 5)
@@ -24,7 +24,7 @@ struct LongformView: View {
var body: some View { var body: some View {
EventShell(state: state, event: event.event, options: options) { EventShell(state: state, event: event.event, options: options) {
SelectableText(damus_state: state, event: event.event, attributedString: AttributedString(stringLiteral: event.title ?? NSLocalizedString("Untitled", comment: "Title of longform event if it is untitled.")), size: .title) SelectableText(damus_state: state, event: event.event, attributedString: AttributedString(stringLiteral: event.title ?? "Untitled"), size: .title)
NoteContentView(damus_state: state, event: event.event, blur_images: false, size: .selected, options: options) NoteContentView(damus_state: state, event: event.event, blur_images: false, size: .selected, options: options)
} }
@@ -1,140 +0,0 @@
//
// DamusFullScreenCover.swift
// damus
//
// Created by Daniel DAquino on 2024-10-25.
//
import SwiftUI
// MARK: - Private view modifier implementations of DamusFullScreenCover
/// This implements a full screen cover made for use in Damus.
/// This was created as a way to facilitate video coordination throughout the app, by handling the necessary logic without requiring any special handling in the usages of video player views.
///
/// In the future this could be used to faciliate other full screen logic as well.
///
/// # Features
///
/// This has the following features:
/// 1. It automatically tells the video coordinator about full screen mode changes, so that the video coordinator always knows if the app is in normal mode or in full screen mode for video coordination
/// 2. It automatically sets the `view_layer_context`, which is consumed by video player views, allowing those views to communicate about their layer position to the video coordinator
fileprivate struct DamusFullScreenCover<FullScreenContent: View, T: Identifiable & Equatable>: ViewModifier {
/// The `damus_state`, where we can access the video coordinator
let damus_state: DamusState
/// The item to be presented full screen
@Binding var item: T?
/// The view to be presented full screen
let full_screen_content: (T) -> FullScreenContent
func body(content: Content) -> some View {
content
.environment(\.view_layer_context, .normal_layer) // Let the views under content know they are NOT in a full screen environment
.onChange(of: item, perform: { newValue in
// Inform the video coordinator whether we are in full screen mode or not.
damus_state.video.set_full_screen_mode(newValue != nil)
})
.fullScreenCover(item: $item, content: { item in
full_screen_content(item)
.environment(\.view_layer_context, .full_screen_layer) // Let the views under full screen content know they are in a full screen environment
// Another observer for full screen presentation is needed here because in some cases the underlying view (`body::content`) may have been deinitialized and no longer listen to changes
// One such example is when the underlying navigation stack navigates away from a source view at the same time it opens the full screen view
// Therefore, when the full screen view is dismissed, this content will disappear, and we should notify the video coordinator.
.onDisappear {
damus_state.video.set_full_screen_mode(false)
}
})
}
}
/// A convenience view modifier that provides a different interface than `DamusFullScreenCover`, but is otherwise identical to it.
fileprivate struct DamusFullScreenCoverWithoutItem<FullScreenContent: View>: ViewModifier {
let damus_state: DamusState
@Binding var is_presented: Bool
let full_screen_content: () -> FullScreenContent
private let fake_item: FakeItem = FakeItem()
private var binding_item: Binding<FakeItem?> {
return Binding(
get: { is_presented ? self.fake_item : nil },
set: { is_presented = $0 != nil ? true : false }
)
}
func body(content: Content) -> some View {
content
.damus_full_screen_cover(self.binding_item, damus_state: damus_state, content: { _ in full_screen_content() })
}
private struct FakeItem: Identifiable, Equatable {
let id: Int = 1
}
}
// MARK: - Environment variable definitions
extension EnvironmentValues {
@Entry var view_layer_context: ViewLayerContext? = nil
}
/// Context about the layer a view finds itself in
/// This communicates to a view (e.g. a video player) context about whether it is being displayed inside a full screen layer, or a normal layer
enum ViewLayerContext {
/// This is used for items placed in a scroll view, such as on a timeline or a thread view.
case normal_layer
/// This is used for video players being displayed full screen
case full_screen_layer
}
// MARK: - View extension interfaces to access Damus' full screen cover
extension View {
/// A full screen cover to be used throughout Damus, containing extra functionality that helps with app coordination, and is meant to replace `.fullScreenCover`
///
/// ## Usage notes
///
/// This is the preferred method of doing a full screen cover. This is preferred over `.fullScreenCover` because it helps with certain coordination elements:
///
/// 1. It automatically informs the video coordinator if the app is in full screen or not
/// 2. It provides contextual information that any child view can pickup to introspect whether or not they are in a full screen layer. This can be picked up via the `\.view_layer_context` environment variable
///
/// **CAUTION:**
/// If you are planning to use this from a view that is presented on a timeline or lazy stack, please use `present(full_screen_item: FullScreenItem)` instead to avoid your full screen view to abruptly disappear.
/// Please read the documentation for `present(full_screen_item: FullScreenItem)` for more details.
///
/// - Parameters:
/// - is_presented: whether to show the full screen cover
/// - damus_state: The state of the app
/// - content: The view to show full screen
/// - Returns: the modified view
func damus_full_screen_cover<Content: View>(_ is_presented: Binding<Bool>, damus_state: DamusState, @ViewBuilder content: @escaping () -> Content) -> some View {
return self.modifier(DamusFullScreenCoverWithoutItem(damus_state: damus_state, is_presented: is_presented, full_screen_content: content))
}
/// A full screen cover to be used throughout Damus, containing extra functionality that helps with app coordination, and is meant to replace `.fullScreenCover`
///
/// ## Usage notes
///
/// This is the preferred method of doing a full screen cover. This is preferred over `.fullScreenCover` because it helps with certain coordination elements:
///
/// 1. It automatically informs the video coordinator if the app is in full screen or not
/// 2. It provides contextual information that any child view can pickup to introspect whether or not they are in a full screen layer. This can be picked up via the `\.view_layer_context` environment variable
///
/// **CAUTION:**
/// If you are planning to use this from a view that is presented on a timeline or lazy stack, please use `present(full_screen_item: FullScreenItem)` instead to avoid your full screen view to abruptly disappear.
/// Please read the documentation for `present(full_screen_item: FullScreenItem)` for more details.
///
///
/// - Parameters:
/// - item: The item to be displayed full screen, or `nil` if full screen should be dismissed.
/// - damus_state: The state of the app
/// - content: The view to render `item`
/// - Returns: the modified view
func damus_full_screen_cover<Content: View, T: Identifiable & Equatable>(_ item: Binding<T?>, damus_state: DamusState, @ViewBuilder content: @escaping (T) -> Content) -> some View {
return self.modifier(DamusFullScreenCover(damus_state: damus_state, item: item, full_screen_content: content))
}
}
+14 -106
View File
@@ -10,119 +10,27 @@ import Foundation
import SwiftUI import SwiftUI
extension View { extension View {
/// Watches for visibility changes. Does not detect occlusion func on_visibility_change(perform visibility_change_notifier: @escaping (Bool) -> Void, edge: Alignment = .center) -> some View {
/// self.modifier(VisibilityTracker(visibility_change_notifier: visibility_change_notifier, edge: edge))
/// ## Usage notes
///
/// 1. Detection mechanisms are not perfect, parameters may need fine tuning. Please refer to `VisibilityTracker` documentation for more details.
/// 2. This does **not** detect if the view has been occluded. There are currently no known mechanisms to do that.
/// If occlusion tracking is needed for your usage, consider using layout knowledge/introspection of the different layers that make up the view, and using that information for your logic.
/// For example, when dealing with items on a normal view, and a full screen cover, write your logic based on explicit information about which views are in the full screen layer.
/// Read about `present(full_screen_item: FullScreenItem)`, `damus_full_screen_cover`, and the `.view_layer_context` environment variable.
///
/// - Parameters:
/// - visibility_change_notifier: Function to call once visibility changes
/// - edge: Edge for the visibility overlay sensor
/// - method: The method to use for visibility tracking. Refer to `VisibilityTracker` documentation for more details.
/// - Returns: A modified view.
func on_visibility_change(perform visibility_change_notifier: @escaping (Bool) -> Void, edge: Alignment = .center, method: VisibilityTracker.Method = .standard) -> some View {
self.modifier(VisibilityTracker(visibility_change_notifier: visibility_change_notifier, edge: edge, method: method))
} }
} }
/// Tracks visibility of a SwiftUI view.
/// Built mostly to track visibility states of video players around the app and help the video coordinator pick a video to focus on, but can be used for basically any other view
/// **Caution:** This is not a perfect tracker, please read and fine-tune parameters for your use case, especially `method`
struct VisibilityTracker: ViewModifier { struct VisibilityTracker: ViewModifier {
let visibility_window: CGFloat = 0.8
let visibility_change_notifier: (Bool) -> Void let visibility_change_notifier: (Bool) -> Void
let edge: Alignment let edge: Alignment
let method: Method
init(visibility_change_notifier: @escaping (Bool) -> Void, edge: Alignment, method: Method) {
self.visibility_change_notifier = visibility_change_notifier
self.edge = edge
self.method = method
}
@EnvironmentObject private var orientationTracker: OrientationTracker
/// Holds information about whether the view is "generically" visible, meaning whether it would have been loaded on a lazy stack.
@State private var generic_visible: Bool = false {
didSet {
if oldValue == generic_visible { return } // Save up computing resources if there were no changes
self.visibility_change_notifier(self.is_visible)
}
}
/// Whether the view is visible by checking if its Y position is within a range of the user's screen
@State private var y_scroll_visible: Bool = false {
didSet {
switch self.method {
case .standard:
if oldValue == y_scroll_visible { return } // Save up computing resources if there were no changes
self.visibility_change_notifier(self.is_visible)
case .no_y_scroll_detection:
return // Don't cause re-renders if the visibility method does not use this
}
}
}
/// Whether view is "visible"
var is_visible: Bool {
switch method {
case .standard:
return generic_visible && y_scroll_visible
case .no_y_scroll_detection:
return generic_visible
}
}
func body(content: Content) -> some View { func body(content: Content) -> some View {
content content
.overlay( .overlay(
GeometryReader { geo in LazyVStack {
let localFrame = geo.frame(in: .local) Color.clear
let centerY = globalCoordinate(localX: 0, localY: localFrame.midY, localGeometry: geo).y .onAppear {
LazyVStack { visibility_change_notifier(true)
Color.clear }
// MARK: Detection triggers .onDisappear {
.onAppear { visibility_change_notifier(false)
self.generic_visible = true }
self.y_scroll_visible = self.compute_y_scroll_visible(centerY: centerY) },
} alignment: edge)
.onDisappear {
self.generic_visible = false
}
.onChange(of: centerY) { new_center_y in
if generic_visible == false { return } // Don't bother calculating anything if this is not visible generically, to save up computing resources
self.y_scroll_visible = self.compute_y_scroll_visible(
centerY: new_center_y // Compute the new Y scroll visibility using the newest value to avoid transient issues on device orientation changes
)
}
}
},
alignment: edge)
}
/// Computes whether the view is "visible" in a range of the screen given its Y position
private func compute_y_scroll_visible(centerY: CGFloat) -> Bool {
let screen_center_y = orientationTracker.deviceMajorAxis / 2
let screen_visibility_window_margin = orientationTracker.deviceMajorAxis * visibility_window / 2
let isBelowTop = centerY > screen_center_y - screen_visibility_window_margin,
isAboveBottom = centerY < screen_center_y + screen_visibility_window_margin
return (isBelowTop && isAboveBottom)
}
/// The methods available for visibility detection.
/// Unfortunately, there is currently no perfect visibility detection mechanism, so callers of `VisibilityTracker` should select a method that best suits the context of the view.
enum Method: Equatable {
/// Includes both a generic and Y coordinate based visibility detection.
/// When this option is selected, the view is only deemed visible if both lazy view evaluators load it (when close enough to viewport), and the center Y coordinate is sufficiently in the center
/// This is best for most view presentations, specially for scroll views.
case standard
/// Includes only a generic visibility detection based on a lazy view loader
/// When this option is selected, the view is only deemed visible if the lazy view evaluators load it (which SwiftUI does when it is close enough to viewport), regardless of Y coordinate
/// This is not suitable for scroll views or most presentations because it may trigger too early, leading to false positives. This is more suitable when the standard detection mechanism is triggering too many false negatives, and this is a more "static" view
/// For example: when displaying an item in full screen mode where it is visible in a more stable, static form, and device orientation changes may cause transient visibility triggers
case no_y_scroll_detection
} }
} }
+20
View File
@@ -0,0 +1,20 @@
//
// ImageView.swift
// damus
//
// Created by user232838 on 1/5/23.
//
import SwiftUI
struct ImageView: View {
var body: some View {
Text(verbatim: /*@START_MENU_TOKEN@*/"Hello, World!"/*@END_MENU_TOKEN@*/)
}
}
struct ImageView_Previews: PreviewProvider {
static var previews: some View {
ImageView()
}
}
+30 -71
View File
@@ -8,37 +8,37 @@
import SwiftUI import SwiftUI
struct FullScreenCarouselView<Content: View>: View { struct FullScreenCarouselView<Content: View>: View {
@ObservedObject var video_coordinator: DamusVideoCoordinator let video_controller: VideoController
let urls: [MediaUrl] let urls: [MediaUrl]
@Environment(\.presentationMode) var presentationMode @Environment(\.presentationMode) var presentationMode
@State var showMenu = true @State var showMenu = true
@State private var imageDict: [URL: UIImage] = [:]
let settings: UserSettingsStore let settings: UserSettingsStore
@ObservedObject var carouselSelection: CarouselSelection @Binding var selectedIndex: Int
let content: (() -> Content)? let content: (() -> Content)?
init(video_coordinator: DamusVideoCoordinator, urls: [MediaUrl], showMenu: Bool = true, settings: UserSettingsStore, selectedIndex: Binding<Int>, @ViewBuilder content: @escaping () -> Content) { init(video_controller: VideoController, urls: [MediaUrl], showMenu: Bool = true, settings: UserSettingsStore, selectedIndex: Binding<Int>, @ViewBuilder content: @escaping () -> Content) {
self.video_coordinator = video_coordinator self.video_controller = video_controller
self.urls = urls self.urls = urls
self._showMenu = State(initialValue: showMenu) self._showMenu = State(initialValue: showMenu)
self.settings = settings self.settings = settings
self._carouselSelection = ObservedObject(initialValue: CarouselSelection(index: selectedIndex.wrappedValue)) _selectedIndex = selectedIndex
self.content = content self.content = content
} }
init(video_coordinator: DamusVideoCoordinator, urls: [MediaUrl], showMenu: Bool = true, settings: UserSettingsStore, selectedIndex: Binding<Int>) { init(video_controller: VideoController, urls: [MediaUrl], showMenu: Bool = true, settings: UserSettingsStore, selectedIndex: Binding<Int>) {
self.video_coordinator = video_coordinator self.video_controller = video_controller
self.urls = urls self.urls = urls
self._showMenu = State(initialValue: showMenu) self._showMenu = State(initialValue: showMenu)
self.settings = settings self.settings = settings
self._carouselSelection = ObservedObject(initialValue: CarouselSelection(index: selectedIndex.wrappedValue)) _selectedIndex = selectedIndex
self.content = nil self.content = nil
} }
var background: some ShapeStyle { var background: some ShapeStyle {
if case .video = urls[safe: carouselSelection.index] { if case .video = urls[safe: selectedIndex] {
return AnyShapeStyle(Color.black) return AnyShapeStyle(Color.black)
} }
else { else {
@@ -55,24 +55,23 @@ struct FullScreenCarouselView<Content: View>: View {
Color(self.background_color) Color(self.background_color)
.ignoresSafeArea() .ignoresSafeArea()
TabView(selection: $carouselSelection.index) { TabView(selection: $selectedIndex) {
ForEach(urls.indices, id: \.self) { index in ForEach(urls.indices, id: \.self) { index in
VStack { VStack {
if case .video = urls[safe: index] { if case .video = urls[safe: index] {
ImageContainerView( ImageContainerView(video_controller: video_controller, url: urls[index], settings: settings)
video_coordinator: video_coordinator, .clipped() // SwiftUI hack from https://stackoverflow.com/a/74401288 to make playback controls show up within the TabView
url: urls[index], .aspectRatio(contentMode: .fit)
settings: settings, .padding(.top, Theme.safeAreaInsets?.top)
imageDict: $imageDict .padding(.bottom, Theme.safeAreaInsets?.bottom)
) .modifier(SwipeToDismissModifier(minDistance: 50, onDismiss: {
.modifier(SwipeToDismissModifier(minDistance: 50, onDismiss: { presentationMode.wrappedValue.dismiss()
presentationMode.wrappedValue.dismiss() }))
})) .ignoresSafeArea()
.ignoresSafeArea()
} }
else { else {
ZoomableScrollView { ZoomableScrollView {
ImageContainerView(video_coordinator: video_coordinator, url: urls[index], settings: settings, imageDict: $imageDict) ImageContainerView(video_controller: video_controller, url: urls[index], settings: settings)
.aspectRatio(contentMode: .fit) .aspectRatio(contentMode: .fit)
.padding(.top, Theme.safeAreaInsets?.top) .padding(.top, Theme.safeAreaInsets?.top)
.padding(.bottom, Theme.safeAreaInsets?.bottom) .padding(.bottom, Theme.safeAreaInsets?.bottom)
@@ -97,49 +96,17 @@ struct FullScreenCarouselView<Content: View>: View {
GeometryReader { geo in GeometryReader { geo in
VStack { VStack {
if showMenu { if showMenu {
HStack { NavDismissBarView(showBackgroundCircle: false)
Button(action: { .foregroundColor(.white)
presentationMode.wrappedValue.dismiss()
}, label: {
Image(systemName: "xmark")
.frame(width: 30, height: 30)
})
.buttonStyle(PlayerCircleButtonStyle())
Spacer()
if let url = urls[safe: carouselSelection.index],
let image = imageDict[url.url] {
ShareLink(item: Image(uiImage: image),
preview: SharePreview(NSLocalizedString("Shared Picture",
comment: "Label for the preview of the image being picture"),
image: Image(uiImage: image))) {
Image(systemName: "ellipsis")
.frame(width: 30, height: 30)
}
.buttonStyle(PlayerCircleButtonStyle())
}
}
.padding()
Spacer() Spacer()
VStack { if urls.count > 1 {
if urls.count > 1 { PageControlView(currentPage: $selectedIndex, numberOfPages: urls.count)
PageControlView(currentPage: $carouselSelection.index, numberOfPages: urls.count) .frame(maxWidth: 0, maxHeight: 0)
.frame(maxWidth: 0, maxHeight: 0) .padding(.top, 5)
.padding(.top, 5)
}
if let focused_video = video_coordinator.focused_video {
DamusVideoControlsView(video: focused_video)
}
self.content?()
} }
.padding(.top, 5)
.background(Color.black.opacity(0.7)) self.content?()
} }
} }
.animation(.easeInOut, value: showMenu) .animation(.easeInOut, value: showMenu)
@@ -161,7 +128,7 @@ fileprivate struct FullScreenCarouselPreviewView<Content: View>: View {
} }
var body: some View { var body: some View {
FullScreenCarouselView(video_coordinator: test_damus_state.video, urls: [test_video_url, url], settings: test_damus_state.settings, selectedIndex: $selectedIndex) { FullScreenCarouselView(video_controller: test_damus_state.video, urls: [test_video_url, url], settings: test_damus_state.settings, selectedIndex: $selectedIndex) {
self.custom_content?() self.custom_content?()
} }
.environmentObject(OrientationTracker()) .environmentObject(OrientationTracker())
@@ -189,11 +156,3 @@ struct FullScreenCarouselView_Previews: PreviewProvider {
} }
} }
} }
/// Class to define object for monitoring selectedIndex and updating mutlples views
final class CarouselSelection: ObservableObject {
@Published var index: Int
init(index: Int) {
self.index = index
}
}
+5 -17
View File
@@ -10,29 +10,18 @@ import Kingfisher
struct ImageContainerView: View { struct ImageContainerView: View {
let video_coordinator: DamusVideoCoordinator let video_controller: VideoController
let url: MediaUrl let url: MediaUrl
let settings: UserSettingsStore let settings: UserSettingsStore
@Binding var imageDict: [URL: UIImage]
@State private var image: UIImage? @State private var image: UIImage?
@State private var showShareSheet = false @State private var showShareSheet = false
init(video_coordinator: DamusVideoCoordinator, url: MediaUrl, settings: UserSettingsStore, imageDict: Binding<[URL: UIImage]>) {
self.video_coordinator = video_coordinator
self.url = url
self.settings = settings
self._imageDict = imageDict
}
private struct ImageHandler: ImageModifier { private struct ImageHandler: ImageModifier {
@Binding var handler: UIImage? @Binding var handler: UIImage?
@Binding var imageDict: [URL: UIImage]
let url: URL
func modify(_ image: UIImage) -> UIImage { func modify(_ image: UIImage) -> UIImage {
handler = image handler = image
imageDict[url] = image
return image return image
} }
} }
@@ -43,7 +32,7 @@ struct ImageContainerView: View {
.configure { view in .configure { view in
view.framePreloadCount = 3 view.framePreloadCount = 3
} }
.imageModifier(ImageHandler(handler: $image, imageDict: $imageDict, url: url)) .imageModifier(ImageHandler(handler: $image))
.kfClickable() .kfClickable()
.clipped() .clipped()
.modifier(ImageContextMenuModifier(url: url, image: image, settings: settings, showShareSheet: $showShareSheet)) .modifier(ImageContextMenuModifier(url: url, image: image, settings: settings, showShareSheet: $showShareSheet))
@@ -58,7 +47,7 @@ struct ImageContainerView: View {
case .image(let url): case .image(let url):
Img(url: url) Img(url: url)
case .video(let url): case .video(let url):
DamusVideoPlayerView(url: url, coordinator: video_coordinator, style: .no_controls(on_tap: nil)) DamusVideoPlayer(url: url, video_size: .constant(nil), controller: video_controller, style: .full, visibility_tracking_method: .generic)
} }
} }
} }
@@ -69,11 +58,10 @@ fileprivate let test_video_url = URL(string: "http://cdn.jb55.com/s/zaps-build.m
struct ImageContainerView_Previews: PreviewProvider { struct ImageContainerView_Previews: PreviewProvider {
static var previews: some View { static var previews: some View {
@State var imageDict: [URL: UIImage] = [:]
Group { Group {
ImageContainerView(video_coordinator: test_damus_state.video, url: .image(test_image_url), settings: test_damus_state.settings, imageDict: $imageDict) ImageContainerView(video_controller: test_damus_state.video, url: .image(test_image_url), settings: test_damus_state.settings)
.previewDisplayName("Image") .previewDisplayName("Image")
ImageContainerView(video_coordinator: test_damus_state.video, url: .video(test_video_url), settings: test_damus_state.settings, imageDict: $imageDict) ImageContainerView(video_controller: test_damus_state.video, url: .video(test_video_url), settings: test_damus_state.settings)
.previewDisplayName("Video") .previewDisplayName("Video")
} }
.environmentObject(OrientationTracker()) .environmentObject(OrientationTracker())
@@ -78,7 +78,7 @@ struct ImageContextMenuModifier: ViewModifier {
Label(NSLocalizedString("Share", comment: "Button to share an image."), image: "upload") Label(NSLocalizedString("Share", comment: "Button to share an image."), image: "upload")
} }
} }
.alert(String(format: NSLocalizedString("Found\n %@", comment: "Alert message asking if the user wants to open the link."), qrCodeValue).truncate(maxLength: 50), isPresented: $open_link_confirm) { .alert(NSLocalizedString("Found\n \(qrCodeValue)", comment: "Alert message asking if the user wants to open the link.").truncate(maxLength: 50), isPresented: $open_link_confirm) {
if open_wallet_confirm { if open_wallet_confirm {
Button(NSLocalizedString("Open in wallet", comment: "Button to open the value found in browser."), role: .none) { Button(NSLocalizedString("Open in wallet", comment: "Button to open the value found in browser."), role: .none) {
do { do {
+12 -61
View File
@@ -10,7 +10,8 @@ import Kingfisher
struct ProfileImageContainerView: View { struct ProfileImageContainerView: View {
let url: URL? let url: URL?
let settings: UserSettingsStore let settings: UserSettingsStore
@Binding var image: UIImage?
@State private var image: UIImage?
@State private var showShareSheet = false @State private var showShareSheet = false
private struct ImageHandler: ImageModifier { private struct ImageHandler: ImageModifier {
@@ -39,18 +40,13 @@ struct ProfileImageContainerView: View {
} }
} }
enum NavDismissBarContainer {
case fullScreenCarousel
case profilePicImageView
}
struct NavDismissBarView: View { struct NavDismissBarView: View {
@Environment(\.presentationMode) var presentationMode @Environment(\.presentationMode) var presentationMode
let navDismissBarContainer: NavDismissBarContainer let showBackgroundCircle: Bool
init(navDismissBarContainer: NavDismissBarContainer) { init(showBackgroundCircle: Bool = true) {
self.navDismissBarContainer = navDismissBarContainer self.showBackgroundCircle = showBackgroundCircle
} }
var body: some View { var body: some View {
@@ -58,18 +54,15 @@ struct NavDismissBarView: View {
Button(action: { Button(action: {
presentationMode.wrappedValue.dismiss() presentationMode.wrappedValue.dismiss()
}, label: { }, label: {
switch navDismissBarContainer { if showBackgroundCircle {
case .profilePicImageView:
Image("close") Image("close")
.frame(width: 33, height: 33) .frame(width: 33, height: 33)
.background(.regularMaterial) .background(.regularMaterial)
.clipShape(Circle()) .clipShape(Circle())
}
case .fullScreenCarousel: else {
Image("close") Image("close")
.frame(width: 33, height: 33) .frame(width: 33, height: 33)
.background(.damusBlack)
.clipShape(Circle())
} }
}) })
@@ -83,10 +76,6 @@ struct ProfilePicImageView: View {
let pubkey: Pubkey let pubkey: Pubkey
let profiles: Profiles let profiles: Profiles
let settings: UserSettingsStore let settings: UserSettingsStore
let nav: NavigationCoordinator
let shouldShowEditButton: Bool
@State var image: UIImage?
@State var showMenu = true
@Environment(\.presentationMode) var presentationMode @Environment(\.presentationMode) var presentationMode
@@ -96,57 +85,18 @@ struct ProfilePicImageView: View {
.ignoresSafeArea() .ignoresSafeArea()
ZoomableScrollView { ZoomableScrollView {
ProfileImageContainerView(url: get_profile_url(picture: nil, pubkey: pubkey, profiles: profiles), settings: settings, image: $image) ProfileImageContainerView(url: get_profile_url(picture: nil, pubkey: pubkey, profiles: profiles), settings: settings)
.aspectRatio(contentMode: .fit) .aspectRatio(contentMode: .fit)
.padding(.top, Theme.safeAreaInsets?.top) .padding(.top, Theme.safeAreaInsets?.top)
.padding(.bottom, Theme.safeAreaInsets?.bottom) .padding(.bottom, Theme.safeAreaInsets?.bottom)
.padding(.horizontal) .padding(.horizontal)
.allowsHitTesting(false)
} }
.ignoresSafeArea() .ignoresSafeArea()
.modifier(SwipeToDismissModifier(minDistance: 50, onDismiss: { .modifier(SwipeToDismissModifier(minDistance: 50, onDismiss: {
presentationMode.wrappedValue.dismiss() presentationMode.wrappedValue.dismiss()
})) }))
} }
.overlay( .overlay(NavDismissBarView(), alignment: .top)
Group {
if showMenu {
HStack {
NavDismissBarView(navDismissBarContainer: .profilePicImageView)
if let image = image {
ShareLink(item: Image(uiImage: image),
preview: SharePreview(NSLocalizedString("Damus Profile", comment: "Label for the preview of the profile picture"), image: Image(uiImage: image))) {
Image(systemName: "ellipsis")
.frame(width: 33, height: 33)
.background(.regularMaterial)
.clipShape(Circle())
}
.padding(20)
}
}
}
},
alignment: .top
)
.overlay(
shouldShowEditButton && showMenu ?
Button(action: {
presentationMode.wrappedValue.dismiss()
nav.push(route: Route.EditMetadata)
}) {
Text("Edit", comment: "Edit Button for editing profile")
.font(.system(size: 16, weight: .bold))
.foregroundColor(Color("DamusPurple"))
Spacer()
}
.padding([.vertical, .leading], 20)
: nil,
alignment: .bottomLeading
)
.gesture(TapGesture(count: 1).onEnded {
showMenu.toggle()
})
.animation(.easeInOut, value: showMenu)
} }
} }
@@ -155,6 +105,7 @@ struct ProfileZoomView_Previews: PreviewProvider {
ProfilePicImageView( ProfilePicImageView(
pubkey: test_pubkey, pubkey: test_pubkey,
profiles: make_preview_profiles(test_pubkey), profiles: make_preview_profiles(test_pubkey),
settings: test_damus_state.settings, nav: test_damus_state.nav, shouldShowEditButton: true) settings: test_damus_state.settings
)
} }
} }
+19 -32
View File
@@ -5,7 +5,6 @@
// Created by William Casarin on 2022-05-22. // Created by William Casarin on 2022-05-22.
// //
import CodeScanner
import SwiftUI import SwiftUI
enum ParsedKey { enum ParsedKey {
@@ -104,7 +103,6 @@ struct LoginView: View {
} }
.frame(minWidth: 300, maxWidth: .infinity, maxHeight: 12, alignment: .center) .frame(minWidth: 300, maxWidth: .infinity, maxHeight: 12, alignment: .center)
} }
.accessibilityIdentifier(AppAccessibilityIdentifiers.sign_in_confirm_button.rawValue)
.buttonStyle(GradientButtonStyle()) .buttonStyle(GradientButtonStyle())
.padding(.top, 10) .padding(.top, 10)
} }
@@ -300,35 +298,27 @@ struct KeyInput: View {
var body: some View { var body: some View {
HStack { HStack {
Button(action: { Image(systemName: "doc.on.clipboard")
if let pastedkey = UIPasteboard.general.string { .foregroundColor(.gray)
self.key.wrappedValue = pastedkey .onTapGesture {
if let pastedkey = UIPasteboard.general.string {
self.key.wrappedValue = pastedkey
}
} }
}, label: {
Image(systemName: "doc.on.clipboard")
})
.foregroundColor(.gray)
.accessibilityLabel(NSLocalizedString("Paste private key", comment: "Accessibility label for the private key paste button"))
SignInScan(shouldSaveKey: shouldSaveKey, loginKey: key, privKeyFound: privKeyFound) SignInScan(shouldSaveKey: shouldSaveKey, loginKey: key, privKeyFound: privKeyFound)
if is_secured { if is_secured {
SecureField("", text: key) SecureField("", text: key)
.nsecLoginStyle(key: key.wrappedValue, title: title) .nsecLoginStyle(key: key.wrappedValue, title: title)
.accessibilityLabel(NSLocalizedString("Account private key", comment: "Accessibility label for the private key input field")) } else {
} else { TextField("", text: key)
TextField("", text: key) .nsecLoginStyle(key: key.wrappedValue, title: title)
.nsecLoginStyle(key: key.wrappedValue, title: title) }
.accessibilityLabel(NSLocalizedString("Account private key", comment: "Accessibility label for the private key input field")) Image(systemName: "eye.slash")
} .foregroundColor(.gray)
.onTapGesture {
Button(action: { is_secured.toggle()
is_secured.toggle() }
}, label: {
Image(systemName: "eye.slash")
})
.foregroundColor(.gray)
.accessibilityLabel(NSLocalizedString("Toggle key visibility", comment: "Accessibility label for toggling the visibility of the private key input field"))
} }
.padding(.vertical, 2) .padding(.vertical, 2)
.padding(.horizontal, 10) .padding(.horizontal, 10)
@@ -351,7 +341,6 @@ struct SignInHeader: View {
.frame(width: 56, height: 56, alignment: .center) .frame(width: 56, height: 56, alignment: .center)
.shadow(color: DamusColors.purple, radius: 2) .shadow(color: DamusColors.purple, radius: 2)
.padding(.bottom) .padding(.bottom)
.accessibilityLabel(NSLocalizedString("Damus logo", comment: "Accessibility label for damus logo"))
Text("Sign in", comment: "Title of view to log into an account.") Text("Sign in", comment: "Title of view to log into an account.")
.foregroundColor(DamusColors.neutral6) .foregroundColor(DamusColors.neutral6)
@@ -375,12 +364,10 @@ struct SignInEntry: View {
.fontWeight(.medium) .fontWeight(.medium)
.padding(.top, 30) .padding(.top, 30)
KeyInput(NSLocalizedString("nsec1…", comment: "Prompt for user to enter in an account key to login. This text shows the characters the key could start with if it was a private key."), KeyInput(NSLocalizedString("nsec1...", comment: "Prompt for user to enter in an account key to login. This text shows the characters the key could start with if it was a private key."),
key: key, key: key,
shouldSaveKey: shouldSaveKey, shouldSaveKey: shouldSaveKey,
privKeyFound: $privKeyFound) privKeyFound: $privKeyFound)
.accessibilityIdentifier(AppAccessibilityIdentifiers.sign_in_nsec_key_entry_field.rawValue)
if privKeyFound { if privKeyFound {
Toggle(NSLocalizedString("Save Key in Secure Keychain", comment: "Toggle to save private key to the Apple secure keychain."), isOn: shouldSaveKey) Toggle(NSLocalizedString("Save Key in Secure Keychain", comment: "Toggle to save private key to the Apple secure keychain."), isOn: shouldSaveKey)
} }
@@ -401,7 +388,7 @@ struct SignInScan: View {
Button(action: { showQR.toggle() }, label: { Button(action: { showQR.toggle() }, label: {
Image(systemName: "qrcode.viewfinder")}) Image(systemName: "qrcode.viewfinder")})
.foregroundColor(.gray) .foregroundColor(.gray)
.accessibilityLabel(NSLocalizedString("Scan QR code", comment: "Accessibility label for a button that scans a private key QR code"))
} }
.sheet(isPresented: $showQR, onDismiss: { .sheet(isPresented: $showQR, onDismiss: {
if qrkey == nil { resetView() }} if qrkey == nil { resetView() }}
-3
View File
@@ -66,9 +66,7 @@ struct TabButton: View {
struct TabBar: View { struct TabBar: View {
var nstatus: NotificationStatusModel var nstatus: NotificationStatusModel
var navIsAtRoot: Bool
@Binding var selected: Timeline @Binding var selected: Timeline
@Binding var headerOffset: CGFloat
let settings: UserSettingsStore let settings: UserSettingsStore
let action: (Timeline) -> () let action: (Timeline) -> ()
@@ -83,6 +81,5 @@ struct TabBar: View {
TabButton(timeline: .notifications, img: "notification-bell", selected: $selected, nstatus: nstatus, settings: settings, action: action).keyboardShortcut("4") TabButton(timeline: .notifications, img: "notification-bell", selected: $selected, nstatus: nstatus, settings: settings, action: action).keyboardShortcut("4")
} }
} }
.opacity(selected != .home || (selected == .home && !navIsAtRoot) ? 1.0 : 0.35 + abs(1.25 - (abs(headerOffset/100.0))))
} }
} }
+18 -55
View File
@@ -9,27 +9,18 @@ import UIKit
import SwiftUI import SwiftUI
import PhotosUI import PhotosUI
enum MediaPickerEntry {
case editPictureControl
case postView
}
struct MediaPicker: UIViewControllerRepresentable { struct MediaPicker: UIViewControllerRepresentable {
@Environment(\.presentationMode) @Environment(\.presentationMode)
@Binding private var presentationMode @Binding private var presentationMode
let mediaPickerEntry: MediaPickerEntry
@Binding var image_upload_confirm: Bool @Binding var image_upload_confirm: Bool
var imagesOnly: Bool = false
let onMediaPicked: (PreUploadedMedia) -> Void let onMediaPicked: (PreUploadedMedia) -> Void
final class Coordinator: NSObject, PHPickerViewControllerDelegate { final class Coordinator: NSObject, PHPickerViewControllerDelegate {
var parent: MediaPicker let parent: MediaPicker
// properties used for returning medias in the same order as picking
let dispatchGroup: DispatchGroup = DispatchGroup()
var orderIds: [String] = []
var orderMap: [String: PreUploadedMedia] = [:]
init(_ parent: MediaPicker) { init(_ parent: MediaPicker) {
self.parent = parent self.parent = parent
@@ -40,16 +31,7 @@ struct MediaPicker: UIViewControllerRepresentable {
self.parent.presentationMode.dismiss() self.parent.presentationMode.dismiss()
} }
// When user dismiss the upload confirmation and re-adds again, reset orderIds and orderMap
orderIds.removeAll()
orderMap.removeAll()
for result in results { for result in results {
let orderId = result.assetIdentifier ?? UUID().uuidString
orderIds.append(orderId)
dispatchGroup.enter()
if result.itemProvider.hasItemConformingToTypeIdentifier(UTType.image.identifier) { if result.itemProvider.hasItemConformingToTypeIdentifier(UTType.image.identifier) {
result.itemProvider.loadItem(forTypeIdentifier: UTType.image.identifier, options: nil) { (item, error) in result.itemProvider.loadItem(forTypeIdentifier: UTType.image.identifier, options: nil) { (item, error) in
guard let url = item as? URL else { return } guard let url = item as? URL else { return }
@@ -68,7 +50,7 @@ struct MediaPicker: UIViewControllerRepresentable {
do { do {
try imageData.write(to: destinationURL) try imageData.write(to: destinationURL)
Task { Task {
await self.chooseMedia(.processed_image(destinationURL), orderId: orderId) await self.chooseMedia(.processed_image(destinationURL))
} }
} }
catch { catch {
@@ -82,13 +64,13 @@ struct MediaPicker: UIViewControllerRepresentable {
url: url, url: url,
fallback: processImage, fallback: processImage,
unprocessedEnum: {.unprocessed_image($0)}, unprocessedEnum: {.unprocessed_image($0)},
processedEnum: {.processed_image($0)}, processedEnum: {.processed_image($0)}
orderId: orderId) )
} else { } else {
// Media was taken from camera // Media was taken from camera
result.itemProvider.loadObject(ofClass: UIImage.self) { (image, error) in result.itemProvider.loadObject(ofClass: UIImage.self) { (image, error) in
if let image = image as? UIImage, error == nil { if let image = image as? UIImage, error == nil {
self.chooseMedia(.uiimage(image), orderId: orderId) self.chooseMedia(.uiimage(image))
} }
} }
} }
@@ -101,60 +83,41 @@ struct MediaPicker: UIViewControllerRepresentable {
url: url, url: url,
fallback: processVideo, fallback: processVideo,
unprocessedEnum: {.unprocessed_video($0)}, unprocessedEnum: {.unprocessed_video($0)},
processedEnum: {.processed_video($0)}, orderId: orderId processedEnum: {.processed_video($0)}
) )
} }
} }
} }
dispatchGroup.notify(queue: .main) { [weak self] in
guard let self = self else { return }
var arrMedia: [PreUploadedMedia] = []
for id in self.orderIds {
if let media = self.orderMap[id] {
arrMedia.append(media)
self.parent.onMediaPicked(media)
}
}
}
} }
private func chooseMedia(_ media: PreUploadedMedia) {
private func chooseMedia(_ media: PreUploadedMedia, orderId: String) { self.parent.onMediaPicked(media)
self.parent.image_upload_confirm = true self.parent.image_upload_confirm = true
self.orderMap[orderId] = media
self.dispatchGroup.leave()
} }
private func attemptAcquireResourceAndChooseMedia(url: URL, fallback: (URL) -> URL?, unprocessedEnum: (URL) -> PreUploadedMedia, processedEnum: (URL) -> PreUploadedMedia, orderId: String) { private func attemptAcquireResourceAndChooseMedia(url: URL, fallback: (URL) -> URL?, unprocessedEnum: (URL) -> PreUploadedMedia, processedEnum: (URL) -> PreUploadedMedia) {
if url.startAccessingSecurityScopedResource() { if url.startAccessingSecurityScopedResource() {
// Have permission from system to use url out of scope // Have permission from system to use url out of scope
print("Acquired permission to security scoped resource") print("Acquired permission to security scoped resource")
self.chooseMedia(unprocessedEnum(url), orderId: orderId) self.chooseMedia(unprocessedEnum(url))
} else { } else {
// Need to copy URL to non-security scoped location // Need to copy URL to non-security scoped location
guard let newUrl = fallback(url) else { return } guard let newUrl = fallback(url) else { return }
self.chooseMedia(processedEnum(newUrl), orderId: orderId) self.chooseMedia(processedEnum(newUrl))
} }
} }
} }
func makeCoordinator() -> Coordinator { func makeCoordinator() -> Coordinator {
Coordinator(self) Coordinator(self)
} }
func makeUIViewController(context: Context) -> PHPickerViewController { func makeUIViewController(context: Context) -> PHPickerViewController {
var configuration = PHPickerConfiguration(photoLibrary: .shared()) var configuration = PHPickerConfiguration(photoLibrary: .shared())
switch mediaPickerEntry { configuration.selectionLimit = 1
case .postView: configuration.filter = imagesOnly ? .images : .any(of: [.images, .videos])
configuration.selectionLimit = 0 // allows multiple media selection
configuration.filter = .any(of: [.images, .videos])
configuration.selection = .ordered // images are returned in the order they were selected + numbered badge displayed
case .editPictureControl:
configuration.selectionLimit = 1 // allows one media selection
configuration.filter = .images // allows image only
}
let picker = PHPickerViewController(configuration: configuration) let picker = PHPickerViewController(configuration: configuration)
picker.delegate = context.coordinator as any PHPickerViewControllerDelegate picker.delegate = context.coordinator as any PHPickerViewControllerDelegate
return picker return picker
+8 -15
View File
@@ -13,10 +13,6 @@ struct AddMuteItemView: View {
@Environment(\.dismiss) var dismiss @Environment(\.dismiss) var dismiss
var trimmedText: String {
new_text.trimmingCharacters(in: .whitespaces)
}
var body: some View { var body: some View {
VStack { VStack {
Text("Add mute item", comment: "Title text to indicate user to an add an item to their mutelist.") Text("Add mute item", comment: "Title text to indicate user to an add an item to their mutelist.")
@@ -34,13 +30,12 @@ struct AddMuteItemView: View {
Text("Duration", comment: "The duration in which to mute the given item.") Text("Duration", comment: "The duration in which to mute the given item.")
} }
let trimmedText = self.trimmedText
HStack { HStack {
Label("", image: "copy2") Label("", image: "copy2")
.onTapGesture { .onTapGesture {
if let pasted_text = UIPasteboard.general.string { if let pasted_text = UIPasteboard.general.string {
self.new_text = pasted_text.trimmingCharacters(in: .whitespaces) self.new_text = pasted_text
} }
} }
TextField(NSLocalizedString("npub, #hashtag, phrase", comment: "Placeholder example for relay server address."), text: $new_text) TextField(NSLocalizedString("npub, #hashtag, phrase", comment: "Placeholder example for relay server address."), text: $new_text)
@@ -49,7 +44,7 @@ struct AddMuteItemView: View {
Label("", image: "close-circle") Label("", image: "close-circle")
.foregroundColor(.accentColor) .foregroundColor(.accentColor)
.opacity(trimmedText.isEmpty ? 0.0 : 1.0) .opacity((new_text == "") ? 0.0 : 1.0)
.onTapGesture { .onTapGesture {
self.new_text = "" self.new_text = ""
} }
@@ -61,17 +56,17 @@ struct AddMuteItemView: View {
Button(action: { Button(action: {
let expiration_date: Date? = self.expiration.date_from_now let expiration_date: Date? = self.expiration.date_from_now
let mute_item: MuteItem? = { let mute_item: MuteItem? = {
if trimmedText.starts(with: "npub") { if new_text.starts(with: "npub") {
if let pubkey: Pubkey = bech32_pubkey_decode(trimmedText) { if let pubkey: Pubkey = bech32_pubkey_decode(new_text) {
return .user(pubkey, expiration_date) return .user(pubkey, expiration_date)
} else { } else {
return nil return nil
} }
} else if trimmedText.starts(with: "#") { } else if new_text.starts(with: "#") {
// Remove the starting `#` character // Remove the starting `#` character
return .hashtag(Hashtag(hashtag: String("\(trimmedText)".dropFirst())), expiration_date) return .hashtag(Hashtag(hashtag: String("\(new_text)".dropFirst())), expiration_date)
} else { } else {
return .word(trimmedText, expiration_date) return .word(new_text, expiration_date)
} }
}() }()
@@ -97,15 +92,13 @@ struct AddMuteItemView: View {
dismiss() dismiss()
}) { }) {
HStack { HStack {
Text("Add mute item", comment: "Button to an add an item to the user's mutelist.") Text(verbatim: "Add mute item")
.bold() .bold()
} }
.frame(minWidth: 300, maxWidth: .infinity, alignment: .center) .frame(minWidth: 300, maxWidth: .infinity, alignment: .center)
} }
.buttonStyle(GradientButtonStyle(padding: 10)) .buttonStyle(GradientButtonStyle(padding: 10))
.padding(.vertical) .padding(.vertical)
.opacity(trimmedText.isEmpty ? 0.5 : 1.0)
.disabled(trimmedText.isEmpty)
Spacer() Spacer()
} }
+1 -4
View File
@@ -86,10 +86,7 @@ struct MutelistView: View {
} }
} }
} }
Section( Section(NSLocalizedString("Threads", comment: "Section header title for a list of threads that are muted.")) {
header: Text(NSLocalizedString("Threads", comment: "Section header title for a list of threads that are muted.")),
footer: Text("").padding(.bottom, 10 + tabHeight + getSafeAreaBottom())
) {
ForEach(threads, id: \.self) { item in ForEach(threads, id: \.self) { item in
if case let MuteItem.thread(note_id, _) = item { if case let MuteItem.thread(note_id, _) = item {
if let event = damus_state.events.lookup(note_id) { if let event = damus_state.events.lookup(note_id) {
+9 -3
View File
@@ -119,7 +119,15 @@ struct NoteContentView: View {
} }
func fullscreen_preview(dismiss: @escaping () -> Void) -> some View { func fullscreen_preview(dismiss: @escaping () -> Void) -> some View {
EmptyView() VStack {
EventView(damus: damus_state, event: self.event, options: .embedded_text_only)
.padding(.top)
}
.background(.thickMaterial)
.onTapGesture(perform: {
damus_state.nav.push(route: Route.Thread(thread: .init(event: self.event, damus_state: damus_state)))
dismiss()
})
} }
func MainContent(artifacts: NoteArtifactsSeparated) -> some View { func MainContent(artifacts: NoteArtifactsSeparated) -> some View {
@@ -295,9 +303,7 @@ struct NoteContentView: View {
case .separated(let separated): case .separated(let separated):
if #available(iOS 17.4, macOS 14.4, *) { if #available(iOS 17.4, macOS 14.4, *) {
MainContent(artifacts: separated) MainContent(artifacts: separated)
#if !targetEnvironment(macCatalyst)
.translationPresentation(isPresented: $isAppleTranslationPopoverPresented, text: event.get_content(damus_state.keypair)) .translationPresentation(isPresented: $isAppleTranslationPopoverPresented, text: event.get_content(damus_state.keypair))
#endif
} else { } else {
MainContent(artifacts: separated) MainContent(artifacts: separated)
} }
@@ -143,7 +143,7 @@ func event_group_unique_pubkeys(profiles: Profiles, group: EventGroupType) -> [P
"zapped_your_profile_2" - returned when 2 zaps occurred to the current user's profile "zapped_your_profile_2" - returned when 2 zaps occurred to the current user's profile
"zapped_your_profile_3" - returned when 3 or more zaps occurred to the current user's profile "zapped_your_profile_3" - returned when 3 or more zaps occurred to the current user's profile
*/ */
func reacting_to_text(profiles: Profiles, our_pubkey: Pubkey, group: EventGroupType, ev: NostrEvent?, pubkeys: [Pubkey], locale: Locale = Locale.current) -> String { func reacting_to_text(profiles: Profiles, our_pubkey: Pubkey, group: EventGroupType, ev: NostrEvent?, pubkeys: [Pubkey], locale: Locale? = nil) -> String {
if group.events.count == 0 { if group.events.count == 0 {
return "??" return "??"
} }
@@ -188,8 +188,7 @@ struct EventGroupView: View {
let group: EventGroupType let group: EventGroupType
func GroupDescription(_ pubkeys: [Pubkey]) -> some View { func GroupDescription(_ pubkeys: [Pubkey]) -> some View {
let text = reacting_to_text(profiles: state.profiles, our_pubkey: state.pubkey, group: group, ev: event, pubkeys: pubkeys) Text(verbatim: "\(reacting_to_text(profiles: state.profiles, our_pubkey: state.pubkey, group: group, ev: event, pubkeys: pubkeys))")
return Text(text)
} }
func ZapIcon(_ zapgrp: ZapGroup) -> some View { func ZapIcon(_ zapgrp: ZapGroup) -> some View {
@@ -38,9 +38,7 @@ struct OnboardingSuggestionsView: View {
}, label: { }, label: {
Text("Skip", comment: "Button to dismiss the suggested users screen") Text("Skip", comment: "Button to dismiss the suggested users screen")
.font(.subheadline.weight(.semibold)) .font(.subheadline.weight(.semibold))
}) }))
.accessibilityIdentifier(AppAccessibilityIdentifiers.onboarding_sheet_skip_button.rawValue)
)
.tag(0) .tag(0)
PostView( PostView(
@@ -114,10 +112,7 @@ struct SuggestedUsersSectionHeader: View {
let model: SuggestedUsersViewModel let model: SuggestedUsersViewModel
var body: some View { var body: some View {
HStack { HStack {
let locale = Locale.current Text(group.title.uppercased())
let format = localizedStringFormat(key: group.category, locale: locale)
let categoryName = String(format: format, locale: locale)
Text(categoryName)
Spacer() Spacer()
Button(NSLocalizedString("Follow All", comment: "Button to follow all users in this section")) { Button(NSLocalizedString("Follow All", comment: "Button to follow all users in this section")) {
model.follow(pubkeys: group.users) model.follow(pubkeys: group.users)
@@ -48,10 +48,7 @@ struct SuggestedUserView: View {
.foregroundColor(.gray) .foregroundColor(.gray)
.font(.caption) .font(.caption)
} }
.frame(maxWidth: .infinity, alignment: .leading)
Spacer() Spacer()
GradientFollowButton(target: target, follows_you: false, follow_state: damus_state.contacts.follow_state(target.pubkey)) GradientFollowButton(target: target, follows_you: false, follow_state: damus_state.contacts.follow_state(target.pubkey))
} }
} }
@@ -10,11 +10,11 @@ import Combine
struct SuggestedUserGroup: Identifiable, Codable { struct SuggestedUserGroup: Identifiable, Codable {
let id = UUID() let id = UUID()
let category: String let title: String
let users: [Pubkey] let users: [Pubkey]
enum CodingKeys: String, CodingKey { enum CodingKeys: String, CodingKey {
case category, users case title, users
} }
} }
+10 -9
View File
@@ -1,6 +1,6 @@
[ [
{ {
"category": "suggested_users_nostr", "title": "nostr",
"users": [ "users": [
"ba2f394833658475e91680b898f9be0f1d850166c6a839dbe084d0266ad6e20a", "ba2f394833658475e91680b898f9be0f1d850166c6a839dbe084d0266ad6e20a",
"82341f882b6eabcd2ba7f1ef90aad961cf074af15b9ef44a09f9d2a8fbfbe6a2", "82341f882b6eabcd2ba7f1ef90aad961cf074af15b9ef44a09f9d2a8fbfbe6a2",
@@ -9,29 +9,30 @@
] ]
}, },
{ {
"category": "suggested_users_permaculture_livestock_gardening", "title": "permaculture & livestock & gardening",
"users": [ "users": [
"4b1804c90d59dff195ee0e8f692b98a7c762bf1793b3e126c546d730dcb04477", "4b1804c90d59dff195ee0e8f692b98a7c762bf1793b3e126c546d730dcb04477",
"2c24e1af571fb4ccfeca3981649c1b09c695cd83b129709eb3b41c7ad2854899", "2c24e1af571fb4ccfeca3981649c1b09c695cd83b129709eb3b41c7ad2854899",
"296842eaaed9be5ae0668da09fe48aac0521c4af859ad547d93145e5ac34c17e" "296842eaaed9be5ae0668da09fe48aac0521c4af859ad547d93145e5ac34c17e",
"2c24e1af571fb4ccfeca3981649c1b09c695cd83b129709eb3b41c7ad2854899"
] ]
}, },
{ {
"category": "suggested_users_music", "title": "music",
"users": [ "users": [
"23708a76e7090cb108d33e8801fd36262c475e5499b23eb87eee4a31f4f0144e", "23708a76e7090cb108d33e8801fd36262c475e5499b23eb87eee4a31f4f0144e",
"ca9d68eb25620fc755e1b8c76b5f155f4c7e96d99c532c109a8b36d208bdce55" "ca9d68eb25620fc755e1b8c76b5f155f4c7e96d99c532c109a8b36d208bdce55"
] ]
}, },
{ {
"category": "suggested_users_books", "title": "books",
"users": [ "users": [
"2652f3af10de63bc10a2628871a3fce62e08655e4fcf90a58be16f246bb65da3", "2652f3af10de63bc10a2628871a3fce62e08655e4fcf90a58be16f246bb65da3",
"b83a28b7e4e5d20bd960c5faeb6625f95529166b8bdb045d42634a2f35919450" "b83a28b7e4e5d20bd960c5faeb6625f95529166b8bdb045d42634a2f35919450"
] ]
}, },
{ {
"category": "suggested_users_art_photography", "title": "art & photography",
"users": [ "users": [
"f96c3d76497074c4c83a7b3823380e77dc73d5a9494fd2e053e4a1453e17824b", "f96c3d76497074c4c83a7b3823380e77dc73d5a9494fd2e053e4a1453e17824b",
"11b2d93b26d7e56fb57f0afce0d33bfa7fb35b913e4c0aeb7706464befb9ca97", "11b2d93b26d7e56fb57f0afce0d33bfa7fb35b913e4c0aeb7706464befb9ca97",
@@ -49,7 +50,7 @@
] ]
}, },
{ {
"category": "suggested_users_ai_art", "title": "ai art",
"users": [ "users": [
"431fa2f340f0adf8963d6d7c6e2c20d913278c691fe609fd3857db13d8f39feb", "431fa2f340f0adf8963d6d7c6e2c20d913278c691fe609fd3857db13d8f39feb",
"9936a53def39d712f886ac7e2ed509ce223b534834dd29d95caba9f6bc01ef35", "9936a53def39d712f886ac7e2ed509ce223b534834dd29d95caba9f6bc01ef35",
@@ -59,7 +60,7 @@
] ]
}, },
{ {
"category": "suggested_users_parenting", "title": "parenting",
"users": [ "users": [
"c7c8f645fd45b09055fb6c26d148737ad7ed12ddecde0d4c877b88f8d4196865", "c7c8f645fd45b09055fb6c26d148737ad7ed12ddecde0d4c877b88f8d4196865",
"fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52", "fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52",
@@ -69,7 +70,7 @@
] ]
}, },
{ {
"category": "suggested_users_food", "title": "food",
"users": [ "users": [
"cbb2f023b6aa09626d51d2f4ea99fa9138ea80ec7d5ffdce9feef8dcd6352031" "cbb2f023b6aa09626d51d2f4ea99fa9138ea80ec7d5ffdce9feef8dcd6352031"
] ]
+37 -125
View File
@@ -31,7 +31,6 @@ enum PostAction {
case quoting(NostrEvent) case quoting(NostrEvent)
case posting(PostTarget) case posting(PostTarget)
case highlighting(HighlightContentDraft) case highlighting(HighlightContentDraft)
case sharing(ShareContent)
var ev: NostrEvent? { var ev: NostrEvent? {
switch self { switch self {
@@ -43,8 +42,6 @@ enum PostAction {
return nil return nil
case .highlighting: case .highlighting:
return nil return nil
case .sharing(_):
return nil
} }
} }
} }
@@ -57,16 +54,13 @@ struct PostView: View {
@State var error: String? = nil @State var error: String? = nil
@State var uploadedMedias: [UploadedMedia] = [] @State var uploadedMedias: [UploadedMedia] = []
@State var image_upload_confirm: Bool = false @State var image_upload_confirm: Bool = false
@State var imagePastedFromPasteboard: PreUploadedMedia? = nil
@State var imageUploadConfirmPasteboard: Bool = false
@State var references: [RefId] = [] @State var references: [RefId] = []
@State var imageUploadConfirmDamusShare: Bool = false
@State var filtered_pubkeys: Set<Pubkey> = [] @State var filtered_pubkeys: Set<Pubkey> = []
@State var focusWordAttributes: (String?, NSRange?) = (nil, nil) @State var focusWordAttributes: (String?, NSRange?) = (nil, nil)
@State var newCursorIndex: Int? @State var newCursorIndex: Int?
@State var textHeight: CGFloat? = nil @State var textHeight: CGFloat? = nil
@State var preUploadedMedia: [PreUploadedMedia] = [] @State var preUploadedMedia: PreUploadedMedia? = nil
@StateObject var image_upload: ImageUploadModel = ImageUploadModel() @StateObject var image_upload: ImageUploadModel = ImageUploadModel()
@StateObject var tagModel: TagModel = TagModel() @StateObject var tagModel: TagModel = TagModel()
@@ -157,7 +151,6 @@ struct PostView: View {
var ImageButton: some View { var ImageButton: some View {
Button(action: { Button(action: {
preUploadedMedia.removeAll()
attach_media = true attach_media = true
}, label: { }, label: {
Image("images") Image("images")
@@ -221,8 +214,6 @@ struct PostView: View {
damus_state.drafts.post = nil damus_state.drafts.post = nil
case .highlighting(let draft): case .highlighting(let draft):
damus_state.drafts.highlights.removeValue(forKey: draft.source) damus_state.drafts.highlights.removeValue(forKey: draft.source)
case .sharing(_):
damus_state.drafts.post = nil
} }
} }
@@ -255,9 +246,7 @@ struct PostView: View {
TextViewWrapper( TextViewWrapper(
attributedText: $post, attributedText: $post,
textHeight: $textHeight, textHeight: $textHeight,
initialTextSuffix: initial_text_suffix, initialTextSuffix: initial_text_suffix,
imagePastedFromPasteboard: $imagePastedFromPasteboard,
imageUploadConfirmPasteboard: $imageUploadConfirmPasteboard,
cursorIndex: newCursorIndex, cursorIndex: newCursorIndex,
getFocusWordForMention: { word, range in getFocusWordForMention: { word, range in
focusWordAttributes = (word, range) focusWordAttributes = (word, range)
@@ -304,7 +293,6 @@ struct PostView: View {
.padding(10) .padding(10)
}) })
.buttonStyle(NeutralButtonStyle()) .buttonStyle(NeutralButtonStyle())
.accessibilityIdentifier(AppAccessibilityIdentifiers.post_composer_cancel_button.rawValue)
if let error { if let error {
Text(error) Text(error)
@@ -329,36 +317,34 @@ struct PostView: View {
.padding() .padding()
.padding(.top, 15) .padding(.top, 15)
} }
@discardableResult func handle_upload(media: MediaUpload) {
func handle_upload(media: MediaUpload) async -> Bool {
let uploader = damus_state.settings.default_media_uploader let uploader = damus_state.settings.default_media_uploader
Task {
let img = getImage(media: media) let img = getImage(media: media)
print("img size w:\(img.size.width) h:\(img.size.height)") print("img size w:\(img.size.width) h:\(img.size.height)")
async let blurhash = calculate_blurhash(img: img)
async let blurhash = calculate_blurhash(img: img) let res = await image_upload.start(media: media, uploader: uploader, keypair: damus_state.keypair)
let res = await image_upload.start(media: media, uploader: uploader, keypair: damus_state.keypair)
switch res {
case .success(let url):
guard let url = URL(string: url) else {
self.error = "Error uploading image :("
return false
}
let blurhash = await blurhash
let meta = blurhash.map { bh in calculate_image_metadata(url: url, img: img, blurhash: bh) }
let uploadedMedia = UploadedMedia(localURL: media.localURL, uploadedURL: url, representingImage: img, metadata: meta)
uploadedMedias.append(uploadedMedia)
return true
case .failed(let error): switch res {
if let error { case .success(let url):
self.error = error.localizedDescription guard let url = URL(string: url) else {
} else { self.error = "Error uploading image :("
self.error = "Error uploading image :(" return
}
let blurhash = await blurhash
let meta = blurhash.map { bh in calculate_image_metadata(url: url, img: img, blurhash: bh) }
let uploadedMedia = UploadedMedia(localURL: media.localURL, uploadedURL: url, representingImage: img, metadata: meta)
uploadedMedias.append(uploadedMedia)
case .failed(let error):
if let error {
self.error = error.localizedDescription
} else {
self.error = "Error uploading image :("
}
} }
return false
} }
} }
@@ -398,11 +384,6 @@ struct PostView: View {
else if case .highlighting(let draft) = action { else if case .highlighting(let draft) = action {
HighlightDraftContentView(draft: draft) HighlightDraftContentView(draft: draft)
} }
else if case .sharing(let draft) = action,
let url = draft.getLinkURL() {
LinkViewRepresentable(meta: .url(url))
.frame(height: 50)
}
} }
.padding(.horizontal) .padding(.horizontal)
} }
@@ -427,7 +408,7 @@ struct PostView: View {
GeometryReader { (deviceSize: GeometryProxy) in GeometryReader { (deviceSize: GeometryProxy) in
VStack(alignment: .leading, spacing: 0) { VStack(alignment: .leading, spacing: 0) {
let searching = get_searching_string(focusWordAttributes.0) let searching = get_searching_string(focusWordAttributes.0)
let searchingHashTag = get_searching_hashTag(focusWordAttributes.0)
TopBar TopBar
ScrollViewReader { scroller in ScrollViewReader { scroller in
@@ -441,7 +422,7 @@ struct PostView: View {
.padding(.top, 5) .padding(.top, 5)
} }
} }
.frame(maxHeight: searching == nil && searchingHashTag == nil ? deviceSize.size.height : 70) .frame(maxHeight: searching == nil ? deviceSize.size.height : 70)
.onAppear { .onAppear {
scroll_to_event(scroller: scroller, id: "post", delay: 1.0, animate: true, anchor: .top) scroll_to_event(scroller: scroller, id: "post", delay: 1.0, animate: true, anchor: .top)
} }
@@ -452,17 +433,7 @@ struct PostView: View {
UserSearch(damus_state: damus_state, search: searching, focusWordAttributes: $focusWordAttributes, newCursorIndex: $newCursorIndex, post: $post) UserSearch(damus_state: damus_state, search: searching, focusWordAttributes: $focusWordAttributes, newCursorIndex: $newCursorIndex, post: $post)
.frame(maxHeight: .infinity) .frame(maxHeight: .infinity)
.environmentObject(tagModel) .environmentObject(tagModel)
// This else observes '#' for hash-tag suggestions and creates SuggestedHashtagsView } else {
} else if let searchingHashTag {
SuggestedHashtagsView(damus_state: damus_state,
events: SearchHomeModel(damus_state: damus_state).events,
isFromPostView: true,
queryHashTag: searchingHashTag,
focusWordAttributes: $focusWordAttributes,
newCursorIndex: $newCursorIndex,
post: $post)
.environmentObject(tagModel)
} else {
Divider() Divider()
VStack(alignment: .leading) { VStack(alignment: .leading) {
AttachmentBar AttachmentBar
@@ -473,24 +444,17 @@ struct PostView: View {
} }
.background(DamusColors.adaptableWhite.edgesIgnoringSafeArea(.all)) .background(DamusColors.adaptableWhite.edgesIgnoringSafeArea(.all))
.sheet(isPresented: $attach_media) { .sheet(isPresented: $attach_media) {
MediaPicker(mediaPickerEntry: .postView, image_upload_confirm: $image_upload_confirm){ media in MediaPicker(image_upload_confirm: $image_upload_confirm){ media in
self.preUploadedMedia.append(media) self.preUploadedMedia = media
} }
.alert(NSLocalizedString("Are you sure you want to upload the selected media?", comment: "Alert message asking if the user wants to upload media."), isPresented: $image_upload_confirm) { .alert(NSLocalizedString("Are you sure you want to upload this media?", comment: "Alert message asking if the user wants to upload media."), isPresented: $image_upload_confirm) {
Button(NSLocalizedString("Upload", comment: "Button to proceed with uploading."), role: .none) { Button(NSLocalizedString("Upload", comment: "Button to proceed with uploading."), role: .none) {
// initiate asynchronous uploading Task for multiple-images if let mediaToUpload = generateMediaUpload(preUploadedMedia) {
Task { self.handle_upload(media: mediaToUpload)
for media in preUploadedMedia { self.attach_media = false
if let mediaToUpload = generateMediaUpload(media) {
await self.handle_upload(media: mediaToUpload)
}
}
} }
self.attach_media = false
}
Button(NSLocalizedString("Cancel", comment: "Button to cancel the upload."), role: .cancel) {
preUploadedMedia.removeAll()
} }
Button(NSLocalizedString("Cancel", comment: "Button to cancel the upload."), role: .cancel) {}
} }
} }
.sheet(isPresented: $attach_camera) { .sheet(isPresented: $attach_camera) {
@@ -499,31 +463,6 @@ struct PostView: View {
self.attach_media = true self.attach_media = true
} }
} }
// This alert seeks confirmation about Image-upload when user taps Paste option
.alert(NSLocalizedString("Are you sure you want to upload this media?", comment: "Alert message asking if the user wants to upload media."), isPresented: $imageUploadConfirmPasteboard) {
Button(NSLocalizedString("Upload", comment: "Button to proceed with uploading."), role: .none) {
if let image = imagePastedFromPasteboard,
let mediaToUpload = generateMediaUpload(image) {
Task {
await self.handle_upload(media: mediaToUpload)
}
}
}
Button(NSLocalizedString("Cancel", comment: "Button to cancel the upload."), role: .cancel) {}
}
// This alert seeks confirmation about media-upload from Damus Share Extension
.alert(NSLocalizedString("Are you sure you want to upload the selected media?", comment: "Alert message asking if the user wants to upload media."), isPresented: $imageUploadConfirmDamusShare) {
Button(NSLocalizedString("Upload", comment: "Button to proceed with uploading."), role: .none) {
Task {
for media in preUploadedMedia {
if let mediaToUpload = generateMediaUpload(media) {
await self.handle_upload(media: mediaToUpload)
}
}
}
}
Button(NSLocalizedString("Cancel", comment: "Button to cancel the upload."), role: .cancel) {}
}
.onAppear() { .onAppear() {
let loaded_draft = load_draft() let loaded_draft = load_draft()
@@ -537,15 +476,6 @@ struct PostView: View {
fill_target_content(target: target) fill_target_content(target: target)
case .highlighting(let draft): case .highlighting(let draft):
references = [draft.source.ref()] references = [draft.source.ref()]
case .sharing(let content):
if let url = content.getLinkURL() {
self.post = NSMutableAttributedString(string: "\(content.title)\n\(String(url.absoluteString))")
} else {
self.preUploadedMedia = content.getMediaArray()
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) {
self.imageUploadConfirmDamusShare = true // display Confirm Sheet after 1 sec
}
}
} }
DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) { DispatchQueue.main.asyncAfter(deadline: .now() + 0.5) {
@@ -556,7 +486,6 @@ struct PostView: View {
if isEmpty() { if isEmpty() {
clear_draft() clear_draft()
} }
preUploadedMedia.removeAll()
} }
} }
} }
@@ -584,17 +513,6 @@ func get_searching_string(_ word: String?) -> String? {
return String(word.dropFirst()) return String(word.dropFirst())
} }
fileprivate func get_searching_hashTag(_ word: String?) -> String? {
guard let word,
word.count >= 2,
let first_char = word.first,
first_char == "#" else {
return nil
}
return String(word.dropFirst())
}
struct PostView_Previews: PreviewProvider { struct PostView_Previews: PreviewProvider {
static var previews: some View { static var previews: some View {
PostView(action: .posting(.none), damus_state: test_damus_state) PostView(action: .posting(.none), damus_state: test_damus_state)
@@ -695,8 +613,6 @@ func set_draft_for_post(drafts: Drafts, action: PostAction, artifacts: DraftArti
drafts.post = artifacts drafts.post = artifacts
case .highlighting(let draft): case .highlighting(let draft):
drafts.highlights[draft.source] = artifacts drafts.highlights[draft.source] = artifacts
case .sharing(_):
drafts.post = artifacts
} }
} }
@@ -710,8 +626,6 @@ func load_draft_for_post(drafts: Drafts, action: PostAction) -> DraftArtifacts?
return drafts.post return drafts.post
case .highlighting(let draft): case .highlighting(let draft):
return drafts.highlights[draft.source] return drafts.highlights[draft.source]
case .sharing(_):
return drafts.post
} }
} }
@@ -787,8 +701,6 @@ func build_post(state: DamusState, post: NSMutableAttributedString, action: Post
break break
case .highlighting(let draft): case .highlighting(let draft):
break break
case .sharing(_):
break
} }
// append additional tags // append additional tags
+5 -11
View File
@@ -76,10 +76,10 @@ struct EditMetadataView: View {
return NIP05.parse(nip05) return NIP05.parse(nip05)
} }
func topSection(topLevelGeo: GeometryProxy) -> some View { var TopSection: some View {
ZStack(alignment: .top) { ZStack(alignment: .top) {
GeometryReader { geo in GeometryReader { geo in
EditBannerImageView(damus_state: damus_state, viewModel: bannerUploadObserver, callback: uploadedBanner(image_url:), safeAreaInsets: topLevelGeo.safeAreaInsets, banner_image: URL(string: banner)) EditBannerImageView(damus_state: damus_state, viewModel: bannerUploadObserver, callback: uploadedBanner(image_url:), banner_image: URL(string: banner))
.aspectRatio(contentMode: .fill) .aspectRatio(contentMode: .fill)
.frame(width: geo.size.width, height: BANNER_HEIGHT) .frame(width: geo.size.width, height: BANNER_HEIGHT)
.clipped() .clipped()
@@ -122,14 +122,8 @@ struct EditMetadataView: View {
} }
var body: some View { var body: some View {
GeometryReader { proxy in
self.content(topLevelGeo: proxy)
}
}
func content(topLevelGeo: GeometryProxy) -> some View {
VStack(alignment: .leading) { VStack(alignment: .leading) {
self.topSection(topLevelGeo: topLevelGeo) TopSection
Form { Form {
Section(NSLocalizedString("Your Name", comment: "Label for Your Name section of user profile form.")) { Section(NSLocalizedString("Your Name", comment: "Label for Your Name section of user profile form.")) {
let display_name_placeholder = "Satoshi Nakamoto" let display_name_placeholder = "Satoshi Nakamoto"
@@ -209,7 +203,7 @@ struct EditMetadataView: View {
}) })
.buttonStyle(GradientButtonStyle(padding: 15)) .buttonStyle(GradientButtonStyle(padding: 15))
.padding(.horizontal, 10) .padding(.horizontal, 10)
.padding(.bottom, 10 + tabHeight) .padding(.bottom, 10)
.disabled(!didChange()) .disabled(!didChange())
.opacity(!didChange() ? 0.5 : 1) .opacity(!didChange() ? 0.5 : 1)
.disabled(profileUploadObserver.isLoading || bannerUploadObserver.isLoading) .disabled(profileUploadObserver.isLoading || bannerUploadObserver.isLoading)
@@ -224,7 +218,7 @@ struct EditMetadataView: View {
.background(Color(.systemGroupedBackground)) .background(Color(.systemGroupedBackground))
.navigationBarBackButtonHidden() .navigationBarBackButtonHidden()
.toolbar { .toolbar {
ToolbarItem(placement: .topBarLeading) { ToolbarItem(placement: .principal) {
navBackButton navBackButton
} }
} }
+3 -5
View File
@@ -14,7 +14,6 @@ class ImageUploadingObserver: ObservableObject {
struct EditPictureControl: View { struct EditPictureControl: View {
let uploader: MediaUploader let uploader: MediaUploader
let keypair: Keypair?
let pubkey: Pubkey let pubkey: Pubkey
var size: CGFloat? = 25 var size: CGFloat? = 25
var setup: Bool? = false var setup: Bool? = false
@@ -41,7 +40,6 @@ struct EditPictureControl: View {
}) { }) {
Text("Image URL", comment: "Option to enter a url") Text("Image URL", comment: "Option to enter a url")
} }
.accessibilityIdentifier(AppAccessibilityIdentifiers.own_profile_banner_image_edit_from_url.rawValue)
Button(action: { Button(action: {
self.show_library = true self.show_library = true
@@ -115,7 +113,7 @@ struct EditPictureControl: View {
} }
} }
.sheet(isPresented: $show_library) { .sheet(isPresented: $show_library) {
MediaPicker(mediaPickerEntry: .editPictureControl, image_upload_confirm: $image_upload_confirm) { media in MediaPicker(image_upload_confirm: $image_upload_confirm, imagesOnly: true) { media in
self.preUploadedMedia = media self.preUploadedMedia = media
} }
.alert(NSLocalizedString("Are you sure you want to upload this image?", comment: "Alert message asking if the user wants to upload an image."), isPresented: $image_upload_confirm) { .alert(NSLocalizedString("Are you sure you want to upload this image?", comment: "Alert message asking if the user wants to upload an image."), isPresented: $image_upload_confirm) {
@@ -197,7 +195,7 @@ struct EditPictureControl: View {
private func handle_upload(media: MediaUpload) { private func handle_upload(media: MediaUpload) {
uploadObserver.isLoading = true uploadObserver.isLoading = true
Task { Task {
let res = await image_upload.start(media: media, uploader: uploader, keypair: keypair) let res = await image_upload.start(media: media, uploader: uploader)
switch res { switch res {
case .success(let urlString): case .success(let urlString):
@@ -223,7 +221,7 @@ struct EditPictureControl_Previews: PreviewProvider {
let observer = ImageUploadingObserver() let observer = ImageUploadingObserver()
ZStack { ZStack {
Color.gray Color.gray
EditPictureControl(uploader: .nostrBuild, keypair: test_keypair, pubkey: test_pubkey, size: 100, setup: false, image_url: url, uploadObserver: observer) { _ in EditPictureControl(uploader: .nostrBuild, pubkey: test_pubkey, size: 100, setup: false, image_url: url, uploadObserver: observer) { _ in
// //
} }
} }
@@ -27,7 +27,6 @@ struct ProfileEditButton: View {
.minimumScaleFactor(0.5) .minimumScaleFactor(0.5)
.lineLimit(1) .lineLimit(1)
} }
.accessibilityIdentifier(AppAccessibilityIdentifiers.own_profile_edit_button.rawValue)
} }
func fillColor() -> Color { func fillColor() -> Color {
@@ -33,7 +33,7 @@ struct EditProfilePictureView: View {
.scaledToFill() .scaledToFill()
.kfClickable() .kfClickable()
EditPictureControl(uploader: damus_state?.settings.default_media_uploader ?? .nostrBuild, keypair: damus_state?.keypair, pubkey: pubkey, image_url: $profile_url, uploadObserver: uploadObserver, callback: callback) EditPictureControl(uploader: damus_state?.settings.default_media_uploader ?? .nostrBuild, pubkey: pubkey, image_url: $profile_url, uploadObserver: uploadObserver, callback: callback)
} }
.frame(width: size, height: size) .frame(width: size, height: size)
.clipShape(Circle()) .clipShape(Circle())
+20
View File
@@ -0,0 +1,20 @@
//
// ProfilePopup.swift
// damus
//
// Created by William Casarin on 2023-08-21.
//
import SwiftUI
struct ProfilePopup: View {
var body: some View {
Text(/*@START_MENU_TOKEN@*/"Hello, World!"/*@END_MENU_TOKEN@*/)
}
}
struct ProfilePopup_Previews: PreviewProvider {
static var previews: some View {
ProfilePopup()
}
}
+11 -53
View File
@@ -51,17 +51,13 @@ func followedByString(_ friend_intersection: [Pubkey], ndb: Ndb, locale: Locale
struct VisualEffectView: UIViewRepresentable { struct VisualEffectView: UIViewRepresentable {
var effect: UIVisualEffect? var effect: UIVisualEffect?
var darkeningOpacity: CGFloat = 0.3 // degree of darkening
func makeUIView(context: UIViewRepresentableContext<Self>) -> UIVisualEffectView { func makeUIView(context: UIViewRepresentableContext<Self>) -> UIVisualEffectView {
let effectView = UIVisualEffectView() UIVisualEffectView()
effectView.backgroundColor = UIColor.black.withAlphaComponent(darkeningOpacity)
return effectView
} }
func updateUIView(_ uiView: UIVisualEffectView, context: UIViewRepresentableContext<Self>) { func updateUIView(_ uiView: UIVisualEffectView, context: UIViewRepresentableContext<Self>) {
uiView.effect = effect uiView.effect = effect
uiView.backgroundColor = UIColor.black.withAlphaComponent(darkeningOpacity)
} }
} }
@@ -107,18 +103,6 @@ struct ProfileView: View {
return Double(-yOffset > navbarHeight ? progress : 0) return Double(-yOffset > navbarHeight ? progress : 0)
} }
func getProfileInfo() -> (String, String) {
let profile_txn = self.damus_state.profiles.lookup(id: profile.pubkey)
let ndbprofile = profile_txn?.unsafeUnownedValue
let displayName = Profile.displayName(profile: ndbprofile, pubkey: profile.pubkey).displayName.truncate(maxLength: 25)
let userName = Profile.displayName(profile: ndbprofile, pubkey: profile.pubkey).username.truncate(maxLength: 25)
return (displayName, "@\(userName)")
}
func showFollowBtnInBlurrBanner() -> Bool {
damus_state.contacts.follow_state(profile.pubkey) == .unfollows && bannerBlurViewOpacity() > 1.0
}
func content_filter(_ fstate: FilterState) -> ((NostrEvent) -> Bool) { func content_filter(_ fstate: FilterState) -> ((NostrEvent) -> Bool) {
var filters = ContentFilters.defaults(damus_state: damus_state) var filters = ContentFilters.defaults(damus_state: damus_state)
filters.append(fstate.filter) filters.append(fstate.filter)
@@ -313,8 +297,8 @@ struct ProfileView: View {
.onTapGesture { .onTapGesture {
is_zoomed.toggle() is_zoomed.toggle()
} }
.damus_full_screen_cover($is_zoomed, damus_state: damus_state) { .fullScreenCover(isPresented: $is_zoomed) {
ProfilePicImageView(pubkey: profile.pubkey, profiles: damus_state.profiles, settings: damus_state.settings, nav: damus_state.nav, shouldShowEditButton: damus_state.pubkey == profile.pubkey) ProfilePicImageView(pubkey: profile.pubkey, profiles: damus_state.profiles, settings: damus_state.settings)
} }
Spacer() Spacer()
@@ -460,44 +444,19 @@ struct ProfileView: View {
.zIndex(-yOffset > navbarHeight ? 0 : 1) .zIndex(-yOffset > navbarHeight ? 0 : 1)
} }
} }
.padding(.bottom, tabHeight + getSafeAreaBottom())
.ignoresSafeArea() .ignoresSafeArea()
.navigationTitle("") .navigationTitle("")
.navigationBarBackButtonHidden() .navigationBarBackButtonHidden()
.toolbar { .toolbar {
ToolbarItem(placement: .topBarLeading) { ToolbarItem(placement: .topBarLeading) {
HStack(spacing: 8) { navBackButton
navBackButton .padding(.top, 5)
.padding(.top, 5) .accentColor(DamusColors.white)
.accentColor(DamusColors.white)
VStack(alignment: .leading, spacing: -4.5) {
Text(getProfileInfo().0) // Display name
.font(.headline)
.foregroundColor(.white)
Text(getProfileInfo().1) // Username
.font(.subheadline)
.foregroundColor(.white.opacity(0.8))
}
.opacity(bannerBlurViewOpacity())
.frame(maxWidth: .infinity, alignment: .leading)
.padding(.top, max(5, 15 + (yOffset / 30)))
}
} }
if showFollowBtnInBlurrBanner() { ToolbarItem(placement: .topBarTrailing) {
ToolbarItem(placement: .topBarTrailing) { navActionSheetButton
FollowButtonView( .padding(.top, 5)
target: profile.get_follow_target(), .accentColor(DamusColors.white)
follows_you: profile.follows(pubkey: damus_state.pubkey),
follow_state: damus_state.contacts.follow_state(profile.pubkey)
)
.padding(.top, 8)
}
} else {
ToolbarItem(placement: .topBarTrailing) {
navActionSheetButton
.padding(.top, 5)
.accentColor(DamusColors.white)
}
} }
} }
.toolbarBackground(.hidden) .toolbarBackground(.hidden)
@@ -518,7 +477,7 @@ struct ProfileView: View {
let url = URL(string: "https://damus.io/" + profile.pubkey.npub)! let url = URL(string: "https://damus.io/" + profile.pubkey.npub)!
ShareSheet(activityItems: [url]) ShareSheet(activityItems: [url])
} }
.damus_full_screen_cover($show_qr_code, damus_state: damus_state) { .fullScreenCover(isPresented: $show_qr_code) {
QRCodeView(damus_state: damus_state, pubkey: profile.pubkey) QRCodeView(damus_state: damus_state, pubkey: profile.pubkey)
} }
@@ -526,7 +485,6 @@ struct ProfileView: View {
PostButtonContainer(is_left_handed: damus_state.settings.left_handed) { PostButtonContainer(is_left_handed: damus_state.settings.left_handed) {
notify(.compose(.posting(.user(profile.pubkey)))) notify(.compose(.posting(.user(profile.pubkey))))
} }
.padding(.bottom, tabHeight)
} }
} }
} }
+5 -12
View File
@@ -19,12 +19,9 @@ struct ProfileActionSheetView: View {
@Environment(\.colorScheme) var colorScheme @Environment(\.colorScheme) var colorScheme
@Environment(\.presentationMode) var presentationMode @Environment(\.presentationMode) var presentationMode
var navigationHandler: (() -> Void)? init(damus_state: DamusState, pubkey: Pubkey) {
init(damus_state: DamusState, pubkey: Pubkey, onNavigate navigationHandler: (() -> Void)? = nil) {
self.damus_state = damus_state self.damus_state = damus_state
self._profile = StateObject(wrappedValue: ProfileModel(pubkey: pubkey, damus: damus_state)) self._profile = StateObject(wrappedValue: ProfileModel(pubkey: pubkey, damus: damus_state))
self.navigationHandler = navigationHandler
} }
func imageBorderColor() -> Color { func imageBorderColor() -> Color {
@@ -40,12 +37,6 @@ struct ProfileActionSheetView: View {
return self.profile_data()?.profile return self.profile_data()?.profile
} }
func navigate(route: Route) {
damus_state.nav.push(route: route)
self.navigationHandler?()
dismiss()
}
var followButton: some View { var followButton: some View {
return ProfileActionSheetFollowButton( return ProfileActionSheetFollowButton(
target: .pubkey(self.profile.pubkey), target: .pubkey(self.profile.pubkey),
@@ -74,7 +65,8 @@ struct ProfileActionSheetView: View {
return VStack(alignment: .center, spacing: 10) { return VStack(alignment: .center, spacing: 10) {
Button( Button(
action: { action: {
self.navigate(route: Route.DMChat(dms: dm_model)) damus_state.nav.push(route: Route.DMChat(dms: dm_model))
dismiss()
}, },
label: { label: {
Image("messages") Image("messages")
@@ -134,7 +126,8 @@ struct ProfileActionSheetView: View {
Button( Button(
action: { action: {
self.navigate(route: Route.ProfileByKey(pubkey: profile.pubkey)) damus_state.nav.push(route: Route.ProfileByKey(pubkey: profile.pubkey))
dismiss()
}, },
label: { label: {
HStack { HStack {
+6 -8
View File
@@ -9,7 +9,6 @@ import SwiftUI
struct PubkeyView: View { struct PubkeyView: View {
let pubkey: Pubkey let pubkey: Pubkey
var sidemenu: Bool = false
@Environment(\.colorScheme) var colorScheme @Environment(\.colorScheme) var colorScheme
@@ -46,21 +45,20 @@ struct PubkeyView: View {
let bech32 = pubkey.npub let bech32 = pubkey.npub
HStack { HStack {
Text(verbatim: "\(abbrev_pubkey(bech32, amount: sidemenu ? 12 : 16))") Text(verbatim: "\(abbrev_pubkey(bech32, amount: 16))")
.font(sidemenu ? .system(size: 10) : .footnote) .font(.footnote)
.foregroundColor(keyColor()) .foregroundColor(keyColor())
.padding(5) .padding(5)
.padding([.leading], 5) .padding([.leading], 5)
.lineLimit(1)
HStack { HStack {
if isCopied { if isCopied {
Image("check-circle") Image("check-circle")
.resizable() .resizable()
.foregroundColor(DamusColors.green) .foregroundColor(DamusColors.green)
.frame(width: sidemenu ? 15 : 20, height: sidemenu ? 15 : 20) .frame(width: 20, height: 20)
Text("Copied", comment: "Label indicating that a user's key was copied.") Text("Copied", comment: "Label indicating that a user's key was copied.")
.font(sidemenu ? .system(size: 10) : .footnote) .font(.footnote)
.layoutPriority(1) .layoutPriority(1)
.foregroundColor(DamusColors.green) .foregroundColor(DamusColors.green)
} else { } else {
@@ -74,7 +72,7 @@ struct PubkeyView: View {
.resizable() .resizable()
.contentShape(Rectangle()) .contentShape(Rectangle())
.foregroundColor(colorScheme == .light ? DamusColors.darkGrey : DamusColors.lightGrey) .foregroundColor(colorScheme == .light ? DamusColors.darkGrey : DamusColors.lightGrey)
.frame(width: sidemenu ? 15 : 20, height: sidemenu ? 15 : 20) .frame(width: 20, height: 20)
} }
.labelStyle(IconOnlyLabelStyle()) .labelStyle(IconOnlyLabelStyle())
.symbolRenderingMode(.hierarchical) .symbolRenderingMode(.hierarchical)
@@ -70,7 +70,7 @@ struct DamusPurpleWelcomeView: View {
.opacity(start ? 1.0 : 0.0) .opacity(start ? 1.0 : 0.0)
.animation(Animation.snappy(duration: 2).delay(0), value: start) .animation(Animation.snappy(duration: 2).delay(0), value: start)
Text("Thank you very much for signing up for Damus Purple. Your contribution helps us continue our fight for a more Open and Free internet.\n\nYou will also get access to premium features, and a star badge on your profile.\n\nEnjoy!", comment: "Appreciation to user for purchasing subscription service") Text("Thank you very much for signing up for Damus\u{00A0}Purple. Your contribution helps us continue our fight for a more Open and Free\u{00A0}internet.\n\nYou will also get access to premium features, and a star badge on your profile.\n\nEnjoy!", comment: "Appreciation to user for purchasing subscription service")
.lineSpacing(5) .lineSpacing(5)
.multilineTextAlignment(.center) .multilineTextAlignment(.center)
.foregroundStyle(.white.opacity(0.8)) .foregroundStyle(.white.opacity(0.8))
+149 -310
View File
@@ -7,16 +7,61 @@
import SwiftUI import SwiftUI
import CoreImage.CIFilterBuiltins import CoreImage.CIFilterBuiltins
import CodeScanner
struct ProfileScanResult: Equatable {
let pubkey: Pubkey
init?(hex: String) {
guard let pk = hex_decode(hex).map({ bytes in Pubkey(Data(bytes)) }) else {
return nil
}
self.pubkey = pk
}
init?(string: String) {
var str = string
guard str.count != 0 else {
return nil
}
if str.hasPrefix("nostr:") {
str.removeFirst("nostr:".count)
}
if let decoded = hex_decode(str),
str.count == 64
{
self.pubkey = Pubkey(Data(decoded))
return
}
if str.starts(with: "npub"),
let b32 = try? bech32_decode(str)
{
self.pubkey = Pubkey(b32.data)
return
}
return nil
}
}
struct QRCodeView: View { struct QRCodeView: View {
let damus_state: DamusState let damus_state: DamusState
@State var pubkey: Pubkey @State var pubkey: Pubkey
@Environment(\.dismiss) var dismiss @Environment(\.presentationMode) var presentationMode
@State private var selectedTab = 0 @State private var selectedTab = 0
@State var scanResult: ProfileScanResult? = nil
@State var profile: Profile? = nil
@State var error: String? = nil
@State private var outerTrimEnd: CGFloat = 0
var animationDuration: Double = 0.5
let generator = UIImpactFeedbackGenerator(style: .light)
@ViewBuilder @ViewBuilder
func navImage(systemImage: String) -> some View { func navImage(systemImage: String) -> some View {
@@ -28,7 +73,7 @@ struct QRCodeView: View {
var navBackButton: some View { var navBackButton: some View {
Button { Button {
dismiss() presentationMode.wrappedValue.dismiss()
} label: { } label: {
navImage(systemImage: "chevron.left") navImage(systemImage: "chevron.left")
} }
@@ -53,7 +98,7 @@ struct QRCodeView: View {
TabView(selection: $selectedTab) { TabView(selection: $selectedTab) {
QRView QRView
.tag(0) .tag(0)
self.qrCameraView QRCameraView()
.tag(1) .tag(1)
} }
.tabViewStyle(PageTabViewStyle(indexDisplayMode: .never)) .tabViewStyle(PageTabViewStyle(indexDisplayMode: .never))
@@ -75,9 +120,18 @@ struct QRCodeView: View {
VStack(alignment: .center) { VStack(alignment: .center) {
let profile_txn = damus_state.profiles.lookup(id: pubkey, txn_name: "qrview-profile") let profile_txn = damus_state.profiles.lookup(id: pubkey, txn_name: "qrview-profile")
let profile = profile_txn?.unsafeUnownedValue let profile = profile_txn?.unsafeUnownedValue
let our_profile = profile_txn.flatMap({ ptxn in
damus_state.ndb.lookup_profile_with_txn(damus_state.pubkey, txn: ptxn)?.profile
})
ProfilePicView(pubkey: pubkey, size: 90.0, highlight: .custom(DamusColors.white, 3.0), profiles: damus_state.profiles, disable_animation: damus_state.settings.disable_animation) if our_profile?.picture != nil {
ProfilePicView(pubkey: pubkey, size: 90.0, highlight: .custom(DamusColors.white, 3.0), profiles: damus_state.profiles, disable_animation: damus_state.settings.disable_animation)
.padding(.top, 20) .padding(.top, 20)
} else {
Image(systemName: "person.fill")
.font(.system(size: 60))
.padding(.top, 20)
}
if let display_name = profile?.display_name { if let display_name = profile?.display_name {
Text(display_name) Text(display_name)
@@ -85,7 +139,7 @@ struct QRCodeView: View {
.foregroundColor(.white) .foregroundColor(.white)
} }
if let name = profile?.name { if let name = profile?.name {
Text(verbatim: "@" + name) Text("@" + name)
.font(.body) .font(.body)
.foregroundColor(.white) .foregroundColor(.white)
} }
@@ -105,17 +159,10 @@ struct QRCodeView: View {
Spacer() Spacer()
// apply the same styling to both text-views without code duplication Text("Follow me on Nostr", comment: "Text on QR code view to prompt viewer looking at screen to follow the user.")
Group { .font(.system(size: 24, weight: .heavy))
if damus_state.pubkey.npub == pubkey.npub { .padding(.top, 10)
Text("Follow me on Nostr", comment: "Text on QR code view to prompt viewer looking at screen to follow the user.") .foregroundColor(.white)
} else {
Text("Follow \(profile?.display_name ?? profile?.name ?? "") on Nostr", comment: "Text on QR code view to prompt viewer looking at screen to follow the user.")
}
}
.font(.system(size: 24, weight: .heavy))
.padding(.top, 10)
.foregroundColor(.white)
Text("Scan the code", comment: "Text on QR code view to prompt viewer to scan the QR code on screen with their device camera.") Text("Scan the code", comment: "Text on QR code view to prompt viewer to scan the QR code on screen with their device camera.")
.font(.system(size: 18, weight: .ultraLight)) .font(.system(size: 18, weight: .ultraLight))
@@ -137,8 +184,35 @@ struct QRCodeView: View {
} }
} }
var qrCameraView: some View { func QRCameraView() -> some View {
QRCameraView(damusState: damus_state, bottomContent: { return VStack(alignment: .center) {
Text("Scan a user's pubkey", comment: "Text to prompt scanning a QR code of a user's pubkey to open their profile.")
.padding(.top, 50)
.font(.system(size: 24, weight: .heavy))
.foregroundColor(.white)
Spacer()
CodeScannerView(codeTypes: [.qr], scanMode: .continuous, simulatedData: "npub1k92qsr95jcumkpu6dffurkvwwycwa2euvx4fthv78ru7gqqz0nrs2ngfwd", shouldVibrateOnSuccess: false) { result in
switch result {
case .success(let success):
handleProfileScan(success.string)
case .failure(let failure):
self.error = failure.localizedDescription
}
}
.scaledToFit()
.frame(maxWidth: 300, maxHeight: 300)
.cornerRadius(10)
.overlay(RoundedRectangle(cornerRadius: 10).stroke(DamusColors.white, lineWidth: 5.0).scaledToFit())
.overlay(RoundedRectangle(cornerRadius: 10).trim(from: 0.0, to: outerTrimEnd).stroke(DamusColors.black, lineWidth: 5.5)
.rotationEffect(.degrees(-90)).scaledToFit())
.shadow(radius: 10)
Spacer()
Spacer()
Button(action: { Button(action: {
selectedTab = 0 selectedTab = 0
}) { }) {
@@ -146,11 +220,65 @@ struct QRCodeView: View {
Text("View QR Code", comment: "Button to switch to view users QR Code") Text("View QR Code", comment: "Button to switch to view users QR Code")
.fontWeight(.semibold) .fontWeight(.semibold)
} }
.frame(maxWidth: .infinity, maxHeight: 12, alignment: .center) .frame( maxWidth: .infinity, maxHeight: 12, alignment: .center)
} }
.buttonStyle(GradientButtonStyle()) .buttonStyle(GradientButtonStyle())
.padding(50) .padding(50)
}, dismiss: dismiss) }
}
func handleProfileScan(_ scanned_str: String) {
guard let result = ProfileScanResult(string: scanned_str) else {
self.error = "Invalid profile QR"
return
}
self.error = nil
guard result != self.scanResult else {
return
}
generator.impactOccurred()
cameraAnimate {
scanResult = result
find_event(state: damus_state, query: .profile(pubkey: result.pubkey)) { res in
guard let res else {
error = "Profile not found"
return
}
switch res {
case .invalid_profile:
error = "Profile was found but was corrupt."
case .profile:
show_profile_after_delay()
case .event:
print("invalid search result")
}
}
}
}
func show_profile_after_delay() {
DispatchQueue.main.asyncAfter(deadline: .now() + animationDuration) {
if let scanResult {
damus_state.nav.push(route: Route.ProfileByKey(pubkey: scanResult.pubkey))
presentationMode.wrappedValue.dismiss()
}
}
}
func cameraAnimate(completion: @escaping () -> Void) {
outerTrimEnd = 0.0
withAnimation(.easeInOut(duration: animationDuration)) {
outerTrimEnd = 1.05 // Set to 1.05 instead of 1.0 since sometimes `completion()` runs before the value reaches 1.0. This ensures the animation is done.
}
completion()
} }
func generateQRCode(pubkey: String) -> UIImage { func generateQRCode(pubkey: String) -> UIImage {
@@ -173,295 +301,6 @@ struct QRCodeView: View {
} }
} }
/// A view that scans for pubkeys/npub QR codes and displays a profile when needed.
///
/// ## Implementation notes:
///
/// - Marked as `fileprivate` since it is a relatively niche view, but can be made public with some adaptation if reuse is needed
/// - The main state is tracked by a single enum, to ensure mutual exclusion of states (only one of the states can be active at a time), and that the info for each state is there when needed both enforced at compile-time
fileprivate struct QRCameraView<Content: View>: View {
// MARK: Input parameters
var damusState: DamusState
/// A custom view to display on the bottom of the camera view
var bottomContent: () -> Content
var dismiss: DismissAction
// MARK: State properties
/// The main state of this view.
@State var scannerState: ScannerState = .scanning {
didSet {
switch (oldValue, scannerState) {
case (.scanning, .scanSuccessful), (.incompatibleQRCodeFound, .scanSuccessful):
generator.impactOccurred() // Haptic feedback upon a successful scan
default:
break
}
}
}
// MARK: Helper properties and objects
let generator = UIImpactFeedbackGenerator(style: .light)
/// A timer that ticks every second.
/// We need this to dismiss the incompatible QR code message automatically once the user is no longer pointing the camera at it
let timer = Timer.publish(every: 1.0, on: .main, in: .common).autoconnect()
/// This is used to create a nice border animation when a scan is successful
///
/// Computed property to simplify state management
var outerTrimEnd: CGFloat {
switch scannerState {
case .scanning, .error, .incompatibleQRCodeFound:
return 0.0
case .scanSuccessful:
return 1.0
}
}
/// A computed binding that indicates if there is an error to be displayed.
///
/// This property is computed based on the main state `scannerState`, and is used to manage the error sheet without adding any extra state variables
var errorBinding: Binding<ScannerError?> {
Binding(
get: {
guard case .error(let error) = scannerState else { return nil }
return error
},
set: { newError in
guard let newError else {
self.scannerState = .scanning
return
}
self.scannerState = .error(newError)
})
}
/// A computed binding that indicates if there is a profile scan result to be displayed
///
/// This property is computed based on the main state `scannerState`, and is used to manage the profile sheet without adding any extra state variables
var profileScanResultBinding: Binding<ProfileScanResult?> {
Binding(
get: {
guard case .scanSuccessful(result: let scanResult) = scannerState else { return nil }
return scanResult
},
set: { newProfileScanResult in
guard let newProfileScanResult else {
self.scannerState = .scanning
return
}
self.scannerState = .scanSuccessful(result: newProfileScanResult)
})
}
// MARK: View layouts
var body: some View {
VStack(alignment: .center) {
Text("Scan a user's pubkey", comment: "Text to prompt scanning a QR code of a user's pubkey to open their profile.")
.padding(.top, 50)
.font(.system(size: 24, weight: .heavy))
.foregroundColor(.white)
Spacer()
CodeScannerView(codeTypes: [.qr], scanMode: .continuous, scanInterval: 1, showViewfinder: true, simulatedData: "npub1k92qsr95jcumkpu6dffurkvwwycwa2euvx4fthv78ru7gqqz0nrs2ngfwd", shouldVibrateOnSuccess: false) { result in
self.handleNewProfileScanInfo(result)
}
.scaledToFit()
.frame(maxWidth: 300, maxHeight: 300)
.cornerRadius(10)
.overlay(RoundedRectangle(cornerRadius: 10).stroke(DamusColors.white, lineWidth: 5.0).scaledToFit())
.overlay(RoundedRectangle(cornerRadius: 10).trim(from: 0.0, to: outerTrimEnd).stroke(DamusColors.black, lineWidth: 5.5)
.rotationEffect(.degrees(-90)).scaledToFit())
.shadow(radius: 10)
Spacer()
self.hintMessage
Spacer()
self.bottomContent()
}
// Show an error sheet if we are on an error state
.sheet(item: self.errorBinding, content: { error in
self.errorSheet(error: error)
})
// Show the profile sheet if we have successfully scanned
.sheet(item: self.profileScanResultBinding, content: { scanResult in
ProfileActionSheetView(damus_state: self.damusState, pubkey: scanResult.pubkey, onNavigate: {
dismiss()
})
.tint(DamusColors.adaptableBlack)
.presentationDetents([.large])
})
// Dismiss an incompatible QR code message automatically after a second or two of pointing it elsewhere.
.onReceive(timer) { _ in
switch self.scannerState {
case .incompatibleQRCodeFound(scannedAt: let date):
if abs(date.timeIntervalSinceNow) > 1.5 {
self.scannerState = .scanning
}
default:
break
}
}
}
var hintMessage: some View {
HStack {
switch self.scannerState {
case .scanning:
Text("Point your camera to a QR code…", comment: "Text on QR code camera view instructing user to point to QR code")
case .incompatibleQRCodeFound:
Text("Sorry, this QR code looks incompatible with Damus. Please try another one.", comment: "Text on QR code camera view telling the user a QR is incompatible")
case .scanSuccessful:
Text("Found profile!", comment: "Text on QR code camera view telling user that profile scan was successful.")
case .error:
Text("Error, please try again", comment: "Text on QR code camera view indicating an error")
}
}
.foregroundColor(.white)
.padding()
}
func errorSheet(error: ScannerError) -> some View {
VStack(spacing: 10) {
Image(systemName: "exclamationmark.circle.fill")
Text("Error", comment: "Headline label for an error sheet on the QR code scanner")
.font(.headline)
Text(error.localizedDescription)
}
.presentationDetents([.medium])
.tint(DamusColors.adaptableBlack)
}
// MARK: Scanning and state management logic
/// A base handler anytime the scanner sends new info,
///
/// Behavior depends on the current state. In some states we completely ignore new scanner info (e.g. when looking at a profile)
/// This function mutates our state
func handleNewProfileScanInfo(_ scanInfo: Result<ScanResult, ScanError>) {
switch scannerState {
case .scanning, .incompatibleQRCodeFound:
withAnimation {
self.scannerState = self.processScanAndComputeNextState(scanInfo)
}
case .scanSuccessful, .error:
return // We don't want new scan results to pop-up while in these states
}
}
/// Processes a QR code scan, and computes the next state to be applied to the view
func processScanAndComputeNextState(_ scanInfo: Result<ScanResult, ScanError>) -> ScannerState {
switch scanInfo {
case .success(let successfulScan):
guard let result = ProfileScanResult(string: successfulScan.string) else {
return .incompatibleQRCodeFound(scannedAt: Date.now)
}
return .scanSuccessful(result: result)
case .failure(let error):
return .error(.scanError(error))
}
}
// MARK: Helper types
/// A custom type for `QRCameraView` to track the state of the scanner.
///
/// This is done to avoid having multiple independent variables to track the state, which increases the chance of state inconsistency.
/// By using this we guarantee at compile-time that we will always be in one state at a time, and that the state is coherent/consistent/clear.
enum ScannerState {
/// Camera is on and actively scanning new QR codes
case scanning
/// Scan and decoding was successful. Show profile.
case scanSuccessful(result: ProfileScanResult)
/// Tell the user they scanned a QR code that is incompatible
case incompatibleQRCodeFound(scannedAt: Date)
/// There was an error. Display a human readable and actionable message
case error(ScannerError)
}
/// Represents an error in this view, to be displayed to the user
///
/// **Implementation notes:**
/// 1. This is identifiable because it that is needed for the error sheet view
/// 2. Currently there is only one error type (`ScanError`), but this is still used to allow us to customize it and add future error types outside the scanner.
enum ScannerError: Error, Identifiable {
case scanError(ScanError)
var localizedDescription: String {
switch self {
case .scanError(let scanError):
switch scanError {
case .badInput:
NSLocalizedString("The camera could not be accessed.", comment: "Camera's bad input error label")
case .badOutput:
NSLocalizedString("The camera was not capable of scanning the requested codes.", comment: "Camera's bad output error label")
case .initError(_):
NSLocalizedString("There was an unexpected error in initializing the camera.", comment: "Camera's initialization error label")
case .permissionDenied:
NSLocalizedString("Camera's permission was denied. You can change this in iOS settings.", comment: "Camera's permission denied error label")
}
}
}
var id: String { return self.localizedDescription }
}
/// A struct that holds results of a profile scan
struct ProfileScanResult: Equatable, Identifiable {
var id: Pubkey { return self.pubkey }
let pubkey: Pubkey
init?(hex: String) {
guard let pk = hex_decode(hex).map({ bytes in Pubkey(Data(bytes)) }) else {
return nil
}
self.pubkey = pk
}
init?(string: String) {
var str = string.trimmingCharacters(in: ["\n", "\t", " "])
guard str.count != 0 else {
return nil
}
if str.hasPrefix("nostr:") {
str.removeFirst("nostr:".count)
}
if let decoded = hex_decode(str),
str.count == 64
{
self.pubkey = Pubkey(Data(decoded))
return
}
if str.starts(with: "npub"),
let b32 = try? bech32_decode(str)
{
self.pubkey = Pubkey(b32.data)
return
}
return nil
}
}
}
// MARK: - Previews
struct QRCodeView_Previews: PreviewProvider { struct QRCodeView_Previews: PreviewProvider {
static var previews: some View { static var previews: some View {
QRCodeView(damus_state: test_damus_state, pubkey: test_note.pubkey) QRCodeView(damus_state: test_damus_state, pubkey: test_note.pubkey)
-1
View File
@@ -5,7 +5,6 @@
// Created by Jericho Hasselbush on 9/29/23. // Created by Jericho Hasselbush on 9/29/23.
// //
import CodeScanner
import SwiftUI import SwiftUI
import VisionKit import VisionKit
-1
View File
@@ -22,7 +22,6 @@ struct ReactionsView: View {
} }
.padding() .padding()
} }
.padding(.bottom, tabHeight)
.navigationBarTitle(NSLocalizedString("Reactions", comment: "Navigation bar title for Reactions view.")) .navigationBarTitle(NSLocalizedString("Reactions", comment: "Navigation bar title for Reactions view."))
.onAppear { .onAppear {
model.subscribe() model.subscribe()
+6 -6
View File
@@ -14,9 +14,9 @@ enum RelayTab: Int, CaseIterable{
var title: String{ var title: String{
switch self { switch self {
case .myRelays: case .myRelays:
return NSLocalizedString("My Relays", comment: "Title of the tab that shows the user's list of their own relays.") return "My relays"
case .recommended: case .recommended:
return NSLocalizedString("Recommended", comment: "Title of the tab that shows the list of relays recommended by Damus.") return "Recommended"
} }
} }
} }
@@ -48,10 +48,10 @@ struct RelayConfigView: View {
NavigationView { NavigationView {
ZStack(alignment: .bottom){ ZStack(alignment: .bottom){
TabView(selection: $selectedTab) { TabView(selection: $selectedTab) {
RelayList(title: RelayTab.myRelays.title, relayList: relays, recommended: false) RelayList(title: "My Relays", relayList: relays, recommended: false)
.tag(0) .tag(0)
RelayList(title: RelayTab.recommended.title, relayList: recommended, recommended: true) RelayList(title: "Recommended", relayList: recommended, recommended: true)
.tag(1) .tag(1)
} }
ZStack{ ZStack{
@@ -83,13 +83,13 @@ struct RelayConfigView: View {
.toolbar { .toolbar {
if state.keypair.privkey != nil && selectedTab == 0 { if state.keypair.privkey != nil && selectedTab == 0 {
if showActionButtons { if showActionButtons {
Button(NSLocalizedString("Done", comment: "Button to leave edit mode for modifying the list of relays.")) { Button("Done") {
withAnimation { withAnimation {
showActionButtons.toggle() showActionButtons.toggle()
} }
} }
} else { } else {
Button(NSLocalizedString("Edit", comment: "Button to enter edit mode for modifying the list of relays.")) { Button("Edit") {
withAnimation { withAnimation {
showActionButtons.toggle() showActionButtons.toggle()
} }
+8 -7
View File
@@ -13,14 +13,15 @@ struct SignalView: View {
var body: some View { var body: some View {
Group { Group {
NavigationLink(value: Route.RelayConfig) { if signal.signal != signal.max_signal {
Text("\(signal.signal)/\(signal.max_signal)", comment: "Fraction of how many of the user's relay servers that are operational.") NavigationLink(value: Route.RelayConfig) {
.font(.callout) Text("\(signal.signal)/\(signal.max_signal)", comment: "Fraction of how many of the user's relay servers that are operational.")
.foregroundColor(.gray) .font(.callout)
.foregroundColor(.gray)
}
} else {
Text("")
} }
.frame(width:50,height:30)
.opacity(signal.signal != signal.max_signal ? 1 : 0)
.disabled(signal.signal == signal.max_signal)
} }
} }
@@ -13,7 +13,6 @@ struct QuoteRepostsView: View {
var body: some View { var body: some View {
TimelineView<AnyView>(events: model.events, loading: $model.loading, damus: damus_state, show_friend_icon: true, filter: ContentFilters.default_filters(damus_state: damus_state).filter(ev:)) TimelineView<AnyView>(events: model.events, loading: $model.loading, damus: damus_state, show_friend_icon: true, filter: ContentFilters.default_filters(damus_state: damus_state).filter(ev:))
.padding(.bottom, tabHeight)
.navigationBarTitle(NSLocalizedString("Quotes", comment: "Navigation bar title for Quote Reposts view.")) .navigationBarTitle(NSLocalizedString("Quotes", comment: "Navigation bar title for Quote Reposts view."))
.onAppear { .onAppear {
model.subscribe() model.subscribe()
-1
View File
@@ -20,7 +20,6 @@ struct RepostsView: View {
} }
.padding() .padding()
} }
.padding(.bottom, tabHeight)
.navigationBarTitle(NSLocalizedString("Reposts", comment: "Navigation bar title for Reposts view.")) .navigationBarTitle(NSLocalizedString("Reposts", comment: "Navigation bar title for Reposts view."))
.onAppear { .onAppear {
model.subscribe() model.subscribe()
-53
View File
@@ -1,53 +0,0 @@
//
// NDBSearchView.swift
// damus
//
// Created by eric on 9/9/24.
//
import SwiftUI
struct NDBSearchView: View {
let damus_state: DamusState
@Binding var results: [NostrEvent]
var body: some View {
ScrollView {
if results.count > 0 {
HStack {
Spacer()
Image("search")
Text("Top hits", comment: "A label indicating that the notes being displayed below it are all top note search results")
Spacer()
}
.padding()
.foregroundColor(.secondary)
LazyVStack {
ForEach(results, id: \.self) { note in
EventView(damus: damus_state, event: note, options: [.truncate_content])
.onTapGesture {
let event = note.get_inner_event(cache: damus_state.events) ?? note
let thread = ThreadModel(event: event, damus_state: damus_state)
damus_state.nav.push(route: Route.Thread(thread: thread))
}
.padding(.horizontal)
ThiccDivider()
}
}
} else if results.count == 0 {
HStack {
Spacer()
Image("search")
Text("No results", comment: "A label indicating that note search resulted in no results")
Spacer()
}
.padding()
.foregroundColor(.secondary)
}
}
}
}
+2 -2
View File
@@ -18,8 +18,8 @@ struct PullDownSearchView: View {
let on_cancel: () -> Void let on_cancel: () -> Void
func do_search(query: String) { func do_search(query: String) {
let limit = 128 let limit = 16
let note_keys = state.ndb.text_search(query: query, limit: limit, order: .newest_first) var note_keys = state.ndb.text_search(query: query, limit: limit, order: .newest_first)
var res = [NostrEvent]() var res = [NostrEvent]()
// TODO: fix duplicate results from search // TODO: fix duplicate results from search
var keyset = Set<NoteKey>() var keyset = Set<NoteKey>()
+1 -1
View File
@@ -91,7 +91,7 @@ struct SearchHomeView: View {
.foregroundColor(.secondary) .foregroundColor(.secondary)
.padding(.top, 20) .padding(.top, 20)
.padding(.horizontal) .padding(.horizontal)
}.padding(.bottom, 50)) })
} }
) )
.refreshable { .refreshable {
+6 -83
View File
@@ -8,7 +8,6 @@
import SwiftUI import SwiftUI
struct MultiSearch { struct MultiSearch {
let text: String
let hashtag: String let hashtag: String
let profiles: [Pubkey] let profiles: [Pubkey]
} }
@@ -44,7 +43,6 @@ enum Search: Identifiable {
struct InnerSearchResults: View { struct InnerSearchResults: View {
let damus_state: DamusState let damus_state: DamusState
let search: Search? let search: Search?
@Binding var results: [NostrEvent]
func ProfileSearchResult(pk: Pubkey) -> some View { func ProfileSearchResult(pk: Pubkey) -> some View {
FollowUserView(target: .pubkey(pk), damus_state: damus_state) FollowUserView(target: .pubkey(pk), damus_state: damus_state)
@@ -53,33 +51,7 @@ struct InnerSearchResults: View {
func HashtagSearch(_ ht: String) -> some View { func HashtagSearch(_ ht: String) -> some View {
let search_model = SearchModel(state: damus_state, search: .filter_hashtag([ht])) let search_model = SearchModel(state: damus_state, search: .filter_hashtag([ht]))
return NavigationLink(value: Route.Search(search: search_model)) { return NavigationLink(value: Route.Search(search: search_model)) {
HStack { Text("Search hashtag: #\(ht)", comment: "Navigation link to search hashtag.")
Text("#\(ht)", comment: "Navigation link to search hashtag.")
}
.padding(.horizontal, 15)
.padding(.vertical, 5)
.background(DamusColors.neutral1)
.cornerRadius(20)
.overlay(
RoundedRectangle(cornerRadius: 20)
.stroke(DamusColors.neutral3, lineWidth: 1)
)
}
}
func TextSearch(_ txt: String) -> some View {
return NavigationLink(value: Route.NDBSearch(results: $results)) {
HStack {
Text(txt)
}
.padding(.horizontal, 15)
.padding(.vertical, 5)
.background(DamusColors.neutral1)
.cornerRadius(20)
.overlay(
RoundedRectangle(cornerRadius: 20)
.stroke(DamusColors.neutral3, lineWidth: 1)
)
} }
} }
@@ -116,13 +88,8 @@ struct InnerSearchResults: View {
case .naddr(let naddr): case .naddr(let naddr):
SearchingEventView(state: damus_state, search_type: .naddr(naddr)) SearchingEventView(state: damus_state, search_type: .naddr(naddr))
case .multi(let multi): case .multi(let multi):
VStack(alignment: .leading) { VStack {
HStack(spacing: 20) { HashtagSearch(multi.hashtag)
HashtagSearch(multi.hashtag)
TextSearch(multi.text)
}
.padding(.bottom, 10)
ProfilesSearch(multi.profiles) ProfilesSearch(multi.profiles)
} }
@@ -137,47 +104,10 @@ struct SearchResultsView: View {
let damus_state: DamusState let damus_state: DamusState
@Binding var search: String @Binding var search: String
@State var result: Search? = nil @State var result: Search? = nil
@State var results: [NostrEvent] = []
let debouncer: Debouncer = Debouncer(interval: 0.25)
func do_search(query: String) {
let limit = 128
var note_keys = damus_state.ndb.text_search(query: query, limit: limit, order: .newest_first)
var res = [NostrEvent]()
// TODO: fix duplicate results from search
var keyset = Set<NoteKey>()
// try reverse because newest first is a bit buggy on partial searches
if note_keys.count == 0 {
// don't touch existing results if there are no new ones
return
}
do {
guard let txn = NdbTxn(ndb: damus_state.ndb) else { return }
for note_key in note_keys {
guard let note = damus_state.ndb.lookup_note_by_key_with_txn(note_key, txn: txn) else {
continue
}
if !keyset.contains(note_key) {
let owned_note = note.to_owned()
res.append(owned_note)
keyset.insert(note_key)
}
}
}
let res_ = res
Task { @MainActor [res_] in
results = res_
}
}
var body: some View { var body: some View {
ScrollView { ScrollView {
InnerSearchResults(damus_state: damus_state, search: result, results: $results) InnerSearchResults(damus_state: damus_state, search: result)
.padding() .padding()
} }
.frame(maxHeight: .infinity) .frame(maxHeight: .infinity)
@@ -189,13 +119,6 @@ struct SearchResultsView: View {
guard let txn = NdbTxn.init(ndb: damus_state.ndb) else { return } guard let txn = NdbTxn.init(ndb: damus_state.ndb) else { return }
self.result = search_for_string(profiles: damus_state.profiles, contacts: damus_state.contacts, search: search, txn: txn) self.result = search_for_string(profiles: damus_state.profiles, contacts: damus_state.contacts, search: search, txn: txn)
} }
.onChange(of: search) { query in
debouncer.debounce {
Task.detached {
do_search(query: query)
}
}
}
} }
} }
@@ -251,7 +174,7 @@ func search_for_string<Y>(profiles: Profiles, contacts: Contacts, search new: St
return .naddr(naddr) return .naddr(naddr)
} }
let multisearch = MultiSearch(text: new, hashtag: make_hashtagable(searchQuery), profiles: search_profiles(profiles: profiles, contacts: contacts, search: new, txn: txn)) let multisearch = MultiSearch(hashtag: make_hashtagable(searchQuery), profiles: search_profiles(profiles: profiles, contacts: contacts, search: new, txn: txn))
return .multi(multisearch) return .multi(multisearch)
} }
@@ -285,7 +208,7 @@ func search_profiles<Y>(profiles: Profiles, contacts: Contacts, search: String,
return [pk] return [pk]
} }
return profiles.search(search, limit: 128, txn: txn).sorted { a, b in return profiles.search(search, limit: 10, txn: txn).sorted { a, b in
let aFriendTypePriority = get_friend_type(contacts: contacts, pubkey: a)?.priority ?? 0 let aFriendTypePriority = get_friend_type(contacts: contacts, pubkey: a)?.priority ?? 0
let bFriendTypePriority = get_friend_type(contacts: contacts, pubkey: b)?.priority ?? 0 let bFriendTypePriority = get_friend_type(contacts: contacts, pubkey: b)?.priority ?? 0
@@ -108,7 +108,6 @@ struct AppearanceSettingsView: View {
Section( Section(
header: Text("Profiles", comment: "Section title for profile view configuration."), header: Text("Profiles", comment: "Section title for profile view configuration."),
footer: Text("Profile action sheets allow you to follow, zap, or DM profiles more quickly without having to view their full profile", comment: "Section footer clarifying what the profile action sheet feature does") footer: Text("Profile action sheets allow you to follow, zap, or DM profiles more quickly without having to view their full profile", comment: "Section footer clarifying what the profile action sheet feature does")
.padding(.bottom, tabHeight + getSafeAreaBottom())
) { ) {
Toggle(NSLocalizedString("Show profile action sheets", comment: "Setting to show profile action sheets when clicking on a user's profile picture"), isOn: $settings.show_profile_action_sheet_on_pfp_click) Toggle(NSLocalizedString("Show profile action sheets", comment: "Setting to show profile action sheets when clicking on a user's profile picture"), isOn: $settings.show_profile_action_sheet_on_pfp_click)
.toggleStyle(.switch) .toggleStyle(.switch)
@@ -177,10 +177,7 @@ struct NotificationSettingsView: View {
.toggleStyle(.switch) .toggleStyle(.switch)
} }
Section( Section(header: Text("Notification Dots", comment: "Section header for notification indicator dot settings")) {
header: Text("Notification Dots", comment: "Section header for notification indicator dot settings"),
footer: Text("").padding(.bottom, tabHeight + getSafeAreaBottom())
) {
Toggle(NSLocalizedString("Zaps", comment: "Setting to enable Zap Local Notification"), isOn: indicator_binding(.zaps)) Toggle(NSLocalizedString("Zaps", comment: "Setting to enable Zap Local Notification"), isOn: indicator_binding(.zaps))
.toggleStyle(.switch) .toggleStyle(.switch)
Toggle(NSLocalizedString("Mentions", comment: "Setting to enable Mention Local Notification"), isOn: indicator_binding(.mentions)) Toggle(NSLocalizedString("Mentions", comment: "Setting to enable Mention Local Notification"), isOn: indicator_binding(.mentions))
+12 -12
View File
@@ -56,21 +56,21 @@ struct SetupView: View {
.frame(minWidth: 300, maxWidth: .infinity, maxHeight: 12, alignment: .center) .frame(minWidth: 300, maxWidth: .infinity, maxHeight: 12, alignment: .center)
} }
.buttonStyle(GradientButtonStyle()) .buttonStyle(GradientButtonStyle())
.accessibilityIdentifier(AppAccessibilityIdentifiers.sign_in_option_button.rawValue)
.padding() .padding()
HStack(spacing: 0) {
Text("By continuing you agree to our ")
.font(.subheadline)
.foregroundColor(DamusColors.neutral6)
Button(action: { Button(action: {
navigationCoordinator.push(route: Route.EULA) navigationCoordinator.push(route: Route.EULA)
}, label: { }, label: {
HStack { Text("EULA", comment: "End User License Agreement")
Text("By continuing, you agree to our EULA", comment: "Disclaimer to user that they are agreeing to the End User License Agreement if they create an account or sign in.")
.font(.subheadline) .font(.subheadline)
.foregroundColor(DamusColors.neutral6) })
.padding(.vertical, 5)
Image(systemName: "arrow.forward") }
}
})
.padding(.vertical, 5)
.padding(.bottom) .padding(.bottom)
} }
} }
+102 -88
View File
@@ -11,14 +11,23 @@ import SwiftUI
struct SideMenuView: View { struct SideMenuView: View {
let damus_state: DamusState let damus_state: DamusState
@Binding var isSidebarVisible: Bool @Binding var isSidebarVisible: Bool
@Binding var selected: Timeline
@State var confirm_logout: Bool = false @State var confirm_logout: Bool = false
@State private var showQRCode = false @State private var showQRCode = false
@Environment(\.colorScheme) var colorScheme
var sideBarWidth = min(UIScreen.main.bounds.size.width * 0.65, 400.0) var sideBarWidth = min(UIScreen.main.bounds.size.width * 0.65, 400.0)
let verticalSpacing: CGFloat = 25 let verticalSpacing: CGFloat = 20
let padding: CGFloat = 30 let padding: CGFloat = 30
func fillColor() -> Color {
colorScheme == .light ? DamusColors.white : DamusColors.black
}
func textColor() -> Color {
colorScheme == .light ? DamusColors.black : DamusColors.white
}
var body: some View { var body: some View {
ZStack { ZStack {
GeometryReader { _ in GeometryReader { _ in
@@ -40,7 +49,6 @@ struct SideMenuView: View {
NavigationLink(value: Route.Profile(profile: profile_model, followers: followers)) { NavigationLink(value: Route.Profile(profile: profile_model, followers: followers)) {
navLabel(title: NSLocalizedString("Profile", comment: "Sidebar menu label for Profile view."), img: "user") navLabel(title: NSLocalizedString("Profile", comment: "Sidebar menu label for Profile view."), img: "user")
} }
.accessibilityIdentifier(AppAccessibilityIdentifiers.side_menu_profile_button.rawValue)
NavigationLink(value: Route.Wallet(wallet: damus_state.wallet)) { NavigationLink(value: Route.Wallet(wallet: damus_state.wallet)) {
navLabel(title: NSLocalizedString("Wallet", comment: "Sidebar menu label for Wallet view."), img: "wallet") navLabel(title: NSLocalizedString("Wallet", comment: "Sidebar menu label for Wallet view."), img: "wallet")
@@ -48,11 +56,11 @@ struct SideMenuView: View {
if damus_state.purple.enable_purple { if damus_state.purple.enable_purple {
NavigationLink(destination: DamusPurpleView(damus_state: damus_state)) { NavigationLink(destination: DamusPurpleView(damus_state: damus_state)) {
HStack(spacing: 23) { HStack(spacing: 13) {
Image("nostr-hashtag") Image("nostr-hashtag")
Text("Purple") Text("Purple")
.foregroundColor(DamusColors.purple) .foregroundColor(DamusColors.purple)
.font(.title2.weight(.semibold)) .font(.title2.weight(.bold))
} }
.frame(maxWidth: .infinity, alignment: .leading) .frame(maxWidth: .infinity, alignment: .leading)
} }
@@ -71,22 +79,12 @@ struct SideMenuView: View {
} }
Link(destination: URL(string: "https://store.damus.io/?ref=damus_ios_app")!) { Link(destination: URL(string: "https://store.damus.io/?ref=damus_ios_app")!) {
navLabel(title: NSLocalizedString("Merch", comment: "Sidebar menu label for merch store link."), img: "shop") navLabel(title: NSLocalizedString("Merch", comment: "Sidebar menu label for merch store link."), img: "basket")
} }
NavigationLink(value: Route.Config) { NavigationLink(value: Route.Config) {
navLabel(title: NSLocalizedString("Settings", comment: "Sidebar menu label for accessing the app settings"), img: "settings") navLabel(title: NSLocalizedString("Settings", comment: "Sidebar menu label for accessing the app settings"), img: "settings")
} }
Button(action: {
if damus_state.keypair.privkey == nil {
logout(damus_state)
} else {
confirm_logout = true
}
}, label: {
navLabel(title: NSLocalizedString("Logout", comment: "Sidebar menu label to sign out of the account."), img: "logout")
})
} }
} }
@@ -101,68 +99,38 @@ struct SideMenuView: View {
display_name = profile?.display_name display_name = profile?.display_name
} }
return VStack(alignment: .leading) { return VStack(alignment: .leading, spacing: verticalSpacing) {
HStack(spacing: 10) { HStack {
ProfilePicView(pubkey: damus_state.pubkey, size: 60, highlight: .none, profiles: damus_state.profiles, disable_animation: damus_state.settings.disable_animation)
ProfilePicView(pubkey: damus_state.pubkey, size: 50, highlight: .none, profiles: damus_state.profiles, disable_animation: damus_state.settings.disable_animation)
VStack(alignment: .leading) {
Spacer()
if let display_name {
Button(action: { Text(display_name)
present_sheet(.user_status) .foregroundColor(textColor())
isSidebarVisible = false .font(.title)
}, label: { .lineLimit(1)
Image("add-reaction") }
.resizable() if let name {
.frame(width: 25, height: 25) Text("@" + name)
.padding(5)
.foregroundColor(DamusColors.adaptableBlack)
.background {
Circle()
.foregroundColor(DamusColors.neutral3)
}
})
Button(action: {
showQRCode.toggle()
isSidebarVisible = false
}, label: {
Image("qr-code")
.resizable()
.frame(width: 25, height: 25)
.padding(5)
.foregroundColor(DamusColors.adaptableBlack)
.background {
Circle()
.foregroundColor(DamusColors.neutral3)
}
}).damus_full_screen_cover($showQRCode, damus_state: damus_state) {
QRCodeView(damus_state: damus_state, pubkey: damus_state.pubkey)
}
}
VStack(alignment: .leading) {
if let display_name {
Text(display_name)
.font(.title2.weight(.bold))
.foregroundColor(DamusColors.adaptableBlack)
.frame(maxWidth: .infinity, alignment: .leading)
.dynamicTypeSize(.xSmall)
.lineLimit(1)
}
if let name {
if !name.isEmpty {
Text(verbatim: "@" + name)
.foregroundColor(DamusColors.mediumGrey) .foregroundColor(DamusColors.mediumGrey)
.font(.body) .font(.body)
.lineLimit(1) .lineLimit(1)
} }
} }
PubkeyView(pubkey: damus_state.pubkey, sidemenu: true)
.pubkey_context_menu(pubkey: damus_state.pubkey)
} }
navLabel(title: NSLocalizedString("Set Status", comment: "Sidebar menu label to set user status"), img: "add-reaction")
.font(.title2)
.foregroundColor(textColor())
.frame(maxWidth: .infinity, alignment: .leading)
.dynamicTypeSize(.xSmall)
.onTapGesture {
present_sheet(.user_status)
}
UserStatusView(status: damus_state.profiles.profile_data(damus_state.pubkey).status, show_general: true, show_music: true)
.dynamicTypeSize(.xSmall)
} }
} }
@@ -172,31 +140,68 @@ struct SideMenuView: View {
let profile_model = ProfileModel(pubkey: damus_state.pubkey, damus: damus_state) let profile_model = ProfileModel(pubkey: damus_state.pubkey, damus: damus_state)
NavigationLink(value: Route.Profile(profile: profile_model, followers: followers), label: { NavigationLink(value: Route.Profile(profile: profile_model, followers: followers), label: {
TopProfile TopProfile
.padding(.bottom, verticalSpacing) .padding(.bottom, verticalSpacing)
}) })
.simultaneousGesture(TapGesture().onEnded {
isSidebarVisible = false Divider()
})
ScrollView { ScrollView {
SidemenuItems(profile_model: profile_model, followers: followers) SidemenuItems(profile_model: profile_model, followers: followers)
.simultaneousGesture(TapGesture().onEnded { .labelStyle(SideMenuLabelStyle())
isSidebarVisible = false .padding([.top, .bottom], verticalSpacing)
})
} }
.scrollIndicators(.hidden)
} }
} }
var content: some View { var content: some View {
HStack(alignment: .top) { HStack(alignment: .top) {
ZStack(alignment: .top) { ZStack(alignment: .top) {
DamusColors.adaptableWhite fillColor()
.ignoresSafeArea() .ignoresSafeArea()
MainSidemenu VStack(alignment: .leading, spacing: 0) {
.padding([.leading, .trailing], padding) MainSidemenu
.simultaneousGesture(TapGesture().onEnded {
isSidebarVisible = false
})
Divider()
HStack() {
Button(action: {
//ConfigView(state: damus_state)
if damus_state.keypair.privkey == nil {
logout(damus_state)
} else {
confirm_logout = true
}
}, label: {
Label(NSLocalizedString("Sign out", comment: "Sidebar menu label to sign out of the account."), image: "logout")
.font(.title3)
.foregroundColor(textColor())
.frame(maxWidth: .infinity, alignment: .leading)
.dynamicTypeSize(.xSmall)
})
Spacer()
Button(action: {
showQRCode.toggle()
}, label: {
Image("qr-code")
.font(.title)
.foregroundColor(textColor())
.dynamicTypeSize(.xSmall)
}).fullScreenCover(isPresented: $showQRCode) {
QRCodeView(damus_state: damus_state, pubkey: damus_state.pubkey)
}
}
.padding(.top, verticalSpacing)
}
.padding(.top, -(padding / 2.0))
.padding([.leading, .trailing, .bottom], padding)
} }
.frame(width: sideBarWidth) .frame(width: sideBarWidth)
.offset(x: isSidebarVisible ? 0 : -(sideBarWidth + padding)) .offset(x: isSidebarVisible ? 0 : -(sideBarWidth + padding))
@@ -217,17 +222,26 @@ struct SideMenuView: View {
} }
func navLabel(title: String, img: String) -> some View { func navLabel(title: String, img: String) -> some View {
HStack(spacing: 20) { HStack {
Image(img) Image(img)
.tint(DamusColors.adaptableBlack) .tint(DamusColors.adaptableBlack)
Text(title) Text(title)
.font(.title2.weight(.semibold)) .font(.title2)
.foregroundColor(DamusColors.adaptableBlack) .foregroundColor(textColor())
.frame(maxWidth: .infinity, alignment: .leading) .frame(maxWidth: .infinity, alignment: .leading)
.dynamicTypeSize(.xSmall) .dynamicTypeSize(.xSmall)
.minimumScaleFactor(0.5) }
.lineLimit(1) }
struct SideMenuLabelStyle: LabelStyle {
func makeBody(configuration: Configuration) -> some View {
HStack(alignment: .center, spacing: 8) {
configuration.icon
.frame(width: 24, height: 24)
.aspectRatio(contentMode: .fit)
configuration.title
}
} }
} }
} }
@@ -235,6 +249,6 @@ struct SideMenuView: View {
struct Previews_SideMenuView_Previews: PreviewProvider { struct Previews_SideMenuView_Previews: PreviewProvider {
static var previews: some View { static var previews: some View {
let ds = test_damus_state let ds = test_damus_state
SideMenuView(damus_state: ds, isSidebarVisible: .constant(true), selected: .constant(.home)) SideMenuView(damus_state: ds, isSidebarVisible: .constant(true))
} }
} }
+19 -133
View File
@@ -39,7 +39,6 @@ struct SuggestedHashtagsView: View {
.sorted(by: { a, b in .sorted(by: { a, b in
a.count > b.count a.count > b.count
}) })
SuggestedHashtagsView.lastRefresh_hashtags = all_items // Collecting recent hash-tag data from Search-page
guard let item_limit else { guard let item_limit else {
return all_items return all_items
} }
@@ -47,55 +46,10 @@ struct SuggestedHashtagsView: View {
} }
} }
init(damus_state: DamusState, suggested_hashtags: [String]? = nil, max_items item_limit: Int? = nil, events: EventHolder) {
static var lastRefresh_hashtags: [HashtagWithUserCount] = [] // Holds hash-tag data for PostView
var isFromPostView: Bool
var queryHashTag: String
var filteredSuggestedHashtags: [HashtagWithUserCount] {
let val = SuggestedHashtagsView.lastRefresh_hashtags.filter {$0.hashtag.hasPrefix(returnFirstWordOnly(hashTag: queryHashTag))}
if val.isEmpty {
if SuggestedHashtagsView.lastRefresh_hashtags.isEmpty {
// This is special case when user goes directly to PostView without opening Search-page previously.
var val = hashtags_with_count_to_display // retrieves default hash-tage values
// if not-found, put query hash tag at top
val.insert(HashtagWithUserCount(hashtag: returnFirstWordOnly(hashTag: queryHashTag), count: 0), at: 0)
return val
} else {
// if not-found, put query hash tag at top
var val = SuggestedHashtagsView.lastRefresh_hashtags
val.insert(HashtagWithUserCount(hashtag: returnFirstWordOnly(hashTag: queryHashTag), count: 0), at: 0)
return val
}
} else {
return val
}
}
@Binding var focusWordAttributes: (String?, NSRange?)
@Binding var newCursorIndex: Int?
@Binding var post: NSMutableAttributedString
@EnvironmentObject var tagModel: TagModel
init(damus_state: DamusState,
suggested_hashtags: [String]? = nil,
max_items item_limit: Int? = nil,
events: EventHolder,
isFromPostView: Bool = false,
queryHashTag: String = "",
focusWordAttributes: Binding<(String?, NSRange?)> = .constant((nil, nil)),
newCursorIndex: Binding<Int?> = .constant(nil),
post: Binding<NSMutableAttributedString> = .constant(NSMutableAttributedString(string: ""))) {
self.damus_state = damus_state self.damus_state = damus_state
self.suggested_hashtags = suggested_hashtags ?? DEFAULT_SUGGESTED_HASHTAGS self.suggested_hashtags = suggested_hashtags ?? DEFAULT_SUGGESTED_HASHTAGS
self.item_limit = item_limit self.item_limit = item_limit
self.isFromPostView = isFromPostView
self.queryHashTag = queryHashTag
self._focusWordAttributes = focusWordAttributes
self._newCursorIndex = newCursorIndex
self._post = post
_events = StateObject.init(wrappedValue: events) _events = StateObject.init(wrappedValue: events)
} }
@@ -105,43 +59,24 @@ struct SuggestedHashtagsView: View {
Image(systemName: "sparkles") Image(systemName: "sparkles")
Text("Suggested hashtags", comment: "A label indicating that the items below it are suggested hashtags") Text("Suggested hashtags", comment: "A label indicating that the items below it are suggested hashtags")
Spacer() Spacer()
// Don't show suggestion expand/contract button when user is in PostView Button(action: {
if !isFromPostView { withAnimation(.easeOut(duration: 0.2)) {
Button(action: { show_suggested_hashtags.toggle()
withAnimation(.easeOut(duration: 0.2)) { }
show_suggested_hashtags.toggle() }) {
} if show_suggested_hashtags {
}) { Image(systemName: "rectangle.compress.vertical")
if show_suggested_hashtags { .foregroundStyle(PinkGradient)
Image(systemName: "rectangle.compress.vertical") } else {
.foregroundStyle(PinkGradient) Image(systemName: "rectangle.expand.vertical")
} else { .foregroundStyle(PinkGradient)
Image(systemName: "rectangle.expand.vertical")
.foregroundStyle(PinkGradient)
}
} }
} }
} }
.foregroundColor(.secondary) .foregroundColor(.secondary)
.padding(.vertical, 10) .padding(.vertical, 10)
if isFromPostView { if show_suggested_hashtags {
ScrollView {
LazyVStack {
ForEach(filteredSuggestedHashtags,
id: \.self) { hashtag_with_count in
SuggestedHashtagView(damus_state: damus_state,
hashtag: hashtag_with_count.hashtag,
count: hashtag_with_count.count,
isFromPostView: true,
focusWordAttributes: $focusWordAttributes,
newCursorIndex: $newCursorIndex,
post: $post)
.environmentObject(tagModel)
}
}
}
} else if show_suggested_hashtags {
ForEach(hashtags_with_count_to_display, ForEach(hashtags_with_count_to_display,
id: \.self) { hashtag_with_count in id: \.self) { hashtag_with_count in
SuggestedHashtagView(damus_state: damus_state, hashtag: hashtag_with_count.hashtag, count: hashtag_with_count.count) SuggestedHashtagView(damus_state: damus_state, hashtag: hashtag_with_count.hashtag, count: hashtag_with_count.count)
@@ -156,26 +91,10 @@ struct SuggestedHashtagsView: View {
let hashtag: String let hashtag: String
let count: Int let count: Int
let isFromPostView: Bool init(damus_state: DamusState, hashtag: String, count: Int) {
@Binding var focusWordAttributes: (String?, NSRange?)
@Binding var newCursorIndex: Int?
@Binding var post: NSMutableAttributedString
@EnvironmentObject var tagModel: TagModel
init(damus_state: DamusState,
hashtag: String,
count: Int,
isFromPostView: Bool = false,
focusWordAttributes: Binding<(String?, NSRange?)> = .constant((nil, nil)),
newCursorIndex: Binding<Int?> = .constant(nil),
post: Binding<NSMutableAttributedString> = .constant(NSMutableAttributedString(string: ""))) {
self.damus_state = damus_state self.damus_state = damus_state
self.hashtag = hashtag self.hashtag = hashtag
self.count = count self.count = count
self.isFromPostView = isFromPostView
self._focusWordAttributes = focusWordAttributes
self._newCursorIndex = newCursorIndex
self._post = post
} }
var body: some View { var body: some View {
@@ -186,48 +105,18 @@ struct SuggestedHashtagsView: View {
Text(verbatim: "#\(hashtag)") Text(verbatim: "#\(hashtag)")
.bold() .bold()
// Don't show user-talking label from PostView when the count is 0 let pluralizedString = pluralizedString(key: "users_talking_about_it", count: self.count)
if isFromPostView { Text(pluralizedString)
if count != 0 { .foregroundStyle(.secondary)
let pluralizedString = pluralizedString(key: "users_talking_about_it", count: self.count)
Text(pluralizedString)
.foregroundStyle(.secondary)
}
} else {
let pluralizedString = pluralizedString(key: "users_talking_about_it", count: self.count)
Text(pluralizedString)
.foregroundStyle(.secondary)
}
} }
Spacer() Spacer()
} }
.contentShape(Rectangle()) // make the entire row/rectangle tappable
.onTapGesture { .onTapGesture {
if isFromPostView { let search_model = SearchModel(state: damus_state, search: NostrFilter.init(hashtag: [hashtag]))
let hashTag = NSMutableAttributedString(string: "#\(returnFirstWordOnly(hashTag: hashtag))", damus_state.nav.push(route: Route.Search(search: search_model))
attributes: [
NSAttributedString.Key.foregroundColor: UIColor.black,
NSAttributedString.Key.link: "#\(hashtag)"
])
appendHashTag(withTag: hashTag)
} else {
let search_model = SearchModel(state: damus_state, search: NostrFilter.init(hashtag: [hashtag]))
damus_state.nav.push(route: Route.Search(search: search_model))
}
} }
} }
// Current working-code similar to UserSearch/appendUserTag
private func appendHashTag(withTag tag: NSMutableAttributedString) {
guard let wordRange = focusWordAttributes.1 else { return }
let appended = append_user_tag(tag: tag, post: post, word_range: wordRange)
self.post = appended.post
// adjust cursor position appropriately: ('diff' used in TextViewWrapper / updateUIView after below update of 'post')
tagModel.diff = appended.tag.length - wordRange.length
focusWordAttributes = (nil, nil)
newCursorIndex = wordRange.location + appended.tag.length
}
} }
func users_talking_about(hashtag: Hashtag) -> Int { func users_talking_about(hashtag: Hashtag) -> Int {
@@ -258,6 +147,3 @@ struct SuggestedHashtagsView_Previews: PreviewProvider {
} }
} }
fileprivate func returnFirstWordOnly(hashTag: String) -> String {
return hashTag.components(separatedBy: " ").first?.lowercased() ?? ""
}
+2 -57
View File
@@ -12,16 +12,13 @@ struct TextViewWrapper: UIViewRepresentable {
@EnvironmentObject var tagModel: TagModel @EnvironmentObject var tagModel: TagModel
@Binding var textHeight: CGFloat? @Binding var textHeight: CGFloat?
let initialTextSuffix: String? let initialTextSuffix: String?
@Binding var imagePastedFromPasteboard: PreUploadedMedia?
@Binding var imageUploadConfirmPasteboard: Bool
let cursorIndex: Int? let cursorIndex: Int?
var getFocusWordForMention: ((String?, NSRange?) -> Void)? = nil var getFocusWordForMention: ((String?, NSRange?) -> Void)? = nil
let updateCursorPosition: ((Int) -> Void) let updateCursorPosition: ((Int) -> Void)
func makeUIView(context: Context) -> UITextView { func makeUIView(context: Context) -> UITextView {
let textView = CustomPostTextView(imagePastedFromPasteboard: $imagePastedFromPasteboard, let textView = UITextView()
imageUploadConfirm: $imageUploadConfirmPasteboard)
textView.backgroundColor = UIColor(DamusColors.adaptableWhite) textView.backgroundColor = UIColor(DamusColors.adaptableWhite)
textView.delegate = context.coordinator textView.delegate = context.coordinator
@@ -93,7 +90,7 @@ struct TextViewWrapper: UIViewRepresentable {
let updateCursorPosition: ((Int) -> Void) let updateCursorPosition: ((Int) -> Void)
let initialTextSuffix: String? let initialTextSuffix: String?
var initialTextSuffixWasAdded: Bool = false var initialTextSuffixWasAdded: Bool = false
static let ESCAPE_SEQUENCES = ["\n", "@", " ", ", ", ". ", "! ", "? ", "; ", "#"] static let ESCAPE_SEQUENCES = ["\n", "@", " ", ", ", ". ", "! ", "? ", "; "]
init(attributedText: Binding<NSMutableAttributedString>, init(attributedText: Binding<NSMutableAttributedString>,
getFocusWordForMention: ((String?, NSRange?) -> Void)?, getFocusWordForMention: ((String?, NSRange?) -> Void)?,
@@ -243,55 +240,3 @@ struct TextViewWrapper: UIViewRepresentable {
} }
} }
class CustomPostTextView: UITextView {
@Binding var imagePastedFromPasteboard: PreUploadedMedia?
@Binding var imageUploadConfirm: Bool
// Custom initializer
init(imagePastedFromPasteboard: Binding<PreUploadedMedia?>, imageUploadConfirm: Binding<Bool>) {
self._imagePastedFromPasteboard = imagePastedFromPasteboard
self._imageUploadConfirm = imageUploadConfirm
super.init(frame: .zero, textContainer: nil)
}
required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") }
// Override canPerformAction to enable image pasting
override func canPerformAction(_ action: Selector, withSender sender: Any?) -> Bool {
if action == #selector(UIResponderStandardEditActions.paste(_:)),
UIPasteboard.general.image != nil {
return true // Show `Paste` option while long-pressing if there is an image present in the clipboard
}
return super.canPerformAction(action, withSender: sender) // Default behavior for other actions
}
// Override paste to handle image pasting
override func paste(_ sender: Any?) {
let pasteboard = UIPasteboard.general
if let data = pasteboard.data(forPasteboardType: Constants.GIF_IMAGE_TYPE),
let url = saveGIFToTemporaryDirectory(data) {
imagePastedFromPasteboard = PreUploadedMedia.unprocessed_image(url)
imageUploadConfirm = true
} else if let image = pasteboard.image {
// handle .png, .jpeg files here
imagePastedFromPasteboard = PreUploadedMedia.uiimage(image)
// Show alert view in PostView for Confirming upload
imageUploadConfirm = true
} else {
// fall back to default paste behavior if no image or gif file found
super.paste(sender)
}
}
private func saveGIFToTemporaryDirectory(_ data: Data) -> URL? {
let tempDirectory = FileManager.default.temporaryDirectory
let gifURL = tempDirectory.appendingPathComponent("pasted_image.gif")
do {
try data.write(to: gifURL)
return gifURL
} catch {
return nil
}
}
}
+14 -70
View File
@@ -16,14 +16,11 @@ struct PostingTimelineView: View {
@State var initialOffset: CGFloat? @State var initialOffset: CGFloat?
@State var offset: CGFloat? @State var offset: CGFloat?
@State var showSearch: Bool = true @State var showSearch: Bool = true
@Binding var isSideBarOpened: Bool
@Binding var active_sheet: Sheets? @Binding var active_sheet: Sheets?
@FocusState private var isSearchFocused: Bool @FocusState private var isSearchFocused: Bool
@State private var contentOffset: CGFloat = 0 @State private var contentOffset: CGFloat = 0
@State private var indicatorWidth: CGFloat = 0 @State private var indicatorWidth: CGFloat = 0
@State private var indicatorPosition: CGFloat = 0 @State private var indicatorPosition: CGFloat = 0
@State var headerHeight: CGFloat = 0
@Binding var headerOffset: CGFloat
@SceneStorage("PostingTimelineView.filter_state") var filter_state : FilterState = .posts_and_replies @SceneStorage("PostingTimelineView.filter_state") var filter_state : FilterState = .posts_and_replies
var mystery: some View { var mystery: some View {
@@ -38,56 +35,8 @@ struct PostingTimelineView: View {
} }
func contentTimelineView(filter: (@escaping (NostrEvent) -> Bool)) -> some View { func contentTimelineView(filter: (@escaping (NostrEvent) -> Bool)) -> some View {
TimelineView<AnyView>(events: home.events, loading: .constant(false), headerHeight: $headerHeight, headerOffset: $headerOffset, damus: damus_state, show_friend_icon: false, filter: filter) TimelineView(events: home.events, loading: .constant(false), damus: damus_state, show_friend_icon: false, filter: filter) {
} PullDownSearchView(state: damus_state, on_cancel: {})
func HeaderView()->some View {
VStack {
VStack(spacing: 0) {
// This is needed for the Dynamic Island
HStack {}
.frame(height: getSafeAreaTop())
HStack(alignment: .top) {
TopbarSideMenuButton(damus_state: damus_state, isSideBarOpened: $isSideBarOpened)
Spacer()
Image("damus-home")
.resizable()
.frame(width:30,height:30)
.shadow(color: DamusColors.purple, radius: 2)
.opacity(isSideBarOpened ? 0 : 1)
.animation(isSideBarOpened ? .none : .default, value: isSideBarOpened)
.onTapGesture {
isSideBarOpened.toggle()
}
.padding(.leading)
Spacer()
HStack(alignment: .center) {
SignalView(state: damus_state, signal: home.signal)
}
}
.frame(maxWidth: .infinity, alignment: .trailing)
}
.padding(.horizontal, 20)
VStack(spacing: 0) {
CustomPicker(tabs: [
(NSLocalizedString("Notes", comment: "Label for filter for seeing only notes (instead of notes and replies)."), FilterState.posts),
(NSLocalizedString("Notes & Replies", comment: "Label for filter for seeing notes and replies (instead of only notes)."), FilterState.posts_and_replies)
],
selection: $filter_state)
Divider()
.frame(height: 1)
}
}
.background {
DamusColors.adaptableWhite
.ignoresSafeArea()
} }
} }
@@ -111,26 +60,21 @@ struct PostingTimelineView: View {
PostButtonContainer(is_left_handed: damus_state.settings.left_handed) { PostButtonContainer(is_left_handed: damus_state.settings.left_handed) {
self.active_sheet = .post(.posting(.none)) self.active_sheet = .post(.posting(.none))
} }
.padding(.bottom, tabHeight + getSafeAreaBottom())
.opacity(0.35 + abs(1.25 - (abs(headerOffset/100.0))))
} }
} }
} }
.overlay(alignment: .top) { .safeAreaInset(edge: .top, spacing: 0) {
HeaderView() VStack(spacing: 0) {
.anchorPreference(key: HeaderBoundsKey.self, value: .bounds){$0} CustomPicker(tabs: [
.overlayPreferenceValue(HeaderBoundsKey.self) { value in (NSLocalizedString("Notes", comment: "Label for filter for seeing only notes (instead of notes and replies)."), FilterState.posts),
GeometryReader{ proxy in (NSLocalizedString("Notes & Replies", comment: "Label for filter for seeing notes and replies (instead of only notes)."), FilterState.posts_and_replies)
if let anchor = value{ ],
Color.clear selection: $filter_state)
.onAppear {
headerHeight = proxy[anchor].height Divider()
} .frame(height: 1)
} }
} .background(DamusColors.adaptableWhite)
}
.offset(y: -headerOffset < headerHeight ? headerOffset : (headerOffset < 0 ? headerOffset : 0))
.opacity(1.0 - (abs(headerOffset/100.0)))
} }
} }
} }
+7 -49
View File
@@ -10,11 +10,6 @@ import SwiftUI
struct TimelineView<Content: View>: View { struct TimelineView<Content: View>: View {
@ObservedObject var events: EventHolder @ObservedObject var events: EventHolder
@Binding var loading: Bool @Binding var loading: Bool
@Binding var headerHeight: CGFloat
@Binding var headerOffset: CGFloat
@State var shiftOffset: CGFloat = 0
@State var lastHeaderOffset: CGFloat = 0
@State var direction: SwipeDirection = .none
let damus: DamusState let damus: DamusState
let show_friend_icon: Bool let show_friend_icon: Bool
@@ -22,23 +17,9 @@ struct TimelineView<Content: View>: View {
let content: Content? let content: Content?
let apply_mute_rules: Bool let apply_mute_rules: Bool
init(events: EventHolder, loading: Binding<Bool>, headerHeight: Binding<CGFloat>, headerOffset: Binding<CGFloat>, damus: DamusState, show_friend_icon: Bool, filter: @escaping (NostrEvent) -> Bool, apply_mute_rules: Bool = true, content: (() -> Content)? = nil) {
self.events = events
self._loading = loading
self._headerHeight = headerHeight
self._headerOffset = headerOffset
self.damus = damus
self.show_friend_icon = show_friend_icon
self.filter = filter
self.apply_mute_rules = apply_mute_rules
self.content = content?()
}
init(events: EventHolder, loading: Binding<Bool>, damus: DamusState, show_friend_icon: Bool, filter: @escaping (NostrEvent) -> Bool, apply_mute_rules: Bool = true, content: (() -> Content)? = nil) { init(events: EventHolder, loading: Binding<Bool>, damus: DamusState, show_friend_icon: Bool, filter: @escaping (NostrEvent) -> Bool, apply_mute_rules: Bool = true, content: (() -> Content)? = nil) {
self.events = events self.events = events
self._loading = loading self._loading = loading
self._headerHeight = .constant(0.0)
self._headerOffset = .constant(0.0)
self.damus = damus self.damus = damus
self.show_friend_icon = show_friend_icon self.show_friend_icon = show_friend_icon
self.filter = filter self.filter = filter
@@ -57,43 +38,20 @@ struct TimelineView<Content: View>: View {
content content
} }
Color.clear Color.white.opacity(0)
.id("startblock") .id("startblock")
.frame(height: 0) .frame(height: 1)
InnerTimelineView(events: events, damus: damus, filter: loading ? { _ in true } : filter, apply_mute_rules: self.apply_mute_rules) InnerTimelineView(events: events, damus: damus, filter: loading ? { _ in true } : filter, apply_mute_rules: self.apply_mute_rules)
.redacted(reason: loading ? .placeholder : []) .redacted(reason: loading ? .placeholder : [])
.shimmer(loading) .shimmer(loading)
.disabled(loading) .disabled(loading)
.padding(.top, headerHeight - getSafeAreaTop()) .background(GeometryReader { proxy -> Color in
.offsetY { previous, current in handle_scroll_queue(proxy, queue: self.events)
if previous > current{ return Color.clear
if direction != .up && current < 0 { })
shiftOffset = current - headerOffset
direction = .up
lastHeaderOffset = headerOffset
}
let offset = current < 0 ? (current - shiftOffset) : 0
headerOffset = (-offset < headerHeight ? (offset < 0 ? offset : 0) : -headerHeight)
}else {
if direction != .down {
shiftOffset = current
direction = .down
lastHeaderOffset = headerOffset
}
let offset = lastHeaderOffset + (current - shiftOffset)
headerOffset = (offset > 0 ? 0 : offset)
}
}
.background {
GeometryReader { proxy -> Color in
handle_scroll_queue(proxy, queue: self.events)
return Color.clear
}
}
} }
//.buttonStyle(BorderlessButtonStyle())
.coordinateSpace(name: "scroll") .coordinateSpace(name: "scroll")
.onReceive(handle_notify(.scroll_to_top)) { () in .onReceive(handle_notify(.scroll_to_top)) { () in
events.flush() events.flush()
+34
View File
@@ -0,0 +1,34 @@
//
// AVPlayerView.swift
// damus
//
// Created by Bryan Montz on 9/4/23.
//
import Foundation
import AVKit
import SwiftUI
struct DamusAVPlayerView: UIViewControllerRepresentable {
let player: AVPlayer
var controller: AVPlayerViewController
let show_playback_controls: Bool
func makeUIViewController(context: Context) -> AVPlayerViewController {
self.controller.showsPlaybackControls = show_playback_controls
return self.controller
}
func updateUIViewController(_ uiViewController: AVPlayerViewController, context: Context) {
if uiViewController.player == nil {
uiViewController.player = player
player.play()
}
}
static func dismantleUIViewController(_ uiViewController: AVPlayerViewController, coordinator: ()) {
uiViewController.player?.pause()
uiViewController.player = nil
}
}
@@ -1,106 +0,0 @@
//
// DamusVideoControlsView.swift
// damus
//
// Created by Daniel DAquino on 2024-10-18.
//
import SwiftUI
import AVFoundation
/// A view with playback video controls, made to work seamlessly with `DamusVideoPlayer`
struct DamusVideoControlsView: View {
@ObservedObject var video: DamusVideoPlayer
var body: some View {
VStack {
HStack {
Text(video_timestamp_indicator)
.bold()
.foregroundStyle(.white)
Spacer()
Button(action: {
video.is_muted.toggle()
}, label: {
if video.is_muted {
Image(systemName: "speaker.slash")
.frame(width: 30, height: 30)
}
else {
Image(systemName: "speaker.wave.2.fill")
.frame(width: 30, height: 30)
}
})
.buttonStyle(PlayerCircleButtonStyle())
}
HStack {
Button(action: {
video.is_playing.toggle()
}, label: {
if video.is_playing {
Image(systemName: "pause.fill")
.frame(width: 30, height: 30)
}
else {
Image(systemName: "play.fill")
.frame(width: 30, height: 30)
}
})
.buttonStyle(PlayerCircleButtonStyle())
if let video_duration = video.duration, video_duration > 0 {
Slider(value: $video.current_time, in: 0...video_duration, onEditingChanged: { editing in
video.is_editing_current_time = editing
})
.tint(.white)
}
else {
Spacer()
}
}
}
.padding(10)
}
var video_timestamp_indicator: String {
guard let video_duration = video.duration else {
return "\(formatTimeInterval(video.current_time))"
}
return "\(formatTimeInterval(video.current_time)) / \(formatTimeInterval(video_duration))"
}
func formatTimeInterval(_ interval: TimeInterval) -> String {
if interval.isNaN {
return "--:--"
}
let formatter = DateComponentsFormatter()
formatter.allowedUnits = interval >= 60 * 60 ? [.hour, .minute, .second] : [.minute, .second]
formatter.unitsStyle = .positional
formatter.zeroFormattingBehavior = [.pad]
guard let formattedString = formatter.string(from: interval) else {
return ""
}
return formattedString
}
}
struct PlayerCircleButtonStyle: ButtonStyle {
let padding: CGFloat
init(padding: CGFloat = 8.0) {
self.padding = padding
}
func makeBody(configuration: Self.Configuration) -> some View {
return configuration.label
.padding(padding)
.foregroundColor(Color.white)
.background {
Circle()
.fill(Color.black.opacity(0.5))
}
.scaleEffect(configuration.isPressed ? 0.95 : 1)
}
}
@@ -1,131 +0,0 @@
//
// DamusVideoCoordinator.swift
// damus
//
// Created by Bryan Montz on 9/3/23.
//
import Combine
import Foundation
import SwiftUICore
import AVFoundation
/// DamusVideoCoordinator is responsible for coordinating the various video players throughout the app, and providing a nicely orchestrated experience.
/// The goals of this object are to:
/// - ensure some video playing states (such as mute state and current time) are consistent across different video player view instances of the same video
/// - ensure only one video is playing at a time
/// - Provide global video playback controls to control the currently playing video
///
/// This is used as a singleton object (one global object per `DamusState`), which gets passed around to video players, which can then interact with the coordinator to ensure an app-wide coherent experience
///
/// A good analogy here is that video players and their models/states are like individual cars and their drivers, and this coordinator is like a traffic control person + traffic lights that ensures cars don't crash each other.
final class DamusVideoCoordinator: ObservableObject {
// MARK: - States
// MARK: State and information about each video
private var players: [URL: DamusVideoPlayer] = [:]
// MARK: Main stage requests from player views
// The stacks of video player views that have marked themselves as visible on the user screen.
//
// Because our visibility tracker cannot tell if a player is obscured by a view in front of it,
// we need to implement two stacks representing the different view layers:
// - Normal layer: For timelines, threads, etc
// - Full screen layer: For full screen views
private var normal_layer_main_stage_requests: [MainStageRequest] = []
private var full_screen_layer_stage_requests: [MainStageRequest] = []
// MARK: Coordinator state
// Members representing the state of the coordinator itself
private var full_screen_mode: Bool = false {
didSet {
self.select_focused_video()
}
}
/// The video currently in focus
/// This can only be chosen by the coordinator. To get a video in focus, use one of the instance methods that provide an interface for focus control.
@MainActor
@Published private(set) var focused_video: DamusVideoPlayer? {
didSet {
oldValue?.pause()
focused_video?.play()
Log.info("VIDEO_COORDINATOR: %s paused, playing %s", for: .video_coordination, oldValue?.url.absoluteString ?? "no video", focused_video?.url.absoluteString ?? "no video")
}
}
// MARK: - Interface to set and fetch information about each different video
@MainActor
func get_player(for url: URL) -> DamusVideoPlayer {
if let player = self.players[url] {
return player
}
let player = DamusVideoPlayer(url: url)
self.players[url] = player
return player
}
// MARK: - Interface for video players to come to the foreground
// This portion provides an interface for video players to signal their visibility changes,
// and implements some coordination logic to choose which video to play and pause at a given time.
func request_main_stage(_ request: MainStageRequest) {
Log.info("VIDEO_COORDINATOR: %s requested main stage", for: .video_coordination, request.requestor_id.uuidString)
switch request.layer_context {
case .normal_layer:
if normal_layer_main_stage_requests.first(where: { $0.requestor_id == request.requestor_id }) != nil { return } // Entry exists already
normal_layer_main_stage_requests.append(request)
case .full_screen_layer:
if full_screen_layer_stage_requests.first(where: { $0.requestor_id == request.requestor_id }) != nil { return } // Entry exists already
full_screen_layer_stage_requests.append(request)
}
self.select_focused_video()
}
func give_up_main_stage(request_id: UUID) {
Log.info("VIDEO_COORDINATOR: %s gave up the main stage", for: .video_coordination, request_id.uuidString)
normal_layer_main_stage_requests.removeAll(where: { $0.requestor_id == request_id })
full_screen_layer_stage_requests.removeAll(where: { $0.requestor_id == request_id })
self.select_focused_video()
}
// MARK: - Additional interface to help with video coordination
func set_full_screen_mode(_ is_full_screen: Bool) {
full_screen_mode = is_full_screen
}
// MARK: - Internal video coordination logic
private func select_focused_video() {
// This function may be called during a SwiftUI view update,
// so schedule this change for the next render pass to ensure state immutability/stability within a single render pass
DispatchQueue.main.async { [weak self] in // [weak self] to safeguard in cases this object is deallocated by the time we execute this task
guard let self else { return }
// The focused video will always be the last one that was inserted similar to a LIFO stack
// The reason is that:
// - both a LIFO stack and a FIFO queue are decent at selecting videos when scrolling on the Y axis (timeline),
// - The LIFO stack is better at selecting videos when navigating on the Z axis (e.g. opening and closing full screen covers or sheets), since those sheets operate like a stack as well
let winning_request = self.full_screen_mode ? self.full_screen_layer_stage_requests.last : self.normal_layer_main_stage_requests.last
self.focused_video = winning_request?.player
winning_request?.main_stage_granted?()
}
Log.info("VIDEO_COORDINATOR: fullscreen layer main stage request stack: %s", for: .video_coordination, full_screen_layer_stage_requests.map({ $0.requestor_id.uuidString }).debugDescription)
Log.info("VIDEO_COORDINATOR: normal layer main stage request stack: %s", for: .video_coordination, normal_layer_main_stage_requests.map({ $0.requestor_id.uuidString }).debugDescription)
Log.info("VIDEO_COORDINATOR: full_screen_mode: %s", for: .video_coordination, String(describing: self.full_screen_mode))
}
// MARK: - Helper structures
struct MainStageRequest {
var requestor_id: UUID
var layer_context: ViewLayerContext
var player: DamusVideoPlayer
var main_stage_granted: (() -> Void)?
}
}
+146 -216
View File
@@ -1,248 +1,178 @@
// //
// DamusVideoPlayer.swift // VideoPlayerView.swift
// damus // damus
// //
// Created by Bryan Montz on 9/5/23. // Created by William Casarin on 2023-04-05.
// //
import AVFoundation
import AVKit
import Combine
import Foundation
import SwiftUI import SwiftUI
/// DamusVideoPlayer has the function of wrapping `AVPlayer` and exposing a control interface that integrates seamlessly with SwiftUI views /// get coordinates in Global reference frame given a Local point & geometry
/// func globalCoordinate(localX x: CGFloat, localY y: CGFloat,
/// This is **NOT** a video player view. This is a headless video object concerned about the video and its playback. To display a video, you need `DamusVideoPlayerView` localGeometry geo: GeometryProxy) -> CGPoint {
/// This is also **NOT** a control view. Please see `DamusVideoControlsView` for that. let localPoint = CGPoint(x: x, y: y)
/// return geo.frame(in: .global).origin.applying(
/// **Implementation notes:** .init(translationX: localPoint.x, y: localPoint.y)
/// - `@MainActor` is needed because `@Published` properties need to be updated on the main thread to avoid SwiftUI mutations within a single render pass )
/// - `@Published` variables are the chosen interface because they integrate very seamlessly with SwiftUI views. Avoid the use of procedural functions to avoid SwiftUI state desync. }
@MainActor final class DamusVideoPlayer: ObservableObject {
struct DamusVideoPlayer: View {
// MARK: Immutable foundational instance members
/// The URL of the video
let url: URL let url: URL
/// The underlying AVPlayer that we are wrapping. @StateObject var model: DamusVideoPlayerViewModel
/// This is not public because we don't want any callers of this class controlling the `AVPlayer` directly, we want them to go through our interface @EnvironmentObject private var orientationTracker: OrientationTracker
/// This measure helps avoid state inconsistencies and other flakiness. DO NOT USE THIS OUTSIDE `DamusVideoPlayer` let style: Style
private let player: AVPlayer let visibility_tracking_method: VisibilityTrackingMethod
@State var isVisible: Bool = false
init(url: URL, video_size: Binding<CGSize?>, controller: VideoController, style: Style, visibility_tracking_method: VisibilityTrackingMethod = .y_scroll) {
// MARK: SwiftUI-friendly interface self.url = url
let mute: Bool?
/// Indicates whether the video has audio at all if case .full = style {
@Published private(set) var has_audio = false mute = false
/// Whether whether this is a live video
@Published private(set) var is_live = false
/// The video size
@Published private(set) var video_size: CGSize?
/// Whether or not to mute the video
@Published var is_muted = true {
didSet {
if oldValue == is_muted { return }
player.isMuted = is_muted
} }
else {
mute = nil
}
_model = StateObject(wrappedValue: DamusVideoPlayerViewModel(url: url, video_size: video_size, controller: controller, mute: mute))
self.visibility_tracking_method = visibility_tracking_method
self.style = style
} }
/// Whether the video is loading
@Published private(set) var is_loading = true var body: some View {
/// The current time of playback, in seconds GeometryReader { geo in
/// Usage note: If editing (such as in a slider), make sure to set `is_editing_current_time` to `true` to detach this value from the current playback let localFrame = geo.frame(in: .local)
@Published var current_time: TimeInterval = .zero let centerY = globalCoordinate(localX: 0, localY: localFrame.midY, localGeometry: geo).y
/// Whether video is playing or not ZStack {
@Published var is_playing = false { if case .full = self.style {
didSet { DamusAVPlayerView(player: model.player, controller: model.player_view_controller, show_playback_controls: true)
if oldValue == is_playing { return } }
// When scrubbing, the playback control is temporarily decoupled, so don't play/pause our `AVPlayer` if case .preview(let on_tap) = self.style {
// When scrubbing stops, the `is_editing_current_time` handler will automatically play/pause depending on `is_playing` DamusAVPlayerView(player: model.player, controller: model.player_view_controller, show_playback_controls: false)
if is_editing_current_time { return } .simultaneousGesture(TapGesture().onEnded({
if is_playing { on_tap?()
player.play() }))
}
if model.is_loading {
ProgressView()
.progressViewStyle(.circular)
.tint(.white)
.scaleEffect(CGSize(width: 1.5, height: 1.5))
}
if case .preview = self.style {
if model.has_audio {
mute_button
}
}
if model.is_live {
live_indicator
}
} }
else { .onChange(of: centerY) { _ in
player.pause() if case .y_scroll = visibility_tracking_method {
update_is_visible(centerY: centerY)
}
}
.on_visibility_change(perform: { new_visibility in
if case .generic = visibility_tracking_method {
model.set_view_is_visible(new_visibility)
}
})
.onAppear {
if case .y_scroll = visibility_tracking_method {
update_is_visible(centerY: centerY)
}
}
}
.onDisappear {
if case .y_scroll = visibility_tracking_method {
model.view_did_disappear()
} }
} }
} }
/// Whether the current time is being manually edited (e.g. when user is scrubbing through the video)
/// **Implementation note:** When set to `true`, this decouples the `current_time` from the video playback observer in a way analogous to a clutch on a standard transmission car, if you are into Automotive engineering. private func update_is_visible(centerY: CGFloat) {
@Published var is_editing_current_time = false { let isBelowTop = centerY > 100, /// 100 =~ approx. bottom (y) of ContentView's TabView
didSet { isAboveBottom = centerY < orientationTracker.deviceMajorAxis
if oldValue == is_editing_current_time { return } model.set_view_is_visible(isBelowTop && isAboveBottom)
if !is_editing_current_time { }
Task {
await self.player.seek(to: CMTime(seconds: current_time, preferredTimescale: 60)) private var mute_icon: String {
// Start playing video again, if we were playing before scrubbing !model.has_audio || model.is_muted ? "speaker.slash" : "speaker"
if self.is_playing { }
self.player.play()
private var mute_icon_color: Color {
model.has_audio ? .white : .red
}
private var mute_button: some View {
HStack {
Spacer()
VStack {
Spacer()
Button {
model.did_tap_mute_button()
} label: {
ZStack {
Circle()
.opacity(0.2)
.frame(width: 32, height: 32)
.foregroundColor(.black)
Image(systemName: mute_icon)
.padding()
.foregroundColor(mute_icon_color)
} }
} }
} }
else {
// Pause playing video, if we were playing before we started scrubbing
if self.is_playing { self.player.pause() }
}
}
}
/// The duration of the video, in seconds.
var duration: TimeInterval? {
return player.currentItem?.duration.seconds
}
// MARK: Internal instance members
private var cancellables = Set<AnyCancellable>()
private var videoSizeObserver: NSKeyValueObservation?
private var videoDurationObserver: NSKeyValueObservation?
private var videoCurrentTimeObserver: Any?
private var videoIsPlayingObserver: NSKeyValueObservation?
// MARK: - Initialization
public init(url: URL) {
self.url = url
self.player = AVPlayer(playerItem: AVPlayerItem(url: url))
self.video_size = nil
Task {
await load()
}
player.isMuted = is_muted
NotificationCenter.default.addObserver(
self,
selector: #selector(did_play_to_end),
name: Notification.Name.AVPlayerItemDidPlayToEndTime,
object: player.currentItem
)
observeVideoSize()
observeDuration()
observeCurrentTime()
observeVideoIsPlaying()
}
// MARK: - Observers
// Functions that allow us to observe certain variables and publish their changes for view updates
// These are all private because they are part of the internal logic
private func observeVideoSize() {
videoSizeObserver = player.currentItem?.observe(\.presentationSize, options: [.new], changeHandler: { [weak self] (playerItem, change) in
guard let self else { return }
if let newSize = change.newValue, newSize != .zero {
DispatchQueue.main.async {
self.video_size = newSize // Update the bound value
}
}
})
}
private func observeDuration() {
videoDurationObserver = player.currentItem?.observe(\.duration, options: [.new], changeHandler: { [weak self] (playerItem, change) in
guard let self else { return }
if let newDuration = change.newValue, newDuration != .zero {
DispatchQueue.main.async {
self.is_live = newDuration == .indefinite
}
}
})
}
private func observeCurrentTime() {
videoCurrentTimeObserver = player.addPeriodicTimeObserver(forInterval: CMTime(seconds: 1, preferredTimescale: 600), queue: .main) { [weak self] time in
guard let self else { return }
DispatchQueue.main.async { // Must use main thread to update @Published properties
if self.is_editing_current_time == false {
self.current_time = time.seconds
}
}
} }
} }
private func observeVideoIsPlaying() { private var live_indicator: some View {
videoIsPlayingObserver = player.observe(\.rate, changeHandler: { [weak self] (player, change) in VStack {
guard let self else { return } HStack {
guard let new_rate = change.newValue else { return } Text("LIVE", comment: "Text indicator that the video is a livestream.")
DispatchQueue.main.async { .bold()
self.is_playing = new_rate > 0 .foregroundColor(.red)
.padding(.horizontal)
.padding(.vertical, 5)
.background(
Capsule()
.fill(Color.black.opacity(0.5))
)
.padding([.top, .leading])
Spacer()
} }
}) Spacer()
}
// MARK: - Other internal logic functions
private func load() async {
has_audio = await self.video_has_audio()
is_loading = false
}
private func video_has_audio() async -> Bool {
do {
let hasAudibleTracks = ((try await player.currentItem?.asset.loadMediaSelectionGroup(for: .audible)) != nil)
let tracks = try? await player.currentItem?.asset.load(.tracks)
let hasAudioTrack = tracks?.filter({ t in t.mediaType == .audio }).first != nil // Deal with odd cases of audio only MOV
return hasAudibleTracks || hasAudioTrack
} catch {
return false
} }
} }
@objc private func did_play_to_end() { enum Style {
player.seek(to: CMTime.zero) /// A full video player with playback controls
player.play() case full
/// A style suitable for muted, auto-playing videos on a feed
case preview(on_tap: (() -> Void)?)
} }
// MARK: - Deinit enum VisibilityTrackingMethod {
/// Detects visibility based on its Y position relative to viewport. Ideal for long feeds
deinit { case y_scroll
videoSizeObserver?.invalidate() /// Detects visibility based whether the view intersects with the viewport
videoDurationObserver?.invalidate() case generic
videoIsPlayingObserver?.invalidate()
}
// MARK: - Convenience interface functions
func play() {
self.is_playing = true
}
func pause() {
self.is_playing = false
} }
} }
struct DamusVideoPlayer_Previews: PreviewProvider {
extension DamusVideoPlayer { static var previews: some View {
/// The simplest view for a `DamusVideoPlayer` object. Group {
/// DamusVideoPlayer(url: URL(string: "http://cdn.jb55.com/s/zaps-build.mp4")!, video_size: .constant(nil), controller: VideoController(), style: .full)
/// Other views with more features should use this as a base. .environmentObject(OrientationTracker())
/// .previewDisplayName("Full video player")
/// ## Implementation notes:
/// DamusVideoPlayer(url: URL(string: "http://cdn.jb55.com/s/zaps-build.mp4")!, video_size: .constant(nil), controller: VideoController(), style: .preview(on_tap: nil))
/// 1. This is defined inside `DamusVideoPlayer` to allow it to access the private `AVPlayer` instance required to initialize it, which is otherwise hidden away from every other class. .environmentObject(OrientationTracker())
/// 2. DO NOT write any `AVPlayer` control/manipulation code, the `AVPlayer` instance is owned by `DamusVideoPlayer` and only managed there to keep things sane. .previewDisplayName("Preview video player")
struct BaseView: UIViewControllerRepresentable {
let player: DamusVideoPlayer
let show_playback_controls: Bool
func makeUIViewController(context: Context) -> AVPlayerViewController {
let controller = AVPlayerViewController()
controller.showsPlaybackControls = show_playback_controls
return controller
}
func updateUIViewController(_ uiViewController: AVPlayerViewController, context: Context) {
if uiViewController.player == nil {
uiViewController.player = player.player
}
}
static func dismantleUIViewController(_ uiViewController: AVPlayerViewController, coordinator: ()) {
uiViewController.player = nil
} }
} }
} }
@@ -1,199 +0,0 @@
//
// DamusVideoPlayerView.swift
// damus
//
// Created by William Casarin on 2023-04-05.
//
import SwiftUI
/// get coordinates in Global reference frame given a Local point & geometry
func globalCoordinate(localX x: CGFloat, localY y: CGFloat,
localGeometry geo: GeometryProxy) -> CGPoint {
let localPoint = CGPoint(x: x, y: y)
return geo.frame(in: .global).origin.applying(
.init(translationX: localPoint.x, y: localPoint.y)
)
}
/// A feature-rich, generic video player view that plays along well with the multi-video coordinator
struct DamusVideoPlayerView: View {
let url: URL
@ObservedObject var model: DamusVideoPlayer
let style: Style
let main_state_requestor_id: UUID = UUID()
@State var is_visible: Bool = false {
didSet {
if self.is_visible {
// We are visible, request main stage
video_coordinator.request_main_stage(
DamusVideoCoordinator.MainStageRequest(
requestor_id: self.main_state_requestor_id,
layer_context: self.view_layer,
player: self.model,
main_stage_granted: self.main_stage_granted
)
)
}
else {
// We are no longer visible, give up the main stage
video_coordinator.give_up_main_stage(request_id: self.main_state_requestor_id)
}
}
}
/// The context this video player is in.
@Environment(\.view_layer_context) var view_layer_context
/// The video coordinator in this environment
let video_coordinator: DamusVideoCoordinator
var view_layer: ViewLayerContext {
return view_layer_context ?? .normal_layer
}
init(url: URL, coordinator: DamusVideoCoordinator, style: Style) {
self.url = url
self.model = coordinator.get_player(for: url)
self.video_coordinator = coordinator
self.style = style
}
init(model: DamusVideoPlayer, coordinator: DamusVideoCoordinator, style: Style) {
self.url = model.url
self.model = model
self.video_coordinator = coordinator
self.style = style
}
var body: some View {
ZStack {
switch self.style {
case .full:
DamusVideoPlayer.BaseView(player: model, show_playback_controls: true)
case .preview(on_tap: let on_tap), .no_controls(on_tap: let on_tap):
if let on_tap {
DamusVideoPlayer.BaseView(player: model, show_playback_controls: false)
.highPriorityGesture(TapGesture().onEnded({
on_tap()
}))
}
else {
DamusVideoPlayer.BaseView(player: model, show_playback_controls: false)
}
}
if model.is_loading {
ProgressView()
.progressViewStyle(.circular)
.tint(.white)
.scaleEffect(CGSize(width: 1.5, height: 1.5))
}
if case .preview = self.style {
if model.has_audio {
mute_button
}
}
if model.is_live {
live_indicator
}
}
.on_visibility_change(perform: { new_is_visible in
self.is_visible = new_is_visible
}, method: self.visibility_tracking_method)
}
private var visibility_tracking_method: VisibilityTracker.Method {
switch self.view_layer {
case .normal_layer:
return .standard
case .full_screen_layer:
return .no_y_scroll_detection
}
}
func main_stage_granted() {
switch self.style {
case .full, .no_controls:
self.model.is_muted = false
case .preview:
self.model.is_muted = true
}
}
private var mute_icon: String {
!model.has_audio || model.is_muted ? "speaker.slash" : "speaker"
}
private var mute_icon_color: Color {
model.has_audio ? .white : .red
}
private var mute_button: some View {
HStack {
Spacer()
VStack {
ZStack {
Circle()
.opacity(0.2)
.frame(width: 32, height: 32)
.foregroundColor(.black)
Image(systemName: mute_icon)
.padding()
.foregroundColor(mute_icon_color)
}
.highPriorityGesture(TapGesture().onEnded {
model.is_muted.toggle()
})
Spacer()
}
}
}
private var live_indicator: some View {
VStack {
HStack {
Text("LIVE", comment: "Text indicator that the video is a livestream.")
.bold()
.foregroundColor(.red)
.padding(.horizontal)
.padding(.vertical, 5)
.background(
Capsule()
.fill(Color.black.opacity(0.5))
)
.padding([.top, .leading])
Spacer()
}
Spacer()
}
}
// MARK: - Helper structures
enum Style {
/// A full video player with playback controls
case full
/// A style suitable for muted, auto-playing videos on a feed
case preview(on_tap: (() -> Void)?)
/// A video player without any playback controls, suitable if using custom controls elsewhere.
case no_controls(on_tap: (() -> Void)?)
}
}
struct DamusVideoPlayer_Previews: PreviewProvider {
static var previews: some View {
Group {
DamusVideoPlayerView(url: URL(string: "http://cdn.jb55.com/s/zaps-build.mp4")!, coordinator: DamusVideoCoordinator(), style: .full)
.environmentObject(OrientationTracker())
.environmentObject(DamusVideoCoordinator())
.previewDisplayName("Full video player")
DamusVideoPlayerView(url: URL(string: "http://cdn.jb55.com/s/zaps-build.mp4")!, coordinator: DamusVideoCoordinator(), style: .preview(on_tap: nil))
.environmentObject(OrientationTracker())
.environmentObject(DamusVideoCoordinator())
.previewDisplayName("Preview video player")
}
}
}
@@ -0,0 +1,147 @@
//
// DamusVideoPlayerViewModel.swift
// damus
//
// Created by Bryan Montz on 9/5/23.
//
import AVFoundation
import AVKit
import Combine
import Foundation
import SwiftUI
func video_has_audio(player: AVPlayer) async -> Bool {
do {
let hasAudibleTracks = ((try await player.currentItem?.asset.loadMediaSelectionGroup(for: .audible)) != nil)
let tracks = try? await player.currentItem?.asset.load(.tracks)
let hasAudioTrack = tracks?.filter({ t in t.mediaType == .audio }).first != nil // Deal with odd cases of audio only MOV
return hasAudibleTracks || hasAudioTrack
} catch {
return false
}
}
@MainActor
final class DamusVideoPlayerViewModel: ObservableObject {
private let url: URL
private let player_item: AVPlayerItem
let player: AVPlayer
fileprivate let controller: VideoController
let player_view_controller = AVPlayerViewController()
let id = UUID()
@Published var has_audio = false
@Published var is_live = false
@Binding var video_size: CGSize?
@Published var is_muted = true
@Published var is_loading = true
private var cancellables = Set<AnyCancellable>()
private var videoSizeObserver: NSKeyValueObservation?
private var videoDurationObserver: NSKeyValueObservation?
private var is_scrolled_into_view = false {
didSet {
if is_scrolled_into_view && !oldValue {
// we have just scrolled from out of view into view
controller.focused_model_id = id
} else if !is_scrolled_into_view && oldValue {
// we have just scrolled from in view to out of view
if controller.focused_model_id == id {
controller.focused_model_id = nil
}
}
}
}
init(url: URL, video_size: Binding<CGSize?>, controller: VideoController, mute: Bool? = nil) {
self.url = url
player_item = AVPlayerItem(url: url)
player = AVPlayer(playerItem: player_item)
self.controller = controller
_video_size = video_size
Task {
await load()
}
is_muted = mute ?? controller.should_mute_video(url: url)
player.isMuted = is_muted
NotificationCenter.default.addObserver(
self,
selector: #selector(did_play_to_end),
name: Notification.Name.AVPlayerItemDidPlayToEndTime,
object: player_item
)
controller.$focused_model_id
.sink { [weak self] model_id in
model_id == self?.id ? self?.player.play() : self?.player.pause()
}
.store(in: &cancellables)
observeVideoSize()
observeDuration()
}
private func observeVideoSize() {
videoSizeObserver = player.currentItem?.observe(\.presentationSize, options: [.new], changeHandler: { [weak self] (playerItem, change) in
guard let self else { return }
if let newSize = change.newValue, newSize != .zero {
DispatchQueue.main.async {
self.video_size = newSize // Update the bound value
}
}
})
}
private func observeDuration() {
videoDurationObserver = player.currentItem?.observe(\.duration, options: [.new], changeHandler: { [weak self] (playerItem, change) in
guard let self else { return }
if let newDuration = change.newValue, newDuration != .zero {
DispatchQueue.main.async {
self.is_live = newDuration == .indefinite
}
}
})
}
private func load() async {
if let meta = controller.metadata(for: url) {
has_audio = meta.has_audio
video_size = meta.size
} else {
has_audio = await video_has_audio(player: player)
}
is_loading = false
}
func did_tap_mute_button() {
is_muted.toggle()
player.isMuted = is_muted
controller.toggle_should_mute_video(url: url)
}
func set_view_is_visible(_ is_visible: Bool) {
is_scrolled_into_view = is_visible
}
func view_did_disappear() {
set_view_is_visible(false)
}
@objc private func did_play_to_end() {
player.seek(to: CMTime.zero)
player.play()
}
deinit {
videoSizeObserver?.invalidate()
videoDurationObserver?.invalidate()
}
}
+44
View File
@@ -0,0 +1,44 @@
//
// VideoController.swift
// damus
//
// Created by Bryan Montz on 9/3/23.
//
import Combine
import Foundation
struct VideoMetadata {
let has_audio: Bool
let size: CGSize
}
final class VideoController: ObservableObject {
private var mute_states: [URL: Bool] = [:]
private var metadatas: [URL: VideoMetadata] = [:]
@Published var focused_model_id: UUID?
func toggle_should_mute_video(url: URL) {
let state = mute_states[url] ?? true
mute_states[url] = !state
objectWillChange.send()
}
func should_mute_video(url: URL) -> Bool {
mute_states[url] ?? true
}
func set_metadata(_ metadata: VideoMetadata, url: URL) {
metadatas[url] = metadata
}
func metadata(for url: URL) -> VideoMetadata? {
metadatas[url]
}
func size_for_url(_ url: URL) -> CGSize? {
metadatas[url]?.size
}
}
+1 -1
View File
@@ -178,7 +178,7 @@ struct ConnectWalletView: View {
Text("Damus Wallet", comment: "Title text for Damus Wallet view.") Text("Damus Wallet", comment: "Title text for Damus Wallet view.")
.fontWeight(.bold) .fontWeight(.bold)
Text("Securely connect your Damus app to your wallet using Nostr Wallet Connect", comment: "Text to prompt user to connect their wallet using 'Nostr Wallet Connect'.") Text("Securely connect your Damus app to your wallet using Nostr\u{00A0}Wallet\u{00A0}Connect", comment: "Text to prompt user to connect their wallet using 'Nostr Wallet Connect'.")
.font(.caption) .font(.caption)
.multilineTextAlignment(.center) .multilineTextAlignment(.center)
} }
-1
View File
@@ -6,7 +6,6 @@
// //
import SwiftUI import SwiftUI
import CodeScanner
enum WalletScanResult: Equatable { enum WalletScanResult: Equatable {
static func == (lhs: WalletScanResult, rhs: WalletScanResult) -> Bool { static func == (lhs: WalletScanResult, rhs: WalletScanResult) -> Bool {
-1
View File
@@ -28,7 +28,6 @@ struct ZapsView: View {
} }
} }
} }
.padding(.bottom, tabHeight)
.navigationBarTitle(NSLocalizedString("Zaps", comment: "Navigation bar title for the Zaps view.")) .navigationBarTitle(NSLocalizedString("Zaps", comment: "Navigation bar title for the Zaps view."))
.onAppear { .onAppear {
model.subscribe() model.subscribe()
Binary file not shown.
Binary file not shown.
+49 -169
View File
@@ -2,30 +2,6 @@
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0"> <plist version="1.0">
<dict> <dict>
<key>followed_by_three_and_others</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@OTHERS@</string>
<key>OTHERS</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>d</string>
<key>zero</key>
<string>متابع من قبل %2$@, %3$@, %4$@ &amp; %1$d وآخرين</string>
<key>one</key>
<string>متابع من قبل %2$@, %3$@, %4$@ &amp; %1$d آخر</string>
<key>two</key>
<string>متابع من قبل %2$@, %3$@, %4$@ &amp; %1$d وآخرين</string>
<key>few</key>
<string>متابع من قبل %2$@, %3$@, %4$@ &amp; %1$d وآخرين</string>
<key>many</key>
<string>متابع من قبل %2$@, %3$@, %4$@ &amp; %1$d وآخرين</string>
<key>other</key>
<string>متابع من قبل %2$@, %3$@, %4$@ &amp; %1$d وآخرين</string>
</dict>
</dict>
<key>followers_count</key> <key>followers_count</key>
<dict> <dict>
<key>NSStringLocalizedFormatKey</key> <key>NSStringLocalizedFormatKey</key>
@@ -74,30 +50,6 @@
<string>المتابَعون</string> <string>المتابَعون</string>
</dict> </dict>
</dict> </dict>
<key>imports_count</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@IMPORTS@</string>
<key>IMPORTS</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>d</string>
<key>zero</key>
<string>المستوردات</string>
<key>one</key>
<string>استورد</string>
<key>two</key>
<string>المستوردات</string>
<key>few</key>
<string>المستوردات</string>
<key>many</key>
<string>المستوردات</string>
<key>other</key>
<string>المستوردات</string>
</dict>
</dict>
<key>reacted_tagged_in_3</key> <key>reacted_tagged_in_3</key>
<dict> <dict>
<key>NSStringLocalizedFormatKey</key> <key>NSStringLocalizedFormatKey</key>
@@ -109,20 +61,20 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>d</string> <string>d</string>
<key>zero</key> <key>zero</key>
<string>%2$@ و %1$ وغيرهم تفاعلوا مع منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر تفاعل مع منشور تمت الإشارة لك فيه</string>
<key>one</key> <key>one</key>
<string>%2$@ و %1$ تفاعل مع منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر تفاعل مع منشور تمت الإشارة لك فيه</string>
<key>two</key> <key>two</key>
<string>%2$@ و %1$ تفاعلا مع منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d آخران تفاعلوا مع منشور تمت الإشارة لك فيه</string>
<key>few</key> <key>few</key>
<string>%2$@ و %1$ تفاعلا مع منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d آخرون تفاعلوا مع منشور تمت الإشارة لك فيه</string>
<key>many</key> <key>many</key>
<string>%2$@ و %1$ وغيرهم تفاعلوا مع منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر تفاعلوا مع منشور تمت الإشارة لك فيه</string>
<key>other</key> <key>other</key>
<string>%2$@ و %1$ وغيرهم تفاعلوا مع منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر تفاعلوا مع منشور تمت الإشارة لك فيه</string>
</dict> </dict>
</dict> </dict>
<key>reacted_your_note_3</key> <key>reacted_your_post_3</key>
<dict> <dict>
<key>NSStringLocalizedFormatKey</key> <key>NSStringLocalizedFormatKey</key>
<string>%#@REACTED@</string> <string>%#@REACTED@</string>
@@ -133,17 +85,17 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>d</string> <string>d</string>
<key>zero</key> <key>zero</key>
<string>%2$@ و %1$d وغيرهم تفاعلوا مع منشورك</string> <string>%2$@ و %1$d مستخدم آخر تفاعل مع منشورك</string>
<key>one</key> <key>one</key>
<string>%2$@ و %1$d أخر تفاعل مع منشورك</string> <string>%2$@ و %1$d مستخدم آخر تفاعل مع منشورك</string>
<key>two</key> <key>two</key>
<string>%2$@ و %1$d غيرهم تفاعلا مع منشورك</string> <string>%2$@ و %1$d آخران تفاعلا مع منشورك</string>
<key>few</key> <key>few</key>
<string>%2$@ و %1$d غيرهم تفاعلوا مع منشورك</string> <string>%2$@ و %1$d آخرون تفاعلوا مع منشورك</string>
<key>many</key> <key>many</key>
<string>%2$@ و %1$d غيرهم تفاعلوا مع منشورك</string> <string>%2$@ و %1$d مستخدم آخر تفاعلوا مع منشورك</string>
<key>other</key> <key>other</key>
<string>%2$@ و %1$d غيرهم تفاعلوا مع منشورك</string> <string>%2$@ و %1$d مستخدم آخر تفاعلوا مع منشورك</string>
</dict> </dict>
</dict> </dict>
<key>reacted_your_profile_3</key> <key>reacted_your_profile_3</key>
@@ -253,20 +205,20 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>d</string> <string>d</string>
<key>zero</key> <key>zero</key>
<string>%2$@ و %1$d غيرهم أعادوا مشاركة منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر نشر منشورا تمت الإشارة لك فيه</string>
<key>one</key> <key>one</key>
<string>%2$@ و %1$d أخر أعاد مشاركة منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر نشر منشورا تمت الإشارة لك فيه</string>
<key>two</key> <key>two</key>
<string>%2$@ و %1$d غيرهم أعادا مشاركة منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d آخران نشروا منشورا تمت الإشارة لك فيه</string>
<key>few</key> <key>few</key>
<string>%2$@ و %1$d غيرهم أعادوا مشاركة منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d آخرون نشروا منشورا تمت الإشارة لك فيه</string>
<key>many</key> <key>many</key>
<string>%2$@ و %1$d غيرهم أعادوا مشاركة منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر نشروا منشورا تمت الإشارة لك فيه</string>
<key>other</key> <key>other</key>
<string>%2$@ و %1$d غيرهم أعادوا مشاركة منشور تم ذكر حسابك فيه</string> <string>%2$@ و %1$d مستخدم آخر نشروا منشورا تمت الإشارة لك فيه</string>
</dict> </dict>
</dict> </dict>
<key>reposted_your_note_3</key> <key>reposted_your_post_3</key>
<dict> <dict>
<key>NSStringLocalizedFormatKey</key> <key>NSStringLocalizedFormatKey</key>
<string>%#@REPOSTED@</string> <string>%#@REPOSTED@</string>
@@ -277,17 +229,17 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>d</string> <string>d</string>
<key>zero</key> <key>zero</key>
<string>%2$@ و %1$d أعاد مشاركة منشورك</string> <string>%2$@ و %1$d مستخدم آخر نشر منشورك</string>
<key>one</key> <key>one</key>
<string>%2$@ و %1$d أعاد مشاركة منشورك</string> <string>%2$@ و %1$d مستخدم آخر نشر منشورك</string>
<key>two</key> <key>two</key>
<string>%2$@ و %1$d أعادا مشاركة منشورك</string> <string>%2$@ و %1$d آخران نشروا منشورك</string>
<key>few</key> <key>few</key>
<string>%2$@ و %1$d أعادوا مشاركة منشورك</string> <string>%2$@ و %1$d آخرون نشروا منشورك</string>
<key>many</key> <key>many</key>
<string>%2$@ و %1$d أعادوا مشاركة منشورك</string> <string>%2$@ و %1$d مستخدم آخر نشروا منشورك</string>
<key>other</key> <key>other</key>
<string>%2$@ و %1$d أعادوا مشاركة منشورك</string> <string>%2$@ و %1$d مستخدم آخر نشروا منشورك</string>
</dict> </dict>
</dict> </dict>
<key>reposted_your_profile_3</key> <key>reposted_your_profile_3</key>
@@ -338,30 +290,6 @@
<string>اعادة نشر</string> <string>اعادة نشر</string>
</dict> </dict>
</dict> </dict>
<key>quoted_reposts_count</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@QUOTE_REPOSTS@</string>
<key>QUOTE_REPOSTS</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>d</string>
<key>zero</key>
<string>اقتباسات</string>
<key>one</key>
<string>اقتباس</string>
<key>two</key>
<string>اقتباسات</string>
<key>few</key>
<string>اقتباسات</string>
<key>many</key>
<string>اقتباسات</string>
<key>other</key>
<string>اقتباسات</string>
</dict>
</dict>
<key>sats</key> <key>sats</key>
<dict> <dict>
<key>NSStringLocalizedFormatKey</key> <key>NSStringLocalizedFormatKey</key>
@@ -410,54 +338,6 @@
<string>%2$@ ساتوشي</string> <string>%2$@ ساتوشي</string>
</dict> </dict>
</dict> </dict>
<key>users_talking_about_it</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@USERS@</string>
<key>USERS</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>d</string>
<key>zero</key>
<string>%d يتكلم عنك</string>
<key>one</key>
<string>%d يتكلم عنك</string>
<key>two</key>
<string>%d يتكلمون عنك</string>
<key>few</key>
<string>%d يتكلمون عنك</string>
<key>many</key>
<string>%d يتكلمون عنك</string>
<key>other</key>
<string>%d يتكلمون عنك</string>
</dict>
</dict>
<key>word_count</key>
<dict>
<key>NSStringLocalizedFormatKey</key>
<string>%#@WORDS@</string>
<key>WORDS</key>
<dict>
<key>NSStringFormatSpecTypeKey</key>
<string>NSStringPluralRuleType</string>
<key>NSStringFormatValueTypeKey</key>
<string>d</string>
<key>zero</key>
<string>%d كلمة</string>
<key>one</key>
<string>%d كلمة</string>
<key>two</key>
<string>%d كلمتان</string>
<key>few</key>
<string>%d من الكلمات</string>
<key>many</key>
<string>%d من الكلمات</string>
<key>other</key>
<string>%d من الكلمات</string>
</dict>
</dict>
<key>zap_notification_no_message</key> <key>zap_notification_no_message</key>
<dict> <dict>
<key>NSStringLocalizedFormatKey</key> <key>NSStringLocalizedFormatKey</key>
@@ -493,17 +373,17 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>@</string> <string>@</string>
<key>zero</key> <key>zero</key>
<string>لقد وصلك %2$@ سات من %3$@: &quot;%4$@&quot;</string> <string>تم استلام %2$@ من %3$@: "%4$@"</string>
<key>one</key> <key>one</key>
<string>لقد وصلك %2$@ سات من %3$@: &quot;%4$@&quot;</string> <string>تم استلام %2$@ من %3$@: "%4$@"</string>
<key>two</key> <key>two</key>
<string>لقد وصلك %2$@ ساتس من %3$@: &quot;%4$@&quot;</string> <string>تم استلام %2$@ من %3$@: "%4$@"</string>
<key>few</key> <key>few</key>
<string>لقد وصلك %2$@ ساتس من %3$@: &quot;%4$@&quot;</string> <string>تم استلام %2$@ من %3$@: "%4$@"</string>
<key>many</key> <key>many</key>
<string>لقد وصلك %2$@ ساتس من %3$@: &quot;%4$@&quot;</string> <string>تم استلام %2$@ من %3$@: "%4$@"</string>
<key>other</key> <key>other</key>
<string>لقد وصلك %2$@ ساتس من %3$@: &quot;%4$@&quot;</string> <string>تم استلام %2$@ من %3$@: "%4$@"</string>
</dict> </dict>
</dict> </dict>
<key>zapped_tagged_in_3</key> <key>zapped_tagged_in_3</key>
@@ -517,20 +397,20 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>d</string> <string>d</string>
<key>zero</key> <key>zero</key>
<string>%2$@ و %1$d غيرهم ومضوا منشور تم ذكر حستبك فيه</string> <string>%2$@ و %1$d مستخدم آخر ومّض منشورا تمت الإشارة لك فيه</string>
<key>one</key> <key>one</key>
<string>%2$@ و %1$d آخر ومض منشور تم ذكر حستبك فيه</string> <string>%2$@ و %1$d مستخدم آخر ومّض منشورا تمت الإشارة لك فيه</string>
<key>two</key> <key>two</key>
<string>%2$@ و %1$d آخرين ومضوا منشور تم ذكر حستبك فيه</string> <string>%2$@ و %1$d آخران ومّضوا منشورا تمت الإشارة لك فيه</string>
<key>few</key> <key>few</key>
<string>%2$@ و %1$d آخرين ومضوا منشور تم ذكر حستبك فيه</string> <string>%2$@ و %1$d آخررن ومّضوا منشورا تمت الإشارة لك فيه</string>
<key>many</key> <key>many</key>
<string>%2$@ و %1$d آخرين ومضوا منشور تم ذكر حستبك فيه</string> <string>%2$@ و %1$d مستخدم آخر ومّضوا منشورا تمت الإشارة لك فيه</string>
<key>other</key> <key>other</key>
<string>%2$@ و %1$d آخرين ومضوا منشور تم ذكر حستبك فيه</string> <string>%2$@ و %1$d مستخدم آخر ومّضوا منشورا تمت الإشارة لك فيه</string>
</dict> </dict>
</dict> </dict>
<key>zapped_your_note_3</key> <key>zapped_your_post_3</key>
<dict> <dict>
<key>NSStringLocalizedFormatKey</key> <key>NSStringLocalizedFormatKey</key>
<string>%#@ZAPPED@</string> <string>%#@ZAPPED@</string>
@@ -541,13 +421,13 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>d</string> <string>d</string>
<key>zero</key> <key>zero</key>
<string>%2$@ و %1$d مستخدم آخر ومّضوا منشورك</string> <string>%2$@ و %1$d مستخدم آخر ومّض منشورك</string>
<key>one</key> <key>one</key>
<string>%2$@ و %1$d مستخدم آخر ومّضوا منشورك</string> <string>%2$@ و %1$d مستخدم آخر ومّض منشورك</string>
<key>two</key> <key>two</key>
<string>%2$@ و %1$d مستخدم آخر ومّضوا منشورك</string> <string>%2$@ و %1$d آخران ومّضوا منشورك</string>
<key>few</key> <key>few</key>
<string>%2$@ و %1$d مستخدم آخر ومّضوا منشورك</string> <string>%2$@ و %1$d آخرون ومّضوا منشورك</string>
<key>many</key> <key>many</key>
<string>%2$@ و %1$d مستخدم آخر ومّضوا منشورك</string> <string>%2$@ و %1$d مستخدم آخر ومّضوا منشورك</string>
<key>other</key> <key>other</key>
@@ -565,17 +445,17 @@
<key>NSStringFormatValueTypeKey</key> <key>NSStringFormatValueTypeKey</key>
<string>d</string> <string>d</string>
<key>zero</key> <key>zero</key>
<string>%2$@ و %1$d مستخدم آخر ومضوا حسابك</string> <string>%2$@ و %1$d مستخدم آخر ومّض حسابك</string>
<key>one</key> <key>one</key>
<string>%2$@ و %1$d مستخدم آخر ومضوا حسابك</string> <string>%2$@ و %1$d مستخدم آخر ومّض حسابك</string>
<key>two</key> <key>two</key>
<string>%2$@ و %1$d مستخدم آخر ومضوا حسابك</string> <string>%2$@ و %1$d آخران ومّضوا حسابك</string>
<key>few</key> <key>few</key>
<string>%2$@ و %1$d مستخدم آخر ومضوا حسابك</string> <string>%2$@ و %1$d آخرون ومّضوا حسابك</string>
<key>many</key> <key>many</key>
<string>%2$@ و %1$d مستخدم آخر ومضوا حسابك</string> <string>%2$@ و %1$d مستخدم آخر ومّضوا حسابك</string>
<key>other</key> <key>other</key>
<string>%2$@ و %1$d مستخدم آخر ومضوا حسابك</string> <string>%2$@ و %1$d مستخدم آخر ومّضوا حسابك</string>
</dict> </dict>
</dict> </dict>
<key>zaps_count</key> <key>zaps_count</key>
Binary file not shown.
-1
View File
@@ -44,7 +44,6 @@ struct MainView: View {
.onReceive(handle_notify(.logout)) { () in .onReceive(handle_notify(.logout)) { () in
try? clear_keypair() try? clear_keypair()
keypair = nil keypair = nil
SuggestedHashtagsView.lastRefresh_hashtags.removeAll()
// We need to disconnect and reconnect to all relays when the user signs out // We need to disconnect and reconnect to all relays when the user signs out
// This is to conform to NIP-42 and ensure we aren't persisting old connections // This is to conform to NIP-42 and ensure we aren't persisting old connections
notify(.disconnect_relays) notify(.disconnect_relays)
Binary file not shown.
Binary file not shown.

Some files were not shown because too many files have changed in this diff Show More