RoomDialog.tsx 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. import React, { useRef } from "react";
  2. import { copyTextToSystemClipboard } from "../../clipboard";
  3. import { Dialog } from "../../components/Dialog";
  4. import {
  5. clipboard,
  6. start,
  7. stop,
  8. share,
  9. shareIOS,
  10. shareWindows,
  11. } from "../../components/icons";
  12. import { ToolButton } from "../../components/ToolButton";
  13. import { t } from "../../i18n";
  14. import "./RoomDialog.scss";
  15. import Stack from "../../components/Stack";
  16. import { AppState } from "../../types";
  17. const getShareIcon = () => {
  18. const navigator = window.navigator as any;
  19. const isAppleBrowser = /Apple/.test(navigator.vendor);
  20. const isWindowsBrowser = navigator.appVersion.indexOf("Win") !== -1;
  21. if (isAppleBrowser) {
  22. return shareIOS;
  23. } else if (isWindowsBrowser) {
  24. return shareWindows;
  25. }
  26. return share;
  27. };
  28. const RoomDialog = ({
  29. handleClose,
  30. activeRoomLink,
  31. username,
  32. onUsernameChange,
  33. onRoomCreate,
  34. onRoomDestroy,
  35. setErrorMessage,
  36. theme,
  37. }: {
  38. handleClose: () => void;
  39. activeRoomLink: string;
  40. username: string;
  41. onUsernameChange: (username: string) => void;
  42. onRoomCreate: () => void;
  43. onRoomDestroy: () => void;
  44. setErrorMessage: (message: string) => void;
  45. theme: AppState["theme"];
  46. }) => {
  47. const roomLinkInput = useRef<HTMLInputElement>(null);
  48. const copyRoomLink = async () => {
  49. try {
  50. await copyTextToSystemClipboard(activeRoomLink);
  51. } catch (error: any) {
  52. setErrorMessage(error.message);
  53. }
  54. if (roomLinkInput.current) {
  55. roomLinkInput.current.select();
  56. }
  57. };
  58. const shareRoomLink = async () => {
  59. try {
  60. await navigator.share({
  61. title: t("roomDialog.shareTitle"),
  62. text: t("roomDialog.shareTitle"),
  63. url: activeRoomLink,
  64. });
  65. } catch (error: any) {
  66. // Just ignore.
  67. }
  68. };
  69. const selectInput = (event: React.MouseEvent<HTMLInputElement>) => {
  70. if (event.target !== document.activeElement) {
  71. event.preventDefault();
  72. (event.target as HTMLInputElement).select();
  73. }
  74. };
  75. const renderRoomDialog = () => {
  76. return (
  77. <div className="RoomDialog-modal">
  78. {!activeRoomLink && (
  79. <>
  80. <p>{t("roomDialog.desc_intro")}</p>
  81. <p>{`🔒 ${t("roomDialog.desc_privacy")}`}</p>
  82. <div className="RoomDialog-sessionStartButtonContainer">
  83. <ToolButton
  84. className="RoomDialog-startSession"
  85. type="button"
  86. icon={start}
  87. title={t("roomDialog.button_startSession")}
  88. aria-label={t("roomDialog.button_startSession")}
  89. showAriaLabel={true}
  90. onClick={onRoomCreate}
  91. />
  92. </div>
  93. </>
  94. )}
  95. {activeRoomLink && (
  96. <>
  97. <p>{t("roomDialog.desc_inProgressIntro")}</p>
  98. <p>{t("roomDialog.desc_shareLink")}</p>
  99. <div className="RoomDialog-linkContainer">
  100. <Stack.Row gap={2}>
  101. {"share" in navigator ? (
  102. <ToolButton
  103. type="button"
  104. icon={getShareIcon()}
  105. title={t("labels.share")}
  106. aria-label={t("labels.share")}
  107. onClick={shareRoomLink}
  108. />
  109. ) : null}
  110. <ToolButton
  111. type="button"
  112. icon={clipboard}
  113. title={t("labels.copy")}
  114. aria-label={t("labels.copy")}
  115. onClick={copyRoomLink}
  116. />
  117. </Stack.Row>
  118. <input
  119. value={activeRoomLink}
  120. readOnly={true}
  121. className="RoomDialog-link"
  122. ref={roomLinkInput}
  123. onPointerDown={selectInput}
  124. />
  125. </div>
  126. <div className="RoomDialog-usernameContainer">
  127. <label className="RoomDialog-usernameLabel" htmlFor="username">
  128. {t("labels.yourName")}
  129. </label>
  130. <input
  131. id="username"
  132. value={username || ""}
  133. className="RoomDialog-username TextInput"
  134. onChange={(event) => onUsernameChange(event.target.value)}
  135. onKeyPress={(event) => event.key === "Enter" && handleClose()}
  136. />
  137. </div>
  138. <p>
  139. <span role="img" aria-hidden="true" className="RoomDialog-emoji">
  140. {"🔒"}
  141. </span>{" "}
  142. {t("roomDialog.desc_privacy")}
  143. </p>
  144. <p>{t("roomDialog.desc_exitSession")}</p>
  145. <div className="RoomDialog-sessionStartButtonContainer">
  146. <ToolButton
  147. className="RoomDialog-stopSession"
  148. type="button"
  149. icon={stop}
  150. title={t("roomDialog.button_stopSession")}
  151. aria-label={t("roomDialog.button_stopSession")}
  152. showAriaLabel={true}
  153. onClick={onRoomDestroy}
  154. />
  155. </div>
  156. </>
  157. )}
  158. </div>
  159. );
  160. };
  161. return (
  162. <Dialog
  163. small
  164. onCloseRequest={handleClose}
  165. title={t("labels.liveCollaboration")}
  166. theme={theme}
  167. >
  168. {renderRoomDialog()}
  169. </Dialog>
  170. );
  171. };
  172. export default RoomDialog;