123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356 |
- <script setup>
- import { ref, watch, onMounted, computed, onBeforeMount } from 'vue'
- import { loadTicker } from './api'
- const API_KEY =
- '49c2392a697fa3880cecd2a3b029b7e993aa3fd7906502e7634d797e92e5164d'
- const ticker = ref('')
- const tickers = ref([])
- const sel = ref(null)
- const graph = ref([])
- const err = ref('')
- const valutes = ref([])
- const page = ref(1)
- const filter = ref('')
- const windowData = Object.fromEntries(
- new URL(window.location).searchParams.entries(),
- )
- page.value = windowData.page ?? 1
- filter.value = windowData.filter ?? ''
- function subscribeToUpdates(tickerName) {
- setInterval(async () => {
- const f = await fetch(
- `https://min-api.cryptocompare.com/data/price?fsym=${tickerName}&tsyms=USD&api_key=${API_KEY}`,
- )
- const data = await f.json()
- // тут я добавил проверку data.USD - некоторые валюты возвращают ошибку
- if (typeof data.USD != 'undefined') {
- tickers.value.find(t => t.name === tickerName).price =
- data.USD > 1 ? data.USD.toFixed(2) : data.USD.toPrecision(2)
- if (sel.value?.name === tickerName) {
- graph.value.push(data.USD)
- }
- }
- }, 5000)
- }
- function add(nameToAdd) {
- const tickerName = nameToAdd.toUpperCase()
- const isInvalid = valutes.value.findIndex(v => v === tickerName) == -1
- console.log('isInvalid:', isInvalid)
- const newTicker = {
- name: tickerName,
- price: '-',
- isInvalid,
- }
- // Проверяем, существует ли уже тикер с таким названием
- if (tickers.value.findIndex(n => n.name === newTicker.name) >= 0) {
- // Если тикер уже существует, помечаем его как некорректный
- err.value = 1
- }
- tickers.value = [...tickers.value, newTicker] // Добавляем некорректный тикер в список
- filter.value = '' // Сбрасываем фильтр
- }
- onBeforeMount(() => {
- const tickersData = localStorage.getItem('cryptonomicon-list') ?? '[]'
- if (tickersData) {
- tickers.value = JSON.parse(tickersData)
- }
- setInterval(() => {
- updateTickers()
- }, 5000)
- })
- watch(tickers, () => {
- localStorage.setItem('cryptonomicon-list', JSON.stringify(tickers.value))
- })
- onMounted(async () => {
- const f = await fetch(
- `https://min-api.cryptocompare.com/data/all/coinlist?summary=true&api_key=${API_KEY}`,
- )
- const data = await f.json()
- if (data.Response == 'Success') {
- valutes.value = Object.keys(data.Data)
- // console.log(valutes.value)
- }
- })
- const shortValutes = computed(() => {
- if (ticker.value !== '')
- return valutes.value
- .filter(v => v.includes(ticker.value.toUpperCase()))
- .slice(0, 4)
- })
- function handleDelete(tickerToRemove) {
- tickers.value = tickers.value.filter(t => t != tickerToRemove)
- if (sel.value == tickerToRemove) {
- sel.value = null
- }
- }
- async function updateTickers() {
- if (!tickers.value.length) return
- const data = await loadTicker(tickers.value.map(t => t.name))
- tickers.value.forEach(t => {
- t.price = data[t.name] ?? '-'
- })
- if (sel.value?.name) {
- graph.value.push(data[sel.value.name])
- }
- }
- const pageStateOptions = computed(() => {
- return {
- filter: filter.value,
- page: page.value,
- }
- })
- function formatPrice(price) {
- if (typeof price == 'number') {
- return price > 1 ? price.toFixed(2) : price.toPrecision(2)
- }
- return price
- }
- watch(filter, () => {
- page.value = 1
- window.history.pushState(
- null,
- document.title,
- `${window.location.pathname}?filter=${filter.value} `,
- )
- })
- watch(pageStateOptions, value => {
- window.history.pushState(
- null,
- document.title,
- `${window.location.pathname}?filter=${value.filter}&page=${value.page}`,
- )
- })
- const startIndex = computed(() => {
- return (page.value - 1) * 6
- })
- const endIndex = computed(() => {
- return page.value * 6
- })
- const filteredTickers = computed(() => {
- return tickers.value.filter(t => t.name.includes(filter.value.toUpperCase()))
- })
- const paginatedTickers = computed(() => {
- return filteredTickers.value.slice(startIndex.value, endIndex.value)
- })
- watch(paginatedTickers, value => {
- if (value.length === 0 && page.value > 1) page.value--
- })
- function select(ticker) {
- sel.value = ticker
- }
- watch(sel, () => {
- graph.value = []
- })
- const hasNextPage = computed(() => {
- return filteredTickers.value.length > endIndex.value
- })
- const normalizedGraph = computed(() => {
- const maxValue = Math.max(...graph.value)
- const minValue = Math.min(...graph.value)
- if (maxValue == minValue) {
- return graph.value.map(() => 50)
- }
- return graph.value.map(
- price => 5 + ((price - minValue) * 95) / (maxValue - minValue),
- )
- })
- </script>
- <template>
- <div class="container mx-auto flex flex-col items-center bg-gray-100 p-4">
- <div class="container">
- <section>
- <div>
- Фильтр: <input v-model="filter" @input="page = '1'" />
- <button
- class="my-4 mx-2 inline-flex items-center py-2 px-4 border border-transparent shadow-sm text-sm leading-4 font-medium rounded-full text-white bg-gray-600 hover:bg-gray-700 transition-colors duration-300 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500"
- @click="page = page - 1"
- v-if="page > 1"
- >
- Назад
- </button>
- <button
- class="my-4 mx-2 inline-flex items-center py-2 px-4 border border-transparent shadow-sm text-sm leading-4 font-medium rounded-full text-white bg-gray-600 hover:bg-gray-700 transition-colors duration-300 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500"
- @click="page = page + 1"
- v-if="hasNextPage"
- >
- Вперед
- </button>
- </div>
- <div class="flex">
- <div class="max-w-xs">
- <label for="wallet" class="block text-sm font-medium text-gray-700"
- >Тикер
- </label>
- <div class="mt-1 relative rounded-md shadow-md">
- <input
- v-model="ticker"
- type="text"
- name="wallet"
- id="wallet"
- class="block w-full pr-10 border-gray-300 text-gray-900 focus:outline-none focus:ring-gray-500 focus:border-gray-500 sm:text-sm rounded-md"
- placeholder="Например DOGE"
- />
- </div>
- <div
- class="flex bg-white shadow-md p-1 rounded-md shadow-md flex-wrap"
- >
- <span
- v-for="(p, index) in shortValutes"
- @click="add(p)"
- class="inline-flex items-center px-2 m-1 rounded-md text-xs font-medium bg-gray-300 text-gray-800 cursor-pointer"
- >
- {{ p }}
- </span>
- </div>
- <div v-if="err === 1" class="text-sm text-red-600">
- Такой тикер уже добавлен
- </div>
- </div>
- </div>
- <button
- @click="add(ticker)"
- v-on:keydown.enter="add"
- type="button"
- class="my-4 inline-flex items-center py-2 px-4 border border-transparent shadow-sm text-sm leading-4 font-medium rounded-full text-white bg-gray-600 hover:bg-gray-700 transition-colors duration-300 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-gray-500"
- >
- <!-- Heroicon name: solid/mail -->
- <svg
- class="-ml-0.5 mr-2 h-6 w-6"
- xmlns="http://www.w3.org/2000/svg"
- width="30"
- height="30"
- viewBox="0 0 24 24"
- fill="#ffffff"
- >
- <path
- d="M13 7h-2v4H7v2h4v4h2v-4h4v-2h-4V7zm-1-5C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"
- ></path>
- </svg>
- Добавить
- </button>
- </section>
- <div v-if="tickers.length > 0">
- <hr class="w-full border-t border-gray-600 my-4" />
- <dl class="mt-5 grid grid-cols-1 gap-5 sm:grid-cols-3">
- <div
- v-for="(t, index) in paginatedTickers"
- :key="index"
- @click="select(t)"
- :class="{
- 'border-4': sel == t,
- 'bg-red-100': t.isInvalid,
- 'bg-white': !t.isInvalid,
- }"
- class="overflow-hidden shadow rounded-lg border-purple-800 border-solid cursor-pointer"
- >
- <div class="px-4 py-5 sm:p-6 text-center">
- <dt class="text-sm font-medium text-gray-500 truncate">
- {{ t.name }} - USD
- </dt>
- <dd class="mt-1 text-3xl font-semibold text-gray-900">
- {{ t.price }}
- </dd>
- </div>
- <div class="w-full border-t border-gray-200"></div>
- <button
- @click.stop="handleDelete(t)"
- class="flex items-center justify-center font-medium w-full bg-gray-100 px-4 py-4 sm:px-6 text-md text-gray-500 hover:text-gray-600 hover:bg-gray-200 hover:opacity-20 transition-all focus:outline-none"
- >
- <svg
- class="h-5 w-5"
- xmlns="http://www.w3.org/2000/svg"
- viewBox="0 0 20 20"
- fill="#718096"
- aria-hidden="true"
- >
- <path
- fill-rule="evenodd"
- d="M9 2a1 1 0 00-.894.553L7.382 4H4a1 1 0 000 2v10a2 2 0 002 2h8a2 2 0 002-2V6a1 1 0 100-2h-3.382l-.724-1.447A1 1 0 0011 2H9zM7 8a1 1 0 012 0v6a1 1 0 11-2 0V8zm5-1a1 1 0 00-1 1v6a1 1 0 102 0V8a1 1 0 00-1-1z"
- clip-rule="evenodd"
- ></path></svg
- >Удалить
- </button>
- </div>
- </dl>
- <hr class="w-full border-t border-gray-600 my-4" />
- </div>
- <section v-if="sel" class="relative">
- <h3 class="text-lg leading-6 font-medium text-gray-900 my-8">
- {{ sel.name }} - USD
- </h3>
- <div class="flex items-end border-gray-600 border-b border-l h-64">
- <div
- v-for="(bar, idx) in normalizedGraph"
- :key="idx"
- :style="{ height: `${bar}%` }"
- class="bg-purple-800 border w-10 h-16"
- ></div>
- </div>
- <button
- @click="sel = null"
- type="button"
- class="absolute top-0 right-0"
- >
- <svg
- xmlns="http://www.w3.org/2000/svg"
- xmlns:xlink="http://www.w3.org/1999/xlink"
- xmlns:svgjs="http://svgjs.com/svgjs"
- version="1.1"
- width="30"
- height="30"
- x="0"
- y="0"
- viewBox="0 0 511.76 511.76"
- style="enable-background: new 0 0 512 512"
- xml:space="preserve"
- >
- <g>
- <path
- d="M436.896,74.869c-99.84-99.819-262.208-99.819-362.048,0c-99.797,99.819-99.797,262.229,0,362.048 c49.92,49.899,115.477,74.837,181.035,74.837s131.093-24.939,181.013-74.837C536.715,337.099,536.715,174.688,436.896,74.869z M361.461,331.317c8.341,8.341,8.341,21.824,0,30.165c-4.16,4.16-9.621,6.251-15.083,6.251c-5.461,0-10.923-2.091-15.083-6.251 l-75.413-75.435l-75.392,75.413c-4.181,4.16-9.643,6.251-15.083,6.251c-5.461,0-10.923-2.091-15.083-6.251 c-8.341-8.341-8.341-21.845,0-30.165l75.392-75.413l-75.413-75.413c-8.341-8.341-8.341-21.845,0-30.165 c8.32-8.341,21.824-8.341,30.165,0l75.413,75.413l75.413-75.413c8.341-8.341,21.824-8.341,30.165,0 c8.341,8.32,8.341,21.824,0,30.165l-75.413,75.413L361.461,331.317z"
- fill="#718096"
- data-original="#000000"
- ></path>
- </g>
- </svg>
- </button>
- </section>
- </div>
- </div>
- </template>
|