Source: ui/widget/chat-log.js

  1. import { TextArea } from './text-area.js';
  2. import { TextAreaInput } from './text-area-input.js';
  3. import { Label } from './label.js';
  4. import { RemoteBrowser } from './remote-browser.js';
  5. import { VRSpaceAPI } from './../../client/rest-api.js'
  6. import { VRSPACEUI } from '../vrspace-ui.js';
  7. import { ServerCapabilities } from '../../client/openapi/model/ServerCapabilities.js';
  8. class ChatLogInput extends TextAreaInput {
  9. inputFocused(input, focused) {
  10. super.inputFocused(input,focused);
  11. if ( focused ) {
  12. console.log("Focused ", this.textArea);
  13. ChatLog.activeInstance = this.textArea;
  14. }
  15. }
  16. }
  17. class Link {
  18. constructor( text, enterWorld=false ) {
  19. this.url = text;
  20. this.enterWorld = enterWorld;
  21. let pos = text.indexOf("://");
  22. if ( pos > 0 ) {
  23. text = text.substring(pos+3);
  24. } else {
  25. this.url = "https://"+this.url;
  26. }
  27. pos = text.indexOf("/");
  28. if ( pos > 0 ) {
  29. this.site = text.substring(0,pos);
  30. } else {
  31. this.site = text;
  32. }
  33. console.log('new link: '+this.url+" "+this.site);
  34. this.label = null;
  35. this.buttons = [];
  36. }
  37. openHere(local) {
  38. if ( local ) {
  39. console.log("TODO: Enter the world like AvatarSelection.enterWorld does");
  40. }
  41. window.location.href = this.url;
  42. }
  43. openTab() {
  44. window.open(this.url, '_blank').focus();
  45. }
  46. dispose() {
  47. this.label.dispose();
  48. this.buttons.forEach(b=>b.dispose());
  49. }
  50. }
  51. class LinkStack {
  52. /** @type {ServerCapabilities} */
  53. static serverCapablities = null;
  54. constructor(scene, parent, position, scaling = new BABYLON.Vector3(.02,.02,.02)) {
  55. this.scene = scene;
  56. this.parent = parent;
  57. this.scaling = scaling;
  58. this.position = position;
  59. this.capacity = 5;
  60. this.links = [];
  61. this.meshes = []; // XR selectables
  62. this.clickHandler = this.scene.onPointerObservable.add((pointerInfo) => {
  63. if ( pointerInfo.type == BABYLON.PointerEventTypes.POINTERDOWN
  64. && pointerInfo.pickInfo.hit
  65. ) {
  66. for ( let i = 0; i < this.links.length; i++ ) {
  67. if ( this.links[i].label.textPlane == pointerInfo.pickInfo.pickedMesh ) {
  68. this.clicked(this.links[i]);
  69. break;
  70. }
  71. }
  72. }
  73. });
  74. if ( LinkStack.serverCapablities == null ) {
  75. VRSpaceAPI.getInstance().endpoint.server.getServerCapabilities().then(c=>LinkStack.serverCapablities=c);
  76. }
  77. }
  78. addLink(word, enterWorld, local){
  79. let link = new Link(word, enterWorld);
  80. this.scroll();
  81. // add buttons to open in new tab, this tab, optionally internal browser
  82. if ( enterWorld ) {
  83. this.addButton(link, "enter", () => link.openHere(local));
  84. } else {
  85. this.addButton(link, "external-link", () => link.openTab());
  86. if (LinkStack.serverCapablities.remoteBrowser) {
  87. this.addButton(link, "play", () => this.openBrowser(link.url));
  88. }
  89. }
  90. let x = this.scaling.x*link.buttons.length+this.position.x+link.site.length/(Label.fontRatio*2)*this.scaling.x;
  91. let pos = new BABYLON.Vector3(x,this.position.y,this.position.z);
  92. let label = new Label(link.site,pos,this.parent);
  93. //label.background = "black";
  94. label.display();
  95. label.textPlane.scaling = this.scaling;
  96. link.label = label;
  97. this.links.push(link);
  98. this.meshes.push(label.textPlane);
  99. return link;
  100. }
  101. addButton(link, name, callback) {
  102. let button = BABYLON.GUI.Button.CreateImageOnlyButton(name+"-"+link.site, VRSPACEUI.contentBase+"/content/icons/"+name+".png");
  103. let buttonPlane = BABYLON.MeshBuilder.CreatePlane(name+"-"+link.site, {height:1,width:1});
  104. buttonPlane.parent = this.parent;
  105. buttonPlane.position = new BABYLON.Vector3(this.position.x + this.scaling.x*link.buttons.length,this.position.y,this.position.z);
  106. buttonPlane.scaling = this.scaling;
  107. let buttonTexture = BABYLON.GUI.AdvancedDynamicTexture.CreateForMesh(
  108. buttonPlane,
  109. 128,
  110. 128,
  111. false // mouse events disabled
  112. );
  113. buttonTexture.addControl(button);
  114. button.onPointerDownObservable.add( () => callback());
  115. link.buttons.push(buttonPlane);
  116. this.meshes.push(buttonPlane);
  117. }
  118. /** @param {Link} link */
  119. async clicked(link) {
  120. // process invitations
  121. console.log("Clicked "+link.url);
  122. if ( link.enterWorld ) {
  123. link.openHere();
  124. } else if (LinkStack.serverCapablities.remoteBrowser) {
  125. this.openBrowser(link.url);
  126. } else {
  127. link.openTab();
  128. }
  129. }
  130. openBrowser(url) {
  131. if ( this.browser ) {
  132. this.browser.dispose();
  133. }
  134. this.browser = new RemoteBrowser(this.scene);
  135. this.browser.show();
  136. //this.browser.attachToCamera();
  137. this.browser.attachToHud();
  138. if ( url.toLowerCase().endsWith(".jpg") || url.toLowerCase().endsWith(".jpg") ) {
  139. this.browser.loadUrl(url);
  140. } else {
  141. this.browser.get(url);
  142. }
  143. }
  144. scroll() {
  145. if ( this.links.length == this.capacity ) {
  146. this.links[0].dispose();
  147. let link = this.links.splice(0,1)[0];
  148. link.dispose();
  149. }
  150. for ( let i = 0; i < this.links.length; i++ ) {
  151. let label = this.links[i].label;
  152. let y = label.textPlane.position.y + label.textPlane.scaling.y*1.5;
  153. label.textPlane.position = new BABYLON.Vector3( label.textPlane.position.x, y, label.textPlane.position.z );
  154. this.links[i].buttons.forEach(b=>b.position.y = y );
  155. }
  156. }
  157. dispose() {
  158. this.scene.onPointerObservable.remove(this.clickHandler);
  159. this.links.forEach(l=>l.dispose());
  160. this.meshes = [];
  161. if ( this.browser ) {
  162. this.browser.dispose();
  163. }
  164. }
  165. isSelectableMesh(mesh) {
  166. return this.meshes.includes(mesh);
  167. }
  168. }
  169. /**
  170. * Chat log with TextArea and TextAreaInput, attached by to HUD.
  171. * By default alligned to left side of the screen.
  172. */
  173. export class ChatLog extends TextArea {
  174. static instanceCount = 0;
  175. static instances = {}
  176. /** @type {ChatLog} */
  177. static activeInstance = null;
  178. static instanceId(name, title) {
  179. return name+"_"+title;
  180. }
  181. /** @return {ChatLog} */
  182. static findInstance(title, name="ChatLog") {
  183. if ( ChatLog.instances.hasOwnProperty(ChatLog.instanceId(name,title)) ) {
  184. return ChatLog.instances[ChatLog.instanceId(name,title)];
  185. }
  186. }
  187. static getInstance(scene, title, name="ChatLog") {
  188. if ( ChatLog.instances.hasOwnProperty(ChatLog.instanceId(name,title)) ) {
  189. return ChatLog.instances[ChatLog.instanceId(name,title)];
  190. }
  191. return new ChatLog(scene, title, name);
  192. }
  193. constructor(scene, title, name="ChatLog") {
  194. super(scene, name, title);
  195. if ( ChatLog.instances.hasOwnProperty(this.instanceId()) ) {
  196. throw "Instance already exists: "+this.instanceId();
  197. }
  198. this.input = new ChatLogInput(this, "Say", title);
  199. this.input.submitName = "send";
  200. this.input.showNoMatch = false;
  201. this.inputPrefix = "ME";
  202. this.showLinks = true;
  203. this.minimizeInput = false;
  204. this.minimizeTitle = true;
  205. this.autoHide = true;
  206. this.size = .3;
  207. this.baseAnchor = -.4;
  208. this.verticalAnchor = -.1;
  209. //this.baseAnchor = 0;
  210. this.anchor = this.baseAnchor;
  211. this.leftSide();
  212. this.linkStack = new LinkStack(this.scene, this.group, new BABYLON.Vector3(this.size/2*1.25,-this.size/2,0));
  213. this.listeners = [];
  214. ChatLog.instanceCount++;
  215. ChatLog.instances[this.instanceId()] = this;
  216. }
  217. instanceId() {
  218. return ChatLog.instanceId(this.name, this.titleText);
  219. }
  220. /**
  221. * Show both TextArea and TextAreaInput, and attach to HUD.
  222. */
  223. show() {
  224. super.show();
  225. this.setActiveInstance();
  226. this.input.inputPrefix = this.inputPrefix;
  227. this.input.addListener( text => this.notifyListeners(text) );
  228. // order matters: InputArea.init() shows the title, so call hide after
  229. this.input.init();
  230. if ( this.handles ) {
  231. if ( !this.minimizeInput ) {
  232. this.handles.dontMinimize.push(this.input.plane);
  233. }
  234. if ( this.title && !this.minimizeTitle ) {
  235. this.handles.dontMinimize.push(this.title.textPlane);
  236. // reposition the title
  237. this.handles.onMinMax = (minimized) => {
  238. if ( minimized ) {
  239. this.title.position.y = -1.2 * this.size/2 + this.title.height/2;
  240. this.clearActiveInstance();
  241. } else {
  242. // CHECKME: copied from TextArea.showTitle:
  243. this.title.position.y = 1.2 * this.size/2 + this.title.height/2;
  244. this.setActiveInstance();
  245. }
  246. };
  247. } else {
  248. this.handles.onMinMax = (minimized) => {
  249. if ( minimized ) {
  250. this.clearActiveInstance();
  251. } else {
  252. this.setActiveInstance();
  253. }
  254. };
  255. }
  256. }
  257. this.hide(this.autoHide);
  258. this.attachToHud();
  259. this.handleResize();
  260. this.resizeHandler = () => this.handleResize();
  261. window.addEventListener("resize", this.resizeHandler);
  262. }
  263. /**
  264. * Log something written by someone.
  265. * @param {String} who who wrote that
  266. * @param {String} what what they wrote
  267. * @param {String} link optional url to open
  268. */
  269. log( who, what, link, local ) {
  270. this.input.write(what,who);
  271. if ( link ) {
  272. this.showLink(link, true);
  273. }
  274. }
  275. attachToHud(){
  276. super.attachToHud();
  277. }
  278. /**
  279. * Move to left side of the screen
  280. */
  281. leftSide() {
  282. this.anchor = - Math.abs(this.anchor);
  283. this.moveToAnchor();
  284. }
  285. /**
  286. * Move to right side of the screen
  287. */
  288. rightSide() {
  289. this.anchor = Math.abs(this.anchor);
  290. this.moveToAnchor();
  291. }
  292. /**
  293. * Move either left or right, whatever is the current anchor
  294. */
  295. moveToAnchor() {
  296. //this.position = new BABYLON.Vector3(this.anchor, this.size/2-.025, 0);
  297. this.position = new BABYLON.Vector3(this.anchor, this.size/2+this.verticalAnchor, 0.2);
  298. this.group.position = this.position;
  299. }
  300. /**
  301. * Handle window resize, recalculates the current anchor and positions appropriatelly.
  302. */
  303. handleResize() {
  304. let aspectRatio = this.scene.getEngine().getAspectRatio(this.scene.activeCamera);
  305. // 0.67 -> anchor 0.1 (e.g. smartphone vertical)
  306. // 2 -> anchor 0.4 (pc, smartphone horizontal)
  307. let diff = (aspectRatio-0.67)/1.33;
  308. //this.anchor = -this.baseAnchor * diff * Math.sign(this.anchor);
  309. this.anchor = this.baseAnchor * diff;
  310. //console.log("Aspect ratio: "+aspectRatio+" anchor "+Math.sign(this.anchor)+" "+this.anchor+" base "+this.baseAnchor+" diff "+diff);
  311. this.moveToAnchor();
  312. }
  313. hasLink(line) {
  314. // TODO improve link detection
  315. return line.indexOf("://") > -1 || line.indexOf('www.') > -1 ;
  316. }
  317. processLinks(line) {
  318. if ( this.showLinks && typeof(line) === "string" && this.hasLink(line)) {
  319. line.split(' ').forEach((word)=>{
  320. if ( this.hasLink(word) ) {
  321. this.showLink(word);
  322. }
  323. });
  324. }
  325. }
  326. showLink(word, enterWorld, local) {
  327. console.log("Link found: "+word);
  328. this.linkStack.addLink(word, enterWorld);
  329. }
  330. write(string) {
  331. this.processLinks(string);
  332. super.write(string);
  333. this.hide(false);
  334. }
  335. setActiveInstance() {
  336. ChatLog.activeInstance = this;
  337. console.log("Focused ", this);
  338. }
  339. clearActiveInstance() {
  340. if ( ChatLog.activeInstance == this ) {
  341. console.log("Focus removed from ", this);
  342. ChatLog.activeInstance = null;
  343. }
  344. }
  345. notifyListeners(text,link) {
  346. this.listeners.forEach(l=>l(text, link));
  347. }
  348. /**
  349. * Add a listener to be called when input text is changed
  350. */
  351. addListener(listener) {
  352. this.listeners.push(listener);
  353. }
  354. /** Remove a listener */
  355. removeListener(listener) {
  356. let pos = this.listeners.indexOf(listener);
  357. if ( pos > -1 ) {
  358. this.listeners.splice(pos,1);
  359. }
  360. }
  361. /** Clean up */
  362. dispose() {
  363. window.removeEventListener("resize", this.resizeHandler);
  364. this.input.dispose();
  365. super.dispose();
  366. this.linkStack.dispose();
  367. this.clearActiveInstance();
  368. delete ChatLog.instances[this.instanceId()];
  369. ChatLog.instanceCount--;
  370. }
  371. /** XR pointer selection support */
  372. isSelectableMesh(mesh) {
  373. return super.isSelectableMesh(mesh) || this.input.isSelectableMesh(mesh) || this.linkStack.isSelectableMesh(mesh);
  374. }
  375. }