247 lines
6.5 KiB
C++
247 lines
6.5 KiB
C++
/*
|
|
* Platform Specification Implementation
|
|
* Copyright(C) 2003-2013 Jinhao(cnjinhao@hotmail.com)
|
|
*
|
|
* Distributed under the Boost Software License, Version 1.0.
|
|
* (See accompanying file LICENSE_1_0.txt or copy at
|
|
* http://www.boost.org/LICENSE_1_0.txt)
|
|
*
|
|
* @file: nana/detail/platform_spec.cpp
|
|
*
|
|
* This file provides basis class and data structrue that required by nana
|
|
*/
|
|
#include <nana/config.hpp>
|
|
|
|
#include PLATFORM_SPEC_HPP
|
|
#include <shellapi.h>
|
|
#include <stdexcept>
|
|
|
|
namespace nana
|
|
{
|
|
namespace detail
|
|
{
|
|
drawable_impl_type::drawable_impl_type()
|
|
: pixbuf_ptr(nullptr), bytes_per_line(0),
|
|
fgcolor_(0xFFFFFFFF)
|
|
{
|
|
pen.handle = nullptr;
|
|
pen.color = nana::null_color;
|
|
pen.style = -1;
|
|
pen.width = -1;
|
|
|
|
brush.handle = nullptr;
|
|
brush.style = brush_spec::Solid;
|
|
brush.color = nana::null_color;
|
|
|
|
round_region.handle = nullptr;
|
|
round_region.radius_x = round_region.radius_y = 0;
|
|
|
|
string.tab_length = 4;
|
|
string.tab_pixels = 0;
|
|
string.whitespace_pixels = 0;
|
|
}
|
|
|
|
drawable_impl_type::~drawable_impl_type()
|
|
{
|
|
::DeleteDC(context);
|
|
::DeleteObject(pixmap);
|
|
::DeleteObject(pen.handle);
|
|
::DeleteObject(brush.handle);
|
|
::DeleteObject(round_region.handle);
|
|
}
|
|
|
|
void drawable_impl_type::fgcolor(nana::color_t col)
|
|
{
|
|
if(this->fgcolor_ != col)
|
|
{
|
|
::SetTextColor(context, NANA_RGB(col));
|
|
fgcolor_ = col;
|
|
}
|
|
}
|
|
|
|
void drawable_impl_type::pen_spec::set(HDC context, int style, int width, nana::color_t color)
|
|
{
|
|
if(this->color != color || this->width != width || this->style != style)
|
|
{
|
|
this->color = color;
|
|
this->width = width;
|
|
this->style = style;
|
|
this->handle = ::CreatePen(style, width, NANA_RGB(color));
|
|
::DeleteObject(::SelectObject(context, this->handle));
|
|
}
|
|
}
|
|
|
|
void drawable_impl_type::brush_spec::set(HDC context, drawable_impl_type::brush_spec::t style, nana::color_t color)
|
|
{
|
|
if(this->color != color || this->style != style)
|
|
{
|
|
this->color = color;
|
|
this->style = style;
|
|
switch(style)
|
|
{
|
|
case brush_spec::HatchBDiagonal:
|
|
this->handle = ::CreateHatchBrush(HS_BDIAGONAL, NANA_RGB(color));
|
|
break;
|
|
case brush_spec::Solid:
|
|
default:
|
|
this->style = brush_spec::Solid;
|
|
this->handle = ::CreateSolidBrush(NANA_RGB(color));
|
|
break;
|
|
}
|
|
::DeleteObject(::SelectObject(context, this->handle));
|
|
}
|
|
}
|
|
|
|
void drawable_impl_type::round_region_spec::set(const nana::rectangle& r, unsigned radius_x, unsigned radius_y)
|
|
{
|
|
if(this->r != r || this->radius_x != radius_x || this->radius_y != radius_y)
|
|
{
|
|
if(handle)
|
|
::DeleteObject(this->handle);
|
|
this->r = r;
|
|
this->radius_x = radius_x;
|
|
this->radius_y = radius_y;
|
|
handle = ::CreateRoundRectRgn(r.x, r.y, r.x + static_cast<int>(r.width) + 1, r.y + static_cast<int>(r.height) + 1, static_cast<int>(radius_x + 1), static_cast<int>(radius_y + 1));
|
|
}
|
|
}
|
|
|
|
//struct font_tag::deleter
|
|
void font_tag::deleter::operator()(const font_tag* tag) const
|
|
{
|
|
if(tag && tag->handle)
|
|
::DeleteObject(tag->handle);
|
|
delete tag;
|
|
}
|
|
//end struct font_tag::deleter
|
|
|
|
//class platform_spec
|
|
platform_spec::co_initializer::co_initializer()
|
|
: ole32_(::LoadLibrary(STR("OLE32.DLL")))
|
|
{
|
|
if(ole32_)
|
|
{
|
|
typedef HRESULT (__stdcall *CoInitializeEx_t)(LPVOID, DWORD);
|
|
CoInitializeEx_t fn_init = reinterpret_cast<CoInitializeEx_t>(::GetProcAddress(ole32_, "CoInitializeEx"));
|
|
if(0 == fn_init)
|
|
{
|
|
::FreeLibrary(ole32_);
|
|
ole32_ = 0;
|
|
throw std::runtime_error("Nana.PlatformSpec.Co_initializer: Can't locate the CoInitializeEx().");
|
|
}
|
|
else
|
|
fn_init(0, COINIT_APARTMENTTHREADED | /*COINIT_DISABLE_OLE1DDE =*/0x4);
|
|
}
|
|
else
|
|
throw std::runtime_error("Nana.PlatformSpec.Co_initializer: No Ole32.DLL Loaded.");
|
|
}
|
|
|
|
platform_spec::co_initializer::~co_initializer()
|
|
{
|
|
if(ole32_)
|
|
{
|
|
typedef void (__stdcall *CoUninitialize_t)(void);
|
|
CoUninitialize_t fn_unin = reinterpret_cast<CoUninitialize_t>(::GetProcAddress(ole32_, "CoUninitialize"));
|
|
if(fn_unin)
|
|
fn_unin();
|
|
::FreeLibrary(ole32_);
|
|
}
|
|
}
|
|
|
|
platform_spec::platform_spec()
|
|
{
|
|
//Create default font object.
|
|
NONCLIENTMETRICS metrics = {};
|
|
metrics.cbSize = sizeof metrics;
|
|
#if(WINVER >= 0x0600)
|
|
OSVERSIONINFO osvi = {};
|
|
osvi.dwOSVersionInfoSize = sizeof(osvi);
|
|
::GetVersionEx(&osvi);
|
|
if (osvi.dwMajorVersion < 6)
|
|
metrics.cbSize -= sizeof(metrics.iPaddedBorderWidth);
|
|
#endif
|
|
::SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof metrics, &metrics, 0);
|
|
def_font_ptr_ = make_native_font(metrics.lfMessageFont.lfFaceName, font_size_to_height(9), 400, false, false, false);
|
|
}
|
|
|
|
const platform_spec::font_ptr_t& platform_spec::default_native_font() const
|
|
{
|
|
return def_font_ptr_;
|
|
}
|
|
|
|
void platform_spec::default_native_font(const font_ptr_t& fp)
|
|
{
|
|
def_font_ptr_ = fp;
|
|
}
|
|
|
|
unsigned platform_spec::font_size_to_height(unsigned size) const
|
|
{
|
|
HDC hdc = ::GetDC(0);
|
|
size = ::MulDiv(int(size), ::GetDeviceCaps(hdc, LOGPIXELSY), 72);
|
|
::ReleaseDC(0, hdc);
|
|
return size;
|
|
}
|
|
|
|
unsigned platform_spec::font_height_to_size(unsigned height) const
|
|
{
|
|
HDC hdc = ::GetDC(0);
|
|
unsigned pixels = ::GetDeviceCaps(hdc, LOGPIXELSY);
|
|
::ReleaseDC(0, hdc);
|
|
|
|
height = static_cast<unsigned>(static_cast<long long>(72) * height / pixels);
|
|
return height;
|
|
}
|
|
|
|
platform_spec::font_ptr_t platform_spec::make_native_font(const nana::char_t* name, unsigned height, unsigned weight, bool italic, bool underline, bool strike_out)
|
|
{
|
|
::LOGFONT logfont;
|
|
memset(&logfont, 0, sizeof logfont);
|
|
|
|
strcpy(logfont.lfFaceName, (name && *name ? name : def_font_ptr_->name.c_str()));
|
|
|
|
logfont.lfCharSet = DEFAULT_CHARSET;
|
|
HDC hdc = ::GetDC(0);
|
|
logfont.lfHeight = -static_cast<int>(height);
|
|
::ReleaseDC(0, hdc);
|
|
|
|
logfont.lfWidth = 0;
|
|
logfont.lfWeight = weight;
|
|
logfont.lfQuality = PROOF_QUALITY;
|
|
logfont.lfPitchAndFamily = FIXED_PITCH;
|
|
logfont.lfItalic = italic;
|
|
logfont.lfUnderline = underline;
|
|
logfont.lfStrikeOut = strike_out;
|
|
HFONT result = ::CreateFontIndirect(&logfont);
|
|
|
|
if(result)
|
|
{
|
|
font_tag * impl = new font_tag;
|
|
impl->name = logfont.lfFaceName;
|
|
impl->height = height;
|
|
impl->weight = weight;
|
|
impl->italic = italic;
|
|
impl->underline = underline;
|
|
impl->strikeout = strike_out;
|
|
impl->handle = result;
|
|
return std::shared_ptr<font_tag>(impl, font_tag::deleter());
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
platform_spec& platform_spec::instance()
|
|
{
|
|
static platform_spec object;
|
|
return object;
|
|
}
|
|
|
|
void platform_spec::keep_window_icon(native_window_type wd, const paint::image& img)
|
|
{
|
|
iconbase_[wd] = img;
|
|
}
|
|
|
|
void platform_spec::release_window_icon(native_window_type wd)
|
|
{
|
|
iconbase_.erase(wd);
|
|
}
|
|
}//end namespace detail
|
|
}//end namespace nana
|