2
0

display_server_windows.cpp 131 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377
  1. /**************************************************************************/
  2. /* display_server_windows.cpp */
  3. /**************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /**************************************************************************/
  8. /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
  9. /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /**************************************************************************/
  30. #include "display_server_windows.h"
  31. #include "core/io/marshalls.h"
  32. #include "core/math/geometry_2d.h"
  33. #include "main/main.h"
  34. #include "os_windows.h"
  35. #include "scene/resources/texture.h"
  36. #include <avrt.h>
  37. #include <dwmapi.h>
  38. #ifndef DWMWA_USE_IMMERSIVE_DARK_MODE
  39. #define DWMWA_USE_IMMERSIVE_DARK_MODE 20
  40. #endif
  41. #if defined(GLES3_ENABLED)
  42. #include "drivers/gles3/rasterizer_gles3.h"
  43. #endif
  44. #if defined(__GNUC__)
  45. // Workaround GCC warning from -Wcast-function-type.
  46. #define GetProcAddress (void *)GetProcAddress
  47. #endif
  48. static String format_error_message(DWORD id) {
  49. LPWSTR messageBuffer = nullptr;
  50. size_t size = FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
  51. nullptr, id, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPWSTR)&messageBuffer, 0, nullptr);
  52. String msg = "Error " + itos(id) + ": " + String::utf16((const char16_t *)messageBuffer, size);
  53. LocalFree(messageBuffer);
  54. return msg;
  55. }
  56. static void track_mouse_leave_event(HWND hWnd) {
  57. TRACKMOUSEEVENT tme;
  58. tme.cbSize = sizeof(TRACKMOUSEEVENT);
  59. tme.dwFlags = TME_LEAVE;
  60. tme.hwndTrack = hWnd;
  61. tme.dwHoverTime = HOVER_DEFAULT;
  62. TrackMouseEvent(&tme);
  63. }
  64. bool DisplayServerWindows::has_feature(Feature p_feature) const {
  65. switch (p_feature) {
  66. case FEATURE_SUBWINDOWS:
  67. case FEATURE_TOUCHSCREEN:
  68. case FEATURE_MOUSE:
  69. case FEATURE_MOUSE_WARP:
  70. case FEATURE_CLIPBOARD:
  71. case FEATURE_CURSOR_SHAPE:
  72. case FEATURE_CUSTOM_CURSOR_SHAPE:
  73. case FEATURE_IME:
  74. case FEATURE_WINDOW_TRANSPARENCY:
  75. case FEATURE_HIDPI:
  76. case FEATURE_ICON:
  77. case FEATURE_NATIVE_ICON:
  78. case FEATURE_SWAP_BUFFERS:
  79. case FEATURE_KEEP_SCREEN_ON:
  80. case FEATURE_TEXT_TO_SPEECH:
  81. return true;
  82. default:
  83. return false;
  84. }
  85. }
  86. String DisplayServerWindows::get_name() const {
  87. return "Windows";
  88. }
  89. void DisplayServerWindows::_set_mouse_mode_impl(MouseMode p_mode) {
  90. if (windows.has(MAIN_WINDOW_ID) && (p_mode == MOUSE_MODE_CAPTURED || p_mode == MOUSE_MODE_CONFINED || p_mode == MOUSE_MODE_CONFINED_HIDDEN)) {
  91. // Mouse is grabbed (captured or confined).
  92. WindowID window_id = _get_focused_window_or_popup();
  93. if (!windows.has(window_id)) {
  94. window_id = MAIN_WINDOW_ID;
  95. }
  96. WindowData &wd = windows[window_id];
  97. RECT clipRect;
  98. GetClientRect(wd.hWnd, &clipRect);
  99. ClientToScreen(wd.hWnd, (POINT *)&clipRect.left);
  100. ClientToScreen(wd.hWnd, (POINT *)&clipRect.right);
  101. ClipCursor(&clipRect);
  102. if (p_mode == MOUSE_MODE_CAPTURED) {
  103. center = window_get_size() / 2;
  104. POINT pos = { (int)center.x, (int)center.y };
  105. ClientToScreen(wd.hWnd, &pos);
  106. SetCursorPos(pos.x, pos.y);
  107. SetCapture(wd.hWnd);
  108. _register_raw_input_devices(window_id);
  109. }
  110. } else {
  111. // Mouse is free to move around (not captured or confined).
  112. ReleaseCapture();
  113. ClipCursor(nullptr);
  114. _register_raw_input_devices(INVALID_WINDOW_ID);
  115. }
  116. if (p_mode == MOUSE_MODE_HIDDEN || p_mode == MOUSE_MODE_CAPTURED || p_mode == MOUSE_MODE_CONFINED_HIDDEN) {
  117. if (hCursor == nullptr) {
  118. hCursor = SetCursor(nullptr);
  119. } else {
  120. SetCursor(nullptr);
  121. }
  122. } else {
  123. CursorShape c = cursor_shape;
  124. cursor_shape = CURSOR_MAX;
  125. cursor_set_shape(c);
  126. }
  127. }
  128. DisplayServer::WindowID DisplayServerWindows::_get_focused_window_or_popup() const {
  129. const List<WindowID>::Element *E = popup_list.back();
  130. if (E) {
  131. return E->get();
  132. }
  133. return last_focused_window;
  134. }
  135. void DisplayServerWindows::_register_raw_input_devices(WindowID p_target_window) {
  136. use_raw_input = true;
  137. RAWINPUTDEVICE rid[1] = {};
  138. rid[0].usUsagePage = 0x01;
  139. rid[0].usUsage = 0x02;
  140. rid[0].dwFlags = 0;
  141. if (p_target_window != INVALID_WINDOW_ID && windows.has(p_target_window)) {
  142. // Follow the defined window
  143. rid[0].hwndTarget = windows[p_target_window].hWnd;
  144. } else {
  145. // Follow the keyboard focus
  146. rid[0].hwndTarget = 0;
  147. }
  148. if (RegisterRawInputDevices(rid, 1, sizeof(rid[0])) == FALSE) {
  149. // Registration failed.
  150. use_raw_input = false;
  151. }
  152. }
  153. bool DisplayServerWindows::tts_is_speaking() const {
  154. ERR_FAIL_COND_V(!tts, false);
  155. return tts->is_speaking();
  156. }
  157. bool DisplayServerWindows::tts_is_paused() const {
  158. ERR_FAIL_COND_V(!tts, false);
  159. return tts->is_paused();
  160. }
  161. TypedArray<Dictionary> DisplayServerWindows::tts_get_voices() const {
  162. ERR_FAIL_COND_V(!tts, TypedArray<Dictionary>());
  163. return tts->get_voices();
  164. }
  165. void DisplayServerWindows::tts_speak(const String &p_text, const String &p_voice, int p_volume, float p_pitch, float p_rate, int p_utterance_id, bool p_interrupt) {
  166. ERR_FAIL_COND(!tts);
  167. tts->speak(p_text, p_voice, p_volume, p_pitch, p_rate, p_utterance_id, p_interrupt);
  168. }
  169. void DisplayServerWindows::tts_pause() {
  170. ERR_FAIL_COND(!tts);
  171. tts->pause();
  172. }
  173. void DisplayServerWindows::tts_resume() {
  174. ERR_FAIL_COND(!tts);
  175. tts->resume();
  176. }
  177. void DisplayServerWindows::tts_stop() {
  178. ERR_FAIL_COND(!tts);
  179. tts->stop();
  180. }
  181. void DisplayServerWindows::mouse_set_mode(MouseMode p_mode) {
  182. _THREAD_SAFE_METHOD_
  183. if (mouse_mode == p_mode) {
  184. // Already in the same mode; do nothing.
  185. return;
  186. }
  187. mouse_mode = p_mode;
  188. _set_mouse_mode_impl(p_mode);
  189. }
  190. DisplayServer::MouseMode DisplayServerWindows::mouse_get_mode() const {
  191. return mouse_mode;
  192. }
  193. void DisplayServerWindows::warp_mouse(const Point2i &p_position) {
  194. _THREAD_SAFE_METHOD_
  195. WindowID window_id = _get_focused_window_or_popup();
  196. if (!windows.has(window_id)) {
  197. return; // No focused window?
  198. }
  199. if (mouse_mode == MOUSE_MODE_CAPTURED) {
  200. old_x = p_position.x;
  201. old_y = p_position.y;
  202. } else {
  203. POINT p;
  204. p.x = p_position.x;
  205. p.y = p_position.y;
  206. ClientToScreen(windows[window_id].hWnd, &p);
  207. SetCursorPos(p.x, p.y);
  208. }
  209. }
  210. Point2i DisplayServerWindows::mouse_get_position() const {
  211. POINT p;
  212. GetCursorPos(&p);
  213. return Point2i(p.x, p.y) - _get_screens_origin();
  214. }
  215. BitField<MouseButtonMask> DisplayServerWindows::mouse_get_button_state() const {
  216. return last_button_state;
  217. }
  218. void DisplayServerWindows::clipboard_set(const String &p_text) {
  219. _THREAD_SAFE_METHOD_
  220. if (!windows.has(last_focused_window)) {
  221. return; // No focused window?
  222. }
  223. // Convert LF line endings to CRLF in clipboard content.
  224. // Otherwise, line endings won't be visible when pasted in other software.
  225. String text = p_text.replace("\r\n", "\n").replace("\n", "\r\n"); // Avoid \r\r\n.
  226. if (!OpenClipboard(windows[last_focused_window].hWnd)) {
  227. ERR_FAIL_MSG("Unable to open clipboard.");
  228. }
  229. EmptyClipboard();
  230. Char16String utf16 = text.utf16();
  231. HGLOBAL mem = GlobalAlloc(GMEM_MOVEABLE, (utf16.length() + 1) * sizeof(WCHAR));
  232. ERR_FAIL_COND_MSG(mem == nullptr, "Unable to allocate memory for clipboard contents.");
  233. LPWSTR lptstrCopy = (LPWSTR)GlobalLock(mem);
  234. memcpy(lptstrCopy, utf16.get_data(), (utf16.length() + 1) * sizeof(WCHAR));
  235. GlobalUnlock(mem);
  236. SetClipboardData(CF_UNICODETEXT, mem);
  237. // Set the CF_TEXT version (not needed?).
  238. CharString utf8 = text.utf8();
  239. mem = GlobalAlloc(GMEM_MOVEABLE, utf8.length() + 1);
  240. ERR_FAIL_COND_MSG(mem == nullptr, "Unable to allocate memory for clipboard contents.");
  241. LPTSTR ptr = (LPTSTR)GlobalLock(mem);
  242. memcpy(ptr, utf8.get_data(), utf8.length());
  243. ptr[utf8.length()] = 0;
  244. GlobalUnlock(mem);
  245. SetClipboardData(CF_TEXT, mem);
  246. CloseClipboard();
  247. }
  248. String DisplayServerWindows::clipboard_get() const {
  249. _THREAD_SAFE_METHOD_
  250. if (!windows.has(last_focused_window)) {
  251. return String(); // No focused window?
  252. }
  253. String ret;
  254. if (!OpenClipboard(windows[last_focused_window].hWnd)) {
  255. ERR_FAIL_V_MSG("", "Unable to open clipboard.");
  256. }
  257. if (IsClipboardFormatAvailable(CF_UNICODETEXT)) {
  258. HGLOBAL mem = GetClipboardData(CF_UNICODETEXT);
  259. if (mem != nullptr) {
  260. LPWSTR ptr = (LPWSTR)GlobalLock(mem);
  261. if (ptr != nullptr) {
  262. ret = String::utf16((const char16_t *)ptr);
  263. GlobalUnlock(mem);
  264. }
  265. }
  266. } else if (IsClipboardFormatAvailable(CF_TEXT)) {
  267. HGLOBAL mem = GetClipboardData(CF_UNICODETEXT);
  268. if (mem != nullptr) {
  269. LPTSTR ptr = (LPTSTR)GlobalLock(mem);
  270. if (ptr != nullptr) {
  271. ret.parse_utf8((const char *)ptr);
  272. GlobalUnlock(mem);
  273. }
  274. }
  275. }
  276. CloseClipboard();
  277. return ret;
  278. }
  279. typedef struct {
  280. int count;
  281. int screen;
  282. HMONITOR monitor;
  283. } EnumScreenData;
  284. static BOOL CALLBACK _MonitorEnumProcPrim(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  285. EnumScreenData *data = (EnumScreenData *)dwData;
  286. if ((lprcMonitor->left == 0) && (lprcMonitor->top == 0)) {
  287. data->screen = data->count;
  288. return FALSE;
  289. }
  290. data->count++;
  291. return TRUE;
  292. }
  293. static BOOL CALLBACK _MonitorEnumProcScreen(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  294. EnumScreenData *data = (EnumScreenData *)dwData;
  295. if (data->monitor == hMonitor) {
  296. data->screen = data->count;
  297. }
  298. data->count++;
  299. return TRUE;
  300. }
  301. static BOOL CALLBACK _MonitorEnumProcCount(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  302. int *data = (int *)dwData;
  303. (*data)++;
  304. return TRUE;
  305. }
  306. int DisplayServerWindows::get_screen_count() const {
  307. _THREAD_SAFE_METHOD_
  308. int data = 0;
  309. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcCount, (LPARAM)&data);
  310. return data;
  311. }
  312. int DisplayServerWindows::get_primary_screen() const {
  313. EnumScreenData data = { 0, 0, 0 };
  314. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcPrim, (LPARAM)&data);
  315. return data.screen;
  316. }
  317. typedef struct {
  318. int count;
  319. int screen;
  320. Point2 pos;
  321. } EnumPosData;
  322. static BOOL CALLBACK _MonitorEnumProcPos(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  323. EnumPosData *data = (EnumPosData *)dwData;
  324. if (data->count == data->screen) {
  325. data->pos.x = lprcMonitor->left;
  326. data->pos.y = lprcMonitor->top;
  327. }
  328. data->count++;
  329. return TRUE;
  330. }
  331. static BOOL CALLBACK _MonitorEnumProcOrigin(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  332. EnumPosData *data = (EnumPosData *)dwData;
  333. data->pos.x = MIN(data->pos.x, lprcMonitor->left);
  334. data->pos.y = MIN(data->pos.y, lprcMonitor->top);
  335. return TRUE;
  336. }
  337. Point2i DisplayServerWindows::_get_screens_origin() const {
  338. _THREAD_SAFE_METHOD_
  339. EnumPosData data = { 0, 0, Point2() };
  340. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcOrigin, (LPARAM)&data);
  341. return data.pos;
  342. }
  343. Point2i DisplayServerWindows::screen_get_position(int p_screen) const {
  344. _THREAD_SAFE_METHOD_
  345. switch (p_screen) {
  346. case SCREEN_PRIMARY: {
  347. p_screen = get_primary_screen();
  348. } break;
  349. case SCREEN_OF_MAIN_WINDOW: {
  350. p_screen = window_get_current_screen(MAIN_WINDOW_ID);
  351. } break;
  352. default:
  353. break;
  354. }
  355. EnumPosData data = { 0, p_screen, Point2() };
  356. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcPos, (LPARAM)&data);
  357. return data.pos - _get_screens_origin();
  358. }
  359. typedef struct {
  360. int count;
  361. int screen;
  362. Size2 size;
  363. } EnumSizeData;
  364. typedef struct {
  365. int count;
  366. int screen;
  367. Rect2i rect;
  368. } EnumRectData;
  369. typedef struct {
  370. int count;
  371. int screen;
  372. float rate;
  373. } EnumRefreshRateData;
  374. static BOOL CALLBACK _MonitorEnumProcSize(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  375. EnumSizeData *data = (EnumSizeData *)dwData;
  376. if (data->count == data->screen) {
  377. data->size.x = lprcMonitor->right - lprcMonitor->left;
  378. data->size.y = lprcMonitor->bottom - lprcMonitor->top;
  379. }
  380. data->count++;
  381. return TRUE;
  382. }
  383. Size2i DisplayServerWindows::screen_get_size(int p_screen) const {
  384. _THREAD_SAFE_METHOD_
  385. switch (p_screen) {
  386. case SCREEN_PRIMARY: {
  387. p_screen = get_primary_screen();
  388. } break;
  389. case SCREEN_OF_MAIN_WINDOW: {
  390. p_screen = window_get_current_screen(MAIN_WINDOW_ID);
  391. } break;
  392. default:
  393. break;
  394. }
  395. EnumSizeData data = { 0, p_screen, Size2() };
  396. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcSize, (LPARAM)&data);
  397. return data.size;
  398. }
  399. static BOOL CALLBACK _MonitorEnumProcUsableSize(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  400. EnumRectData *data = (EnumRectData *)dwData;
  401. if (data->count == data->screen) {
  402. MONITORINFO minfo;
  403. memset(&minfo, 0, sizeof(MONITORINFO));
  404. minfo.cbSize = sizeof(MONITORINFO);
  405. GetMonitorInfoA(hMonitor, &minfo);
  406. data->rect.position.x = minfo.rcWork.left;
  407. data->rect.position.y = minfo.rcWork.top;
  408. data->rect.size.x = minfo.rcWork.right - minfo.rcWork.left;
  409. data->rect.size.y = minfo.rcWork.bottom - minfo.rcWork.top;
  410. }
  411. data->count++;
  412. return TRUE;
  413. }
  414. static BOOL CALLBACK _MonitorEnumProcRefreshRate(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  415. EnumRefreshRateData *data = (EnumRefreshRateData *)dwData;
  416. if (data->count == data->screen) {
  417. MONITORINFOEXW minfo;
  418. memset(&minfo, 0, sizeof(minfo));
  419. minfo.cbSize = sizeof(minfo);
  420. GetMonitorInfoW(hMonitor, &minfo);
  421. DEVMODEW dm;
  422. memset(&dm, 0, sizeof(dm));
  423. dm.dmSize = sizeof(dm);
  424. EnumDisplaySettingsW(minfo.szDevice, ENUM_CURRENT_SETTINGS, &dm);
  425. data->rate = dm.dmDisplayFrequency;
  426. }
  427. data->count++;
  428. return TRUE;
  429. }
  430. Rect2i DisplayServerWindows::screen_get_usable_rect(int p_screen) const {
  431. _THREAD_SAFE_METHOD_
  432. switch (p_screen) {
  433. case SCREEN_PRIMARY: {
  434. p_screen = get_primary_screen();
  435. } break;
  436. case SCREEN_OF_MAIN_WINDOW: {
  437. p_screen = window_get_current_screen(MAIN_WINDOW_ID);
  438. } break;
  439. default:
  440. break;
  441. }
  442. EnumRectData data = { 0, p_screen, Rect2i() };
  443. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcUsableSize, (LPARAM)&data);
  444. data.rect.position -= _get_screens_origin();
  445. return data.rect;
  446. }
  447. typedef struct {
  448. int count;
  449. int screen;
  450. int dpi;
  451. } EnumDpiData;
  452. enum _MonitorDpiType {
  453. MDT_Effective_DPI = 0,
  454. MDT_Angular_DPI = 1,
  455. MDT_Raw_DPI = 2,
  456. MDT_Default = MDT_Effective_DPI
  457. };
  458. static int QueryDpiForMonitor(HMONITOR hmon, _MonitorDpiType dpiType = MDT_Default) {
  459. int dpiX = 96, dpiY = 96;
  460. static HMODULE Shcore = nullptr;
  461. typedef HRESULT(WINAPI * GetDPIForMonitor_t)(HMONITOR hmonitor, _MonitorDpiType dpiType, UINT * dpiX, UINT * dpiY);
  462. static GetDPIForMonitor_t getDPIForMonitor = nullptr;
  463. if (Shcore == nullptr) {
  464. Shcore = LoadLibraryW(L"Shcore.dll");
  465. getDPIForMonitor = Shcore ? (GetDPIForMonitor_t)GetProcAddress(Shcore, "GetDpiForMonitor") : nullptr;
  466. if ((Shcore == nullptr) || (getDPIForMonitor == nullptr)) {
  467. if (Shcore) {
  468. FreeLibrary(Shcore);
  469. }
  470. Shcore = (HMODULE)INVALID_HANDLE_VALUE;
  471. }
  472. }
  473. UINT x = 0, y = 0;
  474. if (hmon && (Shcore != (HMODULE)INVALID_HANDLE_VALUE)) {
  475. HRESULT hr = getDPIForMonitor(hmon, dpiType /*MDT_Effective_DPI*/, &x, &y);
  476. if (SUCCEEDED(hr) && (x > 0) && (y > 0)) {
  477. dpiX = (int)x;
  478. dpiY = (int)y;
  479. }
  480. } else {
  481. static int overallX = 0, overallY = 0;
  482. if (overallX <= 0 || overallY <= 0) {
  483. HDC hdc = GetDC(nullptr);
  484. if (hdc) {
  485. overallX = GetDeviceCaps(hdc, LOGPIXELSX);
  486. overallY = GetDeviceCaps(hdc, LOGPIXELSY);
  487. ReleaseDC(nullptr, hdc);
  488. }
  489. }
  490. if (overallX > 0 && overallY > 0) {
  491. dpiX = overallX;
  492. dpiY = overallY;
  493. }
  494. }
  495. return (dpiX + dpiY) / 2;
  496. }
  497. static BOOL CALLBACK _MonitorEnumProcDpi(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData) {
  498. EnumDpiData *data = (EnumDpiData *)dwData;
  499. if (data->count == data->screen) {
  500. data->dpi = QueryDpiForMonitor(hMonitor);
  501. }
  502. data->count++;
  503. return TRUE;
  504. }
  505. int DisplayServerWindows::screen_get_dpi(int p_screen) const {
  506. _THREAD_SAFE_METHOD_
  507. switch (p_screen) {
  508. case SCREEN_PRIMARY: {
  509. p_screen = get_primary_screen();
  510. } break;
  511. case SCREEN_OF_MAIN_WINDOW: {
  512. p_screen = window_get_current_screen(MAIN_WINDOW_ID);
  513. } break;
  514. default:
  515. break;
  516. }
  517. EnumDpiData data = { 0, p_screen, 72 };
  518. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcDpi, (LPARAM)&data);
  519. return data.dpi;
  520. }
  521. float DisplayServerWindows::screen_get_refresh_rate(int p_screen) const {
  522. _THREAD_SAFE_METHOD_
  523. switch (p_screen) {
  524. case SCREEN_PRIMARY: {
  525. p_screen = get_primary_screen();
  526. } break;
  527. case SCREEN_OF_MAIN_WINDOW: {
  528. p_screen = window_get_current_screen(MAIN_WINDOW_ID);
  529. } break;
  530. default:
  531. break;
  532. }
  533. EnumRefreshRateData data = { 0, p_screen, SCREEN_REFRESH_RATE_FALLBACK };
  534. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcRefreshRate, (LPARAM)&data);
  535. return data.rate;
  536. }
  537. void DisplayServerWindows::screen_set_keep_on(bool p_enable) {
  538. if (keep_screen_on == p_enable) {
  539. return;
  540. }
  541. if (p_enable) {
  542. const String reason = "Godot Engine running with display/window/energy_saving/keep_screen_on = true";
  543. Char16String reason_utf16 = reason.utf16();
  544. REASON_CONTEXT context;
  545. context.Version = POWER_REQUEST_CONTEXT_VERSION;
  546. context.Flags = POWER_REQUEST_CONTEXT_SIMPLE_STRING;
  547. context.Reason.SimpleReasonString = (LPWSTR)(reason_utf16.ptrw());
  548. power_request = PowerCreateRequest(&context);
  549. if (power_request == INVALID_HANDLE_VALUE) {
  550. print_error("Failed to enable screen_keep_on.");
  551. return;
  552. }
  553. if (PowerSetRequest(power_request, POWER_REQUEST_TYPE::PowerRequestSystemRequired) == 0) {
  554. print_error("Failed to request system sleep override.");
  555. return;
  556. }
  557. if (PowerSetRequest(power_request, POWER_REQUEST_TYPE::PowerRequestDisplayRequired) == 0) {
  558. print_error("Failed to request display timeout override.");
  559. return;
  560. }
  561. } else {
  562. PowerClearRequest(power_request, POWER_REQUEST_TYPE::PowerRequestSystemRequired);
  563. PowerClearRequest(power_request, POWER_REQUEST_TYPE::PowerRequestDisplayRequired);
  564. CloseHandle(power_request);
  565. power_request = nullptr;
  566. }
  567. keep_screen_on = p_enable;
  568. }
  569. bool DisplayServerWindows::screen_is_kept_on() const {
  570. return keep_screen_on;
  571. }
  572. Vector<DisplayServer::WindowID> DisplayServerWindows::get_window_list() const {
  573. _THREAD_SAFE_METHOD_
  574. Vector<DisplayServer::WindowID> ret;
  575. for (const KeyValue<WindowID, WindowData> &E : windows) {
  576. ret.push_back(E.key);
  577. }
  578. return ret;
  579. }
  580. DisplayServer::WindowID DisplayServerWindows::get_window_at_screen_position(const Point2i &p_position) const {
  581. Point2i offset = _get_screens_origin();
  582. POINT p;
  583. p.x = p_position.x + offset.x;
  584. p.y = p_position.y + offset.y;
  585. HWND hwnd = WindowFromPoint(p);
  586. for (const KeyValue<WindowID, WindowData> &E : windows) {
  587. if (E.value.hWnd == hwnd) {
  588. return E.key;
  589. }
  590. }
  591. return INVALID_WINDOW_ID;
  592. }
  593. DisplayServer::WindowID DisplayServerWindows::create_sub_window(WindowMode p_mode, VSyncMode p_vsync_mode, uint32_t p_flags, const Rect2i &p_rect) {
  594. _THREAD_SAFE_METHOD_
  595. WindowID window_id = _create_window(p_mode, p_vsync_mode, p_flags, p_rect);
  596. ERR_FAIL_COND_V_MSG(window_id == INVALID_WINDOW_ID, INVALID_WINDOW_ID, "Failed to create sub window.");
  597. WindowData &wd = windows[window_id];
  598. if (p_flags & WINDOW_FLAG_RESIZE_DISABLED_BIT) {
  599. wd.resizable = false;
  600. }
  601. if (p_flags & WINDOW_FLAG_BORDERLESS_BIT) {
  602. wd.borderless = true;
  603. }
  604. if (p_flags & WINDOW_FLAG_ALWAYS_ON_TOP_BIT && p_mode != WINDOW_MODE_FULLSCREEN && p_mode != WINDOW_MODE_EXCLUSIVE_FULLSCREEN) {
  605. wd.always_on_top = true;
  606. }
  607. if (p_flags & WINDOW_FLAG_NO_FOCUS_BIT) {
  608. wd.no_focus = true;
  609. }
  610. if (p_flags & WINDOW_FLAG_MOUSE_PASSTHROUGH_BIT) {
  611. wd.mpass = true;
  612. }
  613. if (p_flags & WINDOW_FLAG_POPUP_BIT) {
  614. wd.is_popup = true;
  615. }
  616. if (p_flags & WINDOW_FLAG_TRANSPARENT_BIT) {
  617. DWM_BLURBEHIND bb;
  618. ZeroMemory(&bb, sizeof(bb));
  619. HRGN hRgn = CreateRectRgn(0, 0, -1, -1);
  620. bb.dwFlags = DWM_BB_ENABLE | DWM_BB_BLURREGION;
  621. bb.hRgnBlur = hRgn;
  622. bb.fEnable = TRUE;
  623. DwmEnableBlurBehindWindow(wd.hWnd, &bb);
  624. wd.layered_window = true;
  625. }
  626. // Inherit icons from MAIN_WINDOW for all sub windows.
  627. HICON mainwindow_icon = (HICON)SendMessage(windows[MAIN_WINDOW_ID].hWnd, WM_GETICON, ICON_SMALL, 0);
  628. if (mainwindow_icon) {
  629. SendMessage(windows[window_id].hWnd, WM_SETICON, ICON_SMALL, (LPARAM)mainwindow_icon);
  630. }
  631. mainwindow_icon = (HICON)SendMessage(windows[MAIN_WINDOW_ID].hWnd, WM_GETICON, ICON_BIG, 0);
  632. if (mainwindow_icon) {
  633. SendMessage(windows[window_id].hWnd, WM_SETICON, ICON_BIG, (LPARAM)mainwindow_icon);
  634. }
  635. return window_id;
  636. }
  637. void DisplayServerWindows::show_window(WindowID p_id) {
  638. ERR_FAIL_COND(!windows.has(p_id));
  639. WindowData &wd = windows[p_id];
  640. popup_open(p_id);
  641. if (p_id != MAIN_WINDOW_ID) {
  642. _update_window_style(p_id);
  643. }
  644. if (wd.maximized) {
  645. ShowWindow(wd.hWnd, SW_SHOWMAXIMIZED);
  646. SetForegroundWindow(wd.hWnd); // Slightly higher priority.
  647. SetFocus(wd.hWnd); // Set keyboard focus.
  648. } else if (wd.minimized) {
  649. ShowWindow(wd.hWnd, SW_SHOWMINIMIZED);
  650. } else if (wd.no_focus || wd.is_popup) {
  651. // https://docs.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-showwindow
  652. ShowWindow(wd.hWnd, SW_SHOWNA);
  653. } else {
  654. ShowWindow(wd.hWnd, SW_SHOW);
  655. SetForegroundWindow(wd.hWnd); // Slightly higher priority.
  656. SetFocus(wd.hWnd); // Set keyboard focus.
  657. }
  658. if (wd.always_on_top) {
  659. SetWindowPos(wd.hWnd, HWND_TOPMOST, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOMOVE | SWP_NOSIZE | ((wd.no_focus || wd.is_popup) ? SWP_NOACTIVATE : 0));
  660. }
  661. }
  662. void DisplayServerWindows::delete_sub_window(WindowID p_window) {
  663. _THREAD_SAFE_METHOD_
  664. ERR_FAIL_COND(!windows.has(p_window));
  665. ERR_FAIL_COND_MSG(p_window == MAIN_WINDOW_ID, "Main window cannot be deleted.");
  666. popup_close(p_window);
  667. WindowData &wd = windows[p_window];
  668. while (wd.transient_children.size()) {
  669. window_set_transient(*wd.transient_children.begin(), INVALID_WINDOW_ID);
  670. }
  671. if (wd.transient_parent != INVALID_WINDOW_ID) {
  672. window_set_transient(p_window, INVALID_WINDOW_ID);
  673. }
  674. #ifdef VULKAN_ENABLED
  675. if (context_vulkan) {
  676. context_vulkan->window_destroy(p_window);
  677. }
  678. #endif
  679. #ifdef GLES3_ENABLED
  680. if (gl_manager) {
  681. gl_manager->window_destroy(p_window);
  682. }
  683. #endif
  684. if ((tablet_get_current_driver() == "wintab") && wintab_available && windows[p_window].wtctx) {
  685. wintab_WTClose(windows[p_window].wtctx);
  686. windows[p_window].wtctx = 0;
  687. }
  688. DestroyWindow(windows[p_window].hWnd);
  689. windows.erase(p_window);
  690. }
  691. void DisplayServerWindows::gl_window_make_current(DisplayServer::WindowID p_window_id) {
  692. #if defined(GLES3_ENABLED)
  693. if (gl_manager) {
  694. gl_manager->window_make_current(p_window_id);
  695. }
  696. #endif
  697. }
  698. int64_t DisplayServerWindows::window_get_native_handle(HandleType p_handle_type, WindowID p_window) const {
  699. ERR_FAIL_COND_V(!windows.has(p_window), 0);
  700. switch (p_handle_type) {
  701. case DISPLAY_HANDLE: {
  702. return 0; // Not supported.
  703. }
  704. case WINDOW_HANDLE: {
  705. return (int64_t)windows[p_window].hWnd;
  706. }
  707. #if defined(GLES3_ENABLED)
  708. case WINDOW_VIEW: {
  709. if (gl_manager) {
  710. return (int64_t)gl_manager->get_hdc(p_window);
  711. }
  712. return 0;
  713. }
  714. case OPENGL_CONTEXT: {
  715. if (gl_manager) {
  716. return (int64_t)gl_manager->get_hglrc(p_window);
  717. }
  718. return 0;
  719. }
  720. #endif
  721. default: {
  722. return 0;
  723. }
  724. }
  725. }
  726. void DisplayServerWindows::window_attach_instance_id(ObjectID p_instance, WindowID p_window) {
  727. _THREAD_SAFE_METHOD_
  728. ERR_FAIL_COND(!windows.has(p_window));
  729. windows[p_window].instance_id = p_instance;
  730. }
  731. ObjectID DisplayServerWindows::window_get_attached_instance_id(WindowID p_window) const {
  732. _THREAD_SAFE_METHOD_
  733. ERR_FAIL_COND_V(!windows.has(p_window), ObjectID());
  734. return windows[p_window].instance_id;
  735. }
  736. void DisplayServerWindows::window_set_rect_changed_callback(const Callable &p_callable, WindowID p_window) {
  737. _THREAD_SAFE_METHOD_
  738. ERR_FAIL_COND(!windows.has(p_window));
  739. windows[p_window].rect_changed_callback = p_callable;
  740. }
  741. void DisplayServerWindows::window_set_window_event_callback(const Callable &p_callable, WindowID p_window) {
  742. _THREAD_SAFE_METHOD_
  743. ERR_FAIL_COND(!windows.has(p_window));
  744. windows[p_window].event_callback = p_callable;
  745. }
  746. void DisplayServerWindows::window_set_input_event_callback(const Callable &p_callable, WindowID p_window) {
  747. _THREAD_SAFE_METHOD_
  748. ERR_FAIL_COND(!windows.has(p_window));
  749. windows[p_window].input_event_callback = p_callable;
  750. }
  751. void DisplayServerWindows::window_set_input_text_callback(const Callable &p_callable, WindowID p_window) {
  752. _THREAD_SAFE_METHOD_
  753. ERR_FAIL_COND(!windows.has(p_window));
  754. windows[p_window].input_text_callback = p_callable;
  755. }
  756. void DisplayServerWindows::window_set_drop_files_callback(const Callable &p_callable, WindowID p_window) {
  757. _THREAD_SAFE_METHOD_
  758. ERR_FAIL_COND(!windows.has(p_window));
  759. windows[p_window].drop_files_callback = p_callable;
  760. }
  761. void DisplayServerWindows::window_set_title(const String &p_title, WindowID p_window) {
  762. _THREAD_SAFE_METHOD_
  763. ERR_FAIL_COND(!windows.has(p_window));
  764. SetWindowTextW(windows[p_window].hWnd, (LPCWSTR)(p_title.utf16().get_data()));
  765. }
  766. void DisplayServerWindows::window_set_mouse_passthrough(const Vector<Vector2> &p_region, WindowID p_window) {
  767. _THREAD_SAFE_METHOD_
  768. ERR_FAIL_COND(!windows.has(p_window));
  769. windows[p_window].mpath = p_region;
  770. _update_window_mouse_passthrough(p_window);
  771. }
  772. void DisplayServerWindows::_update_window_mouse_passthrough(WindowID p_window) {
  773. ERR_FAIL_COND(!windows.has(p_window));
  774. if (windows[p_window].mpass || windows[p_window].mpath.size() == 0) {
  775. SetWindowRgn(windows[p_window].hWnd, nullptr, TRUE);
  776. } else {
  777. POINT *points = (POINT *)memalloc(sizeof(POINT) * windows[p_window].mpath.size());
  778. for (int i = 0; i < windows[p_window].mpath.size(); i++) {
  779. if (windows[p_window].borderless) {
  780. points[i].x = windows[p_window].mpath[i].x;
  781. points[i].y = windows[p_window].mpath[i].y;
  782. } else {
  783. points[i].x = windows[p_window].mpath[i].x + GetSystemMetrics(SM_CXSIZEFRAME);
  784. points[i].y = windows[p_window].mpath[i].y + GetSystemMetrics(SM_CYSIZEFRAME) + GetSystemMetrics(SM_CYCAPTION);
  785. }
  786. }
  787. HRGN region = CreatePolygonRgn(points, windows[p_window].mpath.size(), ALTERNATE);
  788. SetWindowRgn(windows[p_window].hWnd, region, TRUE);
  789. DeleteObject(region);
  790. memfree(points);
  791. }
  792. }
  793. int DisplayServerWindows::window_get_current_screen(WindowID p_window) const {
  794. _THREAD_SAFE_METHOD_
  795. ERR_FAIL_COND_V(!windows.has(p_window), -1);
  796. EnumScreenData data = { 0, 0, MonitorFromWindow(windows[p_window].hWnd, MONITOR_DEFAULTTONEAREST) };
  797. EnumDisplayMonitors(nullptr, nullptr, _MonitorEnumProcScreen, (LPARAM)&data);
  798. return data.screen;
  799. }
  800. void DisplayServerWindows::window_set_current_screen(int p_screen, WindowID p_window) {
  801. _THREAD_SAFE_METHOD_
  802. ERR_FAIL_COND(!windows.has(p_window));
  803. ERR_FAIL_INDEX(p_screen, get_screen_count());
  804. if (window_get_current_screen(p_window) == p_screen) {
  805. return;
  806. }
  807. const WindowData &wd = windows[p_window];
  808. if (wd.fullscreen) {
  809. Point2 pos = screen_get_position(p_screen) + _get_screens_origin();
  810. Size2 size = screen_get_size(p_screen);
  811. MoveWindow(wd.hWnd, pos.x, pos.y, size.width, size.height, TRUE);
  812. } else {
  813. Rect2i srect = screen_get_usable_rect(p_screen);
  814. Point2i wpos = window_get_position(p_window) - screen_get_position(window_get_current_screen(p_window));
  815. Size2i wsize = window_get_size(p_window);
  816. wpos += srect.position;
  817. wpos.x = CLAMP(wpos.x, srect.position.x, srect.position.x + srect.size.width - wsize.width / 3);
  818. wpos.y = CLAMP(wpos.y, srect.position.y, srect.position.y + srect.size.height - wsize.height / 3);
  819. window_set_position(wpos, p_window);
  820. }
  821. // Don't let the mouse leave the window when resizing to a smaller resolution.
  822. if (mouse_mode == MOUSE_MODE_CONFINED || mouse_mode == MOUSE_MODE_CONFINED_HIDDEN) {
  823. RECT crect;
  824. GetClientRect(wd.hWnd, &crect);
  825. ClientToScreen(wd.hWnd, (POINT *)&crect.left);
  826. ClientToScreen(wd.hWnd, (POINT *)&crect.right);
  827. ClipCursor(&crect);
  828. }
  829. }
  830. Point2i DisplayServerWindows::window_get_position(WindowID p_window) const {
  831. _THREAD_SAFE_METHOD_
  832. ERR_FAIL_COND_V(!windows.has(p_window), Point2i());
  833. const WindowData &wd = windows[p_window];
  834. if (wd.minimized) {
  835. return wd.last_pos;
  836. }
  837. POINT point;
  838. point.x = 0;
  839. point.y = 0;
  840. ClientToScreen(wd.hWnd, &point);
  841. return Point2i(point.x, point.y) - _get_screens_origin();
  842. }
  843. Point2i DisplayServerWindows::window_get_position_with_decorations(WindowID p_window) const {
  844. _THREAD_SAFE_METHOD_
  845. ERR_FAIL_COND_V(!windows.has(p_window), Point2i());
  846. const WindowData &wd = windows[p_window];
  847. if (wd.minimized) {
  848. return wd.last_pos;
  849. }
  850. RECT r;
  851. if (GetWindowRect(wd.hWnd, &r)) {
  852. return Point2i(r.left, r.top) - _get_screens_origin();
  853. }
  854. return Point2i();
  855. }
  856. void DisplayServerWindows::_update_real_mouse_position(WindowID p_window) {
  857. ERR_FAIL_COND(!windows.has(p_window));
  858. POINT mouse_pos;
  859. if (GetCursorPos(&mouse_pos) && ScreenToClient(windows[p_window].hWnd, &mouse_pos)) {
  860. if (mouse_pos.x > 0 && mouse_pos.y > 0 && mouse_pos.x <= windows[p_window].width && mouse_pos.y <= windows[p_window].height) {
  861. old_x = mouse_pos.x;
  862. old_y = mouse_pos.y;
  863. old_invalid = false;
  864. Input::get_singleton()->set_mouse_position(Point2i(mouse_pos.x, mouse_pos.y));
  865. }
  866. }
  867. }
  868. void DisplayServerWindows::window_set_position(const Point2i &p_position, WindowID p_window) {
  869. _THREAD_SAFE_METHOD_
  870. ERR_FAIL_COND(!windows.has(p_window));
  871. WindowData &wd = windows[p_window];
  872. if (wd.fullscreen || wd.maximized) {
  873. return;
  874. }
  875. Point2i offset = _get_screens_origin();
  876. RECT rc;
  877. rc.left = p_position.x + offset.x;
  878. rc.right = p_position.x + wd.width + offset.x;
  879. rc.bottom = p_position.y + wd.height + offset.y;
  880. rc.top = p_position.y + offset.y;
  881. const DWORD style = GetWindowLongPtr(wd.hWnd, GWL_STYLE);
  882. const DWORD exStyle = GetWindowLongPtr(wd.hWnd, GWL_EXSTYLE);
  883. AdjustWindowRectEx(&rc, style, false, exStyle);
  884. MoveWindow(wd.hWnd, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, TRUE);
  885. // Don't let the mouse leave the window when moved.
  886. if (mouse_mode == MOUSE_MODE_CONFINED || mouse_mode == MOUSE_MODE_CONFINED_HIDDEN) {
  887. RECT rect;
  888. GetClientRect(wd.hWnd, &rect);
  889. ClientToScreen(wd.hWnd, (POINT *)&rect.left);
  890. ClientToScreen(wd.hWnd, (POINT *)&rect.right);
  891. ClipCursor(&rect);
  892. }
  893. wd.last_pos = p_position;
  894. _update_real_mouse_position(p_window);
  895. }
  896. void DisplayServerWindows::window_set_exclusive(WindowID p_window, bool p_exclusive) {
  897. _THREAD_SAFE_METHOD_
  898. ERR_FAIL_COND(!windows.has(p_window));
  899. WindowData &wd = windows[p_window];
  900. if (wd.exclusive != p_exclusive) {
  901. wd.exclusive = p_exclusive;
  902. if (wd.transient_parent != INVALID_WINDOW_ID) {
  903. if (wd.exclusive) {
  904. WindowData &wd_parent = windows[wd.transient_parent];
  905. SetWindowLongPtr(wd.hWnd, GWLP_HWNDPARENT, (LONG_PTR)wd_parent.hWnd);
  906. } else {
  907. SetWindowLongPtr(wd.hWnd, GWLP_HWNDPARENT, (LONG_PTR) nullptr);
  908. }
  909. }
  910. }
  911. }
  912. void DisplayServerWindows::window_set_transient(WindowID p_window, WindowID p_parent) {
  913. _THREAD_SAFE_METHOD_
  914. ERR_FAIL_COND(p_window == p_parent);
  915. ERR_FAIL_COND(!windows.has(p_window));
  916. WindowData &wd_window = windows[p_window];
  917. ERR_FAIL_COND(wd_window.transient_parent == p_parent);
  918. ERR_FAIL_COND_MSG(wd_window.always_on_top, "Windows with the 'on top' can't become transient.");
  919. if (p_parent == INVALID_WINDOW_ID) {
  920. // Remove transient.
  921. ERR_FAIL_COND(wd_window.transient_parent == INVALID_WINDOW_ID);
  922. ERR_FAIL_COND(!windows.has(wd_window.transient_parent));
  923. WindowData &wd_parent = windows[wd_window.transient_parent];
  924. wd_window.transient_parent = INVALID_WINDOW_ID;
  925. wd_parent.transient_children.erase(p_window);
  926. if (wd_window.exclusive) {
  927. SetWindowLongPtr(wd_window.hWnd, GWLP_HWNDPARENT, (LONG_PTR) nullptr);
  928. }
  929. } else {
  930. ERR_FAIL_COND(!windows.has(p_parent));
  931. ERR_FAIL_COND_MSG(wd_window.transient_parent != INVALID_WINDOW_ID, "Window already has a transient parent");
  932. WindowData &wd_parent = windows[p_parent];
  933. wd_window.transient_parent = p_parent;
  934. wd_parent.transient_children.insert(p_window);
  935. if (wd_window.exclusive) {
  936. SetWindowLongPtr(wd_window.hWnd, GWLP_HWNDPARENT, (LONG_PTR)wd_parent.hWnd);
  937. }
  938. }
  939. }
  940. void DisplayServerWindows::window_set_max_size(const Size2i p_size, WindowID p_window) {
  941. _THREAD_SAFE_METHOD_
  942. ERR_FAIL_COND(!windows.has(p_window));
  943. WindowData &wd = windows[p_window];
  944. if ((p_size != Size2()) && ((p_size.x < wd.min_size.x) || (p_size.y < wd.min_size.y))) {
  945. ERR_PRINT("Maximum window size can't be smaller than minimum window size!");
  946. return;
  947. }
  948. wd.max_size = p_size;
  949. }
  950. Size2i DisplayServerWindows::window_get_max_size(WindowID p_window) const {
  951. _THREAD_SAFE_METHOD_
  952. ERR_FAIL_COND_V(!windows.has(p_window), Size2i());
  953. const WindowData &wd = windows[p_window];
  954. return wd.max_size;
  955. }
  956. void DisplayServerWindows::window_set_min_size(const Size2i p_size, WindowID p_window) {
  957. _THREAD_SAFE_METHOD_
  958. ERR_FAIL_COND(!windows.has(p_window));
  959. WindowData &wd = windows[p_window];
  960. if ((p_size != Size2()) && (wd.max_size != Size2()) && ((p_size.x > wd.max_size.x) || (p_size.y > wd.max_size.y))) {
  961. ERR_PRINT("Minimum window size can't be larger than maximum window size!");
  962. return;
  963. }
  964. wd.min_size = p_size;
  965. }
  966. Size2i DisplayServerWindows::window_get_min_size(WindowID p_window) const {
  967. _THREAD_SAFE_METHOD_
  968. ERR_FAIL_COND_V(!windows.has(p_window), Size2i());
  969. const WindowData &wd = windows[p_window];
  970. return wd.min_size;
  971. }
  972. void DisplayServerWindows::window_set_size(const Size2i p_size, WindowID p_window) {
  973. _THREAD_SAFE_METHOD_
  974. ERR_FAIL_COND(!windows.has(p_window));
  975. WindowData &wd = windows[p_window];
  976. if (wd.fullscreen || wd.maximized) {
  977. return;
  978. }
  979. int w = p_size.width;
  980. int h = p_size.height;
  981. wd.width = w;
  982. wd.height = h;
  983. #if defined(VULKAN_ENABLED)
  984. if (context_vulkan) {
  985. context_vulkan->window_resize(p_window, w, h);
  986. }
  987. #endif
  988. #if defined(GLES3_ENABLED)
  989. if (gl_manager) {
  990. gl_manager->window_resize(p_window, w, h);
  991. }
  992. #endif
  993. RECT rect;
  994. GetWindowRect(wd.hWnd, &rect);
  995. if (!wd.borderless) {
  996. RECT crect;
  997. GetClientRect(wd.hWnd, &crect);
  998. w += (rect.right - rect.left) - (crect.right - crect.left);
  999. h += (rect.bottom - rect.top) - (crect.bottom - crect.top);
  1000. }
  1001. MoveWindow(wd.hWnd, rect.left, rect.top, w, h, TRUE);
  1002. // Don't let the mouse leave the window when resizing to a smaller resolution.
  1003. if (mouse_mode == MOUSE_MODE_CONFINED || mouse_mode == MOUSE_MODE_CONFINED_HIDDEN) {
  1004. RECT crect;
  1005. GetClientRect(wd.hWnd, &crect);
  1006. ClientToScreen(wd.hWnd, (POINT *)&crect.left);
  1007. ClientToScreen(wd.hWnd, (POINT *)&crect.right);
  1008. ClipCursor(&crect);
  1009. }
  1010. }
  1011. Size2i DisplayServerWindows::window_get_size(WindowID p_window) const {
  1012. _THREAD_SAFE_METHOD_
  1013. ERR_FAIL_COND_V(!windows.has(p_window), Size2i());
  1014. const WindowData &wd = windows[p_window];
  1015. // GetClientRect() returns a zero rect for a minimized window, so we need to get the size in another way.
  1016. if (wd.minimized) {
  1017. return Size2(wd.width, wd.height);
  1018. }
  1019. RECT r;
  1020. if (GetClientRect(wd.hWnd, &r)) { // Retrieves area inside of window border, including decoration.
  1021. return Size2(r.right - r.left, r.bottom - r.top);
  1022. }
  1023. return Size2();
  1024. }
  1025. Size2i DisplayServerWindows::window_get_size_with_decorations(WindowID p_window) const {
  1026. _THREAD_SAFE_METHOD_
  1027. ERR_FAIL_COND_V(!windows.has(p_window), Size2i());
  1028. const WindowData &wd = windows[p_window];
  1029. RECT r;
  1030. if (GetWindowRect(wd.hWnd, &r)) { // Retrieves area inside of window border, including decoration.
  1031. return Size2(r.right - r.left, r.bottom - r.top);
  1032. }
  1033. return Size2();
  1034. }
  1035. void DisplayServerWindows::_get_window_style(bool p_main_window, bool p_fullscreen, bool p_multiwindow_fs, bool p_borderless, bool p_resizable, bool p_maximized, bool p_no_activate_focus, DWORD &r_style, DWORD &r_style_ex) {
  1036. // Windows docs for window styles:
  1037. // https://docs.microsoft.com/en-us/windows/win32/winmsg/window-styles
  1038. // https://docs.microsoft.com/en-us/windows/win32/winmsg/extended-window-styles
  1039. r_style = 0;
  1040. r_style_ex = WS_EX_WINDOWEDGE;
  1041. if (p_main_window) {
  1042. r_style_ex |= WS_EX_APPWINDOW;
  1043. r_style |= WS_VISIBLE;
  1044. }
  1045. if (p_fullscreen || p_borderless) {
  1046. r_style |= WS_POPUP; // p_borderless was WS_EX_TOOLWINDOW in the past.
  1047. if (p_fullscreen && p_multiwindow_fs) {
  1048. r_style |= WS_BORDER; // Allows child windows to be displayed on top of full screen.
  1049. }
  1050. } else {
  1051. if (p_resizable) {
  1052. if (p_maximized) {
  1053. r_style = WS_OVERLAPPEDWINDOW | WS_MAXIMIZE;
  1054. } else {
  1055. r_style = WS_OVERLAPPEDWINDOW;
  1056. }
  1057. } else {
  1058. r_style = WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU;
  1059. }
  1060. }
  1061. if (p_no_activate_focus) {
  1062. r_style_ex |= WS_EX_TOPMOST | WS_EX_NOACTIVATE;
  1063. }
  1064. if (!p_borderless && !p_no_activate_focus) {
  1065. r_style |= WS_VISIBLE;
  1066. }
  1067. r_style |= WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
  1068. r_style_ex |= WS_EX_ACCEPTFILES;
  1069. }
  1070. void DisplayServerWindows::_update_window_style(WindowID p_window, bool p_repaint) {
  1071. _THREAD_SAFE_METHOD_
  1072. ERR_FAIL_COND(!windows.has(p_window));
  1073. WindowData &wd = windows[p_window];
  1074. DWORD style = 0;
  1075. DWORD style_ex = 0;
  1076. _get_window_style(p_window == MAIN_WINDOW_ID, wd.fullscreen, wd.multiwindow_fs, wd.borderless, wd.resizable, wd.maximized, wd.no_focus || wd.is_popup, style, style_ex);
  1077. SetWindowLongPtr(wd.hWnd, GWL_STYLE, style);
  1078. SetWindowLongPtr(wd.hWnd, GWL_EXSTYLE, style_ex);
  1079. if (icon.is_valid()) {
  1080. set_icon(icon);
  1081. }
  1082. SetWindowPos(wd.hWnd, wd.always_on_top ? HWND_TOPMOST : HWND_NOTOPMOST, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOMOVE | SWP_NOSIZE | ((wd.no_focus || wd.is_popup) ? SWP_NOACTIVATE : 0));
  1083. if (p_repaint) {
  1084. RECT rect;
  1085. GetWindowRect(wd.hWnd, &rect);
  1086. MoveWindow(wd.hWnd, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, TRUE);
  1087. }
  1088. }
  1089. void DisplayServerWindows::window_set_mode(WindowMode p_mode, WindowID p_window) {
  1090. _THREAD_SAFE_METHOD_
  1091. ERR_FAIL_COND(!windows.has(p_window));
  1092. WindowData &wd = windows[p_window];
  1093. if (wd.fullscreen && p_mode != WINDOW_MODE_FULLSCREEN && p_mode != WINDOW_MODE_EXCLUSIVE_FULLSCREEN) {
  1094. RECT rect;
  1095. wd.fullscreen = false;
  1096. wd.multiwindow_fs = false;
  1097. wd.maximized = wd.was_maximized;
  1098. if (wd.pre_fs_valid) {
  1099. rect = wd.pre_fs_rect;
  1100. } else {
  1101. rect.left = 0;
  1102. rect.right = wd.width;
  1103. rect.top = 0;
  1104. rect.bottom = wd.height;
  1105. wd.pre_fs_valid = true;
  1106. }
  1107. _update_window_style(p_window, false);
  1108. MoveWindow(wd.hWnd, rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top, TRUE);
  1109. if (restore_mouse_trails > 1) {
  1110. SystemParametersInfoA(SPI_SETMOUSETRAILS, restore_mouse_trails, 0, 0);
  1111. restore_mouse_trails = 0;
  1112. }
  1113. } else if (p_mode == WINDOW_MODE_WINDOWED) {
  1114. ShowWindow(wd.hWnd, SW_RESTORE);
  1115. wd.maximized = false;
  1116. wd.minimized = false;
  1117. }
  1118. if (p_mode == WINDOW_MODE_MAXIMIZED) {
  1119. ShowWindow(wd.hWnd, SW_MAXIMIZE);
  1120. wd.maximized = true;
  1121. wd.minimized = false;
  1122. }
  1123. if (p_mode == WINDOW_MODE_MINIMIZED) {
  1124. ShowWindow(wd.hWnd, SW_MINIMIZE);
  1125. wd.maximized = false;
  1126. wd.minimized = true;
  1127. }
  1128. if (p_mode == WINDOW_MODE_EXCLUSIVE_FULLSCREEN) {
  1129. wd.multiwindow_fs = false;
  1130. _update_window_style(p_window, false);
  1131. } else {
  1132. wd.multiwindow_fs = true;
  1133. _update_window_style(p_window, false);
  1134. }
  1135. if ((p_mode == WINDOW_MODE_FULLSCREEN || p_mode == WINDOW_MODE_EXCLUSIVE_FULLSCREEN) && !wd.fullscreen) {
  1136. if (wd.minimized) {
  1137. ShowWindow(wd.hWnd, SW_RESTORE);
  1138. }
  1139. wd.was_maximized = wd.maximized;
  1140. if (wd.pre_fs_valid) {
  1141. GetWindowRect(wd.hWnd, &wd.pre_fs_rect);
  1142. }
  1143. int cs = window_get_current_screen(p_window);
  1144. Point2 pos = screen_get_position(cs) + _get_screens_origin();
  1145. Size2 size = screen_get_size(cs);
  1146. wd.fullscreen = true;
  1147. wd.maximized = false;
  1148. wd.minimized = false;
  1149. _update_window_style(p_window, false);
  1150. MoveWindow(wd.hWnd, pos.x, pos.y, size.width, size.height, TRUE);
  1151. // If the user has mouse trails enabled in windows, then sometimes the cursor disappears in fullscreen mode.
  1152. // Save number of trails so we can restore when exiting, then turn off mouse trails
  1153. SystemParametersInfoA(SPI_GETMOUSETRAILS, 0, &restore_mouse_trails, 0);
  1154. if (restore_mouse_trails > 1) {
  1155. SystemParametersInfoA(SPI_SETMOUSETRAILS, 0, 0, 0);
  1156. }
  1157. }
  1158. // Don't let the mouse leave the window when resizing to a smaller resolution.
  1159. if (mouse_mode == MOUSE_MODE_CONFINED || mouse_mode == MOUSE_MODE_CONFINED_HIDDEN) {
  1160. RECT crect;
  1161. GetClientRect(wd.hWnd, &crect);
  1162. ClientToScreen(wd.hWnd, (POINT *)&crect.left);
  1163. ClientToScreen(wd.hWnd, (POINT *)&crect.right);
  1164. ClipCursor(&crect);
  1165. }
  1166. }
  1167. DisplayServer::WindowMode DisplayServerWindows::window_get_mode(WindowID p_window) const {
  1168. _THREAD_SAFE_METHOD_
  1169. ERR_FAIL_COND_V(!windows.has(p_window), WINDOW_MODE_WINDOWED);
  1170. const WindowData &wd = windows[p_window];
  1171. if (wd.fullscreen) {
  1172. if (wd.multiwindow_fs) {
  1173. return WINDOW_MODE_FULLSCREEN;
  1174. } else {
  1175. return WINDOW_MODE_EXCLUSIVE_FULLSCREEN;
  1176. }
  1177. } else if (wd.minimized) {
  1178. return WINDOW_MODE_MINIMIZED;
  1179. } else if (wd.maximized) {
  1180. return WINDOW_MODE_MAXIMIZED;
  1181. } else {
  1182. return WINDOW_MODE_WINDOWED;
  1183. }
  1184. }
  1185. bool DisplayServerWindows::window_is_maximize_allowed(WindowID p_window) const {
  1186. _THREAD_SAFE_METHOD_
  1187. ERR_FAIL_COND_V(!windows.has(p_window), false);
  1188. // FIXME: Implement this, or confirm that it should always be true.
  1189. return true;
  1190. }
  1191. void DisplayServerWindows::window_set_flag(WindowFlags p_flag, bool p_enabled, WindowID p_window) {
  1192. _THREAD_SAFE_METHOD_
  1193. ERR_FAIL_COND(!windows.has(p_window));
  1194. WindowData &wd = windows[p_window];
  1195. switch (p_flag) {
  1196. case WINDOW_FLAG_RESIZE_DISABLED: {
  1197. wd.resizable = !p_enabled;
  1198. _update_window_style(p_window);
  1199. } break;
  1200. case WINDOW_FLAG_BORDERLESS: {
  1201. wd.borderless = p_enabled;
  1202. _update_window_style(p_window);
  1203. _update_window_mouse_passthrough(p_window);
  1204. ShowWindow(wd.hWnd, (wd.no_focus || wd.is_popup) ? SW_SHOWNOACTIVATE : SW_SHOW); // Show the window.
  1205. } break;
  1206. case WINDOW_FLAG_ALWAYS_ON_TOP: {
  1207. ERR_FAIL_COND_MSG(wd.transient_parent != INVALID_WINDOW_ID && p_enabled, "Transient windows can't become on top");
  1208. wd.always_on_top = p_enabled;
  1209. _update_window_style(p_window);
  1210. } break;
  1211. case WINDOW_FLAG_TRANSPARENT: {
  1212. if (p_enabled) {
  1213. //enable per-pixel alpha
  1214. DWM_BLURBEHIND bb;
  1215. ZeroMemory(&bb, sizeof(bb));
  1216. HRGN hRgn = CreateRectRgn(0, 0, -1, -1);
  1217. bb.dwFlags = DWM_BB_ENABLE | DWM_BB_BLURREGION;
  1218. bb.hRgnBlur = hRgn;
  1219. bb.fEnable = TRUE;
  1220. DwmEnableBlurBehindWindow(wd.hWnd, &bb);
  1221. wd.layered_window = true;
  1222. } else {
  1223. //disable per-pixel alpha
  1224. wd.layered_window = false;
  1225. DWM_BLURBEHIND bb;
  1226. ZeroMemory(&bb, sizeof(bb));
  1227. HRGN hRgn = CreateRectRgn(0, 0, -1, -1);
  1228. bb.dwFlags = DWM_BB_ENABLE | DWM_BB_BLURREGION;
  1229. bb.hRgnBlur = hRgn;
  1230. bb.fEnable = FALSE;
  1231. DwmEnableBlurBehindWindow(wd.hWnd, &bb);
  1232. }
  1233. } break;
  1234. case WINDOW_FLAG_NO_FOCUS: {
  1235. wd.no_focus = p_enabled;
  1236. _update_window_style(p_window);
  1237. } break;
  1238. case WINDOW_FLAG_MOUSE_PASSTHROUGH: {
  1239. wd.mpass = p_enabled;
  1240. _update_window_mouse_passthrough(p_window);
  1241. } break;
  1242. case WINDOW_FLAG_POPUP: {
  1243. ERR_FAIL_COND_MSG(p_window == MAIN_WINDOW_ID, "Main window can't be popup.");
  1244. ERR_FAIL_COND_MSG(IsWindowVisible(wd.hWnd) && (wd.is_popup != p_enabled), "Popup flag can't changed while window is opened.");
  1245. wd.is_popup = p_enabled;
  1246. } break;
  1247. default:
  1248. break;
  1249. }
  1250. }
  1251. bool DisplayServerWindows::window_get_flag(WindowFlags p_flag, WindowID p_window) const {
  1252. _THREAD_SAFE_METHOD_
  1253. ERR_FAIL_COND_V(!windows.has(p_window), false);
  1254. const WindowData &wd = windows[p_window];
  1255. switch (p_flag) {
  1256. case WINDOW_FLAG_RESIZE_DISABLED: {
  1257. return !wd.resizable;
  1258. } break;
  1259. case WINDOW_FLAG_BORDERLESS: {
  1260. return wd.borderless;
  1261. } break;
  1262. case WINDOW_FLAG_ALWAYS_ON_TOP: {
  1263. return wd.always_on_top;
  1264. } break;
  1265. case WINDOW_FLAG_TRANSPARENT: {
  1266. return wd.layered_window;
  1267. } break;
  1268. case WINDOW_FLAG_NO_FOCUS: {
  1269. return wd.no_focus;
  1270. } break;
  1271. case WINDOW_FLAG_MOUSE_PASSTHROUGH: {
  1272. return wd.mpass;
  1273. } break;
  1274. case WINDOW_FLAG_POPUP: {
  1275. return wd.is_popup;
  1276. } break;
  1277. default:
  1278. break;
  1279. }
  1280. return false;
  1281. }
  1282. void DisplayServerWindows::window_request_attention(WindowID p_window) {
  1283. _THREAD_SAFE_METHOD_
  1284. ERR_FAIL_COND(!windows.has(p_window));
  1285. const WindowData &wd = windows[p_window];
  1286. FLASHWINFO info;
  1287. info.cbSize = sizeof(FLASHWINFO);
  1288. info.hwnd = wd.hWnd;
  1289. info.dwFlags = FLASHW_TRAY;
  1290. info.dwTimeout = 0;
  1291. info.uCount = 2;
  1292. FlashWindowEx(&info);
  1293. }
  1294. void DisplayServerWindows::window_move_to_foreground(WindowID p_window) {
  1295. _THREAD_SAFE_METHOD_
  1296. ERR_FAIL_COND(!windows.has(p_window));
  1297. WindowData &wd = windows[p_window];
  1298. if (!wd.no_focus && !wd.is_popup) {
  1299. SetForegroundWindow(wd.hWnd);
  1300. }
  1301. }
  1302. bool DisplayServerWindows::window_can_draw(WindowID p_window) const {
  1303. _THREAD_SAFE_METHOD_
  1304. ERR_FAIL_COND_V(!windows.has(p_window), false);
  1305. const WindowData &wd = windows[p_window];
  1306. return !wd.minimized;
  1307. }
  1308. bool DisplayServerWindows::can_any_window_draw() const {
  1309. _THREAD_SAFE_METHOD_
  1310. for (const KeyValue<WindowID, WindowData> &E : windows) {
  1311. if (!E.value.minimized) {
  1312. return true;
  1313. }
  1314. }
  1315. return false;
  1316. }
  1317. Vector2i DisplayServerWindows::ime_get_selection() const {
  1318. _THREAD_SAFE_METHOD_
  1319. DisplayServer::WindowID window_id = _get_focused_window_or_popup();
  1320. const WindowData &wd = windows[window_id];
  1321. if (!wd.ime_active) {
  1322. return Vector2i();
  1323. }
  1324. int cursor = ImmGetCompositionStringW(wd.im_himc, GCS_CURSORPOS, nullptr, 0);
  1325. int32_t length = ImmGetCompositionStringW(wd.im_himc, GCS_COMPSTR, nullptr, 0);
  1326. wchar_t *string = reinterpret_cast<wchar_t *>(memalloc(length));
  1327. ImmGetCompositionStringW(wd.im_himc, GCS_COMPSTR, string, length);
  1328. int32_t utf32_cursor = 0;
  1329. for (int32_t i = 0; i < length / sizeof(wchar_t); i++) {
  1330. if ((string[i] & 0xfffffc00) == 0xd800) {
  1331. i++;
  1332. }
  1333. if (i < cursor) {
  1334. utf32_cursor++;
  1335. } else {
  1336. break;
  1337. }
  1338. }
  1339. memdelete(string);
  1340. return Vector2i(utf32_cursor, 0);
  1341. }
  1342. String DisplayServerWindows::ime_get_text() const {
  1343. _THREAD_SAFE_METHOD_
  1344. DisplayServer::WindowID window_id = _get_focused_window_or_popup();
  1345. const WindowData &wd = windows[window_id];
  1346. if (!wd.ime_active) {
  1347. return String();
  1348. }
  1349. String ret;
  1350. int32_t length = ImmGetCompositionStringW(wd.im_himc, GCS_COMPSTR, nullptr, 0);
  1351. wchar_t *string = reinterpret_cast<wchar_t *>(memalloc(length));
  1352. ImmGetCompositionStringW(wd.im_himc, GCS_COMPSTR, string, length);
  1353. ret.parse_utf16((char16_t *)string, length / sizeof(wchar_t));
  1354. memdelete(string);
  1355. return ret;
  1356. }
  1357. void DisplayServerWindows::window_set_ime_active(const bool p_active, WindowID p_window) {
  1358. _THREAD_SAFE_METHOD_
  1359. ERR_FAIL_COND(!windows.has(p_window));
  1360. WindowData &wd = windows[p_window];
  1361. if (p_active) {
  1362. wd.ime_active = true;
  1363. ImmAssociateContext(wd.hWnd, wd.im_himc);
  1364. CreateCaret(wd.hWnd, NULL, 1, 1);
  1365. window_set_ime_position(wd.im_position, p_window);
  1366. } else {
  1367. ImmAssociateContext(wd.hWnd, (HIMC)0);
  1368. DestroyCaret();
  1369. wd.ime_active = false;
  1370. }
  1371. }
  1372. void DisplayServerWindows::window_set_ime_position(const Point2i &p_pos, WindowID p_window) {
  1373. _THREAD_SAFE_METHOD_
  1374. ERR_FAIL_COND(!windows.has(p_window));
  1375. WindowData &wd = windows[p_window];
  1376. wd.im_position = p_pos;
  1377. HIMC himc = ImmGetContext(wd.hWnd);
  1378. if (himc == (HIMC)0) {
  1379. return;
  1380. }
  1381. COMPOSITIONFORM cps;
  1382. cps.dwStyle = CFS_POINT;
  1383. cps.ptCurrentPos.x = wd.im_position.x;
  1384. cps.ptCurrentPos.y = wd.im_position.y;
  1385. ImmSetCompositionWindow(himc, &cps);
  1386. ImmReleaseContext(wd.hWnd, himc);
  1387. }
  1388. void DisplayServerWindows::cursor_set_shape(CursorShape p_shape) {
  1389. _THREAD_SAFE_METHOD_
  1390. ERR_FAIL_INDEX(p_shape, CURSOR_MAX);
  1391. if (cursor_shape == p_shape) {
  1392. return;
  1393. }
  1394. if (mouse_mode != MOUSE_MODE_VISIBLE && mouse_mode != MOUSE_MODE_CONFINED) {
  1395. cursor_shape = p_shape;
  1396. return;
  1397. }
  1398. static const LPCTSTR win_cursors[CURSOR_MAX] = {
  1399. IDC_ARROW,
  1400. IDC_IBEAM,
  1401. IDC_HAND, // Finger.
  1402. IDC_CROSS,
  1403. IDC_WAIT,
  1404. IDC_APPSTARTING,
  1405. IDC_SIZEALL,
  1406. IDC_ARROW,
  1407. IDC_NO,
  1408. IDC_SIZENS,
  1409. IDC_SIZEWE,
  1410. IDC_SIZENESW,
  1411. IDC_SIZENWSE,
  1412. IDC_SIZEALL,
  1413. IDC_SIZENS,
  1414. IDC_SIZEWE,
  1415. IDC_HELP
  1416. };
  1417. if (cursors_cache.has(p_shape)) {
  1418. SetCursor(cursors[p_shape]);
  1419. } else {
  1420. SetCursor(LoadCursor(hInstance, win_cursors[p_shape]));
  1421. }
  1422. cursor_shape = p_shape;
  1423. }
  1424. DisplayServer::CursorShape DisplayServerWindows::cursor_get_shape() const {
  1425. return cursor_shape;
  1426. }
  1427. void DisplayServerWindows::cursor_set_custom_image(const Ref<Resource> &p_cursor, CursorShape p_shape, const Vector2 &p_hotspot) {
  1428. _THREAD_SAFE_METHOD_
  1429. if (p_cursor.is_valid()) {
  1430. RBMap<CursorShape, Vector<Variant>>::Element *cursor_c = cursors_cache.find(p_shape);
  1431. if (cursor_c) {
  1432. if (cursor_c->get()[0] == p_cursor && cursor_c->get()[1] == p_hotspot) {
  1433. cursor_set_shape(p_shape);
  1434. return;
  1435. }
  1436. cursors_cache.erase(p_shape);
  1437. }
  1438. Ref<Texture2D> texture = p_cursor;
  1439. Ref<AtlasTexture> atlas_texture = p_cursor;
  1440. Ref<Image> image;
  1441. Size2 texture_size;
  1442. Rect2 atlas_rect;
  1443. if (texture.is_valid()) {
  1444. image = texture->get_image();
  1445. }
  1446. if (!image.is_valid() && atlas_texture.is_valid()) {
  1447. texture = atlas_texture->get_atlas();
  1448. atlas_rect.size.width = texture->get_width();
  1449. atlas_rect.size.height = texture->get_height();
  1450. atlas_rect.position.x = atlas_texture->get_region().position.x;
  1451. atlas_rect.position.y = atlas_texture->get_region().position.y;
  1452. texture_size.width = atlas_texture->get_region().size.x;
  1453. texture_size.height = atlas_texture->get_region().size.y;
  1454. } else if (image.is_valid()) {
  1455. texture_size.width = texture->get_width();
  1456. texture_size.height = texture->get_height();
  1457. }
  1458. ERR_FAIL_COND(!texture.is_valid());
  1459. ERR_FAIL_COND(p_hotspot.x < 0 || p_hotspot.y < 0);
  1460. ERR_FAIL_COND(texture_size.width > 256 || texture_size.height > 256);
  1461. ERR_FAIL_COND(p_hotspot.x > texture_size.width || p_hotspot.y > texture_size.height);
  1462. image = texture->get_image();
  1463. ERR_FAIL_COND(!image.is_valid());
  1464. UINT image_size = texture_size.width * texture_size.height;
  1465. // Create the BITMAP with alpha channel.
  1466. COLORREF *buffer = nullptr;
  1467. BITMAPV5HEADER bi;
  1468. ZeroMemory(&bi, sizeof(bi));
  1469. bi.bV5Size = sizeof(bi);
  1470. bi.bV5Width = texture_size.width;
  1471. bi.bV5Height = -texture_size.height;
  1472. bi.bV5Planes = 1;
  1473. bi.bV5BitCount = 32;
  1474. bi.bV5Compression = BI_BITFIELDS;
  1475. bi.bV5RedMask = 0x00ff0000;
  1476. bi.bV5GreenMask = 0x0000ff00;
  1477. bi.bV5BlueMask = 0x000000ff;
  1478. bi.bV5AlphaMask = 0xff000000;
  1479. HDC dc = GetDC(nullptr);
  1480. HBITMAP bitmap = CreateDIBSection(dc, reinterpret_cast<BITMAPINFO *>(&bi), DIB_RGB_COLORS, reinterpret_cast<void **>(&buffer), nullptr, 0);
  1481. HBITMAP mask = CreateBitmap(texture_size.width, texture_size.height, 1, 1, nullptr);
  1482. bool fully_transparent = true;
  1483. for (UINT index = 0; index < image_size; index++) {
  1484. int row_index = floor(index / texture_size.width) + atlas_rect.position.y;
  1485. int column_index = (index % int(texture_size.width)) + atlas_rect.position.x;
  1486. if (atlas_texture.is_valid()) {
  1487. column_index = MIN(column_index, atlas_rect.size.width - 1);
  1488. row_index = MIN(row_index, atlas_rect.size.height - 1);
  1489. }
  1490. const Color &c = image->get_pixel(column_index, row_index);
  1491. fully_transparent = fully_transparent && (c.a == 0.f);
  1492. *(buffer + index) = c.to_argb32();
  1493. }
  1494. // Finally, create the icon.
  1495. if (cursors[p_shape]) {
  1496. DestroyIcon(cursors[p_shape]);
  1497. }
  1498. if (fully_transparent) {
  1499. cursors[p_shape] = nullptr;
  1500. } else {
  1501. ICONINFO iconinfo;
  1502. iconinfo.fIcon = FALSE;
  1503. iconinfo.xHotspot = p_hotspot.x;
  1504. iconinfo.yHotspot = p_hotspot.y;
  1505. iconinfo.hbmMask = mask;
  1506. iconinfo.hbmColor = bitmap;
  1507. cursors[p_shape] = CreateIconIndirect(&iconinfo);
  1508. }
  1509. Vector<Variant> params;
  1510. params.push_back(p_cursor);
  1511. params.push_back(p_hotspot);
  1512. cursors_cache.insert(p_shape, params);
  1513. if (p_shape == cursor_shape) {
  1514. if (mouse_mode == MOUSE_MODE_VISIBLE || mouse_mode == MOUSE_MODE_CONFINED) {
  1515. SetCursor(cursors[p_shape]);
  1516. }
  1517. }
  1518. DeleteObject(mask);
  1519. DeleteObject(bitmap);
  1520. ReleaseDC(nullptr, dc);
  1521. } else {
  1522. // Reset to default system cursor.
  1523. if (cursors[p_shape]) {
  1524. DestroyIcon(cursors[p_shape]);
  1525. }
  1526. cursors[p_shape] = nullptr;
  1527. CursorShape c = cursor_shape;
  1528. cursor_shape = CURSOR_MAX;
  1529. cursor_set_shape(c);
  1530. cursors_cache.erase(p_shape);
  1531. }
  1532. }
  1533. bool DisplayServerWindows::get_swap_cancel_ok() {
  1534. return true;
  1535. }
  1536. void DisplayServerWindows::enable_for_stealing_focus(OS::ProcessID pid) {
  1537. _THREAD_SAFE_METHOD_
  1538. AllowSetForegroundWindow(pid);
  1539. }
  1540. int DisplayServerWindows::keyboard_get_layout_count() const {
  1541. return GetKeyboardLayoutList(0, nullptr);
  1542. }
  1543. int DisplayServerWindows::keyboard_get_current_layout() const {
  1544. HKL cur_layout = GetKeyboardLayout(0);
  1545. int layout_count = GetKeyboardLayoutList(0, nullptr);
  1546. HKL *layouts = (HKL *)memalloc(layout_count * sizeof(HKL));
  1547. GetKeyboardLayoutList(layout_count, layouts);
  1548. for (int i = 0; i < layout_count; i++) {
  1549. if (cur_layout == layouts[i]) {
  1550. memfree(layouts);
  1551. return i;
  1552. }
  1553. }
  1554. memfree(layouts);
  1555. return -1;
  1556. }
  1557. void DisplayServerWindows::keyboard_set_current_layout(int p_index) {
  1558. int layout_count = GetKeyboardLayoutList(0, nullptr);
  1559. ERR_FAIL_INDEX(p_index, layout_count);
  1560. HKL *layouts = (HKL *)memalloc(layout_count * sizeof(HKL));
  1561. GetKeyboardLayoutList(layout_count, layouts);
  1562. ActivateKeyboardLayout(layouts[p_index], KLF_SETFORPROCESS);
  1563. memfree(layouts);
  1564. }
  1565. String DisplayServerWindows::keyboard_get_layout_language(int p_index) const {
  1566. int layout_count = GetKeyboardLayoutList(0, nullptr);
  1567. ERR_FAIL_INDEX_V(p_index, layout_count, "");
  1568. HKL *layouts = (HKL *)memalloc(layout_count * sizeof(HKL));
  1569. GetKeyboardLayoutList(layout_count, layouts);
  1570. WCHAR buf[LOCALE_NAME_MAX_LENGTH];
  1571. memset(buf, 0, LOCALE_NAME_MAX_LENGTH * sizeof(WCHAR));
  1572. LCIDToLocaleName(MAKELCID(LOWORD(layouts[p_index]), SORT_DEFAULT), buf, LOCALE_NAME_MAX_LENGTH, 0);
  1573. memfree(layouts);
  1574. return String::utf16((const char16_t *)buf).substr(0, 2);
  1575. }
  1576. Key DisplayServerWindows::keyboard_get_keycode_from_physical(Key p_keycode) const {
  1577. Key modifiers = p_keycode & KeyModifierMask::MODIFIER_MASK;
  1578. Key keycode_no_mod = (Key)(p_keycode & KeyModifierMask::CODE_MASK);
  1579. if (keycode_no_mod == Key::PRINT ||
  1580. keycode_no_mod == Key::KP_ADD ||
  1581. keycode_no_mod == Key::KP_5 ||
  1582. (keycode_no_mod >= Key::KEY_0 && keycode_no_mod <= Key::KEY_9)) {
  1583. return p_keycode;
  1584. }
  1585. unsigned int scancode = KeyMappingWindows::get_scancode(keycode_no_mod);
  1586. if (scancode == 0) {
  1587. return p_keycode;
  1588. }
  1589. HKL current_layout = GetKeyboardLayout(0);
  1590. UINT vk = MapVirtualKeyEx(scancode, MAPVK_VSC_TO_VK, current_layout);
  1591. if (vk == 0) {
  1592. return p_keycode;
  1593. }
  1594. UINT char_code = MapVirtualKeyEx(vk, MAPVK_VK_TO_CHAR, current_layout) & 0x7FFF;
  1595. // Unlike a similar Linux/BSD check which matches full Latin-1 range,
  1596. // we limit these to ASCII to fix some layouts, including Arabic ones
  1597. if (char_code >= 32 && char_code <= 127) {
  1598. // Godot uses 'braces' instead of 'brackets'
  1599. if (char_code == (unsigned int)Key::BRACKETLEFT || char_code == (unsigned int)Key::BRACKETRIGHT) {
  1600. char_code += 32;
  1601. }
  1602. return (Key)(char_code | (unsigned int)modifiers);
  1603. }
  1604. return (Key)(KeyMappingWindows::get_keysym(vk) | modifiers);
  1605. }
  1606. String _get_full_layout_name_from_registry(HKL p_layout) {
  1607. String id = "SYSTEM\\CurrentControlSet\\Control\\Keyboard Layouts\\" + String::num_int64((int64_t)p_layout, 16, false).lpad(8, "0");
  1608. String ret;
  1609. HKEY hkey;
  1610. WCHAR layout_text[1024];
  1611. memset(layout_text, 0, 1024 * sizeof(WCHAR));
  1612. if (RegOpenKeyExW(HKEY_LOCAL_MACHINE, (LPCWSTR)(id.utf16().get_data()), 0, KEY_QUERY_VALUE, &hkey) != ERROR_SUCCESS) {
  1613. return ret;
  1614. }
  1615. DWORD buffer = 1024;
  1616. DWORD vtype = REG_SZ;
  1617. if (RegQueryValueExW(hkey, L"Layout Text", nullptr, &vtype, (LPBYTE)layout_text, &buffer) == ERROR_SUCCESS) {
  1618. ret = String::utf16((const char16_t *)layout_text);
  1619. }
  1620. RegCloseKey(hkey);
  1621. return ret;
  1622. }
  1623. String DisplayServerWindows::keyboard_get_layout_name(int p_index) const {
  1624. int layout_count = GetKeyboardLayoutList(0, nullptr);
  1625. ERR_FAIL_INDEX_V(p_index, layout_count, "");
  1626. HKL *layouts = (HKL *)memalloc(layout_count * sizeof(HKL));
  1627. GetKeyboardLayoutList(layout_count, layouts);
  1628. String ret = _get_full_layout_name_from_registry(layouts[p_index]); // Try reading full name from Windows registry, fallback to locale name if failed (e.g. on Wine).
  1629. if (ret.is_empty()) {
  1630. WCHAR buf[LOCALE_NAME_MAX_LENGTH];
  1631. memset(buf, 0, LOCALE_NAME_MAX_LENGTH * sizeof(WCHAR));
  1632. LCIDToLocaleName(MAKELCID(LOWORD(layouts[p_index]), SORT_DEFAULT), buf, LOCALE_NAME_MAX_LENGTH, 0);
  1633. WCHAR name[1024];
  1634. memset(name, 0, 1024 * sizeof(WCHAR));
  1635. GetLocaleInfoEx(buf, LOCALE_SLOCALIZEDDISPLAYNAME, (LPWSTR)&name, 1024);
  1636. ret = String::utf16((const char16_t *)name);
  1637. }
  1638. memfree(layouts);
  1639. return ret;
  1640. }
  1641. void DisplayServerWindows::process_events() {
  1642. _THREAD_SAFE_METHOD_
  1643. MSG msg;
  1644. if (!drop_events) {
  1645. joypad->process_joypads();
  1646. }
  1647. while (PeekMessageW(&msg, nullptr, 0, 0, PM_REMOVE)) {
  1648. TranslateMessage(&msg);
  1649. DispatchMessageW(&msg);
  1650. }
  1651. if (!drop_events) {
  1652. _process_key_events();
  1653. Input::get_singleton()->flush_buffered_events();
  1654. }
  1655. }
  1656. void DisplayServerWindows::force_process_and_drop_events() {
  1657. _THREAD_SAFE_METHOD_
  1658. drop_events = true;
  1659. process_events();
  1660. drop_events = false;
  1661. }
  1662. void DisplayServerWindows::release_rendering_thread() {
  1663. }
  1664. void DisplayServerWindows::make_rendering_thread() {
  1665. }
  1666. void DisplayServerWindows::swap_buffers() {
  1667. #if defined(GLES3_ENABLED)
  1668. if (gl_manager) {
  1669. gl_manager->swap_buffers();
  1670. }
  1671. #endif
  1672. }
  1673. void DisplayServerWindows::set_native_icon(const String &p_filename) {
  1674. _THREAD_SAFE_METHOD_
  1675. Ref<FileAccess> f = FileAccess::open(p_filename, FileAccess::READ);
  1676. ERR_FAIL_COND_MSG(f.is_null(), "Cannot open file with icon '" + p_filename + "'.");
  1677. ICONDIR *icon_dir = (ICONDIR *)memalloc(sizeof(ICONDIR));
  1678. int pos = 0;
  1679. icon_dir->idReserved = f->get_32();
  1680. pos += sizeof(WORD);
  1681. f->seek(pos);
  1682. icon_dir->idType = f->get_32();
  1683. pos += sizeof(WORD);
  1684. f->seek(pos);
  1685. ERR_FAIL_COND_MSG(icon_dir->idType != 1, "Invalid icon file format!");
  1686. icon_dir->idCount = f->get_32();
  1687. pos += sizeof(WORD);
  1688. f->seek(pos);
  1689. icon_dir = (ICONDIR *)memrealloc(icon_dir, sizeof(ICONDIR) - sizeof(ICONDIRENTRY) + icon_dir->idCount * sizeof(ICONDIRENTRY));
  1690. f->get_buffer((uint8_t *)&icon_dir->idEntries[0], icon_dir->idCount * sizeof(ICONDIRENTRY));
  1691. int small_icon_index = -1; // Select 16x16 with largest color count.
  1692. int small_icon_cc = 0;
  1693. int big_icon_index = -1; // Select largest.
  1694. int big_icon_width = 16;
  1695. int big_icon_cc = 0;
  1696. for (int i = 0; i < icon_dir->idCount; i++) {
  1697. int colors = (icon_dir->idEntries[i].bColorCount == 0) ? 32768 : icon_dir->idEntries[i].bColorCount;
  1698. int width = (icon_dir->idEntries[i].bWidth == 0) ? 256 : icon_dir->idEntries[i].bWidth;
  1699. if (width == 16) {
  1700. if (colors >= small_icon_cc) {
  1701. small_icon_index = i;
  1702. small_icon_cc = colors;
  1703. }
  1704. }
  1705. if (width >= big_icon_width) {
  1706. if (colors >= big_icon_cc) {
  1707. big_icon_index = i;
  1708. big_icon_width = width;
  1709. big_icon_cc = colors;
  1710. }
  1711. }
  1712. }
  1713. ERR_FAIL_COND_MSG(big_icon_index == -1, "No valid icons found!");
  1714. if (small_icon_index == -1) {
  1715. WARN_PRINT("No small icon found, reusing " + itos(big_icon_width) + "x" + itos(big_icon_width) + " @" + itos(big_icon_cc) + " icon!");
  1716. small_icon_index = big_icon_index;
  1717. small_icon_cc = big_icon_cc;
  1718. }
  1719. // Read the big icon.
  1720. DWORD bytecount_big = icon_dir->idEntries[big_icon_index].dwBytesInRes;
  1721. Vector<uint8_t> data_big;
  1722. data_big.resize(bytecount_big);
  1723. pos = icon_dir->idEntries[big_icon_index].dwImageOffset;
  1724. f->seek(pos);
  1725. f->get_buffer((uint8_t *)&data_big.write[0], bytecount_big);
  1726. HICON icon_big = CreateIconFromResource((PBYTE)&data_big.write[0], bytecount_big, TRUE, 0x00030000);
  1727. ERR_FAIL_COND_MSG(!icon_big, "Could not create " + itos(big_icon_width) + "x" + itos(big_icon_width) + " @" + itos(big_icon_cc) + " icon, error: " + format_error_message(GetLastError()) + ".");
  1728. // Read the small icon.
  1729. DWORD bytecount_small = icon_dir->idEntries[small_icon_index].dwBytesInRes;
  1730. Vector<uint8_t> data_small;
  1731. data_small.resize(bytecount_small);
  1732. pos = icon_dir->idEntries[small_icon_index].dwImageOffset;
  1733. f->seek(pos);
  1734. f->get_buffer((uint8_t *)&data_small.write[0], bytecount_small);
  1735. HICON icon_small = CreateIconFromResource((PBYTE)&data_small.write[0], bytecount_small, TRUE, 0x00030000);
  1736. ERR_FAIL_COND_MSG(!icon_small, "Could not create 16x16 @" + itos(small_icon_cc) + " icon, error: " + format_error_message(GetLastError()) + ".");
  1737. // Online tradition says to be sure last error is cleared and set the small icon first.
  1738. int err = 0;
  1739. SetLastError(err);
  1740. SendMessage(windows[MAIN_WINDOW_ID].hWnd, WM_SETICON, ICON_SMALL, (LPARAM)icon_small);
  1741. err = GetLastError();
  1742. ERR_FAIL_COND_MSG(err, "Error setting ICON_SMALL: " + format_error_message(err) + ".");
  1743. SendMessage(windows[MAIN_WINDOW_ID].hWnd, WM_SETICON, ICON_BIG, (LPARAM)icon_big);
  1744. err = GetLastError();
  1745. ERR_FAIL_COND_MSG(err, "Error setting ICON_BIG: " + format_error_message(err) + ".");
  1746. memdelete(icon_dir);
  1747. }
  1748. void DisplayServerWindows::set_icon(const Ref<Image> &p_icon) {
  1749. _THREAD_SAFE_METHOD_
  1750. ERR_FAIL_COND(!p_icon.is_valid());
  1751. if (icon != p_icon) {
  1752. icon = p_icon->duplicate();
  1753. if (icon->get_format() != Image::FORMAT_RGBA8) {
  1754. icon->convert(Image::FORMAT_RGBA8);
  1755. }
  1756. }
  1757. int w = icon->get_width();
  1758. int h = icon->get_height();
  1759. // Create temporary bitmap buffer.
  1760. int icon_len = 40 + h * w * 4;
  1761. Vector<BYTE> v;
  1762. v.resize(icon_len);
  1763. BYTE *icon_bmp = v.ptrw();
  1764. encode_uint32(40, &icon_bmp[0]);
  1765. encode_uint32(w, &icon_bmp[4]);
  1766. encode_uint32(h * 2, &icon_bmp[8]);
  1767. encode_uint16(1, &icon_bmp[12]);
  1768. encode_uint16(32, &icon_bmp[14]);
  1769. encode_uint32(BI_RGB, &icon_bmp[16]);
  1770. encode_uint32(w * h * 4, &icon_bmp[20]);
  1771. encode_uint32(0, &icon_bmp[24]);
  1772. encode_uint32(0, &icon_bmp[28]);
  1773. encode_uint32(0, &icon_bmp[32]);
  1774. encode_uint32(0, &icon_bmp[36]);
  1775. uint8_t *wr = &icon_bmp[40];
  1776. const uint8_t *r = icon->get_data().ptr();
  1777. for (int i = 0; i < h; i++) {
  1778. for (int j = 0; j < w; j++) {
  1779. const uint8_t *rpx = &r[((h - i - 1) * w + j) * 4];
  1780. uint8_t *wpx = &wr[(i * w + j) * 4];
  1781. wpx[0] = rpx[2];
  1782. wpx[1] = rpx[1];
  1783. wpx[2] = rpx[0];
  1784. wpx[3] = rpx[3];
  1785. }
  1786. }
  1787. HICON hicon = CreateIconFromResource(icon_bmp, icon_len, TRUE, 0x00030000);
  1788. // Set the icon for the window.
  1789. SendMessage(windows[MAIN_WINDOW_ID].hWnd, WM_SETICON, ICON_SMALL, (LPARAM)hicon);
  1790. // Set the icon in the task manager (should we do this?).
  1791. SendMessage(windows[MAIN_WINDOW_ID].hWnd, WM_SETICON, ICON_BIG, (LPARAM)hicon);
  1792. }
  1793. void DisplayServerWindows::window_set_vsync_mode(DisplayServer::VSyncMode p_vsync_mode, WindowID p_window) {
  1794. _THREAD_SAFE_METHOD_
  1795. #if defined(VULKAN_ENABLED)
  1796. if (context_vulkan) {
  1797. context_vulkan->set_vsync_mode(p_window, p_vsync_mode);
  1798. }
  1799. #endif
  1800. #if defined(GLES3_ENABLED)
  1801. if (gl_manager) {
  1802. gl_manager->set_use_vsync(p_window, p_vsync_mode != DisplayServer::VSYNC_DISABLED);
  1803. }
  1804. #endif
  1805. }
  1806. DisplayServer::VSyncMode DisplayServerWindows::window_get_vsync_mode(WindowID p_window) const {
  1807. _THREAD_SAFE_METHOD_
  1808. #if defined(VULKAN_ENABLED)
  1809. if (context_vulkan) {
  1810. return context_vulkan->get_vsync_mode(p_window);
  1811. }
  1812. #endif
  1813. #if defined(GLES3_ENABLED)
  1814. if (gl_manager) {
  1815. return gl_manager->is_using_vsync(p_window) ? DisplayServer::VSYNC_ENABLED : DisplayServer::VSYNC_DISABLED;
  1816. }
  1817. #endif
  1818. return DisplayServer::VSYNC_ENABLED;
  1819. }
  1820. void DisplayServerWindows::set_context(Context p_context) {
  1821. }
  1822. #define MI_WP_SIGNATURE 0xFF515700
  1823. #define SIGNATURE_MASK 0xFFFFFF00
  1824. // Keeping the name suggested by Microsoft, but this macro really answers:
  1825. // Is this mouse event emulated from touch or pen input?
  1826. #define IsPenEvent(dw) (((dw)&SIGNATURE_MASK) == MI_WP_SIGNATURE)
  1827. // This one tells whether the event comes from touchscreen (and not from pen).
  1828. #define IsTouchEvent(dw) (IsPenEvent(dw) && ((dw)&0x80))
  1829. void DisplayServerWindows::_touch_event(WindowID p_window, bool p_pressed, float p_x, float p_y, int idx) {
  1830. if (touch_state.has(idx) == p_pressed) {
  1831. return;
  1832. }
  1833. if (p_pressed) {
  1834. touch_state.insert(idx, Vector2(p_x, p_y));
  1835. } else {
  1836. touch_state.erase(idx);
  1837. }
  1838. Ref<InputEventScreenTouch> event;
  1839. event.instantiate();
  1840. event->set_index(idx);
  1841. event->set_window_id(p_window);
  1842. event->set_pressed(p_pressed);
  1843. event->set_position(Vector2(p_x, p_y));
  1844. Input::get_singleton()->parse_input_event(event);
  1845. }
  1846. void DisplayServerWindows::_drag_event(WindowID p_window, float p_x, float p_y, int idx) {
  1847. RBMap<int, Vector2>::Element *curr = touch_state.find(idx);
  1848. if (!curr) {
  1849. return;
  1850. }
  1851. if (curr->get() == Vector2(p_x, p_y)) {
  1852. return;
  1853. }
  1854. Ref<InputEventScreenDrag> event;
  1855. event.instantiate();
  1856. event->set_window_id(p_window);
  1857. event->set_index(idx);
  1858. event->set_position(Vector2(p_x, p_y));
  1859. event->set_relative(Vector2(p_x, p_y) - curr->get());
  1860. Input::get_singleton()->parse_input_event(event);
  1861. curr->get() = Vector2(p_x, p_y);
  1862. }
  1863. void DisplayServerWindows::_send_window_event(const WindowData &wd, WindowEvent p_event) {
  1864. if (!wd.event_callback.is_null()) {
  1865. Variant event = int(p_event);
  1866. Variant *eventp = &event;
  1867. Variant ret;
  1868. Callable::CallError ce;
  1869. wd.event_callback.callp((const Variant **)&eventp, 1, ret, ce);
  1870. }
  1871. }
  1872. void DisplayServerWindows::_dispatch_input_events(const Ref<InputEvent> &p_event) {
  1873. static_cast<DisplayServerWindows *>(get_singleton())->_dispatch_input_event(p_event);
  1874. }
  1875. void DisplayServerWindows::_dispatch_input_event(const Ref<InputEvent> &p_event) {
  1876. _THREAD_SAFE_METHOD_
  1877. if (in_dispatch_input_event) {
  1878. return;
  1879. }
  1880. in_dispatch_input_event = true;
  1881. Variant ev = p_event;
  1882. Variant *evp = &ev;
  1883. Variant ret;
  1884. Callable::CallError ce;
  1885. {
  1886. List<WindowID>::Element *E = popup_list.back();
  1887. if (E && Object::cast_to<InputEventKey>(*p_event)) {
  1888. // Redirect keyboard input to active popup.
  1889. if (windows.has(E->get())) {
  1890. Callable callable = windows[E->get()].input_event_callback;
  1891. if (callable.is_valid()) {
  1892. callable.callp((const Variant **)&evp, 1, ret, ce);
  1893. }
  1894. }
  1895. in_dispatch_input_event = false;
  1896. return;
  1897. }
  1898. }
  1899. Ref<InputEventFromWindow> event_from_window = p_event;
  1900. if (event_from_window.is_valid() && event_from_window->get_window_id() != INVALID_WINDOW_ID) {
  1901. // Send to a single window.
  1902. if (windows.has(event_from_window->get_window_id())) {
  1903. Callable callable = windows[event_from_window->get_window_id()].input_event_callback;
  1904. if (callable.is_valid()) {
  1905. callable.callp((const Variant **)&evp, 1, ret, ce);
  1906. }
  1907. }
  1908. } else {
  1909. // Send to all windows.
  1910. for (const KeyValue<WindowID, WindowData> &E : windows) {
  1911. const Callable callable = E.value.input_event_callback;
  1912. if (callable.is_valid()) {
  1913. callable.callp((const Variant **)&evp, 1, ret, ce);
  1914. }
  1915. }
  1916. }
  1917. in_dispatch_input_event = false;
  1918. }
  1919. LRESULT CALLBACK MouseProc(int code, WPARAM wParam, LPARAM lParam) {
  1920. DisplayServerWindows *ds_win = static_cast<DisplayServerWindows *>(DisplayServer::get_singleton());
  1921. if (ds_win) {
  1922. return ds_win->MouseProc(code, wParam, lParam);
  1923. } else {
  1924. return ::CallNextHookEx(nullptr, code, wParam, lParam);
  1925. }
  1926. }
  1927. DisplayServer::WindowID DisplayServerWindows::window_get_active_popup() const {
  1928. const List<WindowID>::Element *E = popup_list.back();
  1929. if (E) {
  1930. return E->get();
  1931. } else {
  1932. return INVALID_WINDOW_ID;
  1933. }
  1934. }
  1935. void DisplayServerWindows::window_set_popup_safe_rect(WindowID p_window, const Rect2i &p_rect) {
  1936. _THREAD_SAFE_METHOD_
  1937. ERR_FAIL_COND(!windows.has(p_window));
  1938. WindowData &wd = windows[p_window];
  1939. wd.parent_safe_rect = p_rect;
  1940. }
  1941. Rect2i DisplayServerWindows::window_get_popup_safe_rect(WindowID p_window) const {
  1942. _THREAD_SAFE_METHOD_
  1943. ERR_FAIL_COND_V(!windows.has(p_window), Rect2i());
  1944. const WindowData &wd = windows[p_window];
  1945. return wd.parent_safe_rect;
  1946. }
  1947. void DisplayServerWindows::popup_open(WindowID p_window) {
  1948. _THREAD_SAFE_METHOD_
  1949. const WindowData &wd = windows[p_window];
  1950. if (wd.is_popup) {
  1951. // Find current popup parent, or root popup if new window is not transient.
  1952. List<WindowID>::Element *C = nullptr;
  1953. List<WindowID>::Element *E = popup_list.back();
  1954. while (E) {
  1955. if (wd.transient_parent != E->get() || wd.transient_parent == INVALID_WINDOW_ID) {
  1956. C = E;
  1957. E = E->prev();
  1958. } else {
  1959. break;
  1960. }
  1961. }
  1962. if (C) {
  1963. _send_window_event(windows[C->get()], DisplayServerWindows::WINDOW_EVENT_CLOSE_REQUEST);
  1964. }
  1965. time_since_popup = OS::get_singleton()->get_ticks_msec();
  1966. popup_list.push_back(p_window);
  1967. }
  1968. }
  1969. void DisplayServerWindows::popup_close(WindowID p_window) {
  1970. _THREAD_SAFE_METHOD_
  1971. List<WindowID>::Element *E = popup_list.find(p_window);
  1972. while (E) {
  1973. List<WindowID>::Element *F = E->next();
  1974. WindowID win_id = E->get();
  1975. popup_list.erase(E);
  1976. if (win_id != p_window) {
  1977. // Only request close on related windows, not this window. We are already processing it.
  1978. _send_window_event(windows[win_id], DisplayServerWindows::WINDOW_EVENT_CLOSE_REQUEST);
  1979. }
  1980. E = F;
  1981. }
  1982. }
  1983. LRESULT DisplayServerWindows::MouseProc(int code, WPARAM wParam, LPARAM lParam) {
  1984. _THREAD_SAFE_METHOD_
  1985. uint64_t delta = OS::get_singleton()->get_ticks_msec() - time_since_popup;
  1986. if (delta > 250) {
  1987. switch (wParam) {
  1988. case WM_NCLBUTTONDOWN:
  1989. case WM_NCRBUTTONDOWN:
  1990. case WM_NCMBUTTONDOWN:
  1991. case WM_LBUTTONDOWN:
  1992. case WM_RBUTTONDOWN:
  1993. case WM_MBUTTONDOWN: {
  1994. MOUSEHOOKSTRUCT *ms = (MOUSEHOOKSTRUCT *)lParam;
  1995. Point2i pos = Point2i(ms->pt.x, ms->pt.y) - _get_screens_origin();
  1996. List<WindowID>::Element *C = nullptr;
  1997. List<WindowID>::Element *E = popup_list.back();
  1998. // Find top popup to close.
  1999. while (E) {
  2000. // Popup window area.
  2001. Rect2i win_rect = Rect2i(window_get_position(E->get()), window_get_size(E->get()));
  2002. // Area of the parent window, which responsible for opening sub-menu.
  2003. Rect2i safe_rect = window_get_popup_safe_rect(E->get());
  2004. if (win_rect.has_point(pos)) {
  2005. break;
  2006. } else if (safe_rect != Rect2i() && safe_rect.has_point(pos)) {
  2007. break;
  2008. } else {
  2009. C = E;
  2010. E = E->prev();
  2011. }
  2012. }
  2013. if (C) {
  2014. _send_window_event(windows[C->get()], DisplayServerWindows::WINDOW_EVENT_CLOSE_REQUEST);
  2015. return 1;
  2016. }
  2017. } break;
  2018. }
  2019. }
  2020. return ::CallNextHookEx(mouse_monitor, code, wParam, lParam);
  2021. }
  2022. // Handle a single window message received while CreateWindowEx is still on the stack and our data
  2023. // structures are not fully initialized.
  2024. LRESULT DisplayServerWindows::_handle_early_window_message(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) {
  2025. switch (uMsg) {
  2026. case WM_GETMINMAXINFO: {
  2027. // We receive this during CreateWindowEx and we haven't initialized the window
  2028. // struct, so let Windows figure out the maximized size.
  2029. // Silently forward to user/default.
  2030. } break;
  2031. case WM_NCCREATE: {
  2032. // We tunnel an unowned pointer to our window context (WindowData) through the
  2033. // first possible message (WM_NCCREATE) to fix up our window context collection.
  2034. CREATESTRUCTW *pCreate = (CREATESTRUCTW *)lParam;
  2035. WindowData *pWindowData = reinterpret_cast<WindowData *>(pCreate->lpCreateParams);
  2036. // Fix this up so we can recognize the remaining messages.
  2037. pWindowData->hWnd = hWnd;
  2038. } break;
  2039. default: {
  2040. // Additional messages during window creation should happen after we fixed
  2041. // up the data structures on WM_NCCREATE, but this might change in the future,
  2042. // so report an error here and then we can implement them.
  2043. ERR_PRINT_ONCE(vformat("Unexpected window message 0x%x received for window we cannot recognize in our collection; sequence error.", uMsg));
  2044. } break;
  2045. }
  2046. if (user_proc) {
  2047. return CallWindowProcW(user_proc, hWnd, uMsg, wParam, lParam);
  2048. }
  2049. return DefWindowProcW(hWnd, uMsg, wParam, lParam);
  2050. }
  2051. // The window procedure for our window class "Engine", used to handle processing of window-related system messages/events.
  2052. // See: https://docs.microsoft.com/en-us/windows/win32/winmsg/window-procedures
  2053. LRESULT DisplayServerWindows::WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) {
  2054. if (drop_events) {
  2055. if (user_proc) {
  2056. return CallWindowProcW(user_proc, hWnd, uMsg, wParam, lParam);
  2057. } else {
  2058. return DefWindowProcW(hWnd, uMsg, wParam, lParam);
  2059. }
  2060. }
  2061. WindowID window_id = INVALID_WINDOW_ID;
  2062. bool window_created = false;
  2063. // Check whether window exists
  2064. // FIXME this is O(n), where n is the set of currently open windows and subwindows
  2065. // we should have a secondary map from HWND to WindowID or even WindowData* alias, if we want to eliminate all the map lookups below
  2066. for (const KeyValue<WindowID, WindowData> &E : windows) {
  2067. if (E.value.hWnd == hWnd) {
  2068. window_id = E.key;
  2069. window_created = true;
  2070. break;
  2071. }
  2072. }
  2073. // WARNING: we get called with events before the window is registered in our collection
  2074. // specifically, even the call to CreateWindowEx already calls here while still on the stack,
  2075. // so there is no way to store the window handle in our collection before we get here
  2076. if (!window_created) {
  2077. // don't let code below operate on incompletely initialized window objects or missing window_id
  2078. return _handle_early_window_message(hWnd, uMsg, wParam, lParam);
  2079. }
  2080. // Process window messages.
  2081. switch (uMsg) {
  2082. case WM_NCHITTEST: {
  2083. if (windows[window_id].mpass) {
  2084. return HTTRANSPARENT;
  2085. }
  2086. } break;
  2087. case WM_MOUSEACTIVATE: {
  2088. if (windows[window_id].no_focus) {
  2089. return MA_NOACTIVATEANDEAT; // Do not activate, and discard mouse messages.
  2090. } else if (windows[window_id].is_popup) {
  2091. return MA_NOACTIVATE; // Do not activate, but process mouse messages.
  2092. }
  2093. } break;
  2094. case WM_SETFOCUS: {
  2095. windows[window_id].window_has_focus = true;
  2096. last_focused_window = window_id;
  2097. // Restore mouse mode.
  2098. _set_mouse_mode_impl(mouse_mode);
  2099. if (!app_focused) {
  2100. if (OS::get_singleton()->get_main_loop()) {
  2101. OS::get_singleton()->get_main_loop()->notification(MainLoop::NOTIFICATION_APPLICATION_FOCUS_IN);
  2102. }
  2103. app_focused = true;
  2104. }
  2105. } break;
  2106. case WM_KILLFOCUS: {
  2107. windows[window_id].window_has_focus = false;
  2108. last_focused_window = window_id;
  2109. // Release capture unconditionally because it can be set due to dragging, in addition to captured mode.
  2110. ReleaseCapture();
  2111. // Release every touch to avoid sticky points.
  2112. for (const KeyValue<int, Vector2> &E : touch_state) {
  2113. _touch_event(window_id, false, E.value.x, E.value.y, E.key);
  2114. }
  2115. touch_state.clear();
  2116. bool self_steal = false;
  2117. HWND new_hwnd = (HWND)wParam;
  2118. if (IsWindow(new_hwnd)) {
  2119. self_steal = true;
  2120. }
  2121. if (!self_steal) {
  2122. if (OS::get_singleton()->get_main_loop()) {
  2123. OS::get_singleton()->get_main_loop()->notification(MainLoop::NOTIFICATION_APPLICATION_FOCUS_OUT);
  2124. }
  2125. app_focused = false;
  2126. }
  2127. } break;
  2128. case WM_ACTIVATE: { // Watch for window activate message.
  2129. if (!windows[window_id].window_focused) {
  2130. _process_activate_event(window_id, wParam, lParam);
  2131. } else {
  2132. windows[window_id].saved_wparam = wParam;
  2133. windows[window_id].saved_lparam = lParam;
  2134. // Run a timer to prevent event catching warning if the focused window is closing.
  2135. windows[window_id].focus_timer_id = SetTimer(windows[window_id].hWnd, 2, USER_TIMER_MINIMUM, (TIMERPROC) nullptr);
  2136. }
  2137. if (wParam != WA_INACTIVE) {
  2138. track_mouse_leave_event(hWnd);
  2139. }
  2140. return 0; // Return to the message loop.
  2141. } break;
  2142. case WM_GETMINMAXINFO: {
  2143. if (windows[window_id].resizable && !windows[window_id].fullscreen) {
  2144. // Size of window decorations.
  2145. Size2 decor = window_get_size_with_decorations(window_id) - window_get_size(window_id);
  2146. MINMAXINFO *min_max_info = (MINMAXINFO *)lParam;
  2147. if (windows[window_id].min_size != Size2()) {
  2148. min_max_info->ptMinTrackSize.x = windows[window_id].min_size.x + decor.x;
  2149. min_max_info->ptMinTrackSize.y = windows[window_id].min_size.y + decor.y;
  2150. }
  2151. if (windows[window_id].max_size != Size2()) {
  2152. min_max_info->ptMaxTrackSize.x = windows[window_id].max_size.x + decor.x;
  2153. min_max_info->ptMaxTrackSize.y = windows[window_id].max_size.y + decor.y;
  2154. }
  2155. return 0;
  2156. }
  2157. } break;
  2158. case WM_ERASEBKGND: {
  2159. Color early_color;
  2160. if (!_get_window_early_clear_override(early_color)) {
  2161. break;
  2162. }
  2163. bool must_recreate_brush = !window_bkg_brush || window_bkg_brush_color != early_color.to_argb32();
  2164. if (must_recreate_brush) {
  2165. if (window_bkg_brush) {
  2166. DeleteObject(window_bkg_brush);
  2167. }
  2168. window_bkg_brush = CreateSolidBrush(RGB(early_color.get_r8(), early_color.get_g8(), early_color.get_b8()));
  2169. }
  2170. HDC hdc = (HDC)wParam;
  2171. RECT rect = {};
  2172. if (GetUpdateRect(hWnd, &rect, true)) {
  2173. FillRect(hdc, &rect, window_bkg_brush);
  2174. }
  2175. return 1;
  2176. } break;
  2177. case WM_PAINT: {
  2178. Main::force_redraw();
  2179. } break;
  2180. case WM_SETTINGCHANGE: {
  2181. if (lParam && CompareStringOrdinal(reinterpret_cast<LPCWCH>(lParam), -1, L"ImmersiveColorSet", -1, true) == CSTR_EQUAL) {
  2182. if (is_dark_mode_supported() && dark_title_available) {
  2183. BOOL value = is_dark_mode();
  2184. ::DwmSetWindowAttribute(windows[window_id].hWnd, DWMWA_USE_IMMERSIVE_DARK_MODE, &value, sizeof(value));
  2185. }
  2186. }
  2187. } break;
  2188. case WM_THEMECHANGED: {
  2189. if (is_dark_mode_supported() && dark_title_available) {
  2190. BOOL value = is_dark_mode();
  2191. ::DwmSetWindowAttribute(windows[window_id].hWnd, DWMWA_USE_IMMERSIVE_DARK_MODE, &value, sizeof(value));
  2192. }
  2193. } break;
  2194. case WM_SYSCOMMAND: // Intercept system commands.
  2195. {
  2196. switch (wParam) // Check system calls.
  2197. {
  2198. case SC_SCREENSAVE: // Screensaver trying to start?
  2199. case SC_MONITORPOWER: // Monitor trying to enter powersave?
  2200. return 0; // Prevent from happening.
  2201. case SC_KEYMENU:
  2202. if ((lParam >> 16) <= 0) {
  2203. return 0;
  2204. }
  2205. }
  2206. } break;
  2207. case WM_CLOSE: // Did we receive a close message?
  2208. {
  2209. if (windows[window_id].focus_timer_id != 0U) {
  2210. KillTimer(windows[window_id].hWnd, windows[window_id].focus_timer_id);
  2211. }
  2212. _send_window_event(windows[window_id], WINDOW_EVENT_CLOSE_REQUEST);
  2213. return 0; // Jump back.
  2214. }
  2215. case WM_MOUSELEAVE: {
  2216. if (window_mouseover_id == window_id) {
  2217. old_invalid = true;
  2218. window_mouseover_id = INVALID_WINDOW_ID;
  2219. _send_window_event(windows[window_id], WINDOW_EVENT_MOUSE_EXIT);
  2220. } else if (window_mouseover_id != INVALID_WINDOW_ID && windows.has(window_mouseover_id)) {
  2221. // This is reached during drag and drop, after dropping in a different window.
  2222. // Once-off notification, must call again.
  2223. track_mouse_leave_event(windows[window_mouseover_id].hWnd);
  2224. }
  2225. } break;
  2226. case WM_INPUT: {
  2227. if (mouse_mode != MOUSE_MODE_CAPTURED || !use_raw_input) {
  2228. break;
  2229. }
  2230. UINT dwSize;
  2231. GetRawInputData((HRAWINPUT)lParam, RID_INPUT, nullptr, &dwSize, sizeof(RAWINPUTHEADER));
  2232. LPBYTE lpb = new BYTE[dwSize];
  2233. if (lpb == nullptr) {
  2234. return 0;
  2235. }
  2236. if (GetRawInputData((HRAWINPUT)lParam, RID_INPUT, lpb, &dwSize, sizeof(RAWINPUTHEADER)) != dwSize) {
  2237. OutputDebugString(TEXT("GetRawInputData does not return correct size !\n"));
  2238. }
  2239. RAWINPUT *raw = (RAWINPUT *)lpb;
  2240. if (raw->header.dwType == RIM_TYPEMOUSE) {
  2241. Ref<InputEventMouseMotion> mm;
  2242. mm.instantiate();
  2243. mm->set_window_id(window_id);
  2244. mm->set_ctrl_pressed(control_mem);
  2245. mm->set_shift_pressed(shift_mem);
  2246. mm->set_alt_pressed(alt_mem);
  2247. mm->set_pressure((raw->data.mouse.ulButtons & RI_MOUSE_LEFT_BUTTON_DOWN) ? 1.0f : 0.0f);
  2248. mm->set_button_mask(last_button_state);
  2249. Point2i c(windows[window_id].width / 2, windows[window_id].height / 2);
  2250. // Centering just so it works as before.
  2251. POINT pos = { (int)c.x, (int)c.y };
  2252. ClientToScreen(windows[window_id].hWnd, &pos);
  2253. SetCursorPos(pos.x, pos.y);
  2254. mm->set_position(c);
  2255. mm->set_global_position(c);
  2256. mm->set_velocity(Vector2(0, 0));
  2257. if (raw->data.mouse.usFlags == MOUSE_MOVE_RELATIVE) {
  2258. mm->set_relative(Vector2(raw->data.mouse.lLastX, raw->data.mouse.lLastY));
  2259. } else if (raw->data.mouse.usFlags == MOUSE_MOVE_ABSOLUTE) {
  2260. int nScreenWidth = GetSystemMetrics(SM_CXVIRTUALSCREEN);
  2261. int nScreenHeight = GetSystemMetrics(SM_CYVIRTUALSCREEN);
  2262. int nScreenLeft = GetSystemMetrics(SM_XVIRTUALSCREEN);
  2263. int nScreenTop = GetSystemMetrics(SM_YVIRTUALSCREEN);
  2264. Vector2 abs_pos(
  2265. (double(raw->data.mouse.lLastX) - 65536.0 / (nScreenWidth)) * nScreenWidth / 65536.0 + nScreenLeft,
  2266. (double(raw->data.mouse.lLastY) - 65536.0 / (nScreenHeight)) * nScreenHeight / 65536.0 + nScreenTop);
  2267. POINT coords; // Client coords.
  2268. coords.x = abs_pos.x;
  2269. coords.y = abs_pos.y;
  2270. ScreenToClient(hWnd, &coords);
  2271. mm->set_relative(Vector2(coords.x - old_x, coords.y - old_y));
  2272. old_x = coords.x;
  2273. old_y = coords.y;
  2274. }
  2275. if ((windows[window_id].window_has_focus || windows[window_id].is_popup) && mm->get_relative() != Vector2()) {
  2276. Input::get_singleton()->parse_input_event(mm);
  2277. }
  2278. }
  2279. delete[] lpb;
  2280. } break;
  2281. case WT_CSRCHANGE:
  2282. case WT_PROXIMITY: {
  2283. if ((tablet_get_current_driver() == "wintab") && wintab_available && windows[window_id].wtctx) {
  2284. AXIS pressure;
  2285. if (wintab_WTInfo(WTI_DEVICES + windows[window_id].wtlc.lcDevice, DVC_NPRESSURE, &pressure)) {
  2286. windows[window_id].min_pressure = int(pressure.axMin);
  2287. windows[window_id].max_pressure = int(pressure.axMax);
  2288. }
  2289. AXIS orientation[3];
  2290. if (wintab_WTInfo(WTI_DEVICES + windows[window_id].wtlc.lcDevice, DVC_ORIENTATION, &orientation)) {
  2291. windows[window_id].tilt_supported = orientation[0].axResolution && orientation[1].axResolution;
  2292. }
  2293. return 0;
  2294. }
  2295. } break;
  2296. case WT_PACKET: {
  2297. if ((tablet_get_current_driver() == "wintab") && wintab_available && windows[window_id].wtctx) {
  2298. PACKET packet;
  2299. if (wintab_WTPacket(windows[window_id].wtctx, wParam, &packet)) {
  2300. POINT coords;
  2301. GetCursorPos(&coords);
  2302. ScreenToClient(windows[window_id].hWnd, &coords);
  2303. windows[window_id].last_pressure_update = 0;
  2304. float pressure = float(packet.pkNormalPressure - windows[window_id].min_pressure) / float(windows[window_id].max_pressure - windows[window_id].min_pressure);
  2305. double azim = (packet.pkOrientation.orAzimuth / 10.0f) * (Math_PI / 180);
  2306. double alt = Math::tan((Math::abs(packet.pkOrientation.orAltitude / 10.0f)) * (Math_PI / 180));
  2307. bool inverted = packet.pkStatus & TPS_INVERT;
  2308. Vector2 tilt = (windows[window_id].tilt_supported) ? Vector2(Math::atan(Math::sin(azim) / alt), Math::atan(Math::cos(azim) / alt)) : Vector2();
  2309. // Nothing changed, ignore event.
  2310. if (!old_invalid && coords.x == old_x && coords.y == old_y && windows[window_id].last_pressure == pressure && windows[window_id].last_tilt == tilt && windows[window_id].last_pen_inverted == inverted) {
  2311. break;
  2312. }
  2313. windows[window_id].last_pressure = pressure;
  2314. windows[window_id].last_tilt = tilt;
  2315. windows[window_id].last_pen_inverted = inverted;
  2316. // Don't calculate relative mouse movement if we don't have focus in CAPTURED mode.
  2317. if (!windows[window_id].window_has_focus && mouse_mode == MOUSE_MODE_CAPTURED) {
  2318. break;
  2319. }
  2320. Ref<InputEventMouseMotion> mm;
  2321. mm.instantiate();
  2322. mm->set_window_id(window_id);
  2323. mm->set_ctrl_pressed(GetKeyState(VK_CONTROL) < 0);
  2324. mm->set_shift_pressed(GetKeyState(VK_SHIFT) < 0);
  2325. mm->set_alt_pressed(alt_mem);
  2326. mm->set_pressure(windows[window_id].last_pressure);
  2327. mm->set_tilt(windows[window_id].last_tilt);
  2328. mm->set_pen_inverted(windows[window_id].last_pen_inverted);
  2329. mm->set_button_mask(last_button_state);
  2330. mm->set_position(Vector2(coords.x, coords.y));
  2331. mm->set_global_position(Vector2(coords.x, coords.y));
  2332. if (mouse_mode == MOUSE_MODE_CAPTURED) {
  2333. Point2i c(windows[window_id].width / 2, windows[window_id].height / 2);
  2334. old_x = c.x;
  2335. old_y = c.y;
  2336. if (mm->get_position() == c) {
  2337. center = c;
  2338. return 0;
  2339. }
  2340. Point2i ncenter = mm->get_position();
  2341. center = ncenter;
  2342. POINT pos = { (int)c.x, (int)c.y };
  2343. ClientToScreen(windows[window_id].hWnd, &pos);
  2344. SetCursorPos(pos.x, pos.y);
  2345. }
  2346. mm->set_velocity(Input::get_singleton()->get_last_mouse_velocity());
  2347. if (old_invalid) {
  2348. old_x = mm->get_position().x;
  2349. old_y = mm->get_position().y;
  2350. old_invalid = false;
  2351. }
  2352. mm->set_relative(Vector2(mm->get_position() - Vector2(old_x, old_y)));
  2353. old_x = mm->get_position().x;
  2354. old_y = mm->get_position().y;
  2355. if (windows[window_id].window_has_focus || window_get_active_popup() == window_id) {
  2356. Input::get_singleton()->parse_input_event(mm);
  2357. }
  2358. }
  2359. return 0;
  2360. }
  2361. } break;
  2362. case WM_POINTERENTER: {
  2363. if (mouse_mode == MOUSE_MODE_CAPTURED && use_raw_input) {
  2364. break;
  2365. }
  2366. if ((tablet_get_current_driver() != "winink") || !winink_available) {
  2367. break;
  2368. }
  2369. uint32_t pointer_id = LOWORD(wParam);
  2370. POINTER_INPUT_TYPE pointer_type = PT_POINTER;
  2371. if (!win8p_GetPointerType(pointer_id, &pointer_type)) {
  2372. break;
  2373. }
  2374. if (pointer_type != PT_PEN) {
  2375. break;
  2376. }
  2377. windows[window_id].block_mm = true;
  2378. return 0;
  2379. } break;
  2380. case WM_POINTERLEAVE: {
  2381. windows[window_id].block_mm = false;
  2382. return 0;
  2383. } break;
  2384. case WM_POINTERUPDATE: {
  2385. if (mouse_mode == MOUSE_MODE_CAPTURED && use_raw_input) {
  2386. break;
  2387. }
  2388. if ((tablet_get_current_driver() != "winink") || !winink_available) {
  2389. break;
  2390. }
  2391. uint32_t pointer_id = LOWORD(wParam);
  2392. POINTER_INPUT_TYPE pointer_type = PT_POINTER;
  2393. if (!win8p_GetPointerType(pointer_id, &pointer_type)) {
  2394. break;
  2395. }
  2396. if (pointer_type != PT_PEN) {
  2397. break;
  2398. }
  2399. POINTER_PEN_INFO pen_info;
  2400. if (!win8p_GetPointerPenInfo(pointer_id, &pen_info)) {
  2401. break;
  2402. }
  2403. if (Input::get_singleton()->is_emulating_mouse_from_touch()) {
  2404. // Universal translation enabled; ignore OS translation.
  2405. LPARAM extra = GetMessageExtraInfo();
  2406. if (IsTouchEvent(extra)) {
  2407. break;
  2408. }
  2409. }
  2410. if (window_mouseover_id != window_id) {
  2411. // Mouse enter.
  2412. if (mouse_mode != MOUSE_MODE_CAPTURED) {
  2413. if (window_mouseover_id != INVALID_WINDOW_ID && windows.has(window_mouseover_id)) {
  2414. // Leave previous window.
  2415. _send_window_event(windows[window_mouseover_id], WINDOW_EVENT_MOUSE_EXIT);
  2416. }
  2417. _send_window_event(windows[window_id], WINDOW_EVENT_MOUSE_ENTER);
  2418. }
  2419. CursorShape c = cursor_shape;
  2420. cursor_shape = CURSOR_MAX;
  2421. cursor_set_shape(c);
  2422. window_mouseover_id = window_id;
  2423. // Once-off notification, must call again.
  2424. track_mouse_leave_event(hWnd);
  2425. }
  2426. // Don't calculate relative mouse movement if we don't have focus in CAPTURED mode.
  2427. if (!windows[window_id].window_has_focus && mouse_mode == MOUSE_MODE_CAPTURED) {
  2428. break;
  2429. }
  2430. Ref<InputEventMouseMotion> mm;
  2431. mm.instantiate();
  2432. mm->set_window_id(window_id);
  2433. if (pen_info.penMask & PEN_MASK_PRESSURE) {
  2434. mm->set_pressure((float)pen_info.pressure / 1024);
  2435. } else {
  2436. mm->set_pressure((HIWORD(wParam) & POINTER_MESSAGE_FLAG_FIRSTBUTTON) ? 1.0f : 0.0f);
  2437. }
  2438. if ((pen_info.penMask & PEN_MASK_TILT_X) && (pen_info.penMask & PEN_MASK_TILT_Y)) {
  2439. mm->set_tilt(Vector2((float)pen_info.tiltX / 90, (float)pen_info.tiltY / 90));
  2440. }
  2441. mm->set_pen_inverted(pen_info.penFlags & (PEN_FLAG_INVERTED | PEN_FLAG_ERASER));
  2442. mm->set_ctrl_pressed(GetKeyState(VK_CONTROL) < 0);
  2443. mm->set_shift_pressed(GetKeyState(VK_SHIFT) < 0);
  2444. mm->set_alt_pressed(alt_mem);
  2445. mm->set_button_mask(last_button_state);
  2446. POINT coords; // Client coords.
  2447. coords.x = GET_X_LPARAM(lParam);
  2448. coords.y = GET_Y_LPARAM(lParam);
  2449. ScreenToClient(windows[window_id].hWnd, &coords);
  2450. mm->set_position(Vector2(coords.x, coords.y));
  2451. mm->set_global_position(Vector2(coords.x, coords.y));
  2452. if (mouse_mode == MOUSE_MODE_CAPTURED) {
  2453. Point2i c(windows[window_id].width / 2, windows[window_id].height / 2);
  2454. old_x = c.x;
  2455. old_y = c.y;
  2456. if (mm->get_position() == c) {
  2457. center = c;
  2458. return 0;
  2459. }
  2460. Point2i ncenter = mm->get_position();
  2461. center = ncenter;
  2462. POINT pos = { (int)c.x, (int)c.y };
  2463. ClientToScreen(hWnd, &pos);
  2464. SetCursorPos(pos.x, pos.y);
  2465. }
  2466. mm->set_velocity(Input::get_singleton()->get_last_mouse_velocity());
  2467. if (old_invalid) {
  2468. old_x = mm->get_position().x;
  2469. old_y = mm->get_position().y;
  2470. old_invalid = false;
  2471. }
  2472. mm->set_relative(Vector2(mm->get_position() - Vector2(old_x, old_y)));
  2473. old_x = mm->get_position().x;
  2474. old_y = mm->get_position().y;
  2475. if (windows[window_id].window_has_focus || window_get_active_popup() == window_id) {
  2476. Input::get_singleton()->parse_input_event(mm);
  2477. }
  2478. return 0; // Pointer event handled return 0 to avoid duplicate WM_MOUSEMOVE event.
  2479. } break;
  2480. case WM_MOUSEMOVE: {
  2481. if (windows[window_id].block_mm) {
  2482. break;
  2483. }
  2484. if (mouse_mode == MOUSE_MODE_CAPTURED && use_raw_input) {
  2485. break;
  2486. }
  2487. if (Input::get_singleton()->is_emulating_mouse_from_touch()) {
  2488. // Universal translation enabled; ignore OS translation.
  2489. LPARAM extra = GetMessageExtraInfo();
  2490. if (IsTouchEvent(extra)) {
  2491. break;
  2492. }
  2493. }
  2494. DisplayServer::WindowID over_id = get_window_at_screen_position(mouse_get_position());
  2495. if (windows.has(over_id) && !Rect2(window_get_position(over_id), Point2(windows[over_id].width, windows[over_id].height)).has_point(mouse_get_position())) {
  2496. // Don't consider the windowborder as part of the window.
  2497. over_id = INVALID_WINDOW_ID;
  2498. }
  2499. if (window_mouseover_id != over_id) {
  2500. // Mouse enter.
  2501. if (mouse_mode != MOUSE_MODE_CAPTURED) {
  2502. if (window_mouseover_id != INVALID_WINDOW_ID && windows.has(window_mouseover_id)) {
  2503. // Leave previous window.
  2504. _send_window_event(windows[window_mouseover_id], WINDOW_EVENT_MOUSE_EXIT);
  2505. }
  2506. if (over_id != INVALID_WINDOW_ID && windows.has(over_id)) {
  2507. _send_window_event(windows[over_id], WINDOW_EVENT_MOUSE_ENTER);
  2508. }
  2509. }
  2510. CursorShape c = cursor_shape;
  2511. cursor_shape = CURSOR_MAX;
  2512. cursor_set_shape(c);
  2513. window_mouseover_id = over_id;
  2514. // Once-off notification, must call again.
  2515. track_mouse_leave_event(hWnd);
  2516. }
  2517. // Don't calculate relative mouse movement if we don't have focus in CAPTURED mode.
  2518. if (!windows[window_id].window_has_focus && mouse_mode == MOUSE_MODE_CAPTURED) {
  2519. break;
  2520. }
  2521. DisplayServer::WindowID receiving_window_id = _get_focused_window_or_popup();
  2522. if (receiving_window_id == INVALID_WINDOW_ID) {
  2523. receiving_window_id = window_id;
  2524. }
  2525. Ref<InputEventMouseMotion> mm;
  2526. mm.instantiate();
  2527. mm->set_window_id(receiving_window_id);
  2528. mm->set_ctrl_pressed((wParam & MK_CONTROL) != 0);
  2529. mm->set_shift_pressed((wParam & MK_SHIFT) != 0);
  2530. mm->set_alt_pressed(alt_mem);
  2531. if ((tablet_get_current_driver() == "wintab") && wintab_available && windows[window_id].wtctx) {
  2532. // Note: WinTab sends both WT_PACKET and WM_xBUTTONDOWN/UP/MOUSEMOVE events, use mouse 1/0 pressure only when last_pressure was not updated recently.
  2533. if (windows[window_id].last_pressure_update < 10) {
  2534. windows[window_id].last_pressure_update++;
  2535. } else {
  2536. windows[window_id].last_tilt = Vector2();
  2537. windows[window_id].last_pressure = (wParam & MK_LBUTTON) ? 1.0f : 0.0f;
  2538. windows[window_id].last_pen_inverted = false;
  2539. }
  2540. } else {
  2541. windows[window_id].last_tilt = Vector2();
  2542. windows[window_id].last_pressure = (wParam & MK_LBUTTON) ? 1.0f : 0.0f;
  2543. windows[window_id].last_pen_inverted = false;
  2544. }
  2545. mm->set_pressure(windows[window_id].last_pressure);
  2546. mm->set_tilt(windows[window_id].last_tilt);
  2547. mm->set_pen_inverted(windows[window_id].last_pen_inverted);
  2548. mm->set_button_mask(last_button_state);
  2549. mm->set_position(Vector2(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)));
  2550. mm->set_global_position(Vector2(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)));
  2551. if (mouse_mode == MOUSE_MODE_CAPTURED) {
  2552. Point2i c(windows[window_id].width / 2, windows[window_id].height / 2);
  2553. old_x = c.x;
  2554. old_y = c.y;
  2555. if (mm->get_position() == c) {
  2556. center = c;
  2557. return 0;
  2558. }
  2559. Point2i ncenter = mm->get_position();
  2560. center = ncenter;
  2561. POINT pos = { (int)c.x, (int)c.y };
  2562. ClientToScreen(windows[window_id].hWnd, &pos);
  2563. SetCursorPos(pos.x, pos.y);
  2564. }
  2565. mm->set_velocity(Input::get_singleton()->get_last_mouse_velocity());
  2566. if (old_invalid) {
  2567. old_x = mm->get_position().x;
  2568. old_y = mm->get_position().y;
  2569. old_invalid = false;
  2570. }
  2571. mm->set_relative(Vector2(mm->get_position() - Vector2(old_x, old_y)));
  2572. old_x = mm->get_position().x;
  2573. old_y = mm->get_position().y;
  2574. if (receiving_window_id != window_id) {
  2575. // Adjust event position relative to window distance when event is sent to a different window.
  2576. mm->set_position(mm->get_position() - window_get_position(receiving_window_id) + window_get_position(window_id));
  2577. mm->set_global_position(mm->get_position());
  2578. }
  2579. Input::get_singleton()->parse_input_event(mm);
  2580. } break;
  2581. case WM_LBUTTONDOWN:
  2582. case WM_LBUTTONUP:
  2583. if (Input::get_singleton()->is_emulating_mouse_from_touch()) {
  2584. // Universal translation enabled; ignore OS translations for left button.
  2585. LPARAM extra = GetMessageExtraInfo();
  2586. if (IsTouchEvent(extra)) {
  2587. break;
  2588. }
  2589. }
  2590. [[fallthrough]];
  2591. case WM_MBUTTONDOWN:
  2592. case WM_MBUTTONUP:
  2593. case WM_RBUTTONDOWN:
  2594. case WM_RBUTTONUP:
  2595. case WM_MOUSEWHEEL:
  2596. case WM_MOUSEHWHEEL:
  2597. case WM_LBUTTONDBLCLK:
  2598. case WM_MBUTTONDBLCLK:
  2599. case WM_RBUTTONDBLCLK:
  2600. case WM_XBUTTONDBLCLK:
  2601. case WM_XBUTTONDOWN:
  2602. case WM_XBUTTONUP: {
  2603. Ref<InputEventMouseButton> mb;
  2604. mb.instantiate();
  2605. mb->set_window_id(window_id);
  2606. switch (uMsg) {
  2607. case WM_LBUTTONDOWN: {
  2608. mb->set_pressed(true);
  2609. mb->set_button_index(MouseButton::LEFT);
  2610. } break;
  2611. case WM_LBUTTONUP: {
  2612. mb->set_pressed(false);
  2613. mb->set_button_index(MouseButton::LEFT);
  2614. } break;
  2615. case WM_MBUTTONDOWN: {
  2616. mb->set_pressed(true);
  2617. mb->set_button_index(MouseButton::MIDDLE);
  2618. } break;
  2619. case WM_MBUTTONUP: {
  2620. mb->set_pressed(false);
  2621. mb->set_button_index(MouseButton::MIDDLE);
  2622. } break;
  2623. case WM_RBUTTONDOWN: {
  2624. mb->set_pressed(true);
  2625. mb->set_button_index(MouseButton::RIGHT);
  2626. } break;
  2627. case WM_RBUTTONUP: {
  2628. mb->set_pressed(false);
  2629. mb->set_button_index(MouseButton::RIGHT);
  2630. } break;
  2631. case WM_LBUTTONDBLCLK: {
  2632. mb->set_pressed(true);
  2633. mb->set_button_index(MouseButton::LEFT);
  2634. mb->set_double_click(true);
  2635. } break;
  2636. case WM_RBUTTONDBLCLK: {
  2637. mb->set_pressed(true);
  2638. mb->set_button_index(MouseButton::RIGHT);
  2639. mb->set_double_click(true);
  2640. } break;
  2641. case WM_MBUTTONDBLCLK: {
  2642. mb->set_pressed(true);
  2643. mb->set_button_index(MouseButton::MIDDLE);
  2644. mb->set_double_click(true);
  2645. } break;
  2646. case WM_MOUSEWHEEL: {
  2647. mb->set_pressed(true);
  2648. int motion = (short)HIWORD(wParam);
  2649. if (!motion) {
  2650. return 0;
  2651. }
  2652. if (motion > 0) {
  2653. mb->set_button_index(MouseButton::WHEEL_UP);
  2654. } else {
  2655. mb->set_button_index(MouseButton::WHEEL_DOWN);
  2656. }
  2657. mb->set_factor(fabs((double)motion / (double)WHEEL_DELTA));
  2658. } break;
  2659. case WM_MOUSEHWHEEL: {
  2660. mb->set_pressed(true);
  2661. int motion = (short)HIWORD(wParam);
  2662. if (!motion) {
  2663. return 0;
  2664. }
  2665. if (motion < 0) {
  2666. mb->set_button_index(MouseButton::WHEEL_LEFT);
  2667. } else {
  2668. mb->set_button_index(MouseButton::WHEEL_RIGHT);
  2669. }
  2670. mb->set_factor(fabs((double)motion / (double)WHEEL_DELTA));
  2671. } break;
  2672. case WM_XBUTTONDOWN: {
  2673. mb->set_pressed(true);
  2674. if (HIWORD(wParam) == XBUTTON1) {
  2675. mb->set_button_index(MouseButton::MB_XBUTTON1);
  2676. } else {
  2677. mb->set_button_index(MouseButton::MB_XBUTTON2);
  2678. }
  2679. } break;
  2680. case WM_XBUTTONUP: {
  2681. mb->set_pressed(false);
  2682. if (HIWORD(wParam) == XBUTTON1) {
  2683. mb->set_button_index(MouseButton::MB_XBUTTON1);
  2684. } else {
  2685. mb->set_button_index(MouseButton::MB_XBUTTON2);
  2686. }
  2687. } break;
  2688. case WM_XBUTTONDBLCLK: {
  2689. mb->set_pressed(true);
  2690. if (HIWORD(wParam) == XBUTTON1) {
  2691. mb->set_button_index(MouseButton::MB_XBUTTON1);
  2692. } else {
  2693. mb->set_button_index(MouseButton::MB_XBUTTON2);
  2694. }
  2695. mb->set_double_click(true);
  2696. } break;
  2697. default: {
  2698. return 0;
  2699. }
  2700. }
  2701. mb->set_ctrl_pressed((wParam & MK_CONTROL) != 0);
  2702. mb->set_shift_pressed((wParam & MK_SHIFT) != 0);
  2703. mb->set_alt_pressed(alt_mem);
  2704. // mb->is_alt_pressed()=(wParam&MK_MENU)!=0;
  2705. if (mb->is_pressed()) {
  2706. last_button_state.set_flag(mouse_button_to_mask(mb->get_button_index()));
  2707. } else {
  2708. last_button_state.clear_flag(mouse_button_to_mask(mb->get_button_index()));
  2709. }
  2710. mb->set_button_mask(last_button_state);
  2711. mb->set_position(Vector2(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)));
  2712. if (mouse_mode == MOUSE_MODE_CAPTURED && !use_raw_input) {
  2713. mb->set_position(Vector2(old_x, old_y));
  2714. }
  2715. if (uMsg != WM_MOUSEWHEEL && uMsg != WM_MOUSEHWHEEL) {
  2716. if (mb->is_pressed()) {
  2717. if (++pressrc > 0 && mouse_mode != MOUSE_MODE_CAPTURED) {
  2718. SetCapture(hWnd);
  2719. }
  2720. } else {
  2721. if (--pressrc <= 0) {
  2722. if (mouse_mode != MOUSE_MODE_CAPTURED) {
  2723. ReleaseCapture();
  2724. }
  2725. pressrc = 0;
  2726. }
  2727. }
  2728. } else {
  2729. // For reasons unknown to humanity, wheel comes in screen coordinates.
  2730. POINT coords;
  2731. coords.x = mb->get_position().x;
  2732. coords.y = mb->get_position().y;
  2733. ScreenToClient(hWnd, &coords);
  2734. mb->set_position(Vector2(coords.x, coords.y));
  2735. }
  2736. mb->set_global_position(mb->get_position());
  2737. Input::get_singleton()->parse_input_event(mb);
  2738. if (mb->is_pressed() && mb->get_button_index() >= MouseButton::WHEEL_UP && mb->get_button_index() <= MouseButton::WHEEL_RIGHT) {
  2739. // Send release for mouse wheel.
  2740. Ref<InputEventMouseButton> mbd = mb->duplicate();
  2741. mbd->set_window_id(window_id);
  2742. last_button_state.clear_flag(mouse_button_to_mask(mbd->get_button_index()));
  2743. mbd->set_button_mask(last_button_state);
  2744. mbd->set_pressed(false);
  2745. Input::get_singleton()->parse_input_event(mbd);
  2746. }
  2747. } break;
  2748. case WM_WINDOWPOSCHANGED: {
  2749. Rect2i window_client_rect;
  2750. Rect2i window_rect;
  2751. {
  2752. RECT rect;
  2753. GetClientRect(hWnd, &rect);
  2754. ClientToScreen(hWnd, (POINT *)&rect.left);
  2755. ClientToScreen(hWnd, (POINT *)&rect.right);
  2756. window_client_rect = Rect2i(rect.left, rect.top, rect.right - rect.left, rect.bottom - rect.top);
  2757. window_client_rect.position -= _get_screens_origin();
  2758. RECT wrect;
  2759. GetWindowRect(hWnd, &wrect);
  2760. window_rect = Rect2i(wrect.left, wrect.top, wrect.right - wrect.left, wrect.bottom - wrect.top);
  2761. window_rect.position -= _get_screens_origin();
  2762. }
  2763. WINDOWPOS *window_pos_params = (WINDOWPOS *)lParam;
  2764. WindowData &window = windows[window_id];
  2765. bool rect_changed = false;
  2766. if (!(window_pos_params->flags & SWP_NOSIZE) || window_pos_params->flags & SWP_FRAMECHANGED) {
  2767. int screen_id = window_get_current_screen(window_id);
  2768. Size2i screen_size = screen_get_size(screen_id);
  2769. Point2i screen_position = screen_get_position(screen_id);
  2770. window.maximized = false;
  2771. window.minimized = false;
  2772. window.fullscreen = false;
  2773. if (IsIconic(hWnd)) {
  2774. window.minimized = true;
  2775. } else if (IsZoomed(hWnd)) {
  2776. window.maximized = true;
  2777. } else if (window_rect.position == screen_position && window_rect.size == screen_size) {
  2778. window.fullscreen = true;
  2779. }
  2780. if (!window.minimized) {
  2781. window.width = window_client_rect.size.width;
  2782. window.height = window_client_rect.size.height;
  2783. rect_changed = true;
  2784. }
  2785. #if defined(VULKAN_ENABLED)
  2786. if (context_vulkan && window.context_created) {
  2787. // Note: Trigger resize event to update swapchains when window is minimized/restored, even if size is not changed.
  2788. context_vulkan->window_resize(window_id, window.width, window.height);
  2789. }
  2790. #endif
  2791. }
  2792. if (!window.minimized && (!(window_pos_params->flags & SWP_NOMOVE) || window_pos_params->flags & SWP_FRAMECHANGED)) {
  2793. window.last_pos = window_client_rect.position;
  2794. rect_changed = true;
  2795. }
  2796. if (rect_changed) {
  2797. if (!window.rect_changed_callback.is_null()) {
  2798. Variant size = Rect2i(window.last_pos.x, window.last_pos.y, window.width, window.height);
  2799. const Variant *args[] = { &size };
  2800. Variant ret;
  2801. Callable::CallError ce;
  2802. window.rect_changed_callback.callp(args, 1, ret, ce);
  2803. }
  2804. }
  2805. // Return here to prevent WM_MOVE and WM_SIZE from being sent
  2806. // See: https://docs.microsoft.com/en-us/windows/win32/winmsg/wm-windowposchanged#remarks
  2807. return 0;
  2808. } break;
  2809. case WM_ENTERSIZEMOVE: {
  2810. Input::get_singleton()->release_pressed_events();
  2811. windows[window_id].move_timer_id = SetTimer(windows[window_id].hWnd, 1, USER_TIMER_MINIMUM, (TIMERPROC) nullptr);
  2812. } break;
  2813. case WM_EXITSIZEMOVE: {
  2814. KillTimer(windows[window_id].hWnd, windows[window_id].move_timer_id);
  2815. } break;
  2816. case WM_TIMER: {
  2817. if (wParam == windows[window_id].move_timer_id) {
  2818. _process_key_events();
  2819. if (!Main::is_iterating()) {
  2820. Main::iteration();
  2821. }
  2822. } else if (wParam == windows[window_id].focus_timer_id) {
  2823. _process_activate_event(window_id, windows[window_id].saved_wparam, windows[window_id].saved_lparam);
  2824. KillTimer(windows[window_id].hWnd, wParam);
  2825. windows[window_id].focus_timer_id = 0U;
  2826. }
  2827. } break;
  2828. case WM_SYSKEYUP:
  2829. case WM_KEYUP:
  2830. if (windows[window_id].ime_suppress_next_keyup) {
  2831. windows[window_id].ime_suppress_next_keyup = false;
  2832. break;
  2833. }
  2834. [[fallthrough]];
  2835. case WM_SYSKEYDOWN:
  2836. case WM_KEYDOWN: {
  2837. if (windows[window_id].ime_in_progress) {
  2838. break;
  2839. }
  2840. if (wParam == VK_SHIFT) {
  2841. shift_mem = (uMsg == WM_KEYDOWN || uMsg == WM_SYSKEYDOWN);
  2842. }
  2843. if (wParam == VK_CONTROL) {
  2844. control_mem = (uMsg == WM_KEYDOWN || uMsg == WM_SYSKEYDOWN);
  2845. }
  2846. if (wParam == VK_MENU) {
  2847. alt_mem = (uMsg == WM_KEYDOWN || uMsg == WM_SYSKEYDOWN);
  2848. if (lParam & (1 << 24)) {
  2849. gr_mem = alt_mem;
  2850. }
  2851. }
  2852. if (mouse_mode == MOUSE_MODE_CAPTURED) {
  2853. // When SetCapture is used, ALT+F4 hotkey is ignored by Windows, so handle it ourselves
  2854. if (wParam == VK_F4 && alt_mem && (uMsg == WM_KEYDOWN || uMsg == WM_SYSKEYDOWN)) {
  2855. _send_window_event(windows[window_id], WINDOW_EVENT_CLOSE_REQUEST);
  2856. }
  2857. }
  2858. [[fallthrough]];
  2859. }
  2860. case WM_CHAR: {
  2861. if (windows[window_id].ime_in_progress) {
  2862. break;
  2863. }
  2864. ERR_BREAK(key_event_pos >= KEY_EVENT_BUFFER_SIZE);
  2865. // Make sure we don't include modifiers for the modifier key itself.
  2866. KeyEvent ke;
  2867. ke.shift = (wParam != VK_SHIFT) ? shift_mem : false;
  2868. ke.alt = (!(wParam == VK_MENU && (uMsg == WM_KEYDOWN || uMsg == WM_SYSKEYDOWN))) ? alt_mem : false;
  2869. ke.control = (wParam != VK_CONTROL) ? control_mem : false;
  2870. ke.meta = meta_mem;
  2871. ke.uMsg = uMsg;
  2872. ke.window_id = window_id;
  2873. if (ke.uMsg == WM_SYSKEYDOWN) {
  2874. ke.uMsg = WM_KEYDOWN;
  2875. }
  2876. if (ke.uMsg == WM_SYSKEYUP) {
  2877. ke.uMsg = WM_KEYUP;
  2878. }
  2879. ke.wParam = wParam;
  2880. ke.lParam = lParam;
  2881. key_event_buffer[key_event_pos++] = ke;
  2882. } break;
  2883. case WM_IME_COMPOSITION: {
  2884. CANDIDATEFORM cf;
  2885. cf.dwIndex = 0;
  2886. cf.dwStyle = CFS_CANDIDATEPOS;
  2887. cf.ptCurrentPos.x = windows[window_id].im_position.x;
  2888. cf.ptCurrentPos.y = windows[window_id].im_position.y;
  2889. ImmSetCandidateWindow(windows[window_id].im_himc, &cf);
  2890. cf.dwStyle = CFS_EXCLUDE;
  2891. cf.rcArea.left = windows[window_id].im_position.x;
  2892. cf.rcArea.right = windows[window_id].im_position.x;
  2893. cf.rcArea.top = windows[window_id].im_position.y;
  2894. cf.rcArea.bottom = windows[window_id].im_position.y;
  2895. ImmSetCandidateWindow(windows[window_id].im_himc, &cf);
  2896. if (windows[window_id].ime_active) {
  2897. SetCaretPos(windows[window_id].im_position.x, windows[window_id].im_position.y);
  2898. OS::get_singleton()->get_main_loop()->notification(MainLoop::NOTIFICATION_OS_IME_UPDATE);
  2899. }
  2900. } break;
  2901. case WM_INPUTLANGCHANGEREQUEST: {
  2902. // FIXME: Do something?
  2903. } break;
  2904. case WM_IME_STARTCOMPOSITION: {
  2905. if (windows[window_id].ime_active) {
  2906. windows[window_id].ime_in_progress = true;
  2907. if (key_event_pos > 0) {
  2908. key_event_pos--;
  2909. }
  2910. }
  2911. return 0;
  2912. } break;
  2913. case WM_IME_ENDCOMPOSITION: {
  2914. if (windows[window_id].ime_active) {
  2915. windows[window_id].ime_in_progress = false;
  2916. windows[window_id].ime_suppress_next_keyup = true;
  2917. }
  2918. return 0;
  2919. } break;
  2920. case WM_IME_NOTIFY: {
  2921. return 0;
  2922. } break;
  2923. case WM_TOUCH: {
  2924. BOOL bHandled = FALSE;
  2925. UINT cInputs = LOWORD(wParam);
  2926. PTOUCHINPUT pInputs = memnew_arr(TOUCHINPUT, cInputs);
  2927. if (pInputs) {
  2928. if (GetTouchInputInfo((HTOUCHINPUT)lParam, cInputs, pInputs, sizeof(TOUCHINPUT))) {
  2929. for (UINT i = 0; i < cInputs; i++) {
  2930. TOUCHINPUT ti = pInputs[i];
  2931. POINT touch_pos = {
  2932. TOUCH_COORD_TO_PIXEL(ti.x),
  2933. TOUCH_COORD_TO_PIXEL(ti.y),
  2934. };
  2935. ScreenToClient(hWnd, &touch_pos);
  2936. // Do something with each touch input entry.
  2937. if (ti.dwFlags & TOUCHEVENTF_MOVE) {
  2938. _drag_event(window_id, touch_pos.x, touch_pos.y, ti.dwID);
  2939. } else if (ti.dwFlags & (TOUCHEVENTF_UP | TOUCHEVENTF_DOWN)) {
  2940. _touch_event(window_id, ti.dwFlags & TOUCHEVENTF_DOWN, touch_pos.x, touch_pos.y, ti.dwID);
  2941. }
  2942. }
  2943. bHandled = TRUE;
  2944. } else {
  2945. // TODO: Handle the error here.
  2946. }
  2947. memdelete_arr(pInputs);
  2948. } else {
  2949. // TODO: Handle the error here, probably out of memory.
  2950. }
  2951. if (bHandled) {
  2952. CloseTouchInputHandle((HTOUCHINPUT)lParam);
  2953. return 0;
  2954. }
  2955. } break;
  2956. case WM_DEVICECHANGE: {
  2957. joypad->probe_joypads();
  2958. } break;
  2959. case WM_DESTROY: {
  2960. Input::get_singleton()->flush_buffered_events();
  2961. } break;
  2962. case WM_SETCURSOR: {
  2963. if (LOWORD(lParam) == HTCLIENT) {
  2964. if (windows[window_id].window_has_focus && (mouse_mode == MOUSE_MODE_HIDDEN || mouse_mode == MOUSE_MODE_CAPTURED || mouse_mode == MOUSE_MODE_CONFINED_HIDDEN)) {
  2965. // Hide the cursor.
  2966. if (hCursor == nullptr) {
  2967. hCursor = SetCursor(nullptr);
  2968. } else {
  2969. SetCursor(nullptr);
  2970. }
  2971. } else {
  2972. if (hCursor != nullptr) {
  2973. CursorShape c = cursor_shape;
  2974. cursor_shape = CURSOR_MAX;
  2975. cursor_set_shape(c);
  2976. hCursor = nullptr;
  2977. }
  2978. }
  2979. }
  2980. } break;
  2981. case WM_DROPFILES: {
  2982. HDROP hDropInfo = (HDROP)wParam;
  2983. const int buffsize = 4096;
  2984. WCHAR buf[buffsize];
  2985. int fcount = DragQueryFileW(hDropInfo, 0xFFFFFFFF, nullptr, 0);
  2986. Vector<String> files;
  2987. for (int i = 0; i < fcount; i++) {
  2988. DragQueryFileW(hDropInfo, i, buf, buffsize);
  2989. String file = String::utf16((const char16_t *)buf);
  2990. files.push_back(file);
  2991. }
  2992. if (files.size() && !windows[window_id].drop_files_callback.is_null()) {
  2993. Variant v = files;
  2994. Variant *vp = &v;
  2995. Variant ret;
  2996. Callable::CallError ce;
  2997. windows[window_id].drop_files_callback.callp((const Variant **)&vp, 1, ret, ce);
  2998. }
  2999. } break;
  3000. default: {
  3001. if (user_proc) {
  3002. return CallWindowProcW(user_proc, hWnd, uMsg, wParam, lParam);
  3003. }
  3004. }
  3005. }
  3006. return DefWindowProcW(hWnd, uMsg, wParam, lParam);
  3007. }
  3008. LRESULT CALLBACK WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) {
  3009. DisplayServerWindows *ds_win = static_cast<DisplayServerWindows *>(DisplayServer::get_singleton());
  3010. if (ds_win) {
  3011. return ds_win->WndProc(hWnd, uMsg, wParam, lParam);
  3012. } else {
  3013. return DefWindowProcW(hWnd, uMsg, wParam, lParam);
  3014. }
  3015. }
  3016. void DisplayServerWindows::_process_activate_event(WindowID p_window_id, WPARAM wParam, LPARAM lParam) {
  3017. if (LOWORD(wParam) == WA_ACTIVE || LOWORD(wParam) == WA_CLICKACTIVE) {
  3018. _send_window_event(windows[p_window_id], WINDOW_EVENT_FOCUS_IN);
  3019. windows[p_window_id].window_focused = true;
  3020. alt_mem = false;
  3021. control_mem = false;
  3022. shift_mem = false;
  3023. gr_mem = false;
  3024. // Restore mouse mode.
  3025. _set_mouse_mode_impl(mouse_mode);
  3026. } else { // WM_INACTIVE.
  3027. Input::get_singleton()->release_pressed_events();
  3028. _send_window_event(windows[p_window_id], WINDOW_EVENT_FOCUS_OUT);
  3029. windows[p_window_id].window_focused = false;
  3030. alt_mem = false;
  3031. }
  3032. if ((tablet_get_current_driver() == "wintab") && wintab_available && windows[p_window_id].wtctx) {
  3033. wintab_WTEnable(windows[p_window_id].wtctx, GET_WM_ACTIVATE_STATE(wParam, lParam));
  3034. }
  3035. }
  3036. void DisplayServerWindows::_process_key_events() {
  3037. for (int i = 0; i < key_event_pos; i++) {
  3038. KeyEvent &ke = key_event_buffer[i];
  3039. switch (ke.uMsg) {
  3040. case WM_CHAR: {
  3041. // Extended keys should only be processed as WM_KEYDOWN message.
  3042. if (!KeyMappingWindows::is_extended_key(ke.wParam) && ((i == 0 && ke.uMsg == WM_CHAR) || (i > 0 && key_event_buffer[i - 1].uMsg == WM_CHAR))) {
  3043. static char32_t prev_wc = 0;
  3044. char32_t unicode = ke.wParam;
  3045. if ((unicode & 0xfffffc00) == 0xd800) {
  3046. if (prev_wc != 0) {
  3047. ERR_PRINT("invalid utf16 surrogate input");
  3048. }
  3049. prev_wc = unicode;
  3050. break; // Skip surrogate.
  3051. } else if ((unicode & 0xfffffc00) == 0xdc00) {
  3052. if (prev_wc == 0) {
  3053. ERR_PRINT("invalid utf16 surrogate input");
  3054. break; // Skip invalid surrogate.
  3055. }
  3056. unicode = (prev_wc << 10UL) + unicode - ((0xd800 << 10UL) + 0xdc00 - 0x10000);
  3057. prev_wc = 0;
  3058. } else {
  3059. prev_wc = 0;
  3060. }
  3061. Ref<InputEventKey> k;
  3062. k.instantiate();
  3063. Key keycode = KeyMappingWindows::get_keysym(MapVirtualKey((ke.lParam >> 16) & 0xFF, MAPVK_VSC_TO_VK));
  3064. Key key_label = keycode;
  3065. Key physical_keycode = KeyMappingWindows::get_scansym((ke.lParam >> 16) & 0xFF, ke.lParam & (1 << 24));
  3066. static BYTE keyboard_state[256];
  3067. memset(keyboard_state, 0, 256);
  3068. wchar_t chars[256] = {};
  3069. UINT extended_code = MapVirtualKey((ke.lParam >> 16) & 0xFF, MAPVK_VSC_TO_VK_EX);
  3070. if (!(ke.lParam & (1 << 24)) && ToUnicodeEx(extended_code, (ke.lParam >> 16) & 0xFF, keyboard_state, chars, 255, 4, GetKeyboardLayout(0)) > 0) {
  3071. String keysym = String::utf16((char16_t *)chars, 255);
  3072. if (!keysym.is_empty()) {
  3073. key_label = fix_key_label(keysym[0], keycode);
  3074. }
  3075. }
  3076. k->set_window_id(ke.window_id);
  3077. k->set_shift_pressed(ke.shift);
  3078. k->set_alt_pressed(ke.alt);
  3079. k->set_ctrl_pressed(ke.control);
  3080. k->set_meta_pressed(ke.meta);
  3081. k->set_pressed(true);
  3082. k->set_keycode(keycode);
  3083. k->set_physical_keycode(physical_keycode);
  3084. k->set_key_label(key_label);
  3085. k->set_unicode(fix_unicode(unicode));
  3086. if (k->get_unicode() && gr_mem) {
  3087. k->set_alt_pressed(false);
  3088. k->set_ctrl_pressed(false);
  3089. }
  3090. Input::get_singleton()->parse_input_event(k);
  3091. } else {
  3092. // Do nothing.
  3093. }
  3094. } break;
  3095. case WM_KEYUP:
  3096. case WM_KEYDOWN: {
  3097. Ref<InputEventKey> k;
  3098. k.instantiate();
  3099. k->set_window_id(ke.window_id);
  3100. k->set_shift_pressed(ke.shift);
  3101. k->set_alt_pressed(ke.alt);
  3102. k->set_ctrl_pressed(ke.control);
  3103. k->set_meta_pressed(ke.meta);
  3104. k->set_pressed(ke.uMsg == WM_KEYDOWN);
  3105. Key keycode = KeyMappingWindows::get_keysym(ke.wParam);
  3106. if ((ke.lParam & (1 << 24)) && (ke.wParam == VK_RETURN)) {
  3107. // Special case for Numpad Enter key.
  3108. keycode = Key::KP_ENTER;
  3109. }
  3110. Key key_label = keycode;
  3111. Key physical_keycode = KeyMappingWindows::get_scansym((ke.lParam >> 16) & 0xFF, ke.lParam & (1 << 24));
  3112. static BYTE keyboard_state[256];
  3113. memset(keyboard_state, 0, 256);
  3114. wchar_t chars[256] = {};
  3115. UINT extended_code = MapVirtualKey((ke.lParam >> 16) & 0xFF, MAPVK_VSC_TO_VK_EX);
  3116. if (!(ke.lParam & (1 << 24)) && ToUnicodeEx(extended_code, (ke.lParam >> 16) & 0xFF, keyboard_state, chars, 255, 4, GetKeyboardLayout(0)) > 0) {
  3117. String keysym = String::utf16((char16_t *)chars, 255);
  3118. if (!keysym.is_empty()) {
  3119. key_label = fix_key_label(keysym[0], keycode);
  3120. }
  3121. }
  3122. k->set_keycode(keycode);
  3123. k->set_physical_keycode(physical_keycode);
  3124. k->set_key_label(key_label);
  3125. if (i + 1 < key_event_pos && key_event_buffer[i + 1].uMsg == WM_CHAR) {
  3126. char32_t unicode = key_event_buffer[i + 1].wParam;
  3127. static char32_t prev_wck = 0;
  3128. if ((unicode & 0xfffffc00) == 0xd800) {
  3129. if (prev_wck != 0) {
  3130. ERR_PRINT("invalid utf16 surrogate input");
  3131. }
  3132. prev_wck = unicode;
  3133. break; // Skip surrogate.
  3134. } else if ((unicode & 0xfffffc00) == 0xdc00) {
  3135. if (prev_wck == 0) {
  3136. ERR_PRINT("invalid utf16 surrogate input");
  3137. break; // Skip invalid surrogate.
  3138. }
  3139. unicode = (prev_wck << 10UL) + unicode - ((0xd800 << 10UL) + 0xdc00 - 0x10000);
  3140. prev_wck = 0;
  3141. } else {
  3142. prev_wck = 0;
  3143. }
  3144. k->set_unicode(fix_unicode(unicode));
  3145. }
  3146. if (k->get_unicode() && gr_mem) {
  3147. k->set_alt_pressed(false);
  3148. k->set_ctrl_pressed(false);
  3149. }
  3150. k->set_echo((ke.uMsg == WM_KEYDOWN && (ke.lParam & (1 << 30))));
  3151. Input::get_singleton()->parse_input_event(k);
  3152. } break;
  3153. }
  3154. }
  3155. key_event_pos = 0;
  3156. }
  3157. void DisplayServerWindows::_update_tablet_ctx(const String &p_old_driver, const String &p_new_driver) {
  3158. for (KeyValue<WindowID, WindowData> &E : windows) {
  3159. WindowData &wd = E.value;
  3160. wd.block_mm = false;
  3161. if ((p_old_driver == "wintab") && wintab_available && wd.wtctx) {
  3162. wintab_WTEnable(wd.wtctx, false);
  3163. wintab_WTClose(wd.wtctx);
  3164. wd.wtctx = 0;
  3165. }
  3166. if ((p_new_driver == "wintab") && wintab_available) {
  3167. wintab_WTInfo(WTI_DEFSYSCTX, 0, &wd.wtlc);
  3168. wd.wtlc.lcOptions |= CXO_MESSAGES;
  3169. wd.wtlc.lcPktData = PK_STATUS | PK_NORMAL_PRESSURE | PK_TANGENT_PRESSURE | PK_ORIENTATION;
  3170. wd.wtlc.lcMoveMask = PK_STATUS | PK_NORMAL_PRESSURE | PK_TANGENT_PRESSURE;
  3171. wd.wtlc.lcPktMode = 0;
  3172. wd.wtlc.lcOutOrgX = 0;
  3173. wd.wtlc.lcOutExtX = wd.wtlc.lcInExtX;
  3174. wd.wtlc.lcOutOrgY = 0;
  3175. wd.wtlc.lcOutExtY = -wd.wtlc.lcInExtY;
  3176. wd.wtctx = wintab_WTOpen(wd.hWnd, &wd.wtlc, false);
  3177. if (wd.wtctx) {
  3178. wintab_WTEnable(wd.wtctx, true);
  3179. AXIS pressure;
  3180. if (wintab_WTInfo(WTI_DEVICES + wd.wtlc.lcDevice, DVC_NPRESSURE, &pressure)) {
  3181. wd.min_pressure = int(pressure.axMin);
  3182. wd.max_pressure = int(pressure.axMax);
  3183. }
  3184. AXIS orientation[3];
  3185. if (wintab_WTInfo(WTI_DEVICES + wd.wtlc.lcDevice, DVC_ORIENTATION, &orientation)) {
  3186. wd.tilt_supported = orientation[0].axResolution && orientation[1].axResolution;
  3187. }
  3188. wintab_WTEnable(wd.wtctx, true);
  3189. } else {
  3190. print_verbose("WinTab context creation failed.");
  3191. }
  3192. }
  3193. }
  3194. }
  3195. DisplayServer::WindowID DisplayServerWindows::_create_window(WindowMode p_mode, VSyncMode p_vsync_mode, uint32_t p_flags, const Rect2i &p_rect) {
  3196. DWORD dwExStyle;
  3197. DWORD dwStyle;
  3198. _get_window_style(window_id_counter == MAIN_WINDOW_ID, (p_mode == WINDOW_MODE_FULLSCREEN || p_mode == WINDOW_MODE_EXCLUSIVE_FULLSCREEN), p_mode != WINDOW_MODE_EXCLUSIVE_FULLSCREEN, p_flags & WINDOW_FLAG_BORDERLESS_BIT, !(p_flags & WINDOW_FLAG_RESIZE_DISABLED_BIT), p_mode == WINDOW_MODE_MAXIMIZED, (p_flags & WINDOW_FLAG_NO_FOCUS_BIT) | (p_flags & WINDOW_FLAG_POPUP), dwStyle, dwExStyle);
  3199. RECT WindowRect;
  3200. WindowRect.left = p_rect.position.x;
  3201. WindowRect.right = p_rect.position.x + p_rect.size.x;
  3202. WindowRect.top = p_rect.position.y;
  3203. WindowRect.bottom = p_rect.position.y + p_rect.size.y;
  3204. int rq_screen = get_screen_from_rect(p_rect);
  3205. if (rq_screen < 0) {
  3206. rq_screen = get_primary_screen(); // Requested window rect is outside any screen bounds.
  3207. }
  3208. if (p_mode == WINDOW_MODE_FULLSCREEN || p_mode == WINDOW_MODE_EXCLUSIVE_FULLSCREEN) {
  3209. Rect2i screen_rect = Rect2i(screen_get_position(rq_screen), screen_get_size(rq_screen));
  3210. WindowRect.left = screen_rect.position.x;
  3211. WindowRect.right = screen_rect.position.x + screen_rect.size.x;
  3212. WindowRect.top = screen_rect.position.y;
  3213. WindowRect.bottom = screen_rect.position.y + screen_rect.size.y;
  3214. } else {
  3215. Rect2i srect = screen_get_usable_rect(rq_screen);
  3216. Point2i wpos = p_rect.position;
  3217. if (srect != Rect2i()) {
  3218. wpos.x = CLAMP(wpos.x, srect.position.x, srect.position.x + srect.size.width - p_rect.size.width / 3);
  3219. wpos.y = CLAMP(wpos.y, srect.position.y, srect.position.y + srect.size.height - p_rect.size.height / 3);
  3220. }
  3221. WindowRect.left = wpos.x;
  3222. WindowRect.right = wpos.x + p_rect.size.x;
  3223. WindowRect.top = wpos.y;
  3224. WindowRect.bottom = wpos.y + p_rect.size.y;
  3225. }
  3226. Point2i offset = _get_screens_origin();
  3227. WindowRect.left += offset.x;
  3228. WindowRect.right += offset.x;
  3229. WindowRect.top += offset.y;
  3230. WindowRect.bottom += offset.y;
  3231. AdjustWindowRectEx(&WindowRect, dwStyle, FALSE, dwExStyle);
  3232. WindowID id = window_id_counter;
  3233. {
  3234. WindowData &wd = windows[id];
  3235. wd.hWnd = CreateWindowExW(
  3236. dwExStyle,
  3237. L"Engine", L"",
  3238. dwStyle,
  3239. // (GetSystemMetrics(SM_CXSCREEN) - WindowRect.right) / 2,
  3240. // (GetSystemMetrics(SM_CYSCREEN) - WindowRect.bottom) / 2,
  3241. WindowRect.left,
  3242. WindowRect.top,
  3243. WindowRect.right - WindowRect.left,
  3244. WindowRect.bottom - WindowRect.top,
  3245. nullptr,
  3246. nullptr,
  3247. hInstance,
  3248. // tunnel the WindowData we need to handle creation message
  3249. // lifetime is ensured because we are still on the stack when this is
  3250. // processed in the window proc
  3251. reinterpret_cast<void *>(&wd));
  3252. if (!wd.hWnd) {
  3253. MessageBoxW(nullptr, L"Window Creation Error.", L"ERROR", MB_OK | MB_ICONEXCLAMATION);
  3254. windows.erase(id);
  3255. ERR_FAIL_V_MSG(INVALID_WINDOW_ID, "Failed to create Windows OS window.");
  3256. }
  3257. if (p_mode == WINDOW_MODE_FULLSCREEN || p_mode == WINDOW_MODE_EXCLUSIVE_FULLSCREEN) {
  3258. wd.fullscreen = true;
  3259. if (p_mode == WINDOW_MODE_FULLSCREEN) {
  3260. wd.multiwindow_fs = true;
  3261. }
  3262. }
  3263. if (p_mode != WINDOW_MODE_FULLSCREEN && p_mode != WINDOW_MODE_EXCLUSIVE_FULLSCREEN) {
  3264. wd.pre_fs_valid = true;
  3265. }
  3266. if (is_dark_mode_supported() && dark_title_available) {
  3267. BOOL value = is_dark_mode();
  3268. ::DwmSetWindowAttribute(wd.hWnd, DWMWA_USE_IMMERSIVE_DARK_MODE, &value, sizeof(value));
  3269. }
  3270. #ifdef VULKAN_ENABLED
  3271. if (context_vulkan) {
  3272. if (context_vulkan->window_create(id, p_vsync_mode, wd.hWnd, hInstance, WindowRect.right - WindowRect.left, WindowRect.bottom - WindowRect.top) != OK) {
  3273. memdelete(context_vulkan);
  3274. context_vulkan = nullptr;
  3275. windows.erase(id);
  3276. ERR_FAIL_V_MSG(INVALID_WINDOW_ID, "Failed to create Vulkan Window.");
  3277. }
  3278. wd.context_created = true;
  3279. }
  3280. #endif
  3281. #ifdef GLES3_ENABLED
  3282. if (gl_manager) {
  3283. if (gl_manager->window_create(id, wd.hWnd, hInstance, WindowRect.right - WindowRect.left, WindowRect.bottom - WindowRect.top) != OK) {
  3284. memdelete(gl_manager);
  3285. gl_manager = nullptr;
  3286. windows.erase(id);
  3287. ERR_FAIL_V_MSG(INVALID_WINDOW_ID, "Failed to create an OpenGL window.");
  3288. }
  3289. window_set_vsync_mode(p_vsync_mode, id);
  3290. }
  3291. #endif
  3292. RegisterTouchWindow(wd.hWnd, 0);
  3293. DragAcceptFiles(wd.hWnd, true);
  3294. if ((tablet_get_current_driver() == "wintab") && wintab_available) {
  3295. wintab_WTInfo(WTI_DEFSYSCTX, 0, &wd.wtlc);
  3296. wd.wtlc.lcOptions |= CXO_MESSAGES;
  3297. wd.wtlc.lcPktData = PK_STATUS | PK_NORMAL_PRESSURE | PK_TANGENT_PRESSURE | PK_ORIENTATION;
  3298. wd.wtlc.lcMoveMask = PK_STATUS | PK_NORMAL_PRESSURE | PK_TANGENT_PRESSURE;
  3299. wd.wtlc.lcPktMode = 0;
  3300. wd.wtlc.lcOutOrgX = 0;
  3301. wd.wtlc.lcOutExtX = wd.wtlc.lcInExtX;
  3302. wd.wtlc.lcOutOrgY = 0;
  3303. wd.wtlc.lcOutExtY = -wd.wtlc.lcInExtY;
  3304. wd.wtctx = wintab_WTOpen(wd.hWnd, &wd.wtlc, false);
  3305. if (wd.wtctx) {
  3306. wintab_WTEnable(wd.wtctx, true);
  3307. AXIS pressure;
  3308. if (wintab_WTInfo(WTI_DEVICES + wd.wtlc.lcDevice, DVC_NPRESSURE, &pressure)) {
  3309. wd.min_pressure = int(pressure.axMin);
  3310. wd.max_pressure = int(pressure.axMax);
  3311. }
  3312. AXIS orientation[3];
  3313. if (wintab_WTInfo(WTI_DEVICES + wd.wtlc.lcDevice, DVC_ORIENTATION, &orientation)) {
  3314. wd.tilt_supported = orientation[0].axResolution && orientation[1].axResolution;
  3315. }
  3316. } else {
  3317. print_verbose("WinTab context creation failed.");
  3318. }
  3319. } else {
  3320. wd.wtctx = 0;
  3321. }
  3322. if (p_mode == WINDOW_MODE_MAXIMIZED) {
  3323. wd.maximized = true;
  3324. wd.minimized = false;
  3325. }
  3326. if (p_mode == WINDOW_MODE_MINIMIZED) {
  3327. wd.maximized = false;
  3328. wd.minimized = true;
  3329. }
  3330. wd.last_pressure = 0;
  3331. wd.last_pressure_update = 0;
  3332. wd.last_tilt = Vector2();
  3333. // IME.
  3334. wd.im_himc = ImmGetContext(wd.hWnd);
  3335. ImmAssociateContext(wd.hWnd, (HIMC)0);
  3336. wd.im_position = Vector2();
  3337. // FIXME this is wrong in cases where the window coordinates were changed due to full screen mode; use WindowRect
  3338. wd.last_pos = p_rect.position;
  3339. wd.width = p_rect.size.width;
  3340. wd.height = p_rect.size.height;
  3341. window_id_counter++;
  3342. }
  3343. return id;
  3344. }
  3345. // WinTab API.
  3346. bool DisplayServerWindows::wintab_available = false;
  3347. WTOpenPtr DisplayServerWindows::wintab_WTOpen = nullptr;
  3348. WTClosePtr DisplayServerWindows::wintab_WTClose = nullptr;
  3349. WTInfoPtr DisplayServerWindows::wintab_WTInfo = nullptr;
  3350. WTPacketPtr DisplayServerWindows::wintab_WTPacket = nullptr;
  3351. WTEnablePtr DisplayServerWindows::wintab_WTEnable = nullptr;
  3352. // UXTheme API.
  3353. bool DisplayServerWindows::dark_title_available = false;
  3354. bool DisplayServerWindows::ux_theme_available = false;
  3355. ShouldAppsUseDarkModePtr DisplayServerWindows::ShouldAppsUseDarkMode = nullptr;
  3356. GetImmersiveColorFromColorSetExPtr DisplayServerWindows::GetImmersiveColorFromColorSetEx = nullptr;
  3357. GetImmersiveColorTypeFromNamePtr DisplayServerWindows::GetImmersiveColorTypeFromName = nullptr;
  3358. GetImmersiveUserColorSetPreferencePtr DisplayServerWindows::GetImmersiveUserColorSetPreference = nullptr;
  3359. // Windows Ink API.
  3360. bool DisplayServerWindows::winink_available = false;
  3361. GetPointerTypePtr DisplayServerWindows::win8p_GetPointerType = nullptr;
  3362. GetPointerPenInfoPtr DisplayServerWindows::win8p_GetPointerPenInfo = nullptr;
  3363. typedef enum _SHC_PROCESS_DPI_AWARENESS {
  3364. SHC_PROCESS_DPI_UNAWARE = 0,
  3365. SHC_PROCESS_SYSTEM_DPI_AWARE = 1,
  3366. SHC_PROCESS_PER_MONITOR_DPI_AWARE = 2
  3367. } SHC_PROCESS_DPI_AWARENESS;
  3368. bool DisplayServerWindows::is_dark_mode_supported() const {
  3369. return ux_theme_available;
  3370. }
  3371. bool DisplayServerWindows::is_dark_mode() const {
  3372. return ux_theme_available && ShouldAppsUseDarkMode();
  3373. }
  3374. Color DisplayServerWindows::get_accent_color() const {
  3375. if (!ux_theme_available) {
  3376. return Color(0, 0, 0, 0);
  3377. }
  3378. int argb = GetImmersiveColorFromColorSetEx((UINT)GetImmersiveUserColorSetPreference(false, false), GetImmersiveColorTypeFromName(L"ImmersiveSystemAccent"), false, 0);
  3379. return Color((argb & 0xFF) / 255.f, ((argb & 0xFF00) >> 8) / 255.f, ((argb & 0xFF0000) >> 16) / 255.f, ((argb & 0xFF000000) >> 24) / 255.f);
  3380. }
  3381. int DisplayServerWindows::tablet_get_driver_count() const {
  3382. return tablet_drivers.size();
  3383. }
  3384. String DisplayServerWindows::tablet_get_driver_name(int p_driver) const {
  3385. if (p_driver < 0 || p_driver >= tablet_drivers.size()) {
  3386. return "";
  3387. } else {
  3388. return tablet_drivers[p_driver];
  3389. }
  3390. }
  3391. String DisplayServerWindows::tablet_get_current_driver() const {
  3392. return tablet_driver;
  3393. }
  3394. void DisplayServerWindows::tablet_set_current_driver(const String &p_driver) {
  3395. if (tablet_get_driver_count() == 0) {
  3396. return;
  3397. }
  3398. bool found = false;
  3399. for (int i = 0; i < tablet_get_driver_count(); i++) {
  3400. if (p_driver == tablet_get_driver_name(i)) {
  3401. found = true;
  3402. }
  3403. }
  3404. if (found) {
  3405. _update_tablet_ctx(tablet_driver, p_driver);
  3406. tablet_driver = p_driver;
  3407. } else {
  3408. ERR_PRINT("Unknown tablet driver " + p_driver + ".");
  3409. }
  3410. }
  3411. DisplayServerWindows::DisplayServerWindows(const String &p_rendering_driver, WindowMode p_mode, VSyncMode p_vsync_mode, uint32_t p_flags, const Vector2i *p_position, const Vector2i &p_resolution, int p_screen, Error &r_error) {
  3412. KeyMappingWindows::initialize();
  3413. drop_events = false;
  3414. key_event_pos = 0;
  3415. alt_mem = false;
  3416. gr_mem = false;
  3417. shift_mem = false;
  3418. control_mem = false;
  3419. meta_mem = false;
  3420. hInstance = static_cast<OS_Windows *>(OS::get_singleton())->get_hinstance();
  3421. pressrc = 0;
  3422. old_invalid = true;
  3423. mouse_mode = MOUSE_MODE_VISIBLE;
  3424. rendering_driver = p_rendering_driver;
  3425. // Init TTS
  3426. tts = memnew(TTS_Windows);
  3427. // Enforce default keep screen on value.
  3428. screen_set_keep_on(GLOBAL_GET("display/window/energy_saving/keep_screen_on"));
  3429. // Load Windows version info.
  3430. OSVERSIONINFOW os_ver;
  3431. ZeroMemory(&os_ver, sizeof(OSVERSIONINFOW));
  3432. os_ver.dwOSVersionInfoSize = sizeof(OSVERSIONINFOW);
  3433. HMODULE nt_lib = LoadLibraryW(L"ntdll.dll");
  3434. if (nt_lib) {
  3435. RtlGetVersionPtr RtlGetVersion = (RtlGetVersionPtr)GetProcAddress(nt_lib, "RtlGetVersion");
  3436. if (RtlGetVersion) {
  3437. RtlGetVersion(&os_ver);
  3438. }
  3439. FreeLibrary(nt_lib);
  3440. }
  3441. // Load UXTheme.
  3442. HMODULE ux_theme_lib = LoadLibraryW(L"uxtheme.dll");
  3443. if (ux_theme_lib) {
  3444. ShouldAppsUseDarkMode = (ShouldAppsUseDarkModePtr)GetProcAddress(ux_theme_lib, MAKEINTRESOURCEA(132));
  3445. GetImmersiveColorFromColorSetEx = (GetImmersiveColorFromColorSetExPtr)GetProcAddress(ux_theme_lib, MAKEINTRESOURCEA(95));
  3446. GetImmersiveColorTypeFromName = (GetImmersiveColorTypeFromNamePtr)GetProcAddress(ux_theme_lib, MAKEINTRESOURCEA(96));
  3447. GetImmersiveUserColorSetPreference = (GetImmersiveUserColorSetPreferencePtr)GetProcAddress(ux_theme_lib, MAKEINTRESOURCEA(98));
  3448. ux_theme_available = ShouldAppsUseDarkMode && GetImmersiveColorFromColorSetEx && GetImmersiveColorTypeFromName && GetImmersiveUserColorSetPreference;
  3449. if (os_ver.dwBuildNumber >= 22000) {
  3450. dark_title_available = true;
  3451. }
  3452. }
  3453. // Note: Wacom WinTab driver API for pen input, for devices incompatible with Windows Ink.
  3454. HMODULE wintab_lib = LoadLibraryW(L"wintab32.dll");
  3455. if (wintab_lib) {
  3456. wintab_WTOpen = (WTOpenPtr)GetProcAddress(wintab_lib, "WTOpenW");
  3457. wintab_WTClose = (WTClosePtr)GetProcAddress(wintab_lib, "WTClose");
  3458. wintab_WTInfo = (WTInfoPtr)GetProcAddress(wintab_lib, "WTInfoW");
  3459. wintab_WTPacket = (WTPacketPtr)GetProcAddress(wintab_lib, "WTPacket");
  3460. wintab_WTEnable = (WTEnablePtr)GetProcAddress(wintab_lib, "WTEnable");
  3461. wintab_available = wintab_WTOpen && wintab_WTClose && wintab_WTInfo && wintab_WTPacket && wintab_WTEnable;
  3462. }
  3463. if (wintab_available) {
  3464. tablet_drivers.push_back("wintab");
  3465. }
  3466. // Note: Windows Ink API for pen input, available on Windows 8+ only.
  3467. HMODULE user32_lib = LoadLibraryW(L"user32.dll");
  3468. if (user32_lib) {
  3469. win8p_GetPointerType = (GetPointerTypePtr)GetProcAddress(user32_lib, "GetPointerType");
  3470. win8p_GetPointerPenInfo = (GetPointerPenInfoPtr)GetProcAddress(user32_lib, "GetPointerPenInfo");
  3471. winink_available = win8p_GetPointerType && win8p_GetPointerPenInfo;
  3472. }
  3473. if (winink_available) {
  3474. tablet_drivers.push_back("winink");
  3475. }
  3476. if (OS::get_singleton()->is_hidpi_allowed()) {
  3477. HMODULE Shcore = LoadLibraryW(L"Shcore.dll");
  3478. if (Shcore != nullptr) {
  3479. typedef HRESULT(WINAPI * SetProcessDpiAwareness_t)(SHC_PROCESS_DPI_AWARENESS);
  3480. SetProcessDpiAwareness_t SetProcessDpiAwareness = (SetProcessDpiAwareness_t)GetProcAddress(Shcore, "SetProcessDpiAwareness");
  3481. if (SetProcessDpiAwareness) {
  3482. SetProcessDpiAwareness(SHC_PROCESS_SYSTEM_DPI_AWARE);
  3483. }
  3484. }
  3485. }
  3486. memset(&wc, 0, sizeof(WNDCLASSEXW));
  3487. wc.cbSize = sizeof(WNDCLASSEXW);
  3488. wc.style = CS_OWNDC | CS_DBLCLKS;
  3489. wc.lpfnWndProc = (WNDPROC)::WndProc;
  3490. wc.cbClsExtra = 0;
  3491. wc.cbWndExtra = 0;
  3492. wc.hInstance = hInstance ? hInstance : GetModuleHandle(nullptr);
  3493. wc.hIcon = LoadIcon(nullptr, IDI_WINLOGO);
  3494. wc.hCursor = nullptr;
  3495. wc.hbrBackground = nullptr;
  3496. wc.lpszMenuName = nullptr;
  3497. wc.lpszClassName = L"Engine";
  3498. if (!RegisterClassExW(&wc)) {
  3499. MessageBoxW(nullptr, L"Failed To Register The Window Class.", L"ERROR", MB_OK | MB_ICONEXCLAMATION);
  3500. r_error = ERR_UNAVAILABLE;
  3501. return;
  3502. }
  3503. _register_raw_input_devices(INVALID_WINDOW_ID);
  3504. #if defined(VULKAN_ENABLED)
  3505. if (rendering_driver == "vulkan") {
  3506. context_vulkan = memnew(VulkanContextWindows);
  3507. if (context_vulkan->initialize() != OK) {
  3508. memdelete(context_vulkan);
  3509. context_vulkan = nullptr;
  3510. r_error = ERR_UNAVAILABLE;
  3511. return;
  3512. }
  3513. }
  3514. #endif
  3515. // Init context and rendering device
  3516. #if defined(GLES3_ENABLED)
  3517. if (rendering_driver == "opengl3") {
  3518. GLManager_Windows::ContextType opengl_api_type = GLManager_Windows::GLES_3_0_COMPATIBLE;
  3519. gl_manager = memnew(GLManager_Windows(opengl_api_type));
  3520. if (gl_manager->initialize() != OK) {
  3521. memdelete(gl_manager);
  3522. gl_manager = nullptr;
  3523. r_error = ERR_UNAVAILABLE;
  3524. return;
  3525. }
  3526. RasterizerGLES3::make_current();
  3527. }
  3528. #endif
  3529. mouse_monitor = SetWindowsHookEx(WH_MOUSE, ::MouseProc, nullptr, GetCurrentThreadId());
  3530. Point2i window_position;
  3531. if (p_position != nullptr) {
  3532. window_position = *p_position;
  3533. } else {
  3534. if (p_screen == SCREEN_OF_MAIN_WINDOW) {
  3535. p_screen = SCREEN_PRIMARY;
  3536. }
  3537. window_position = screen_get_position(p_screen) + (screen_get_size(p_screen) - p_resolution) / 2;
  3538. }
  3539. WindowID main_window = _create_window(p_mode, p_vsync_mode, 0, Rect2i(window_position, p_resolution));
  3540. ERR_FAIL_COND_MSG(main_window == INVALID_WINDOW_ID, "Failed to create main window.");
  3541. joypad = new JoypadWindows(&windows[MAIN_WINDOW_ID].hWnd);
  3542. for (int i = 0; i < WINDOW_FLAG_MAX; i++) {
  3543. if (p_flags & (1 << i)) {
  3544. window_set_flag(WindowFlags(i), true, main_window);
  3545. }
  3546. }
  3547. show_window(MAIN_WINDOW_ID);
  3548. #if defined(VULKAN_ENABLED)
  3549. if (rendering_driver == "vulkan") {
  3550. rendering_device_vulkan = memnew(RenderingDeviceVulkan);
  3551. rendering_device_vulkan->initialize(context_vulkan);
  3552. RendererCompositorRD::make_current();
  3553. }
  3554. #endif
  3555. if (!Engine::get_singleton()->is_editor_hint() && !OS::get_singleton()->is_in_low_processor_usage_mode()) {
  3556. // Increase priority for projects that are not in low-processor mode (typically games)
  3557. // to reduce the risk of frame stuttering.
  3558. // This is not done for the editor to prevent importers or resource bakers
  3559. // from making the system unresponsive.
  3560. SetPriorityClass(GetCurrentProcess(), ABOVE_NORMAL_PRIORITY_CLASS);
  3561. DWORD index = 0;
  3562. HANDLE handle = AvSetMmThreadCharacteristicsW(L"Games", &index);
  3563. if (handle) {
  3564. AvSetMmThreadPriority(handle, AVRT_PRIORITY_CRITICAL);
  3565. }
  3566. // This is needed to make sure that background work does not starve the main thread.
  3567. // This is only setting the priority of this thread, not the whole process.
  3568. SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_TIME_CRITICAL);
  3569. }
  3570. cursor_shape = CURSOR_ARROW;
  3571. _update_real_mouse_position(MAIN_WINDOW_ID);
  3572. r_error = OK;
  3573. static_cast<OS_Windows *>(OS::get_singleton())->set_main_window(windows[MAIN_WINDOW_ID].hWnd);
  3574. Input::get_singleton()->set_event_dispatch_function(_dispatch_input_events);
  3575. }
  3576. Vector<String> DisplayServerWindows::get_rendering_drivers_func() {
  3577. Vector<String> drivers;
  3578. #ifdef VULKAN_ENABLED
  3579. drivers.push_back("vulkan");
  3580. #endif
  3581. #ifdef GLES3_ENABLED
  3582. drivers.push_back("opengl3");
  3583. #endif
  3584. return drivers;
  3585. }
  3586. DisplayServer *DisplayServerWindows::create_func(const String &p_rendering_driver, WindowMode p_mode, VSyncMode p_vsync_mode, uint32_t p_flags, const Vector2i *p_position, const Vector2i &p_resolution, int p_screen, Error &r_error) {
  3587. DisplayServer *ds = memnew(DisplayServerWindows(p_rendering_driver, p_mode, p_vsync_mode, p_flags, p_position, p_resolution, p_screen, r_error));
  3588. if (r_error != OK) {
  3589. if (p_rendering_driver == "vulkan") {
  3590. String executable_name = OS::get_singleton()->get_executable_path().get_file();
  3591. OS::get_singleton()->alert(
  3592. vformat("Your video card drivers seem not to support the required Vulkan version.\n\n"
  3593. "If possible, consider updating your video card drivers or using the OpenGL 3 driver.\n\n"
  3594. "You can enable the OpenGL 3 driver by starting the engine from the\n"
  3595. "command line with the command:\n'%s --rendering-driver opengl3'\n\n"
  3596. "If you have recently updated your video card drivers, try rebooting.",
  3597. executable_name),
  3598. "Unable to initialize Vulkan video driver");
  3599. } else {
  3600. OS::get_singleton()->alert(
  3601. "Your video card drivers seem not to support the required OpenGL 3.3 version.\n\n"
  3602. "If possible, consider updating your video card drivers.\n\n"
  3603. "If you have recently updated your video card drivers, try rebooting.",
  3604. "Unable to initialize OpenGL video driver");
  3605. }
  3606. }
  3607. return ds;
  3608. }
  3609. void DisplayServerWindows::register_windows_driver() {
  3610. register_create_function("windows", create_func, get_rendering_drivers_func);
  3611. }
  3612. DisplayServerWindows::~DisplayServerWindows() {
  3613. delete joypad;
  3614. touch_state.clear();
  3615. cursors_cache.clear();
  3616. if (mouse_monitor) {
  3617. UnhookWindowsHookEx(mouse_monitor);
  3618. }
  3619. if (user_proc) {
  3620. SetWindowLongPtr(windows[MAIN_WINDOW_ID].hWnd, GWLP_WNDPROC, (LONG_PTR)user_proc);
  3621. }
  3622. // Close power request handle.
  3623. screen_set_keep_on(false);
  3624. #ifdef GLES3_ENABLED
  3625. // destroy windows .. NYI?
  3626. // FIXME wglDeleteContext is never called
  3627. #endif
  3628. if (windows.has(MAIN_WINDOW_ID)) {
  3629. #ifdef VULKAN_ENABLED
  3630. if (context_vulkan) {
  3631. context_vulkan->window_destroy(MAIN_WINDOW_ID);
  3632. }
  3633. #endif
  3634. if (wintab_available && windows[MAIN_WINDOW_ID].wtctx) {
  3635. wintab_WTClose(windows[MAIN_WINDOW_ID].wtctx);
  3636. windows[MAIN_WINDOW_ID].wtctx = 0;
  3637. }
  3638. DestroyWindow(windows[MAIN_WINDOW_ID].hWnd);
  3639. }
  3640. #if defined(VULKAN_ENABLED)
  3641. if (rendering_device_vulkan) {
  3642. rendering_device_vulkan->finalize();
  3643. memdelete(rendering_device_vulkan);
  3644. rendering_device_vulkan = nullptr;
  3645. }
  3646. if (context_vulkan) {
  3647. memdelete(context_vulkan);
  3648. context_vulkan = nullptr;
  3649. }
  3650. #endif
  3651. if (restore_mouse_trails > 1) {
  3652. SystemParametersInfoA(SPI_SETMOUSETRAILS, restore_mouse_trails, 0, 0);
  3653. }
  3654. #ifdef GLES3_ENABLED
  3655. if (gl_manager) {
  3656. memdelete(gl_manager);
  3657. gl_manager = nullptr;
  3658. }
  3659. #endif
  3660. if (tts) {
  3661. memdelete(tts);
  3662. }
  3663. CoUninitialize();
  3664. }