123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- <!DOCTYPE html>
- <html lang="en">
- <head>
- <meta charset="UTF-8" />
- <title>Title</title>
- <style>
- body {
- background: oldlace;
- }
- img {
- width: 150px;
- margin-left: 20px;
- }
- h1 {
- text-align: center;
- color: brown;
- }
- #div {
- display: flex;
- flex-direction: column;
- justify-content: center;
- margin: 0 auto;
- align-items: center;
- }
- #img-block {
- display: flex;
- align-items: center;
- }
- #textBlock {
- display: flex;
- }
- p {
- padding-right: 20px;
- color: brown;
- font-size: 20px;
- }
- button {
- margin: 0 auto;
- width: 200px;
- height: 40px;
- background-color: bisque;
- box-shadow: 6px 6px 20px rgba(0, 0, 0, 0.3);
- }
- button:hover {
- box-shadow: 0px;
- }
- .buttomP {
- color: red;
- margin: 0;
- text-align: center;
- font-size: 20px;
- padding-left: 15px;
- }
- #list,
- #amount,
- #sum {
- margin-bottom: 20px;
- width: 300px;
- height: 20px;
- background-color: bisque;
- }
- #amount,
- #sum {
- width: 294px;
- }
- #text {
- padding-left: 60px;
- }
- </style>
- </head>
- <body>
- <h1>ГОПОТА МАРКЕТ</h1>
- <div id="div">
- <div id="img-block">
- <div>
- <img
- src=""
- alt=""
- />
- <p id="text">20 денег</p>
- </div>
- <div>
- <img
- src="https://euro-opt.shop/4119-large_default/chipsy-pringles-original-pringls-original-165g.jpg"
- alt=""
- width=""
- />
- <p id="text">10 денег</p>
- </div>
- <div>
- <img
- 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"
- alt=""
- />
- <p id="text">30 денег</p>
- </div>
- </div>
- <div id="textBlock">
- <p>В наличии:</p>
- <br />
- </div>
- <p>Чё надо?</p>
- <select name="store" id="list" class="s-10">
- <option value="пиво">Пиво</option>
- <option value="чипсы">Чипсы</option>
- <option value="сиги">Сиги</option>
- </select>
- <p>Скок надо?</p>
- <input type="number" id="sum" class="i-3" />
- <p>Давай бабки! Сдачи нету!</p>
- <input type="number" id="amount" class="i-2" />
- <button id="button" class="b-2">
- <p class="buttomP">КУПИТЬ</p>
- </button>
- <p class="out-2" id="kassa">КАССА: 0 денег</p>
- </div>
- <script>
- setTimeout(function () {
- let parent = document.querySelector("#textBlock");
- parent.lastElementChild.remove();
- store.dispatch({ type: "КУПИТЬ", ШО: "пиво", СКОКА: 0 });
- }, 1);
- function reducer(state, { type, ШО, СКОКА, БАБЛО }) {
- //объект action деструктуризируется на три переменных
- if (!state) {
- //начальная уборка в ларьке:
- return {
- пиво: { price: 20, amount: 100 },
- чипсы: { price: 10, amount: 100 },
- сиги: { price: 30, amount: 100 },
- касса: { cash: 0 },
- };
- }
- let stateSHO = state[ШО].amount;
- let statePrice = state[ШО].price;
- if (
- type === "КУПИТЬ" &&
- stateSHO != 0 &&
- stateSHO >= 0 &&
- sum.value <= 100
- ) {
- if((stateSHO - sum.value) >= 0) {
- if (statePrice * sum.value <= amount.value) {
- return {
- ...state,
- [ШО]: {
- price: state[ШО].price,
- amount: stateSHO - СКОКА,
- },
- касса: {
- cash: +state.касса.cash + +amount.value,
- },
- };
- } else {
- alert("Мало денеге, пойди в мамы попроси!")
- }
- } else {
- alert("У нас нету столько товара")
-
- }
-
- } else {
- alert("Либо у нас нету столько товара, либо говорить научись!")
-
- }
- return state;
- //если мы не поняли, что от нас просят в `action` - оставляем все как есть
- }
- function createStore(reducer) {
- let state = reducer(undefined, {}); //стартовая инициализация состояния, запуск редьюсера со state === undefined
- let cbs = []; //массив подписчиков
- const getState = () => state; //функция, возвращающая переменную из замыкания
- const subscribe = (cb) => (
- cbs.push(cb), //запоминаем подписчиков в массиве
- () => (cbs = cbs.filter((c) => c !== cb))
- ); //возвращаем функцию unsubscribe, которая удаляет подписчика из списка
- const dispatch = (action) => {
- const newState = reducer(state, action); //пробуем запустить редьюсер
- if (newState !== state) {
- //проверяем, смог ли редьюсер обработать action
- state = newState; //если смог, то обновляем state
- for (let cb of cbs) cb(); //и запускаем подписчиков
- }
- };
- return {
- getState, //добавление функции getState в результирующий объект
- dispatch,
- subscribe, //добавление subscribe в объект
- };
- }
- const store = createStore(reducer);
- //запомнит функцию во внутреннем массиве cbs.
- //она будет запущена при любом успешном dispatch
- const unsubscribe = store.subscribe(() =>
- console.log(store.getState())
- );
- setTimeout(unsubscribe, 10000); //отпишемся через 10 секунд, например
- //происходит запуск редьюсера, который создает новый state.
- //dispatch запускает всех подписчиков из массива cbs
-
- console.log(store.getState());
- store.subscribe(() => console.log(store.getState()));
- for (let [good, amount] of Object.entries(store.getState())) {
- console.log(good, amount);
- let p = document.createElement("p");
- p.innerHTML = `${good} (${amount})`;
- textBlock.append(p);
- button.onclick = () => {
- store.dispatch({
- type: "КУПИТЬ",
- ШО: list.value,
- СКОКА: sum.value,
- });
- kassa.innerHTML =
- "КАССА: " + store.getState().касса.cash + " денег";
-
-
- };
- store.subscribe(
- () =>
- (p.innerText = `${good} (${
- store.getState()[good].amount
- })`)
- );
- }
-
- </script>
- </body>
- </html>
|