1use gpui::{AppContext, Global, Subscription, ViewContext};
2
3#[derive(Default)]
4struct FeatureFlags {
5 flags: Vec<String>,
6 staff: bool,
7}
8
9impl FeatureFlags {
10 fn has_flag(&self, flag: &str) -> bool {
11 self.staff || self.flags.iter().any(|f| f.as_str() == flag)
12 }
13}
14
15impl Global for FeatureFlags {}
16
17/// To create a feature flag, implement this trait on a trivial type and use it as
18/// a generic parameter when called [`FeatureFlagAppExt::has_flag`].
19///
20/// Feature flags are always enabled for members of Zed staff. To disable this behavior
21/// so you can test flags being disabled, set ZED_DISABLE_STAFF=1 in your environment,
22/// which will force Zed to treat the current user as non-staff.
23pub trait FeatureFlag {
24 const NAME: &'static str;
25}
26
27pub struct Remoting {}
28impl FeatureFlag for Remoting {
29 const NAME: &'static str = "remoting";
30}
31
32pub struct TerminalInlineAssist {}
33impl FeatureFlag for TerminalInlineAssist {
34 const NAME: &'static str = "terminal-inline-assist";
35}
36
37pub trait FeatureFlagViewExt<V: 'static> {
38 fn observe_flag<T: FeatureFlag, F>(&mut self, callback: F) -> Subscription
39 where
40 F: Fn(bool, &mut V, &mut ViewContext<V>) + Send + Sync + 'static;
41}
42
43impl<V> FeatureFlagViewExt<V> for ViewContext<'_, V>
44where
45 V: 'static,
46{
47 fn observe_flag<T: FeatureFlag, F>(&mut self, callback: F) -> Subscription
48 where
49 F: Fn(bool, &mut V, &mut ViewContext<V>) + 'static,
50 {
51 self.observe_global::<FeatureFlags>(move |v, cx| {
52 let feature_flags = cx.global::<FeatureFlags>();
53 callback(feature_flags.has_flag(<T as FeatureFlag>::NAME), v, cx);
54 })
55 }
56}
57
58pub trait FeatureFlagAppExt {
59 fn update_flags(&mut self, staff: bool, flags: Vec<String>);
60 fn set_staff(&mut self, staff: bool);
61 fn has_flag<T: FeatureFlag>(&self) -> bool;
62 fn is_staff(&self) -> bool;
63}
64
65impl FeatureFlagAppExt for AppContext {
66 fn update_flags(&mut self, staff: bool, flags: Vec<String>) {
67 let feature_flags = self.default_global::<FeatureFlags>();
68 feature_flags.staff = staff;
69 feature_flags.flags = flags;
70 }
71
72 fn set_staff(&mut self, staff: bool) {
73 let feature_flags = self.default_global::<FeatureFlags>();
74 feature_flags.staff = staff;
75 }
76
77 fn has_flag<T: FeatureFlag>(&self) -> bool {
78 self.try_global::<FeatureFlags>()
79 .map(|flags| flags.has_flag(T::NAME))
80 .unwrap_or(false)
81 }
82
83 fn is_staff(&self) -> bool {
84 self.try_global::<FeatureFlags>()
85 .map(|flags| flags.staff)
86 .unwrap_or(false)
87 }
88}