HW16.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. // fetch basic
  2. // Исследуйте сайт swapi.dev, рассмотрите JSON - ы, которые предоставляются этим сервисом(например: https://swapi.dev/api/people/1/, https://swapi.dev/api/people/2/, https://swapi.dev/api/starships/12/
  3. //С помощью следующего кода считать и вывести информацию о Люке Скайвокере:
  4. // Напишите функцию для отображения любого JSON в форме таблицы.Функция должна принимать два параметра:
  5. // DOM - элемент, в котором строится таблица
  6. // JSON, который нужно отобразить.
  7. {
  8. fetch('https://swapi.dev/api/people/1/')
  9. .then(res => res.json())
  10. .then(luke => {
  11. starWarsData(document.body, luke)
  12. })
  13. function starWarsData(parent, data) {
  14. const table = document.createElement('table')
  15. table.style.cssText = `
  16. border: 1px solid black;
  17. border-collapse: collapse;`
  18. parent.append(table)
  19. for (let [key, value] of Object.entries(data)) {
  20. const row = document.createElement('tr')
  21. table.append(row)
  22. let col = document.createElement('td')
  23. col.innerText = `${key}`
  24. col.style.cssText = `
  25. border: 1px solid black;
  26. padding: 5px 10px;`
  27. row.append(col)
  28. col = document.createElement('td')
  29. col.innerText = `${value}`
  30. col.style.cssText = `
  31. border: 1px solid black;
  32. padding: 5px 10px;`
  33. row.append(col)
  34. }
  35. }
  36. }
  37. // fetch improved
  38. // Расширить функцию отображения:
  39. // Если одно из полей строка или массив.
  40. // Если строки или строка содержат в себе https://swapi.dev/api/
  41. // То выводить вместо текста строки кнопку, при нажатии на которую:
  42. // делается fetch данных по этой ссылке
  43. // функция отображения запускает сама себя(рекурсивно) для отображения новых данных в том же элементе.
  44. {
  45. function starWarsData(parent, data) {
  46. // create Table
  47. const table = document.createElement('table')
  48. table.style.cssText = `
  49. border: 1px solid black;
  50. border-collapse: collapse;
  51. margin: 10px;`
  52. parent.append(table)
  53. // create table Header
  54. const tableHead = document.createElement('tr')
  55. table.append(tableHead)
  56. const colHead = document.createElement('td')
  57. colHead.innerText = Object.values(data)[0]
  58. colHead.colSpan = 2
  59. colHead.style.cssText = `
  60. background-color: #a0a0a0;
  61. text-align: center;
  62. font-size: 1.5em;
  63. font-weight: 900;
  64. line-height: 1.5em;`
  65. tableHead.append(colHead)
  66. // create table Body
  67. for (const [key, values] of Object.entries(data)) {
  68. // create row
  69. const row = document.createElement('tr')
  70. table.append(row)
  71. // create first cell with key
  72. let col = document.createElement('td')
  73. col.innerText = key
  74. col.style.cssText = `
  75. border: 1px solid black;
  76. padding: 5px 10px;`
  77. row.append(col)
  78. // create second cell
  79. col = document.createElement('td')
  80. col.style.cssText = `
  81. border: 1px solid black;
  82. padding: 5px 10px;`
  83. // cheking string or array(object)
  84. if (typeof values !== 'object') {
  85. if (!(values.toString().includes('swapi.dev/api'))) {
  86. col.innerText = values
  87. col.style.cssText = `
  88. border: 1px solid black;
  89. padding: 5px 10px;`
  90. row.append(col)
  91. } else {
  92. row.append(col)
  93. createBtn(col, values)
  94. }
  95. } else {
  96. row.append(col)
  97. for (const value of values) {
  98. createBtn(col, value)
  99. }
  100. }
  101. }
  102. // create button
  103. function createBtn(parent, param) {
  104. const btn = document.createElement('button')
  105. btn.innerText = param
  106. btn.style.margin = '5px'
  107. parent.append(btn)
  108. btn.onclick = () => {
  109. fetch(param)
  110. .then(res => res.json())
  111. .then(item => {
  112. starWarsData(parent, item)
  113. })
  114. btn.disabled = true
  115. }
  116. }
  117. }
  118. // start parsing
  119. fetch('https://swapi.dev/api/people/1/')
  120. .then(res => res.json())
  121. .then(luke => {
  122. starWarsData(document.body, luke)
  123. })
  124. }
  125. // race
  126. // Используя Promise.race запустите запрос на API(swapi.dev) параллельно с delay.По результату определите, что было быстрее, запрос по сети, или определенный интервал времени.Подберите параметр delay так, что бы результат был неизвестен изначально, и при многократных запусках быстрее был то delay, то myfetch.
  127. {
  128. const delay = ms => new Promise(ok => setTimeout(() => ok(ms), ms))
  129. const promiseDelay = delay(Math.random() * 250)
  130. .then(result => result)
  131. const promiseApi = fetch('https://swapi.dev/api/people/1/')
  132. .then(res => res.json())
  133. .then(result => result)
  134. Promise.race([promiseDelay, promiseApi]).then((value) => console.log(value))
  135. }
  136. // Promisify: confirm
  137. // Промисифицируйте confirm.Напишите функцию, которая возвращает промис, который переходит в состояние fulfilled при нажатии "OK" и реджектится при нажатии "Cancel".Функция должна принимать строку для confirm:
  138. // Промисификация не делает confirm неблокирующей функцией.Данное задание имеет только образовательный смысл.
  139. {
  140. function confirmPromise(text) {
  141. function executor(resolve, reject) {
  142. confirm(text) ? resolve() : reject()
  143. }
  144. return new Promise(executor)
  145. }
  146. confirmPromise('Промисы это сложно?').then(() => console.log('не так уже и сложно'), () => console.log('respect за усидчивость и внимательность'))
  147. }
  148. // Promisify: prompt
  149. // Аналогично предыдующему заданию промисифицируйте prompt.В случае нажатия "ОК" промис резолвится и его результатом становится текст, введенный пользователем в окне prompt.В случае нажатия "Отмены" - промис реджектится.Параметром функции будет текст сообщения в prompt
  150. {
  151. function promptPromise(text) {
  152. function executor(resolve, reject) {
  153. let question = prompt(text)
  154. question !== null && question !== '' ? resolve(question) : reject()
  155. }
  156. return new Promise(executor)
  157. }
  158. promptPromise("Как тебя зовут?").then(name => console.log(`Тебя зовут ${name}`),
  159. () => console.log('Ну зачем морозиться, нормально же общались'))
  160. }
  161. // Promisify: LoginForm
  162. // Промисифицируйте конструктор LoginForm.В промисифицированную функцию передается DOM - элемент - родитель для формы, В колбэке, предназначенном для получения логина и пароля в момент нажатия кнопки "Login...", который вы назначаете в объекте LoginForm, зарезолвите промис.Результатом промиса должен быть объект с ключами login и password, ключи должны содержать значения полей ввода.
  163. {
  164. function LoginForm(parent) {
  165. function Password(parent, open) {
  166. // create password fild
  167. const inputPassword = document.createElement('input')
  168. inputPassword.type = 'password'
  169. inputPassword.placeholder = 'Insert password'
  170. parent.append(inputPassword)
  171. // create checkbox
  172. const inputCheckbox = document.createElement('input')
  173. inputCheckbox.type = 'checkbox'
  174. inputCheckbox.checked = false
  175. parent.append(inputCheckbox)
  176. // create getters
  177. this.getValue = () => inputPassword.value
  178. this.getOpen = () => inputCheckbox.checked
  179. // create setters
  180. this.setValue = (value) => inputPassword.value = value
  181. this.setOpen = (open) => {
  182. if (open === true) {
  183. inputPassword.type = 'text'
  184. inputCheckbox.checked = true
  185. }
  186. if (open === false) {
  187. inputPassword.type = 'password'
  188. inputCheckbox.checked = false
  189. }
  190. return inputPassword.type, inputCheckbox.checked
  191. }
  192. // starting onChange
  193. inputPassword.addEventListener('input', () => {
  194. this.onChange(inputPassword.value)
  195. })
  196. // starting onOpenChange + change inputPasswor hide/see
  197. inputCheckbox.addEventListener('change', () => {
  198. this.setOpen(inputCheckbox.checked),
  199. this.onOpenChange(inputCheckbox.checked)
  200. })
  201. }
  202. function Login(parent) {
  203. // create login fild
  204. const inputLogin = document.createElement('input')
  205. inputLogin.type = 'text'
  206. inputLogin.placeholder = 'Insert login'
  207. parent.append(inputLogin)
  208. // create getters
  209. this.getValue = () => inputLogin.value
  210. // create setters
  211. this.setValue = (value) => inputLogin.value = value
  212. // starting onChange
  213. inputLogin.addEventListener('input', () => {
  214. this.onChange(inputLogin.value)
  215. })
  216. }
  217. // create form with password and login
  218. const form = document.createElement('form')
  219. parent.append(form)
  220. const loginLabel = document.createElement('label')
  221. loginLabel.innerText = 'Login:'
  222. form.append(loginLabel)
  223. let breakSymbol = document.createElement('br')
  224. form.append(breakSymbol)
  225. const login = new Login(form)
  226. breakSymbol = document.createElement('br')
  227. form.append(breakSymbol)
  228. const passwordLabel = document.createElement('label')
  229. passwordLabel.innerText = 'Password:'
  230. form.append(passwordLabel)
  231. breakSymbol = document.createElement('br')
  232. form.append(breakSymbol)
  233. const password = new Password(form, true)
  234. breakSymbol = document.createElement('br')
  235. form.append(breakSymbol)
  236. const confirmBtn = document.createElement('button')
  237. confirmBtn.innerText = 'Confirm'
  238. confirmBtn.type = 'button'
  239. confirmBtn.style.marginTop = '10px'
  240. confirmBtn.disabled = true
  241. form.append(confirmBtn)
  242. // change confirmBtn.disabled
  243. function checkButton() {
  244. if (login.getValue() !== '' && password.getValue() !== '') {
  245. confirmBtn.disabled = false
  246. } else {
  247. confirmBtn.disabled = true
  248. }
  249. return confirmBtn.disabled
  250. }
  251. checkButton()
  252. // listening password and login
  253. login.onChange = password.onChange = checkButton
  254. // create getters
  255. this.getPasswordValue = () => password.getValue()
  256. this.getPasswordOpen = () => password.getOpen()
  257. this.getLoginValue = () => login.getValue()
  258. this.getButtonStatus = () => confirmBtn.disabled
  259. // create setters
  260. this.setPasswordValue = (value) => password.setValue(value)
  261. this.setLoginValue = (value) => login.setValue(value)
  262. this.setPasswordOpen = (status) => password.setOpen(status)
  263. // create callbacks
  264. this.onOpenChange = open => password.onOpenChange = open
  265. // on lobig Button
  266. confirmBtn.addEventListener('click', () => {
  267. this.confirmBtnListener()
  268. })
  269. }
  270. function loginPromise(parent) {
  271. function executor(resolve, reject) {
  272. const form = new LoginForm(parent)
  273. form.confirmBtnListener = () => resolve({
  274. 'login': form.getLoginValue(),
  275. 'password': form.getPasswordValue()
  276. })
  277. }
  278. return new Promise(executor)
  279. }
  280. loginPromise(document.body).then(({ login, password }) => console.log(`Вы ввели ${login} и ${password}`))
  281. }