mirror of
https://github.com/microsoft/PowerToys.git
synced 2025-01-04 18:27:56 +08:00
53f830bb38
* Migrate FancyZones data persisting from Registry to JSON file * Address PR comment: Remove redundant check * Addres PR comment: Remove unused Dpi and add CmdArgs enum * Address PR comment: Make methods const and inline * Address PR comments: Expose GenerateUniqueId function and use const ref instead of passing wstring by value * Address PR comment: Use lamdba as callback * Address PR comment: Move GenerateUniqueId to ZoneWindowUtils namespace * Address PR comment: Use regular comparison instead of std::wstring::compare * Address PR comment: Use std::wstring_view for tmp file paths * Address PR comment: Use scoped lock when accessing member data * Address PR comment: Remove typedefs to increase code readability * Address PR comment: removed nullptr checks with corresponding tests * Address PR comment: Move ZoneSet object instead of copying * Address PR comment: Make FancyZonesData instance const where possible * Remove unnecessary gutter variable during calculating zone coordinates * Remove uneeded subclass * Avoid unnecessary copying and reserve space for vector if possible * Save FancyZones data after exiting editor * App zone history (#18) * added window and zone set ids to app zone history * Rename JSON file * Remove AppZoneHistory migration * Move parsing of ZoneWindow independent temp files outside of it * Unit tests update (#19) * check device existence in map * updated ZoneSet tests * updated JsonHelpers tests * Use single zone count information * Remove uneeded tests * Remove one more test * Remove uneeded line * Address PR comments - Missing whitespace * Update zoneset data for new virtual desktops (#21) * update active zone set with actual data * Introduce Blank zone set (used to indicate that no layout applied yet). Move parsing completely outside of ZoneWindow. * Fix unit tests to match modifications in implementation * Fix applying layouts on startup (second monitor) Co-authored-by: vldmr11080 <57061786+vldmr11080@users.noreply.github.com> Co-authored-by: Seraphima <zykovas91@gmail.com>
630 lines
20 KiB
C++
630 lines
20 KiB
C++
#include "pch.h"
|
|
|
|
#include "util.h"
|
|
#include "lib/ZoneSet.h"
|
|
#include "lib/RegistryHelpers.h"
|
|
|
|
#include <common/dpi_aware.h>
|
|
|
|
namespace
|
|
{
|
|
constexpr int C_MULTIPLIER = 10000;
|
|
|
|
/*
|
|
struct GridLayoutInfo {
|
|
int rows;
|
|
int columns;
|
|
int rowsPercents[MAX_ZONE_COUNT];
|
|
int columnsPercents[MAX_ZONE_COUNT];
|
|
int cellChildMap[MAX_ZONE_COUNT][MAX_ZONE_COUNT];
|
|
};
|
|
*/
|
|
|
|
auto l = JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Minimal{ .rows = 1, .columns = 1 });
|
|
// PriorityGrid layout is unique for zoneCount <= 11. For zoneCount > 11 PriorityGrid is same as Grid
|
|
JSONHelpers::GridLayoutInfo predefinedPriorityGridLayouts[11] = {
|
|
/* 1 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 1,
|
|
.columns = 1,
|
|
.rowsPercents = { 10000 },
|
|
.columnsPercents = { 10000 },
|
|
.cellChildMap = { { 0 } } }),
|
|
/* 2 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 1,
|
|
.columns = 2,
|
|
.rowsPercents = { 10000 },
|
|
.columnsPercents = { 6667, 3333 },
|
|
.cellChildMap = { { 0, 1 } } }),
|
|
/* 3 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 1,
|
|
.columns = 3,
|
|
.rowsPercents = { 10000 },
|
|
.columnsPercents = { 2500, 5000, 2500 },
|
|
.cellChildMap = { { 0, 1, 2 } } }),
|
|
/* 4 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 2,
|
|
.columns = 3,
|
|
.rowsPercents = { 5000, 5000 },
|
|
.columnsPercents = { 2500, 5000, 2500 },
|
|
.cellChildMap = { { 0, 1, 2 }, { 0, 1, 3 } } }),
|
|
/* 5 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 2,
|
|
.columns = 3,
|
|
.rowsPercents = { 5000, 5000 },
|
|
.columnsPercents = { 2500, 5000, 2500 },
|
|
.cellChildMap = { { 0, 1, 2 }, { 3, 1, 4 } } }),
|
|
/* 6 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 3,
|
|
.columns = 3,
|
|
.rowsPercents = { 3333, 3334, 3333 },
|
|
.columnsPercents = { 2500, 5000, 2500 },
|
|
.cellChildMap = { { 0, 1, 2 }, { 0, 1, 3 }, { 4, 1, 5 } } }),
|
|
/* 7 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 3,
|
|
.columns = 3,
|
|
.rowsPercents = { 3333, 3334, 3333 },
|
|
.columnsPercents = { 2500, 5000, 2500 },
|
|
.cellChildMap = { { 0, 1, 2 }, { 3, 1, 4 }, { 5, 1, 6 } } }),
|
|
/* 8 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 3,
|
|
.columns = 4,
|
|
.rowsPercents = { 3333, 3334, 3333 },
|
|
.columnsPercents = { 2500, 2500, 2500, 2500 },
|
|
.cellChildMap = { { 0, 1, 2, 3 }, { 4, 1, 2, 5 }, { 6, 1, 2, 7 } } }),
|
|
/* 9 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 3,
|
|
.columns = 4,
|
|
.rowsPercents = { 3333, 3334, 3333 },
|
|
.columnsPercents = { 2500, 2500, 2500, 2500 },
|
|
.cellChildMap = { { 0, 1, 2, 3 }, { 4, 1, 2, 5 }, { 6, 1, 7, 8 } } }),
|
|
/* 10 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 3,
|
|
.columns = 4,
|
|
.rowsPercents = { 3333, 3334, 3333 },
|
|
.columnsPercents = { 2500, 2500, 2500, 2500 },
|
|
.cellChildMap = { { 0, 1, 2, 3 }, { 4, 1, 5, 6 }, { 7, 1, 8, 9 } } }),
|
|
/* 11 */
|
|
JSONHelpers::GridLayoutInfo(JSONHelpers::GridLayoutInfo::Full{
|
|
.rows = 3,
|
|
.columns = 4,
|
|
.rowsPercents = { 3333, 3334, 3333 },
|
|
.columnsPercents = { 2500, 2500, 2500, 2500 },
|
|
.cellChildMap = { { 0, 1, 2, 3 }, { 4, 1, 5, 6 }, { 7, 8, 9, 10 } } }),
|
|
};
|
|
}
|
|
|
|
struct ZoneSet : winrt::implements<ZoneSet, IZoneSet>
|
|
{
|
|
public:
|
|
ZoneSet(ZoneSetConfig const& config) :
|
|
m_config(config)
|
|
{
|
|
}
|
|
|
|
ZoneSet(ZoneSetConfig const& config, std::vector<winrt::com_ptr<IZone>> zones) :
|
|
m_config(config),
|
|
m_zones(zones)
|
|
{
|
|
}
|
|
|
|
IFACEMETHODIMP_(GUID)
|
|
Id() noexcept { return m_config.Id; }
|
|
IFACEMETHODIMP_(JSONHelpers::ZoneSetLayoutType)
|
|
LayoutType() noexcept { return m_config.LayoutType; }
|
|
IFACEMETHODIMP AddZone(winrt::com_ptr<IZone> zone) noexcept;
|
|
IFACEMETHODIMP_(winrt::com_ptr<IZone>)
|
|
ZoneFromPoint(POINT pt) noexcept;
|
|
IFACEMETHODIMP_(int)
|
|
GetZoneIndexFromWindow(HWND window) noexcept;
|
|
IFACEMETHODIMP_(std::vector<winrt::com_ptr<IZone>>)
|
|
GetZones() noexcept { return m_zones; }
|
|
IFACEMETHODIMP_(void)
|
|
MoveWindowIntoZoneByIndex(HWND window, HWND zoneWindow, int index) noexcept;
|
|
IFACEMETHODIMP_(void)
|
|
MoveWindowIntoZoneByDirection(HWND window, HWND zoneWindow, DWORD vkCode) noexcept;
|
|
IFACEMETHODIMP_(void)
|
|
MoveWindowIntoZoneByPoint(HWND window, HWND zoneWindow, POINT ptClient) noexcept;
|
|
IFACEMETHODIMP_(bool)
|
|
CalculateZones(MONITORINFO monitorInfo, int zoneCount, int spacing) noexcept;
|
|
|
|
private:
|
|
bool CalculateFocusLayout(Rect workArea, int zoneCount) noexcept;
|
|
bool CalculateColumnsAndRowsLayout(Rect workArea, JSONHelpers::ZoneSetLayoutType type, int zoneCount, int spacing) noexcept;
|
|
bool CalculateGridLayout(Rect workArea, JSONHelpers::ZoneSetLayoutType type, int zoneCount, int spacing) noexcept;
|
|
bool CalculateUniquePriorityGridLayout(Rect workArea, int zoneCount, int spacing) noexcept;
|
|
bool CalculateCustomLayout(Rect workArea, int spacing) noexcept;
|
|
|
|
bool CalculateGridZones(Rect workArea, JSONHelpers::GridLayoutInfo gridLayoutInfo, int spacing);
|
|
|
|
winrt::com_ptr<IZone> ZoneFromWindow(HWND window) noexcept;
|
|
|
|
std::vector<winrt::com_ptr<IZone>> m_zones;
|
|
ZoneSetConfig m_config;
|
|
};
|
|
|
|
IFACEMETHODIMP ZoneSet::AddZone(winrt::com_ptr<IZone> zone) noexcept
|
|
{
|
|
m_zones.emplace_back(zone);
|
|
|
|
// Important not to set Id 0 since we store it in the HWND using SetProp.
|
|
// SetProp(0) doesn't really work.
|
|
zone->SetId(m_zones.size());
|
|
return S_OK;
|
|
}
|
|
|
|
IFACEMETHODIMP_(winrt::com_ptr<IZone>)
|
|
ZoneSet::ZoneFromPoint(POINT pt) noexcept
|
|
{
|
|
winrt::com_ptr<IZone> smallestKnownZone = nullptr;
|
|
// To reduce redundant calculations, we will store the last known zones area.
|
|
int smallestKnownZoneArea = INT32_MAX;
|
|
for (auto iter = m_zones.rbegin(); iter != m_zones.rend(); iter++)
|
|
{
|
|
if (winrt::com_ptr<IZone> zone = iter->try_as<IZone>())
|
|
{
|
|
RECT* newZoneRect = &zone->GetZoneRect();
|
|
if (PtInRect(newZoneRect, pt))
|
|
{
|
|
if (smallestKnownZone == nullptr)
|
|
{
|
|
smallestKnownZone = zone;
|
|
|
|
RECT* r = &smallestKnownZone->GetZoneRect();
|
|
smallestKnownZoneArea = (r->right - r->left) * (r->bottom - r->top);
|
|
}
|
|
else
|
|
{
|
|
int newZoneArea = (newZoneRect->right - newZoneRect->left) * (newZoneRect->bottom - newZoneRect->top);
|
|
|
|
if (newZoneArea < smallestKnownZoneArea)
|
|
{
|
|
smallestKnownZone = zone;
|
|
smallestKnownZoneArea = newZoneArea;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return smallestKnownZone;
|
|
}
|
|
|
|
IFACEMETHODIMP_(int)
|
|
ZoneSet::GetZoneIndexFromWindow(HWND window) noexcept
|
|
{
|
|
int zoneIndex = 0;
|
|
for (auto iter = m_zones.begin(); iter != m_zones.end(); iter++, zoneIndex++)
|
|
{
|
|
if (winrt::com_ptr<IZone> zone = iter->try_as<IZone>())
|
|
{
|
|
if (zone->ContainsWindow(window))
|
|
{
|
|
return zoneIndex;
|
|
}
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
IFACEMETHODIMP_(void)
|
|
ZoneSet::MoveWindowIntoZoneByIndex(HWND window, HWND windowZone, int index) noexcept
|
|
{
|
|
if (m_zones.empty())
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (index >= int(m_zones.size()))
|
|
{
|
|
index = 0;
|
|
}
|
|
|
|
while (auto zoneDrop = ZoneFromWindow(window))
|
|
{
|
|
zoneDrop->RemoveWindowFromZone(window, !IsZoomed(window));
|
|
}
|
|
|
|
if (auto zone = m_zones.at(index))
|
|
{
|
|
zone->AddWindowToZone(window, windowZone, false);
|
|
}
|
|
}
|
|
|
|
IFACEMETHODIMP_(void)
|
|
ZoneSet::MoveWindowIntoZoneByDirection(HWND window, HWND windowZone, DWORD vkCode) noexcept
|
|
{
|
|
if (m_zones.empty())
|
|
{
|
|
return;
|
|
}
|
|
|
|
winrt::com_ptr<IZone> oldZone = nullptr;
|
|
winrt::com_ptr<IZone> newZone = nullptr;
|
|
|
|
auto iter = std::find(m_zones.begin(), m_zones.end(), ZoneFromWindow(window));
|
|
if (iter == m_zones.end())
|
|
{
|
|
iter = (vkCode == VK_RIGHT) ? m_zones.begin() : m_zones.end() - 1;
|
|
}
|
|
else if (oldZone = iter->as<IZone>())
|
|
{
|
|
if (vkCode == VK_LEFT)
|
|
{
|
|
if (iter == m_zones.begin())
|
|
{
|
|
iter = m_zones.end();
|
|
}
|
|
iter--;
|
|
}
|
|
else if (vkCode == VK_RIGHT)
|
|
{
|
|
iter++;
|
|
if (iter == m_zones.end())
|
|
{
|
|
iter = m_zones.begin();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (newZone = iter->as<IZone>())
|
|
{
|
|
if (oldZone)
|
|
{
|
|
oldZone->RemoveWindowFromZone(window, false);
|
|
}
|
|
newZone->AddWindowToZone(window, windowZone, true);
|
|
}
|
|
}
|
|
|
|
IFACEMETHODIMP_(void)
|
|
ZoneSet::MoveWindowIntoZoneByPoint(HWND window, HWND zoneWindow, POINT ptClient) noexcept
|
|
{
|
|
while (auto zoneDrop = ZoneFromWindow(window))
|
|
{
|
|
zoneDrop->RemoveWindowFromZone(window, !IsZoomed(window));
|
|
}
|
|
|
|
if (auto zone = ZoneFromPoint(ptClient))
|
|
{
|
|
zone->AddWindowToZone(window, zoneWindow, true);
|
|
}
|
|
}
|
|
|
|
IFACEMETHODIMP_(bool)
|
|
ZoneSet::CalculateZones(MONITORINFO monitorInfo, int zoneCount, int spacing) noexcept
|
|
{
|
|
Rect const workArea(monitorInfo.rcWork);
|
|
//invalid work area
|
|
if (workArea.width() == 0 || workArea.height() == 0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//invalid zoneCount, may cause division by zero
|
|
if (zoneCount <= 0 && m_config.LayoutType != JSONHelpers::ZoneSetLayoutType::Custom)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool success = true;
|
|
switch (m_config.LayoutType)
|
|
{
|
|
case JSONHelpers::ZoneSetLayoutType::Focus:
|
|
success = CalculateFocusLayout(workArea, zoneCount);
|
|
break;
|
|
case JSONHelpers::ZoneSetLayoutType::Columns:
|
|
case JSONHelpers::ZoneSetLayoutType::Rows:
|
|
success = CalculateColumnsAndRowsLayout(workArea, m_config.LayoutType, zoneCount, spacing);
|
|
break;
|
|
case JSONHelpers::ZoneSetLayoutType::Grid:
|
|
case JSONHelpers::ZoneSetLayoutType::PriorityGrid:
|
|
success = CalculateGridLayout(workArea, m_config.LayoutType, zoneCount, spacing);
|
|
break;
|
|
case JSONHelpers::ZoneSetLayoutType::Custom:
|
|
success = CalculateCustomLayout(workArea, spacing);
|
|
break;
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
bool ZoneSet::CalculateFocusLayout(Rect workArea, int zoneCount) noexcept
|
|
{
|
|
bool success = true;
|
|
|
|
long left{ long(workArea.width() * 0.1) };
|
|
long top{ long(workArea.height() * 0.1) };
|
|
long right{ long(workArea.width() * 0.6) };
|
|
long bottom{ long(workArea.height() * 0.6) };
|
|
|
|
RECT focusZoneRect{ left, top, right, bottom };
|
|
|
|
long focusRectXIncrement = (zoneCount <= 1) ? 0 : (int)(workArea.width() * 0.2) / (zoneCount - 1);
|
|
long focusRectYIncrement = (zoneCount <= 1) ? 0 : (int)(workArea.height() * 0.2) / (zoneCount - 1);
|
|
|
|
if (left >= right || top >= bottom || left < 0 || right < 0 || top < 0 || bottom < 0)
|
|
{
|
|
success = false;
|
|
}
|
|
|
|
for (int i = 0; i < zoneCount; i++)
|
|
{
|
|
AddZone(MakeZone(focusZoneRect));
|
|
focusZoneRect.left += focusRectXIncrement;
|
|
focusZoneRect.right += focusRectXIncrement;
|
|
focusZoneRect.bottom += focusRectYIncrement;
|
|
focusZoneRect.top += focusRectYIncrement;
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
bool ZoneSet::CalculateColumnsAndRowsLayout(Rect workArea, JSONHelpers::ZoneSetLayoutType type, int zoneCount, int spacing) noexcept
|
|
{
|
|
bool success = true;
|
|
|
|
int zonePercent = C_MULTIPLIER / zoneCount;
|
|
|
|
long totalWidth;
|
|
long totalHeight;
|
|
|
|
long cellWidth;
|
|
long cellHeight;
|
|
|
|
if (type == JSONHelpers::ZoneSetLayoutType::Columns)
|
|
{
|
|
totalWidth = workArea.width() - (spacing * (zoneCount + 1));
|
|
totalHeight = workArea.height() - (spacing * 2);
|
|
cellWidth = totalWidth * zonePercent / C_MULTIPLIER;
|
|
cellHeight = totalHeight;
|
|
}
|
|
else
|
|
{ //Rows
|
|
totalWidth = workArea.width() - (spacing * 2);
|
|
totalHeight = workArea.height() - (spacing * (zoneCount + 1));
|
|
cellWidth = totalWidth;
|
|
cellHeight = totalHeight * zonePercent / C_MULTIPLIER;
|
|
}
|
|
|
|
long top = spacing;
|
|
long left = spacing;
|
|
long bottom = top + cellHeight;
|
|
long right = left + cellWidth;
|
|
|
|
for (int zone = 0; zone < zoneCount; zone++)
|
|
{
|
|
if (left >= right || top >= bottom || left < 0 || right < 0 || top < 0 || bottom < 0)
|
|
{
|
|
success = false;
|
|
}
|
|
|
|
RECT focusZoneRect{ left, top, right, bottom };
|
|
AddZone(MakeZone(focusZoneRect));
|
|
|
|
if (type == JSONHelpers::ZoneSetLayoutType::Columns)
|
|
{
|
|
left += cellWidth + spacing;
|
|
right = left + cellWidth;
|
|
}
|
|
else
|
|
{ //Rows
|
|
top += cellHeight + spacing;
|
|
bottom = top + cellHeight;
|
|
}
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
bool ZoneSet::CalculateGridLayout(Rect workArea, JSONHelpers::ZoneSetLayoutType type, int zoneCount, int spacing) noexcept
|
|
{
|
|
const auto count = sizeof(predefinedPriorityGridLayouts) / sizeof(JSONHelpers::GridLayoutInfo);
|
|
if (type == JSONHelpers::ZoneSetLayoutType::PriorityGrid && zoneCount < count)
|
|
{
|
|
return CalculateUniquePriorityGridLayout(workArea, zoneCount, spacing);
|
|
}
|
|
|
|
int rows = 1, columns = 1;
|
|
while (zoneCount / rows >= rows)
|
|
{
|
|
rows++;
|
|
}
|
|
rows--;
|
|
columns = zoneCount / rows;
|
|
if (zoneCount % rows == 0)
|
|
{
|
|
// even grid
|
|
}
|
|
else
|
|
{
|
|
columns++;
|
|
}
|
|
|
|
JSONHelpers::GridLayoutInfo gridLayoutInfo(JSONHelpers::GridLayoutInfo::Minimal{ .rows = rows, .columns = columns });
|
|
|
|
for (int row = 0; row < rows; row++)
|
|
{
|
|
gridLayoutInfo.rowsPercents()[row] = C_MULTIPLIER / rows;
|
|
}
|
|
for (int col = 0; col < columns; col++)
|
|
{
|
|
gridLayoutInfo.columnsPercents()[col] = C_MULTIPLIER / columns;
|
|
}
|
|
|
|
for (int i = 0; i < rows; ++i)
|
|
{
|
|
gridLayoutInfo.cellChildMap()[i] = std::vector<int>(columns);
|
|
}
|
|
|
|
int index = 0;
|
|
for (int col = columns - 1; col >= 0; col--)
|
|
{
|
|
for (int row = rows - 1; row >= 0; row--)
|
|
{
|
|
gridLayoutInfo.cellChildMap()[row][col] = index++;
|
|
if (index == zoneCount)
|
|
{
|
|
index--;
|
|
}
|
|
}
|
|
}
|
|
return CalculateGridZones(workArea, gridLayoutInfo, spacing);
|
|
}
|
|
|
|
bool ZoneSet::CalculateUniquePriorityGridLayout(Rect workArea, int zoneCount, int spacing) noexcept
|
|
{
|
|
if (zoneCount <= 0 || zoneCount >= sizeof(predefinedPriorityGridLayouts))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return CalculateGridZones(workArea, predefinedPriorityGridLayouts[zoneCount - 1], spacing);
|
|
}
|
|
|
|
bool ZoneSet::CalculateCustomLayout(Rect workArea, int spacing) noexcept
|
|
{
|
|
wil::unique_cotaskmem_string guuidStr;
|
|
if (SUCCEEDED_LOG(StringFromCLSID(m_config.Id, &guuidStr)))
|
|
{
|
|
const std::wstring guuid = guuidStr.get();
|
|
const auto& customZoneSets = JSONHelpers::FancyZonesDataInstance().GetCustomZoneSetsMap();
|
|
if (!customZoneSets.contains(guuid))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
const auto& zoneSet = customZoneSets.at(guuid);
|
|
if (zoneSet.type == JSONHelpers::CustomLayoutType::Canvas && std::holds_alternative<JSONHelpers::CanvasLayoutInfo>(zoneSet.info))
|
|
{
|
|
const auto& zoneSetInfo = std::get<JSONHelpers::CanvasLayoutInfo>(zoneSet.info);
|
|
for (const auto& zone : zoneSetInfo.zones)
|
|
{
|
|
int x = zone.x;
|
|
int y = zone.y;
|
|
int width = zone.width;
|
|
int height = zone.height;
|
|
|
|
if (x < 0 || y < 0 || width < 0 || height < 0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
DPIAware::Convert(m_config.Monitor, x, y);
|
|
DPIAware::Convert(m_config.Monitor, width, height);
|
|
|
|
AddZone(MakeZone(RECT{ x, y, x + width, y + height }));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
else if (zoneSet.type == JSONHelpers::CustomLayoutType::Grid && std::holds_alternative<JSONHelpers::GridLayoutInfo>(zoneSet.info))
|
|
{
|
|
const auto& info = std::get<JSONHelpers::GridLayoutInfo>(zoneSet.info);
|
|
return CalculateGridZones(workArea, info, spacing);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool ZoneSet::CalculateGridZones(Rect workArea, JSONHelpers::GridLayoutInfo gridLayoutInfo, int spacing)
|
|
{
|
|
bool success = true;
|
|
|
|
long totalWidth = workArea.width() - (spacing * (gridLayoutInfo.columns() + 1));
|
|
long totalHeight = workArea.height() - (spacing * (gridLayoutInfo.rows() + 1));
|
|
struct Info
|
|
{
|
|
long Extent;
|
|
long Start;
|
|
long End;
|
|
};
|
|
Info rowInfo[JSONHelpers::MAX_ZONE_COUNT];
|
|
Info columnInfo[JSONHelpers::MAX_ZONE_COUNT];
|
|
|
|
long top = spacing;
|
|
for (int row = 0; row < gridLayoutInfo.rows(); row++)
|
|
{
|
|
rowInfo[row].Start = top;
|
|
rowInfo[row].Extent = totalHeight * gridLayoutInfo.rowsPercents()[row] / C_MULTIPLIER;
|
|
rowInfo[row].End = rowInfo[row].Start + rowInfo[row].Extent;
|
|
top += rowInfo[row].Extent + spacing;
|
|
}
|
|
|
|
long left = spacing;
|
|
for (int col = 0; col < gridLayoutInfo.columns(); col++)
|
|
{
|
|
columnInfo[col].Start = left;
|
|
columnInfo[col].Extent = totalWidth * gridLayoutInfo.columnsPercents()[col] / C_MULTIPLIER;
|
|
columnInfo[col].End = columnInfo[col].Start + columnInfo[col].Extent;
|
|
left += columnInfo[col].Extent + spacing;
|
|
}
|
|
|
|
for (int row = 0; row < gridLayoutInfo.rows(); row++)
|
|
{
|
|
for (int col = 0; col < gridLayoutInfo.columns(); col++)
|
|
{
|
|
int i = gridLayoutInfo.cellChildMap()[row][col];
|
|
if (((row == 0) || (gridLayoutInfo.cellChildMap()[row - 1][col] != i)) &&
|
|
((col == 0) || (gridLayoutInfo.cellChildMap()[row][col - 1] != i)))
|
|
{
|
|
left = columnInfo[col].Start;
|
|
top = rowInfo[row].Start;
|
|
|
|
int maxRow = row;
|
|
while (((maxRow + 1) < gridLayoutInfo.rows()) && (gridLayoutInfo.cellChildMap()[maxRow + 1][col] == i))
|
|
{
|
|
maxRow++;
|
|
}
|
|
int maxCol = col;
|
|
while (((maxCol + 1) < gridLayoutInfo.columns()) && (gridLayoutInfo.cellChildMap()[row][maxCol + 1] == i))
|
|
{
|
|
maxCol++;
|
|
}
|
|
|
|
long right = columnInfo[maxCol].End;
|
|
long bottom = rowInfo[maxRow].End;
|
|
|
|
if (left >= right || top >= bottom || left < 0 || right < 0 || top < 0 || bottom < 0)
|
|
{
|
|
success = false;
|
|
}
|
|
|
|
AddZone(MakeZone(RECT{ left, top, right, bottom }));
|
|
}
|
|
}
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
winrt::com_ptr<IZone> ZoneSet::ZoneFromWindow(HWND window) noexcept
|
|
{
|
|
for (auto iter = m_zones.begin(); iter != m_zones.end(); iter++)
|
|
{
|
|
if (winrt::com_ptr<IZone> zone = iter->try_as<IZone>())
|
|
{
|
|
if (zone->ContainsWindow(window))
|
|
{
|
|
return zone;
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
winrt::com_ptr<IZoneSet> MakeZoneSet(ZoneSetConfig const& config) noexcept
|
|
{
|
|
return winrt::make_self<ZoneSet>(config);
|
|
}
|