index.js 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. /*function jsonPost(url, data)
  2. {
  3. return new Promise((resolve, reject) => {
  4. var x = new XMLHttpRequest();
  5. x.onerror = () => reject(new Error('jsonPost failed'))
  6. //x.setRequestHeader('Content-Type', 'application/json');
  7. x.open("POST", url, true);
  8. x.send(JSON.stringify(data))
  9. x.onreadystatechange = () => {
  10. if (x.readyState == XMLHttpRequest.DONE && x.status == 200){
  11. resolve(JSON.parse(x.responseText))
  12. }
  13. else if (x.status != 200){
  14. reject(new Error('status is not 200'))
  15. }
  16. }
  17. })
  18. }*/
  19. async function jsonPost(url, data){
  20. try {
  21. let response = await fetch(url, {
  22. method: 'POST',
  23. body: JSON.stringify(data)
  24. })
  25. if (response.ok) {
  26. let json = await response.json()
  27. return json}
  28. } catch (error) {
  29. console.log('ошибка: ' + error) //ловим ошибки
  30. }
  31. }
  32. //событие по кнопке SEND с проверкой заполненности полей
  33. sendButton.onclick = sendAndCheck
  34. async function sendMessage(nick, message) {
  35. if (nick && message) {
  36. jsonPost("http://students.a-level.com.ua:10012", { func: 'addMessage', nick, message })
  37. }
  38. }
  39. async function getMessages (id) {
  40. const res = await jsonPost("http://students.a-level.com.ua:10012", { func: "getMessages", messageId: id}).then(res => res)
  41. nextMessageId = res.nextMessageId
  42. for(const msg of res.data){
  43. const msgDiv = document.createElement('div')
  44. chat.prepend(msgDiv)
  45. const userNick = document.createElement('span')
  46. msgDiv.append(userNick)
  47. userNick.innerText = msg.nick + ': '
  48. userNick.classList.add('nick')
  49. const userMessage = document.createElement('span')
  50. msgDiv.append(userMessage)
  51. userMessage.innerText = msg.message
  52. const timeMessage = document.createElement('span')
  53. msgDiv.prepend(timeMessage)
  54. const msgTime = new Date(msg.timestamp)
  55. let timeFormatHhMmSs = ''
  56. timeFormatHhMmSs += msgTime.getFullYear() + '.'
  57. if (msgTime.getMonth() < 9) timeFormatHhMmSs += '0' + (msgTime.getMonth()+1) + '.'
  58. else timeFormatHhMmSs += (msgTime.getMonth()+1) + '.'
  59. if (msgTime.getDate() < 10) timeFormatHhMmSs += '0' + msgTime.getDate() + ' '
  60. else timeFormatHhMmSs += msgTime.getDate() + ' '
  61. if (msgTime.getHours() < 10) timeFormatHhMmSs += '0' + msgTime.getHours() + ':'
  62. else timeFormatHhMmSs += msgTime.getHours() + ':'
  63. if (msgTime.getMinutes() < 10) timeFormatHhMmSs += '0' + msgTime.getMinutes() + ':'
  64. else timeFormatHhMmSs += msgTime.getMinutes() + ':'
  65. if (msgTime.getSeconds() < 10) timeFormatHhMmSs += '0' + msgTime.getSeconds()
  66. else timeFormatHhMmSs += msgTime.getSeconds()
  67. timeMessage.innerText = timeFormatHhMmSs
  68. }
  69. }
  70. async function sendAndCheck(){
  71. await sendMessage(userName.value, userMessage.value)
  72. getMessages (nextMessageId)
  73. }
  74. let nextMessageId = 0
  75. //первоначальная загрузка и вывод на экран сообщений из чата
  76. //getMessages (nextMessageId)
  77. //каждые две секунды проверяем новые сообщения, если есть, добавляем их к тем что уже на экране
  78. //setInterval( () => getMessages (nextMessageId), 2000 )
  79. function delay(ms){ //промисифицированная функция для создания промиса, который резолвится через определенное время
  80. //функция-исполнитель, принимает два параметра - функции для передачи промису результата работы
  81. function executor(fulfill, reject){ //reject объявлен для вида.
  82. setTimeout(() => fulfill(ms), ms) //setTimeout запустит функцию, которая запустит fulfill через ms миллисекунд. Результатом промиса будет время задержки
  83. }
  84. //возврщаем новый промис, передав в него executor. Промис тут же его запускает, передав в него колбэки для управления состоянием
  85. return new Promise(executor)
  86. }
  87. async function checkLoop() {
  88. while(true) {
  89. getMessages (nextMessageId)
  90. await delay(2000)
  91. console.log('2сек')
  92. }
  93. }
  94. checkLoop()
  95. /*SWAPI Links
  96. Напишите промисифицированную функцию, которая принимать параметром ссылку на swapi.dev (например, https://swapi.dev/api/people/20) и скачивать всю информацию по ссылке, включая вложенные ссылки. Все ссылки внутри скачанного объекта должны заменяться на скачанные объекты. Полученный общий объект должен стать результатом промиса, который возвращает функция.
  97. Например, ссылка, приведенная выше позволяет информацию о Йода Мастере скачать:
  98. Ваша функция должна перебрать весь объект, найти там ссылки и массивы со ссылками, скачать эти ссылки и сложить это всё в общий объект, заместив строки со ссылками. Получится что-то такое:
  99. Ваша функция может быть обычной (с цепочкой .then) или асинхронной, по желанию. Главное-вовремя зарезолвить промис-результат функции. Сделайте запросы по вложенным в объект ссылкам параллельно. Для проверки корректности работы функции - выводите в консоль JSON-строку, вместо оригинального объекта. Это позволит убедиться в том, что ваша функция резолвит промис именно тогда, когда все данные по вложенным ссылкам скачаны:
  100. swapiLinks("https://swapi.dev/api/people/20/")
  101. .then(yodaWithLinks => console.log(JSON.stringify(yodaWithLinks, null, 4)))
  102. Если же вы просто выведете объект (console.log(yodaWithLinks)), то за счет ссылочности объектов в Javascript, в него можно добавить что-то уже после вывода. Раскрыв этот объект при отладке в консоли спустя пару секунд после вывода, вы увидите что всё ок. Однако это не значит, что все реально ОК на момент работы console.log.
  103. Вывод JSON покажет вам правду - состояние объекта на момент резолва промиса-результата вашей функции.
  104. В теле функции new Promise не нужен.*/
  105. swapiLinks("https://swapi.dev/api/people/20/")
  106. .then(yodaWithLinks => console.log(JSON.stringify(yodaWithLinks, null, 4)))
  107. async function downloadLink(link){
  108. let jsonObj = await fetch(link).then(res => res.json())
  109. return jsonObj
  110. }
  111. async function swapiLinks(link) {
  112. const allP = [];
  113. try {
  114. let jsonObj = await downloadLink(link)
  115. for (const key in jsonObj){
  116. let regexp = /^\s*https?:\/\/./i; //тут проверяем является ли значение ссылкой
  117. let value = jsonObj[key];
  118. if (regexp.test(value) && typeof value === "string") { //если значение ссылка но не массив
  119. allP.push(downloadLink(value).then(resultObj => jsonObj[key] = resultObj))
  120. } else if (Array.isArray(value)) {
  121. allP.push(Promise.all(
  122. value.map(downloadLink)
  123. ).then(
  124. resultObj => {
  125. jsonObj[key] = resultObj
  126. }
  127. ))
  128. }
  129. }
  130. return Promise.all(allP)
  131. .then(() => jsonObj)
  132. } catch (error) {
  133. console.log("Ошибка: " + error)
  134. }
  135. }
  136. /*domEventPromise
  137. Реализуйте промисифицированную функцию, которая резолвит промис по событию в DOM:
  138. function domEventPromise(element, eventName){
  139. function executor(resolve){
  140. //happy hacking
  141. }
  142. return new Promise(executor)
  143. }
  144. domEventPromise(knopka, 'click').then( e => console.log('event click happens', e))
  145. Функция должна:
  146. используя addEventListener повесить свой обработчик события на DOM element событие eventName
  147. по событию зарезолвить промис отдав в качестве результата объект события
  148. убрать обработчик с DOM-элемента, используя removeEventListener.
  149. Убедитесь, что действительно убираете обработчик после резолва промиса. Он больше не нужен.*/
  150. function domEventPromise(element, eventName){
  151. function executor(resolve){
  152. //happy hacking
  153. function myEvent(event) {
  154. element.removeEventListener(eventName, myEvent)
  155. resolve(event)
  156. }
  157. element.addEventListener(eventName, myEvent)
  158. }
  159. return new Promise(executor)
  160. }
  161. let div = document.createElement('div')
  162. document.body.append(div)
  163. div.style.width = '100px'
  164. div.style.height = '100px'
  165. div.style.backgroundColor = 'red'
  166. domEventPromise(div, 'click').then( e => console.log('event click happens', e))