index.html 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8" />
  5. <title>Title</title>
  6. <style>
  7. body {
  8. background: oldlace;
  9. }
  10. img {
  11. width: 150px;
  12. margin-left: 20px;
  13. }
  14. h1 {
  15. text-align: center;
  16. color: brown;
  17. }
  18. #div {
  19. display: flex;
  20. flex-direction: column;
  21. justify-content: center;
  22. margin: 0 auto;
  23. align-items: center;
  24. }
  25. #img-block {
  26. display: flex;
  27. align-items: center;
  28. }
  29. #textBlock {
  30. display: flex;
  31. }
  32. p {
  33. padding-right: 20px;
  34. color: brown;
  35. font-size: 20px;
  36. }
  37. button {
  38. margin: 0 auto;
  39. width: 200px;
  40. height: 40px;
  41. background-color: bisque;
  42. box-shadow: 6px 6px 20px rgba(0, 0, 0, 0.3);
  43. }
  44. button:hover {
  45. box-shadow: 0px;
  46. }
  47. .buttomP {
  48. color: red;
  49. margin: 0;
  50. text-align: center;
  51. font-size: 20px;
  52. padding-left: 15px;
  53. }
  54. #list,
  55. #amount,
  56. #sum {
  57. margin-bottom: 20px;
  58. width: 300px;
  59. height: 20px;
  60. background-color: bisque;
  61. }
  62. #amount,
  63. #sum {
  64. width: 294px;
  65. }
  66. #text {
  67. padding-left: 60px;
  68. }
  69. </style>
  70. </head>
  71. <body>
  72. <h1>ГОПОТА МАРКЕТ</h1>
  73. <div id="div">
  74. <div id="img-block">
  75. <div>
  76. <img
  77. src=""
  78. alt=""
  79. />
  80. <p id="text">20 денег</p>
  81. </div>
  82. <div>
  83. <img
  84. src="https://euro-opt.shop/4119-large_default/chipsy-pringles-original-pringls-original-165g.jpg"
  85. alt=""
  86. width=""
  87. />
  88. <p id="text">10 денег</p>
  89. </div>
  90. <div>
  91. <img
  92. src="https://img3.zakaz.ua/1.1620828397.ad72436478c_2021-05-12_Tatiana/1.1620828397.SNCPSG10.obj.0.1.jpg.oe.jpg.pf.jpg.1350nowm.jpg.1350x.jpg"
  93. alt=""
  94. />
  95. <p id="text">30 денег</p>
  96. </div>
  97. </div>
  98. <div id="textBlock">
  99. <p>В наличии:</p>
  100. <br />
  101. </div>
  102. <p>Чё надо?</p>
  103. <select name="store" id="list" class="s-10">
  104. <option value="пиво">Пиво</option>
  105. <option value="чипсы">Чипсы</option>
  106. <option value="сиги">Сиги</option>
  107. </select>
  108. <p>Скок надо?</p>
  109. <input type="number" id="sum" class="i-3" />
  110. <p>Давай бабки! Сдачи нету!</p>
  111. <input type="number" id="amount" class="i-2" />
  112. <button id="button" class="b-2">
  113. <p class="buttomP">КУПИТЬ</p>
  114. </button>
  115. <p class="out-2" id="kassa">КАССА: 0 денег</p>
  116. </div>
  117. <script>
  118. setTimeout(function () {
  119. let parent = document.querySelector("#textBlock");
  120. parent.lastElementChild.remove();
  121. store.dispatch({ type: "КУПИТЬ", ШО: "пиво", СКОКА: 0 });
  122. }, 1);
  123. function reducer(state, { type, ШО, СКОКА, БАБЛО }) {
  124. //объект action деструктуризируется на три переменных
  125. if (!state) {
  126. //начальная уборка в ларьке:
  127. return {
  128. пиво: { price: 20, amount: 100 },
  129. чипсы: { price: 10, amount: 100 },
  130. сиги: { price: 30, amount: 100 },
  131. касса: { cash: 0 },
  132. };
  133. }
  134. let stateSHO = state[ШО].amount;
  135. let statePrice = state[ШО].price;
  136. if (
  137. type === "КУПИТЬ" &&
  138. stateSHO != 0 &&
  139. stateSHO >= 0 &&
  140. sum.value <= 100
  141. ) {
  142. if((stateSHO - sum.value) >= 0) {
  143. if (statePrice * sum.value <= amount.value) {
  144. return {
  145. ...state,
  146. [ШО]: {
  147. price: state[ШО].price,
  148. amount: stateSHO - СКОКА,
  149. },
  150. касса: {
  151. cash: +state.касса.cash + +amount.value,
  152. },
  153. };
  154. } else {
  155. alert("Мало денеге, пойди в мамы попроси!")
  156. }
  157. } else {
  158. alert("У нас нету столько товара")
  159. }
  160. } else {
  161. alert("Либо у нас нету столько товара, либо говорить научись!")
  162. }
  163. return state;
  164. //если мы не поняли, что от нас просят в `action` - оставляем все как есть
  165. }
  166. function createStore(reducer) {
  167. let state = reducer(undefined, {}); //стартовая инициализация состояния, запуск редьюсера со state === undefined
  168. let cbs = []; //массив подписчиков
  169. const getState = () => state; //функция, возвращающая переменную из замыкания
  170. const subscribe = (cb) => (
  171. cbs.push(cb), //запоминаем подписчиков в массиве
  172. () => (cbs = cbs.filter((c) => c !== cb))
  173. ); //возвращаем функцию unsubscribe, которая удаляет подписчика из списка
  174. const dispatch = (action) => {
  175. const newState = reducer(state, action); //пробуем запустить редьюсер
  176. if (newState !== state) {
  177. //проверяем, смог ли редьюсер обработать action
  178. state = newState; //если смог, то обновляем state
  179. for (let cb of cbs) cb(); //и запускаем подписчиков
  180. }
  181. };
  182. return {
  183. getState, //добавление функции getState в результирующий объект
  184. dispatch,
  185. subscribe, //добавление subscribe в объект
  186. };
  187. }
  188. const store = createStore(reducer);
  189. //запомнит функцию во внутреннем массиве cbs.
  190. //она будет запущена при любом успешном dispatch
  191. const unsubscribe = store.subscribe(() =>
  192. console.log(store.getState())
  193. );
  194. setTimeout(unsubscribe, 10000); //отпишемся через 10 секунд, например
  195. //происходит запуск редьюсера, который создает новый state.
  196. //dispatch запускает всех подписчиков из массива cbs
  197. console.log(store.getState());
  198. store.subscribe(() => console.log(store.getState()));
  199. for (let [good, amount] of Object.entries(store.getState())) {
  200. console.log(good, amount);
  201. let p = document.createElement("p");
  202. p.innerHTML = `${good} (${amount})`;
  203. textBlock.append(p);
  204. button.onclick = () => {
  205. store.dispatch({
  206. type: "КУПИТЬ",
  207. ШО: list.value,
  208. СКОКА: sum.value,
  209. });
  210. kassa.innerHTML =
  211. "КАССА: " + store.getState().касса.cash + " денег";
  212. };
  213. store.subscribe(
  214. () =>
  215. (p.innerText = `${good} (${
  216. store.getState()[good].amount
  217. })`)
  218. );
  219. }
  220. </script>
  221. </body>
  222. </html>