blob: 2254961d146cb1f5f80d20be0d2606a2377f995f [file] [log] [blame]
bungeman@google.com72cf4fc2014-03-21 22:48:32 +00001/*
2 * Copyright 2014 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "SkDWrite.h"
9#include "SkHRESULT.h"
10#include "SkOnce.h"
11#include "SkString.h"
12
13#include <dwrite.h>
14
halcanary96fcdcc2015-08-27 07:41:13 -070015static IDWriteFactory* gDWriteFactory = nullptr;
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000016
mtklein1b818772014-06-02 11:26:59 -070017static void release_dwrite_factory() {
18 if (gDWriteFactory) {
19 gDWriteFactory->Release();
20 }
21}
22
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000023static void create_dwrite_factory(IDWriteFactory** factory) {
24 typedef decltype(DWriteCreateFactory)* DWriteCreateFactoryProc;
25 DWriteCreateFactoryProc dWriteCreateFactoryProc = reinterpret_cast<DWriteCreateFactoryProc>(
26 GetProcAddress(LoadLibraryW(L"dwrite.dll"), "DWriteCreateFactory"));
27
28 if (!dWriteCreateFactoryProc) {
29 HRESULT hr = HRESULT_FROM_WIN32(GetLastError());
30 if (!IS_ERROR(hr)) {
31 hr = ERROR_PROC_NOT_FOUND;
32 }
33 HRVM(hr, "Could not get DWriteCreateFactory proc.");
34 }
35
36 HRVM(dWriteCreateFactoryProc(DWRITE_FACTORY_TYPE_SHARED,
37 __uuidof(IDWriteFactory),
38 reinterpret_cast<IUnknown**>(factory)),
39 "Could not create DirectWrite factory.");
mtklein1b818772014-06-02 11:26:59 -070040 atexit(release_dwrite_factory);
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000041}
42
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000043
mtklein199ba8e2015-01-13 08:40:23 -080044SK_DECLARE_STATIC_ONCE(once);
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000045IDWriteFactory* sk_get_dwrite_factory() {
mtklein1b818772014-06-02 11:26:59 -070046 SkOnce(&once, create_dwrite_factory, &gDWriteFactory);
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000047 return gDWriteFactory;
48}
49
50////////////////////////////////////////////////////////////////////////////////
51// String conversion
52
53/** Converts a utf8 string to a WCHAR string. */
54HRESULT sk_cstring_to_wchar(const char* skname, SkSMallocWCHAR* name) {
halcanary96fcdcc2015-08-27 07:41:13 -070055 int wlen = MultiByteToWideChar(CP_UTF8, 0, skname, -1, nullptr, 0);
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000056 if (0 == wlen) {
57 HRM(HRESULT_FROM_WIN32(GetLastError()),
58 "Could not get length for wchar to utf-8 conversion.");
59 }
60 name->reset(wlen);
61 wlen = MultiByteToWideChar(CP_UTF8, 0, skname, -1, name->get(), wlen);
62 if (0 == wlen) {
63 HRM(HRESULT_FROM_WIN32(GetLastError()), "Could not convert wchar to utf-8.");
64 }
65 return S_OK;
66}
67
68/** Converts a WCHAR string to a utf8 string. */
bungeman5e7b4f92014-08-25 10:16:01 -070069HRESULT sk_wchar_to_skstring(WCHAR* name, int nameLen, SkString* skname) {
halcanary96fcdcc2015-08-27 07:41:13 -070070 int len = WideCharToMultiByte(CP_UTF8, 0, name, nameLen, nullptr, 0, nullptr, nullptr);
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000071 if (0 == len) {
bungeman5e7b4f92014-08-25 10:16:01 -070072 if (nameLen <= 0) {
73 skname->reset();
74 return S_OK;
75 }
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000076 HRM(HRESULT_FROM_WIN32(GetLastError()),
77 "Could not get length for utf-8 to wchar conversion.");
78 }
bungeman5e7b4f92014-08-25 10:16:01 -070079 skname->resize(len);
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000080
halcanary96fcdcc2015-08-27 07:41:13 -070081 len = WideCharToMultiByte(CP_UTF8, 0, name, nameLen, skname->writable_str(), len, nullptr, nullptr);
bungeman@google.com72cf4fc2014-03-21 22:48:32 +000082 if (0 == len) {
83 HRM(HRESULT_FROM_WIN32(GetLastError()), "Could not convert utf-8 to wchar.");
84 }
85 return S_OK;
86}
87
88////////////////////////////////////////////////////////////////////////////////
89// Locale
90
91void sk_get_locale_string(IDWriteLocalizedStrings* names, const WCHAR* preferedLocale,
92 SkString* skname) {
93 UINT32 nameIndex = 0;
94 if (preferedLocale) {
95 // Ignore any errors and continue with index 0 if there is a problem.
96 BOOL nameExists;
97 names->FindLocaleName(preferedLocale, &nameIndex, &nameExists);
98 if (!nameExists) {
99 nameIndex = 0;
100 }
101 }
102
bungeman5e7b4f92014-08-25 10:16:01 -0700103 UINT32 nameLen;
104 HRVM(names->GetStringLength(nameIndex, &nameLen), "Could not get name length.");
bungeman@google.com72cf4fc2014-03-21 22:48:32 +0000105
bungeman5e7b4f92014-08-25 10:16:01 -0700106 SkSMallocWCHAR name(nameLen+1);
107 HRVM(names->GetString(nameIndex, name.get(), nameLen+1), "Could not get string.");
bungeman@google.com72cf4fc2014-03-21 22:48:32 +0000108
bungeman5e7b4f92014-08-25 10:16:01 -0700109 HRV(sk_wchar_to_skstring(name.get(), nameLen, skname));
bungeman@google.com72cf4fc2014-03-21 22:48:32 +0000110}
111
112HRESULT SkGetGetUserDefaultLocaleNameProc(SkGetUserDefaultLocaleNameProc* proc) {
113 *proc = reinterpret_cast<SkGetUserDefaultLocaleNameProc>(
114 GetProcAddress(LoadLibraryW(L"Kernel32.dll"), "GetUserDefaultLocaleName")
115 );
116 if (!*proc) {
117 HRESULT hr = HRESULT_FROM_WIN32(GetLastError());
118 if (!IS_ERROR(hr)) {
119 hr = ERROR_PROC_NOT_FOUND;
120 }
121 return hr;
122 }
123 return S_OK;
124}