Compare commits

..

841 commits
1.7.6 ... main

Author SHA1 Message Date
Thomas Ricouard
9af485b2be Correctly updated grouped notifications 2024-09-20 09:23:36 +02:00
Thomas Ricouard
71027e2cc3 Add Picasso screenshots 2024-09-19 13:46:22 +02:00
Thomas Ricouard
4338e0a355 Fix the marker restore 2024-09-19 13:46:13 +02:00
Thomas Ricouard
df19135e19 Typos 2024-09-19 13:46:06 +02:00
Thomas Ricouard
12b6c1af36 Restore tap on status counter 2024-09-19 10:10:25 +02:00
Thomas Ricouard
db7155423a Add back posts streaming behind a setting, default to off 2024-09-19 09:55:56 +02:00
Thomas Ricouard
7eeaf4d902 Add explanations to some settings 2024-09-19 09:30:31 +02:00
Thomas Ricouard
b19a5706e2 Update packages 2024-09-19 09:16:49 +02:00
Thomas Ricouard
a1ce85b188 Bump version to 1.10.59 2024-09-19 09:06:38 +02:00
Thomas Ricouard
009150bc3e Fix MediaUI close button on Catalyst 2024-09-19 08:59:36 +02:00
Thomas Ricouard
23960d6d0f Embed Threads quote 2024-09-18 16:53:54 +02:00
Thomas Ricouard
4dd4968bb2 Fix image preview sheet 2024-09-18 15:53:55 +02:00
Thomas Ricouard
b0846c9a3d Fix issue with various editor sheets 2024-09-18 15:38:42 +02:00
Thomas Ricouard
7cef1c5786 Bump version to 1.10.58 2024-09-18 15:14:38 +02:00
Thomas Ricouard
a167bcdf28 Fix iPad app refresh on background 2024-09-17 20:49:51 +02:00
Thomas Ricouard
d2eca1d646 Fetch timeline 40 per page + 200 per batch 2024-09-17 14:26:57 +02:00
Thomas Ricouard
9ec6a4ef66 Cleanup + use ScrollPoxy 2024-09-17 14:01:32 +02:00
Thomas Ricouard
576b52e8c8 Bump version to 1.10.57 2024-09-17 08:49:00 +02:00
Thomas Ricouard
46f481aabd More fixes 2024-09-17 08:23:53 +02:00
Thomas Ricouard
81d5d4396f More fixes for VideoPlayer 2024-09-16 17:41:55 +02:00
Thomas Ricouard
f610c3f047 Fix timeline next page 2024-09-16 17:32:46 +02:00
Thomas Ricouard
7a35779b29 Fix notifications filter 2024-09-16 17:32:37 +02:00
Thomas Ricouard
e4d1196301 Fix video player 2024-09-16 17:32:31 +02:00
Thomas Ricouard
93421c56d9 Timeline: cleanup datasource near bottom 2024-09-15 13:25:51 +02:00
Thomas Ricouard
0b7c6a799f Bump version to 1.10.56 2024-09-15 08:27:43 +02:00
Thomas Ricouard
8f71f6649a Fixes 2024-09-15 08:06:10 +02:00
Thomas Ricouard
122e57d8ac Restore ScrollToTop on timeline tab 2024-09-14 08:09:51 +02:00
Thomas Ricouard
594fb3ea07 New iOS 18 Icon 2024-09-13 15:21:14 +02:00
Thomas Ricouard
2f1632b950 Bump version to 1.10.55 2024-09-13 14:03:05 +02:00
Thomas Ricouard
c1dfb0a085 Fix tests 2024-09-13 13:22:41 +02:00
Thomas Ricouard
ae1bfbeb45 Lint 2024-09-13 12:06:08 +02:00
Thomas Ricouard
6d289ebd09 Completely remove stream for new statuses 2024-09-13 12:04:55 +02:00
Thomas Ricouard
be2939b13c Fix copy link 2024-09-13 11:27:09 +02:00
Thomas Ricouard
491090e373 New pending statuses behavior 2024-09-13 11:27:01 +02:00
Thomas Ricouard
7c5f2aea81 Bump version to 1.10.54 2024-09-13 11:18:44 +02:00
Thomas Ricouard
24f8982bd4 Fiw boot on macOS 14 2024-09-13 10:08:38 +02:00
Thomas Ricouard
7b67ba0294 Bump version to 1.10.53 2024-09-13 10:02:59 +02:00
Thomas Ricouard
698edc4d58 Catalyst: Fix window size 2024-09-11 21:48:03 +02:00
Thomas Ricouard
9d5d341764 fix macOS window 2024-09-11 11:00:34 +02:00
Thomas Ricouard
896c031ab0 Merge branch 'hotfix-sidebar' 2024-09-11 10:03:32 +02:00
Thomas Ricouard
c0b0a3ee1c Fix sidebar 2024-09-11 09:50:32 +02:00
Thomas Ricouard
30b9a95cac Bump version to 1.10.52 2024-09-11 08:50:02 +02:00
Thomas Ricouard
0a15f7ff1c fix visionOS build 2024-09-10 11:44:14 +02:00
Thomas Ricouard
8f7df06d21 Fix tests 2024-09-10 11:31:16 +02:00
Thomas Ricouard
6297a428a3
Full Xcode 16 supports + iOS 18 support (#2100)
* Compile on iOS 18

* Fix more warnings

* Tweak build settings

* Migrate to Swift Tests

* better tests

* Fix

* Fix tests

* More TabView cleanup

Bump to iOS 18 only + remove custom sidebar

* Revert "More TabView cleanup"

This reverts commit e051437fcb.

* Tabbar fix + bump to iOS 18

* Remove popToRoot

* Cleanup scrollToTop

* Support both TapBar

* Better TabView support

* Better TabView support

* Cleanup

* Disable TabView animations

* Remove id in ForEach

* Remove external init for StatusRowView

* Cleanup

* More Swift 6 concurrency

* Swift 6 mode

* Fixes

* Full Swift 6 packages support

* For now compile env in Swift 5 mode

* Fix archive

* More fix to Archive

* Address `dispatch_assert_queue_fail` (#2161)

See https://twitter.com/dimillian/status/1823089444397724003?s=61&t=SC3rvyJQWn1NQqAgMVrT0Q

* Bump Env to Swift 6

* Fix push notification

* Remove unecessary workaround

* Cleanup

* Move to @Entry

* Fix TabView on Catalyst

* Fix build

* Fix build 2

* fix warning

* Fix icons for iOS 18

---------

Co-authored-by: NachoSoto <NachoSoto@users.noreply.github.com>
2024-09-10 06:53:19 +02:00
Thomas Ricouard
4cce9d6333 Fix build 2024-09-06 11:34:20 +02:00
Thomas Ricouard
435f28dda9 UI tweaks 2024-09-06 11:32:49 +02:00
Thomas Ricouard
904cd3dbd7 Fix Catalyst build 2024-09-06 07:53:01 +02:00
Jesús Jiménez Sánchez
ce3e2d2344
Update ES localization (#2177) 2024-09-05 15:57:23 +02:00
Jerry Zhang
5080a536db
Update Simplified Chinese localization (#2173)
* Update Simplified Chinese localization

* Update term on telemetry for SC localization

* Update SC translation for Telemetry

Co-Authored-By: nixzhu <zhuhongxu@gmail.com>

* Fix TelemetryDeck name

---------

Co-authored-by: nixzhu <zhuhongxu@gmail.com>
2024-09-05 15:57:15 +02:00
Thomas Ricouard
55cbd9ea6f Fix setting background 2024-09-03 16:37:01 +02:00
Thomas Ricouard
24f77a99e1 Fix TelemetryDeck copy 2024-09-03 16:34:10 +02:00
Alessio Mason
45628b5a5f
Updated Italian localization (#2172)
* Latest Italian localizations

* Updated Italian localization
2024-09-03 10:15:27 +02:00
Klaus Dresbach
1b7720c9c4
Prevent a tab item from being set twice (#2166) 2024-09-03 10:15:19 +02:00
Thomas Ricouard
199749b809 Bump version to 1.10.51 2024-08-30 11:48:14 +02:00
Thomas Ricouard
e0eed97bcf Direct subscribe to sub.club 2024-08-21 18:22:38 +02:00
Thomas Ricouard
f80fca91e2 Add TelemetryDeck about 2024-08-20 09:32:33 +02:00
Thomas Ricouard
dc85557d40 Tweak to notifications policy 2024-08-16 07:24:19 +02:00
Thomas Ricouard
412f475d1d Connect to notification filters V2 2024-08-15 22:30:31 +02:00
Thomas Ricouard
dfbbb84e9d Bump version to 1.10.50 2024-08-15 21:37:37 +02:00
Thomas Ricouard
83bf872cca Add timer for sub.club account 2024-08-15 21:25:04 +02:00
Thomas Ricouard
8c6d2bee7f Add Wishlist Kit 2024-08-15 09:23:07 +02:00
Thomas Ricouard
824b2de23f Bump version to 1.10.49 2024-08-14 21:21:06 +02:00
Thomas Ricouard
48febd628f Ignore cancelled error on timeline 2024-08-14 20:27:26 +02:00
Thomas Ricouard
dd1a4585e0 Fix build on visionOS 2024-08-14 20:19:20 +02:00
Thomas Ricouard
1ad4a245f3
Add support for sub.club support (#2162)
* Account tipping

* Tryout full flow

* Add link params

* WIP

* Progress flow

* Fixes

* More progress

* Refresh user profile on notification

* Tweaks

* Fix follow button not refreshing

* Refactor proxy url

* Code cleanup

* Subscribe to a premium account from a standard linked account

* Premium posts tab on linked standard account

* Fix flow

* New domain

* Fix flow

* More fixes to follow flow

* Update so to sub.club

* Add colorScheme in URL

* rollback domain

* Back to sub.club

* Use SubClub API for Subscription info

* Fix

* Merge

* Merge branch 'iOS-18' + fixes
2024-08-14 20:07:43 +02:00
Thomas Ricouard
9d11814e49 Add TelemetryDeck 2024-08-14 17:23:10 +02:00
Thomas Ricouard
56869e3a2f Allow cancel add account sheet even when no account 2024-08-14 17:23:00 +02:00
Thomas Ricouard
a737ac19e5 Add Labels in settings 2024-08-14 17:22:44 +02:00
Thomas Ricouard
456d85eda2 New alternative icon settings 2024-08-14 17:22:27 +02:00
Thomas Ricouard
4158f7c959 Bump version to 1.10.48 2024-08-14 08:47:57 +02:00
Thomas Ricouard
113c4f1c84 Display push notification while in the app 2024-08-14 08:45:20 +02:00
Ahnaf Mahmud
77bec1fae3
Remove space between Settings and ellipsis (#2125) 2024-08-14 08:43:35 +02:00
Thomas Ricouard
6f608efa7f Switch to gpt4 o-mini 2024-08-13 12:41:46 +02:00
Thomas Ricouard
6c9d9161dc Extract function from TimelineViewModel 2024-08-05 13:59:38 +02:00
Thomas Ricouard
a72f290038 Lint 2024-08-01 08:58:54 +02:00
Thomas Ricouard
2a3da72239 Update purchase ios 2024-08-01 08:56:34 +02:00
Thomas Ricouard
9fa19aa132 Add media grid on user profile 2024-07-31 18:44:29 +02:00
Thomas Ricouard
123f05538a Paginate search results fix #2143 2024-07-31 11:19:43 +02:00
Thomas Ricouard
719c023369 Bump version to 1.10.47 2024-07-31 11:02:02 +02:00
Thomas Ricouard
0338d54d81 Fix #2147 2024-07-31 10:47:56 +02:00
Jesús Jiménez Sánchez
6fa4ac6f79
Update ES localization (#2148) 2024-07-30 09:24:16 +02:00
Jerry Zhang
cf494fd07a
Update Simplified Chinese localization (#2140) 2024-07-30 09:24:09 +02:00
Adem Özsayın
6766ed496d
Fix nav bar crash and empty settings screen after logout (#2124) 2024-07-29 07:57:33 +02:00
fwcd
33c2646ea1
Make status rows draggable (#2141) 2024-07-29 07:56:42 +02:00
fwcd
9af98c3921
Add context menu and draggability to attachment image views (#2142)
* Factor out MediaUIShareLink

* Add share link to attachment image view

* Make attachment image views draggable

* Add copy actions to attachment image view
2024-07-29 07:56:22 +02:00
Keita Watanabe
4f9cb2e86a
fix sendable (#2144) 2024-07-29 07:55:44 +02:00
Thomas Ricouard
4c1ba2168d Fix double media in share sheet 2024-07-22 13:35:30 +02:00
Thomas Ricouard
be02b2ea76 Fix toolbar 2024-07-22 08:28:42 +02:00
Klaus Dresbach
a3326c3fc2
AddAccountView: fix error handling (#2137) 2024-07-22 07:17:49 +02:00
Ege Sucu
563213d98f
Changed a Translation Wording (#2136)
"Yükselişteki" is not a 100% translation for "Trend", we also have "Trend" in Turkish & this old longer version breaks UI on the tab because of more characters.
2024-07-22 07:17:42 +02:00
Thomas Ricouard
82338f815a Small Refactor TimelineViewModel 2024-07-21 11:20:36 +02:00
Thomas Ricouard
7a9b6cc0e0 Fix icons credit 2024-07-20 21:36:01 +02:00
Thomas Ricouard
339e2ab1c3 new community icons 2024-07-20 21:32:20 +02:00
Thomas Ricouard
f18f3e0e84 Bump version to 1.10.46 2024-07-20 19:14:31 +02:00
Thomas Ricouard
fc6b2129dd Cleanup 2024-07-18 21:37:18 +02:00
Thomas Ricouard
54768772b5 Remove external init for StatusRowView 2024-07-18 21:37:18 +02:00
Thomas Ricouard
3f2fbeeec4 Remove id in ForEach 2024-07-18 21:37:18 +02:00
Ico Davids
47b5fdf92e
Updated NL translations (#2129)
* Updated NL translations

* Some ellipsis tuning
2024-07-17 17:30:48 +02:00
Klaus Dresbach
d320caaa4f
AddRemoteTimelineView: show error message when instance is invalid (#2105) (#2133) 2024-07-17 17:30:30 +02:00
Thomas Ricouard
fbff1d6dfe Adjust content gradient + settings 2024-07-16 09:00:11 +02:00
Thomas Ricouard
3c5c9adc03 Cache tag + gradient on home timeline 2024-07-10 16:13:41 +02:00
nixzhu
5969e8a166
update Simplified Chinese localization (#2121) 2024-07-09 16:08:46 +02:00
Thomas Ricouard
8a33b6c0d0 better attribution UI 2024-07-05 08:45:40 +02:00
Thomas Ricouard
d7429c078f Timeline subtitle 2024-07-04 08:54:23 +02:00
Thomas Ricouard
6f3f8e9dd0 Fix #2116 2024-07-03 14:17:35 +02:00
Thomas Ricouard
b98a90ced6 Bump version to 1.10.45 2024-07-03 09:22:49 +02:00
Christopher Schindler
fdb213e4bf
AccountDetail: fix wrong space on top of the header view (#2113) 2024-07-03 08:12:29 +02:00
Xabi
4b8d7113f1
Update EU localisation (#2112)
Added delete header and delete avatar
2024-07-03 08:12:16 +02:00
Thomas Ricouard
7b9cfc2863 Enable links in iPad / macOS sidebar 2024-07-02 20:22:09 +02:00
Thomas Ricouard
478a788f87 Redesign News + support links attributions 2024-07-02 19:59:21 +02:00
Pierre-Yves Lapersonne
2d04433783
fix: french localization (#2117) (#2118)
Some wordings were not translated in french, others needed to be improved a bit.
This fix brings evolutions written by myself and tested locally on a real device (french guy with french as mother language)

Signed-off-by: Pierre-Yves Lapersonne <dev@pylapersonne.info>
2024-07-02 15:22:38 +02:00
Thomas Ricouard
59a333cc20 Bump version to 1.10.44 2024-06-26 14:47:35 +02:00
Thomas Ricouard
9bc9961f34 Fix filtered notifications 2024-06-26 13:32:36 +02:00
Thomas Ricouard
0af3732ea9 Fix typo 2024-06-26 11:06:50 +02:00
Thomas Ricouard
5960014da9 Fix warnings 2024-06-26 11:05:35 +02:00
Thomas Ricouard
2ab52d3d3e Optimize some text 2024-06-26 09:43:27 +02:00
Thomas Ricouard
cd60e0ce1a Fixed action view height 2024-06-26 09:26:34 +02:00
Thomas Ricouard
ff1d5733a0 Tryout: Fixed size optimizations 2024-06-26 09:03:00 +02:00
Nam
9887a81ef0
Make toolbar account selector button avatar match shape option. (#2098) 2024-06-25 10:59:43 +02:00
Christopher Schindler
f2ba08e1cc
Add an option to delete the avatar or header (#2109)
* Endpoint: add Profile endpoint

* EditAccount: add an option to delete the avatar or header

* EditAccount: always display avatar view
2024-06-25 10:59:19 +02:00
Thomas Ricouard
2b8bc2ecd3 Pin EmojiText + NukeUI 2024-06-24 16:01:20 +02:00
Thomas Ricouard
513c686b64 Back to main Bodega 2024-06-24 09:01:45 +02:00
Christopher Schindler
aaeb9eaa36
Fix the upload of heavy avatar or header image (#2103)
* Compressor: allows custom parameters for compressing image (maxSize, maxHeight, maxWidth)

* Account: configures the maximum image size for uploading avatar and header
2024-06-24 08:58:06 +02:00
Christopher Schindler
02a8cb12e9
AccountDetailView: fix white spacing on top of the view (#2110) 2024-06-24 08:56:17 +02:00
Jerry Zhang
be54a58ae6
Update Simplified Chinese localization (#2111)
* Update Simplified Chinese localization

* Refine Simplified Chinese localization

Co-Authored-By: nixzhu <zhuhongxu@gmail.com>

* Fix: SC localization

Co-Authored-By: nixzhu <zhuhongxu@gmail.com>

* Fix: Missing SC translations

Co-Authored-By: nixzhu <zhuhongxu@gmail.com>

---------

Co-authored-by: nixzhu <zhuhongxu@gmail.com>
2024-06-24 08:55:54 +02:00
Thomas Ricouard
8e86e6d205 Back to main Introspect 2024-06-18 08:28:19 +02:00
Thomas Ricouard
283e537c44 Bump version to 1.10.43 2024-06-12 21:24:55 +02:00
Thomas Ricouard
551697eb2c Refix Introspect 2024-06-12 20:51:08 +02:00
Thomas Ricouard
ab99ef9a0a Fix 2024-06-12 20:41:19 +02:00
Thomas Ricouard
02d73de113 Fix iOS 18 build + performances 2024-06-12 20:37:43 +02:00
Thomas Ricouard
375ea665b4 Default logout experience to trending timeline + remove cache when logout 2024-06-12 20:02:01 +02:00
Thomas Ricouard
a88b9a7fd9 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-06-08 18:21:35 +02:00
Thomas Ricouard
520315d50f Update packages 2024-06-08 18:21:33 +02:00
Xabi
1bbbdc8194
Update EU localisation (#2095)
Still unsure whether TimelineFilter and AppAccount need to be translated.
2024-06-08 18:02:31 +02:00
Adem Özsayın
d36930b7af
Updated TR translations (#2088) 2024-06-01 13:55:26 +02:00
David Whetstone
c06e3b59e4
Add "Settings ..." menu and ⌘, hotkey (#2079) 2024-06-01 13:55:06 +02:00
Thomas Ricouard
8cca261e43 Bump version to 1.10.42 2024-06-01 13:54:44 +02:00
Thomas Ricouard
f40aeb9cac Add followers count widget 2024-05-17 14:22:00 +02:00
Thomas Ricouard
1578896b3e Immersive short modal 2024-05-17 13:56:03 +02:00
Thomas Ricouard
ba3d8b1882 Composer: disable predictive type on all platforms 2024-05-17 13:55:55 +02:00
Thomas Ricouard
04af087c4b Bump version to 1.10.41 2024-05-16 07:05:17 +02:00
Thomas Ricouard
a9398c25af fix #2064 2024-05-15 10:52:41 +02:00
Thomas Ricouard
13d721912b Add lists widget 2024-05-15 09:27:35 +02:00
Thomas Ricouard
e3d4e693d2 More improvement to alt edit 2024-05-15 08:30:57 +02:00
Thomas Ricouard
86c053344b Improve media alt edit 2024-05-15 08:28:05 +02:00
Thomas Ricouard
a996aace80 Add translate for image alt 2024-05-14 19:43:52 +02:00
Jesús Jiménez Sánchez
18a1d17230
Update ES localization (#2076) 2024-05-14 19:37:40 +02:00
Thomas Ricouard
69cb9a20f9 Add native translate for media description edit + profile bio 2024-05-14 19:36:25 +02:00
Thomas Ricouard
bab2b4be9c Fix localization 2024-05-14 12:20:25 +02:00
Thomas Ricouard
bb005386df Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-05-13 22:20:45 +02:00
Thomas Ricouard
c77bb992b4 Update OpenAI models to gpt-4o 2024-05-13 22:20:43 +02:00
Paul Schuetz
7caf00d07d
Resolve escaped characters in a status (#2071)
* Resolve escaped characters in a status

Escaped characters are now returned to their original form for
HTMLString.asRawText.

* Unescape the markdown version too

The HTMLString.asMarkdown string is now also unescaped, &amp; and
similar are resolved.

* Fix a internal fallback

If one of the unescape(...) commands fails, the original, unescaped
text is used instead of an empty string.
2024-05-13 21:32:38 +02:00
Thomas Ricouard
6ed760a775 Bump version to 1.10.40 2024-05-13 20:20:24 +02:00
Thomas Ricouard
ecd149b3d2 Fix a crash in quote post editor on macOS 2024-05-13 19:26:48 +02:00
Xabi
9aaf0b2350
Update EU localisation (#2062)
Round 2
2024-05-13 13:28:24 +02:00
Cthulhux
2d6cce6b01
de: translated one more string (#2063) 2024-05-13 13:27:58 +02:00
Paul Schuetz
48faddebea
Implement Apple Translate (#2065)
* Implement a first version of Apple's Translation

The user can now choose between his instance's server, DeepL (with API
key) and Apple's Translation framework. A translation is cleared if
the translation type is changed. The strings aren't yet written, but
the translations settings view's inconsistent background is now fixed.

* Transfer the old "always_use_deepl" setting

The "always_use_deepl"-setting is now deleted, but its content is
transferred to the equivalent value in "preferred_translation_type".

* Show the user if the DeepL-API key is still stored

The user is now shown a prompt if they've switched away from
.useDeepl, but there's still an API key stored. The API key is not
deleted if the user doesn't instruct the app to do so, so this change
makes it more transparent, since a user might not expect the key to
be stored and might not want this to be the case.

* Localize Labels

The labels for the buttons and options are now localized. "DeepL API Key" is written consistently (with uppercase Key)

* Run all the strings through localization

The strings "DeepL" and "Apple Translate" are now also saved in
localizable.strings and addressed through keys. They were taken
directly previously, which was inconsistent.

* Fix storage

The selected value for preferredTranslationType wasn't stored, the
synchronization between UserPreferences and Storage is now in place.

* Hide Apple Translate if not yet on iOS 17.4

The Apple Translate option is hidden if the user hasn't updated their
phone to at least iOS 17.4. If the Apple Translate option is selected
but the user has downgraded to before iOS 17.4, the standard instance
option is selected.

* Consistently show Apple Translate

Apple Translate was previously only shown if the standard translate
button was visible, that is now fixed. It's now attached to the
StatusRowView, which is always present.

* Animate the removal of translations

The reset of a translation when the translation type is changed is now
animated, which is important for iPad users if they've translated a
post in the sidebar.

* Add support for the Mac Catalyst build

The Mac Catalyst Version doesn't allow the import of the api, so
compiler flags now check if the import isn't allowed and then remove
all references to Apple Translate.

* Swift Format

* Revert "Run all the strings through localization"

This reverts commit 86c5099662.

# Conflicts:
#	Packages/Env/Sources/Env/TranslationType.swift

* Remove the DeepL fallback

The DeepL fallback for the instance translation service is removed,
error messages are shown if a translation fails.

* Allow for the use of an User API Key as fallback

The DeepL fallback is reinstated if the user has put in their own API
Key

* Make the localization keys clear strings

* Make Apple and the instance a fallback

Apple Translate is now a fallback for both other translation types,
the instance service is a fallback for DeepL.
2024-05-13 13:27:21 +02:00
Thomas Ricouard
a8039df22d Don't open link on secondary column 2024-05-13 09:27:24 +02:00
Thomas Ricouard
e21ec0bd1f Add expanded sidebar layout 2024-05-08 11:51:28 +02:00
Thomas Ricouard
9c42a3d7cc Add copy button for alt text 2024-05-08 11:03:25 +02:00
Thomas Ricouard
54a16b2c9a Fix unboost icon 2024-05-08 11:00:40 +02:00
Thomas Ricouard
a6f3068728 Add accounts list placeholder 2024-05-08 10:59:31 +02:00
Thomas Ricouard
f04258ec04 Revert "Delete unused functions in TimelineDatasource.swift (#2037)"
This reverts commit e9a2d3e151.
2024-05-08 10:50:22 +02:00
Cthulhux
8468e51c17
de: Update Localizable.xcstrings (#2057)
(Not entirely sure whether to translate "TimelineFilter" et al.)
2024-05-08 10:39:09 +02:00
Noah Martin
e9a2d3e151
Delete unused functions in TimelineDatasource.swift (#2037) 2024-05-08 10:38:36 +02:00
Igor Camilo
1f56fa1b9b
Add tooltip to sidebar buttons. (#2040) 2024-05-08 10:38:27 +02:00
Jerry Zhang
ccad00a094
Update Simplified Chinese localization (#2052)
* Update Simplified Chinese localization

* Fix

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2024-05-08 10:32:26 +02:00
Thomas Ricouard
51fecb01f5 Bump version to 1.10.39 2024-05-06 09:25:36 +02:00
Thomas Ricouard
c29de44d8c Widget: Mentions only allow large size 2024-05-06 09:20:01 +02:00
Thomas Ricouard
1d79832544 Bump version to 1.10.38 2024-05-06 08:41:33 +02:00
Thomas Ricouard
a37316c56f Lint 2024-05-06 08:38:37 +02:00
Thomas Ricouard
189e10f2b4 Widget: Add mentions widget 2024-05-06 08:37:58 +02:00
Thomas Ricouard
24d5ecd119 Shortcuts: Fix image 2024-05-05 21:13:34 +02:00
Thomas Ricouard
ee6f003073 Widget: Improve text size 2024-05-05 20:03:12 +02:00
Thomas Ricouard
7328c00006 Widget: Remove optional parameters 2024-05-05 19:41:04 +02:00
Thomas Ricouard
a6fd8d1137 Widget: Renaming 2024-05-05 19:37:06 +02:00
Thomas Ricouard
ea31cda3c2 Widget: Add Hashtag widget 2024-05-05 19:31:28 +02:00
Thomas Ricouard
8ab7b5ac69 Fix app group 2024-05-05 19:03:25 +02:00
Thomas Ricouard
7aebe530dd Widget: More UI refinements 2024-05-05 18:29:48 +02:00
Thomas Ricouard
a2afd4f58f Fix widget bundle identifier 2024-05-05 18:16:20 +02:00
Thomas Ricouard
88218cd6ec Style fix 2024-05-05 18:10:32 +02:00
Thomas Ricouard
c4dee39efe More fix for timeline widget 2024-05-05 18:06:47 +02:00
Thomas Ricouard
73651cb7f1 Polish on timeline widget 2024-05-05 17:47:08 +02:00
Thomas Ricouard
dd1615f0e3 Fix widget entitlements 2024-05-05 17:30:09 +02:00
Thomas Ricouard
6bd14e0f8d Don't embed widgets on visionOS 2024-05-05 13:34:22 +02:00
Thomas Ricouard
1ca4a74ff0 Initial widget support 2024-05-05 13:12:19 +02:00
Thomas Ricouard
c3edabb183 Lint 2024-05-04 13:19:19 +02:00
Thomas Ricouard
ba4cc899f8 Add inline post Intent 2024-05-04 13:12:43 +02:00
Thomas Ricouard
5a93184c6d Rename Intent 2024-05-04 11:34:51 +02:00
Thomas Ricouard
66754ecc7c Fix editor progress bar 2024-05-04 11:27:52 +02:00
Thomas Ricouard
e857439a02 Bump version to 1.10.37 2024-05-03 16:25:30 +02:00
Jesús Jiménez Sánchez
ed620e86ca
Update ES localization (#2048)
Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2024-05-03 16:21:40 +02:00
Cthulhux
936bc96ff7
de:Update Localizable.xcstrings (#2051)
Time for a few new strings

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2024-05-03 16:19:32 +02:00
Xabi
37b441a43d
Update EU localisation (#2049)
Added recent strings
2024-05-03 16:18:28 +02:00
Thomas Ricouard
07af494dcb Allow text connect 2024-05-02 12:05:48 +02:00
Thomas Ricouard
49a5c6a56a Add more shortcuts 2024-05-02 11:37:38 +02:00
Thomas Ricouard
4e4d903c44 Add AppIntent service + post to Mastodon intent 2024-05-02 09:32:19 +02:00
Thomas Ricouard
abcd4cc321 Add muted and blocked accounts list 2024-05-02 08:43:58 +02:00
Euigyom Kim
6a7df1065d
Fix scrolling issue on emoji picker (#2032)
* Fix scrolling issue in emoji picker

* Fix design on emoji section header
2024-04-22 16:38:31 +02:00
Thomas Ricouard
c0b855ea55 Bump version to 1.10.36 2024-04-21 15:07:48 +02:00
Thomas Ricouard
4c3047b0b9 Fix packages 2024-04-19 06:56:48 -07:00
Thomas Ricouard
899b92e390 Revert "Fix StatusRowContentView invade SwipeActions area (#2007)"
This reverts commit 3782300b27.
2024-04-19 06:46:56 -07:00
Thomas Ricouard
e71c55b488 Bump version to 1.10.35 2024-04-19 06:46:49 -07:00
Thomas Ricouard
361b5f1d84 Fix tests 2024-04-17 11:11:22 -07:00
Thomas Ricouard
ad61600328 Fix localizations 2024-04-17 11:05:21 -07:00
Thomas Ricouard
5f1f71068c Update packages 2024-04-17 10:55:00 -07:00
tkgka
3782300b27
Fix StatusRowContentView invade SwipeActions area (#2007)
* Fix StatusRowContentView invade SwipeActions area

* ./ add padding inside StatusRowMediaPreviewView
2024-04-17 10:54:46 -07:00
Andrzej Rózga
7d47834903
Polish localization update (#2029) 2024-04-07 09:16:47 +02:00
Jesús Jiménez Sánchez
65a83fa636
Update ES localization (#2028) 2024-04-07 09:16:41 +02:00
Nathan Reed
8038e8e6af
Improve deep link handling on cold start (#2026)
Previously, if the app was not already running when the Safari action extension was used to open a post in the app, the post would open in the in-app Safari instead of using the Ice Cubes UI.
The action extension only worked well if Ice Cubes was already running but backgrounded when it was used.
This was because of the `hasConnection(with:)` check used to ensure that the current server has a federation relationship with the server the post is on.
Early in app launch, the list of federated peers has not come back from the API request yet, so `hasConnection(with:)` was always returning `false`.

To fix, issue a request to fetch the peers as part of the URL handling process, before checking `hasConnection(with:)` to make the final navigation decision.
As an optimization, only do this if `hasConnection(with:)` returns `false` initially -- if it returns `true`, we already know a connection exists so no need to check again.
2024-04-02 08:26:58 +02:00
Ahnaf Mahmud
eb82a67671
Update menu bar and copyright (#2025)
* Update menu bar

* Update copyright
2024-04-02 08:25:52 +02:00
Ege Sucu
bc5bb8272a
TR language updated & Changes Reviewed (#2023) 2024-03-29 10:24:44 +01:00
Xabi
d2ead5b6d1
Update EU localisation (#2017)
newest strings
2024-03-28 15:32:06 +01:00
Jerry Zhang
d22370959c
Update Simplified Chinese localization (#2018)
* Update Simplified Chinese localization

* fix SC localization
2024-03-28 15:31:59 +01:00
Alessio Mason
2c9b841f30
Latest Italian localizations (#2020) 2024-03-28 15:31:39 +01:00
Cthulhux
2e3cf4aace
de: translated the new filters (#2022) 2024-03-28 15:31:31 +01:00
Thomas Ricouard
7de563a6eb Fix packages 2024-03-27 09:17:33 +01:00
Thomas Ricouard
3aae2e6623 bump version to 1.10.34 2024-03-27 09:04:13 +01:00
Thomas Ricouard
5c32c24ae5 Add supports for notifications filter API 2024-03-26 15:49:43 +01:00
Roddy Munro
bb56047ee2
Add SwiftPolyglot to validate localizations (#2011)
* Add step to validate translations on PRs without errors

* Move workflow

* Get specific release of SwiftPolyglot
2024-03-25 08:48:44 +01:00
sh95014
924ada6606
Update zh_Hant loc (#2013)
* checkpoint

* checkpoint

* plurals and a couple of minor fixes

* Update Localizable.strings

* Update Localizable.strings

* Update zh-Hant localizations

* improve translation of "by" in "filtered by"

* update zh-Hant localization

* update Hant localization

* Update zh_Hant

* improve status.poll.closes-in

* Update zh_Hant

* Custom layout for App Store links

* update zh-Hant

* update zh_Hant

* update zh_Hant

* update zh_Hant

* update zh_Hant

* update zh_Hant

* update zh_Hant
2024-03-25 08:47:21 +01:00
Jerry Zhang
e6f96d1899
Update Simplified Chinese localization (#2005) 2024-03-14 09:45:11 +01:00
Thomas Ricouard
058500f91e Swiftformat . 2024-03-11 09:05:52 +01:00
Thomas Ricouard
fd3d9fc2bc Add missing localization 2024-03-11 09:02:32 +01:00
Thomas Ricouard
9a7e6b7cb0 Various fixes for Xcode 15.3 2024-03-11 08:59:29 +01:00
Max von Webel
bc2a09891a
Added a "Moved To" Button to accounts that moved to other instances (#2001)
* added moved information to Account model

* Added "Moved To" button to account details for accounts that have moved
2024-03-11 08:57:35 +01:00
Ahnaf Mahmud
7c343eb4e9
Update visionOS availability (#1999) 2024-03-11 08:56:50 +01:00
Thai D. V
15d7d1dabd
handle edge cases for StatusRowCardView (#1985) 2024-02-26 11:50:10 +01:00
Xabi
f4ec69a37f
Update EU localisation (#1986) 2024-02-26 11:49:32 +01:00
Thomas Ricouard
732a253c7a More EN locale fix 2024-02-26 09:17:59 +01:00
Thomas Ricouard
9c67af8451 Fix EN locale 2024-02-26 09:16:16 +01:00
Thomas Ricouard
b56da94a7c Add more sheets to shared + link to filters in timeline top filters 2024-02-21 09:45:29 +01:00
Alessio Mason
e612fbdf7c
Italian localization fix (#1982)
* General Italian localization overhaul

* Quick IT fix

* Another quick IT fix

* Italian localization fix
2024-02-17 19:26:14 +01:00
Cthulhux
f46a0cee17
de: two more missing strings (#1975)
While using the app, I found that I had not seen two strings. Probably they were marked "translated" too early. Sorry! Here they are.
2024-02-16 08:50:41 +01:00
Andrzej Rózga
4a90d979e3
Polish localization update (#1976)
* Polish localization update

* pl: two more missing strings
2024-02-16 08:50:34 +01:00
Jerry Zhang
9e4323f317
Update Simplified Chinese localization (#1974) 2024-02-14 13:34:19 +01:00
Thomas Ricouard
24ce872849 Add previews + refactor placeholder view 2024-02-14 13:34:06 +01:00
Thomas Ricouard
1f858414d8 format . 2024-02-14 12:48:14 +01:00
Thomas Ricouard
2d988d48c1 Remove some button from status row 2024-02-14 10:48:17 +01:00
Thomas Ricouard
21d9fd7b59 Bump version to 1.10.33 2024-02-14 07:37:15 +01:00
Thomas Ricouard
cca6472a32 Update to Nuke 12.4.0 2024-02-13 18:51:00 +01:00
Thomas Ricouard
c769e80bb6 Add preview for status row 2024-02-13 18:50:51 +01:00
Thomas Ricouard
2986d2b177 Fix env 2024-02-13 17:21:33 +01:00
sh95014
29312d1be2
select a contrasting color for label of "show sensitive content" button (#1965)
* Custom layout for App Store links

* select a contrasting color for label of "show sensitive content" button

fix https://github.com/Dimillian/IceCubesApp/issues/1932

* move contrasting color to Theme and cache computed var
2024-02-13 11:33:59 +01:00
Cthulhux
9ddf0e65fc
Update Localizable.xcstrings (#1969) 2024-02-13 07:50:05 +01:00
Thomas Ricouard
bc74a50a6a Fix some English strings 2024-02-12 07:22:28 +01:00
sh95014
d55d6a0371
Use horizontal link preview card for Apple Podcasts as well (#1966)
* Custom layout for App Store links

* generalize the logic to include links known to be associated with square icons

- such as Apple Music and Spotify

* show Apple Podcasts in horizontal link preview
2024-02-12 07:18:18 +01:00
sh95014
773fdc318b
Plurals for various strings (#1968)
fix https://github.com/Dimillian/IceCubesApp/issues/1936
2024-02-12 07:18:08 +01:00
Thomas Ricouard
7423aba92a Fix crash on visionOS in AboutView 2024-02-11 18:59:34 +01:00
Thomas Ricouard
77aa50ef19 Fix #1873 2024-02-11 18:52:58 +01:00
Thomas Ricouard
dfc213a19a Remove spacer 2024-02-11 18:45:38 +01:00
Andrzej Rózga
20900f573f
Polish localization update (#1958) 2024-02-11 11:14:44 +01:00
Ege Sucu
046a41e8ef
TR Localizations are Updated & some are formatted (#1957) 2024-02-11 11:14:36 +01:00
Thomas Ricouard
fcd56ab7a0 Fix #1960 2024-02-11 11:13:03 +01:00
Thomas Ricouard
923927cddd Cleanup 2024-02-11 11:12:34 +01:00
Thomas Ricouard
219703ecc7 Refactor to NextPageView + handle next page loading failure 2024-02-11 10:58:51 +01:00
Thomas Ricouard
0739264005 Fix background 2024-02-10 12:16:32 +01:00
Thomas Ricouard
6f8bec4737 Add localization 2024-02-10 11:34:49 +01:00
Thomas Ricouard
d8e6e6cfb1 Share Sheet: set cursor before shared content 2024-02-10 11:32:58 +01:00
Thomas Ricouard
e7bc857231 News trending links experience 2024-02-10 11:26:22 +01:00
Thomas Ricouard
35d249f7c9 Bump to 1.10.32 2024-02-10 09:57:56 +01:00
Thomas Ricouard
7b7e65bf31
Update LICENSE 2024-02-10 08:53:49 +01:00
Thomas Ricouard
9542002534 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-02-06 19:19:54 +01:00
Thomas Ricouard
3020d831e4 Various fixes 2024-02-06 19:19:53 +01:00
Thomas Ricouard
a0e022b8de Fix #1948 2024-02-06 17:32:42 +01:00
Thomas Ricouard
b9b3d0e727 Enhance visionOS support 2024-02-06 15:17:20 +01:00
Jerry Zhang
4bf476daea
Update Simplified Chinese localization (#1947) 2024-02-06 13:30:00 +01:00
Ege Sucu
d1fd97794a
Updated TR Localization (#1945) 2024-02-06 13:29:54 +01:00
Thomas Ricouard
f14ca6e529 Various visionOS fixes 2024-02-06 09:15:22 +01:00
Thomas Ricouard
75bb4f43dd More fix for #1943 2024-02-05 14:24:29 +01:00
Thomas Ricouard
cfd6eed159 Fix #1943 2024-02-05 08:59:27 +01:00
Thomas Ricouard
d10adf1fd9 Update packages 2024-02-05 08:56:38 +01:00
Thomas Ricouard
3b07d56b1d Bump version to 1.10.31 2024-02-05 08:56:03 +01:00
Thomas Ricouard
b4dbda8722 Migrate EmojiText API 2024-02-05 08:55:24 +01:00
David Walter
827765f251
EmojiText 4.0.0 (#1941) 2024-02-05 08:49:29 +01:00
Chanhwi Joo
e7702e1ad0
Update Korean localization (#1942) 2024-02-05 08:49:14 +01:00
Thomas Ricouard
70f58aa08d Fix #1939 2024-02-04 12:02:14 +01:00
Thomas Ricouard
cf81054366 Fix isCompact 2024-02-02 18:39:39 +01:00
Thomas Ricouard
f67163e4b0 Cleanup print + use OSLog 2024-02-02 18:26:24 +01:00
Thomas Ricouard
9bd967cddf Fix #1938 2024-02-02 08:53:59 +01:00
Thomas Ricouard
551e6b1412
Delete .github/workflows directory 2024-01-31 10:30:04 +01:00
Thomas Ricouard
1c76d50bde
Create ios.yml 2024-01-31 10:27:51 +01:00
Thomas Ricouard
2a6afb4092 Card title limit to two lines instead of one 2024-01-31 07:57:59 +01:00
Thomas Ricouard
b348f37f1a Add block confirmation 2024-01-31 07:56:50 +01:00
sh95014
de757c58f8
update zh_Hant (#1934)
* checkpoint

* checkpoint

* plurals and a couple of minor fixes

* Update Localizable.strings

* Update Localizable.strings

* Update zh-Hant localizations

* improve translation of "by" in "filtered by"

* update zh-Hant localization

* update Hant localization

* Update zh_Hant

* improve status.poll.closes-in

* Update zh_Hant

* Custom layout for App Store links

* update zh-Hant

* update zh_Hant

* update zh_Hant

* update zh_Hant
2024-01-30 09:22:32 +01:00
Thai D. V
7268b5a38e
Improve StatusPollView (#1929)
* fix `StatusPollView`

* fix text alignment
2024-01-30 09:22:20 +01:00
Thomas Ricouard
b8cf446406 Bump version to 1.10.30 2024-01-29 08:59:35 +01:00
Thomas Ricouard
6e497fae5b Adjust action buttons size 2024-01-29 08:48:24 +01:00
Thai D. V
586e4f525e
Fix timeline media size (#1928)
* fix layout for post with 1 media item

* fix corner radius

* Fixes

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2024-01-28 17:32:16 +01:00
Thomas Ricouard
7f689bbb9c Bump version to 1.10.29 2024-01-27 10:32:30 +01:00
Thomas Ricouard
9dfd9c27c7 Fix actions button on macOS 2024-01-27 10:31:25 +01:00
Thomas Ricouard
9cf16b2f30 Fix memory leak related to video #1925 2024-01-27 10:20:12 +01:00
Thomas Ricouard
1299202bba Fix #1927 2024-01-27 09:52:27 +01:00
Thomas Ricouard
f16f0d514b Fix Swift strict concurrency warnings 2024-01-26 13:01:23 +01:00
Thomas Ricouard
096996c242 Video: Increase compression when in extension 2024-01-24 16:56:35 +01:00
Thomas Ricouard
c7bd5a1d94 Cache Account display name 2024-01-24 16:22:44 +01:00
Thomas Ricouard
20f4eb9c71 Remove section 2024-01-24 13:38:46 +01:00
Thomas Ricouard
74590542bc Custom emojis, botom sheet only 2024-01-24 09:25:34 +01:00
Thomas Ricouard
49b1b0e96c Bump version to 1.10.28 2024-01-24 08:11:32 +01:00
Thomas Ricouard
3eec5c0eec new top / bottom bar 2024-01-23 08:51:58 +01:00
Thomas Ricouard
016e4d5d57 Refactor CancelToolbarItem 2024-01-23 08:13:45 +01:00
Thomas Ricouard
ba071eb4c8 Fix status menu on Catalyst 2024-01-23 06:55:12 +01:00
Thomas Ricouard
d2014d3aec Fix #1918 2024-01-23 06:32:58 +01:00
Thomas Ricouard
621f0d0864 Play video on a separate window on macOS 2024-01-22 22:04:28 +01:00
Thomas Ricouard
eeff60bf98 Add quick look for videos 2024-01-22 21:54:28 +01:00
Thomas Ricouard
245d35db82 Fix buttons touch zone 2024-01-22 21:20:43 +01:00
Thai D. V
62eeba5334
use default frame for preview images that API returns incorrect (width: 0, height: 0) (#1915) 2024-01-22 13:58:01 +01:00
Thomas Ricouard
46b8fbde29 New status row context menu 2024-01-22 09:14:45 +01:00
Thomas Ricouard
9a8568d3fa Fix notifications layout 2024-01-22 09:05:22 +01:00
Thomas Ricouard
a6ccdc029b Timeline filter: Add ControlGroup 2024-01-22 06:38:56 +01:00
Le-Roy Karunaratne
ed9a4a598d
Resolve #359 Optional Missing Alt-Text warning (#1895)
* Resolve #359 Optional Missing Alt-Text warning

Add toggle in settings to require alt text (default off)
If setting is enabled, posting show an error if any attached media is missing alt text

* Re-localized strings
2024-01-22 06:28:03 +01:00
Thai D. V
13af2d7e3f
fix indentation lines (#1914) 2024-01-22 06:27:56 +01:00
Thomas Ricouard
2b446833da Notifications: Full media size + autoplay 2024-01-21 19:46:29 +01:00
Thomas Ricouard
0b96b76641 Bump version to 1.10.27 2024-01-21 19:26:57 +01:00
Thomas Ricouard
78eee1e855 Fix status embed 2024-01-21 18:49:45 +01:00
Thomas Ricouard
b7937e3580 Fix language sheet 2024-01-21 18:27:55 +01:00
Thomas Ricouard
9320b2f114 Fix local timeline -> home switch 2024-01-21 17:05:05 +01:00
Thomas Ricouard
ad7bc999d3 Fix alternate icon 2024-01-21 12:11:36 +01:00
Thomas Ricouard
b41fd2d6ce Bump version to 1.10.26 2024-01-21 12:02:37 +01:00
Thomas Ricouard
a79a181d6f Content Filter view: Blurred background 2024-01-21 11:22:07 +01:00
Thomas Ricouard
fb944f9c48 Editor: Cleanup focus state 2024-01-21 11:13:47 +01:00
Thomas Ricouard
3c82af0273 Custom emojis: blurred background 2024-01-21 11:13:38 +01:00
Thomas Ricouard
3577254f08 VisionOS: Disable theming 2024-01-21 11:13:25 +01:00
Thomas Ricouard
abff6218a4 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-01-21 10:49:46 +01:00
Thomas Ricouard
1be9a7b941 Fix custom emojis loading 2024-01-21 10:49:40 +01:00
Cthulhux
18381e22e2
de: tab 1-5 were untranslated (#1908) 2024-01-21 10:45:19 +01:00
Jerry Zhang
e2273c436a
Update SC Localization (#1909) 2024-01-21 10:45:12 +01:00
Ege Sucu
2296dd4658
Stale localization string removed (#1910) 2024-01-21 10:45:07 +01:00
Thai D. V
92662665b9
fix divide by zero (#1911) 2024-01-21 10:45:00 +01:00
Thomas Ricouard
17387626b8 New App Icon 2024-01-21 09:31:50 +01:00
Thomas Ricouard
e00fd49d89 fix movie supported type 2024-01-20 19:41:44 +01:00
Thomas Ricouard
97798b2c35 Refactor NSItemProvider handler 2024-01-20 19:17:59 +01:00
Thomas Ricouard
90a2a19bb1 Fill poll view 2024-01-20 08:55:12 +01:00
Thomas Ricouard
21d54cc546 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-01-20 07:31:46 +01:00
Thomas Ricouard
76638911ee Bump version to 1.10.25 2024-01-20 07:31:39 +01:00
Thai D. V
0b7fed2e9a
Fix Preview Image Size of StatusRowCardView. (#1904)
* fix preview image size and text spacing

* Fix bg

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2024-01-19 17:48:52 +01:00
Ghulam Mustafa
328ee2d090
Fix #1885 blank options on profile page on macos (#1906) 2024-01-19 17:48:20 +01:00
Thomas Ricouard
ebdd5b9feb Fix #1905 2024-01-19 12:35:41 +01:00
Thai D. V
f79117eff1
fix StatusRowCardView layout (#1901) 2024-01-19 09:01:40 +01:00
Thomas Ricouard
709dd79e25 Fix for visionOS + Remove shimmer 2024-01-19 08:51:29 +01:00
Thai D. V
bf7cdc3712
add scroll bar in StatusRowMediaPreviewView for macOS version (#1896) 2024-01-18 06:43:40 +01:00
Alessio Mason
f12d0600f7
General Italian localization overhaul (#1897)
* General Italian localization overhaul

* Quick IT fix

* Another quick IT fix
2024-01-18 06:42:03 +01:00
emmanuel
b6f11e4e08
Update README.md (#1898) 2024-01-18 06:41:55 +01:00
Thai D. V
76a8f45478
hide context indicator for statuses inside StatusDetailView (#1899) 2024-01-18 06:41:40 +01:00
Thomas Ricouard
e03747aa45 Fix hover effect 2024-01-16 20:55:55 +01:00
Thomas Ricouard
8568d6cc59 Bump to 1.10.24 2024-01-16 19:59:35 +01:00
Thomas Ricouard
1a0b52d268 VisionOS Fixes 2024-01-16 19:32:36 +01:00
Thomas Ricouard
0dea624060 VisionOS fixes 2024-01-16 18:43:09 +01:00
Thomas Ricouard
a4927fd30c VisionOS fixes 2024-01-15 21:15:40 +01:00
Thomas Ricouard
b8be6b79af Bump to 1.10.23 2024-01-15 17:37:33 +01:00
Thomas Ricouard
ddaf4f9fde UI Fixes 2024-01-15 15:03:34 +01:00
Thomas Ricouard
d9f115ba67 Editor: Fix lag 2024-01-15 15:03:28 +01:00
Jesús Jiménez Sánchez
a8f8933e11
Update Spanish translation (#1889) 2024-01-15 10:45:46 +01:00
Yusuke Arakawa
35f6dc8d14
ja: Update Localizable.xcstrings (#1890) 2024-01-15 10:45:32 +01:00
Cthulhux
70eea46aef
de: Update Localizable.xcstrings (#1887) 2024-01-14 10:52:16 +01:00
Thomas Durand
788fab930b
Using AsyncButton from button kit to improve following and related buttons (#1888) 2024-01-14 10:51:54 +01:00
Ahnaf Mahmud
7c8ea23ae9
Correct platform name depending on device (#1883) 2024-01-13 17:51:32 +01:00
Ege Sucu
54dd7521c0
TR Localization Update #1878 (#1880) 2024-01-13 17:50:38 +01:00
Andrzej Rózga
5eccb5c294
Polish localization update (#1882) 2024-01-13 17:50:32 +01:00
Ahnaf Mahmud
254962c5c0
Correct English UK localisation (#1884) 2024-01-13 17:50:26 +01:00
Thomas Ricouard
a737c61d15 Add window default size 2024-01-13 15:50:20 +01:00
Thomas Ricouard
27da37e9ec Fix #1859 2024-01-13 08:48:29 +01:00
Thomas Ricouard
6d12f2528d Fix #1873 2024-01-13 08:39:51 +01:00
Eric
231b622f4e
Added toggleable button to show/hide poll results + animations (#1877)
* Added toggleable button to show/hide poll results

* Animations and localizations

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2024-01-13 08:30:25 +01:00
Jerry Zhang
73a02db57f
Update SC Localization (#1876) 2024-01-13 07:12:22 +01:00
Xabi
ab69a0683b
Update EU localisation (#1872) 2024-01-12 19:52:27 +01:00
Thomas Ricouard
34d8d8bb46 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-01-12 07:04:55 +01:00
Thomas Ricouard
79d062f168 Bump version to 1.10.22 2024-01-12 07:04:49 +01:00
Cthulhux
49e47f3dfd
de: translated timeline content filter (#1864) 2024-01-12 07:04:13 +01:00
Andrzej Rózga
c55259efbb
Polish localization update (#1865) 2024-01-12 07:04:04 +01:00
Ico Davids
99c8ea6f99
Updated NL translations (#1866) 2024-01-12 07:03:57 +01:00
sh95014
a11b6ac2a5
update zh_Hant (#1867)
* checkpoint

* checkpoint

* plurals and a couple of minor fixes

* Update Localizable.strings

* Update Localizable.strings

* Update zh-Hant localizations

* improve translation of "by" in "filtered by"

* update zh-Hant localization

* update Hant localization

* Update zh_Hant

* improve status.poll.closes-in

* Update zh_Hant

* Custom layout for App Store links

* update zh-Hant

* update zh_Hant

* update zh_Hant
2024-01-12 07:03:49 +01:00
Thomas Ricouard
8c68aa711d Content filter detent 2024-01-11 20:02:48 +01:00
Thomas Ricouard
68b7d469f5 Remove some Spacer() 2024-01-11 19:53:21 +01:00
Thomas Ricouard
fab23cbafc Cleanup 2024-01-11 18:56:06 +01:00
Thomas Ricouard
2f5307bfc7 Add timeline content filter 2024-01-11 18:55:35 +01:00
Cthulhux
801b6c5682
de: Update Localizable.xcstrings (#1860)
.. new string
2024-01-11 08:13:21 +01:00
Andrzej Rózga
802eae750d
Polish localization update (#1861) 2024-01-11 08:13:13 +01:00
Thomas Ricouard
afa31411e1 Bump version to 1.10.21 2024-01-11 08:13:02 +01:00
Thomas Ricouard
af2a69fdb1 Fix image sizing 2024-01-10 16:50:18 +01:00
Thomas Ricouard
0da8228e61 Followed Tags + Lists tab. + sidebar customization 2024-01-10 13:26:55 +01:00
Thomas Ricouard
6246d7b0a5 Fill tab only if selected 2024-01-10 08:56:35 +01:00
Thomas Ricouard
b6c3b07ad6 Post context menu sound + haptic feedback + correct icons 2024-01-10 08:48:28 +01:00
Chanhwi Joo
71495181c6
Update Korean localization (#1858) 2024-01-10 07:28:43 +01:00
Henrik Nyh
d3558b761a
Fix "status.action.delete.confirm.message" capitalisation (#1854)
Skipped be (Belarusian) since I wasn't sure about that one, and AI couldn't make up its mind…
2024-01-10 07:28:37 +01:00
Thomas Ricouard
f758b672f1 Fix #1855 2024-01-10 07:28:08 +01:00
Thomas Ricouard
1c1a612d56 Bump version to 1.10.20 2024-01-09 19:11:06 +01:00
Thomas Ricouard
8ea3fa73e5 Fix #1851 (both) 2024-01-09 19:06:54 +01:00
Thomas Ricouard
5c1f113c54 Add a progress indicator when posting a new post 2024-01-09 16:17:34 +01:00
Thomas Ricouard
b7e8f63e86 Multi window visionOS support 2024-01-09 13:28:57 +01:00
Thomas Ricouard
e7864f7089 Add following search fix #1846 2024-01-09 13:28:51 +01:00
Thomas Ricouard
3a173a8cae Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-01-09 10:32:56 +01:00
Thomas Ricouard
b2be3778c1 Prevent video sleep in full screen 2024-01-09 10:32:50 +01:00
Cthulhux
bda192fdc4
de: Update Localizable.xcstrings (#1839) 2024-01-09 10:32:42 +01:00
Ege Sucu
adca09dfcb
TR Localization Updated (#1847) 2024-01-09 10:32:34 +01:00
Jerry Zhang
3e67113c19
Update Simplified Chinese localization (#1850) 2024-01-09 10:32:25 +01:00
Thomas Ricouard
a4cdc6fc18 Don't show count when it's 0 2024-01-09 09:16:01 +01:00
Thomas Ricouard
c847de8f47 Fix #1849 2024-01-09 09:02:06 +01:00
Thomas Ricouard
7ac9a750cb More fixes 2024-01-09 08:44:51 +01:00
Thomas Ricouard
c86d627cee VisionOS fixes 2024-01-09 08:12:22 +01:00
Thomas Ricouard
916c0d9831 Bump version to 1.10.19 2024-01-09 08:12:18 +01:00
Thomas Ricouard
14b25830ff Editor fixes 2024-01-08 21:21:37 +01:00
Thomas Ricouard
556eb15fb4 Button fixes 2024-01-08 21:21:32 +01:00
Thomas Ricouard
8e8713886a Bigger tap area for buttons 2024-01-08 21:21:22 +01:00
Thomas Ricouard
e79ead5efe visionOS fix 2024-01-08 21:21:14 +01:00
Thomas Ricouard
5d24c4d2e8 Fix #1836 2024-01-08 18:22:44 +01:00
Thomas Ricouard
e725b6be4d Create new lists from timeline home menu even if no lists 2024-01-08 18:22:32 +01:00
Thomas Ricouard
6af0c36740 Fix #1840 2024-01-08 18:22:10 +01:00
Thomas Ricouard
4ebe486816 Fix #1841 2024-01-08 12:43:45 +01:00
Thomas Ricouard
753a0574b1 Fix #1842 2024-01-08 12:38:40 +01:00
Thomas Ricouard
9e1b1780c9 Fix invalid images 2024-01-08 12:19:25 +01:00
Thomas Ricouard
738180665e Fix tests 2024-01-07 19:02:22 +01:00
Clemens Beck
334b09ebe3
Feature: Use tagGroup icon in timeline quick access feature (#1834)
* Use tagGroup icon in timeline quick access feature

* Make tagGroup symbol optional
2024-01-07 18:38:45 +01:00
Ege Sucu
6f42ed8de0
Latest localization changes reviewed (#1837) 2024-01-07 18:38:18 +01:00
Thomas Ricouard
c09f9727f1 More video tweaks 2024-01-07 18:33:13 +01:00
Thomas Ricouard
cd63c9ddff Mute video by default 2024-01-07 18:14:12 +01:00
Thomas Ricouard
c4c86e1434 Fix audio session 2024-01-07 17:52:28 +01:00
Thomas Ricouard
2c7ca2ca81 Add new setting to mute video 2024-01-07 17:33:37 +01:00
Thomas Ricouard
7a7066baa4 More fixes to video 2024-01-07 16:49:49 +01:00
Thomas Ricouard
71d12aec15 No play button in video compact mode 2024-01-07 16:33:20 +01:00
Thomas Ricouard
d378341914 Don't autoplay in compact mode 2024-01-07 16:32:36 +01:00
Thomas Ricouard
5ca5dfbd24 better video player 2024-01-07 15:29:59 +01:00
Thomas Ricouard
5ce4d0b41d
Create dependabot.yml 2024-01-07 13:11:23 +01:00
Thomas Ricouard
75987d74aa Fix extensions recent tags 2024-01-07 12:52:45 +01:00
Thomas Ricouard
6aae6f7e40 Fix timeline when resuming from marker 2024-01-07 11:59:15 +01:00
Thomas Ricouard
e6c2146217 Bump version to 1.10.18 2024-01-07 11:08:31 +01:00
Thomas Ricouard
6e981a99fc Various fixes 2024-01-07 10:35:11 +01:00
Thomas Ricouard
27ce7fe916 Editor toolbar icons 2024-01-07 09:47:55 +01:00
Thomas Ricouard
d952601528 Fix crash on macOS 2024-01-07 09:37:18 +01:00
Thomas Ricouard
34a482f01f More layout rework for the composer 2024-01-07 07:03:39 +01:00
Thomas Ricouard
8e8737b040 Move lang in the editor 2024-01-06 22:26:12 +01:00
Thomas Ricouard
a80d36227e Allow video to play sound full screen 2024-01-06 22:08:41 +01:00
Thomas Ricouard
ca9dd5b469 Layout adjustments to the editor 2024-01-06 20:02:16 +01:00
Thomas Ricouard
7eb382c052 Status -> StatusKit 2024-01-06 19:27:26 +01:00
Thomas Ricouard
d65510493a Namespace StatusEditor 2024-01-06 18:43:26 +01:00
Ege Sucu
9ade571f53
Fixed the "Substitution x was never referenced." (#1831)
* Fixed the "Substitution x was never referenced."

* Fixed other locations not getting proper plurals
2024-01-06 14:52:38 +01:00
Thomas Ricouard
c24403094c Naming 2024-01-06 14:49:07 +01:00
Thomas Ricouard
36cc3d5207 ProxyRepresentation -> DataRepresentation 2024-01-06 14:24:52 +01:00
Thomas Ricouard
9329bdf19b Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-01-06 14:19:54 +01:00
Ege Sucu
0058ce36b7
Translated the last added string (#1830) 2024-01-06 14:19:38 +01:00
Thomas Ricouard
8a3c540967 Fix typo 2024-01-06 14:19:28 +01:00
Thomas Ricouard
e5bb521502 composer tweaks 2024-01-06 12:54:51 +01:00
Thomas Ricouard
6435b40a51 Add followed tags to tag suggestion 2024-01-06 12:32:23 +01:00
Thomas Ricouard
1297331407 Fix more warnings 2024-01-06 11:24:41 +01:00
Thomas Ricouard
8c8c551686 Fix warnings + better recently used tags 2024-01-06 11:21:07 +01:00
Thomas Ricouard
bb55154b75 Split/Refactor editor autocomplete 2024-01-06 10:52:04 +01:00
Thomas Ricouard
e4f7a6954b Split settings view + add recently used tags list 2024-01-06 10:51:47 +01:00
Thomas Ricouard
73882b0806 Bump version to 1.10.17 2024-01-06 09:55:23 +01:00
Ege Sucu
e473981841
Turkish content is improved & added missing ones. (#1827) 2024-01-06 09:31:49 +01:00
Thomas Ricouard
0916a80a2e Fix status layout 2024-01-06 06:57:46 +01:00
Thomas Ricouard
a0ff3596cb Revert bottom status padding to 12 2024-01-05 22:48:11 +01:00
Thomas Ricouard
f401d4094d Fix tag follow button 2024-01-05 21:28:46 +01:00
Thomas Ricouard
ef204cf6fd Bump version to 1.10.16 2024-01-05 21:27:06 +01:00
Thomas Ricouard
91f0df0f26 Status actions: Bigger tap target 2024-01-05 20:33:47 +01:00
Thomas Ricouard
a3f29aa15b Add gesture to close the suggested tags 2024-01-05 19:32:17 +01:00
Thomas Ricouard
b3af5f1c45 Fix layout issue in the composer 2024-01-05 19:25:27 +01:00
Thomas Ricouard
0501ce9828 Fix poll for current account 2024-01-05 19:10:31 +01:00
Thomas Ricouard
bfc1f61e4b Add access to profile from account selector 2024-01-05 18:57:02 +01:00
Pavlo Shadov
8152db745d
Fix the "About" button's label in Accounts Selector view (#1826)
* Fix the key used for About in Accounts Selector view

* Improve the translation for the word "Application" in Ukrainian
2024-01-05 18:17:11 +01:00
Thomas Ricouard
1f6e8e8d18 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2024-01-05 18:04:17 +01:00
Thomas Ricouard
d07427b919 Add detent support for composer 2024-01-05 18:04:15 +01:00
Chanhwi Joo
f674c2fa46
Update Korean localization (#1824) 2024-01-05 15:47:13 +01:00
Thomas Ricouard
fcf00796b8 Show recent tags inline with just the # char 2024-01-05 12:13:52 +01:00
Thomas Ricouard
d94e816d63 Dismiss tag on select 2024-01-05 11:35:32 +01:00
Thomas Ricouard
f428118fa0 Recently used tags 2024-01-05 10:57:26 +01:00
Pavlo Shadov
3e968525ac
Update Ukrainian localization (#1817)
* Remove empty string

* Update and add missing localizations

* Revert empty string removal

* Mark "parent view for EditTagGroupView" as translated

* Remove the text dedicated only for Preview
2024-01-05 08:51:53 +01:00
sh95014
e6a4bd383c
shift the square.and.pencil icon so that the square is visually centered in the background rectangle (#1819) 2024-01-05 08:50:51 +01:00
sh95014
72b4a92bfe
update zh_Hant (#1820)
* checkpoint

* checkpoint

* plurals and a couple of minor fixes

* Update Localizable.strings

* Update Localizable.strings

* Update zh-Hant localizations

* improve translation of "by" in "filtered by"

* update zh-Hant localization

* update Hant localization

* Update zh_Hant

* improve status.poll.closes-in

* Update zh_Hant

* Custom layout for App Store links

* update zh-Hant

* update zh_Hant
2024-01-05 08:50:35 +01:00
Jesús Jiménez Sánchez
e61a2a32e4
Add missing Spanish translations (#1822) 2024-01-05 08:50:21 +01:00
Thomas Ricouard
6bb6a02912 Add a new custom Post tab for the tabbar 2024-01-05 08:36:06 +01:00
Thomas Ricouard
c0a78ef007 Bump version to 1.10.15 2024-01-04 21:47:20 +01:00
Thomas Ricouard
80a22e55fa Fix sidebar bottom egde 2024-01-04 21:28:45 +01:00
Thomas Ricouard
c3adb37da0 More fixes for slideover mode 2024-01-04 21:24:22 +01:00
Thomas Ricouard
d399c18a82 Bump version to 1.10.14 2024-01-04 17:48:29 +01:00
Thomas Ricouard
3d29c9e600 Fix account swap for favorites and bookmarks 2024-01-04 16:53:10 +01:00
Thomas Ricouard
9ec9c94c9a DM reply now open on a window on macOS 2024-01-04 16:52:58 +01:00
Thomas Ricouard
60ade66251 Sidebar fixes 2024-01-04 16:42:03 +01:00
Thomas Ricouard
469b99f3c9 Sidebar is now backed by TabView + restore slideover tabbar transition 2024-01-04 16:21:15 +01:00
Thomas Ricouard
fd190378c6 Timeline: Add spacing between loader and unread count 2024-01-04 15:34:40 +01:00
Thomas Ricouard
6e9bff575d Fix #1821 2024-01-04 14:08:24 +01:00
Thomas Ricouard
3a3cae21b0 Fix #1376 for real 2024-01-04 13:19:36 +01:00
Thomas Ricouard
3229bf0cb5 Timeline: Add indicator when loading new posts 2024-01-04 12:56:46 +01:00
Thomas Ricouard
c43d1d0dda Bump version to 1.10.13 2024-01-04 07:48:58 +01:00
Thomas Ricouard
e733dc3f2a iPad hotfix: Disable tabbar slideover 2024-01-04 07:47:59 +01:00
Thomas Ricouard
0e9a006483 Optimize images 2024-01-03 20:53:48 +01:00
Xabi
8459224ab1
Update EU localisation (#1813)
New strings, standardisation and fixes
2024-01-03 16:48:25 +01:00
Cthulhux
d46d47f97d
de: updated fast refresh disclaimer, fixed english grammar ;-) (#1816) 2024-01-03 16:48:19 +01:00
Jerry Zhang
6babd50d6e
Update Simplified Chinese localization (#1814)
* Update SC Localization

* fix: wrong order in tabs SC translation

Co-Authored-By: nixzhu <zhuhongxu@gmail.com>

---------

Co-authored-by: nixzhu <zhuhongxu@gmail.com>
2024-01-03 16:48:06 +01:00
Thomas Ricouard
72f3af7255 Bump version to 1.10.12 2024-01-03 16:47:46 +01:00
Thomas Ricouard
2eb15b48d4 Don't cache non filterable timeline 2024-01-03 14:59:28 +01:00
Thomas Ricouard
ad4995ad70 Catalyst: Fix mention window 2024-01-03 13:40:53 +01:00
Thomas Ricouard
75a61cb534 Fix for status card 2024-01-03 13:22:21 +01:00
Thomas Ricouard
1bdd31e848 Timeline: Fixes 2024-01-03 12:33:06 +01:00
Thomas Ricouard
2e23b08b88 Cache and restore position on all timelines 2024-01-03 11:34:50 +01:00
Thomas Ricouard
6854df4b89 Better fast refresh disclaimer 2024-01-03 10:54:48 +01:00
Thomas Ricouard
73323f8460 Allow quote post preview + link preview fix #1812 2024-01-03 09:23:28 +01:00
Thomas Ricouard
f39005c118 Timeline: Tag heeader now tappable 2024-01-03 09:16:24 +01:00
Thomas Ricouard
326d6e5d50 Bump version to 1.10.11 2024-01-03 06:44:27 +01:00
Thomas Ricouard
6cc14f8249 Better UX for avatar / header selection 2024-01-02 21:50:11 +01:00
Thomas Ricouard
1eb33466ca Edit profile: Update avatar & header 2024-01-02 21:16:27 +01:00
Thomas Ricouard
f699c33dfb Make status action button areas bigger 2024-01-02 19:35:14 +01:00
Thomas Ricouard
f6b7b9807f Fix #1808 2024-01-02 18:55:36 +01:00
Hao Song
245f13d59f
Fix settings done button logic (#1809)
* Fix settings done button logic

With the new customizable tab bar, "Settings" can be rendered directly
as the root view of a tab bar entry instead of as a "modal" presented
from the root view. The "Done" button to dismiss the modal should be
hidden if `isModal` is `false`.

* Hide Settings secondary column button for iPad when presented as a modal
2024-01-02 18:50:50 +01:00
Andrea Draghetti
442a7938ca
Improved the Italian translation (#1810) 2024-01-02 15:16:07 +01:00
sh95014
c4de2d6784
zh-Hant localization (#1804)
* checkpoint

* checkpoint

* plurals and a couple of minor fixes

* Update Localizable.strings

* Update Localizable.strings

* Update zh-Hant localizations

* improve translation of "by" in "filtered by"

* update zh-Hant localization

* update Hant localization

* Update zh_Hant

* improve status.poll.closes-in

* Update zh_Hant

* Custom layout for App Store links

* update zh-Hant
2024-01-02 15:16:00 +01:00
Thomas Ricouard
632b3f5734 Statuses: Cleanup viewId 2024-01-02 14:06:53 +01:00
Thomas Ricouard
924e1b6057 Bump version to 1.10.10 2024-01-02 13:54:26 +01:00
Thomas Ricouard
983c22886a Rework icons order 2024-01-02 13:29:21 +01:00
Thomas Ricouard
f19ab2b130 Increase contrast on the unread counter 2024-01-02 12:14:53 +01:00
Thomas Ricouard
13e87b41e9 Fix list not updating in the quick access pills 2024-01-02 11:54:17 +01:00
Thomas Ricouard
c4b85679a2 StatusRow: Light cleanup 2024-01-02 08:58:20 +01:00
Thomas Ricouard
0c13cbd61f Filters: Add current editing keyword on save 2024-01-01 21:29:38 +01:00
Thomas Ricouard
aee6459bcf Status detail: Don't cancel the task 2024-01-01 21:29:28 +01:00
Thomas Ricouard
f235ebb720 Remove the double # in quick access pills 2024-01-01 21:29:12 +01:00
Thomas Ricouard
47436daaf2 Proper fix for looping timeline 2024-01-01 21:29:03 +01:00
Thomas Ricouard
1e7c25993a Fix timeline filtering + looping 2024-01-01 21:06:10 +01:00
Thomas Ricouard
6f1896caf3 Bump version to 1.10.9 2024-01-01 19:33:28 +01:00
Thomas Ricouard
4fee875fa7 Fix timeline top padding 2024-01-01 19:06:14 +01:00
Thomas Ricouard
e41dcd6976 Fix timeline tag group filter 2024-01-01 19:00:23 +01:00
Thomas Ricouard
bd51dfc0b6 Fix safe area on media viewer 2024-01-01 18:31:55 +01:00
Thomas Ricouard
ce845cd6b3 Better timeline top pin view 2024-01-01 18:23:03 +01:00
Thomas Ricouard
f0061b36ca Account: Show nav bar title for main, followers, following 2024-01-01 17:29:15 +01:00
Thomas Ricouard
8ee5da319c Reflect edit / post / delete status better 2024-01-01 16:46:34 +01:00
Thomas Ricouard
8c72b627df Fix scroll to id 2024-01-01 14:16:42 +01:00
Thomas Ricouard
b10ee3091c Make edit stickier 2024-01-01 14:13:25 +01:00
Thomas Ricouard
b93df71431 Don't resize images if same size 2024-01-01 11:39:32 +01:00
Thomas Ricouard
b6317d7324 Add tests for Router 2024-01-01 09:48:53 +01:00
Thomas Ricouard
7222d530dd Fix url router 2024-01-01 09:23:06 +01:00
Thomas Ricouard
b4757621f2 Only show the new icon card on iPad / macOS 2023-12-31 13:43:13 +01:00
Thomas Ricouard
9b70519798 Fix sensitive content transition 2023-12-31 13:28:27 +01:00
Thomas Ricouard
a85c701f50 Fix #1639 2023-12-31 13:17:09 +01:00
Thomas Ricouard
7add850fe6 Rename timeline related files 2023-12-31 11:18:42 +01:00
Thomas Ricouard
acccdb8041 Load new posts per 100 instead of 200 2023-12-31 08:11:53 +01:00
Thomas Ricouard
3a721d3280 Bump version to 1.10.8 2023-12-30 19:09:25 +01:00
Xabi
e694dd5529
Update EU localisation (#1801)
The following strings were marked as translated even if they were not:
- tabs customizations
- first tab
- second tab
- fourth tab
- fifth tab

Is there a way to sort strings by date on Xcode?
2023-12-30 16:16:43 +01:00
Thomas Ricouard
0497191acf Add icons for all filters 2023-12-30 16:16:19 +01:00
Thomas Ricouard
781121d1d4 Add more tests 2023-12-30 15:40:04 +01:00
Thomas Ricouard
fe66acbd39 Timeline: Add pills quick access 2023-12-30 14:54:09 +01:00
Thomas Ricouard
631707a798 Refactor TimelineView 2023-12-30 12:30:02 +01:00
Thomas Ricouard
6ea4888ae5 Add more timeline tests 2023-12-30 09:51:34 +01:00
Thomas Ricouard
b0cc02541e Fix modelTests 2023-12-30 09:16:18 +01:00
Thomas Ricouard
8a2861b37f Add stream tests 2023-12-30 09:08:19 +01:00
Thomas Ricouard
176e4feaf8 Move tabbar label settings 2023-12-30 07:34:47 +01:00
Thomas Ricouard
b4013e39c0 Fix DM view 2023-12-29 18:56:50 +01:00
Thomas Ricouard
c328c6c0be Refactor App level to App + App View -> Slideover = phone layout on iPad 2023-12-29 18:50:53 +01:00
Thomas Ricouard
84898c3b8e Fix #1798 2023-12-29 18:16:23 +01:00
Thomas Ricouard
2bdef66da0 Actions buttons: Bigger tap area 2023-12-29 17:57:37 +01:00
Thomas Ricouard
bb39f07503 Bump to 1.10.7 2023-12-29 17:06:37 +01:00
Thomas Ricouard
6359349a40 VisionOS fix 2023-12-29 17:06:27 +01:00
Thomas Ricouard
c9dc24d02a visionOS icon 2023-12-29 16:29:48 +01:00
Thomas Ricouard
59bd8a437a Update macOS icon 2023-12-29 16:14:35 +01:00
Thomas Ricouard
67969f595a Fix About view crash on macOS 2023-12-29 14:21:12 +01:00
Thomas Ricouard
59c0b841c2 Update readme 2023-12-29 13:00:17 +01:00
Thomas Ricouard
f39f9e1363 Add more features to the readme 2023-12-29 12:31:34 +01:00
Thomas Ricouard
5209ab80fc New readme images and content 2023-12-29 12:03:46 +01:00
Thomas Ricouard
89c060aeea WIP New readme 2023-12-29 11:52:12 +01:00
Thomas Ricouard
1a366c7bd7 New default icon 2023-12-29 09:40:20 +01:00
Thomas Ricouard
d04f6d34ce Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2023-12-29 08:01:11 +01:00
Thomas Ricouard
4ba8d004d1 Add Threads themes 2023-12-29 08:01:09 +01:00
Thomas Ricouard
83e752ce63 Bump version to 1.10.6 2023-12-29 07:34:30 +01:00
Cthulhux
32119e67c1
de: Update Localizable.xcstrings (#1791)
More string translations.. :-)

Note: you forgot to set "needs_review" for some of them this time.
2023-12-29 07:33:10 +01:00
Xabi
52d726f9b4
Update EU localisation (#1793)
Translated strings that needed review
2023-12-29 07:33:04 +01:00
Thomas Ricouard
19715bb1f6 Fix local / trending tab showing federated timeline 2023-12-29 07:32:33 +01:00
Thomas Ricouard
7b484fc8e1 Add loading indicator on profile 2023-12-28 22:18:13 +01:00
Thomas Ricouard
dac9fc55e9 Scale share button 2023-12-28 22:03:24 +01:00
Thomas Ricouard
dcd63cfd54 Fix #1789 2023-12-28 21:57:55 +01:00
Thomas Ricouard
a1093c8052 Animate favorite / boost count update 2023-12-28 21:57:41 +01:00
Thomas Ricouard
01cd65e6ac Bump version to 1.10.5 2023-12-28 21:49:50 +01:00
Thomas Ricouard
f761fa7117 Bump version to 1.10.4 2023-12-28 16:15:30 +01:00
Thomas Ricouard
6850fcd928 Add support / about shortcuts in account selector bottom sheet 2023-12-28 16:03:16 +01:00
Thomas Ricouard
b6370aef98 Fix about icons 2023-12-28 15:44:40 +01:00
Thomas Ricouard
b83f7e9a55 Fix notifications reload 2023-12-28 13:48:50 +01:00
Thomas Ricouard
00e35be2d5 Fix counter 2023-12-28 13:14:11 +01:00
Thomas Ricouard
f3043b608c Better date pre compute 2023-12-28 12:31:16 +01:00
Thomas Ricouard
8f0548f45d Optimize timeline 2023-12-28 12:26:09 +01:00
Thomas Ricouard
fa4603e77c Fix notifications locked type 2023-12-28 12:03:01 +01:00
Thomas Ricouard
a09b2fa95e monospacedDigit 2023-12-28 11:56:23 +01:00
Thomas Ricouard
d0c2cd4520 Better unread counter + animation 2023-12-28 11:54:41 +01:00
Thomas Ricouard
5c2148104c Add Tabs customization on iOS 2023-12-28 11:26:00 +01:00
Thomas Ricouard
b0ba6c15da Add favorites / bookmarks tab on macOS / iPadOS 2023-12-28 09:37:02 +01:00
Thomas Ricouard
f79580f746 Bump version to 1.10.3 2023-12-28 07:58:38 +01:00
Thomas Ricouard
e0563122a7 Layout fix & tweak 2023-12-28 07:55:17 +01:00
Cthulhux
aa41f24de9
de: Update Localizable.xcstrings (#1781)
new string..
2023-12-28 07:49:01 +01:00
Jerry Zhang
506a158fa4
Update Simplified Chinese Localization (#1782)
* Update SC Localization

* Fix: refresh timeline SC localization

Co-Authored-By: nixzhu <zhuhongxu@gmail.com>

---------

Co-authored-by: nixzhu <zhuhongxu@gmail.com>
2023-12-28 07:48:55 +01:00
Jerry Zhang
82d0e3e576
Update GB Localization (#1783) 2023-12-28 07:48:49 +01:00
Thomas Ricouard
d065ae6aa8 Add settings to toggle between share and bookmark button 2023-12-28 07:48:35 +01:00
Thomas Ricouard
44dbd379ba Further layout tweaks 2023-12-27 19:28:16 +01:00
Thomas Ricouard
d300bee96f Bump to version 1.10.2 2023-12-27 18:53:11 +01:00
Thomas Ricouard
21ac4cfa21 Further tweak the layout 2023-12-27 18:16:59 +01:00
Thomas Ricouard
bfa717bfa2 Shorter date for status < to 24H 2023-12-27 18:05:41 +01:00
Thomas Ricouard
e53a3d0f61 Fix reasons check 2023-12-27 17:12:48 +01:00
Thomas Ricouard
2b16b10987 Tweak status detail layout 2023-12-27 16:48:50 +01:00
Thomas Ricouard
9a5457946b Bump version to 1.10.1 2023-12-27 16:40:29 +01:00
Thomas Ricouard
e6b3113090 Add more information for the fast refresh toggle 2023-12-27 16:27:31 +01:00
Thomas Ricouard
8a0cf44834 Visually align new post button 2023-12-27 16:27:21 +01:00
Thomas Ricouard
1a3bded101 New default timeline layout 2023-12-27 16:07:16 +01:00
Thomas Ricouard
2e1652ef53 Fix "Only buttons" settings 2023-12-27 15:16:53 +01:00
Thomas Ricouard
962c7c0295 Timeline: Basic timeline sync using the marker API 2023-12-27 13:26:30 +01:00
Thomas Ricouard
590299d102 Sync with markers API for notifications 2023-12-26 16:01:02 +01:00
Thomas Ricouard
4de4c7c82a Don't remove notifications 2023-12-26 15:12:08 +01:00
Thomas Ricouard
ede45a9d46 Fix #1769 2023-12-26 13:56:28 +01:00
Thomas Ricouard
f3a48118f0 Fix #1772 2023-12-26 13:51:41 +01:00
Thomas Ricouard
c4bff07c40 Bump version to 1.10.0 2023-12-26 13:40:39 +01:00
Thai D. V
f326bbefe6
Fix: Search Instances Feature (#1766)
* fix: search logic and performance

* Remove overlay

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-12-26 13:31:22 +01:00
Thomas Ricouard
3ac1bf362b Use alternate icons, remove duplicate assets 2023-12-26 13:24:39 +01:00
Thomas Ricouard
5f05248523 Fix #1768 2023-12-21 21:10:46 +01:00
Thomas Ricouard
f04f5c701c Fix #1767 2023-12-21 21:00:44 +01:00
Thomas Ricouard
3eb373550e visionOS: Fixes 2023-12-19 15:07:51 +01:00
Thomas Ricouard
2fdaed7df4 Add direct access to push notifications settings in notifications tab 2023-12-19 09:58:35 +01:00
Thomas Ricouard
5a2478c791
VisionOS native support (#1758)
* Initial support

* UI Adjustments

* WIP icons

* More UI
2023-12-19 09:51:20 +01:00
Thai D. V
ca13e61b53
fix: AddAccountView (#1764)
* fix typo

* format number of users and posts

* add thumbnail and re-layout
2023-12-19 07:25:31 +01:00
Jerry Zhang
260dbd351a
Update SC Localization (#1761) 2023-12-19 06:46:47 +01:00
sh95014
d69696b726
separator should extend to leading margin (#1763) 2023-12-19 06:46:39 +01:00
Thomas Ricouard
9fb8d4e484 Bump version to 1.9.19 2023-12-18 09:00:29 +01:00
Thomas Ricouard
8ff3e22d9f SwiftFormat 2023-12-18 08:22:59 +01:00
sh95014
2145bd5971
AppStore Link Preview (#1756)
* Custom layout for App Store links

* generalize the logic to include links known to be associated with square icons

- such as Apple Music and Spotify
2023-12-18 07:01:46 +01:00
David Walter
d755396119
Update EmojiText to 3.2.1 (#1753) 2023-12-18 06:43:52 +01:00
Cthulhux
5aa9f22b42
de: translated popover setting (#1755) 2023-12-18 06:43:34 +01:00
Thomas Ricouard
6f6e352baf Mixin portrait and landscape media in the status medias carousel 2023-12-17 10:41:42 +01:00
Thomas Ricouard
1fa54afc3a Add support for GIPHY + rework loading of the media in the editor 2023-12-17 10:27:01 +01:00
Thomas Ricouard
4985e69200 Allow popover settings everywhere 2023-12-17 09:21:09 +01:00
sh95014
f9da958047
add a setting to disable the account popover on hover (#1750)
* add a setting to disable the account popover on hover

- not entirely pleased with the AnyView() cast but don't really know of a less invasive change

* Fixes

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-12-17 07:57:20 +01:00
Thomas Ricouard
a7f982e827 Fix posting order for threads 2023-12-17 07:52:16 +01:00
sh95014
8f37465be0
Update zh-Hant localization (#1745)
* checkpoint

* checkpoint

* plurals and a couple of minor fixes

* Update Localizable.strings

* Update Localizable.strings

* Update zh-Hant localizations

* improve translation of "by" in "filtered by"

* update zh-Hant localization

* update Hant localization

* Update zh_Hant

* improve status.poll.closes-in

* Update zh_Hant
2023-12-17 07:34:08 +01:00
Cthulhux
81f4276596
de: Update Localizable.xcstrings (#1743) 2023-12-17 07:34:01 +01:00
sh95014
038d029022
group slider and text of "max reply indentation" setting (#1749) 2023-12-17 07:33:55 +01:00
Thomas Ricouard
0d37cdf64b Bump version to 1.9.18 2023-12-17 07:28:59 +01:00
David Walter
47326b3f7a
Update EmojiText to 3.2.0 (#1751)
Fixes #1738
2023-12-17 07:28:18 +01:00
Thomas Ricouard
8f31e34e1d Fix colors 2023-12-15 20:13:50 +01:00
Thomas Ricouard
98c732e6fb Bump version to 1.9.17 2023-12-15 20:08:14 +01:00
Jerry Zhang
880277c6f3
Update SC Localization (#1742)
* Update SC Localization

* Update SC Localization on indentation
2023-12-15 05:55:25 +01:00
Thai D. V
1977b1a572
Feature: Post and Reply by Threads (#1740)
* refactor: `StatusEditorView`

* feat: post and reply by threads

* Tidy up

* Fixes

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-12-14 08:06:24 +01:00
Paul Schuetz
d8a686be51
Allow the user to customize the thread indentation (#1737)
* Allow the user to customize the thread indentation

The user can now select if they want to indent threads/replies, and how much
the replies should be indented.

* Make the wording clearer

The wording is now clearer since "thread" is replaced by "reply".

* Fix localizations

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-12-14 07:17:09 +01:00
Paul Schuetz
e4df8a8b69
Link to parent post (#1736)
The "reply to ..."-text is now a link to the parent post. A tap scrolls to the
parent if the whole hierarchy over a post is shown (detail view). Otherwise,
the detail view for the parent is opened.
2023-12-14 07:12:12 +01:00
Thai D. V
81ba1e9bee
Add Select Status Text Action (#1731)
* add select text action

* Fixes

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-12-14 06:46:16 +01:00
Thomas Ricouard
71f090552a Bump version to 1.9.16 2023-12-13 19:49:15 +01:00
Thomas Ricouard
8d7b6f382e Fixes & optimizations 2023-12-13 12:37:07 +01:00
Thomas Ricouard
232e031559 Add charts for tags 2023-12-13 09:05:30 +01:00
Thomas Ricouard
d31af12bb6 Add pull to refresh on post detail 2023-12-10 08:42:26 +01:00
Thomas Ricouard
c11a31955c Bump version to 1.9.15 2023-12-09 13:08:07 +01:00
Thai D. V
f3ef79b297
Relayout media on status editor (#1728)
* relayout media display

* animate media layout

* fix layout
2023-12-09 10:59:10 +01:00
Thomas Ricouard
52208ab20e Poxy OpenAI calls and remove OpenAI secrets from the app 2023-12-09 10:58:42 +01:00
Thomas Ricouard
da6c5ed76c Add follow section in about 2023-12-08 08:04:35 +01:00
Thomas Ricouard
382ebd77e6 Bump version to 1.9.14 2023-12-08 06:58:26 +01:00
Xabi
9aa64f261a
Missing strings (#1725) 2023-12-07 20:15:18 +01:00
Thomas Ricouard
9ab394272f Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2023-12-07 18:48:19 +01:00
Thomas Ricouard
052afd5931 New media carrousel 2023-12-07 18:48:18 +01:00
Jesús Jiménez Sánchez
fdcdf3453f
Missing Spanish translations (#1724) 2023-12-07 13:51:47 +01:00
Chanhwi Joo
6a34b4c9df
Update Korean localization (#1723) 2023-12-07 11:23:08 +01:00
Thomas Ricouard
51656794fc Use env webAuthenticationSession 2023-12-07 09:45:34 +01:00
Thomas Ricouard
5941276145 Update dependencies 2023-12-07 09:29:03 +01:00
Thai D. V
774ba834bd
Improve media selection on the status editor. (#1722)
* show menu buttons on media item

* fix media preparing logic
- not removing photo pickers when removing media on the post editor
- pickers don't have identifiers after being selected
- preparing tasks (creating containers, uploading media) don't run in parallel
- re-preparing the whole media list every time adding new ones

* remove measurement code

* rename variables

* fix MainActor mutation
2023-12-07 06:39:34 +01:00
Andrzej Rózga
9fe5994bb2
Polish localization update (#1719) 2023-12-07 06:10:01 +01:00
Thomas Ricouard
f2cd05968e Add a translate button after generating image description 2023-12-06 21:04:47 +01:00
Thomas Ricouard
4f9e23296f Cleanup 2023-12-06 18:56:19 +01:00
Thomas Ricouard
3d2171d716 Refactor auth to ASWebAuthenticationSession 2023-12-06 08:05:26 +01:00
Jerry Zhang
a6f6aa3a02
Update Simplified Chinese localization (#1714)
* Update SC Localization

* Adjust SC localization

Co-Authored-By: nixzhu <zhuhongxu@gmail.com>

---------

Co-authored-by: nixzhu <zhuhongxu@gmail.com>
2023-12-06 06:41:41 +01:00
Thai D. V
2e350f5fce
move environment runtime check to compile time (#1709) 2023-12-06 06:41:26 +01:00
Thomas Ricouard
df1a44cc21 Use V1 accounts API for autocomplete 2023-12-05 21:03:47 +01:00
Thai D. V
330aa93437
photo view ignores bottom edge of safe area (#1708) 2023-12-05 08:23:42 +01:00
Xabi
a86048de33
Update Basque localisation (#1707)
Translated newest strings.
2023-12-05 08:15:38 +01:00
Cthulhux
afcd49cb69
de: update Localizable.xcstrings: added "Generate description" (#1706) 2023-12-05 08:15:31 +01:00
Thomas Ricouard
2ff724c268 Bump version to 1.9.13 2023-12-04 20:53:01 +01:00
Thomas Ricouard
4dbe04a5d4 Better image alt prompt 2023-12-04 20:36:15 +01:00
Thomas Ricouard
28ab417b0a Generation image description using GPT Vision 2023-12-04 20:04:12 +01:00
Thomas Ricouard
5c204fd06f New Christmas icons 2023-12-04 20:03:41 +01:00
Thomas Ricouard
fcaf48ce53 foregroundColor -> foregroundStyle 2023-12-04 15:49:44 +01:00
Thomas Ricouard
3840b8fb28 presentationMode -> dismiss 2023-12-04 14:14:42 +01:00
Thomas Ricouard
4a09989160 Fix tag condition 2023-12-04 11:49:36 +01:00
Thomas Ricouard
947a684ce3 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2023-12-04 09:52:20 +01:00
Thomas Ricouard
76219f553b Add tag status indicator for home timeline 2023-12-04 09:52:18 +01:00
Cthulhux
0166a892d8
Update Localizable.xcstrings: added de string (#1705) 2023-12-04 09:04:30 +01:00
Thomas Ricouard
bfc2994cfb Fix iCloue container 2023-12-03 14:15:35 +01:00
Thomas Durand
8d54f1a359
Using BUNDLE_ID_PREFIX to fix fork building with iCloud container (#1703) 2023-12-03 13:55:15 +01:00
Thomas Durand
ad2adadf87
Added a "Continues a thread" label to status rows (#1704) 2023-12-03 13:54:57 +01:00
Thai D. V
56360ae821
fix: make windowWidth and windowHeight of SceneDelegate observable (#1693) 2023-12-03 12:43:15 +01:00
Cthulhux
bf65c386e6
Update Localizable.xcstrings: updated de (#1699) 2023-12-03 08:18:22 +01:00
Thomas Ricouard
b249b37612 Add a fast refresh option 2023-12-01 08:51:19 +01:00
Thomas Ricouard
f89b3d2761 Bump version to 1.9.12 2023-12-01 08:51:12 +01:00
Thomas Ricouard
719eb34701 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2023-12-01 08:15:00 +01:00
Thomas Ricouard
885a134eaf Fix list on older instances 2023-12-01 08:14:57 +01:00
Chanhwi Joo
1d04a51fb0
Update Korean localization (#1697) 2023-12-01 08:14:17 +01:00
Thomas Ricouard
32be7d4460 bump version to 1.9.11 2023-11-30 12:26:54 +01:00
Thomas Ricouard
28c8e4d60e Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2023-11-30 12:02:52 +01:00
Thomas Ricouard
222daae47c Add new icons 2023-11-30 12:02:50 +01:00
Cthulhux
efe0bdcdad
German translation: added a few strings, trying to fix #1584 (#1694) 2023-11-29 09:02:24 +01:00
Hugo Saynac
69d5f265fe
Improve indentation level design for indentations level > 1 (#1695) 2023-11-29 09:02:01 +01:00
Jerry Zhang
94670762a4
Update SC Localization (#1696) 2023-11-29 09:01:36 +01:00
Thomas Ricouard
12419a77e2 Fix build for real 2023-11-28 14:41:17 +01:00
Thomas Ricouard
662f1002f5 Fix build 2023-11-28 14:32:35 +01:00
Thomas Ricouard
f2606b4614 Search users in list edit 2023-11-28 14:16:04 +01:00
Thomas Ricouard
ab07fb5906 Update localizations 2023-11-28 09:35:26 +01:00
Thai D. V
1f703fc1f4
add localization to EditTagGroupView (#1692) 2023-11-28 09:19:48 +01:00
Thomas Ricouard
2e2a9f5f14 Add more lists setttings 2023-11-28 09:18:52 +01:00
Thomas Ricouard
d2f7ab1464 Move AccountPopoverView 2023-11-27 09:19:43 +01:00
Nathan Reed
06a8ca67c3
Improve display of HTML ul (bullet list) and ol (numbered list) (#1690)
While SwiftUI's `Text` view won't display these in an `AttributedString` even if they get parsed from Markdown (which would also require the use of the `.full` option instead of the `.inlineOnlyPresrevingWhitespace` option), we can improve the appearance somewhat.
Currently, list elements are clumped together with no spaces between them, and there's no indication whatsoever that the author indicated these to be a list.
Change to insert Markdown list syntax with linebreaks and dashes, so users can at least understand there's a list there.
Similar change for ordered lists.

This will still be broken for nested lists, but it didn't seem worth it to put a lot of effort into this (or other revamps, like making bold/italics/code work properly) because it seems like the current text handling in Ice Cubes is suboptimal and eventually slated for improvement (according to https://github.com/Dimillian/IceCubesApp/issues/1459#issuecomment-1638562657).
So this is more designed to make lists "less broken" in some cases, rather than be a comprehensive fix for all lists in all cases.
2023-11-27 09:15:27 +01:00
Thai D. V
de83b8ec90
Fix EditTagGroupView (#1686)
* refactor data of `EditTagGroupView`

* lower case tags before saving because API is case-insensitive

* fix: "add new tag" `TextField` is not focused after adding the first tag (on both macOS and iOS)

* perf: improve symbol search performance

* improve layout and animation of symbol search

* fix: sort tags and remove duplicate tags

* fix: crash when open timeline for an empty tag group

* fix: revert concurrency code because performance issue at 1d3f271 is a false alarm

* add warning labels to help the users

* fix: state `tagGroup`

* fix: selecting symbol logic and warning labels

* refactor `EditTagGroupView.body`

* refactor warning labels

* Fix theme

* Move to its own folder

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-11-27 09:13:07 +01:00
Thai D. V
ea5480ef46
add account popovers for display name and handle (#1687) 2023-11-27 09:00:52 +01:00
Simone Margio
98e8ffe4a3
Update Italian translation (#1688) 2023-11-27 08:57:27 +01:00
Jerry Zhang
1af75fbede
Update SC Localization (#1684) 2023-11-24 19:30:00 +01:00
Thomas Ricouard
3a23afed89 Bump version to 1.9.10 2023-11-20 19:27:20 +01:00
Thomas Ricouard
b3153289c4 Fix to account selector on iOS 2023-11-20 18:43:16 +01:00
Thomas Ricouard
47d54fd9e6 Fixes 2023-11-20 17:20:09 +01:00
Thai D. V
94172cef27
Feature: popover the account overview when hovering on the avatar (#1682)
* fix avatar scale

* refactor avatar  config data

* add `AvatarView_Previews`

* refactor shape and placeholder of avatar

* refactor `AvatarView` and add `AvatarPopup`

* add `hoverEffect` for iPad

* fix auto-dismiss bug

* fix `showPopup` bug

* disable inappropriate avatar popups
2023-11-20 10:59:49 +01:00
Oleg
534b098ca6
Hide settings link on macos (#1681) 2023-11-20 09:27:58 +01:00
Thomas Ricouard
cc03465956 Fix missing localizations 2023-11-20 08:51:44 +01:00
Thomas Ricouard
18f95bdf92 Bump version to 1.9.9 2023-11-19 08:45:43 +01:00
Thomas Ricouard
71ab8d558a Fix / simplify account content warning 2023-11-19 08:26:07 +01:00
Paul Schuetz
12d92ab1ec
Add hint if the server post options are overridden (#1679)
If the content settings specify their own post settings and override the
instance settings, a hint (and link to the content settings) is added to the
instance settings (infos) since that setting might introduce confusion (As
happened in #1677).
2023-11-19 08:10:53 +01:00
Paul Schuetz
8bf36709ea
Fix reply indentation when the post has pictures (#1678)
The size of the image is now set correctly to prevent the shifting of the
vertical bars. The handling of the compact view (regarding the indentation) is
now centrally handled in StatusDetailView.
2023-11-19 08:10:44 +01:00
Euigyom Kim
d3b52b3206
Make categorized emoji picker (#1680)
Signed-off-by: Euigyom Kim <egkim@dehol.kr>
2023-11-19 08:09:41 +01:00
Thomas Ricouard
1e35ffb82b Fix settings close button 2023-11-18 11:58:04 +01:00
Thomas Ricouard
c1c7c666cb Disable indentation in compact post 2023-11-18 10:44:27 +01:00
sh95014
11388757f3
Limit image height to screen height (#1675)
* limit image height to window height minus a hardcoded value

https://github.com/Dimillian/IceCubesApp/issues/1554

* Limit image to screen height

- limit available height to 80% of screen/window height
- if image fits in available width and height, just display it at 1x (to avoid ugly resizing artifacts)
- otherwise, shrink it proportionally to fit

https://github.com/Dimillian/IceCubesApp/issues/1554
2023-11-17 09:42:33 +01:00
Paul Schuetz
59e5eba860
Improve the display of replies (#1672)
Threads/replies are now shown more clearly. Each reply has an indentation level
(and therefore the number of vertical lines) one more than its direct parent.
This leads to siblings having the same indentation level. It makes
understanding somewhat complex thread structures way easier. Previously, a
reply was only indented if it came directly after its parent. If a toot had
more than one reply, the structure was nearly indecipherable, as it wasn't
clear which the parent post of the second (or later) toot was. An example is
"https://mastodon.social/@mhoye/110452462852364819" and all of its replies.

Signed-off-by: Paul Schuetz <pa.schuetz@web.de>
2023-11-16 09:56:00 +01:00
Thai D. V
4b74532048
Feature: store selected notification filter (#1627) (#1663)
* store selected notification filter (#1627)

* store one filter for all accounts
2023-11-16 09:53:16 +01:00
Thomas Ricouard
58d6a3b472 Bump version to 1.9.8 2023-11-16 09:51:56 +01:00
Thomas Ricouard
f451d7cb8c Bigger media viewer window 2023-11-14 19:48:14 +01:00
Thomas Ricouard
bf618d3c5f Update packages 2023-11-14 19:47:51 +01:00
Thomas Ricouard
0c50071ae6 Fix scheme 2023-11-13 08:34:36 +01:00
Madeline
d30fcb8c9b
Update project PRODUCT_NAME (#1664)
Add constant value for product_name in target IceCubesApp.

Replaces $(TARGET_NAME), which resulted in an incorrect displayed name
when ran under macOS Catalyst.

Value is now set to constant "Ice Cubes".
2023-11-13 08:22:44 +01:00
sh95014
33145eaafc
Update zh_Hant (#1671)
* checkpoint

* checkpoint

* plurals and a couple of minor fixes

* Update Localizable.strings

* Update Localizable.strings

* Update zh-Hant localizations

* improve translation of "by" in "filtered by"

* update zh-Hant localization

* update Hant localization

* Update zh_Hant
2023-11-13 08:16:12 +01:00
Thomas Ricouard
32f96ac1ce Format 2023-11-07 11:24:03 +01:00
Théo Arrouye
4266ac4b42
Improve SoundEffectManager & HapticManager (#1662)
* Remove unnecessary vars and switches

* Improve SoundEffectManager call-site API

* Improve HapticManager call-site API
2023-11-07 11:22:36 +01:00
Thai D. V
6e1e83cace
Refactor StatusRowMediaPreviewView (#1654)
* improve the sensitive content overlay animation and refactor subviews

* fix alt text button and refactor views

* refactor `StatusRowMediaPreviewView.onTapGesture`

* simplify `MediaPreview` and `FeaturedImagePreView`

* make alt text button adaptable
2023-11-07 11:20:35 +01:00
Andrzej Rózga
9e4b333981
Polish localization update (#1655)
* Polish localization update

* Merge branch 'main' into pr/1655

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-11-06 11:51:16 +01:00
Chanhwi Joo
a7ac559225
Update Korean localization (#1658) 2023-11-06 11:40:11 +01:00
Thomas Ricouard
0424b62684 Fix build 2023-11-01 19:55:48 +01:00
Thomas Ricouard
3e3c69c41c format 2023-11-01 18:58:44 +01:00
Thomas Ricouard
4c7a7986c5 fix build 2023-11-01 18:58:19 +01:00
Hugo Saynac
b2933b8c75
Fix flickering issues when resizing window (#1644)
* Fix flickering issues when resizing window, or hiding notifications on macOS

* Restore processor and add debouncing to the processor updates

* Fix indentation

* Add LazyResizableImage to the Design system module
2023-11-01 18:57:13 +01:00
Echo
b2550d28ac
Grammar (#1634) 2023-11-01 18:52:00 +01:00
Paul Schuetz
f68bc3e306
Show the pending-counter in any corners (#1638)
The pending-button can now be shown in any corner the user prefers. This is
accomplished by allowing the user to move the counter left in addition to the
already present option to move it down. Fixes #1637

Signed-off-by: Paul Schuetz <pa.schuetz@web.de>
2023-11-01 18:51:46 +01:00
Thai D. V
20ecc49e31
refactor MediaUIView state and logic (#1651) 2023-11-01 18:50:02 +01:00
Thomas Ricouard
cb1f3dc548 Bump version to 1.9.7 2023-10-29 11:57:40 +01:00
Thomas Ricouard
db64dd726e Enable CloudKit sync for tag groups, local timeline and drafts 2023-10-29 08:51:20 +01:00
Thomas Ricouard
bbce55e703 MediaViewer: Add loading state for quicklook 2023-10-29 08:27:26 +01:00
Cthulhux
cd0e9c10ac
de: Update Localizable.xcstrings (#1643) 2023-10-28 13:46:50 +02:00
Andrzej Rózga
df3d8e9ea3
Polish localization update (#1645)
* Polish localization update

* Polish localization update
2023-10-28 13:46:42 +02:00
Jerry
8943588645
Update Simplified Chinese localization (#1646) 2023-10-28 13:46:34 +02:00
Xabi
104c308cb2
Update EU Localizable.xcstrings (#1648)
New window
2023-10-28 13:46:24 +02:00
Thomas Ricouard
4fa2f3a10b Bump version to 1.9.6 2023-10-28 13:45:50 +02:00
Thomas Ricouard
8a49409b26 Tag group: fix first tag not being included 2023-10-27 11:39:31 +02:00
Thomas Ricouard
e9e1992806 Bump version to 1.9.5 2023-10-27 11:39:19 +02:00
Thomas Ricouard
641853ed8d Editor: Open in window on mac catalyst 2023-10-27 11:39:11 +02:00
Thomas Ricouard
3f3ea4ff68 Editor: Properly close window on Catalyst 2023-10-27 11:39:02 +02:00
Thomas Ricouard
5a52eb50e9 Fix login on Catalyst 2023-10-26 17:47:19 +02:00
Thomas Ricouard
2a936adca0 Add missing localization 2023-10-26 13:57:00 +02:00
Thomas Ricouard
cf0f0fd891 Refactor + add more shortcuts on macOS 2023-10-26 06:23:00 +02:00
Thomas Ricouard
494b0df0e3 Embed extensions in Catalyst 2023-10-24 19:19:53 +02:00
Thomas Ricouard
434247f3ea Fixes for macOS 2023-10-24 18:34:45 +02:00
Thomas Ricouard
07bfd8cd0e Initial macOS Catalyst support 2023-10-23 19:12:25 +02:00
Thomas Ricouard
b257bfc576 Bump version to 1.9.3 2023-10-19 08:55:10 +02:00
Thomas Ricouard
1b228d504f Media viewer: various fixes 2023-10-18 12:19:39 +02:00
Xabi
e9b322e289
Update EU Localizable.xcstrings (#1625)
Added:
- Block %@
- Do you want to block this user?
2023-10-17 09:39:54 +02:00
Thomas Ricouard
32cbb1699e Bump version to 1.9.2 2023-10-17 09:19:41 +02:00
Thomas Ricouard
ccae4e0e3d Add button to save photo in the new media viewer 2023-10-17 08:52:05 +02:00
Thomas Ricouard
8ed6d548eb Always autoplay video in the new media viewer 2023-10-17 08:24:11 +02:00
Thomas Ricouard
1743b3bc08 Bump version to 1.9.1 2023-10-17 08:19:14 +02:00
Thomas Ricouard
3cee46d4ef Fix build 2023-10-16 19:40:58 +02:00
Thomas Ricouard
ff5ed48a6e Bump version to 1.9.0 2023-10-16 19:11:08 +02:00
Thomas Ricouard
fd55020533 New media viewer 2023-10-16 19:08:59 +02:00
Thomas Ricouard
017275ec69 Fix status embed 2023-10-16 09:26:49 +02:00
Thomas Ricouard
f08c90f8a0 Fix #1419 2023-10-16 09:16:17 +02:00
Andrzej Rózga
518e69d49d
Polish localization update (#1621) 2023-10-15 08:11:29 +02:00
Jerry
f6abd5ddf0
Update Simplified Chinese localization for block user confirmation (#1617) 2023-10-11 08:51:15 +02:00
Cthulhux
855fde2eb4
de: Update Localizable.xcstrings (#1619)
translated the user block stuff
2023-10-11 08:51:07 +02:00
Chanhwi Joo
1bd9d15a8f
Update Korean localization & localize the user block confirmation dialog (#1616)
* Update Korean localization

* Localize the user block confirmation dialog
2023-10-10 18:49:16 +02:00
Thomas Ricouard
ee725f15f7 Revert "Enable iCloud sync for SwiftData"
This reverts commit 89c611ed62.
2023-10-10 18:48:17 +02:00
Thomas Ricouard
89c611ed62 Enable iCloud sync for SwiftData 2023-10-10 18:32:11 +02:00
Ico Davids
ee2dbf2965
Updated NL translations (#1613) 2023-10-10 08:23:05 +02:00
Jerry
31ae9cb952
Update Simplified Chinese localization (#1612) 2023-10-06 06:29:10 +02:00
Cthulhux
6480014148
de: Update Localizable.xcstrings (#1611) 2023-10-06 06:28:59 +02:00
Thomas Ricouard
beec49a7e6 Bump version to 1.8.6 2023-10-05 20:41:19 +02:00
Thomas Ricouard
3fd9013dbd Compiler check 2023-10-05 17:10:24 +02:00
Thomas Ricouard
aa7c1b87e4 Migrate to string catalog 2023-10-05 10:28:39 +02:00
Thomas Ricouard
1275b09f20 Add / Remove tag from tag groups from timeline view 2023-10-05 09:47:51 +02:00
Bosco Ho
1bf4d9e398
Feature: Tab bar scroll to top (#1598)
* - *WIP* Explore tab: Tap on tab to scroll to top.

* - Explore tab: Tap tab to scroll to top.

* - Explore: Tap tab again to focus on search bar.
- Explore: Set `.defaultMinListRowHeight` so scroll to view doesn't occupy more than 1pt height in grouped style list.
- Explore: Add padding to get Explore list view to look the same.

* - Explore: Minor adjust to padding.

* - Messages: Add tap tab to scroll to top.

* - Notifications: Add tap tab to scroll to top.

* - Profile: Add tap tab to scroll to top.

* Add `ScrollToView` that can be used across all views.

* Move scroll-to-top constants to ScrollToView.

* Format

---------

Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-10-05 08:22:45 +02:00
Eric
4bbfdcd256
Feature request: Block user confirmation dialog (#1606)
- Using State property and Binding between ContextMenu and AccountDetailView to show a confirmation dialog when the block button is pressed.

Co-authored-by: Eric Chaing <eric@Erics-MacBook-Pro.local>
2023-10-04 09:40:54 +02:00
Yasura Dodo
e8cb090baf
Fix a crash bug at AccountsListRow (#1602)
When you long tap a `AccountsListRow`, a `contextMenu` will be called, and then the app will be crashed.
This happens because two environments are missing; `QuickLook` and `RouterPath`
2023-10-02 11:58:13 +02:00
Yasura Dodo
e3f7eb31e4
Fix a crash bug at Client.makeURL (#1601)
The crash will happen when you type something unexpected instance URL.

Example
```swift
let server = "mstdn.jp/"

var components = URLComponents()
components.scheme = "https"
components.host = server
components.path = "/api/v1/instance"
components.url! // 💥 error: Execution was interrupted, reason: EXC_BREAKPOINT (code=1, subcode=0x18c986650).
```
2023-10-02 09:31:59 +02:00
Thomas Ricouard
23a83d69cc Remove legacy migrations 2023-10-01 09:48:27 +02:00
Thomas Ricouard
d5896b95e9 format 2023-10-01 09:37:37 +02:00
Paul Schuetz
0b5e764556
Automatically remove spaces in server names (#1600)
* Automatically remove spaces in server names

If a server name includes a space (which can happen if the string is pasted /
autocompleted), this space is removed, which results in the app not crashing.
Fixes #1599

Signed-off-by: Paul Schuetz <pa.schuetz@web.de>

* Format

---------

Signed-off-by: Paul Schuetz <pa.schuetz@web.de>
Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
2023-10-01 09:37:09 +02:00
Benoît Clouet
d32c5c004c
Added the ability to display the Pending/Unread button at bottom of the screen for bigger displays or smaller hands (#1595) 2023-10-01 09:24:37 +02:00
Bosco Ho
1f44c502dd
Use NavigationLink with value to push Explore trending links" (#1594)
- Fixes trending links "see more" not getting added to navigation path.
2023-09-27 08:38:17 +02:00
Paul Schuetz
1f28595d39
Remove "Translate with DeepL"-option (#1593)
The "Translate with DeepL"-option is removed to make the app better understandable for the average user. A person who wants to use DeepL can still insert their own API key to always use DeepL.
Fixes #1583

Signed-off-by: Paul Schuetz <pa.schuetz@web.de>
2023-09-26 14:11:00 +02:00
Thomas Ricouard
717ef16628 Correctly save / restore any previously selected timeline on home tab 2023-09-25 14:43:29 +02:00
Thomas Ricouard
46d4f3c4f4 Bump version to 1.8.5 2023-09-25 14:37:39 +02:00
Thomas Ricouard
e0663bf177 Fix #1590 2023-09-25 14:12:35 +02:00
Thomas Ricouard
fd1ec73773 Bump version to 1.8.4 2023-09-25 14:09:04 +02:00
Thomas Ricouard
7efd8ed7cb Fix #1585 2023-09-22 22:41:06 +02:00
Thomas Ricouard
cc32845134 Revert "Switch to iOS 17 inspector"
This reverts commit 7589ab75f8.
2023-09-22 22:39:35 +02:00
Thomas Ricouard
4870b202d6 Migrate TagGroup to SwiftData 2023-09-22 19:33:53 +02:00
Thomas Ricouard
527d982dce Migrate LocalTimeline to SwiftData 2023-09-22 12:49:25 +02:00
Thomas Ricouard
cb1b5b69df Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2023-09-22 09:31:37 +02:00
Thomas Ricouard
0c4bde40af Migrate drafts to SwiftData 2023-09-22 09:31:35 +02:00
Jerry
21782c9e02
Update missing Simplified Chinese translation (#1580) 2023-09-22 08:35:44 +02:00
Thomas Ricouard
7eec1b8439 Share sheet: Fix account selector 2023-09-22 08:35:21 +02:00
Thomas Ricouard
60713101a7 Remove some .shared usage 2023-09-22 08:32:13 +02:00
Thomas Ricouard
90fc2907d3 Bump to 1.8.3 2023-09-21 08:56:43 +02:00
Thomas Ricouard
4adbff1342 Fix avatar shape and position settings not being saved 2023-09-20 21:19:45 +02:00
Thomas Ricouard
7589ab75f8 Switch to iOS 17 inspector 2023-09-20 21:19:31 +02:00
Thomas Ricouard
15f498037d Fix thread bar in status detail 2023-09-20 21:19:02 +02:00
Thomas Ricouard
90337bd3ea Bump version to 1.8.2 2023-09-20 08:20:01 +02:00
Thomas Ricouard
46df3bb7f9 Fix #1579 2023-09-20 07:28:04 +02:00
Thomas Ricouard
75a00907ea Bump version to 1.8.1 2023-09-19 18:40:32 +02:00
Thomas Ricouard
e2f0863ff6 Fix timeline filter update 2023-09-19 18:33:13 +02:00
Thomas Ricouard
3743e6d870 Fix share sheet 2023-09-19 09:25:48 +02:00
Thomas Ricouard
6c23569d15 UserPreferences -> Observable 2023-09-19 09:18:20 +02:00
Thomas Ricouard
fd09276d49 Refactor notifications count 2023-09-19 08:44:11 +02:00
Thomas Ricouard
f9c0355f1d Convert Theme to Observable 2023-09-18 21:03:52 +02:00
Thomas Ricouard
625b7f8137 Fix tab icon only on iOS 17 2023-09-18 20:14:26 +02:00
Thomas Ricouard
e6455304ac Fix draft label color 2023-09-18 19:07:14 +02:00
Thomas Ricouard
1b0ddf4fd9 Fix #1457 2023-09-18 19:04:54 +02:00
Thomas Ricouard
bc554da678 Merge branch 'main' of https://github.com/Dimillian/IceCubesApp 2023-09-18 18:55:12 +02:00
Thomas Ricouard
8bb102cd67 Fix #1466 2023-09-18 18:55:11 +02:00
Cthulhux
cd253c7dc1
de: added 1 translation (#1573)
I've become sloppy over the past few months... sorry :)) totally missed this!
2023-09-18 18:39:31 +02:00
Thomas Ricouard
27102cbae3 oops 2023-09-18 14:30:36 +02:00
Thomas Ricouard
bd3d2008c2 Fix #1483 2023-09-18 14:30:24 +02:00
Thomas Ricouard
1ac25cc417 Fix icon image files 2023-09-18 09:39:20 +02:00
Thomas Ricouard
8b445324e0 Fix #1548 2023-09-18 09:29:42 +02:00
Thomas Ricouard
379d2f36fb Fix #1545 2023-09-18 09:18:48 +02:00
Thomas Ricouard
7388cc4a86 Add two new icons 2023-09-18 09:14:51 +02:00
Thomas Ricouard
4189a59cf6
iOS 17+ only support + migrating to Observation framework (#1571)
* Initial iOS 17 + Observable migration

* More Observation

* More observation

* Checkpoint

* Checkpoint

* Bump version to 1.8.0

* SwiftFormat

* Fix home timeline switch on login

* Fix sidebar routerPath

* Fixes on detail view

* Remove print changes

* Simply detail view

* More opt

* Migrate DisplaySettingsLocalValues

* Better post detail transition

* Status detail animation finally right

* Cleanup
2023-09-18 07:01:23 +02:00
Eslam Nahel
3853eff065
Fix text field bottom padding in EditorView (#1570) 2023-09-17 08:47:15 +02:00
Thomas Ricouard
5b0f10f0a2 Bump version to 1.7.9 2023-09-17 07:32:09 +02:00
Thomas Ricouard
98035e8530 Better status focused screen transition 2023-09-16 15:04:42 +02:00
Thomas Ricouard
aaafac8e5a Bump version to 1.7.8 2023-09-16 14:31:26 +02:00
Thomas Ricouard
8a3c971402 Swiftformat 2023-09-16 14:15:03 +02:00
Thomas Ricouard
584a0d0432 Composer: Fix nav bar background 2023-09-16 14:02:50 +02:00
Thomas Ricouard
8e3584ee79 Upgrade to Swift tools version 5.9 + strict Swift concurrency everywhere 2023-09-15 12:46:15 +02:00
Thomas Ricouard
1bbb0dc82d Fix sound effects + upgrade swift concurrency settings 2023-09-14 11:04:14 +02:00
Xabi
53b442eb33
Update EU localisation (#1567)
Added:
- account.action.privacy-settings
2023-09-04 09:07:33 +02:00
Thomas Ricouard
5f4fef859c
Update README.md 2023-08-31 19:35:48 +02:00
Jesús Jiménez Sánchez
1f46691279
Fix incorrect Spanish translation (#1562) 2023-08-31 07:42:07 +02:00
Andrzej Rózga
59f2023497
Polish localization update (#1564) 2023-08-31 07:42:00 +02:00
Jerry
462d2355f4
Update Simplified Chinese localization (#1566)
Simplified Chinese translation for "Privacy Settings"
2023-08-31 07:41:54 +02:00
Thomas Ricouard
e921f2cdd4
Fix dark secondary color (#1565) 2023-08-30 09:10:22 +02:00
Thomas Ricouard
bdce052dc7 Bump version to 1.7.7 2023-08-30 08:41:31 +02:00
768 changed files with 107738 additions and 24570 deletions

11
.github/dependabot.yml vendored Normal file
View file

@ -0,0 +1,11 @@
# To get started with Dependabot version updates, you'll need to specify which
# package ecosystems to update and where the package manifests are located.
# Please see the documentation for all configuration options:
# https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates
version: 2
updates:
- package-ecosystem: "swift" # See documentation for possible values
directory: "/" # Location of package manifests
schedule:
interval: "weekly"

View file

@ -0,0 +1,35 @@
name: Validate Translations
on:
pull_request:
types: [synchronize, opened, reopened, labeled, unlabeled, edited]
jobs:
main:
name: Validate Translations
runs-on: macOS-latest
steps:
- name: git checkout
uses: actions/checkout@v3
- name: ruby versions
run: |
ruby --version
gem --version
bundler --version
- name: ruby setup
uses: ruby/setup-ruby@v1
with:
ruby-version: 3.3
bundler-cache: true
# additional steps here, if needed
- name: Clone SwiftPolyglot
run: git clone https://github.com/appdecostudio/SwiftPolyglot.git --branch 0.2.0
- name: Build and Run SwiftPolyglot
run: |
swift build --package-path ./SwiftPolyglot --configuration release
swift run --package-path ./SwiftPolyglot swiftpolyglot "en,eu,be,ca,zh-Hans,zh-Hant,nl,en-GB,fr,de,it,ja,ko,nb,pl,pt-BR,es,tr,uk"

View file

@ -13,7 +13,7 @@ import Models
import Network import Network
// Sample code was sending this from a thread to another, let asume @Sendable for this // Sample code was sending this from a thread to another, let asume @Sendable for this
extension NSExtensionContext: @unchecked Sendable {} extension NSExtensionContext: @unchecked @retroactive Sendable {}
final class ActionRequestHandler: NSObject, NSExtensionRequestHandling, Sendable { final class ActionRequestHandler: NSObject, NSExtensionRequestHandling, Sendable {
enum Error: Swift.Error { enum Error: Swift.Error {

View file

@ -2,9 +2,9 @@
<!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>OPENAI_SECRET</key> <key>com.apple.security.app-sandbox</key>
<string>NICE_TRY</string> <true/>
<key>DEEPL_SECRET</key> <key>com.apple.security.network.client</key>
<string>NICE_TRY_AGAIN</string> <true/>
</dict> </dict>
</plist> </plist>

File diff suppressed because it is too large Load diff

View file

@ -1,12 +1,22 @@
{ {
"originHash" : "b7af8c2ab18771d4cebfbeb66d91559df500516a12027cd67834b2a576eb3df0",
"pins" : [ "pins" : [
{ {
"identity" : "bodega", "identity" : "bodega",
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/mergesort/Bodega", "location" : "https://github.com/mergesort/Bodega",
"state" : { "state" : {
"revision" : "f0554077c178088ba11557bbdbb71775cc6a1b84", "revision" : "bfd8871e9c2590d31b200e54c75428a71483afdf",
"version" : "2.1.0" "version" : "2.1.3"
}
},
{
"identity" : "buttonkit",
"kind" : "remoteSourceControl",
"location" : "https://github.com/Dean151/ButtonKit",
"state" : {
"revision" : "d567519b297777c38dee56ef10201fef4962ff75",
"version" : "0.4.1"
} }
}, },
{ {
@ -14,8 +24,17 @@
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/divadretlaw/EmojiText", "location" : "https://github.com/divadretlaw/EmojiText",
"state" : { "state" : {
"revision" : "a4ddf5077c241170e8ac0d3a9480c511e27c1ae9", "revision" : "174a7bc7bd75650ad1acb5679dbb754296093de0",
"version" : "2.8.0" "version" : "4.0.0"
}
},
{
"identity" : "giphy-ios-sdk",
"kind" : "remoteSourceControl",
"location" : "https://github.com/Giphy/giphy-ios-sdk",
"state" : {
"revision" : "fb61ec12738133eb3b9bf62ed11d1bf93d9b4b20",
"version" : "2.2.10"
} }
}, },
{ {
@ -24,7 +43,16 @@
"location" : "https://github.com/evgenyneu/keychain-swift", "location" : "https://github.com/evgenyneu/keychain-swift",
"state" : { "state" : {
"branch" : "master", "branch" : "master",
"revision" : "c1fde55798b164cad44b5e23cfa2f0f1ebcd76af" "revision" : "5e1b02b6a9dac2a759a1d5dbc175c86bd192a608"
}
},
{
"identity" : "libwebp-xcode",
"kind" : "remoteSourceControl",
"location" : "https://github.com/SDWebImage/libwebp-Xcode",
"state" : {
"revision" : "b2b1d20a90b14d11f6ef4241da6b81c1d3f171e4",
"version" : "1.3.2"
} }
}, },
{ {
@ -32,8 +60,8 @@
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/nicklockwood/LRUCache", "location" : "https://github.com/nicklockwood/LRUCache",
"state" : { "state" : {
"revision" : "6d2b5246c9c98dcd498552bb22f08d55b12a8371", "revision" : "542f0449556327415409ededc9c43a4bd0a397dc",
"version" : "1.0.4" "version" : "1.0.7"
} }
}, },
{ {
@ -41,17 +69,17 @@
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/kean/Nuke", "location" : "https://github.com/kean/Nuke",
"state" : { "state" : {
"revision" : "3f666f120b63ea7de57d42e9a7c9b47f8e7a290b", "revision" : "0ead44350d2737db384908569c012fe67c421e4d",
"version" : "12.1.6" "version" : "12.8.0"
} }
}, },
{ {
"identity" : "purchases-ios", "identity" : "purchases-ios",
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/RevenueCat/purchases-ios.git", "location" : "https://github.com/RevenueCat/purchases-ios",
"state" : { "state" : {
"revision" : "4601c1e0c246f3d74094229737e894a9f2339e6a", "revision" : "7d55b964114a3d4a76791227cdc28577617596db",
"version" : "4.25.7" "version" : "4.43.2"
} }
}, },
{ {
@ -68,8 +96,35 @@
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/stephencelis/SQLite.swift.git", "location" : "https://github.com/stephencelis/SQLite.swift.git",
"state" : { "state" : {
"revision" : "7a2e3cd27de56f6d396e84f63beefd0267b55ccb", "revision" : "a95fc6df17d108bd99210db5e8a9bac90fe984b8",
"version" : "0.14.1" "version" : "0.15.3"
}
},
{
"identity" : "swift-cmark",
"kind" : "remoteSourceControl",
"location" : "https://github.com/apple/swift-cmark.git",
"state" : {
"revision" : "3ccff77b2dc5b96b77db3da0d68d28068593fa53",
"version" : "0.5.0"
}
},
{
"identity" : "swift-markdown",
"kind" : "remoteSourceControl",
"location" : "https://github.com/apple/swift-markdown",
"state" : {
"revision" : "8f79cb175981458a0a27e76cb42fee8e17b1a993",
"version" : "0.5.0"
}
},
{
"identity" : "swiftsdk",
"kind" : "remoteSourceControl",
"location" : "https://github.com/TelemetryDeck/SwiftSDK",
"state" : {
"revision" : "13a26cf125b70d695913eb9bea9f9b9c29da5790",
"version" : "2.3.0"
} }
}, },
{ {
@ -77,28 +132,46 @@
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/scinfu/SwiftSoup.git", "location" : "https://github.com/scinfu/SwiftSoup.git",
"state" : { "state" : {
"revision" : "8b6cf29eead8841a1fa7822481cb3af4ddaadba6", "revision" : "3c2c7e1e72b8abd96eafbae80323c5c1e5317437",
"version" : "2.6.1" "version" : "2.7.5"
} }
}, },
{ {
"identity" : "swiftui-introspect", "identity" : "swiftui-introspect",
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/siteline/SwiftUI-Introspect.git", "location" : "https://github.com/siteline/swiftui-introspect",
"state" : { "state" : {
"revision" : "9da0f9b7bffe96a7c98a0128f1e214f62728a39a", "revision" : "668a65735751432b640260c56dfa621cec568368",
"version" : "0.11.1" "version" : "1.2.0"
} }
}, },
{ {
"identity" : "swiftui-shimmer", "identity" : "wishkit-ios",
"kind" : "remoteSourceControl", "kind" : "remoteSourceControl",
"location" : "https://github.com/markiv/SwiftUI-Shimmer", "location" : "https://github.com/wishkit/wishkit-ios.git",
"state" : { "state" : {
"revision" : "965a7cbcbf094cbcf22b9251a2323bdc3432e171", "revision" : "2b5eb8d1fb13706f8c14767a5239e34e403375f1",
"version" : "1.1.0" "version" : "4.2.2"
}
},
{
"identity" : "wishkit-ios-shared",
"kind" : "remoteSourceControl",
"location" : "https://github.com/wishkit/wishkit-ios-shared.git",
"state" : {
"revision" : "118c9c482e4ad57c65d664283516425b98616483",
"version" : "1.4.3"
}
},
{
"identity" : "wrappinghstack",
"kind" : "remoteSourceControl",
"location" : "https://github.com/dkk/WrappingHStack",
"state" : {
"revision" : "425d9488ba55f58f0b34498c64c054c77fc2a44b",
"version" : "2.2.11"
} }
} }
], ],
"version" : 2 "version" : 3
} }

View file

@ -0,0 +1,96 @@
<?xml version="1.0" encoding="UTF-8"?>
<Scheme
LastUpgradeVersion = "1600"
wasCreatedForAppExtension = "YES"
version = "2.0">
<BuildAction
parallelizeBuildables = "YES"
buildImplicitDependencies = "YES">
<BuildActionEntries>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "E9DF41F929830FEC0003AAD2"
BuildableName = "IceCubesActionExtension.appex"
BlueprintName = "IceCubesActionExtension"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
</BuildActionEntries>
</BuildAction>
<TestAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
shouldUseLaunchSchemeArgsEnv = "YES"
shouldAutocreateTestPlan = "YES">
</TestAction>
<LaunchAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = ""
selectedLauncherIdentifier = "Xcode.IDEFoundation.Launcher.PosixSpawn"
launchStyle = "0"
askForAppToLaunch = "Yes"
useCustomWorkingDirectory = "NO"
ignoresPersistentStateOnLaunch = "NO"
debugDocumentVersioning = "YES"
debugServiceExtension = "internal"
allowLocationSimulation = "YES"
launchAutomaticallySubstyle = "2">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</LaunchAction>
<ProfileAction
buildConfiguration = "Release"
shouldUseLaunchSchemeArgsEnv = "YES"
savedToolIdentifier = ""
useCustomWorkingDirectory = "NO"
debugDocumentVersioning = "YES"
askForAppToLaunch = "Yes"
launchAutomaticallySubstyle = "2">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</ProfileAction>
<AnalyzeAction
buildConfiguration = "Debug">
</AnalyzeAction>
<ArchiveAction
buildConfiguration = "Release"
revealArchiveInOrganizer = "YES">
</ArchiveAction>
</Scheme>

View file

@ -1,6 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<Scheme <Scheme
LastUpgradeVersion = "1420" LastUpgradeVersion = "1600"
version = "1.3"> version = "1.3">
<BuildAction <BuildAction
parallelizeBuildables = "YES" parallelizeBuildables = "YES"
@ -15,7 +15,7 @@
<BuildableReference <BuildableReference
BuildableIdentifier = "primary" BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9" BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "IceCubesApp.app" BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp" BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj"> ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference> </BuildableReference>
@ -45,7 +45,7 @@
<BuildableReference <BuildableReference
BuildableIdentifier = "primary" BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9" BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "IceCubesApp.app" BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp" BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj"> ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference> </BuildableReference>
@ -62,7 +62,7 @@
<BuildableReference <BuildableReference
BuildableIdentifier = "primary" BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9" BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "IceCubesApp.app" BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp" BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj"> ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference> </BuildableReference>

View file

@ -0,0 +1,114 @@
<?xml version="1.0" encoding="UTF-8"?>
<Scheme
LastUpgradeVersion = "1600"
wasCreatedForAppExtension = "YES"
version = "2.0">
<BuildAction
parallelizeBuildables = "YES"
buildImplicitDependencies = "YES"
buildArchitectures = "Automatic">
<BuildActionEntries>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9F7788C42BE652B1004E6BEF"
BuildableName = "IceCubesAppWidgetsExtensionExtension.appex"
BlueprintName = "IceCubesAppWidgetsExtensionExtension"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
</BuildActionEntries>
</BuildAction>
<TestAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
shouldUseLaunchSchemeArgsEnv = "YES"
shouldAutocreateTestPlan = "YES">
</TestAction>
<LaunchAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = ""
selectedLauncherIdentifier = "Xcode.IDEFoundation.Launcher.PosixSpawn"
launchStyle = "0"
askForAppToLaunch = "Yes"
useCustomWorkingDirectory = "NO"
ignoresPersistentStateOnLaunch = "NO"
debugDocumentVersioning = "YES"
debugServiceExtension = "internal"
allowLocationSimulation = "YES"
launchAutomaticallySubstyle = "2">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
<EnvironmentVariables>
<EnvironmentVariable
key = "_XCWidgetKind"
value = ""
isEnabled = "YES">
</EnvironmentVariable>
<EnvironmentVariable
key = "_XCWidgetDefaultView"
value = "timeline"
isEnabled = "YES">
</EnvironmentVariable>
<EnvironmentVariable
key = "_XCWidgetFamily"
value = "systemMedium"
isEnabled = "YES">
</EnvironmentVariable>
</EnvironmentVariables>
</LaunchAction>
<ProfileAction
buildConfiguration = "Release"
shouldUseLaunchSchemeArgsEnv = "YES"
savedToolIdentifier = ""
useCustomWorkingDirectory = "NO"
debugDocumentVersioning = "YES"
askForAppToLaunch = "Yes"
launchAutomaticallySubstyle = "2">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</ProfileAction>
<AnalyzeAction
buildConfiguration = "Debug">
</AnalyzeAction>
<ArchiveAction
buildConfiguration = "Release"
revealArchiveInOrganizer = "YES">
</ArchiveAction>
</Scheme>

View file

@ -0,0 +1,100 @@
<?xml version="1.0" encoding="UTF-8"?>
<Scheme
LastUpgradeVersion = "1600"
wasCreatedForAppExtension = "YES"
version = "2.0">
<BuildAction
parallelizeBuildables = "YES"
buildImplicitDependencies = "YES">
<BuildActionEntries>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9F2A5415296AB631009B2D7C"
BuildableName = "IceCubesNotifications.appex"
BlueprintName = "IceCubesNotifications"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
</BuildActionEntries>
</BuildAction>
<TestAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
shouldUseLaunchSchemeArgsEnv = "YES"
shouldAutocreateTestPlan = "YES">
</TestAction>
<LaunchAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = ""
selectedLauncherIdentifier = "Xcode.IDEFoundation.Launcher.PosixSpawn"
launchStyle = "0"
askForAppToLaunch = "Yes"
useCustomWorkingDirectory = "NO"
ignoresPersistentStateOnLaunch = "NO"
debugDocumentVersioning = "YES"
debugServiceExtension = "internal"
allowLocationSimulation = "YES"
launchAutomaticallySubstyle = "2">
<RemoteRunnable
runnableDebuggingMode = "1"
BundleIdentifier = "com.thomasricouard.IceCubesApp"
RemotePath = "/Users/dimillian/Library/Developer/CoreSimulator/Devices/8EF923D0-4CF1-49B6-B287-5F05AD5440C1/data/Containers/Bundle/Application/C447A1D1-9BC9-49C9-8FA5-130E8403972F/Ice Cubes.app">
</RemoteRunnable>
<MacroExpansion>
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</MacroExpansion>
</LaunchAction>
<ProfileAction
buildConfiguration = "Release"
shouldUseLaunchSchemeArgsEnv = "YES"
savedToolIdentifier = ""
useCustomWorkingDirectory = "NO"
debugDocumentVersioning = "YES"
askForAppToLaunch = "Yes"
launchAutomaticallySubstyle = "2">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</ProfileAction>
<AnalyzeAction
buildConfiguration = "Debug">
</AnalyzeAction>
<ArchiveAction
buildConfiguration = "Release"
revealArchiveInOrganizer = "YES">
</ArchiveAction>
</Scheme>

View file

@ -0,0 +1,96 @@
<?xml version="1.0" encoding="UTF-8"?>
<Scheme
LastUpgradeVersion = "1600"
wasCreatedForAppExtension = "YES"
version = "2.0">
<BuildAction
parallelizeBuildables = "YES"
buildImplicitDependencies = "YES">
<BuildActionEntries>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FAD858729743F7400496AB1"
BuildableName = "IceCubesShareExtension.appex"
BlueprintName = "IceCubesShareExtension"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
<BuildActionEntry
buildForTesting = "YES"
buildForRunning = "YES"
buildForProfiling = "YES"
buildForArchiving = "YES"
buildForAnalyzing = "YES">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildActionEntry>
</BuildActionEntries>
</BuildAction>
<TestAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
shouldUseLaunchSchemeArgsEnv = "YES"
shouldAutocreateTestPlan = "YES">
</TestAction>
<LaunchAction
buildConfiguration = "Debug"
selectedDebuggerIdentifier = ""
selectedLauncherIdentifier = "Xcode.IDEFoundation.Launcher.PosixSpawn"
launchStyle = "0"
askForAppToLaunch = "Yes"
useCustomWorkingDirectory = "NO"
ignoresPersistentStateOnLaunch = "NO"
debugDocumentVersioning = "YES"
debugServiceExtension = "internal"
allowLocationSimulation = "YES"
launchAutomaticallySubstyle = "2">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</LaunchAction>
<ProfileAction
buildConfiguration = "Release"
shouldUseLaunchSchemeArgsEnv = "YES"
savedToolIdentifier = ""
useCustomWorkingDirectory = "NO"
debugDocumentVersioning = "YES"
askForAppToLaunch = "Yes"
launchAutomaticallySubstyle = "2">
<BuildableProductRunnable
runnableDebuggingMode = "0">
<BuildableReference
BuildableIdentifier = "primary"
BlueprintIdentifier = "9FBFE638292A715500C250E9"
BuildableName = "Ice Cubes.app"
BlueprintName = "IceCubesApp"
ReferencedContainer = "container:IceCubesApp.xcodeproj">
</BuildableReference>
</BuildableProductRunnable>
</ProfileAction>
<AnalyzeAction
buildConfiguration = "Debug">
</AnalyzeAction>
<ArchiveAction
buildConfiguration = "Release"
revealArchiveInOrganizer = "YES">
</ArchiveAction>
</Scheme>

View file

@ -6,8 +6,10 @@ import Env
import Explore import Explore
import LinkPresentation import LinkPresentation
import Lists import Lists
import MediaUI
import Models import Models
import Status import Notifications
import StatusKit
import SwiftUI import SwiftUI
import Timeline import Timeline
@ -22,6 +24,8 @@ extension View {
AccountDetailView(account: account) AccountDetailView(account: account)
case let .accountSettingsWithAccount(account, appAccount): case let .accountSettingsWithAccount(account, appAccount):
AccountSettingsView(account: account, appAccount: appAccount) AccountSettingsView(account: account, appAccount: appAccount)
case let .accountMediaGridView(account, initialMedia):
AccountDetailMediaGridView(account: account, initialMediaStatuses: initialMedia)
case let .statusDetail(id): case let .statusDetail(id):
StatusDetailView(statusId: id) StatusDetailView(statusId: id)
case let .statusDetailWithStatus(status): case let .statusDetailWithStatus(status):
@ -31,9 +35,20 @@ extension View {
case let .conversationDetail(conversation): case let .conversationDetail(conversation):
ConversationDetailView(conversation: conversation) ConversationDetailView(conversation: conversation)
case let .hashTag(tag, accountId): case let .hashTag(tag, accountId):
TimelineView(timeline: .constant(.hashtag(tag: tag, accountId: accountId)), scrollToTopSignal: .constant(0), canFilterTimeline: false) TimelineView(timeline: .constant(.hashtag(tag: tag, accountId: accountId)),
pinnedFilters: .constant([]),
selectedTagGroup: .constant(nil),
canFilterTimeline: false)
case let .list(list): case let .list(list):
TimelineView(timeline: .constant(.list(list: list)), scrollToTopSignal: .constant(0), canFilterTimeline: false) TimelineView(timeline: .constant(.list(list: list)),
pinnedFilters: .constant([]),
selectedTagGroup: .constant(nil),
canFilterTimeline: false)
case let .linkTimeline(url, title):
TimelineView(timeline: .constant(.link(url: url, title: title)),
pinnedFilters: .constant([]),
selectedTagGroup: .constant(nil),
canFilterTimeline: false)
case let .following(id): case let .following(id):
AccountsListView(mode: .following(accountId: id)) AccountsListView(mode: .following(accountId: id))
case let .followers(id): case let .followers(id):
@ -45,9 +60,23 @@ extension View {
case let .accountsList(accounts): case let .accountsList(accounts):
AccountsListView(mode: .accountsList(accounts: accounts)) AccountsListView(mode: .accountsList(accounts: accounts))
case .trendingTimeline: case .trendingTimeline:
TimelineView(timeline: .constant(.trending), scrollToTopSignal: .constant(0), canFilterTimeline: false) TimelineView(timeline: .constant(.trending),
pinnedFilters: .constant([]),
selectedTagGroup: .constant(nil),
canFilterTimeline: false)
case let .trendingLinks(cards):
TrendingLinksListView(cards: cards)
case let .tagsList(tags): case let .tagsList(tags):
TagsListView(tags: tags) TagsListView(tags: tags)
case .notificationsRequests:
NotificationsRequestsListView()
case let .notificationForAccount(accountId):
NotificationsListView(lockedType: nil,
lockedAccountId: accountId)
case .blockedAccounts:
AccountsListView(mode: .blocked)
case .mutedAccounts:
AccountsListView(mode: .muted)
} }
} }
} }
@ -56,19 +85,31 @@ extension View {
sheet(item: sheetDestinations) { destination in sheet(item: sheetDestinations) { destination in
switch destination { switch destination {
case let .replyToStatusEditor(status): case let .replyToStatusEditor(status):
StatusEditorView(mode: .replyTo(status: status)) StatusEditor.MainView(mode: .replyTo(status: status))
.withEnvironments() .withEnvironments()
case let .newStatusEditor(visibility): case let .newStatusEditor(visibility):
StatusEditorView(mode: .new(visibility: visibility)) StatusEditor.MainView(mode: .new(text: nil, visibility: visibility))
.withEnvironments()
case let .prefilledStatusEditor(text, visibility):
StatusEditor.MainView(mode: .new(text: text, visibility: visibility))
.withEnvironments()
case let .imageURL(urls, visibility):
StatusEditor.MainView(mode: .imageURL(urls: urls, visibility: visibility))
.withEnvironments() .withEnvironments()
case let .editStatusEditor(status): case let .editStatusEditor(status):
StatusEditorView(mode: .edit(status: status)) StatusEditor.MainView(mode: .edit(status: status))
.withEnvironments() .withEnvironments()
case let .quoteStatusEditor(status): case let .quoteStatusEditor(status):
StatusEditorView(mode: .quote(status: status)) StatusEditor.MainView(mode: .quote(status: status))
.withEnvironments()
case let .quoteLinkStatusEditor(link):
StatusEditor.MainView(mode: .quoteLink(link: link))
.withEnvironments() .withEnvironments()
case let .mentionStatusEditor(account, visibility): case let .mentionStatusEditor(account, visibility):
StatusEditorView(mode: .mention(account: account, visibility: visibility)) StatusEditor.MainView(mode: .mention(account: account, visibility: visibility))
.withEnvironments()
case .listCreate:
ListCreateView()
.withEnvironments() .withEnvironments()
case let .listEdit(list): case let .listEdit(list):
ListEditView(list: list) ListEditView(list: list)
@ -89,36 +130,64 @@ extension View {
StatusEditHistoryView(statusId: status) StatusEditHistoryView(statusId: status)
.withEnvironments() .withEnvironments()
case .settings: case .settings:
SettingsTabs(popToRootTab: .constant(.settings)) SettingsTabs(isModal: true)
.withEnvironments() .withEnvironments()
.preferredColorScheme(Theme.shared.selectedScheme == .dark ? .dark : .light) .preferredColorScheme(Theme.shared.selectedScheme == .dark ? .dark : .light)
case .accountPushNotficationsSettings: case .accountPushNotficationsSettings:
if let subscription = PushNotificationsService.shared.subscriptions.first(where: { $0.account.token == AppAccountsManager.shared.currentAccount.oauthToken }) { if let subscription = PushNotificationsService.shared.subscriptions.first(where: { $0.account.token == AppAccountsManager.shared.currentAccount.oauthToken }) {
PushNotificationsView(subscription: subscription) NavigationSheet { PushNotificationsView(subscription: subscription) }
.withEnvironments() .withEnvironments()
} else { } else {
EmptyView() EmptyView()
} }
case .about:
NavigationSheet { AboutView() }
.withEnvironments()
case .support:
NavigationSheet { SupportAppView() }
.withEnvironments()
case let .report(status): case let .report(status):
ReportView(status: status) ReportView(status: status)
.withEnvironments() .withEnvironments()
case let .shareImage(image, status): case let .shareImage(image, status):
ActivityView(image: image, status: status) ActivityView(image: image, status: status)
.withEnvironments()
case let .editTagGroup(tagGroup, onSaved): case let .editTagGroup(tagGroup, onSaved):
EditTagGroupView(editingTagGroup: tagGroup, onSaved: onSaved) EditTagGroupView(tagGroup: tagGroup, onSaved: onSaved)
.withEnvironments()
case .timelineContentFilter:
NavigationSheet { TimelineContentFilterView() }
.presentationDetents([.medium])
.presentationBackground(.thinMaterial)
.withEnvironments()
case .accountEditInfo:
EditAccountView()
.withEnvironments()
case .accountFiltersList:
FiltersListView()
.withEnvironments() .withEnvironments()
} }
} }
} }
func withEnvironments() -> some View { func withEnvironments() -> some View {
environmentObject(CurrentAccount.shared) environment(CurrentAccount.shared)
.environmentObject(UserPreferences.shared) .environment(UserPreferences.shared)
.environmentObject(CurrentInstance.shared) .environment(CurrentInstance.shared)
.environmentObject(Theme.shared) .environment(Theme.shared)
.environmentObject(AppAccountsManager.shared) .environment(AppAccountsManager.shared)
.environmentObject(PushNotificationsService.shared) .environment(PushNotificationsService.shared)
.environmentObject(AppAccountsManager.shared.currentClient) .environment(AppAccountsManager.shared.currentClient)
.environment(QuickLook.shared)
}
func withModelContainer() -> some View {
modelContainer(for: [
Draft.self,
LocalTimeline.self,
TagGroup.self,
RecentTag.self,
])
} }
} }
@ -155,9 +224,15 @@ struct ActivityView: UIViewControllerRepresentable {
} }
func makeUIViewController(context _: UIViewControllerRepresentableContext<ActivityView>) -> UIActivityViewController { func makeUIViewController(context _: UIViewControllerRepresentableContext<ActivityView>) -> UIActivityViewController {
return UIActivityViewController(activityItems: [image, LinkDelegate(image: image, status: status)], UIActivityViewController(activityItems: [image, LinkDelegate(image: image, status: status)],
applicationActivities: nil) applicationActivities: nil)
} }
func updateUIViewController(_: UIActivityViewController, context _: UIViewControllerRepresentableContext<ActivityView>) {} func updateUIViewController(_: UIActivityViewController, context _: UIViewControllerRepresentableContext<ActivityView>) {}
} }
extension URL: @retroactive Identifiable {
public var id: String {
absoluteString
}
}

View file

@ -0,0 +1,36 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>aps-environment</key>
<string>development</string>
<key>com.apple.developer.icloud-container-identifiers</key>
<array>
<string>iCloud.icecubesapp</string>
</array>
<key>com.apple.developer.icloud-services</key>
<array>
<string>CloudKit</string>
</array>
<key>com.apple.developer.user-fonts</key>
<array>
<string>app-usage</string>
</array>
<key>com.apple.developer.usernotifications.communication</key>
<true/>
<key>com.apple.security.app-sandbox</key>
<true/>
<key>com.apple.security.application-groups</key>
<array>
<string>group.$(BUNDLE_ID_PREFIX).IceCubesApp</string>
</array>
<key>com.apple.security.files.user-selected.read-write</key>
<true/>
<key>com.apple.security.network.client</key>
<true/>
<key>keychain-access-groups</key>
<array>
<string>$(AppIdentifierPrefix)$(BUNDLE_ID_PREFIX).IceCubesApp</string>
</array>
</dict>
</plist>

View file

@ -4,6 +4,14 @@
<dict> <dict>
<key>aps-environment</key> <key>aps-environment</key>
<string>development</string> <string>development</string>
<key>com.apple.developer.icloud-container-identifiers</key>
<array>
<string>iCloud.icecubesapp</string>
</array>
<key>com.apple.developer.icloud-services</key>
<array>
<string>CloudKit</string>
</array>
<key>com.apple.developer.user-fonts</key> <key>com.apple.developer.user-fonts</key>
<array> <array>
<string>app-usage</string> <string>app-usage</string>
@ -16,7 +24,9 @@
<array> <array>
<string>group.$(BUNDLE_ID_PREFIX).IceCubesApp</string> <string>group.$(BUNDLE_ID_PREFIX).IceCubesApp</string>
</array> </array>
<key>com.apple.security.files.user-selected.read-only</key> <key>com.apple.security.files.user-selected.read-write</key>
<true/>
<key>com.apple.security.network.client</key>
<true/> <true/>
<key>keychain-access-groups</key> <key>keychain-access-groups</key>
<array> <array>

View file

@ -1,306 +0,0 @@
import Account
import AppAccount
import AVFoundation
import DesignSystem
import Env
import KeychainSwift
import Network
import RevenueCat
import SwiftUI
import Timeline
@main
struct IceCubesApp: App {
@UIApplicationDelegateAdaptor private var appDelegate: AppDelegate
@Environment(\.scenePhase) private var scenePhase
@StateObject private var appAccountsManager = AppAccountsManager.shared
@StateObject private var currentInstance = CurrentInstance.shared
@StateObject private var currentAccount = CurrentAccount.shared
@StateObject private var userPreferences = UserPreferences.shared
@StateObject private var pushNotificationsService = PushNotificationsService.shared
@StateObject private var watcher = StreamWatcher()
@StateObject private var quickLook = QuickLook()
@StateObject private var theme = Theme.shared
@StateObject private var sidebarRouterPath = RouterPath()
@State private var selectedTab: Tab = .timeline
@State private var popToRootTab: Tab = .other
@State private var sideBarLoadedTabs: Set<Tab> = Set()
@State private var isSupporter: Bool = false
private var availableTabs: [Tab] {
appAccountsManager.currentClient.isAuth ? Tab.loggedInTabs() : Tab.loggedOutTab()
}
var body: some Scene {
WindowGroup {
appView
.applyTheme(theme)
.onAppear {
setNewClientsInEnv(client: appAccountsManager.currentClient)
setupRevenueCat()
refreshPushSubs()
}
.environmentObject(appAccountsManager)
.environmentObject(appAccountsManager.currentClient)
.environmentObject(quickLook)
.environmentObject(currentAccount)
.environmentObject(currentInstance)
.environmentObject(userPreferences)
.environmentObject(theme)
.environmentObject(watcher)
.environmentObject(pushNotificationsService)
.environment(\.isSupporter, isSupporter)
.fullScreenCover(item: $quickLook.url, content: { url in
QuickLookPreview(selectedURL: url, urls: quickLook.urls)
.edgesIgnoringSafeArea(.bottom)
.background(TransparentBackground())
})
.onChange(of: pushNotificationsService.handledNotification) { notification in
if notification != nil {
pushNotificationsService.handledNotification = nil
if appAccountsManager.currentAccount.oauthToken?.accessToken != notification?.account.token.accessToken,
let account = appAccountsManager.availableAccounts.first(where:
{ $0.oauthToken?.accessToken == notification?.account.token.accessToken })
{
appAccountsManager.currentAccount = account
DispatchQueue.main.asyncAfter(deadline: .now() + 0.2) {
selectedTab = .notifications
pushNotificationsService.handledNotification = notification
}
} else {
selectedTab = .notifications
}
}
}
}
.commands {
appMenu
}
.onChange(of: scenePhase) { scenePhase in
handleScenePhase(scenePhase: scenePhase)
}
.onChange(of: appAccountsManager.currentClient) { newClient in
setNewClientsInEnv(client: newClient)
if newClient.isAuth {
watcher.watch(streams: [.user, .direct])
}
}
}
@ViewBuilder
private var appView: some View {
if UIDevice.current.userInterfaceIdiom == .pad || UIDevice.current.userInterfaceIdiom == .mac {
sidebarView
} else {
tabBarView
}
}
private func badgeFor(tab: Tab) -> Int {
if tab == .notifications && selectedTab != tab,
let token = appAccountsManager.currentAccount.oauthToken
{
return watcher.unreadNotificationsCount + userPreferences.getNotificationsCount(for: token)
}
return 0
}
private var sidebarView: some View {
SideBarView(selectedTab: $selectedTab,
popToRootTab: $popToRootTab,
tabs: availableTabs,
routerPath: sidebarRouterPath)
{
GeometryReader { _ in
HStack(spacing: 0) {
ZStack {
if selectedTab == .profile {
ProfileTab(popToRootTab: $popToRootTab)
}
ForEach(availableTabs) { tab in
if tab == selectedTab || sideBarLoadedTabs.contains(tab) {
tab
.makeContentView(popToRootTab: $popToRootTab)
.opacity(tab == selectedTab ? 1 : 0)
.transition(.opacity)
.id("\(tab)\(appAccountsManager.currentAccount.id)")
.onAppear {
sideBarLoadedTabs.insert(tab)
}
} else {
EmptyView()
}
}
}
if appAccountsManager.currentClient.isAuth,
userPreferences.showiPadSecondaryColumn
{
Divider().edgesIgnoringSafeArea(.all)
notificationsSecondaryColumn
}
}
}
}.onChange(of: $appAccountsManager.currentAccount.id) { _ in
sideBarLoadedTabs.removeAll()
}
}
private var notificationsSecondaryColumn: some View {
NotificationsTab(popToRootTab: $popToRootTab, lockedType: nil)
.environment(\.isSecondaryColumn, true)
.frame(maxWidth: .secondaryColumnWidth)
.id(appAccountsManager.currentAccount.id)
}
private var tabBarView: some View {
TabView(selection: .init(get: {
selectedTab
}, set: { newTab in
if newTab == selectedTab {
/// Stupid hack to trigger onChange binding in tab views.
popToRootTab = .other
DispatchQueue.main.asyncAfter(deadline: .now() + 0.01) {
popToRootTab = selectedTab
}
}
HapticManager.shared.fireHaptic(of: .tabSelection)
SoundEffectManager.shared.playSound(of: .tabSelection)
selectedTab = newTab
DispatchQueue.main.async {
if selectedTab == .notifications,
let token = appAccountsManager.currentAccount.oauthToken
{
userPreferences.setNotification(count: 0, token: token)
watcher.unreadNotificationsCount = 0
}
}
})) {
ForEach(availableTabs) { tab in
tab.makeContentView(popToRootTab: $popToRootTab)
.tabItem {
if userPreferences.showiPhoneTabLabel {
tab.label
.labelStyle(TitleAndIconLabelStyle())
} else {
tab.label
.labelStyle(IconOnlyLabelStyle())
}
}
.tag(tab)
.badge(badgeFor(tab: tab))
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .tabBar)
}
}
.id(appAccountsManager.currentClient.id)
}
private func setNewClientsInEnv(client: Client) {
currentAccount.setClient(client: client)
currentInstance.setClient(client: client)
userPreferences.setClient(client: client)
Task {
await currentInstance.fetchCurrentInstance()
watcher.setClient(client: client, instanceStreamingURL: currentInstance.instance?.urls?.streamingApi)
watcher.watch(streams: [.user, .direct])
}
}
private func handleScenePhase(scenePhase: ScenePhase) {
switch scenePhase {
case .background:
watcher.stopWatching()
case .active:
watcher.watch(streams: [.user, .direct])
UIApplication.shared.applicationIconBadgeNumber = 0
Task {
await userPreferences.refreshServerPreferences()
}
default:
break
}
}
private func setupRevenueCat() {
Purchases.logLevel = .error
Purchases.configure(withAPIKey: "appl_JXmiRckOzXXTsHKitQiicXCvMQi")
Purchases.shared.getCustomerInfo { info, _ in
if info?.entitlements["Supporter"]?.isActive == true {
isSupporter = true
}
}
}
private func refreshPushSubs() {
PushNotificationsService.shared.requestPushNotifications()
}
@CommandsBuilder
private var appMenu: some Commands {
CommandGroup(replacing: .newItem) {
Button("menu.new-post") {
sidebarRouterPath.presentedSheet = .newStatusEditor(visibility: userPreferences.postVisibility)
}
}
CommandGroup(replacing: .textFormatting) {
Menu("menu.font") {
Button("menu.font.bigger") {
if theme.fontSizeScale < 1.5 {
theme.fontSizeScale += 0.1
}
}
Button("menu.font.smaller") {
if theme.fontSizeScale > 0.5 {
theme.fontSizeScale -= 0.1
}
}
}
}
}
}
class AppDelegate: NSObject, UIApplicationDelegate {
let themeObserver = ThemeObserverViewController(nibName: nil, bundle: nil)
func application(_: UIApplication,
didFinishLaunchingWithOptions _: [UIApplication.LaunchOptionsKey: Any]? = nil) -> Bool
{
try? AVAudioSession.sharedInstance().setCategory(.ambient)
PushNotificationsService.shared.setAccounts(accounts: AppAccountsManager.shared.pushAccounts)
return true
}
func application(_: UIApplication,
didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data)
{
PushNotificationsService.shared.pushToken = deviceToken
Task {
PushNotificationsService.shared.setAccounts(accounts: AppAccountsManager.shared.pushAccounts)
await PushNotificationsService.shared.updateSubscriptions(forceCreate: false)
}
}
func application(_: UIApplication, didFailToRegisterForRemoteNotificationsWithError _: Error) {}
func application(_: UIApplication, configurationForConnecting connectingSceneSession: UISceneSession, options _: UIScene.ConnectionOptions) -> UISceneConfiguration {
let configuration = UISceneConfiguration(name: nil, sessionRole: connectingSceneSession.role)
if connectingSceneSession.role == .windowApplication {
configuration.delegateClass = SceneDelegate.self
}
return configuration
}
}
class ThemeObserverViewController: UIViewController {
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
super.traitCollectionDidChange(previousTraitCollection)
print(traitCollection.userInterfaceStyle.rawValue)
}
}

View file

@ -0,0 +1,180 @@
import Account
import AppAccount
import AVFoundation
import DesignSystem
import Env
import KeychainSwift
import MediaUI
import Network
import RevenueCat
import StatusKit
import SwiftUI
import Timeline
@MainActor
struct AppView: View {
@Environment(AppAccountsManager.self) private var appAccountsManager
@Environment(UserPreferences.self) private var userPreferences
@Environment(Theme.self) private var theme
@Environment(StreamWatcher.self) private var watcher
@Environment(\.openWindow) var openWindow
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
@Binding var selectedTab: AppTab
@Binding var appRouterPath: RouterPath
@State var iosTabs = iOSTabs.shared
@State var sidebarTabs = SidebarTabs.shared
@State var selectedTabScrollToTop: Int = -1
var body: some View {
#if os(visionOS)
tabBarView
#else
if UIDevice.current.userInterfaceIdiom == .pad || UIDevice.current.userInterfaceIdiom == .mac {
sidebarView
} else {
tabBarView
}
#endif
}
var availableTabs: [AppTab] {
guard appAccountsManager.currentClient.isAuth else {
return AppTab.loggedOutTab()
}
if UIDevice.current.userInterfaceIdiom == .phone || horizontalSizeClass == .compact {
return iosTabs.tabs
} else if UIDevice.current.userInterfaceIdiom == .vision {
return AppTab.visionOSTab()
}
return sidebarTabs.tabs.map { $0.tab }
}
@ViewBuilder
var tabBarView: some View {
TabView(selection: .init(get: {
selectedTab
}, set: { newTab in
updateTab(with: newTab)
})) {
ForEach(availableTabs) { tab in
tab.makeContentView(selectedTab: $selectedTab)
.tabItem {
if userPreferences.showiPhoneTabLabel {
tab.label
.environment(\.symbolVariants, tab == selectedTab ? .fill : .none)
} else {
Image(systemName: tab.iconName)
}
}
.tag(tab)
.badge(badgeFor(tab: tab))
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .tabBar)
}
}
.id(appAccountsManager.currentClient.id)
.withSheetDestinations(sheetDestinations: $appRouterPath.presentedSheet)
.environment(\.selectedTabScrollToTop, selectedTabScrollToTop)
}
private func updateTab(with newTab: AppTab) {
if newTab == .post {
#if os(visionOS)
openWindow(value: WindowDestinationEditor.newStatusEditor(visibility: userPreferences.postVisibility))
#else
appRouterPath.presentedSheet = .newStatusEditor(visibility: userPreferences.postVisibility)
#endif
return
}
HapticManager.shared.fireHaptic(.tabSelection)
SoundEffectManager.shared.playSound(.tabSelection)
if selectedTab == newTab {
selectedTabScrollToTop = newTab.rawValue
DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) {
selectedTabScrollToTop = -1
}
} else {
selectedTabScrollToTop = -1
}
selectedTab = newTab
}
private func badgeFor(tab: AppTab) -> Int {
if tab == .notifications, selectedTab != tab,
let token = appAccountsManager.currentAccount.oauthToken
{
return watcher.unreadNotificationsCount + (userPreferences.notificationsCount[token] ?? 0)
}
return 0
}
#if !os(visionOS)
var sidebarView: some View {
SideBarView(selectedTab: .init(get: {
selectedTab
}, set: { newTab in
updateTab(with: newTab)
}), tabs: availableTabs)
{
HStack(spacing: 0) {
if #available(iOS 18.0, *) {
baseTabView
#if targetEnvironment(macCatalyst)
.tabViewStyle(.sidebarAdaptable)
.introspect(.tabView, on: .iOS(.v17, .v18)) { (tabview: UITabBarController) in
tabview.sidebar.isHidden = true
}
#else
.tabViewStyle(.tabBarOnly)
#endif
} else {
baseTabView
}
if horizontalSizeClass == .regular,
appAccountsManager.currentClient.isAuth,
userPreferences.showiPadSecondaryColumn
{
Divider().edgesIgnoringSafeArea(.all)
notificationsSecondaryColumn
}
}
}
.environment(appRouterPath)
.environment(\.selectedTabScrollToTop, selectedTabScrollToTop)
}
#endif
private var baseTabView: some View {
TabView(selection: $selectedTab) {
ForEach(availableTabs) { tab in
tab
.makeContentView(selectedTab: $selectedTab)
.toolbar(horizontalSizeClass == .regular ? .hidden : .visible, for: .tabBar)
.tabItem {
tab.label
}
.tag(tab)
}
}
#if !os(visionOS)
.introspect(.tabView, on: .iOS(.v17, .v18)) { (tabview: UITabBarController) in
tabview.tabBar.isHidden = horizontalSizeClass == .regular
tabview.customizableViewControllers = []
tabview.moreNavigationController.isNavigationBarHidden = true
}
#endif
}
var notificationsSecondaryColumn: some View {
NotificationsTab(selectedTab: .constant(.notifications)
, lockedType: nil)
.environment(\.isSecondaryColumn, true)
.frame(maxWidth: .secondaryColumnWidth)
.id(appAccountsManager.currentAccount.id)
}
}

View file

@ -0,0 +1,70 @@
import Env
import SwiftUI
extension IceCubesApp {
@CommandsBuilder
var appMenu: some Commands {
CommandGroup(replacing: .appSettings) {
Button("menu.settings") {
appRouterPath.presentedSheet = .settings
}
.keyboardShortcut(",", modifiers: .command)
}
CommandGroup(replacing: .newItem) {
Button("menu.new-window") {
openWindow(id: "MainWindow")
}
.keyboardShortcut("n", modifiers: .shift)
Button("menu.new-post") {
#if targetEnvironment(macCatalyst)
openWindow(value: WindowDestinationEditor.newStatusEditor(visibility: userPreferences.postVisibility))
#else
appRouterPath.presentedSheet = .newStatusEditor(visibility: userPreferences.postVisibility)
#endif
}
.keyboardShortcut("n", modifiers: .command)
}
CommandGroup(replacing: .textFormatting) {
Menu("menu.font") {
Button("menu.font.bigger") {
if theme.fontSizeScale < 1.5 {
theme.fontSizeScale += 0.1
}
}
Button("menu.font.smaller") {
if theme.fontSizeScale > 0.5 {
theme.fontSizeScale -= 0.1
}
}
}
}
CommandMenu("tab.timeline") {
Button("timeline.latest") {
NotificationCenter.default.post(name: .refreshTimeline, object: nil)
}
.keyboardShortcut("r", modifiers: .command)
Button("timeline.home") {
NotificationCenter.default.post(name: .homeTimeline, object: nil)
}
.keyboardShortcut("h", modifiers: .shift)
Button("timeline.trending") {
NotificationCenter.default.post(name: .trendingTimeline, object: nil)
}
.keyboardShortcut("t", modifiers: .shift)
Button("timeline.federated") {
NotificationCenter.default.post(name: .federatedTimeline, object: nil)
}
.keyboardShortcut("f", modifiers: .shift)
Button("timeline.local") {
NotificationCenter.default.post(name: .localTimeline, object: nil)
}
.keyboardShortcut("l", modifiers: .shift)
}
CommandGroup(replacing: .help) {
Button("menu.help.github") {
let url = URL(string: "https://github.com/Dimillian/IceCubesApp/issues")!
UIApplication.shared.open(url)
}
}
}
}

View file

@ -0,0 +1,169 @@
import AppIntents
import Env
import MediaUI
import StatusKit
import SwiftUI
extension IceCubesApp {
var appScene: some Scene {
WindowGroup(id: "MainWindow") {
AppView(selectedTab: $selectedTab, appRouterPath: $appRouterPath)
.applyTheme(theme)
.onAppear {
setNewClientsInEnv(client: appAccountsManager.currentClient)
setupRevenueCat()
refreshPushSubs()
}
.environment(appAccountsManager)
.environment(appAccountsManager.currentClient)
.environment(quickLook)
.environment(currentAccount)
.environment(currentInstance)
.environment(userPreferences)
.environment(theme)
.environment(watcher)
.environment(pushNotificationsService)
.environment(appIntentService)
.environment(\.isSupporter, isSupporter)
.sheet(item: $quickLook.selectedMediaAttachment) { selectedMediaAttachment in
if #available(iOS 18.0, *) {
MediaUIView(selectedAttachment: selectedMediaAttachment,
attachments: quickLook.mediaAttachments)
.presentationBackground(.ultraThinMaterial)
.presentationCornerRadius(16)
.presentationSizing(.page)
.withEnvironments()
} else {
MediaUIView(selectedAttachment: selectedMediaAttachment,
attachments: quickLook.mediaAttachments)
.presentationBackground(.ultraThinMaterial)
.presentationCornerRadius(16)
.withEnvironments()
}
}
.onChange(of: pushNotificationsService.handledNotification) { _, newValue in
if newValue != nil {
pushNotificationsService.handledNotification = nil
if appAccountsManager.currentAccount.oauthToken?.accessToken != newValue?.account.token.accessToken,
let account = appAccountsManager.availableAccounts.first(where:
{ $0.oauthToken?.accessToken == newValue?.account.token.accessToken })
{
appAccountsManager.currentAccount = account
DispatchQueue.main.asyncAfter(deadline: .now() + 0.2) {
selectedTab = .notifications
pushNotificationsService.handledNotification = newValue
}
} else {
selectedTab = .notifications
}
}
}
.onChange(of: appIntentService.handledIntent) { _, _ in
if let intent = appIntentService.handledIntent?.intent {
handleIntent(intent)
appIntentService.handledIntent = nil
}
}
.withModelContainer()
}
.commands {
appMenu
}
.onChange(of: scenePhase) { _, newValue in
handleScenePhase(scenePhase: newValue)
}
.onChange(of: appAccountsManager.currentClient) { _, newValue in
setNewClientsInEnv(client: newValue)
if newValue.isAuth {
watcher.watch(streams: [.user, .direct])
}
}
#if targetEnvironment(macCatalyst)
.windowResize()
#elseif os(visionOS)
.defaultSize(width: 800, height: 1200)
#endif
}
@SceneBuilder
var otherScenes: some Scene {
WindowGroup(for: WindowDestinationEditor.self) { destination in
Group {
switch destination.wrappedValue {
case let .newStatusEditor(visibility):
StatusEditor.MainView(mode: .new(text: nil, visibility: visibility))
case let .prefilledStatusEditor(text, visibility):
StatusEditor.MainView(mode: .new(text: text, visibility: visibility))
case let .editStatusEditor(status):
StatusEditor.MainView(mode: .edit(status: status))
case let .quoteStatusEditor(status):
StatusEditor.MainView(mode: .quote(status: status))
case let .replyToStatusEditor(status):
StatusEditor.MainView(mode: .replyTo(status: status))
case let .mentionStatusEditor(account, visibility):
StatusEditor.MainView(mode: .mention(account: account, visibility: visibility))
case let .quoteLinkStatusEditor(link):
StatusEditor.MainView(mode: .quoteLink(link: link))
case .none:
EmptyView()
}
}
.withEnvironments()
.environment(\.isCatalystWindow, true)
.environment(RouterPath())
.withModelContainer()
.applyTheme(theme)
.frame(minWidth: 300, minHeight: 400)
}
.defaultSize(width: 600, height: 800)
.windowResizability(.contentMinSize)
WindowGroup(for: WindowDestinationMedia.self) { destination in
Group {
switch destination.wrappedValue {
case let .mediaViewer(attachments, selectedAttachment):
MediaUIView(selectedAttachment: selectedAttachment,
attachments: attachments)
case .none:
EmptyView()
}
}
.withEnvironments()
.withModelContainer()
.applyTheme(theme)
.environment(\.isCatalystWindow, true)
.frame(minWidth: 300, minHeight: 400)
}
.defaultSize(width: 1200, height: 1000)
.windowResizability(.contentMinSize)
}
private func handleIntent(_: any AppIntent) {
if let postIntent = appIntentService.handledIntent?.intent as? PostIntent {
#if os(visionOS) || os(macOS)
openWindow(value: WindowDestinationEditor.prefilledStatusEditor(text: postIntent.content ?? "",
visibility: userPreferences.postVisibility))
#else
appRouterPath.presentedSheet = .prefilledStatusEditor(text: postIntent.content ?? "",
visibility: userPreferences.postVisibility)
#endif
} else if let tabIntent = appIntentService.handledIntent?.intent as? TabIntent {
selectedTab = tabIntent.tab.toAppTab
} else if let imageIntent = appIntentService.handledIntent?.intent as? PostImageIntent,
let urls = imageIntent.images?.compactMap({ $0.fileURL })
{
appRouterPath.presentedSheet = .imageURL(urls: urls,
visibility: userPreferences.postVisibility)
}
}
}
extension Scene {
func windowResize() -> some Scene {
if #available(iOS 18.0, *) {
return self.windowResizability(.contentSize)
} else {
return self.defaultSize(width: 1100, height: 1400)
}
}
}

View file

@ -0,0 +1,128 @@
import Account
import AppAccount
import AVFoundation
import DesignSystem
import Env
import KeychainSwift
import MediaUI
import Network
import RevenueCat
import StatusKit
import SwiftUI
import Timeline
import WishKit
@main
struct IceCubesApp: App {
@UIApplicationDelegateAdaptor private var appDelegate: AppDelegate
@Environment(\.scenePhase) var scenePhase
@Environment(\.openWindow) var openWindow
@State var appAccountsManager = AppAccountsManager.shared
@State var currentInstance = CurrentInstance.shared
@State var currentAccount = CurrentAccount.shared
@State var userPreferences = UserPreferences.shared
@State var pushNotificationsService = PushNotificationsService.shared
@State var appIntentService = AppIntentService.shared
@State var watcher = StreamWatcher.shared
@State var quickLook = QuickLook.shared
@State var theme = Theme.shared
@State var selectedTab: AppTab = .timeline
@State var appRouterPath = RouterPath()
@State var isSupporter: Bool = false
var body: some Scene {
appScene
otherScenes
}
func setNewClientsInEnv(client: Client) {
currentAccount.setClient(client: client)
currentInstance.setClient(client: client)
userPreferences.setClient(client: client)
Task {
await currentInstance.fetchCurrentInstance()
watcher.setClient(client: client, instanceStreamingURL: currentInstance.instance?.urls?.streamingApi)
watcher.watch(streams: [.user, .direct])
}
}
func handleScenePhase(scenePhase: ScenePhase) {
switch scenePhase {
case .background:
watcher.stopWatching()
case .active:
watcher.watch(streams: [.user, .direct])
UNUserNotificationCenter.current().setBadgeCount(0)
userPreferences.reloadNotificationsCount(tokens: appAccountsManager.availableAccounts.compactMap(\.oauthToken))
Task {
await userPreferences.refreshServerPreferences()
}
default:
break
}
}
func setupRevenueCat() {
Purchases.logLevel = .error
Purchases.configure(withAPIKey: "appl_JXmiRckOzXXTsHKitQiicXCvMQi")
Purchases.shared.getCustomerInfo { info, _ in
if info?.entitlements["Supporter"]?.isActive == true {
isSupporter = true
}
}
}
func refreshPushSubs() {
PushNotificationsService.shared.requestPushNotifications()
}
}
class AppDelegate: UIResponder, UIApplicationDelegate {
func application(_: UIApplication,
didFinishLaunchingWithOptions _: [UIApplication.LaunchOptionsKey: Any]? = nil) -> Bool
{
try? AVAudioSession.sharedInstance().setCategory(.ambient, options: .mixWithOthers)
try? AVAudioSession.sharedInstance().setActive(true)
PushNotificationsService.shared.setAccounts(accounts: AppAccountsManager.shared.pushAccounts)
Telemetry.setup()
Telemetry.signal("app.launched")
WishKit.configure(with: "AF21AE07-3BA9-4FE2-BFB1-59A3B3941730")
return true
}
func application(_: UIApplication,
didRegisterForRemoteNotificationsWithDeviceToken deviceToken: Data)
{
PushNotificationsService.shared.pushToken = deviceToken
Task {
PushNotificationsService.shared.setAccounts(accounts: AppAccountsManager.shared.pushAccounts)
await PushNotificationsService.shared.updateSubscriptions(forceCreate: false)
}
}
func application(_: UIApplication, didFailToRegisterForRemoteNotificationsWithError _: Error) {}
func application(_: UIApplication, didReceiveRemoteNotification _: [AnyHashable: Any]) async -> UIBackgroundFetchResult {
UserPreferences.shared.reloadNotificationsCount(tokens: AppAccountsManager.shared.availableAccounts.compactMap(\.oauthToken))
return .noData
}
func application(_: UIApplication, configurationForConnecting connectingSceneSession: UISceneSession, options _: UIScene.ConnectionOptions) -> UISceneConfiguration {
let configuration = UISceneConfiguration(name: nil, sessionRole: connectingSceneSession.role)
if connectingSceneSession.role == .windowApplication {
configuration.delegateClass = SceneDelegate.self
}
return configuration
}
override func buildMenu(with builder: UIMenuBuilder) {
super.buildMenu(with: builder)
builder.remove(menu: .document)
builder.remove(menu: .toolbar)
builder.remove(menu: .sidebar)
}
}

View file

@ -1,96 +0,0 @@
import QuickLook
import SwiftUI
import UIKit
extension URL: Identifiable {
public var id: String {
absoluteString
}
}
struct QuickLookPreview: UIViewControllerRepresentable {
let selectedURL: URL
let urls: [URL]
func makeUIViewController(context _: Context) -> UIViewController {
return AppQLPreviewController(selectedURL: selectedURL, urls: urls)
}
func updateUIViewController(
_: UIViewController, context _: Context
) {}
}
class AppQLPreviewController: UIViewController {
let selectedURL: URL
let urls: [URL]
var qlController: QLPreviewController?
init(selectedURL: URL, urls: [URL]) {
self.selectedURL = selectedURL
self.urls = urls
super.init(nibName: nil, bundle: nil)
}
@available(*, unavailable)
required init?(coder _: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
if qlController == nil {
qlController = QLPreviewController()
qlController?.dataSource = self
qlController?.delegate = self
qlController?.currentPreviewItemIndex = urls.firstIndex(of: selectedURL) ?? 0
present(qlController!, animated: true)
}
}
}
extension AppQLPreviewController: QLPreviewControllerDataSource {
func numberOfPreviewItems(in _: QLPreviewController) -> Int {
return urls.count
}
func previewController(_: QLPreviewController, previewItemAt index: Int) -> QLPreviewItem {
return urls[index] as QLPreviewItem
}
}
extension AppQLPreviewController: QLPreviewControllerDelegate {
func previewController(_: QLPreviewController, editingModeFor _: QLPreviewItem) -> QLPreviewItemEditingMode {
.createCopy
}
func previewControllerWillDismiss(_: QLPreviewController) {
dismiss(animated: true)
}
func previewControllerDidDismiss(_: QLPreviewController) {
dismiss(animated: true)
}
}
struct TransparentBackground: UIViewControllerRepresentable {
public func makeUIViewController(context _: Context) -> UIViewController {
return TransparentController()
}
public func updateUIViewController(_: UIViewController, context _: Context) {}
class TransparentController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
view.backgroundColor = .clear
}
override func willMove(toParent parent: UIViewController?) {
super.willMove(toParent: parent)
parent?.view?.backgroundColor = .clear
parent?.modalPresentationStyle = .overCurrentContext
}
}
}

View file

@ -2,14 +2,14 @@ import DesignSystem
import Env import Env
import Models import Models
import Network import Network
import Status import StatusKit
import SwiftUI import SwiftUI
public struct ReportView: View { public struct ReportView: View {
@Environment(\.dismiss) private var dismiss @Environment(\.dismiss) private var dismiss
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var client: Client @Environment(Client.self) private var client
let status: Status let status: Status
@State private var commentText: String = "" @State private var commentText: String = ""
@ -35,42 +35,38 @@ public struct ReportView: View {
} }
.navigationTitle("report.title") .navigationTitle("report.title")
.navigationBarTitleDisplayMode(.inline) .navigationBarTitleDisplayMode(.inline)
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.scrollDismissesKeyboard(.immediately) .background(theme.secondaryBackgroundColor)
.toolbar { .scrollDismissesKeyboard(.immediately)
ToolbarItem(placement: .navigationBarTrailing) { #endif
Button { .toolbar {
isSendingReport = true ToolbarItem(placement: .navigationBarTrailing) {
Task { Button {
do { isSendingReport = true
let _: ReportSent = Task {
try await client.post(endpoint: Statuses.report(accountId: status.account.id, do {
statusId: status.id, let _: ReportSent =
comment: commentText)) try await client.post(endpoint: Statuses.report(accountId: status.account.id,
dismiss() statusId: status.id,
isSendingReport = false comment: commentText))
} catch { dismiss()
isSendingReport = false isSendingReport = false
} catch {
isSendingReport = false
}
}
} label: {
if isSendingReport {
ProgressView()
} else {
Text("report.action.send")
} }
} }
} label: {
if isSendingReport {
ProgressView()
} else {
Text("report.action.send")
}
} }
}
ToolbarItem(placement: .navigationBarLeading) { CancelToolbarItem()
Button {
dismiss()
} label: {
Text("action.cancel")
}
} }
}
} }
} }
} }

View file

@ -1,32 +1,49 @@
import DesignSystem import DesignSystem
import Env import Env
import Models
import Observation
import SafariServices import SafariServices
import SwiftUI import SwiftUI
import AppAccount
import WebKit
extension View { extension View {
func withSafariRouter() -> some View { @MainActor func withSafariRouter() -> some View {
modifier(SafariRouter()) modifier(SafariRouter())
} }
} }
@MainActor
private struct SafariRouter: ViewModifier { private struct SafariRouter: ViewModifier {
@EnvironmentObject private var theme: Theme @Environment(\.isSecondaryColumn) private var isSecondaryColumn: Bool
@EnvironmentObject private var preferences: UserPreferences @Environment(Theme.self) private var theme
@EnvironmentObject private var routerPath: RouterPath @Environment(UserPreferences.self) private var preferences
@Environment(RouterPath.self) private var routerPath
@Environment(AppAccountsManager.self) private var appAccount
@StateObject private var safariManager = InAppSafariManager() #if !os(visionOS)
@State private var safariManager = InAppSafariManager()
#endif
func body(content: Content) -> some View { func body(content: Content) -> some View {
content content
.environment(\.openURL, OpenURLAction { url in .environment(\.openURL, OpenURLAction { url in
// Open internal URL. // Open internal URL.
routerPath.handle(url: url) guard !isSecondaryColumn else { return .discarded }
return routerPath.handle(url: url)
}) })
.onOpenURL { url in .onOpenURL { url in
// Open external URL (from icecubesapp://) // Open external URL (from icecubesapp://)
let urlString = url.absoluteString.replacingOccurrences(of: "icecubesapp://", with: "https://") guard !isSecondaryColumn else { return }
if url.absoluteString == "icecubesapp://subclub" {
#if !os(visionOS)
safariManager.dismiss()
#endif
return
}
let urlString = url.absoluteString.replacingOccurrences(of: AppInfo.scheme, with: "https://")
guard let url = URL(string: urlString), url.host != nil else { return } guard let url = URL(string: urlString), url.host != nil else { return }
_ = routerPath.handle(url: url) _ = routerPath.handleDeepLink(url: url)
} }
.onAppear { .onAppear {
routerPath.urlHandler = { url in routerPath.urlHandler = { url in
@ -39,72 +56,107 @@ private struct SafariRouter: ViewModifier {
UIApplication.shared.open(url) UIApplication.shared.open(url)
return .handled return .handled
} }
} } else if url.query()?.contains("callback=") == false,
guard preferences.preferredBrowser == .inAppSafari, !ProcessInfo.processInfo.isiOSAppOnMac else { return .systemAction } url.host() == AppInfo.premiumInstance,
// SFSafariViewController only supports initial URLs with http:// or https:// schemes. let accountName = appAccount.currentAccount.accountName {
guard let scheme = url.scheme, ["https", "http"].contains(scheme.lowercased()) else { let newURL = url.appending(queryItems: [
.init(name: "callback", value: "icecubesapp://subclub"),
.init(name: "id", value: "@\(accountName)")
])
#if !os(visionOS)
return safariManager.open(newURL)
#else
return .systemAction return .systemAction
#endif
} }
return safariManager.open(url) #if !targetEnvironment(macCatalyst)
guard preferences.preferredBrowser == .inAppSafari else { return .systemAction }
// SFSafariViewController only supports initial URLs with http:// or https:// schemes.
guard let scheme = url.scheme, ["https", "http"].contains(scheme.lowercased()) else {
return .systemAction
}
#if os(visionOS)
return .systemAction
#else
return safariManager.open(url)
#endif
#else
return .systemAction
#endif
} }
} }
#if !os(visionOS)
.background { .background {
WindowReader { window in WindowReader { window in
self.safariManager.windowScene = window.windowScene safariManager.windowScene = window.windowScene
} }
} }
#endif
} }
} }
private class InAppSafariManager: NSObject, ObservableObject, SFSafariViewControllerDelegate { #if !os(visionOS)
var windowScene: UIWindowScene?
let viewController: UIViewController = .init()
var window: UIWindow?
@MainActor @MainActor
func open(_ url: URL) -> OpenURLAction.Result { @Observable private class InAppSafariManager: NSObject, SFSafariViewControllerDelegate {
guard let windowScene = windowScene else { return .systemAction } var windowScene: UIWindowScene?
let viewController: UIViewController = .init()
var window: UIWindow?
window = setupWindow(windowScene: windowScene) @MainActor
func open(_ url: URL) -> OpenURLAction.Result {
guard let windowScene else { return .systemAction }
let configuration = SFSafariViewController.Configuration() window = setupWindow(windowScene: windowScene)
configuration.entersReaderIfAvailable = UserPreferences.shared.inAppBrowserReaderView
let safari = SFSafariViewController(url: url, configuration: configuration) let configuration = SFSafariViewController.Configuration()
safari.preferredBarTintColor = UIColor(Theme.shared.primaryBackgroundColor) configuration.entersReaderIfAvailable = UserPreferences.shared.inAppBrowserReaderView
safari.preferredControlTintColor = UIColor(Theme.shared.tintColor)
safari.delegate = self
DispatchQueue.main.async { [weak self] in let safari = SFSafariViewController(url: url, configuration: configuration)
self?.viewController.present(safari, animated: true) safari.preferredBarTintColor = UIColor(Theme.shared.primaryBackgroundColor)
safari.preferredControlTintColor = UIColor(Theme.shared.tintColor)
safari.delegate = self
DispatchQueue.main.async { [weak self] in
self?.viewController.present(safari, animated: true)
}
return .handled
}
func dismiss() {
viewController.presentedViewController?.dismiss(animated: true)
window?.resignKey()
window?.isHidden = false
window = nil
} }
return .handled func setupWindow(windowScene: UIWindowScene) -> UIWindow {
} let window = window ?? UIWindow(windowScene: windowScene)
func setupWindow(windowScene: UIWindowScene) -> UIWindow { window.rootViewController = viewController
let window = self.window ?? UIWindow(windowScene: windowScene) window.makeKeyAndVisible()
window.rootViewController = viewController switch Theme.shared.selectedScheme {
window.makeKeyAndVisible() case .dark:
window.overrideUserInterfaceStyle = .dark
case .light:
window.overrideUserInterfaceStyle = .light
}
switch Theme.shared.selectedScheme { self.window = window
case .dark: return window
window.overrideUserInterfaceStyle = .dark
case .light:
window.overrideUserInterfaceStyle = .light
} }
self.window = window nonisolated func safariViewControllerDidFinish(_: SFSafariViewController) {
return window Task { @MainActor in
window?.resignKey()
window?.isHidden = false
window = nil
}
}
} }
#endif
func safariViewControllerDidFinish(_: SFSafariViewController) {
window?.resignKey()
window?.isHidden = false
window = nil
}
}
private struct WindowReader: UIViewRepresentable { private struct WindowReader: UIViewRepresentable {
var onUpdate: (UIWindow) -> Void var onUpdate: (UIWindow) -> Void

View file

@ -4,40 +4,61 @@ import DesignSystem
import Env import Env
import Models import Models
import SwiftUI import SwiftUI
import SwiftUIIntrospect
@MainActor
struct SideBarView<Content: View>: View { struct SideBarView<Content: View>: View {
@EnvironmentObject private var appAccounts: AppAccountsManager @Environment(\.openWindow) private var openWindow
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(\.horizontalSizeClass) private var horizontalSizeClass
@EnvironmentObject private var theme: Theme
@EnvironmentObject private var watcher: StreamWatcher
@EnvironmentObject private var userPreferences: UserPreferences
@Binding var selectedTab: Tab @Environment(AppAccountsManager.self) private var appAccounts
@Binding var popToRootTab: Tab @Environment(CurrentAccount.self) private var currentAccount
var tabs: [Tab] @Environment(Theme.self) private var theme
@ObservedObject var routerPath = RouterPath() @Environment(StreamWatcher.self) private var watcher
@Environment(UserPreferences.self) private var userPreferences
@Environment(RouterPath.self) private var routerPath
@Binding var selectedTab: AppTab
var tabs: [AppTab]
@ViewBuilder var content: () -> Content @ViewBuilder var content: () -> Content
private func badgeFor(tab: Tab) -> Int { @State private var sidebarTabs = SidebarTabs.shared
if tab == .notifications && selectedTab != tab,
private func badgeFor(tab: AppTab) -> Int {
if tab == .notifications, selectedTab != tab,
let token = appAccounts.currentAccount.oauthToken let token = appAccounts.currentAccount.oauthToken
{ {
return watcher.unreadNotificationsCount + userPreferences.getNotificationsCount(for: token) return watcher.unreadNotificationsCount + (userPreferences.notificationsCount[token] ?? 0)
} }
return 0 return 0
} }
private func makeIconForTab(tab: Tab) -> some View { private func makeIconForTab(tab: AppTab) -> some View {
ZStack(alignment: .topTrailing) { ZStack(alignment: .topTrailing) {
SideBarIcon(systemIconName: tab.iconName, HStack {
isSelected: tab == selectedTab) SideBarIcon(systemIconName: tab.iconName,
isSelected: tab == selectedTab)
if userPreferences.isSidebarExpanded {
Text(tab.title)
.font(.headline)
.foregroundColor(tab == selectedTab ? theme.tintColor : theme.labelColor)
.frame(maxWidth: .infinity, alignment: .leading)
}
}
.frame(width: (userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth) - 24, height: 50)
.background(tab == selectedTab ? theme.primaryBackgroundColor : .clear,
in: RoundedRectangle(cornerRadius: 8))
.cornerRadius(8)
.shadow(color: tab == selectedTab ? .black.opacity(0.2) : .clear, radius: 5)
.overlay(
RoundedRectangle(cornerRadius: 8)
.stroke(tab == selectedTab ? theme.labelColor.opacity(0.1) : .clear, lineWidth: 1)
)
let badge = badgeFor(tab: tab) let badge = badgeFor(tab: tab)
if badge > 0 { if badge > 0 {
makeBadgeView(count: badge) makeBadgeView(count: badge)
} }
} }
.contentShape(Rectangle())
.frame(width: .sidebarWidth, height: 50)
} }
private func makeBadgeView(count: Int) -> some View { private func makeBadgeView(count: Int) -> some View {
@ -49,27 +70,32 @@ struct SideBarView<Content: View>: View {
.font(.caption2) .font(.caption2)
} }
.frame(width: 24, height: 24) .frame(width: 24, height: 24)
.offset(x: 14, y: -14) .offset(x: 5, y: -5)
} }
private var postButton: some View { private var postButton: some View {
Button { Button {
routerPath.presentedSheet = .newStatusEditor(visibility: userPreferences.postVisibility) #if targetEnvironment(macCatalyst) || os(visionOS)
openWindow(value: WindowDestinationEditor.newStatusEditor(visibility: userPreferences.postVisibility))
#else
routerPath.presentedSheet = .newStatusEditor(visibility: userPreferences.postVisibility)
#endif
} label: { } label: {
Image(systemName: "square.and.pencil") Image(systemName: "square.and.pencil")
.resizable() .resizable()
.aspectRatio(contentMode: .fit) .aspectRatio(contentMode: .fit)
.frame(width: 20, height: 30) .frame(width: 20, height: 30)
.offset(x: 2, y: -2)
} }
.buttonStyle(.borderedProminent) .buttonStyle(.borderedProminent)
.keyboardShortcut("n", modifiers: .command) .help(AppTab.post.title)
} }
private func makeAccountButton(account: AppAccount, showBadge: Bool) -> some View { private func makeAccountButton(account: AppAccount, showBadge: Bool) -> some View {
Button { Button {
if account.id == appAccounts.currentAccount.id { if account.id == appAccounts.currentAccount.id {
selectedTab = .profile selectedTab = .profile
SoundEffectManager.shared.playSound(of: .tabSelection) SoundEffectManager.shared.playSound(.tabSelection)
} else { } else {
var transation = Transaction() var transation = Transaction()
transation.disablesAnimations = true transation.disablesAnimations = true
@ -79,82 +105,114 @@ struct SideBarView<Content: View>: View {
} }
} label: { } label: {
ZStack(alignment: .topTrailing) { ZStack(alignment: .topTrailing) {
AppAccountView(viewModel: .init(appAccount: account, isCompact: true)) if userPreferences.isSidebarExpanded {
if showBadge, AppAccountView(viewModel: .init(appAccount: account,
isCompact: false,
isInSettings: false),
isParentPresented: .constant(false))
} else {
AppAccountView(viewModel: .init(appAccount: account,
isCompact: true,
isInSettings: false),
isParentPresented: .constant(false))
}
if !userPreferences.isSidebarExpanded,
showBadge,
let token = account.oauthToken, let token = account.oauthToken,
userPreferences.getNotificationsCount(for: token) > 0 let notificationsCount = userPreferences.notificationsCount[token],
notificationsCount > 0
{ {
makeBadgeView(count: userPreferences.getNotificationsCount(for: token)) makeBadgeView(count: notificationsCount)
} }
} }
.padding(.leading, userPreferences.isSidebarExpanded ? 16 : 0)
} }
.frame(width: .sidebarWidth, height: 50) .help(accountButtonTitle(accountName: account.accountName))
.frame(width: userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth, height: 50)
.padding(.vertical, 8) .padding(.vertical, 8)
.background(selectedTab == .profile && account.id == appAccounts.currentAccount.id ? .background(selectedTab == .profile && account.id == appAccounts.currentAccount.id ?
theme.secondaryBackgroundColor : .clear) theme.secondaryBackgroundColor : .clear)
} }
private func accountButtonTitle(accountName: String?) -> LocalizedStringKey {
if let accountName {
"tab.profile-account-\(accountName)"
} else {
AppTab.profile.title
}
}
private var tabsView: some View { private var tabsView: some View {
ForEach(tabs) { tab in ForEach(tabs) { tab in
Button { if tab != .profile && sidebarTabs.isEnabled(tab) {
// ensure keyboard is always dismissed when selecting a tab Button {
hideKeyboard() // ensure keyboard is always dismissed when selecting a tab
hideKeyboard()
if tab == selectedTab { selectedTab = tab
popToRootTab = .other SoundEffectManager.shared.playSound(.tabSelection)
DispatchQueue.main.asyncAfter(deadline: .now() + 0.01) { if tab == .notifications {
popToRootTab = tab if let token = appAccounts.currentAccount.oauthToken {
userPreferences.notificationsCount[token] = 0
}
watcher.unreadNotificationsCount = 0
} }
} label: {
makeIconForTab(tab: tab)
} }
selectedTab = tab .help(tab.title)
SoundEffectManager.shared.playSound(of: .tabSelection)
if tab == .notifications {
if let token = appAccounts.currentAccount.oauthToken {
userPreferences.setNotification(count: 0, token: token)
}
watcher.unreadNotificationsCount = 0
}
} label: {
makeIconForTab(tab: tab)
} }
.background(tab == selectedTab ? theme.secondaryBackgroundColor : .clear)
} }
} }
var body: some View { var body: some View {
@Bindable var routerPath = routerPath
HStack(spacing: 0) { HStack(spacing: 0) {
ScrollView { if horizontalSizeClass == .regular {
VStack(alignment: .center) { ScrollView {
if appAccounts.availableAccounts.isEmpty { VStack(alignment: .center) {
tabsView if appAccounts.availableAccounts.isEmpty {
} else { tabsView
ForEach(appAccounts.availableAccounts) { account in } else {
makeAccountButton(account: account, ForEach(appAccounts.availableAccounts) { account in
showBadge: account.id != appAccounts.currentAccount.id) makeAccountButton(account: account,
if account.id == appAccounts.currentAccount.id { showBadge: account.id != appAccounts.currentAccount.id)
tabsView if account.id == appAccounts.currentAccount.id {
tabsView
}
} }
} }
} }
postButton
.padding(.top, 12)
Spacer()
} }
.frame(width: userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth)
.scrollContentBackground(.hidden)
.background(.thinMaterial)
.safeAreaInset(edge: .bottom, content: {
HStack(spacing: 16) {
postButton
.padding(.vertical, 24)
.padding(.leading, userPreferences.isSidebarExpanded ? 18 : 0)
if userPreferences.isSidebarExpanded {
Text("menu.new-post")
.font(.subheadline)
.foregroundColor(theme.labelColor)
.frame(maxWidth: .infinity, alignment: .leading)
}
}
.frame(width: userPreferences.isSidebarExpanded ? .sidebarWidthExpanded : .sidebarWidth)
.background(.thinMaterial)
})
Divider().edgesIgnoringSafeArea(.all)
} }
.frame(width: .sidebarWidth)
.scrollContentBackground(.hidden)
.background(.thinMaterial)
Divider()
.edgesIgnoringSafeArea(.top)
content() content()
} }
.background(.thinMaterial) .background(.thinMaterial)
.edgesIgnoringSafeArea(.bottom)
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet) .withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
} }
} }
private struct SideBarIcon: View { private struct SideBarIcon: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
let systemIconName: String let systemIconName: String
let isSelected: Bool let isSelected: Bool
@ -166,17 +224,19 @@ private struct SideBarIcon: View {
.font(.title2) .font(.title2)
.fontWeight(.medium) .fontWeight(.medium)
.foregroundColor(isSelected ? theme.tintColor : theme.labelColor) .foregroundColor(isSelected ? theme.tintColor : theme.labelColor)
.symbolVariant(isSelected ? .fill : .none)
.scaleEffect(isHovered ? 0.8 : 1.0) .scaleEffect(isHovered ? 0.8 : 1.0)
.onHover { isHovered in .onHover { isHovered in
withAnimation(.interpolatingSpring(stiffness: 300, damping: 15)) { withAnimation(.interpolatingSpring(stiffness: 300, damping: 15)) {
self.isHovered = isHovered self.isHovered = isHovered
} }
} }
.frame(width: 50, height: 40)
} }
} }
extension View { extension View {
func hideKeyboard() { @MainActor func hideKeyboard() {
let resign = #selector(UIResponder.resignFirstResponder) let resign = #selector(UIResponder.resignFirstResponder)
UIApplication.shared.sendAction(resign, to: nil, from: nil, for: nil) UIApplication.shared.sendAction(resign, to: nil, from: nil, for: nil)
} }

View file

@ -4,44 +4,29 @@ import Env
import Explore import Explore
import Models import Models
import Network import Network
import Shimmer
import SwiftUI import SwiftUI
@MainActor
struct ExploreTab: View { struct ExploreTab: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var preferences: UserPreferences @Environment(UserPreferences.self) private var preferences
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(CurrentAccount.self) private var currentAccount
@EnvironmentObject private var client: Client @Environment(Client.self) private var client
@StateObject private var routerPath = RouterPath() @State private var routerPath = RouterPath()
@Binding var popToRootTab: Tab
var body: some View { var body: some View {
NavigationStack(path: $routerPath.path) { NavigationStack(path: $routerPath.path) {
ExploreView() ExploreView()
.withAppRouter() .withAppRouter()
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet) .withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .navigationBar) .toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .navigationBar)
.toolbar { .toolbar {
statusEditorToolbarItem(routerPath: routerPath, ToolbarTab(routerPath: $routerPath)
visibility: preferences.postVisibility)
if UIDevice.current.userInterfaceIdiom != .pad {
ToolbarItem(placement: .navigationBarLeading) {
AppAccountsSelectorView(routerPath: routerPath)
}
}
if UIDevice.current.userInterfaceIdiom == .pad && !preferences.showiPadSecondaryColumn {
SecondaryColumnToolbarItem()
}
} }
} }
.withSafariRouter() .withSafariRouter()
.environmentObject(routerPath) .environment(routerPath)
.onChange(of: $popToRootTab.wrappedValue) { popToRootTab in .onChange(of: client.id) {
if popToRootTab == .explore {
routerPath.path = []
}
}
.onChange(of: client.id) { _ in
routerPath.path = [] routerPath.path = []
} }
.onAppear { .onAppear {

View file

@ -5,17 +5,16 @@ import DesignSystem
import Env import Env
import Models import Models
import Network import Network
import Shimmer
import SwiftUI import SwiftUI
@MainActor
struct MessagesTab: View { struct MessagesTab: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var watcher: StreamWatcher @Environment(StreamWatcher.self) private var watcher
@EnvironmentObject private var client: Client @Environment(Client.self) private var client
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(CurrentAccount.self) private var currentAccount
@EnvironmentObject private var appAccount: AppAccountsManager @Environment(AppAccountsManager.self) private var appAccount
@StateObject private var routerPath = RouterPath() @State private var routerPath = RouterPath()
@Binding var popToRootTab: Tab
var body: some View { var body: some View {
NavigationStack(path: $routerPath.path) { NavigationStack(path: $routerPath.path) {
@ -23,27 +22,18 @@ struct MessagesTab: View {
.withAppRouter() .withAppRouter()
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet) .withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
.toolbar { .toolbar {
if UIDevice.current.userInterfaceIdiom != .pad { ToolbarTab(routerPath: $routerPath)
ToolbarItem(placement: .navigationBarLeading) {
AppAccountsSelectorView(routerPath: routerPath)
}
}
} }
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .navigationBar) .toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .navigationBar)
.id(client.id) .id(client.id)
} }
.onChange(of: $popToRootTab.wrappedValue) { popToRootTab in .onChange(of: client.id) {
if popToRootTab == .messages {
routerPath.path = []
}
}
.onChange(of: client.id) { _ in
routerPath.path = [] routerPath.path = []
} }
.onAppear { .onAppear {
routerPath.client = client routerPath.client = client
} }
.withSafariRouter() .withSafariRouter()
.environmentObject(routerPath) .environment(routerPath)
} }
} }

View file

@ -0,0 +1,24 @@
import AppAccount
import DesignSystem
import Env
import SwiftUI
@MainActor
struct NavigationSheet<Content: View>: View {
@Environment(\.dismiss) private var dismiss
var content: () -> Content
init(@ViewBuilder content: @escaping () -> Content) {
self.content = content
}
var body: some View {
NavigationStack {
content()
.toolbar {
CloseToolbarItem()
}
}
}
}

View file

@ -0,0 +1,46 @@
import AppAccount
import DesignSystem
import Env
import Network
import SwiftUI
@MainActor
struct NavigationTab<Content: View>: View {
@Environment(\.isSecondaryColumn) private var isSecondaryColumn: Bool
@Environment(AppAccountsManager.self) private var appAccount
@Environment(CurrentAccount.self) private var currentAccount
@Environment(UserPreferences.self) private var userPreferences
@Environment(Theme.self) private var theme
@Environment(Client.self) private var client
var content: () -> Content
@State private var routerPath = RouterPath()
init(@ViewBuilder content: @escaping () -> Content) {
self.content = content
}
var body: some View {
NavigationStack(path: $routerPath.path) {
content()
.withEnvironments()
.withAppRouter()
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
.withSafariRouter()
.toolbar {
ToolbarTab(routerPath: $routerPath)
}
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .navigationBar)
.onChange(of: client.id) {
routerPath.path = []
}
.onAppear {
routerPath.client = client
}
.withSafariRouter()
}
.environment(routerPath)
}
}

View file

@ -7,19 +7,21 @@ import Notifications
import SwiftUI import SwiftUI
import Timeline import Timeline
@MainActor
struct NotificationsTab: View { struct NotificationsTab: View {
@Environment(\.isSecondaryColumn) private var isSecondaryColumn: Bool @Environment(\.isSecondaryColumn) private var isSecondaryColumn: Bool
@Environment(\.scenePhase) private var scenePhase @Environment(\.scenePhase) private var scenePhase
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var client: Client @Environment(Client.self) private var client
@EnvironmentObject private var watcher: StreamWatcher @Environment(StreamWatcher.self) private var watcher
@EnvironmentObject private var appAccount: AppAccountsManager @Environment(AppAccountsManager.self) private var appAccount
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(CurrentAccount.self) private var currentAccount
@EnvironmentObject private var userPreferences: UserPreferences @Environment(UserPreferences.self) private var userPreferences
@EnvironmentObject private var pushNotificationsService: PushNotificationsService @Environment(PushNotificationsService.self) private var pushNotificationsService
@StateObject private var routerPath = RouterPath() @State private var routerPath = RouterPath()
@Binding var popToRootTab: Tab
@Binding var selectedTab: AppTab
let lockedType: Models.Notification.NotificationType? let lockedType: Models.Notification.NotificationType?
@ -29,70 +31,62 @@ struct NotificationsTab: View {
.withAppRouter() .withAppRouter()
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet) .withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
.toolbar { .toolbar {
if !isSecondaryColumn { ToolbarItem(placement: .topBarTrailing) {
statusEditorToolbarItem(routerPath: routerPath, Button {
visibility: userPreferences.postVisibility) routerPath.presentedSheet = .accountPushNotficationsSettings
if UIDevice.current.userInterfaceIdiom != .pad { } label: {
ToolbarItem(placement: .navigationBarLeading) { Image(systemName: "bell")
AppAccountsSelectorView(routerPath: routerPath)
}
}
}
if UIDevice.current.userInterfaceIdiom == .pad {
if (!isSecondaryColumn && !userPreferences.showiPadSecondaryColumn) || isSecondaryColumn {
SecondaryColumnToolbarItem()
} }
} }
ToolbarTab(routerPath: $routerPath)
} }
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .navigationBar) .toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .navigationBar)
.id(client.id) .id(client.id)
} }
.onAppear { .onAppear {
routerPath.client = client routerPath.client = client
if isSecondaryColumn { clearNotifications()
clearNotifications()
}
} }
.withSafariRouter() .withSafariRouter()
.environmentObject(routerPath) .environment(routerPath)
.onChange(of: $popToRootTab.wrappedValue) { popToRootTab in .onChange(of: selectedTab) { _, _ in
if popToRootTab == .notifications { clearNotifications()
routerPath.path = []
}
} }
.onChange(of: pushNotificationsService.handledNotification) { notification in .onChange(of: pushNotificationsService.handledNotification) { _, newValue in
if let notification, let type = notification.notification.supportedType { if let newValue, let type = newValue.notification.supportedType {
DispatchQueue.main.asyncAfter(deadline: .now() + 0.3) { DispatchQueue.main.asyncAfter(deadline: .now() + 0.3) {
switch type { switch type {
case .follow, .follow_request: case .follow, .follow_request:
routerPath.navigate(to: .accountDetailWithAccount(account: notification.notification.account)) routerPath.navigate(to: .accountDetailWithAccount(account: newValue.notification.account))
default: default:
if let status = notification.notification.status { if let status = newValue.notification.status {
routerPath.navigate(to: .statusDetailWithStatus(status: status)) routerPath.navigate(to: .statusDetailWithStatus(status: status))
} }
} }
} }
} }
} }
.onChange(of: scenePhase, perform: { scenePhase in .onChange(of: scenePhase) { _, newValue in
switch scenePhase { switch newValue {
case .active: case .active:
clearNotifications() clearNotifications()
default: default:
break break
} }
}) }
.onChange(of: client.id) { _ in .onChange(of: client.id) {
routerPath.path = [] routerPath.path = []
} }
} }
private func clearNotifications() { private func clearNotifications() {
if isSecondaryColumn { if selectedTab == .notifications || isSecondaryColumn {
if let token = appAccount.currentAccount.oauthToken { if let token = appAccount.currentAccount.oauthToken, userPreferences.notificationsCount[token] ?? 0 > 0 {
userPreferences.setNotification(count: 0, token: token) userPreferences.notificationsCount[token] = 0
}
if watcher.unreadNotificationsCount > 0 {
watcher.unreadNotificationsCount = 0
} }
watcher.unreadNotificationsCount = 0
} }
} }
} }

View file

@ -5,16 +5,15 @@ import DesignSystem
import Env import Env
import Models import Models
import Network import Network
import Shimmer
import SwiftUI import SwiftUI
@MainActor
struct ProfileTab: View { struct ProfileTab: View {
@EnvironmentObject private var appAccount: AppAccountsManager @Environment(AppAccountsManager.self) private var appAccount
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var client: Client @Environment(Client.self) private var client
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(CurrentAccount.self) private var currentAccount
@StateObject private var routerPath = RouterPath() @State private var routerPath = RouterPath()
@Binding var popToRootTab: Tab
var body: some View { var body: some View {
NavigationStack(path: $routerPath.path) { NavigationStack(path: $routerPath.path) {
@ -22,25 +21,21 @@ struct ProfileTab: View {
AccountDetailView(account: account) AccountDetailView(account: account)
.withAppRouter() .withAppRouter()
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet) .withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .navigationBar) .toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .navigationBar)
.id(account.id) .id(account.id)
} else { } else {
AccountDetailView(account: .placeholder()) AccountDetailView(account: .placeholder())
.redacted(reason: .placeholder) .redacted(reason: .placeholder)
.allowsHitTesting(false)
} }
} }
.onChange(of: $popToRootTab.wrappedValue) { popToRootTab in .onChange(of: client.id) {
if popToRootTab == .profile {
routerPath.path = []
}
}
.onChange(of: client.id) { _ in
routerPath.path = [] routerPath.path = []
} }
.onAppear { .onAppear {
routerPath.client = client routerPath.client = client
} }
.withSafariRouter() .withSafariRouter()
.environmentObject(routerPath) .environment(routerPath)
} }
} }

View file

@ -1,10 +1,18 @@
import Account
import DesignSystem import DesignSystem
import Env import Env
import Models
import Network
import SwiftUI import SwiftUI
@MainActor
struct AboutView: View { struct AboutView: View {
@EnvironmentObject private var routerPath: RouterPath @Environment(RouterPath.self) private var routerPath
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@Environment(Client.self) private var client
@State private var dimillianAccount: AccountsListRowViewModel?
@State private var iceCubesAccount: AccountsListRowViewModel?
let versionNumber: String let versionNumber: String
@ -19,27 +27,28 @@ struct AboutView: View {
var body: some View { var body: some View {
List { List {
Section { Section {
HStack { #if !targetEnvironment(macCatalyst) && !os(visionOS)
Spacer() HStack {
Image("icon0") Spacer()
.resizable() Image(uiImage: .init(named: "AppIconAlternate0-image") ?? .init())
.frame(width: 50, height: 50) .resizable()
.cornerRadius(4) .frame(width: 50, height: 50)
Image("icon14") .cornerRadius(4)
.resizable() Image(uiImage: .init(named: "AppIconAlternate46-image") ?? .init())
.frame(width: 50, height: 50) .resizable()
.cornerRadius(4) .frame(width: 50, height: 50)
Image("icon17") .cornerRadius(4)
.resizable() Image(uiImage: .init(named: "AppIconAlternate17-image") ?? .init())
.frame(width: 50, height: 50) .resizable()
.cornerRadius(4) .frame(width: 50, height: 50)
Image("icon23") .cornerRadius(4)
.resizable() Image(uiImage: .init(named: "AppIconAlternate23-image") ?? .init())
.frame(width: 50, height: 50) .resizable()
.cornerRadius(4) .frame(width: 50, height: 50)
Spacer() .cornerRadius(4)
} Spacer()
}
#endif
Link(destination: URL(string: "https://github.com/Dimillian/IceCubesApp/blob/main/PRIVACY.MD")!) { Link(destination: URL(string: "https://github.com/Dimillian/IceCubesApp/blob/main/PRIVACY.MD")!) {
Label("settings.support.privacy-policy", systemImage: "lock") Label("settings.support.privacy-policy", systemImage: "lock")
} }
@ -48,9 +57,25 @@ struct AboutView: View {
Label("settings.support.terms-of-use", systemImage: "checkmark.shield") Label("settings.support.terms-of-use", systemImage: "checkmark.shield")
} }
} footer: { } footer: {
Text("\(versionNumber)©2023 Thomas Ricouard") Text("\(versionNumber)© 2024 Thomas Ricouard")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
followAccountsSection
Section("Telemetry") {
Link(destination: .init(string: "https://telemetrydeck.com")!) {
Label("Telemetry by TelemetryDeck", systemImage: "link")
}
Link(destination: .init(string: "https://telemetrydeck.com/privacy/")!) {
Label("Privacy Policy", systemImage: "checkmark.shield")
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
Section { Section {
Text(""" Text("""
@ -75,32 +100,82 @@ struct AboutView: View {
[SwiftUI-Introspect](https://github.com/siteline/SwiftUI-Introspect) [SwiftUI-Introspect](https://github.com/siteline/SwiftUI-Introspect)
[RevenueCat](https://github.com/RevenueCat/purchases-ios) [RevenueCat](https://github.com/RevenueCat/purchases-ios)
[SFSafeSymbols](https://github.com/SFSafeSymbols/SFSafeSymbols) [SFSafeSymbols](https://github.com/SFSafeSymbols/SFSafeSymbols)
""") """)
.multilineTextAlignment(.leading) .multilineTextAlignment(.leading)
.font(.scaledSubheadline) .font(.scaledSubheadline)
.foregroundColor(.gray) .foregroundStyle(.secondary)
} header: { } header: {
Text("settings.about.built-with") Text("settings.about.built-with")
.textCase(nil) .textCase(nil)
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
}
.task {
await fetchAccounts()
} }
.listStyle(.insetGrouped) .listStyle(.insetGrouped)
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.navigationTitle(Text("settings.about.title")) .background(theme.secondaryBackgroundColor)
.navigationBarTitleDisplayMode(.large) #endif
.environment(\.openURL, OpenURLAction { url in .navigationTitle(Text("settings.about.title"))
routerPath.handle(url: url) .navigationBarTitleDisplayMode(.large)
}) .environment(\.openURL, OpenURLAction { url in
routerPath.handle(url: url)
})
}
@ViewBuilder
private var followAccountsSection: some View {
if let iceCubesAccount, let dimillianAccount {
Section {
AccountsListRow(viewModel: iceCubesAccount)
AccountsListRow(viewModel: dimillianAccount)
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
} else {
Section {
ProgressView()
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
}
private func fetchAccounts() async {
await withThrowingTaskGroup(of: Void.self) { group in
group.addTask {
let viewModel = try await fetchAccountViewModel(client, account: "dimillian@mastodon.social")
await MainActor.run {
dimillianAccount = viewModel
}
}
group.addTask {
let viewModel = try await fetchAccountViewModel(client, account: "icecubesapp@mastodon.online")
await MainActor.run {
iceCubesAccount = viewModel
}
}
}
}
private func fetchAccountViewModel(_ client: Client, account: String) async throws -> AccountsListRowViewModel {
let dimillianAccount: Account = try await client.get(endpoint: Accounts.lookup(name: account))
let rel: [Relationship] = try await client.get(endpoint: Accounts.relationships(ids: [dimillianAccount.id]))
return .init(account: dimillianAccount, relationShip: rel.first)
} }
} }
struct AboutView_Previews: PreviewProvider { struct AboutView_Previews: PreviewProvider {
static var previews: some View { static var previews: some View {
AboutView() AboutView()
.environmentObject(Theme.shared) .environment(Theme.shared)
} }
} }

View file

@ -11,16 +11,16 @@ struct AccountSettingsView: View {
@Environment(\.dismiss) private var dismiss @Environment(\.dismiss) private var dismiss
@Environment(\.openURL) private var openURL @Environment(\.openURL) private var openURL
@EnvironmentObject private var pushNotifications: PushNotificationsService @Environment(PushNotificationsService.self) private var pushNotifications
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(CurrentAccount.self) private var currentAccount
@EnvironmentObject private var currentInstance: CurrentInstance @Environment(CurrentInstance.self) private var currentInstance
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var appAccountsManager: AppAccountsManager @Environment(AppAccountsManager.self) private var appAccountsManager
@EnvironmentObject private var client: Client @Environment(Client.self) private var client
@Environment(RouterPath.self) private var routerPath
@State private var isEditingAccount: Bool = false
@State private var isEditingFilters: Bool = false
@State private var cachedPostsCount: Int = 0 @State private var cachedPostsCount: Int = 0
@State private var timelineCache = TimelineCache()
let account: Account let account: Account
let appAccount: AppAccount let appAccount: AppAccount
@ -29,7 +29,7 @@ struct AccountSettingsView: View {
Form { Form {
Section { Section {
Button { Button {
isEditingAccount = true routerPath.presentedSheet = .accountEditInfo
} label: { } label: {
Label("account.action.edit-info", systemImage: "pencil") Label("account.action.edit-info", systemImage: "pencil")
.frame(maxWidth: .infinity, alignment: .leading) .frame(maxWidth: .infinity, alignment: .leading)
@ -39,7 +39,7 @@ struct AccountSettingsView: View {
if currentInstance.isFiltersSupported { if currentInstance.isFiltersSupported {
Button { Button {
isEditingFilters = true routerPath.presentedSheet = .accountFiltersList
} label: { } label: {
Label("account.action.edit-filters", systemImage: "line.3.horizontal.decrease.circle") Label("account.action.edit-filters", systemImage: "line.3.horizontal.decrease.circle")
.frame(maxWidth: .infinity, alignment: .leading) .frame(maxWidth: .infinity, alignment: .leading)
@ -59,8 +59,8 @@ struct AccountSettingsView: View {
Label("settings.account.cached-posts-\(String(cachedPostsCount))", systemImage: "internaldrive") Label("settings.account.cached-posts-\(String(cachedPostsCount))", systemImage: "internaldrive")
Button("settings.account.action.delete-cache", role: .destructive) { Button("settings.account.action.delete-cache", role: .destructive) {
Task { Task {
await TimelineCache.shared.clearCache(for: appAccountsManager.currentClient.id) await timelineCache.clearCache(for: appAccountsManager.currentClient.id)
cachedPostsCount = await TimelineCache.shared.cachedPostsCount(for: appAccountsManager.currentClient.id) cachedPostsCount = await timelineCache.cachedPostsCount(for: appAccountsManager.currentClient.id)
} }
} }
} }
@ -80,41 +80,38 @@ struct AccountSettingsView: View {
if let token = appAccount.oauthToken { if let token = appAccount.oauthToken {
Task { Task {
let client = Client(server: appAccount.server, oauthToken: token) let client = Client(server: appAccount.server, oauthToken: token)
await TimelineCache.shared.clearCache(for: client.id) await timelineCache.clearCache(for: client.id)
if let sub = pushNotifications.subscriptions.first(where: { $0.account.token == token }) { if let sub = pushNotifications.subscriptions.first(where: { $0.account.token == token }) {
await sub.deleteSubscription() await sub.deleteSubscription()
} }
appAccountsManager.delete(account: appAccount) appAccountsManager.delete(account: appAccount)
Telemetry.signal("account.removed")
dismiss() dismiss()
} }
} }
} label: { } label: {
Text("account.action.logout") Label("account.action.logout", systemImage: "trash")
.frame(maxWidth: .infinity) .frame(maxWidth: .infinity)
} }
} }
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
} }
.sheet(isPresented: $isEditingAccount, content: {
EditAccountView()
})
.sheet(isPresented: $isEditingFilters, content: {
FiltersListView()
})
.toolbar { .toolbar {
ToolbarItem(placement: .principal) { ToolbarItem(placement: .principal) {
HStack { HStack {
AvatarView(url: account.avatar, size: .embed) AvatarView(account.avatar, config: .embed)
Text(account.safeDisplayName) Text(account.safeDisplayName)
.font(.headline) .font(.headline)
} }
} }
} }
.task { .task {
cachedPostsCount = await TimelineCache.shared.cachedPostsCount(for: appAccountsManager.currentClient.id) cachedPostsCount = await timelineCache.cachedPostsCount(for: appAccountsManager.currentClient.id)
} }
.navigationTitle(account.safeDisplayName) .navigationTitle(account.safeDisplayName)
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
#endif
} }
} }

View file

@ -1,4 +1,5 @@
import AppAccount import AppAccount
import AuthenticationServices
import Combine import Combine
import DesignSystem import DesignSystem
import Env import Env
@ -6,18 +7,20 @@ import Models
import Network import Network
import NukeUI import NukeUI
import SafariServices import SafariServices
import Shimmer
import SwiftUI import SwiftUI
@MainActor
struct AddAccountView: View { struct AddAccountView: View {
@Environment(\.webAuthenticationSession) private var webAuthenticationSession
@Environment(\.dismiss) private var dismiss @Environment(\.dismiss) private var dismiss
@Environment(\.scenePhase) private var scenePhase @Environment(\.scenePhase) private var scenePhase
@Environment(\.openURL) private var openURL
@EnvironmentObject private var appAccountsManager: AppAccountsManager @Environment(AppAccountsManager.self) private var appAccountsManager
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(CurrentAccount.self) private var currentAccount
@EnvironmentObject private var currentInstance: CurrentInstance @Environment(CurrentInstance.self) private var currentInstance
@EnvironmentObject private var pushNotifications: PushNotificationsService @Environment(PushNotificationsService.self) private var pushNotifications
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@State private var instanceName: String = "" @State private var instanceName: String = ""
@State private var instance: Instance? @State private var instance: Instance?
@ -25,7 +28,9 @@ struct AddAccountView: View {
@State private var signInClient: Client? @State private var signInClient: Client?
@State private var instances: [InstanceSocial] = [] @State private var instances: [InstanceSocial] = []
@State private var instanceFetchError: LocalizedStringKey? @State private var instanceFetchError: LocalizedStringKey?
@State private var oauthURL: URL? @State private var instanceSocialClient = InstanceSocialClient()
@State private var searchingTask = Task<Void, Never> {}
@State private var getInstanceDetailTask = Task<Void, Never> {}
private let instanceNamePublisher = PassthroughSubject<String, Never>() private let instanceNamePublisher = PassthroughSubject<String, Never>()
@ -43,16 +48,25 @@ struct AddAccountView: View {
@FocusState private var isInstanceURLFieldFocused: Bool @FocusState private var isInstanceURLFieldFocused: Bool
private func cleanServerStr(_ server: String) -> String {
server.replacingOccurrences(of: " ", with: "")
}
var body: some View { var body: some View {
NavigationStack { NavigationStack {
Form { Form {
TextField("instance.url", text: $instanceName) TextField("instance.url", text: $instanceName)
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
.keyboardType(.URL) .keyboardType(.URL)
.textContentType(.URL) .textContentType(.URL)
.textInputAutocapitalization(.never) .textInputAutocapitalization(.never)
.autocorrectionDisabled() .autocorrectionDisabled()
.focused($isInstanceURLFieldFocused) .focused($isInstanceURLFieldFocused)
.onChange(of: instanceName) { _, _ in
instanceName = cleanServerStr(instanceName)
}
if let instanceFetchError { if let instanceFetchError {
Text(instanceFetchError) Text(instanceFetchError)
} }
@ -68,78 +82,78 @@ struct AddAccountView: View {
.formStyle(.grouped) .formStyle(.grouped)
.navigationTitle("account.add.navigation-title") .navigationTitle("account.add.navigation-title")
.navigationBarTitleDisplayMode(.inline) .navigationBarTitleDisplayMode(.inline)
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.scrollDismissesKeyboard(.immediately) .background(theme.secondaryBackgroundColor)
.toolbar { .scrollDismissesKeyboard(.immediately)
if !appAccountsManager.availableAccounts.isEmpty { #endif
ToolbarItem(placement: .navigationBarLeading) { .toolbar {
Button("action.cancel", action: { dismiss() }) CancelToolbarItem()
}
} }
} .onAppear {
.onAppear { isInstanceURLFieldFocused = true
isInstanceURLFieldFocused = true let instanceName = instanceName
let client = InstanceSocialClient() Task {
Task { let instances = await instanceSocialClient.fetchInstances(keyword: instanceName)
let instances = await client.fetchInstances() withAnimation {
withAnimation { self.instances = instances
self.instances = instances }
} }
isSigninIn = false
} }
isSigninIn = false .onChange(of: instanceName) {
} searchingTask.cancel()
.onChange(of: instanceName) { newValue in let instanceName = instanceName
instanceNamePublisher.send(newValue) let instanceSocialClient = instanceSocialClient
} searchingTask = Task {
.onReceive(instanceNamePublisher.debounce(for: .milliseconds(500), scheduler: DispatchQueue.main)) { _ in try? await Task.sleep(for: .seconds(0.1))
// let newValue = newValue guard !Task.isCancelled else { return }
// .replacingOccurrences(of: "http://", with: "")
// .replacingOccurrences(of: "https://", with: "") let instances = await instanceSocialClient.fetchInstances(keyword: instanceName)
let client = Client(server: sanitizedName) withAnimation {
Task { self.instances = instances
do { }
// bare bones preflight for domain validity }
if client.server.contains(".") && client.server.last != "." {
let instance: Instance = try await client.get(endpoint: Instances.instance) getInstanceDetailTask.cancel()
withAnimation { getInstanceDetailTask = Task {
self.instance = instance try? await Task.sleep(for: .seconds(0.1))
self.instanceName = sanitizedName // clean up the text box, principally to chop off the username if present so it's clear that you might not wind up siging in as the thing in the box guard !Task.isCancelled else { return }
do {
// bare bones preflight for domain validity
let instanceDetailClient = Client(server: sanitizedName)
if
instanceDetailClient.server.contains("."),
instanceDetailClient.server.last != "."
{
let instance: Instance = try await instanceDetailClient.get(endpoint: Instances.instance)
withAnimation {
self.instance = instance
self.instanceName = sanitizedName // clean up the text box, principally to chop off the username if present so it's clear that you might not wind up siging in as the thing in the box
}
instanceFetchError = nil
} else {
instance = nil
instanceFetchError = nil
} }
instanceFetchError = nil } catch _ as ServerError {
} else { instance = nil
instanceFetchError = "account.add.error.instance-not-supported"
} catch {
instance = nil instance = nil
instanceFetchError = nil instanceFetchError = nil
} }
} catch _ as DecodingError {
instance = nil
instanceFetchError = "account.add.error.instance-not-supported"
} catch {
instance = nil
} }
} }
} .onChange(of: scenePhase) { _, newValue in
.onChange(of: scenePhase, perform: { scenePhase in switch newValue {
switch scenePhase { case .active:
case .active: isSigninIn = false
isSigninIn = false default:
default: break
break }
} }
})
.onOpenURL(perform: { url in
Task {
await continueSignIn(url: url)
}
})
.onChange(of: oauthURL, perform: { newValue in
if newValue == nil {
isSigninIn = false
}
})
.sheet(item: $oauthURL, content: { url in
SafariView(url: url)
})
} }
} }
@ -168,7 +182,9 @@ struct AddAccountView: View {
} }
.buttonStyle(.borderedProminent) .buttonStyle(.borderedProminent)
} }
#if !os(visionOS)
.listRowBackground(theme.tintColor) .listRowBackground(theme.tintColor)
#endif
} }
private var instancesListView: some View { private var instancesListView: some View {
@ -176,30 +192,65 @@ struct AddAccountView: View {
if instances.isEmpty { if instances.isEmpty {
placeholderRow placeholderRow
} else { } else {
ForEach(sanitizedName.isEmpty ? instances : instances.filter { $0.name.contains(sanitizedName.lowercased()) }) { instance in ForEach(instances) { instance in
Button { Button {
self.instanceName = instance.name instanceName = instance.name
} label: { } label: {
VStack(alignment: .leading, spacing: 4) { VStack(alignment: .leading, spacing: 4) {
Text(instance.name) LazyImage(url: instance.thumbnail) { state in
.font(.scaledHeadline) if let image = state.image {
.foregroundColor(.primary) image
Text(instance.info?.shortDescription ?? "") .resizable()
.font(.scaledBody) .scaledToFill()
.foregroundColor(.gray) } else {
(Text("instance.list.users-\(instance.users)") Rectangle().fill(theme.tintColor.opacity(0.1))
+ Text("") }
+ Text("instance.list.posts-\(instance.statuses)")) }
.font(.scaledFootnote) .frame(height: 100)
.foregroundColor(.gray) .frame(maxWidth: .infinity)
.clipped()
VStack(alignment: .leading) {
HStack {
Text(instance.name)
.font(.scaledHeadline)
.foregroundColor(.primary)
Spacer()
(Text("instance.list.users-\(formatAsNumber(instance.users))")
+ Text("")
+ Text("instance.list.posts-\(formatAsNumber(instance.statuses))"))
.foregroundStyle(theme.tintColor)
}
.padding(.bottom, 5)
Text(instance.info?.shortDescription?.trimmingCharacters(in: .whitespacesAndNewlines) ?? "")
.foregroundStyle(Color.secondary)
.lineLimit(10)
}
.font(.scaledFootnote)
.padding(10)
} }
} }
.listRowBackground(theme.primaryBackgroundColor) #if !os(visionOS)
.background(theme.primaryBackgroundColor)
.listRowBackground(Color.clear)
.listRowInsets(EdgeInsets(top: 10, leading: 0, bottom: 10, trailing: 0))
.listRowSeparator(.hidden)
.clipShape(RoundedRectangle(cornerRadius: 4))
#endif
} }
} }
} }
} }
private func formatAsNumber(_ string: String) -> String {
(Int(string) ?? 0)
.formatted(
.number
.notation(.compactName)
.locale(.current)
)
}
private var placeholderRow: some View { private var placeholderRow: some View {
VStack(alignment: .leading, spacing: 4) { VStack(alignment: .leading, spacing: 4) {
Text("placeholder.loading.short") Text("placeholder.loading.short")
@ -207,25 +258,26 @@ struct AddAccountView: View {
.foregroundColor(.primary) .foregroundColor(.primary)
Text("placeholder.loading.long") Text("placeholder.loading.long")
.font(.scaledBody) .font(.scaledBody)
.foregroundColor(.gray) .foregroundStyle(.secondary)
Text("placeholder.loading.short") Text("placeholder.loading.short")
.font(.scaledFootnote) .font(.scaledFootnote)
.foregroundColor(.gray) .foregroundStyle(.secondary)
} }
.redacted(reason: .placeholder) .redacted(reason: .placeholder)
.shimmering() .allowsHitTesting(false)
.listRowBackground(theme.primaryBackgroundColor) #if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private func signIn() async { private func signIn() async {
do { signInClient = .init(server: sanitizedName)
signInClient = .init(server: sanitizedName) if let oauthURL = try? await signInClient?.oauthURL(),
if let oauthURL = try await signInClient?.oauthURL() { let url = try? await webAuthenticationSession.authenticate(using: oauthURL,
self.oauthURL = oauthURL callbackURLScheme: AppInfo.scheme.replacingOccurrences(of: "://", with: ""))
} else { {
isSigninIn = false await continueSignIn(url: url)
} } else {
} catch {
isSigninIn = false isSigninIn = false
} }
} }
@ -236,10 +288,10 @@ struct AddAccountView: View {
return return
} }
do { do {
oauthURL = nil
let oauthToken = try await client.continueOauthFlow(url: url) let oauthToken = try await client.continueOauthFlow(url: url)
let client = Client(server: client.server, oauthToken: oauthToken) let client = Client(server: client.server, oauthToken: oauthToken)
let account: Account = try await client.get(endpoint: Accounts.verifyCredentials) let account: Account = try await client.get(endpoint: Accounts.verifyCredentials)
Telemetry.signal("account.added")
appAccountsManager.add(account: AppAccount(server: client.server, appAccountsManager.add(account: AppAccount(server: client.server,
accountName: "\(account.acct)@\(client.server)", accountName: "\(account.acct)@\(client.server)",
oauthToken: oauthToken)) oauthToken: oauthToken))
@ -250,18 +302,7 @@ struct AddAccountView: View {
isSigninIn = false isSigninIn = false
dismiss() dismiss()
} catch { } catch {
oauthURL = nil
isSigninIn = false isSigninIn = false
} }
} }
} }
struct SafariView: UIViewControllerRepresentable {
let url: URL
func makeUIViewController(context _: UIViewControllerRepresentableContext<SafariView>) -> SFSafariViewController {
SFSafariViewController(url: url)
}
func updateUIViewController(_: SFSafariViewController, context _: UIViewControllerRepresentableContext<SafariView>) {}
}

View file

@ -5,28 +5,33 @@ import Models
import Network import Network
import NukeUI import NukeUI
import SwiftUI import SwiftUI
import Timeline
import UserNotifications import UserNotifications
@MainActor
struct ContentSettingsView: View { struct ContentSettingsView: View {
@EnvironmentObject private var userPreferences: UserPreferences @Environment(UserPreferences.self) private var userPreferences
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@State private var contentFilter = TimelineContentFilter.shared
var body: some View { var body: some View {
@Bindable var userPreferences = userPreferences
Form { Form {
Section("settings.content.boosts") {
Toggle(isOn: $userPreferences.suppressDupeReblogs) {
Text("settings.content.hide-repeated-boosts")
}
}.listRowBackground(theme.primaryBackgroundColor)
Section("settings.content.media") { Section("settings.content.media") {
Toggle(isOn: $userPreferences.autoPlayVideo) { Toggle(isOn: $userPreferences.autoPlayVideo) {
Text("settings.other.autoplay-video") Text("settings.other.autoplay-video")
} }
Toggle(isOn: $userPreferences.muteVideo) {
Text("settings.other.mute-video")
}
Toggle(isOn: $userPreferences.showAltTextForMedia) { Toggle(isOn: $userPreferences.showAltTextForMedia) {
Text("settings.content.media.show.alt") Text("settings.content.media.show.alt")
} }
}.listRowBackground(theme.primaryBackgroundColor) }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
Section("settings.content.sharing") { Section("settings.content.sharing") {
Picker("settings.content.sharing.share-button-behavior", selection: $userPreferences.shareButtonBehavior) { Picker("settings.content.sharing.share-button-behavior", selection: $userPreferences.shareButtonBehavior) {
@ -36,20 +41,25 @@ struct ContentSettingsView: View {
} }
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
Section("settings.content.instance-settings") { Section("settings.content.instance-settings") {
Toggle(isOn: $userPreferences.useInstanceContentSettings) { Toggle(isOn: $userPreferences.useInstanceContentSettings) {
Text("settings.content.use-instance-settings") Text("settings.content.use-instance-settings")
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
.onChange(of: userPreferences.useInstanceContentSettings) { newVal in #endif
.onChange(of: userPreferences.useInstanceContentSettings) { _, newVal in
if newVal { if newVal {
userPreferences.appAutoExpandSpoilers = userPreferences.autoExpandSpoilers userPreferences.appAutoExpandSpoilers = userPreferences.autoExpandSpoilers
userPreferences.appAutoExpandMedia = userPreferences.autoExpandMedia userPreferences.appAutoExpandMedia = userPreferences.autoExpandMedia
userPreferences.appDefaultPostsSensitive = userPreferences.postIsSensitive userPreferences.appDefaultPostsSensitive = userPreferences.postIsSensitive
userPreferences.appDefaultPostVisibility = userPreferences.postVisibility userPreferences.appDefaultPostVisibility = userPreferences.postVisibility
userPreferences.appRequireAltText = userPreferences.appRequireAltText
} }
} }
@ -74,7 +84,9 @@ struct ContentSettingsView: View {
} footer: { } footer: {
Text("settings.content.collapse-long-posts-hint") Text("settings.content.collapse-long-posts-hint")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
Section("settings.content.posting") { Section("settings.content.posting") {
Picker("settings.content.default-visibility", selection: $userPreferences.appDefaultPostVisibility) { Picker("settings.content.default-visibility", selection: $userPreferences.appDefaultPostVisibility) {
@ -83,29 +95,55 @@ struct ContentSettingsView: View {
} }
} }
.disabled(userPreferences.useInstanceContentSettings) .disabled(userPreferences.useInstanceContentSettings)
Picker("settings.content.default-reply-visibility", selection: $userPreferences.appDefaultReplyVisibility) { Picker("settings.content.default-reply-visibility", selection: $userPreferences.appDefaultReplyVisibility) {
ForEach(Visibility.allCases, id: \.rawValue) { vis in ForEach(Visibility.allCases, id: \.rawValue) { vis in
if UserPreferences.getIntOfVisibility(vis) <= if UserPreferences.getIntOfVisibility(vis) <=
UserPreferences.getIntOfVisibility(userPreferences.postVisibility) { UserPreferences.getIntOfVisibility(userPreferences.postVisibility)
{
Text(vis.title).tag(vis) Text(vis.title).tag(vis)
} }
} }
} }
.onChange(of: userPreferences.postVisibility) { newValue in .onChange(of: userPreferences.postVisibility) {
userPreferences.conformReplyVisibilityConstraints() userPreferences.conformReplyVisibilityConstraints()
} }
Toggle(isOn: $userPreferences.appDefaultPostsSensitive) { Toggle(isOn: $userPreferences.appDefaultPostsSensitive) {
Text("settings.content.default-sensitive") Text("settings.content.default-sensitive")
} }
.disabled(userPreferences.useInstanceContentSettings) .disabled(userPreferences.useInstanceContentSettings)
}
Toggle(isOn: $userPreferences.appRequireAltText) {
Text("settings.content.require-alt-text")
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
Section("timeline.content-filter.title") {
Toggle(isOn: $contentFilter.showBoosts) {
Label("timeline.filter.show-boosts", image: "Rocket")
}
Toggle(isOn: $contentFilter.showReplies) {
Label("timeline.filter.show-replies", systemImage: "bubble.left.and.bubble.right")
}
Toggle(isOn: $contentFilter.showThreads) {
Label("timeline.filter.show-threads", systemImage: "bubble.left.and.text.bubble.right")
}
Toggle(isOn: $contentFilter.showQuotePosts) {
Label("timeline.filter.show-quote", systemImage: "quote.bubble")
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
} }
.navigationTitle("settings.content.navigation-title") .navigationTitle("settings.content.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
#endif
} }
} }

View file

@ -3,57 +3,29 @@ import DesignSystem
import Env import Env
import Models import Models
import Network import Network
import Status import Observation
import StatusKit
import SwiftUI import SwiftUI
class DisplaySettingsLocalValues: ObservableObject { @MainActor
@Published var tintColor = Theme.shared.tintColor @Observable class DisplaySettingsLocalValues {
@Published var primaryBackgroundColor = Theme.shared.primaryBackgroundColor var tintColor = Theme.shared.tintColor
@Published var secondaryBackgroundColor = Theme.shared.secondaryBackgroundColor var primaryBackgroundColor = Theme.shared.primaryBackgroundColor
@Published var labelColor = Theme.shared.labelColor var secondaryBackgroundColor = Theme.shared.secondaryBackgroundColor
@Published var lineSpacing = Theme.shared.lineSpacing var labelColor = Theme.shared.labelColor
@Published var fontSizeScale = Theme.shared.fontSizeScale var lineSpacing = Theme.shared.lineSpacing
var fontSizeScale = Theme.shared.fontSizeScale
private let debouncesDelay: DispatchQueue.SchedulerTimeType.Stride = .seconds(0.5)
private var subscriptions = Set<AnyCancellable>()
init() {
$tintColor
.debounce(for: debouncesDelay, scheduler: DispatchQueue.main)
.sink(receiveValue: { newColor in Theme.shared.tintColor = newColor })
.store(in: &subscriptions)
$primaryBackgroundColor
.debounce(for: debouncesDelay, scheduler: DispatchQueue.main)
.sink(receiveValue: { newColor in Theme.shared.primaryBackgroundColor = newColor })
.store(in: &subscriptions)
$secondaryBackgroundColor
.debounce(for: debouncesDelay, scheduler: DispatchQueue.main)
.sink(receiveValue: { newColor in Theme.shared.secondaryBackgroundColor = newColor })
.store(in: &subscriptions)
$labelColor
.debounce(for: debouncesDelay, scheduler: DispatchQueue.main)
.sink(receiveValue: { newColor in Theme.shared.labelColor = newColor })
.store(in: &subscriptions)
$lineSpacing
.debounce(for: debouncesDelay, scheduler: DispatchQueue.main)
.sink(receiveValue: { newSpacing in Theme.shared.lineSpacing = newSpacing })
.store(in: &subscriptions)
$fontSizeScale
.debounce(for: debouncesDelay, scheduler: DispatchQueue.main)
.sink(receiveValue: { newScale in Theme.shared.fontSizeScale = newScale })
.store(in: &subscriptions)
}
} }
@MainActor
struct DisplaySettingsView: View { struct DisplaySettingsView: View {
typealias FontState = Theme.FontState typealias FontState = Theme.FontState
@Environment(\.colorScheme) private var colorScheme @Environment(\.colorScheme) private var colorScheme
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var userPreferences: UserPreferences @Environment(UserPreferences.self) private var userPreferences
@StateObject private var localValues = DisplaySettingsLocalValues() @State private var localValues = DisplaySettingsLocalValues()
@State private var isFontSelectorPresented = false @State private var isFontSelectorPresented = false
@ -64,26 +36,56 @@ struct DisplaySettingsView: View {
var body: some View { var body: some View {
ZStack(alignment: .top) { ZStack(alignment: .top) {
Form { Form {
StatusRowView(viewModel: { previewStatusViewModel }) #if !os(visionOS)
.allowsHitTesting(false) StatusRowExternalView(viewModel: previewStatusViewModel)
.opacity(0) .allowsHitTesting(false)
.hidden() .opacity(0)
themeSection .hidden()
themeSection
#endif
fontSection fontSection
layoutSection layoutSection
platformsSection platformsSection
resetSection resetSection
} }
.navigationTitle("settings.display.navigation-title") .navigationTitle("settings.display.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
examplePost .background(theme.secondaryBackgroundColor)
#endif
.task(id: localValues.tintColor) {
do { try await Task.sleep(for: .microseconds(500)) } catch {}
theme.tintColor = localValues.tintColor
}
.task(id: localValues.primaryBackgroundColor) {
do { try await Task.sleep(for: .microseconds(500)) } catch {}
theme.primaryBackgroundColor = localValues.primaryBackgroundColor
}
.task(id: localValues.secondaryBackgroundColor) {
do { try await Task.sleep(for: .microseconds(500)) } catch {}
theme.secondaryBackgroundColor = localValues.secondaryBackgroundColor
}
.task(id: localValues.labelColor) {
do { try await Task.sleep(for: .microseconds(500)) } catch {}
theme.labelColor = localValues.labelColor
}
.task(id: localValues.lineSpacing) {
do { try await Task.sleep(for: .microseconds(500)) } catch {}
theme.lineSpacing = localValues.lineSpacing
}
.task(id: localValues.fontSizeScale) {
do { try await Task.sleep(for: .microseconds(500)) } catch {}
theme.fontSizeScale = localValues.fontSizeScale
}
#if !os(visionOS)
examplePost
#endif
} }
} }
private var examplePost: some View { private var examplePost: some View {
VStack(spacing: 0) { VStack(spacing: 0) {
StatusRowView(viewModel: { previewStatusViewModel }) StatusRowExternalView(viewModel: previewStatusViewModel)
.allowsHitTesting(false) .allowsHitTesting(false)
.padding(.layoutPadding) .padding(.layoutPadding)
.background(theme.primaryBackgroundColor) .background(theme.primaryBackgroundColor)
@ -99,7 +101,9 @@ struct DisplaySettingsView: View {
} }
} }
@ViewBuilder
private var themeSection: some View { private var themeSection: some View {
@Bindable var theme = theme
Section { Section {
Toggle("settings.display.theme.systemColor", isOn: $theme.followSystemColorScheme) Toggle("settings.display.theme.systemColor", isOn: $theme.followSystemColorScheme)
themeSelectorButton themeSelectorButton
@ -111,7 +115,7 @@ struct DisplaySettingsView: View {
} }
.disabled(theme.followSystemColorScheme) .disabled(theme.followSystemColorScheme)
.opacity(theme.followSystemColorScheme ? 0.5 : 1.0) .opacity(theme.followSystemColorScheme ? 0.5 : 1.0)
.onChange(of: theme.selectedSet) { _ in .onChange(of: theme.selectedSet) {
localValues.tintColor = theme.tintColor localValues.tintColor = theme.tintColor
localValues.primaryBackgroundColor = theme.primaryBackgroundColor localValues.primaryBackgroundColor = theme.primaryBackgroundColor
localValues.secondaryBackgroundColor = theme.secondaryBackgroundColor localValues.secondaryBackgroundColor = theme.secondaryBackgroundColor
@ -124,7 +128,9 @@ struct DisplaySettingsView: View {
Text("settings.display.section.theme.footer") Text("settings.display.section.theme.footer")
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private var fontSection: some View { private var fontSection: some View {
@ -176,10 +182,15 @@ struct DisplaySettingsView: View {
d[.leading] d[.leading]
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
@ViewBuilder
private var layoutSection: some View { private var layoutSection: some View {
@Bindable var theme = theme
@Bindable var userPreferences = userPreferences
Section("settings.display.section.display") { Section("settings.display.section.display") {
Picker("settings.display.avatar.position", selection: $theme.avatarPosition) { Picker("settings.display.avatar.position", selection: $theme.avatarPosition) {
ForEach(Theme.AvatarPosition.allCases, id: \.rawValue) { position in ForEach(Theme.AvatarPosition.allCases, id: \.rawValue) { position in
@ -197,53 +208,67 @@ struct DisplaySettingsView: View {
Text(buttonStyle.description).tag(buttonStyle) Text(buttonStyle.description).tag(buttonStyle)
} }
} }
Picker("settings.display.status.action-secondary", selection: $theme.statusActionSecondary) {
ForEach(Theme.StatusActionSecondary.allCases, id: \.rawValue) { action in
Text(action.description).tag(action)
}
}
Picker("settings.display.status.media-style", selection: $theme.statusDisplayStyle) { Picker("settings.display.status.media-style", selection: $theme.statusDisplayStyle) {
ForEach(Theme.StatusDisplayStyle.allCases, id: \.rawValue) { buttonStyle in ForEach(Theme.StatusDisplayStyle.allCases, id: \.rawValue) { buttonStyle in
Text(buttonStyle.description).tag(buttonStyle) Text(buttonStyle.description).tag(buttonStyle)
} }
} }
Toggle("settings.display.translate-button", isOn: $userPreferences.showTranslateButton) Toggle("settings.display.translate-button", isOn: $userPreferences.showTranslateButton)
Toggle("settings.display.pending-at-bottom", isOn: $userPreferences.pendingShownAtBottom)
Toggle("settings.display.pending-left", isOn: $userPreferences.pendingShownLeft)
Toggle("settings.display.show-reply-indentation", isOn: $userPreferences.showReplyIndentation)
if userPreferences.showReplyIndentation {
VStack {
Slider(value: .init(get: {
Double(userPreferences.maxReplyIndentation)
}, set: { newVal in
userPreferences.maxReplyIndentation = UInt(newVal)
}), in: 1 ... 20, step: 1)
Text("settings.display.max-reply-indentation-\(String(userPreferences.maxReplyIndentation))")
.font(.scaledBody)
}
.alignmentGuide(.listRowSeparatorLeading) { d in
d[.leading]
}
}
Toggle("settings.display.show-account-popover", isOn: $userPreferences.showAccountPopover)
Toggle("Show Content Gradient", isOn: $theme.showContentGradient)
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
@ViewBuilder @ViewBuilder
private var platformsSection: some View { private var platformsSection: some View {
if UIDevice.current.userInterfaceIdiom == .phone { @Bindable var userPreferences = userPreferences
Section("iPhone") {
Toggle("settings.display.show-tab-label", isOn: $userPreferences.showiPhoneTabLabel)
}
.listRowBackground(theme.primaryBackgroundColor)
}
if UIDevice.current.userInterfaceIdiom == .pad { if UIDevice.current.userInterfaceIdiom == .pad {
Section("iPad") { Section("settings.display.section.platform") {
Toggle("settings.display.show-ipad-column", isOn: $userPreferences.showiPadSecondaryColumn) Toggle("settings.display.show-ipad-column", isOn: $userPreferences.showiPadSecondaryColumn)
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
} }
private var resetSection: some View { private var resetSection: some View {
Section { Section {
Button { Button {
theme.followSystemColorScheme = true theme.restoreDefault()
theme.selectedSet = colorScheme == .dark ? .iceCubeDark : .iceCubeLight
theme.avatarShape = .rounded
theme.avatarPosition = .top
theme.statusActionsDisplay = .full
localValues.tintColor = theme.tintColor
localValues.primaryBackgroundColor = theme.primaryBackgroundColor
localValues.secondaryBackgroundColor = theme.secondaryBackgroundColor
localValues.labelColor = theme.labelColor
} label: { } label: {
Text("settings.display.restore") Text("settings.display.restore")
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private var themeSelectorButton: some View { private var themeSelectorButton: some View {

View file

@ -1,24 +1,30 @@
import DesignSystem import DesignSystem
import Env import Env
import Models import Models
import Status import StatusKit
import SwiftUI import SwiftUI
@MainActor
struct HapticSettingsView: View { struct HapticSettingsView: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var userPreferences: UserPreferences @Environment(UserPreferences.self) private var userPreferences
var body: some View { var body: some View {
@Bindable var userPreferences = userPreferences
Form { Form {
Section { Section {
Toggle("settings.haptic.timeline", isOn: $userPreferences.hapticTimelineEnabled) Toggle("settings.haptic.timeline", isOn: $userPreferences.hapticTimelineEnabled)
Toggle("settings.haptic.tab-selection", isOn: $userPreferences.hapticTabSelectionEnabled) Toggle("settings.haptic.tab-selection", isOn: $userPreferences.hapticTabSelectionEnabled)
Toggle("settings.haptic.buttons", isOn: $userPreferences.hapticButtonPressEnabled) Toggle("settings.haptic.buttons", isOn: $userPreferences.hapticButtonPressEnabled)
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
.navigationTitle("settings.haptic.navigation-title") .navigationTitle("settings.haptic.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
#endif
} }
} }

View file

@ -1,6 +1,7 @@
import DesignSystem import DesignSystem
import SwiftUI import SwiftUI
@MainActor
struct IconSelectorView: View { struct IconSelectorView: View {
enum Icon: Int, CaseIterable, Identifiable { enum Icon: Int, CaseIterable, Identifiable {
var id: String { var id: String {
@ -8,7 +9,7 @@ struct IconSelectorView: View {
} }
init(string: String) { init(string: String) {
if string == Icon.primary.appIconName { if string == "AppIcon" {
self = .primary self = .primary
} else { } else {
self = .init(rawValue: Int(String(string.replacing("AppIconAlternate", with: "")))!)! self = .init(rawValue: Int(String(string.replacing("AppIconAlternate", with: "")))!)!
@ -16,30 +17,24 @@ struct IconSelectorView: View {
} }
case primary = 0 case primary = 0
case alt1, alt2, alt3, alt4, alt5, alt6, alt7, alt8 case alt1, alt2, alt3, alt4, alt5, alt6, alt7, alt8, alt9, alt10, alt11, alt12, alt13, alt14, alt15
case alt9, alt10, alt11, alt12, alt13, alt14 case alt16, alt17, alt18, alt19, alt20, alt21
case alt15, alt16, alt17, alt18, alt19, alt20, alt21 case alt22, alt23, alt24, alt25, alt26
case alt22, alt23, alt24, alt25 case alt27, alt28, alt29
case alt26, alt27, alt28 case alt30, alt31, alt32, alt33, alt34, alt35, alt36
case alt29, alt30, alt31, alt32
case alt33
case alt34, alt35
case alt36
case alt37 case alt37
case alt38, alt39 case alt38
case alt40 case alt39, alt40, alt41, alt42, alt43
case alt44, alt45
case alt46, alt47, alt48
case alt49
var appIconName: String { var appIconName: String {
switch self { return "AppIconAlternate\(rawValue)"
case .primary:
return "AppIcon"
default:
return "AppIconAlternate\(rawValue)"
}
} }
var iconName: String { var previewImageName: String {
"icon\(rawValue)" return "AppIconAlternate\(rawValue)-image"
} }
} }
@ -49,20 +44,24 @@ struct IconSelectorView: View {
let icons: [Icon] let icons: [Icon]
static let items = [ static let items = [
IconSelector(title: "settings.app.icon.official".localized, icons: [.primary, .alt1, .alt2, .alt3, .alt4, .alt5, .alt6, .alt7, .alt8, IconSelector(title: "settings.app.icon.official".localized, icons: [
.alt9, .alt10, .alt11, .alt12, .alt13, .alt14, .primary, .alt46, .alt1, .alt2, .alt3, .alt4,
.alt15, .alt16, .alt17, .alt18, .alt19, .alt25]), .alt5, .alt6, .alt7, .alt8,
IconSelector(title: "\("settings.app.icon.designed-by".localized) Albert Kinng", icons: [.alt20, .alt21, .alt22, .alt23, .alt24]), .alt9, .alt10, .alt11, .alt12, .alt13, .alt14, .alt15,
IconSelector(title: "\("settings.app.icon.designed-by".localized) Dan van Moll", icons: [.alt26, .alt27, .alt28]), .alt16, .alt17, .alt18, .alt19, .alt20, .alt21]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) Chanhwi Joo (GitHub @te6-in)", icons: [.alt29, .alt34, .alt31, .alt35, .alt30, .alt32, .alt40]), IconSelector(title: "\("settings.app.icon.designed-by".localized) Albert Kinng", icons: [.alt22, .alt23, .alt24, .alt25, .alt26]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) W. Kovács Ágnes (@wildgica)", icons: [.alt33]), IconSelector(title: "\("settings.app.icon.designed-by".localized) Dan van Moll", icons: [.alt27, .alt28, .alt29]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) Duncan Horne", icons: [.alt36]), IconSelector(title: "\("settings.app.icon.designed-by".localized) Chanhwi Joo (GitHub @te6-in)", icons: [.alt30, .alt31, .alt32, .alt33, .alt34, .alt35, .alt36]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) BeAware@social.beaware.live", icons: [.alt37]), IconSelector(title: "\("settings.app.icon.designed-by".localized) W. Kovács Ágnes (@wildgica)", icons: [.alt37]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) Simone Margio", icons: [.alt38, .alt39]), IconSelector(title: "\("settings.app.icon.designed-by".localized) Duncan Horne", icons: [.alt38]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) BeAware@social.beaware.live", icons: [.alt39, .alt40, .alt41, .alt42, .alt43]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) Simone Margio", icons: [.alt44, .alt45]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) Peter Broqvist (@PKB)", icons: [.alt47, .alt48]),
IconSelector(title: "\("settings.app.icon.designed-by".localized) Oz Tsori (@oztsori)", icons: [.alt49]),
] ]
} }
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@State private var currentIcon = UIApplication.shared.alternateIconName ?? Icon.primary.appIconName @State private var currentIcon = UIApplication.shared.alternateIconName ?? Icon.primary.appIconName
private let columns = [GridItem(.adaptive(minimum: 125, maximum: 1024))] private let columns = [GridItem(.adaptive(minimum: 125, maximum: 1024))]
@ -82,7 +81,9 @@ struct IconSelectorView: View {
.padding(6) .padding(6)
.navigationTitle("settings.app.icon.navigation-title") .navigationTitle("settings.app.icon.navigation-title")
} }
#if !os(visionOS)
.background(theme.primaryBackgroundColor) .background(theme.primaryBackgroundColor)
#endif
} }
private func makeIconGridView(icons: [Icon]) -> some View { private func makeIconGridView(icons: [Icon]) -> some View {
@ -100,7 +101,7 @@ struct IconSelectorView: View {
} }
} label: { } label: {
ZStack(alignment: .bottomTrailing) { ZStack(alignment: .bottomTrailing) {
Image(uiImage: .init(named: icon.iconName) ?? .init()) Image(icon.previewImageName)
.resizable() .resizable()
.aspectRatio(contentMode: .fit) .aspectRatio(contentMode: .fit)
.frame(minHeight: 125, maxHeight: 1024) .frame(minHeight: 125, maxHeight: 1024)
@ -113,6 +114,7 @@ struct IconSelectorView: View {
} }
} }
} }
.buttonStyle(.plain)
} }
} }
} }
@ -120,6 +122,6 @@ struct IconSelectorView: View {
extension String { extension String {
var localized: String { var localized: String {
return NSLocalizedString(self, comment: "") NSLocalizedString(self, comment: "")
} }
} }

View file

@ -4,7 +4,7 @@ import NukeUI
import SwiftUI import SwiftUI
struct InstanceInfoView: View { struct InstanceInfoView: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
let instance: Instance let instance: Instance
@ -13,13 +13,15 @@ struct InstanceInfoView: View {
InstanceInfoSection(instance: instance) InstanceInfoSection(instance: instance)
} }
.navigationTitle("instance.info.navigation-title") .navigationTitle("instance.info.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
#endif
} }
} }
public struct InstanceInfoSection: View { public struct InstanceInfoSection: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
let instance: Instance let instance: Instance
@ -35,7 +37,9 @@ public struct InstanceInfoSection: View {
LabeledContent("instance.info.posts", value: format(instance.stats.statusCount)) LabeledContent("instance.info.posts", value: format(instance.stats.statusCount))
LabeledContent("instance.info.domains", value: format(instance.stats.domainCount)) LabeledContent("instance.info.domains", value: format(instance.stats.domainCount))
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
if let rules = instance.rules { if let rules = instance.rules {
Section("instance.info.section.rules") { Section("instance.info.section.rules") {
@ -43,7 +47,9 @@ public struct InstanceInfoSection: View {
Text(rule.text.trimmingCharacters(in: .whitespacesAndNewlines)) Text(rule.text.trimmingCharacters(in: .whitespacesAndNewlines))
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
} }

View file

@ -7,12 +7,13 @@ import NukeUI
import SwiftUI import SwiftUI
import UserNotifications import UserNotifications
@MainActor
struct PushNotificationsView: View { struct PushNotificationsView: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var appAccountsManager: AppAccountsManager @Environment(AppAccountsManager.self) private var appAccountsManager
@EnvironmentObject private var pushNotifications: PushNotificationsService @Environment(PushNotificationsService.self) private var pushNotifications
@StateObject public var subscription: PushNotificationSubscriptionSettings @State public var subscription: PushNotificationSubscriptionSettings
var body: some View { var body: some View {
Form { Form {
@ -32,7 +33,9 @@ struct PushNotificationsView: View {
} footer: { } footer: {
Text("settings.push.main-toggle.description") Text("settings.push.main-toggle.description")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
if subscription.isEnabled { if subscription.isEnabled {
Section { Section {
@ -85,7 +88,9 @@ struct PushNotificationsView: View {
Label("settings.push.new-posts", systemImage: "bubble.right") Label("settings.push.new-posts", systemImage: "bubble.right")
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
Section { Section {
@ -100,14 +105,18 @@ struct PushNotificationsView: View {
} footer: { } footer: {
Text("settings.push.duplicate.footer") Text("settings.push.duplicate.footer")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
.navigationTitle("settings.push.navigation-title") .navigationTitle("settings.push.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.task { .background(theme.secondaryBackgroundColor)
await subscription.fetchSubscription() #endif
} .task {
await subscription.fetchSubscription()
}
} }
private func updateSubscription() { private func updateSubscription() {

View file

@ -0,0 +1,44 @@
import DesignSystem
import Env
import Models
import SwiftData
import SwiftUI
struct RecenTagsSettingView: View {
@Environment(\.modelContext) private var context
@Environment(RouterPath.self) private var routerPath
@Environment(Theme.self) private var theme
@Query(sort: \RecentTag.lastUse, order: .reverse) var tags: [RecentTag]
var body: some View {
Form {
ForEach(tags) { tag in
VStack(alignment: .leading) {
Text("#\(tag.title)")
.font(.scaledBody)
.foregroundColor(.primary)
Text(tag.formattedDate)
.font(.scaledFootnote)
.foregroundStyle(.secondary)
}
}.onDelete { indexes in
if let index = indexes.first {
context.delete(tags[index])
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
.navigationTitle("settings.general.recent-tags")
.scrollContentBackground(.hidden)
#if !os(visionOS)
.background(theme.secondaryBackgroundColor)
#endif
.toolbar {
EditButton()
}
}
}

View file

@ -0,0 +1,45 @@
import DesignSystem
import Env
import Models
import SwiftData
import SwiftUI
struct RemoteTimelinesSettingView: View {
@Environment(\.modelContext) private var context
@Environment(RouterPath.self) private var routerPath
@Environment(Theme.self) private var theme
@Query(sort: \LocalTimeline.creationDate, order: .reverse) var localTimelines: [LocalTimeline]
var body: some View {
Form {
ForEach(localTimelines) { timeline in
Text(timeline.instance)
}.onDelete { indexes in
if let index = indexes.first {
context.delete(localTimelines[index])
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
Button {
routerPath.presentedSheet = .addRemoteLocalTimeline
} label: {
Label("settings.timeline.add", systemImage: "badge.plus.radiowaves.right")
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
.navigationTitle("settings.general.remote-timelines")
.scrollContentBackground(.hidden)
#if !os(visionOS)
.background(theme.secondaryBackgroundColor)
#endif
.toolbar {
EditButton()
}
}
}

View file

@ -6,26 +6,31 @@ import Foundation
import Models import Models
import Network import Network
import Nuke import Nuke
import SwiftData
import SwiftUI import SwiftUI
import Timeline import Timeline
@MainActor
struct SettingsTabs: View { struct SettingsTabs: View {
@Environment(\.dismiss) private var dismiss @Environment(\.dismiss) private var dismiss
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
@EnvironmentObject private var pushNotifications: PushNotificationsService @Environment(PushNotificationsService.self) private var pushNotifications
@EnvironmentObject private var preferences: UserPreferences @Environment(UserPreferences.self) private var preferences
@EnvironmentObject private var client: Client @Environment(Client.self) private var client
@EnvironmentObject private var currentInstance: CurrentInstance @Environment(CurrentInstance.self) private var currentInstance
@EnvironmentObject private var appAccountsManager: AppAccountsManager @Environment(AppAccountsManager.self) private var appAccountsManager
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@StateObject private var routerPath = RouterPath()
@State private var routerPath = RouterPath()
@State private var addAccountSheetPresented = false @State private var addAccountSheetPresented = false
@State private var isEditingAccount = false @State private var isEditingAccount = false
@State private var cachedRemoved = false @State private var cachedRemoved = false
@State private var timelineCache = TimelineCache()
@Binding var popToRootTab: Tab let isModal: Bool
@State private var startingPoint: SettingsStartingPoint? = nil
var body: some View { var body: some View {
NavigationStack(path: $routerPath.path) { NavigationStack(path: $routerPath.path) {
@ -34,29 +39,59 @@ struct SettingsTabs: View {
accountsSection accountsSection
generalSection generalSection
otherSections otherSections
postStreamingSection
AISection
cacheSection cacheSection
} }
.scrollContentBackground(.hidden) .scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor) #if !os(visionOS)
.navigationTitle(Text("settings.title")) .background(theme.secondaryBackgroundColor)
.navigationBarTitleDisplayMode(.inline) #endif
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .navigationBar) .navigationTitle(Text("settings.title"))
.toolbar { .navigationBarTitleDisplayMode(.inline)
if UIDevice.current.userInterfaceIdiom == .phone { .toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .navigationBar)
ToolbarItem { .toolbar {
Button { if isModal {
dismiss() ToolbarItem {
} label: { Button {
Text("action.done").bold() dismiss()
} label: {
Text("action.done").bold()
}
} }
} }
if UIDevice.current.userInterfaceIdiom == .pad, !preferences.showiPadSecondaryColumn, !isModal {
SecondaryColumnToolbarItem()
}
} }
if UIDevice.current.userInterfaceIdiom == .pad && !preferences.showiPadSecondaryColumn { .withAppRouter()
SecondaryColumnToolbarItem() .withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
.onAppear {
startingPoint = RouterPath.settingsStartingPoint
RouterPath.settingsStartingPoint = nil
}
.navigationDestination(item: $startingPoint) { targetView in
switch targetView {
case .display:
DisplaySettingsView()
case .haptic:
HapticSettingsView()
case .remoteTimelines:
RemoteTimelinesSettingView()
case .tagGroups:
TagsGroupSettingView()
case .recentTags:
RecenTagsSettingView()
case .content:
ContentSettingsView()
case .swipeActions:
SwipeActionsSettingsView()
case .tabAndSidebarEntries:
EmptyView()
case .translation:
TranslationSettingsView()
}
} }
}
.withAppRouter()
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
} }
.onAppear { .onAppear {
routerPath.client = client routerPath.client = client
@ -67,12 +102,7 @@ struct SettingsTabs: View {
} }
} }
.withSafariRouter() .withSafariRouter()
.environmentObject(routerPath) .environment(routerPath)
.onChange(of: $popToRootTab.wrappedValue) { popToRootTab in
if popToRootTab == .notifications {
routerPath.path = []
}
}
} }
private var accountsSection: some View { private var accountsSection: some View {
@ -90,7 +120,7 @@ struct SettingsTabs: View {
.tint(.red) .tint(.red)
} }
} }
AppAccountView(viewModel: .init(appAccount: account)) AppAccountView(viewModel: .init(appAccount: account), isParentPresented: .constant(false))
} }
} }
.onDelete { indexSet in .onDelete { indexSet in
@ -101,12 +131,14 @@ struct SettingsTabs: View {
} }
} }
} }
addAccountButton
if !appAccountsManager.availableAccounts.isEmpty { if !appAccountsManager.availableAccounts.isEmpty {
editAccountButton editAccountButton
} }
addAccountButton
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private func logoutAccount(account: AppAccount) async { private func logoutAccount(account: AppAccount) async {
@ -114,9 +146,10 @@ struct SettingsTabs: View {
let sub = pushNotifications.subscriptions.first(where: { $0.account.token == token }) let sub = pushNotifications.subscriptions.first(where: { $0.account.token == token })
{ {
let client = Client(server: account.server, oauthToken: token) let client = Client(server: account.server, oauthToken: token)
await TimelineCache.shared.clearCache(for: client.id) await timelineCache.clearCache(for: client.id)
await sub.deleteSubscription() await sub.deleteSubscription()
appAccountsManager.delete(account: account) appAccountsManager.delete(account: account)
Telemetry.signal("account.removed")
} }
} }
@ -136,32 +169,50 @@ struct SettingsTabs: View {
Label("settings.general.haptic", systemImage: "waveform.path") Label("settings.general.haptic", systemImage: "waveform.path")
} }
} }
NavigationLink(destination: remoteLocalTimelinesView) { NavigationLink(destination: RemoteTimelinesSettingView()) {
Label("settings.general.remote-timelines", systemImage: "dot.radiowaves.right") Label("settings.general.remote-timelines", systemImage: "dot.radiowaves.right")
} }
NavigationLink(destination: tagGroupsView) { NavigationLink(destination: TagsGroupSettingView()) {
Label("timeline.filter.tag-groups", systemImage: "number") Label("timeline.filter.tag-groups", systemImage: "number")
} }
NavigationLink(destination: RecenTagsSettingView()) {
Label("settings.general.recent-tags", systemImage: "clock")
}
NavigationLink(destination: ContentSettingsView()) { NavigationLink(destination: ContentSettingsView()) {
Label("settings.general.content", systemImage: "rectangle.stack") Label("settings.general.content", systemImage: "rectangle.stack")
} }
NavigationLink(destination: SwipeActionsSettingsView()) { NavigationLink(destination: SwipeActionsSettingsView()) {
Label("settings.general.swipeactions", systemImage: "hand.draw") Label("settings.general.swipeactions", systemImage: "hand.draw")
} }
if UIDevice.current.userInterfaceIdiom == .phone || horizontalSizeClass == .compact {
NavigationLink(destination: TabbarEntriesSettingsView()) {
Label("settings.general.tabbarEntries", systemImage: "platter.filled.bottom.iphone")
}
} else if UIDevice.current.userInterfaceIdiom == .pad || UIDevice.current.userInterfaceIdiom == .mac {
NavigationLink(destination: SidebarEntriesSettingsView()) {
Label("settings.general.sidebarEntries", systemImage: "sidebar.squares.leading")
}
}
NavigationLink(destination: TranslationSettingsView()) { NavigationLink(destination: TranslationSettingsView()) {
Label("settings.general.translate", systemImage: "captions.bubble") Label("settings.general.translate", systemImage: "captions.bubble")
} }
Link(destination: URL(string: UIApplication.openSettingsURLString)!) { #if !targetEnvironment(macCatalyst)
Label("settings.system", systemImage: "gear") Link(destination: URL(string: UIApplication.openSettingsURLString)!) {
} Label("settings.system", systemImage: "gear")
.tint(theme.labelColor) }
.tint(theme.labelColor)
#endif
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
@ViewBuilder
private var otherSections: some View { private var otherSections: some View {
Section("settings.section.other") { @Bindable var preferences = preferences
if !ProcessInfo.processInfo.isiOSAppOnMac { Section {
#if !targetEnvironment(macCatalyst)
Picker(selection: $preferences.preferredBrowser) { Picker(selection: $preferences.preferredBrowser) {
ForEach(PreferredBrowser.allCases, id: \.rawValue) { browser in ForEach(PreferredBrowser.allCases, id: \.rawValue) { browser in
switch browser { switch browser {
@ -178,35 +229,79 @@ struct SettingsTabs: View {
Label("settings.general.browser.in-app.readerview", systemImage: "doc.plaintext") Label("settings.general.browser.in-app.readerview", systemImage: "doc.plaintext")
} }
.disabled(preferences.preferredBrowser != PreferredBrowser.inAppSafari) .disabled(preferences.preferredBrowser != PreferredBrowser.inAppSafari)
} #endif
Toggle(isOn: $preferences.isOpenAIEnabled) {
Label("settings.other.hide-openai", systemImage: "faxmachine")
}
Toggle(isOn: $preferences.isSocialKeyboardEnabled) { Toggle(isOn: $preferences.isSocialKeyboardEnabled) {
Label("settings.other.social-keyboard", systemImage: "keyboard") Label("settings.other.social-keyboard", systemImage: "keyboard")
} }
Toggle(isOn: $preferences.soundEffectEnabled) { Toggle(isOn: $preferences.soundEffectEnabled) {
Label("settings.other.sound-effect", systemImage: "hifispeaker") Label("settings.other.sound-effect", systemImage: "hifispeaker")
} }
Toggle(isOn: $preferences.fastRefreshEnabled) {
Label("settings.other.fast-refresh", systemImage: "arrow.clockwise")
}
} header: {
Text("settings.section.other")
} footer: {
Text("settings.section.other.footer")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
}
@ViewBuilder
private var postStreamingSection: some View {
@Bindable var preferences = preferences
Section {
Toggle(isOn: $preferences.isPostsStreamingEnabled) {
Label("Posts streaming", systemImage: "clock.badge")
}
} header: {
Text("Streaming")
} footer: {
Text("Enabling post streaming will automatically add new posts at the top of your home timeline. Disable if you get performance issues.")
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
@ViewBuilder
private var AISection: some View {
@Bindable var preferences = preferences
Section {
Toggle(isOn: $preferences.isOpenAIEnabled) {
Label("settings.other.hide-openai", systemImage: "faxmachine")
}
} header: {
Text("AI")
} footer: {
Text("Disable to hide AI assisted tool options such as copywritting and alt-image description generated using AI. Uses OpenAI API. See our Privacy Policy for more information.")
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private var appSection: some View { private var appSection: some View {
Section { Section {
if !ProcessInfo.processInfo.isiOSAppOnMac { #if !targetEnvironment(macCatalyst) && !os(visionOS)
NavigationLink(destination: IconSelectorView()) { NavigationLink(destination: IconSelectorView()) {
Label { Label {
Text("settings.app.icon") Text("settings.app.icon")
} icon: { } icon: {
let icon = IconSelectorView.Icon(string: UIApplication.shared.alternateIconName ?? "AppIcon") let icon = IconSelectorView.Icon(string: UIApplication.shared.alternateIconName ?? "AppIcon")
Image(uiImage: .init(named: icon.iconName)!) if let image: UIImage = .init(named: icon.previewImageName) {
.resizable() Image(uiImage: image)
.frame(width: 25, height: 25) .resizable()
.cornerRadius(4) .frame(width: 25, height: 25)
.cornerRadius(4)
} else {
EmptyView()
}
} }
} }
} #endif
Link(destination: URL(string: "https://github.com/Dimillian/IceCubesApp")!) { Link(destination: URL(string: "https://github.com/Dimillian/IceCubesApp")!) {
Label("settings.app.source", systemImage: "link") Label("settings.app.source", systemImage: "link")
@ -226,9 +321,17 @@ struct SettingsTabs: View {
.tint(theme.labelColor) .tint(theme.labelColor)
} }
NavigationLink(destination: AboutView()) { NavigationLink {
AboutView()
} label: {
Label("settings.app.about", systemImage: "info.circle") Label("settings.app.about", systemImage: "info.circle")
} }
NavigationLink {
WishlistView()
} label: {
Label("Feature Requests", systemImage: "list.bullet.rectangle.portrait")
}
} header: { } header: {
Text("settings.section.app") Text("settings.section.app")
@ -237,14 +340,16 @@ struct SettingsTabs: View {
Text("settings.section.app.footer \(appVersion)").frame(maxWidth: .infinity, alignment: .center) Text("settings.section.app.footer \(appVersion)").frame(maxWidth: .infinity, alignment: .center)
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private var addAccountButton: some View { private var addAccountButton: some View {
Button { Button {
addAccountSheetPresented.toggle() addAccountSheetPresented.toggle()
} label: { } label: {
Text("settings.account.add") Label("settings.account.add", systemImage: "person.badge.plus")
} }
.sheet(isPresented: $addAccountSheetPresented) { .sheet(isPresented: $addAccountSheetPresented) {
AddAccountView() AddAccountView()
@ -252,84 +357,23 @@ struct SettingsTabs: View {
} }
private var editAccountButton: some View { private var editAccountButton: some View {
Button(role: isEditingAccount ? .none : .destructive) { Button(role: .destructive) {
withAnimation { withAnimation {
isEditingAccount.toggle() isEditingAccount.toggle()
} }
} label: { } label: {
if isEditingAccount { if isEditingAccount {
Text("action.done") Label("action.done", systemImage: "person.badge.minus")
.foregroundStyle(.red)
} else { } else {
Text("account.action.logout") Label("account.action.logout", systemImage: "person.badge.minus")
.foregroundStyle(.red)
} }
} }
} }
private var tagGroupsView: some View {
Form {
ForEach(preferences.tagGroups, id: \.self) { group in
Label(group.title, systemImage: group.sfSymbolName)
.onTapGesture {
routerPath.presentedSheet = .editTagGroup(tagGroup: group, onSaved: nil)
}
}
.onDelete { indexes in
if let index = indexes.first {
_ = preferences.tagGroups.remove(at: index)
}
}
.onMove { source, destination in
preferences.tagGroups.move(fromOffsets: source, toOffset: destination)
}
.listRowBackground(theme.primaryBackgroundColor)
Button {
routerPath.presentedSheet = .addTagGroup
} label: {
Label("timeline.filter.add-tag-groups", systemImage: "plus")
}
.listRowBackground(theme.primaryBackgroundColor)
}
.navigationTitle("timeline.filter.tag-groups")
.scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
.toolbar {
EditButton()
}
}
private var remoteLocalTimelinesView: some View {
Form {
ForEach(preferences.remoteLocalTimelines, id: \.self) { server in
Text(server)
}.onDelete { indexes in
if let index = indexes.first {
_ = preferences.remoteLocalTimelines.remove(at: index)
}
}
.onMove(perform: moveTimelineItems)
.listRowBackground(theme.primaryBackgroundColor)
Button {
routerPath.presentedSheet = .addRemoteLocalTimeline
} label: {
Label("settings.timeline.add", systemImage: "badge.plus.radiowaves.right")
}
.listRowBackground(theme.primaryBackgroundColor)
}
.navigationTitle("settings.general.remote-timelines")
.scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
.toolbar {
EditButton()
}
}
private func moveTimelineItems(from source: IndexSet, to destination: Int) {
preferences.remoteLocalTimelines.move(fromOffsets: source, toOffset: destination)
}
private var cacheSection: some View { private var cacheSection: some View {
Section("settings.section.cache") { Section {
if cachedRemoved { if cachedRemoved {
Text("action.done") Text("action.done")
.transition(.move(edge: .leading)) .transition(.move(edge: .leading))
@ -341,7 +385,13 @@ struct SettingsTabs: View {
} }
} }
} }
} header: {
Text("settings.section.cache")
} footer: {
Text("Remove all cached images and videos")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
} }

View file

@ -0,0 +1,40 @@
import DesignSystem
import Env
import SwiftUI
@MainActor
struct SidebarEntriesSettingsView: View {
@Environment(Theme.self) private var theme
@Environment(UserPreferences.self) private var userPreferences
@State private var sidebarTabs = SidebarTabs.shared
var body: some View {
@Bindable var userPreferences = userPreferences
Form {
Section {
ForEach($sidebarTabs.tabs, id: \.tab) { $tab in
if tab.tab != .profile && tab.tab != .settings {
Toggle(isOn: $tab.enabled) {
tab.tab.label
}
}
}
.onMove(perform: move)
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
.environment(\.editMode, .constant(.active))
.navigationTitle("settings.general.sidebarEntries")
#if !os(visionOS)
.scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
#endif
}
func move(from source: IndexSet, to destination: Int) {
sidebarTabs.tabs.move(fromOffsets: source, toOffset: destination)
}
}

View file

@ -1,9 +1,9 @@
import DesignSystem import DesignSystem
import Env import Env
import RevenueCat import RevenueCat
import Shimmer
import SwiftUI import SwiftUI
@MainActor
struct SupportAppView: View { struct SupportAppView: View {
enum Tip: String, CaseIterable { enum Tip: String, CaseIterable {
case one, two, three, four, supporter case one, two, three, four, supporter
@ -19,35 +19,35 @@ struct SupportAppView: View {
var title: LocalizedStringKey { var title: LocalizedStringKey {
switch self { switch self {
case .one: case .one:
return "settings.support.one.title" "settings.support.one.title"
case .two: case .two:
return "settings.support.two.title" "settings.support.two.title"
case .three: case .three:
return "settings.support.three.title" "settings.support.three.title"
case .four: case .four:
return "settings.support.four.title" "settings.support.four.title"
case .supporter: case .supporter:
return "settings.support.supporter.title" "settings.support.supporter.title"
} }
} }
var subtitle: LocalizedStringKey { var subtitle: LocalizedStringKey {
switch self { switch self {
case .one: case .one:
return "settings.support.one.subtitle" "settings.support.one.subtitle"
case .two: case .two:
return "settings.support.two.subtitle" "settings.support.two.subtitle"
case .three: case .three:
return "settings.support.three.subtitle" "settings.support.three.subtitle"
case .four: case .four:
return "settings.support.four.subtitle" "settings.support.four.subtitle"
case .supporter: case .supporter:
return "settings.support.supporter.subtitle" "settings.support.supporter.subtitle"
} }
} }
} }
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@Environment(\.openURL) private var openURL @Environment(\.openURL) private var openURL
@ -68,23 +68,25 @@ struct SupportAppView: View {
linksSection linksSection
} }
.navigationTitle("settings.support.navigation-title") .navigationTitle("settings.support.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.alert("settings.support.alert.title", isPresented: $purchaseSuccessDisplayed, actions: { .background(theme.secondaryBackgroundColor)
Button { purchaseSuccessDisplayed = false } label: { Text("alert.button.ok") } #endif
}, message: { .alert("settings.support.alert.title", isPresented: $purchaseSuccessDisplayed, actions: {
Text("settings.support.alert.message") Button { purchaseSuccessDisplayed = false } label: { Text("alert.button.ok") }
}) }, message: {
.alert("alert.error", isPresented: $purchaseErrorDisplayed, actions: { Text("settings.support.alert.message")
Button { purchaseErrorDisplayed = false } label: { Text("alert.button.ok") } })
}, message: { .alert("alert.error", isPresented: $purchaseErrorDisplayed, actions: {
Text("settings.support.alert.error.message") Button { purchaseErrorDisplayed = false } label: { Text("alert.button.ok") }
}) }, message: {
.onAppear { Text("settings.support.alert.error.message")
loadingProducts = true })
fetchStoreProducts() .onAppear {
refreshUserInfo() loadingProducts = true
} fetchStoreProducts()
refreshUserInfo()
}
} }
private func purchase(product: StoreProduct) async { private func purchase(product: StoreProduct) async {
@ -103,8 +105,8 @@ struct SupportAppView: View {
} }
private func fetchStoreProducts() { private func fetchStoreProducts() {
Purchases.shared.getProducts(Tip.allCases.map { $0.productId }) { products in Purchases.shared.getProducts(Tip.allCases.map(\.productId)) { products in
self.subscription = products.first(where: { $0.productIdentifier == Tip.supporter.productId }) subscription = products.first(where: { $0.productIdentifier == Tip.supporter.productId })
self.products = products.filter { $0.productIdentifier != Tip.supporter.productId }.sorted(by: { $0.price < $1.price }) self.products = products.filter { $0.productIdentifier != Tip.supporter.productId }.sorted(by: { $0.price < $1.price })
withAnimation { withAnimation {
loadingProducts = false loadingProducts = false
@ -114,7 +116,7 @@ struct SupportAppView: View {
private func refreshUserInfo() { private func refreshUserInfo() {
Purchases.shared.getCustomerInfo { info, _ in Purchases.shared.getCustomerInfo { info, _ in
self.customerInfo = info customerInfo = info
} }
} }
@ -150,7 +152,9 @@ struct SupportAppView: View {
Text("settings.support.message-from-dev") Text("settings.support.message-from-dev")
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private var subscriptionSection: some View { private var subscriptionSection: some View {
@ -174,7 +178,7 @@ struct SupportAppView: View {
.font(.scaledSubheadline) .font(.scaledSubheadline)
Text(Tip.supporter.subtitle) Text(Tip.supporter.subtitle)
.font(.scaledFootnote) .font(.scaledFootnote)
.foregroundColor(.gray) .foregroundStyle(.secondary)
} }
Spacer() Spacer()
makePurchaseButton(product: subscription) makePurchaseButton(product: subscription)
@ -187,7 +191,9 @@ struct SupportAppView: View {
Text("settings.support.supporter.subscription-info") Text("settings.support.supporter.subscription-info")
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private var tipsSection: some View { private var tipsSection: some View {
@ -203,7 +209,7 @@ struct SupportAppView: View {
.font(.scaledSubheadline) .font(.scaledSubheadline)
Text(tip.subtitle) Text(tip.subtitle)
.font(.scaledFootnote) .font(.scaledFootnote)
.foregroundColor(.gray) .foregroundStyle(.secondary)
} }
Spacer() Spacer()
makePurchaseButton(product: product) makePurchaseButton(product: product)
@ -212,7 +218,9 @@ struct SupportAppView: View {
} }
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
private var restorePurchase: some View { private var restorePurchase: some View {
@ -221,7 +229,7 @@ struct SupportAppView: View {
Spacer() Spacer()
Button { Button {
Purchases.shared.restorePurchases { info, _ in Purchases.shared.restorePurchases { info, _ in
self.customerInfo = info customerInfo = info
} }
} label: { } label: {
Text("settings.support.restore-purchase.button") Text("settings.support.restore-purchase.button")
@ -231,7 +239,9 @@ struct SupportAppView: View {
} footer: { } footer: {
Text("settings.support.restore-purchase.explanation") Text("settings.support.restore-purchase.explanation")
} }
#if !os(visionOS)
.listRowBackground(theme.secondaryBackgroundColor) .listRowBackground(theme.secondaryBackgroundColor)
#endif
} }
private var linksSection: some View { private var linksSection: some View {
@ -251,21 +261,23 @@ struct SupportAppView: View {
.buttonStyle(.borderless) .buttonStyle(.borderless)
} }
} }
#if !os(visionOS)
.listRowBackground(theme.secondaryBackgroundColor) .listRowBackground(theme.secondaryBackgroundColor)
#endif
} }
private var loadingPlaceholder: some View { private var loadingPlaceholder: some View {
HStack { HStack {
VStack(alignment: .leading) { VStack(alignment: .leading) {
Text("placeholder.loading.short.") Text("placeholder.loading.short")
.font(.scaledSubheadline) .font(.scaledSubheadline)
Text("settings.support.placeholder.loading-subtitle") Text("settings.support.placeholder.loading-subtitle")
.font(.scaledFootnote) .font(.scaledFootnote)
.foregroundColor(.gray) .foregroundStyle(.secondary)
} }
.padding(.vertical, 8) .padding(.vertical, 8)
} }
.redacted(reason: .placeholder) .redacted(reason: .placeholder)
.shimmering() .allowsHitTesting(false)
} }
} }

View file

@ -2,11 +2,13 @@ import DesignSystem
import Env import Env
import SwiftUI import SwiftUI
@MainActor
struct SwipeActionsSettingsView: View { struct SwipeActionsSettingsView: View {
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@EnvironmentObject private var userPreferences: UserPreferences @Environment(UserPreferences.self) private var userPreferences
var body: some View { var body: some View {
@Bindable var userPreferences = userPreferences
Form { Form {
Section { Section {
Label("settings.swipeactions.status.leading", systemImage: "arrow.right") Label("settings.swipeactions.status.leading", systemImage: "arrow.right")
@ -14,7 +16,7 @@ struct SwipeActionsSettingsView: View {
createStatusActionPicker(selection: $userPreferences.swipeActionsStatusLeadingLeft, createStatusActionPicker(selection: $userPreferences.swipeActionsStatusLeadingLeft,
label: "settings.swipeactions.primary") label: "settings.swipeactions.primary")
.onChange(of: userPreferences.swipeActionsStatusLeadingLeft) { action in .onChange(of: userPreferences.swipeActionsStatusLeadingLeft) { _, action in
if action == .none { if action == .none {
userPreferences.swipeActionsStatusLeadingRight = .none userPreferences.swipeActionsStatusLeadingRight = .none
} }
@ -29,7 +31,7 @@ struct SwipeActionsSettingsView: View {
createStatusActionPicker(selection: $userPreferences.swipeActionsStatusTrailingRight, createStatusActionPicker(selection: $userPreferences.swipeActionsStatusTrailingRight,
label: "settings.swipeactions.primary") label: "settings.swipeactions.primary")
.onChange(of: userPreferences.swipeActionsStatusTrailingRight) { action in .onChange(of: userPreferences.swipeActionsStatusTrailingRight) { _, action in
if action == .none { if action == .none {
userPreferences.swipeActionsStatusTrailingLeft = .none userPreferences.swipeActionsStatusTrailingLeft = .none
} }
@ -44,7 +46,9 @@ struct SwipeActionsSettingsView: View {
} footer: { } footer: {
Text("settings.swipeactions.status.explanation") Text("settings.swipeactions.status.explanation")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
Section { Section {
Picker(selection: $userPreferences.swipeActionsIconStyle, label: Text("settings.swipeactions.icon-style")) { Picker(selection: $userPreferences.swipeActionsIconStyle, label: Text("settings.swipeactions.icon-style")) {
@ -60,15 +64,19 @@ struct SwipeActionsSettingsView: View {
} footer: { } footer: {
Text("settings.swipeactions.use-theme-colors-explanation") Text("settings.swipeactions.use-theme-colors-explanation")
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
.navigationTitle("settings.swipeactions.navigation-title") .navigationTitle("settings.swipeactions.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
#endif
} }
private func createStatusActionPicker(selection: Binding<StatusAction>, label: LocalizedStringKey) -> some View { private func createStatusActionPicker(selection: Binding<StatusAction>, label: LocalizedStringKey) -> some View {
return Picker(selection: selection, label: Text(label)) { Picker(selection: selection, label: Text(label)) {
Section { Section {
Text(StatusAction.none.displayName()).tag(StatusAction.none) Text(StatusAction.none.displayName()).tag(StatusAction.none)
} }

View file

@ -0,0 +1,69 @@
import DesignSystem
import Env
import SwiftUI
@MainActor
struct TabbarEntriesSettingsView: View {
@Environment(Theme.self) private var theme
@Environment(UserPreferences.self) private var userPreferences
@State private var tabs = iOSTabs.shared
var body: some View {
@Bindable var userPreferences = userPreferences
Form {
Section {
Picker("settings.tabs.first-tab", selection: $tabs.firstTab) {
ForEach(AppTab.allCases) { tab in
if tab == tabs.firstTab || !tabs.tabs.contains(tab) {
tab.label.tag(tab)
}
}
}
Picker("settings.tabs.second-tab", selection: $tabs.secondTab) {
ForEach(AppTab.allCases) { tab in
if tab == tabs.secondTab || !tabs.tabs.contains(tab) {
tab.label.tag(tab)
}
}
}
Picker("settings.tabs.third-tab", selection: $tabs.thirdTab) {
ForEach(AppTab.allCases) { tab in
if tab == tabs.thirdTab || !tabs.tabs.contains(tab) {
tab.label.tag(tab)
}
}
}
Picker("settings.tabs.fourth-tab", selection: $tabs.fourthTab) {
ForEach(AppTab.allCases) { tab in
if tab == tabs.fourthTab || !tabs.tabs.contains(tab) {
tab.label.tag(tab)
}
}
}
Picker("settings.tabs.fifth-tab", selection: $tabs.fifthTab) {
ForEach(AppTab.allCases) { tab in
if tab == tabs.fifthTab || !tabs.tabs.contains(tab) {
tab.label.tag(tab)
}
}
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
Section {
Toggle("settings.display.show-tab-label", isOn: $userPreferences.showiPhoneTabLabel)
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
.navigationTitle("settings.general.tabbarEntries")
#if !os(visionOS)
.scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
#endif
}
}

View file

@ -0,0 +1,50 @@
import DesignSystem
import Env
import Models
import SwiftData
import SwiftUI
struct TagsGroupSettingView: View {
@Environment(\.modelContext) private var context
@Environment(RouterPath.self) private var routerPath
@Environment(Theme.self) private var theme
@Query(sort: \TagGroup.creationDate, order: .reverse) var tagGroups: [TagGroup]
var body: some View {
Form {
ForEach(tagGroups) { group in
Label(group.title, systemImage: group.symbolName)
.onTapGesture {
routerPath.presentedSheet = .editTagGroup(tagGroup: group, onSaved: nil)
}
}
.onDelete { indexes in
if let index = indexes.first {
context.delete(tagGroups[index])
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
Button {
routerPath.presentedSheet = .addTagGroup
} label: {
Label("timeline.filter.add-tag-groups", systemImage: "plus")
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
.navigationTitle("timeline.filter.tag-groups")
.scrollContentBackground(.hidden)
#if !os(visionOS)
.background(theme.secondaryBackgroundColor)
#endif
.toolbar {
EditButton()
}
}
}

View file

@ -2,31 +2,25 @@ import DesignSystem
import Env import Env
import SwiftUI import SwiftUI
@MainActor
struct TranslationSettingsView: View { struct TranslationSettingsView: View {
@EnvironmentObject private var preferences: UserPreferences @Environment(UserPreferences.self) private var preferences
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@State private var apiKey: String = "" @State private var apiKey: String = ""
var body: some View { var body: some View {
Form { Form {
Toggle(isOn: preferences.$alwaysUseDeepl) { translationSelector
Text("settings.translation.always-deepl") if preferences.preferredTranslationType == .useDeepl {
}
.listRowBackground(theme.primaryBackgroundColor)
if preferences.alwaysUseDeepl {
Section("settings.translation.user-api-key") { Section("settings.translation.user-api-key") {
Picker("settings.translation.api-key-type", selection: $preferences.userDeeplAPIFree) { deepLPicker
Text("DeepL API Free").tag(true)
Text("DeepL API Pro").tag(false)
}
SecureField("settings.translation.user-api-key", text: $apiKey) SecureField("settings.translation.user-api-key", text: $apiKey)
.textContentType(.password) .textContentType(.password)
} }
.onAppear(perform: readValue) #if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
if apiKey.isEmpty { if apiKey.isEmpty {
Section { Section {
@ -35,23 +29,103 @@ struct TranslationSettingsView: View {
.foregroundColor(.red) .foregroundColor(.red)
} }
} }
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
#endif
} }
} }
backgroundAPIKey
Section { autoDetectSection
Toggle(isOn: preferences.$autoDetectPostLanguage) {
Text("settings.translation.auto-detect-post-language")
}
} footer: {
Text("settings.translation.auto-detect-post-language-footer")
}
} }
.navigationTitle("settings.translation.navigation-title") .navigationTitle("settings.translation.navigation-title")
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.onChange(of: apiKey, perform: writeNewValue) .background(theme.secondaryBackgroundColor)
.onAppear(perform: updatePrefs) #endif
.onChange(of: apiKey) {
writeNewValue()
}
.onAppear(perform: updatePrefs)
.onAppear(perform: readValue)
}
@ViewBuilder
private var translationSelector: some View {
@Bindable var preferences = preferences
Picker("Translation Service", selection: $preferences.preferredTranslationType) {
ForEach(allTTCases, id: \.self) { type in
Text(type.description).tag(type)
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
var allTTCases: [TranslationType] {
TranslationType.allCases.filter { type in
if type != .useApple {
return true
}
#if canImport(_Translation_SwiftUI)
if #available(iOS 17.4, *) {
return true
} else {
return false
}
#else
return false
#endif
}
}
@ViewBuilder
private var deepLPicker: some View {
@Bindable var preferences = preferences
Picker("settings.translation.api-key-type", selection: $preferences.userDeeplAPIFree) {
Text("DeepL API Free").tag(true)
Text("DeepL API Pro").tag(false)
}
}
@ViewBuilder
private var autoDetectSection: some View {
@Bindable var preferences = preferences
Section {
Toggle(isOn: $preferences.autoDetectPostLanguage) {
Text("settings.translation.auto-detect-post-language")
}
} footer: {
Text("settings.translation.auto-detect-post-language-footer")
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
@ViewBuilder
private var backgroundAPIKey: some View {
if preferences.preferredTranslationType != .useDeepl,
!apiKey.isEmpty
{
Section {
Text("The DeepL API Key is still stored!")
if preferences.preferredTranslationType == .useServerIfPossible {
Text("It can however still be used as a fallback for your instance's translation service.")
}
Button(role: .destructive) {
withAnimation {
writeNewValue(value: "")
readValue()
}
} label: {
Text("action.delete")
}
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
} }
private func writeNewValue() { private func writeNewValue() {
@ -63,11 +137,7 @@ struct TranslationSettingsView: View {
} }
private func readValue() { private func readValue() {
if let apiKey = DeepLUserAPIHandler.readIfAllowed() { apiKey = DeepLUserAPIHandler.readKey()
self.apiKey = apiKey
} else {
apiKey = ""
}
} }
private func updatePrefs() { private func updatePrefs() {
@ -78,6 +148,6 @@ struct TranslationSettingsView: View {
struct TranslationSettingsView_Previews: PreviewProvider { struct TranslationSettingsView_Previews: PreviewProvider {
static var previews: some View { static var previews: some View {
TranslationSettingsView() TranslationSettingsView()
.environmentObject(UserPreferences.shared) .environment(UserPreferences.shared)
} }
} }

View file

@ -0,0 +1,8 @@
import SwiftUI
import WishKit
struct WishlistView: View {
var body: some View {
WishKit.view
}
}

View file

@ -1,54 +1,78 @@
import Account import Account
import AppIntents
import DesignSystem import DesignSystem
import Explore import Explore
import Foundation import Foundation
import Status import StatusKit
import SwiftUI import SwiftUI
enum Tab: Int, Identifiable, Hashable { @MainActor
enum AppTab: Int, Identifiable, Hashable, CaseIterable, Codable {
case timeline, notifications, mentions, explore, messages, settings, other case timeline, notifications, mentions, explore, messages, settings, other
case trending, federated, local case trending, federated, local
case profile case profile
case bookmarks
case favorites
case post
case followedTags
case lists
case links
var id: Int { nonisolated var id: Int {
rawValue rawValue
} }
static func loggedOutTab() -> [Tab] { static func loggedOutTab() -> [AppTab] {
[.timeline, .settings] [.timeline, .settings]
} }
static func loggedInTabs() -> [Tab] { static func visionOSTab() -> [AppTab] {
if UIDevice.current.userInterfaceIdiom == .pad || UIDevice.current.userInterfaceIdiom == .mac { [.profile, .timeline, .notifications, .mentions, .explore, .post, .settings]
return [.timeline, .trending, .federated, .local, .notifications, .mentions, .explore, .messages, .settings]
} else {
return [.timeline, .notifications, .explore, .messages, .profile]
}
} }
@ViewBuilder @ViewBuilder
func makeContentView(popToRootTab: Binding<Tab>) -> some View { func makeContentView(selectedTab: Binding<AppTab>) -> some View {
switch self { switch self {
case .timeline: case .timeline:
TimelineTab(popToRootTab: popToRootTab) TimelineTab()
case .trending: case .trending:
TimelineTab(popToRootTab: popToRootTab, timeline: .trending) TimelineTab(timeline: .trending)
case .local: case .local:
TimelineTab(popToRootTab: popToRootTab, timeline: .local) TimelineTab(timeline: .local)
case .federated: case .federated:
TimelineTab(popToRootTab: popToRootTab, timeline: .federated) TimelineTab(timeline: .federated)
case .notifications: case .notifications:
NotificationsTab(popToRootTab: popToRootTab, lockedType: nil) NotificationsTab(selectedTab: selectedTab, lockedType: nil)
case .mentions: case .mentions:
NotificationsTab(popToRootTab: popToRootTab, lockedType: .mention) NotificationsTab(selectedTab: selectedTab, lockedType: .mention)
case .explore: case .explore:
ExploreTab(popToRootTab: popToRootTab) ExploreTab()
case .messages: case .messages:
MessagesTab(popToRootTab: popToRootTab) MessagesTab()
case .settings: case .settings:
SettingsTabs(popToRootTab: popToRootTab) SettingsTabs(isModal: false)
case .profile: case .profile:
ProfileTab(popToRootTab: popToRootTab) ProfileTab()
case .bookmarks:
NavigationTab {
AccountStatusesListView(mode: .bookmarks)
}
case .favorites:
NavigationTab {
AccountStatusesListView(mode: .favorites)
}
case .followedTags:
NavigationTab {
FollowedTagsListView()
}
case .lists:
NavigationTab {
ListsListView()
}
case .links:
NavigationTab { TrendingLinksListView(cards: []) }
case .post:
VStack {}
case .other: case .other:
EmptyView() EmptyView()
} }
@ -56,56 +80,194 @@ enum Tab: Int, Identifiable, Hashable {
@ViewBuilder @ViewBuilder
var label: some View { var label: some View {
if self != .other {
Label(title, systemImage: iconName)
}
}
var title: LocalizedStringKey {
switch self { switch self {
case .timeline: case .timeline:
Label("tab.timeline", systemImage: iconName) "tab.timeline"
case .trending: case .trending:
Label("tab.trending", systemImage: iconName) "tab.trending"
case .local: case .local:
Label("tab.local", systemImage: iconName) "tab.local"
case .federated: case .federated:
Label("tab.federated", systemImage: iconName) "tab.federated"
case .notifications: case .notifications:
Label("tab.notifications", systemImage: iconName) "tab.notifications"
case .mentions: case .mentions:
Label("tab.notifications", systemImage: iconName) "tab.mentions"
case .explore: case .explore:
Label("tab.explore", systemImage: iconName) "tab.explore"
case .messages: case .messages:
Label("tab.messages", systemImage: iconName) "tab.messages"
case .settings: case .settings:
Label("tab.settings", systemImage: iconName) "tab.settings"
case .profile: case .profile:
Label("tab.profile", systemImage: iconName) "tab.profile"
case .bookmarks:
"accessibility.tabs.profile.picker.bookmarks"
case .favorites:
"accessibility.tabs.profile.picker.favorites"
case .post:
"menu.new-post"
case .followedTags:
"timeline.filter.tags"
case .lists:
"timeline.filter.lists"
case .links:
"explore.section.trending.links"
case .other: case .other:
EmptyView() ""
} }
} }
var iconName: String { var iconName: String {
switch self { switch self {
case .timeline: case .timeline:
return "rectangle.stack" "rectangle.stack"
case .trending: case .trending:
return "chart.line.uptrend.xyaxis" "chart.line.uptrend.xyaxis"
case .local: case .local:
return "person.2" "person.2"
case .federated: case .federated:
return "globe.americas" "globe.americas"
case .notifications: case .notifications:
return "bell" "bell"
case .mentions: case .mentions:
return "at" "at"
case .explore: case .explore:
return "magnifyingglass" "magnifyingglass"
case .messages: case .messages:
return "tray" "tray"
case .settings: case .settings:
return "gear" "gear"
case .profile: case .profile:
return "person.crop.circle" "person.crop.circle"
case .bookmarks:
"bookmark"
case .favorites:
"star"
case .post:
"square.and.pencil"
case .followedTags:
"tag"
case .lists:
"list.bullet"
case .links:
"newspaper"
case .other: case .other:
return "" ""
} }
} }
} }
@MainActor
@Observable
class SidebarTabs {
struct SidedebarTab: Hashable, Codable {
let tab: AppTab
var enabled: Bool
}
class Storage {
@AppStorage("sidebar_tabs") var tabs: [SidedebarTab] = [
.init(tab: .timeline, enabled: true),
.init(tab: .trending, enabled: true),
.init(tab: .federated, enabled: true),
.init(tab: .local, enabled: true),
.init(tab: .notifications, enabled: true),
.init(tab: .mentions, enabled: true),
.init(tab: .messages, enabled: true),
.init(tab: .explore, enabled: true),
.init(tab: .bookmarks, enabled: true),
.init(tab: .favorites, enabled: true),
.init(tab: .followedTags, enabled: true),
.init(tab: .lists, enabled: true),
.init(tab: .links, enabled: true),
.init(tab: .settings, enabled: true),
.init(tab: .profile, enabled: true),
]
}
private let storage = Storage()
public static let shared = SidebarTabs()
var tabs: [SidedebarTab] {
didSet {
storage.tabs = tabs
}
}
func isEnabled(_ tab: AppTab) -> Bool {
tabs.first(where: { $0.tab.id == tab.id })?.enabled == true
}
private init() {
tabs = storage.tabs
}
}
@MainActor
@Observable
class iOSTabs {
enum TabEntries: String {
case first, second, third, fourth, fifth
}
class Storage {
@AppStorage(TabEntries.first.rawValue) var firstTab = AppTab.timeline
@AppStorage(TabEntries.second.rawValue) var secondTab = AppTab.notifications
@AppStorage(TabEntries.third.rawValue) var thirdTab = AppTab.explore
@AppStorage(TabEntries.fourth.rawValue) var fourthTab = AppTab.links
@AppStorage(TabEntries.fifth.rawValue) var fifthTab = AppTab.profile
}
private let storage = Storage()
public static let shared = iOSTabs()
var tabs: [AppTab] {
[firstTab, secondTab, thirdTab, fourthTab, fifthTab]
}
var firstTab: AppTab {
didSet {
storage.firstTab = firstTab
}
}
var secondTab: AppTab {
didSet {
storage.secondTab = secondTab
}
}
var thirdTab: AppTab {
didSet {
storage.thirdTab = thirdTab
}
}
var fourthTab: AppTab {
didSet {
storage.fourthTab = fourthTab
}
}
var fifthTab: AppTab {
didSet {
storage.fifthTab = fifthTab
}
}
private init() {
firstTab = storage.firstTab
secondTab = storage.secondTab
thirdTab = storage.thirdTab
fourthTab = storage.fourthTab
fifthTab = storage.fifthTab
}
}

View file

@ -0,0 +1,448 @@
import Combine
import DesignSystem
import Env
import Models
import Network
import NukeUI
import SFSafeSymbols
import SwiftData
import SwiftUI
@MainActor
struct EditTagGroupView: View {
@Environment(\.dismiss) private var dismiss
@Environment(\.modelContext) private var context
@Environment(Theme.self) private var theme
@State private var tagGroup: TagGroup
private let onSaved: ((TagGroup) -> Void)?
private let isNewGroup: Bool
@FocusState private var focusedField: Focus?
var body: some View {
NavigationStack {
Form {
Section {
TitleInputView(
title: $tagGroup.title,
titleValidationStatus: tagGroup.titleValidationStatus,
focusedField: $focusedField,
isNewGroup: isNewGroup
)
SymbolInputView(
selectedSymbol: $tagGroup.symbolName,
selectedSymbolValidationStatus: tagGroup.symbolNameValidationStatus,
focusedField: $focusedField
)
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
Section("add-tag-groups.edit.tags") {
TagsInputView(
tags: $tagGroup.tags,
tagsValidationStatus: tagGroup.tagsValidationStatus,
focusedField: $focusedField
)
}
#if !os(visionOS)
.listRowBackground(theme.primaryBackgroundColor)
#endif
}
.formStyle(.grouped)
.navigationTitle(
isNewGroup
? "timeline.filter.add-tag-groups"
: "timeline.filter.edit-tag-groups"
)
.navigationBarTitleDisplayMode(.inline)
#if !os(visionOS)
.scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
.scrollDismissesKeyboard(.interactively)
#endif
.toolbar {
CancelToolbarItem()
ToolbarItem(placement: .navigationBarTrailing) {
Button("action.save", action: { save() })
.disabled(!tagGroup.isValid)
}
}
.onAppear {
focusedField = .title
}
}
}
init(tagGroup: TagGroup = .emptyGroup(), onSaved: ((TagGroup) -> Void)? = nil) {
_tagGroup = State(wrappedValue: tagGroup)
self.onSaved = onSaved
isNewGroup = tagGroup.title.isEmpty
}
private func save() {
tagGroup.format()
context.insert(tagGroup)
onSaved?(tagGroup)
dismiss()
}
enum Focus {
case title
case symbol
case new
}
}
struct AddTagGroupView_Previews: PreviewProvider {
static var previews: some View {
let container = try? ModelContainer(for: TagGroup.self, configurations: ModelConfiguration())
// need to use `sheet` to show `symbolsSuggestionView` in preview
return Text(verbatim: "parent view for EditTagGroupView")
.sheet(isPresented: .constant(true)) {
EditTagGroupView()
.withEnvironments()
.modelContainer(container!)
}
}
}
private struct TitleInputView: View {
@Binding var title: String
let titleValidationStatus: TagGroup.TitleValidationStatus
@FocusState.Binding var focusedField: EditTagGroupView.Focus?
@Query var tagGroups: [TagGroup]
let isNewGroup: Bool
var body: some View {
VStack(alignment: .leading) {
TextField("add-tag-groups.edit.title.field", text: $title, axis: .horizontal)
.focused($focusedField, equals: .title)
.onSubmit {
focusedField = .symbol
}
if focusedField == .title, warningText != "" {
Text(warningText).warningLabel()
}
}
}
var warningText: LocalizedStringKey {
if case let .invalid(description) = titleValidationStatus {
return description
} else if
isNewGroup,
tagGroups.contains(where: { $0.title == title })
{
return "\(title) add-tag-groups.edit.title.field.warning.already-exists"
}
return ""
}
}
private struct SymbolInputView: View {
@State private var symbolQuery = ""
@Binding var selectedSymbol: String
let selectedSymbolValidationStatus: TagGroup.SymbolNameValidationStatus
@FocusState.Binding var focusedField: EditTagGroupView.Focus?
var body: some View {
VStack(alignment: .leading) {
HStack {
TextField("add-tag-groups.edit.icon.field", text: $symbolQuery, axis: .horizontal)
.textInputAutocapitalization(.never)
.autocorrectionDisabled()
.focused($focusedField, equals: .symbol)
.onSubmit {
if TagGroup.allSymbols.contains(symbolQuery) {
selectedSymbol = symbolQuery
}
focusedField = .new
}
.onChange(of: focusedField) {
symbolQuery = selectedSymbol
}
Image(systemName: selectedSymbol)
.frame(height: 30)
}
if case let .invalid(description) = selectedSymbolValidationStatus,
focusedField == .symbol
{
Text(description).warningLabel()
}
if focusedField == .symbol {
SymbolSearchResultsView(
symbolQuery: $symbolQuery,
selectedSymbol: $selectedSymbol,
focusedField: $focusedField
)
.frame(maxWidth: .infinity, alignment: .leading)
.frame(height: 40)
}
}
}
}
private struct TagsInputView: View {
@State private var newTag: String = ""
@Binding var tags: [String]
let tagsValidationStatus: TagGroup.TagsValidationStatus
@FocusState.Binding var focusedField: EditTagGroupView.Focus?
var body: some View {
ForEach(tags, id: \.self) { tag in
HStack {
Text(tag)
Spacer()
Button { deleteTag(tag) } label: {
Image(systemName: "trash")
.foregroundStyle(.red)
}
.buttonStyle(.plain)
}
}
.onDelete { indexes in
if let index = indexes.first {
let tag = tags[index]
deleteTag(tag)
}
}
// this `VStack` need to be here to overcome a SwiftUI bug
// "add new tag" `TextField` is not focused after adding the first tag
VStack(alignment: .leading) {
HStack {
// this condition is using to overcome a SwiftUI bug
// "add new tag" `TextField` is not focused after adding the first tag
if tags.isEmpty {
addNewTagTextField()
} else {
addNewTagTextField()
.onAppear { focusedField = .new }
}
Spacer()
if !newTag.isEmpty, !tags.contains(newTag) {
Button { addNewTag() } label: {
Image(systemName: "checkmark.circle.fill").tint(.green)
}
}
}
if focusedField == .new, warningText != "" {
Text(warningText).warningLabel()
}
}
var warningText: LocalizedStringKey {
if tags.contains(newTag) {
return "add-tag-groups.edit.tags.field.warning.duplicated-tag"
} else if case let .invalid(description) = tagsValidationStatus {
return description
}
return ""
}
}
private func addNewTagTextField() -> some View {
VStack(alignment: .leading) {
TextField("add-tag-groups.edit.tags.add", text: $newTag, axis: .horizontal)
.textInputAutocapitalization(.never)
.autocorrectionDisabled()
.onSubmit {
addNewTag()
}
.focused($focusedField, equals: .new)
}
}
private func addNewTag() {
addTag(newTag.trimmingCharacters(in: .whitespaces).lowercased())
newTag = ""
focusedField = .new
}
private func addTag(_ tag: String) {
guard !tag.isEmpty,
!tags.contains(tag)
else { return }
tags.append(tag)
tags.sort()
}
private func deleteTag(_ tag: String) {
tags.removeAll(where: { $0 == tag })
}
}
private struct SymbolSearchResultsView: View {
@Binding var symbolQuery: String
@Binding var selectedSymbol: String
@State private var results: [String] = []
@FocusState.Binding var focusedField: EditTagGroupView.Focus?
var body: some View {
Group {
switch validationStatus {
case .valid:
ScrollView(.horizontal, showsIndicators: false) {
LazyHStack {
ForEach(results, id: \.self) { name in
Button {
results = TagGroup.searchSymbol(for: symbolQuery, exclude: selectedSymbol)
selectedSymbol = name
symbolQuery = name
focusedField = .new
} label: {
Image(systemName: name)
}
.buttonStyle(.plain)
}
}
.animation(.spring(duration: 0.2), value: results)
}
.onAppear {
results = TagGroup.searchSymbol(for: symbolQuery, exclude: selectedSymbol)
}
case let .invalid(description):
Text(description)
.font(.subheadline)
.foregroundStyle(.secondary)
}
}
.onAppear {
results = TagGroup.searchSymbol(for: symbolQuery, exclude: selectedSymbol)
}
.onChange(of: symbolQuery) {
results = TagGroup.searchSymbol(for: symbolQuery, exclude: selectedSymbol)
}
}
// MARK: search results validation
enum ValidationStatus: Equatable {
case valid
case invalid(description: LocalizedStringKey)
}
var validationStatus: ValidationStatus {
if results.isEmpty {
if symbolQuery == selectedSymbol,
!symbolQuery.isEmpty,
results.count == 0
{
.invalid(description: "\(symbolQuery) add-tag-groups.edit.tags.field.warning.search-results.already-selected")
} else {
.invalid(description: "add-tag-groups.edit.tags.field.warning.search-results.no-symbol-found")
}
} else {
.valid
}
}
}
extension TagGroup {
// MARK: title validation
enum TitleValidationStatus: Equatable {
case valid
case invalid(description: LocalizedStringKey)
}
var titleValidationStatus: TitleValidationStatus {
title.isEmpty
? .invalid(description: "add-tag-groups.edit.title.field.warning.empty-title")
: .valid
}
// MARK: symbolName validation
enum SymbolNameValidationStatus: Equatable {
case valid
case invalid(description: LocalizedStringKey)
}
var symbolNameValidationStatus: SymbolNameValidationStatus {
if symbolName.isEmpty {
return .invalid(description: "add-tag-groups.edit.title.field.warning.no-symbol-selected")
} else if !Self.allSymbols.contains(symbolName) {
return .invalid(description: "\(symbolName) add-tag-groups.edit.title.field.warning.invalid-sfsymbol-name")
}
return .valid
}
// MARK: tags validation
enum TagsValidationStatus: Equatable {
case valid
case invalid(description: LocalizedStringKey)
}
var tagsValidationStatus: TagsValidationStatus {
if tags.count < 2 {
return .invalid(description: "add-tag-groups.edit.tags.field.warning.number-of-tags")
}
return .valid
}
// MARK: TagGroup validation
var isValid: Bool {
titleValidationStatus == .valid
&& symbolNameValidationStatus == .valid
&& tagsValidationStatus == .valid
}
// MARK: format
func format() {
title = title.trimmingCharacters(in: .whitespacesAndNewlines)
tags = tags.map { $0.lowercased() }
}
// MARK: static members
static func emptyGroup() -> TagGroup {
TagGroup(title: "", symbolName: "", tags: [])
}
static func searchSymbol(for query: String, exclude excludedSymbol: String) -> [String] {
guard !query.isEmpty else { return [] }
return allSymbols.filter {
$0.contains(query) &&
$0 != excludedSymbol
}
}
static let allSymbols: [String] = SFSymbol.allSymbols.map { symbol in
symbol.rawValue
}
}
extension Text {
func warningLabel() -> Text {
font(.caption)
.foregroundStyle(.red)
}
}

View file

@ -4,14 +4,15 @@ import Env
import Models import Models
import Network import Network
import NukeUI import NukeUI
import Shimmer
import SwiftUI import SwiftUI
@MainActor
struct AddRemoteTimelineView: View { struct AddRemoteTimelineView: View {
@Environment(\.dismiss) private var dismiss @Environment(\.dismiss) private var dismiss
@Environment(\.modelContext) private var context
@EnvironmentObject private var preferences: UserPreferences @Environment(UserPreferences.self) private var preferences
@EnvironmentObject private var theme: Theme @Environment(Theme.self) private var theme
@State private var instanceName: String = "" @State private var instanceName: String = ""
@State private var instance: Instance? @State private var instance: Instance?
@ -36,44 +37,51 @@ struct AddRemoteTimelineView: View {
.foregroundColor(.green) .foregroundColor(.green)
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
} }
if !instanceName.isEmpty && instance == nil {
Label("timeline.\(instanceName)-not-valid", systemImage: "xmark.seal.fill")
.foregroundColor(.red)
.listRowBackground(theme.primaryBackgroundColor)
}
Button { Button {
guard instance != nil else { return } guard instance != nil else { return }
preferences.remoteLocalTimelines.append(instanceName) context.insert(LocalTimeline(instance: instanceName))
dismiss() dismiss()
} label: { } label: {
Text("timeline.add.action.add") Text("timeline.add.action.add")
} }
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)
.disabled(instance == nil)
instancesListView instancesListView
} }
.formStyle(.grouped) .formStyle(.grouped)
.navigationTitle("timeline.add-remote.title") .navigationTitle("timeline.add-remote.title")
.navigationBarTitleDisplayMode(.inline) .navigationBarTitleDisplayMode(.inline)
.scrollContentBackground(.hidden) #if !os(visionOS)
.background(theme.secondaryBackgroundColor) .scrollContentBackground(.hidden)
.scrollDismissesKeyboard(.immediately) .background(theme.secondaryBackgroundColor)
.toolbar { .scrollDismissesKeyboard(.immediately)
ToolbarItem(placement: .navigationBarLeading) { #endif
Button("action.cancel", action: { dismiss() }) .toolbar {
CancelToolbarItem()
} }
} .onChange(of: instanceName) { _, newValue in
.onChange(of: instanceName) { newValue in instanceNamePublisher.send(newValue)
instanceNamePublisher.send(newValue)
}
.onReceive(instanceNamePublisher.debounce(for: .milliseconds(500), scheduler: DispatchQueue.main)) { newValue in
Task {
let client = Client(server: newValue)
instance = try? await client.get(endpoint: Instances.instance)
} }
} .onReceive(instanceNamePublisher.debounce(for: .milliseconds(500), scheduler: DispatchQueue.main)) { newValue in
.onAppear { Task {
isInstanceURLFieldFocused = true let client = Client(server: newValue)
let client = InstanceSocialClient() instance = try? await client.get(endpoint: Instances.instance)
Task { }
self.instances = await client.fetchInstances() }
.onAppear {
isInstanceURLFieldFocused = true
let client = InstanceSocialClient()
let instanceName = instanceName
Task {
instances = await client.fetchInstances(keyword: instanceName)
}
} }
}
} }
} }
@ -85,7 +93,7 @@ struct AddRemoteTimelineView: View {
} else { } else {
ForEach(instanceName.isEmpty ? instances : instances.filter { $0.name.contains(instanceName.lowercased()) }) { instance in ForEach(instanceName.isEmpty ? instances : instances.filter { $0.name.contains(instanceName.lowercased()) }) { instance in
Button { Button {
self.instanceName = instance.name instanceName = instance.name
} label: { } label: {
VStack(alignment: .leading, spacing: 4) { VStack(alignment: .leading, spacing: 4) {
Text(instance.name) Text(instance.name)
@ -93,13 +101,13 @@ struct AddRemoteTimelineView: View {
.foregroundColor(.primary) .foregroundColor(.primary)
Text(instance.info?.shortDescription ?? "") Text(instance.info?.shortDescription ?? "")
.font(.scaledBody) .font(.scaledBody)
.foregroundColor(.gray) .foregroundStyle(Color.secondary)
(Text("instance.list.users-\(instance.users)") (Text("instance.list.users-\(instance.users)")
+ Text("") + Text("")
+ Text("instance.list.posts-\(instance.statuses)")) + Text("instance.list.posts-\(instance.statuses)"))
.font(.scaledFootnote) .font(.scaledFootnote)
.foregroundColor(.gray) .foregroundStyle(Color.secondary)
} }
} }
.listRowBackground(theme.primaryBackgroundColor) .listRowBackground(theme.primaryBackgroundColor)

View file

@ -1,216 +0,0 @@
import Combine
import DesignSystem
import Env
import Models
import Network
import NukeUI
import Shimmer
import SwiftUI
struct EditTagGroupView: View {
@Environment(\.dismiss) private var dismiss
@EnvironmentObject private var preferences: UserPreferences
@EnvironmentObject private var theme: Theme
@State private var title: String = ""
@State private var sfSymbolName: String = ""
@State private var tags: [String] = []
@State private var newTag: String = ""
@State private var popupTagsPresented = false
private var editingTagGroup: TagGroup?
private var onSaved: ((TagGroup) -> Void)?
private var canSave: Bool {
!title.isEmpty &&
// At least have 2 tags, one main and one additional.
tags.count >= 2
}
@FocusState private var focusedField: Focus?
enum Focus {
case title
case symbol
case new
}
init(editingTagGroup: TagGroup? = nil, onSaved: ((TagGroup) -> Void)? = nil) {
self.editingTagGroup = editingTagGroup
self.onSaved = onSaved
}
var body: some View {
NavigationStack {
ZStack(alignment: .bottom) {
Form {
metadataSection
keywordsSection
}
.formStyle(.grouped)
.navigationTitle(editingTagGroup != nil ? "timeline.filter.edit-tag-groups" : "timeline.filter.add-tag-groups")
.navigationBarTitleDisplayMode(.inline)
.scrollContentBackground(.hidden)
.background(theme.secondaryBackgroundColor)
.scrollDismissesKeyboard(.immediately)
.toolbar {
ToolbarItem(placement: .navigationBarLeading) {
Button("action.cancel", action: { dismiss() })
}
ToolbarItem(placement: .navigationBarTrailing) {
Button("action.save", action: { save() })
.disabled(!canSave)
}
}
symbolsSuggestionView
}
.onAppear {
focusedField = .title
if let editingTagGroup {
title = editingTagGroup.title
sfSymbolName = editingTagGroup.sfSymbolName
tags = editingTagGroup.tags
}
}
}
}
@ViewBuilder
private var metadataSection: some View {
Section {
TextField("add-tag-groups.edit.title.field", text: $title, axis: .horizontal)
.focused($focusedField, equals: Focus.title)
.onSubmit {
focusedField = Focus.symbol
}
HStack {
TextField("add-tag-groups.edit.icon.field", text: $sfSymbolName, axis: .horizontal)
.textInputAutocapitalization(.never)
.autocorrectionDisabled()
.focused($focusedField, equals: Focus.symbol)
.onSubmit {
focusedField = Focus.new
}
.onChange(of: sfSymbolName) { _ in
popupTagsPresented = true
}
Image(systemName: sfSymbolName)
}
}
.listRowBackground(theme.primaryBackgroundColor)
}
private var keywordsSection: some View {
Section("add-tag-groups.edit.tags") {
ForEach(tags, id: \.self) { tag in
HStack {
Text(tag)
Spacer()
Button {
deleteTag(tag)
} label: {
Image(systemName: "trash")
.tint(.red)
}
}
}
.onDelete { indexes in
if let index = indexes.first {
let tag = tags[index]
deleteTag(tag)
}
}
HStack {
TextField("add-tag-groups.edit.tags.add", text: $newTag, axis: .horizontal)
.textInputAutocapitalization(.never)
.autocorrectionDisabled()
.onSubmit {
addNewTag()
}
.focused($focusedField, equals: Focus.new)
Spacer()
if !newTag.isEmpty {
Button {
addNewTag()
} label: {
Image(systemName: "checkmark.circle.fill")
.tint(.green)
}
}
}
}
.listRowBackground(theme.primaryBackgroundColor)
}
private func addNewTag() {
addTag(newTag.trimmingCharacters(in: .whitespaces))
newTag = ""
focusedField = Focus.new
}
private func addTag(_ tag: String) {
guard !tag.isEmpty else { return }
tags.append(tag)
}
private func deleteTag(_ tag: String) {
tags.removeAll(where: { $0 == tag })
}
private func save() {
var toSave = tags
let main = toSave.removeFirst()
let tagGroup: TagGroup = .init(
title: title.trimmingCharacters(in: .whitespaces),
sfSymbolName: sfSymbolName,
main: main,
additional: toSave
)
if let editingTagGroup,
let index = preferences.tagGroups.firstIndex(of: editingTagGroup) {
preferences.tagGroups[index] = tagGroup
} else {
preferences.tagGroups.append(tagGroup)
}
dismiss()
onSaved?(tagGroup)
}
@ViewBuilder
private var symbolsSuggestionView: some View {
if focusedField == .symbol && !sfSymbolName.isEmpty {
let filteredMatches = allSymbols
.filter { $0.contains(sfSymbolName) }
if !filteredMatches.isEmpty {
ScrollView(.horizontal, showsIndicators: false) {
LazyHStack {
ForEach(filteredMatches, id: \.self) { symbolName in
Button {
sfSymbolName = symbolName
} label: {
Image(systemName: symbolName)
}
}
}
.padding(.horizontal, .layoutPadding)
}
.frame(height: 40)
.background(.ultraThinMaterial)
}
} else {
EmptyView()
}
}
}
struct AddTagGroupView_Previews: PreviewProvider {
static var previews: some View {
EditTagGroupView()
.withEnvironments()
}
}

View file

@ -1,13 +0,0 @@
//
// Symbols.swift
// IceCubesApp
//
// Created by Alejandro Martinez on 18/7/23.
//
import Foundation
import SFSafeSymbols
let allSymbols: [String] = SFSymbol.allSymbols.map { symbol in
symbol.rawValue
}

View file

@ -4,51 +4,60 @@ import DesignSystem
import Env import Env
import Models import Models
import Network import Network
import SwiftData
import SwiftUI import SwiftUI
import Timeline import Timeline
@MainActor
struct TimelineTab: View { struct TimelineTab: View {
@EnvironmentObject private var appAccount: AppAccountsManager @Environment(\.modelContext) private var context
@EnvironmentObject private var theme: Theme
@EnvironmentObject private var currentAccount: CurrentAccount @Environment(AppAccountsManager.self) private var appAccount
@EnvironmentObject private var preferences: UserPreferences @Environment(Theme.self) private var theme
@EnvironmentObject private var client: Client @Environment(CurrentAccount.self) private var currentAccount
@StateObject private var routerPath = RouterPath() @Environment(UserPreferences.self) private var preferences
@Binding var popToRootTab: Tab @Environment(Client.self) private var client
@State private var routerPath = RouterPath()
@State private var didAppear: Bool = false @State private var didAppear: Bool = false
@State private var timeline: TimelineFilter @State private var timeline: TimelineFilter = .home
@State private var scrollToTopSignal: Int = 0 @State private var selectedTagGroup: TagGroup?
@AppStorage("last_timeline_filter") public var lastTimelineFilter: TimelineFilter = .home @Query(sort: \LocalTimeline.creationDate, order: .reverse) var localTimelines: [LocalTimeline]
@Query(sort: \TagGroup.creationDate, order: .reverse) var tagGroups: [TagGroup]
@AppStorage("last_timeline_filter") var lastTimelineFilter: TimelineFilter = .home
@AppStorage("timeline_pinned_filters") private var pinnedFilters: [TimelineFilter] = []
private let canFilterTimeline: Bool private let canFilterTimeline: Bool
init(popToRootTab: Binding<Tab>, timeline: TimelineFilter? = nil) { init(timeline: TimelineFilter? = nil) {
canFilterTimeline = timeline == nil canFilterTimeline = timeline == nil
self.timeline = timeline ?? .home _timeline = .init(initialValue: timeline ?? .home)
_popToRootTab = popToRootTab
} }
var body: some View { var body: some View {
NavigationStack(path: $routerPath.path) { NavigationStack(path: $routerPath.path) {
TimelineView(timeline: $timeline, scrollToTopSignal: $scrollToTopSignal, canFilterTimeline: canFilterTimeline) TimelineView(timeline: $timeline,
pinnedFilters: $pinnedFilters,
selectedTagGroup: $selectedTagGroup,
canFilterTimeline: canFilterTimeline)
.withAppRouter() .withAppRouter()
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet) .withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
.toolbar { .toolbar {
toolbarView toolbarView
} }
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .navigationBar) .toolbarBackground(theme.primaryBackgroundColor.opacity(0.30), for: .navigationBar)
.id(client.id) .id(client.id)
} }
.onAppear { .onAppear {
routerPath.client = client routerPath.client = client
if !didAppear && canFilterTimeline { if !didAppear, canFilterTimeline {
didAppear = true didAppear = true
if client.isAuth { if client.isAuth {
timeline = lastTimelineFilter timeline = lastTimelineFilter
} else { } else {
timeline = .federated timeline = .trending
} }
} }
Task { Task {
@ -58,118 +67,59 @@ struct TimelineTab: View {
routerPath.presentedSheet = .addAccount routerPath.presentedSheet = .addAccount
} }
} }
.onChange(of: client.isAuth, perform: { _ in .onChange(of: client.isAuth) {
if client.isAuth { resetTimelineFilter()
timeline = lastTimelineFilter
} else {
timeline = .federated
}
})
.onChange(of: currentAccount.account?.id, perform: { _ in
if client.isAuth, canFilterTimeline {
timeline = lastTimelineFilter
} else {
timeline = .federated
}
})
.onChange(of: $popToRootTab.wrappedValue) { popToRootTab in
if popToRootTab == .timeline {
if routerPath.path.isEmpty {
scrollToTopSignal += 1
} else {
routerPath.path = []
}
}
} }
.onChange(of: client.id) { _ in .onChange(of: currentAccount.account?.id) {
resetTimelineFilter()
}
.onChange(of: client.id) {
routerPath.path = [] routerPath.path = []
} }
.onChange(of: timeline) { timeline in .onChange(of: timeline) { _, newValue in
if timeline == .home || timeline == .federated || timeline == .local { if client.isAuth, canFilterTimeline {
lastTimelineFilter = timeline lastTimelineFilter = newValue
}
switch newValue {
case let .tagGroup(title, _, _):
if let group = tagGroups.first(where: { $0.title == title }) {
selectedTagGroup = group
}
default:
selectedTagGroup = nil
} }
} }
.onReceive(NotificationCenter.default.publisher(for: .refreshTimeline)) { _ in
timeline = .latest
}
.onReceive(NotificationCenter.default.publisher(for: .trendingTimeline)) { _ in
timeline = .trending
}
.onReceive(NotificationCenter.default.publisher(for: .localTimeline)) { _ in
timeline = .local
}
.onReceive(NotificationCenter.default.publisher(for: .federatedTimeline)) { _ in
timeline = .federated
}
.onReceive(NotificationCenter.default.publisher(for: .homeTimeline)) { _ in
timeline = .home
}
.withSafariRouter() .withSafariRouter()
.environmentObject(routerPath) .environment(routerPath)
} }
@ViewBuilder @ViewBuilder
private var timelineFilterButton: some View { private var timelineFilterButton: some View {
if timeline.supportNewestPagination { headerGroup
Button { timelineFiltersButtons
self.timeline = .latest if client.isAuth {
} label: { listsFiltersButons
Label(TimelineFilter.latest.localizedTitle(), systemImage: TimelineFilter.latest.iconName() ?? "") tagsFiltersButtons
}
.keyboardShortcut("r", modifiers: .command)
Divider()
}
ForEach(TimelineFilter.availableTimeline(client: client), id: \.self) { timeline in
Button {
self.timeline = timeline
} label: {
Label(timeline.localizedTitle(), systemImage: timeline.iconName() ?? "")
}
}
if !currentAccount.lists.isEmpty {
Menu("timeline.filter.lists") {
ForEach(currentAccount.sortedLists) { list in
Button {
timeline = .list(list: list)
} label: {
Label(list.title, systemImage: "list.bullet")
}
}
}
}
if !currentAccount.tags.isEmpty {
Menu("timeline.filter.tags") {
ForEach(currentAccount.sortedTags) { tag in
Button {
timeline = .hashtag(tag: tag.name, accountId: nil)
} label: {
Label("#\(tag.name)", systemImage: "number")
}
}
}
}
Menu("timeline.filter.local") {
ForEach(preferences.remoteLocalTimelines, id: \.self) { server in
Button {
timeline = .remoteLocal(server: server, filter: .local)
} label: {
VStack {
Label(server, systemImage: "dot.radiowaves.right")
}
}
}
Button {
routerPath.presentedSheet = .addRemoteLocalTimeline
} label: {
Label("timeline.filter.add-local", systemImage: "badge.plus.radiowaves.right")
}
}
Menu("timeline.filter.tag-groups") {
ForEach(preferences.tagGroups, id: \.self) { group in
Button {
timeline = .tagGroup(group)
} label: {
VStack {
let icon = group.sfSymbolName.isEmpty ? "number" : group.sfSymbolName
Label(group.title, systemImage: icon)
}
}
}
Button {
routerPath.presentedSheet = .addTagGroup
} label: {
Label("timeline.filter.add-tag-groups", systemImage: "plus")
}
} }
localTimelinesFiltersButtons
tagGroupsFiltersButtons
Divider()
contentFilterButton
} }
private var addAccountButton: some View { private var addAccountButton: some View {
@ -189,17 +139,7 @@ struct TimelineTab: View {
} }
} }
if client.isAuth { if client.isAuth {
if UIDevice.current.userInterfaceIdiom != .pad { ToolbarTab(routerPath: $routerPath)
ToolbarItem(placement: .navigationBarLeading) {
AppAccountsSelectorView(routerPath: routerPath)
.id(currentAccount.account?.id)
}
}
statusEditorToolbarItem(routerPath: routerPath,
visibility: preferences.postVisibility)
if UIDevice.current.userInterfaceIdiom == .pad && !preferences.showiPadSecondaryColumn {
SecondaryColumnToolbarItem()
}
} else { } else {
ToolbarItem(placement: .navigationBarTrailing) { ToolbarItem(placement: .navigationBarTrailing) {
addAccountButton addAccountButton
@ -234,4 +174,149 @@ struct TimelineTab: View {
} }
} }
} }
@ViewBuilder
private var headerGroup: some View {
ControlGroup {
if timeline.supportNewestPagination {
Button {
timeline = .latest
} label: {
Label(TimelineFilter.latest.localizedTitle(), systemImage: TimelineFilter.latest.iconName())
}
}
if timeline == .home {
Button {
timeline = .resume
} label: {
VStack {
Label(TimelineFilter.resume.localizedTitle(),
systemImage: TimelineFilter.resume.iconName())
}
}
}
pinButton
}
}
@ViewBuilder
private var pinButton: some View {
let index = pinnedFilters.firstIndex(where: { $0.id == timeline.id })
Button {
withAnimation {
if let index {
let timeline = pinnedFilters.remove(at: index)
Telemetry.signal("timeline.pin.removed", parameters: ["timeline" : timeline.rawValue])
} else {
pinnedFilters.append(timeline)
Telemetry.signal("timeline.pin.added", parameters: ["timeline" : timeline.rawValue])
}
}
} label: {
if index != nil {
Label("status.action.unpin", systemImage: "pin.slash")
} else {
Label("status.action.pin", systemImage: "pin")
}
}
}
private var timelineFiltersButtons: some View {
ForEach(TimelineFilter.availableTimeline(client: client), id: \.self) { timeline in
Button {
self.timeline = timeline
} label: {
Label(timeline.localizedTitle(), systemImage: timeline.iconName())
}
}
}
@ViewBuilder
private var listsFiltersButons: some View {
Menu("timeline.filter.lists") {
Button {
routerPath.presentedSheet = .listCreate
} label: {
Label("account.list.create", systemImage: "plus")
}
ForEach(currentAccount.sortedLists) { list in
Button {
timeline = .list(list: list)
} label: {
Label(list.title, systemImage: "list.bullet")
}
}
}
}
@ViewBuilder
private var tagsFiltersButtons: some View {
if !currentAccount.tags.isEmpty {
Menu("timeline.filter.tags") {
ForEach(currentAccount.sortedTags) { tag in
Button {
timeline = .hashtag(tag: tag.name, accountId: nil)
} label: {
Label("#\(tag.name)", systemImage: "number")
}
}
}
}
}
private var localTimelinesFiltersButtons: some View {
Menu("timeline.filter.local") {
ForEach(localTimelines) { remoteLocal in
Button {
timeline = .remoteLocal(server: remoteLocal.instance, filter: .local)
} label: {
VStack {
Label(remoteLocal.instance, systemImage: "dot.radiowaves.right")
}
}
}
Button {
routerPath.presentedSheet = .addRemoteLocalTimeline
} label: {
Label("timeline.filter.add-local", systemImage: "badge.plus.radiowaves.right")
}
}
}
private var tagGroupsFiltersButtons: some View {
Menu("timeline.filter.tag-groups") {
ForEach(tagGroups) { group in
Button {
timeline = .tagGroup(title: group.title, tags: group.tags, symbolName: group.symbolName)
} label: {
VStack {
let icon = group.symbolName.isEmpty ? "number" : group.symbolName
Label(group.title, systemImage: icon)
}
}
}
Button {
routerPath.presentedSheet = .addTagGroup
} label: {
Label("timeline.filter.add-tag-groups", systemImage: "plus")
}
}
}
private var contentFilterButton: some View {
Button(action: {
routerPath.presentedSheet = .timelineContentFilter
}, label: {
Label("timeline.content-filter.title", systemSymbol: .line3HorizontalDecrease)
})
}
private func resetTimelineFilter() {
if client.isAuth, canFilterTimeline {
timeline = lastTimelineFilter
} else if !client.isAuth {
timeline = .trending
}
}
} }

View file

@ -0,0 +1,51 @@
import AppAccount
import DesignSystem
import Env
import SwiftUI
@MainActor
struct ToolbarTab: ToolbarContent {
@Environment(\.isSecondaryColumn) private var isSecondaryColumn: Bool
@Environment(\.horizontalSizeClass) private var horizontalSizeClass
@Environment(UserPreferences.self) private var userPreferences
@Environment(Theme.self) private var theme
@Binding var routerPath: RouterPath
var body: some ToolbarContent {
if !isSecondaryColumn {
if horizontalSizeClass == .regular {
ToolbarItem(placement: .topBarLeading) {
if UIDevice.current.userInterfaceIdiom == .pad || UIDevice.current.userInterfaceIdiom == .mac {
Button {
withAnimation {
userPreferences.isSidebarExpanded.toggle()
}
} label: {
if userPreferences.isSidebarExpanded {
Image(systemName: "sidebar.squares.left")
} else {
Image(systemName: "sidebar.left")
}
}
}
}
}
statusEditorToolbarItem(routerPath: routerPath,
visibility: userPreferences.postVisibility)
if UIDevice.current.userInterfaceIdiom != .pad ||
(UIDevice.current.userInterfaceIdiom == .pad && horizontalSizeClass == .compact)
{
ToolbarItem(placement: .navigationBarLeading) {
AppAccountsSelectorView(routerPath: routerPath, avatarConfig: theme.avatarShape == .circle ? .badge : .badgeRounded)
}
}
}
if UIDevice.current.userInterfaceIdiom == .pad && horizontalSizeClass == .regular {
if (!isSecondaryColumn && !userPreferences.showiPadSecondaryColumn) || isSecondaryColumn {
SecondaryColumnToolbarItem()
}
}
}
}

Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 22 KiB

BIN
IceCubesApp/Assets.xcassets/AppIcon.appiconset/128.png Executable file → Normal file

Binary file not shown.

Before

Width:  |  Height:  |  Size: 24 KiB

After

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 30 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 33 KiB

BIN
IceCubesApp/Assets.xcassets/AppIcon.appiconset/16.png Executable file → Normal file

Binary file not shown.

Before

Width:  |  Height:  |  Size: 819 B

After

Width:  |  Height:  |  Size: 1.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 39 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 45 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.1 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 47 KiB

BIN
IceCubesApp/Assets.xcassets/AppIcon.appiconset/256.png Executable file → Normal file

Binary file not shown.

Before

Width:  |  Height:  |  Size: 84 KiB

After

Width:  |  Height:  |  Size: 47 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.9 KiB

BIN
IceCubesApp/Assets.xcassets/AppIcon.appiconset/32.png Executable file → Normal file

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.3 KiB

After

Width:  |  Height:  |  Size: 2.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.3 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 4.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 147 KiB

BIN
IceCubesApp/Assets.xcassets/AppIcon.appiconset/512.png Executable file → Normal file

Binary file not shown.

Before

Width:  |  Height:  |  Size: 308 KiB

After

Width:  |  Height:  |  Size: 147 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6.3 KiB

BIN
IceCubesApp/Assets.xcassets/AppIcon.appiconset/64.png Executable file → Normal file

Binary file not shown.

Before

Width:  |  Height:  |  Size: 7.1 KiB

After

Width:  |  Height:  |  Size: 5.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 8.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 9.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 10 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 12 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 156 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 501 KiB

View file

@ -1,284 +1,98 @@
{ {
"images": [ "images" : [
{ {
"size": "60x60", "filename" : "AppIcon-fs8.png",
"expected-size": "180", "idiom" : "universal",
"filename": "180.png", "platform" : "ios",
"folder": "Assets.xcassets/AppIcon.appiconset/", "size" : "1024x1024"
"idiom": "iphone", },
"scale": "3x" {
}, "appearances" : [
{ {
"size": "40x40", "appearance" : "luminosity",
"expected-size": "80", "value" : "dark"
"filename": "80.png", }
"folder": "Assets.xcassets/AppIcon.appiconset/", ],
"idiom": "iphone", "filename" : "dark.png",
"scale": "2x" "idiom" : "universal",
}, "platform" : "ios",
{ "size" : "1024x1024"
"size": "40x40", },
"expected-size": "120", {
"filename": "120.png", "appearances" : [
"folder": "Assets.xcassets/AppIcon.appiconset/", {
"idiom": "iphone", "appearance" : "luminosity",
"scale": "3x" "value" : "tinted"
}, }
{ ],
"size": "60x60", "filename" : "tinted.png",
"expected-size": "120", "idiom" : "universal",
"filename": "120.png", "platform" : "ios",
"folder": "Assets.xcassets/AppIcon.appiconset/", "size" : "1024x1024"
"idiom": "iphone", },
"scale": "2x" {
}, "filename" : "16.png",
{ "idiom" : "mac",
"size": "57x57", "scale" : "1x",
"expected-size": "57", "size" : "16x16"
"filename": "57.png", },
"folder": "Assets.xcassets/AppIcon.appiconset/", {
"idiom": "iphone", "filename" : "32.png",
"scale": "1x" "idiom" : "mac",
}, "scale" : "2x",
{ "size" : "16x16"
"size": "29x29", },
"expected-size": "58", {
"filename": "58.png", "filename" : "32 1.png",
"folder": "Assets.xcassets/AppIcon.appiconset/", "idiom" : "mac",
"idiom": "iphone", "scale" : "1x",
"scale": "2x" "size" : "32x32"
}, },
{ {
"size": "29x29", "filename" : "64.png",
"expected-size": "29", "idiom" : "mac",
"filename": "29.png", "scale" : "2x",
"folder": "Assets.xcassets/AppIcon.appiconset/", "size" : "32x32"
"idiom": "iphone", },
"scale": "1x" {
}, "filename" : "128.png",
{ "idiom" : "mac",
"size": "29x29", "scale" : "1x",
"expected-size": "87", "size" : "128x128"
"filename": "87.png", },
"folder": "Assets.xcassets/AppIcon.appiconset/", {
"idiom": "iphone", "filename" : "256 1.png",
"scale": "3x" "idiom" : "mac",
}, "scale" : "2x",
{ "size" : "128x128"
"size": "57x57", },
"expected-size": "114", {
"filename": "114.png", "filename" : "256.png",
"folder": "Assets.xcassets/AppIcon.appiconset/", "idiom" : "mac",
"idiom": "iphone", "scale" : "1x",
"scale": "2x" "size" : "256x256"
}, },
{ {
"size": "20x20", "filename" : "512 1.png",
"expected-size": "40", "idiom" : "mac",
"filename": "40.png", "scale" : "2x",
"folder": "Assets.xcassets/AppIcon.appiconset/", "size" : "256x256"
"idiom": "iphone", },
"scale": "2x" {
}, "filename" : "512.png",
{ "idiom" : "mac",
"size": "20x20", "scale" : "1x",
"expected-size": "60", "size" : "512x512"
"filename": "60.png", },
"folder": "Assets.xcassets/AppIcon.appiconset/", {
"idiom": "iphone", "filename" : "Content.png",
"scale": "3x" "idiom" : "mac",
}, "scale" : "2x",
{ "size" : "512x512"
"size": "1024x1024", }
"filename": "1024.png", ],
"expected-size": "1024", "info" : {
"idiom": "ios-marketing", "author" : "xcode",
"folder": "Assets.xcassets/AppIcon.appiconset/", "version" : 1
"scale": "1x" }
}, }
{
"size": "40x40",
"expected-size": "80",
"filename": "80.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "2x"
},
{
"size": "72x72",
"expected-size": "72",
"filename": "72.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "1x"
},
{
"size": "76x76",
"expected-size": "152",
"filename": "152.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "2x"
},
{
"size": "50x50",
"expected-size": "100",
"filename": "100.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "2x"
},
{
"size": "29x29",
"expected-size": "58",
"filename": "58.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "2x"
},
{
"size": "76x76",
"expected-size": "76",
"filename": "76.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "1x"
},
{
"size": "29x29",
"expected-size": "29",
"filename": "29.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "1x"
},
{
"size": "50x50",
"expected-size": "50",
"filename": "50.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "1x"
},
{
"size": "72x72",
"expected-size": "144",
"filename": "144.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "2x"
},
{
"size": "40x40",
"expected-size": "40",
"filename": "40.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "1x"
},
{
"size": "83.5x83.5",
"expected-size": "167",
"filename": "167.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "2x"
},
{
"size": "20x20",
"expected-size": "20",
"filename": "20.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "1x"
},
{
"size": "20x20",
"expected-size": "40",
"filename": "40.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "ipad",
"scale": "2x"
},
{
"size": "128x128",
"expected-size": "128",
"filename": "128.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "1x"
},
{
"size": "256x256",
"expected-size": "256",
"filename": "256.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "1x"
},
{
"size": "128x128",
"expected-size": "256",
"filename": "256.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "2x"
},
{
"size": "256x256",
"expected-size": "512",
"filename": "512.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "2x"
},
{
"size": "32x32",
"expected-size": "32",
"filename": "32.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "1x"
},
{
"size": "512x512",
"expected-size": "512",
"filename": "512.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "1x"
},
{
"size": "16x16",
"expected-size": "16",
"filename": "16.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "1x"
},
{
"size": "16x16",
"expected-size": "32",
"filename": "32.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "2x"
},
{
"size": "32x32",
"expected-size": "64",
"filename": "64.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "2x"
},
{
"size": "512x512",
"expected-size": "1024",
"filename": "1024.png",
"folder": "Assets.xcassets/AppIcon.appiconset/",
"idiom": "mac",
"scale": "2x"
}
]
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 370 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 545 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 764 KiB

View file

@ -0,0 +1,13 @@
{
"images" : [
{
"filename" : "Background.png",
"idiom" : "vision",
"scale" : "2x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

View file

@ -0,0 +1,6 @@
{
"info" : {
"author" : "xcode",
"version" : 1
}
}

View file

@ -0,0 +1,17 @@
{
"info" : {
"author" : "xcode",
"version" : 1
},
"layers" : [
{
"filename" : "Front.solidimagestacklayer"
},
{
"filename" : "Mid.solidimagestacklayer"
},
{
"filename" : "Back.solidimagestacklayer"
}
]
}

View file

@ -0,0 +1,13 @@
{
"images" : [
{
"filename" : "Layer 1.png",
"idiom" : "vision",
"scale" : "2x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 549 KiB

View file

@ -0,0 +1,6 @@
{
"info" : {
"author" : "xcode",
"version" : 1
}
}

View file

@ -0,0 +1,12 @@
{
"images" : [
{
"idiom" : "vision",
"scale" : "2x"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

View file

@ -0,0 +1,6 @@
{
"info" : {
"author" : "xcode",
"version" : 1
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 79 KiB

View file

@ -1,7 +1,7 @@
{ {
"images" : [ "images" : [
{ {
"filename" : "icon-fs8.heic", "filename" : "AppIcon-fs8.png",
"idiom" : "universal", "idiom" : "universal",
"scale" : "1x" "scale" : "1x"
}, },

Binary file not shown.

After

Width:  |  Height:  |  Size: 156 KiB

View file

@ -0,0 +1,14 @@
{
"images" : [
{
"filename" : "AppIcon-fs8.png",
"idiom" : "universal",
"platform" : "ios",
"size" : "1024x1024"
}
],
"info" : {
"author" : "xcode",
"version" : 1
}
}

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