2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-05-21 04:12:37 -07:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#include "nsScreenManagerWin.h"
|
|
|
|
#include "nsScreenWin.h"
|
2013-04-01 02:10:33 -07:00
|
|
|
#include "gfxWindowsPlatform.h"
|
2013-04-09 01:39:47 -07:00
|
|
|
#include "nsIWidget.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
|
2014-07-14 00:52:00 -07:00
|
|
|
BOOL CALLBACK CountMonitors(HMONITOR, HDC, LPRECT, LPARAM ioCount);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2014-07-14 00:52:00 -07:00
|
|
|
nsScreenManagerWin::nsScreenManagerWin()
|
2007-03-22 10:30:00 -07:00
|
|
|
: mNumberOfScreens(0)
|
|
|
|
{
|
|
|
|
// nothing to do. I guess we could cache a bunch of information
|
|
|
|
// here, but we want to ask the device at runtime in case anything
|
|
|
|
// has changed.
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-14 00:52:00 -07:00
|
|
|
nsScreenManagerWin::~nsScreenManagerWin()
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// addref, release, QI
|
2014-04-27 00:06:00 -07:00
|
|
|
NS_IMPL_ISUPPORTS(nsScreenManagerWin, nsIScreenManager)
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// CreateNewScreenObject
|
|
|
|
//
|
|
|
|
// Utility routine. Creates a new screen object from the given device handle
|
|
|
|
//
|
|
|
|
// NOTE: For this "single-monitor" impl, we just always return the cached primary
|
|
|
|
// screen. This should change when a multi-monitor impl is done.
|
|
|
|
//
|
|
|
|
nsIScreen*
|
2014-07-14 00:52:00 -07:00
|
|
|
nsScreenManagerWin::CreateNewScreenObject(HMONITOR inScreen)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2012-07-30 07:20:58 -07:00
|
|
|
nsIScreen* retScreen = nullptr;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// look through our screen list, hoping to find it. If it's not there,
|
|
|
|
// add it and return the new one.
|
2014-07-14 00:52:00 -07:00
|
|
|
for (unsigned i = 0; i < mScreenList.Length(); ++i) {
|
2009-03-10 19:26:03 -07:00
|
|
|
ScreenListItem& curr = mScreenList[i];
|
2014-07-14 00:52:00 -07:00
|
|
|
if (inScreen == curr.mMon) {
|
2009-03-10 19:26:03 -07:00
|
|
|
NS_IF_ADDREF(retScreen = curr.mScreen.get());
|
2007-03-22 10:30:00 -07:00
|
|
|
return retScreen;
|
|
|
|
}
|
|
|
|
} // for each screen.
|
|
|
|
|
|
|
|
retScreen = new nsScreenWin(inScreen);
|
2014-07-14 00:52:00 -07:00
|
|
|
mScreenList.AppendElement(ScreenListItem(inScreen, retScreen));
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
NS_IF_ADDREF(retScreen);
|
|
|
|
return retScreen;
|
|
|
|
}
|
|
|
|
|
2014-07-14 10:22:26 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsScreenManagerWin::ScreenForId(uint32_t aId, nsIScreen **outScreen)
|
|
|
|
{
|
|
|
|
*outScreen = nullptr;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < mScreenList.Length(); ++i) {
|
|
|
|
ScreenListItem& curr = mScreenList[i];
|
|
|
|
uint32_t id;
|
|
|
|
nsresult rv = curr.mScreen->GetId(&id);
|
|
|
|
if (NS_SUCCEEDED(rv) && id == aId) {
|
|
|
|
NS_IF_ADDREF(*outScreen = curr.mScreen.get());
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// ScreenForRect
|
|
|
|
//
|
|
|
|
// Returns the screen that contains the rectangle. If the rect overlaps
|
|
|
|
// multiple screens, it picks the screen with the greatest area of intersection.
|
|
|
|
//
|
2013-04-01 02:10:33 -07:00
|
|
|
// The coordinates are in pixels (not twips) and in logical screen coordinates.
|
2007-03-22 10:30:00 -07:00
|
|
|
//
|
|
|
|
NS_IMETHODIMP
|
2014-07-14 00:52:00 -07:00
|
|
|
nsScreenManagerWin::ScreenForRect(int32_t inLeft, int32_t inTop, int32_t inWidth, int32_t inHeight,
|
|
|
|
nsIScreen **outScreen)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2014-07-14 00:52:00 -07:00
|
|
|
if (!(inWidth || inHeight)) {
|
|
|
|
NS_WARNING("trying to find screen for sizeless window, using primary monitor");
|
|
|
|
*outScreen = CreateNewScreenObject(nullptr); // addrefs
|
2007-03-22 10:30:00 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2013-04-01 02:10:33 -07:00
|
|
|
// convert coordinates from logical to device pixels for MonitorFromRect
|
2013-04-09 01:39:47 -07:00
|
|
|
double dpiScale = nsIWidget::DefaultScaleOverride();
|
|
|
|
if (dpiScale <= 0.0) {
|
|
|
|
dpiScale = gfxWindowsPlatform::GetPlatform()->GetDPIScale();
|
|
|
|
}
|
2013-04-01 02:10:33 -07:00
|
|
|
RECT globalWindowBounds = {
|
|
|
|
NSToIntRound(dpiScale * inLeft),
|
|
|
|
NSToIntRound(dpiScale * inTop),
|
|
|
|
NSToIntRound(dpiScale * (inLeft + inWidth)),
|
|
|
|
NSToIntRound(dpiScale * (inTop + inHeight))
|
|
|
|
};
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2014-07-14 00:52:00 -07:00
|
|
|
HMONITOR genScreen = ::MonitorFromRect(&globalWindowBounds, MONITOR_DEFAULTTOPRIMARY);
|
|
|
|
|
|
|
|
*outScreen = CreateNewScreenObject(genScreen); // addrefs
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
} // ScreenForRect
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// GetPrimaryScreen
|
|
|
|
//
|
|
|
|
// The screen with the menubar/taskbar. This shouldn't be needed very
|
|
|
|
// often.
|
|
|
|
//
|
|
|
|
NS_IMETHODIMP
|
2014-07-14 00:52:00 -07:00
|
|
|
nsScreenManagerWin::GetPrimaryScreen(nsIScreen** aPrimaryScreen)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2014-07-14 00:52:00 -07:00
|
|
|
*aPrimaryScreen = CreateNewScreenObject(nullptr); // addrefs
|
2007-03-22 10:30:00 -07:00
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
} // GetPrimaryScreen
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// CountMonitors
|
|
|
|
//
|
|
|
|
// Will be called once for every monitor in the system. Just
|
|
|
|
// increments the parameter, which holds a ptr to a PRUin32 holding the
|
|
|
|
// count up to this point.
|
|
|
|
//
|
|
|
|
BOOL CALLBACK
|
2014-07-14 00:52:00 -07:00
|
|
|
CountMonitors(HMONITOR, HDC, LPRECT, LPARAM ioParam)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2012-08-22 08:56:38 -07:00
|
|
|
uint32_t* countPtr = reinterpret_cast<uint32_t*>(ioParam);
|
2007-03-22 10:30:00 -07:00
|
|
|
++(*countPtr);
|
|
|
|
|
|
|
|
return TRUE; // continue the enumeration
|
|
|
|
|
|
|
|
} // CountMonitors
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// GetNumberOfScreens
|
|
|
|
//
|
|
|
|
// Returns how many physical screens are available.
|
|
|
|
//
|
|
|
|
NS_IMETHODIMP
|
2014-07-14 00:52:00 -07:00
|
|
|
nsScreenManagerWin::GetNumberOfScreens(uint32_t *aNumberOfScreens)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2014-07-14 00:52:00 -07:00
|
|
|
if (mNumberOfScreens)
|
2007-03-22 10:30:00 -07:00
|
|
|
*aNumberOfScreens = mNumberOfScreens;
|
|
|
|
else {
|
2012-08-22 08:56:38 -07:00
|
|
|
uint32_t count = 0;
|
2012-07-30 07:20:58 -07:00
|
|
|
BOOL result = ::EnumDisplayMonitors(nullptr, nullptr, (MONITORENUMPROC)CountMonitors, (LPARAM)&count);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!result)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
*aNumberOfScreens = mNumberOfScreens = count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
} // GetNumberOfScreens
|
|
|
|
|
2013-04-09 14:07:02 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsScreenManagerWin::GetSystemDefaultScale(float *aDefaultScale)
|
|
|
|
{
|
|
|
|
*aDefaultScale = float(gfxWindowsPlatform::GetPlatform()->GetDPIScale());
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_IMETHODIMP
|
2014-07-14 00:52:00 -07:00
|
|
|
nsScreenManagerWin::ScreenForNativeWidget(void *aWidget, nsIScreen **outScreen)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2014-07-14 00:52:00 -07:00
|
|
|
HMONITOR mon = MonitorFromWindow((HWND) aWidget, MONITOR_DEFAULTTOPRIMARY);
|
|
|
|
*outScreen = CreateNewScreenObject(mon);
|
2007-03-22 10:30:00 -07:00
|
|
|
return NS_OK;
|
|
|
|
}
|