mirror of
https://github.com/skyline-emu/skyline.git
synced 2025-01-26 18:27:54 +03:00
Implement Display Settings + Attach Texture to FenceCycle
This commit is contained in:
parent
b2bb855a02
commit
e511e158e3
@ -19,6 +19,8 @@ namespace skyline {
|
||||
PREF_ELEM("log_level", logLevel, static_cast<Logger::LogLevel>(element.text().as_uint(static_cast<unsigned int>(Logger::LogLevel::Info)))),
|
||||
PREF_ELEM("username_value", username, element.text().as_string()),
|
||||
PREF_ELEM("operation_mode", operationMode, element.attribute("value").as_bool()),
|
||||
PREF_ELEM("force_triple_buffering", forceTripleBuffering, element.attribute("value").as_bool()),
|
||||
PREF_ELEM("disable_frame_throttling", disableFrameThrottling, element.attribute("value").as_bool()),
|
||||
};
|
||||
|
||||
#undef PREF_ELEM
|
||||
|
@ -14,7 +14,8 @@ namespace skyline {
|
||||
Logger::LogLevel logLevel; //!< The minimum level that logs need to be for them to be printed
|
||||
std::string username; //!< The name set by the user to be supplied to the guest
|
||||
bool operationMode; //!< If the emulated Switch should be handheld or docked
|
||||
bool forceTripleBuffering{true}; //!< If the presentation should always triple buffer even if the game double buffers
|
||||
bool forceTripleBuffering; //!< If the presentation engine should always triple buffer even if the swapchain supports double buffering
|
||||
bool disableFrameThrottling; //!< Allow the guest to submit frames without any blocking calls
|
||||
|
||||
/**
|
||||
* @param fd An FD to the preference XML file
|
||||
|
@ -114,35 +114,20 @@ namespace skyline::gpu {
|
||||
*/
|
||||
void AttachObjects(std::initializer_list<std::shared_ptr<FenceCycleDependency>> dependencies) {
|
||||
if (!signalled.test(std::memory_order_consume)) {
|
||||
{
|
||||
auto it{dependencies.begin()};
|
||||
while (it != dependencies.end()) {
|
||||
auto next{std::next(it)};
|
||||
auto it{dependencies.begin()}, next{std::next(it)};
|
||||
if (it != dependencies.end()) {
|
||||
while (next != dependencies.end()) {
|
||||
(*it)->next = *next;
|
||||
it = next;
|
||||
next = std::next(next);
|
||||
}
|
||||
}
|
||||
|
||||
const auto &first{*dependencies.begin()};
|
||||
const auto &last{*dependencies.end()};
|
||||
std::shared_ptr<FenceCycleDependency> next{std::atomic_load_explicit(&list, std::memory_order_consume)};
|
||||
do {
|
||||
last->next = next;
|
||||
if (!next && signalled.test(std::memory_order_consume)) {
|
||||
std::shared_ptr<FenceCycleDependency> current{first};
|
||||
while (current) {
|
||||
next.swap(first->next);
|
||||
current.swap(next);
|
||||
next.reset();
|
||||
}
|
||||
return;
|
||||
}
|
||||
} while (std::atomic_compare_exchange_strong(&list, &next, first));
|
||||
AttachObject(*dependencies.begin());
|
||||
}
|
||||
}
|
||||
|
||||
template<typename... Dependencies>
|
||||
void AttachObjects(Dependencies... dependencies) {
|
||||
void AttachObjects(Dependencies &&... dependencies) {
|
||||
AttachObjects(std::initializer_list<std::shared_ptr<FenceCycleDependency>>{std::forward<Dependencies>(dependencies)...});
|
||||
}
|
||||
};
|
||||
|
@ -69,7 +69,7 @@ namespace skyline::gpu {
|
||||
}
|
||||
|
||||
void PresentationEngine::UpdateSwapchain(texture::Format format, texture::Dimensions extent) {
|
||||
auto minImageCount{std::max(vkSurfaceCapabilities.minImageCount, state.settings->forceTripleBuffering ? 3U : 0U)};
|
||||
auto minImageCount{std::max(vkSurfaceCapabilities.minImageCount, state.settings->forceTripleBuffering ? 3U : 2U)};
|
||||
if (minImageCount > MaxSwapchainImageCount)
|
||||
throw exception("Requesting swapchain with higher image count ({}) than maximum slot count ({})", minImageCount, MaxSwapchainImageCount);
|
||||
|
||||
@ -99,13 +99,14 @@ namespace skyline::gpu {
|
||||
.imageUsage = presentUsage,
|
||||
.imageSharingMode = vk::SharingMode::eExclusive,
|
||||
.compositeAlpha = vk::CompositeAlphaFlagBitsKHR::eInherit,
|
||||
.presentMode = vk::PresentModeKHR::eMailbox,
|
||||
.presentMode = state.settings->disableFrameThrottling ? vk::PresentModeKHR::eMailbox : vk::PresentModeKHR::eFifo,
|
||||
.clipped = true,
|
||||
});
|
||||
|
||||
auto vkImages{vkSwapchain->getImages()};
|
||||
if (vkImages.size() > MaxSwapchainImageCount)
|
||||
throw exception("Swapchain has higher image count ({}) than maximum slot count ({})", minImageCount, MaxSwapchainImageCount);
|
||||
state.logger->Error("Buffer Count: {}", vkImages.size());
|
||||
|
||||
for (size_t index{}; index < vkImages.size(); index++) {
|
||||
auto &slot{images[index]};
|
||||
|
@ -262,8 +262,7 @@ namespace skyline::gpu {
|
||||
},
|
||||
});
|
||||
});
|
||||
|
||||
cycle->AttachObject(stagingBuffer);
|
||||
cycle->AttachObjects(stagingBuffer, shared_from_this());
|
||||
}
|
||||
}
|
||||
|
||||
@ -376,6 +375,6 @@ namespace skyline::gpu {
|
||||
},
|
||||
});
|
||||
});
|
||||
cycle->AttachObject(source);
|
||||
cycle->AttachObjects(source, shared_from_this());
|
||||
}
|
||||
}
|
||||
|
@ -224,7 +224,7 @@ namespace skyline::gpu {
|
||||
* @brief A texture which is backed by host constructs while being synchronized with the underlying guest texture
|
||||
* @note This class conforms to the Lockable and BasicLockable C++ named requirements
|
||||
*/
|
||||
class Texture : public FenceCycleDependency {
|
||||
class Texture : public std::enable_shared_from_this<Texture>, public FenceCycleDependency {
|
||||
private:
|
||||
GPU &gpu;
|
||||
std::mutex mutex; //!< Synchronizes any mutations to the texture or its backing
|
||||
|
@ -212,7 +212,10 @@ class EmulationActivity : AppCompatActivity(), SurfaceHolder.Callback, View.OnTo
|
||||
}
|
||||
|
||||
@Suppress("DEPRECATION") val display = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) display!! else windowManager.defaultDisplay
|
||||
display?.supportedModes?.maxByOrNull { it.refreshRate * it.physicalHeight * it.physicalWidth }?.let { window.attributes.preferredDisplayModeId = it.modeId }
|
||||
if (settings.maxRefreshRate)
|
||||
display?.supportedModes?.maxByOrNull { it.refreshRate * it.physicalHeight * it.physicalWidth }?.let { window.attributes.preferredDisplayModeId = it.modeId }
|
||||
else
|
||||
display?.supportedModes?.minByOrNull { abs(it.refreshRate - 60f) }?.let { window.attributes.preferredDisplayModeId = it.modeId }
|
||||
|
||||
binding.gameView.setOnTouchListener(this)
|
||||
|
||||
|
@ -35,4 +35,6 @@ class Settings @Inject constructor(@ApplicationContext private val context : Con
|
||||
var logLevel by sharedPreferences(context, "3")
|
||||
|
||||
var filter by sharedPreferences(context, 0)
|
||||
|
||||
var maxRefreshRate by sharedPreferences(context, false)
|
||||
}
|
||||
|
@ -25,7 +25,7 @@
|
||||
<string name="file_missing">The log file was not found</string>
|
||||
<string name="upload_logs">The logs are being uploaded</string>
|
||||
<string name="cleared">The logs have been cleared</string>
|
||||
<!-- Settings -->
|
||||
<!-- Settings - Emulator -->
|
||||
<string name="emulator">Emulator</string>
|
||||
<string name="search_location">Search Location</string>
|
||||
<string name="theme">Theme</string>
|
||||
@ -40,17 +40,30 @@
|
||||
<string name="log_compact">Compact Logs</string>
|
||||
<string name="log_compact_desc_on">Logs will be displayed in a compact form factor</string>
|
||||
<string name="log_compact_desc_off">Logs will be displayed in a verbose form factor</string>
|
||||
<!-- Settings - System -->
|
||||
<string name="system">System</string>
|
||||
<string name="use_docked">Use Docked Mode</string>
|
||||
<string name="handheld_enabled">The system will emulate being in handheld mode</string>
|
||||
<string name="docked_enabled">The system will emulate being in docked mode</string>
|
||||
<string name="username">Username</string>
|
||||
<string name="username_default">@string/app_name</string>
|
||||
<!-- Settings - Keys -->
|
||||
<string name="keys">Keys</string>
|
||||
<string name="prod_keys">Production Keys</string>
|
||||
<string name="title_keys">Title Keys</string>
|
||||
<string name="import_keys_success">Successfully imported keys</string>
|
||||
<string name="import_keys_failed">Failed to import keys</string>
|
||||
<!-- Settings - Display -->
|
||||
<string name="display">Display</string>
|
||||
<string name="force_triple_buffering">Force Triple Buffering</string>
|
||||
<string name="triple_buffering_enabled">Utilize at least 3 swapchain buffers (Higher FPS with higher input lag)</string>
|
||||
<string name="triple_buffering_disabled">Utilize at least 2 swapchain buffers (Lower FPS with lower input lag)</string>
|
||||
<string name="disable_frame_throttling">Disable Frame Throttling</string>
|
||||
<string name="disable_frame_throttling_enabled">Game is allowed to submit frames as fast as possible (Only for benchmarking)</string>
|
||||
<string name="disable_frame_throttling_disabled">Only allow the game to submit frames at the display refresh rate</string>
|
||||
<string name="max_refresh_rate">Use Maximum Display Refresh Rate</string>
|
||||
<string name="max_refresh_rate_enabled">Sets the display refresh rate as high as possible (Will break most games)</string>
|
||||
<string name="max_refresh_rate_disabled">Sets the display refresh rate to 60Hz</string>
|
||||
<!-- Input -->
|
||||
<string name="input">Input</string>
|
||||
<string name="osc">On-Screen Controls</string>
|
||||
|
@ -1,90 +1,105 @@
|
||||
<androidx.preference.PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:app="http://schemas.android.com/apk/res-auto">
|
||||
xmlns:app="http://schemas.android.com/apk/res-auto">
|
||||
<PreferenceCategory
|
||||
android:key="category_emulator"
|
||||
android:title="@string/emulator">
|
||||
android:key="category_emulator"
|
||||
android:title="@string/emulator">
|
||||
<emu.skyline.preference.FolderPickerPreference
|
||||
app:key="search_location"
|
||||
app:title="@string/search_location" />
|
||||
app:key="search_location"
|
||||
app:title="@string/search_location" />
|
||||
<emu.skyline.preference.ThemePreference
|
||||
android:defaultValue="2"
|
||||
android:entries="@array/app_theme"
|
||||
android:entryValues="@array/app_theme_val"
|
||||
app:key="app_theme"
|
||||
app:title="@string/theme"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
android:defaultValue="2"
|
||||
android:entries="@array/app_theme"
|
||||
android:entryValues="@array/app_theme_val"
|
||||
app:key="app_theme"
|
||||
app:title="@string/theme"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
<ListPreference
|
||||
android:defaultValue="1"
|
||||
android:entries="@array/layout_type"
|
||||
android:entryValues="@array/layout_type_val"
|
||||
app:key="layout_type"
|
||||
app:title="@string/layout_type"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
android:defaultValue="1"
|
||||
android:entries="@array/layout_type"
|
||||
android:entryValues="@array/layout_type_val"
|
||||
app:key="layout_type"
|
||||
app:title="@string/layout_type"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="false"
|
||||
android:summaryOff="@string/select_action_desc_off"
|
||||
android:summaryOn="@string/select_action_desc_on"
|
||||
app:key="select_action"
|
||||
app:title="@string/select_action" />
|
||||
android:defaultValue="false"
|
||||
android:summaryOff="@string/select_action_desc_off"
|
||||
android:summaryOn="@string/select_action_desc_on"
|
||||
app:key="select_action"
|
||||
app:title="@string/select_action" />
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="false"
|
||||
android:summaryOff="@string/perf_stats_desc_off"
|
||||
android:summaryOn="@string/perf_stats_desc_on"
|
||||
app:key="perf_stats"
|
||||
app:title="@string/perf_stats" />
|
||||
android:defaultValue="false"
|
||||
android:summaryOff="@string/perf_stats_desc_off"
|
||||
android:summaryOn="@string/perf_stats_desc_on"
|
||||
app:key="perf_stats"
|
||||
app:title="@string/perf_stats" />
|
||||
<ListPreference
|
||||
android:defaultValue="2"
|
||||
android:entries="@array/log_level"
|
||||
android:entryValues="@array/log_level_val"
|
||||
app:key="log_level"
|
||||
app:title="@string/log_level"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
android:defaultValue="2"
|
||||
android:entries="@array/log_level"
|
||||
android:entryValues="@array/log_level_val"
|
||||
app:key="log_level"
|
||||
app:title="@string/log_level"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="false"
|
||||
android:summaryOff="@string/log_compact_desc_off"
|
||||
android:summaryOn="@string/log_compact_desc_on"
|
||||
app:key="log_compact"
|
||||
app:title="@string/log_compact" />
|
||||
android:defaultValue="false"
|
||||
android:summaryOff="@string/log_compact_desc_off"
|
||||
android:summaryOn="@string/log_compact_desc_on"
|
||||
app:key="log_compact"
|
||||
app:title="@string/log_compact" />
|
||||
</PreferenceCategory>
|
||||
<PreferenceCategory
|
||||
android:key="category_keys"
|
||||
android:title="@string/keys">
|
||||
<emu.skyline.preference.KeyPickerPreference
|
||||
app:key="prod_keys"
|
||||
app:title="@string/prod_keys"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
<emu.skyline.preference.KeyPickerPreference
|
||||
app:key="title_keys"
|
||||
app:title="@string/title_keys"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
</PreferenceCategory>
|
||||
<PreferenceCategory
|
||||
android:key="category_system"
|
||||
android:title="@string/system">
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="true"
|
||||
android:summaryOff="@string/handheld_enabled"
|
||||
android:summaryOn="@string/docked_enabled"
|
||||
app:key="operation_mode"
|
||||
app:title="@string/use_docked" />
|
||||
<emu.skyline.preference.CustomEditTextPreference
|
||||
android:defaultValue="@string/username_default"
|
||||
app:key="username_value"
|
||||
app:limit="31"
|
||||
app:title="@string/username" />
|
||||
android:defaultValue="@string/username_default"
|
||||
app:key="username_value"
|
||||
app:limit="31"
|
||||
app:title="@string/username" />
|
||||
</PreferenceCategory>
|
||||
<PreferenceCategory
|
||||
android:key="category_keys"
|
||||
android:title="@string/keys">
|
||||
<emu.skyline.preference.KeyPickerPreference
|
||||
app:key="prod_keys"
|
||||
app:title="@string/prod_keys"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
<emu.skyline.preference.KeyPickerPreference
|
||||
app:key="title_keys"
|
||||
app:title="@string/title_keys"
|
||||
app:useSimpleSummaryProvider="true" />
|
||||
</PreferenceCategory>
|
||||
<PreferenceCategory
|
||||
android:key="category_system"
|
||||
android:title="@string/system">
|
||||
android:key="category_presentation"
|
||||
android:title="@string/display">
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="true"
|
||||
android:summaryOff="@string/handheld_enabled"
|
||||
android:summaryOn="@string/docked_enabled"
|
||||
app:key="operation_mode"
|
||||
app:title="@string/use_docked" />
|
||||
android:defaultValue="true"
|
||||
android:summaryOff="@string/triple_buffering_disabled"
|
||||
android:summaryOn="@string/triple_buffering_enabled"
|
||||
app:key="force_triple_buffering"
|
||||
app:title="@string/force_triple_buffering" />
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="false"
|
||||
android:dependency="force_triple_buffering"
|
||||
android:summaryOff="@string/disable_frame_throttling_disabled"
|
||||
android:summaryOn="@string/disable_frame_throttling_enabled"
|
||||
app:key="disable_frame_throttling"
|
||||
app:title="@string/disable_frame_throttling" />
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="false"
|
||||
android:summaryOff="@string/max_refresh_rate_disabled"
|
||||
android:summaryOn="@string/max_refresh_rate_enabled"
|
||||
app:key="max_refresh_rate"
|
||||
app:title="@string/max_refresh_rate" />
|
||||
</PreferenceCategory>
|
||||
<PreferenceCategory
|
||||
android:key="category_input"
|
||||
android:title="@string/input"
|
||||
app:initialExpandedChildrenCount="4">
|
||||
<!--
|
||||
<CheckBoxPreference
|
||||
android:defaultValue="true"
|
||||
android:summaryOff="@string/osc_not_shown"
|
||||
android:summaryOn="@string/osc_shown"
|
||||
app:key="show_osc"
|
||||
app:title="@string/show_osc" />
|
||||
-->
|
||||
android:key="category_input"
|
||||
android:title="@string/input"
|
||||
app:initialExpandedChildrenCount="4">
|
||||
<emu.skyline.preference.ControllerPreference index="0" />
|
||||
<emu.skyline.preference.ControllerPreference index="1" />
|
||||
<emu.skyline.preference.ControllerPreference index="2" />
|
||||
@ -95,73 +110,73 @@
|
||||
<emu.skyline.preference.ControllerPreference index="7" />
|
||||
</PreferenceCategory>
|
||||
<PreferenceCategory
|
||||
android:key="category_licenses"
|
||||
android:title="@string/licenses"
|
||||
app:initialExpandedChildrenCount="3">
|
||||
android:key="category_licenses"
|
||||
android:title="@string/licenses"
|
||||
app:initialExpandedChildrenCount="3">
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/mpl2_license"
|
||||
libraryUrl="https://github.com/skyline-emu/skyline"
|
||||
app:summary="@string/skyline_license_description"
|
||||
app:title="@string/app_name" />
|
||||
libraryLicense="@string/mpl2_license"
|
||||
libraryUrl="https://github.com/skyline-emu/skyline"
|
||||
app:summary="@string/skyline_license_description"
|
||||
app:title="@string/app_name" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/fmtlib_license"
|
||||
libraryUrl="https://github.com/fmtlib/fmt"
|
||||
app:summary="@string/fmtlib_description"
|
||||
app:title="@string/fmtlib" />
|
||||
libraryLicense="@string/fmtlib_license"
|
||||
libraryUrl="https://github.com/fmtlib/fmt"
|
||||
app:summary="@string/fmtlib_description"
|
||||
app:title="@string/fmtlib" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/google/oboe"
|
||||
app:summary="@string/oboe_description"
|
||||
app:title="@string/oboe" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/google/oboe"
|
||||
app:summary="@string/oboe_description"
|
||||
app:title="@string/oboe" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/KhronosGroup/Vulkan-Hpp"
|
||||
app:summary="@string/vkhpp_description"
|
||||
app:title="@string/vkhpp" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/KhronosGroup/Vulkan-Hpp"
|
||||
app:summary="@string/vkhpp_description"
|
||||
app:title="@string/vkhpp" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/zlib_license"
|
||||
libraryUrl="https://github.com/leethomason/tinyxml2"
|
||||
app:summary="@string/txml2_description"
|
||||
app:title="@string/txml2" />
|
||||
libraryLicense="@string/zlib_license"
|
||||
libraryUrl="https://github.com/leethomason/tinyxml2"
|
||||
app:summary="@string/txml2_description"
|
||||
app:title="@string/txml2" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/tdebatty/java-string-similarity"
|
||||
app:summary="@string/jssim_description"
|
||||
app:title="@string/jssim" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/tdebatty/java-string-similarity"
|
||||
app:summary="@string/jssim_description"
|
||||
app:title="@string/jssim" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://developer.android.com/jetpack/androidx"
|
||||
app:summary="@string/andx_description"
|
||||
app:title="@string/andx" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://developer.android.com/jetpack/androidx"
|
||||
app:summary="@string/andx_description"
|
||||
app:title="@string/andx" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/material-components/material-components-android"
|
||||
app:summary="@string/amat_description"
|
||||
app:title="@string/amat" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://github.com/material-components/material-components-android"
|
||||
app:summary="@string/amat_description"
|
||||
app:title="@string/amat" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://kotlinlang.org/api/latest/jvm/stdlib"
|
||||
app:summary="@string/ktstd_description"
|
||||
app:title="@string/ktstd" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://kotlinlang.org/api/latest/jvm/stdlib"
|
||||
app:summary="@string/ktstd_description"
|
||||
app:title="@string/ktstd" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://material.io/resources/icons"
|
||||
app:summary="@string/mtico_description"
|
||||
app:title="@string/mtico" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://material.io/resources/icons"
|
||||
app:summary="@string/mtico_description"
|
||||
app:title="@string/mtico" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://fonts.google.com/specimen/Open+Sans"
|
||||
app:summary="@string/open_sans_description"
|
||||
app:title="@string/open_sans" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://fonts.google.com/specimen/Open+Sans"
|
||||
app:summary="@string/open_sans_description"
|
||||
app:title="@string/open_sans" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://fonts.google.com/specimen/Roboto"
|
||||
app:summary="@string/roboto_description"
|
||||
app:title="@string/roboto" />
|
||||
libraryLicense="@string/apache2_license"
|
||||
libraryUrl="https://fonts.google.com/specimen/Roboto"
|
||||
app:summary="@string/roboto_description"
|
||||
app:title="@string/roboto" />
|
||||
<emu.skyline.preference.LicensePreference
|
||||
libraryLicense="@string/sil_open_font_license"
|
||||
libraryUrl="https://fonts.google.com/specimen/Source+Sans+Pro"
|
||||
app:summary="@string/source_sans_pro_description"
|
||||
app:title="@string/source_sans_pro" />
|
||||
libraryLicense="@string/sil_open_font_license"
|
||||
libraryUrl="https://fonts.google.com/specimen/Source+Sans+Pro"
|
||||
app:summary="@string/source_sans_pro_description"
|
||||
app:title="@string/source_sans_pro" />
|
||||
</PreferenceCategory>
|
||||
</androidx.preference.PreferenceScreen>
|
||||
|
Loading…
x
Reference in New Issue
Block a user