webvr.xml 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942
  1. <krpano>
  2. <!--
  3. webvr.xml
  4. krpano 1.21
  5. https://krpano.com/plugins/xmlextensions/#webvr
  6. https://krpano.com/plugins/webvr/
  7. -->
  8. <!-- load the WebVR plugin and assign it to a global 'webvr' variable -->
  9. <plugin name="webvr" devices="html5" keep="true"
  10. url="webvr.js"
  11. mobilevr_support="true"
  12. mobilevr_touch_support="true"
  13. mobilevr_fake_support="true"
  14. mobilevr_profile.normal="90|60|42|0|0|0"
  15. mobilevr_profile.mobile="80|60|42|35|0.441|0.156"
  16. mobilevr_wakelock="true"
  17. fullscreen_mirroring="false"
  18. mouse_pointerlock="true"
  19. vr_cursor_onover="if(handcursor, tween(hotspot[vr_cursor].scale,0.4,0.1); vr_auto_click(get(vr_timeout)); );"
  20. vr_cursor_onout="tween(hotspot[vr_cursor].scale,0.3,0.1);"
  21. onavailable="webvr_onavailable();"
  22. onunavailable=""
  23. onunknowndevice="webvr_onunknowndevice();"
  24. onvrcontrollers="webvr_onvrcontrollers();"
  25. onentervr="webvr_onentervr();"
  26. onexitvr="webvr_onexitvr();"
  27. ondenied="webvr_ondenied();"
  28. />
  29. <!-- the VR cursor hotspot -->
  30. <style name="vr_cursor_style"
  31. url="webvr_vrcursor.png"
  32. visible="false"
  33. enabled="false"
  34. distorted="true"
  35. crop="0|0|80|80"
  36. scale="0.3"
  37. depth="120"
  38. />
  39. <action name="webvr_load_vr_cursor_hs" scope="local">
  40. addhotspot('vr_cursor', hs);
  41. hs.loadstyle(vr_cursor_style);
  42. set(hs.keep, true);
  43. set(webvr.vr_cursor, 'hotspot[vr_cursor]');
  44. </action>
  45. <action name="webvr_load_vr_controller_hs" scope="private:VRCONTROLLERS" args="controllerstyle">
  46. removehotspot('vr_controller_l');
  47. removehotspot('vr_controller_r');
  48. addhotspot('vr_controller_l', vr_ctrl_l);
  49. addhotspot('vr_controller_r', vr_ctrl_r);
  50. set(vr_ctrl_l.keep, true);
  51. set(vr_ctrl_r.keep, true);
  52. vr_ctrl_l.loadstyle(calc(controllerstyle ? controllerstyle : (global.display.depthbuffer ? 'vrcontroller_laser' : 'vrcontroller_light_and_point')));
  53. vr_ctrl_r.loadstyle(calc(controllerstyle ? controllerstyle : (global.display.depthbuffer ? 'vrcontroller_laser' : 'vrcontroller_light_and_point')));
  54. <!-- optional: vibrate the controllers on hovering:
  55. vr_ctrl_l.addevent('onover', pulse(1.0, 0.25) );
  56. vr_ctrl_r.addevent('onover', pulse(1.0, 0.25) ); -->
  57. if(!global.webvr.iswebxr,
  58. vr_ctrl_l.addevent('onvrcontrollerbutton', 'if(vrbuttonindex == 3 AND vrbuttonstate == "up", vrsetup_open(); );' );
  59. vr_ctrl_r.addevent('onvrcontrollerbutton', 'if(vrbuttonindex == 3 AND vrbuttonstate == "up", vrsetup_open(); );' );
  60. );
  61. set(global.webvr.vr_controller, 'vr_controller_l,vr_controller_r');
  62. set(global.have_vr_controllers, true);
  63. if(global.webvr.isvrbrowser AND global.webvr.vrcontrollers[0].buttons.length LE 2,
  64. <!-- when there are only two buttons on the VR controller (e.g. Oculus Go) use an extra hotspot for the VR-setup -->
  65. addhotspot('webvr_vrsetup', vr_setup_hs);
  66. vr_setup_hs.loadstyle('webvr_button_style');
  67. set(vr_setup_hs, keep=true, ath=0, atv=90, depth=160, zorder=999, torigin=view, html='VR SETUP', oversampling=3, scale=0.2, onclick='vrsetup_open();', onloaded='renderloop( copy(ath,view.hlookat); );');
  68. );
  69. </action>
  70. <action name="webvr_onvrcontrollers" scope="private:VRCONTROLLERS">
  71. if(!global.have_vr_controllers, webvr_load_vr_controller_hs(); );
  72. </action>
  73. <style name="vrcontroller_laser"
  74. url="webvr_laser.png"
  75. distorted="true"
  76. enabled="false"
  77. visible="false"
  78. width="0.5" height="1000" edge="bottom" oref="1" oy="0"
  79. torigin="world" tx="0" ty="0" tz="0" depth="0"
  80. zorder="99999"
  81. depthbuffer="true"
  82. targethitd.number="0"
  83. onover="copy(targethitd, target.hitd);"
  84. onout="set(targethitd, 0);"
  85. onloaded="asyncloop(loaded, calc(height, (targethitd GT 0 ? targethitd : (hitd GT 0 ? hitd : 1000)) / display.hotspotworldscale); );"
  86. />
  87. <style name="vrcontroller_light_and_point"
  88. url="webvr_light.png"
  89. distorted="true"
  90. enabled="false"
  91. visible="false"
  92. width="0.5" height="18" edge="bottom" oref="1" oy="0"
  93. torigin="world" tx="0" ty="0" tz="0" depth="0"
  94. zorder="99999"
  95. depthbuffer="true"
  96. onloaded="vrcontroller_target_point();"
  97. />
  98. <style name="vrcontroller_handcursor_and_point"
  99. url="webvr_handcursor.png"
  100. distorted="true"
  101. enabled="false"
  102. visible="false"
  103. width="10" height="10" edge="center" oref="1" oy="-1"
  104. torigin="world" tx="0" ty="0" tz="0" depth="0"
  105. zorder="99999"
  106. depthbuffer="true"
  107. onloaded="vrcontroller_target_point();"
  108. />
  109. <action name="vrcontroller_target_point" scope="localonly">
  110. addhotspot(auto, hs);
  111. set(hs, keep=true, type=text, bgcolor=0xFFFFFF, bgalpha=1.0, width=10, height=10, bgroundedge=5, bgborder='1 0x000000 1.0', oversampling=2,
  112. scale=0.4, torigin=world, depth=0, distorted=false, zoom=true, zorder=99998, enabled=false
  113. );
  114. renderloop(
  115. if(!caller.loaded,
  116. removehotspot(get(hs.name));
  117. stoprenderloop();
  118. ,
  119. if(global.display.havedepthmap,
  120. <!-- use the laser for depthmap panos -->
  121. removehotspot(get(hs.name));
  122. stoprenderloop();
  123. webvr_load_vr_controller_hs('vrcontroller_laser');
  124. ,
  125. calc(hs.bgcolor, caller.pressed ? 0x049AFF : (caller.hovering ? 0x00FF00 : 0xFFFFFF));
  126. calc(distance, (caller.target AND caller.target.hitd GT 0 ? caller.target.hitd : (caller.hitd GT 0 ? caller.hitd : 1000)));
  127. calc(hs.scale, 0.4 * (distance GT 1000 ? distance / 1000 : (distance LT 200 ? 0.25 : (0.25 + (distance - 200)/800 * 0.75))));
  128. calc(hs.tx, caller.tx + caller.dx * distance);
  129. calc(hs.ty, caller.ty + caller.dy * distance);
  130. calc(hs.tz, caller.tz + caller.dz * distance);
  131. );
  132. );
  133. );
  134. </action>
  135. <!-- vr_auto_click() - call this action in the onover event of a
  136. hotspot to trigger automatically a click after some time. -->
  137. <action name="vr_auto_click" scope="local" args="vr_aclk_timeout">
  138. if(webvr.isenabled,
  139. if(vr_aclk_timeout == null, set(vr_aclk_timeout, 2000));
  140. copy(vr_aclk_t1, timertick);
  141. set(vr_aclk_waiting, true);
  142. copy(webvr.vr_aclk_hotspot, caller.name);
  143. set(hotspot[vr_cursor].crop,'0|0|80|80');
  144. asyncloop(vr_aclk_waiting AND webvr.vr_aclk_hotspot == caller.name,
  145. sub(dt, timertick, vr_aclk_t1);
  146. if(!caller.hovering,
  147. set(vr_aclk_waiting, false);
  148. set(hotspot[vr_cursor].crop,'0|0|80|80');
  149. ,
  150. div(f, dt, vr_aclk_timeout);
  151. mul(f, 16);
  152. roundval(f);
  153. Math.min(f, 16);
  154. mul(f, 80);
  155. txtadd(hotspot[vr_cursor].crop,get(f),'|0|80|80');
  156. <!-- wait another 100ms delay after finishing the animation before doing the click -->
  157. sub(dt, 100);
  158. if(dt GT vr_aclk_timeout,
  159. set(vr_aclk_waiting,false);
  160. set(hotspot[vr_cursor].crop,'0|0|80|80');
  161. <!-- call onclick -->
  162. callwith(caller, scope(global, ondown();onup();onclick(); ); );
  163. );
  164. );
  165. );
  166. );
  167. </action>
  168. <!-- by pressing SPACE the headset could be re-centered -->
  169. <events name="webvr_events" devices="html5" keep="true"
  170. onmousedown="if(webvr AND webvr.isenabled, webvr_showbuttons() );"
  171. />
  172. <!-- when WebVR support is available show an EnterVR button -->
  173. <action name="webvr_onavailable">
  174. webvr.loadsettings();
  175. if(layer[webvr_enterbutton], delayedcall(0.5, tween(layer[webvr_enterbutton].alpha,1.0); ); );
  176. </action>
  177. <action name="webvr_onunknowndevice">
  178. if(webvr.isfake AND device.desktop AND webvr.havesettings == false,
  179. <!-- set the 'no distortion' headset for fake desktop usage -->
  180. set(webvr.mobilevr_lens_overlap, 1.0);
  181. set(webvr.mobilevr_lens_fov, 96.0);
  182. set(webvr.mobilevr_lens_dist, 0.0);
  183. set(webvr.mobilevr_lens_dist2, '1|0|0|0');
  184. set(webvr.mobilevr_lens_ca, 0.0);
  185. set(webvr.mobilevr_lens_vign, 100);
  186. );
  187. </action>
  188. <action name="webvr_onentervr">
  189. if(layer[webvr_enterbutton], tween(layer[webvr_enterbutton].alpha,0,0); );
  190. webvr_showbuttons();
  191. webvr_hide_all_non_vr_layers();
  192. if(webvr.isfake, webvr_show_fakemode_info(true); );
  193. webvr_load_vr_cursor_hs();
  194. </action>
  195. <action name="webvr_onexitvr">
  196. removehotspot('vr_cursor');
  197. removehotspot('vr_controller_l');
  198. removehotspot('vr_controller_r');
  199. set(have_vr_controllers, false);
  200. stopdelayedcall(vr_button_fadeout);
  201. if(layer[webvr_enterbutton], tween(layer[webvr_enterbutton].alpha,1); );
  202. tween(layer[webvr_exitbutton].alpha,0);
  203. tween(layer[webvr_setupbutton].alpha,0);
  204. webvr_show_fakemode_info(false);
  205. webvr_restore_layers();
  206. </action>
  207. <action name="webvr_ondenied" scope="local">
  208. addlayer(webvr_ondenied_info);
  209. layer[webvr_ondenied_info].loadstyle(webvr_button_style);
  210. set(layer[webvr_ondenied_info],
  211. align='center',
  212. html='Entering VR mode was denied!',
  213. onclick='set(enabled,false); tween(alpha,0,0.5,default,removelayer(get(name)));'
  214. );
  215. delayedcall(2.0,
  216. callwith(layer[webvr_ondenied_info], onclick(); );
  217. );
  218. </action>
  219. <action name="webvr_hide_all_non_vr_layers" scope="local">
  220. for(set(i,0), i LT layer.count, inc(i),
  221. copy(lr, layer[get(i)]);
  222. if(lr.vr !== true,
  223. copy(lr.vr_backup_visible, lr.visible);
  224. set(lr.visible, false);
  225. );
  226. );
  227. </action>
  228. <action name="webvr_restore_layers" scope="local">
  229. for(set(i,0), i LT layer.count, inc(i),
  230. copy(lr, layer[get(i)]);
  231. if(lr.vr_backup_visible,
  232. copy(lr.visible, lr.vr_backup_visible);
  233. delete(lr.vr_backup_visible);
  234. );
  235. );
  236. </action>
  237. <action name="webvr_show_fakemode_info" scope="local" args="show">
  238. if(show == true,
  239. addlayer(webvr_fakemode_info);
  240. set(layer[webvr_fakemode_info],
  241. type='text',
  242. keep=true,
  243. align='bottom',
  244. y=80,
  245. bg=false,
  246. css='color:#FFFFFF;text-align:center;',
  247. html='[i][u]Simulated WebVR Mode![/u][br]For real WebVR with headset tracking use a [a href="http://webvr.info" target="_blank" style="color:#FFFFFF;"]WebVR-capable[/a] browser or a mobile device and a VR headset.[/i]'
  248. );
  249. ,
  250. removelayer(webvr_fakemode_info);
  251. );
  252. </action>
  253. <!-- ensure the same scaling on mobiles (regardless if mobilescale is 0.5 or 1.0) -->
  254. <krpano webvr_setup_scale="calc:(1.0 + 1.0*(device.mobile AND stagescale LT 1.0)) / (1.0 + 1.0*device.mobile)"
  255. webvr_button_scale.normal="1.0"
  256. webvr_button_scale.mobile="1.6"
  257. />
  258. <!-- the EnterVR/ExitVR and SetupVR buttons -->
  259. <style name="webvr_button_style"
  260. type="text"
  261. bgcolor="0x000000"
  262. bgalpha="0.5"
  263. bgroundedge="0"
  264. css="calc:'color:#FFFFFF;font-size:' + 20*webvr_setup_scale*webvr_button_scale + 'px;'"
  265. padding="calc:6*webvr_setup_scale*webvr_button_scale + ' ' + 10*webvr_setup_scale*webvr_button_scale"
  266. />
  267. <layer name="webvr_enterbutton" keep="true" vr="true"
  268. style="webvr_button_style"
  269. html="Enter VR"
  270. align="top" y="24"
  271. autoalpha="true" alpha="0.0"
  272. onclick="webvr.enterVR();"
  273. />
  274. <layer name="webvr_exitbutton" keep="true" vr="true"
  275. style="webvr_button_style"
  276. html="Exit VR"
  277. align="top" y="24"
  278. autoalpha="true" alpha="0.0"
  279. onclick="webvr.exitVR();"
  280. />
  281. <layer name="webvr_setupbutton" keep="true" vr="true"
  282. style="webvr_button_style"
  283. html="VR Setup"
  284. align="bottom" y="24"
  285. autoalpha="true" alpha="0.0"
  286. onclick="vrsetup_open();"
  287. />
  288. <action name="webvr_showbuttons">
  289. stopdelayedcall(vr_button_fadeout);
  290. tween(layer[webvr_exitbutton].alpha|layer[webvr_setupbutton].alpha, 1.0|1.0, 0.25);
  291. delayedcall(vr_button_fadeout, 3.0, tween(layer[webvr_exitbutton].alpha|layer[webvr_setupbutton].alpha, 0.0|0.0, 1.0); );
  292. </action>
  293. <!-- VR SETUP -->
  294. <mobilevr_presets>
  295. <headset name="cbv1" caption="Cardboard V1" profile="80|60|42|35|0.441|0.156" />
  296. <headset name="cbv2" caption="Cardboard V2" profile="120|64|39|35|0.34|0.55" />
  297. <headset name="dydm" caption="Daydream" profile="104|60|41|35|0.42|0.51" />
  298. <headset name="nodt" caption="No Distortion" profile="90|60|42|0|0|0" />
  299. </mobilevr_presets>
  300. <action name="vrsetup_open">
  301. if(!vrsetup_open_js, vrsetup_init(); );
  302. vrsetup_open_js();
  303. </action>
  304. <action name="vrsetup_init" type="Javascript"><![CDATA[
  305. var webvr = krpano.webvr;
  306. var padding = 20;
  307. function vrsetup_dlg_create(type)
  308. {
  309. var dlg = {type:type, bg:null, y:0, scale:1, elements:[]};
  310. if (type == 'layer')
  311. {
  312. dlg.bg = krpano.addlayer();
  313. dlg.bg.keep = true;
  314. dlg.bg.type = 'container';
  315. dlg.bg.align = 'center';
  316. }
  317. else // 'hotspot'
  318. {
  319. dlg.scale = 0.15;
  320. dlg.bg = krpano.addhotspot();
  321. dlg.bg.keep = true;
  322. dlg.bg.type = 'text';
  323. dlg.bg.distorted = true;
  324. dlg.bg.ath = krpano.view.hlookat;
  325. dlg.bg.atv = 0;
  326. dlg.bg.depth = 150;
  327. dlg.bg.torigin = 'world';
  328. dlg.bg.tx = krpano.view.tx;
  329. dlg.bg.ty = krpano.view.ty;
  330. dlg.bg.tz = krpano.view.tz;
  331. }
  332. dlg.bg.bgcolor = 0x000000;
  333. dlg.bg.bgalpha = 0.5;
  334. dlg.bg.bgcapture = true;
  335. dlg.bg.handcursor = false;
  336. dlg.bg.capture = false;
  337. dlg.bg.zorder = 100;
  338. dlg.bg.visible = false;
  339. dlg.y = 0;
  340. return dlg;
  341. }
  342. function vrsetup_dlg_addline(dlg, linetext, customcss, onclick)
  343. {
  344. var txt;
  345. if (dlg.type == 'layer')
  346. {
  347. txt = krpano.addlayer();
  348. txt.keep = true;
  349. txt.type = 'text';
  350. txt.align = 'center';
  351. txt.zorder = 101;
  352. }
  353. else // 'hotspot'
  354. {
  355. txt = krpano.addhotspot();
  356. txt.keep = true;
  357. txt.type = 'text';
  358. txt.distorted = true;
  359. txt.zorder = 101;
  360. txt.ath = dlg.bg.ath;
  361. txt.atv = dlg.bg.atv;
  362. txt.depth = dlg.bg.depth - 1;
  363. txt.oversampling = 2;
  364. txt.scale = dlg.scale;
  365. txt.torigin = dlg.bg.torigin;
  366. txt.tx = dlg.bg.tx;
  367. txt.ty = dlg.bg.ty;
  368. txt.tz = dlg.bg.tz;
  369. }
  370. txt.onautosized = function()
  371. {
  372. txt.havesize = true;
  373. }
  374. txt.edge = 'top';
  375. txt.visible = false;
  376. txt.bg = false;
  377. txt.html = "" + linetext;
  378. txt.css = 'font-size:32px;font-weight:bold;color:#FFFFFF; line-height:90%;' + (customcss || '');
  379. if (onclick)
  380. {
  381. txt.onclick = onclick;
  382. }
  383. else
  384. {
  385. txt.enabled = false;
  386. }
  387. dlg.elements.push( txt );
  388. return txt;
  389. }
  390. function vrsetup_dlg_addctrl(dlg, changedelay, callback)
  391. {
  392. var txt = vrsetup_dlg_addline(dlg, callback(0) );
  393. var inc;
  394. var dec;
  395. if(dlg.type == 'layer')
  396. {
  397. inc = krpano.addlayer();
  398. inc.keep = true;
  399. inc.type = 'text';
  400. inc.align = 'center';
  401. inc.zorder = 101;
  402. dec = krpano.addlayer();
  403. dec.keep = true;
  404. dec.type = 'text';
  405. dec.align = 'center';
  406. dec.zorder = 101;
  407. }
  408. else // 'hotspot'
  409. {
  410. inc = krpano.addhotspot();
  411. inc.keep = true;
  412. inc.type = 'text';
  413. inc.distorted = true;
  414. inc.zorder = 101;
  415. inc.ath = dlg.bg.ath;
  416. inc.atv = dlg.bg.atv;
  417. inc.depth = dlg.bg.depth - 1;
  418. inc.oversampling = 2;
  419. inc.scale = dlg.scale;
  420. inc.torigin = dlg.bg.torigin;
  421. inc.tx = dlg.bg.tx;
  422. inc.ty = dlg.bg.ty;
  423. inc.tz = dlg.bg.tz;
  424. dec = krpano.addhotspot();
  425. dec.keep = true;
  426. dec.type = 'text';
  427. dec.distorted = true;
  428. dec.zorder = 101;
  429. dec.ath = dlg.bg.ath;
  430. dec.atv = dlg.bg.atv;
  431. dec.depth = dlg.bg.depth - 1;
  432. dec.oversampling = 2;
  433. dec.scale = dlg.scale;
  434. dec.torigin = dlg.bg.torigin;
  435. dec.tx = dlg.bg.tx;
  436. dec.ty = dlg.bg.ty;
  437. dec.tz = dlg.bg.tz;
  438. }
  439. inc.edge = 'top';
  440. inc.visible = false;
  441. inc.bg = false;
  442. inc.html = '&#62;';
  443. inc.css = 'font-size:32px;font-weight:bold;color:#FFFFFF; line-height:90%;';
  444. inc.padding = '0 10';
  445. dec.edge = 'top';
  446. dec.visible = false;
  447. dec.bg = false;
  448. dec.html = '&#60;';
  449. dec.css = 'font-size:32px;font-weight:bold;color:#FFFFFF; line-height:90%;';
  450. dec.padding = '0 10';
  451. inc.vr_timeout = changedelay * 1000;
  452. dec.vr_timeout = changedelay * 1000;
  453. inc.ondown = function(){ txt.html = ""+callback(+1); inc.enabled = false; setTimeout(function(){ inc.enabled = true; },0); };
  454. dec.ondown = function(){ txt.html = ""+callback(-1); dec.enabled = false; setTimeout(function(){ dec.enabled = true; },0); };
  455. txt.ctrlchilds = [inc,dec];
  456. txt.updateControl = function()
  457. {
  458. txt.html = callback(0);
  459. }
  460. return txt;
  461. }
  462. function vrsetup_dlg_addspace(dlg, customspace)
  463. {
  464. dlg.elements.push(customspace|| padding);
  465. }
  466. function vrsetup_dlg_finish(dlg)
  467. {
  468. var i, w=0, h=0;
  469. var waitforsizes=false;
  470. for (i=0; i < dlg.elements.length; i++)
  471. {
  472. var e = dlg.elements[i];
  473. if ( isNaN(e) )
  474. {
  475. if ( e.havesize )
  476. {
  477. w = Math.max( w, e.width * 1);
  478. h += e.height * 1;
  479. }
  480. else
  481. {
  482. waitforsizes = true;
  483. break;
  484. }
  485. }
  486. else
  487. {
  488. h += e;
  489. }
  490. }
  491. if (waitforsizes)
  492. {
  493. setTimeout( function(){ vrsetup_dlg_finish(dlg); }, 16 );
  494. }
  495. else
  496. {
  497. var y = 0;
  498. dlg.bg.width = Math.ceil((w + padding*2)*dlg.scale);
  499. dlg.bg.height = Math.ceil((h + padding*2)*dlg.scale);
  500. for (i=0; i < dlg.elements.length; i++)
  501. {
  502. var e = dlg.elements[i];
  503. if ( isNaN(e) )
  504. {
  505. e.oy = Math.round((-h/2 + y) * dlg.scale);
  506. e.visible = true;
  507. y += e.height * 1;
  508. if (e.ctrlchilds )
  509. {
  510. for (var j=0; j < e.ctrlchilds.length; j++)
  511. {
  512. var sube = e.ctrlchilds[j];
  513. sube.ox = Math.round(((j&1)-0.5) * (-w) * dlg.scale);
  514. sube.oy = e.oy;
  515. sube.visible = true;
  516. }
  517. }
  518. }
  519. else
  520. {
  521. y += e;
  522. }
  523. }
  524. dlg.bg.visible = true;
  525. }
  526. }
  527. function vrsetup_dlg_remove(dlg)
  528. {
  529. var removefu = dlg.type == 'layer' ? krpano.removelayer : krpano.removehotspot;
  530. var i,j;
  531. for (i=0; i < dlg.elements.length; i++)
  532. {
  533. var e = dlg.elements[i];
  534. if ( isNaN(e) )
  535. {
  536. if (e.ctrlchilds )
  537. {
  538. for (j=0; j < e.ctrlchilds.length; j++)
  539. {
  540. var sube = e.ctrlchilds[j];
  541. removefu(sube.name);
  542. }
  543. }
  544. removefu(e.name);
  545. }
  546. }
  547. removefu(dlg.bg.name);
  548. }
  549. function vrsetup_webvr_dialog()
  550. {
  551. // WebVR API rendering
  552. var dlg = vrsetup_dlg_create('hotspot');
  553. vrsetup_dlg_addline(dlg, 'WebVR Setup');
  554. vrsetup_dlg_addspace(dlg);
  555. vrsetup_dlg_addline(dlg, 'Oversampling:', 'font-size:16px;')
  556. var ctrl_ss = vrsetup_dlg_addctrl(dlg, 1.0, function(change)
  557. {
  558. var p = webvr.oversampling;
  559. if (change < 0) { p = Math.max(0.2, Number(p) - 0.1); webvr.oversampling = p; } else
  560. if (change > 0) { p = Math.min(4.0, Number(p) + 0.1); webvr.oversampling = p; }
  561. krpano.actions.delayedcall(0.3, function()
  562. {
  563. renderres.html = webvr.renderwidth + "x" + webvr.renderheight;
  564. });
  565. return p.toFixed(1);
  566. });
  567. vrsetup_dlg_addspace(dlg,8);
  568. vrsetup_dlg_addline(dlg, 'Rendering Resolution:', 'font-size:16px;');
  569. var renderres = vrsetup_dlg_addline(dlg, '');
  570. vrsetup_dlg_addspace(dlg);
  571. vrsetup_dlg_addline(dlg, 'CLOSE', '', vrsetup_close);
  572. vrsetup_dlg_finish(dlg);
  573. return dlg;
  574. }
  575. function vrsetup_mobilevr_dialog()
  576. {
  577. // MobileVR / Cardboard rendering
  578. var dlg = vrsetup_dlg_create(0 ? 'hotspot' : 'layer');
  579. vrsetup_dlg_addline(dlg, 'MobileVR SETUP');
  580. vrsetup_dlg_addspace(dlg);
  581. vrsetup_dlg_addline(dlg, 'Screensize (inch):', 'font-size:16px;')
  582. vrsetup_dlg_addctrl(dlg, 1.0, function(change){ var ss = Number(webvr.mobilevr_screensize); if (isNaN(ss)) ss = 5.0; if (change < 0) { ss = Math.max(4.0, ss - 0.1); webvr.mobilevr_screensize = ss; } else if (change > 0) { ss = Math.min(10.0, ss + 0.1); webvr.mobilevr_screensize = ss; } return ss.toFixed(1); });
  583. vrsetup_dlg_addspace(dlg);
  584. vrsetup_dlg_addline(dlg, 'VR Headset Preset:', 'font-size:16px;')
  585. var ctrl_ps = vrsetup_dlg_addctrl(dlg, 1.0, function(change)
  586. {
  587. var preset_index = -1;
  588. var i;
  589. var profile = webvr.mobilevr_profile;
  590. var presets = krpano.get("mobilevr_presets.headset");
  591. if (presets)
  592. {
  593. presets = presets.getArray();
  594. for (i=0; i < presets.length; i++)
  595. {
  596. if ( profile == presets[i].profile )
  597. {
  598. preset_index = i;
  599. break;
  600. }
  601. }
  602. if (change < 0)
  603. {
  604. preset_index--;
  605. if (preset_index < 0)
  606. preset_index = presets.length - 1;
  607. webvr.mobilevr_profile = presets[preset_index].profile;
  608. }
  609. else if (change > 0)
  610. {
  611. preset_index++;
  612. if (preset_index >= presets.length)
  613. preset_index = 0;
  614. webvr.mobilevr_profile = presets[preset_index].profile;
  615. }
  616. }
  617. if (preset_index >= 0)
  618. {
  619. return presets[preset_index].caption;
  620. }
  621. return 'Custom';
  622. });
  623. vrsetup_dlg_addspace(dlg);
  624. vrsetup_dlg_addline(dlg, 'Customize Headset', 'font-size:25px;', function()
  625. {
  626. vrsetup_dlg_remove(dlg);
  627. dlg=null;
  628. vrsetup_dialog = vrsetup_mobilevr_interactive_dialog();
  629. });
  630. vrsetup_dlg_addline(dlg, '(Interactive Adjustment in VR)', 'font-size:12px;');
  631. vrsetup_dlg_addspace(dlg);
  632. vrsetup_dlg_addline(dlg, 'CLOSE', '', vrsetup_close);
  633. vrsetup_dlg_finish(dlg);
  634. return dlg;
  635. }
  636. function vrsetup_mobilevr_interactive_dialog()
  637. {
  638. // MobileVR / Cardboard rendering
  639. var ctrl_preset, ctrl_fov, ctrl_ild, ctrl_stl, ctrl_ttl, ctrl_k1, ctrl_k2, ctrl_os;
  640. var dlg = vrsetup_dlg_create('hotspot');
  641. vrsetup_dlg_addline(dlg, 'MobileVR SETUP');
  642. vrsetup_dlg_addspace(dlg);
  643. vrsetup_dlg_addline(dlg, 'Preset:', 'font-size:16px;')
  644. ctrl_preset = vrsetup_dlg_addctrl(dlg, 1.0, function(change)
  645. {
  646. var preset_index = -1;
  647. var i;
  648. var profile = webvr.mobilevr_profile;
  649. var presets = krpano.get("mobilevr_presets.headset");
  650. if (presets)
  651. {
  652. presets = presets.getArray();
  653. for (i=0; i < presets.length; i++)
  654. {
  655. if ( profile == presets[i].profile )
  656. {
  657. preset_index = i;
  658. break;
  659. }
  660. }
  661. if (change < 0)
  662. {
  663. preset_index--;
  664. if (preset_index < 0)
  665. preset_index = presets.length - 1;
  666. webvr.mobilevr_profile = presets[preset_index].profile;
  667. }
  668. else if (change > 0)
  669. {
  670. preset_index++;
  671. if (preset_index >= presets.length)
  672. preset_index = 0;
  673. webvr.mobilevr_profile = presets[preset_index].profile;
  674. }
  675. if (change != 0)
  676. {
  677. ctrl_fov.updateControl();
  678. ctrl_ild.updateControl();
  679. ctrl_stl.updateControl();
  680. if (ctrl_ttl) ctrl_ttl.updateControl();
  681. ctrl_k1.updateControl();
  682. ctrl_k2.updateControl();
  683. }
  684. }
  685. if (preset_index >= 0)
  686. {
  687. return presets[preset_index].caption;
  688. }
  689. return 'Custom';
  690. });
  691. vrsetup_dlg_addspace(dlg);
  692. vrsetup_dlg_addline(dlg, 'Lens-Field-of-View:', 'font-size:16px;');
  693. ctrl_fov = vrsetup_dlg_addctrl(dlg, 0.25, function(change){ var p = webvr.mobilevr_profile.split("|"); if (change < 0) { p[0] = Number(p[0]) - 1.0; webvr.mobilevr_profile = p.join("|"); } else if (change > 0) { p[0] = Number(p[0]) + 1.0; webvr.mobilevr_profile = p.join("|"); } if (change != 0) ctrl_preset.updateControl(); return Number(p[0]).toFixed(0); });
  694. vrsetup_dlg_addspace(dlg, 8);
  695. vrsetup_dlg_addline(dlg, 'Inter-Lens-Distance (mm):', 'font-size:16px;');
  696. ctrl_ild = vrsetup_dlg_addctrl(dlg, 0.25, function(change){ var p = webvr.mobilevr_profile.split("|"); if (change < 0) { p[1] = Number(p[1]) - 1.0; webvr.mobilevr_profile = p.join("|"); } else if (change > 0) { p[1] = Number(p[1]) + 1.0; webvr.mobilevr_profile = p.join("|"); } if (change != 0) ctrl_preset.updateControl(); return Number(p[1]).toFixed(0); });
  697. vrsetup_dlg_addspace(dlg, 8);
  698. vrsetup_dlg_addline(dlg, 'Screen-to-Lens-Distance (mm):', 'font-size:16px;');
  699. ctrl_stl = vrsetup_dlg_addctrl(dlg, 0.25, function(change){ var p = webvr.mobilevr_profile.split("|"); if (change < 0) { p[2] = Number(p[2]) - 1.0; webvr.mobilevr_profile = p.join("|"); } else if (change > 0) { p[2] = Number(p[2]) + 1.0; webvr.mobilevr_profile = p.join("|"); } if (change != 0) ctrl_preset.updateControl(); return Number(p[2]).toFixed(0); });
  700. vrsetup_dlg_addspace(dlg, 8);
  701. if (!webvr.isfake)
  702. {
  703. vrsetup_dlg_addline(dlg, 'Tray-to-Lens-Center-Distance: (mm):', 'font-size:16px;');
  704. ctrl_ttl = vrsetup_dlg_addctrl(dlg, 0.25, function(change){ var p = webvr.mobilevr_profile.split("|"); if (change < 0) { p[3] = Number(p[3]) - 1.0; webvr.mobilevr_profile = p.join("|"); } else if (change > 0) { p[3] = Number(p[3]) + 1.0; webvr.mobilevr_profile = p.join("|"); } if (change != 0) ctrl_preset.updateControl(); return Number(p[3]).toFixed(0); });
  705. vrsetup_dlg_addspace(dlg, 8);
  706. }
  707. vrsetup_dlg_addline(dlg, 'Lens-Distortion Coefficients:', 'font-size:16px;');
  708. ctrl_k1 = vrsetup_dlg_addctrl(dlg, 0.1, function(change){ var p = webvr.mobilevr_profile.split("|"); if (change < 0) { p[4] = Number(p[4]) - 0.01; webvr.mobilevr_profile = p.join("|"); } else if (change > 0) { p[4] = Number(p[4]) + 0.01; webvr.mobilevr_profile = p.join("|"); } if (change != 0) ctrl_preset.updateControl(); return Number(p[4]).toFixed(3); });
  709. ctrl_k2 = vrsetup_dlg_addctrl(dlg, 0.1, function(change){ var p = webvr.mobilevr_profile.split("|"); if (change < 0) { p[5] = Number(p[5]) - 0.01; webvr.mobilevr_profile = p.join("|"); } else if (change > 0) { p[5] = Number(p[5]) + 0.01; webvr.mobilevr_profile = p.join("|"); } if (change != 0) ctrl_preset.updateControl(); return Number(p[5]).toFixed(3); });
  710. vrsetup_dlg_addspace(dlg);
  711. vrsetup_dlg_addline(dlg, 'Oversampling:', 'font-size:16px;');
  712. ctrl_os = vrsetup_dlg_addctrl(dlg, 0.25, function(change){ var p = webvr.oversampling; if (change < 0) { p = Math.max(0.2, Number(p) - 0.1); webvr.oversampling = p; } else if (change > 0) { p = Math.min(4.0, Number(p) + 0.1); webvr.oversampling = p; } return p.toFixed(1); });
  713. vrsetup_dlg_addspace(dlg);
  714. vrsetup_dlg_addline(dlg, 'CLOSE', '', vrsetup_close);
  715. vrsetup_dlg_finish(dlg);
  716. return dlg;
  717. }
  718. var vrsetup_dialog = null;
  719. var vrsetup_events = null;
  720. function vrsetup_close()
  721. {
  722. if (vrsetup_dialog)
  723. {
  724. webvr.savesettings();
  725. vrsetup_dlg_remove(vrsetup_dialog);
  726. vrsetup_dialog = null;
  727. }
  728. if (vrsetup_events)
  729. {
  730. krpano.events.removeItem(vrsetup_events.name);
  731. vrsetup_events = null;
  732. }
  733. }
  734. krpano.vrsetup_close = function()
  735. {
  736. vrsetup_close();
  737. }
  738. krpano.vrsetup_open_js = function()
  739. {
  740. if (vrsetup_dialog != null)
  741. {
  742. // already open...
  743. return;
  744. }
  745. if (webvr.iswebvr)
  746. {
  747. vrsetup_dialog = vrsetup_webvr_dialog();
  748. }
  749. else
  750. {
  751. vrsetup_dialog = vrsetup_mobilevr_dialog();
  752. }
  753. vrsetup_events = krpano.events.createItem("auto");
  754. vrsetup_events.keep = true;
  755. vrsetup_events.webvr_onexitvr = function()
  756. {
  757. vrsetup_close();
  758. }
  759. }
  760. ]]></action>
  761. <!-- A list of devices and their screensize -->
  762. <mobilevr_device_database>
  763. <device name="iPhone 5/5S/SE" screen="320x568x2" size="4.0" bevel="3" />
  764. <device name="iPhone 6/6S/7/8" screen="375x667x2" size="4.7" />
  765. <device name="iPhone 6/6S/7/8 Plus" screen="414x736" size="5.5" />
  766. <device name="iPhone 6/6S/7/8 Plus (zoomed)" screen="375x667x3" size="5.5" />
  767. <device name="iPhone X/XS/11Pro" screen="375x812x3" size="5.85|5.33" />
  768. <device name="iPhone XS/11Pro Max" screen="414x896x3" size="6.46|5.95" />
  769. <device name="iPhone XR/11" screen="414x896x2" size="6.06|5.58" />
  770. <device name="iPhone 12/12Pro" screen="390x844" size="6.1" />
  771. <device name="iPhone 12Pro Max" screen="428x926" size="6.7" />
  772. <device name="Samsung S6" ua="sm-g930" size="5.1" />
  773. <device name="Samsung S6 Edge" ua="sm-g925" size="5.1" />
  774. <device name="Samsung S6 Edge Plus" ua="sm-g928" size="5.7" />
  775. <device name="Samsung S7" ua="sm-g930" size="5.1" />
  776. <device name="Samsung S7 Edge" ua="sm-g935" size="5.5" />
  777. <device name="Samsung S8" ua="sm-g950" size="5.8" />
  778. <device name="Samsung S8+" ua="sm-g955" size="6.2" />
  779. <device name="Samsung S9" ua="sm-g960" size="5.8" />
  780. <device name="Samsung S9+" ua="sm-g965" size="6.2" />
  781. <device name="Samsung S10e" ua="sm-g970" size="5.8" />
  782. <device name="Samsung S10" ua="sm-g973" size="6.1" />
  783. <device name="Samsung S10+" ua="sm-g975" size="6.4" />
  784. <device name="Samsung S10 5G" ua="sm-g977" size="6.7" />
  785. <device name="Samsung Note 7" ua="sm-n930" size="5.7" />
  786. <device name="Samsung Note 8" ua="sm-n950" size="6.3" />
  787. <device name="Samsung Note 9" ua="sm-n960" size="6.4" />
  788. <device name="Samsung Note 10" ua="sm-n970" size="6.3" />
  789. <device name="Samsung Note 10 5G" ua="sm-n971" size="6.3" />
  790. <device name="Samsung Note 10+" ua="sm-n975" size="6.8" />
  791. <device name="Samsung Note 10+ 5G" ua="sm-n976" size="6.8" />
  792. <device name="Huawei P10 Lite" ua="HUAWEIWAS-TL10" size="5.2" />
  793. <device name="Huawei P20 Lite" ua="HUAWEIANE-LX1" size="5.84" />
  794. </mobilevr_device_database>
  795. </krpano>