Code Coverage |
||||||||||
Lines |
Functions and Methods |
Classes and Traits |
||||||||
| Total | |
100.00% |
39 / 39 |
|
100.00% |
14 / 14 |
CRAP | |
100.00% |
1 / 1 |
| Features | |
100.00% |
39 / 39 |
|
100.00% |
14 / 14 |
33 | |
100.00% |
1 / 1 |
| enabled | |
100.00% |
3 / 3 |
|
100.00% |
1 / 1 |
1 | |||
| hasHtmlPages | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
| hasBladePages | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
| hasMarkdownPages | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
| hasMarkdownPosts | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
| hasDocumentationPages | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
| hasDocumentationSearch | |
100.00% |
3 / 3 |
|
100.00% |
1 / 1 |
3 | |||
| hasDarkmode | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
| hasThemeToggleButtons | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
2 | |||
| hasTorchlight | |
100.00% |
3 / 3 |
|
100.00% |
1 / 1 |
3 | |||
| htmlPages | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| bladePages | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| markdownPages | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| markdownPosts | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| documentationPages | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| documentationSearch | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| darkmode | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| torchlight | n/a |
0 / 0 |
n/a |
0 / 0 |
1 | |||||
| sitemap | |
100.00% |
3 / 3 |
|
100.00% |
1 / 1 |
3 | |||
| rss | |
100.00% |
5 / 5 |
|
100.00% |
1 / 1 |
5 | |||
| toArray | |
100.00% |
5 / 5 |
|
100.00% |
1 / 1 |
1 | |||
| getDefaultOptions | |
100.00% |
10 / 10 |
|
100.00% |
1 / 1 |
1 | |||
| 1 | <?php |
| 2 | |
| 3 | declare(strict_types=1); |
| 4 | |
| 5 | namespace Hyde\Facades; |
| 6 | |
| 7 | use Hyde\Hyde; |
| 8 | use Hyde\Enums\Feature; |
| 9 | use Hyde\Pages\MarkdownPost; |
| 10 | use Hyde\Pages\DocumentationPage; |
| 11 | use JetBrains\PhpStorm\Deprecated; |
| 12 | use Hyde\Support\Concerns\Serializable; |
| 13 | use Hyde\Support\Contracts\SerializableContract; |
| 14 | use Hyde\Framework\Concerns\Internal\MockableFeatures; |
| 15 | use Illuminate\Support\Str; |
| 16 | |
| 17 | use function get_class_methods; |
| 18 | use function extension_loaded; |
| 19 | use function str_starts_with; |
| 20 | use function in_array; |
| 21 | use function collect; |
| 22 | use function substr; |
| 23 | use function count; |
| 24 | use function app; |
| 25 | |
| 26 | /** |
| 27 | * Allows features to be enabled and disabled in a simple object-oriented manner. |
| 28 | * |
| 29 | * @internal Until this class is split into a service/manager class, it should not be used outside of Hyde as the API is subject to change. |
| 30 | * |
| 31 | * @todo Split facade logic to service/manager class. (Initial and mock data could be set with boot/set methods) |
| 32 | * Based entirely on Laravel Jetstream (License MIT) |
| 33 | * |
| 34 | * @see https://jetstream.laravel.com/ |
| 35 | */ |
| 36 | class Features implements SerializableContract |
| 37 | { |
| 38 | use Serializable; |
| 39 | use MockableFeatures; |
| 40 | |
| 41 | /** |
| 42 | * Determine if the given specified is enabled. |
| 43 | */ |
| 44 | public static function enabled(Feature $feature): bool |
| 45 | { |
| 46 | return static::resolveMockedInstance($feature->name) ?? in_array( |
| 47 | $feature, Config::getArray('hyde.features', static::getDefaultOptions()) |
| 48 | ); |
| 49 | } |
| 50 | |
| 51 | // ================================================ |
| 52 | // Determine if a given feature is enabled. |
| 53 | // ================================================ |
| 54 | |
| 55 | public static function hasHtmlPages(): bool |
| 56 | { |
| 57 | return static::enabled(Feature::HtmlPages); |
| 58 | } |
| 59 | |
| 60 | public static function hasBladePages(): bool |
| 61 | { |
| 62 | return static::enabled(Feature::BladePages); |
| 63 | } |
| 64 | |
| 65 | public static function hasMarkdownPages(): bool |
| 66 | { |
| 67 | return static::enabled(Feature::MarkdownPages); |
| 68 | } |
| 69 | |
| 70 | public static function hasMarkdownPosts(): bool |
| 71 | { |
| 72 | return static::enabled(Feature::MarkdownPosts); |
| 73 | } |
| 74 | |
| 75 | public static function hasDocumentationPages(): bool |
| 76 | { |
| 77 | return static::enabled(Feature::DocumentationPages); |
| 78 | } |
| 79 | |
| 80 | public static function hasDocumentationSearch(): bool |
| 81 | { |
| 82 | return static::enabled(Feature::DocumentationSearch) |
| 83 | && static::hasDocumentationPages() |
| 84 | && count(DocumentationPage::files()) > 0; |
| 85 | } |
| 86 | |
| 87 | public static function hasDarkmode(): bool |
| 88 | { |
| 89 | return static::enabled(Feature::Darkmode); |
| 90 | } |
| 91 | |
| 92 | public static function hasThemeToggleButtons(): bool |
| 93 | { |
| 94 | return static::hasDarkmode() && Config::getBool('hyde.theme_toggle_buttons', true); |
| 95 | } |
| 96 | |
| 97 | /** |
| 98 | * Torchlight is by default enabled automatically when an API token |
| 99 | * is set in the .env file but is disabled when running tests. |
| 100 | */ |
| 101 | public static function hasTorchlight(): bool |
| 102 | { |
| 103 | return static::enabled(Feature::Torchlight) |
| 104 | && (Config::getNullableString('torchlight.token') !== null) |
| 105 | && (app('env') !== 'testing'); |
| 106 | } |
| 107 | |
| 108 | // ================================================= |
| 109 | // Configure features to be used in the config file. |
| 110 | // ================================================= |
| 111 | |
| 112 | /** |
| 113 | * @codeCoverageIgnore Deprecated method. |
| 114 | * |
| 115 | * @deprecated This method will be removed in v2.0. Please use `Feature::HtmlPages` instead. |
| 116 | */ |
| 117 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::HtmlPages Enum case', replacement: 'Feature::HtmlPages', since: '1.6.0')] |
| 118 | public static function htmlPages(): Feature |
| 119 | { |
| 120 | return Feature::HtmlPages; |
| 121 | } |
| 122 | |
| 123 | /** |
| 124 | * @codeCoverageIgnore Deprecated method. |
| 125 | * |
| 126 | * @deprecated This method will be removed in v2.0. Please use `Feature::BladePages` instead. |
| 127 | */ |
| 128 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::BladePages Enum case', replacement: 'Feature::BladePages', since: '1.6.0')] |
| 129 | public static function bladePages(): Feature |
| 130 | { |
| 131 | return Feature::BladePages; |
| 132 | } |
| 133 | |
| 134 | /** |
| 135 | * @codeCoverageIgnore Deprecated method. |
| 136 | * |
| 137 | * @deprecated This method will be removed in v2.0. Please use `Feature::MarkdownPages` instead. |
| 138 | */ |
| 139 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::MarkdownPages Enum case', replacement: 'Feature::MarkdownPages', since: '1.6.0')] |
| 140 | public static function markdownPages(): Feature |
| 141 | { |
| 142 | return Feature::MarkdownPages; |
| 143 | } |
| 144 | |
| 145 | /** |
| 146 | * @codeCoverageIgnore Deprecated method. |
| 147 | * |
| 148 | * @deprecated This method will be removed in v2.0. Please use `Feature::MarkdownPosts` instead. |
| 149 | */ |
| 150 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::MarkdownPosts Enum case', replacement: 'Feature::MarkdownPosts', since: '1.6.0')] |
| 151 | public static function markdownPosts(): Feature |
| 152 | { |
| 153 | return Feature::MarkdownPosts; |
| 154 | } |
| 155 | |
| 156 | /** |
| 157 | * @codeCoverageIgnore Deprecated method. |
| 158 | * |
| 159 | * @deprecated This method will be removed in v2.0. Please use `Feature::DocumentationPages` instead. |
| 160 | */ |
| 161 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::DocumentationPages Enum case', replacement: 'Feature::DocumentationPages', since: '1.6.0')] |
| 162 | public static function documentationPages(): Feature |
| 163 | { |
| 164 | return Feature::DocumentationPages; |
| 165 | } |
| 166 | |
| 167 | /** |
| 168 | * @codeCoverageIgnore Deprecated method. |
| 169 | * |
| 170 | * @deprecated This method will be removed in v2.0. Please use `Feature::DocumentationSearch` instead. |
| 171 | */ |
| 172 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::DocumentationSearch Enum case', replacement: 'Feature::DocumentationSearch', since: '1.6.0')] |
| 173 | public static function documentationSearch(): Feature |
| 174 | { |
| 175 | return Feature::DocumentationSearch; |
| 176 | } |
| 177 | |
| 178 | /** |
| 179 | * @codeCoverageIgnore Deprecated method. |
| 180 | * |
| 181 | * @deprecated This method will be removed in v2.0. Please use `Feature::Darkmode` instead. |
| 182 | */ |
| 183 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::Darkmode Enum case', replacement: 'Feature::Darkmode', since: '1.6.0')] |
| 184 | public static function darkmode(): Feature |
| 185 | { |
| 186 | return Feature::Darkmode; |
| 187 | } |
| 188 | |
| 189 | /** |
| 190 | * @codeCoverageIgnore Deprecated method. |
| 191 | * |
| 192 | * @deprecated This method will be removed in v2.0. Please use `Feature::Torchlight` instead. |
| 193 | */ |
| 194 | #[Deprecated(reason: 'Replaced by the \Hyde\Enums\Feature::Torchlight Enum case', replacement: 'Feature::Torchlight', since: '1.6.0')] |
| 195 | public static function torchlight(): Feature |
| 196 | { |
| 197 | return Feature::Torchlight; |
| 198 | } |
| 199 | |
| 200 | // ==================================================== |
| 201 | // Dynamic features that in addition to being enabled |
| 202 | // in the config file, require preconditions to be met. |
| 203 | // ==================================================== |
| 204 | |
| 205 | /** Can a sitemap be generated? */ |
| 206 | public static function sitemap(): bool |
| 207 | { |
| 208 | return static::resolveMockedInstance('sitemap') ?? Hyde::hasSiteUrl() |
| 209 | && Config::getBool('hyde.generate_sitemap', true) |
| 210 | && extension_loaded('simplexml'); |
| 211 | } |
| 212 | |
| 213 | /** Can an RSS feed be generated? */ |
| 214 | public static function rss(): bool |
| 215 | { |
| 216 | return static::resolveMockedInstance('rss') ?? Hyde::hasSiteUrl() |
| 217 | && static::hasMarkdownPosts() |
| 218 | && Config::getBool('hyde.rss.enabled', true) |
| 219 | && extension_loaded('simplexml') |
| 220 | && count(MarkdownPost::files()) > 0; |
| 221 | } |
| 222 | |
| 223 | /** |
| 224 | * Get an array representation of the features and their status. |
| 225 | * |
| 226 | * @return array<string, bool> |
| 227 | * |
| 228 | * @example ['html-pages' => true, 'markdown-pages' => false, ...] |
| 229 | */ |
| 230 | public function toArray(): array |
| 231 | { |
| 232 | return collect(get_class_methods(static::class)) |
| 233 | ->filter(fn (string $method): bool => str_starts_with($method, 'has')) |
| 234 | ->mapWithKeys(fn (string $method): array => [ |
| 235 | Str::kebab(substr($method, 3)) => static::{$method}(), |
| 236 | ])->toArray(); |
| 237 | } |
| 238 | |
| 239 | protected static function getDefaultOptions(): array |
| 240 | { |
| 241 | return [ |
| 242 | // Page Modules |
| 243 | Feature::HtmlPages, |
| 244 | Feature::MarkdownPosts, |
| 245 | Feature::BladePages, |
| 246 | Feature::MarkdownPages, |
| 247 | Feature::DocumentationPages, |
| 248 | |
| 249 | // Frontend Features |
| 250 | Feature::Darkmode, |
| 251 | Feature::DocumentationSearch, |
| 252 | |
| 253 | // Integrations |
| 254 | Feature::Torchlight, |
| 255 | ]; |
| 256 | } |
| 257 | } |