Bluesky app fork with some witchin' additions 馃挮
witchsky.app
bluesky
fork
client
1import {type Insets, Platform} from 'react-native'
2import {type AppBskyActorDefs, BSKY_LABELER_DID} from '@atproto/api'
3
4import {type ProxyHeaderValue} from '#/state/session/agent'
5import {BLUESKY_PROXY_DID, CHAT_PROXY_DID} from '#/env'
6
7export const LOCAL_DEV_SERVICE =
8 Platform.OS === 'android' ? 'http://10.0.2.2:2583' : 'http://localhost:2583'
9export const STAGING_SERVICE = 'https://staging.bsky.dev'
10export const BSKY_SERVICE = 'https://bsky.social'
11export const BSKY_SERVICE_DID = 'did:web:bsky.social'
12export const PUBLIC_BSKY_SERVICE = 'https://public.api.bsky.app'
13export const DEFAULT_SERVICE = BSKY_SERVICE
14export const HELP_DESK_URL = `https://tangled.org/jollywhoppers.com/witchsky.app/`
15export const EMBED_SERVICE = 'https://embed.bsky.app'
16export const EMBED_SCRIPT = `${EMBED_SERVICE}/static/embed.js`
17export const BSKY_DOWNLOAD_URL = 'https://bsky.app/download'
18export const STARTER_PACK_MAX_SIZE = 150
19export const CARD_ASPECT_RATIO = 1200 / 630
20
21// HACK
22// Yes, this is exactly what it looks like. It's a hard-coded constant
23// reflecting the number of new users in the last week. We don't have
24// time to add a route to the servers for this so we're just going to hard
25// code and update this number with each release until we can get the
26// server route done.
27// -prf
28export const JOINED_THIS_WEEK = 560000 // estimate as of 12/18/24
29
30export const DISCOVER_DEBUG_DIDS: Record<string, true> = {
31 'did:plc:oisofpd7lj26yvgiivf3lxsi': true, // hailey.at
32 'did:plc:p2cp5gopk7mgjegy6wadk3ep': true, // samuel.bsky.team
33 'did:plc:ragtjsm2j2vknwkz3zp4oxrd': true, // pfrazee.com
34 'did:plc:vpkhqolt662uhesyj6nxm7ys': true, // why.bsky.team
35 'did:plc:3jpt2mvvsumj2r7eqk4gzzjz': true, // esb.lol
36 'did:plc:vjug55kidv6sye7ykr5faxxn': true, // emilyliu.me
37 'did:plc:tgqseeot47ymot4zro244fj3': true, // iwsmith.bsky.social
38 'did:plc:2dzyut5lxna5ljiaasgeuffz': true, // darrin.bsky.team
39}
40
41const BASE_FEEDBACK_FORM_URL = `${HELP_DESK_URL}`
42export function FEEDBACK_FORM_URL(_params: {
43 email?: string
44 handle?: string
45}): string {
46 return BASE_FEEDBACK_FORM_URL
47}
48
49export const MAX_DISPLAY_NAME = 64
50export const MAX_DESCRIPTION = 256
51
52export const MAX_GRAPHEME_LENGTH = 300
53
54export const MAX_DRAFT_GRAPHEME_LENGTH = 1000
55
56export const MAX_DM_GRAPHEME_LENGTH = 1000
57
58// Recommended is 100 per: https://www.w3.org/WAI/GL/WCAG20/tests/test3.html
59// but increasing limit per user feedback
60export const MAX_ALT_TEXT = 2000
61export const DEFAULT_ALT_TEXT_AI_MODEL = 'google/gemma-3-27b-it:free'
62
63export const MAX_REPORT_REASON_GRAPHEME_LENGTH = 2000
64
65export function IS_TEST_USER(handle?: string) {
66 return handle && handle?.endsWith('.test')
67}
68
69export function IS_PROD_SERVICE(url?: string) {
70 return url && url !== STAGING_SERVICE && !url.startsWith(LOCAL_DEV_SERVICE)
71}
72
73export const PROD_DEFAULT_FEED = (rkey: string) =>
74 `at://did:plc:z72i7hdynmk6r22z27h6tvur/app.bsky.feed.generator/${rkey}`
75
76export const STAGING_DEFAULT_FEED = (rkey: string) =>
77 `at://did:plc:yofh3kx63drvfljkibw5zuxo/app.bsky.feed.generator/${rkey}`
78
79export const PROD_FEEDS = [
80 `feedgen|${PROD_DEFAULT_FEED('whats-hot')}`,
81 `feedgen|${PROD_DEFAULT_FEED('thevids')}`,
82]
83
84export const STAGING_FEEDS = [
85 `feedgen|${STAGING_DEFAULT_FEED('whats-hot')}`,
86 `feedgen|${STAGING_DEFAULT_FEED('thevids')}`,
87]
88
89export const POST_IMG_MAX = {
90 width: 2000,
91 height: 2000,
92 size: 1000000,
93}
94
95export const STAGING_LINK_META_PROXY =
96 'https://cardyb.staging.bsky.dev/v1/extract?url='
97
98export const PROD_LINK_META_PROXY = 'https://cardyb.bsky.app/v1/extract?url='
99
100export function LINK_META_PROXY(serviceUrl: string) {
101 if (IS_PROD_SERVICE(serviceUrl)) {
102 return PROD_LINK_META_PROXY
103 }
104
105 return STAGING_LINK_META_PROXY
106}
107
108export const STATUS_PAGE_URL = 'https://status.bsky.app/'
109
110// Hitslop constants
111export const createHitslop = (size: number): Insets => ({
112 top: size,
113 left: size,
114 bottom: size,
115 right: size,
116})
117export const HITSLOP_10 = createHitslop(10)
118export const HITSLOP_20 = createHitslop(20)
119export const HITSLOP_30 = createHitslop(30)
120export const LANG_DROPDOWN_HITSLOP = {top: 10, bottom: 10, left: 4, right: 4}
121export const BACK_HITSLOP = HITSLOP_30
122export const MAX_POST_LINES = 25
123
124export const BSKY_APP_ACCOUNT_DID = 'did:plc:z72i7hdynmk6r22z27h6tvur'
125
126export const BSKY_FEED_OWNER_DIDS = [
127 BSKY_APP_ACCOUNT_DID,
128 'did:plc:vpkhqolt662uhesyj6nxm7ys',
129 'did:plc:q6gjnaw2blty4crticxkmujt',
130]
131
132export const DISCOVER_FEED_URI =
133 'at://did:plc:z72i7hdynmk6r22z27h6tvur/app.bsky.feed.generator/whats-hot'
134export const VIDEO_FEED_URI =
135 'at://did:plc:z72i7hdynmk6r22z27h6tvur/app.bsky.feed.generator/thevids'
136export const STAGING_VIDEO_FEED_URI =
137 'at://did:plc:yofh3kx63drvfljkibw5zuxo/app.bsky.feed.generator/thevids'
138export const VIDEO_FEED_URIS = [VIDEO_FEED_URI, STAGING_VIDEO_FEED_URI]
139export const DISCOVER_SAVED_FEED = {
140 type: 'feed',
141 value: DISCOVER_FEED_URI,
142 pinned: true,
143}
144export const TIMELINE_SAVED_FEED = {
145 type: 'timeline',
146 value: 'following',
147 pinned: true,
148}
149export const VIDEO_SAVED_FEED = {
150 type: 'feed',
151 value: VIDEO_FEED_URI,
152 pinned: true,
153}
154
155export const RECOMMENDED_SAVED_FEEDS: Pick<
156 AppBskyActorDefs.SavedFeed,
157 'type' | 'value' | 'pinned'
158>[] = [DISCOVER_SAVED_FEED, TIMELINE_SAVED_FEED]
159
160export const KNOWN_SHUTDOWN_FEEDS = [
161 'at://did:plc:wqowuobffl66jv3kpsvo7ak4/app.bsky.feed.generator/the-algorithm', // for you by skygaze
162]
163
164export const GIF_SERVICE = 'https://gifs.bsky.app'
165
166export const GIF_SEARCH = (params: string) =>
167 `${GIF_SERVICE}/tenor/v2/search?${params}`
168export const GIF_FEATURED = (params: string) =>
169 `${GIF_SERVICE}/tenor/v2/featured?${params}`
170
171export const MAX_LABELERS = 20
172
173export const VIDEO_SERVICE = 'https://video.bsky.app'
174export const VIDEO_SERVICE_DID = 'did:web:video.bsky.app'
175
176export const VIDEO_MAX_DURATION_MS = 3 * 60 * 1000 // 3 minutes in milliseconds
177/**
178 * Maximum size of a video in megabytes, _not_ mebibytes. Backend uses
179 * ISO megabytes.
180 */
181export const VIDEO_MAX_SIZE = 1000 * 1000 * 100 // 100mb
182
183export const SUPPORTED_MIME_TYPES = [
184 'video/mp4',
185 'video/mpeg',
186 'video/webm',
187 'video/quicktime',
188 'image/gif',
189] as const
190
191export type SupportedMimeTypes = (typeof SUPPORTED_MIME_TYPES)[number]
192
193export const EMOJI_REACTION_LIMIT = 5
194
195export const urls = {
196 website: {
197 blog: {
198 findFriendsAnnouncement:
199 'https://bsky.social/about/blog/12-16-2025-find-friends',
200 initialVerificationAnnouncement: `https://bsky.social/about/blog/04-21-2025-verification`,
201 searchTipsAndTricks: 'https://bsky.social/about/blog/05-31-2024-search',
202 },
203 support: {
204 findFriendsPrivacyPolicy:
205 'https://bsky.social/about/support/find-friends-privacy-policy',
206 },
207 },
208}
209
210export const PUBLIC_APPVIEW = 'https://api.bsky.app'
211export const PUBLIC_APPVIEW_DID = 'did:web:api.bsky.app'
212export const PUBLIC_STAGING_APPVIEW_DID = 'did:web:api.staging.bsky.dev'
213
214export const DEV_ENV_APPVIEW = `http://localhost:2584` // always the same
215export const DEV_ENV_APPVIEW_DID = `did:plc:dw4kbjf5mn7nhenabiqpkyh3` // always the same
216
217// temp hack for e2e - esb
218export const BLUESKY_PROXY_HEADER = {
219 value: `${BLUESKY_PROXY_DID}#bsky_appview`,
220 get() {
221 return this.value as ProxyHeaderValue
222 },
223 set(value: string) {
224 this.value = value
225 },
226}
227
228export const DM_SERVICE_HEADERS = {
229 'atproto-proxy': `${CHAT_PROXY_DID}#bsky_chat`,
230}
231
232export const BLUESKY_MOD_SERVICE_HEADERS = {
233 'atproto-proxy': `${BSKY_LABELER_DID}#atproto_labeler`,
234}
235
236export const BLUESKY_NOTIF_SERVICE_HEADERS = {
237 'atproto-proxy': `${BLUESKY_PROXY_DID}#bsky_notif`,
238}
239
240export const webLinks = {
241 tos: `https://bsky.social/about/support/tos`,
242 privacy: `https://bsky.social/about/support/privacy-policy`,
243 community: `https://bsky.social/about/support/community-guidelines`,
244 communityDeprecated: `https://bsky.social/about/support/community-guidelines-deprecated`,
245}