-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdivisible_room_primary.js
3549 lines (2983 loc) · 138 KB
/
divisible_room_primary.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/*
Copyright (c) 2023 Cisco and/or its affiliates.
This software is licensed to you under the terms of the Cisco Sample
Code License, Version 1.1 (the "License"). You may obtain a copy of the
License at
https://developer.cisco.com/docs/licenses
All use of the material herein must be in accordance with the terms of
the License. All rights not expressly granted by the License are
reserved. Unless required by applicable law or agreed to separately in
writing, software distributed under the License is distributed on an "AS
IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
or implied.
*
*
* Repository: gve_devnet_divisible_conference_rooms_webex_devices_macros
* Macro file: divisible_room_primary
* Version: 1.0.1
* Released: May 7, 2024
* Latest RoomOS version tested: 11.15.1.6
*
* Macro Author: Gerardo Chaves
* Technical Solutions Architect
* Cisco Systems
*
* Consulting Engineer: Robert(Bobby) McGonigle Jr
* Technical Marketing Engineer
* Cisco Systems
*
*
*
* As a macro, the features and functions of this webex divisibe conference
* rooms macro are not supported by Cisco TAC
*
* Hardware and Software support are provided by their respective manufacturers
* and the service agreements they offer
*
* Should you need assistance with this macro, reach out to your Cisco sales representative
* so they can engage the GVE DevNet team.
*/
import xapi from 'xapi';
import { GMM } from './GMM_Lib'
const minOS10Version = '10.17.1.0';
const minOS11Version = '11.2.1.0';
const JS_PRIMARY = 1, JS_SECONDARY = 2, JS_AUXILIARY = 3, JS_LOCAL = 0
/*
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ SECTION 1 - SECTION 1 - SECTION 1 - SECTION 1 - SECTION 1 - SECTION 1 +
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
*/
// In this section, write in the values for the constants below.
// If you fill out the SECONDARY_CODECS_USERNAME and SECONDARY_CODECS_PASSWORD with the credentials to be able to log
// into the the Secondary codec (if configuring Primary) or Primary codec (if configuring Secondary)
// they will be used to establish an HTTP connection with that other codec, but these credentials will be
// stored clear text in the macro.
// If you wish to slightly obfuscate the credentials, use a Base64 encoded string for SECONDARY_CODECS_USERNAME and
// leave SECONDARY_CODECS_PASSWORD blank. If you do that, you would need to combine the username and password in one string
// separated by a colon (i.e. "username:password") before Base64 encoding with a tool such as https://www.base64encode.org/
// Instructions for creating these admin accounts are in the "Installation Instructions" document.
const SECONDARY_CODECS_USERNAME = ''
const SECONDARY_CODECS_PASSWORD = ''
// You can fill out the SECONDARIES_BOT_TOKEN value intead of SECONDARY_CODECS_USERNAME/SECONDARY_CODECS_PASSWORD to use the Webex cloud to
// communicate with other codecs in the system. it should contain the Bot access token you wish to use to have the codec use
// when sending commands to the other codecs by using Webex messaging.
// NOTE: You must add the Bot that corresponds to the bot token you intend to use to the API access list in the Workspace where the device is configured
// To do so in Control Hub, go to the Workspace for each device, click on the "Edit API Access" button and add the bot to the list (search for it by name)
// with "Full Access" access level.
const SECONDARIES_BOT_TOKEN = ''
/*
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ SECTION 2 - SECTION 2 - SECTION 2 - SECTION 2 - SECTION 2 - SECTION 2
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
*/
// If you wish to pin-protect the room combine/split control
// panel (when not using wall sensor), enter a numeric value for COMBINE_CONTROL_PIN
// otherwise leave it blank: ""
const COMBINE_CONTROL_PIN = "";
// For more reliability when combining and dividing rooms you can use a custom cable connecting the
// GPIO pins 2-4 between the primary codec and secondary codecs. This cable cannot be used if you have
// a setup where you need to "promote" a secondary room to primary to accomodate specific room layouts
// in which case the value should be false.
const USE_GPIO_INTERCODEC = false;
// USE_WALL_SENSOR controls if you use a physical wall sensor or not
// If set to false, you will get a custom panel to manually switch rooms from join to split
// If set to true, you will get a PIN protected override button, in case the wall sensor is broken
// and you need to override manually
const USE_WALL_SENSOR = false;
// WALL_SENSOR_COMBINED_STATE shoud contain the state of PIN 1 when the rooms is
// combined. This could be 'High' or 'Low' depending on how the sensor is wired
const WALL_SENSOR_COMBINED_STATE = 'Low';
/*
If you set USE_WALL_SENSOR to true above, you can
change the override protect PINs here if needed.
*/
const COMBINE_PIN = "1234";
const SPLIT_PIN = "4321";
const FIXED_SENSOR = "5678";
// USE_ALTERNATE_COMBINED_PRESENTERTRACK_SETTINGS speficies if different settings should be used for presentertrack on primary codec
// for combined and split modes. If set to true, you must modify the settings for presentertrack to use for each scenario in the
// SPLIT_PRESENTERTRACK_SETTINGS and COMBINED_PRESENTERTRACK_SETTINGS object constants below.
// Instructions on how setup and to obtain the settings from the primary codec can be found in
// the "How_to_Setup_Two-PresenterTrack_Zones.pdf" document in the same repository for this macro.
const USE_ALTERNATE_COMBINED_PRESENTERTRACK_SETTINGS = false;
const SPLIT_PRESENTERTRACK_SETTINGS = {
PAN: -1000,
TILT: -309,
ZOOM: 4104,
TRIGGERZONE: '0,95,400,850'
};//Replace these placeholder values with your actual values.
// Each key in the N_COMBINED_PRESENTERTRACK_SETTINGS object refers to the
// name of compositions associated to the secondary rooms selected (separated by ':' ), in addition
// to the primary room, when combining rooms for which you wish to use the set
// of values for presenter track reflected in the value of the entry.
// For example, entry with key 'RoomSecondaryRight' will be used when the primary room
// plus the secondary codec associated to the RoomSecondaryRight are combined, and
// entry with key 'RoomSecondaryLeft:RoomSecondaryRight' will be used when the primary room
// plus the secondary codecs associated to both the RoomSecondaryRight and RoomSecondaryRight are combined
// into a 3 way combined room
const N_COMBINED_PRESENTERTRACK_SETTINGS = {
'RoomSecondaryRight':
{
PAN: -1378,
TILT: -309,
ZOOM: 4104,
TRIGGERZONE: '0,89,549,898'
},
'RoomSecondaryLeft':
{
PAN: -1378,
TILT: -309,
ZOOM: 4104,
TRIGGERZONE: '0,89,549,898'
},
'RoomSecondaryLeft:RoomSecondaryRight':
{
PAN: -1378,
TILT: -309,
ZOOM: 4104,
TRIGGERZONE: '0,89,549,898'
},
'RoomSecondaryLeft:RoomSecondaryRight:RoomSecondaryFarRight':
{
PAN: -1378,
TILT: -309,
ZOOM: 4104,
TRIGGERZONE: '0,89,549,898'
}
}; //Replace these placeholder values with your actual values.
// CHK_VUMETER_LOUDSPEAKER specifies if we check the LoudspeakerActivity flag from the VuMeter events
// to ignore any microphone activity while the loudspeakers are active to reduce the possibility of
// switching due to sound coming in from remote participants in the meeting if the AfterAEC setting
// is not being effective. Set to true to perform the check for each microphone activity event.
const CHK_VUMETER_LOUDSPEAKER = false;
/*
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ SECTION 3 - SECTION 3 - SECTION 3 - SECTION 3 - SECTION 3 - SECTION 3 +
+
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
*/
// To set the volume of the primary codecs to a specific value when combined vs when standalone, set the
// the PRIMARY_COMBINED_VOLUME_COMBINED and PRIMARY_COMBINED_VOLUME_STANDALONE constants
// if you leave them with value 0 they will be ignored
const PRIMARY_COMBINED_VOLUME_COMBINED = 0;
const PRIMARY_COMBINED_VOLUME_STANDALONE = 0;
/*
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ SECTION 4 - SECTION 4 - SECTION 4 - SECTION 4 - SECTION 4 - SECTION 4 +
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
General microphones and video sources for both primary and secondary codecs
The monitorMics, ethernetMics and usbMics arrays refer to locally connected microphones for which the macro will monitor vuMeter levels.
The ID range for monitorMics is 1-8 since it refers to the physical analog mic input connectors on the codec.
The ID range for ethernetMics is 11-18, 21-28 an so forth until 81-88 since we support up to 8 ethernet mics with 8
sub-ids each. So, for example , ethernec mic ID 12 as specified in this array refers to Ethernet Mic 1, sub-ID 2
The ID range for usbMics is 101-104 an maps to USB mic IDs 1-4 even though at the moment just one USB Mic input is supported (101)
The externalMics array refers to externally connected microphones where a controller sends the codec text messages over SSH or
serial interface indicating which of those external microphones is currently active.
The text message should be sent by the controller in the format “MIC_ACTIVE_XX” where XX is a distinct
“microphone” id from 01 o 99. We are reserving 00 to indicate that there is relative silence in the room or that mute is active.
Even though the receiving of unformatted “MIC_ACTIVE_XX” type strings is supported, for better logging it is strongly
recommended that the controller sends the message wrapped as an object as shown in the following examples.
sending the MIC_ACTIVE_01 message via serial:
xCommand Message Send Text: "{\x5C"App\x5C":\x5C"Crestron\x5C",\x5C"Source\x5C":{},\x5C"Type\x5C":\x5C"Command\x5C",\x5C"Value\x5C":\x5C"MIC_ACTIVE_01\x5C"}"\x0D\x0A
sending the MIC_ACTIVE_01 message via SSH:
xCommand Message Send Text: "{\"App\":\"Crestron\",\"Source\":{},\"Type\":\"Command\",\"Value\":\"MIC_ACTIVE_01\"}"
NOTE: Any combination of microphone types specified in the monitorMics, ethernetMics , usbMics and externalMics is supported by
the macro, but given the differences in echo cancellation processing perfomed by the different microphone categories it is strongly
advised to stick to only one type of microphone to use for each installation.
NOTE: See section 6 for PresenterTrack QA mode configuration and the PRESENTER_QA_AUDIENCE_MIC_IDS array
*/
const config = {
monitorMics: [1, 2, 3, 8], // (ex: [1, 2, 3, 4, 5, 6, 7, 8] ) analog input connectors (1-8) associated to microphones monitored
ethernetMics: [], // (ex: 11, 12, 13, 14] ) IDs associated to Ethernet mics, up to 8 mics with 8 sub-ids: e.j. 12 is Ethernet Mic 1, sub-ID 2.
usbMics: [], // (ex: [101]) Mic input connectors associated to the USB microphones being used in the main codec: 101 is USB Mic 1
externalMics: [], // (ex: [901, 902]) input ids associated to microphones connected to an external controller received as message format MIC_ACTIVE_XX where XX is an external mic id 01-99
compositions: [ // Create your array of compositions, if room role is JS_SECONDARY, these are for local cameras and AUX codecs only, no JS_SECONDARY source compositions allowed
{ // example for quadcam directly connected to connector 1 in main room
name: 'RoomMain', // Name for your composition.
codecAddress: '', // No codecAddress needed if source is JS_LOCAL
mics: [1, 2], // Mics you want to associate with this composition. Example: [1, 2, 3]
connectors: [1], // Video input connector Ids to use
source: JS_LOCAL, // Always use JS_LOCAL in Primary or Secondary when referring to locally connected camera
layout: 'Prominent',// Layout to use
preset: 0 // use a camera preset instead of a layout with specific connectors.
},
{ // example for video from Secondary codec 'RoomSecondaryRight'
name: 'RoomSecondaryRight', //Name for your composition. name will be used in toggle UI if source is JS_SECONDARY
codecAddress: '10.0.0.112', // IP address or Webex ID (if SECONDARIES_BOT_TOKEN is set) of the secondary codec. To obtain codec ID: xStatus Webex DeveloperId
mics: [8], // in this example, audio tieline coming from secondary codec RoomSecondaryRight is connected into analog/mic connector 8
connectors: [2], // in this example, video tie line from secondary codec RoomSecondaryRight is connected to input connector 2
source: JS_SECONDARY, // all compositions related to secondary codecs must speecify source: JS_SECONDARY
layout: 'Prominent', // Layout to use
preset: 0 // use a camera preset instead of a layout with specific connectors.
},
/*
{ // example for video from Secondary codec 'RoomSecondaryLeft'
name: 'RoomSecondaryLeft', //Name for your composition. name will be used in toggle UI if source is JS_SECONDARY
codecAddress: '10.0.0.113', // IP address or Webex ID (if SECONDARIES_BOT_TOKEN is set) of the secondary codec. To obtain codec ID: xStatus Webex DeveloperId
mics: [7], // in this example, audio tieline coming from secondary codec RoomSecondaryRight is connected into analog/mic connector 8
connectors: [3], // in this example, video tie line from secondary codec RoomSecondaryRight is connected to input connector 2
source: JS_SECONDARY, // all compositions related to secondary codecs must speecify source: JS_SECONDARY
layout: 'Prominent', // Layout to use
preset: 0 // use a camera preset instead of a layout with specific connectors.
},
*/
{
// NOTE: If you want to always show and overview shot irrespective of microphone input, just
// set SIDE_BY_SIDE_TIME in section 5 below to 0
// Also, if you wish to show several presets in a composition or a combination of presets and
// non-preset camera or tie line video inputs, specify the presets to use in the preset key below
// as an array (i.e. [11,12]) but also include the video connector ID for the cameras for those
// presets in the connectors array below in the right order so the macro knows how to lay them out in the composition
// (i.e. connectors:[2,3,1,4] if the connectorID for the camera associated for preset 11 is 2,
// the connectorID for the camera associated for preset 12 is 3 and you want to also include input from quadcam
// at connector 1 and video from tieline from secondary in connector 4 as the overview shot.)
name: 'Overview', // IMPORTANT: There needs to be an overview compositino with mics: [0]
codecAddress: '', // No codecAddress needed if source is JS_LOCAL
mics: [0], // always just [0] for overview compositions
connectors: [1, 2], // Specify here the video inputs and order to use to compose the "overview" shot. Ex: [2,1] including those for preset related cameras
source: JS_LOCAL, // Overview composition always has source JS_LOCAL
layout: 'Equal', // Layout to use
preset: 0 // use a camera preset instead of a layout with specific connectors. Specify a single preset or an array of preset Ids
// NOTE: do not set preset to just one integer if you want more than one video input to be layed out, if you only
// have one preset but still want to specify other connectos in the layout then specify and array of just one preset
// (i.e. preset: [11] if only preset 11 will be used and connectors:[2,1,4] if you want to compose it input from the
// camera doing the preset with connectors 1 and 4 as well.)
// Setting preset to just one integeter will force it to ignore the connectors value
// Set preset to 0 if no presets will be used.
}
]
};
// If you are using a SpeakerTrack 60, set QUAD_CAM_ID to the connector ID where the first camera of the array is connected
// and also use that ID in the connetors array in the compositions above
// If you are using a QuadCam, set this value to the connector ID being used for it.
// If you do not have any speakertracking capable cameras, just set this value to 0
const QUAD_CAM_ID = 1;
// In RoomOS 11 there are multiple SpeakerTrack default behaviors to choose from on the navigator or
// Touch10 device. Set ST_DEFAULT_BEHAVIOR to the one you want this macro to use from these choices:
// Auto: The same as BestOverview.
// BestOverview: The default framing mode is Best overview.
// Closeup: The default framing mode is Closeup (speaker tracking).
// Current: The framing mode is kept unchanged when leaving a call.
// Frames: The default framing mode is Frames.
const ST_DEFAULT_BEHAVIOR = 'Closeup';
/*
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ SECTION 5 - SECTION 5 - SECTION 5 - SECTION 5 - SECTION 5 - SECTION 5 +
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
TIMERS and THRESHOLDS
*/
// Time to wait for silence before setting Speakertrack Side-by-Side (Overview) mode
// set SIDE_BY_SIDE_TIME to 0 if you always want to show that mode
const SIDE_BY_SIDE_TIME = 10000; // 10 seconds
// Time to wait before switching to a new speaker
const NEW_SPEAKER_TIME = 2000; // 2 seconds
// Time to wait before activating automatic mode at the beginning of a call
const INITIAL_CALL_TIME = 15000; // 15 seconds
// WEBRTC_VIDEO_UNMUTE_WAIT_TIME only applies to RoomOS version 10 since
// have to to implement a woraround there to be able to switch cameras
// while in a WebRTC call. Values less than 1500 ms do not seem to work, but
// if you are having trouble getting switching to work in WebRTC calls you can increase
// this value although that will affect the overall experience since during this time
// the remote participants just see a black screen instead of the video feed.
const WEBRTC_VIDEO_UNMUTE_WAIT_TIME = 1500;
// Microphone High/Low Thresholds
const MICROPHONELOW = 6;
const MICROPHONEHIGH = 25;
/*
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ SECTION 6 - SECTION 6 - SECTION 6 - SECTION 6 - SECTION 6 - SECTION 6 +
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Presenter Track Q&A Mode
*/
// ALLOW_PRESENTER_QA_MODE controls if the custom panel for activating PresenterTrack with or without
// Q&A Mode is shown in the Touch10 or Navigator. Without it, you cannot activate PresenterTrack Q&A mode
const ALLOW_PRESENTER_QA_MODE = false;
//PRESENTER_QA_AUDIENCE_MIC_IDS is an array for Mic IDs that are being used for the audience.
const PRESENTER_QA_AUDIENCE_MIC_IDS = [1, 2, 8]; // in this example, the missing mic from monitorMics is 3 which is the presenter's mic
// PRESENTER_QA_KEEP_COMPOSITION_TIME is the time in ms that the macro will keep sending
// a composed image of the presenter and an audience member asking a question after the question
// has been asked by any audience member. If different audience members ask questions while the composition
// is being shown after NEW_SPEAKER_TIME milliseconds have passed, the composition will change
// to use that new audience member instead of the original. This will continue until no other audience members have
// spoken for PRESENTER_QA_KEEP_COMPOSITION_TIME milliseconds and then the code will resume sending only the
// full video feed from the Presenter camera
const PRESENTER_QA_KEEP_COMPOSITION_TIME = 7000
/*
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
+ DO NOT EDIT ANYTHING BELOW THIS LINE +
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
*/
const enableKeepAlive = false;
const keepAliveReportOnlyFails = true;
// KA_FREQUENCY_SECONDS is the frequency in which to send keep alives to secondaries, in seconds,
// no less than 3 and at least 1 more than KA_CHECK_REPLIES_TIMEOUT_MS/1000
const KA_FREQUENCY_SECONDS = 15;
const KA_CHECK_REPLIES_TIMEOUT_MS = 2000; // time in ms to check for KA replies no less than 1000
async function isCodecPro() {
let ProductPlatform = await xapi.Status.SystemUnit.ProductPlatform.get()
return (ProductPlatform == "Codec Pro")
}
let secondariesKAStatus = {};
function priHandleKeepAliveResponse(ipAddress) {
// called when received "VTC_KA_OK"
secondariesKAStatus[ipAddress].online = true;
}
function priKeepAliveStatuses() {
let allReportsOnline = true;
// check status of all KA responses and report if any missing
Object.entries(secondariesKAStatus).forEach(([key, val]) => {
if (!val.online) {
console.warn(`Secondary at IP: ${key} did not respond to latest keep alive`);
allReportsOnline = false;
}
})
if (!keepAliveReportOnlyFails && allReportsOnline)
console.log(`Received KeepAlive responses from all secondaries after ${KA_CHECK_REPLIES_TIMEOUT_MS} milliseconds. `)
}
async function priSendKeepAlive() {
//send message "VTC_KA_req" to all secondaries
if (!keepAliveReportOnlyFails)
console.log(`Sending KeepAlive messages to all secondary codecs...`)
Object.entries(secondariesKAStatus).forEach(([key, val]) => {
val.online = false;
})
await sendIntercodecMessage("VTC_KA_req");
//check for keepAlive replies KA_CHECK_REPLIES_TIMEOUT_MS miliseconds after sending
setTimeout(priKeepAliveStatuses, KA_CHECK_REPLIES_TIMEOUT_MS);
}
// Validate config settings
async function validate_config() {
let hasOverview = true;
// only allow CodecPro or CodecEQ with advanced AV integrator option key
const ProductPlatform = await xapi.Status.SystemUnit.ProductPlatform.get()
if (ProductPlatform == "Room Kit EQ") {
try {
console.log(`Is Codec EQ`);
const hasAVOptionInstalled = await xapi.Status.SystemUnit.Software.OptionKeys.AVIntegrator.get()
if (hasAVOptionInstalled != 'True') {
await disableMacro(`config validation fail: Platform ${ProductPlatform} without AV Integrator Option key not supported.`);
}
}
catch (e) {
await disableMacro(`config validation fail: Platform ${ProductPlatform} could not validate AV Option key.`);
}
}
else if (ProductPlatform == "Codec Pro") {
console.log(`Is Codec Pro`)
}
else {
await disableMacro(`config validation fail: Platform ${ProductPlatform} not supported.`);
}
if (module.name.replace('./', '') != 'divisible_room_primary')
await disableMacro(`config validation fail: macro name has changed to: ${module.name.replace('./', '')}. Please set back to: divisible_room_primary`);
if (SECONDARY_CODECS_USERNAME == '')
await disableMacro(`config validation fail: SECONDARY_CODECS credentials must be set. Current values: SECONDARY_CODECS_USERNAME: ${SECONDARY_CODECS_USERNAME} SECONDARY_CODECS_PASSWORD= ${SECONDARY_CODECS_PASSWORD}`);
// allow up to 8 analog mics
let allowedMics = [1, 2, 3, 4, 5, 6, 7, 8];
let allowedEthernetMics = []
// allow up to 8 ethernet mics with 8 lobes each
for (let i = 1; i <= 8; i++) {
for (let j = 1; j <= 8; j++) {
allowedEthernetMics.push((i * 10) + j)
}
}
let allowedUSBMics = []
// allow up to 4 USB mics
for (let i = 1; i <= 4; i++) {
allowedUSBMics.push(100 + i)
}
let allowedExternalMics = []
// allow up to 99 External mics
for (let i = 1; i <= 99; i++) {
allowedExternalMics.push(900 + i)
}
// only allow up to 8 analog microphones
if (config.monitorMics.length > 8)
await disableMacro(`config validation fail: config.monitorMics can only have up to 8 entries. Current value: ${config.MonitorMics} `);
// only allow up to 8 analog microphones
if (config.ethernetMics.length > 64)
await disableMacro(`config validation fail: config.ethernetMics can only have up to 64 entries. Current value: ${config.ethernetMics} `);
// only allow up to 8 analog microphones
if (config.usbMics.length > 4)
await disableMacro(`config validation fail: config.usbMics can only have up to 4 entries. Current value: ${config.usbMics} `);
if (config.externalMics.length > 99)
await disableMacro(`config validation fail: config.externalMics can only have up to 99 entries. Current value: ${config.ethernetMics} `);
if ((config.monitorMics.length + config.ethernetMics + config.usbMics.length + config.externalMics.length) < 1)
await disableMacro(`config validation fail: there must be at least one microphone configured between config.monitorMics, config.ethernetMics and config.usbMics.`);
// Check if using USB mic/input, that Echo control is turned on
if (config.usbMics.length > 0) {
const usbEchoControl = await xapi.config.Audio.Input.USBInterface[1].EchoControl.Mode.get()
if (usbEchoControl != 'On')
await disableMacro(`config validation fail: when using USB microphone input, Echo Control needs to be enabled. Only asynchronous USB devices are supported. Please enable and re-activate macro`);
}
// make sure the mics are within those specified in the monitorMics array
if (!config.monitorMics.every(r => allowedMics.includes(r)))
await disableMacro(`config validation fail: config.monitorMics can only have analog mic ids 1-8. Current value: ${config.monitorMics} `);
if (!config.ethernetMics.every(r => allowedEthernetMics.includes(r)))
await disableMacro(`config validation fail: config.ethernetMics can only include Ethernet mics 1-8(8 lobes each). Current value: ${config.ethernetMics} `);
if (!config.usbMics.every(r => allowedUSBMics.includes(r)))
await disableMacro(`config validation fail: config.usbMics can only include USB mics 1-4 (values 101-104). Current value: ${config.usbMics} `);
if (!config.externalMics.every(r => allowedExternalMics.includes(r)))
await disableMacro(`config validation fail: config.externalMics can only include external mics 01-99 (values 901-999). Current value: ${config.externalMics} `);
// check for duplicates in config.monitorMics
if (new Set(config.monitorMics).size !== config.monitorMics.length)
await disableMacro(`config validation fail: config.monitorMics cannot have duplicates. Current value: ${config.monitorMics} `);
if (new Set(config.ethernetMics).size !== config.ethernetMics.length)
await disableMacro(`config validation fail: config.ethernetMics cannot have duplicates. Current value: ${config.ethernetMics} `);
if (new Set(config.usbMics).size !== config.usbMics.length)
await disableMacro(`config validation fail: config.usbMics cannot have duplicates. Current value: ${config.usbMics} `);
// Check for valid audience mics CONFIGURED for the Presenter QA Mode feature
if (ALLOW_PRESENTER_QA_MODE)
if (!PRESENTER_QA_AUDIENCE_MIC_IDS.every(r => config.monitorMics.includes(r)) &&
!PRESENTER_QA_AUDIENCE_MIC_IDS.every(r => config.ethernetMics.includes(r)) &&
!PRESENTER_QA_AUDIENCE_MIC_IDS.every(r => config.externalMics.includes(r)) &&
!PRESENTER_QA_AUDIENCE_MIC_IDS.every(r => config.usbMics.includes(r)))
await disableMacro(`config validation fail: PRESENTER_QA_AUDIENCE_MIC_IDS can only specify values contained in config.monitorMics, config.ethernetMics or config.usbMics . Current values PRESENTER_QA_AUDIENCE_MIC_IDS: ${PRESENTER_QA_AUDIENCE_MIC_IDS}`);
hasOverview = false;
// add value 0 to allowedMics array to include overview composition
allowedMics.push(0)
// consolidate all allowed mics to check each composition for valid mics.
allowedMics = allowedMics.concat(allowedEthernetMics, allowedUSBMics, allowedExternalMics)
// now let's check each composition
for (let i = 0; i < config.compositions.length; i++) {
let compose = config.compositions[i];
// make sure each composition is marked JS_PRIMARY or JS_SECONDARY
if (![JS_PRIMARY, JS_SECONDARY, JS_AUXILIARY, JS_LOCAL].includes(compose.source)) await disableMacro(`config validation fail: composition named ${compose.name} should have a valid value for key 'source' (JS_PRIMARY, JS_SECONDARY, JS_AUXILIARY or JS_LOCAL).`);
// only allow up to 8 mics and at least 1 specified for each composition
if (compose.mics.length > 8 || compose.mics.length < 1)
await disableMacro(`config validation fail: mics for each composition can only have between 1 and 8 entries. Current value: ${compose.mics} `);
// make sure the mics are within those specified in the monitorMics array, plus 0 for overview
if (!compose.mics.every(r => allowedMics.includes(r)))
await disableMacro(`config validation fail: mics for each composition can only have mic ids 0-8, 11-88, 101-104 or 901-999. Current value: ${compose.mics} `);
// keep track that we have at least one composition with mics [0] to check at the end and that it is JS_PRIMARY sourced
if (JSON.stringify(compose.mics) == JSON.stringify([0]) && compose.source == JS_LOCAL) hasOverview = true;
}
// check that there was at least one Overview composition with mics==[0]
if (!hasOverview)
await disableMacro('config validation fail: no overview composition configured or it does not have source set to JS_LOCAL');
// all went well, can return true!
return true;
}
function delay(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
async function disableMacro(reason = 'N/A') {
console.warn(reason)
let act = `Disabling [${module.name.replace('./', '')}] in 10 seconds`
console.error({ Error: reason, Action: act })
await xapi.Command.UserInterface.Message.Alert.Display({ Title: '⚠️ Macro Error ⚠️', Text: `${reason}<p>${act}`, Duration: 9 });
await delay(10000);
await xapi.Command.Macros.Macro.Deactivate({ Name: module.name.replace('./', '') });
await delay(100);
await xapi.Command.Macros.Runtime.Restart();
}
async function checkOverviewPreset() {
console.log('Checking for existence of preset 30')
let pre_list = await xapi.Command.Camera.Preset.List(
{ CameraId: 1 })
let pre_exists = false;
if ('Preset' in pre_list) {
pre_list.Preset.forEach(preObj => {
if (preObj.PresetId == '30') pre_exists = true;
})
}
if (!pre_exists) {
console.log('Preset 30 does not exist, need to create it....')
await xapi.Command.Camera.PositionReset({ CameraId: 1 });
await delay(1000);
await xapi.Command.Camera.Preset.Store(
{ CameraId: 1, Name: "Overview", PresetId: 30 });
console.log('Preset 30 created')
}
}
const PANEL_room_combine_PIN = `<Extensions><Version>1.8</Version>
<Panel>
<Order>2</Order>
<PanelId>room_combine_PIN</PanelId>
<Type>Home</Type>
<Icon>Sliders</Icon>
<Color>#CF7900</Color>
<Name>Wall Sensor Override Control</Name>
<ActivityType>Custom</ActivityType>
</Panel>
</Extensions>`;
let panel_combine_split_str = `<Extensions><Version>1.8</Version>
<Panel>
<Order>2</Order>
<PanelId>panel_combine_split</PanelId>
<Origin>local</Origin>
<Type>Home</Type>
<Icon>Sliders</Icon>
<Color>#00D6A2</Color>
<Name>Room Combine Control</Name>
<ActivityType>Custom</ActivityType>
<Page>
<Name>Room Combine Control</Name>
<Row>
<Name>Row</Name>
<Widget>
<WidgetId>widget_text_combine</WidgetId>
<Name>Combine with selected rooms</Name>
<Type>Text</Type>
<Options>size=3;fontSize=normal;align=left</Options>
</Widget>
<Widget>
<WidgetId>widget_toggle_combine</WidgetId>
<Type>ToggleButton</Type>
<Options>size=1</Options>
</Widget>
</Row>`
let panel_combine_split_secondaries_str = `<Row>
<Name>Row</Name>
<Widget>
<WidgetId>notice_text</WidgetId>
<Name>When not combined, select at least one:</Name>
<Type>Text</Type>
<Options>size=3;fontSize=small;align=left</Options>
</Widget>
</Row>`
let secondaries_count = 0;
config.compositions.forEach(compose => {
if (compose.source == JS_SECONDARY && compose.codecAddress != '') {
secondaries_count += 1;
let theWidgetId = 'widget_tog_' + compose.codecAddress.replace(/\./g, "_")
let theName = compose.name
panel_combine_split_secondaries_str = panel_combine_split_secondaries_str + `<Row>
<Name>Row</Name>
<Widget>
<WidgetId>${theWidgetId}</WidgetId>
<Type>ToggleButton</Type>
<Options>size=1</Options>
</Widget>
<Widget>
<WidgetId>${theWidgetId}_text</WidgetId>
<Name>${theName}</Name>
<Type>Text</Type>
<Options>size=2;fontSize=small;align=center</Options>
</Widget>
</Row>`
}
})
// Only show selectable secondaries if there is more than one secondary.
if (secondaries_count > 1)
panel_combine_split_str = panel_combine_split_str + panel_combine_split_secondaries_str;
const PANEL_panel_combine_split = panel_combine_split_str + `
<Options>hideRowNames=1</Options>
</Page>
</Panel>
</Extensions>`;
//Declare your object for GMM communication
var secondaryCodecs = {};
//Run your init script asynchronously
async function init_intercodec() {
if (SECONDARY_CODECS_USERNAME != '') {
let stored_setStatus = {}
stored_setStatus = await GMM.read.global('JoinSplit_secondariesStatus').catch(async e => {
console.log("No initial JoinSplit_secondariesStatus global detected, using constants in macro to create new one")
return {};
})
let codecAddressArray = [];
config.compositions.forEach(compose => {
if (compose.source == JS_SECONDARY && compose.codecAddress != '') {
console.log(`Setting up connection to secondary codec with address ${compose.codecAddress}`);
//otherCodec[compose.codecAddress] = new GMM.Connect.IP(SECONDARY_CODECS_USERNAME, SECONDARY_CODECS_PASSWORD, compose.codecAddress)
codecAddressArray.push(compose.codecAddress);
console.log(`Creating secondaries keep alive status objects`);
secondariesKAStatus[compose.codecAddress] = { 'online': false };
console.log(`Creating secondaries status object for this secondary codec...`)
//make sure there is an entry for compose.codecAddress in secondariesStatus, if not, create a new one
if (!(compose.codecAddress in stored_setStatus)) { // this secondary codec info was not in permanent storage, create
secondariesStatus[compose.codecAddress] = { 'inCall': false, 'inPreview': false, 'online': false, 'selected': true };
}
else {
secondariesStatus[compose.codecAddress] = stored_setStatus[compose.codecAddress]; // copy over what was in storage, mainly the 'selected' state
secondariesStatus[compose.codecAddress]['inCall'] = false; // the inCall attribute should never be true when re-initting macro
secondariesStatus[compose.codecAddress]['inPreview'] = false; // the inPreview attribute should never be true when re-initting macro
}
connector_to_codec_map[compose.connectors[0]] = compose.codecAddress; // mapping connectors to IP of corresponding secondary
}
})
if (SECONDARIES_BOT_TOKEN == '')
secondaryCodecs = new GMM.Connect.IP(SECONDARY_CODECS_USERNAME, SECONDARY_CODECS_PASSWORD, codecAddressArray);
else
secondaryCodecs = new GMM.Connect.Webex(SECONDARIES_BOT_TOKEN, codecAddressArray);
//console.log(secondaryCodecs)
await GMM.write.global('JoinSplit_secondariesStatus', secondariesStatus).then(() => {
console.log({ Message: 'ChangeState', Action: 'Secondary codecs state stored.' })
})
}
// This schedules the keep alive messages to send from primary to secondaries, if enabled.
if (enableKeepAlive && true) {
if (KA_FREQUENCY_SECONDS >= 3 && KA_CHECK_REPLIES_TIMEOUT_MS >= 1000)
if ((KA_FREQUENCY_SECONDS * 1000) > KA_CHECK_REPLIES_TIMEOUT_MS + 1000)
setInterval(priSendKeepAlive, KA_FREQUENCY_SECONDS * 1000);
}
}
const localCallout = new GMM.Connect.Local(module.name.replace('./', ''))
/////////////////////////////////////////////////////////////////////////////////////////
// VARIABLES
/////////////////////////////////////////////////////////////////////////////////////////
// roomCombined keeps the current state of join/split for the codec. It is normally also reflected in
// permanent storage (GMMMemory macro) in the JoinSplit_combinedState global
var roomCombined = false;
// wallSensorOverride keeps the current state of the wall sensor functionality. If it is working well it is set to false
// If users detect a failure of the sensor, they will use the wall sensor override custom panel (PIN based or toggle button based)
// and from then on the macro will ignore the wall sensor input on GPIO PIN 1.
// The value of this boolean will always be reflected in permanent storage (GMMMemory macro) in the JoinSplit_wallSensorOverride global
// Once the wall sensor is repaired, somebody with access to this macro will have to manually edit the Memory_Storage macro file and set
// JoinSplit_wallSensorOverride to false and re-start the macro
var wallSensorOverride = false;
// below we are just initializing JoinSplit_primary_settings, no need to fill out values
var JoinSplit_primary_settings = {
VideoMonitors: ''
}
let micArrays = {};
for (var i in config.monitorMics) {
micArrays[config.monitorMics[i].toString()] = [0, 0, 0, 0];
}
for (var i in config.ethernetMics) {
micArrays[config.ethernetMics[i].toString()] = [0, 0, 0, 0];
}
for (var i in config.usbMics) {
micArrays[config.usbMics[i].toString()] = [0, 0, 0, 0];
}
let lowWasRecalled = false;
let lastActiveHighInput = 0;
let lastSourceDict = { SourceID: '1' }
let allowSideBySide = true;
let sideBySideTimer = null;
let InitialCallTimer = null;
let allowCameraSwitching = false;
let allowNewSpeaker = true;
let newSpeakerTimer = null;
let manual_mode = true;
let primarySingleScreen = false;
let micHandler = () => void 0;
let micHandlerEthernet = () => void 0;
let micHandlerUSB = () => void 0;
let overviewShowDouble = true; //Always setting overviewShowDouble to true so we always evaluate the overview composition now
let inSideBySide = false;
let presenterTracking = false;
let presenterDetected = true;
let presenterTrackConfigured = false;
let presenterQAKeepComposition = false;
let qaCompositionTimer = null;
let usb_mode = false;
let webrtc_mode = false;
let primaryInCall = false;
let primaryInPreview = false;
let secondariesStatus = {};
let connector_to_codec_map = {}
function secondariesInCall() { //now also check for secondaries in Preview mode
let result = false;
Object.entries(secondariesStatus).forEach(([key, val]) => {
if ((val.inCall || val.inPreview) && val.selected) result = true;
})
return result;
}
function secondariesOnline() {
let result = false;
Object.entries(secondariesStatus).forEach(([key, val]) => {
if (val.online) result = true;
})
return result;
}
let PRESENTER_QA_MODE = false
let ST_ACTIVE_CONNECTOR = 0;
let macroTurnedOnST = false;
let macroTurnedOffST = false;
let isOSTen = false;
let isOSEleven = false;
// Initial check for the Video Monitor configuration
async function check4_Video_Monitor_Config() {
const videoMonitorConfig = await xapi.Config.Video.Monitors.get()
return new Promise((resolve, reject) => {
if (videoMonitorConfig != 'Auto' && (videoMonitorConfig != 'Triple' && videoMonitorConfig != 'TriplePresentationOnly')) {
resolve(videoMonitorConfig)
} else {
reject(new Error('xConfiguration Video Monitors can not be set to Auto, Triple or TriplePresentationOnly for the Join/Split macro to work properly'))
}
})
}
async function getPresetCamera(prID) {
const value = await xapi.Command.Camera.Preset.Show({ PresetId: prID });
return (value.CameraId)
}
async function check4_Minimum_Version_Required(minimumOs) {
const reg = /^\D*(?<MAJOR>\d*)\.(?<MINOR>\d*)\.(?<EXTRAVERSION>\d*)\.(?<BUILDID>\d*).*$/i;
const minOs = minimumOs;
const os = await xapi.Status.SystemUnit.Software.Version.get();
console.log(os)
const x = (reg.exec(os)).groups;
const y = (reg.exec(minOs)).groups;
if (parseInt(x.MAJOR) > parseInt(y.MAJOR)) return true;
if (parseInt(x.MAJOR) < parseInt(y.MAJOR)) return false;
if (parseInt(x.MINOR) > parseInt(y.MINOR)) return true;
if (parseInt(x.MINOR) < parseInt(y.MINOR)) return false;
if (parseInt(x.EXTRAVERSION) > parseInt(y.EXTRAVERSION)) return true;
if (parseInt(x.EXTRAVERSION) < parseInt(y.EXTRAVERSION)) return false;
if (parseInt(x.BUILDID) > parseInt(y.BUILDID)) return true;
if (parseInt(x.BUILDID) < parseInt(y.BUILDID)) return false;
return false;
}
/**
* The following functions allow the ability to set the Pins High or Low
**/
function setGPIOPin2ToHigh() {
xapi.command('GPIO ManualState Set', { Pin2: 'High' }).catch(e => console.debug(e));
console.log('Pin 2 has been set to High; MUTE is inactive');
}
function setGPIOPin2ToLow() {
xapi.command('GPIO ManualState Set', { Pin2: 'Low' }).catch(e => console.debug(e));
console.log('Pin 2 has been set to Low; MUTE is active');
}
function setGPIOPin3ToHigh() {
xapi.command('GPIO ManualState Set', { Pin3: 'High' }).catch(e => console.debug(e));
console.log('Pin 3 has been set to High; STANDBY is inactive');
}
function setGPIOPin3ToLow() {
xapi.command('GPIO ManualState Set', { Pin3: 'Low' }).catch(e => console.debug(e));
console.log('Pin 3 has been set to Low; STANDBY is active');
}
function setGPIOPin4ToHigh() {
xapi.command('GPIO ManualState Set', { Pin4: 'High' }).catch(e => console.debug(e));
console.log('Pin 4 has been set to High; Rooms are Standalone');
}
function setGPIOPin4ToLow() {
xapi.command('GPIO ManualState Set', { Pin4: 'Low' }).catch(e => console.debug(e));
console.log('Pin 4 has been set to Low; Rooms are Combined');
}
async function setCombinedMode(combinedValue) {
roomCombined = combinedValue;
await GMM.write.global('JoinSplit_combinedState', roomCombined).then(() => {
console.log({ Message: 'ChangeState', Action: 'Combined state stored.' })
})
}
async function setWallSensorOverride(overrideValue) {
wallSensorOverride = overrideValue;
await GMM.write.global('JoinSplit_wallSensorOverride', wallSensorOverride).then(() => {
console.log({ Message: 'ChangeState', Action: 'Wall Sensor Override state stored.' })
})
}
/**
* This will initialize the room state to Combined or Divided based on the setting in Memory Macro (persistent storage)
**/
async function initialCombinedJoinState() {
// Change all these to whatever is needed to trigger on the Primary when it goes into combined
if (roomCombined) {
console.log('Primary Room is in Combined Mode');
if (true) {
primaryCombinedMode();
if (await isCodecPro())
if (USE_GPIO_INTERCODEC) setGPIOPin4ToLow();
if (!USE_WALL_SENSOR) {
xapi.command('UserInterface Extensions Widget SetValue', { WidgetId: 'widget_toggle_combine', Value: 'On' });
}
}
setCombinedMode(true);
} else {
console.log('Primary Room is in Divided Mode');
if (true) {
setPrimaryDefaultConfig();
if (USE_GPIO_INTERCODEC) setGPIOPin4ToHigh();
}
setCombinedMode(false);
}
}
/**
* This will initialize the room state to Combined or Divided based on the Pin 4 set by Primary
**/
async function checkCombinedStateSecondary() {
if (USE_GPIO_INTERCODEC) Promise.all([xapi.status.get('GPIO Pin 4')]).then(promises => {
let [pin4] = promises;
console.log('Pin4: ' + pin4.State);
// Change all these to whatever is needed to trigger on the Secondary when it goes into combined
if (pin4.State === 'Low' && (!secondarySelected)) {
console.log('Secondary Room is in Combined Mode');
setSecondaryToCombined();
} else {
if (!secondarySelected) console.log('Secondary Room is not selected in Primary...');
console.log('Secondary Room is in Divided Mode');
setSecondaryToSplit();
// since we are in secondary codec and in split configuration, we need to
// prepare to do basic switching to support PresenterTrack QA mode.
init_switching();
}
}).catch(e => console.debug(e));
else { // not using GPIO PINs
if (roomCombined) {
console.log('Secondary was set to Combined Mode in permanent storage');
setSecondaryToCombined();
}
else {
console.log('Secondary set to Divided Mode in permanent storage');
setSecondaryToSplit();
}
}
}
/**
* The following functions display a message on the touch panel to alert the users
* that the rooms are either being separated or joined together
**/
function alertJoinedScreen() {
xapi.command('UserInterface Message Alert Display', {
Title: 'Combining Rooms ...',
Text: 'Please wait',
Duration: 10,
});
}
function alertSplitScreen() {
xapi.command('UserInterface Message Alert Display', {
Title: 'Dividing Rooms ...',
Text: 'Please wait',
Duration: 10,
});
}
/**
* The following functions display a message on the touch panel to alert the users
* that the rooms are either being selected or unselected
**/
function alertSelectedScreen() {
xapi.command('UserInterface Message Alert Display', {
Title: 'Selecting room to combine ...',