2018-01-13 16:22:39 -05:00
|
|
|
// Copyright 2018 yuzu emulator team
|
2016-05-03 00:07:17 -06:00
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-01-11 22:24:09 -05:00
|
|
|
#include <cmath>
|
|
|
|
|
2016-05-03 00:07:17 -06:00
|
|
|
#include "common/assert.h"
|
2017-05-27 16:10:25 -07:00
|
|
|
#include "core/frontend/framebuffer_layout.h"
|
2018-08-31 14:16:16 +08:00
|
|
|
#include "core/settings.h"
|
2016-05-03 00:07:17 -06:00
|
|
|
|
|
|
|
namespace Layout {
|
|
|
|
|
2016-11-05 01:47:05 -06:00
|
|
|
// Finds the largest size subrectangle contained in window area that is confined to the aspect ratio
|
|
|
|
template <class T>
|
2019-02-26 22:47:49 -05:00
|
|
|
static Common::Rectangle<T> MaxRectangle(Common::Rectangle<T> window_area,
|
|
|
|
float screen_aspect_ratio) {
|
2016-11-05 01:47:05 -06:00
|
|
|
float scale = std::min(static_cast<float>(window_area.GetWidth()),
|
|
|
|
window_area.GetHeight() / screen_aspect_ratio);
|
2019-02-26 22:47:49 -05:00
|
|
|
return Common::Rectangle<T>{0, 0, static_cast<T>(std::round(scale)),
|
|
|
|
static_cast<T>(std::round(scale * screen_aspect_ratio))};
|
2016-05-03 00:07:17 -06:00
|
|
|
}
|
|
|
|
|
2019-05-29 02:14:24 -04:00
|
|
|
FramebufferLayout DefaultFrameLayout(u32 width, u32 height) {
|
2016-05-03 00:07:17 -06:00
|
|
|
ASSERT(width > 0);
|
|
|
|
ASSERT(height > 0);
|
2016-11-05 01:47:05 -06:00
|
|
|
// The drawing code needs at least somewhat valid values for both screens
|
|
|
|
// so just calculate them both even if the other isn't showing.
|
2018-01-09 22:36:07 -05:00
|
|
|
FramebufferLayout res{width, height};
|
2016-11-05 01:47:05 -06:00
|
|
|
|
2019-05-29 02:14:24 -04:00
|
|
|
const auto window_aspect_ratio = static_cast<float>(height) / width;
|
2020-02-13 22:17:28 -05:00
|
|
|
float emulation_aspect_ratio;
|
|
|
|
|
2020-02-13 23:13:23 -05:00
|
|
|
switch (static_cast<Aspect>(Settings::values.aspect_ratio)) {
|
|
|
|
case Aspect::AspectDefault:
|
2020-02-13 22:17:28 -05:00
|
|
|
emulation_aspect_ratio = static_cast<float>(ScreenUndocked::Height) / ScreenUndocked::Width;
|
|
|
|
break;
|
2020-02-13 23:13:23 -05:00
|
|
|
case Aspect::Aspect21by9:
|
2020-02-13 22:17:28 -05:00
|
|
|
emulation_aspect_ratio = 9.f / 21;
|
|
|
|
break;
|
2020-02-13 23:13:23 -05:00
|
|
|
case Aspect::AspectStretch:
|
2020-02-13 22:17:28 -05:00
|
|
|
emulation_aspect_ratio = window_aspect_ratio;
|
|
|
|
break;
|
2020-02-13 23:13:23 -05:00
|
|
|
default:
|
2020-02-13 22:17:28 -05:00
|
|
|
emulation_aspect_ratio = static_cast<float>(ScreenUndocked::Height) / ScreenUndocked::Width;
|
|
|
|
}
|
2017-08-25 23:53:07 +02:00
|
|
|
|
2019-05-29 02:14:24 -04:00
|
|
|
const Common::Rectangle<u32> screen_window_area{0, 0, width, height};
|
|
|
|
Common::Rectangle<u32> screen = MaxRectangle(screen_window_area, emulation_aspect_ratio);
|
2017-08-25 23:53:07 +02:00
|
|
|
|
|
|
|
if (window_aspect_ratio < emulation_aspect_ratio) {
|
2018-01-09 22:36:07 -05:00
|
|
|
screen = screen.TranslateX((screen_window_area.GetWidth() - screen.GetWidth()) / 2);
|
2017-08-25 23:53:07 +02:00
|
|
|
} else {
|
2018-01-09 22:36:07 -05:00
|
|
|
screen = screen.TranslateY((height - screen.GetHeight()) / 2);
|
2017-08-25 23:53:07 +02:00
|
|
|
}
|
2019-05-29 02:14:24 -04:00
|
|
|
|
2018-01-09 22:36:07 -05:00
|
|
|
res.screen = screen;
|
2017-08-25 23:53:07 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2019-05-29 02:14:24 -04:00
|
|
|
FramebufferLayout FrameLayoutFromResolutionScale(u32 res_scale) {
|
|
|
|
u32 width, height;
|
2018-08-31 14:16:16 +08:00
|
|
|
|
|
|
|
if (Settings::values.use_docked_mode) {
|
|
|
|
width = ScreenDocked::WidthDocked * res_scale;
|
|
|
|
height = ScreenDocked::HeightDocked * res_scale;
|
|
|
|
} else {
|
|
|
|
width = ScreenUndocked::Width * res_scale;
|
|
|
|
height = ScreenUndocked::Height * res_scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DefaultFrameLayout(width, height);
|
|
|
|
}
|
|
|
|
|
2017-08-25 23:53:07 +02:00
|
|
|
} // namespace Layout
|