Source: fvsessions.js

  1. /// <reference path="/ss/st/fvterm/web/Scripts/FVTermParent.js" />
  2. /// <reference path="/ss/st/fvterm/web/Scripts/FVTermMsgApi.js" />
  3. (function ()
  4. {
  5. 'use strict';
  6. function log(text)
  7. {
  8. /* not active
  9. if (typeof (console) != 'undefined')
  10. console.log(text);
  11. */
  12. }
  13. /**
  14. * @fileOverview The FVSessions Module contains the FVSessions.Manager class function and the FVSessions.Session class
  15. * @author Inventu Corporation <support@inventu.com">
  16. * @version 1.0.20
  17. */
  18. var FVSMGR, FVSESS,
  19. enterKeys = 'EC123456789abcdefghijklmnoxyz';
  20. //trim polyfill
  21. if (!String.prototype.trim)
  22. {
  23. String.prototype.trim = function ()
  24. {
  25. return this.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, '');
  26. };
  27. }
  28. /**
  29. * @namespace FVSessions
  30. */
  31. if (!window.FVSessions)
  32. window.FVSessions = {};
  33. FVSessions.config = {
  34. fvBaseUrl: '/fvterm/',
  35. fvTermPage: 'scterm.html',
  36. fvMacrosMgr: 'macros/FlowMacros.aspx',
  37. fvProfileMgr: 'SCProfile.aspx',
  38. msgAPI:true
  39. };
  40. // #region Manager class
  41. /** The Manager Class is used to initialize the environment and host Session objects
  42. * In September 2023, the FVSessions.config setting msgAPI was added with a default value of true.
  43. * This now assumes that FVTermMsgApi.js instead of FVTermParent.js is the corresponding JS file
  44. * providing the iFrame API management. FVTermMsgApi.js utilizes inter-window messaging instead
  45. * of direct calls that require document.domain to be set in many environments.
  46. * If you are using FVSessions.js with the direct API module FVTermParent.js, you will need to set
  47. * FVSessions.config.msgAPI=false prior to creating a new connection.
  48. * @typedef { FVSessions.Manager } Manager
  49. */
  50. /** @typedef {Object} Connection
  51. * @description Defines how a Session connects when started--matches all documented properties for the FVTermParent ConnectFVTerm function
  52. * @property {string} application (optional--overrides the SessionControl property of same name if defined) Name of application defined in web.config for screen recognition
  53. * @property {string} hostName Host name defined for the Inventu service that this session connects to
  54. * @property {boolean} autoStart If true the session will automatically connect to the host at start time
  55. * @property {boolean} keepAliveOnClose If true, the session will not be terminated/closed by the browser logic if the user closes the browser at the desktop level
  56. * @property {string} userID Can be set prior to start session--this is the UserID to be associated with the session in the Inventu Service
  57. * @property {Session} fnSessClass Needed at constructor time--defines class for sessions, such as FVSessions.HODSession-default is FVSessions.Session
  58. */
  59. /** @typedef {Object} SessionControl
  60. * @description Keyed by SessionID -- included in the SessionDef object passed as the control parameter to the Manager Class Constructor
  61. * @property {function(Session, rows, screenName)} onnewscreen Event function called any time a new screen is received from the host
  62. * @property {function(Session)} onclose Event function called when the session is closed
  63. * @property {function(Session, enterKey)} onsend Event function called whenever the user enters a screen
  64. * @property {string} frameId Optional frameID that this session should be opened/started in--if not set the name is 'FVSWin_' + the session id
  65. * @property {Connection} connection Defines how the session will connect during the Manager.StartSession function
  66. */
  67. /** @typedef {Object} SessionDef
  68. * @property {string} application Default name of application defined in web.config for screen recognition -- can be overridden in session connection property
  69. * @property {function(Session, rows, screenName)} onnewscreen Default Event function called any time a new screen is received from the host--
  70. * can be overridden for each Session in the sessions SessionControl object
  71. * @property {function(Session)} onclose Default Event function called when the session is closed--
  72. * can be overridden for each Session in the sessions SessionControl object
  73. * @property {function(Session, enterKey)} onsend Default Event function called whenever the user enters a screen--
  74. * can be overridden for each Session in the sessions SessionControl object
  75. * @property {SessionControl} sessions One or more session definitions keyed by the sessionID
  76. */
  77. /** @constructor
  78. * @description Manager Class Constructor function--please ensure that the FVSessions.config.msgAPI is set
  79. * to correspond with the API JS file you are using (see FVSessions.Manager class documentation)
  80. * @param {Object} win The Window object for the environment this Manager will run in
  81. * @param {SessionDef} control Settings for the events and sessions, with objects keyed by sessionID key property for each session is connection
  82. * @param {function(Object)} fnReady Completion function called when ready--includes a reference to the sessions object, indexed by session ids
  83. */
  84. FVSessions.Manager = function (win, control, fnReady)
  85. {
  86. var key, me = this, session, sessionKey, sessionChecks = [],
  87. sessions = (control && control.sessions) ? control.sessions : null, bindings, sessObj,
  88. fnSessClass = (control.fnSessClass) ? control.fnSessClass : FVSessions.Session;
  89. this.sessions = {};
  90. if (control.application)
  91. this.application = control.application;
  92. this.win = win;
  93. this.doc = win.document;
  94. if (sessions)
  95. {
  96. for (key in sessions)
  97. {
  98. if (sessions.hasOwnProperty(key))
  99. {
  100. bindings = {};
  101. sessObj = sessions[key];
  102. if (control.onalert || sessObj.onalert)
  103. bindings.onalert = (sessObj.onalert) ? sessObj.onalert : control.onalert;
  104. if (control.onsend || sessObj.onsend)
  105. bindings.onsend = (sessObj.onsend) ? sessObj.onsend : control.onsend;
  106. if (control.onclose || sessObj.onclose)
  107. bindings.onclose = (sessObj.onclose) ? sessObj.onclose : control.onclose;
  108. if (control.onnewscreen || sessObj.onnewscreen)
  109. bindings.onnewscreen = (sessObj.onnewscreen) ? sessObj.onnewscreen : control.onnewscreen;
  110. this.sessions[key] = session = new fnSessClass(key, this, sessObj, bindings);
  111. if (session.sessionKey)
  112. sessionChecks.push(session.sessionKey);
  113. }
  114. }
  115. if (sessionChecks.length)
  116. {
  117. this.ServerCall(FVSessions.config.fvBaseUrl + FVSessions.config.fvMacrosMgr + '?action=verifySessions', sessionChecks, { method: 'POST' }, function (statuses)
  118. {
  119. var i, id;
  120. for (i = 0; i < statuses.length; i++)
  121. {
  122. session = me.FindSessionByKey(sessionChecks[i]);
  123. session.SetStatus(statuses[i]);
  124. }
  125. if (fnReady)
  126. fnReady(me.sessions);
  127. });
  128. }
  129. else
  130. {
  131. if (fnReady)
  132. window.setTimeout(function () { fnReady(me.sessions) }, 1);
  133. }
  134. }
  135. };
  136. FVSMGR = FVSessions.Manager.prototype;
  137. /** @memberof! FVSessions.Manager.prototype
  138. * @description Find a Session object based on a sessionKey
  139. * @param {String} sessionKey The SessionKey to use in the search
  140. * @returns {Session} Found session or null
  141. */
  142. FVSMGR.FindSessionByKey = function (sessionKey)
  143. {
  144. var key;
  145. for (key in this.sessions)
  146. {
  147. if (this.sessions[key].sessionKey == sessionKey)
  148. return this.sessions[key];
  149. }
  150. }
  151. /** @memberof! FVSessions.Manager.prototype
  152. * @description Resize all Sessions
  153. * @param {Integer} width New Width for each Session
  154. * @param {Integer} height New height for each sessions
  155. */
  156. FVSMGR.ResizeSessions = function (width, height)
  157. {
  158. var key, session;
  159. for (key in this.sessions)
  160. {
  161. if (!this.sessions.hasOwnProperty(key))
  162. continue;
  163. session = this.sessions[key];
  164. if (session.fvWin)
  165. {
  166. session.fvWin.style.width = width + 'px';
  167. session.fvWin.style.height = height + 'px';
  168. }
  169. }
  170. };
  171. /** @memberof! FVSessions.Manager.prototype
  172. * @description Start a Session
  173. * @param {String} id The ID of the session - must match one of the sessions defined in the constructor
  174. * @param {Object} styles (Optional) - If not null, an object with property names matching styles to attach to the new session's iFrame
  175. * @param {Object} parentElem Parent DOM element--if not included, the document.body element will be the parent of the new iFrame
  176. * @param {function(Session)} fnReady Completion function called when the session has been started
  177. */
  178. FVSMGR.StartSession = function (id, styles, parentElem, fnReady)
  179. {
  180. var key, me = this, session = this.sessions[id], parentDiv,
  181. fvWin = (session.fvWin) ? session.fvWin : this.doc.getElementById(session.frameId),
  182. url = FVSessions.config.fvBaseUrl + FVSessions.config.fvTermPage,
  183. connection = session.connection,
  184. fnConnect = (FVSessions.config.msgAPI) ? FVMsgApi.ConnectMsgFVTerm : ConnectFVTerm;
  185. if (!fvWin)
  186. {
  187. if (!parentElem)
  188. parentElem = this.doc.body;
  189. parentDiv = this.doc.createElement('div');
  190. parentDiv.style.display = 'inline';
  191. parentElem.appendChild(parentDiv);
  192. parentDiv.innerHTML = '<iframe id="' + session.frameId + '" />';
  193. fvWin = this.doc.getElementById(session.frameId);
  194. session.fvWin = fvWin;
  195. }
  196. /* If msgApi, turn fvWin into a proxy and save the real fvWin
  197. */
  198. if (styles)
  199. {
  200. for (key in styles)
  201. {
  202. if (styles.hasOwnProperty(key))
  203. {
  204. fvWin.style[key] = styles[key];
  205. }
  206. }
  207. }
  208. if (session.status != 'offline')
  209. {
  210. connection = { "sessionKey": session.sessionKey };
  211. if (session.connection.keepAliveOnClose)
  212. connection.keepAliveOnClose = session.connection.keepAliveOnClose;
  213. if (this.application)
  214. connection.application = this.application;
  215. }
  216. //fnConnect(fvWin.id, url, connection, async function (fvApi, sessionKey) need async?
  217. fnConnect(fvWin.id, url, connection, async function (fvApi, sessionKey)
  218. {
  219. if (fvApi)
  220. await session.BindNewSession(fvApi, sessionKey);
  221. else
  222. session.connectError = sessionKey;
  223. if (fnReady)
  224. fnReady(session);
  225. });
  226. };
  227. /** @memberof! FVSessions.Manager.prototype
  228. * @description Call a server function that supports the session manager
  229. * @param {String} url Action portion of the url
  230. * @param {Object} data (Optional) - For a post, the object that will be sent as JSON stringified null value is ignored
  231. * @param {Object} options (optional) properties include method (GET/POST), mimeType, credUser and credPW all optional--null is ignored
  232. * @param {function(Object)} fnReady Completion function called with returned object converted from JSON
  233. * @returns {FVSMGR.ServerCall} object
  234. */
  235. FVSMGR.ServerCall = function (url, data, options, fnReady)
  236. {
  237. var xmlHttp = null,
  238. me = this, key,
  239. err, method = (options && options.method) ? options.method : 'GET',
  240. mimeType = (options && options.mimeType) ? options.mimeType : 'application/json',
  241. credUser = (options && options.credUser) ? options.credUser : null,
  242. credPW = (options && options.credPW) ? options.credPW : null,
  243. query = [], postData = null;
  244. try
  245. {
  246. if (window.XMLHttpRequest)
  247. {
  248. // If IE7, Mozilla, Safari, etc: Use native object
  249. xmlHttp = new XMLHttpRequest();
  250. if (xmlHttp.overrideMimeType)
  251. xmlHttp.overrideMimeType(mimeType);
  252. }
  253. else if (window.ActiveXObject)
  254. {
  255. // ...otherwise, use the ActiveX control for IE5.x and IE6
  256. xmlHttp = new ActiveXObject('MSXML2.XMLHTTP.3.0');
  257. }
  258. if (data)
  259. {
  260. if ((method == 'GET') ||
  261. (mimeType != 'application/json'))
  262. {
  263. for (key in data)
  264. {
  265. if (data.hasOwnProperty(key))
  266. query.push(encodeURIComponent(key) + '=' + encodeURIComponent(data[key]));
  267. }
  268. }
  269. if (method == 'GET')
  270. {
  271. if (url.indexOf('?') == -1)
  272. url += '?';
  273. url += query.join('&');
  274. }
  275. else
  276. {
  277. if (mimeType == 'application/json')
  278. postData = JSON.stringify(data);
  279. else
  280. postData = query.join('&');
  281. }
  282. }
  283. xmlHttp.open(method, url, true, credUser, credPW);
  284. xmlHttp.onreadystatechange = function ()
  285. {
  286. if (xmlHttp.readyState >= 4)
  287. {
  288. var result;
  289. if (xmlHttp.status == 200)
  290. {
  291. if (mimeType == 'application/json')
  292. result = JSON.parse(xmlHttp.responseText);
  293. else
  294. result = xmlHttp.responseText;
  295. me.httpError = false;
  296. }
  297. else
  298. {
  299. result = { status: 'error', code: xmlHttp.status };
  300. me.httpError = true;
  301. }
  302. if (fnReady)
  303. {
  304. fnReady(result);
  305. }
  306. }
  307. };
  308. if (method == 'POST')
  309. xmlHttp.setRequestHeader('Content-type', 'application/json; charset=utf-8');
  310. xmlHttp.send(postData);
  311. }
  312. catch (e)
  313. {
  314. alert(e.message);
  315. return null;
  316. }
  317. return xmlHttp;
  318. };
  319. // #endregion Manager Class
  320. // #region Session Class
  321. /** The Session class manages an individual FVTerm session and is created using the Manager
  322. * @typedef {FVSessions.Session} Session
  323. */
  324. /** @description Session Class Constructor function
  325. * @constructor
  326. * @param {String} id The id for this session, used for access and naming
  327. * @param {Object} mgr The owning FVSessions.Manager object
  328. * @param {Object} settings The source properties including connection which has connection details
  329. * @param {Object} bindings Any event/call forwarders
  330. */
  331. FVSessions.Session = function (id, mgr, settings, bindings)
  332. {
  333. if (!id)
  334. return;
  335. this.id = id;
  336. this.mgr = mgr;
  337. this.settings = settings;
  338. this.connection = settings.connection;
  339. if (!this.connection.application && mgr.application)
  340. this.connection.application = mgr.application;
  341. this.status = 'offline';
  342. this.frameId = (settings.frameId) ? settings.frameId : 'FVSWin_' + id;
  343. this.storeId = 'FVSK_' + id;
  344. this.sessionKey = this.GetSessionKey();
  345. this.bindings = bindings;
  346. if (FVSessions.config.msgAPI)
  347. FVSessions.SetMsgMethods();
  348. else
  349. FVSessions.SetLocalMethods();
  350. };
  351. FVSESS = FVSessions.Session.prototype;
  352. /** @memberof! FVSessions.Session.prototype
  353. * @description Internal function that binds the FVTermParent events to the client's onnewscreen,
  354. * onclose, onsend and onalert (when each handler is present) in support of the FVTermMsgApi.js module
  355. * implemented September of 2023, this function is flagged async as the fvApi.GetCookieEx must run
  356. * with an await to retain synchronicity in the messaging environment.
  357. * @param {Object} fvApi Api Control Object
  358. * @param {String} sessionKey SessionKey if for active session
  359. * */
  360. FVSESS.BindNewSession = async function (fvApi, sessionKey)
  361. {
  362. var me = this,
  363. realSessionKey = (sessionKey && (sessionKey.substr(0, 4) == 'FVC_')) ? await fvApi.GetCookieEx(sessionKey) : sessionKey;
  364. this.fvApi = fvApi;
  365. if (sessionKey)
  366. log('BindNewSession for ' + sessionKey + ' (' + realSessionKey + ')');
  367. else
  368. log('BindNewSession for failed session');
  369. if (sessionKey)
  370. this.SaveSessionKey(realSessionKey);
  371. this.SetStatus('started');
  372. fvApi.onnewscreen = function (rows, screenName)
  373. {
  374. me.screenName = screenName;
  375. if (me.bindings.onnewscreen)
  376. me.bindings.onnewscreen(me, rows, screenName);
  377. };
  378. fvApi.onclose = function (closeType)
  379. {
  380. me.SetStatus('offline'); // clears sessionkey
  381. if (me.bindings.onclose)
  382. me.bindings.onclose(me, closeType);
  383. };
  384. fvApi.onsend = function (enterKey)
  385. {
  386. if (me.bindings.onsend)
  387. return me.bindings.onsend(me, enterKey);
  388. else
  389. return true;
  390. };
  391. fvApi.onalert = function (alertObj, msg, bgColor, fnOnClick, buttons)
  392. {
  393. if (me.bindings.onalert)
  394. return me.bindings.onalert(me, alertObj, msg, bgColor, fnOnClick, buttons);
  395. else
  396. return null; // not handle
  397. };
  398. };
  399. /** @memberof! FVSessions.Session.prototype
  400. * @description Internal use only function used to set functions when FVSessions.config.msgAPI is true
  401. */
  402. FVSessions.SetMsgMethods = function ()
  403. {
  404. /** @memberof! FVSessions.Session.prototype
  405. * @description Close the Session if active
  406. */
  407. FVSESS.Close = function ()
  408. {
  409. if (this.status != 'offline')
  410. {
  411. try
  412. {
  413. this.fvApi.CloseTerm();
  414. }
  415. catch (e) { }
  416. }
  417. };
  418. /** @memberof! FVSessions.Session.prototype
  419. * @description Control if the session has focus - Note for MsgAPI need to await this if you want it
  420. * to be synchronous
  421. * @param {boolean} focusOn If true the session will have focus set, if false, focus will be off and locked off until
  422. * @param {boolean} activate If true, when focusOn is true, will also set browser focus into the FVTerm body
  423. * set back on with this call
  424. * @returns {Promise} For MsgAPI returns a Promise
  425. */
  426. FVSESS.SetFocus = async function (focusOn, activate)
  427. {
  428. var me = this;
  429. return new Promise(async (resolve) =>
  430. {
  431. if (this.status != 'offline')
  432. {
  433. try
  434. {
  435. if (focusOn)
  436. {
  437. await me.fvApi.EmActive(2, 'FVSession');
  438. if (activate)
  439. await me.fvApi.SCSetTermFocus();
  440. }
  441. else
  442. {
  443. await me.fvApi.EmActive(-1, 'FVSession');
  444. }
  445. resolve(true);
  446. }
  447. catch (e)
  448. {
  449. log('Exception during SetFocus=' + e);
  450. resolve(false);
  451. }
  452. }
  453. });
  454. };
  455. /** @memberof! FVSessions.Session.prototype
  456. * @description Read from the active screen--will trim blanks from the text that is read--
  457. * Note that for MsgAPI you need to await this call to get the right value
  458. * @param {Integer} row Row to read from where 1 is the first row
  459. * @param {Integer} column Column to read from where 1 is the first column
  460. * @param {Integer} length Length to read--if row/column is entry field this is ignored
  461. * @returns {String} Text from the screen at row, column for length
  462. */
  463. FVSESS.GetScreenText = async function (row, column, length)
  464. {
  465. return this.fvApi.GetScnTextEx(row, column, length); // returns a promise
  466. };
  467. /** @memberof! FVSessions.Session.prototype
  468. * @description Send a set of keys to the session, should end with Enter or Function key
  469. * @param {String} transactions Transactions to run separated by the vertical bar (|) -- grammar for each transaction:
  470. * <ul style=list-style:none;">
  471. * <li>F^row^column=move cursor to location;</li>
  472. * <li>H^enterKeys^waitForScreen= enterkeys that end with enter or function key and waitForScreen is a screen in the active application's definition file</li>
  473. * <li>I^keystrokes= keystrokes are a mix of text and @T for tab and so on (default action if no I^ is present at start of the transaction)</li>
  474. * <li>T^text^row^column=Enter text into an unprotected field at row and column where each starts with 1</li>
  475. * <li>S^milliseconds=Sleep for the milliseconds like S^2000 for 2 second sleep</li>
  476. * <li>W^milliseconds^waitForScreen=Sleep for the milliseconds like S^2000 for 2 second sleep but if screen comes in move on</li>
  477. * </ul>
  478. * Example: H^RMIS@E^RMIS|T^123456789^3^23|T^10081958^5^23|H^@E^RMIM will enter the transaction "RMIS" then wait for a screen identified as RMIS;
  479. * then it will key "123456789" at a field at row 3 column 23; then a field "10081958" at row 5, column 23; then will send an ENTER and wait again for RMIS to appear
  480. * @param {function(screenName)} onComplete Optional completion function for when the transactions are completed
  481. * @param {boolean} bubble Set true for screen updates to be bubbled to the onload
  482. */
  483. FVSESS.Transactions = function (transactions, onComplete, bubble)
  484. {
  485. this.fvApi.Transactions(transactions, onComplete, bubble);
  486. };
  487. /** @memberof! FVSessions.Session.prototype
  488. * @description Set a field's value for MsgAPI use await to ensure synchronous operation
  489. * @param {String} text Text to set into an unprotected entry field on the screen
  490. * @param {integer} row Row to set the text on starting with 1
  491. * @param {integer} column Column to set thet ext on starting with 1
  492. * @returns {Boolean} Function succeeded (MsgAPI only)
  493. */
  494. FVSESS.SetField = function (text, row, column)
  495. {
  496. var me = this;
  497. return new Promise(async (resolve) =>
  498. {
  499. var ok=await me.fvApi.SetField(row, column, text);
  500. resolve(ok);
  501. });
  502. };
  503. };
  504. /** @memberof! FVSessions.Session.prototype
  505. * @description Internal use only function used to set functions when FVSessions.config.msgAPI is false
  506. */
  507. FVSessions.SetLocalMethods = function ()
  508. {
  509. /** @memberof! FVSessions.Session.prototype
  510. * @description Split keys so that hllapi enters and function keys break into individual objects --
  511. * NOT SUPPORTED for MsgAPI=true
  512. * @param {String} text Hllapi string with @ mnemonics (not [enter] style for now!)
  513. * @returns {Object} Array of strings where each has keys ending with an AID (enter/function)
  514. */
  515. FVSESS.SplitKeys = function (text)
  516. {
  517. try
  518. {
  519. var i, ch, parts = [], keys, start = 0, state = 0;
  520. for (i = 0; i < text.length; i++)
  521. {
  522. ch = text.substr(i, 1);
  523. switch (state)
  524. {
  525. case 0:
  526. start = i;
  527. state = 1;
  528. // fall through
  529. case 1:
  530. if (ch == '@')
  531. state = 2;
  532. break;
  533. case 2:
  534. if (enterKeys.indexOf(ch) != -1)
  535. {
  536. parts.push(text.substr(start, (i - start) + 1));
  537. state = 0;
  538. }
  539. else
  540. state = 1;
  541. break;
  542. }
  543. }
  544. return parts;
  545. }
  546. catch (e)
  547. {
  548. log('SplitKeys exception=' + e);
  549. return [text];
  550. }
  551. };
  552. FVSESS.Close = function ()
  553. {
  554. if (this.status != 'offline')
  555. {
  556. try
  557. {
  558. this.fvApi.fvWin.CloseTerm();
  559. }
  560. catch (e) { }
  561. }
  562. };
  563. FVSESS.SetFocus = function (focusOn, activate)
  564. {
  565. if (this.status != 'offline')
  566. {
  567. try
  568. {
  569. if (focusOn)
  570. {
  571. this.fvApi.fvWin.EmActive(2, 'FVSession');
  572. if (activate)
  573. this.fvApi.fvWin.SCSetTermFocus();
  574. }
  575. else
  576. {
  577. this.fvApi.fvWin.EmActive(-1, 'FVSession');
  578. }
  579. }
  580. catch (e) { }
  581. }
  582. };
  583. FVSESS.GetScreenText = async function (row, column, length)
  584. {
  585. return new Promise((resolve) =>
  586. {
  587. resolve(this.fvApi.GetScnTextEx(row, column, length));
  588. });
  589. };
  590. FVSESS.Transactions = function (transactions, onComplete, bubble)
  591. {
  592. var me = this, lastScreen = this.screenName, timeoutHandle = null, atTran = 0, parts, tran, wait = false, waitingFor = '*', trans = transactions.split('|'),
  593. topHasFocus = this.fvApi.fvWin.topHasFocus,
  594. nextTran = function ()
  595. {
  596. atTran++;
  597. if (timeoutHandle)
  598. {
  599. clearTimeout(timeoutHandle);
  600. timeoutHandle = null;
  601. }
  602. if (atTran < trans.length)
  603. {
  604. if (trans[atTran] == '')
  605. nextTran();
  606. else
  607. window.setTimeout(function () { innerExec() }, 1); // let current screen event complete!
  608. }
  609. else
  610. {
  611. fixFocus();
  612. if (onComplete)
  613. onComplete(lastScreen);
  614. }
  615. },
  616. fixFocus = function ()
  617. {
  618. if (me.fvApi.fvWin.topHasFocus != topHasFocus)
  619. me.fvApi.fvWin.EmActive((topHasFocus) ? 1 : 0, 'FVSess.Transactions');
  620. //log('Transactions end topHasfocus=' + me.fvApi.fvWin.topHasFocus);
  621. },
  622. screenWaiter = function (rows, screenName)
  623. {
  624. //log(me.id + ' Transactions screenWaiter waitingFor=' + waitingFor + ' result screenName=' + screenName);
  625. lastScreen = screenName;
  626. if (waitingFor == '*') // any screen
  627. nextTran();
  628. else
  629. {
  630. if (screenName != waitingFor)
  631. me.fvApi.fnNextScreen = screenWaiter;
  632. else
  633. nextTran();
  634. }
  635. return bubble; // determines if next screen is bubbled up
  636. },
  637. innerExec = function ()
  638. {
  639. tran = trans[atTran];
  640. parts = (tran.indexOf('^') == -1) ? ['I', tran] : tran.split('^');
  641. //log(me.id + ' ('+me.sessionKey+') Transactions innerExec atTran=' + atTran + ' trans.length=' + trans.length+' transaction=' + tran);
  642. try
  643. {
  644. switch (parts[0])
  645. {
  646. case 'F': // move cursor to location
  647. me.fvApi.SetCursor(parseInt(parts[1]), parseInt(parts[2]));
  648. nextTran();
  649. break;
  650. case 'H': // send hllapi and wait for screen
  651. waitingFor = parts[2];
  652. //log(me.id + ' Transactions send and wait=' + parts[1] + ', ' + waitingFor);
  653. me.fvApi.SendKeys(parts[1], screenWaiter);
  654. timeoutHandle = window.setTimeout(function ()
  655. {
  656. // If we hit this a wait function went longer than 30 seconds
  657. atTran = trans.length;
  658. fixFocus();
  659. if (onComplete)
  660. onComplete(lastScreen, 'TIMED-OUT waiting for ' + waitingFor);
  661. }, 30000);
  662. break;
  663. case 'W': // wait for screen or timeout
  664. waitingFor = parts[2];
  665. me.fvApi.fnNextScreen = screenWaiter;
  666. timeoutHandle = window.setTimeout(function ()
  667. {
  668. // If we hit this a wait function went longer than wait point
  669. nextTran();
  670. }, parseInt(parts[1]));
  671. break;
  672. case 'I': // just perform keystrokes
  673. waitingFor = '*';
  674. //me.fvApi.SendKeys(parts[1], screenWaiter);
  675. if (me.TypeKeys(parts[1], screenWaiter))
  676. nextTran();
  677. break;
  678. case 'T': // Type text into row column
  679. me.TypeAtRowCol(parts[1], parseInt(parts[2]), parseInt(parts[3]));
  680. nextTran();
  681. break;
  682. case 'S':
  683. window.setTimeout(function () { nextTran(); }, parseInt(parts[1]));
  684. break;
  685. default: // extra | maybe?
  686. nextTran();
  687. break;
  688. }
  689. }
  690. catch (e)
  691. {
  692. log(me.id + ' Transactions innerExec exception=' + e);
  693. }
  694. }
  695. bubble = (typeof (bubble) == 'undefined') ? false : bubble;
  696. //log('Transactions start topHasfocus=' + this.fvApi.fvWin.topHasFocus);
  697. innerExec();
  698. };
  699. /** @memberof! FVSessions.Session.prototype
  700. * @description Type HLLAPI string starting at current row and column
  701. * NOT SUPPORTED for MsgAPI=true
  702. * @param {String} text Text to type into unprotected entry field(s) on the screen, can include @T for tab and end
  703. * with @E or other AID string, which will be sent to the host for processing
  704. * @param {function(screenRows,screenName)} screenWaiter Provides callback instead of onLoad
  705. * @returns {boolean} Keys were typed OK
  706. */
  707. FVSESS.TypeKeys = function (text, screenWaiter)
  708. {
  709. var me = this, i, ch, keys, state = 0,
  710. ev, kev, em = me.fvApi.emSess,
  711. fvWin = me.fvApi.fvWin,
  712. SetEV = function (chr)
  713. {
  714. ev = {};
  715. if (chr)
  716. ev.keyCode = chr.charCodeAt(0);
  717. kev = fvWin.SCKEV(em, -1, ev);
  718. kev.apiCall = true;
  719. };
  720. try
  721. {
  722. for (i = 0; i < text.length; i++)
  723. {
  724. ch = text.substr(i, 1);
  725. switch (state)
  726. {
  727. case 0:
  728. if (ch == '@')
  729. state = 1;
  730. else
  731. {
  732. // type character
  733. SetEV(ch);
  734. fvWin.SCKeyPress(em, ev, kev);
  735. }
  736. break;
  737. case 1:
  738. if (enterKeys.indexOf(ch) != -1) // AID, end of this!
  739. {
  740. me.fvApi.SendKeys(text.substr(i - 1), screenWaiter);
  741. return false;
  742. }
  743. else
  744. {
  745. state = 0; // assume single character
  746. SetEV();
  747. switch (ch)
  748. {
  749. case 'D':
  750. fvWin.SCDelete(em, kev);
  751. break;
  752. case 'q':
  753. fvWin.SCCursorEnd(em, kev);
  754. break;
  755. case '0':
  756. fvWin.CursorHome(em, kev, true);
  757. break;
  758. case 'A':
  759. state = 2;
  760. break;
  761. case 'C':
  762. state = 3;
  763. break;
  764. case 'F': // FieldExit
  765. fvWin.FieldExit(em, kev, 'NT');
  766. break;
  767. case 'T': // tab
  768. fvWin.SCTab(em, ev, kev);
  769. break;
  770. case 'B':
  771. fvWin.SCBackTab(em, ev, kev);
  772. break;
  773. case 'L':
  774. fvWin.CursorLeft(em, kev);
  775. break;
  776. case 'U':
  777. fvWin.CursorUp(em, kev);
  778. break;
  779. case 'Z':
  780. fvWin.CursorRight(em, kev);
  781. break;
  782. case 'V':
  783. fvWin.CursorDown(em, kev);
  784. break;
  785. default:
  786. log('TypeKeys unknown HLLAPI Code-@' + ch);
  787. break;
  788. }
  789. }
  790. break;
  791. case 2: // first char was 'A'
  792. if (ch == 'E')
  793. fvWin.FieldExit(em, kev, '+');
  794. else
  795. log('TypeKeys unknown HLLAPI Code-@A@' + ch);
  796. state = 0;
  797. break;
  798. case 3: // first char was 'C'
  799. if (ch == 'H')
  800. fvWin.SCBackSp(em, kev);
  801. else
  802. log('TypeKeys unknown HLLAPI Code-@C@' + ch);
  803. state = 0;
  804. break;
  805. }
  806. }
  807. }
  808. catch (e)
  809. {
  810. log('TypeKeys exception=' + e);
  811. }
  812. return true;
  813. };
  814. /** @memberof! FVSessions.Session.prototype
  815. * @description Type text starting at a row and column - if length of text exceeds first field will autoskip to next
  816. * NOT SUPPORTED FOR MsgAPI=true
  817. * @param {String} text Text to type into unprotected entry field(s) on the screen
  818. * @param {integer} row Row to type the text on starting with 1
  819. * @param {integer} column Column to type the text on starting with 1
  820. */
  821. FVSESS.TypeAtRowCol = function (text, row, column)
  822. {
  823. var ev = {}, kev;
  824. ev.keyCode = 600;
  825. //log('TypeAtRowCol: "' + text + '" @' + row + ',' + column);
  826. this.fvApi.SetCursor(row, column);
  827. kev = this.fvApi.fvWin.SCKEV(this.fvApi.emSess, -1, ev);
  828. kev.apiCall = true;
  829. this.fvApi.fvWin.SCPaste(this.fvApi.emSess, kev, text);
  830. };
  831. FVSESS.SetField = function (text, row, column)
  832. {
  833. this.fvApi.SetField(row, column, text);
  834. };
  835. };
  836. /** @memberof! FVSessions.Session.prototype
  837. * @description Enter the active screen using a hllapi string
  838. * @param {String} enterKeys Should be one or more keys with an enter/function/action key at the end like @E or @C ([enter] or [clear] or @T for Tabbing with ASCII host)
  839. * @param {function(screenName)} onComplete Optional completion function for when the screen has been entered
  840. */
  841. FVSESS.EnterScreen = function (enterKeys, onComplete)
  842. {
  843. this.fvApi.SendKeys(enterKeys, function (rows, screenName)
  844. {
  845. if (onComplete)
  846. onComplete(screenName);
  847. });
  848. };
  849. /** @memberof! FVSessions.Session.prototype
  850. * @description Set the status of the Session
  851. * @param {String} newStatus Status which includes offline, started (needs logon) and ready (not on any logon screens)
  852. */
  853. FVSESS.SetStatus = function (newStatus)
  854. {
  855. this.status = newStatus;
  856. switch (newStatus)
  857. {
  858. case 'offline':
  859. this.SaveSessionKey();
  860. break;
  861. }
  862. };
  863. /** @memberof! FVSessions.Session.prototype
  864. * @description Get the current session key from local storage
  865. * @returns {String} Sessionkey or null if not found
  866. */
  867. FVSESS.GetSessionKey = function ()
  868. {
  869. var sessionKey = localStorage.getItem(this.storeId);
  870. log('GetSessionKey for ' + this.storeId + '=' + sessionKey);
  871. return sessionKey;
  872. };
  873. /** @memberof! FVSessions.Session.prototype
  874. * @description Save a sessionKey for this session to local storage
  875. * @param {String} sessionKey SessionKey to save -- if null, the localstorage will be cleared
  876. */
  877. FVSESS.SaveSessionKey = function (sessionKey)
  878. {
  879. if (!this.connection.keepAliveOnClose)
  880. return; // only save sessionKey when keepAliveOnClose is true...
  881. if (!sessionKey)
  882. {
  883. this.sessionKey = null;
  884. log('SaveSessionKey deleted key for ' + this.storeId);
  885. localStorage.removeItem(this.storeId);
  886. }
  887. else
  888. {
  889. log('SaveSessionKey saved ' + sessionKey + ' for ' + this.storeId);
  890. this.sessionKey = sessionKey;
  891. localStorage.setItem(this.storeId, sessionKey);
  892. }
  893. };
  894. // #endregion Session Class
  895. }());